mirror of
https://github.com/gchq/CyberChef.git
synced 2025-05-08 07:21:02 -04:00
[#181] add some rudimentary code to open and close operations at the appropriate moments
This commit is contained in:
parent
55695cb7f1
commit
7648c6dce4
8 changed files with 168 additions and 49 deletions
|
@ -642,7 +642,7 @@ class App {
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the compile message.
|
* Sets the compile message ( "notice" in #banner ).
|
||||||
*/
|
*/
|
||||||
setCompileMessage() {
|
setCompileMessage() {
|
||||||
// Display time since last build and compile message
|
// Display time since last build and compile message
|
||||||
|
|
|
@ -143,8 +143,9 @@ class Manager {
|
||||||
this.addMultiEventListeners("#save-texts textarea", "keyup paste", this.controls.saveTextChange, this.controls);
|
this.addMultiEventListeners("#save-texts textarea", "keyup paste", this.controls.saveTextChange, this.controls);
|
||||||
|
|
||||||
// Operations
|
// Operations
|
||||||
this.addMultiEventListener("#search", "keyup paste search", this.ops.searchOperations, this.ops);
|
this.addMultiEventListener("#search", "keyup paste search click", this.ops.searchOperations, this.ops);
|
||||||
this.addDynamicListener(".op-list li.operation", "dblclick", this.ops.operationDblclick, this.ops);
|
this.addDynamicListener(".op-list li.operation", "dblclick", this.ops.operationDblclick, this.ops);
|
||||||
|
document.getElementById("reset-operations").addEventListener("click", this.ops.resetOperationsClick.bind(this.ops));
|
||||||
document.getElementById("edit-favourites").addEventListener("click", this.ops.editFavouritesClick.bind(this.ops));
|
document.getElementById("edit-favourites").addEventListener("click", this.ops.editFavouritesClick.bind(this.ops));
|
||||||
document.getElementById("save-favourites").addEventListener("click", this.ops.saveFavouritesClick.bind(this.ops));
|
document.getElementById("save-favourites").addEventListener("click", this.ops.saveFavouritesClick.bind(this.ops));
|
||||||
document.getElementById("reset-favourites").addEventListener("click", this.ops.resetFavouritesClick.bind(this.ops));
|
document.getElementById("reset-favourites").addEventListener("click", this.ops.resetFavouritesClick.bind(this.ops));
|
||||||
|
|
|
@ -198,7 +198,7 @@
|
||||||
data-help="<p>The Operations list contains all the operations in CyberChef arranged into categories. Some operations may be present in multiple categories. You can search for operations using the search box.</p><p>To use an operation, either double click it, or drag it into the Recipe pane. You will then be able to configure its arguments (or 'Ingredients' in CyberChef terminology).</p>">
|
data-help="<p>The Operations list contains all the operations in CyberChef arranged into categories. Some operations may be present in multiple categories. You can search for operations using the search box.</p><p>To use an operation, either double click it, or drag it into the Recipe pane. You will then be able to configure its arguments (or 'Ingredients' in CyberChef terminology).</p>">
|
||||||
Operations
|
Operations
|
||||||
|
|
||||||
<!--@TODO: this should just be removed, however that currently breaks some code-->
|
<!--@TODO: this should just be removed, however that currently breaks some code-->
|
||||||
<span class="pane-controls hide-on-maximised-output">
|
<span class="pane-controls hide-on-maximised-output">
|
||||||
<button type="button"
|
<button type="button"
|
||||||
class="btn btn-warning bmd-btn-icon"
|
class="btn btn-warning bmd-btn-icon"
|
||||||
|
@ -207,7 +207,15 @@
|
||||||
title="Edit favourites">
|
title="Edit favourites">
|
||||||
<i class="material-icons">star</i>
|
<i class="material-icons">star</i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
<button type="button"
|
||||||
|
class="btn bmd-btn-icon mobile-only hidden"
|
||||||
|
id="reset-operations"
|
||||||
|
title="Reset operations">
|
||||||
|
<i class="material-icons">close</i>
|
||||||
|
</button>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="foo">
|
<div class="foo">
|
||||||
<input id="search"
|
<input id="search"
|
||||||
|
@ -220,8 +228,8 @@
|
||||||
data-help="<p>Use the search box to find useful operations.</p><p>Both operation names and descriptions are queried using a fuzzy matching algorithm.</p>"
|
data-help="<p>Use the search box to find useful operations.</p><p>Both operation names and descriptions are queried using a fuzzy matching algorithm.</p>"
|
||||||
/>
|
/>
|
||||||
<!--operation list and categories-->
|
<!--operation list and categories-->
|
||||||
<ul id="search-results" class="op-list"></ul>
|
<ul id="search-results" class="op-list hidden"></ul>
|
||||||
<div id="categories" class="panel-group no-select"></div>
|
<div id="categories" class="panel-group no-select hidden"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -243,31 +251,9 @@
|
||||||
</div>
|
</div>
|
||||||
<!--selected recipes list-->
|
<!--selected recipes list-->
|
||||||
<ul id="rec-list" class="list-area no-select"></ul>
|
<ul id="rec-list" class="list-area no-select"></ul>
|
||||||
|
|
||||||
<!--controls-->
|
|
||||||
<div id="controls" class="no-select hide-on-maximised-output">
|
|
||||||
<div id="controls-content">
|
|
||||||
<button type="button" class="mx-2 btn btn-lg btn-secondary" id="step" data-toggle="tooltip" title="Step through the recipe" data-help-title="Stepping through the Recipe" data-help="<p>The Step button allows you to execute one operation at a time, rather than running the whole Recipe from beginning to end.</p><p>Step allows you to inspect the data at each stage of the Recipe and understand what is being passed to the next operation.</p>">
|
|
||||||
Step
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<button type="button" class="mx-2 btn btn-lg btn-success btn-raised btn-block" id="bake" data-help-title="Baking" data-help="<p>Baking causes CyberChef to run the Recipe against your data. This involves three steps:</p><ol><li>The data in the Input is encoded into bytes using the character encoding selected in the Input status bar.</li><li>The data is run through each of the operations in the Recipe in turn with the output of one operation being fed into the next operation as its input.</li><li>The outcome of the final operation in the Recipe is decoded into Output text using the character encoding selected in the Output status bar.</li></ol><p>If there are multiple Inputs, the Bake button causes every Input to be baked simultaneously.</p>">
|
|
||||||
<img aria-hidden="true" src="<%- require('../static/images/cook_male-32x32.png') %>" alt="Chef Icon"/>
|
|
||||||
<span>Bake!</span>
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<div class="form-group" style="display: contents;">
|
|
||||||
<div class="mx-1 checkbox" data-help-title="Auto-bake" data-help="<p>When Auto-bake is turned on, CyberChef will bake the Input using the Recipe whenever anything in the Input or Recipe changes.</p>This includes:<ul><li>Adding or removing operations</li><li>Modifying operation arguments</li><li>Editing the Input</li><li>Changing the Input character encoding</li></ul><p>If there are multiple inputs, only the currently active tab will be baked when Auto-bake triggers. You can bake all inputs manually using the Bake button.</p>">
|
|
||||||
<label id="auto-bake-label">
|
|
||||||
<input type="checkbox" checked="checked" id="auto-bake">
|
|
||||||
<br>Auto Bake
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!--IO: input/output-->
|
||||||
<div id="IO">
|
<div id="IO">
|
||||||
|
|
||||||
<!--input-->
|
<!--input-->
|
||||||
|
@ -393,6 +379,29 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!--controls-->
|
||||||
|
<div id="controls" class="no-select hide-on-maximised-output">
|
||||||
|
<div id="controls-content">
|
||||||
|
<button type="button" class="mx-2 btn btn-lg btn-secondary" id="step" data-toggle="tooltip" title="Step through the recipe" data-help-title="Stepping through the Recipe" data-help="<p>The Step button allows you to execute one operation at a time, rather than running the whole Recipe from beginning to end.</p><p>Step allows you to inspect the data at each stage of the Recipe and understand what is being passed to the next operation.</p>">
|
||||||
|
Step
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button type="button" class="mx-2 btn btn-lg btn-success btn-raised btn-block" id="bake" data-help-title="Baking" data-help="<p>Baking causes CyberChef to run the Recipe against your data. This involves three steps:</p><ol><li>The data in the Input is encoded into bytes using the character encoding selected in the Input status bar.</li><li>The data is run through each of the operations in the Recipe in turn with the output of one operation being fed into the next operation as its input.</li><li>The outcome of the final operation in the Recipe is decoded into Output text using the character encoding selected in the Output status bar.</li></ol><p>If there are multiple Inputs, the Bake button causes every Input to be baked simultaneously.</p>">
|
||||||
|
<img aria-hidden="true" src="<%- require('../static/images/cook_male-32x32.png') %>" alt="Chef Icon"/>
|
||||||
|
<span>Bake!</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<div class="form-group" style="display: contents;">
|
||||||
|
<div class="mx-1 checkbox" data-help-title="Auto-bake" data-help="<p>When Auto-bake is turned on, CyberChef will bake the Input using the Recipe whenever anything in the Input or Recipe changes.</p>This includes:<ul><li>Adding or removing operations</li><li>Modifying operation arguments</li><li>Editing the Input</li><li>Changing the Input character encoding</li></ul><p>If there are multiple inputs, only the currently active tab will be baked when Auto-bake triggers. You can bake all inputs manually using the Bake button.</p>">
|
||||||
|
<label id="auto-bake-label">
|
||||||
|
<input type="checkbox" checked="checked" id="auto-bake">
|
||||||
|
<br>Auto Bake
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -322,9 +322,6 @@ input.toggle-string {
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ================ experimentation ================ */
|
/* ================ experimentation ================ */
|
||||||
.foo {
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
|
|
||||||
#search-results,
|
#search-results,
|
||||||
#categories {
|
#categories {
|
||||||
|
@ -332,10 +329,39 @@ input.toggle-string {
|
||||||
top: 40px; /* the height of the search input */
|
top: 40px; /* the height of the search input */
|
||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
z-index: 10;
|
|
||||||
height: auto;
|
height: auto;
|
||||||
max-height: 60vh;
|
max-height: 60vh;
|
||||||
overflow: scroll;
|
overflow: scroll;
|
||||||
border-bottom: 1px solid var(--primary-border-colour);
|
border-bottom: 1px solid var(--primary-border-colour);
|
||||||
|
|
||||||
|
z-index: 20;
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
#reset-operations.hidden,
|
||||||
|
#search-results.hidden,
|
||||||
|
#categories.hidden {
|
||||||
|
z-index: -10;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.foo,
|
||||||
|
#recipe {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
#rec-list {
|
||||||
|
position: absolute;
|
||||||
|
top: 48px; /* the height of recipe title */
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#controls {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,11 @@
|
||||||
user-select: auto;
|
user-select: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#input .cm-scroller:hover,
|
||||||
|
#output .cm-scroller:hover {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
#output-text.html-output .cm-content,
|
#output-text.html-output .cm-content,
|
||||||
#output-text.html-output .cm-line,
|
#output-text.html-output .cm-line,
|
||||||
#output-html {
|
#output-html {
|
||||||
|
|
|
@ -19,6 +19,7 @@ body {
|
||||||
}
|
}
|
||||||
|
|
||||||
/*@TODO: test with keyboard popping up because that messes with the viewheight probably*/
|
/*@TODO: test with keyboard popping up because that messes with the viewheight probably*/
|
||||||
|
/*@TODO: make sure the panels are nicely divided / flex-grow to make up the full height of workspace-wrapper*/
|
||||||
|
|
||||||
#workspace-wrapper {
|
#workspace-wrapper {
|
||||||
/* The workspaces' available height minus the top banner and control element at the bottom */
|
/* The workspaces' available height minus the top banner and control element at the bottom */
|
||||||
|
@ -32,13 +33,17 @@ body {
|
||||||
|
|
||||||
#recipe {
|
#recipe {
|
||||||
height: 15vh;
|
height: 15vh;
|
||||||
overflow-y: auto;
|
}
|
||||||
|
|
||||||
|
#input,
|
||||||
|
#output {
|
||||||
|
height: 25vh;
|
||||||
}
|
}
|
||||||
|
|
||||||
#input .cm-scroller,
|
#input .cm-scroller,
|
||||||
#output .cm-scroller {
|
#output .cm-scroller {
|
||||||
height: 20vh;
|
height: 100%;
|
||||||
overflow-y: auto;
|
overflow-y: scroll;
|
||||||
}
|
}
|
||||||
|
|
||||||
#operations .title,
|
#operations .title,
|
||||||
|
@ -55,18 +60,21 @@ label[for="output-text"] {
|
||||||
line-height: revert;
|
line-height: revert;
|
||||||
}
|
}
|
||||||
|
|
||||||
.gutter {
|
@media only screen and ( min-width: 768px ) {
|
||||||
background-color: var(--secondary-border-colour);
|
.gutter {
|
||||||
background-repeat: no-repeat;
|
background-color: var(--secondary-border-colour);
|
||||||
background-position: 50%;
|
background-repeat: no-repeat;
|
||||||
|
background-position: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.gutter.gutter-horizontal {
|
||||||
|
background-image: url('');
|
||||||
|
cursor: ew-resize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.gutter.gutter-vertical {
|
||||||
|
background-image: url('');
|
||||||
|
cursor: ns-resize;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.gutter.gutter-horizontal {
|
|
||||||
background-image: url('');
|
|
||||||
cursor: ew-resize;
|
|
||||||
}
|
|
||||||
|
|
||||||
.gutter.gutter-vertical {
|
|
||||||
background-image: url('');
|
|
||||||
cursor: ns-resize;
|
|
||||||
}
|
|
||||||
|
|
|
@ -97,3 +97,9 @@ body {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@media only screen and ( min-width: 768px ){
|
||||||
|
.mobile-only {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -28,6 +28,25 @@ class OperationsWaiter {
|
||||||
this.removeIntent = false;
|
this.removeIntent = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Toggle element visibility
|
||||||
|
*
|
||||||
|
* @param {HTMLElement} elm
|
||||||
|
* @param {boolean} isVisible
|
||||||
|
*
|
||||||
|
* @TODO: this is pretty generic so probably move it ( to manager? )
|
||||||
|
*/
|
||||||
|
isVisible( elm, isVisible ){
|
||||||
|
if ( isVisible ) {
|
||||||
|
if ( elm.classList.contains("hidden")) {
|
||||||
|
elm.classList.remove("hidden");
|
||||||
|
}
|
||||||
|
} else if ( isVisible === false ) {
|
||||||
|
if ( !elm.classList.contains("hidden")){
|
||||||
|
elm.classList.add("hidden");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handler for search events.
|
* Handler for search events.
|
||||||
|
@ -36,7 +55,31 @@ class OperationsWaiter {
|
||||||
* @param {event} e
|
* @param {event} e
|
||||||
*/
|
*/
|
||||||
searchOperations(e) {
|
searchOperations(e) {
|
||||||
let ops, selected;
|
let ops, selected, categories, hideOperations, searchResults;
|
||||||
|
|
||||||
|
if (e.type === "click" && !e.target.value.length){
|
||||||
|
categories = document.getElementById("categories");
|
||||||
|
hideOperations = document.getElementById("reset-operations");
|
||||||
|
searchResults = document.getElementById("search-results" );
|
||||||
|
|
||||||
|
this.isVisible(categories, true);
|
||||||
|
this.isVisible(hideOperations, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.type === "keyup"){
|
||||||
|
categories = document.getElementById("categories");
|
||||||
|
hideOperations = document.getElementById("reset-operations");
|
||||||
|
searchResults = document.getElementById("search-results" );
|
||||||
|
|
||||||
|
if ( e.target.value.length === 0 ) {
|
||||||
|
this.isVisible(categories, true);
|
||||||
|
this.isVisible(hideOperations, true);
|
||||||
|
} else {
|
||||||
|
this.isVisible(categories, false );
|
||||||
|
this.isVisible(searchResults, true );
|
||||||
|
this.isVisible(hideOperations, true );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (e.type === "search" || e.keyCode === 13) { // Search or Return
|
if (e.type === "search" || e.keyCode === 13) { // Search or Return
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
@ -270,6 +313,27 @@ class OperationsWaiter {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hide any operation lists ( #categories or #search-results ) and the close button on click
|
||||||
|
*/
|
||||||
|
resetOperationsClick(){
|
||||||
|
let search, categories, searchResults, resetOperations;
|
||||||
|
|
||||||
|
search = document.getElementById("search");
|
||||||
|
categories = document.getElementById( "categories");
|
||||||
|
searchResults = document.getElementById( "search-results");
|
||||||
|
resetOperations = document.getElementById("reset-operations");
|
||||||
|
|
||||||
|
// if any input remains in #search, clear it
|
||||||
|
if (search.value.length) {
|
||||||
|
search.value = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
this.isVisible(categories, false );
|
||||||
|
this.isVisible(searchResults, false );
|
||||||
|
this.isVisible(resetOperations, false );
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handler for save favourites click events.
|
* Handler for save favourites click events.
|
||||||
* Saves the selected favourites and reloads them.
|
* Saves the selected favourites and reloads them.
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue