import Utils from "../core/Utils.js"; /** * Waiter to handle events related to the output. * * @author n1474335 [n1474335@gmail.com] * @copyright Crown Copyright 2016 * @license Apache-2.0 * * @constructor * @param {App} app - The main view object for CyberChef. * @param {Manager} manager - The CyberChef event manager. */ const OutputWaiter = function(app, manager) { this.app = app; this.manager = manager; }; /** * Gets the output string from the output textarea. * * @returns {string} */ OutputWaiter.prototype.get = function() { return document.getElementById("output-text").value; }; /** * Sets the output in the output textarea. * * @param {string} dataStr - The output string/HTML * @param {string} type - The data type of the output * @param {number} duration - The length of time (ms) it took to generate the output */ OutputWaiter.prototype.set = function(dataStr, type, duration) { const outputText = document.getElementById("output-text"); const outputHtml = document.getElementById("output-html"); const outputHighlighter = document.getElementById("output-highlighter"); const inputHighlighter = document.getElementById("input-highlighter"); if (type === "html") { outputText.style.display = "none"; outputHtml.style.display = "block"; outputHighlighter.display = "none"; inputHighlighter.display = "none"; outputText.value = ""; outputHtml.innerHTML = dataStr; // Execute script sections const scriptElements = outputHtml.querySelectorAll("script"); for (let i = 0; i < scriptElements.length; i++) { try { eval(scriptElements[i].innerHTML); // eslint-disable-line no-eval } catch (err) { console.error(err); } } } else { outputText.style.display = "block"; outputHtml.style.display = "none"; outputHighlighter.display = "block"; inputHighlighter.display = "block"; outputText.value = Utils.printable(dataStr, true); outputHtml.innerHTML = ""; } this.manager.highlighter.removeHighlights(); const lines = dataStr.count("\n") + 1; this.setOutputInfo(dataStr.length, lines, duration); }; /** * Displays information about the output. * * @param {number} length - The length of the current output string * @param {number} lines - The number of the lines in the current output string * @param {number} duration - The length of time (ms) it took to generate the output */ OutputWaiter.prototype.setOutputInfo = function(length, lines, duration) { let width = length.toString().length; width = width < 4 ? 4 : width; const lengthStr = Utils.pad(length.toString(), width, " ").replace(/ /g, " "); const linesStr = Utils.pad(lines.toString(), width, " ").replace(/ /g, " "); const timeStr = Utils.pad(duration.toString() + "ms", width, " ").replace(/ /g, " "); document.getElementById("output-info").innerHTML = "time: " + timeStr + "
length: " + lengthStr + "
lines: " + linesStr; document.getElementById("input-selection-info").innerHTML = ""; document.getElementById("output-selection-info").innerHTML = ""; }; /** * Adjusts the display properties of the output buttons so that they fit within the current width * without wrapping or overflowing. */ OutputWaiter.prototype.adjustWidth = function() { const output = document.getElementById("output"); const saveToFile = document.getElementById("save-to-file"); const switchIO = document.getElementById("switch"); const undoSwitch = document.getElementById("undo-switch"); const maximiseOutput = document.getElementById("maximise-output"); if (output.clientWidth < 680) { saveToFile.childNodes[1].nodeValue = ""; switchIO.childNodes[1].nodeValue = ""; undoSwitch.childNodes[1].nodeValue = ""; maximiseOutput.childNodes[1].nodeValue = ""; } else { saveToFile.childNodes[1].nodeValue = " Save to file"; switchIO.childNodes[1].nodeValue = " Move output to input"; undoSwitch.childNodes[1].nodeValue = " Undo"; maximiseOutput.childNodes[1].nodeValue = maximiseOutput.getAttribute("title") === "Maximise" ? " Max" : " Restore"; } }; /** * Handler for save click events. * Saves the current output to a file, downloaded as a URL octet stream. */ OutputWaiter.prototype.saveClick = function() { const data = Utils.toBase64(this.app.dishStr); const filename = window.prompt("Please enter a filename:", "download.dat"); if (filename) { const el = document.createElement("a"); el.setAttribute("href", "data:application/octet-stream;base64;charset=utf-8," + data); el.setAttribute("download", filename); // Firefox requires that the element be added to the DOM before it can be clicked el.style.display = "none"; document.body.appendChild(el); el.click(); el.remove(); } }; /** * Handler for switch click events. * Moves the current output into the input textarea. */ OutputWaiter.prototype.switchClick = function() { this.switchOrigData = this.manager.input.get(); document.getElementById("undo-switch").disabled = false; this.app.setInput(this.app.dishStr); }; /** * Handler for undo switch click events. * Removes the output from the input and replaces the input that was removed. */ OutputWaiter.prototype.undoSwitchClick = function() { this.app.setInput(this.switchOrigData); document.getElementById("undo-switch").disabled = true; }; /** * Handler for file switch click events. * Moves a files data for items created via Utils.displayFilesAsHTML to the input. */ OutputWaiter.prototype.fileSwitch = function(e) { e.preventDefault(); this.switchOrigData = this.manager.input.get(); this.app.setInput(e.target.getAttribute("fileValue")); document.getElementById("undo-switch").disabled = false; }; /** * Handler for maximise output click events. * Resizes the output frame to be as large as possible, or restores it to its original size. */ OutputWaiter.prototype.maximiseOutputClick = function(e) { const el = e.target.id === "maximise-output" ? e.target : e.target.parentNode; if (el.getAttribute("title") === "Maximise") { this.app.columnSplitter.collapse(0); this.app.columnSplitter.collapse(1); this.app.ioSplitter.collapse(0); el.setAttribute("title", "Restore"); el.innerHTML = " Restore"; this.adjustWidth(); } else { el.setAttribute("title", "Maximise"); el.innerHTML = " Max"; this.app.resetLayout(); } }; /** * Shows or hides the loading icon. * * @param {boolean} value */ OutputWaiter.prototype.toggleLoader = function(value) { const outputLoader = document.getElementById("output-loader"), outputElement = document.getElementById("output-text"); if (value) { this.manager.controls.hideStaleIndicator(); this.bakingStatusTimeout = setTimeout(function() { outputElement.disabled = true; outputLoader.style.visibility = "visible"; outputLoader.style.opacity = 1; this.manager.controls.toggleBakeButtonFunction(true); }.bind(this), 200); } else { clearTimeout(this.bakingStatusTimeout); outputElement.disabled = false; outputLoader.style.opacity = 0; outputLoader.style.visibility = "hidden"; this.manager.controls.toggleBakeButtonFunction(false); this.setStatusMsg(""); } }; /** * Sets the baking status message value. * * @param {string} msg */ OutputWaiter.prototype.setStatusMsg = function(msg) { const el = document.querySelector("#output-loader .loading-msg"); el.textContent = msg; }; export default OutputWaiter;