From 7648c6dce49cc0dcf631a9d7be77ac24633802e1 Mon Sep 17 00:00:00 2001 From: Robin Scholtes Date: Mon, 17 Apr 2023 20:44:47 +1200 Subject: [PATCH] [#181] add some rudimentary code to open and close operations at the appropriate moments --- src/web/App.mjs | 2 +- src/web/Manager.mjs | 3 +- src/web/html/index.html | 61 +++++++++-------- src/web/stylesheets/components/_operation.css | 34 ++++++++-- src/web/stylesheets/layout/_io.css | 5 ++ src/web/stylesheets/layout/_structure.css | 40 ++++++----- src/web/stylesheets/utils/_general.css | 6 ++ src/web/waiters/OperationsWaiter.mjs | 66 ++++++++++++++++++- 8 files changed, 168 insertions(+), 49 deletions(-) diff --git a/src/web/App.mjs b/src/web/App.mjs index 5db558b7..92d1277a 100755 --- a/src/web/App.mjs +++ b/src/web/App.mjs @@ -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 diff --git a/src/web/Manager.mjs b/src/web/Manager.mjs index b02a7eee..6ef8bd53 100755 --- a/src/web/Manager.mjs +++ b/src/web/Manager.mjs @@ -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)); diff --git a/src/web/html/index.html b/src/web/html/index.html index 4c3d60c1..66c08ecf 100755 --- a/src/web/html/index.html +++ b/src/web/html/index.html @@ -198,7 +198,7 @@ data-help="

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.

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

"> Operations - + + + +
- -
+ +
@@ -243,31 +251,9 @@ - - -
-
- - - - -
-
- -
-
-
-
+
@@ -393,6 +379,29 @@
+ + +
+
+ + + + +
+
+ +
+
+
+
diff --git a/src/web/stylesheets/components/_operation.css b/src/web/stylesheets/components/_operation.css index 2700aa6e..25722bfe 100755 --- a/src/web/stylesheets/components/_operation.css +++ b/src/web/stylesheets/components/_operation.css @@ -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; } diff --git a/src/web/stylesheets/layout/_io.css b/src/web/stylesheets/layout/_io.css index 2c74ee1f..0805b2a4 100755 --- a/src/web/stylesheets/layout/_io.css +++ b/src/web/stylesheets/layout/_io.css @@ -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 { diff --git a/src/web/stylesheets/layout/_structure.css b/src/web/stylesheets/layout/_structure.css index 62188440..75d79c89 100755 --- a/src/web/stylesheets/layout/_structure.css +++ b/src/web/stylesheets/layout/_structure.css @@ -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; -} diff --git a/src/web/stylesheets/utils/_general.css b/src/web/stylesheets/utils/_general.css index 481f27bf..2d59918c 100755 --- a/src/web/stylesheets/utils/_general.css +++ b/src/web/stylesheets/utils/_general.css @@ -97,3 +97,9 @@ body { display: inline-block; } } + +@media only screen and ( min-width: 768px ){ + .mobile-only { + display: none; + } +} diff --git a/src/web/waiters/OperationsWaiter.mjs b/src/web/waiters/OperationsWaiter.mjs index dee0dd06..862f96d3 100755 --- a/src/web/waiters/OperationsWaiter.mjs +++ b/src/web/waiters/OperationsWaiter.mjs @@ -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.