bfg-common 1.5.443 → 1.5.445

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (139) hide show
  1. package/CODE_STYLE.md +109 -109
  2. package/assets/img/icons/icons-sprite-dark-3.svg +227 -227
  3. package/assets/img/icons/icons-sprite-dark-5.svg +488 -488
  4. package/assets/img/icons/icons-sprite-light-3.svg +227 -227
  5. package/assets/img/icons/icons-sprite-light-5.svg +488 -488
  6. package/assets/localization/local_be.json +1 -0
  7. package/assets/localization/local_en.json +3 -2
  8. package/assets/localization/local_hy.json +1 -0
  9. package/assets/localization/local_kk.json +1 -0
  10. package/assets/localization/local_ru.json +1 -0
  11. package/assets/localization/local_zh.json +1 -0
  12. package/components/atoms/TheIcon3.vue +50 -50
  13. package/components/atoms/collapse/CollapseNav.vue +170 -170
  14. package/components/atoms/perPage/PerPage.vue +58 -58
  15. package/components/atoms/table/dataGrid/DataGrid.vue +1694 -1694
  16. package/components/atoms/table/dataGrid/DataGridPagination.vue +97 -97
  17. package/components/atoms/table/dataGrid/lib/config/settingsTable.ts +94 -94
  18. package/components/atoms/table/dataGrid/lib/utils/export.ts +16 -16
  19. package/components/common/browse/blocks/Container.vue +235 -235
  20. package/components/common/browse/blocks/contents/filesNew/Skeleton.vue +18 -18
  21. package/components/common/browse/blocks/lib/models/types.ts +1 -1
  22. package/components/common/browse/lib/models/interfaces.ts +5 -5
  23. package/components/common/context/lib/models/interfaces.ts +33 -33
  24. package/components/common/countdownTimer/CountdownTimer.vue +15 -26
  25. package/components/common/countdownTimer/CountdownTimerNew.vue +53 -0
  26. package/components/common/countdownTimer/CountdownTimerOld.vue +33 -0
  27. package/components/common/diagramMain/lib/config/initial.ts +50 -50
  28. package/components/common/diagramMain/lib/models/types.ts +21 -21
  29. package/components/common/diagramMain/lib/utils/utils.ts +331 -331
  30. package/components/common/diagramMain/modals/lib/config/networkModal.ts +398 -398
  31. package/components/common/diagramMain/modals/lib/config/portModal.ts +251 -251
  32. package/components/common/diagramMain/modals/lib/config/vCenterModal.ts +48 -48
  33. package/components/common/diagramMain/modals/lib/config/vmKernelAdapter.ts +90 -90
  34. package/components/common/diagramMain/modals/lib/mappers/mappers.ts +87 -87
  35. package/components/common/diagramMain/modals/lib/utils/index.ts +24 -24
  36. package/components/common/diagramMain/modals/migrateVmkernelAdapter/validations/common.ts +14 -14
  37. package/components/common/diagramMain/modals/migrateVmkernelAdapter/validations/validations.ts +19 -19
  38. package/components/common/diagramMain/port/Port.vue +580 -580
  39. package/components/common/diagramMain/port/Ports.vue +47 -47
  40. package/components/common/layout/theHeader/helpMenu/About.vue +82 -82
  41. package/components/common/layout/theHeader/modals/{Reconnect.vue → reconnect/Reconnect.vue} +43 -56
  42. package/components/common/layout/theHeader/modals/reconnect/ReconnectNew.vue +74 -0
  43. package/components/common/layout/theHeader/modals/reconnect/ReconnectOld.vue +67 -0
  44. package/components/common/pages/backups/Backups.vue +102 -102
  45. package/components/common/pages/backups/DetailView.vue +52 -52
  46. package/components/common/pages/backups/lib/models/interfaces.ts +36 -36
  47. package/components/common/pages/backups/modals/createBackup/configuration/Configuration.vue +29 -29
  48. package/components/common/pages/backups/modals/createBackup/configuration/backupWindow/BackupWindow.vue +26 -26
  49. package/components/common/pages/backups/modals/createBackup/configuration/maxBandwidth/MaxBandwidth.vue +66 -66
  50. package/components/common/pages/backups/modals/createBackup/configuration/maxBandwidth/lib/config/options.ts +6 -6
  51. package/components/common/pages/backups/modals/createBackup/configuration/strategy/Strategy.vue +35 -35
  52. package/components/common/pages/backups/modals/createBackup/datastores/Datastores.vue +59 -59
  53. package/components/common/pages/backups/modals/createBackup/datastores/tableView/TableView.vue +95 -95
  54. package/components/common/pages/backups/modals/createBackup/disks/tableView/lib/config/table.ts +117 -117
  55. package/components/common/pages/backups/modals/createBackup/general/General.vue +135 -135
  56. package/components/common/pages/backups/modals/createBackup/lib/config/strategyOptions.ts +12 -12
  57. package/components/common/pages/backups/modals/lib/config/restore.ts +115 -115
  58. package/components/common/pages/backups/modals/lib/models/interfaces.ts +186 -186
  59. package/components/common/pages/backups/modals/restore/disks/Disks.vue +27 -27
  60. package/components/common/pages/backups/modals/restore/disks/tableView/TableView.vue +102 -102
  61. package/components/common/pages/backups/modals/restore/disks/tableView/lib/config/table.ts +117 -117
  62. package/components/common/pages/backups/modals/restore/name/Name.vue +160 -160
  63. package/components/common/pages/backups/modals/restore/name/lib/models/interfaces.ts +6 -6
  64. package/components/common/pages/backups/modals/restore/networks/Networks.vue +67 -67
  65. package/components/common/pages/backups/modals/restore/networks/table/Table.vue +214 -214
  66. package/components/common/pages/backups/modals/restore/types/lib/config/typeOptions.ts +25 -25
  67. package/components/common/pages/backups/tools/Tools.vue +75 -75
  68. package/components/common/pages/backups/tools/lib/config/tabs.ts +36 -36
  69. package/components/common/pages/home/headline/HeadlineOld.vue +42 -42
  70. package/components/common/pages/home/lib/models/interfaces.ts +48 -48
  71. package/components/common/pages/home/widgets/hosts/Hosts.vue +27 -27
  72. package/components/common/pages/home/widgets/hosts/lib/config/items.ts +23 -23
  73. package/components/common/pages/home/widgets/vms/VmsOld.vue +35 -35
  74. package/components/common/pages/home/widgets/vms/lib/config/items.ts +19 -19
  75. package/components/common/pages/scheduledTasks/lib/utils/utils.ts +84 -84
  76. package/components/common/readyToComplete/ReadyToComplete.vue +17 -17
  77. package/components/common/select/radio/RadioGroup.vue +137 -137
  78. package/components/common/spiceConsole/Drawer.vue +381 -381
  79. package/components/common/spiceConsole/SpiceConsole.vue +127 -127
  80. package/components/common/spiceConsole/lib/models/interfaces.ts +5 -5
  81. package/components/common/tools/Actions.vue +207 -207
  82. package/components/common/treeView/TreeView.vue +52 -52
  83. package/components/common/vm/actions/add/folderTreeView/FolderTreeView.vue +72 -72
  84. package/components/common/vm/actions/add/folderTreeView/New.vue +40 -40
  85. package/components/common/vm/actions/common/customizeHardware/virtualHardware/browseView/BrowseView.vue +2 -1
  86. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/media/Media.vue +25 -25
  87. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/media/MediaNew.vue +78 -78
  88. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/media/MediaOld.vue +50 -50
  89. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cpu/Cpu.vue +2 -0
  90. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cpu/shares/Shares.vue +140 -140
  91. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cpu/shares/lib/config/options.ts +28 -28
  92. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newHardDisk/location/Location.vue +154 -154
  93. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newHardDisk/location/LocationOld.vue +85 -85
  94. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newNetwork/location/new/table/Table.vue +99 -99
  95. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newNetwork/macAddress/MacAddress.vue +119 -119
  96. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/directPathIo/DirectPathIoNew.vue +66 -66
  97. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/directPathIo/DirectPathIoOld.vue +62 -62
  98. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/dynamicDirectPathIo/DynamicDirectPathIo.vue +31 -31
  99. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/dynamicDirectPathIo/DynamicDirectPathIoOld.vue +76 -76
  100. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/note/Note.vue +15 -15
  101. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/note/NoteNew.vue +42 -42
  102. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/note/NoteOld.vue +30 -30
  103. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/nvidiaGrid/NvidiaGridOld.vue +84 -84
  104. package/components/common/vm/actions/common/customizeHardware/virtualHardware/other/Other.vue +16 -16
  105. package/components/common/vm/actions/common/customizeHardware/virtualHardware/videoCard/numberDisplays/NumberDisplays.vue +53 -53
  106. package/components/common/vm/actions/common/customizeHardware/vmoptions/bootOptions/delay/Delay.vue +32 -32
  107. package/components/common/vm/actions/common/customizeHardware/vmoptions/bootOptions/firmware/Firmware.vue +60 -60
  108. package/components/common/vm/actions/common/customizeHardware/vmoptions/bootOptions/order/Order.vue +174 -174
  109. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/keymap/Keymap.vue +32 -32
  110. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/keymap/KeymapOld.vue +44 -44
  111. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/password/Password.vue +103 -103
  112. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/usbRedirection/UsbRedirection.vue +28 -28
  113. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/usbRedirection/UsbRedirectionOld.vue +44 -44
  114. package/components/common/vm/actions/common/select/compatibility/Old.vue +107 -107
  115. package/components/common/vm/actions/common/select/computeResource/ComputeResource.vue +143 -143
  116. package/components/common/vm/actions/common/select/computeResource/New.vue +184 -184
  117. package/components/common/vm/actions/common/select/computeResource/treeView/TreeView.vue +131 -131
  118. package/components/common/vm/actions/common/select/createType/CreateType.vue +38 -38
  119. package/components/common/vm/actions/common/select/createType/lib/models/interfaces.ts +5 -5
  120. package/components/common/vm/actions/common/select/os/Old.vue +152 -152
  121. package/components/common/vm/actions/common/select/os/Os.vue +139 -139
  122. package/components/common/vm/actions/common/select/storage/new/New.vue +320 -320
  123. package/components/common/vm/actions/common/select/storage/new/lib/models/interfaces.ts +5 -5
  124. package/components/common/vm/actions/common/select/storage/new/lib/utils/utils.ts +21 -21
  125. package/components/common/vm/actions/common/select/template/old/Old.vue +50 -50
  126. package/components/common/vm/actions/editSettings/new/Skeleton.vue +88 -88
  127. package/components/common/wizards/datastore/add/steps/typeMode/lib/config/typeOptions.ts +43 -43
  128. package/components/common/wizards/vm/migrate/select/network/Network.vue +103 -103
  129. package/components/common/wizards/vm/migrate/select/type/lib/config/typeOptions.ts +45 -45
  130. package/composables/productNameLocal.ts +30 -30
  131. package/composables/useAppVersion.ts +21 -21
  132. package/package.json +1 -1
  133. package/plugins/date.ts +233 -233
  134. package/plugins/panelStates.ts +70 -70
  135. package/plugins/text.ts +59 -59
  136. package/public/spice-console/lib/images/bitmap.js +203 -203
  137. package/public/spice-console/network/spicechannel.js +390 -390
  138. package/store/main/mutations.ts +7 -7
  139. package/store/main/state.ts +7 -7
