mirror of
https://github.com/gchq/CyberChef.git
synced 2025-05-07 06:57:12 -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() {
|
||||
// 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);
|
||||
|
||||
// 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);
|
||||
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("save-favourites").addEventListener("click", this.ops.saveFavouritesClick.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>">
|
||||
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">
|
||||
<button type="button"
|
||||
class="btn btn-warning bmd-btn-icon"
|
||||
|
@ -207,7 +207,15 @@
|
|||
title="Edit favourites">
|
||||
<i class="material-icons">star</i>
|
||||
</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>
|
||||
|
||||
</div>
|
||||
<div class="foo">
|
||||
<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>"
|
||||
/>
|
||||
<!--operation list and categories-->
|
||||
<ul id="search-results" class="op-list"></ul>
|
||||
<div id="categories" class="panel-group no-select"></div>
|
||||
<ul id="search-results" class="op-list hidden"></ul>
|
||||
<div id="categories" class="panel-group no-select hidden"></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
@ -243,31 +251,9 @@
|
|||
</div>
|
||||
<!--selected recipes list-->
|
||||
<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>
|
||||
|
||||
<!--IO: input/output-->
|
||||
<div id="IO">
|
||||
|
||||
<!--input-->
|
||||
|
@ -393,6 +379,29 @@
|
|||
</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>
|
||||
|
||||
|
|
|
@ -322,9 +322,6 @@ input.toggle-string {
|
|||
}
|
||||
|
||||
/* ================ experimentation ================ */
|
||||
.foo {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
#search-results,
|
||||
#categories {
|
||||
|
@ -332,10 +329,39 @@ input.toggle-string {
|
|||
top: 40px; /* the height of the search input */
|
||||
left: 0;
|
||||
right: 0;
|
||||
z-index: 10;
|
||||
height: auto;
|
||||
max-height: 60vh;
|
||||
overflow: scroll;
|
||||
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;
|
||||
}
|
||||
|
||||
#input .cm-scroller:hover,
|
||||
#output .cm-scroller:hover {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
#output-text.html-output .cm-content,
|
||||
#output-text.html-output .cm-line,
|
||||
#output-html {
|
||||
|
|
|
@ -19,6 +19,7 @@ body {
|
|||
}
|
||||
|
||||
/*@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 {
|
||||
/* The workspaces' available height minus the top banner and control element at the bottom */
|
||||
|
@ -32,13 +33,17 @@ body {
|
|||
|
||||
#recipe {
|
||||
height: 15vh;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
#input,
|
||||
#output {
|
||||
height: 25vh;
|
||||
}
|
||||
|
||||
#input .cm-scroller,
|
||||
#output .cm-scroller {
|
||||
height: 20vh;
|
||||
overflow-y: auto;
|
||||
height: 100%;
|
||||
overflow-y: scroll;
|
||||
}
|
||||
|
||||
#operations .title,
|
||||
|
@ -55,18 +60,21 @@ label[for="output-text"] {
|
|||
line-height: revert;
|
||||
}
|
||||
|
||||
.gutter {
|
||||
background-color: var(--secondary-border-colour);
|
||||
background-repeat: no-repeat;
|
||||
background-position: 50%;
|
||||
@media only screen and ( min-width: 768px ) {
|
||||
.gutter {
|
||||
background-color: var(--secondary-border-colour);
|
||||
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;
|
||||
}
|
||||
}
|
||||
|
||||
@media only screen and ( min-width: 768px ){
|
||||
.mobile-only {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -28,6 +28,25 @@ class OperationsWaiter {
|
|||
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.
|
||||
|
@ -36,7 +55,31 @@ class OperationsWaiter {
|
|||
* @param {event} 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
|
||||
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.
|
||||
* Saves the selected favourites and reloads them.
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue