2018-05-15 17:36:45 +00:00
/ * *
* @ author n1474335 [ n1474335 @ gmail . com ]
2019-04-02 15:58:36 +00:00
* @ author j433866 [ j433866 @ gmail . com ]
2018-05-15 17:36:45 +00:00
* @ copyright Crown Copyright 2016
* @ license Apache - 2.0
* /
import Utils from "../core/Utils" ;
import FileSaver from "file-saver" ;
2019-04-03 11:00:47 +00:00
import zip from "zlibjs/bin/zip.min" ;
const Zlib = zip . Zlib ;
2018-05-15 17:36:45 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Waiter to handle events related to the output
* /
2018-05-15 17:36:45 +00:00
class OutputWaiter {
/ * *
* OutputWaiter constructor .
*
* @ param { App } app - The main view object for CyberChef .
2019-04-02 15:58:36 +00:00
* @ param { Manager } manager - The CyberChef event manager
2018-05-15 17:36:45 +00:00
* /
constructor ( app , manager ) {
this . app = app ;
this . manager = manager ;
2019-04-25 15:32:48 +00:00
this . outputs = { } ;
2019-04-26 14:15:44 +00:00
this . activeTab = - 1 ;
2019-04-04 09:15:13 +00:00
2019-04-02 15:58:36 +00:00
this . maxTabs = 4 ; // Calculate this
2018-05-15 17:36:45 +00:00
}
2019-04-04 09:15:13 +00:00
/ * *
* Calculates the maximum number of tabs to display
* /
calcMaxTabs ( ) {
const numTabs = Math . floor ( ( document . getElementById ( "IO" ) . offsetWidth - 75 ) / 120 ) ;
this . maxTabs = numTabs ;
}
2018-05-15 17:36:45 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Gets the output for the specified input number
2018-05-15 17:36:45 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
2019-04-03 11:00:47 +00:00
* @ returns { string | ArrayBuffer }
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
getOutput ( inputNum ) {
2019-04-25 15:32:48 +00:00
if ( this . outputs [ inputNum ] === undefined || this . outputs [ inputNum ] === null ) return - 1 ;
2018-05-15 17:36:45 +00:00
2019-04-25 15:32:48 +00:00
if ( this . outputs [ inputNum ] . data === null ) return "" ;
2018-05-15 17:36:45 +00:00
2019-05-03 10:49:14 +00:00
if ( typeof this . outputs [ inputNum ] . data . result === "string" ) {
return this . outputs [ inputNum ] . data . result ;
2019-04-25 15:32:48 +00:00
} else {
2019-05-03 10:49:14 +00:00
return this . outputs [ inputNum ] . data . result || "" ;
2019-03-27 09:05:10 +00:00
}
}
2018-05-15 17:36:45 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Gets the output string or FileBuffer for the active input
2018-05-15 17:36:45 +00:00
*
2019-04-02 15:58:36 +00:00
* @ returns { string | ArrayBuffer }
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
getActive ( ) {
2019-04-03 11:00:47 +00:00
return this . getOutput ( this . getActiveTab ( ) ) ;
2018-05-15 17:36:45 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Adds a new output to the output array .
* Creates a new tab if we have less than maxtabs tabs open
2018-05-15 17:36:45 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
* @ param { boolean } [ changeTab = true ]
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
addOutput ( inputNum , changeTab = true ) {
2019-04-25 15:32:48 +00:00
const output = this . getOutput ( inputNum ) ;
if ( output !== - 1 ) {
2019-04-02 15:58:36 +00:00
// Remove the output if it already exists
2019-04-25 15:32:48 +00:00
delete this . outputs [ inputNum ] ;
2019-04-02 15:58:36 +00:00
}
const newOutput = {
data : null ,
inputNum : inputNum ,
2019-05-01 14:19:01 +00:00
statusMessage : ` Input ${ inputNum } has not been baked yet. ` ,
2019-04-02 15:58:36 +00:00
error : null ,
status : "inactive"
} ;
2019-04-25 15:32:48 +00:00
this . outputs [ inputNum ] = newOutput ;
2019-04-02 15:58:36 +00:00
// add new tab
this . addTab ( inputNum , changeTab ) ;
2018-05-15 17:36:45 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Updates the value for the output in the output array .
* If this is the active output tab , updates the output textarea
*
* @ param { Object } data
* @ param { number } inputNum
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
updateOutputValue ( data , inputNum ) {
2019-04-25 15:32:48 +00:00
if ( this . getOutput ( inputNum ) === - 1 ) {
this . addOutput ( inputNum ) ;
2019-04-02 15:58:36 +00:00
}
2018-05-15 17:36:45 +00:00
2019-04-25 15:32:48 +00:00
this . outputs [ inputNum ] . data = data ;
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
// set output here
this . set ( inputNum ) ;
2018-05-15 17:36:45 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Updates the status message for the output in the output array .
* If this is the active output tab , updates the output textarea
*
* @ param { string } statusMessage
* @ param { number } inputNum
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
updateOutputMessage ( statusMessage , inputNum ) {
2019-04-25 15:32:48 +00:00
if ( this . getOutput ( inputNum ) === - 1 ) return ;
this . outputs [ inputNum ] . statusMessage = statusMessage ;
2019-04-02 15:58:36 +00:00
this . set ( inputNum ) ;
2018-05-15 17:36:45 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Updates the error value for the output in the output array .
* If this is the active output tab , calls app . handleError .
* Otherwise , the error will be handled when the output is switched to
2018-05-15 17:36:45 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { Error } error
* @ param { number } inputNum
2019-05-01 16:08:36 +00:00
* @ param { number } [ progress = 0 ]
2018-05-15 17:36:45 +00:00
* /
2019-05-01 16:08:36 +00:00
updateOutputError ( error , inputNum , progress = 0 ) {
2019-04-25 15:32:48 +00:00
if ( this . getOutput ( inputNum ) === - 1 ) return ;
2018-05-15 17:36:45 +00:00
2019-04-25 15:32:48 +00:00
this . outputs [ inputNum ] . error = error ;
2019-05-01 16:08:36 +00:00
this . outputs [ inputNum ] . progress = progress ;
this . updateOutputStatus ( "error" , inputNum ) ;
2019-04-02 15:58:36 +00:00
// call handle error here
// or make the error handling part of set()
}
2018-05-15 17:36:45 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Updates the status value for the output in the output array
2018-05-15 17:36:45 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { string } status
* @ param { number } inputNum
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
updateOutputStatus ( status , inputNum ) {
2019-04-25 15:32:48 +00:00
if ( this . getOutput ( inputNum ) === - 1 ) return ;
this . outputs [ inputNum ] . status = status ;
2018-05-15 17:36:45 +00:00
2019-05-01 16:08:36 +00:00
if ( status !== "error" ) {
delete this . outputs [ inputNum ] . error ;
delete this . outputs [ inputNum ] . progress ;
}
2019-04-02 15:58:36 +00:00
this . set ( inputNum ) ;
2018-05-15 17:36:45 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Removes an output from the output array .
*
* @ param { number } inputNum
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
removeOutput ( inputNum ) {
2019-04-25 15:32:48 +00:00
if ( this . getOutput ( inputNum ) === - 1 ) return ;
2018-05-15 17:36:45 +00:00
2019-04-25 15:32:48 +00:00
delete ( this . outputs [ inputNum ] ) ;
2019-04-02 15:58:36 +00:00
}
2018-05-15 17:36:45 +00:00
2019-04-30 12:18:22 +00:00
/ * *
* Removes all output tabs
* /
removeAllOutputs ( ) {
this . outputs = { } ;
const tabs = document . getElementById ( "output-tabs" ) . children ;
for ( let i = tabs . length - 1 ; i >= 0 ; i -- ) {
tabs . item ( i ) . remove ( ) ;
}
}
2018-05-15 17:36:45 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Sets the output in the output textarea .
*
* @ param { number } inputNum
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
set ( inputNum ) {
2019-04-25 15:32:48 +00:00
const output = this . outputs [ inputNum ] ;
if ( output === undefined || output === null ) return ;
if ( typeof inputNum !== "number" ) inputNum = parseInt ( inputNum , 10 ) ;
2019-05-01 16:08:36 +00:00
if ( inputNum !== this . getActiveTab ( ) ) return ;
2019-04-02 15:58:36 +00:00
const outputText = document . getElementById ( "output-text" ) ;
const outputHtml = document . getElementById ( "output-html" ) ;
const outputFile = document . getElementById ( "output-file" ) ;
const outputHighlighter = document . getElementById ( "output-highlighter" ) ;
const inputHighlighter = document . getElementById ( "input-highlighter" ) ;
// If inactive, show blank
// If pending or baking, show loader and status message
// If error, style the tab and handle the error
// If done, display the output if it's the active tab
2019-04-25 15:32:48 +00:00
if ( output . status === "inactive" || output . status === "stale" ) {
this . manager . controls . showStaleIndicator ( ) ;
} else {
this . manager . controls . hideStaleIndicator ( ) ;
}
2019-04-02 15:58:36 +00:00
2019-05-01 16:08:36 +00:00
this . manager . recipe . updateBreakpointIndicator ( false ) ;
if ( output . status === "pending" || output . status === "baking" ) {
// show the loader and the status message if it's being shown
// otherwise don't do anything
this . toggleLoader ( true ) ;
document . querySelector ( "#output-loader .loading-msg" ) . textContent = output . statusMessage ;
} else if ( output . status === "error" ) {
// style the tab if it's being shown
// run app.handleError()
this . toggleLoader ( false ) ;
outputText . style . display = "block" ;
outputHtml . style . display = "none" ;
outputFile . style . display = "none" ;
outputHighlighter . display = "none" ;
inputHighlighter . display = "none" ;
outputText . value = output . error ;
outputHtml . innerHTML = "" ;
this . manager . recipe . updateBreakpointIndicator ( output . progress ) ;
} else if ( output . status === "baked" || output . status === "inactive" ) {
this . displayTabInfo ( inputNum ) ;
this . toggleLoader ( false ) ;
this . closeFile ( ) ;
let scriptElements , lines , length ;
if ( output . data === null ) {
2019-04-02 15:58:36 +00:00
outputText . style . display = "block" ;
outputHtml . style . display = "none" ;
outputFile . style . display = "none" ;
outputHighlighter . display = "block" ;
inputHighlighter . display = "block" ;
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
outputText . value = "" ;
outputHtml . innerHTML = "" ;
2018-05-15 17:36:45 +00:00
2019-05-01 16:08:36 +00:00
lines = 0 ;
length = 0 ;
return ;
2019-04-02 15:58:36 +00:00
}
2018-05-15 17:36:45 +00:00
2019-05-01 16:08:36 +00:00
switch ( output . data . type ) {
case "html" :
outputText . style . display = "none" ;
outputHtml . style . display = "block" ;
outputFile . style . display = "none" ;
outputHighlighter . style . display = "none" ;
inputHighlighter . style . display = "none" ;
outputText . value = "" ;
outputHtml . innerHTML = output . data . result ;
// Execute script sections
scriptElements = outputHtml . querySelectorAll ( "script" ) ;
for ( let i = 0 ; i < scriptElements . length ; i ++ ) {
try {
eval ( scriptElements [ i ] . innerHTML ) ; // eslint-disable-line no-eval
} catch ( err ) {
log . error ( err ) ;
2019-04-02 15:58:36 +00:00
}
2019-05-01 16:08:36 +00:00
}
length = output . data . dish . value . length ;
break ;
case "ArrayBuffer" :
outputText . style . display = "block" ;
outputHtml . style . display = "none" ;
outputHighlighter . display = "none" ;
inputHighlighter . display = "none" ;
outputText . value = "" ;
outputHtml . innerHTML = "" ;
length = output . data . result . length ;
this . setFile ( output . data . result ) ;
break ;
case "string" :
default :
outputText . style . display = "block" ;
outputHtml . style . display = "none" ;
outputFile . style . display = "none" ;
outputHighlighter . display = "block" ;
inputHighlighter . display = "block" ;
outputText . value = Utils . printable ( output . data . result , true ) ;
outputHtml . innerHTML = "" ;
lines = output . data . result . count ( "\n" ) + 1 ;
length = output . data . result . length ;
break ;
2019-04-02 15:58:36 +00:00
}
2019-05-01 16:08:36 +00:00
this . setOutputInfo ( length , lines , output . data . duration ) ;
this . backgroundMagic ( ) ;
2018-05-15 17:36:45 +00:00
}
}
/ * *
2019-04-02 15:58:36 +00:00
* Shows file details
*
* @ param { ArrayBuffer } buf
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
setFile ( buf ) {
const file = new File ( [ buf ] , "output.dat" ) ;
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
// Display file overlay in output area with details
const fileOverlay = document . getElementById ( "output-file" ) ,
fileSize = document . getElementById ( "output-file-size" ) ,
outputText = document . getElementById ( "output-text" ) ,
fileSlice = buf . slice ( 0 , 4096 ) ;
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
fileOverlay . style . display = "block" ;
fileSize . textContent = file . size . toLocaleString ( ) + " bytes" ;
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
outputText . classList . add ( "blur" ) ;
outputText . value = Utils . printable ( Utils . arrayBufferToStr ( fileSlice ) ) ;
2018-05-15 17:36:45 +00:00
}
2019-04-02 15:58:36 +00:00
/ * *
* Clears output file details
* /
closeFile ( ) {
document . getElementById ( "output-file" ) . style . display = "none" ;
document . getElementById ( "output-text" ) . classList . remove ( "blur" ) ;
}
2018-05-15 17:36:45 +00:00
/ * *
2019-01-16 12:29:34 +00:00
* Save bombe object then remove it from the DOM so that it does not cause performance issues .
2019-01-15 19:03:17 +00:00
* /
saveBombe ( ) {
2019-01-16 12:29:34 +00:00
this . bombeEl = document . getElementById ( "bombe" ) ;
this . bombeEl . parentNode . removeChild ( this . bombeEl ) ;
2019-01-15 19:03:17 +00:00
}
/ * *
* Shows or hides the output loading screen .
* The animated Bombe SVG , whilst quite aesthetically pleasing , is reasonably CPU
* intensive , so we remove it from the DOM when not in use . We only show it if the
* recipe is taking longer than 200 ms . We add it to the DOM just before that so that
* it is ready to fade in without stuttering .
2018-05-15 17:36:45 +00:00
*
2019-01-15 19:03:17 +00:00
* @ param { boolean } value - true == show loader
2018-05-15 17:36:45 +00:00
* /
toggleLoader ( value ) {
2019-01-15 19:03:17 +00:00
clearTimeout ( this . appendBombeTimeout ) ;
clearTimeout ( this . outputLoaderTimeout ) ;
2018-05-15 17:36:45 +00:00
const outputLoader = document . getElementById ( "output-loader" ) ,
2019-01-15 19:03:17 +00:00
outputElement = document . getElementById ( "output-text" ) ,
2019-01-16 12:29:34 +00:00
animation = document . getElementById ( "output-loader-animation" ) ;
2018-05-15 17:36:45 +00:00
if ( value ) {
this . manager . controls . hideStaleIndicator ( ) ;
2019-01-15 19:03:17 +00:00
2019-05-01 14:19:01 +00:00
// Don't add the bombe if it's already there!
if ( animation . children . length > 0 ) return ;
2019-01-15 19:03:17 +00:00
// Start a timer to add the Bombe to the DOM just before we make it
// visible so that there is no stuttering
this . appendBombeTimeout = setTimeout ( function ( ) {
2019-01-16 12:29:34 +00:00
animation . appendChild ( this . bombeEl ) ;
2019-01-15 19:03:17 +00:00
} . bind ( this ) , 150 ) ;
// Show the loading screen
this . outputLoaderTimeout = setTimeout ( function ( ) {
2018-05-15 17:36:45 +00:00
outputElement . disabled = true ;
outputLoader . style . visibility = "visible" ;
outputLoader . style . opacity = 1 ;
2019-04-25 15:32:48 +00:00
} , 200 ) ;
2018-05-15 17:36:45 +00:00
} else {
2019-01-15 19:03:17 +00:00
// Remove the Bombe from the DOM to save resources
this . outputLoaderTimeout = setTimeout ( function ( ) {
try {
2019-01-16 12:29:34 +00:00
animation . removeChild ( this . bombeEl ) ;
2019-01-15 19:03:17 +00:00
} catch ( err ) { }
} . bind ( this ) , 500 ) ;
2018-05-15 17:36:45 +00:00
outputElement . disabled = false ;
outputLoader . style . opacity = 0 ;
outputLoader . style . visibility = "hidden" ;
2019-04-02 15:58:36 +00:00
// this.setStatusMsg("");
2018-05-15 17:36:45 +00:00
}
}
2019-04-03 11:00:47 +00:00
/ * *
* Handler for save click events .
* Saves the current output to a file .
* /
saveClick ( ) {
2019-04-25 15:32:48 +00:00
this . downloadFile ( ) ;
2019-04-03 11:00:47 +00:00
}
/ * *
* Handler for file download events .
* /
async downloadFile ( ) {
const fileName = window . prompt ( "Please enter a filename: " , "download.dat" ) ;
const file = new File ( [ this . getActive ( ) ] , fileName ) ;
FileSaver . saveAs ( file , fileName , false ) ;
}
/ * *
* Handler for save all click event
* Saves all outputs to a single archvie file
* /
saveAllClick ( ) {
this . downloadAllFiles ( ) ;
}
/ * *
* Handler for download all files events .
* /
async downloadAllFiles ( ) {
const fileName = window . prompt ( "Please enter a filename: " , "download.zip" ) ;
const fileExt = window . prompt ( "Please enter a file extension for the files: " , ".txt" ) ;
const zip = new Zlib . Zip ( ) ;
2019-05-01 13:46:05 +00:00
const inputNums = Object . keys ( this . outputs ) ;
for ( let i = 0 ; i < inputNums . length ; i ++ ) {
const name = Utils . strToByteArray ( inputNums [ i ] + fileExt ) ;
let out = this . getOutput ( inputNums [ i ] ) ;
2019-04-03 11:00:47 +00:00
if ( typeof out === "string" ) {
out = Utils . strToUtf8ByteArray ( out ) ;
}
out = new Uint8Array ( out ) ;
// options.filename = Utils.strToByteArray(this.outputs[i].inputNum + ".dat");
zip . addFile ( out , { filename : name } ) ;
}
const file = new File ( [ zip . compress ( ) ] , fileName ) ;
FileSaver . saveAs ( file , fileName , false ) ;
}
2018-05-15 17:36:45 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Adds a new output tab .
2018-05-15 17:36:45 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
* @ param { boolean } [ changeTab = true ]
2018-05-15 17:36:45 +00:00
* /
2019-04-02 15:58:36 +00:00
addTab ( inputNum , changeTab = true ) {
const tabsWrapper = document . getElementById ( "output-tabs" ) ;
const numTabs = tabsWrapper . children . length ;
2018-05-15 17:36:45 +00:00
2019-04-30 12:18:22 +00:00
if ( this . getTabItem ( inputNum ) === undefined && numTabs < this . maxTabs ) {
2019-04-02 15:58:36 +00:00
// Create a new tab element
const newTab = this . createTabElement ( inputNum ) ;
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
tabsWrapper . appendChild ( newTab ) ;
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
if ( numTabs > 0 ) {
tabsWrapper . parentElement . style . display = "block" ;
2019-04-03 11:00:47 +00:00
2019-04-02 15:58:36 +00:00
document . getElementById ( "output-wrapper" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
document . getElementById ( "output-highlighter" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
document . getElementById ( "output-file" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
document . getElementById ( "output-loader" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
2019-04-25 15:32:48 +00:00
document . getElementById ( "save-all-to-file" ) . style . display = "inline-block" ;
2019-04-30 12:18:22 +00:00
} else {
tabsWrapper . parentElement . style . display = "none" ;
document . getElementById ( "output-wrapper" ) . style . height = "calc(100% - var(--title-height))" ;
document . getElementById ( "output-highlighter" ) . style . height = "calc(100% - var(--title-height))" ;
document . getElementById ( "output-file" ) . style . height = "calc(100% - var(--title-height))" ;
document . getElementById ( "output-loader" ) . style . height = "calc(100% - var(--title-height))" ;
document . getElementById ( "save-all-to-file" ) . style . display = "none" ;
2019-04-02 15:58:36 +00:00
}
}
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
if ( changeTab ) {
2019-04-03 11:00:47 +00:00
this . changeTab ( inputNum , false ) ;
2019-04-02 15:58:36 +00:00
}
2018-05-15 17:36:45 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Changes the active tab
2018-05-15 17:36:45 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
2019-04-03 11:00:47 +00:00
* @ param { boolean } [ changeInput = false ]
2018-05-15 17:36:45 +00:00
* /
2019-04-03 11:00:47 +00:00
changeTab ( inputNum , changeInput = false ) {
2019-04-02 15:58:36 +00:00
const currentNum = this . getActiveTab ( ) ;
2019-04-25 15:32:48 +00:00
if ( this . getOutput ( inputNum ) === - 1 ) return ;
2019-05-01 13:46:05 +00:00
this . hideMagicButton ( ) ;
2019-04-02 15:58:36 +00:00
const tabsWrapper = document . getElementById ( "output-tabs" ) ;
const tabs = tabsWrapper . children ;
let found = false ;
for ( let i = 0 ; i < tabs . length ; i ++ ) {
if ( tabs . item ( i ) . getAttribute ( "inputNum" ) === inputNum . toString ( ) ) {
tabs . item ( i ) . classList . add ( "active-output-tab" ) ;
2019-04-26 14:15:44 +00:00
this . activeTab = inputNum ;
2019-04-02 15:58:36 +00:00
found = true ;
} else {
tabs . item ( i ) . classList . remove ( "active-output-tab" ) ;
}
}
if ( ! found ) {
let direction = "right" ;
if ( currentNum > inputNum ) {
direction = "left" ;
}
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
const newOutputs = this . getNearbyNums ( inputNum , direction ) ;
for ( let i = 0 ; i < newOutputs . length ; i ++ ) {
tabs . item ( i ) . setAttribute ( "inputNum" , newOutputs [ i ] . toString ( ) ) ;
this . displayTabInfo ( newOutputs [ i ] ) ;
if ( newOutputs [ i ] === inputNum ) {
2019-04-26 14:15:44 +00:00
this . activeTab = inputNum ;
2019-04-03 11:00:47 +00:00
tabs . item ( i ) . classList . add ( "active-output-tab" ) ;
2019-04-02 15:58:36 +00:00
}
}
}
2018-05-15 17:36:45 +00:00
2019-04-02 15:58:36 +00:00
this . set ( inputNum ) ;
2019-04-03 11:00:47 +00:00
if ( changeInput ) {
this . manager . input . changeTab ( inputNum , false ) ;
}
2019-04-02 15:58:36 +00:00
}
2018-06-03 16:33:13 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Handler for changing tabs event
*
* @ param { event } mouseEvent
2018-06-03 16:33:13 +00:00
* /
2019-04-02 15:58:36 +00:00
changeTabClick ( mouseEvent ) {
2019-04-04 12:13:21 +00:00
if ( ! mouseEvent . target ) return ;
const tabNum = mouseEvent . target . parentElement . getAttribute ( "inputNum" ) ;
2019-04-02 15:58:36 +00:00
if ( tabNum ) {
2019-04-03 11:00:47 +00:00
this . changeTab ( parseInt ( tabNum , 10 ) , this . app . options . syncTabs ) ;
}
}
/ * *
* Handler for changing to the left tab
* /
changeTabLeft ( ) {
const currentTab = this . getActiveTab ( ) ;
2019-04-25 15:32:48 +00:00
this . changeTab ( this . getPreviousInputNum ( currentTab ) , this . app . options . syncTabs ) ;
2019-04-03 11:00:47 +00:00
}
/ * *
* Handler for changing to the right tab
* /
changeTabRight ( ) {
const currentTab = this . getActiveTab ( ) ;
this . changeTab ( this . getNextInputNum ( currentTab ) , this . app . options . syncTabs ) ;
}
/ * *
* Handler for go to tab button clicked
* /
goToTab ( ) {
const tabNum = parseInt ( window . prompt ( "Enter tab number:" , this . getActiveTab ( ) . toString ( ) ) , 10 ) ;
2019-05-01 16:08:36 +00:00
if ( this . getOutput ( tabNum ) !== undefined ) {
2019-04-03 11:00:47 +00:00
this . changeTab ( tabNum , this . app . options . syncTabs ) ;
2018-06-03 16:33:13 +00:00
}
}
/ * *
2019-04-02 15:58:36 +00:00
* Generates a list of the nearby inputNums
2019-04-04 09:15:13 +00:00
* @ param inputNum
* @ param direction
2018-06-03 16:33:13 +00:00
* /
2019-04-02 15:58:36 +00:00
getNearbyNums ( inputNum , direction ) {
const nums = [ ] ;
2019-04-04 09:15:13 +00:00
for ( let i = 0 ; i < this . maxTabs ; i ++ ) {
let newNum ;
if ( i === 0 ) {
newNum = inputNum ;
} else {
switch ( direction ) {
case "left" :
newNum = this . getNextInputNum ( nums [ i - 1 ] ) ;
if ( newNum === nums [ i - 1 ] ) {
direction = "right" ;
newNum = this . getPreviousInputNum ( nums [ i - 1 ] ) ;
}
2019-04-03 15:05:10 +00:00
break ;
2019-04-04 09:15:13 +00:00
case "right" :
newNum = this . getPreviousInputNum ( nums [ i - 1 ] ) ;
if ( newNum === nums [ i - 1 ] ) {
direction = "left" ;
newNum = this . getNextInputNum ( nums [ i - 1 ] ) ;
}
2019-04-02 15:58:36 +00:00
}
}
2019-04-04 09:15:13 +00:00
if ( ! nums . includes ( newNum ) && ( newNum > 0 ) ) {
nums . push ( newNum ) ;
}
2019-04-02 15:58:36 +00:00
}
nums . sort ( function ( a , b ) {
return a - b ;
} ) ;
return nums ;
2018-07-27 13:37:38 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Gets the largest inputNum
2018-07-27 13:37:38 +00:00
*
2019-04-02 15:58:36 +00:00
* @ returns { number }
2018-07-27 13:37:38 +00:00
* /
2019-04-02 15:58:36 +00:00
getLargestInputNum ( ) {
let largest = 0 ;
2019-04-25 15:32:48 +00:00
const inputNums = Object . keys ( this . outputs ) ;
for ( let i = 0 ; i < inputNums . length ; i ++ ) {
const iNum = parseInt ( inputNums [ i ] , 10 ) ;
if ( iNum > largest ) {
largest = iNum ;
2019-04-02 15:58:36 +00:00
}
}
return largest ;
2018-07-27 13:37:38 +00:00
}
2019-04-03 11:00:47 +00:00
/ * *
* Gets the smallest inputNum
*
* @ returns { number }
* /
getSmallestInputNum ( ) {
let smallest = this . getLargestInputNum ( ) ;
2019-04-25 15:32:48 +00:00
const inputNums = Object . keys ( this . outputs ) ;
for ( let i = 0 ; i < inputNums . length ; i ++ ) {
const iNum = parseInt ( inputNums [ i ] , 10 ) ;
if ( iNum < smallest ) {
smallest = iNum ;
2019-04-03 11:00:47 +00:00
}
}
return smallest ;
}
2018-07-27 13:37:38 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Gets the previous inputNum
2018-07-27 13:37:38 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum - The current input number
* @ returns { number }
2018-07-27 13:37:38 +00:00
* /
2019-04-02 15:58:36 +00:00
getPreviousInputNum ( inputNum ) {
2019-04-03 11:00:47 +00:00
let num = this . getSmallestInputNum ( ) ;
2019-04-25 15:32:48 +00:00
const inputNums = Object . keys ( this . outputs ) ;
for ( let i = 0 ; i < inputNums . length ; i ++ ) {
const iNum = parseInt ( inputNums [ i ] , 10 ) ;
if ( iNum < inputNum ) {
if ( iNum > num ) {
num = iNum ;
2019-04-02 15:58:36 +00:00
}
}
}
return num ;
2018-07-27 13:37:38 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Gets the next inputNum
*
* @ param { number } inputNum - The current input number
* @ returns { number }
2018-07-27 13:37:38 +00:00
* /
2019-04-02 15:58:36 +00:00
getNextInputNum ( inputNum ) {
let num = this . getLargestInputNum ( ) ;
2019-04-25 15:32:48 +00:00
const inputNums = Object . keys ( this . outputs ) ;
for ( let i = 0 ; i < inputNums . length ; i ++ ) {
const iNum = parseInt ( inputNums [ i ] , 10 ) ;
if ( iNum > inputNum ) {
if ( iNum < num ) {
num = iNum ;
2019-04-02 15:58:36 +00:00
}
}
}
return num ;
2018-06-03 16:33:13 +00:00
}
2019-03-09 06:25:27 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Removes a tab and it ' s corresponding output
2019-03-09 06:25:27 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
2019-03-09 06:25:27 +00:00
* /
2019-04-02 15:58:36 +00:00
removeTab ( inputNum ) {
2019-04-03 11:00:47 +00:00
let activeTab = this . getActiveTab ( ) ;
2019-04-25 15:32:48 +00:00
if ( this . getOutput ( inputNum ) === - 1 ) return ;
2019-03-09 06:25:27 +00:00
2019-04-02 15:58:36 +00:00
const tabElement = this . getTabItem ( inputNum ) ;
2019-03-09 06:25:27 +00:00
2019-04-02 15:58:36 +00:00
this . removeOutput ( inputNum ) ;
if ( tabElement !== null ) {
// find new tab number?
2019-04-03 11:00:47 +00:00
if ( inputNum === activeTab ) {
activeTab = this . getPreviousInputNum ( activeTab ) ;
if ( activeTab === this . getActiveTab ( ) ) {
activeTab = this . getNextInputNum ( activeTab ) ;
}
}
this . refreshTabs ( activeTab ) ;
}
}
/ * *
* Redraw the entire tab bar to remove any outdated tabs
* @ param { number } activeTab
* /
refreshTabs ( activeTab ) {
const tabsList = document . getElementById ( "output-tabs" ) ;
let newInputs = this . getNearbyNums ( activeTab , "right" ) ;
if ( newInputs . length < this . maxTabs ) {
newInputs = this . getNearbyNums ( activeTab , "left" ) ;
}
for ( let i = tabsList . children . length - 1 ; i >= 0 ; i -- ) {
tabsList . children . item ( i ) . remove ( ) ;
}
for ( let i = 0 ; i < newInputs . length ; i ++ ) {
tabsList . appendChild ( this . createTabElement ( newInputs [ i ] ) ) ;
this . displayTabInfo ( newInputs [ i ] ) ;
}
if ( newInputs . length > 1 ) {
tabsList . parentElement . style . display = "block" ;
document . getElementById ( "output-wrapper" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
document . getElementById ( "output-highlighter" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
document . getElementById ( "output-file" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
document . getElementById ( "output-loader" ) . style . height = "calc(100% - var(--tab-height) - var(--title-height))" ;
2019-04-25 15:32:48 +00:00
document . getElementById ( "save-all-to-file" ) . style . display = "inline-block" ;
2019-04-03 11:00:47 +00:00
} else {
tabsList . parentElement . style . display = "none" ;
document . getElementById ( "output-wrapper" ) . style . height = "calc(100% - var(--title-height))" ;
document . getElementById ( "output-highlighter" ) . style . height = "calc(100% - var(--title-height))" ;
document . getElementById ( "output-file" ) . style . height = "calc(100% - var(--title-height))" ;
document . getElementById ( "output-loader" ) . style . height = "calc(100% - var(--title-height))" ;
2019-04-25 15:32:48 +00:00
document . getElementById ( "save-all-to-file" ) . style . display = "none" ;
2019-04-02 15:58:36 +00:00
}
2019-04-03 11:00:47 +00:00
this . changeTab ( activeTab ) ;
2019-03-09 06:25:27 +00:00
}
2019-03-27 09:05:10 +00:00
/ * *
2019-04-02 15:58:36 +00:00
* Creates a new tab element to be added to the tab bar
2019-03-27 09:05:10 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
2019-03-27 09:05:10 +00:00
* /
2019-04-02 15:58:36 +00:00
createTabElement ( inputNum ) {
2019-03-27 09:05:10 +00:00
const newTab = document . createElement ( "li" ) ;
2019-04-02 15:58:36 +00:00
newTab . setAttribute ( "inputNum" , inputNum . toString ( ) ) ;
2019-03-27 09:05:10 +00:00
const newTabContent = document . createElement ( "div" ) ;
newTabContent . classList . add ( "output-tab-content" ) ;
2019-04-02 15:58:36 +00:00
newTabContent . innerText = ` Tab ${ inputNum . toString ( ) } ` ;
2019-03-27 09:05:10 +00:00
2019-04-02 15:58:36 +00:00
// Do we want remove tab button on output?
2019-03-27 09:05:10 +00:00
newTab . appendChild ( newTabContent ) ;
2019-04-02 15:58:36 +00:00
return newTab ;
2019-03-27 09:05:10 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Gets the number of the current active tab
2019-03-27 09:05:10 +00:00
*
2019-04-02 15:58:36 +00:00
* @ returns { number }
2019-03-27 09:05:10 +00:00
* /
2019-04-02 15:58:36 +00:00
getActiveTab ( ) {
2019-04-26 14:15:44 +00:00
return this . activeTab ;
2019-03-27 13:48:54 +00:00
}
/ * *
2019-04-02 15:58:36 +00:00
* Gets the li element for a tab
2019-03-27 13:48:54 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
2019-03-27 13:48:54 +00:00
* /
2019-04-02 15:58:36 +00:00
getTabItem ( inputNum ) {
const tabs = document . getElementById ( "output-tabs" ) . children ;
for ( let i = 0 ; i < tabs . length ; i ++ ) {
if ( parseInt ( tabs . item ( i ) . getAttribute ( "inputNum" ) , 10 ) === inputNum ) {
return tabs . item ( i ) ;
}
2019-03-27 13:48:54 +00:00
}
}
/ * *
2019-04-02 15:58:36 +00:00
* Display output information in the tab header
2019-03-27 13:48:54 +00:00
*
2019-04-02 15:58:36 +00:00
* @ param { number } inputNum
2019-03-27 13:48:54 +00:00
* /
2019-04-02 15:58:36 +00:00
displayTabInfo ( inputNum ) {
const tabItem = this . getTabItem ( inputNum ) ;
2019-03-27 13:48:54 +00:00
2019-04-02 15:58:36 +00:00
if ( ! tabItem ) return ;
2019-03-27 13:48:54 +00:00
2019-04-02 15:58:36 +00:00
const tabContent = tabItem . firstElementChild ;
2019-03-27 13:48:54 +00:00
2019-04-02 15:58:36 +00:00
tabContent . innerText = ` Tab ${ inputNum } ` ;
2019-03-27 13:48:54 +00:00
2019-03-27 09:05:10 +00:00
}
2019-04-25 15:32:48 +00:00
/ * *
* Displays information about the output .
*
* @ param { number } length - The length of the current output string
* @ param { number } lines - The number of the lines in the current output string
* @ param { number } duration - The length of time ( ms ) it took to generate the output
* /
setOutputInfo ( length , lines , duration ) {
if ( ! length ) return ;
let width = length . toString ( ) . length ;
width = width < 4 ? 4 : width ;
const lengthStr = length . toString ( ) . padStart ( width , " " ) . replace ( / /g , " " ) ;
const timeStr = ( duration . toString ( ) + "ms" ) . padStart ( width , " " ) . replace ( / /g , " " ) ;
let msg = "time: " + timeStr + "<br>length: " + lengthStr ;
if ( typeof lines === "number" ) {
const linesStr = lines . toString ( ) . padStart ( width , " " ) . replace ( / /g , " " ) ;
msg += "<br>lines: " + linesStr ;
}
document . getElementById ( "output-info" ) . innerHTML = msg ;
document . getElementById ( "input-selection-info" ) . innerHTML = "" ;
document . getElementById ( "output-selection-info" ) . innerHTML = "" ;
}
/ * *
* Triggers the BackgroundWorker to attempt Magic on the current output .
* /
backgroundMagic ( ) {
this . hideMagicButton ( ) ;
2019-05-01 13:46:05 +00:00
if ( ! this . app . options . autoMagic || ! this . getActive ( ) ) return ;
2019-04-25 15:32:48 +00:00
const sample = this . getActive ( ) . slice ( 0 , 1000 ) || "" ;
2019-05-01 13:46:05 +00:00
if ( sample . length || sample . byteLength ) {
2019-04-25 15:32:48 +00:00
this . manager . background . magic ( sample ) ;
}
}
/ * *
* Handles the results of a background Magic call .
*
* @ param { Object [ ] } options
* /
backgroundMagicResult ( options ) {
if ( ! options . length ||
! options [ 0 ] . recipe . length )
return ;
const currentRecipeConfig = this . app . getRecipeConfig ( ) ;
const newRecipeConfig = currentRecipeConfig . concat ( options [ 0 ] . recipe ) ;
const opSequence = options [ 0 ] . recipe . map ( o => o . op ) . join ( ", " ) ;
this . showMagicButton ( opSequence , options [ 0 ] . data , newRecipeConfig ) ;
}
/ * *
* Handler for Magic click events .
*
* Loads the Magic recipe .
*
* @ fires Manager # statechange
* /
magicClick ( ) {
const magicButton = document . getElementById ( "magic" ) ;
this . app . setRecipeConfig ( JSON . parse ( magicButton . getAttribute ( "data-recipe" ) ) ) ;
window . dispatchEvent ( this . manager . statechange ) ;
this . hideMagicButton ( ) ;
}
/ * *
* Displays the Magic button with a title and adds a link to a complete recipe .
*
* @ param { string } opSequence
* @ param { string } result
* @ param { Object [ ] } recipeConfig
* /
showMagicButton ( opSequence , result , recipeConfig ) {
const magicButton = document . getElementById ( "magic" ) ;
magicButton . setAttribute ( "data-original-title" , ` <i> ${ opSequence } </i> will produce <span class="data-text">" ${ Utils . escapeHtml ( Utils . truncate ( result ) , 30 ) } "</span> ` ) ;
magicButton . setAttribute ( "data-recipe" , JSON . stringify ( recipeConfig ) , null , "" ) ;
magicButton . classList . remove ( "hidden" ) ;
}
/ * *
* Hides the Magic button and resets its values .
* /
hideMagicButton ( ) {
const magicButton = document . getElementById ( "magic" ) ;
magicButton . classList . add ( "hidden" ) ;
magicButton . setAttribute ( "data-recipe" , "" ) ;
magicButton . setAttribute ( "data-original-title" , "Magic!" ) ;
}
/ * *
* Handler for file slice display events .
* /
displayFileSlice ( ) {
const startTime = new Date ( ) . getTime ( ) ,
showFileOverlay = document . getElementById ( "show-file-overlay" ) ,
sliceFromEl = document . getElementById ( "output-file-slice-from" ) ,
sliceToEl = document . getElementById ( "output-file-slice-to" ) ,
sliceFrom = parseInt ( sliceFromEl . value , 10 ) ,
sliceTo = parseInt ( sliceToEl . value , 10 ) ,
str = Utils . arrayBufferToStr ( this . getActive ( ) . slice ( sliceFrom , sliceTo ) ) ;
document . getElementById ( "output-text" ) . classList . remove ( "blur" ) ;
showFileOverlay . style . display = "block" ;
}
/ * *
* Handler for copy click events .
* Copies the output to the clipboard
* /
copyClick ( ) {
const output = this . getActive ( ) ;
// Create invisible textarea to populate with the raw dish string (not the printable version that
// contains dots instead of the actual bytes)
const textarea = document . createElement ( "textarea" ) ;
textarea . style . position = "fixed" ;
textarea . style . top = 0 ;
textarea . style . left = 0 ;
textarea . style . width = 0 ;
textarea . style . height = 0 ;
textarea . style . border = "none" ;
textarea . value = output ;
document . body . appendChild ( textarea ) ;
let success = false ;
try {
textarea . select ( ) ;
success = output && document . execCommand ( "copy" ) ;
} catch ( err ) {
success = false ;
}
if ( success ) {
this . app . alert ( "Copied raw output successfully." , 2000 ) ;
} else {
this . app . alert ( "Sorry, the output could not be copied." , 3000 ) ;
}
// Clean up
document . body . removeChild ( textarea ) ;
}
2018-05-15 17:36:45 +00:00
}
export default OutputWaiter ;