bfg-common 1.5.310 → 1.5.311

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 (128) 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/components/atoms/TheIcon3.vue +50 -50
  7. package/components/atoms/collapse/CollapseNav.vue +170 -170
  8. package/components/atoms/collapse/CollapseNavItem.vue +226 -226
  9. package/components/atoms/nav/NavBar.vue +147 -147
  10. package/components/atoms/perPage/PerPage.vue +58 -58
  11. package/components/atoms/table/dataGrid/DataGrid.vue +1717 -1717
  12. package/components/atoms/table/dataGrid/lib/config/settingsTable.ts +94 -94
  13. package/components/atoms/table/dataGrid/lib/utils/export.ts +16 -16
  14. package/components/atoms/tabs/VerticalTabs.vue +105 -105
  15. package/components/common/accordion/Recursion.vue +225 -225
  16. package/components/common/backup/storage/actions/add/lib/config/steps.ts +168 -168
  17. package/components/common/backup/storage/actions/add/steps/hostAccessibility/HostAccessibility.vue +52 -52
  18. package/components/common/backup/storage/actions/add/steps/readyComplete/ReadyComplete.vue +45 -45
  19. package/components/common/backup/storage/actions/delete/Delete.vue +65 -65
  20. package/components/common/browse/blocks/contents/filesNew/Skeleton.vue +18 -18
  21. package/components/common/context/lib/models/interfaces.ts +33 -33
  22. package/components/common/diagramMain/DiagramMain.vue +897 -897
  23. package/components/common/diagramMain/modals/lib/config/networkModal.ts +398 -398
  24. package/components/common/diagramMain/network/Network.vue +141 -141
  25. package/components/common/layout/theHeader/helpMenu/About.vue +82 -82
  26. package/components/common/layout/theHeader/userMenu/modals/preferences/view/ViewOld.vue +112 -112
  27. package/components/common/monitor/overview/OverviewOld.vue +139 -139
  28. package/components/common/pages/home/headline/Headline.vue +45 -45
  29. package/components/common/pages/home/headline/HeadlineOld.vue +42 -42
  30. package/components/common/pages/home/lib/models/interfaces.ts +48 -48
  31. package/components/common/pages/home/widgets/Widgets.vue +49 -49
  32. package/components/common/pages/home/widgets/WidgetsNew.vue +86 -86
  33. package/components/common/pages/home/widgets/WidgetsOld.vue +34 -34
  34. package/components/common/pages/home/widgets/hosts/Hosts.vue +27 -27
  35. package/components/common/pages/home/widgets/hosts/lib/config/items.ts +23 -23
  36. package/components/common/pages/home/widgets/vms/Vms.vue +26 -26
  37. package/components/common/pages/home/widgets/vms/VmsOld.vue +35 -35
  38. package/components/common/pages/home/widgets/vms/lib/config/items.ts +19 -19
  39. package/components/common/pages/packages/Packages.vue +208 -208
  40. package/components/common/pages/shortcuts/block/BlockOld.vue +68 -68
  41. package/components/common/readyToComplete/ReadyToComplete.vue +17 -17
  42. package/components/common/recursionTree/RecursionTree.vue +223 -223
  43. package/components/common/select/button/ButtonDropdown.vue +112 -112
  44. package/components/common/spiceConsole/Drawer.vue +377 -377
  45. package/components/common/spiceConsole/SpiceConsole.vue +127 -127
  46. package/components/common/spiceConsole/lib/models/interfaces.ts +5 -5
  47. package/components/common/tools/Actions.vue +207 -207
  48. package/components/common/vm/actions/add/folderTreeView/FolderTreeView.vue +72 -72
  49. package/components/common/vm/actions/add/folderTreeView/New.vue +40 -40
  50. package/components/common/vm/actions/add/folderTreeView/Old.vue +50 -50
  51. package/components/common/vm/actions/clone/lib/config/steps.ts +291 -291
  52. package/components/common/vm/actions/clone/toTemplate/lib/config/steps.ts +116 -116
  53. package/components/common/vm/actions/common/customizeHardware/virtualHardware/VirtualHardwareOld.vue +321 -321
  54. package/components/common/vm/actions/common/customizeHardware/virtualHardware/browseView/BrowseView.vue +227 -227
  55. package/components/common/vm/actions/common/customizeHardware/virtualHardware/bus/Bus.vue +100 -100
  56. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/CdDvdDrive.vue +232 -232
  57. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/CdDvdDriveOld.vue +168 -168
  58. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/media/Media.vue +25 -25
  59. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/media/MediaNew.vue +78 -78
  60. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cdDvdDrive/media/MediaOld.vue +50 -50
  61. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cpu/shares/Shares.vue +140 -140
  62. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cpu/shares/lib/config/options.ts +28 -28
  63. package/components/common/vm/actions/common/customizeHardware/virtualHardware/cpu/tooltip/TooltipNew.vue +154 -154
  64. package/components/common/vm/actions/common/customizeHardware/virtualHardware/limit/Limit.vue +220 -220
  65. package/components/common/vm/actions/common/customizeHardware/virtualHardware/memory/Memory.vue +307 -307
  66. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newHardDisk/NewHardDisk.vue +385 -385
  67. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newHardDisk/location/Location.vue +154 -154
  68. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newHardDisk/location/LocationOld.vue +85 -85
  69. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newNetwork/NewNetwork.vue +288 -288
  70. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newNetwork/location/new/table/lib/config/config.ts +94 -94
  71. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newNetwork/macAddress/MacAddress.vue +119 -119
  72. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/NewPciDevice.vue +205 -205
  73. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/directPathIo/DirectPathIoNew.vue +66 -66
  74. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/directPathIo/DirectPathIoOld.vue +62 -62
  75. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/dynamicDirectPathIo/DynamicDirectPathIo.vue +31 -31
  76. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/dynamicDirectPathIo/DynamicDirectPathIoOld.vue +76 -76
  77. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/note/Note.vue +15 -15
  78. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/note/NoteNew.vue +42 -42
  79. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/note/NoteOld.vue +30 -30
  80. package/components/common/vm/actions/common/customizeHardware/virtualHardware/newPciDevice/nvidiaGrid/NvidiaGridOld.vue +84 -84
  81. package/components/common/vm/actions/common/customizeHardware/virtualHardware/other/Other.vue +16 -16
  82. package/components/common/vm/actions/common/customizeHardware/virtualHardware/videoCard/VideoCard.vue +154 -154
  83. package/components/common/vm/actions/common/customizeHardware/virtualHardware/videoCard/numberDisplays/NumberDisplays.vue +53 -53
  84. package/components/common/vm/actions/common/customizeHardware/vmoptions/Vmoptions.vue +152 -152
  85. package/components/common/vm/actions/common/customizeHardware/vmoptions/VmoptionsNew.vue +128 -128
  86. package/components/common/vm/actions/common/customizeHardware/vmoptions/VmoptionsOld.vue +111 -111
  87. package/components/common/vm/actions/common/customizeHardware/vmoptions/bootOptions/delay/Delay.vue +32 -32
  88. package/components/common/vm/actions/common/customizeHardware/vmoptions/bootOptions/firmware/Firmware.vue +60 -60
  89. package/components/common/vm/actions/common/customizeHardware/vmoptions/bootOptions/order/Order.vue +174 -174
  90. package/components/common/vm/actions/common/customizeHardware/vmoptions/generalOptions/GeneralOptions.vue +83 -83
  91. package/components/common/vm/actions/common/customizeHardware/vmoptions/generalOptions/GeneralOptionsOld.vue +154 -154
  92. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/keymap/Keymap.vue +32 -32
  93. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/keymap/KeymapOld.vue +44 -44
  94. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/password/Password.vue +103 -103
  95. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/usbRedirection/UsbRedirection.vue +28 -28
  96. package/components/common/vm/actions/common/customizeHardware/vmoptions/remoteConsoleOptions/usbRedirection/UsbRedirectionOld.vue +44 -44
  97. package/components/common/vm/actions/common/select/compatibility/Old.vue +107 -107
  98. package/components/common/vm/actions/common/select/computeResource/treeView/TreeView.vue +118 -118
  99. package/components/common/vm/actions/common/select/createType/CreateType.vue +38 -38
  100. package/components/common/vm/actions/common/select/createType/lib/models/interfaces.ts +5 -5
  101. package/components/common/vm/actions/common/select/name/Name.vue +174 -174
  102. package/components/common/vm/actions/common/select/os/Old.vue +152 -152
  103. package/components/common/vm/actions/common/select/os/Os.vue +139 -139
  104. package/components/common/vm/actions/common/select/storage/new/lib/models/interfaces.ts +5 -5
  105. package/components/common/vm/actions/common/select/storage/new/lib/utils/utils.ts +21 -21
  106. package/components/common/vm/actions/editSettings/new/Skeleton.vue +88 -88
  107. package/components/common/vmt/actions/add/Add.vue +641 -641
  108. package/components/common/vmt/actions/add/lib/config/steps.ts +107 -107
  109. package/components/common/vmt/actions/add/lib/models/interfaces.ts +23 -23
  110. package/components/common/wizards/network/add/steps/SelectConnectionType.vue +104 -104
  111. package/components/common/wizards/network/add/steps/SelectedTargetDevice.vue +467 -467
  112. package/components/common/wizards/vm/migrate/Migrate.vue +349 -349
  113. package/components/common/wizards/vm/migrate/lib/config/constructDataReady.ts +220 -220
  114. package/components/common/wizards/vm/migrate/lib/config/steps.ts +157 -157
  115. package/components/common/wizards/vm/migrate/lib/validations.ts +68 -68
  116. package/components/common/wizards/vm/migrate/select/network/Network.vue +103 -103
  117. package/components/common/wizards/vm/migrate/select/network/table/network/lib/config/advancedTable.ts +91 -91
  118. package/composables/productNameLocal.ts +30 -30
  119. package/composables/useAppVersion.ts +21 -21
  120. package/package.json +1 -1
  121. package/plugins/date.ts +233 -233
  122. package/plugins/panelStates.ts +70 -70
  123. package/plugins/text.ts +59 -59
  124. package/public/spice-console/lib/images/bitmap.js +203 -203
  125. package/public/spice-console/network/spicechannel.js +387 -387
  126. package/store/main/mutations.ts +7 -7
  127. package/store/main/state.ts +7 -7
  128. package/store/tasks/mappers/recentTasks.ts +64 -64
@@ -1,377 +1,377 @@
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
- v-model="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 codec = ref<UI_T_CODEC>(1)
125
- const onChangeCodec = (): void => {
126
- // @ts-ignore
127
- if (!window.app) return
128
- // @ts-ignore
129
- const channels = window.app.spiceConnection.channels
130
- for (const i in channels) {
131
- const channel = channels[i]
132
- if (channel.channel === wdi.SpiceVars.SPICE_CHANNEL_DISPLAY) {
133
- channel.changeCodec(codec.value)
134
- }
135
- }
136
- document.getElementById('inputmanager')?.focus()
137
- }
138
-
139
- const isKeyboardShown = ref<boolean>(false)
140
- const onToggleKeyboard = (): void => {
141
- isKeyboardShown.value = !isKeyboardShown.value
142
- }
143
-
144
- const isShow = ref<boolean>(false)
145
- const grab = ref<HTMLDivElement | null>(null)
146
- const localization = computed<UI_I_Localization>(() => useLocal())
147
- const initialTop = window.innerHeight / 2 - 15
148
- const { y } = useDraggable(grab, {
149
- initialValue: { x: -30, y: initialTop },
150
- })
151
-
152
- const showOrHideKeyboard = computed<string>(() =>
153
- isKeyboardShown.value
154
- ? localization.value.common.hideKeyboard
155
- : localization.value.common.showKeyboard
156
- )
157
-
158
- watch(y, () => {
159
- isDrag = true
160
- })
161
-
162
- let isDrag = false
163
- const toggleDrawer = (): void => {
164
- if (isDrag) {
165
- isDrag = false
166
- return
167
- }
168
- isShow.value = !isShow.value
169
- }
170
-
171
- const hover = ref<boolean>(false)
172
- const onInputManagerFocus = (): void => {
173
- if (hover.value) return
174
- document.getElementById('inputmanager')?.focus()
175
- }
176
-
177
- const viewSize = ref<string>('0 x 0')
178
- const canvasSize = ref<string>('0 x 0')
179
- const displaySizeInfo = (): void => {
180
- const layout = document.getElementById('eventLayer')
181
- viewSize.value = `${window.innerWidth}px x ${window.innerHeight}px`
182
- if (layout) {
183
- canvasSize.value = `${layout.width || 0}px x ${layout.height || 0}px`
184
- }
185
-
186
- requestAnimationFrame(displaySizeInfo)
187
- }
188
- displaySizeInfo()
189
-
190
- const usbDevice = ref<number | string>(-1)
191
- const usbDevices = ref<UI_I_DeviceOption[]>([
192
- { label: 'USB Device', value: -1, disabled: true },
193
- { label: 'Add USB device', value: -2 },
194
- ])
195
- const onChangeUsbDevice = (data: any): void => {
196
- if (data.target.value === '-2') {
197
- getUSBDeviceInformation()
198
- }
199
- }
200
-
201
- const getUSBDeviceInformation = async (): Promise<void> => {
202
- try {
203
- // Запросить устройство
204
- const device = await navigator.usb.requestDevice({ filters: [] })
205
- usbDevice.value = setDevice(device)
206
- } catch (error) {
207
- console.error('Error:', error)
208
- }
209
- }
210
-
211
- const setDevice = (device: any): string => {
212
- const interfaces = device.configuration.interfaces
213
-
214
- const types: any = new Set()
215
- interfaces.forEach((item: any) => {
216
- const classCode = item.alternate.interfaceClass
217
- const subclassCode = item.alternate.interfaceSubclass
218
- const protocolCode = item.alternate.interfaceProtocol
219
-
220
- if (classCode === 0x03) {
221
- // HID класс
222
- types.add(identifyHIDDevice(subclassCode, protocolCode))
223
- } else {
224
- types.add(getUSBDeviceType(classCode))
225
- }
226
- })
227
-
228
- const value = `${device.productId}_${device.vendorId}`
229
- if (usbDevices.value.every((device) => device.value !== value)) {
230
- usbDevices.value.push({
231
- value,
232
- label: Array.from(types).join(' / '),
233
- })
234
- }
235
-
236
- return value
237
- }
238
-
239
- const setDefaultDevices = async (): Promise<void> => {
240
- const devices = await navigator.usb.getDevices()
241
- devices.forEach((device: any) => {
242
- setDevice(device)
243
- })
244
- }
245
- setDefaultDevices()
246
- </script>
247
- <style lang="scss" scoped>
248
- .vmw-drawer {
249
- background-color: #314351;
250
- position: absolute;
251
- top: 0;
252
- right: -300px;
253
- width: 300px;
254
- height: 100vh;
255
- padding: 20px;
256
- z-index: var(--z-modal);
257
-
258
- &.show {
259
- right: 0;
260
-
261
- .vmw-drawer__open {
262
- z-index: var(--z-negative);
263
- }
264
- }
265
-
266
- &__open {
267
- width: 65px;
268
- height: 40px;
269
- background-color: #314351;
270
- position: absolute;
271
- left: -30px;
272
- cursor: pointer;
273
- border-top-left-radius: 20px;
274
- border-bottom-left-radius: 20px;
275
- box-shadow: 0 0 15px 5px #31435169;
276
- transition: left 0.5s;
277
- user-select: none;
278
-
279
- &.moving {
280
- cursor: grabbing;
281
- }
282
-
283
- &:hover {
284
- left: -60px;
285
- }
286
-
287
- & .vmw-drawer__open-icon {
288
- fill: #ffffff;
289
- transform: rotate(-90deg);
290
- width: 30px;
291
- height: 40px;
292
- z-index: var(--z-dropdown);
293
- }
294
-
295
- & .vmw-drawer__drag-icon {
296
- fill: #ffffff;
297
- width: 40px;
298
- position: absolute;
299
- left: 21px;
300
- z-index: calc(var(--z-default) + 1);
301
- cursor: grabbing;
302
- }
303
- }
304
-
305
- .vmw-drawer-header {
306
- & h3 {
307
- color: #fff;
308
- font-weight: bold;
309
- font-size: 20px;
310
- text-align: center;
311
- }
312
-
313
- &__close {
314
- width: 30px;
315
- fill: #fff;
316
- position: absolute;
317
- right: 10px;
318
- top: 10px;
319
- cursor: pointer;
320
- }
321
- }
322
-
323
- .vmw-drawer-body {
324
- margin-top: 50px;
325
-
326
- & select,
327
- &__btn {
328
- display: block;
329
- width: 100%;
330
- color: #fff;
331
- cursor: pointer;
332
- font-size: 15px;
333
- border: 1px solid #fff;
334
- padding: 5px;
335
- border-radius: 5px;
336
- text-align: center;
337
- margin-bottom: 20px;
338
- background-color: transparent;
339
-
340
- &:not(.disable):hover {
341
- background-color: #ffffff;
342
- color: #314351;
343
- }
344
-
345
- &.disable {
346
- opacity: 0.5;
347
- cursor: not-allowed;
348
- }
349
-
350
- input[type='file'] {
351
- position: absolute;
352
- top: 0;
353
- left: 0;
354
- right: 0;
355
- bottom: 0;
356
- opacity: 0;
357
- }
358
- }
359
-
360
- & > hr {
361
- margin-bottom: 20px;
362
- border-color: #ffffff40;
363
- box-shadow: 0 0 20px 0.5px #ffffff2e;
364
- }
365
- }
366
-
367
- .vmw-drawer-footer {
368
- .size-info {
369
- margin-bottom: 10px;
370
-
371
- .size-info-text {
372
- color: #ffffff;
373
- }
374
- }
375
- }
376
- }
377
- </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
+ v-model="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 codec = ref<UI_T_CODEC>(1)
125
+ const onChangeCodec = (): void => {
126
+ // @ts-ignore
127
+ if (!window.app) return
128
+ // @ts-ignore
129
+ const channels = window.app.spiceConnection.channels
130
+ for (const i in channels) {
131
+ const channel = channels[i]
132
+ if (channel.channel === wdi.SpiceVars.SPICE_CHANNEL_DISPLAY) {
133
+ channel.changeCodec(codec.value)
134
+ }
135
+ }
136
+ document.getElementById('inputmanager')?.focus()
137
+ }
138
+
139
+ const isKeyboardShown = ref<boolean>(false)
140
+ const onToggleKeyboard = (): void => {
141
+ isKeyboardShown.value = !isKeyboardShown.value
142
+ }
143
+
144
+ const isShow = ref<boolean>(false)
145
+ const grab = ref<HTMLDivElement | null>(null)
146
+ const localization = computed<UI_I_Localization>(() => useLocal())
147
+ const initialTop = window.innerHeight / 2 - 15
148
+ const { y } = useDraggable(grab, {
149
+ initialValue: { x: -30, y: initialTop },
150
+ })
151
+
152
+ const showOrHideKeyboard = computed<string>(() =>
153
+ isKeyboardShown.value
154
+ ? localization.value.common.hideKeyboard
155
+ : localization.value.common.showKeyboard
156
+ )
157
+
158
+ watch(y, () => {
159
+ isDrag = true
160
+ })
161
+
162
+ let isDrag = false
163
+ const toggleDrawer = (): void => {
164
+ if (isDrag) {
165
+ isDrag = false
166
+ return
167
+ }
168
+ isShow.value = !isShow.value
169
+ }
170
+
171
+ const hover = ref<boolean>(false)
172
+ const onInputManagerFocus = (): void => {
173
+ if (hover.value) return
174
+ document.getElementById('inputmanager')?.focus()
175
+ }
176
+
177
+ const viewSize = ref<string>('0 x 0')
178
+ const canvasSize = ref<string>('0 x 0')
179
+ const displaySizeInfo = (): void => {
180
+ const layout = document.getElementById('eventLayer')
181
+ viewSize.value = `${window.innerWidth}px x ${window.innerHeight}px`
182
+ if (layout) {
183
+ canvasSize.value = `${layout.width || 0}px x ${layout.height || 0}px`
184
+ }
185
+
186
+ requestAnimationFrame(displaySizeInfo)
187
+ }
188
+ displaySizeInfo()
189
+
190
+ const usbDevice = ref<number | string>(-1)
191
+ const usbDevices = ref<UI_I_DeviceOption[]>([
192
+ { label: 'USB Device', value: -1, disabled: true },
193
+ { label: 'Add USB device', value: -2 },
194
+ ])
195
+ const onChangeUsbDevice = (data: any): void => {
196
+ if (data.target.value === '-2') {
197
+ getUSBDeviceInformation()
198
+ }
199
+ }
200
+
201
+ const getUSBDeviceInformation = async (): Promise<void> => {
202
+ try {
203
+ // Запросить устройство
204
+ const device = await navigator.usb.requestDevice({ filters: [] })
205
+ usbDevice.value = setDevice(device)
206
+ } catch (error) {
207
+ console.error('Error:', error)
208
+ }
209
+ }
210
+
211
+ const setDevice = (device: any): string => {
212
+ const interfaces = device.configuration.interfaces
213
+
214
+ const types: any = new Set()
215
+ interfaces.forEach((item: any) => {
216
+ const classCode = item.alternate.interfaceClass
217
+ const subclassCode = item.alternate.interfaceSubclass
218
+ const protocolCode = item.alternate.interfaceProtocol
219
+
220
+ if (classCode === 0x03) {
221
+ // HID класс
222
+ types.add(identifyHIDDevice(subclassCode, protocolCode))
223
+ } else {
224
+ types.add(getUSBDeviceType(classCode))
225
+ }
226
+ })
227
+
228
+ const value = `${device.productId}_${device.vendorId}`
229
+ if (usbDevices.value.every((device) => device.value !== value)) {
230
+ usbDevices.value.push({
231
+ value,
232
+ label: Array.from(types).join(' / '),
233
+ })
234
+ }
235
+
236
+ return value
237
+ }
238
+
239
+ const setDefaultDevices = async (): Promise<void> => {
240
+ const devices = await navigator.usb.getDevices()
241
+ devices.forEach((device: any) => {
242
+ setDevice(device)
243
+ })
244
+ }
245
+ setDefaultDevices()
246
+ </script>
247
+ <style lang="scss" scoped>
248
+ .vmw-drawer {
249
+ background-color: #314351;
250
+ position: absolute;
251
+ top: 0;
252
+ right: -300px;
253
+ width: 300px;
254
+ height: 100vh;
255
+ padding: 20px;
256
+ z-index: var(--z-modal);
257
+
258
+ &.show {
259
+ right: 0;
260
+
261
+ .vmw-drawer__open {
262
+ z-index: var(--z-negative);
263
+ }
264
+ }
265
+
266
+ &__open {
267
+ width: 65px;
268
+ height: 40px;
269
+ background-color: #314351;
270
+ position: absolute;
271
+ left: -30px;
272
+ cursor: pointer;
273
+ border-top-left-radius: 20px;
274
+ border-bottom-left-radius: 20px;
275
+ box-shadow: 0 0 15px 5px #31435169;
276
+ transition: left 0.5s;
277
+ user-select: none;
278
+
279
+ &.moving {
280
+ cursor: grabbing;
281
+ }
282
+
283
+ &:hover {
284
+ left: -60px;
285
+ }
286
+
287
+ & .vmw-drawer__open-icon {
288
+ fill: #ffffff;
289
+ transform: rotate(-90deg);
290
+ width: 30px;
291
+ height: 40px;
292
+ z-index: var(--z-dropdown);
293
+ }
294
+
295
+ & .vmw-drawer__drag-icon {
296
+ fill: #ffffff;
297
+ width: 40px;
298
+ position: absolute;
299
+ left: 21px;
300
+ z-index: calc(var(--z-default) + 1);
301
+ cursor: grabbing;
302
+ }
303
+ }
304
+
305
+ .vmw-drawer-header {
306
+ & h3 {
307
+ color: #fff;
308
+ font-weight: bold;
309
+ font-size: 20px;
310
+ text-align: center;
311
+ }
312
+
313
+ &__close {
314
+ width: 30px;
315
+ fill: #fff;
316
+ position: absolute;
317
+ right: 10px;
318
+ top: 10px;
319
+ cursor: pointer;
320
+ }
321
+ }
322
+
323
+ .vmw-drawer-body {
324
+ margin-top: 50px;
325
+
326
+ & select,
327
+ &__btn {
328
+ display: block;
329
+ width: 100%;
330
+ color: #fff;
331
+ cursor: pointer;
332
+ font-size: 15px;
333
+ border: 1px solid #fff;
334
+ padding: 5px;
335
+ border-radius: 5px;
336
+ text-align: center;
337
+ margin-bottom: 20px;
338
+ background-color: transparent;
339
+
340
+ &:not(.disable):hover {
341
+ background-color: #ffffff;
342
+ color: #314351;
343
+ }
344
+
345
+ &.disable {
346
+ opacity: 0.5;
347
+ cursor: not-allowed;
348
+ }
349
+
350
+ input[type='file'] {
351
+ position: absolute;
352
+ top: 0;
353
+ left: 0;
354
+ right: 0;
355
+ bottom: 0;
356
+ opacity: 0;
357
+ }
358
+ }
359
+
360
+ & > hr {
361
+ margin-bottom: 20px;
362
+ border-color: #ffffff40;
363
+ box-shadow: 0 0 20px 0.5px #ffffff2e;
364
+ }
365
+ }
366
+
367
+ .vmw-drawer-footer {
368
+ .size-info {
369
+ margin-bottom: 10px;
370
+
371
+ .size-info-text {
372
+ color: #ffffff;
373
+ }
374
+ }
375
+ }
376
+ }
377
+ </style>