@sinequa/atomic-angular 0.0.140

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 (174) hide show
  1. package/assets/tailwind.css +248 -0
  2. package/esm2022/lib/assistant/index.mjs +2 -0
  3. package/esm2022/lib/assistant/signalR.web.service.mjs +81 -0
  4. package/esm2022/lib/components/dropdown.mjs +127 -0
  5. package/esm2022/lib/components/index.mjs +5 -0
  6. package/esm2022/lib/components/menu/index.mjs +3 -0
  7. package/esm2022/lib/components/menu/menu-item.mjs +22 -0
  8. package/esm2022/lib/components/menu/menu.mjs +99 -0
  9. package/esm2022/lib/components/metadata/index.mjs +2 -0
  10. package/esm2022/lib/components/metadata/metadata.component.mjs +65 -0
  11. package/esm2022/lib/components/theme/index.mjs +3 -0
  12. package/esm2022/lib/components/theme/theme-selector.component.mjs +67 -0
  13. package/esm2022/lib/components/theme/theme-toggle.component.mjs +67 -0
  14. package/esm2022/lib/directives/index.mjs +5 -0
  15. package/esm2022/lib/directives/infinite-scroll.directive.mjs +47 -0
  16. package/esm2022/lib/directives/select-article-on-click.directive.mjs +39 -0
  17. package/esm2022/lib/directives/show-bookmark.directive.mjs +55 -0
  18. package/esm2022/lib/directives/theme-provider.directive.mjs +31 -0
  19. package/esm2022/lib/guards/auth.guard.mjs +27 -0
  20. package/esm2022/lib/guards/index.mjs +3 -0
  21. package/esm2022/lib/guards/initialization.guard.mjs +41 -0
  22. package/esm2022/lib/interceptors/audit.interceptor.mjs +23 -0
  23. package/esm2022/lib/interceptors/auth.interceptor.mjs +49 -0
  24. package/esm2022/lib/interceptors/body.interceptor.mjs +24 -0
  25. package/esm2022/lib/interceptors/error.interceptor.mjs +35 -0
  26. package/esm2022/lib/interceptors/index.mjs +7 -0
  27. package/esm2022/lib/interceptors/toast.interceptor.mjs +27 -0
  28. package/esm2022/lib/models/aggregation.mjs +2 -0
  29. package/esm2022/lib/models/article-metadata.mjs +2 -0
  30. package/esm2022/lib/models/autocomplete.mjs +2 -0
  31. package/esm2022/lib/models/custom-json.mjs +2 -0
  32. package/esm2022/lib/models/filter-dropdown.mjs +2 -0
  33. package/esm2022/lib/models/index.mjs +7 -0
  34. package/esm2022/lib/models/user-settings.mjs +2 -0
  35. package/esm2022/lib/pipes/highlight-word.pipe.mjs +33 -0
  36. package/esm2022/lib/pipes/index.mjs +3 -0
  37. package/esm2022/lib/pipes/source-icon.pipe.mjs +43 -0
  38. package/esm2022/lib/providers/eager-provider.mjs +24 -0
  39. package/esm2022/lib/providers/index.mjs +2 -0
  40. package/esm2022/lib/public-api.mjs +19 -0
  41. package/esm2022/lib/resolvers/index.mjs +2 -0
  42. package/esm2022/lib/resolvers/query-name-resolver.mjs +14 -0
  43. package/esm2022/lib/resources/index.mjs +2 -0
  44. package/esm2022/lib/resources/themes.mjs +53 -0
  45. package/esm2022/lib/services/application.service.mjs +245 -0
  46. package/esm2022/lib/services/autocomplete.service.mjs +85 -0
  47. package/esm2022/lib/services/drawer/backdrop.service.mjs +23 -0
  48. package/esm2022/lib/services/drawer/drawer-stack.service.mjs +152 -0
  49. package/esm2022/lib/services/drawer/drawer.service.mjs +38 -0
  50. package/esm2022/lib/services/index.mjs +12 -0
  51. package/esm2022/lib/services/label.service.mjs +161 -0
  52. package/esm2022/lib/services/navigation.service.mjs +59 -0
  53. package/esm2022/lib/services/saved-searches.service.mjs +75 -0
  54. package/esm2022/lib/services/search.service.mjs +89 -0
  55. package/esm2022/lib/services/selection-history.service.mjs +92 -0
  56. package/esm2022/lib/services/selection.service.mjs +87 -0
  57. package/esm2022/lib/stores/aggregations.store.mjs +62 -0
  58. package/esm2022/lib/stores/app.store.mjs +265 -0
  59. package/esm2022/lib/stores/application.store.mjs +93 -0
  60. package/esm2022/lib/stores/index.mjs +9 -0
  61. package/esm2022/lib/stores/principal.store.mjs +47 -0
  62. package/esm2022/lib/stores/query-params.store.mjs +208 -0
  63. package/esm2022/lib/stores/selection.store.mjs +46 -0
  64. package/esm2022/lib/stores/theme.store.mjs +116 -0
  65. package/esm2022/lib/stores/user-settings.store.mjs +272 -0
  66. package/esm2022/lib/tokens/highlights.mjs +32 -0
  67. package/esm2022/lib/tokens/index.mjs +2 -0
  68. package/esm2022/lib/utils/debounced-signal.mjs +38 -0
  69. package/esm2022/lib/utils/index.mjs +8 -0
  70. package/esm2022/lib/utils/inline-worker.mjs +40 -0
  71. package/esm2022/lib/utils/query.mjs +58 -0
  72. package/esm2022/lib/utils/routes.mjs +28 -0
  73. package/esm2022/lib/utils/tailwind-utils.mjs +6 -0
  74. package/esm2022/lib/utils/theme-body-hook.mjs +18 -0
  75. package/esm2022/lib/utils/theme-registry.mjs +6 -0
  76. package/esm2022/lib/web-services/aggregations.service.mjs +104 -0
  77. package/esm2022/lib/web-services/app.service.mjs +48 -0
  78. package/esm2022/lib/web-services/audit.service.mjs +122 -0
  79. package/esm2022/lib/web-services/index.mjs +10 -0
  80. package/esm2022/lib/web-services/json-method-plugin.service.mjs +54 -0
  81. package/esm2022/lib/web-services/preview.service.mjs +327 -0
  82. package/esm2022/lib/web-services/principal.service.mjs +46 -0
  83. package/esm2022/lib/web-services/query.service.mjs +123 -0
  84. package/esm2022/lib/web-services/text-chunck.service.mjs +46 -0
  85. package/esm2022/public-api.mjs +5 -0
  86. package/esm2022/sinequa-atomic-angular.mjs +5 -0
  87. package/fesm2022/sinequa-atomic-angular.mjs +4204 -0
  88. package/fesm2022/sinequa-atomic-angular.mjs.map +1 -0
  89. package/index.d.ts +5 -0
  90. package/lib/assistant/index.d.ts +1 -0
  91. package/lib/assistant/signalR.web.service.d.ts +46 -0
  92. package/lib/components/dropdown.d.ts +50 -0
  93. package/lib/components/index.d.ts +4 -0
  94. package/lib/components/menu/index.d.ts +2 -0
  95. package/lib/components/menu/menu-item.d.ts +8 -0
  96. package/lib/components/menu/menu.d.ts +24 -0
  97. package/lib/components/metadata/index.d.ts +1 -0
  98. package/lib/components/metadata/metadata.component.d.ts +24 -0
  99. package/lib/components/theme/index.d.ts +2 -0
  100. package/lib/components/theme/theme-selector.component.d.ts +70 -0
  101. package/lib/components/theme/theme-toggle.component.d.ts +10 -0
  102. package/lib/directives/index.d.ts +4 -0
  103. package/lib/directives/infinite-scroll.directive.d.ts +30 -0
  104. package/lib/directives/select-article-on-click.directive.d.ts +14 -0
  105. package/lib/directives/show-bookmark.directive.d.ts +52 -0
  106. package/lib/directives/theme-provider.directive.d.ts +20 -0
  107. package/lib/guards/auth.guard.d.ts +7 -0
  108. package/lib/guards/index.d.ts +2 -0
  109. package/lib/guards/initialization.guard.d.ts +20 -0
  110. package/lib/interceptors/audit.interceptor.d.ts +13 -0
  111. package/lib/interceptors/auth.interceptor.d.ts +14 -0
  112. package/lib/interceptors/body.interceptor.d.ts +11 -0
  113. package/lib/interceptors/error.interceptor.d.ts +9 -0
  114. package/lib/interceptors/index.d.ts +5 -0
  115. package/lib/interceptors/toast.interceptor.d.ts +13 -0
  116. package/lib/models/aggregation.d.ts +12 -0
  117. package/lib/models/article-metadata.d.ts +5 -0
  118. package/lib/models/autocomplete.d.ts +5 -0
  119. package/lib/models/custom-json.d.ts +58 -0
  120. package/lib/models/filter-dropdown.d.ts +10 -0
  121. package/lib/models/index.d.ts +6 -0
  122. package/lib/models/user-settings.d.ts +32 -0
  123. package/lib/pipes/highlight-word.pipe.d.ts +22 -0
  124. package/lib/pipes/index.d.ts +2 -0
  125. package/lib/pipes/source-icon.pipe.d.ts +54 -0
  126. package/lib/providers/eager-provider.d.ts +11 -0
  127. package/lib/providers/index.d.ts +1 -0
  128. package/lib/public-api.d.ts +15 -0
  129. package/lib/resolvers/index.d.ts +1 -0
  130. package/lib/resolvers/query-name-resolver.d.ts +9 -0
  131. package/lib/resources/index.d.ts +1 -0
  132. package/lib/resources/themes.d.ts +51 -0
  133. package/lib/services/application.service.d.ts +178 -0
  134. package/lib/services/autocomplete.service.d.ts +91 -0
  135. package/lib/services/drawer/backdrop.service.d.ts +9 -0
  136. package/lib/services/drawer/drawer-stack.service.d.ts +70 -0
  137. package/lib/services/drawer/drawer.service.d.ts +15 -0
  138. package/lib/services/index.d.ts +11 -0
  139. package/lib/services/label.service.d.ts +117 -0
  140. package/lib/services/navigation.service.d.ts +33 -0
  141. package/lib/services/saved-searches.service.d.ts +145 -0
  142. package/lib/services/search.service.d.ts +155 -0
  143. package/lib/services/selection-history.service.d.ts +50 -0
  144. package/lib/services/selection.service.d.ts +127 -0
  145. package/lib/stores/aggregations.store.d.ts +50 -0
  146. package/lib/stores/app.store.d.ts +208 -0
  147. package/lib/stores/application.store.d.ts +106 -0
  148. package/lib/stores/index.d.ts +8 -0
  149. package/lib/stores/principal.store.d.ts +53 -0
  150. package/lib/stores/query-params.store.d.ts +187 -0
  151. package/lib/stores/selection.store.d.ts +62 -0
  152. package/lib/stores/theme.store.d.ts +55 -0
  153. package/lib/stores/user-settings.store.d.ts +161 -0
  154. package/lib/tokens/highlights.d.ts +8 -0
  155. package/lib/tokens/index.d.ts +1 -0
  156. package/lib/utils/debounced-signal.d.ts +25 -0
  157. package/lib/utils/index.d.ts +7 -0
  158. package/lib/utils/inline-worker.d.ts +11 -0
  159. package/lib/utils/query.d.ts +26 -0
  160. package/lib/utils/routes.d.ts +16 -0
  161. package/lib/utils/tailwind-utils.d.ts +2 -0
  162. package/lib/utils/theme-body-hook.d.ts +6 -0
  163. package/lib/utils/theme-registry.d.ts +3 -0
  164. package/lib/web-services/aggregations.service.d.ts +60 -0
  165. package/lib/web-services/app.service.d.ts +30 -0
  166. package/lib/web-services/audit.service.d.ts +75 -0
  167. package/lib/web-services/index.d.ts +9 -0
  168. package/lib/web-services/json-method-plugin.service.d.ts +41 -0
  169. package/lib/web-services/preview.service.d.ts +295 -0
  170. package/lib/web-services/principal.service.d.ts +28 -0
  171. package/lib/web-services/query.service.d.ts +29 -0
  172. package/lib/web-services/text-chunck.service.d.ts +22 -0
  173. package/package.json +28 -0
  174. package/public-api.d.ts +1 -0
