import { scheduleOnce } from "@ember/runloop"; import Component from "@ember/component"; import discourseDebounce from "discourse/lib/debounce"; import { renderSpinner } from "discourse/helpers/loading-spinner"; import { escapeExpression } from "discourse/lib/utilities"; import { bufferedRender } from "discourse-common/lib/buffered-render"; import { observes, on } from "discourse-common/utils/decorators"; export default Component.extend( bufferedRender({ classNames: ["admin-backups-logs"], init() { this._super(...arguments); this._reset(); }, _reset() { this.setProperties({ formattedLogs: "", index: 0 }); }, _scrollDown() { const div = this.element; div.scrollTop = div.scrollHeight; }, @on("init") @observes("logs.[]") _resetFormattedLogs() { if (this.logs.length === 0) { this._reset(); // reset the cached logs whenever the model is reset this.rerenderBuffer(); } }, @on("init") @observes("logs.[]") _updateFormattedLogs: discourseDebounce(function() { const logs = this.logs; if (logs.length === 0) return; // do the log formatting only once for HELLish performance let formattedLogs = this.formattedLogs; for (let i = this.index, length = logs.length; i < length; i++) { const date = logs[i].get("timestamp"), message = escapeExpression(logs[i].get("message")); formattedLogs += "[" + date + "] " + message + "\n"; } // update the formatted logs & cache index this.setProperties({ formattedLogs: formattedLogs, index: logs.length }); // force rerender this.rerenderBuffer(); scheduleOnce("afterRender", this, this._scrollDown); }, 150), buildBuffer(buffer) { const formattedLogs = this.formattedLogs; if (formattedLogs && formattedLogs.length > 0) { buffer.push("
");
        buffer.push(formattedLogs);
        buffer.push("
"); } else { buffer.push("

" + I18n.t("admin.backups.logs.none") + "

"); } // add a loading indicator if (this.get("status.isOperationRunning")) { buffer.push(renderSpinner("small")); } } }) );