[#181] add some rudimentary code to open and close operations at the appropriate moments

This commit is contained in:
Robin Scholtes 2023-04-17 20:44:47 +12:00
parent 55695cb7f1
commit 7648c6dce4
8 changed files with 168 additions and 49 deletions

View file

@ -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

View file

@ -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));

View file

@ -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>

View file

@ -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;
}

View file

@ -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 {

View file

@ -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;
}

View file

@ -97,3 +97,9 @@ body {
display: inline-block;
}
}
@media only screen and ( min-width: 768px ){
.mobile-only {
display: none;
}
}

View file

@ -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.