@@ -0,0 +1,245 @@
1
+ import { computed, inject, Injectable, InjectionToken } from '@angular/core';
2
+ import { ActivatedRoute, Router } from '@angular/router';
3
+ import { getState } from '@ngrx/signals';
4
+ import { toast } from 'ngx-sonner';
5
+ import { globalConfig, labels, login, setGlobalConfig } from '@sinequa/atomic';
6
+ import { AuthGuard, InitializationGuard } from '../guards';
7
+ import { AppStore } from '../stores/app.store';
8
+ import { PrincipalStore } from '../stores/principal.store';
9
+ import { UserSettingsStore } from '../stores/user-settings.store';
10
+ import { AppService } from '../web-services/app.service';
11
+ import { AuditService } from '../web-services/audit.service';
12
+ import { ApplicationStore } from '../stores/application.store';
13
+ import * as i0 from "@angular/core";
14
+ export const ROUTE_COMPONENTS = new InjectionToken('ROUTE_COMPONENTS', { factory: () => [] });
15
+ export class ApplicationService {
16
+ constructor() {
17
+ this.userSettingsStore = inject(UserSettingsStore);
18
+ this.principalStore = inject(PrincipalStore);
19
+ this.appService = inject(AppService);
20
+ this.appStore = inject(AppStore);
21
+ this.applicationStore = inject(ApplicationStore);
22
+ this.route = inject(ActivatedRoute);
23
+ this.router = inject(Router);
24
+ this.auditService = inject(AuditService);
25
+ this.components = inject(ROUTE_COMPONENTS);
26
+ this.defaultComponent = computed(() => this.components.find(c => c.path === 'all')?.component);
27
+ this.defaultLayoutComponent = computed(() => this.components.find(c => c.isRoot)?.component);
28
+ }
29
+ /**
30
+ * Registers a list of components.
31
+ *
32
+ * * For each path, the corresponding component is registered.
33
+ * * The default component is the component registered with the path 'all'.
34
+ * * The default layout component is the component registered with the isLayout flag set to true.
35
+ * * If no layout component is registered, the default layout component is the default component.
36
+ * * If no default component is registered, the default component is the first component in the list.
37
+ *
38
+ * Those components will be used to create the routes.
39
+ *
40
+ * @deprecated use the ROUTE_COMPONENTS injection token instead
41
+ *
42
+ * @param components - An array of ComponentMapping objects to be registered.
43
+ */
44
+ register(components) {
45
+ this.components = [...components];
46
+ }
47
+ /**
48
+ * Authenticates the user with the provided credentials, initializes and creates routes.  
49
+ * * in case of failure, it rejects with an error message.
50
+ * * in case of success, it resolves with `true`, shows a toast notification to welcome the user,
51
+ * and updates the application state to ready.
52
+ *
53
+ * @param {Credentials} [credentials] - The user's login credentials.
54
+ * @returns {Promise<boolean>} - A promise that resolves to `true` if login is successful,
55
+ * or rejects with an error message if login fails.
56
+ * @throws {Error} - Throws an error if the login process fails.
57
+ */
58
+ async autoLogin({ credentials, withCreateRoutes = true } = {}) {
59
+ const { useCredentialsOrSSO } = globalConfig;
60
+ try {
61
+ const authenticated = await login(credentials);
62
+ if (authenticated) {
63
+ await this.initAndCreateRoutes(withCreateRoutes).catch((error) => {
64
+ console.warn('Error initializing the application (authenticated)', error);
65
+ return Promise.reject(error);
66
+ });
67
+ // Show a toast notification to welcome the user
68
+ // and the application is ready to use
69
+ const { fullName, name } = getState(this.principalStore).principal;
70
+ toast(`Welcome back ${fullName || name}!`, { duration: 2000 });
71
+ this.applicationStore.updateReadyState();
72
+ return Promise.resolve(true);
73
+ }
74
+ else {
75
+ return Promise.reject('Login failed');
76
+ }
77
+ }
78
+ catch (error) {
79
+ console.error('Error logging in', error);
80
+ if (useCredentialsOrSSO) {
81
+ await this.initAndCreateRoutes(withCreateRoutes).catch((error) => {
82
+ console.warn('Error initializing the application (useCredentialsOrSSO)', error);
83
+ setGlobalConfig({ useCredentials: true });
84
+ return Promise.reject(error);
85
+ });
86
+ setGlobalConfig({ useSSO: true });
87
+ // Show a toast notification to welcome the user
88
+ // and the application is ready to use
89
+ const { fullName, name } = getState(this.principalStore).principal;
90
+ toast(`Welcome back ${fullName || name}!`, { duration: 2000 });
91
+ this.applicationStore.updateReadyState();
92
+ return Promise.resolve(true);
93
+ }
94
+ else {
95
+ return Promise.reject(error);
96
+ }
97
+ }
98
+ }
99
+ /**
100
+ * Initializes the application.
101
+ * - Fetches the application configuration.
102
+ * - Sets the fetched application configuration in the app store.
103
+ * - Loads the user settings and logs the state of the user settings store.
104
+ */
105
+ async init() {
106
+ // Fetch the application configuration
107
+ await this.appStore.initialize();
108
+ console.log("appStore", getState(this.appStore));
109
+ // Load the principal (user information)
110
+ await this.principalStore.initialize();
111
+ console.log("principalStore", getState(this.principalStore));
112
+ // Load the user settings
113
+ await this.userSettingsStore.initialize();
114
+ console.log("userSettingsStore", getState(this.userSettingsStore));
115
+ // Labels access
116
+ const service = (this.appStore.getWebServiceByType('labels'));
117
+ if (!service) {
118
+ this.applicationStore.updateHasLabelsAccess(false);
119
+ return;
120
+ }
121
+ try {
122
+ const rights = await labels.getUserRights();
123
+ this.applicationStore.updateHasLabelsAccess(rights.canEditPublicLabels || rights.canManagePublicLabels);
124
+ }
125
+ catch (error) {
126
+ console.log("labels.canHandleLabels failure - error: ", error);
127
+ this.applicationStore.updateHasLabelsAccess(false);
128
+ }
129
+ }
130
+ /**
131
+ * Initializes the application and creates routes.
132
+ *
133
+ * This method performs the following actions:
134
+ * 1. Calls the `init` method to initialize the application.
135
+ * 2. Calls the `createRoutes` method to set up the application routes.
136
+ * 3. Notifies the audit service of a login event.
137
+ *
138
+ * @returns {Promise<void>} A promise that resolves when the initialization and route creation are complete.
139
+ */
140
+ async initAndCreateRoutes(withCreateRoutes = true) {
141
+ // throw an error if no components are registered and withCreateRoutes is true
142
+ // components are required to create the routes
143
+ if (withCreateRoutes && !this.components.length)
144
+ throw new Error('No components registered');
145
+ await this.init();
146
+ if (withCreateRoutes) {
147
+ this.createRoutes();
148
+ }
149
+ }
150
+ /**
151
+ * Creates dynamic routes based on the application's queries and custom JSON configurations.
152
+ *
153
+ * This method performs the following steps:
154
+ * 1. Retrieves the queries and custom JSON data from the application state.
155
+ * 2. Maps the queries to an array of objects containing query names and tabs.
156
+ * 3. Throws an error if no queries are found.
157
+ * 4. Retrieves route data from custom JSONs or falls back to default data.
158
+ * 5. Creates routes for each tab in each query, or uses the query name if no tabs are found.
159
+ * 6. Removes the current search route from the router configuration.
160
+ * 7. Creates child routes based on the provided routes data or the first query's tabs.
161
+ * 8. Updates the search route with the new child routes.
162
+ * 9. Resets the router configuration with the new routes.
163
+ *
164
+ * @throws {Error} If no queries are found.
165
+ */
166
+ createRoutes() {
167
+ // Now we can create the dynamic routes based on the queries's tabs
168
+ const { queries, data: custom, customJSONs = [] } = getState(this.appStore);
169
+ // contains an array of objects with the query name and the tabs
170
+ const queriesMap = Object.entries(queries).map(([key, value]) => ({ key, ...value }));
171
+ // ! if no queries are found, throw an error
172
+ if (!queriesMap.length) {
173
+ throw new Error('No queries found');
174
+ }
175
+ // Retrieves the routes data from the custom JSONs array or fall back to the default data (custom json).
176
+ let { data: cjRoutes } = Array.isArray(customJSONs) ? customJSONs?.find((c => c.name === 'routes')) || {} : { data: custom?.['routes'] };
177
+ // check if cjRoutes is an array and not an object to avoid errors
178
+ if (cjRoutes && !Array.isArray(cjRoutes)) {
179
+ cjRoutes = undefined;
180
+ }
181
+ const routes = (cjRoutes || custom?.['routes']);
182
+ // take only first query
183
+ const firstQuery = queriesMap[0];
184
+ // We need to create a route for each tab in each query
185
+ // if a query has no tabs, we create a route with the query name as the tab name
186
+ const firstQueryConfig = { tabs: firstQuery.tabSearch.tabs || [{ name: firstQuery.name }] };
187
+ // We need to remove the current search route from the router config
188
+ // the route exists in the router config because it was created in the app-routing.module.ts and we need it
189
+ // to be able to navigate to the search page. We will recreate it with the new tabs
190
+ const currentConfig = this.router.config.filter(route => route.path !== 'search');
191
+ let children = [];
192
+ // if the routes data is provided, we create the children routes based on the routes data
193
+ if (routes) {
194
+ // update routes pathDisplayName with the query display name when not provided
195
+ // for each path we need to find the corresponding query and tab in the firstQuery object
196
+ // and create a child route with the query name and tab name
197
+ const displayNamesMap = firstQuery.tabSearch.tabs.reduce((acc, tab) => {
198
+ acc[tab.name] = { display: tab.display };
199
+ return acc;
200
+ }, {});
201
+ children = routes.map((route) => {
202
+ return ({
203
+ path: route.path,
204
+ component: this.components.find(c => c.path === route.path)?.component || this.defaultComponent(),
205
+ data: {
206
+ queryName: route.wsName || firstQuery.name,
207
+ display: route.pathDisplayName || displayNamesMap[route.wsQueryTab]?.display || route.path,
208
+ wsQueryTab: route.wsQueryTab,
209
+ iconClass: route.icon
210
+ }
211
+ });
212
+ });
213
+ }
214
+ else {
215
+ // if the routes data is not provided, we create the children routes based on the first query's tabs
216
+ // create the children routes for the search route
217
+ children = firstQueryConfig.tabs.map(tab => ({
218
+ path: tab.name,
219
+ component: this.components.find(c => c.path === tab.name)?.component || this.defaultComponent(),
220
+ data: { queryName: firstQuery.name, display: tab.display || tab.name }
221
+ }));
222
+ }
223
+ const searchPath = this.router.config.find(route => route.path === 'search')
224
+ || {
225
+ path: 'search',
226
+ component: this.components.find(c => c.path === 'search')?.component || this.defaultLayoutComponent(),
227
+ canActivate: [AuthGuard(), InitializationGuard()],
228
+ children: []
229
+ };
230
+ searchPath.component = this.components.find(c => c.path === 'search')?.component || this.defaultLayoutComponent();
231
+ searchPath.children = [...children, { path: '**', redirectTo: 'all', pathMatch: 'full' }];
232
+ const newConfig = [searchPath, ...currentConfig];
233
+ // finally we reset the router config with the new routes
234
+ this.router.resetConfig(newConfig);
235
+ }
236
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: ApplicationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
237
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: ApplicationService, providedIn: 'root' }); }
238
+ }
239
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: ApplicationService, decorators: [{
240
+ type: Injectable,
241
+ args: [{
242
+ providedIn: 'root'
243
+ }]
244
+ }] });
245
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"application.service.js","sourceRoot":"","sources":["../../../../../projects/atomic-angular/src/lib/services/application.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { computed, inject, Injectable, InjectionToken, signal, Type } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { getState } from '@ngrx/signals';\nimport { toast } from 'ngx-sonner';\n\nimport { CCApp, Credentials, globalConfig, labels, login, setGlobalConfig } from '@sinequa/atomic';\n\nimport { AuthGuard, InitializationGuard } from '../guards';\nimport { AppStore } from '../stores/app.store';\nimport { PrincipalStore } from '../stores/principal.store';\nimport { UserSettingsStore } from '../stores/user-settings.store';\nimport { AppService } from '../web-services/app.service';\nimport { AuditService } from '../web-services/audit.service';\nimport { ApplicationStore } from '../stores/application.store';\n\nexport const ROUTE_COMPONENTS = new InjectionToken<ComponentMapping[]>('ROUTE_COMPONENTS', { factory: () => [] });\n\nexport type ComponentMapping = {\n  path: string,\n  component: Type<unknown>\n  isRoot?: boolean\n};\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ApplicationService {\n\n  userSettingsStore = inject(UserSettingsStore);\n  principalStore = inject(PrincipalStore);\n\n  appService = inject(AppService);\n  appStore = inject(AppStore);\n  applicationStore = inject(ApplicationStore);\n\n  route = inject(ActivatedRoute);\n  router = inject(Router);\n\n  auditService = inject(AuditService);\n\n  components = inject(ROUTE_COMPONENTS);\n  defaultComponent = computed(() => this.components.find(c => c.path === 'all')?.component);\n  defaultLayoutComponent = computed(() => this.components.find(c => c.isRoot)?.component);\n\n  /**\n   * Registers a list of components.\n   *\n   * * For each path, the corresponding component is registered.\n   * * The default component is the component registered with the path 'all'.\n   * * The default layout component is the component registered with the isLayout flag set to true.\n   * * If no layout component is registered, the default layout component is the default component.\n   * * If no default component is registered, the default component is the first component in the list.\n   *\n   * Those components will be used to create the routes.\n   *\n   * @deprecated use the ROUTE_COMPONENTS injection token instead\n   *\n   * @param components - An array of ComponentMapping objects to be registered.\n   */\n  register(components: ComponentMapping[]) {\n    this.components = [...components];\n  }\n\n  /**\n   * Authenticates the user with the provided credentials, initializes and creates routes.&nbsp;&nbsp;\n   * * in case of failure, it rejects with an error message.\n   * * in case of success, it resolves with `true`, shows a toast notification to welcome the user,\n   * and updates the application state to ready.\n   *\n   * @param {Credentials} [credentials] - The user's login credentials.\n   * @returns {Promise<boolean>} - A promise that resolves to `true` if login is successful,\n   *                               or rejects with an error message if login fails.\n   * @throws {Error} - Throws an error if the login process fails.\n   */\n  async autoLogin({ credentials, withCreateRoutes = true }: { credentials?: Credentials; withCreateRoutes?: boolean; } = {}): Promise<boolean> {\n    const { useCredentialsOrSSO } = globalConfig;\n    try {\n      const authenticated = await login(credentials);\n      if (authenticated) {\n        await this.initAndCreateRoutes(withCreateRoutes).catch((error) => {\n          console.warn('Error initializing the application (authenticated)', error);\n          return Promise.reject(error);\n        });\n\n        // Show a toast notification to welcome the user\n        // and the application is ready to use\n        const { fullName, name } = getState(this.principalStore).principal;\n        toast(`Welcome back ${fullName || name}!`, { duration: 2000 })\n        this.applicationStore.updateReadyState();\n\n        return Promise.resolve(true);\n      }\n      else {\n        return Promise.reject('Login failed');\n      }\n    }\n    catch (error) {\n      console.error('Error logging in', error);\n      if (useCredentialsOrSSO) {\n        await this.initAndCreateRoutes(withCreateRoutes).catch((error) => {\n          console.warn('Error initializing the application (useCredentialsOrSSO)', error);\n          setGlobalConfig({ useCredentials: true });\n          return Promise.reject(error);\n        })\n\n        setGlobalConfig({ useSSO: true });\n        // Show a toast notification to welcome the user\n        // and the application is ready to use\n        const { fullName, name } = getState(this.principalStore).principal;\n        toast(`Welcome back ${fullName || name}!`, { duration: 2000 })\n        this.applicationStore.updateReadyState();\n\n        return Promise.resolve(true);\n      }\n      else {\n        return Promise.reject(error);\n      }\n    }\n  }\n\n  /**\n   * Initializes the application.\n   * - Fetches the application configuration.\n   * - Sets the fetched application configuration in the app store.\n   * - Loads the user settings and logs the state of the user settings store.\n   */\n  async init() {\n    // Fetch the application configuration\n    await this.appStore.initialize();\n    console.log(\"appStore\", getState(this.appStore));\n\n    // Load the principal (user information)\n    await this.principalStore.initialize()\n    console.log(\"principalStore\", getState(this.principalStore))\n\n    // Load the user settings\n    await this.userSettingsStore.initialize();\n    console.log(\"userSettingsStore\", getState(this.userSettingsStore))\n\n    // Labels access\n    const service = (this.appStore.getWebServiceByType('labels'));\n    if (!service) {\n      this.applicationStore.updateHasLabelsAccess(false);\n      return;\n    }\n    try {\n      const rights = await labels.getUserRights();\n      this.applicationStore.updateHasLabelsAccess(rights.canEditPublicLabels || rights.canManagePublicLabels);\n    } catch (error) {\n      console.log(\"labels.canHandleLabels failure - error: \", error);\n      this.applicationStore.updateHasLabelsAccess(false);\n    }\n  }\n\n\n  /**\n   * Initializes the application and creates routes.\n   *\n   * This method performs the following actions:\n   * 1. Calls the `init` method to initialize the application.\n   * 2. Calls the `createRoutes` method to set up the application routes.\n   * 3. Notifies the audit service of a login event.\n   *\n   * @returns {Promise<void>} A promise that resolves when the initialization and route creation are complete.\n   */\n  async initAndCreateRoutes(withCreateRoutes = true) {\n    // throw an error if no components are registered and withCreateRoutes is true\n    // components are required to create the routes\n    if (withCreateRoutes && !this.components.length) throw new Error('No components registered');\n\n    await this.init();\n    if (withCreateRoutes) {\n      this.createRoutes();\n    }\n  }\n\n  /**\n   * Creates dynamic routes based on the application's queries and custom JSON configurations.\n   *\n   * This method performs the following steps:\n   * 1. Retrieves the queries and custom JSON data from the application state.\n   * 2. Maps the queries to an array of objects containing query names and tabs.\n   * 3. Throws an error if no queries are found.\n   * 4. Retrieves route data from custom JSONs or falls back to default data.\n   * 5. Creates routes for each tab in each query, or uses the query name if no tabs are found.\n   * 6. Removes the current search route from the router configuration.\n   * 7. Creates child routes based on the provided routes data or the first query's tabs.\n   * 8. Updates the search route with the new child routes.\n   * 9. Resets the router configuration with the new routes.\n   *\n   * @throws {Error} If no queries are found.\n   */\n  private createRoutes() {\n    // Now we can create the dynamic routes based on the queries's tabs\n    const { queries, data: custom, customJSONs = [] } = getState(this.appStore) as CCApp;\n\n    // contains an array of objects with the query name and the tabs\n    const queriesMap = Object.entries(queries).map(([key, value]) => ({ key, ...value }));\n\n    // ! if no queries are found, throw an error\n    if (!queriesMap.length) {\n      throw new Error('No queries found');\n    }\n\n    // Retrieves the routes data from the custom JSONs array or fall back to the default data (custom json).\n    let { data: cjRoutes } = Array.isArray(customJSONs) ? customJSONs?.find((c => c.name === 'routes')) || {} : { data: custom?.['routes'] };\n    // check if cjRoutes is an array and not an object to avoid errors\n    if (cjRoutes && !Array.isArray(cjRoutes)) {\n      cjRoutes = undefined;\n    }\n    const routes = (cjRoutes || custom?.['routes']) as { path: string, wsName: string, wsQueryTab: string, pathDisplayName: string, icon: string }[];\n\n    // take only first query\n    const firstQuery = queriesMap[0];\n\n    // We need to create a route for each tab in each query\n    // if a query has no tabs, we create a route with the query name as the tab name\n    const firstQueryConfig = { tabs: firstQuery.tabSearch.tabs || [{ name: firstQuery.name }] };\n\n    // We need to remove the current search route from the router config\n    // the route exists in the router config because it was created in the app-routing.module.ts and we need it\n    // to be able to navigate to the search page. We will recreate it with the new tabs\n    const currentConfig = this.router.config.filter(route => route.path !== 'search');\n\n    let children = [];\n\n    // if the routes data is provided, we create the children routes based on the routes data\n    if (routes) {\n      // update routes pathDisplayName with the query display name when not provided\n      // for each path we need to find the corresponding query and tab in the firstQuery object\n      // and create a child route with the query name and tab name\n      const displayNamesMap = firstQuery.tabSearch.tabs.reduce((acc, tab) => {\n        acc[tab.name] = { display: tab.display };\n        return acc;\n      }, {} as Record<string, { display: string }>);\n\n      children = routes.map((route) => {\n        return ({\n          path: route.path,\n          component: this.components.find(c => c.path === route.path)?.component || this.defaultComponent(),\n          data: {\n            queryName: route.wsName || firstQuery.name,\n            display: route.pathDisplayName || displayNamesMap[route.wsQueryTab]?.display || route.path,\n            wsQueryTab: route.wsQueryTab,\n            iconClass: route.icon\n          }\n        })\n      });\n    }\n    else {\n      // if the routes data is not provided, we create the children routes based on the first query's tabs\n      // create the children routes for the search route\n      children = firstQueryConfig.tabs.map(tab => ({\n        path: tab.name,\n        component: this.components.find(c => c.path === tab.name)?.component || this.defaultComponent(),\n        data: { queryName: firstQuery.name, display: tab.display || tab.name }\n      }));\n    }\n\n    const searchPath = this.router.config.find(route => route.path === 'search')\n      || {\n      path: 'search',\n      component: this.components.find(c => c.path === 'search')?.component || this.defaultLayoutComponent(),\n      canActivate: [AuthGuard(), InitializationGuard()],\n      children: []\n    };\n\n    searchPath.component = this.components.find(c => c.path === 'search')?.component || this.defaultLayoutComponent();\n    searchPath.children = [...children, { path: '**', redirectTo: 'all', pathMatch: 'full' }];\n\n    const newConfig = [searchPath, ...currentConfig];\n    // finally we reset the router config with the new routes\n    this.router.resetConfig(newConfig);\n  }\n}"]}
@@ -0,0 +1,85 @@
1
+ import { Injectable, Injector, inject, signal } from '@angular/core';
2
+ import { forkJoin, from, of } from 'rxjs';
3
+ import { getState } from '@ngrx/signals';
4
+ import { fetchSuggest } from '@sinequa/atomic';
5
+ import { UserSettingsStore } from '../stores/user-settings.store';
6
+ import { AppStore } from '../stores/app.store';
7
+ import * as i0 from "@angular/core";
8
+ export class AutocompleteService {
9
+ constructor() {
10
+ this.opened = signal(false);
11
+ this.injector = inject(Injector);
12
+ this.userSettingsStore = inject(UserSettingsStore);
13
+ this.appStore = inject(AppStore);
14
+ }
15
+ /**
16
+ * Retrieves autocomplete items for the given text, max count for each
17
+ * category handled by the service can be specified in the admin
18
+ *
19
+ * @param text Text to retrieve autocomplete items for
20
+ * @returns An observable of an array of {@link Suggestion} arrays grouped by
21
+ * suggestion queries configured in the admin
22
+ */
23
+ getFromSuggestQueriesForText(text) {
24
+ // Do not ask for autocomplete items if the text is empty
25
+ if (!text)
26
+ return of([]);
27
+ const queries = this.appStore.getWebServiceByType('autocomplete')?.suggestQueries?.split(',') ?? [];
28
+ const obss = queries.reduce((acc, curr) => {
29
+ acc.push(from(fetchSuggest(curr, text)));
30
+ return acc;
31
+ }, []);
32
+ return forkJoin(obss);
33
+ }
34
+ /**
35
+ * Retrieves autocomplete items for the given text from the user settings
36
+ *
37
+ * @param text Text to retrieve autocomplete items for
38
+ * @param maxCount Maximum number of items to retrieve
39
+ * @returns An observable of an array of {@link Suggestion} arrays grouped by
40
+ * `recent-searches`, `saved-searches`, `bookmarks` from the user settings
41
+ */
42
+ getFromUserSettingsForText(text, maxCount) {
43
+ const { bookmarks, recentSearches, savedSearches } = getState(this.userSettingsStore);
44
+ const items = [];
45
+ if (typeof maxCount === 'number')
46
+ maxCount = { recentSearches: maxCount, savedSearches: maxCount, bookmarks: maxCount };
47
+ if (recentSearches) {
48
+ // don't filter if the text is empty
49
+ const matchingRecentSearches = text
50
+ ? recentSearches.filter(recentSearch => recentSearch.display?.toLocaleLowerCase().includes(text.toLocaleLowerCase()))
51
+ : recentSearches;
52
+ const searches = matchingRecentSearches.slice(0, maxCount?.recentSearches);
53
+ if (searches.length > 0)
54
+ items.push(...searches.map(search => ({ category: 'recent-search', ...search })));
55
+ }
56
+ if (savedSearches) {
57
+ // don't filter if the text is empty
58
+ const matchingSavedSearches = text
59
+ ? savedSearches.filter(savedSearch => savedSearch.display?.toLocaleLowerCase().includes(text.toLocaleLowerCase()))
60
+ : savedSearches;
61
+ const searches = matchingSavedSearches.slice(0, maxCount?.savedSearches);
62
+ if (searches.length > 0)
63
+ items.push(...searches.map(search => ({ category: 'saved-search', ...search })));
64
+ }
65
+ if (bookmarks) {
66
+ // don't filter if the text is empty
67
+ const matchingBookmarks = text
68
+ ? bookmarks.filter(bookmark => bookmark.label?.toLowerCase().includes(text.toLowerCase()))
69
+ : bookmarks;
70
+ const searches = matchingBookmarks.slice(0, maxCount?.bookmarks);
71
+ if (searches.length > 0)
72
+ items.push(...searches.map(search => ({ category: 'bookmark', display: search.label ?? '', ...search })));
73
+ }
74
+ return items;
75
+ }
76
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: AutocompleteService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
77
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: AutocompleteService, providedIn: 'root' }); }
78
+ }
79
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: AutocompleteService, decorators: [{
80
+ type: Injectable,
81
+ args: [{
82
+ providedIn: 'root'
83
+ }]
84
+ }] });
85
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete.service.js","sourceRoot":"","sources":["../../../../../projects/atomic-angular/src/lib/services/autocomplete.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Injector, inject, signal } from '@angular/core';\nimport { Observable, forkJoin, from, of } from 'rxjs';\nimport { getState } from '@ngrx/signals';\n\nimport { CCWebService, Suggestion, fetchSuggest } from '@sinequa/atomic';\n\nimport { UserSettingsStore } from '../stores/user-settings.store';\nimport { AppStore } from '../stores/app.store';\nimport { Autocomplete } from '../models/autocomplete';\n\n\ntype AutocompleteWebService = CCWebService & {\n  allowedWithAnySBA?: boolean,\n  revision?: number,\n  suggestQueries?: string,\n  useDistributionRegex?: boolean,\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class AutocompleteService {\n  readonly opened = signal(false);\n\n  readonly injector = inject(Injector)\n\n  userSettingsStore = inject(UserSettingsStore);\n  appStore = inject(AppStore);\n\n  /**\n   * Retrieves autocomplete items for the given text, max count for each\n   * category handled by the service can be specified in the admin\n   *\n   * @param text Text to retrieve autocomplete items for\n   * @returns An observable of an array of {@link Suggestion} arrays grouped by\n   * suggestion queries configured in the admin\n   */\n  getFromSuggestQueriesForText(text: string): Observable<Suggestion[][]> {\n    // Do not ask for autocomplete items if the text is empty\n    if (!text) return of([]);\n\n    const queries = (this.appStore.getWebServiceByType('autocomplete') as AutocompleteWebService)?.suggestQueries?.split(',') ?? [];\n    const obss = queries.reduce((acc: Observable<Suggestion[]>[], curr: string) => {\n      acc.push(from(fetchSuggest(curr, text)));\n      return acc;\n    }, [] as Observable<Suggestion[]>[]);\n\n    return forkJoin<Suggestion[][]>(obss);\n  }\n\n  /**\n   * Retrieves autocomplete items for the given text from the user settings\n   *\n   * @param text Text to retrieve autocomplete items for\n   * @param maxCount Maximum number of items to retrieve\n   * @returns An observable of an array of {@link Suggestion} arrays grouped by\n   * `recent-searches`, `saved-searches`, `bookmarks` from the user settings\n   */\n  getFromUserSettingsForText(text: string, maxCount?: number | Autocomplete): Suggestion[] {\n    const { bookmarks, recentSearches, savedSearches } = getState(this.userSettingsStore);\n    const items: Suggestion[] = [];\n\n    if (typeof maxCount === 'number')\n      maxCount = { recentSearches: maxCount, savedSearches: maxCount, bookmarks: maxCount };\n\n    if (recentSearches) {\n\n      // don't filter if the text is empty\n      const matchingRecentSearches = text\n        ? recentSearches.filter(recentSearch => recentSearch.display?.toLocaleLowerCase().includes(text.toLocaleLowerCase()))\n        : recentSearches;\n\n      const searches = matchingRecentSearches.slice(0, maxCount?.recentSearches);\n\n      if (searches.length > 0)\n        items.push(...searches.map(search => ({ category: 'recent-search', ...search })));\n    }\n\n    if (savedSearches) {\n\n      // don't filter if the text is empty\n      const matchingSavedSearches = text\n        ? savedSearches.filter(savedSearch => savedSearch.display?.toLocaleLowerCase().includes(text.toLocaleLowerCase()))\n        : savedSearches;\n\n      const searches = matchingSavedSearches.slice(0, maxCount?.savedSearches);\n\n      if (searches.length > 0)\n        items.push(...searches.map(search => ({ category: 'saved-search', ...search })));\n    }\n\n    if (bookmarks) {\n\n      // don't filter if the text is empty\n      const matchingBookmarks = text\n        ? bookmarks.filter(bookmark => bookmark.label?.toLowerCase().includes(text.toLowerCase()))\n        : bookmarks;\n\n      const searches = matchingBookmarks.slice(0, maxCount?.bookmarks);\n\n      if (searches.length > 0)\n        items.push(...searches.map(search => ({ category: 'bookmark', display: search.label ?? '', ...search })));\n    }\n\n    return items;\n  }\n}\n"]}
@@ -0,0 +1,23 @@
1
+ import { Injectable } from '@angular/core';
2
+ import { BehaviorSubject } from 'rxjs';
3
+ import * as i0 from "@angular/core";
4
+ export class BackdropService {
5
+ constructor() {
6
+ this.isVisible = new BehaviorSubject(false);
7
+ }
8
+ show() {
9
+ this.isVisible.next(true);
10
+ }
11
+ hide() {
12
+ this.isVisible.next(false);
13
+ }
14
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: BackdropService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
15
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: BackdropService, providedIn: 'root' }); }
16
+ }
17
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: BackdropService, decorators: [{
18
+ type: Injectable,
19
+ args: [{
20
+ providedIn: 'root'
21
+ }]
22
+ }] });
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFja2Ryb3Auc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2F0b21pYy1hbmd1bGFyL3NyYy9saWIvc2VydmljZXMvZHJhd2VyL2JhY2tkcm9wLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzQyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sTUFBTSxDQUFDOztBQUt2QyxNQUFNLE9BQU8sZUFBZTtJQUg1QjtRQUlTLGNBQVMsR0FBRyxJQUFJLGVBQWUsQ0FBVSxLQUFLLENBQUMsQ0FBQztLQVN4RDtJQVBRLElBQUk7UUFDVCxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUM1QixDQUFDO0lBRU0sSUFBSTtRQUNULElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzdCLENBQUM7OEdBVFUsZUFBZTtrSEFBZixlQUFlLGNBRmQsTUFBTTs7MkZBRVAsZUFBZTtrQkFIM0IsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBCZWhhdmlvclN1YmplY3QgfSBmcm9tICdyeGpzJztcblxuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgQmFja2Ryb3BTZXJ2aWNlIHtcbiAgcHVibGljIGlzVmlzaWJsZSA9IG5ldyBCZWhhdmlvclN1YmplY3Q8Ym9vbGVhbj4oZmFsc2UpO1xuXG4gIHB1YmxpYyBzaG93KCk6IHZvaWQge1xuICAgIHRoaXMuaXNWaXNpYmxlLm5leHQodHJ1ZSk7XG4gIH1cblxuICBwdWJsaWMgaGlkZSgpOiB2b2lkIHtcbiAgICB0aGlzLmlzVmlzaWJsZS5uZXh0KGZhbHNlKTtcbiAgfVxufVxuIl19
@@ -0,0 +1,152 @@
1
+ import { EventEmitter, Injectable, inject } from '@angular/core';
2
+ import { BehaviorSubject, Subscription } from 'rxjs';
3
+ import { getState } from '@ngrx/signals';
4
+ import { SelectionHistoryService } from '../selection-history.service';
5
+ import { SelectionService } from '../selection.service';
6
+ import { NavigationService } from '../navigation.service';
7
+ import { SelectionStore } from '../../stores/selection.store';
8
+ import { BackdropService } from './backdrop.service';
9
+ import * as i0 from "@angular/core";
10
+ export class DrawerStackService {
11
+ constructor() {
12
+ this.isOpened = new BehaviorSubject(false);
13
+ this.toggleTopDrawerExtension$ = new EventEmitter();
14
+ this.forceTopDrawerCollapse$ = new EventEmitter();
15
+ this.closeTopDrawer$ = new EventEmitter();
16
+ this.closeAllDrawers$ = new EventEmitter();
17
+ this.isChatOpened = new BehaviorSubject(false);
18
+ this.openChatDrawer$ = new EventEmitter();
19
+ this.closeChatDrawer$ = new EventEmitter();
20
+ this.askAI$ = new EventEmitter();
21
+ this.selection = inject(SelectionService);
22
+ this.selectionHistory = inject(SelectionHistoryService);
23
+ this.selectionStore = inject(SelectionStore);
24
+ this.navigationService = inject(NavigationService);
25
+ this.backdropService = inject(BackdropService);
26
+ this.subscriptions = new Subscription();
27
+ // on new search, close all drawers including chat drawer
28
+ this.subscriptions.add(this.navigationService.navigationEnd$.subscribe(() => {
29
+ this.closeAssistant();
30
+ this.closeAll();
31
+ }));
32
+ }
33
+ ngOnDestroy() {
34
+ this.subscriptions.unsubscribe();
35
+ }
36
+ /**
37
+ * Sets current drawer stack status to open
38
+ */
39
+ open(componentType) {
40
+ this.componentType = componentType;
41
+ this.isOpened.next(true);
42
+ }
43
+ /**
44
+ * Emits event to extend the top drawer
45
+ */
46
+ extend() {
47
+ this.toggleTopDrawerExtension$.next();
48
+ }
49
+ /**
50
+ * Emits event to close the top drawer, checks if history is empty, if so,
51
+ * sets current drawer stack status to closed and clears history and current
52
+ * selection
53
+ */
54
+ close() {
55
+ this.closeTopDrawer$.next();
56
+ if (this.selectionHistory.getCurrentSelectionIndex() === -1) {
57
+ this.isOpened.next(false);
58
+ this.selectionHistory.clearHistory();
59
+ }
60
+ }
61
+ /**
62
+ * Emits event to close the top drawer
63
+ */
64
+ closeTop() {
65
+ this.closeTopDrawer$.next();
66
+ }
67
+ /**
68
+ * Sets current drawer stack status to closed, clears history and emits event
69
+ * to close all drawers
70
+ *
71
+ * @param keepDrawerOpen if true, do not trigger layout animation
72
+ */
73
+ closeAll(keepDrawerOpen = false) {
74
+ if (!keepDrawerOpen)
75
+ this.isOpened.next(false);
76
+ this.selectionHistory.clearHistory();
77
+ this.closeAllDrawers$.next();
78
+ }
79
+ /**
80
+ * Replace the current selection with the given article by closing all
81
+ * drawers and opening the drawer with the new selection without triggering
82
+ * layout animation
83
+ *
84
+ * @param article the article to replace the current selection with
85
+ */
86
+ replace(article) {
87
+ const { id } = getState(this.selectionStore);
88
+ if (id && (!article || article.id === id))
89
+ return;
90
+ // close everything without trigger layout animation
91
+ this.closeAll(true);
92
+ // set selection
93
+ this.selection.setCurrentArticle(article);
94
+ // open drawer
95
+ this.open();
96
+ }
97
+ /**
98
+ * Stack the given article by setting the current selection and opening the
99
+ * drawer
100
+ *
101
+ * @param article the article to stack
102
+ */
103
+ stack(article, withQueryText) {
104
+ const { id } = getState(this.selectionStore);
105
+ if (id && (!article || article.id === id))
106
+ return;
107
+ // force top drawer to collapse
108
+ this.forceTopDrawerCollapse$.next();
109
+ // set selection
110
+ this.selection.setCurrentArticle(article, withQueryText);
111
+ // open drawer
112
+ this.open();
113
+ }
114
+ toggleAssistant() {
115
+ if (this.isChatOpened.getValue()) {
116
+ this.backdropService.hide();
117
+ this.closeAssistant();
118
+ }
119
+ else {
120
+ this.backdropService.show();
121
+ this.openAssistant();
122
+ }
123
+ }
124
+ openAssistant() {
125
+ this.isChatOpened.next(true);
126
+ this.isOpened.next(true);
127
+ this.openChatDrawer$.next();
128
+ this.backdropService.show();
129
+ }
130
+ closeAssistant(keepDrawerOpen = false) {
131
+ this.isChatOpened.next(false);
132
+ if (!keepDrawerOpen && this.selectionHistory.getCurrentSelectionIndex() === -1) {
133
+ this.isOpened.next(false);
134
+ this.closeAllDrawers$.next();
135
+ }
136
+ this.closeChatDrawer$.next();
137
+ this.backdropService.hide();
138
+ }
139
+ askAI(text) {
140
+ this.openAssistant();
141
+ this.askAI$.next(text);
142
+ }
143
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: DrawerStackService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
144
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: DrawerStackService, providedIn: 'root' }); }
145
+ }
146
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: DrawerStackService, decorators: [{
147
+ type: Injectable,
148
+ args: [{
149
+ providedIn: 'root'
150
+ }]
151
+ }], ctorParameters: () => [] });
152
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drawer-stack.service.js","sourceRoot":"","sources":["../../../../../../projects/atomic-angular/src/lib/services/drawer/drawer-stack.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter, Injectable, OnDestroy, Type, inject } from '@angular/core';\nimport { BehaviorSubject, Subscription } from 'rxjs';\nimport { getState } from '@ngrx/signals';\n\nimport { Article } from '@sinequa/atomic';\n\nimport { SelectionHistoryService } from '../selection-history.service';\nimport { SelectionService } from '../selection.service';\nimport { NavigationService } from '../navigation.service';\nimport { SelectionStore } from '../../stores/selection.store';\nimport { BackdropService } from './backdrop.service';\n\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class DrawerStackService implements OnDestroy {\n  public readonly isOpened = new BehaviorSubject<boolean>(false);\n  public readonly toggleTopDrawerExtension$ = new EventEmitter<void>();\n  public readonly forceTopDrawerCollapse$ = new EventEmitter<void>();\n  public readonly closeTopDrawer$ = new EventEmitter<void>();\n  public readonly closeAllDrawers$ = new EventEmitter<void>();\n\n  public readonly isChatOpened = new BehaviorSubject<boolean>(false);\n  public readonly openChatDrawer$ = new EventEmitter<void>();\n  public readonly closeChatDrawer$ = new EventEmitter<void>();\n  public readonly askAI$ = new EventEmitter<string>();\n\n  private readonly selection = inject(SelectionService);\n  private readonly selectionHistory = inject(SelectionHistoryService);\n  private readonly selectionStore = inject(SelectionStore);\n  private readonly navigationService = inject(NavigationService);\n  private readonly backdropService = inject(BackdropService);\n\n  private readonly subscriptions = new Subscription();\n\n  public componentType?: Type<any>;\n\n  constructor() {\n    // on new search, close all drawers including chat drawer\n    this.subscriptions.add(\n      this.navigationService.navigationEnd$.subscribe(() => {\n        this.closeAssistant();\n        this.closeAll();\n      })\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n  }\n\n  /**\n   * Sets current drawer stack status to open\n   */\n  public open(componentType?: Type<any>): void {\n    this.componentType = componentType;\n    this.isOpened.next(true);\n  }\n\n  /**\n   * Emits event to extend the top drawer\n   */\n  public extend(): void {\n    this.toggleTopDrawerExtension$.next();\n  }\n\n  /**\n   * Emits event to close the top drawer, checks if history is empty, if so,\n   * sets current drawer stack status to closed and clears history and current\n   * selection\n   */\n  public close(): void {\n    this.closeTopDrawer$.next();\n\n    if (this.selectionHistory.getCurrentSelectionIndex() === -1) {\n      this.isOpened.next(false);\n      this.selectionHistory.clearHistory();\n    }\n  }\n\n  /**\n   * Emits event to close the top drawer\n   */\n  public closeTop(): void {\n    this.closeTopDrawer$.next();\n  }\n\n  /**\n   * Sets current drawer stack status to closed, clears history and emits event\n   * to close all drawers\n   *\n   * @param keepDrawerOpen if true, do not trigger layout animation\n   */\n  public closeAll(keepDrawerOpen: boolean = false): void {\n    if (!keepDrawerOpen) this.isOpened.next(false);\n\n    this.selectionHistory.clearHistory();\n\n    this.closeAllDrawers$.next();\n  }\n\n  /**\n   * Replace the current selection with the given article by closing all\n   * drawers and opening the drawer with the new selection without triggering\n   * layout animation\n   *\n   * @param article the article to replace the current selection with\n   */\n  public replace(article: Article | undefined): void {\n    const { id } = getState(this.selectionStore);\n\n    if (id && (!article || article.id === id)) return;\n\n    // close everything without trigger layout animation\n    this.closeAll(true);\n\n    // set selection\n    this.selection.setCurrentArticle(article);\n    // open drawer\n    this.open();\n  }\n\n  /**\n   * Stack the given article by setting the current selection and opening the\n   * drawer\n   *\n   * @param article the article to stack\n   */\n  public stack(article: Article | undefined, withQueryText?: boolean): void {\n    const { id } = getState(this.selectionStore);\n\n    if (id && (!article || article.id === id)) return;\n\n    // force top drawer to collapse\n    this.forceTopDrawerCollapse$.next();\n    // set selection\n    this.selection.setCurrentArticle(article, withQueryText);\n    // open drawer\n    this.open();\n  }\n\n  public toggleAssistant(): void {\n    if (this.isChatOpened.getValue()) {\n      this.backdropService.hide();\n      this.closeAssistant();\n    } else {\n      this.backdropService.show();\n      this.openAssistant();\n    }\n  }\n\n  public openAssistant(): void {\n    this.isChatOpened.next(true);\n    this.isOpened.next(true);\n    this.openChatDrawer$.next();\n\n    this.backdropService.show();\n  }\n\n  public closeAssistant(keepDrawerOpen: boolean = false): void {\n    this.isChatOpened.next(false);\n\n    if (!keepDrawerOpen && this.selectionHistory.getCurrentSelectionIndex() === -1) {\n      this.isOpened.next(false);\n      this.closeAllDrawers$.next();\n    }\n\n    this.closeChatDrawer$.next();\n    this.backdropService.hide();\n  }\n\n  public askAI(text: string): void {\n    this.openAssistant();\n    this.askAI$.next(text);\n  }\n}\n"]}