@alauda-fe/common
Version:
Alauda frontend team common codes.
140 lines • 19 kB
JavaScript
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"]}