Terminal.vue 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  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 ws from '@/lib/websocket'
  8. let term: Terminal | null
  9. let ping: null | NodeJS.Timer
  10. const websocket = ws('/api/pty')
  11. onMounted(() => {
  12. initTerm()
  13. websocket.onmessage = wsOnMessage
  14. websocket.onopen = wsOnOpen
  15. })
  16. interface Message {
  17. Type: Number,
  18. Data: any | null
  19. }
  20. const fitAddon = new FitAddon()
  21. const fit = _.throttle(function () {
  22. fitAddon.fit()
  23. }, 50)
  24. function initTerm() {
  25. term = new Terminal({
  26. rendererType: 'canvas',
  27. convertEol: true,
  28. fontSize: 14,
  29. cursorStyle: 'block',
  30. scrollback: 1000,
  31. theme: {
  32. background: 'rgba(3,14,32,0.7)'
  33. },
  34. })
  35. term.loadAddon(fitAddon)
  36. // this.fitAddon = 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(function (key) {
  44. let 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: any }) {
  61. term!.write(msg.data)
  62. }
  63. function wsOnOpen() {
  64. ping = setInterval(function () {
  65. sendMessage({Type: 3, Data: null})
  66. }, 30000)
  67. }
  68. onUnmounted(() => {
  69. window.removeEventListener('resize', fit)
  70. clearInterval(ping!)
  71. ping = null
  72. websocket.close()
  73. })
  74. </script>
  75. <template>
  76. <a-card :title="$gettext('Terminal')">
  77. <div class="console" id="terminal"></div>
  78. </a-card>
  79. </template>
  80. <style lang="less" scoped>
  81. .console {
  82. min-height: calc(100vh - 300px);
  83. }
  84. </style>