mirror of
https://github.com/gchq/CyberChef
synced 2025-01-24 02:05:02 +00:00
Improve searching for tabs UI.
Content / filename search is now a regex. Use .toLocaleString() for loading / baking stats
This commit is contained in:
parent
9b2f44efb1
commit
d94a431537
7 changed files with 250 additions and 112 deletions
|
@ -284,6 +284,9 @@ class InputWaiter {
|
|||
case "displayTabSearchResults":
|
||||
this.displayTabSearchResults(r.data);
|
||||
break;
|
||||
case "filterTabError":
|
||||
this.app.handleError(r.data);
|
||||
break;
|
||||
case "setUrl":
|
||||
this.setUrl(r.data);
|
||||
break;
|
||||
|
@ -886,20 +889,20 @@ class InputWaiter {
|
|||
const loaded = loadedData.loaded;
|
||||
const total = loadedData.total;
|
||||
|
||||
let width = total.toString().length;
|
||||
let width = total.toLocaleString().length;
|
||||
width = width < 2 ? 2 : width;
|
||||
|
||||
const totalStr = total.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const totalStr = total.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
let msg = "Total: " + totalStr;
|
||||
|
||||
const loadedStr = loaded.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const loadedStr = loaded.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
msg += "<br>Loaded: " + loadedStr;
|
||||
|
||||
if (pending > 0) {
|
||||
const pendingStr = pending.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const pendingStr = pending.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
msg += "<br>Pending: " + pendingStr;
|
||||
} else if (loading > 0) {
|
||||
const loadingStr = loading.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const loadingStr = loading.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
msg += "<br>Loading: " + loadingStr;
|
||||
}
|
||||
|
||||
|
@ -1377,8 +1380,8 @@ class InputWaiter {
|
|||
const showLoading = document.getElementById("input-show-loading").checked;
|
||||
const showLoaded = document.getElementById("input-show-loaded").checked;
|
||||
|
||||
const fileNameFilter = document.getElementById("input-filename-filter").value;
|
||||
const contentFilter = document.getElementById("input-content-filter").value;
|
||||
const filter = document.getElementById("input-filter").value;
|
||||
const filterType = document.getElementById("input-filter-button").innerText;
|
||||
const numResults = parseInt(document.getElementById("input-num-results").value, 10);
|
||||
|
||||
this.inputWorker.postMessage({
|
||||
|
@ -1387,13 +1390,23 @@ class InputWaiter {
|
|||
showPending: showPending,
|
||||
showLoading: showLoading,
|
||||
showLoaded: showLoaded,
|
||||
fileNameFilter: fileNameFilter,
|
||||
contentFilter: contentFilter,
|
||||
filter: filter,
|
||||
filterType: filterType,
|
||||
numResults: numResults
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Handle when an option in the filter drop down box is clicked
|
||||
*
|
||||
* @param {event} mouseEvent
|
||||
*/
|
||||
filterOptionClick(mouseEvent) {
|
||||
document.getElementById("input-filter-button").innerText = mouseEvent.target.innerText;
|
||||
this.filterTabSearch();
|
||||
}
|
||||
|
||||
/**
|
||||
* Displays the results of a tab search in the find tab box
|
||||
*
|
||||
|
|
|
@ -908,17 +908,26 @@ self.updateMaxTabs = function(maxTabs, activeTab) {
|
|||
* @param {boolean} searchData.showPending - If true, include pending inputs in the results
|
||||
* @param {boolean} searchData.showLoading - If true, include loading inputs in the results
|
||||
* @param {boolean} searchData.showLoaded - If true, include loaded inputs in the results
|
||||
* @param {string} searchData.fileNameFilter - A string that must be present in the filename of an input
|
||||
* @param {string} searchData.contentFilter - A string that must be present in the input contents
|
||||
* @param {string} searchData.filter - A regular expression to match the inputs on
|
||||
* @param {string} searchData.filterType - Either "CONTENT" or "FILENAME". Detemines what should be matched with filter
|
||||
* @param {number} searchData.numResults - The maximum number of results to be returned
|
||||
*/
|
||||
self.filterTabs = function(searchData) {
|
||||
const showPending = searchData.showPending;
|
||||
const showLoading = searchData.showLoading;
|
||||
const showLoaded = searchData.showLoaded;
|
||||
const showPending = searchData.showPending,
|
||||
showLoading = searchData.showLoading,
|
||||
showLoaded = searchData.showLoaded,
|
||||
filterType = searchData.filterType;
|
||||
|
||||
const fileNameFilter = searchData.fileNameFilter;
|
||||
const contentFilter = searchData.contentFilter;
|
||||
let filterExp;
|
||||
try {
|
||||
filterExp = new RegExp(searchData.filter, "i");
|
||||
} catch (error) {
|
||||
self.postMessage({
|
||||
action: "filterTabError",
|
||||
data: error.message
|
||||
});
|
||||
return;
|
||||
}
|
||||
const numResults = searchData.numResults;
|
||||
|
||||
const inputs = [];
|
||||
|
@ -930,18 +939,29 @@ self.filterTabs = function(searchData) {
|
|||
if (self.inputs[iNum].status === "pending" && showPending ||
|
||||
self.inputs[iNum].status === "loading" && showLoading ||
|
||||
self.inputs[iNum].status === "loaded" && showLoaded) {
|
||||
try {
|
||||
if (typeof self.inputs[iNum].data === "string") {
|
||||
if (self.inputs[iNum].data.slice(0, 4096).toLowerCase().includes(contentFilter)) {
|
||||
if (filterType.toLowerCase() === "content" &&
|
||||
filterExp.test(self.inputs[iNum].data.slice(0, 4096))) {
|
||||
textDisplay = self.inputs[iNum].data.slice(0, 4096);
|
||||
addInput = true;
|
||||
}
|
||||
} else {
|
||||
if (self.inputs[iNum].data.name.toLowerCase().includes(fileNameFilter) &&
|
||||
Utils.arrayBufferToStr(self.inputs[iNum].data.fileBuffer.slice(0, 4096)).toLowerCase().includes(contentFilter)) {
|
||||
if ((filterType.toLowerCase() === "filename" &&
|
||||
filterExp.test(self.inputs[iNum].data.name)) ||
|
||||
filterType.toLowerCase() === "content" &&
|
||||
filterExp.test(Utils.arrayBufferToStr(self.inputs[iNum].data.fileBuffer.slice(0, 4096)))) {
|
||||
textDisplay = self.inputs[iNum].data.name;
|
||||
addInput = true;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
self.postMessage({
|
||||
action: "filterTabError",
|
||||
data: error.message
|
||||
});
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (addInput) {
|
||||
|
|
|
@ -166,10 +166,9 @@ class Manager {
|
|||
document.getElementById("input-show-pending").addEventListener("change", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-show-loading").addEventListener("change", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-show-loaded").addEventListener("change", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-filename-filter").addEventListener("change", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-filename-filter").addEventListener("keyup", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-content-filter").addEventListener("change", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-content-filter").addEventListener("keyup", this.input.filterTabSearch.bind(this.input));
|
||||
this.addListeners("#input-filter-content,#input-filter-filename", "click", this.input.filterOptionClick, this.input);
|
||||
document.getElementById("input-filter").addEventListener("change", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-filter").addEventListener("keyup", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-num-results").addEventListener("change", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-num-results").addEventListener("keyup", this.input.filterTabSearch.bind(this.input));
|
||||
document.getElementById("input-filter-refresh").addEventListener("click", this.input.filterTabSearch.bind(this.input));
|
||||
|
|
|
@ -705,6 +705,28 @@ class OutputWaiter {
|
|||
}
|
||||
|
||||
const newOutputs = this.getNearbyNums(inputNum, direction);
|
||||
|
||||
const tabsLeft = (newOutputs[0] !== this.getSmallestInputNum());
|
||||
const tabsRight = (newOutputs[newOutputs.length - 1] !== this.getLargestInputNum());
|
||||
|
||||
const firstTabElement = document.getElementById("output-tabs").firstElementChild;
|
||||
const lastTabElement = document.getElementById("output-tabs").lastElementChild;
|
||||
|
||||
if (firstTabElement) {
|
||||
if (tabsLeft) {
|
||||
firstTabElement.style.boxShadow = "15px 0px 15px -15px var(--primary-border-colour) inset";
|
||||
} else {
|
||||
firstTabElement.style.boxShadow = "";
|
||||
}
|
||||
}
|
||||
if (lastTabElement) {
|
||||
if (tabsRight) {
|
||||
lastTabElement.style.boxShadow = "-15px 0px 15px -15px var(--primary-border-colour) inset";
|
||||
} else {
|
||||
lastTabElement.style.boxShadow = "";
|
||||
}
|
||||
}
|
||||
|
||||
for (let i = 0; i < newOutputs.length; i++) {
|
||||
tabs.item(i).setAttribute("inputNum", newOutputs[i].toString());
|
||||
this.displayTabInfo(newOutputs[i]);
|
||||
|
@ -908,6 +930,27 @@ class OutputWaiter {
|
|||
this.displayTabInfo(newInputs[i]);
|
||||
}
|
||||
|
||||
const tabsLeft = (newInputs[0] !== this.getSmallestInputNum());
|
||||
const tabsRight = (newInputs[newInputs.length - 1] !== this.getLargestInputNum());
|
||||
|
||||
const firstTabElement = document.getElementById("output-tabs").firstElementChild;
|
||||
const lastTabElement = document.getElementById("output-tabs").lastElementChild;
|
||||
|
||||
if (firstTabElement) {
|
||||
if (tabsLeft) {
|
||||
firstTabElement.style.boxShadow = "15px 0px 15px -15px var(--primary-border-colour) inset";
|
||||
} else {
|
||||
firstTabElement.style.boxShadow = "";
|
||||
}
|
||||
}
|
||||
if (lastTabElement) {
|
||||
if (tabsRight) {
|
||||
lastTabElement.style.boxShadow = "-15px 0px 15px -15px var(--primary-border-colour) inset";
|
||||
} else {
|
||||
lastTabElement.style.boxShadow = "";
|
||||
}
|
||||
}
|
||||
|
||||
if (newInputs.length > 1) {
|
||||
tabsList.parentElement.style.display = "block";
|
||||
|
||||
|
@ -1320,6 +1363,14 @@ class OutputWaiter {
|
|||
inputNums = Object.keys(this.outputs),
|
||||
results = [];
|
||||
|
||||
let contentFilterExp;
|
||||
try {
|
||||
contentFilterExp = new RegExp(contentFilter, "i");
|
||||
} catch (error) {
|
||||
this.app.handleError(error);
|
||||
return;
|
||||
}
|
||||
|
||||
// Search through the outputs for matching output results
|
||||
for (let i = 0; i < inputNums.length; i++) {
|
||||
const iNum = inputNums[i],
|
||||
|
@ -1347,7 +1398,7 @@ class OutputWaiter {
|
|||
data = Utils.arrayBufferToStr(data);
|
||||
}
|
||||
data = data.replace(/[\r\n]/g, "");
|
||||
if (data.toLowerCase().includes(contentFilter)) {
|
||||
if (contentFilterExp.test(data)) {
|
||||
results.push({
|
||||
inputNum: iNum,
|
||||
textDisplay: data.slice(0, 100)
|
||||
|
@ -1359,7 +1410,7 @@ class OutputWaiter {
|
|||
data = Utils.arrayBufferToStr(data);
|
||||
}
|
||||
data = data.replace(/[\r\n]/g, "");
|
||||
if (data.toLowerCase().includes(contentFilter)) {
|
||||
if (contentFilterExp.test(data)) {
|
||||
results.push({
|
||||
inputNum: iNum,
|
||||
textDisplay: data.slice(0, 100)
|
||||
|
|
|
@ -626,13 +626,13 @@ class WorkerWaiter {
|
|||
}
|
||||
|
||||
const bakeInfo = document.getElementById("bake-info");
|
||||
let width = progress.total.toString().length;
|
||||
let width = progress.total.toLocaleString().length;
|
||||
width = width < 2 ? 2 : width;
|
||||
|
||||
const totalStr = progress.total.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const bakedStr = progress.baked.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const pendingStr = progress.pending.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const bakingStr = progress.baking.toString().padStart(width, " ").replace(/ /g, " ");
|
||||
const totalStr = progress.total.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
const bakedStr = progress.baked.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
const pendingStr = progress.pending.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
const bakingStr = progress.baking.toLocaleString().padStart(width, " ").replace(/ /g, " ");
|
||||
|
||||
let msg = "Total: " + totalStr;
|
||||
msg += "<br>Baked: " + bakedStr;
|
||||
|
|
|
@ -768,7 +768,8 @@
|
|||
</div>
|
||||
<div class="modal-body" id="input-tab-body">
|
||||
<h6>Load Status</h6>
|
||||
<ul id="input-find-options">
|
||||
<div id="input-find-options">
|
||||
<ul id="input-find-options-checkboxes">
|
||||
<li class="checkbox input-find-option">
|
||||
<label for="input-show-pending">
|
||||
<input type="checkbox" id="input-show-pending" checked="">
|
||||
|
@ -788,18 +789,23 @@
|
|||
</label>
|
||||
</li>
|
||||
</ul>
|
||||
<br>
|
||||
<div class="form-group input-find-option">
|
||||
<label for="input-filename-filter" class="bmd-label-floating">Filename</label>
|
||||
<input type="text" class="form-control" id="input-filename-filter">
|
||||
</div>
|
||||
<div class="form-group input-find-option">
|
||||
<label for="input-content-filter" class="bmd-label-floating">Content</label>
|
||||
<input type="text" class="form-control" id="input-content-filter">
|
||||
</div>
|
||||
<div class="form-group input-find-option">
|
||||
<div class="form-group input-find-option" id="input-num-results-container">
|
||||
<label for="input-num-results" class="bmd-label-floating">Number of results</label>
|
||||
<input type="number" class="form-control" id="input-num-results" value="20">
|
||||
<input type="number" class="form-control" id="input-num-results" value="20" min="1">
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-group input-group">
|
||||
<div class="toggle-string">
|
||||
<label for="input-filter" class="bmd-label-floating toggle-string">Filter (regex)</label>
|
||||
<input type="text" class="form-control toggle-string" id="input-filter">
|
||||
</div>
|
||||
<div class="input-group-append">
|
||||
<button class="btn btn-secondary dropdown-toggle" id="input-filter-button" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">CONTENT</button>
|
||||
<div class="dropdown-menu toggle-dropdown">
|
||||
<a class="dropdown-item" id="input-filter-content">Content</a>
|
||||
<a class="dropdown-item" id="input-filter-filename">Filename</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<br>
|
||||
<h6>Results</h6>
|
||||
|
@ -821,7 +827,8 @@
|
|||
</div>
|
||||
<div class="modal-body" id="output-tab-body">
|
||||
<h6>Bake Status</h6>
|
||||
<ul id="output-find-options">
|
||||
<div id="output-find-options">
|
||||
<ul id="output-find-options-checkboxes">
|
||||
<li class="checkbox output-find-option">
|
||||
<label for="output-show-pending">
|
||||
<input type="checkbox" id="output-show-pending" checked="">
|
||||
|
@ -853,15 +860,15 @@
|
|||
</label>
|
||||
</li>
|
||||
</ul>
|
||||
<br>
|
||||
<div class="form-group output-find-option">
|
||||
<label for="output-content-filter" class="bmd-label-floating">Content</label>
|
||||
<label for="output-content-filter" class="bmd-label-floating">Content filter (regex)</label>
|
||||
<input type="text" class="form-control" id="output-content-filter">
|
||||
</div>
|
||||
<div class="form-group output-find-option">
|
||||
<div class="form-group output-find-option" id="output-num-results-container">
|
||||
<label for="output-num-results" class="bmd-label-floating">Number of results</label>
|
||||
<input type="number" class="form-control" id="output-num-results" value="20">
|
||||
</div>
|
||||
</div>
|
||||
<br>
|
||||
<h6>Results</h6>
|
||||
<ul id="output-search-results"></ul>
|
||||
|
|
|
@ -297,19 +297,67 @@
|
|||
|
||||
#input-find-options,
|
||||
#output-find-options {
|
||||
list-style: none;
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
flex-wrap: wrap;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
#input-find-options li,
|
||||
#output-find-options li {
|
||||
.form-group.input-find-option {
|
||||
width: 80%;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
.input-find-option .toggle-string {
|
||||
width: 70%;
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
.input-find-option-append button {
|
||||
border-top-right-radius: 4px;
|
||||
background-color: var(--arg-background) !important;
|
||||
margin: unset;
|
||||
}
|
||||
|
||||
.input-find-option-append button:hover {
|
||||
filter: brightness(97%);
|
||||
}
|
||||
|
||||
#input-num-results-container {
|
||||
width: 20%;
|
||||
float: right;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.form-group.output-find-option {
|
||||
width: 70%;
|
||||
float: left;
|
||||
}
|
||||
|
||||
#output-num-results-container {
|
||||
width: 20%;
|
||||
float: right;
|
||||
margin: 0;
|
||||
margin-left: 10%;
|
||||
}
|
||||
|
||||
#input-find-options-checkboxes,
|
||||
#output-find-options-checkboxes {
|
||||
list-style: none;
|
||||
width: 70%;
|
||||
margin-top: auto;
|
||||
margin-bottom: auto;
|
||||
}
|
||||
|
||||
#input-find-options-checkboxes li,
|
||||
#output-find-options-checkboxes li {
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
float: left;
|
||||
padding: 10px;
|
||||
width: auto;
|
||||
text-align: center;
|
||||
vertical-align: middle;
|
||||
}
|
||||
|
||||
|
||||
|
|
Loading…
Reference in a new issue