UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

140 lines 19 kB
import { __decorate, __metadata } from "tslib"; import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, Input, Output, ViewChild, } from '@angular/core'; import { debounce } from 'lodash-es'; import { Observable } from 'rxjs'; import { Terminal } from 'xterm'; import { FitAddon } from 'xterm-addon-fit'; import { SearchAddon } from 'xterm-addon-search'; import { WebLinksAddon } from 'xterm-addon-web-links'; import { clearWindowsXtermTools, setWindowsXtermTools, } from '../core/public-api'; import { ObservableInput, ValueHook } from '../core/utils/decorators'; import { EffectDirectiveModule } from '../effect-directive/module'; import { XTERM_DARK_THEME, XTERM_LIGHT_THEME } from '../exec/constants'; import { initRendererAddon } from '../exec/terminal/component'; import { CommonLayoutModule } from '../page-scaffold/page-header/common-layout/common-layout.module'; import * as i0 from "@angular/core"; const _c0 = ["anchor"]; const SHELL_THEMES = { dark: XTERM_DARK_THEME, light: XTERM_LIGHT_THEME, }; export class XtermComponent { constructor() { this.feedLines = 1; this.searchAddonInit = new EventEmitter(); this.terminalInit = new EventEmitter(); this.searchAddon = new SearchAddon(); this.baseTerminalOptions = { fontSize: 12, lineHeight: 1.5, letterSpacing: 0, fontWeight: 'normal', fontFamily: 'Menlo, Monaco, "Consolas", "Courier New", monospace', cursorBlink: false, convertEol: true, scrollback: Number.MAX_SAFE_INTEGER, allowProposedApi: true, }; } initFitAddon() { this.fitAddon = new FitAddon(); const resizeObserver = new ResizeObserver(debounce(([entry]) => { if (entry.contentRect.width !== 0) { this.fitAddon.fit(); } }, 200)); resizeObserver.observe(this.terminalWrapper.nativeElement); return this.fitAddon; } initSearchAddon() { this.searchAddon = new SearchAddon(); this.searchAddonInit.next(this.searchAddon); return this.searchAddon; } initXterm() { this.terminal = new Terminal(this.baseTerminalOptions); this.terminal.loadAddon(this.initFitAddon()); this.terminal.loadAddon(this.initSearchAddon()); this.terminal.loadAddon(new WebLinksAddon()); this.terminal.loadAddon(initRendererAddon()); this.terminal.open(this.terminalWrapper.nativeElement); this.setTheme(); this.bindCopyEvent(); setWindowsXtermTools(this.terminal); this.terminalInit.next(this.terminal); } ngAfterViewInit() { this.initXterm(); this.logs$.subscribe(logs => { this.fitAddon.fit(); this.write(logs, () => { requestAnimationFrame(() => this.terminal.scrollToBottom()); }); }); } ngOnDestroy() { if (this.terminal) { this.terminal.dispose(); } clearWindowsXtermTools(); } write(logs, callback) { if (this.terminal?.write) { this.terminal.clear(); Array.isArray(logs) ? logs.forEach(v => this.terminal.write(v, callback)) : this.terminal.write(logs || '', callback); } } bindCopyEvent() { this.terminal.attachCustomKeyEventHandler((e) => { if (e.ctrlKey && e.key?.toLowerCase() === 'c') { document.execCommand('copy'); } return false; }); } setTheme(theme = this.theme) { if (!this.terminal) { return; } this.terminal.options.theme = SHELL_THEMES[theme]; } static { this.ɵfac = function XtermComponent_Factory(t) { return new (t || XtermComponent)(); }; } static { this.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: XtermComponent, selectors: [["acl-xterm"]], viewQuery: function XtermComponent_Query(rf, ctx) { if (rf & 1) { i0.ɵɵviewQuery(_c0, 7); } if (rf & 2) { let _t; i0.ɵɵqueryRefresh(_t = i0.ɵɵloadQuery()) && (ctx.terminalWrapper = _t.first); } }, inputs: { theme: "theme", logs: "logs" }, outputs: { searchAddonInit: "searchAddonInit", terminalInit: "terminalInit" }, standalone: true, features: [i0.ɵɵStandaloneFeature], decls: 2, vars: 0, consts: [["anchor", ""], [1, "logs-xterm"]], template: function XtermComponent_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelement(0, "div", 1, 0); } }, dependencies: [CommonModule, CommonLayoutModule, EffectDirectiveModule], styles: [".logs-xterm[_ngcontent-%COMP%]{color:rgb(var(--aui-color-n-8));width:100%;height:100%}.logs-xterm[_ngcontent-%COMP%] .xterm{padding:0 20px;height:100%}.logs-xterm[_ngcontent-%COMP%] .xterm .xterm-screen{pointer-events:none}.logs-xterm[_ngcontent-%COMP%] .xterm .xterm-viewport{width:100%!important;border-radius:0 0 2px 2px}.logs-xterm[_ngcontent-%COMP%] .xterm .xterm-viewport::-webkit-scrollbar{width:8px;height:4px}.logs-xterm[_ngcontent-%COMP%] .xterm .xterm-viewport::-webkit-scrollbar-thumb{border-radius:4px;background-color:rgb(var(--aui-color-n-7))}.logs-xterm[_ngcontent-%COMP%] .xterm .xterm-viewport::-webkit-scrollbar-thumb:hover{background-color:rgb(var(--aui-color-n-5))}.logs-xterm[_ngcontent-%COMP%] .xterm .xterm-viewport::-webkit-scrollbar-corner{background-color:transparent}"], changeDetection: 0 }); } } __decorate([ ValueHook(function (theme) { this.setTheme(theme); }), __metadata("design:type", String) ], XtermComponent.prototype, "theme", void 0); __decorate([ ObservableInput(), __metadata("design:type", Observable) ], XtermComponent.prototype, "logs$", void 0); (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(XtermComponent, [{ type: Component, args: [{ selector: 'acl-xterm', template: '<div class="logs-xterm" #anchor></div>', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, CommonLayoutModule, EffectDirectiveModule], styles: [".logs-xterm{color:rgb(var(--aui-color-n-8));width:100%;height:100%}.logs-xterm ::ng-deep .xterm{padding:0 20px;height:100%}.logs-xterm ::ng-deep .xterm .xterm-screen{pointer-events:none}.logs-xterm ::ng-deep .xterm .xterm-viewport{width:100%!important;border-radius:0 0 2px 2px}.logs-xterm ::ng-deep .xterm .xterm-viewport::-webkit-scrollbar{width:8px;height:4px}.logs-xterm ::ng-deep .xterm .xterm-viewport::-webkit-scrollbar-thumb{border-radius:4px;background-color:rgb(var(--aui-color-n-7))}.logs-xterm ::ng-deep .xterm .xterm-viewport::-webkit-scrollbar-thumb:hover{background-color:rgb(var(--aui-color-n-5))}.logs-xterm ::ng-deep .xterm .xterm-viewport::-webkit-scrollbar-corner{background-color:transparent}\n"] }] }], null, { theme: [{ type: Input }], logs$: [], logs: [{ type: Input }], terminalWrapper: [{ type: ViewChild, args: ['anchor', { static: true }] }], searchAddonInit: [{ type: Output }], terminalInit: [{ type: Output }] }); })(); (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(XtermComponent, { className: "XtermComponent" }); })(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"xterm.component.js","sourceRoot":"","sources":["../../../../../libs/common/src/xterm/xterm.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Theme } from '@alauda/ui';\nimport { CommonModule } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport { debounce } from 'lodash-es';\nimport { Observable } from 'rxjs';\nimport { ITerminalOptions, ITheme, Terminal } from 'xterm';\nimport { FitAddon } from 'xterm-addon-fit';\nimport { SearchAddon } from 'xterm-addon-search';\nimport { WebLinksAddon } from 'xterm-addon-web-links';\n\nimport {\n  clearWindowsXtermTools,\n  setWindowsXtermTools,\n} from '../core/public-api';\nimport { ObservableInput, ValueHook } from '../core/utils/decorators';\nimport { EffectDirectiveModule } from '../effect-directive/module';\nimport { XTERM_DARK_THEME, XTERM_LIGHT_THEME } from '../exec/constants';\nimport { initRendererAddon } from '../exec/terminal/component';\nimport { CommonLayoutModule } from '../page-scaffold/page-header/common-layout/common-layout.module';\n\nconst SHELL_THEMES: Record<Theme, ITheme> = {\n  dark: XTERM_DARK_THEME,\n  light: XTERM_LIGHT_THEME,\n};\n\n@Component({\n  selector: 'acl-xterm',\n  template: '<div class=\"logs-xterm\" #anchor></div>',\n  styleUrls: ['xterm.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [CommonModule, CommonLayoutModule, EffectDirectiveModule],\n})\nexport class XtermComponent implements AfterViewInit, OnDestroy {\n  @ValueHook(function (theme: Theme) {\n    this.setTheme(theme);\n  })\n  @Input()\n  theme: Theme;\n\n  feedLines = 1;\n\n  @ObservableInput()\n  logs$: Observable<string[] | string>;\n\n  @Input()\n  logs: string[] | string;\n\n  @ViewChild('anchor', { static: true })\n  terminalWrapper: ElementRef;\n\n  @Output()\n  searchAddonInit = new EventEmitter<SearchAddon>();\n\n  @Output()\n  terminalInit = new EventEmitter<Terminal>();\n\n  terminal: Terminal;\n  searchAddon = new SearchAddon();\n\n  fitAddon: FitAddon;\n\n  baseTerminalOptions: ITerminalOptions = {\n    fontSize: 12,\n    lineHeight: 1.5,\n    letterSpacing: 0,\n    fontWeight: 'normal',\n    fontFamily: 'Menlo, Monaco, \"Consolas\", \"Courier New\", monospace',\n    cursorBlink: false,\n    convertEol: true,\n    scrollback: Number.MAX_SAFE_INTEGER,\n    allowProposedApi: true,\n  };\n\n  private initFitAddon() {\n    this.fitAddon = new FitAddon();\n\n    const resizeObserver = new ResizeObserver(\n      debounce(([entry]) => {\n        if (entry.contentRect.width !== 0) {\n          this.fitAddon.fit();\n        }\n      }, 200),\n    );\n\n    resizeObserver.observe(this.terminalWrapper.nativeElement);\n\n    return this.fitAddon;\n  }\n\n  private initSearchAddon() {\n    this.searchAddon = new SearchAddon();\n    this.searchAddonInit.next(this.searchAddon);\n    return this.searchAddon;\n  }\n\n  private initXterm() {\n    this.terminal = new Terminal(this.baseTerminalOptions);\n    this.terminal.loadAddon(this.initFitAddon());\n    this.terminal.loadAddon(this.initSearchAddon());\n    this.terminal.loadAddon(new WebLinksAddon());\n    this.terminal.loadAddon(initRendererAddon());\n\n    this.terminal.open(this.terminalWrapper.nativeElement);\n    this.setTheme();\n    this.bindCopyEvent();\n\n    setWindowsXtermTools(this.terminal);\n    this.terminalInit.next(this.terminal);\n  }\n\n  ngAfterViewInit(): void {\n    this.initXterm();\n\n    this.logs$.subscribe(logs => {\n      this.fitAddon.fit();\n      this.write(logs, () => {\n        requestAnimationFrame(() => this.terminal.scrollToBottom());\n      });\n    });\n  }\n\n  ngOnDestroy() {\n    if (this.terminal) {\n      this.terminal.dispose();\n    }\n    clearWindowsXtermTools();\n  }\n\n  write(logs: string | string[], callback?: () => void) {\n    if (this.terminal?.write) {\n      this.terminal.clear();\n      Array.isArray(logs)\n        ? logs.forEach(v => this.terminal.write(v, callback))\n        : this.terminal.write(logs || '', callback);\n    }\n  }\n\n  bindCopyEvent() {\n    this.terminal.attachCustomKeyEventHandler((e: KeyboardEvent) => {\n      if (e.ctrlKey && e.key?.toLowerCase() === 'c') {\n        document.execCommand('copy');\n      }\n      return false;\n    });\n  }\n\n  setTheme(theme: Theme = this.theme) {\n    if (!this.terminal) {\n      return;\n    }\n\n    this.terminal.options.theme = SHELL_THEMES[theme];\n  }\n}\n"]}