@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.
- package/assets/tailwind.css +248 -0
- package/esm2022/lib/assistant/index.mjs +2 -0
- package/esm2022/lib/assistant/signalR.web.service.mjs +81 -0
- package/esm2022/lib/components/dropdown.mjs +127 -0
- package/esm2022/lib/components/index.mjs +5 -0
- package/esm2022/lib/components/menu/index.mjs +3 -0
- package/esm2022/lib/components/menu/menu-item.mjs +22 -0
- package/esm2022/lib/components/menu/menu.mjs +99 -0
- package/esm2022/lib/components/metadata/index.mjs +2 -0
- package/esm2022/lib/components/metadata/metadata.component.mjs +65 -0
- package/esm2022/lib/components/theme/index.mjs +3 -0
- package/esm2022/lib/components/theme/theme-selector.component.mjs +67 -0
- package/esm2022/lib/components/theme/theme-toggle.component.mjs +67 -0
- package/esm2022/lib/directives/index.mjs +5 -0
- package/esm2022/lib/directives/infinite-scroll.directive.mjs +47 -0
- package/esm2022/lib/directives/select-article-on-click.directive.mjs +39 -0
- package/esm2022/lib/directives/show-bookmark.directive.mjs +55 -0
- package/esm2022/lib/directives/theme-provider.directive.mjs +31 -0
- package/esm2022/lib/guards/auth.guard.mjs +27 -0
- package/esm2022/lib/guards/index.mjs +3 -0
- package/esm2022/lib/guards/initialization.guard.mjs +41 -0
- package/esm2022/lib/interceptors/audit.interceptor.mjs +23 -0
- package/esm2022/lib/interceptors/auth.interceptor.mjs +49 -0
- package/esm2022/lib/interceptors/body.interceptor.mjs +24 -0
- package/esm2022/lib/interceptors/error.interceptor.mjs +35 -0
- package/esm2022/lib/interceptors/index.mjs +7 -0
- package/esm2022/lib/interceptors/toast.interceptor.mjs +27 -0
- package/esm2022/lib/models/aggregation.mjs +2 -0
- package/esm2022/lib/models/article-metadata.mjs +2 -0
- package/esm2022/lib/models/autocomplete.mjs +2 -0
- package/esm2022/lib/models/custom-json.mjs +2 -0
- package/esm2022/lib/models/filter-dropdown.mjs +2 -0
- package/esm2022/lib/models/index.mjs +7 -0
- package/esm2022/lib/models/user-settings.mjs +2 -0
- package/esm2022/lib/pipes/highlight-word.pipe.mjs +33 -0
- package/esm2022/lib/pipes/index.mjs +3 -0
- package/esm2022/lib/pipes/source-icon.pipe.mjs +43 -0
- package/esm2022/lib/providers/eager-provider.mjs +24 -0
- package/esm2022/lib/providers/index.mjs +2 -0
- package/esm2022/lib/public-api.mjs +19 -0
- package/esm2022/lib/resolvers/index.mjs +2 -0
- package/esm2022/lib/resolvers/query-name-resolver.mjs +14 -0
- package/esm2022/lib/resources/index.mjs +2 -0
- package/esm2022/lib/resources/themes.mjs +53 -0
- package/esm2022/lib/services/application.service.mjs +245 -0
- package/esm2022/lib/services/autocomplete.service.mjs +85 -0
- package/esm2022/lib/services/drawer/backdrop.service.mjs +23 -0
- package/esm2022/lib/services/drawer/drawer-stack.service.mjs +152 -0
- package/esm2022/lib/services/drawer/drawer.service.mjs +38 -0
- package/esm2022/lib/services/index.mjs +12 -0
- package/esm2022/lib/services/label.service.mjs +161 -0
- package/esm2022/lib/services/navigation.service.mjs +59 -0
- package/esm2022/lib/services/saved-searches.service.mjs +75 -0
- package/esm2022/lib/services/search.service.mjs +89 -0
- package/esm2022/lib/services/selection-history.service.mjs +92 -0
- package/esm2022/lib/services/selection.service.mjs +87 -0
- package/esm2022/lib/stores/aggregations.store.mjs +62 -0
- package/esm2022/lib/stores/app.store.mjs +265 -0
- package/esm2022/lib/stores/application.store.mjs +93 -0
- package/esm2022/lib/stores/index.mjs +9 -0
- package/esm2022/lib/stores/principal.store.mjs +47 -0
- package/esm2022/lib/stores/query-params.store.mjs +208 -0
- package/esm2022/lib/stores/selection.store.mjs +46 -0
- package/esm2022/lib/stores/theme.store.mjs +116 -0
- package/esm2022/lib/stores/user-settings.store.mjs +272 -0
- package/esm2022/lib/tokens/highlights.mjs +32 -0
- package/esm2022/lib/tokens/index.mjs +2 -0
- package/esm2022/lib/utils/debounced-signal.mjs +38 -0
- package/esm2022/lib/utils/index.mjs +8 -0
- package/esm2022/lib/utils/inline-worker.mjs +40 -0
- package/esm2022/lib/utils/query.mjs +58 -0
- package/esm2022/lib/utils/routes.mjs +28 -0
- package/esm2022/lib/utils/tailwind-utils.mjs +6 -0
- package/esm2022/lib/utils/theme-body-hook.mjs +18 -0
- package/esm2022/lib/utils/theme-registry.mjs +6 -0
- package/esm2022/lib/web-services/aggregations.service.mjs +104 -0
- package/esm2022/lib/web-services/app.service.mjs +48 -0
- package/esm2022/lib/web-services/audit.service.mjs +122 -0
- package/esm2022/lib/web-services/index.mjs +10 -0
- package/esm2022/lib/web-services/json-method-plugin.service.mjs +54 -0
- package/esm2022/lib/web-services/preview.service.mjs +327 -0
- package/esm2022/lib/web-services/principal.service.mjs +46 -0
- package/esm2022/lib/web-services/query.service.mjs +123 -0
- package/esm2022/lib/web-services/text-chunck.service.mjs +46 -0
- package/esm2022/public-api.mjs +5 -0
- package/esm2022/sinequa-atomic-angular.mjs +5 -0
- package/fesm2022/sinequa-atomic-angular.mjs +4204 -0
- package/fesm2022/sinequa-atomic-angular.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/assistant/index.d.ts +1 -0
- package/lib/assistant/signalR.web.service.d.ts +46 -0
- package/lib/components/dropdown.d.ts +50 -0
- package/lib/components/index.d.ts +4 -0
- package/lib/components/menu/index.d.ts +2 -0
- package/lib/components/menu/menu-item.d.ts +8 -0
- package/lib/components/menu/menu.d.ts +24 -0
- package/lib/components/metadata/index.d.ts +1 -0
- package/lib/components/metadata/metadata.component.d.ts +24 -0
- package/lib/components/theme/index.d.ts +2 -0
- package/lib/components/theme/theme-selector.component.d.ts +70 -0
- package/lib/components/theme/theme-toggle.component.d.ts +10 -0
- package/lib/directives/index.d.ts +4 -0
- package/lib/directives/infinite-scroll.directive.d.ts +30 -0
- package/lib/directives/select-article-on-click.directive.d.ts +14 -0
- package/lib/directives/show-bookmark.directive.d.ts +52 -0
- package/lib/directives/theme-provider.directive.d.ts +20 -0
- package/lib/guards/auth.guard.d.ts +7 -0
- package/lib/guards/index.d.ts +2 -0
- package/lib/guards/initialization.guard.d.ts +20 -0
- package/lib/interceptors/audit.interceptor.d.ts +13 -0
- package/lib/interceptors/auth.interceptor.d.ts +14 -0
- package/lib/interceptors/body.interceptor.d.ts +11 -0
- package/lib/interceptors/error.interceptor.d.ts +9 -0
- package/lib/interceptors/index.d.ts +5 -0
- package/lib/interceptors/toast.interceptor.d.ts +13 -0
- package/lib/models/aggregation.d.ts +12 -0
- package/lib/models/article-metadata.d.ts +5 -0
- package/lib/models/autocomplete.d.ts +5 -0
- package/lib/models/custom-json.d.ts +58 -0
- package/lib/models/filter-dropdown.d.ts +10 -0
- package/lib/models/index.d.ts +6 -0
- package/lib/models/user-settings.d.ts +32 -0
- package/lib/pipes/highlight-word.pipe.d.ts +22 -0
- package/lib/pipes/index.d.ts +2 -0
- package/lib/pipes/source-icon.pipe.d.ts +54 -0
- package/lib/providers/eager-provider.d.ts +11 -0
- package/lib/providers/index.d.ts +1 -0
- package/lib/public-api.d.ts +15 -0
- package/lib/resolvers/index.d.ts +1 -0
- package/lib/resolvers/query-name-resolver.d.ts +9 -0
- package/lib/resources/index.d.ts +1 -0
- package/lib/resources/themes.d.ts +51 -0
- package/lib/services/application.service.d.ts +178 -0
- package/lib/services/autocomplete.service.d.ts +91 -0
- package/lib/services/drawer/backdrop.service.d.ts +9 -0
- package/lib/services/drawer/drawer-stack.service.d.ts +70 -0
- package/lib/services/drawer/drawer.service.d.ts +15 -0
- package/lib/services/index.d.ts +11 -0
- package/lib/services/label.service.d.ts +117 -0
- package/lib/services/navigation.service.d.ts +33 -0
- package/lib/services/saved-searches.service.d.ts +145 -0
- package/lib/services/search.service.d.ts +155 -0
- package/lib/services/selection-history.service.d.ts +50 -0
- package/lib/services/selection.service.d.ts +127 -0
- package/lib/stores/aggregations.store.d.ts +50 -0
- package/lib/stores/app.store.d.ts +208 -0
- package/lib/stores/application.store.d.ts +106 -0
- package/lib/stores/index.d.ts +8 -0
- package/lib/stores/principal.store.d.ts +53 -0
- package/lib/stores/query-params.store.d.ts +187 -0
- package/lib/stores/selection.store.d.ts +62 -0
- package/lib/stores/theme.store.d.ts +55 -0
- package/lib/stores/user-settings.store.d.ts +161 -0
- package/lib/tokens/highlights.d.ts +8 -0
- package/lib/tokens/index.d.ts +1 -0
- package/lib/utils/debounced-signal.d.ts +25 -0
- package/lib/utils/index.d.ts +7 -0
- package/lib/utils/inline-worker.d.ts +11 -0
- package/lib/utils/query.d.ts +26 -0
- package/lib/utils/routes.d.ts +16 -0
- package/lib/utils/tailwind-utils.d.ts +2 -0
- package/lib/utils/theme-body-hook.d.ts +6 -0
- package/lib/utils/theme-registry.d.ts +3 -0
- package/lib/web-services/aggregations.service.d.ts +60 -0
- package/lib/web-services/app.service.d.ts +30 -0
- package/lib/web-services/audit.service.d.ts +75 -0
- package/lib/web-services/index.d.ts +9 -0
- package/lib/web-services/json-method-plugin.service.d.ts +41 -0
- package/lib/web-services/preview.service.d.ts +295 -0
- package/lib/web-services/principal.service.d.ts +28 -0
- package/lib/web-services/query.service.d.ts +29 -0
- package/lib/web-services/text-chunck.service.d.ts +22 -0
- package/package.json +28 -0
- 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"]}
|