CyberChef/src/web/OutputWaiter.js

245 lines
8.5 KiB
JavaScript
Raw Normal View History

import Utils from "../core/Utils.js";
2016-11-28 10:42:58 +00:00
/**
* 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.
2016-11-28 10:42:58 +00:00
* @param {Manager} manager - The CyberChef event manager.
*/
2017-04-13 17:08:50 +00:00
const OutputWaiter = function(app, manager) {
2016-11-28 10:42:58 +00:00
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
2016-11-28 10:42:58 +00:00
* @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) {
2017-05-02 23:40:39 +00:00
const outputText = document.getElementById("output-text");
const outputHtml = document.getElementById("output-html");
const outputHighlighter = document.getElementById("output-highlighter");
const inputHighlighter = document.getElementById("input-highlighter");
2016-11-28 10:42:58 +00:00
2016-12-14 16:39:17 +00:00
if (type === "html") {
outputText.style.display = "none";
outputHtml.style.display = "block";
outputHighlighter.display = "none";
inputHighlighter.display = "none";
2017-02-09 15:09:33 +00:00
outputText.value = "";
outputHtml.innerHTML = dataStr;
2017-02-09 15:09:33 +00:00
2016-11-28 10:42:58 +00:00
// Execute script sections
2017-04-13 17:08:50 +00:00
const scriptElements = outputHtml.querySelectorAll("script");
for (let i = 0; i < scriptElements.length; i++) {
2016-11-28 10:42:58 +00:00
try {
eval(scriptElements[i].innerHTML); // eslint-disable-line no-eval
2016-11-28 10:42:58 +00:00
} catch (err) {
console.error(err);
}
}
} else {
outputText.style.display = "block";
outputHtml.style.display = "none";
outputHighlighter.display = "block";
inputHighlighter.display = "block";
2017-02-09 15:09:33 +00:00
outputText.value = Utils.printable(dataStr, true);
outputHtml.innerHTML = "";
2016-11-28 10:42:58 +00:00
}
2017-02-09 15:09:33 +00:00
this.manager.highlighter.removeHighlights();
2017-04-13 17:08:50 +00:00
const lines = dataStr.count("\n") + 1;
this.setOutputInfo(dataStr.length, lines, duration);
2016-11-28 10:42:58 +00:00
};
/**
* 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) {
2017-04-13 17:08:50 +00:00
let width = length.toString().length;
2016-11-28 10:42:58 +00:00
width = width < 4 ? 4 : width;
2017-02-09 15:09:33 +00:00
2017-04-13 17:08:50 +00:00
const lengthStr = Utils.pad(length.toString(), width, " ").replace(/ /g, "&nbsp;");
const linesStr = Utils.pad(lines.toString(), width, " ").replace(/ /g, "&nbsp;");
const timeStr = Utils.pad(duration.toString() + "ms", width, " ").replace(/ /g, "&nbsp;");
2017-02-09 15:09:33 +00:00
document.getElementById("output-info").innerHTML = "time: " + timeStr +
"<br>length: " + lengthStr +
"<br>lines: " + linesStr;
2016-11-28 10:42:58 +00:00
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() {
2017-05-02 23:40:39 +00:00
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");
2017-02-09 15:09:33 +00:00
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";
}
};
2016-11-28 10:42:58 +00:00
/**
* Handler for save click events.
* Saves the current output to a file, downloaded as a URL octet stream.
*/
OutputWaiter.prototype.saveClick = function() {
2017-05-02 23:40:39 +00:00
const data = Utils.toBase64(this.app.dishStr);
const filename = window.prompt("Please enter a filename:", "download.dat");
2017-02-09 15:09:33 +00:00
2016-11-28 10:42:58 +00:00
if (filename) {
2017-04-13 17:08:50 +00:00
const el = document.createElement("a");
2016-11-28 10:42:58 +00:00
el.setAttribute("href", "data:application/octet-stream;base64;charset=utf-8," + data);
el.setAttribute("download", filename);
2017-02-09 15:09:33 +00:00
2016-11-28 10:42:58 +00:00
// Firefox requires that the element be added to the DOM before it can be clicked
el.style.display = "none";
document.body.appendChild(el);
2017-02-09 15:09:33 +00:00
2016-11-28 10:42:58 +00:00
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();
2016-11-28 10:42:58 +00:00
document.getElementById("undo-switch").disabled = false;
this.app.setInput(this.app.dishStr);
2016-11-28 10:42:58 +00:00
};
/**
* 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);
2016-11-28 10:42:58 +00:00
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) {
2017-04-13 17:08:50 +00:00
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 = "<img src=''> Restore";
this.adjustWidth();
} else {
el.setAttribute("title", "Maximise");
el.innerHTML = "<img src=''> 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;