2022-01-16 00:54:15 +01:00
|
|
|
// create a xterm for replay
|
2022-01-15 13:14:17 +01:00
|
|
|
function createReplayTerminal() {
|
2022-01-16 00:54:15 +01:00
|
|
|
// vscode-snazzy https://github.com/Tyriar/vscode-snazzy
|
|
|
|
// copied from xterm.js website
|
|
|
|
var baseTheme = {
|
|
|
|
foreground: '#eff0eb',
|
|
|
|
background: '#282a36',
|
|
|
|
selection: '#97979b33',
|
|
|
|
black: '#282a36',
|
|
|
|
brightBlack: '#686868',
|
|
|
|
red: '#ff5c57',
|
|
|
|
brightRed: '#ff5c57',
|
|
|
|
green: '#5af78e',
|
|
|
|
brightGreen: '#5af78e',
|
|
|
|
yellow: '#f3f99d',
|
|
|
|
brightYellow: '#f3f99d',
|
|
|
|
blue: '#57c7ff',
|
|
|
|
brightBlue: '#57c7ff',
|
|
|
|
magenta: '#ff6ac1',
|
|
|
|
brightMagenta: '#ff6ac1',
|
|
|
|
cyan: '#9aedfe',
|
|
|
|
brightCyan: '#9aedfe',
|
|
|
|
white: '#f1f1f0',
|
|
|
|
brightWhite: '#eff0eb'
|
|
|
|
};
|
|
|
|
|
|
|
|
const term = new Terminal({
|
|
|
|
fontFamily: `'Fira Code', ui-monospace,SFMono-Regular,'SF Mono',Menlo,Consolas,'Liberation Mono',monospace`,
|
|
|
|
fontSize: 12,
|
|
|
|
theme: baseTheme,
|
|
|
|
convertEol: true,
|
|
|
|
cursorBlink: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
term.open(document.getElementById('terminal_view'));
|
|
|
|
term.resize(124, 37);
|
|
|
|
|
|
|
|
const weblinksAddon = new WebLinksAddon.WebLinksAddon();
|
|
|
|
term.loadAddon(weblinksAddon);
|
|
|
|
|
|
|
|
// fit the xterm viewpoint to parent element
|
|
|
|
const fitAddon = new FitAddon.FitAddon();
|
|
|
|
term.loadAddon(fitAddon);
|
|
|
|
fitAddon.fit();
|
|
|
|
|
|
|
|
return term;
|
|
|
|
}
|
|
|
|
|
|
|
|
// sleep for ms seconds
|
|
|
|
function _sleep(ms) {
|
|
|
|
return new Promise(resolve => setTimeout(resolve, ms));
|
|
|
|
}
|
|
|
|
|
|
|
|
// we could sleep for a long time
|
|
|
|
// periodically check if we need to end replay.
|
|
|
|
// This is pretty ugly but the callback mess otherwise
|
2022-01-16 03:17:03 +01:00
|
|
|
async function sleep(ms, paused) {
|
2022-01-16 00:54:15 +01:00
|
|
|
var loop_cnt = parseInt(ms / 20) + 1
|
|
|
|
|
|
|
|
for (i = 0; i < loop_cnt; i++) {
|
2022-01-16 03:17:03 +01:00
|
|
|
if (paused()) {
|
|
|
|
return paused()
|
2022-01-16 00:54:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
await _sleep(20)
|
|
|
|
}
|
|
|
|
|
2022-01-16 03:17:03 +01:00
|
|
|
return paused()
|
2022-01-16 00:54:15 +01:00
|
|
|
}
|
|
|
|
// convert data to uint8array, we cannot convert it to string as
|
|
|
|
// it will mess up special characters
|
|
|
|
function base64ToUint8array(base64) {
|
|
|
|
var raw = window.atob(base64);
|
|
|
|
var rawLength = raw.length;
|
|
|
|
var array = new Uint8Array(new ArrayBuffer(rawLength));
|
|
|
|
|
|
|
|
for (i = 0; i < rawLength; i++) {
|
|
|
|
array[i] = raw.charCodeAt(i);
|
|
|
|
}
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2022-01-18 19:06:25 +01:00
|
|
|
async function play_ctrl(term, session, start, total_dur, paused, shifted, prog) {
|
|
|
|
var cur = 0
|
|
|
|
var new_pos = -1
|
|
|
|
|
|
|
|
start = parseInt(total_dur * start / 100)
|
|
|
|
term.reset()
|
|
|
|
|
|
|
|
for (const item of session) {
|
|
|
|
new_pos = shifted()
|
|
|
|
|
|
|
|
if (new_pos != -1) {
|
|
|
|
return new_pos
|
|
|
|
}
|
|
|
|
|
|
|
|
// we will blast through the beginning of the session
|
|
|
|
if (cur >= start) {
|
|
|
|
// we are cheating a little bit here, we do not want to wait for too long
|
2022-01-18 19:11:39 +01:00
|
|
|
exit = await sleep(Math.min(item.Duration, 1000), paused)
|
2022-01-18 19:06:25 +01:00
|
|
|
|
|
|
|
if (exit) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
term.write(base64ToUint8array(item.Data))
|
|
|
|
cur += item.Duration
|
|
|
|
|
|
|
|
if (cur > start) {
|
|
|
|
prog(parseInt(cur * 100 / total_dur))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1
|
|
|
|
}
|
2022-01-16 00:54:15 +01:00
|
|
|
// replay session
|
|
|
|
// term: xterm, path: session file to replay,
|
|
|
|
// start: start position to replay in percentile, range 0-100
|
2022-01-18 19:06:25 +01:00
|
|
|
// paused: callback whether to stop play
|
|
|
|
// prog: callback to update the progress bar
|
|
|
|
// shift: callback whether we should change position
|
|
|
|
// end: callback when playback is finished
|
|
|
|
async function replay_session(term, path, start, paused, shifted, prog, end) {
|
2022-01-16 00:54:15 +01:00
|
|
|
var session
|
2022-01-18 19:06:25 +01:00
|
|
|
var total_dur = 0
|
|
|
|
var cur = 0
|
|
|
|
var new_pos = 0
|
|
|
|
var ret = 0
|
2022-01-16 00:54:15 +01:00
|
|
|
|
|
|
|
// read file from server
|
|
|
|
await fetch(path)
|
|
|
|
.then(res => res.json())
|
|
|
|
.then(out => {
|
|
|
|
session = out
|
|
|
|
})
|
|
|
|
|
|
|
|
//calculate the total duration
|
|
|
|
for (const item of session) {
|
|
|
|
item.Duration = parseInt(item.Duration / 1000000)
|
|
|
|
total_dur += item.Duration
|
|
|
|
}
|
|
|
|
|
2022-01-18 19:06:25 +01:00
|
|
|
console.log("Total duration:", total_dur, "start replay on position", start)
|
2022-01-16 00:54:15 +01:00
|
|
|
|
2022-01-18 19:06:25 +01:00
|
|
|
while (true) {
|
|
|
|
ret = await play_ctrl(term, session, start, total_dur, paused, shifted, prog)
|
|
|
|
if (ret == -1) {
|
|
|
|
break
|
2022-01-16 00:54:15 +01:00
|
|
|
}
|
|
|
|
|
2022-01-18 19:06:25 +01:00
|
|
|
start = ret
|
2022-01-16 00:54:15 +01:00
|
|
|
}
|
2022-01-16 03:17:03 +01:00
|
|
|
|
2022-01-18 19:06:25 +01:00
|
|
|
term.reset()
|
2022-01-16 03:17:03 +01:00
|
|
|
end()
|
2022-01-16 00:54:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function Init() {
|
|
|
|
let term = createReplayTerminal();
|
|
|
|
var str = [
|
|
|
|
' ┌────────────────────────────────────────────────────────────────────────────┐\n',
|
|
|
|
' │ \u001b[32;1mhttps://github.com/syssecfsu/witty\x1b[0m <- click it! │\n',
|
|
|
|
' └────────────────────────────────────────────────────────────────────────────┘\n',
|
|
|
|
''
|
|
|
|
].join('');
|
|
|
|
|
|
|
|
term.writeln(str);
|
|
|
|
|
|
|
|
// adjust the progress bar size to that of terminal
|
|
|
|
var view = document.querySelector("#terminal")
|
|
|
|
var pbar = document.querySelector("#replay-control")
|
|
|
|
pbar.setAttribute("style", "width:" + (view.offsetWidth - 32) + "px");
|
|
|
|
|
|
|
|
return term
|
|
|
|
}
|