Input and output character encodings can now be set

This commit is contained in:
n1474335 2022-09-02 12:56:04 +01:00
parent 7c8a185a3d
commit e93aa42697
15 changed files with 482 additions and 423 deletions

View file

@ -68,16 +68,10 @@ class Chef {
// Present the raw result // Present the raw result
await recipe.present(this.dish); await recipe.present(this.dish);
// Depending on the size of the output, we may send it back as a string or an ArrayBuffer.
// This can prevent unnecessary casting as an ArrayBuffer can be easily downloaded as a file.
// The threshold is specified in KiB.
const threshold = (options.ioDisplayThreshold || 1024) * 1024;
const returnType = const returnType =
this.dish.type === Dish.HTML ? this.dish.type === Dish.HTML ?
Dish.HTML : Dish.HTML :
this.dish.size > threshold ? Dish.ARRAY_BUFFER;
Dish.ARRAY_BUFFER :
Dish.STRING;
return { return {
dish: rawDish, dish: rawDish,

View file

@ -101,14 +101,17 @@ async function bake(data) {
// Ensure the relevant modules are loaded // Ensure the relevant modules are loaded
self.loadRequiredModules(data.recipeConfig); self.loadRequiredModules(data.recipeConfig);
try { try {
self.inputNum = (data.inputNum !== undefined) ? data.inputNum : -1; self.inputNum = data.inputNum === undefined ? -1 : data.inputNum;
const response = await self.chef.bake( const response = await self.chef.bake(
data.input, // The user's input data.input, // The user's input
data.recipeConfig, // The configuration of the recipe data.recipeConfig, // The configuration of the recipe
data.options // Options set by the user data.options // Options set by the user
); );
const transferable = (data.input instanceof ArrayBuffer) ? [data.input] : undefined; const transferable = (response.dish.value instanceof ArrayBuffer) ?
[response.dish.value] :
undefined;
self.postMessage({ self.postMessage({
action: "bakeComplete", action: "bakeComplete",
data: Object.assign(response, { data: Object.assign(response, {

View file

@ -406,6 +406,7 @@ class Utils {
* Utils.strToArrayBuffer("你好"); * Utils.strToArrayBuffer("你好");
*/ */
static strToArrayBuffer(str) { static strToArrayBuffer(str) {
log.debug("Converting string to array buffer");
const arr = new Uint8Array(str.length); const arr = new Uint8Array(str.length);
let i = str.length, b; let i = str.length, b;
while (i--) { while (i--) {
@ -432,6 +433,7 @@ class Utils {
* Utils.strToUtf8ArrayBuffer("你好"); * Utils.strToUtf8ArrayBuffer("你好");
*/ */
static strToUtf8ArrayBuffer(str) { static strToUtf8ArrayBuffer(str) {
log.debug("Converting string to UTF8 array buffer");
const utf8Str = utf8.encode(str); const utf8Str = utf8.encode(str);
if (str.length !== utf8Str.length) { if (str.length !== utf8Str.length) {
@ -461,6 +463,7 @@ class Utils {
* Utils.strToByteArray("你好"); * Utils.strToByteArray("你好");
*/ */
static strToByteArray(str) { static strToByteArray(str) {
log.debug("Converting string to byte array");
const byteArray = new Array(str.length); const byteArray = new Array(str.length);
let i = str.length, b; let i = str.length, b;
while (i--) { while (i--) {
@ -487,6 +490,7 @@ class Utils {
* Utils.strToUtf8ByteArray("你好"); * Utils.strToUtf8ByteArray("你好");
*/ */
static strToUtf8ByteArray(str) { static strToUtf8ByteArray(str) {
log.debug("Converting string to UTF8 byte array");
const utf8Str = utf8.encode(str); const utf8Str = utf8.encode(str);
if (str.length !== utf8Str.length) { if (str.length !== utf8Str.length) {
@ -515,6 +519,7 @@ class Utils {
* Utils.strToCharcode("你好"); * Utils.strToCharcode("你好");
*/ */
static strToCharcode(str) { static strToCharcode(str) {
log.debug("Converting string to charcode");
const charcode = []; const charcode = [];
for (let i = 0; i < str.length; i++) { for (let i = 0; i < str.length; i++) {
@ -549,6 +554,7 @@ class Utils {
* Utils.byteArrayToUtf8([228,189,160,229,165,189]); * Utils.byteArrayToUtf8([228,189,160,229,165,189]);
*/ */
static byteArrayToUtf8(byteArray) { static byteArrayToUtf8(byteArray) {
log.debug("Converting byte array to UTF8");
const str = Utils.byteArrayToChars(byteArray); const str = Utils.byteArrayToChars(byteArray);
try { try {
const utf8Str = utf8.decode(str); const utf8Str = utf8.decode(str);
@ -581,6 +587,7 @@ class Utils {
* Utils.byteArrayToChars([20320,22909]); * Utils.byteArrayToChars([20320,22909]);
*/ */
static byteArrayToChars(byteArray) { static byteArrayToChars(byteArray) {
log.debug("Converting byte array to chars");
if (!byteArray) return ""; if (!byteArray) return "";
let str = ""; let str = "";
// String concatenation appears to be faster than an array join // String concatenation appears to be faster than an array join
@ -603,6 +610,7 @@ class Utils {
* Utils.arrayBufferToStr(Uint8Array.from([104,101,108,108,111]).buffer); * Utils.arrayBufferToStr(Uint8Array.from([104,101,108,108,111]).buffer);
*/ */
static arrayBufferToStr(arrayBuffer, utf8=true) { static arrayBufferToStr(arrayBuffer, utf8=true) {
log.debug("Converting array buffer to str");
const arr = new Uint8Array(arrayBuffer); const arr = new Uint8Array(arrayBuffer);
return utf8 ? Utils.byteArrayToUtf8(arr) : Utils.byteArrayToChars(arr); return utf8 ? Utils.byteArrayToUtf8(arr) : Utils.byteArrayToChars(arr);
} }

View file

@ -9,7 +9,7 @@
/** /**
* Character encoding format mappings. * Character encoding format mappings.
*/ */
export const IO_FORMAT = { export const CHR_ENC_CODE_PAGES = {
"UTF-8 (65001)": 65001, "UTF-8 (65001)": 65001,
"UTF-7 (65000)": 65000, "UTF-7 (65000)": 65000,
"UTF-16LE (1200)": 1200, "UTF-16LE (1200)": 1200,
@ -164,6 +164,17 @@ export const IO_FORMAT = {
"Simplified Chinese GB18030 (54936)": 54936, "Simplified Chinese GB18030 (54936)": 54936,
}; };
export const CHR_ENC_SIMPLE_LOOKUP = {};
export const CHR_ENC_SIMPLE_REVERSE_LOOKUP = {};
for (const name in CHR_ENC_CODE_PAGES) {
const simpleName = name.match(/(^.+)\([\d/]+\)$/)[1];
CHR_ENC_SIMPLE_LOOKUP[simpleName] = CHR_ENC_CODE_PAGES[name];
CHR_ENC_SIMPLE_REVERSE_LOOKUP[CHR_ENC_CODE_PAGES[name]] = simpleName;
}
/** /**
* Unicode Normalisation Forms * Unicode Normalisation Forms
* *

View file

@ -6,7 +6,7 @@
import Operation from "../Operation.mjs"; import Operation from "../Operation.mjs";
import cptable from "codepage"; import cptable from "codepage";
import {IO_FORMAT} from "../lib/ChrEnc.mjs"; import {CHR_ENC_CODE_PAGES} from "../lib/ChrEnc.mjs";
/** /**
* Decode text operation * Decode text operation
@ -26,7 +26,7 @@ class DecodeText extends Operation {
"<br><br>", "<br><br>",
"Supported charsets are:", "Supported charsets are:",
"<ul>", "<ul>",
Object.keys(IO_FORMAT).map(e => `<li>${e}</li>`).join("\n"), Object.keys(CHR_ENC_CODE_PAGES).map(e => `<li>${e}</li>`).join("\n"),
"</ul>", "</ul>",
].join("\n"); ].join("\n");
this.infoURL = "https://wikipedia.org/wiki/Character_encoding"; this.infoURL = "https://wikipedia.org/wiki/Character_encoding";
@ -36,7 +36,7 @@ class DecodeText extends Operation {
{ {
"name": "Encoding", "name": "Encoding",
"type": "option", "type": "option",
"value": Object.keys(IO_FORMAT) "value": Object.keys(CHR_ENC_CODE_PAGES)
} }
]; ];
} }
@ -47,7 +47,7 @@ class DecodeText extends Operation {
* @returns {string} * @returns {string}
*/ */
run(input, args) { run(input, args) {
const format = IO_FORMAT[args[0]]; const format = CHR_ENC_CODE_PAGES[args[0]];
return cptable.utils.decode(format, new Uint8Array(input)); return cptable.utils.decode(format, new Uint8Array(input));
} }

View file

@ -6,7 +6,7 @@
import Operation from "../Operation.mjs"; import Operation from "../Operation.mjs";
import cptable from "codepage"; import cptable from "codepage";
import {IO_FORMAT} from "../lib/ChrEnc.mjs"; import {CHR_ENC_CODE_PAGES} from "../lib/ChrEnc.mjs";
/** /**
* Encode text operation * Encode text operation
@ -26,7 +26,7 @@ class EncodeText extends Operation {
"<br><br>", "<br><br>",
"Supported charsets are:", "Supported charsets are:",
"<ul>", "<ul>",
Object.keys(IO_FORMAT).map(e => `<li>${e}</li>`).join("\n"), Object.keys(CHR_ENC_CODE_PAGES).map(e => `<li>${e}</li>`).join("\n"),
"</ul>", "</ul>",
].join("\n"); ].join("\n");
this.infoURL = "https://wikipedia.org/wiki/Character_encoding"; this.infoURL = "https://wikipedia.org/wiki/Character_encoding";
@ -36,7 +36,7 @@ class EncodeText extends Operation {
{ {
"name": "Encoding", "name": "Encoding",
"type": "option", "type": "option",
"value": Object.keys(IO_FORMAT) "value": Object.keys(CHR_ENC_CODE_PAGES)
} }
]; ];
} }
@ -47,7 +47,7 @@ class EncodeText extends Operation {
* @returns {ArrayBuffer} * @returns {ArrayBuffer}
*/ */
run(input, args) { run(input, args) {
const format = IO_FORMAT[args[0]]; const format = CHR_ENC_CODE_PAGES[args[0]];
const encoded = cptable.utils.encode(format, input); const encoded = cptable.utils.encode(format, input);
return new Uint8Array(encoded).buffer; return new Uint8Array(encoded).buffer;
} }

View file

@ -8,7 +8,7 @@
import Operation from "../Operation.mjs"; import Operation from "../Operation.mjs";
import Utils from "../Utils.mjs"; import Utils from "../Utils.mjs";
import cptable from "codepage"; import cptable from "codepage";
import {IO_FORMAT} from "../lib/ChrEnc.mjs"; import {CHR_ENC_CODE_PAGES} from "../lib/ChrEnc.mjs";
/** /**
* Text Encoding Brute Force operation * Text Encoding Brute Force operation
@ -28,7 +28,7 @@ class TextEncodingBruteForce extends Operation {
"<br><br>", "<br><br>",
"Supported charsets are:", "Supported charsets are:",
"<ul>", "<ul>",
Object.keys(IO_FORMAT).map(e => `<li>${e}</li>`).join("\n"), Object.keys(CHR_ENC_CODE_PAGES).map(e => `<li>${e}</li>`).join("\n"),
"</ul>" "</ul>"
].join("\n"); ].join("\n");
this.infoURL = "https://wikipedia.org/wiki/Character_encoding"; this.infoURL = "https://wikipedia.org/wiki/Character_encoding";
@ -51,15 +51,15 @@ class TextEncodingBruteForce extends Operation {
*/ */
run(input, args) { run(input, args) {
const output = {}, const output = {},
charsets = Object.keys(IO_FORMAT), charsets = Object.keys(CHR_ENC_CODE_PAGES),
mode = args[0]; mode = args[0];
charsets.forEach(charset => { charsets.forEach(charset => {
try { try {
if (mode === "Decode") { if (mode === "Decode") {
output[charset] = cptable.utils.decode(IO_FORMAT[charset], input); output[charset] = cptable.utils.decode(CHR_ENC_CODE_PAGES[charset], input);
} else { } else {
output[charset] = Utils.arrayBufferToStr(cptable.utils.encode(IO_FORMAT[charset], input)); output[charset] = Utils.arrayBufferToStr(cptable.utils.encode(CHR_ENC_CODE_PAGES[charset], input));
} }
} catch (err) { } catch (err) {
output[charset] = "Could not decode."; output[charset] = "Could not decode.";

View file

@ -180,7 +180,6 @@ class Manager {
document.getElementById("save-all-to-file").addEventListener("click", this.output.saveAllClick.bind(this.output)); document.getElementById("save-all-to-file").addEventListener("click", this.output.saveAllClick.bind(this.output));
document.getElementById("copy-output").addEventListener("click", this.output.copyClick.bind(this.output)); document.getElementById("copy-output").addEventListener("click", this.output.copyClick.bind(this.output));
document.getElementById("switch").addEventListener("click", this.output.switchClick.bind(this.output)); document.getElementById("switch").addEventListener("click", this.output.switchClick.bind(this.output));
document.getElementById("undo-switch").addEventListener("click", this.output.undoSwitchClick.bind(this.output));
document.getElementById("maximise-output").addEventListener("click", this.output.maximiseOutputClick.bind(this.output)); document.getElementById("maximise-output").addEventListener("click", this.output.maximiseOutputClick.bind(this.output));
document.getElementById("magic").addEventListener("click", this.output.magicClick.bind(this.output)); document.getElementById("magic").addEventListener("click", this.output.magicClick.bind(this.output));
this.addDynamicListener("#output-file-download", "click", this.output.downloadFile, this.output); this.addDynamicListener("#output-file-download", "click", this.output.downloadFile, this.output);

View file

@ -300,9 +300,6 @@
<button type="button" class="btn btn-primary bmd-btn-icon" id="switch" data-toggle="tooltip" title="Replace input with output"> <button type="button" class="btn btn-primary bmd-btn-icon" id="switch" data-toggle="tooltip" title="Replace input with output">
<i class="material-icons">open_in_browser</i> <i class="material-icons">open_in_browser</i>
</button> </button>
<button type="button" class="btn btn-primary bmd-btn-icon" id="undo-switch" data-toggle="tooltip" title="Undo" disabled="disabled">
<i class="material-icons">undo</i>
</button>
<button type="button" class="btn btn-primary bmd-btn-icon" id="maximise-output" data-toggle="tooltip" title="Maximise output pane"> <button type="button" class="btn btn-primary bmd-btn-icon" id="maximise-output" data-toggle="tooltip" title="Maximise output pane">
<i class="material-icons">fullscreen</i> <i class="material-icons">fullscreen</i>
</button> </button>

View file

@ -224,7 +224,7 @@
#output-file { #output-file {
position: absolute; position: absolute;
left: 0; left: 0;
bottom: 0; top: 50%;
width: 100%; width: 100%;
display: none; display: none;
} }
@ -446,6 +446,10 @@
/* Status bar */ /* Status bar */
.cm-panel input::placeholder {
font-size: 12px !important;
}
.ͼ2 .cm-panels { .ͼ2 .cm-panels {
background-color: var(--secondary-background-colour); background-color: var(--secondary-background-colour);
border-color: var(--secondary-border-colour); border-color: var(--secondary-border-colour);
@ -509,12 +513,38 @@
background-color: #ddd background-color: #ddd
} }
/* Show the dropup menu on hover */
.cm-status-bar-select:hover .cm-status-bar-select-content {
display: block;
}
/* Change the background color of the dropup button when the dropup content is shown */ /* Change the background color of the dropup button when the dropup content is shown */
.cm-status-bar-select:hover .cm-status-bar-select-btn { .cm-status-bar-select:hover .cm-status-bar-select-btn {
background-color: #f1f1f1; background-color: #f1f1f1;
} }
/* The search field */
.cm-status-bar-filter-input {
box-sizing: border-box;
font-size: 12px;
padding-left: 10px !important;
border: none;
}
.cm-status-bar-filter-search {
border-top: 1px solid #ddd;
}
/* Show the dropup menu */
.cm-status-bar-select .show {
display: block;
}
.cm-status-bar-select-scroll {
overflow-y: auto;
max-height: 300px;
}
.chr-enc-value {
max-width: 150px;
display: inline-block;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
vertical-align: middle;
}

View file

@ -65,10 +65,12 @@ class HTMLWidget extends WidgetType {
*/ */
replaceControlChars(textNode) { replaceControlChars(textNode) {
const val = escapeControlChars(textNode.nodeValue, true, this.view.state.lineBreak); const val = escapeControlChars(textNode.nodeValue, true, this.view.state.lineBreak);
const node = document.createElement("null"); if (val.length !== textNode.nodeValue.length) {
const node = document.createElement("span");
node.innerHTML = val; node.innerHTML = val;
textNode.parentNode.replaceChild(node, textNode); textNode.parentNode.replaceChild(node, textNode);
} }
}
} }
@ -119,8 +121,7 @@ export function htmlPlugin(htmlOutput) {
} }
} }
}, { }, {
decorations: v => v.decorations, decorations: v => v.decorations
} }
); );

View file

@ -5,6 +5,7 @@
*/ */
import {showPanel} from "@codemirror/view"; import {showPanel} from "@codemirror/view";
import {CHR_ENC_SIMPLE_LOOKUP, CHR_ENC_SIMPLE_REVERSE_LOOKUP} from "../../core/lib/ChrEnc.mjs";
/** /**
* A Status bar extension for CodeMirror * A Status bar extension for CodeMirror
@ -19,6 +20,10 @@ class StatusBarPanel {
this.label = opts.label; this.label = opts.label;
this.bakeStats = opts.bakeStats ? opts.bakeStats : null; this.bakeStats = opts.bakeStats ? opts.bakeStats : null;
this.eolHandler = opts.eolHandler; this.eolHandler = opts.eolHandler;
this.chrEncHandler = opts.chrEncHandler;
this.eolVal = null;
this.chrEncVal = null;
this.dom = this.buildDOM(); this.dom = this.buildDOM();
} }
@ -40,19 +45,42 @@ class StatusBarPanel {
dom.appendChild(rhs); dom.appendChild(rhs);
// Event listeners // Event listeners
dom.addEventListener("click", this.eolSelectClick.bind(this), false); dom.querySelectorAll(".cm-status-bar-select-btn").forEach(
el => el.addEventListener("click", this.showDropUp.bind(this), false)
);
dom.querySelector(".eol-select").addEventListener("click", this.eolSelectClick.bind(this), false);
dom.querySelector(".chr-enc-select").addEventListener("click", this.chrEncSelectClick.bind(this), false);
dom.querySelector(".cm-status-bar-filter-input").addEventListener("keyup", this.chrEncFilter.bind(this), false);
return dom; return dom;
} }
/**
* Handler for dropup clicks
* Shows/Hides the dropup
* @param {Event} e
*/
showDropUp(e) {
const el = e.target
.closest(".cm-status-bar-select")
.querySelector(".cm-status-bar-select-content");
el.classList.add("show");
// Focus the filter input if present
const filter = el.querySelector(".cm-status-bar-filter-input");
if (filter) filter.focus();
// Set up a listener to close the menu if the user clicks outside of it
hideOnClickOutside(el, e);
}
/** /**
* Handler for EOL Select clicks * Handler for EOL Select clicks
* Sets the line separator * Sets the line separator
* @param {Event} e * @param {Event} e
*/ */
eolSelectClick(e) { eolSelectClick(e) {
e.preventDefault();
const eolLookup = { const eolLookup = {
"LF": "\u000a", "LF": "\u000a",
"VT": "\u000b", "VT": "\u000b",
@ -65,8 +93,46 @@ class StatusBarPanel {
}; };
const eolval = eolLookup[e.target.getAttribute("data-val")]; const eolval = eolLookup[e.target.getAttribute("data-val")];
if (eolval === undefined) return;
// Call relevant EOL change handler // Call relevant EOL change handler
this.eolHandler(eolval); this.eolHandler(eolval);
hideElement(e.target.closest(".cm-status-bar-select-content"));
}
/**
* Handler for Chr Enc Select clicks
* Sets the character encoding
* @param {Event} e
*/
chrEncSelectClick(e) {
const chrEncVal = parseInt(e.target.getAttribute("data-val"), 10);
if (isNaN(chrEncVal)) return;
this.chrEncHandler(chrEncVal);
this.updateCharEnc(chrEncVal);
hideElement(e.target.closest(".cm-status-bar-select-content"));
}
/**
* Handler for Chr Enc keyup events
* Filters the list of selectable character encodings
* @param {Event} e
*/
chrEncFilter(e) {
const input = e.target;
const filter = input.value.toLowerCase();
const div = input.closest(".cm-status-bar-select-content");
const a = div.getElementsByTagName("a");
for (let i = 0; i < a.length; i++) {
const txtValue = a[i].textContent || a[i].innerText;
if (txtValue.toLowerCase().includes(filter)) {
a[i].style.display = "block";
} else {
a[i].style.display = "none";
}
}
} }
/** /**
@ -121,33 +187,48 @@ class StatusBarPanel {
} }
/** /**
* Gets the current character encoding of the document * Sets the current EOL separator in the status bar
* @param {EditorState} state
*/
updateCharEnc(state) {
// const charenc = this.dom.querySelector("#char-enc-value");
// TODO
// charenc.textContent = "TODO";
}
/**
* Returns what the current EOL separator is set to
* @param {EditorState} state * @param {EditorState} state
*/ */
updateEOL(state) { updateEOL(state) {
if (state.lineBreak === this.eolVal) return;
const eolLookup = { const eolLookup = {
"\u000a": "LF", "\u000a": ["LF", "Line Feed"],
"\u000b": "VT", "\u000b": ["VT", "Vertical Tab"],
"\u000c": "FF", "\u000c": ["FF", "Form Feed"],
"\u000d": "CR", "\u000d": ["CR", "Carriage Return"],
"\u000d\u000a": "CRLF", "\u000d\u000a": ["CRLF", "Carriage Return + Line Feed"],
"\u0085": "NEL", "\u0085": ["NEL", "Next Line"],
"\u2028": "LS", "\u2028": ["LS", "Line Separator"],
"\u2029": "PS" "\u2029": ["PS", "Paragraph Separator"]
}; };
const val = this.dom.querySelector(".eol-value"); const val = this.dom.querySelector(".eol-value");
val.textContent = eolLookup[state.lineBreak]; const button = val.closest(".cm-status-bar-select-btn");
const eolName = eolLookup[state.lineBreak];
val.textContent = eolName[0];
button.setAttribute("title", `End of line sequence: ${eolName[1]}`);
button.setAttribute("data-original-title", `End of line sequence: ${eolName[1]}`);
this.eolVal = state.lineBreak;
}
/**
* Gets the current character encoding of the document
* @param {number} chrEncVal
*/
updateCharEnc(chrEncVal) {
if (chrEncVal === this.chrEncVal) return;
const name = CHR_ENC_SIMPLE_REVERSE_LOOKUP[chrEncVal] ? CHR_ENC_SIMPLE_REVERSE_LOOKUP[chrEncVal] : "Raw Bytes";
const val = this.dom.querySelector(".chr-enc-value");
const button = val.closest(".cm-status-bar-select-btn");
val.textContent = name;
button.setAttribute("title", `${this.label} character encoding: ${name}`);
button.setAttribute("data-original-title", `${this.label} character encoding: ${name}`);
this.chrEncVal = chrEncVal;
} }
/** /**
@ -168,6 +249,19 @@ class StatusBarPanel {
} }
} }
/**
* Updates the sizing of elements that need to fit correctly
* @param {EditorView} view
*/
updateSizing(view) {
const viewHeight = view.contentDOM.clientHeight;
this.dom.querySelectorAll(".cm-status-bar-select-scroll").forEach(
el => {
el.style.maxHeight = (viewHeight - 50) + "px";
}
);
}
/** /**
* Builds the Left-hand-side widgets * Builds the Left-hand-side widgets
* @returns {string} * @returns {string}
@ -197,39 +291,98 @@ class StatusBarPanel {
/** /**
* Builds the Right-hand-side widgets * Builds the Right-hand-side widgets
* Event listener set up in Manager * Event listener set up in Manager
*
* @returns {string} * @returns {string}
*/ */
constructRHS() { constructRHS() {
const chrEncOptions = Object.keys(CHR_ENC_SIMPLE_LOOKUP).map(name =>
`<a href="#" draggable="false" data-val="${CHR_ENC_SIMPLE_LOOKUP[name]}">${name}</a>`
).join("");
return ` return `
<span class="baking-time-info" style="display: none" data-toggle="tooltip" title="Baking time"> <span class="baking-time-info" style="display: none" data-toggle="tooltip" title="Baking time">
<i class="material-icons">schedule</i> <i class="material-icons">schedule</i>
<span class="baking-time-value"></span>ms <span class="baking-time-value"></span>ms
</span> </span>
<span data-toggle="tooltip" title="${this.label} character encoding"> <div class="cm-status-bar-select chr-enc-select">
<i class="material-icons">language</i> <span class="cm-status-bar-select-btn" data-toggle="tooltip" data-placement="left" title="${this.label} character encoding">
<span class="char-enc-value">UTF-16</span> <i class="material-icons">text_fields</i> <span class="chr-enc-value">Raw Bytes</span>
</span> </span>
<div class="cm-status-bar-select-content">
<div class="cm-status-bar-select-scroll no-select">
<a href="#" draggable="false" data-val="0">Raw Bytes</a>
${chrEncOptions}
</div>
<div class="input-group cm-status-bar-filter-search">
<div class="input-group-prepend">
<span class="input-group-text">
<i class="material-icons">search</i>
</span>
</div>
<input type="text" class="form-control cm-status-bar-filter-input" placeholder="Filter...">
</div>
</div>
</div>
<div class="cm-status-bar-select eol-select"> <div class="cm-status-bar-select eol-select">
<span class="cm-status-bar-select-btn" data-toggle="tooltip" data-placement="left" title="End of line sequence"> <span class="cm-status-bar-select-btn" data-toggle="tooltip" data-placement="left" title="End of line sequence">
<i class="material-icons">keyboard_return</i> <span class="eol-value"></span> <i class="material-icons">keyboard_return</i> <span class="eol-value"></span>
</span> </span>
<div class="cm-status-bar-select-content"> <div class="cm-status-bar-select-content no-select">
<a href="#" data-val="LF">Line Feed, U+000A</a> <a href="#" draggable="false" data-val="LF">Line Feed, U+000A</a>
<a href="#" data-val="VT">Vertical Tab, U+000B</a> <a href="#" draggable="false" data-val="VT">Vertical Tab, U+000B</a>
<a href="#" data-val="FF">Form Feed, U+000C</a> <a href="#" draggable="false" data-val="FF">Form Feed, U+000C</a>
<a href="#" data-val="CR">Carriage Return, U+000D</a> <a href="#" draggable="false" data-val="CR">Carriage Return, U+000D</a>
<a href="#" data-val="CRLF">CR+LF, U+000D U+000A</a> <a href="#" draggable="false" data-val="CRLF">CR+LF, U+000D U+000A</a>
<!-- <a href="#" data-val="NL">Next Line, U+0085</a> This causes problems. --> <!-- <a href="#" draggable="false" data-val="NL">Next Line, U+0085</a> This causes problems. -->
<a href="#" data-val="LS">Line Separator, U+2028</a> <a href="#" draggable="false" data-val="LS">Line Separator, U+2028</a>
<a href="#" data-val="PS">Paragraph Separator, U+2029</a> <a href="#" draggable="false" data-val="PS">Paragraph Separator, U+2029</a>
</div> </div>
</div>`; </div>`;
} }
} }
const elementsWithListeners = {};
/**
* Hides the provided element when a click is made outside of it
* @param {Element} element
* @param {Event} instantiatingEvent
*/
function hideOnClickOutside(element, instantiatingEvent) {
/**
* Handler for document click events
* Closes element if click is outside it.
* @param {Event} event
*/
const outsideClickListener = event => {
// Don't trigger if we're clicking inside the element, or if the element
// is not visible, or if this is the same click event that opened it.
if (!element.contains(event.target) &&
event.timeStamp !== instantiatingEvent.timeStamp) {
hideElement(element);
}
};
if (!Object.keys(elementsWithListeners).includes(element)) {
document.addEventListener("click", outsideClickListener);
elementsWithListeners[element] = outsideClickListener;
}
}
/**
* Hides the specified element and removes the click listener for it
* @param {Element} element
*/
function hideElement(element) {
element.classList.remove("show");
document.removeEventListener("click", elementsWithListeners[element]);
delete elementsWithListeners[element];
}
/** /**
* A panel constructor factory building a panel that re-counts the stats every time the document changes. * A panel constructor factory building a panel that re-counts the stats every time the document changes.
* @param {Object} opts * @param {Object} opts
@ -240,7 +393,7 @@ function makePanel(opts) {
return (view) => { return (view) => {
sbPanel.updateEOL(view.state); sbPanel.updateEOL(view.state);
sbPanel.updateCharEnc(view.state); sbPanel.updateCharEnc(opts.initialChrEncVal);
sbPanel.updateBakeStats(); sbPanel.updateBakeStats();
sbPanel.updateStats(view.state.doc); sbPanel.updateStats(view.state.doc);
sbPanel.updateSelection(view.state, false); sbPanel.updateSelection(view.state, false);
@ -250,8 +403,10 @@ function makePanel(opts) {
update(update) { update(update) {
sbPanel.updateEOL(update.state); sbPanel.updateEOL(update.state);
sbPanel.updateSelection(update.state, update.selectionSet); sbPanel.updateSelection(update.state, update.selectionSet);
sbPanel.updateCharEnc(update.state);
sbPanel.updateBakeStats(); sbPanel.updateBakeStats();
if (update.geometryChanged) {
sbPanel.updateSizing(update.view);
}
if (update.docChanged) { if (update.docChanged) {
sbPanel.updateStats(update.state.doc); sbPanel.updateStats(update.state.doc);
} }

View file

@ -10,6 +10,7 @@ import InputWorker from "worker-loader?inline=no-fallback!../workers/InputWorker
import Utils, {debounce} from "../../core/Utils.mjs"; import Utils, {debounce} from "../../core/Utils.mjs";
import {toBase64} from "../../core/lib/Base64.mjs"; import {toBase64} from "../../core/lib/Base64.mjs";
import {isImage} from "../../core/lib/FileType.mjs"; import {isImage} from "../../core/lib/FileType.mjs";
import cptable from "codepage";
import { import {
EditorView, keymap, highlightSpecialChars, drawSelection, rectangularSelection, crosshairCursor, dropCursor EditorView, keymap, highlightSpecialChars, drawSelection, rectangularSelection, crosshairCursor, dropCursor
@ -39,6 +40,7 @@ class InputWaiter {
this.manager = manager; this.manager = manager;
this.inputTextEl = document.getElementById("input-text"); this.inputTextEl = document.getElementById("input-text");
this.inputChrEnc = 0;
this.initEditor(); this.initEditor();
this.inputWorker = null; this.inputWorker = null;
@ -84,7 +86,9 @@ class InputWaiter {
// Custom extensions // Custom extensions
statusBar({ statusBar({
label: "Input", label: "Input",
eolHandler: this.eolChange.bind(this) eolHandler: this.eolChange.bind(this),
chrEncHandler: this.chrEncChange.bind(this),
initialChrEncVal: this.inputChrEnc
}), }),
// Mutable state // Mutable state
@ -122,19 +126,30 @@ class InputWaiter {
/** /**
* Handler for EOL change events * Handler for EOL change events
* Sets the line separator * Sets the line separator
* @param {string} eolVal
*/ */
eolChange(eolval) { eolChange(eolVal) {
const oldInputVal = this.getInput(); const oldInputVal = this.getInput();
// Update the EOL value // Update the EOL value
this.inputEditorView.dispatch({ this.inputEditorView.dispatch({
effects: this.inputEditorConf.eol.reconfigure(EditorState.lineSeparator.of(eolval)) effects: this.inputEditorConf.eol.reconfigure(EditorState.lineSeparator.of(eolVal))
}); });
// Reset the input so that lines are recalculated, preserving the old EOL values // Reset the input so that lines are recalculated, preserving the old EOL values
this.setInput(oldInputVal); this.setInput(oldInputVal);
} }
/**
* Handler for Chr Enc change events
* Sets the input character encoding
* @param {number} chrEncVal
*/
chrEncChange(chrEncVal) {
this.inputChrEnc = chrEncVal;
this.inputChange();
}
/** /**
* Sets word wrap on the input editor * Sets word wrap on the input editor
* @param {boolean} wrap * @param {boolean} wrap
@ -380,7 +395,7 @@ class InputWaiter {
this.showLoadingInfo(r.data, true); this.showLoadingInfo(r.data, true);
break; break;
case "setInput": case "setInput":
this.set(r.data.inputObj, r.data.silent); this.set(r.data.inputNum, r.data.inputObj, r.data.silent);
break; break;
case "inputAdded": case "inputAdded":
this.inputAdded(r.data.changeTab, r.data.inputNum); this.inputAdded(r.data.changeTab, r.data.inputNum);
@ -403,9 +418,6 @@ class InputWaiter {
case "setUrl": case "setUrl":
this.setUrl(r.data); this.setUrl(r.data);
break; break;
case "inputSwitch":
this.manager.output.inputSwitch(r.data);
break;
case "getInput": case "getInput":
case "getInputNums": case "getInputNums":
this.callbacks[r.data.id](r.data); this.callbacks[r.data.id](r.data);
@ -435,22 +447,36 @@ class InputWaiter {
/** /**
* Sets the input in the input area * Sets the input in the input area
* *
* @param {object} inputData - Object containing the input and its metadata * @param {number} inputNum
* @param {number} inputData.inputNum - The unique inputNum for the selected input * @param {Object} inputData - Object containing the input and its metadata
* @param {string | object} inputData.input - The actual input data * @param {string} type
* @param {string} inputData.name - The name of the input file * @param {ArrayBuffer} buffer
* @param {number} inputData.size - The size in bytes of the input file * @param {string} stringSample
* @param {string} inputData.type - The MIME type of the input file * @param {Object} file
* @param {number} inputData.progress - The load progress of the input file * @param {string} file.name
* @param {number} file.size
* @param {string} file.type
* @param {string} status
* @param {number} progress
* @param {boolean} [silent=false] - If false, fires the manager statechange event * @param {boolean} [silent=false] - If false, fires the manager statechange event
*/ */
async set(inputData, silent=false) { async set(inputNum, inputData, silent=false) {
return new Promise(function(resolve, reject) { return new Promise(function(resolve, reject) {
const activeTab = this.manager.tabs.getActiveInputTab(); const activeTab = this.manager.tabs.getActiveInputTab();
if (inputData.inputNum !== activeTab) return; if (inputNum !== activeTab) return;
if (typeof inputData.input === "string") { if (inputData.file) {
this.setInput(inputData.input); this.setFile(inputNum, inputData, silent);
} else {
// TODO Per-tab encodings?
let inputVal;
if (this.inputChrEnc > 0) {
inputVal = cptable.utils.decode(this.inputChrEnc, new Uint8Array(inputData.buffer));
} else {
inputVal = Utils.arrayBufferToStr(inputData.buffer);
}
this.setInput(inputVal);
const fileOverlay = document.getElementById("input-file"), const fileOverlay = document.getElementById("input-file"),
fileName = document.getElementById("input-file-name"), fileName = document.getElementById("input-file-name"),
fileSize = document.getElementById("input-file-size"), fileSize = document.getElementById("input-file-size"),
@ -466,8 +492,8 @@ class InputWaiter {
this.inputTextEl.classList.remove("blur"); this.inputTextEl.classList.remove("blur");
// Set URL to current input // Set URL to current input
const inputStr = toBase64(inputData.input, "A-Za-z0-9+/"); if (inputVal.length >= 0 && inputVal.length <= 51200) {
if (inputStr.length >= 0 && inputStr.length <= 68267) { const inputStr = toBase64(inputVal, "A-Za-z0-9+/");
this.setUrl({ this.setUrl({
includeInput: true, includeInput: true,
input: inputStr input: inputStr
@ -475,8 +501,6 @@ class InputWaiter {
} }
if (!silent) window.dispatchEvent(this.manager.statechange); if (!silent) window.dispatchEvent(this.manager.statechange);
} else {
this.setFile(inputData, silent);
} }
}.bind(this)); }.bind(this));
@ -485,18 +509,22 @@ class InputWaiter {
/** /**
* Displays file details * Displays file details
* *
* @param {object} inputData - Object containing the input and its metadata * @param {number} inputNum
* @param {number} inputData.inputNum - The unique inputNum for the selected input * @param {Object} inputData - Object containing the input and its metadata
* @param {string | object} inputData.input - The actual input data * @param {string} type
* @param {string} inputData.name - The name of the input file * @param {ArrayBuffer} buffer
* @param {number} inputData.size - The size in bytes of the input file * @param {string} stringSample
* @param {string} inputData.type - The MIME type of the input file * @param {Object} file
* @param {number} inputData.progress - The load progress of the input file * @param {string} file.name
* @param {number} file.size
* @param {string} file.type
* @param {string} status
* @param {number} progress
* @param {boolean} [silent=true] - If false, fires the manager statechange event * @param {boolean} [silent=true] - If false, fires the manager statechange event
*/ */
setFile(inputData, silent=true) { setFile(inputNum, inputData, silent=true) {
const activeTab = this.manager.tabs.getActiveInputTab(); const activeTab = this.manager.tabs.getActiveInputTab();
if (inputData.inputNum !== activeTab) return; if (inputNum !== activeTab) return;
const fileOverlay = document.getElementById("input-file"), const fileOverlay = document.getElementById("input-file"),
fileName = document.getElementById("input-file-name"), fileName = document.getElementById("input-file-name"),
@ -505,9 +533,9 @@ class InputWaiter {
fileLoaded = document.getElementById("input-file-loaded"); fileLoaded = document.getElementById("input-file-loaded");
fileOverlay.style.display = "block"; fileOverlay.style.display = "block";
fileName.textContent = inputData.name; fileName.textContent = inputData.file.name;
fileSize.textContent = inputData.size + " bytes"; fileSize.textContent = inputData.file.size + " bytes";
fileType.textContent = inputData.type; fileType.textContent = inputData.file.type;
if (inputData.status === "error") { if (inputData.status === "error") {
fileLoaded.textContent = "Error"; fileLoaded.textContent = "Error";
fileLoaded.style.color = "#FF0000"; fileLoaded.style.color = "#FF0000";
@ -516,7 +544,7 @@ class InputWaiter {
fileLoaded.textContent = inputData.progress + "%"; fileLoaded.textContent = inputData.progress + "%";
} }
this.displayFilePreview(inputData); this.displayFilePreview(inputNum, inputData);
if (!silent) window.dispatchEvent(this.manager.statechange); if (!silent) window.dispatchEvent(this.manager.statechange);
} }
@ -583,19 +611,18 @@ class InputWaiter {
/** /**
* Shows a chunk of the file in the input behind the file overlay * Shows a chunk of the file in the input behind the file overlay
* *
* @param {number} inputNum - The inputNum of the file being displayed
* @param {Object} inputData - Object containing the input data * @param {Object} inputData - Object containing the input data
* @param {number} inputData.inputNum - The inputNum of the file being displayed * @param {string} inputData.stringSample - The first 4096 bytes of input as a string
* @param {ArrayBuffer} inputData.input - The actual input to display
*/ */
displayFilePreview(inputData) { displayFilePreview(inputNum, inputData) {
const activeTab = this.manager.tabs.getActiveInputTab(), const activeTab = this.manager.tabs.getActiveInputTab(),
input = inputData.input; input = inputData.buffer;
if (inputData.inputNum !== activeTab) return; if (inputNum !== activeTab) return;
this.inputTextEl.classList.add("blur"); this.inputTextEl.classList.add("blur");
this.setInput(Utils.arrayBufferToStr(input.slice(0, 4096))); this.setInput(input.stringSample);
this.renderFileThumb(); this.renderFileThumb();
} }
/** /**
@ -623,46 +650,40 @@ class InputWaiter {
* *
* @param {number} inputNum * @param {number} inputNum
* @param {string | ArrayBuffer} value * @param {string | ArrayBuffer} value
* @param {boolean} [force=false] - If true, forces the value to be updated even if the type is different to the currently stored type
*/ */
updateInputValue(inputNum, value, force=false) { updateInputValue(inputNum, value, force=false) {
let includeInput = false; // Prepare the value as a buffer (full value) and a string sample (up to 4096 bytes)
const recipeStr = toBase64(value, "A-Za-z0-9+/"); // B64 alphabet with no padding let buffer;
if (recipeStr.length > 0 && recipeStr.length <= 68267) { let stringSample = "";
includeInput = true;
// If value is a string, interpret it using the specified character encoding
if (typeof value === "string") {
stringSample = value.slice(0, 4096);
if (this.inputChrEnc > 0) {
buffer = cptable.utils.encode(this.inputChrEnc, value);
buffer = new Uint8Array(buffer).buffer;
} else {
buffer = Utils.strToArrayBuffer(value);
} }
} else {
buffer = value;
stringSample = Utils.arrayBufferToStr(value.slice(0, 4096));
}
const recipeStr = buffer.byteLength < 51200 ? toBase64(buffer, "A-Za-z0-9+/") : ""; // B64 alphabet with no padding
this.setUrl({ this.setUrl({
includeInput: includeInput, includeInput: recipeStr.length > 0 && buffer.byteLength < 51200,
input: recipeStr input: recipeStr
}); });
// Value is either a string set by the input or an ArrayBuffer from a LoaderWorker, const transferable = [buffer];
// so is safe to use typeof === "string"
const transferable = (typeof value !== "string") ? [value] : undefined;
this.inputWorker.postMessage({ this.inputWorker.postMessage({
action: "updateInputValue", action: "updateInputValue",
data: { data: {
inputNum: inputNum, inputNum: inputNum,
value: value, buffer: buffer,
force: force stringSample: stringSample
}
}, transferable);
}
/**
* Updates the .data property for the input of the specified inputNum.
* Used for switching the output into the input
*
* @param {number} inputNum - The inputNum of the input we're changing
* @param {object} inputData - The new data object
*/
updateInputObj(inputNum, inputData) {
const transferable = (typeof inputData !== "string") ? [inputData.fileBuffer] : undefined;
this.inputWorker.postMessage({
action: "updateInputObj",
data: {
inputNum: inputNum,
data: inputData
} }
}, transferable); }, transferable);
} }
@ -1052,9 +1073,8 @@ class InputWaiter {
this.updateInputValue(inputNum, "", true); this.updateInputValue(inputNum, "", true);
this.set({ this.set(inputNum, {
inputNum: inputNum, buffer: new ArrayBuffer()
input: ""
}); });
this.manager.tabs.updateInputTabHeader(inputNum, ""); this.manager.tabs.updateInputTabHeader(inputNum, "");

View file

@ -9,6 +9,7 @@ import Utils, {debounce} from "../../core/Utils.mjs";
import Dish from "../../core/Dish.mjs"; import Dish from "../../core/Dish.mjs";
import FileSaver from "file-saver"; import FileSaver from "file-saver";
import ZipWorker from "worker-loader?inline=no-fallback!../workers/ZipWorker.mjs"; import ZipWorker from "worker-loader?inline=no-fallback!../workers/ZipWorker.mjs";
import cptable from "codepage";
import { import {
EditorView, keymap, highlightSpecialChars, drawSelection, rectangularSelection, crosshairCursor EditorView, keymap, highlightSpecialChars, drawSelection, rectangularSelection, crosshairCursor
@ -48,6 +49,7 @@ class OutputWaiter {
html: "", html: "",
changed: false changed: false
}; };
this.outputChrEnc = 0;
this.initEditor(); this.initEditor();
this.outputs = {}; this.outputs = {};
@ -86,7 +88,9 @@ class OutputWaiter {
statusBar({ statusBar({
label: "Output", label: "Output",
bakeStats: this.bakeStats, bakeStats: this.bakeStats,
eolHandler: this.eolChange.bind(this) eolHandler: this.eolChange.bind(this),
chrEncHandler: this.chrEncChange.bind(this),
initialChrEncVal: this.outputChrEnc
}), }),
htmlPlugin(this.htmlOutput), htmlPlugin(this.htmlOutput),
copyOverride(), copyOverride(),
@ -119,19 +123,29 @@ class OutputWaiter {
/** /**
* Handler for EOL change events * Handler for EOL change events
* Sets the line separator * Sets the line separator
* @param {string} eolVal
*/ */
eolChange(eolval) { eolChange(eolVal) {
const oldOutputVal = this.getOutput(); const oldOutputVal = this.getOutput();
// Update the EOL value // Update the EOL value
this.outputEditorView.dispatch({ this.outputEditorView.dispatch({
effects: this.outputEditorConf.eol.reconfigure(EditorState.lineSeparator.of(eolval)) effects: this.outputEditorConf.eol.reconfigure(EditorState.lineSeparator.of(eolVal))
}); });
// Reset the output so that lines are recalculated, preserving the old EOL values // Reset the output so that lines are recalculated, preserving the old EOL values
this.setOutput(oldOutputVal); this.setOutput(oldOutputVal);
} }
/**
* Handler for Chr Enc change events
* Sets the output character encoding
* @param {number} chrEncVal
*/
chrEncChange(chrEncVal) {
this.outputChrEnc = chrEncVal;
}
/** /**
* Sets word wrap on the output editor * Sets word wrap on the output editor
* @param {boolean} wrap * @param {boolean} wrap
@ -193,7 +207,8 @@ class OutputWaiter {
}); });
// Execute script sections // Execute script sections
const scriptElements = document.getElementById("output-html").querySelectorAll("script"); const outputHTML = document.getElementById("output-html");
const scriptElements = outputHTML ? outputHTML.querySelectorAll("script") : [];
for (let i = 0; i < scriptElements.length; i++) { for (let i = 0; i < scriptElements.length; i++) {
try { try {
eval(scriptElements[i].innerHTML); // eslint-disable-line no-eval eval(scriptElements[i].innerHTML); // eslint-disable-line no-eval
@ -405,8 +420,6 @@ class OutputWaiter {
removeAllOutputs() { removeAllOutputs() {
this.outputs = {}; this.outputs = {};
this.resetSwitch();
const tabsList = document.getElementById("output-tabs"); const tabsList = document.getElementById("output-tabs");
const tabsListChildren = tabsList.children; const tabsListChildren = tabsList.children;
@ -418,19 +431,18 @@ class OutputWaiter {
} }
/** /**
* Sets the output in the output textarea. * Sets the output in the output pane.
* *
* @param {number} inputNum * @param {number} inputNum
*/ */
async set(inputNum) { async set(inputNum) {
inputNum = parseInt(inputNum, 10);
if (inputNum !== this.manager.tabs.getActiveOutputTab() || if (inputNum !== this.manager.tabs.getActiveOutputTab() ||
!this.outputExists(inputNum)) return; !this.outputExists(inputNum)) return;
this.toggleLoader(true); this.toggleLoader(true);
return new Promise(async function(resolve, reject) { return new Promise(async function(resolve, reject) {
const output = this.outputs[inputNum], const output = this.outputs[inputNum];
activeTab = this.manager.tabs.getActiveOutputTab();
if (typeof inputNum !== "number") inputNum = parseInt(inputNum, 10);
const outputFile = document.getElementById("output-file"); const outputFile = document.getElementById("output-file");
@ -491,17 +503,33 @@ class OutputWaiter {
switch (output.data.type) { switch (output.data.type) {
case "html": case "html":
outputFile.style.display = "none"; outputFile.style.display = "none";
// TODO what if the HTML content needs to be in a certain character encoding?
// Grey out chr enc selection? Set back to Raw Bytes?
this.setHTMLOutput(output.data.result); this.setHTMLOutput(output.data.result);
break; break;
case "ArrayBuffer": case "ArrayBuffer": {
this.outputTextEl.style.display = "block"; this.outputTextEl.style.display = "block";
outputFile.style.display = "none";
this.clearHTMLOutput(); this.clearHTMLOutput();
this.setOutput("");
this.setFile(await this.getDishBuffer(output.data.dish), activeTab); let outputVal = "";
if (this.outputChrEnc === 0) {
outputVal = Utils.arrayBufferToStr(output.data.result);
} else {
try {
outputVal = cptable.utils.decode(this.outputChrEnc, new Uint8Array(output.data.result));
} catch (err) {
outputVal = err;
}
}
this.setOutput(outputVal);
// this.setFile(await this.getDishBuffer(output.data.dish), activeTab);
break; break;
}
case "string": case "string":
default: default:
this.outputTextEl.style.display = "block"; this.outputTextEl.style.display = "block";
@ -1333,7 +1361,6 @@ class OutputWaiter {
*/ */
async switchClick() { async switchClick() {
const activeTab = this.manager.tabs.getActiveOutputTab(); const activeTab = this.manager.tabs.getActiveOutputTab();
const transferable = [];
const switchButton = document.getElementById("switch"); const switchButton = document.getElementById("switch");
switchButton.classList.add("spin"); switchButton.classList.add("spin");
@ -1341,82 +1368,15 @@ class OutputWaiter {
switchButton.firstElementChild.innerHTML = "autorenew"; switchButton.firstElementChild.innerHTML = "autorenew";
$(switchButton).tooltip("hide"); $(switchButton).tooltip("hide");
let active = await this.getDishBuffer(this.getOutputDish(activeTab)); const activeData = await this.getDishBuffer(this.getOutputDish(activeTab));
if (!this.outputExists(activeTab)) { if (this.outputExists(activeTab)) {
this.resetSwitchButton(); this.manager.input.set({
return;
}
if (this.outputs[activeTab].data.type === "string" &&
active.byteLength <= this.app.options.ioDisplayThreshold * 1024) {
const dishString = await this.getDishStr(this.getOutputDish(activeTab));
active = dishString;
} else {
transferable.push(active);
}
this.manager.input.inputWorker.postMessage({
action: "inputSwitch",
data: {
inputNum: activeTab, inputNum: activeTab,
outputData: active input: activeData
} });
}, transferable);
} }
/**
* Handler for when the inputWorker has switched the inputs.
* Stores the old input
*
* @param {object} switchData
* @param {number} switchData.inputNum
* @param {string | object} switchData.data
* @param {ArrayBuffer} switchData.data.fileBuffer
* @param {number} switchData.data.size
* @param {string} switchData.data.type
* @param {string} switchData.data.name
*/
inputSwitch(switchData) {
this.switchOrigData = switchData;
document.getElementById("undo-switch").disabled = false;
this.resetSwitchButton();
}
/**
* Handler for undo switch click events.
* Removes the output from the input and replaces the input that was removed.
*/
undoSwitchClick() {
this.manager.input.updateInputObj(this.switchOrigData.inputNum, this.switchOrigData.data);
this.manager.input.fileLoaded(this.switchOrigData.inputNum);
this.resetSwitch();
}
/**
* Removes the switch data and resets the switch buttons
*/
resetSwitch() {
if (this.switchOrigData !== undefined) {
delete this.switchOrigData;
}
const undoSwitch = document.getElementById("undo-switch");
undoSwitch.disabled = true;
$(undoSwitch).tooltip("hide");
this.resetSwitchButton();
}
/**
* Resets the switch button to its usual state
*/
resetSwitchButton() {
const switchButton = document.getElementById("switch");
switchButton.classList.remove("spin"); switchButton.classList.remove("spin");
switchButton.disabled = false; switchButton.disabled = false;
switchButton.firstElementChild.innerHTML = "open_in_browser"; switchButton.firstElementChild.innerHTML = "open_in_browser";

View file

@ -3,12 +3,12 @@
* Handles storage, modification and retrieval of the inputs. * Handles storage, modification and retrieval of the inputs.
* *
* @author j433866 [j433866@gmail.com] * @author j433866 [j433866@gmail.com]
* @author n1474335 [n1474335@gmail.com]
* @copyright Crown Copyright 2019 * @copyright Crown Copyright 2019
* @license Apache-2.0 * @license Apache-2.0
*/ */
import Utils from "../../core/Utils.mjs"; import Utils from "../../core/Utils.mjs";
import {detectFileType} from "../../core/lib/FileType.mjs";
// Default max values // Default max values
// These will be correctly calculated automatically // These will be correctly calculated automatically
@ -16,6 +16,21 @@ self.maxWorkers = 4;
self.maxTabs = 1; self.maxTabs = 1;
self.pendingFiles = []; self.pendingFiles = [];
/**
* Dictionary of inputs keyed on the inputNum
* Each entry is an object with the following type:
* @typedef {Object} Input
* @property {string} type
* @property {ArrayBuffer} buffer
* @property {string} stringSample
* @property {Object} file
* @property {string} file.name
* @property {number} file.size
* @property {string} file.type
* @property {string} status
* @property {number} progress
*/
self.inputs = {}; self.inputs = {};
self.loaderWorkers = []; self.loaderWorkers = [];
self.currentInputNum = 1; self.currentInputNum = 1;
@ -53,9 +68,6 @@ self.addEventListener("message", function(e) {
case "updateInputValue": case "updateInputValue":
self.updateInputValue(r.data); self.updateInputValue(r.data);
break; break;
case "updateInputObj":
self.updateInputObj(r.data);
break;
case "updateInputProgress": case "updateInputProgress":
self.updateInputProgress(r.data); self.updateInputProgress(r.data);
break; break;
@ -75,7 +87,7 @@ self.addEventListener("message", function(e) {
log.setLevel(r.data, false); log.setLevel(r.data, false);
break; break;
case "addInput": case "addInput":
self.addInput(r.data, "string"); self.addInput(r.data, "userinput");
break; break;
case "refreshTabs": case "refreshTabs":
self.refreshTabs(r.data.inputNum, r.data.direction); self.refreshTabs(r.data.inputNum, r.data.direction);
@ -98,9 +110,6 @@ self.addEventListener("message", function(e) {
case "loaderWorkerMessage": case "loaderWorkerMessage":
self.handleLoaderMessage(r.data); self.handleLoaderMessage(r.data);
break; break;
case "inputSwitch":
self.inputSwitch(r.data);
break;
case "updateTabHeader": case "updateTabHeader":
self.updateTabHeader(r.data); self.updateTabHeader(r.data);
break; break;
@ -213,13 +222,10 @@ self.bakeInput = function(inputNum, bakeId) {
return; return;
} }
let inputData = inputObj.data;
if (typeof inputData !== "string") inputData = inputData.fileBuffer;
self.postMessage({ self.postMessage({
action: "queueInput", action: "queueInput",
data: { data: {
input: inputData, input: inputObj.buffer,
inputNum: inputNum, inputNum: inputNum,
bakeId: bakeId bakeId: bakeId
} }
@ -236,23 +242,6 @@ self.getInputObj = function(inputNum) {
return self.inputs[inputNum]; return self.inputs[inputNum];
}; };
/**
* Gets the stored value for a specific inputNum.
*
* @param {number} inputNum - The input we want to get the value of
* @returns {string | ArrayBuffer}
*/
self.getInputValue = function(inputNum) {
if (self.inputs[inputNum]) {
if (typeof self.inputs[inputNum].data === "string") {
return self.inputs[inputNum].data;
} else {
return self.inputs[inputNum].data.fileBuffer;
}
}
return "";
};
/** /**
* Gets the stored value or object for a specific inputNum and sends it to the inputWaiter. * Gets the stored value or object for a specific inputNum and sends it to the inputWaiter.
* *
@ -263,7 +252,7 @@ self.getInputValue = function(inputNum) {
*/ */
self.getInput = function(inputData) { self.getInput = function(inputData) {
const inputNum = inputData.inputNum, const inputNum = inputData.inputNum,
data = (inputData.getObj) ? self.getInputObj(inputNum) : self.getInputValue(inputNum); data = (inputData.getObj) ? self.getInputObj(inputNum) : self.inputs[inputNum].buffer;
self.postMessage({ self.postMessage({
action: "getInput", action: "getInput",
data: { data: {
@ -421,17 +410,15 @@ self.getNearbyNums = function(inputNum, direction) {
self.updateTabHeader = function(inputNum) { self.updateTabHeader = function(inputNum) {
const input = self.getInputObj(inputNum); const input = self.getInputObj(inputNum);
if (input === null || input === undefined) return; if (input === null || input === undefined) return;
let inputData = input.data;
if (typeof inputData !== "string") { let header = input.type === "file" ? input.file.name : input.stringSample;
inputData = input.data.name; header = header.slice(0, 100).replace(/[\n\r]/g, "");
}
inputData = inputData.replace(/[\n\r]/g, "");
self.postMessage({ self.postMessage({
action: "updateTabHeader", action: "updateTabHeader",
data: { data: {
inputNum: inputNum, inputNum: inputNum,
input: inputData.slice(0, 100) input: header
} }
}); });
}; };
@ -450,37 +437,15 @@ self.setInput = function(inputData) {
const input = self.getInputObj(inputNum); const input = self.getInputObj(inputNum);
if (input === undefined || input === null) return; if (input === undefined || input === null) return;
let inputVal = input.data; self.postMessage({
const inputObj = { action: "setInput",
data: {
inputNum: inputNum, inputNum: inputNum,
input: inputVal inputObj: input,
};
if (typeof inputVal !== "string") {
inputObj.name = inputVal.name;
inputObj.size = inputVal.size;
inputObj.type = inputVal.type;
inputObj.progress = input.progress;
inputObj.status = input.status;
inputVal = inputVal.fileBuffer;
const fileSlice = inputVal.slice(0, 512001);
inputObj.input = fileSlice;
self.postMessage({
action: "setInput",
data: {
inputObj: inputObj,
silent: silent
}
}, [fileSlice]);
} else {
self.postMessage({
action: "setInput",
data: {
inputObj: inputObj,
silent: silent silent: silent
} }
}); });
}
self.updateTabHeader(inputNum); self.updateTabHeader(inputNum);
}; };
@ -546,54 +511,23 @@ self.updateInputProgress = function(inputData) {
* *
* @param {object} inputData * @param {object} inputData
* @param {number} inputData.inputNum - The input that's having its value updated * @param {number} inputData.inputNum - The input that's having its value updated
* @param {string | ArrayBuffer} inputData.value - The new value of the input * @param {ArrayBuffer} inputData.buffer - The new value of the input as a buffer
* @param {boolean} inputData.force - If true, still updates the input value if the input type is different to the stored value * @param {string} [inputData.stringSample] - A sample of the value as a string (truncated to 4096 chars)
*/ */
self.updateInputValue = function(inputData) { self.updateInputValue = function(inputData) {
const inputNum = inputData.inputNum; const inputNum = parseInt(inputData.inputNum, 10);
if (inputNum < 1) return; if (inputNum < 1) return;
if (Object.prototype.hasOwnProperty.call(self.inputs[inputNum].data, "fileBuffer") &&
typeof inputData.value === "string" && !inputData.force) return; if (!Object.prototype.hasOwnProperty.call(self.inputs, inputNum))
const value = inputData.value; throw new Error(`No input with ID ${inputNum} exists`);
if (self.inputs[inputNum] !== undefined) {
if (typeof value === "string") { self.inputs[inputNum].buffer = inputData.buffer;
self.inputs[inputNum].data = value; if (!("stringSample" in inputData)) {
} else { inputData.stringSample = Utils.arrayBufferToStr(inputData.buffer.slice(0, 4096));
self.inputs[inputNum].data.fileBuffer = value;
} }
self.inputs[inputNum].stringSample = inputData.stringSample;
self.inputs[inputNum].status = "loaded"; self.inputs[inputNum].status = "loaded";
self.inputs[inputNum].progress = 100; self.inputs[inputNum].progress = 100;
return;
}
// If we get to here, an input for inputNum could not be found,
// so create a new one. Only do this if the value is a string, as
// loadFiles will create the input object for files
if (typeof value === "string") {
self.inputs.push({
inputNum: inputNum,
data: value,
status: "loaded",
progress: 100
});
}
};
/**
* Update the stored data object for an input.
* Used if we need to change a string to an ArrayBuffer
*
* @param {object} inputData
* @param {number} inputData.inputNum - The number of the input we're updating
* @param {object} inputData.data - The new data object for the input
*/
self.updateInputObj = function(inputData) {
const inputNum = inputData.inputNum;
const data = inputData.data;
if (self.getInputObj(inputNum) === undefined) return;
self.inputs[inputNum].data = data;
}; };
/** /**
@ -632,8 +566,7 @@ self.loaderWorkerReady = function(workerData) {
/** /**
* Handler for messages sent by loaderWorkers. * Handler for messages sent by loaderWorkers.
* (Messages are sent between the inputWorker and * (Messages are sent between the inputWorker and loaderWorkers via the main thread)
* loaderWorkers via the main thread)
* *
* @param {object} r - The data sent by the loaderWorker * @param {object} r - The data sent by the loaderWorker
* @param {number} r.inputNum - The inputNum which the message corresponds to * @param {number} r.inputNum - The inputNum which the message corresponds to
@ -667,7 +600,7 @@ self.handleLoaderMessage = function(r) {
self.updateInputValue({ self.updateInputValue({
inputNum: inputNum, inputNum: inputNum,
value: r.fileBuffer buffer: r.fileBuffer
}); });
self.postMessage({ self.postMessage({
@ -757,7 +690,8 @@ self.loadFiles = function(filesData) {
let lastInputNum = -1; let lastInputNum = -1;
const inputNums = []; const inputNums = [];
for (let i = 0; i < files.length; i++) { for (let i = 0; i < files.length; i++) {
if (i === 0 && self.getInputValue(activeTab) === "") { // If the first input is empty, replace it rather than adding a new one
if (i === 0 && (!self.inputs[activeTab].buffer || self.inputs[activeTab].buffer.byteLength === 0)) {
self.removeInput({ self.removeInput({
inputNum: activeTab, inputNum: activeTab,
refreshTabs: false, refreshTabs: false,
@ -798,7 +732,7 @@ self.loadFiles = function(filesData) {
* Adds an input to the input dictionary * Adds an input to the input dictionary
* *
* @param {boolean} [changetab=false] - Whether or not to change to the new input * @param {boolean} [changetab=false] - Whether or not to change to the new input
* @param {string} type - Either "string" or "file" * @param {string} type - Either "userinput" or "file"
* @param {Object} fileData - Contains information about the file to be added to the input (only used when type is "file") * @param {Object} fileData - Contains information about the file to be added to the input (only used when type is "file")
* @param {string} fileData.name - The filename of the input being added * @param {string} fileData.name - The filename of the input being added
* @param {number} fileData.size - The file size (in bytes) of the input being added * @param {number} fileData.size - The file size (in bytes) of the input being added
@ -810,25 +744,30 @@ self.addInput = function(
type, type,
fileData = { fileData = {
name: "unknown", name: "unknown",
size: "unknown", size: 0,
type: "unknown" type: "unknown"
}, },
inputNum = self.currentInputNum++ inputNum = self.currentInputNum++
) { ) {
self.numInputs++; self.numInputs++;
const newInputObj = { const newInputObj = {
inputNum: inputNum type: null,
buffer: new ArrayBuffer(),
stringSample: "",
file: null,
status: "pending",
progress: 0
}; };
switch (type) { switch (type) {
case "string": case "userinput":
newInputObj.data = ""; newInputObj.type = "userinput";
newInputObj.status = "loaded"; newInputObj.status = "loaded";
newInputObj.progress = 100; newInputObj.progress = 100;
break; break;
case "file": case "file":
newInputObj.data = { newInputObj.type = "file";
fileBuffer: new ArrayBuffer(), newInputObj.file = {
name: fileData.name, name: fileData.name,
size: fileData.size, size: fileData.size,
type: fileData.type type: fileData.type
@ -837,7 +776,7 @@ self.addInput = function(
newInputObj.progress = 0; newInputObj.progress = 0;
break; break;
default: default:
log.error(`Invalid type '${type}'.`); log.error(`Invalid input type '${type}'.`);
return -1; return -1;
} }
self.inputs[inputNum] = newInputObj; self.inputs[inputNum] = newInputObj;
@ -976,18 +915,18 @@ self.filterTabs = function(searchData) {
self.inputs[iNum].status === "loading" && showLoading || self.inputs[iNum].status === "loading" && showLoading ||
self.inputs[iNum].status === "loaded" && showLoaded) { self.inputs[iNum].status === "loaded" && showLoaded) {
try { try {
if (typeof self.inputs[iNum].data === "string") { if (self.inputs[iNum].type === "userinput") {
if (filterType.toLowerCase() === "content" && if (filterType.toLowerCase() === "content" &&
filterExp.test(self.inputs[iNum].data.slice(0, 4096))) { filterExp.test(self.inputs[iNum].stringSample)) {
textDisplay = self.inputs[iNum].data.slice(0, 4096); textDisplay = self.inputs[iNum].stringSample;
addInput = true; addInput = true;
} }
} else { } else {
if ((filterType.toLowerCase() === "filename" && if ((filterType.toLowerCase() === "filename" &&
filterExp.test(self.inputs[iNum].data.name)) || filterExp.test(self.inputs[iNum].file.name)) ||
filterType.toLowerCase() === "content" && (filterType.toLowerCase() === "content" &&
filterExp.test(Utils.arrayBufferToStr(self.inputs[iNum].data.fileBuffer.slice(0, 4096)))) { filterExp.test(self.inputs[iNum].stringSample))) {
textDisplay = self.inputs[iNum].data.name; textDisplay = self.inputs[iNum].file.name;
addInput = true; addInput = true;
} }
} }
@ -1021,61 +960,3 @@ self.filterTabs = function(searchData) {
data: inputs data: inputs
}); });
}; };
/**
* Swaps the input and outputs, and sends the old input back to the main thread.
*
* @param {object} switchData
* @param {number} switchData.inputNum - The inputNum of the input to be switched to
* @param {string | ArrayBuffer} switchData.outputData - The data to switch to
*/
self.inputSwitch = function(switchData) {
const currentInput = self.getInputObj(switchData.inputNum);
const currentData = currentInput.data;
if (currentInput === undefined || currentInput === null) return;
if (typeof switchData.outputData !== "string") {
const output = new Uint8Array(switchData.outputData),
types = detectFileType(output);
let type = "unknown",
ext = "dat";
if (types.length) {
type = types[0].mime;
ext = types[0].extension.split(",", 1)[0];
}
// ArrayBuffer
self.updateInputObj({
inputNum: switchData.inputNum,
data: {
fileBuffer: switchData.outputData,
name: `output.${ext}`,
size: switchData.outputData.byteLength.toLocaleString(),
type: type
}
});
} else {
// String
self.updateInputValue({
inputNum: switchData.inputNum,
value: switchData.outputData,
force: true
});
}
self.postMessage({
action: "inputSwitch",
data: {
data: currentData,
inputNum: switchData.inputNum
}
});
self.postMessage({
action: "fileLoaded",
data: {
inputNum: switchData.inputNum
}
});
};