@vc-shell/framework 1.2.2 → 1.2.3-beta.1

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 (239) hide show
  1. package/core/composables/index.ts +2 -0
  2. package/core/composables/useAssets/index.ts +72 -28
  3. package/core/composables/useAsync/index.ts +4 -1
  4. package/core/composables/useBladeRegistry/index.ts +6 -5
  5. package/core/composables/useBreadcrumbs/index.ts +4 -1
  6. package/core/composables/useErrorHandler/index.ts +4 -1
  7. package/core/composables/useFunctions/debounce.ts +0 -1
  8. package/core/composables/useFunctions/delay.ts +0 -1
  9. package/core/composables/useFunctions/index.ts +0 -1
  10. package/core/composables/useFunctions/once.ts +0 -1
  11. package/core/composables/useFunctions/sleep.ts +0 -1
  12. package/core/composables/useFunctions/throttle.ts +0 -1
  13. package/core/composables/useGlobalSearch/index.ts +3 -3
  14. package/core/composables/useMenuService/index.ts +5 -2
  15. package/core/composables/useNotifications/index.ts +5 -2
  16. package/core/composables/useTheme/index.ts +4 -1
  17. package/core/composables/useUser/index.ts +189 -20
  18. package/core/composables/useWidgets/index.ts +5 -2
  19. package/core/constants/defaults.ts +76 -0
  20. package/core/constants/index.ts +2 -0
  21. package/core/constants/ui.ts +68 -0
  22. package/core/interceptors/index.ts +5 -2
  23. package/core/plugins/ai-agent/README.md +336 -0
  24. package/core/plugins/ai-agent/components/VcAiAgentPanel.vue +125 -0
  25. package/core/plugins/ai-agent/components/_internal/VcAiAgentHeader.vue +102 -0
  26. package/core/plugins/ai-agent/components/_internal/VcAiAgentIframe.vue +107 -0
  27. package/core/plugins/ai-agent/components/_internal/VcAiAgentLoader.vue +72 -0
  28. package/core/plugins/ai-agent/components/_internal/VcVirtoOzLogo.vue +136 -0
  29. package/core/plugins/ai-agent/components/index.ts +1 -0
  30. package/core/plugins/ai-agent/composables/index.ts +4 -0
  31. package/core/plugins/ai-agent/composables/useAiAgent.ts +231 -0
  32. package/core/plugins/ai-agent/composables/useAiAgentContext.ts +285 -0
  33. package/core/plugins/ai-agent/constants.ts +89 -0
  34. package/core/plugins/ai-agent/index.ts +91 -0
  35. package/core/plugins/ai-agent/services/ai-agent-service.ts +621 -0
  36. package/core/plugins/ai-agent/types.ts +310 -0
  37. package/core/plugins/modularity/index.ts +8 -6
  38. package/core/plugins/modularity/loader.ts +36 -33
  39. package/core/plugins/signalR/index.ts +6 -3
  40. package/core/services/app-bar-menu-service.ts +4 -1
  41. package/core/services/dashboard-service.ts +4 -1
  42. package/core/services/index.ts +2 -0
  43. package/core/services/menu-service.ts +4 -1
  44. package/core/services/settings-menu-service.ts +4 -1
  45. package/core/services/toolbar-service.ts +18 -3
  46. package/core/services/widget-service.ts +7 -4
  47. package/core/types/index.ts +3 -0
  48. package/core/types/services.ts +194 -0
  49. package/core/utilities/errorTypes.ts +126 -0
  50. package/core/utilities/index.ts +2 -0
  51. package/core/utilities/logger.ts +120 -0
  52. package/dist/core/composables/useAssets/index.d.ts.map +1 -1
  53. package/dist/core/composables/useAsync/index.d.ts.map +1 -1
  54. package/dist/core/composables/useBladeRegistry/index.d.ts.map +1 -1
  55. package/dist/core/composables/useBreadcrumbs/index.d.ts.map +1 -1
  56. package/dist/core/composables/useErrorHandler/index.d.ts.map +1 -1
  57. package/dist/core/composables/useFunctions/debounce.d.ts.map +1 -1
  58. package/dist/core/composables/useFunctions/delay.d.ts.map +1 -1
  59. package/dist/core/composables/useFunctions/index.d.ts.map +1 -1
  60. package/dist/core/composables/useFunctions/once.d.ts.map +1 -1
  61. package/dist/core/composables/useFunctions/sleep.d.ts.map +1 -1
  62. package/dist/core/composables/useFunctions/throttle.d.ts.map +1 -1
  63. package/dist/core/composables/useGlobalSearch/index.d.ts.map +1 -1
  64. package/dist/core/composables/useMenuService/index.d.ts.map +1 -1
  65. package/dist/core/composables/useNotifications/index.d.ts.map +1 -1
  66. package/dist/core/composables/useTheme/index.d.ts.map +1 -1
  67. package/dist/core/composables/useUser/index.d.ts +8 -0
  68. package/dist/core/composables/useUser/index.d.ts.map +1 -1
  69. package/dist/core/composables/useWidgets/index.d.ts.map +1 -1
  70. package/dist/core/constants/defaults.d.ts +63 -0
  71. package/dist/core/constants/defaults.d.ts.map +1 -0
  72. package/dist/core/constants/index.d.ts +2 -0
  73. package/dist/core/constants/index.d.ts.map +1 -1
  74. package/dist/core/constants/ui.d.ts +50 -0
  75. package/dist/core/constants/ui.d.ts.map +1 -0
  76. package/dist/core/interceptors/index.d.ts.map +1 -1
  77. package/dist/core/plugins/ai-agent/components/VcAiAgentPanel.vue.d.ts +3 -0
  78. package/dist/core/plugins/ai-agent/components/VcAiAgentPanel.vue.d.ts.map +1 -0
  79. package/dist/core/plugins/ai-agent/components/_internal/VcAiAgentHeader.vue.d.ts +15 -0
  80. package/dist/core/plugins/ai-agent/components/_internal/VcAiAgentHeader.vue.d.ts.map +1 -0
  81. package/dist/core/plugins/ai-agent/components/_internal/VcAiAgentIframe.vue.d.ts +10 -0
  82. package/dist/core/plugins/ai-agent/components/_internal/VcAiAgentIframe.vue.d.ts.map +1 -0
  83. package/dist/core/plugins/ai-agent/components/_internal/VcAiAgentLoader.vue.d.ts +7 -0
  84. package/dist/core/plugins/ai-agent/components/_internal/VcAiAgentLoader.vue.d.ts.map +1 -0
  85. package/dist/core/plugins/ai-agent/components/_internal/VcVirtoOzLogo.vue.d.ts +15 -0
  86. package/dist/core/plugins/ai-agent/components/_internal/VcVirtoOzLogo.vue.d.ts.map +1 -0
  87. package/dist/core/plugins/ai-agent/components/index.d.ts +2 -0
  88. package/dist/core/plugins/ai-agent/components/index.d.ts.map +1 -0
  89. package/dist/core/plugins/ai-agent/composables/index.d.ts +4 -0
  90. package/dist/core/plugins/ai-agent/composables/index.d.ts.map +1 -0
  91. package/dist/core/plugins/ai-agent/composables/useAiAgent.d.ts +95 -0
  92. package/dist/core/plugins/ai-agent/composables/useAiAgent.d.ts.map +1 -0
  93. package/dist/core/plugins/ai-agent/composables/useAiAgentContext.d.ts +55 -0
  94. package/dist/core/plugins/ai-agent/composables/useAiAgentContext.d.ts.map +1 -0
  95. package/dist/core/plugins/ai-agent/constants.d.ts +47 -0
  96. package/dist/core/plugins/ai-agent/constants.d.ts.map +1 -0
  97. package/dist/core/plugins/ai-agent/index.d.ts +48 -0
  98. package/dist/core/plugins/ai-agent/index.d.ts.map +1 -0
  99. package/dist/core/plugins/ai-agent/services/ai-agent-service.d.ts +45 -0
  100. package/dist/core/plugins/ai-agent/services/ai-agent-service.d.ts.map +1 -0
  101. package/dist/core/plugins/ai-agent/types.d.ts +258 -0
  102. package/dist/core/plugins/ai-agent/types.d.ts.map +1 -0
  103. package/dist/core/plugins/modularity/index.d.ts.map +1 -1
  104. package/dist/core/plugins/modularity/loader.d.ts.map +1 -1
  105. package/dist/core/plugins/signalR/index.d.ts.map +1 -1
  106. package/dist/core/services/app-bar-menu-service.d.ts.map +1 -1
  107. package/dist/core/services/dashboard-service.d.ts.map +1 -1
  108. package/dist/core/services/menu-service.d.ts.map +1 -1
  109. package/dist/core/services/settings-menu-service.d.ts.map +1 -1
  110. package/dist/core/services/toolbar-service.d.ts.map +1 -1
  111. package/dist/core/services/widget-service.d.ts.map +1 -1
  112. package/dist/core/types/index.d.ts.map +1 -1
  113. package/dist/core/types/services.d.ts +169 -0
  114. package/dist/core/types/services.d.ts.map +1 -0
  115. package/dist/core/utilities/errorTypes.d.ts +61 -0
  116. package/dist/core/utilities/errorTypes.d.ts.map +1 -0
  117. package/dist/core/utilities/index.d.ts +2 -0
  118. package/dist/core/utilities/index.d.ts.map +1 -1
  119. package/dist/core/utilities/logger.d.ts +259 -0
  120. package/dist/core/utilities/logger.d.ts.map +1 -0
  121. package/dist/framework.js +9748 -8483
  122. package/dist/index.css +1 -1
  123. package/dist/index.d.ts +19 -0
  124. package/dist/index.d.ts.map +1 -1
  125. package/dist/injection-keys.d.ts +21 -6
  126. package/dist/injection-keys.d.ts.map +1 -1
  127. package/dist/locales/de.json +3 -0
  128. package/dist/locales/en.json +3 -0
  129. package/dist/shared/components/app-switcher/composables/useAppSwitcher/index.d.ts.map +1 -1
  130. package/dist/shared/components/blade-navigation/components/vc-blade-navigation/vc-blade-navigation.vue.d.ts.map +1 -1
  131. package/dist/shared/components/blade-navigation/composables/useBladeNavigation/index.d.ts.map +1 -1
  132. package/dist/shared/components/blade-navigation/composables/useBladeNavigation/internal/bladeActions.d.ts.map +1 -1
  133. package/dist/shared/components/blade-navigation/composables/useBladeNavigation/internal/bladeRouteResolver.d.ts.map +1 -1
  134. package/dist/shared/components/blade-navigation/composables/useBladeNavigation/internal/routerUtils.d.ts.map +1 -1
  135. package/dist/shared/components/draggable-dashboard/composables/useDashboardDragAndDrop.d.ts.map +1 -1
  136. package/dist/shared/components/draggable-dashboard/composables/useLayoutPersistence.d.ts.map +1 -1
  137. package/dist/shared/components/notifications/composables/useContainer/index.d.ts.map +1 -1
  138. package/dist/shared/components/notifications/composables/useInstance/index.d.ts.map +1 -1
  139. package/dist/shared/components/notifications/core/notification.d.ts.map +1 -1
  140. package/dist/shared/components/popup-handler/components/vc-popup-container/vc-popup-container.vue.d.ts.map +1 -1
  141. package/dist/shared/components/sign-in/useExternalProvider.d.ts.map +1 -1
  142. package/dist/shared/composables/index.d.ts +1 -0
  143. package/dist/shared/composables/index.d.ts.map +1 -1
  144. package/dist/shared/composables/useExternalWidgets.d.ts.map +1 -1
  145. package/dist/shared/composables/useMenuExpanded.d.ts.map +1 -1
  146. package/dist/shared/composables/useTableSelection.d.ts +57 -0
  147. package/dist/shared/composables/useTableSelection.d.ts.map +1 -0
  148. package/dist/shared/composables/useTableSort.d.ts.map +1 -1
  149. package/dist/shared/modules/assets-manager/components/assets-manager/assets-manager.vue.d.ts.map +1 -1
  150. package/dist/shared/pages/LoginPage/components/login/Login.vue.d.ts.map +1 -1
  151. package/dist/shared/utilities/colorUtils.d.ts +0 -6
  152. package/dist/shared/utilities/colorUtils.d.ts.map +1 -1
  153. package/dist/tsconfig.tsbuildinfo +1 -1
  154. package/dist/ui/components/atoms/vc-banner/vc-banner.vue.d.ts.map +1 -1
  155. package/dist/ui/components/atoms/vc-button/vc-button.vue.d.ts +0 -15
  156. package/dist/ui/components/atoms/vc-button/vc-button.vue.d.ts.map +1 -1
  157. package/dist/ui/components/atoms/vc-container/vc-container.vue.d.ts.map +1 -1
  158. package/dist/ui/components/atoms/vc-icon/vc-icon.vue.d.ts.map +1 -1
  159. package/dist/ui/components/atoms/vc-icon/vc-lucide-icon.vue.d.ts.map +1 -1
  160. package/dist/ui/components/atoms/vc-image/vc-image.vue.d.ts.map +1 -1
  161. package/dist/ui/components/atoms/vc-link/vc-link.vue.d.ts.map +1 -1
  162. package/dist/ui/components/atoms/vc-loading/vc-loading.vue.d.ts.map +1 -1
  163. package/dist/ui/components/atoms/vc-status/vc-status.vue.d.ts +0 -5
  164. package/dist/ui/components/atoms/vc-status/vc-status.vue.d.ts.map +1 -1
  165. package/dist/ui/components/atoms/vc-tooltip/vc-tooltip.vue.d.ts.map +1 -1
  166. package/dist/ui/components/atoms/vc-video/vc-video.vue.d.ts.map +1 -1
  167. package/dist/ui/components/atoms/vc-widget/vc-widget.vue.d.ts.map +1 -1
  168. package/dist/ui/components/molecules/vc-breadcrumbs/vc-breadcrumbs.vue.d.ts.map +1 -1
  169. package/dist/ui/components/molecules/vc-input/vc-input.vue.d.ts.map +1 -1
  170. package/dist/ui/components/molecules/vc-pagination/vc-pagination.vue.d.ts.map +1 -1
  171. package/dist/ui/components/molecules/vc-toast/vc-toast.vue.d.ts.map +1 -1
  172. package/dist/ui/components/organisms/vc-app/vc-app.vue.d.ts.map +1 -1
  173. package/dist/ui/components/organisms/vc-blade/_internal/vc-blade-toolbar/_internal/vc-blade-toolbar-buttons/_internal/vc-blade-toolbar-button/vc-blade-toolbar-button.vue.d.ts.map +1 -1
  174. package/dist/ui/components/organisms/vc-blade/vc-blade.vue.d.ts.map +1 -1
  175. package/dist/ui/components/organisms/vc-login-form/vc-login-form.vue.d.ts.map +1 -1
  176. package/dist/ui/components/organisms/vc-table/_internal/vc-table-cell/vc-table-cell.vue.d.ts.map +1 -1
  177. package/dist/ui/components/organisms/vc-table/composables/useTableActions.d.ts.map +1 -1
  178. package/dist/ui/components/organisms/vc-table/composables/useTableColumnResize.d.ts.map +1 -1
  179. package/dist/ui/components/organisms/vc-table/composables/useTableRowReorder.d.ts.map +1 -1
  180. package/dist/ui/components/organisms/vc-table/composables/useTableSelection.d.ts.map +1 -1
  181. package/dist/ui/components/organisms/vc-table/composables/useTableState.d.ts.map +1 -1
  182. package/dist/{vendor-lodash-es-BqkGj3Jl.js → vendor-lodash-es-SgOIjJF8.js} +2 -0
  183. package/package.json +5 -5
  184. package/shared/components/app-switcher/composables/useAppSwitcher/index.ts +4 -1
  185. package/shared/components/blade-navigation/components/vc-blade-navigation/vc-blade-navigation.vue +67 -4
  186. package/shared/components/blade-navigation/composables/useBladeNavigation/index.ts +13 -10
  187. package/shared/components/blade-navigation/composables/useBladeNavigation/internal/bladeActions.ts +7 -4
  188. package/shared/components/blade-navigation/composables/useBladeNavigation/internal/bladeRouteResolver.ts +4 -1
  189. package/shared/components/blade-navigation/composables/useBladeNavigation/internal/routerUtils.ts +4 -1
  190. package/shared/components/change-password/change-password.vue +1 -1
  191. package/shared/components/draggable-dashboard/composables/useDashboardDragAndDrop.ts +14 -5
  192. package/shared/components/draggable-dashboard/composables/useLayoutPersistence.ts +5 -2
  193. package/shared/components/index.ts +2 -0
  194. package/shared/components/notifications/composables/useContainer/index.ts +8 -6
  195. package/shared/components/notifications/composables/useInstance/index.ts +4 -1
  196. package/shared/components/notifications/core/notification.ts +10 -7
  197. package/shared/components/popup-handler/components/vc-popup-container/vc-popup-container.vue +20 -1
  198. package/shared/components/sign-in/useExternalProvider.ts +6 -4
  199. package/shared/composables/index.ts +1 -0
  200. package/shared/composables/useExternalWidgets.ts +7 -4
  201. package/shared/composables/useMenuExpanded.ts +15 -1
  202. package/shared/composables/useTableSelection.ts +151 -0
  203. package/shared/composables/useTableSort.ts +4 -4
  204. package/shared/modules/assets-manager/components/assets-manager/assets-manager.vue +6 -3
  205. package/shared/pages/LoginPage/components/login/Login.vue +4 -1
  206. package/shared/utilities/colorUtils.ts +5 -12
  207. package/ui/components/atoms/vc-banner/vc-banner.vue +4 -1
  208. package/ui/components/atoms/vc-button/vc-button.vue +2 -25
  209. package/ui/components/atoms/vc-container/vc-container.vue +12 -3
  210. package/ui/components/atoms/vc-icon/vc-icon.vue +0 -10
  211. package/ui/components/atoms/vc-icon/vc-lucide-icon.vue +5 -2
  212. package/ui/components/atoms/vc-image/vc-image.vue +4 -1
  213. package/ui/components/atoms/vc-link/vc-link.vue +59 -54
  214. package/ui/components/atoms/vc-loading/vc-loading.vue +4 -0
  215. package/ui/components/atoms/vc-status/vc-status.vue +0 -5
  216. package/ui/components/atoms/vc-status-icon/vc-status-icon.vue +4 -4
  217. package/ui/components/atoms/vc-tooltip/vc-tooltip.vue +8 -1
  218. package/ui/components/atoms/vc-video/vc-video.vue +4 -2
  219. package/ui/components/atoms/vc-widget/vc-widget.vue +4 -1
  220. package/ui/components/molecules/vc-breadcrumbs/vc-breadcrumbs.vue +7 -2
  221. package/ui/components/molecules/vc-input/vc-input.vue +0 -1
  222. package/ui/components/molecules/vc-pagination/vc-pagination.vue +6 -1
  223. package/ui/components/molecules/vc-rating/vc-rating.vue +1 -1
  224. package/ui/components/molecules/vc-textarea/vc-textarea.vue +1 -1
  225. package/ui/components/molecules/vc-toast/vc-toast.vue +11 -1
  226. package/ui/components/organisms/vc-app/vc-app.vue +22 -3
  227. package/ui/components/organisms/vc-blade/_internal/vc-blade-toolbar/_internal/vc-blade-toolbar-buttons/_internal/vc-blade-toolbar-button/vc-blade-toolbar-button.vue +4 -1
  228. package/ui/components/organisms/vc-blade/_internal/vc-blade-toolbar/vc-blade-toolbar.vue +14 -14
  229. package/ui/components/organisms/vc-blade/vc-blade.vue +3 -1
  230. package/ui/components/organisms/vc-login-form/vc-login-form.vue +3 -1
  231. package/ui/components/organisms/vc-table/_internal/vc-table-cell/vc-table-cell.vue +34 -2
  232. package/ui/components/organisms/vc-table/composables/useTableActions.ts +7 -10
  233. package/ui/components/organisms/vc-table/composables/useTableColumnResize.ts +4 -1
  234. package/ui/components/organisms/vc-table/composables/useTableRowReorder.ts +5 -2
  235. package/ui/components/organisms/vc-table/composables/useTableSelection.ts +26 -18
  236. package/ui/components/organisms/vc-table/composables/useTableState.ts +4 -1
  237. package/core/services/global-search-service.ts +0 -36
  238. package/dist/core/services/global-search-service.d.ts +0 -10
  239. package/dist/core/services/global-search-service.d.ts.map +0 -1
@@ -0,0 +1,102 @@
1
+ <template>
2
+ <div class="vc-ai-agent-header">
3
+ <div class="vc-ai-agent-header__icon">
4
+ <VcVirtoOzLogo :width="22" :height="24" show-dots />
5
+ </div>
6
+
7
+ <div class="vc-ai-agent-header__wrapper">
8
+ <div class="vc-ai-agent-header__content">
9
+ <div class="vc-ai-agent-header__title">
10
+ {{ title }}
11
+ </div>
12
+ </div>
13
+ </div>
14
+
15
+ <div class="vc-ai-agent-header__controls">
16
+ <div
17
+ v-if="isExpanded"
18
+ class="vc-ai-agent-header__button"
19
+ title="Collapse"
20
+ @click="$emit('collapse')"
21
+ >
22
+ <VcIcon icon="material-minimize" />
23
+ </div>
24
+ <div
25
+ v-else
26
+ class="vc-ai-agent-header__button"
27
+ title="Expand"
28
+ @click="$emit('expand')"
29
+ >
30
+ <VcIcon icon="material-toolbar" />
31
+ </div>
32
+ <div
33
+ class="vc-ai-agent-header__button"
34
+ title="Close"
35
+ @click="$emit('close')"
36
+ >
37
+ <VcIcon icon="material-close" />
38
+ </div>
39
+ </div>
40
+ </div>
41
+ </template>
42
+
43
+ <script lang="ts" setup>
44
+ import { VcIcon } from "../../../../../ui/components";
45
+ import VcVirtoOzLogo from "./VcVirtoOzLogo.vue";
46
+
47
+ defineProps<{
48
+ title?: string;
49
+ isExpanded: boolean;
50
+ }>();
51
+
52
+ defineEmits<{
53
+ (e: "close"): void;
54
+ (e: "expand"): void;
55
+ (e: "collapse"): void;
56
+ }>();
57
+ </script>
58
+
59
+ <style lang="scss">
60
+ .vc-ai-agent-header {
61
+ @apply tw-shrink-0 tw-flex tw-items-center tw-py-0 tw-px-6 tw-border-solid tw-border-b;
62
+ height: var(--blade-header-height, 70px);
63
+ background-color: var(--blade-header-background-color, var(--additional-50));
64
+ border-color: var(--blade-header-border-color, var(--neutrals-200));
65
+
66
+ &__left {
67
+ @apply tw-flex tw-flex-row tw-items-center;
68
+ }
69
+
70
+ &__icon {
71
+ @apply tw-mr-3;
72
+ color: var(--primary-500);
73
+ }
74
+
75
+ &__wrapper {
76
+ @apply tw-flex tw-items-center tw-justify-between tw-grow tw-basis-0 tw-overflow-hidden;
77
+ }
78
+
79
+ &__content {
80
+ @apply tw-overflow-hidden;
81
+ }
82
+
83
+ &__title {
84
+ @apply tw-truncate tw-font-semibold;
85
+ color: var(--blade-header-title-color, var(--neutrals-950));
86
+ font-size: var(--blade-header-title-font-size, 19px);
87
+ }
88
+
89
+ &__controls {
90
+ @apply tw-flex tw-items-center;
91
+ }
92
+
93
+ &__button {
94
+ @apply tw-ml-2.5 tw-cursor-pointer;
95
+ color: var(--blade-header-button-color, var(--neutrals-400));
96
+
97
+ &:hover {
98
+ color: var(--blade-header-button-color-hover, var(--neutrals-500));
99
+ }
100
+ }
101
+ }
102
+ </style>
@@ -0,0 +1,107 @@
1
+ <template>
2
+ <div class="vc-ai-agent-iframe">
3
+ <!-- Loading overlay -->
4
+ <Transition name="fade">
5
+ <VcAiAgentLoader v-if="isLoading && url" :text="$t('AI_AGENT.LOADING')" />
6
+ </Transition>
7
+
8
+ <iframe
9
+ v-if="url"
10
+ ref="iframeRef"
11
+ :src="url"
12
+ class="vc-ai-agent-iframe__frame"
13
+ :class="{ 'vc-ai-agent-iframe__frame--loading': isLoading }"
14
+ sandbox="allow-scripts allow-same-origin allow-forms allow-popups allow-popups-to-escape-sandbox"
15
+ allow="clipboard-read; clipboard-write"
16
+ @load="onLoad"
17
+ />
18
+ <div v-else class="vc-ai-agent-iframe__placeholder">
19
+ <VcIcon icon="lucide-sparkles" size="xl" class="vc-ai-agent-iframe__placeholder-icon" />
20
+ <p class="vc-ai-agent-iframe__placeholder-text">AI Agent URL not configured</p>
21
+ </div>
22
+ </div>
23
+ </template>
24
+
25
+ <script lang="ts" setup>
26
+ import { ref, onMounted, watch } from "vue";
27
+ import { VcIcon } from "../../../../../ui/components";
28
+ import VcAiAgentLoader from "./VcAiAgentLoader.vue";
29
+
30
+ const props = defineProps<{
31
+ url: string;
32
+ }>();
33
+
34
+ const emit = defineEmits<{
35
+ (e: "iframe-ready", iframe: HTMLIFrameElement): void;
36
+ }>();
37
+
38
+ const iframeRef = ref<HTMLIFrameElement | null>(null);
39
+ const isLoading = ref(true);
40
+
41
+ // Emit iframe ref as soon as it's available in the DOM (before @load)
42
+ // This is critical because chatbot sends CHAT_READY before @load fires
43
+ watch(iframeRef, (iframe) => {
44
+ if (iframe) {
45
+ emit("iframe-ready", iframe);
46
+ }
47
+ }, { immediate: true });
48
+
49
+ const onLoad = () => {
50
+ // Hide loader after iframe loads
51
+ isLoading.value = false;
52
+ console.debug("[VcAiAgentIframe] Iframe loaded");
53
+ };
54
+
55
+ onMounted(() => {
56
+ // Emit again on mount in case watch didn't catch it
57
+ if (iframeRef.value) {
58
+ emit("iframe-ready", iframeRef.value);
59
+ }
60
+ });
61
+
62
+ // Reset loading state when URL changes
63
+ watch(() => props.url, () => {
64
+ isLoading.value = true;
65
+ });
66
+ </script>
67
+
68
+ <style lang="scss">
69
+ // Fade transition
70
+ .fade-enter-active,
71
+ .fade-leave-active {
72
+ transition: opacity 0.3s ease;
73
+ }
74
+
75
+ .fade-enter-from,
76
+ .fade-leave-to {
77
+ opacity: 0;
78
+ }
79
+
80
+ .vc-ai-agent-iframe {
81
+ @apply tw-w-full tw-h-full tw-flex tw-flex-col tw-relative;
82
+
83
+ &__frame {
84
+ @apply tw-w-full tw-h-full tw-border-0 tw-flex-1;
85
+ background-color: var(--ai-panel-iframe-bg, var(--additional-50));
86
+ transition: opacity 0.3s ease;
87
+
88
+ &--loading {
89
+ opacity: 0;
90
+ }
91
+ }
92
+
93
+ &__placeholder {
94
+ @apply tw-w-full tw-h-full tw-flex tw-flex-col tw-items-center tw-justify-center tw-gap-4;
95
+ background-color: var(--ai-panel-placeholder-bg, var(--neutrals-50));
96
+ }
97
+
98
+ &__placeholder-icon {
99
+ color: var(--ai-panel-placeholder-icon-color, var(--neutrals-300));
100
+ }
101
+
102
+ &__placeholder-text {
103
+ @apply tw-text-sm tw-m-0;
104
+ color: var(--ai-panel-placeholder-text-color, var(--neutrals-500));
105
+ }
106
+ }
107
+ </style>
@@ -0,0 +1,72 @@
1
+ <template>
2
+ <div class="vc-ai-agent-loader">
3
+ <div class="vc-ai-agent-loader__content">
4
+ <div class="vc-ai-agent-loader__logo">
5
+ <VcVirtoOzLogo :width="80" :height="87" animated />
6
+ </div>
7
+ <p v-if="text" class="vc-ai-agent-loader__text">{{ text }}</p>
8
+ </div>
9
+ </div>
10
+ </template>
11
+
12
+ <script lang="ts" setup>
13
+ import VcVirtoOzLogo from "./VcVirtoOzLogo.vue";
14
+
15
+ export interface Props {
16
+ /** Loading text to display */
17
+ text?: string;
18
+ }
19
+
20
+ defineProps<Props>();
21
+ </script>
22
+
23
+ <style lang="scss">
24
+ @keyframes virto-oz-float {
25
+ 0%,
26
+ 100% {
27
+ transform: translateY(0);
28
+ }
29
+ 50% {
30
+ transform: translateY(-6px);
31
+ }
32
+ }
33
+
34
+ @keyframes virto-oz-glow {
35
+ 0%,
36
+ 100% {
37
+ filter: drop-shadow(0 4px 15px rgba(255, 149, 51, 0.4));
38
+ }
39
+ 25% {
40
+ filter: drop-shadow(0 4px 15px rgba(255, 79, 187, 0.4));
41
+ }
42
+ 50% {
43
+ filter: drop-shadow(0 4px 15px rgba(28, 145, 252, 0.4));
44
+ }
45
+ 75% {
46
+ filter: drop-shadow(0 4px 15px rgba(154, 45, 248, 0.4));
47
+ }
48
+ }
49
+
50
+ .vc-ai-agent-loader {
51
+ @apply tw-absolute tw-inset-0 tw-flex tw-items-center tw-justify-center tw-z-10;
52
+ background: linear-gradient(
53
+ 135deg,
54
+ var(--additional-50) 0%,
55
+ var(--neutrals-50) 100%
56
+ );
57
+
58
+ &__content {
59
+ @apply tw-flex tw-flex-col tw-items-center tw-gap-5;
60
+ }
61
+
62
+ &__logo {
63
+ animation: virto-oz-float 2.5s ease-in-out infinite,
64
+ virto-oz-glow 4s linear infinite;
65
+ }
66
+
67
+ &__text {
68
+ @apply tw-text-sm tw-m-0 tw-font-medium;
69
+ color: var(--neutrals-600);
70
+ }
71
+ }
72
+ </style>
@@ -0,0 +1,136 @@
1
+ <template>
2
+ <svg
3
+ :width="width"
4
+ :height="height"
5
+ viewBox="0 0 99 108"
6
+ fill="none"
7
+ xmlns="http://www.w3.org/2000/svg"
8
+ class="vc-virto-oz-logo"
9
+ >
10
+ <!-- White hexagon background -->
11
+ <path
12
+ d="M40.1199 2.41576C45.6989 -0.805254 52.5724 -0.805253 58.1514 2.41576L89.2556 20.3738C94.8345 23.5948 98.2713 29.5475 98.2713 35.9895V71.9055C98.2713 78.3475 94.8345 84.3002 89.2556 87.5212L58.1514 105.479C52.5724 108.7 45.6989 108.7 40.1199 105.479L9.01574 87.5212C3.43678 84.3002 0 78.3475 0 71.9055V35.9895C0 29.5475 3.43678 23.5948 9.01574 20.3738L40.1199 2.41576Z"
13
+ fill="white"
14
+ />
15
+
16
+ <!-- Static gradient spiral pattern (using foreignObject with conic-gradient) -->
17
+ <g v-if="!animated" :clip-path="`url(#${clipPathId})`">
18
+ <g transform="matrix(0 -0.161514 0.160417 0 49.1357 53.9479)">
19
+ <foreignObject x="-334.01" y="-334.01" width="668.02" height="668.02">
20
+ <div
21
+ xmlns="http://www.w3.org/1999/xhtml"
22
+ style="background: conic-gradient(from 90deg, rgba(255, 149, 51, 1) 0deg, rgba(255, 79, 187, 1) 43.2deg, rgba(234, 255, 79, 1) 143.654deg, rgba(28, 145, 252, 1) 252.692deg, rgba(154, 45, 248, 1) 316.8deg, rgba(255, 149, 51, 1) 360deg); height: 100%; width: 100%; opacity: 1;"
23
+ />
24
+ </foreignObject>
25
+ </g>
26
+ </g>
27
+
28
+ <!-- Animated gradient spiral pattern (using linearGradient with animate) -->
29
+ <path
30
+ v-else
31
+ d="M72.9199 46.4165C73.2527 46.9495 73.5831 47.4996 73.9092 48.0679C78.4022 55.8976 79.0605 66.3551 76.7793 76.6079C74.5635 86.5668 69.6096 96.1504 62.9795 102.692L58.1514 105.48C52.5725 108.701 45.699 108.701 40.1201 105.48L21.665 94.8247C29.1398 95.4723 37.0781 94.4072 44.3848 91.9233C54.5984 88.4512 63.7463 82.1497 68.6807 73.6548C75.124 62.5619 75.4351 53.2694 72.9199 46.4165ZM56.5205 29.5093C65.5697 29.5331 75.1259 33.9964 83.0215 40.8853C89.9848 46.9608 95.5441 54.8286 98.2715 62.9478V71.9058C98.2715 78.3478 94.8348 84.301 89.2559 87.522L69.4658 98.9468C74.0688 92.6731 77.4601 84.9996 79.207 77.1479C81.5723 66.517 80.9666 55.3703 76.0654 46.8296C69.6817 35.7054 61.7917 30.789 54.6006 29.5396C55.2271 29.5179 55.867 29.5076 56.5205 29.5093ZM0 50.688C3.16812 57.476 8.04614 63.7935 13.8389 68.853C21.9305 75.9204 31.9368 80.6465 41.7578 80.6724C54.5833 80.706 62.7851 76.3308 67.4629 70.728C67.1684 71.2812 66.8584 71.8408 66.5303 72.4058C61.9844 80.2318 53.417 86.2251 43.584 89.5679C34.4355 92.6779 24.3375 93.4401 15.6328 91.3423L9.01562 87.522C3.43667 84.301 0 78.3478 0 71.9058V50.688ZM26.7559 10.1313C22.5672 16.2131 19.6599 23.483 18.2695 30.9126C16.3077 41.3962 17.3358 52.3558 22.2188 60.8647C28.6034 71.9905 36.4946 76.9061 43.6865 78.1548C43.0594 78.1765 42.4188 78.1868 41.7646 78.1851C32.7113 78.1612 23.2602 73.7801 15.4746 66.98C8.2197 60.6434 2.52684 52.3035 0 43.7075V35.9897C6.33247e-05 29.5479 3.43687 23.5955 9.01562 20.3745L26.7559 10.1313ZM51.2432 31.6978C54.0287 31.7659 57.0796 32.4389 60.1875 33.9976C58.8529 33.7029 57.5584 33.677 56.3965 33.9722L57.0088 36.3833C58.4774 36.0102 60.6741 36.4134 63.0332 37.8237C65.3553 39.212 67.6865 41.4953 69.375 44.5923C70.7081 47.0376 71.6489 50.015 71.8535 53.4839C71.4414 52.1828 70.818 51.0505 69.9824 50.1929L68.2012 51.9282C69.2585 53.0135 70.0076 55.1176 69.9658 57.8657C69.9246 60.5708 69.1121 63.731 67.2744 66.7417C65.8225 69.1203 63.7129 71.4251 60.8086 73.3374C61.7311 72.329 62.4012 71.2217 62.7266 70.0679L60.333 69.3931C59.9218 70.8514 58.4742 72.5523 56.0732 73.8901C53.71 75.207 50.5672 76.0837 47.041 75.9976C44.2551 75.9294 41.204 75.2549 38.0957 73.6958C39.4307 73.9907 40.7254 74.0184 41.8877 73.7231L41.2754 71.312C39.8069 71.6851 37.6101 71.2818 35.251 69.8716C32.9289 68.4834 30.5977 66.2 28.9092 63.103C27.5754 60.6565 26.6337 57.6775 26.4297 54.2065C26.8418 55.5096 27.4661 56.6438 28.3027 57.5024L30.084 55.7661C29.0267 54.6807 28.2775 52.5767 28.3193 49.8286C28.3606 47.1236 29.173 43.9633 31.0107 40.9526C32.4621 38.5751 34.5702 36.2706 37.4727 34.3589C36.5514 35.3666 35.8827 36.4736 35.5576 37.6265L37.9512 38.3013C38.3625 36.8429 39.8101 35.142 42.2109 33.8042C44.5742 32.4874 47.717 31.6116 51.2432 31.6978ZM40.1201 2.4165C45.6991 -0.804509 52.5724 -0.804509 58.1514 2.4165L75.6777 12.5356C68.4683 12.0708 60.8765 13.1991 53.8691 15.6333C43.6671 19.1774 34.5353 25.5492 29.6035 34.0396C23.1612 45.1305 22.8486 54.4223 25.3623 61.2749C25.0301 60.7429 24.7004 60.1936 24.375 59.6265C19.8639 51.765 18.8286 41.4457 20.7139 31.3706C22.4486 22.1007 26.633 13.1734 32.6709 6.71631L40.1201 2.4165ZM54.6855 17.9829C63.3703 14.966 72.9092 14.0715 81.3008 15.7817L89.2559 20.3745C94.8346 23.5955 98.2714 29.5479 98.2715 35.9897V56.4692C95.0606 49.9731 90.2759 43.9144 84.6562 39.0112C76.4636 31.8632 66.3524 27.0488 56.5273 27.0229C43.6978 26.9893 35.4935 31.3659 30.8164 36.9712C31.1117 36.4161 31.4247 35.8554 31.7539 35.2886C36.3024 27.4582 44.8633 21.395 54.6855 17.9829Z"
32
+ fill="url(#virto-oz-gradient-animated)"
33
+ />
34
+
35
+ <!-- White center circle -->
36
+ <path
37
+ d="M72.1266 54.0945C72.1266 60.0873 74.3028 71.0379 60.6238 82.8517C65.2871 73.0587 55.8339 77.1002 49.1209 77.1002C36.4152 77.1002 26.1152 66.8002 26.1152 54.0945C26.1152 41.3889 36.4152 31.0889 49.1209 31.0889C61.8266 31.0889 72.1266 41.3889 72.1266 54.0945Z"
38
+ fill="white"
39
+ />
40
+
41
+ <!-- Center dots (eyes) -->
42
+ <ellipse cx="49.12" cy="54.09" rx="4.05" ry="4.05" fill="#6C6C6C" />
43
+ <ellipse cx="60.62" cy="54.09" rx="4.05" ry="4.05" fill="#6C6C6C" />
44
+ <ellipse cx="37.62" cy="54.09" rx="4.05" ry="4.05" fill="#6C6C6C" />
45
+
46
+ <!-- Definitions -->
47
+ <defs>
48
+ <!-- Clip path for static conic gradient (spiral shape) -->
49
+ <clipPath v-if="!animated" :id="clipPathId">
50
+ <path
51
+ d="M72.9199 46.4165C73.2527 46.9495 73.5831 47.4996 73.9092 48.0679C78.4022 55.8976 79.0605 66.3551 76.7793 76.6079C74.5635 86.5668 69.6096 96.1504 62.9795 102.692L58.1514 105.48C52.5725 108.701 45.699 108.701 40.1201 105.48L21.665 94.8247C29.1398 95.4723 37.0781 94.4072 44.3848 91.9233C54.5984 88.4512 63.7463 82.1497 68.6807 73.6548C75.124 62.5619 75.4351 53.2694 72.9199 46.4165ZM56.5205 29.5093C65.5697 29.5331 75.1259 33.9964 83.0215 40.8853C89.9848 46.9608 95.5441 54.8286 98.2715 62.9478V71.9058C98.2715 78.3478 94.8348 84.301 89.2559 87.522L69.4658 98.9468C74.0688 92.6731 77.4601 84.9996 79.207 77.1479C81.5723 66.517 80.9666 55.3703 76.0654 46.8296C69.6817 35.7054 61.7917 30.789 54.6006 29.5396C55.2271 29.5179 55.867 29.5076 56.5205 29.5093ZM0 50.688C3.16812 57.476 8.04614 63.7935 13.8389 68.853C21.9305 75.9204 31.9368 80.6465 41.7578 80.6724C54.5833 80.706 62.7851 76.3308 67.4629 70.728C67.1684 71.2812 66.8584 71.8408 66.5303 72.4058C61.9844 80.2318 53.417 86.2251 43.584 89.5679C34.4355 92.6779 24.3375 93.4401 15.6328 91.3423L9.01562 87.522C3.43667 84.301 0 78.3478 0 71.9058V50.688ZM26.7559 10.1313C22.5672 16.2131 19.6599 23.483 18.2695 30.9126C16.3077 41.3962 17.3358 52.3558 22.2188 60.8647C28.6034 71.9905 36.4946 76.9061 43.6865 78.1548C43.0594 78.1765 42.4188 78.1868 41.7646 78.1851C32.7113 78.1612 23.2602 73.7801 15.4746 66.98C8.2197 60.6434 2.52684 52.3035 0 43.7075V35.9897C6.33247e-05 29.5479 3.43687 23.5955 9.01562 20.3745L26.7559 10.1313ZM51.2432 31.6978C54.0287 31.7659 57.0796 32.4389 60.1875 33.9976C58.8529 33.7029 57.5584 33.677 56.3965 33.9722L57.0088 36.3833C58.4774 36.0102 60.6741 36.4134 63.0332 37.8237C65.3553 39.212 67.6865 41.4953 69.375 44.5923C70.7081 47.0376 71.6489 50.015 71.8535 53.4839C71.4414 52.1828 70.818 51.0505 69.9824 50.1929L68.2012 51.9282C69.2585 53.0135 70.0076 55.1176 69.9658 57.8657C69.9246 60.5708 69.1121 63.731 67.2744 66.7417C65.8225 69.1203 63.7129 71.4251 60.8086 73.3374C61.7311 72.329 62.4012 71.2217 62.7266 70.0679L60.333 69.3931C59.9218 70.8514 58.4742 72.5523 56.0732 73.8901C53.71 75.207 50.5672 76.0837 47.041 75.9976C44.2551 75.9294 41.204 75.2549 38.0957 73.6958C39.4307 73.9907 40.7254 74.0184 41.8877 73.7231L41.2754 71.312C39.8069 71.6851 37.6101 71.2818 35.251 69.8716C32.9289 68.4834 30.5977 66.2 28.9092 63.103C27.5754 60.6565 26.6337 57.6775 26.4297 54.2065C26.8418 55.5096 27.4661 56.6438 28.3027 57.5024L30.084 55.7661C29.0267 54.6807 28.2775 52.5767 28.3193 49.8286C28.3606 47.1236 29.173 43.9633 31.0107 40.9526C32.4621 38.5751 34.5702 36.2706 37.4727 34.3589C36.5514 35.3666 35.8827 36.4736 35.5576 37.6265L37.9512 38.3013C38.3625 36.8429 39.8101 35.142 42.2109 33.8042C44.5742 32.4874 47.717 31.6116 51.2432 31.6978ZM40.1201 2.4165C45.6991 -0.804509 52.5724 -0.804509 58.1514 2.4165L75.6777 12.5356C68.4683 12.0708 60.8765 13.1991 53.8691 15.6333C43.6671 19.1774 34.5353 25.5492 29.6035 34.0396C23.1612 45.1305 22.8486 54.4223 25.3623 61.2749C25.0301 60.7429 24.7004 60.1936 24.375 59.6265C19.8639 51.765 18.8286 41.4457 20.7139 31.3706C22.4486 22.1007 26.633 13.1734 32.6709 6.71631L40.1201 2.4165ZM54.6855 17.9829C63.3703 14.966 72.9092 14.0715 81.3008 15.7817L89.2559 20.3745C94.8346 23.5955 98.2714 29.5479 98.2715 35.9897V56.4692C95.0606 49.9731 90.2759 43.9144 84.6562 39.0112C76.4636 31.8632 66.3524 27.0488 56.5273 27.0229C43.6978 26.9893 35.4935 31.3659 30.8164 36.9712C31.1117 36.4161 31.4247 35.8554 31.7539 35.2886C36.3024 27.4582 44.8633 21.395 54.6855 17.9829Z"
52
+ />
53
+ </clipPath>
54
+
55
+ <!-- Animated gradient -->
56
+ <linearGradient
57
+ v-if="animated"
58
+ id="virto-oz-gradient-animated"
59
+ gradientUnits="userSpaceOnUse"
60
+ x1="0"
61
+ y1="0"
62
+ x2="99"
63
+ y2="108"
64
+ >
65
+ <stop offset="0%" stop-color="#FF9533">
66
+ <animate
67
+ attributeName="stop-color"
68
+ values="#FF9533;#FF4FBB;#EAFF4F;#1C91FC;#9A2DF8;#FF9533"
69
+ dur="4s"
70
+ repeatCount="indefinite"
71
+ />
72
+ </stop>
73
+ <stop offset="25%" stop-color="#FF4FBB">
74
+ <animate
75
+ attributeName="stop-color"
76
+ values="#FF4FBB;#EAFF4F;#1C91FC;#9A2DF8;#FF9533;#FF4FBB"
77
+ dur="4s"
78
+ repeatCount="indefinite"
79
+ />
80
+ </stop>
81
+ <stop offset="50%" stop-color="#EAFF4F">
82
+ <animate
83
+ attributeName="stop-color"
84
+ values="#EAFF4F;#1C91FC;#9A2DF8;#FF9533;#FF4FBB;#EAFF4F"
85
+ dur="4s"
86
+ repeatCount="indefinite"
87
+ />
88
+ </stop>
89
+ <stop offset="75%" stop-color="#1C91FC">
90
+ <animate
91
+ attributeName="stop-color"
92
+ values="#1C91FC;#9A2DF8;#FF9533;#FF4FBB;#EAFF4F;#1C91FC"
93
+ dur="4s"
94
+ repeatCount="indefinite"
95
+ />
96
+ </stop>
97
+ <stop offset="100%" stop-color="#9A2DF8">
98
+ <animate
99
+ attributeName="stop-color"
100
+ values="#9A2DF8;#FF9533;#FF4FBB;#EAFF4F;#1C91FC;#9A2DF8"
101
+ dur="4s"
102
+ repeatCount="indefinite"
103
+ />
104
+ </stop>
105
+ </linearGradient>
106
+ </defs>
107
+ </svg>
108
+ </template>
109
+
110
+ <script lang="ts" setup>
111
+ import { computed } from "vue";
112
+
113
+ export interface Props {
114
+ /** Width in pixels */
115
+ width?: number;
116
+ /** Height in pixels */
117
+ height?: number;
118
+ /** Enable animated gradient */
119
+ animated?: boolean;
120
+ }
121
+
122
+ withDefaults(defineProps<Props>(), {
123
+ width: 22,
124
+ height: 24,
125
+ animated: false,
126
+ });
127
+
128
+ // Generate unique ID for clip path to avoid conflicts when multiple logos are rendered
129
+ const clipPathId = computed(() => `virto-oz-clip-${Math.random().toString(36).substring(2, 11)}`);
130
+ </script>
131
+
132
+ <style lang="scss">
133
+ .vc-virto-oz-logo {
134
+ display: block;
135
+ }
136
+ </style>
@@ -0,0 +1 @@
1
+ export { default as VcAiAgentPanel } from "./VcAiAgentPanel.vue";
@@ -0,0 +1,4 @@
1
+ export { useAiAgent, provideAiAgentService, createAiAgentToolbarButton } from "./useAiAgent";
2
+ export type { UseAiAgentReturn, ProvideAiAgentServiceOptions } from "./useAiAgent";
3
+
4
+ export { useAiAgentContext, clearPreviewState } from "./useAiAgentContext";
@@ -0,0 +1,231 @@
1
+ import { inject, provide } from "vue";
2
+ import { createAiAgentService, type IAiAgentServiceInternal } from "../services/ai-agent-service";
3
+ import { AiAgentServiceKey, LanguageServiceKey, ToolbarServiceKey } from "../../../../injection-keys";
4
+ import type {
5
+ IAiAgentService,
6
+ IAiAgentConfig,
7
+ IAiAgentMessage,
8
+ IAiAgentBladeContext,
9
+ } from "../types";
10
+ import { createLogger, InjectionError } from "../../../utilities";
11
+ import { useUser } from "../../../composables/useUser";
12
+ import { useBladeNavigation } from "../../../../shared/components/blade-navigation/composables";
13
+ import { AI_AGENT_TOOLBAR_BUTTON_ID, AI_AGENT_TOOLBAR_BUTTON_ICON, AI_AGENT_TOOLBAR_BUTTON_TITLE } from "../constants";
14
+ import type { IToolbarService } from "../../../services/toolbar-service";
15
+
16
+ const logger = createLogger("use-ai-agent");
17
+
18
+ /**
19
+ * Options for providing the AI agent service
20
+ */
21
+ export interface ProvideAiAgentServiceOptions {
22
+ /** Initial configuration for the AI agent */
23
+ config?: Partial<IAiAgentConfig>;
24
+ /**
25
+ * Whether to add the AI button to all blade toolbars automatically.
26
+ * Default: true
27
+ */
28
+ addGlobalToolbarButton?: boolean;
29
+ }
30
+
31
+ /**
32
+ * Provides the AI agent service at the current component level.
33
+ * Should be called once at the app level (in VcApp or plugin install).
34
+ *
35
+ * @param options - Configuration options
36
+ */
37
+ export function provideAiAgentService(options?: ProvideAiAgentServiceOptions): IAiAgentServiceInternal {
38
+ const languageService = inject(LanguageServiceKey);
39
+ const { user, getAccessToken } = useUser();
40
+ const { blades, openBlade, resolveBladeByName } = useBladeNavigation();
41
+
42
+ // Create the service
43
+ const service = createAiAgentService({
44
+ userGetter: () => {
45
+ if (!user.value) return undefined;
46
+ return {
47
+ id: user.value.id,
48
+ userName: user.value.userName,
49
+ isAdministrator: user.value.isAdministrator,
50
+ permissions: user.value.permissions,
51
+ };
52
+ },
53
+ bladeGetter: (): IAiAgentBladeContext | null => {
54
+ // Get the last (most recent) blade
55
+ const lastBlade = blades.value[blades.value.length - 1];
56
+ if (!lastBlade) return null;
57
+
58
+ return {
59
+ id: lastBlade.type?.name?.toLowerCase() ?? "unknown",
60
+ name: lastBlade.type?.name ?? "Unknown",
61
+ title: lastBlade.props?.navigation?.instance?.title,
62
+ param: lastBlade.props?.param,
63
+ options: lastBlade.props?.options,
64
+ };
65
+ },
66
+ localeGetter: () => languageService?.currentLocale.value ?? "en",
67
+ tokenGetter: getAccessToken,
68
+ navigateToBlade: (bladeName: string, param?: string, bladeOptions?: Record<string, unknown>) => {
69
+ const blade = resolveBladeByName(bladeName);
70
+ if (blade) {
71
+ openBlade({ blade, param, options: bladeOptions });
72
+ logger.debug(`Navigated to blade: ${bladeName}`);
73
+ } else {
74
+ logger.warn(`Blade not found: ${bladeName}`);
75
+ }
76
+ },
77
+ reloadBlade: () => {
78
+ // Emit reload on current blade if available (last blade in the list)
79
+ const lastBlade = blades.value[blades.value.length - 1];
80
+ if (lastBlade?.props?.navigation?.instance?.reload) {
81
+ lastBlade.props.navigation.instance.reload();
82
+ logger.debug("Current blade reloaded");
83
+ }
84
+ },
85
+ initialConfig: options?.config,
86
+ });
87
+
88
+ provide(AiAgentServiceKey, service);
89
+ logger.debug("AiAgentService provided");
90
+
91
+ // Register global toolbar button with wildcard "*" if enabled
92
+ const addGlobalToolbarButton = options?.addGlobalToolbarButton ?? true;
93
+ if (addGlobalToolbarButton) {
94
+ // Get toolbar service from injection (already created by VcShell plugin)
95
+ const toolbarService = inject<IToolbarService | null>(ToolbarServiceKey, null);
96
+ if (toolbarService) {
97
+ toolbarService.registerToolbarItem(
98
+ {
99
+ id: AI_AGENT_TOOLBAR_BUTTON_ID,
100
+ icon: AI_AGENT_TOOLBAR_BUTTON_ICON,
101
+ title: AI_AGENT_TOOLBAR_BUTTON_TITLE,
102
+ clickHandler: () => {
103
+ service.togglePanel();
104
+ },
105
+ },
106
+ "*", // Wildcard for all blades
107
+ );
108
+ logger.debug("Global AI toolbar button registered with wildcard '*'");
109
+ } else {
110
+ logger.warn("Toolbar service not available, AI button not registered");
111
+ }
112
+ }
113
+
114
+ return service;
115
+ }
116
+
117
+ /**
118
+ * Return type for useAiAgent composable
119
+ */
120
+ export interface UseAiAgentReturn {
121
+ /** Current panel state ("closed" | "open" | "expanded") */
122
+ panelState: IAiAgentService["panelState"];
123
+ /** Panel configuration */
124
+ config: IAiAgentService["config"];
125
+ /** Current context (reactive) */
126
+ context: IAiAgentService["context"];
127
+ /** Whether panel is open (not closed) */
128
+ isOpen: IAiAgentService["isOpen"];
129
+ /** Whether panel is in expanded state */
130
+ isExpanded: IAiAgentService["isExpanded"];
131
+ /** Total count of items in current context */
132
+ totalItemsCount: IAiAgentService["totalItemsCount"];
133
+ /** Open the AI panel */
134
+ openPanel: () => void;
135
+ /** Close the AI panel */
136
+ closePanel: () => void;
137
+ /** Toggle panel open/close */
138
+ togglePanel: () => void;
139
+ /** Expand the panel to larger width */
140
+ expandPanel: () => void;
141
+ /** Collapse panel to normal width */
142
+ collapsePanel: () => void;
143
+ /** Update configuration */
144
+ setConfig: (config: Partial<IAiAgentConfig>) => void;
145
+ /** Send message to AI agent iframe */
146
+ sendMessage: IAiAgentService["sendMessage"];
147
+ /** Register message handler, returns unsubscribe function */
148
+ onMessage: (handler: (event: IAiAgentMessage) => void) => () => void;
149
+ }
150
+
151
+ /**
152
+ * Composable for interacting with the AI agent panel.
153
+ *
154
+ * @example
155
+ * ```typescript
156
+ * const { togglePanel, isOpen, onMessage } = useAiAgent();
157
+ *
158
+ * // Toggle AI panel
159
+ * const handleAiClick = () => togglePanel();
160
+ *
161
+ * // Listen for messages from AI agent
162
+ * onMessage((message) => {
163
+ * if (message.type === 'PREVIEW_CHANGES') {
164
+ * // Handle preview changes
165
+ * }
166
+ * });
167
+ * ```
168
+ */
169
+ export function useAiAgent(): UseAiAgentReturn {
170
+ const service = inject(AiAgentServiceKey);
171
+ if (!service) {
172
+ logger.error("AiAgentService not found. Did you forget to call provideAiAgentService()?");
173
+ throw new InjectionError("AiAgentService");
174
+ }
175
+
176
+ return {
177
+ // State
178
+ panelState: service.panelState,
179
+ config: service.config,
180
+ context: service.context,
181
+ isOpen: service.isOpen,
182
+ isExpanded: service.isExpanded,
183
+ totalItemsCount: service.totalItemsCount,
184
+
185
+ // Panel control
186
+ openPanel: service.openPanel,
187
+ closePanel: service.closePanel,
188
+ togglePanel: service.togglePanel,
189
+ expandPanel: service.expandPanel,
190
+ collapsePanel: service.collapsePanel,
191
+
192
+ // Configuration
193
+ setConfig: service.setConfig,
194
+
195
+ // Communication
196
+ sendMessage: service.sendMessage,
197
+ onMessage: service.onMessage,
198
+ };
199
+ }
200
+
201
+ /**
202
+ * Creates a toolbar button configuration for opening the AI agent.
203
+ * Can be used directly in blade toolbar definitions.
204
+ *
205
+ * @example
206
+ * ```typescript
207
+ * const bladeToolbar = [
208
+ * createAiAgentToolbarButton(),
209
+ * // ... other toolbar items
210
+ * ];
211
+ * ```
212
+ */
213
+ export function createAiAgentToolbarButton(options?: { title?: string; icon?: string }) {
214
+ return {
215
+ id: "ai-agent-toggle",
216
+ icon: options?.icon ?? "lucide-sparkles",
217
+ title: options?.title ?? "AI Assistant",
218
+ clickHandler: () => {
219
+ try {
220
+ const service = inject(AiAgentServiceKey);
221
+ if (service) {
222
+ service.togglePanel();
223
+ } else {
224
+ logger.warn("AiAgentService not available for toolbar button");
225
+ }
226
+ } catch (e) {
227
+ logger.error("Failed to toggle AI panel:", e);
228
+ }
229
+ },
230
+ };
231
+ }