Files
Automaaval/dist/zacatraz/_internal/panel/models/terminal.ts
T
2026-03-14 21:48:05 +00:00

159 lines
3.9 KiB
TypeScript

import type * as p from "@bokehjs/core/properties"
import {div} from "@bokehjs/core/dom"
import {ModelEvent} from "@bokehjs/core/bokeh_events"
import type {Attrs} from "@bokehjs/core/types"
import {HTMLBox, HTMLBoxView, set_size} from "./layout"
export class KeystrokeEvent extends ModelEvent {
constructor(readonly key: string) {
super()
}
protected override get event_values(): Attrs {
return {model: this.origin, key: this.key}
}
static {
this.prototype.event_name = "keystroke"
}
}
export class TerminalView extends HTMLBoxView {
declare model: Terminal
term: any // Element
webLinksAddon: any
container: HTMLDivElement
_rendered: boolean
override connect_signals(): void {
super.connect_signals()
const {output, _clears} = this.model.properties
this.on_change(output, this.write)
this.on_change(_clears, this.clear)
}
override render(): void {
super.render()
this.container = div({class: "terminal-container"})
set_size(this.container, this.model)
this.term = this.getNewTerminal()
this.term.onData((value: any) => {
this.handleOnData(value)
})
this.webLinksAddon = this.getNewWebLinksAddon()
this.term.loadAddon(this.webLinksAddon)
this.term.open(this.container)
this.term.onRender(() => {
if (!this._rendered) {
this.fit()
}
})
this.write()
this.shadow_el.appendChild(this.container)
}
getNewTerminal(): any {
const wn = (window as any)
if (wn.Terminal) {
return new wn.Terminal(this.model.options)
} else {
return new wn.xtermjs.Terminal(this.model.options)
}
}
getNewWebLinksAddon(): any {
const wn = (window as any)
return new wn.WebLinksAddon.WebLinksAddon()
}
handleOnData(value: string): void {
this.model.trigger_event(new KeystrokeEvent(value))
}
write(): void {
const text = this.model.output
if (text == null || !text.length) {
return
}
// https://stackoverflow.com/questions/65367607/how-to-handle-new-line-in-xterm-js-while-writing-data-into-the-terminal
const cleaned = text.replace(/\r?\n/g, "\r\n")
// var text = Array.from(cleaned, (x) => x.charCodeAt(0))
this.term.write(cleaned)
}
clear(): void {
this.term.clear()
}
fit(): void {
const width = this.container.offsetWidth
const height = this.container.offsetHeight
const renderer = this.term._core._renderService
const cell_width = renderer.dimensions.actualCellWidth || 9
const cell_height = renderer.dimensions.actualCellHeight || 18
if (width == null || height == null || width <= 0 || height <= 0) {
return
}
const cols = Math.max(2, Math.floor(width / cell_width))
const rows = Math.max(1, Math.floor(height / cell_height))
if (this.term.rows !== rows || this.term.cols !== cols) {
this.term.resize(cols, rows)
}
this.model.ncols = cols
this.model.nrows = rows
this._rendered = true
}
override after_layout(): void {
super.after_layout()
if (this.container != null) {
this.fit()
}
}
}
export namespace Terminal {
export type Attrs = p.AttrsOf<Props>
export type Props = HTMLBox.Props & {
options: p.Property<any>
output: p.Property<string>
ncols: p.Property<number>
nrows: p.Property<number>
_clears: p.Property<number>
}
}
export interface Terminal extends Terminal.Attrs { }
// The Bokeh .ts model corresponding to the Bokeh .py model
export class Terminal extends HTMLBox {
declare properties: Terminal.Props
constructor(attrs?: Partial<Terminal.Attrs>) {
super(attrs)
}
static override __module__ = "panel.models.terminal"
static {
this.prototype.default_view = TerminalView
this.define<Terminal.Props>(({Any, Int, Str}) => ({
_clears: [ Int, 0 ],
options: [ Any, {} ],
output: [ Str, "" ],
ncols: [ Int, 0 ],
nrows: [ Int, 0 ],
}))
}
}