122 lines
3.0 KiB
JavaScript
122 lines
3.0 KiB
JavaScript
class LogTerminal extends HTMLElement {
|
|
|
|
#maxlines = 10;
|
|
#maxstoredlines = 100;
|
|
#rootdoc = null;
|
|
#lines = [];
|
|
#index = null;
|
|
|
|
static get observedAttributes() { return ['index']; }
|
|
|
|
constructor(){
|
|
super()
|
|
this.#maxlines = this.getAttribute("maxlines")
|
|
this.#rootdoc = this.attachShadow({ "mode" : "open"})
|
|
this.render()
|
|
}
|
|
|
|
connectedCallback(){
|
|
this.reloadLines()
|
|
this.refresh()
|
|
}
|
|
|
|
attributeChangedCallback(name, oldValue, newValue) {
|
|
if(name === "index"){
|
|
this.#index = newValue
|
|
this.reloadLines()
|
|
}
|
|
}
|
|
|
|
get lines(){
|
|
return this.#lines
|
|
}
|
|
|
|
addLines(lines){
|
|
this.#lines = this.#lines.concat(lines)
|
|
this.refresh()
|
|
this.storeLines()
|
|
}
|
|
|
|
reloadLines(){
|
|
if(this.#index == null) return;
|
|
if(sessionStorage.getItem("logs-" + this.#index)){
|
|
this.#lines = JSON.parse(sessionStorage.getItem("logs-" + this.#index))
|
|
}else{
|
|
this.#lines = []
|
|
}
|
|
}
|
|
|
|
storeLines(){
|
|
if(this.#lines.length > this.#maxstoredlines){
|
|
this.#lines.splice(0, this.#lines.length - this.#maxstoredlines)
|
|
}
|
|
sessionStorage.setItem("logs-" + this.#index, JSON.stringify(this.#lines))
|
|
}
|
|
|
|
render(){
|
|
this.#rootdoc.innerHTML = `
|
|
<link href="https://fonts.googleapis.com/css2?family=VT323&display=swap" rel="stylesheet">
|
|
<style>
|
|
.terminal-container{
|
|
overflow: hidden;
|
|
}
|
|
.terminal{
|
|
background-color: black;
|
|
max-height: 10rem;
|
|
height: 10rem;
|
|
min-height:10rem;
|
|
padding: 5px;
|
|
overflow:auto;
|
|
}
|
|
|
|
.terminal > .line {
|
|
font-family: 'VT323', monospace;
|
|
color: #88ff00;
|
|
text-shadow: 1px 1px 1px rgba(0, 200, 0, .6), 0 0 .5em rgba(0, 200, 0, .6)
|
|
}
|
|
|
|
.terminal > .line.error {
|
|
color: #ff3300;
|
|
text-shadow: 1px 1px 1px rgba(200, 0, 0, .6), 0 0 .5em rgba(200, 0, 0, .6)
|
|
}
|
|
|
|
.terminal > .line.error {
|
|
color: rgba(255,255,255,0.6);
|
|
text-shadow: 1px 1px 1px rgba(200, 200, 200, .6), 0 0 .5em rgba(200, 0, 0, .6)
|
|
}
|
|
</style>
|
|
<template id="TERMINAL_TEMPLATE">
|
|
<div name="terminal" class="container">
|
|
<div class="terminal" id="t">
|
|
<div class="line"></div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
<div name="terminal" class="container"></div>
|
|
`
|
|
}
|
|
|
|
refresh(){
|
|
BSS.apply(this.#terminal_bss, this.#rootdoc)
|
|
const lt = this.#rootdoc.querySelector("div.terminal")
|
|
lt.scrollTop = lt.scrollHeight;
|
|
}
|
|
|
|
#terminal_bss = {
|
|
template : "#TERMINAL_TEMPLATE",
|
|
target : "div[name=terminal]",
|
|
in : ()=>this,
|
|
recurse : {
|
|
target : "div.line",
|
|
in: (e,d)=>d.lines,
|
|
apply: (e,d,i)=>{
|
|
if(d.source === "stderr") e.classList.add("error");
|
|
if(d.source === "infrastructure") e.classList.add("infrastructure");
|
|
e.innerHTML = `<span>${d.line}</span>`
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
window.customElements.define('d4s-ccp-logterminal', LogTerminal);
|