@@ -1,381 +1,381 @@
1
- <template>
2
- <div
3
- :class="['vmw-drawer animation', { show: isShow }]"
4
- data-id="spice-console-drawer"
5
- @click.capture="onInputManagerFocus"
6
- >
7
- <div
8
- v-if="!isShow"
9
- :style="`top: ${y}px;`"
10
- ref="grab"
11
- data-id="spice-console-drawer-toggle"
12
- class="vmw-drawer__open"
13
- @click="toggleDrawer"
14
- >
15
- <atoms-the-icon name="arrow" class="vmw-drawer__open-icon" />
16
- <atoms-the-icon2 name="drag" class="vmw-drawer__drag-icon" />
17
- </div>
18
-
19
- <div class="vmw-drawer-header">
20
- <h3>{{ localization.common.consolePanel }}</h3>
21
- <atoms-the-icon
22
- class="vmw-drawer-header__close"
23
- data-id="spice-console-drawer-toggle-icon"
24
- name="close"
25
- @click="toggleDrawer"
26
- />
27
- </div>
28
-
29
- <div class="vmw-drawer-body">
30
- <button
31
- class="vmw-drawer-body__btn animation toggle-fullscreen"
32
- data-id="spice-console-drawer-toggle-fullscreen"
33
- @click="emits('toggle-fullscreen')"
34
- >
35
- {{ localization.common.toggleFullscreenMode }}
36
- </button>
37
- <button
38
- class="vmw-drawer-body__btn animation toggle-fullscreen"
39
- @click="emits('send-alt-command')"
40
- >
41
- {{ localization.common.sendAltCommand }}
42
- </button>
43
- <label
44
- v-development="true"
45
- class="vmw-drawer-body__btn animation relative"
46
- >
47
- {{ localization.remoteConsole.uploadFolder }}
48
- <input type="file" webkitdirectory directory multiple />
49
- </label>
50
- <label
51
- v-development="true"
52
- class="vmw-drawer-body__btn animation relative"
53
- >
54
- {{ localization.remoteConsole.uploadFiles }}
55
- <input type="file" multiple />
56
- </label>
57
-
58
- <select
59
- v-model="usbDevice"
60
- v-development="true"
61
- @mouseenter="hover = true"
62
- @mouseleave="hover = false"
63
- @change="onChangeUsbDevice"
64
- >
65
- <option
66
- v-for="item in usbDevices"
67
- :key="item.value"
68
- :value="item.value"
69
- :disabled="item.disabled"
70
- >
71
- {{ item.label }}
72
- </option>
73
- </select>
74
-
75
- <select
76
- :value="codec"
77
- @mouseenter="hover = true"
78
- @mouseleave="hover = false"
79
- @change="onChangeCodec"
80
- >
81
- <option :value="1">MJPEG</option>
82
- <option :value="2">VP8</option>
83
- <option :value="3">H264</option>
84
- <option :value="4" disabled>VP9</option>
85
- <option :value="5">H265</option>
86
- </select>
87
-
88
- <hr />
89
- <button
90
- class="vmw-drawer-body__btn animation show-keyboard"
91
- @click="onToggleKeyboard"
92
- >
93
- {{ showOrHideKeyboard }}
94
- </button>
95
- </div>
96
-
97
- <div class="vmw-drawer-footer">
98
- <div class="size-info">
99
- <p class="size-info-text">View size: {{ viewSize }}</p>
100
- <p class="size-info-text">Canvas size: {{ canvasSize }}</p>
101
- </div>
102
- <div id="debug-stream"></div>
103
- </div>
104
- <div v-if="isKeyboardShown">
105
- <common-spice-console-keyboard />
106
- </div>
107
- </div>
108
- </template>
109
- <script setup lang="ts">
110
- import { useDraggable } from '@vueuse/core'
111
- import type { UI_I_DeviceOption } from '~/components/common/spiceConsole/lib/models/interfaces'
112
- import type { UI_I_Localization } from '~/lib/models/interfaces'
113
- import type { UI_T_CODEC } from '~/components/common/spiceConsole/lib/models/types'
114
- import {
115
- getUSBDeviceType,
116
- identifyHIDDevice,
117
- } from '~/components/common/spiceConsole/lib/utils/getDeviceType'
118
-
119
- const emits = defineEmits<{
120
- (event: 'toggle-fullscreen'): void
121
- (event: 'send-alt-command'): void
122
- }>()
123
-
124
- const { $console }: any = useNuxtApp()
125
-
126
- const codec = computed<UI_T_CODEC>(() => $console.codec.value)
127
- const onChangeCodec = (event: any): void => {
128
- const value = +event.target.value
129
- $console.changeCodec(value)
130
- // @ts-ignore
131
- if (!window.app) return
132
- // @ts-ignore
133
- const channels = window.app.spiceConnection.channels
134
- for (const i in channels) {
135
- const channel = channels[i]
136
- if (channel.channel === wdi.SpiceVars.SPICE_CHANNEL_DISPLAY) {
137
- channel.changeCodec(value)
138
- }
139
- }
140
- document.getElementById('inputmanager')?.focus()
141
- }
142
-
143
- const isKeyboardShown = ref<boolean>(false)
144
- const onToggleKeyboard = (): void => {
145
- isKeyboardShown.value = !isKeyboardShown.value
146
- }
147
-
148
- const isShow = ref<boolean>(false)
149
- const grab = ref<HTMLDivElement | null>(null)
150
- const localization = computed<UI_I_Localization>(() => useLocal())
151
- const initialTop = window.innerHeight / 2 - 15
152
- const { y } = useDraggable(grab, {
153
- initialValue: { x: -30, y: initialTop },
154
- })
155
-
156
- const showOrHideKeyboard = computed<string>(() =>
157
- isKeyboardShown.value
158
- ? localization.value.common.hideKeyboard
159
- : localization.value.common.showKeyboard
160
- )
161
-
162
- watch(y, () => {
163
- isDrag = true
164
- })
165
-
166
- let isDrag = false
167
- const toggleDrawer = (): void => {
168
- if (isDrag) {
169
- isDrag = false
170
- return
171
- }
172
- isShow.value = !isShow.value
173
- }
174
-
175
- const hover = ref<boolean>(false)
176
- const onInputManagerFocus = (): void => {
177
- if (hover.value) return
178
- document.getElementById('inputmanager')?.focus()
179
- }
180
-
181
- const viewSize = ref<string>('0 x 0')
182
- const canvasSize = ref<string>('0 x 0')
183
- const displaySizeInfo = (): void => {
184
- const layout = document.getElementById('eventLayer')
185
- viewSize.value = `${window.innerWidth}px x ${window.innerHeight}px`
186
- if (layout) {
187
- canvasSize.value = `${layout.width || 0}px x ${layout.height || 0}px`
188
- }
189
-
190
- requestAnimationFrame(displaySizeInfo)
191
- }
192
- displaySizeInfo()
193
-
194
- const usbDevice = ref<number | string>(-1)
195
- const usbDevices = ref<UI_I_DeviceOption[]>([
196
- { label: 'USB Device', value: -1, disabled: true },
197
- { label: 'Add USB device', value: -2 },
198
- ])
199
- const onChangeUsbDevice = (data: any): void => {
200
- if (data.target.value === '-2') {
201
- getUSBDeviceInformation()
202
- }
203
- }
204
-
205
- const getUSBDeviceInformation = async (): Promise<void> => {
206
- try {
207
- // Запросить устройство
208
- const device = await navigator.usb.requestDevice({ filters: [] })
209
- usbDevice.value = setDevice(device)
210
- } catch (error) {
211
- console.error('Error:', error)
212
- }
213
- }
214
-
215
- const setDevice = (device: any): string => {
216
- const interfaces = device.configuration.interfaces
217
-
218
- const types: any = new Set()
219
- interfaces.forEach((item: any) => {
220
- const classCode = item.alternate.interfaceClass
221
- const subclassCode = item.alternate.interfaceSubclass
222
- const protocolCode = item.alternate.interfaceProtocol
223
-
224
- if (classCode === 0x03) {
225
- // HID класс
226
- types.add(identifyHIDDevice(subclassCode, protocolCode))
227
- } else {
228
- types.add(getUSBDeviceType(classCode))
229
- }
230
- })
231
-
232
- const value = `${device.productId}_${device.vendorId}`
233
- if (usbDevices.value.every((device) => device.value !== value)) {
234
- usbDevices.value.push({
235
- value,
236
- label: Array.from(types).join(' / '),
237
- })
238
- }
239
-
240
- return value
241
- }
242
-
243
- const setDefaultDevices = async (): Promise<void> => {
244
- const devices = await navigator.usb.getDevices()
245
- devices.forEach((device: any) => {
246
- setDevice(device)
247
- })
248
- }
249
- setDefaultDevices()
250
- </script>
251
- <style lang="scss" scoped>
252
- .vmw-drawer {
253
- background-color: #314351;
254
- position: absolute;
255
- top: 0;
256
- right: -300px;
257
- width: 300px;
258
- height: 100vh;
259
- padding: 20px;
260
- z-index: var(--z-modal);
261
-
262
- &.show {
263
- right: 0;
264
-
265
- .vmw-drawer__open {
266
- z-index: var(--z-negative);
267
- }
268
- }
269
-
270
- &__open {
271
- width: 65px;
272
- height: 40px;
273
- background-color: #314351;
274
- position: absolute;
275
- left: -30px;
276
- cursor: pointer;
277
- border-top-left-radius: 20px;
278
- border-bottom-left-radius: 20px;
279
- box-shadow: 0 0 15px 5px #31435169;
280
- transition: left 0.5s;
281
- user-select: none;
282
-
283
- &.moving {
284
- cursor: grabbing;
285
- }
286
-
287
- &:hover {
288
- left: -60px;
289
- }
290
-
291
- & .vmw-drawer__open-icon {
292
- fill: #ffffff;
293
- transform: rotate(-90deg);
294
- width: 30px;
295
- height: 40px;
296
- z-index: var(--z-dropdown);
297
- }
298
-
299
- & .vmw-drawer__drag-icon {
300
- fill: #ffffff;
301
- width: 40px;
302
- position: absolute;
303
- left: 21px;
304
- z-index: calc(var(--z-default) + 1);
305
- cursor: grabbing;
306
- }
307
- }
308
-
309
- .vmw-drawer-header {
310
- & h3 {
311
- color: #fff;
312
- font-weight: bold;
313
- font-size: 20px;
314
- text-align: center;
315
- }
316
-
317
- &__close {
318
- width: 30px;
319
- fill: #fff;
320
- position: absolute;
321
- right: 10px;
322
- top: 10px;
323
- cursor: pointer;
324
- }
325
- }
326
-
327
- .vmw-drawer-body {
328
- margin-top: 50px;
329
-
330
- & select,
331
- &__btn {
332
- display: block;
333
- width: 100%;
334
- color: #fff;
335
- cursor: pointer;
336
- font-size: 15px;
337
- border: 1px solid #fff;
338
- padding: 5px;
339
- border-radius: 5px;
340
- text-align: center;
341
- margin-bottom: 20px;
342
- background-color: transparent;
343
-
344
- &:not(.disable):hover {
345
- background-color: #ffffff;
346
- color: #314351;
347
- }
348
-
349
- &.disable {
350
- opacity: 0.5;
351
- cursor: not-allowed;
352
- }
353
-
354
- input[type='file'] {
355
- position: absolute;
356
- top: 0;
357
- left: 0;
358
- right: 0;
359
- bottom: 0;
360
- opacity: 0;
361
- }
362
- }
363
-
364
- & > hr {
365
- margin-bottom: 20px;
366
- border-color: #ffffff40;
367
- box-shadow: 0 0 20px 0.5px #ffffff2e;
368
- }
369
- }
370
-
371
- .vmw-drawer-footer {
372
- .size-info {
373
- margin-bottom: 10px;
374
-
375
- .size-info-text {
376
- color: #ffffff;
377
- }
378
- }
379
- }
380
- }
381
- </style>
1
+ <template>
2
+ <div
3
+ :class="['vmw-drawer animation', { show: isShow }]"
4
+ data-id="spice-console-drawer"
5
+ @click.capture="onInputManagerFocus"
6
+ >
7
+ <div
8
+ v-if="!isShow"
9
+ :style="`top: ${y}px;`"
10
+ ref="grab"
11
+ data-id="spice-console-drawer-toggle"
12
+ class="vmw-drawer__open"
13
+ @click="toggleDrawer"
14
+ >
15
+ <atoms-the-icon name="arrow" class="vmw-drawer__open-icon" />
16
+ <atoms-the-icon2 name="drag" class="vmw-drawer__drag-icon" />
17
+ </div>
18
+
19
+ <div class="vmw-drawer-header">
20
+ <h3>{{ localization.common.consolePanel }}</h3>
21
+ <atoms-the-icon
22
+ class="vmw-drawer-header__close"
23
+ data-id="spice-console-drawer-toggle-icon"
24
+ name="close"
25
+ @click="toggleDrawer"
26
+ />
27
+ </div>
28
+
29
+ <div class="vmw-drawer-body">
30
+ <button
31
+ class="vmw-drawer-body__btn animation toggle-fullscreen"
32
+ data-id="spice-console-drawer-toggle-fullscreen"
33
+ @click="emits('toggle-fullscreen')"
34
+ >
35
+ {{ localization.common.toggleFullscreenMode }}
36
+ </button>
37
+ <button
38
+ class="vmw-drawer-body__btn animation toggle-fullscreen"
39
+ @click="emits('send-alt-command')"
40
+ >
41
+ {{ localization.common.sendAltCommand }}
42
+ </button>
43
+ <label
44
+ v-development="true"
45
+ class="vmw-drawer-body__btn animation relative"
46
+ >
47
+ {{ localization.remoteConsole.uploadFolder }}
48
+ <input type="file" webkitdirectory directory multiple />
49
+ </label>
50
+ <label
51
+ v-development="true"
52
+ class="vmw-drawer-body__btn animation relative"
53
+ >
54
+ {{ localization.remoteConsole.uploadFiles }}
55
+ <input type="file" multiple />
56
+ </label>
57
+
58
+ <select
59
+ v-model="usbDevice"
60
+ v-development="true"
61
+ @mouseenter="hover = true"
62
+ @mouseleave="hover = false"
63
+ @change="onChangeUsbDevice"
64
+ >
65
+ <option
66
+ v-for="item in usbDevices"
67
+ :key="item.value"
68
+ :value="item.value"
69
+ :disabled="item.disabled"
70
+ >
71
+ {{ item.label }}
72
+ </option>
73
+ </select>
74
+
75
+ <select
76
+ :value="codec"
77
+ @mouseenter="hover = true"
78
+ @mouseleave="hover = false"
79
+ @change="onChangeCodec"
80
+ >
81
+ <option :value="1">MJPEG</option>
82
+ <option :value="2">VP8</option>
83
+ <option :value="3">H264</option>
84
+ <option :value="4" disabled>VP9</option>
85
+ <option :value="5">H265</option>
86
+ </select>
87
+
88
+ <hr />
89
+ <button
90
+ class="vmw-drawer-body__btn animation show-keyboard"
91
+ @click="onToggleKeyboard"
92
+ >
93
+ {{ showOrHideKeyboard }}
94
+ </button>
95
+ </div>
96
+
97
+ <div class="vmw-drawer-footer">
98
+ <div class="size-info">
99
+ <p class="size-info-text">View size: {{ viewSize }}</p>
100
+ <p class="size-info-text">Canvas size: {{ canvasSize }}</p>
101
+ </div>
102
+ <div id="debug-stream"></div>
103
+ </div>
104
+ <div v-if="isKeyboardShown">
105
+ <common-spice-console-keyboard />
106
+ </div>
107
+ </div>
108
+ </template>
109
+ <script setup lang="ts">
110
+ import { useDraggable } from '@vueuse/core'
111
+ import type { UI_I_DeviceOption } from '~/components/common/spiceConsole/lib/models/interfaces'
112
+ import type { UI_I_Localization } from '~/lib/models/interfaces'
113
+ import type { UI_T_CODEC } from '~/components/common/spiceConsole/lib/models/types'
114
+ import {
115
+ getUSBDeviceType,
116
+ identifyHIDDevice,
117
+ } from '~/components/common/spiceConsole/lib/utils/getDeviceType'
118
+
119
+ const emits = defineEmits<{
120
+ (event: 'toggle-fullscreen'): void
121
+ (event: 'send-alt-command'): void
122
+ }>()
123
+
124
+ const { $console }: any = useNuxtApp()
125
+
126
+ const codec = computed<UI_T_CODEC>(() => $console.codec.value)
127
+ const onChangeCodec = (event: any): void => {
128
+ const value = +event.target.value
129
+ $console.changeCodec(value)
130
+ // @ts-ignore
131
+ if (!window.app) return
132
+ // @ts-ignore
133
+ const channels = window.app.spiceConnection.channels
134
+ for (const i in channels) {
135
+ const channel = channels[i]
136
+ if (channel.channel === wdi.SpiceVars.SPICE_CHANNEL_DISPLAY) {
137
+ channel.changeCodec(value)
138
+ }
139
+ }
140
+ document.getElementById('inputmanager')?.focus()
141
+ }
142
+
143
+ const isKeyboardShown = ref<boolean>(false)
144
+ const onToggleKeyboard = (): void => {
145
+ isKeyboardShown.value = !isKeyboardShown.value
146
+ }
147
+
148
+ const isShow = ref<boolean>(false)
149
+ const grab = ref<HTMLDivElement | null>(null)
150
+ const localization = computed<UI_I_Localization>(() => useLocal())
151
+ const initialTop = window.innerHeight / 2 - 15
152
+ const { y } = useDraggable(grab, {
153
+ initialValue: { x: -30, y: initialTop },
154
+ })
155
+
156
+ const showOrHideKeyboard = computed<string>(() =>
157
+ isKeyboardShown.value
158
+ ? localization.value.common.hideKeyboard
159
+ : localization.value.common.showKeyboard
160
+ )
161
+
162
+ watch(y, () => {
163
+ isDrag = true
164
+ })
165
+
166
+ let isDrag = false
167
+ const toggleDrawer = (): void => {
168
+ if (isDrag) {
169
+ isDrag = false
170
+ return
171
+ }
172
+ isShow.value = !isShow.value
173
+ }
174
+
175
+ const hover = ref<boolean>(false)
176
+ const onInputManagerFocus = (): void => {
177
+ if (hover.value) return
178
+ document.getElementById('inputmanager')?.focus()
179
+ }
180
+
181
+ const viewSize = ref<string>('0 x 0')
182
+ const canvasSize = ref<string>('0 x 0')
183
+ const displaySizeInfo = (): void => {
184
+ const layout = document.getElementById('eventLayer')
185
+ viewSize.value = `${window.innerWidth}px x ${window.innerHeight}px`
186
+ if (layout) {
187
+ canvasSize.value = `${layout.width || 0}px x ${layout.height || 0}px`
188
+ }
189
+
190
+ requestAnimationFrame(displaySizeInfo)
191
+ }
192
+ displaySizeInfo()
193
+
194
+ const usbDevice = ref<number | string>(-1)
195
+ const usbDevices = ref<UI_I_DeviceOption[]>([
196
+ { label: 'USB Device', value: -1, disabled: true },
197
+ { label: 'Add USB device', value: -2 },
198
+ ])
199
+ const onChangeUsbDevice = (data: any): void => {
200
+ if (data.target.value === '-2') {
201
+ getUSBDeviceInformation()
202
+ }
203
+ }
204
+
205
+ const getUSBDeviceInformation = async (): Promise<void> => {
206
+ try {
207
+ // Запросить устройство
208
+ const device = await navigator.usb.requestDevice({ filters: [] })
209
+ usbDevice.value = setDevice(device)
210
+ } catch (error) {
211
+ console.error('Error:', error)
212
+ }
213
+ }
214
+
215
+ const setDevice = (device: any): string => {
216
+ const interfaces = device.configuration.interfaces
217
+
218
+ const types: any = new Set()
219
+ interfaces.forEach((item: any) => {
220
+ const classCode = item.alternate.interfaceClass
221
+ const subclassCode = item.alternate.interfaceSubclass
222
+ const protocolCode = item.alternate.interfaceProtocol
223
+
224
+ if (classCode === 0x03) {
225
+ // HID класс
226
+ types.add(identifyHIDDevice(subclassCode, protocolCode))
227
+ } else {
228
+ types.add(getUSBDeviceType(classCode))
229
+ }
230
+ })
231
+
232
+ const value = `${device.productId}_${device.vendorId}`
233
+ if (usbDevices.value.every((device) => device.value !== value)) {
234
+ usbDevices.value.push({
235
+ value,
236
+ label: Array.from(types).join(' / '),
237
+ })
238
+ }
239
+
240
+ return value
241
+ }
242
+
243
+ const setDefaultDevices = async (): Promise<void> => {
244
+ const devices = await navigator.usb.getDevices()
245
+ devices.forEach((device: any) => {
246
+ setDevice(device)
247
+ })
248
+ }
249
+ setDefaultDevices()
250
+ </script>
251
+ <style lang="scss" scoped>
252
+ .vmw-drawer {
253
+ background-color: #314351;
254
+ position: absolute;
255
+ top: 0;
256
+ right: -300px;
257
+ width: 300px;
258
+ height: 100vh;
259
+ padding: 20px;
260
+ z-index: var(--z-modal);
261
+
262
+ &.show {
263
+ right: 0;
264
+
265
+ .vmw-drawer__open {
266
+ z-index: var(--z-negative);
267
+ }
268
+ }
269
+
270
+ &__open {
271
+ width: 65px;
272
+ height: 40px;
273
+ background-color: #314351;
274
+ position: absolute;
275
+ left: -30px;
276
+ cursor: pointer;
277
+ border-top-left-radius: 20px;
278
+ border-bottom-left-radius: 20px;
279
+ box-shadow: 0 0 15px 5px #31435169;
280
+ transition: left 0.5s;
281
+ user-select: none;
282
+
283
+ &.moving {
284
+ cursor: grabbing;
285
+ }
286
+
287
+ &:hover {
288
+ left: -60px;
289
+ }
290
+
291
+ & .vmw-drawer__open-icon {
292
+ fill: #ffffff;
293
+ transform: rotate(-90deg);
294
+ width: 30px;
295
+ height: 40px;
296
+ z-index: var(--z-dropdown);
297
+ }
298
+
299
+ & .vmw-drawer__drag-icon {
300
+ fill: #ffffff;
301
+ width: 40px;
302
+ position: absolute;
303
+ left: 21px;
304
+ z-index: calc(var(--z-default) + 1);
305
+ cursor: grabbing;
306
+ }
307
+ }
308
+
309
+ .vmw-drawer-header {
310
+ & h3 {
311
+ color: #fff;
312
+ font-weight: bold;
313
+ font-size: 20px;
314
+ text-align: center;
315
+ }
316
+
317
+ &__close {
318
+ width: 30px;
319
+ fill: #fff;
320
+ position: absolute;
321
+ right: 10px;
322
+ top: 10px;
323
+ cursor: pointer;
324
+ }
325
+ }
326
+
327
+ .vmw-drawer-body {
328
+ margin-top: 50px;
329
+
330
+ & select,
331
+ &__btn {
332
+ display: block;
333
+ width: 100%;
334
+ color: #fff;
335
+ cursor: pointer;
336
+ font-size: 15px;
337
+ border: 1px solid #fff;
338
+ padding: 5px;
339
+ border-radius: 5px;
340
+ text-align: center;
341
+ margin-bottom: 20px;
342
+ background-color: transparent;
343
+
344
+ &:not(.disable):hover {
345
+ background-color: #ffffff;
346
+ color: #314351;
347
+ }
348
+
349
+ &.disable {
350
+ opacity: 0.5;
351
+ cursor: not-allowed;
352
+ }
353
+
354
+ input[type='file'] {
355
+ position: absolute;
356
+ top: 0;
357
+ left: 0;
358
+ right: 0;
359
+ bottom: 0;
360
+ opacity: 0;
361
+ }
362
+ }
363
+
364
+ & > hr {
365
+ margin-bottom: 20px;
366
+ border-color: #ffffff40;
367
+ box-shadow: 0 0 20px 0.5px #ffffff2e;
368
+ }
369
+ }
370
+
371
+ .vmw-drawer-footer {
372
+ .size-info {
373
+ margin-bottom: 10px;
374
+
375
+ .size-info-text {
376
+ color: #ffffff;
377
+ }
378
+ }
379
+ }
380
+ }
381
+ </style>