@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
101 lines • 18 kB
JavaScript
import { Component, ViewEncapsulation } from '@angular/core';
import { ActionBarItemComponent, C8yTranslatePipe, IconDirective, TitleComponent } from '@c8y/ngx-components';
import { Terminal } from '@xterm/xterm';
import { FitAddon } from '@xterm/addon-fit';
import { ShellAdapter } from './shell-adapter';
import { RemoteAccessService } from '@c8y/ngx-components/remote-access/data';
import { ActivatedRoute } from '@angular/router';
// workaround as we were unable to import this in component styles
import '@xterm/xterm/css/xterm.css';
import { NgIf } from '@angular/common';
import * as i0 from "@angular/core";
import * as i1 from "@c8y/ngx-components/remote-access/data";
import * as i2 from "@angular/router";
export class TerminalViewerComponent {
constructor(remoteAccess, activatedRoute) {
this.remoteAccess = remoteAccess;
this.activatedRoute = activatedRoute;
this.title = '';
this.container = null;
this.terminal = null;
this.socket = null;
this.firstDeviceMessageReceived = false;
this.configurationId = this.activatedRoute.snapshot.params.configurationId;
this.deviceId = this.activatedRoute.parent.snapshot.params.id;
}
ngOnDestroy() {
this.observer?.disconnect();
const stringToSend = 'exit\n';
const sendQueue = [];
for (let i = 0; i < stringToSend.length; i++) {
sendQueue.push(stringToSend.charCodeAt(i));
}
this.socket?.send(new Uint8Array(sendQueue));
this.socket?.close();
}
ngAfterViewInit() {
this.container = document.getElementById('terminal-screen');
this.terminal = new Terminal({
fontSize: 18,
fontFamily: 'consolas, monospace',
cursorBlink: true
});
const fitAddon = new FitAddon();
this.socket = new WebSocket(this.remoteAccess.getWebSocketUri(this.deviceId, this.configurationId), 'binary');
this.socket.binaryType = 'arraybuffer';
this.terminal.loadAddon(fitAddon);
this.terminal.open(this.container);
fitAddon.fit();
const shellAdapter = new ShellAdapter(this.terminal);
this.observer = new ResizeObserver(() => {
fitAddon.fit();
});
this.observer.observe(this.container);
this.terminal.writeln('\u001b[92mEstablishing Websocket connection... \u001b[39m \r\n');
this.terminal.focus();
this.socket.onopen = () => {
this.terminal.writeln('\u001b[92mWebsocket connection was established successfully, waiting for device... \u001b[39m \r\n');
};
this.socket.onclose = () => {
this.terminal.writeln('');
this.terminal.writeln('\r\n\u001b[91mDevice disconnected. \u001b[39m \r\n');
this.terminal.writeln('\r\n\u001b[91mWebsocket connection was closed. \u001b[39m \r\n');
};
this.socket.onmessage = message => {
if (!this.firstDeviceMessageReceived) {
this.firstDeviceMessageReceived = true;
this.terminal.writeln('\u001b[92mDevice connection was established successfully. \u001b[39m \r\n');
}
const filteredData = shellAdapter.filterReceiveData(message.data);
if (filteredData.dataToSend.length) {
this.socket.send(filteredData.dataToSend);
}
if (filteredData.dataToDisplay.length) {
this.terminal.write(filteredData.dataToDisplay);
}
};
this.terminal.onData(data => {
if (this.firstDeviceMessageReceived) {
const encodedToSend = shellAdapter.filterSendData(data);
if (encodedToSend.length) {
this.socket.send(encodedToSend);
}
}
});
}
toggleFullscreen() {
if (document.fullscreenElement) {
document.exitFullscreen();
}
else {
this.container.requestFullscreen();
}
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TerminalViewerComponent, deps: [{ token: i1.RemoteAccessService }, { token: i2.ActivatedRoute }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: TerminalViewerComponent, isStandalone: true, selector: "c8y-terminal-viewer", ngImport: i0, template: "<c8y-title>Terminal Viewer: {{ title | translate }}</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\" *ngIf=\"firstDeviceMessageReceived\">\n <button\n class=\"btn btn-link\"\n (click)=\"toggleFullscreen()\"\n >\n <i [c8yIcon]=\"'expand'\"></i>\n <span translate>Fullscreen</span>\n </button>\n </c8y-action-bar-item>\n\n<div class=\"content-fullpage\" >\n <div id=\"terminal-screen\" class=\"inner-scroll\"></div>\n</div>", dependencies: [{ kind: "component", type: TitleComponent, selector: "c8y-title", inputs: ["pageTitleUpdate"] }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }, { kind: "component", type: ActionBarItemComponent, selector: "c8y-action-bar-item", inputs: ["placement", "priority", "itemClass", "injector", "groupId", "inGroupPriority"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TerminalViewerComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-terminal-viewer', encapsulation: ViewEncapsulation.None, standalone: true, imports: [TitleComponent, C8yTranslatePipe, ActionBarItemComponent, IconDirective, NgIf], template: "<c8y-title>Terminal Viewer: {{ title | translate }}</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\" *ngIf=\"firstDeviceMessageReceived\">\n <button\n class=\"btn btn-link\"\n (click)=\"toggleFullscreen()\"\n >\n <i [c8yIcon]=\"'expand'\"></i>\n <span translate>Fullscreen</span>\n </button>\n </c8y-action-bar-item>\n\n<div class=\"content-fullpage\" >\n <div id=\"terminal-screen\" class=\"inner-scroll\"></div>\n</div>" }]
}], ctorParameters: () => [{ type: i1.RemoteAccessService }, { type: i2.ActivatedRoute }] });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"terminal-viewer.component.js","sourceRoot":"","sources":["../../../../remote-access/terminal-viewer/terminal-viewer.component.ts","../../../../remote-access/terminal-viewer/terminal-viewer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, OnDestroy, ViewEncapsulation } from '@angular/core';\nimport {\n  ActionBarItemComponent,\n  C8yTranslatePipe,\n  IconDirective,\n  TitleComponent\n} from '@c8y/ngx-components';\nimport { Terminal } from '@xterm/xterm';\nimport { FitAddon } from '@xterm/addon-fit';\nimport { ShellAdapter } from './shell-adapter';\nimport { RemoteAccessService } from '@c8y/ngx-components/remote-access/data';\nimport { ActivatedRoute } from '@angular/router';\n// workaround as we were unable to import this in component styles\nimport '@xterm/xterm/css/xterm.css';\nimport { NgIf } from '@angular/common';\n\n@Component({\n  selector: 'c8y-terminal-viewer',\n  templateUrl: './terminal-viewer.component.html',\n  encapsulation: ViewEncapsulation.None,\n  standalone: true,\n  imports: [TitleComponent, C8yTranslatePipe, ActionBarItemComponent, IconDirective, NgIf]\n})\nexport class TerminalViewerComponent implements AfterViewInit, OnDestroy {\n  title = '';\n  container: HTMLElement | null = null;\n  terminal: Terminal | null = null;\n  socket: WebSocket | null = null;\n  firstDeviceMessageReceived = false;\n  readonly configurationId: string;\n  readonly deviceId: string;\n  protected observer: ResizeObserver;\n\n  constructor(\n    private remoteAccess: RemoteAccessService,\n    private activatedRoute: ActivatedRoute\n  ) {\n    this.configurationId = this.activatedRoute.snapshot.params.configurationId;\n    this.deviceId = this.activatedRoute.parent.snapshot.params.id;\n  }\n\n  ngOnDestroy(): void {\n    this.observer?.disconnect();\n    const stringToSend = 'exit\\n';\n    const sendQueue = [];\n    for (let i = 0; i < stringToSend.length; i++) {\n      sendQueue.push(stringToSend.charCodeAt(i));\n    }\n    this.socket?.send(new Uint8Array(sendQueue));\n    this.socket?.close();\n  }\n\n  ngAfterViewInit(): void {\n    this.container = document.getElementById('terminal-screen');\n    this.terminal = new Terminal({\n      fontSize: 18,\n      fontFamily: 'consolas, monospace',\n      cursorBlink: true\n    });\n    const fitAddon = new FitAddon();\n    this.socket = new WebSocket(\n      this.remoteAccess.getWebSocketUri(this.deviceId, this.configurationId),\n      'binary'\n    );\n    this.socket.binaryType = 'arraybuffer';\n    this.terminal.loadAddon(fitAddon);\n    this.terminal.open(this.container);\n    fitAddon.fit();\n    const shellAdapter = new ShellAdapter(this.terminal);\n    this.observer = new ResizeObserver(() => {\n      fitAddon.fit();\n    });\n    this.observer.observe(this.container);\n\n    this.terminal.writeln('\\u001b[92mEstablishing Websocket connection... \\u001b[39m \\r\\n');\n    this.terminal.focus();\n    this.socket.onopen = () => {\n      this.terminal.writeln(\n        '\\u001b[92mWebsocket connection was established successfully, waiting for device... \\u001b[39m \\r\\n'\n      );\n    };\n\n    this.socket.onclose = () => {\n      this.terminal.writeln('');\n      this.terminal.writeln('\\r\\n\\u001b[91mDevice disconnected. \\u001b[39m \\r\\n');\n      this.terminal.writeln('\\r\\n\\u001b[91mWebsocket connection was closed. \\u001b[39m \\r\\n');\n    };\n\n    this.socket.onmessage = message => {\n      if (!this.firstDeviceMessageReceived) {\n        this.firstDeviceMessageReceived = true;\n        this.terminal.writeln(\n          '\\u001b[92mDevice connection was established successfully. \\u001b[39m \\r\\n'\n        );\n      }\n      const filteredData = shellAdapter.filterReceiveData(message.data);\n      if (filteredData.dataToSend.length) {\n        this.socket.send(filteredData.dataToSend);\n      }\n      if (filteredData.dataToDisplay.length) {\n        this.terminal.write(filteredData.dataToDisplay);\n      }\n    };\n\n    this.terminal.onData(data => {\n      if (this.firstDeviceMessageReceived) {\n        const encodedToSend = shellAdapter.filterSendData(data);\n\n        if (encodedToSend.length) {\n          this.socket.send(encodedToSend);\n        }\n      }\n    });\n  }\n\n  toggleFullscreen() {\n    if (document.fullscreenElement) {\n      document.exitFullscreen();\n    } else {\n      this.container.requestFullscreen();\n    }\n  }\n}\n","<c8y-title>Terminal Viewer: {{ title | translate }}</c8y-title>\n\n<c8y-action-bar-item [placement]=\"'right'\" *ngIf=\"firstDeviceMessageReceived\">\n    <button\n      class=\"btn btn-link\"\n      (click)=\"toggleFullscreen()\"\n    >\n      <i [c8yIcon]=\"'expand'\"></i>\n      <span translate>Fullscreen</span>\n    </button>\n  </c8y-action-bar-item>\n\n<div class=\"content-fullpage\" >\n    <div id=\"terminal-screen\" class=\"inner-scroll\"></div>\n</div>"]}