Terminal.vue 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  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. import {useGettext} from 'vue3-gettext'
  9. const {$gettext} = useGettext()
  10. let term: Terminal | null
  11. let ping: null | NodeJS.Timer
  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: any | null
  21. }
  22. const fitAddon = new FitAddon()
  23. const fit = _.throttle(function () {
  24. fitAddon.fit()
  25. }, 50)
  26. function initTerm() {
  27. term = new Terminal({
  28. rendererType: 'canvas',
  29. convertEol: true,
  30. fontSize: 14,
  31. cursorStyle: 'block',
  32. scrollback: 1000,
  33. theme: {
  34. background: '#000',
  35. },
  36. })
  37. term.loadAddon(fitAddon)
  38. term.open(document.getElementById('terminal')!)
  39. setTimeout(() => {
  40. fitAddon.fit()
  41. }, 60)
  42. window.addEventListener('resize', fit)
  43. term.focus()
  44. term.onData(function (key) {
  45. let order: Message = {
  46. Data: key,
  47. Type: 1
  48. }
  49. sendMessage(order)
  50. })
  51. term.onBinary(data => {
  52. sendMessage({Type: 1, Data: data})
  53. })
  54. term.onResize(data => {
  55. sendMessage({Type: 2, Data: {Cols: data.cols, Rows: data.rows}})
  56. })
  57. }
  58. function sendMessage(data: Message) {
  59. websocket.send(JSON.stringify(data))
  60. }
  61. function wsOnMessage(msg: { data: any }) {
  62. term!.write(msg.data)
  63. }
  64. function wsOnOpen() {
  65. ping = setInterval(function () {
  66. sendMessage({Type: 3, Data: null})
  67. }, 30000)
  68. }
  69. onUnmounted(() => {
  70. window.removeEventListener('resize', fit)
  71. clearInterval(ping!)
  72. term?.dispose()
  73. ping = null
  74. websocket.close()
  75. })
  76. </script>
  77. <template>
  78. <a-card :title="$gettext('Terminal')">
  79. <div class="console" id="terminal"></div>
  80. </a-card>
  81. </template>
  82. <style lang="less" scoped>
  83. .console {
  84. min-height: calc(100vh - 300px);
  85. }
  86. </style>