Terminal.vue 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. <script setup lang="ts">
  2. import 'xterm/css/xterm.css'
  3. import { Terminal } from 'xterm'
  4. import { FitAddon } from '@xterm/addon-fit'
  5. import { onMounted, onUnmounted } from 'vue'
  6. import _ from 'lodash'
  7. import { useGettext } from 'vue3-gettext'
  8. import ws from '@/lib/websocket'
  9. const { $gettext } = useGettext()
  10. let term: Terminal | null
  11. let ping: number
  12. const websocket = ws('/api/pty')
  13. onMounted(() => {
  14. initTerm()
  15. websocket.onmessage = wsOnMessage
  16. websocket.onopen = wsOnOpen
  17. })
  18. interface Message {
  19. Type: number
  20. Data: string | null | { Cols: number; Rows: number }
  21. }
  22. const fitAddon = new FitAddon()
  23. const fit = _.throttle(() => {
  24. fitAddon.fit()
  25. }, 50)
  26. function initTerm() {
  27. term = new Terminal({
  28. convertEol: true,
  29. fontSize: 14,
  30. cursorStyle: 'block',
  31. scrollback: 1000,
  32. theme: {
  33. background: '#000',
  34. },
  35. })
  36. term.loadAddon(fitAddon)
  37. term.open(document.getElementById('terminal')!)
  38. setTimeout(() => {
  39. fitAddon.fit()
  40. }, 60)
  41. window.addEventListener('resize', fit)
  42. term.focus()
  43. term.onData(key => {
  44. const order: Message = {
  45. Data: key,
  46. Type: 1,
  47. }
  48. sendMessage(order)
  49. })
  50. term.onBinary(data => {
  51. sendMessage({ Type: 1, Data: data })
  52. })
  53. term.onResize(data => {
  54. sendMessage({ Type: 2, Data: { Cols: data.cols, Rows: data.rows } })
  55. })
  56. }
  57. function sendMessage(data: Message) {
  58. websocket.send(JSON.stringify(data))
  59. }
  60. function wsOnMessage(msg: { data: string | Uint8Array }) {
  61. term!.write(msg.data)
  62. }
  63. function wsOnOpen() {
  64. ping = setInterval(() => {
  65. sendMessage({ Type: 3, Data: null })
  66. }, 30000)
  67. }
  68. onUnmounted(() => {
  69. window.removeEventListener('resize', fit)
  70. clearInterval(ping)
  71. term?.dispose()
  72. ping = 0
  73. websocket.close()
  74. })
  75. </script>
  76. <template>
  77. <ACard :title="$gettext('Terminal')">
  78. <div
  79. id="terminal"
  80. class="console"
  81. />
  82. </ACard>
  83. </template>
  84. <style lang="less" scoped>
  85. .console {
  86. min-height: calc(100vh - 300px);
  87. :deep(.terminal) {
  88. padding: 10px;
  89. }
  90. :deep(.xterm-viewport) {
  91. border-radius: 5px;
  92. }
  93. }
  94. </style>