CyberChef/src/web/waiters/RecipeWaiter.mjs

426 lines
13 KiB
JavaScript
Raw Normal View History

2018-05-15 17:36:45 +00:00
/**
* @author n1474335 [n1474335@gmail.com]
* @copyright Crown Copyright 2016
* @license Apache-2.0
*/
import Sortable from "sortablejs";
import Utils from "../../core/Utils.mjs";
import {escapeControlChars} from "../utils/editorUtils.mjs";
import {CRecipeLi} from "../components/c-recipe-li.mjs";
2018-05-15 17:36:45 +00:00
/**
* Waiter to handle events related to the recipe.
*/
class RecipeWaiter {
/**
* RecipeWaiter constructor.
*
* @param {App} app - The main view object for CyberChef.
* @param {Manager} manager - The CyberChef event manager.
*/
constructor(app, manager) {
this.app = app;
this.manager = manager;
this.removeIntent = false;
}
/**
* Sets up the drag and drop capability for recipe-list
2018-05-15 17:36:45 +00:00
*/
2023-08-02 17:38:52 +12:00
initDragAndDrop() {
2018-05-15 17:36:45 +00:00
const recList = document.getElementById("rec-list");
2023-08-14 14:30:25 +12:00
let swapThreshold, animation, delay;
// tweak these values for better user experiences per device type and UI
if (this.app.isMobileView()) {
2023-08-16 12:55:41 +12:00
swapThreshold = 0.30;
animation = 300;
delay = 50;
2023-08-14 14:30:25 +12:00
} else {
2023-08-16 12:55:41 +12:00
swapThreshold = 1;
animation = 100;
2023-08-14 14:30:25 +12:00
delay = 0;
}
2018-05-15 17:36:45 +00:00
// Recipe list
Sortable.create(recList, {
group: "recipe",
sort: true,
draggable: "c-recipe-li",
swapThreshold: swapThreshold,
animation: animation,
delay: delay,
filter: ".arg",
2018-05-15 17:36:45 +00:00
preventOnFilter: false,
setData: function(dataTransfer, dragEl) {
2023-08-02 17:38:52 +12:00
dataTransfer.setData("Text", dragEl.querySelector("li").getAttribute("data-name"));
2018-05-15 17:36:45 +00:00
},
2023-08-02 17:38:52 +12:00
onEnd: function(e) {
2018-05-15 17:36:45 +00:00
if (this.removeIntent) {
2023-08-02 17:38:52 +12:00
e.item.remove();
e.target.dispatchEvent(this.manager.operationremove);
2018-05-15 17:36:45 +00:00
}
}.bind(this),
2023-08-02 17:38:52 +12:00
onSort: function(e) {
if (e.from.id === "rec-list") {
2018-05-15 17:36:45 +00:00
document.dispatchEvent(this.manager.statechange);
}
}.bind(this)
});
Sortable.utils.on(recList, "dragover", function() {
this.removeIntent = false;
}.bind(this));
Sortable.utils.on(recList, "dragleave", function() {
this.removeIntent = true;
this.app.progress = 0;
}.bind(this));
Sortable.utils.on(recList, "touchend", function(e) {
const loc = e.changedTouches[0];
const target = document.elementFromPoint(loc.clientX, loc.clientY);
this.removeIntent = !recList.contains(target);
}.bind(this));
// Favourites category
document.querySelector("#categories a").addEventListener("dragover", this.favDragover.bind(this));
document.querySelector("#categories a").addEventListener("dragleave", this.favDragleave.bind(this));
document.querySelector("#categories a").addEventListener("drop", this.favDrop.bind(this));
}
/**
* Handler for favourite dragover events.
* If the element being dragged is an operation, displays a visual cue so that the user knows it can
* be dropped here.
*
* @param {Event} e
2018-05-15 17:36:45 +00:00
*/
favDragover(e) {
if (e.dataTransfer.effectAllowed !== "move")
return false;
e.stopPropagation();
e.preventDefault();
if (e.target?.className?.indexOf("category-title") > -1) {
2018-05-15 17:36:45 +00:00
// Hovering over the a
e.target.classList.add("favourites-hover");
} else if (e.target?.parentNode?.className?.indexOf("category-title") > -1) {
2018-05-15 17:36:45 +00:00
// Hovering over the Edit button
e.target.parentNode.classList.add("favourites-hover");
} else if (e.target?.parentNode?.parentNode?.className?.indexOf("category-title") > -1) {
2018-05-15 17:36:45 +00:00
// Hovering over the image on the Edit button
e.target.parentNode.parentNode.classList.add("favourites-hover");
}
}
/**
* Handler for favourite dragleave events.
* Removes the visual cue.
*
* @param {Event} e
2018-05-15 17:36:45 +00:00
*/
favDragleave(e) {
e.stopPropagation();
e.preventDefault();
document.querySelector("#categories a").classList.remove("favourites-hover");
}
/**
* Handler for favourite drop events.
* Adds the dragged operation to the favourites list.
*
* @param {Event} e
2018-05-15 17:36:45 +00:00
*/
favDrop(e) {
e.stopPropagation();
e.preventDefault();
e.target.classList.remove("favourites-hover");
const opName = e.dataTransfer.getData("Text");
this.app.addFavourite(opName);
}
/**
* Handler for ingredient change events.
*
* @fires Manager#statechange
*/
ingChange(e) {
if (e && e?.target?.classList?.contains("no-state-change")) return;
2018-05-15 17:36:45 +00:00
window.dispatchEvent(this.manager.statechange);
}
/**
* Generates a configuration object to represent the current recipe.
*
* @returns {Object[]} recipeConfig - The recipe configuration
2018-05-15 17:36:45 +00:00
*/
getConfig() {
const config = [];
2023-08-15 15:50:10 +12:00
let ingredients, args, disableIcon, breakPointIcon, item;
2018-05-15 17:36:45 +00:00
const operations = document.querySelectorAll("#rec-list li.operation");
for (let i = 0; i < operations.length; i++) {
ingredients = [];
2023-08-15 15:50:10 +12:00
disableIcon = operations[i].querySelector(".disable-icon");
breakPointIcon = operations[i].querySelector(".breakpoint-icon");
args = operations[i].querySelectorAll(".arg");
2018-05-15 17:36:45 +00:00
2023-08-15 15:50:10 +12:00
for (let j = 0; j < args.length; j++) {
if (args[j].getAttribute("type") === "checkbox") {
2018-05-15 17:36:45 +00:00
// checkbox
2023-08-15 15:50:10 +12:00
ingredients[j] = args[j].checked;
} else if (args[j].classList.contains("toggle-string")) {
2018-05-15 17:36:45 +00:00
// toggleString
ingredients[j] = {
2023-08-15 15:50:10 +12:00
option: args[j].parentNode.parentNode.querySelector("button").textContent,
string: args[j].value
2018-05-15 17:36:45 +00:00
};
2023-08-15 15:50:10 +12:00
} else if (args[j].getAttribute("type") === "number") {
2018-05-15 17:36:45 +00:00
// number
2023-08-15 15:50:10 +12:00
ingredients[j] = parseFloat(args[j].value);
2018-05-15 17:36:45 +00:00
} else {
// all others
2023-08-15 15:50:10 +12:00
ingredients[j] = args[j].value;
2018-05-15 17:36:45 +00:00
}
}
item = {
op: operations[i].getAttribute("data-name"),
2018-05-15 17:36:45 +00:00
args: ingredients
};
2023-08-15 15:50:10 +12:00
if (disableIcon && disableIcon.getAttribute("disabled") === "true") {
2018-05-15 17:36:45 +00:00
item.disabled = true;
}
2023-08-15 15:50:10 +12:00
if (breakPointIcon && breakPointIcon.getAttribute("break") === "true") {
2018-05-15 17:36:45 +00:00
item.breakpoint = true;
}
config.push(item);
}
return config;
}
/**
* Moves or removes the breakpoint indicator in the recipe based on the position.
*
* @param {number|boolean} position - If boolean, turn off all indicators
2018-05-15 17:36:45 +00:00
*/
updateBreakpointIndicator(position) {
const operations = document.querySelectorAll("#rec-list li.operation");
if (typeof position === "boolean") position = operations.length;
2018-05-15 17:36:45 +00:00
for (let i = 0; i < operations.length; i++) {
if (i === position) {
operations[i].classList.add("break");
} else {
operations[i].classList.remove("break");
}
}
}
/**
* Adds the specified operation to the recipe
2018-05-15 17:36:45 +00:00
*
* @fires Manager#operationadd
2023-08-02 17:38:52 +12:00
* @fires Manager#statechange
2018-05-15 17:36:45 +00:00
* @param {string} name - The name of the operation to add
2023-08-16 17:29:26 +12:00
* @param {number} index - The index where the operation should be displayed
2018-05-15 17:36:45 +00:00
*/
2023-08-16 17:29:26 +12:00
addOperation(name, index = undefined) {
2023-08-15 13:51:12 +12:00
const item = new CRecipeLi(this.app, name, this.app.operations[name].args);
2023-08-16 17:29:26 +12:00
const recipeList = document.getElementById("rec-list");
if (index !== undefined) {
recipeList.insertBefore(item, recipeList.children[index + 1]);
} else {
recipeList.appendChild(item);
}
2018-05-15 17:36:45 +00:00
$(item).find("[data-toggle='tooltip']").tooltip();
2018-05-15 17:36:45 +00:00
item.dispatchEvent(this.manager.operationadd);
2023-08-02 17:38:52 +12:00
document.dispatchEvent(this.app.manager.statechange);
2018-05-15 17:36:45 +00:00
return item;
}
/**
* Removes all operations from the recipe.
*
* @fires Manager#operationremove
*/
clearRecipe() {
const recList = document.getElementById("rec-list");
while (recList.firstChild) {
recList.removeChild(recList.firstChild);
}
recList.dispatchEvent(this.manager.operationremove);
2023-08-15 13:51:12 +12:00
2023-08-15 13:02:33 +12:00
window.dispatchEvent(this.app.manager.statechange);
2018-05-15 17:36:45 +00:00
}
/**
* Handler for operation dropdown events from toggleString arguments.
* Sets the selected option as the name of the button.
*
* @param {Event} e
2018-05-15 17:36:45 +00:00
*/
dropdownToggleClick(e) {
e.stopPropagation();
e.preventDefault();
2018-05-15 17:36:45 +00:00
const el = e.target;
const button = el.parentNode.parentNode.querySelector("button");
2018-05-15 17:36:45 +00:00
button.innerHTML = el.textContent;
2018-05-15 17:36:45 +00:00
this.ingChange();
}
/**
* Triggers various change events for operation arguments that have just been initialised.
*
* @param {HTMLElement} op
*/
triggerArgEvents(op) {
// Trigger populateOption and argSelector events
const triggerableOptions = op.querySelectorAll(".populate-option, .arg-selector");
const evt = new Event("change", {bubbles: true});
if (triggerableOptions.length) {
for (const el of triggerableOptions) {
el.dispatchEvent(evt);
}
}
}
2018-05-15 17:36:45 +00:00
/**
* Handler for operationadd events.
*
* @listens Manager#operationadd
* @fires Manager#statechange
* @param {Event} e
2018-05-15 17:36:45 +00:00
*/
opAdd(e) {
2023-08-02 17:38:52 +12:00
console.log(e);
log.debug(`'${e.target.querySelector("li").getAttribute("data-name")}' added to recipe`);
this.triggerArgEvents(e.target);
2018-05-15 17:36:45 +00:00
window.dispatchEvent(this.manager.statechange);
}
2019-01-15 23:42:05 +00:00
/**
* Handler for text argument dragover events.
* Gives the user a visual cue to show that items can be dropped here.
*
* @param {Event} e
2019-01-15 23:42:05 +00:00
*/
textArgDragover (e) {
// This will be set if we're dragging an operation
if (e.dataTransfer.effectAllowed === "move")
return false;
e.stopPropagation();
e.preventDefault();
e.target.closest("textarea.arg").classList.add("dropping-file");
}
2019-01-18 15:34:56 +00:00
2019-01-15 23:42:05 +00:00
/**
* Handler for text argument dragleave events.
* Removes the visual cue.
*
* @param {Event} e
2019-01-15 23:42:05 +00:00
*/
textArgDragLeave (e) {
e.stopPropagation();
e.preventDefault();
e.target.classList.remove("dropping-file");
}
2019-01-18 15:34:56 +00:00
2019-01-15 23:42:05 +00:00
/**
* Handler for text argument drop events.
* Loads the dragged data into the argument textarea.
*
* @param {Event} e
2019-01-15 23:42:05 +00:00
*/
textArgDrop(e) {
// This will be set if we're dragging an operation
if (e.dataTransfer.effectAllowed === "move")
return false;
e.stopPropagation();
e.preventDefault();
const targ = e.target;
const file = e.dataTransfer.files[0];
const text = e.dataTransfer.getData("Text");
targ.classList.remove("dropping-file");
if (text) {
targ.value = text;
return;
}
if (file) {
const reader = new FileReader();
const self = this;
reader.onload = function (e) {
targ.value = e.target.result;
2019-01-18 15:34:56 +00:00
// Trigger floating label move
2019-01-18 15:37:25 +00:00
const changeEvent = new Event("change");
2019-01-18 15:34:56 +00:00
targ.dispatchEvent(changeEvent);
window.dispatchEvent(self.manager.statechange);
2019-01-15 23:42:05 +00:00
};
reader.readAsText(file);
}
}
2018-05-15 17:36:45 +00:00
/**
* Sets register values.
*
* @param {number} opIndex
* @param {number} numPrevRegisters
* @param {string[]} registers
*/
setRegisters(opIndex, numPrevRegisters, registers) {
const op = document.querySelector(`#rec-list .operation:nth-child(${opIndex + 1})`),
prevRegList = op.querySelector(".register-list");
// Remove previous div
if (prevRegList) prevRegList.remove();
const registerList = [];
for (let i = 0; i < registers.length; i++) {
registerList.push(`$R${numPrevRegisters + i} = ${escapeControlChars(Utils.escapeHtml(Utils.truncate(registers[i], 100)))}`);
2018-05-15 17:36:45 +00:00
}
const registerListEl = `<div class="register-list">
${registerList.join("<br>")}
</div>`;
op.insertAdjacentHTML("beforeend", registerListEl);
}
}
export default RecipeWaiter;