beccabroek | 457ca04 | 2018-09-07 17:18:24 -0500 | [diff] [blame] | 1 | import {Terminal} from 'xterm'; |
| 2 | import style from 'xterm/dist/xterm.css'; |
| 3 | import * as attach from 'xterm/lib/addons/attach/attach'; |
| 4 | import * as fit from 'xterm/lib/addons/fit/fit'; |
Yang Cheng | 6d1d600 | 2019-02-27 11:16:24 +0800 | [diff] [blame] | 5 | var configJSON = require('../../../config.json'); |
| 6 | if (configJSON.keyType == 'VT100+') { |
| 7 | var vt100PlusKey = require('./vt100plus'); |
| 8 | } |
beccabroek | 457ca04 | 2018-09-07 17:18:24 -0500 | [diff] [blame] | 9 | |
Yang Cheng | 6d1d600 | 2019-02-27 11:16:24 +0800 | [diff] [blame] | 10 | var customKeyHandlers = function(ev) { |
| 11 | if (configJSON.keyType == 'VT100+') { |
| 12 | return vt100PlusKey.customVT100PlusKey(ev, this); |
| 13 | } |
| 14 | return true; |
| 15 | }; |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 16 | |
Yang Cheng | 511a2bb | 2019-03-06 22:20:35 +0800 | [diff] [blame] | 17 | function measureChar(term) { |
| 18 | var span = document.createElement('span'); |
| 19 | var fontFamily = 'courier-new'; |
| 20 | var fontSize = 15; |
| 21 | var rect; |
| 22 | |
| 23 | span.textContent = 'W'; |
| 24 | try { |
| 25 | fontFamily = term.getOption('fontFamily'); |
| 26 | fontSize = term.getOption('fontSize'); |
| 27 | } catch (err) { |
| 28 | console.log('get option failure'); |
| 29 | } |
| 30 | span.style.fontFamily = fontFamily; |
| 31 | span.style.fontSize = fontSize + 'px'; |
| 32 | document.body.appendChild(span); |
| 33 | rect = span.getBoundingClientRect(); |
| 34 | document.body.removeChild(span); |
| 35 | return rect; |
| 36 | } |
| 37 | |
Ed Tanous | d871359 | 2020-07-16 07:33:07 -0700 | [diff] [blame] | 38 | // Add a TextEncoder polyfill to handle ie9 properly. (does anyone still use |
| 39 | // that anymore? Grabbed from |
| 40 | // https://developer.mozilla.org/en-US/docs/Web/API/TextEncoder#Polyfill |
| 41 | if (typeof TextEncoder === 'undefined') { |
| 42 | TextEncoder = function TextEncoder() {}; |
| 43 | TextEncoder.prototype.encode = function encode(str) { |
| 44 | 'use strict'; |
| 45 | var Len = str.length, resPos = -1; |
| 46 | // The Uint8Array's length must be at least 3x the length of the string |
| 47 | // because an invalid UTF-16 |
| 48 | // takes up the equivelent space of 3 UTF-8 characters to encode it |
| 49 | // properly. However, Array's have an auto expanding length and 1.5x should |
| 50 | // be just the right balance for most uses. |
| 51 | var resArr = typeof Uint8Array === 'undefined' ? new Array(Len * 1.5) : |
| 52 | new Uint8Array(Len * 3); |
| 53 | for (var point = 0, nextcode = 0, i = 0; i !== Len;) { |
| 54 | point = str.charCodeAt(i), i += 1; |
| 55 | if (point >= 0xD800 && point <= 0xDBFF) { |
| 56 | if (i === Len) { |
| 57 | resArr[resPos += 1] = 0xef /*0b11101111*/; |
| 58 | resArr[resPos += 1] = 0xbf /*0b10111111*/; |
| 59 | resArr[resPos += 1] = 0xbd /*0b10111101*/; |
| 60 | break; |
| 61 | } |
| 62 | // https://mathiasbynens.be/notes/javascript-encoding#surrogate-formulae |
| 63 | nextcode = str.charCodeAt(i); |
| 64 | if (nextcode >= 0xDC00 && nextcode <= 0xDFFF) { |
| 65 | point = (point - 0xD800) * 0x400 + nextcode - 0xDC00 + 0x10000; |
| 66 | i += 1; |
| 67 | if (point > 0xffff) { |
| 68 | resArr[resPos += 1] = (0x1e /*0b11110*/ << 3) | (point >>> 18); |
| 69 | resArr[resPos += 1] = |
| 70 | (0x2 /*0b10*/ << 6) | ((point >>> 12) & 0x3f /*0b00111111*/); |
| 71 | resArr[resPos += 1] = |
| 72 | (0x2 /*0b10*/ << 6) | ((point >>> 6) & 0x3f /*0b00111111*/); |
| 73 | resArr[resPos += 1] = |
| 74 | (0x2 /*0b10*/ << 6) | (point & 0x3f /*0b00111111*/); |
| 75 | continue; |
| 76 | } |
| 77 | } else { |
| 78 | resArr[resPos += 1] = 0xef /*0b11101111*/; |
| 79 | resArr[resPos += 1] = 0xbf /*0b10111111*/; |
| 80 | resArr[resPos += 1] = 0xbd /*0b10111101*/; |
| 81 | continue; |
| 82 | } |
| 83 | } |
| 84 | if (point <= 0x007f) { |
| 85 | resArr[resPos += 1] = (0x0 /*0b0*/ << 7) | point; |
| 86 | } else if (point <= 0x07ff) { |
| 87 | resArr[resPos += 1] = (0x6 /*0b110*/ << 5) | (point >>> 6); |
| 88 | resArr[resPos += 1] = |
| 89 | (0x2 /*0b10*/ << 6) | (point & 0x3f /*0b00111111*/); |
| 90 | } else { |
| 91 | resArr[resPos += 1] = (0xe /*0b1110*/ << 4) | (point >>> 12); |
| 92 | resArr[resPos += 1] = |
| 93 | (0x2 /*0b10*/ << 6) | ((point >>> 6) & 0x3f /*0b00111111*/); |
| 94 | resArr[resPos += 1] = |
| 95 | (0x2 /*0b10*/ << 6) | (point & 0x3f /*0b00111111*/); |
| 96 | } |
| 97 | } |
| 98 | if (typeof Uint8Array !== 'undefined') |
| 99 | return resArr.subarray(0, resPos + 1); |
| 100 | // else // IE 6-9 |
| 101 | resArr.length = resPos + 1; // trim off extra weight |
| 102 | return resArr; |
| 103 | }; |
| 104 | TextEncoder.prototype.toString = function() { |
| 105 | return '[object TextEncoder]' |
| 106 | }; |
| 107 | try { // Object.defineProperty only works on DOM prototypes in IE8 |
| 108 | Object.defineProperty(TextEncoder.prototype, 'encoding', { |
| 109 | get: function() { |
| 110 | if (TextEncoder.prototype.isPrototypeOf(this)) |
| 111 | return 'utf-8'; |
| 112 | else |
| 113 | throw TypeError('Illegal invocation'); |
| 114 | } |
| 115 | }); |
| 116 | } catch (e) { /*IE6-8 fallback*/ |
| 117 | TextEncoder.prototype.encoding = 'utf-8'; |
| 118 | } |
| 119 | if (typeof Symbol !== 'undefined') |
| 120 | TextEncoder.prototype[Symbol.toStringTag] = 'TextEncoder'; |
Kuiying Wang | f4a43cc | 2019-08-26 15:56:10 +0800 | [diff] [blame] | 121 | } |
| 122 | |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 123 | window.angular && (function(angular) { |
| 124 | 'use strict'; |
| 125 | |
| 126 | angular.module('app.common.directives').directive('serialConsole', [ |
| 127 | function() { |
| 128 | return { |
| 129 | 'restrict': 'E', |
| 130 | 'template': require('./serial-console.html'), |
beccabroek | a788cf0 | 2018-09-04 09:38:13 -0500 | [diff] [blame] | 131 | 'scope': {'path': '=', 'showTabBtn': '=?'}, |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 132 | 'controller': [ |
James Feist | 6a8d180 | 2020-04-08 14:04:19 -0700 | [diff] [blame] | 133 | '$scope', '$cookies', '$window', 'dataService', '$element', |
| 134 | function($scope, $cookies, $window, dataService, $element) { |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 135 | $scope.dataService = dataService; |
| 136 | |
beccabroek | 457ca04 | 2018-09-07 17:18:24 -0500 | [diff] [blame] | 137 | // See https://github.com/xtermjs/xterm.js/ for available xterm |
| 138 | // options |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 139 | |
beccabroek | 457ca04 | 2018-09-07 17:18:24 -0500 | [diff] [blame] | 140 | Terminal.applyAddon(attach); // Apply the `attach` addon |
| 141 | Terminal.applyAddon(fit); // Apply the `fit` addon |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 142 | |
Yang Cheng | 511a2bb | 2019-03-06 22:20:35 +0800 | [diff] [blame] | 143 | var border = 10; |
beccabroek | 457ca04 | 2018-09-07 17:18:24 -0500 | [diff] [blame] | 144 | var term = new Terminal(); |
Yoshie Muranaka | 198ce1f | 2019-09-20 10:33:04 -0700 | [diff] [blame] | 145 | // Should be a reference to <div id="terminal"></div> |
| 146 | var terminal = $element[0].firstElementChild.firstElementChild; |
Yang Cheng | 511a2bb | 2019-03-06 22:20:35 +0800 | [diff] [blame] | 147 | var customConsole; |
| 148 | var charSize; |
| 149 | var termContainer; |
| 150 | |
| 151 | term.open(terminal); |
| 152 | customConsole = configJSON.customConsoleDisplaySize; |
| 153 | |
| 154 | if (customConsole != null) { |
| 155 | charSize = measureChar(term); |
| 156 | termContainer = document.getElementById('term-container'); |
| 157 | if (termContainer != null) { |
| 158 | if (customConsole.width) { |
| 159 | termContainer.style.width = |
| 160 | (charSize.width * customConsole.width + border) + 'px'; |
| 161 | } |
| 162 | if (customConsole.height) { |
| 163 | terminal.style.height = |
| 164 | (charSize.height * customConsole.height + border) + 'px'; |
| 165 | } |
| 166 | } |
| 167 | } |
beccabroek | 457ca04 | 2018-09-07 17:18:24 -0500 | [diff] [blame] | 168 | term.fit(); |
Yang Cheng | 6d1d600 | 2019-02-27 11:16:24 +0800 | [diff] [blame] | 169 | if (configJSON.customKeyEnable == true) { |
| 170 | term.attachCustomKeyEventHandler(customKeyHandlers); |
| 171 | } |
beccabroek | 457ca04 | 2018-09-07 17:18:24 -0500 | [diff] [blame] | 172 | var SOL_THEME = { |
| 173 | background: '#19273c', |
| 174 | cursor: 'rgba(83, 146, 255, .5)', |
| 175 | scrollbar: 'rgba(83, 146, 255, .5)' |
| 176 | }; |
| 177 | term.setOption('theme', SOL_THEME); |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 178 | var hostname = dataService.getHost().replace('https://', ''); |
| 179 | var host = 'wss://' + hostname + '/console0'; |
James Feist | 6a8d180 | 2020-04-08 14:04:19 -0700 | [diff] [blame] | 180 | var token = $cookies.get('XSRF-TOKEN'); |
Yoshie Muranaka | fa8697b | 2019-08-26 11:09:35 -0700 | [diff] [blame] | 181 | try { |
James Feist | 6a8d180 | 2020-04-08 14:04:19 -0700 | [diff] [blame] | 182 | var ws = new WebSocket(host, [token]); |
Yoshie Muranaka | fa8697b | 2019-08-26 11:09:35 -0700 | [diff] [blame] | 183 | term.attach(ws); |
| 184 | ws.onopen = function() { |
| 185 | console.log('websocket opened'); |
| 186 | }; |
| 187 | ws.onclose = function(event) { |
| 188 | console.log( |
| 189 | 'websocket closed. code: ' + event.code + |
| 190 | ' reason: ' + event.reason); |
| 191 | }; |
| 192 | } catch (error) { |
| 193 | console.log(JSON.stringify(error)); |
| 194 | } |
beccabroek | a788cf0 | 2018-09-04 09:38:13 -0500 | [diff] [blame] | 195 | $scope.openTerminalWindow = function() { |
| 196 | $window.open( |
| 197 | '#/server-control/remote-console-window', |
| 198 | 'Remote Console Window', |
| 199 | 'directories=no,titlebar=no,toolbar=no,location=no,status=no,menubar=no,scrollbars=no,resizable=yes,width=600,height=550'); |
| 200 | }; |
beccabroek | 75697f9 | 2018-09-04 09:34:44 -0500 | [diff] [blame] | 201 | } |
| 202 | ] |
| 203 | }; |
| 204 | } |
| 205 | ]); |
| 206 | })(window.angular); |