@c8y/ngx-components 1021.63.2 → 1021.66.0
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/auth-configuration/sso-configuration/template-parts/signature-configuration.component.d.ts +7 -2
- package/auth-configuration/sso-configuration/template-parts/signature-configuration.component.d.ts.map +1 -1
- package/core/bottom-drawer/bottom-drawer-ref.d.ts +4 -0
- package/core/bottom-drawer/bottom-drawer-ref.d.ts.map +1 -1
- package/core/breadcrumb/breadcrumb-item.component.d.ts +1 -1
- package/core/breadcrumb/breadcrumb-item.component.d.ts.map +1 -1
- package/core/breadcrumb/breadcrumb-outlet.component.d.ts +1 -1
- package/core/breadcrumb/breadcrumb-outlet.component.d.ts.map +1 -1
- package/core/breadcrumb/breadcrumb.component.d.ts +1 -1
- package/core/breadcrumb/breadcrumb.component.d.ts.map +1 -1
- package/core/breadcrumb/breadcrumb.module.d.ts +7 -7
- package/core/breadcrumb/breadcrumb.module.d.ts.map +1 -1
- package/core/common/common.module.d.ts +38 -38
- package/core/common/date-format.service.d.ts +30 -0
- package/core/common/date-format.service.d.ts.map +1 -0
- package/core/common/date.pipe.d.ts +7 -3
- package/core/common/date.pipe.d.ts.map +1 -1
- package/core/common/index.d.ts +1 -0
- package/core/common/index.d.ts.map +1 -1
- package/core/common/outlet.directive.d.ts +1 -1
- package/core/common/outlet.directive.d.ts.map +1 -1
- package/core/common/user-preferences/user-preferences-store-current-user.d.ts +11 -0
- package/core/common/user-preferences/user-preferences-store-current-user.d.ts.map +1 -0
- package/core/common/user-preferences/user-preferences.service.d.ts +40 -1
- package/core/common/user-preferences/user-preferences.service.d.ts.map +1 -1
- package/core/date-picker/date-picker.component.d.ts +3 -1
- package/core/date-picker/date-picker.component.d.ts.map +1 -1
- package/core/date-time-picker/date-time-picker.component.d.ts +6 -3
- package/core/date-time-picker/date-time-picker.component.d.ts.map +1 -1
- package/core/dynamic-forms/date/date.type.component.d.ts +3 -0
- package/core/dynamic-forms/date/date.type.component.d.ts.map +1 -1
- package/core/plugins/plugins.model.d.ts +1 -0
- package/core/plugins/plugins.model.d.ts.map +1 -1
- package/core/plugins/plugins.service.d.ts +7 -2
- package/core/plugins/plugins.service.d.ts.map +1 -1
- package/core/time-interval/time-interval.component.d.ts +4 -2
- package/core/time-interval/time-interval.component.d.ts.map +1 -1
- package/ecosystem/application-plugins/application-plugin-readme.component.d.ts +15 -0
- package/ecosystem/application-plugins/application-plugin-readme.component.d.ts.map +1 -0
- package/ecosystem/application-plugins/application-plugins.component.d.ts +4 -2
- package/ecosystem/application-plugins/application-plugins.component.d.ts.map +1 -1
- package/ecosystem/application-plugins/application-plugins.module.d.ts +4 -3
- package/ecosystem/application-plugins/application-plugins.module.d.ts.map +1 -1
- package/ecosystem/application-plugins/install-plugin.component.d.ts +8 -4
- package/ecosystem/application-plugins/install-plugin.component.d.ts.map +1 -1
- package/ecosystem/application-plugins/plugin-list.component.d.ts +10 -26
- package/ecosystem/application-plugins/plugin-list.component.d.ts.map +1 -1
- package/ecosystem/application-plugins/plugin-list.service.d.ts +27 -0
- package/ecosystem/application-plugins/plugin-list.service.d.ts.map +1 -0
- package/ecosystem/ecosystem.module.d.ts +2 -1
- package/ecosystem/ecosystem.module.d.ts.map +1 -1
- package/ecosystem/packages/package-changelog-tab/package-changelog-tab.component.d.ts +15 -0
- package/ecosystem/packages/package-changelog-tab/package-changelog-tab.component.d.ts.map +1 -0
- package/ecosystem/packages/package-changelog.guard.d.ts +10 -0
- package/ecosystem/packages/package-changelog.guard.d.ts.map +1 -0
- package/ecosystem/packages/package-details/package-details.component.d.ts +7 -9
- package/ecosystem/packages/package-details/package-details.component.d.ts.map +1 -1
- package/ecosystem/packages/package-versions/package-contents/contents-plugins/contents-plugins.component.d.ts +5 -1
- package/ecosystem/packages/package-versions/package-contents/contents-plugins/contents-plugins.component.d.ts.map +1 -1
- package/ecosystem/packages/package-versions/package-contents/packages-contents.component.d.ts +8 -3
- package/ecosystem/packages/package-versions/package-contents/packages-contents.component.d.ts.map +1 -1
- package/ecosystem/packages/package-versions/packages-versions.component.d.ts +9 -2
- package/ecosystem/packages/package-versions/packages-versions.component.d.ts.map +1 -1
- package/ecosystem/shared/index.d.ts +1 -0
- package/ecosystem/shared/index.d.ts.map +1 -1
- package/ecosystem/shared/package-changelog/package-changelog.component.d.ts +24 -0
- package/ecosystem/shared/package-changelog/package-changelog.component.d.ts.map +1 -0
- package/ecosystem/shared/shared-ecosystem.module.d.ts +2 -1
- package/ecosystem/shared/shared-ecosystem.module.d.ts.map +1 -1
- package/esm2022/auth-configuration/sso-configuration/template-parts/signature-configuration.component.mjs +11 -7
- package/esm2022/core/bottom-drawer/bottom-drawer-ref.mjs +9 -3
- package/esm2022/core/breadcrumb/breadcrumb-item.component.mjs +3 -3
- package/esm2022/core/breadcrumb/breadcrumb-outlet.component.mjs +19 -9
- package/esm2022/core/breadcrumb/breadcrumb.component.mjs +3 -3
- package/esm2022/core/breadcrumb/breadcrumb.module.mjs +16 -5
- package/esm2022/core/common/common.module.mjs +6 -6
- package/esm2022/core/common/date-format.service.mjs +81 -0
- package/esm2022/core/common/date.pipe.mjs +29 -6
- package/esm2022/core/common/index.mjs +2 -1
- package/esm2022/core/common/outlet.directive.mjs +4 -3
- package/esm2022/core/common/user-preferences/user-preferences-store-current-user.mjs +22 -0
- package/esm2022/core/common/user-preferences/user-preferences.service.mjs +106 -16
- package/esm2022/core/date-picker/date-picker.component.mjs +11 -4
- package/esm2022/core/date-time-picker/date-time-picker.component.mjs +20 -11
- package/esm2022/core/dynamic-forms/date/date.type.component.mjs +12 -4
- package/esm2022/core/plugins/plugins.model.mjs +1 -1
- package/esm2022/core/plugins/plugins.service.mjs +31 -5
- package/esm2022/core/time-interval/time-interval.component.mjs +9 -2
- package/esm2022/ecosystem/application-plugins/application-plugin-readme.component.mjs +26 -0
- package/esm2022/ecosystem/application-plugins/application-plugins.component.mjs +29 -14
- package/esm2022/ecosystem/application-plugins/application-plugins.module.mjs +6 -3
- package/esm2022/ecosystem/application-plugins/install-plugin.component.mjs +21 -15
- package/esm2022/ecosystem/application-plugins/plugin-list-item.component.mjs +3 -3
- package/esm2022/ecosystem/application-plugins/plugin-list.component.mjs +26 -202
- package/esm2022/ecosystem/application-plugins/plugin-list.service.mjs +200 -0
- package/esm2022/ecosystem/application-plugins/update-plugin-of-app/update-plugin-of-app.component.mjs +3 -3
- package/esm2022/ecosystem/application-properties/update-application-modal/update-application-modal.component.mjs +1 -1
- package/esm2022/ecosystem/ecosystem.module.mjs +21 -5
- package/esm2022/ecosystem/packages/package-changelog-tab/package-changelog-tab.component.mjs +33 -0
- package/esm2022/ecosystem/packages/package-changelog.guard.mjs +22 -0
- package/esm2022/ecosystem/packages/package-details/package-details.component.mjs +25 -44
- package/esm2022/ecosystem/packages/package-versions/package-contents/contents-plugins/contents-plugins.component.mjs +14 -4
- package/esm2022/ecosystem/packages/package-versions/package-contents/packages-contents.component.mjs +27 -8
- package/esm2022/ecosystem/packages/package-versions/packages-versions.component.mjs +22 -8
- package/esm2022/ecosystem/shared/index.mjs +2 -1
- package/esm2022/ecosystem/shared/list-filters/list-filters.component.mjs +3 -3
- package/esm2022/ecosystem/shared/package-changelog/package-changelog.component.mjs +82 -0
- package/esm2022/ecosystem/shared/package-version-select/package-version-select.component.mjs +3 -3
- package/esm2022/ecosystem/shared/shared-ecosystem.module.mjs +12 -6
- package/esm2022/operations/bulk-operation-scheduler/operation-scheduler.component.mjs +9 -7
- package/esm2022/operations/bulk-operations-list/bulk-operations-list.component.mjs +1 -1
- package/esm2022/upgrade/ng1/downgraded.services.mjs +3 -2
- package/esm2022/upgrade/ng1/index.mjs +3 -2
- package/fesm2022/c8y-ngx-components-auth-configuration.mjs +9 -5
- package/fesm2022/c8y-ngx-components-auth-configuration.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-ecosystem-application-plugins.mjs +209 -142
- package/fesm2022/c8y-ngx-components-ecosystem-application-plugins.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-ecosystem-shared.mjs +96 -13
- package/fesm2022/c8y-ngx-components-ecosystem-shared.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-ecosystem.mjs +336 -186
- package/fesm2022/c8y-ngx-components-ecosystem.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-operations-bulk-operation-scheduler.mjs +7 -5
- package/fesm2022/c8y-ngx-components-operations-bulk-operation-scheduler.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-operations-bulk-operations-list.mjs +1 -1
- package/fesm2022/c8y-ngx-components-operations-bulk-operations-list.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-upgrade.mjs +3 -1
- package/fesm2022/c8y-ngx-components-upgrade.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components.mjs +833 -550
- package/fesm2022/c8y-ngx-components.mjs.map +1 -1
- package/locales/de.po +45 -6
- package/locales/es.po +45 -6
- package/locales/fr.po +45 -6
- package/locales/ja_JP.po +45 -6
- package/locales/ko.po +45 -6
- package/locales/locales.pot +42 -6
- package/locales/nl.po +45 -6
- package/locales/pl.po +45 -6
- package/locales/pt_BR.po +45 -6
- package/locales/zh_CN.po +45 -6
- package/locales/zh_TW.po +45 -6
- package/operations/bulk-operation-scheduler/operation-scheduler.component.d.ts +4 -2
- package/operations/bulk-operation-scheduler/operation-scheduler.component.d.ts.map +1 -1
- package/package.json +1 -1
- package/upgrade/ng1/downgraded.services.d.ts +1 -0
- package/upgrade/ng1/downgraded.services.d.ts.map +1 -1
- package/upgrade/ng1/index.d.ts.map +1 -1
|
@@ -1,23 +1,28 @@
|
|
|
1
1
|
import { Injectable } from '@angular/core';
|
|
2
2
|
import { InventoryService, UserService } from '@c8y/client';
|
|
3
|
-
import { concat, from, Subject } from 'rxjs';
|
|
3
|
+
import { concat, firstValueFrom, from, Subject } from 'rxjs';
|
|
4
4
|
import { filter, first, map, switchMap } from 'rxjs/operators';
|
|
5
5
|
import { AppStateService } from '../ui-state.service';
|
|
6
6
|
import { UserPreferencesStorageLocal } from './user-preferences-storage-local';
|
|
7
7
|
import { UserPreferencesStorageInventory } from './user-preferences-store-inventory';
|
|
8
8
|
import { Permissions } from '../permissions.service';
|
|
9
|
+
import { AlertService } from '../../alert/alert.service';
|
|
10
|
+
import { UserPreferencesStorageCurrentUser } from './user-preferences-store-current-user';
|
|
9
11
|
import * as i0 from "@angular/core";
|
|
10
12
|
import * as i1 from "@c8y/client";
|
|
11
13
|
import * as i2 from "../ui-state.service";
|
|
14
|
+
import * as i3 from "../../alert/alert.service";
|
|
12
15
|
export class UserPreferencesService {
|
|
13
|
-
constructor(user, inventory, appState) {
|
|
16
|
+
constructor(user, inventory, appState, alert) {
|
|
14
17
|
this.user = user;
|
|
15
18
|
this.inventory = inventory;
|
|
16
19
|
this.appState = appState;
|
|
20
|
+
this.alert = alert;
|
|
17
21
|
this.preferenceChanges$ = new Subject();
|
|
18
22
|
this.storage = {
|
|
19
23
|
local: new UserPreferencesStorageLocal(),
|
|
20
|
-
inventory: new UserPreferencesStorageInventory(this.inventory)
|
|
24
|
+
inventory: new UserPreferencesStorageInventory(this.inventory),
|
|
25
|
+
currentUser: new UserPreferencesStorageCurrentUser(this.user, this.appState)
|
|
21
26
|
};
|
|
22
27
|
this.currentUser = this.appState.currentUser.pipe(filter(currentUser => currentUser !== null));
|
|
23
28
|
}
|
|
@@ -36,7 +41,7 @@ export class UserPreferencesService {
|
|
|
36
41
|
* @returns An Observable with the value of preference.
|
|
37
42
|
*/
|
|
38
43
|
get(key) {
|
|
39
|
-
return this.currentUser.pipe(first(), switchMap(user => this.
|
|
44
|
+
return this.currentUser.pipe(first(), switchMap(user => this.getForCurrentUser(key, user)));
|
|
40
45
|
}
|
|
41
46
|
/**
|
|
42
47
|
* Sets a value in storage for current user.
|
|
@@ -45,21 +50,19 @@ export class UserPreferencesService {
|
|
|
45
50
|
* @returns A promise with saved value.
|
|
46
51
|
*/
|
|
47
52
|
set(key, value) {
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
this.preferenceChanges$.next({ key, value });
|
|
51
|
-
resolve(this.setForUser(key, value, user));
|
|
52
|
-
}, reject);
|
|
53
|
-
});
|
|
53
|
+
this.preferenceChanges$.next({ key, value });
|
|
54
|
+
return this.setForCurrentUser(key, value);
|
|
54
55
|
}
|
|
55
56
|
/**
|
|
56
57
|
* Get an Observable value of searched key for a specific user.
|
|
57
58
|
* @param key The storage key for searched value.
|
|
58
59
|
* @param user The user for whom the search is done.
|
|
59
60
|
* @returns An Observable with the value of preference.
|
|
61
|
+
*
|
|
62
|
+
* @deprecated Uses depracted inventory approach. Use get instead.
|
|
60
63
|
*/
|
|
61
64
|
getForUser(key, user) {
|
|
62
|
-
const rawKey = this.
|
|
65
|
+
const rawKey = this.getTransformedRawKey(key, user);
|
|
63
66
|
const storage = this.getStorage(user);
|
|
64
67
|
return from(storage.get(rawKey));
|
|
65
68
|
}
|
|
@@ -68,22 +71,100 @@ export class UserPreferencesService {
|
|
|
68
71
|
* @param key The storage key for the value to be set.
|
|
69
72
|
* @param value The storage value to be set.
|
|
70
73
|
* @returns A promise with saved value.
|
|
74
|
+
*
|
|
75
|
+
* @deprecated Uses deprecated inventory approach. Use set instead.
|
|
71
76
|
*/
|
|
72
77
|
setForUser(key, value, user) {
|
|
73
|
-
const rawKey = this.
|
|
78
|
+
const rawKey = this.getTransformedRawKey(key, user);
|
|
74
79
|
const storage = this.getStorage(user);
|
|
75
80
|
return Promise.resolve(storage.set(rawKey, value));
|
|
76
81
|
}
|
|
82
|
+
/**
|
|
83
|
+
* Get value of searched key for current user.
|
|
84
|
+
* If preference is not found in user's customProperties, it will try to get it from inventory or local storage and
|
|
85
|
+
* update user's customProperties with the value and return that value.
|
|
86
|
+
* @param key The preference key for searched value.
|
|
87
|
+
* @param user The user for whom the search is done.
|
|
88
|
+
* @returns A Promise with the value of preference.
|
|
89
|
+
*/
|
|
90
|
+
async getForCurrentUser(key, user) {
|
|
91
|
+
const currentUserStorage = await this.getCurrentUserStorage();
|
|
92
|
+
const customPropertiesKey = this.getCustomPropertiesKey(key);
|
|
93
|
+
const currentUserHasKeyProperty = currentUserStorage && (await currentUserStorage.hasKey(customPropertiesKey));
|
|
94
|
+
if (currentUserHasKeyProperty) {
|
|
95
|
+
return await currentUserStorage.get(customPropertiesKey);
|
|
96
|
+
}
|
|
97
|
+
else {
|
|
98
|
+
const localOrInventoryStorage = this.getStorage(user);
|
|
99
|
+
const transformedRawKey = this.getTransformedRawKey(key, user);
|
|
100
|
+
const rawKey = this.getRawKey(key, user);
|
|
101
|
+
const valueFromStorage = (await localOrInventoryStorage.get(transformedRawKey)) ||
|
|
102
|
+
(await localOrInventoryStorage.get(rawKey));
|
|
103
|
+
if (valueFromStorage === undefined) {
|
|
104
|
+
return undefined;
|
|
105
|
+
}
|
|
106
|
+
try {
|
|
107
|
+
await currentUserStorage?.set(customPropertiesKey, valueFromStorage);
|
|
108
|
+
}
|
|
109
|
+
catch (e) {
|
|
110
|
+
// do nothing
|
|
111
|
+
}
|
|
112
|
+
return valueFromStorage;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Sets a value for current user.
|
|
117
|
+
* @param key The preference key for the value to be set.
|
|
118
|
+
* @param value The preference value to be set.
|
|
119
|
+
*/
|
|
120
|
+
async setForCurrentUser(key, value) {
|
|
121
|
+
const currentUserStorage = await this.getCurrentUserStorage();
|
|
122
|
+
const userPreferencesKey = this.getCustomPropertiesKey(key);
|
|
123
|
+
if (currentUserStorage) {
|
|
124
|
+
try {
|
|
125
|
+
await currentUserStorage.set(userPreferencesKey, value);
|
|
126
|
+
}
|
|
127
|
+
catch (e) {
|
|
128
|
+
this.alert.addServerFailure(e);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
const user = await firstValueFrom(this.currentUser);
|
|
133
|
+
const rawKey = this.getRawKey(key, user);
|
|
134
|
+
this.storage.local.set(rawKey, value);
|
|
135
|
+
}
|
|
136
|
+
}
|
|
77
137
|
/**
|
|
78
138
|
* Get a string of key concatenated with username.
|
|
139
|
+
* Used by deprecated invetory storage approach and for leftovers in local storage.
|
|
140
|
+
* It was used to store preferences for specific users and it as replacing "." with "__"
|
|
141
|
+
* because of MongoDB restrictions.
|
|
79
142
|
* @param key The storage key for searched value.
|
|
80
143
|
* @param user The user for whom the search is done.
|
|
81
144
|
* @returns A string of key concatenated with username.
|
|
82
145
|
*/
|
|
83
|
-
|
|
146
|
+
getTransformedRawKey(key, user) {
|
|
84
147
|
const username = user.userName.replace(/\./g, '__');
|
|
85
148
|
return `${key}${username}`;
|
|
86
149
|
}
|
|
150
|
+
/**
|
|
151
|
+
* Get a string of key concatenated with username.
|
|
152
|
+
* Used by local storage.
|
|
153
|
+
* @param key The storage key for searched value.
|
|
154
|
+
* @param user The user for whom the search is done.
|
|
155
|
+
* @returns A string of key concatenated with username.
|
|
156
|
+
*/
|
|
157
|
+
getRawKey(key, user) {
|
|
158
|
+
return `${key}${user.userName}`;
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Get a key for user preferences. Used current user customProperties.
|
|
162
|
+
* @param key The storage key for searched value.
|
|
163
|
+
* @returns A key for user preferences.
|
|
164
|
+
*/
|
|
165
|
+
getCustomPropertiesKey(key) {
|
|
166
|
+
return `c8y_UserPreference-${key}`;
|
|
167
|
+
}
|
|
87
168
|
/**
|
|
88
169
|
* Gets a proper storage depending on the user roles.
|
|
89
170
|
* @param user The user for whom the role check is done.
|
|
@@ -101,11 +182,20 @@ export class UserPreferencesService {
|
|
|
101
182
|
? this.storage.inventory
|
|
102
183
|
: this.storage.local;
|
|
103
184
|
}
|
|
104
|
-
|
|
185
|
+
async getCurrentUserStorage() {
|
|
186
|
+
const currentUser = await firstValueFrom(this.currentUser);
|
|
187
|
+
const hasRoleToEditCurrentUser = this.user.hasRole(currentUser, Permissions.ROLE_USER_MANAGEMENT_OWN_ADMIN);
|
|
188
|
+
const isExternalUser = currentUser.customProperties?.userOrigin === 'OAUTH2';
|
|
189
|
+
if (!hasRoleToEditCurrentUser || isExternalUser) {
|
|
190
|
+
return null;
|
|
191
|
+
}
|
|
192
|
+
return this.storage.currentUser;
|
|
193
|
+
}
|
|
194
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UserPreferencesService, deps: [{ token: i1.UserService }, { token: i1.InventoryService }, { token: i2.AppStateService }, { token: i3.AlertService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
105
195
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UserPreferencesService, providedIn: 'root' }); }
|
|
106
196
|
}
|
|
107
197
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UserPreferencesService, decorators: [{
|
|
108
198
|
type: Injectable,
|
|
109
199
|
args: [{ providedIn: 'root' }]
|
|
110
|
-
}], ctorParameters: () => [{ type: i1.UserService }, { type: i1.InventoryService }, { type: i2.AppStateService }] });
|
|
111
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user-preferences.service.js","sourceRoot":"","sources":["../../../../../core/common/user-preferences/user-preferences.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { InventoryService, IUser, UserService } from '@c8y/client';\nimport { concat, from, Observable, Subject } from 'rxjs';\nimport { filter, first, map, switchMap } from 'rxjs/operators';\nimport { AppStateService } from '../ui-state.service';\nimport { UserPreferencesStorageLocal } from './user-preferences-storage-local';\nimport { UserPreferencesStorageInventory } from './user-preferences-store-inventory';\nimport { UserPreference } from './user-preference.model';\nimport { Permissions } from '../permissions.service';\n\n@Injectable({ providedIn: 'root' })\nexport class UserPreferencesService {\n  currentUser: Observable<IUser>;\n  preferenceChanges$: Subject<UserPreference> = new Subject();\n  private storage: {\n    local: UserPreferencesStorageLocal;\n    inventory: UserPreferencesStorageInventory;\n  };\n\n  constructor(\n    private user: UserService,\n    private inventory: InventoryService,\n    private appState: AppStateService\n  ) {\n    this.storage = {\n      local: new UserPreferencesStorageLocal(),\n      inventory: new UserPreferencesStorageInventory(this.inventory)\n    };\n    this.currentUser = this.appState.currentUser.pipe(filter(currentUser => currentUser !== null));\n  }\n\n  /**\n   * Returns an observable of a user preference with given key.\n   * Emits its initial value first and then updated values when set by user.\n   * @param key The storage key for searched value.\n   * @returns An Observable of a user preference.\n   */\n  observe<T>(key: string): Observable<T> {\n    return concat(\n      this.get(key),\n      this.preferenceChanges$.pipe(\n        filter(change => change.key === key),\n        map(change => change.value)\n      )\n    );\n  }\n\n  /**\n   * Get an Observable value for searched key for current user.\n   * @param key The storage key for searched value.\n   * @returns An Observable with the value of preference.\n   */\n  get(key: string): Observable<any> {\n    return this.currentUser.pipe(\n      first(),\n      switchMap(user => this.getForUser(key, user))\n    );\n  }\n\n  /**\n   * Sets a value in storage for current user.\n   * @param key The storage key for the value to be set.\n   * @param value The storage value to be set.\n   * @returns A promise with saved value.\n   */\n  set(key: string, value: any): Promise<any> {\n    return new Promise((resolve, reject) => {\n      this.currentUser.pipe(first()).subscribe(user => {\n        this.preferenceChanges$.next({ key, value });\n        resolve(this.setForUser(key, value, user));\n      }, reject);\n    });\n  }\n\n  /**\n   * Get an Observable value of searched key for a specific user.\n   * @param key The storage key for searched value.\n   * @param user The user for whom the search is done.\n   * @returns An Observable with the value of preference.\n   */\n  getForUser(key: string, user: IUser): Observable<any> {\n    const rawKey = this.getRawKey(key, user);\n    const storage = this.getStorage(user);\n    return from(storage.get(rawKey));\n  }\n\n  /**\n   * Sets a value in storage for a specific user.\n   * @param key The storage key for the value to be set.\n   * @param value The storage value to be set.\n   * @returns A promise with saved value.\n   */\n  setForUser(key: string, value: any, user: IUser): Promise<any> {\n    const rawKey = this.getRawKey(key, user);\n    const storage = this.getStorage(user);\n    return Promise.resolve(storage.set(rawKey, value));\n  }\n\n  /**\n   * Get a string of key concatenated with username.\n   * @param key The storage key for searched value.\n   * @param user The user for whom the search is done.\n   * @returns A string of key concatenated with username.\n   */\n  private getRawKey(key: string, user: IUser): string {\n    const username = user.userName.replace(/\\./g, '__');\n    return `${key}${username}`;\n  }\n\n  /**\n   * Gets a proper storage depending on the user roles.\n   * @param user The user for whom the role check is done.\n   * @returns A proper storage.\n   */\n  private getStorage(user: IUser): UserPreferencesStorageInventory | UserPreferencesStorageLocal {\n    return this.user.hasAllRoles(user, [\n      Permissions.ROLE_INVENTORY_READ,\n      Permissions.ROLE_INVENTORY_ADMIN\n    ]) ||\n      this.user.hasAllRoles(user, [\n        Permissions.ROLE_MANAGED_OBJECT_ADMIN,\n        Permissions.ROLE_MANAGED_OBJECT_READ\n      ])\n      ? this.storage.inventory\n      : this.storage.local;\n  }\n}\n"]}
|
|
200
|
+
}], ctorParameters: () => [{ type: i1.UserService }, { type: i1.InventoryService }, { type: i2.AppStateService }, { type: i3.AlertService }] });
|
|
201
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user-preferences.service.js","sourceRoot":"","sources":["../../../../../core/common/user-preferences/user-preferences.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { InventoryService, IUser, UserService } from '@c8y/client';\nimport { concat, firstValueFrom, from, Observable, Subject } from 'rxjs';\nimport { filter, first, map, switchMap } from 'rxjs/operators';\nimport { AppStateService } from '../ui-state.service';\nimport { UserPreferencesStorageLocal } from './user-preferences-storage-local';\nimport { UserPreferencesStorageInventory } from './user-preferences-store-inventory';\nimport { UserPreference } from './user-preference.model';\nimport { Permissions } from '../permissions.service';\nimport { AlertService } from '../../alert/alert.service';\nimport { UserPreferencesStorageCurrentUser } from './user-preferences-store-current-user';\n\n@Injectable({ providedIn: 'root' })\nexport class UserPreferencesService {\n  currentUser: Observable<IUser>;\n  preferenceChanges$: Subject<UserPreference> = new Subject();\n  private storage: {\n    local: UserPreferencesStorageLocal;\n    inventory: UserPreferencesStorageInventory;\n    currentUser: UserPreferencesStorageCurrentUser;\n  };\n\n  constructor(\n    private user: UserService,\n    private inventory: InventoryService,\n    private appState: AppStateService,\n    private alert: AlertService\n  ) {\n    this.storage = {\n      local: new UserPreferencesStorageLocal(),\n      inventory: new UserPreferencesStorageInventory(this.inventory),\n      currentUser: new UserPreferencesStorageCurrentUser(this.user, this.appState)\n    };\n    this.currentUser = this.appState.currentUser.pipe(filter(currentUser => currentUser !== null));\n  }\n\n  /**\n   * Returns an observable of a user preference with given key.\n   * Emits its initial value first and then updated values when set by user.\n   * @param key The storage key for searched value.\n   * @returns An Observable of a user preference.\n   */\n  observe<T>(key: string): Observable<T> {\n    return concat(\n      this.get(key),\n      this.preferenceChanges$.pipe(\n        filter(change => change.key === key),\n        map(change => change.value)\n      )\n    );\n  }\n\n  /**\n   * Get an Observable value for searched key for current user.\n   * @param key The storage key for searched value.\n   * @returns An Observable with the value of preference.\n   */\n  get(key: string): Observable<any> {\n    return this.currentUser.pipe(\n      first(),\n      switchMap(user => this.getForCurrentUser(key, user))\n    );\n  }\n\n  /**\n   * Sets a value in storage for current user.\n   * @param key The storage key for the value to be set.\n   * @param value The storage value to be set.\n   * @returns A promise with saved value.\n   */\n  set(key: string, value: any): Promise<any> {\n    this.preferenceChanges$.next({ key, value });\n    return this.setForCurrentUser(key, value);\n  }\n\n  /**\n   * Get an Observable value of searched key for a specific user.\n   * @param key The storage key for searched value.\n   * @param user The user for whom the search is done.\n   * @returns An Observable with the value of preference.\n   *\n   * @deprecated Uses depracted inventory approach. Use get instead.\n   */\n  getForUser(key: string, user: IUser): Observable<any> {\n    const rawKey = this.getTransformedRawKey(key, user);\n    const storage = this.getStorage(user);\n    return from(storage.get(rawKey));\n  }\n\n  /**\n   * Sets a value in storage for a specific user.\n   * @param key The storage key for the value to be set.\n   * @param value The storage value to be set.\n   * @returns A promise with saved value.\n   *\n   * @deprecated Uses deprecated inventory approach. Use set instead.\n   */\n  setForUser(key: string, value: any, user: IUser): Promise<any> {\n    const rawKey = this.getTransformedRawKey(key, user);\n    const storage = this.getStorage(user);\n    return Promise.resolve(storage.set(rawKey, value));\n  }\n\n  /**\n   * Get value of searched key for current user.\n   * If preference is not found in user's customProperties, it will try to get it from inventory or local storage and\n   * update user's customProperties with the value and return that value.\n   * @param key The preference key for searched value.\n   * @param user The user for whom the search is done.\n   * @returns A Promise with the value of preference.\n   */\n  private async getForCurrentUser(key: string, user: IUser): Promise<any> {\n    const currentUserStorage = await this.getCurrentUserStorage();\n    const customPropertiesKey = this.getCustomPropertiesKey(key);\n    const currentUserHasKeyProperty =\n      currentUserStorage && (await currentUserStorage.hasKey(customPropertiesKey));\n    if (currentUserHasKeyProperty) {\n      return await currentUserStorage.get(customPropertiesKey);\n    } else {\n      const localOrInventoryStorage = this.getStorage(user);\n      const transformedRawKey = this.getTransformedRawKey(key, user);\n      const rawKey = this.getRawKey(key, user);\n\n      const valueFromStorage =\n        (await localOrInventoryStorage.get(transformedRawKey)) ||\n        (await localOrInventoryStorage.get(rawKey));\n      if (valueFromStorage === undefined) {\n        return undefined;\n      }\n      try {\n        await currentUserStorage?.set(customPropertiesKey, valueFromStorage);\n      } catch (e) {\n        // do nothing\n      }\n      return valueFromStorage;\n    }\n  }\n\n  /**\n   * Sets a value for current user.\n   * @param key The preference key for the value to be set.\n   * @param value The preference value to be set.\n   */\n  private async setForCurrentUser(key: string, value: any): Promise<void> {\n    const currentUserStorage = await this.getCurrentUserStorage();\n    const userPreferencesKey = this.getCustomPropertiesKey(key);\n    if (currentUserStorage) {\n      try {\n        await currentUserStorage.set(userPreferencesKey, value);\n      } catch (e) {\n        this.alert.addServerFailure(e);\n      }\n    } else {\n      const user = await firstValueFrom(this.currentUser);\n      const rawKey = this.getRawKey(key, user);\n      this.storage.local.set(rawKey, value);\n    }\n  }\n\n  /**\n   * Get a string of key concatenated with username.\n   * Used by deprecated invetory storage approach and for leftovers in local storage.\n   * It was used to store preferences for specific users and it as replacing \".\" with \"__\"\n   * because of MongoDB restrictions.\n   * @param key The storage key for searched value.\n   * @param user The user for whom the search is done.\n   * @returns A string of key concatenated with username.\n   */\n  private getTransformedRawKey(key: string, user: IUser): string {\n    const username = user.userName.replace(/\\./g, '__');\n    return `${key}${username}`;\n  }\n\n  /**\n   * Get a string of key concatenated with username.\n   * Used by local storage.\n   * @param key The storage key for searched value.\n   * @param user The user for whom the search is done.\n   * @returns A string of key concatenated with username.\n   */\n  private getRawKey(key: string, user: IUser): string {\n    return `${key}${user.userName}`;\n  }\n\n  /**\n   * Get a key for user preferences. Used current user customProperties.\n   * @param key The storage key for searched value.\n   * @returns A key for user preferences.\n   */\n  private getCustomPropertiesKey(key: string): string {\n    return `c8y_UserPreference-${key}`;\n  }\n\n  /**\n   * Gets a proper storage depending on the user roles.\n   * @param user The user for whom the role check is done.\n   * @returns A proper storage.\n   */\n  private getStorage(user: IUser): UserPreferencesStorageInventory | UserPreferencesStorageLocal {\n    return this.user.hasAllRoles(user, [\n      Permissions.ROLE_INVENTORY_READ,\n      Permissions.ROLE_INVENTORY_ADMIN\n    ]) ||\n      this.user.hasAllRoles(user, [\n        Permissions.ROLE_MANAGED_OBJECT_ADMIN,\n        Permissions.ROLE_MANAGED_OBJECT_READ\n      ])\n      ? this.storage.inventory\n      : this.storage.local;\n  }\n\n  private async getCurrentUserStorage(): Promise<UserPreferencesStorageCurrentUser> {\n    const currentUser = await firstValueFrom(this.currentUser);\n    const hasRoleToEditCurrentUser = this.user.hasRole(\n      currentUser,\n      Permissions.ROLE_USER_MANAGEMENT_OWN_ADMIN\n    );\n    const isExternalUser = currentUser.customProperties?.userOrigin === 'OAUTH2';\n    if (!hasRoleToEditCurrentUser || isExternalUser) {\n      return null;\n    }\n    return this.storage.currentUser;\n  }\n}\n"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
1
|
+
import { Component, EventEmitter, Input, Output, inject } from '@angular/core';
|
|
2
2
|
import { FormControl, FormGroup, FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
3
3
|
import { gettext } from '../i18n';
|
|
4
4
|
import { BsDropdownModule } from 'ngx-bootstrap/dropdown';
|
|
@@ -10,6 +10,7 @@ import { C8yTranslateDirective } from '../i18n/c8y-translate.directive';
|
|
|
10
10
|
import { RequiredInputPlaceholderDirective } from '../forms/required-input-placeholder.directive';
|
|
11
11
|
import { BsDatepickerModule } from 'ngx-bootstrap/datepicker';
|
|
12
12
|
import { C8yTranslatePipe } from '../i18n/c8y-translate.pipe';
|
|
13
|
+
import { DateFormatService } from '../common/date-format.service';
|
|
13
14
|
import * as i0 from "@angular/core";
|
|
14
15
|
import * as i1 from "ngx-bootstrap/dropdown";
|
|
15
16
|
import * as i2 from "@angular/forms";
|
|
@@ -18,8 +19,12 @@ export class DatePickerComponent {
|
|
|
18
19
|
constructor() {
|
|
19
20
|
this.onDateSelected = new EventEmitter();
|
|
20
21
|
this.placeholder = gettext('Filter by date…');
|
|
22
|
+
this.dateFormatService = inject(DateFormatService);
|
|
21
23
|
}
|
|
22
24
|
ngOnInit() {
|
|
25
|
+
if (!this.dateInputFormat) {
|
|
26
|
+
this.dateInputFormat = this.dateFormatService.getDateFormat();
|
|
27
|
+
}
|
|
23
28
|
this.fgDatePicker = new FormGroup({
|
|
24
29
|
dateFrom: new FormControl(),
|
|
25
30
|
dateTo: new FormControl()
|
|
@@ -40,7 +45,7 @@ export class DatePickerComponent {
|
|
|
40
45
|
this.dateTo = null;
|
|
41
46
|
}
|
|
42
47
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DatePickerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
43
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DatePickerComponent, isStandalone: true, selector: "c8y-date-picker", inputs: { placeholder: "placeholder" }, outputs: { onDateSelected: "onDateSelected" }, ngImport: i0, template: "<div\n dropdown\n class=\"c8y-child-assets-selector dropdown\"\n #datefilter=\"bs-dropdown\"\n [insideClick]=\"true\"\n placement=\"bottom left\"\n [cdkTrapFocus]=\"datefilter.isOpen\"\n >\n <button\n id=\"date-range\"\n dropdownToggle\n title=\"{{ 'Date filter' | translate }}\"\n type=\"button\"\n class=\"btn dropdown-toggle d-flex a-i-center c8y-dropdown\"\n >\n <i c8yIcon=\"calendar-o\" class=\"m-r-4 text-primary\"></i>\n <span class=\"text-truncate\">\n <span *ngIf=\"dateFrom\">\n <span class=\"text-label-small\">{{ 'From`date`' | translate }}</span>\n {{ dateFrom | date }}\n </span>\n <span *ngIf=\"dateTo\">\n <span class=\"text-label-small\">{{ 'To`date`' | translate }}</span>\n {{ dateTo | date }}\n </span>\n <em *ngIf=\"!dateFrom && !dateTo\" class=\"text-muted\">\n {{ placeholder }}\n </em>\n </span>\n </button>\n\n <form [formGroup]=\"fgDatePicker\"\n id=\"dropdown-date-range\"\n *dropdownMenu\n class=\"dropdown-menu\">\n <div class=\"dropdown-form p-b-0\">\n <c8y-form-group class=\"form-group-sm\">\n <label for=\"dateFrom\" class=\"text-medium m-b-4\" translate>Date from</label>\n <div class=\"form-group datepicker d-block m-b-0\">\n <input\n id=\"dateFrom\"\n formControlName=\"dateFrom\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date from' | translate }}\"\n bsDatepicker\n [maxDate]=\"dateTo\"\n (bsValueChange)=\"dateFrom = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true }\"\n />\n </div>\n </c8y-form-group>\n <c8y-form-group class=\"form-group form-group-sm \">\n <label for=\"dateTo\" translate>Date to</label>\n <div class=\"form-group datepicker m-l-0 d-block m-b-0 \">\n <input\n name=\"dateTo\"\n id=\"dateTo\"\n formControlName=\"dateTo\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date to' | translate }}\"\n bsDatepicker\n [minDate]=\"dateFrom\"\n (bsValueChange)=\"dateTo = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true }\"\n />\n </div>\n </c8y-form-group>\n </div>\n <div class=\"p-16 d-flex separator-top gap-8\">\n <button\n title=\"{{ 'Clear selection' | translate }}\"\n type=\"button\"\n class=\"btn btn-default btn-sm flex-grow\"\n (click)=\"clearFilter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Clear`selection`' | translate }}\n </button>\n <button\n [disabled]=\"!fgDatePicker.get('dateFrom').value && !fgDatePicker.get('dateTo').value\"\n title=\"{{ 'Apply selection' | translate }}\"\n type=\"submit\"\n class=\"btn btn-primary btn-sm flex-grow\"\n (click)=\"filter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Apply`selection`' | translate }}\n </button>\n </div>\n </form>\n</div>\n", dependencies: [{ kind: "ngmodule", type: BsDropdownModule }, { kind: "directive", type: i1.BsDropdownMenuDirective, selector: "[bsDropdownMenu],[dropdownMenu]", exportAs: ["bs-dropdown-menu"] }, { kind: "directive", type: i1.BsDropdownToggleDirective, selector: "[bsDropdownToggle],[dropdownToggle]", exportAs: ["bs-dropdown-toggle"] }, { kind: "directive", type: i1.BsDropdownDirective, selector: "[bsDropdown], [dropdown]", inputs: ["placement", "triggers", "container", "dropup", "autoClose", "isAnimated", "insideClick", "isDisabled", "isOpen"], outputs: ["isOpenChange", "onShown", "onHidden"], exportAs: ["bs-dropdown"] }, { kind: "directive", type: CdkTrapFocus, selector: "[cdkTrapFocus]", inputs: ["cdkTrapFocus", "cdkTrapFocusAutoCapture"], exportAs: ["cdkTrapFocus"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: FormGroupComponent, selector: "c8y-form-group", inputs: ["hasError", "hasWarning", "hasSuccess", "novalidation", "status"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: RequiredInputPlaceholderDirective, selector: "input[required], input[formControlName]" }, { kind: "ngmodule", type: BsDatepickerModule }, { kind: "directive", type: i3.BsDatepickerDirective, selector: "[bsDatepicker]", inputs: ["placement", "triggers", "outsideClick", "container", "outsideEsc", "isDisabled", "minDate", "maxDate", "minMode", "daysDisabled", "datesDisabled", "datesEnabled", "dateCustomClasses", "dateTooltipTexts", "isOpen", "bsValue", "bsConfig"], outputs: ["onShown", "onHidden", "bsValueChange"], exportAs: ["bsDatepicker"] }, { kind: "directive", type: i3.BsDatepickerInputDirective, selector: "input[bsDatepicker]" }, { kind: "pipe", type: DatePipe, name: "date" }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }] }); }
|
|
48
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DatePickerComponent, isStandalone: true, selector: "c8y-date-picker", inputs: { placeholder: "placeholder", dateInputFormat: "dateInputFormat" }, outputs: { onDateSelected: "onDateSelected" }, ngImport: i0, template: "<div\n dropdown\n class=\"c8y-child-assets-selector dropdown\"\n #datefilter=\"bs-dropdown\"\n [insideClick]=\"true\"\n placement=\"bottom left\"\n [cdkTrapFocus]=\"datefilter.isOpen\"\n >\n <button\n id=\"date-range\"\n dropdownToggle\n title=\"{{ 'Date filter' | translate }}\"\n type=\"button\"\n class=\"btn dropdown-toggle d-flex a-i-center c8y-dropdown\"\n >\n <i c8yIcon=\"calendar-o\" class=\"m-r-4 text-primary\"></i>\n <span class=\"text-truncate\">\n <span *ngIf=\"dateFrom\">\n <span class=\"text-label-small\">{{ 'From`date`' | translate }}</span>\n {{ dateFrom | date }}\n </span>\n <span *ngIf=\"dateTo\">\n <span class=\"text-label-small\">{{ 'To`date`' | translate }}</span>\n {{ dateTo | date }}\n </span>\n <em *ngIf=\"!dateFrom && !dateTo\" class=\"text-muted\">\n {{ placeholder }}\n </em>\n </span>\n </button>\n\n <form [formGroup]=\"fgDatePicker\"\n id=\"dropdown-date-range\"\n *dropdownMenu\n class=\"dropdown-menu\">\n <div class=\"dropdown-form p-b-0\">\n <c8y-form-group class=\"form-group-sm\">\n <label for=\"dateFrom\" class=\"text-medium m-b-4\" translate>Date from</label>\n <div class=\"form-group datepicker d-block m-b-0\">\n <input\n id=\"dateFrom\"\n formControlName=\"dateFrom\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date from' | translate }}\"\n bsDatepicker\n [maxDate]=\"dateTo\"\n (bsValueChange)=\"dateFrom = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true, dateInputFormat: dateInputFormat }\"\n />\n </div>\n </c8y-form-group>\n <c8y-form-group class=\"form-group form-group-sm \">\n <label for=\"dateTo\" translate>Date to</label>\n <div class=\"form-group datepicker m-l-0 d-block m-b-0 \">\n <input\n name=\"dateTo\"\n id=\"dateTo\"\n formControlName=\"dateTo\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date to' | translate }}\"\n bsDatepicker\n [minDate]=\"dateFrom\"\n (bsValueChange)=\"dateTo = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true, dateInputFormat: dateInputFormat }\"\n />\n </div>\n </c8y-form-group>\n </div>\n <div class=\"p-16 d-flex separator-top gap-8\">\n <button\n title=\"{{ 'Clear selection' | translate }}\"\n type=\"button\"\n class=\"btn btn-default btn-sm flex-grow\"\n (click)=\"clearFilter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Clear`selection`' | translate }}\n </button>\n <button\n [disabled]=\"!fgDatePicker.get('dateFrom').value && !fgDatePicker.get('dateTo').value\"\n title=\"{{ 'Apply selection' | translate }}\"\n type=\"submit\"\n class=\"btn btn-primary btn-sm flex-grow\"\n (click)=\"filter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Apply`selection`' | translate }}\n </button>\n </div>\n </form>\n</div>\n", dependencies: [{ kind: "ngmodule", type: BsDropdownModule }, { kind: "directive", type: i1.BsDropdownMenuDirective, selector: "[bsDropdownMenu],[dropdownMenu]", exportAs: ["bs-dropdown-menu"] }, { kind: "directive", type: i1.BsDropdownToggleDirective, selector: "[bsDropdownToggle],[dropdownToggle]", exportAs: ["bs-dropdown-toggle"] }, { kind: "directive", type: i1.BsDropdownDirective, selector: "[bsDropdown], [dropdown]", inputs: ["placement", "triggers", "container", "dropup", "autoClose", "isAnimated", "insideClick", "isDisabled", "isOpen"], outputs: ["isOpenChange", "onShown", "onHidden"], exportAs: ["bs-dropdown"] }, { kind: "directive", type: CdkTrapFocus, selector: "[cdkTrapFocus]", inputs: ["cdkTrapFocus", "cdkTrapFocusAutoCapture"], exportAs: ["cdkTrapFocus"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: FormGroupComponent, selector: "c8y-form-group", inputs: ["hasError", "hasWarning", "hasSuccess", "novalidation", "status"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: RequiredInputPlaceholderDirective, selector: "input[required], input[formControlName]" }, { kind: "ngmodule", type: BsDatepickerModule }, { kind: "directive", type: i3.BsDatepickerDirective, selector: "[bsDatepicker]", inputs: ["placement", "triggers", "outsideClick", "container", "outsideEsc", "isDisabled", "minDate", "maxDate", "minMode", "daysDisabled", "datesDisabled", "datesEnabled", "dateCustomClasses", "dateTooltipTexts", "isOpen", "bsValue", "bsConfig"], outputs: ["onShown", "onHidden", "bsValueChange"], exportAs: ["bsDatepicker"] }, { kind: "directive", type: i3.BsDatepickerInputDirective, selector: "input[bsDatepicker]" }, { kind: "pipe", type: DatePipe, name: "date" }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }] }); }
|
|
44
49
|
}
|
|
45
50
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DatePickerComponent, decorators: [{
|
|
46
51
|
type: Component,
|
|
@@ -57,10 +62,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
57
62
|
BsDatepickerModule,
|
|
58
63
|
DatePipe,
|
|
59
64
|
C8yTranslatePipe
|
|
60
|
-
], template: "<div\n dropdown\n class=\"c8y-child-assets-selector dropdown\"\n #datefilter=\"bs-dropdown\"\n [insideClick]=\"true\"\n placement=\"bottom left\"\n [cdkTrapFocus]=\"datefilter.isOpen\"\n >\n <button\n id=\"date-range\"\n dropdownToggle\n title=\"{{ 'Date filter' | translate }}\"\n type=\"button\"\n class=\"btn dropdown-toggle d-flex a-i-center c8y-dropdown\"\n >\n <i c8yIcon=\"calendar-o\" class=\"m-r-4 text-primary\"></i>\n <span class=\"text-truncate\">\n <span *ngIf=\"dateFrom\">\n <span class=\"text-label-small\">{{ 'From`date`' | translate }}</span>\n {{ dateFrom | date }}\n </span>\n <span *ngIf=\"dateTo\">\n <span class=\"text-label-small\">{{ 'To`date`' | translate }}</span>\n {{ dateTo | date }}\n </span>\n <em *ngIf=\"!dateFrom && !dateTo\" class=\"text-muted\">\n {{ placeholder }}\n </em>\n </span>\n </button>\n\n <form [formGroup]=\"fgDatePicker\"\n id=\"dropdown-date-range\"\n *dropdownMenu\n class=\"dropdown-menu\">\n <div class=\"dropdown-form p-b-0\">\n <c8y-form-group class=\"form-group-sm\">\n <label for=\"dateFrom\" class=\"text-medium m-b-4\" translate>Date from</label>\n <div class=\"form-group datepicker d-block m-b-0\">\n <input\n id=\"dateFrom\"\n formControlName=\"dateFrom\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date from' | translate }}\"\n bsDatepicker\n [maxDate]=\"dateTo\"\n (bsValueChange)=\"dateFrom = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true }\"\n />\n </div>\n </c8y-form-group>\n <c8y-form-group class=\"form-group form-group-sm \">\n <label for=\"dateTo\" translate>Date to</label>\n <div class=\"form-group datepicker m-l-0 d-block m-b-0 \">\n <input\n name=\"dateTo\"\n id=\"dateTo\"\n formControlName=\"dateTo\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date to' | translate }}\"\n bsDatepicker\n [minDate]=\"dateFrom\"\n (bsValueChange)=\"dateTo = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true }\"\n />\n </div>\n </c8y-form-group>\n </div>\n <div class=\"p-16 d-flex separator-top gap-8\">\n <button\n title=\"{{ 'Clear selection' | translate }}\"\n type=\"button\"\n class=\"btn btn-default btn-sm flex-grow\"\n (click)=\"clearFilter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Clear`selection`' | translate }}\n </button>\n <button\n [disabled]=\"!fgDatePicker.get('dateFrom').value && !fgDatePicker.get('dateTo').value\"\n title=\"{{ 'Apply selection' | translate }}\"\n type=\"submit\"\n class=\"btn btn-primary btn-sm flex-grow\"\n (click)=\"filter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Apply`selection`' | translate }}\n </button>\n </div>\n </form>\n</div>\n" }]
|
|
65
|
+
], template: "<div\n dropdown\n class=\"c8y-child-assets-selector dropdown\"\n #datefilter=\"bs-dropdown\"\n [insideClick]=\"true\"\n placement=\"bottom left\"\n [cdkTrapFocus]=\"datefilter.isOpen\"\n >\n <button\n id=\"date-range\"\n dropdownToggle\n title=\"{{ 'Date filter' | translate }}\"\n type=\"button\"\n class=\"btn dropdown-toggle d-flex a-i-center c8y-dropdown\"\n >\n <i c8yIcon=\"calendar-o\" class=\"m-r-4 text-primary\"></i>\n <span class=\"text-truncate\">\n <span *ngIf=\"dateFrom\">\n <span class=\"text-label-small\">{{ 'From`date`' | translate }}</span>\n {{ dateFrom | date }}\n </span>\n <span *ngIf=\"dateTo\">\n <span class=\"text-label-small\">{{ 'To`date`' | translate }}</span>\n {{ dateTo | date }}\n </span>\n <em *ngIf=\"!dateFrom && !dateTo\" class=\"text-muted\">\n {{ placeholder }}\n </em>\n </span>\n </button>\n\n <form [formGroup]=\"fgDatePicker\"\n id=\"dropdown-date-range\"\n *dropdownMenu\n class=\"dropdown-menu\">\n <div class=\"dropdown-form p-b-0\">\n <c8y-form-group class=\"form-group-sm\">\n <label for=\"dateFrom\" class=\"text-medium m-b-4\" translate>Date from</label>\n <div class=\"form-group datepicker d-block m-b-0\">\n <input\n id=\"dateFrom\"\n formControlName=\"dateFrom\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date from' | translate }}\"\n bsDatepicker\n [maxDate]=\"dateTo\"\n (bsValueChange)=\"dateFrom = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true, dateInputFormat: dateInputFormat }\"\n />\n </div>\n </c8y-form-group>\n <c8y-form-group class=\"form-group form-group-sm \">\n <label for=\"dateTo\" translate>Date to</label>\n <div class=\"form-group datepicker m-l-0 d-block m-b-0 \">\n <input\n name=\"dateTo\"\n id=\"dateTo\"\n formControlName=\"dateTo\"\n class=\"form-control fit-w text-left\"\n placeholder=\"{{ 'Date to' | translate }}\"\n bsDatepicker\n [minDate]=\"dateFrom\"\n (bsValueChange)=\"dateTo = $event\"\n [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true, dateInputFormat: dateInputFormat }\"\n />\n </div>\n </c8y-form-group>\n </div>\n <div class=\"p-16 d-flex separator-top gap-8\">\n <button\n title=\"{{ 'Clear selection' | translate }}\"\n type=\"button\"\n class=\"btn btn-default btn-sm flex-grow\"\n (click)=\"clearFilter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Clear`selection`' | translate }}\n </button>\n <button\n [disabled]=\"!fgDatePicker.get('dateFrom').value && !fgDatePicker.get('dateTo').value\"\n title=\"{{ 'Apply selection' | translate }}\"\n type=\"submit\"\n class=\"btn btn-primary btn-sm flex-grow\"\n (click)=\"filter(); datefilter.isOpen = !datefilter.isOpen\"\n >\n {{ 'Apply`selection`' | translate }}\n </button>\n </div>\n </form>\n</div>\n" }]
|
|
61
66
|
}], propDecorators: { onDateSelected: [{
|
|
62
67
|
type: Output
|
|
63
68
|
}], placeholder: [{
|
|
64
69
|
type: Input
|
|
70
|
+
}], dateInputFormat: [{
|
|
71
|
+
type: Input
|
|
65
72
|
}] } });
|
|
66
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-picker.component.js","sourceRoot":"","sources":["../../../../core/date-picker/date-picker.component.ts","../../../../core/date-picker/date-picker.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAU,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAE1F,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,kBAAkB,EAAE,MAAM,+BAA+B,CAAC;AACnE,OAAO,EAAE,qBAAqB,EAAE,MAAM,iCAAiC,CAAC;AACxE,OAAO,EAAE,iCAAiC,EAAE,MAAM,+CAA+C,CAAC;AAClG,OAAO,EAAE,kBAAkB,EAAE,MAAM,0BAA0B,CAAC;AAC9D,OAAO,EAAE,gBAAgB,EAAE,MAAM,4BAA4B,CAAC;;;;;AAqB9D,MAAM,OAAO,mBAAmB;IAnBhC;QAoBY,mBAAc,GAA8B,IAAI,YAAY,EAAE,CAAC;QAEzE,gBAAW,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAAC;KA4B1C;IAtBC,QAAQ;QACN,IAAI,CAAC,YAAY,GAAG,IAAI,SAAS,CAAC;YAChC,QAAQ,EAAE,IAAI,WAAW,EAAE;YAC3B,MAAM,EAAE,IAAI,WAAW,EAAE;SAC1B,CAAC,CAAC;IACL,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAClD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC;QACxD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,KAAK,CAAC;IACtD,CAAC;IAED,WAAW;QACT,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC;YACzB,QAAQ,EAAE,IAAI;YACd,MAAM,EAAE,IAAI;SACb,CAAC,CAAC;QACH,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC/B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;QACrB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;IACrB,CAAC;+GA9BU,mBAAmB;mGAAnB,mBAAmB,kKCjChC,2mGAyFA,2CDtEI,gBAAgB,ulBAChB,YAAY,4IACZ,aAAa,2EACb,IAAI,4FACJ,WAAW,2pBACX,mBAAmB,gVACnB,kBAAkB,uIAClB,qBAAqB,wEACrB,iCAAiC,mFACjC,kBAAkB,ihBAClB,QAAQ,wCACR,gBAAgB;;4FAGP,mBAAmB;kBAnB/B,SAAS;+BACE,iBAAiB,cAEf,IAAI,WACP;wBACP,gBAAgB;wBAChB,YAAY;wBACZ,aAAa;wBACb,IAAI;wBACJ,WAAW;wBACX,mBAAmB;wBACnB,kBAAkB;wBAClB,qBAAqB;wBACrB,iCAAiC;wBACjC,kBAAkB;wBAClB,QAAQ;wBACR,gBAAgB;qBACjB;8BAGS,cAAc;sBAAvB,MAAM;gBAEP,WAAW;sBADV,KAAK","sourcesContent":["import { Component, EventEmitter, OnInit, Input, Output } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { PickedDates } from './date-picker.model';\nimport { gettext } from '../i18n';\nimport { BsDropdownModule } from 'ngx-bootstrap/dropdown';\nimport { CdkTrapFocus } from '@angular/cdk/a11y';\nimport { IconDirective } from '../common/icon.directive';\nimport { NgIf, DatePipe } from '@angular/common';\nimport { FormGroupComponent } from '../forms/form-group.component';\nimport { C8yTranslateDirective } from '../i18n/c8y-translate.directive';\nimport { RequiredInputPlaceholderDirective } from '../forms/required-input-placeholder.directive';\nimport { BsDatepickerModule } from 'ngx-bootstrap/datepicker';\nimport { C8yTranslatePipe } from '../i18n/c8y-translate.pipe';\n\n@Component({\n  selector: 'c8y-date-picker',\n  templateUrl: './date-picker.component.html',\n  standalone: true,\n  imports: [\n    BsDropdownModule,\n    CdkTrapFocus,\n    IconDirective,\n    NgIf,\n    FormsModule,\n    ReactiveFormsModule,\n    FormGroupComponent,\n    C8yTranslateDirective,\n    RequiredInputPlaceholderDirective,\n    BsDatepickerModule,\n    DatePipe,\n    C8yTranslatePipe\n  ]\n})\nexport class DatePickerComponent implements OnInit {\n  @Output() onDateSelected: EventEmitter<PickedDates> = new EventEmitter();\n  @Input()\n  placeholder = gettext('Filter by date…');\n\n  dateFrom: string;\n  dateTo: string;\n  fgDatePicker: FormGroup;\n\n  ngOnInit() {\n    this.fgDatePicker = new FormGroup({\n      dateFrom: new FormControl(),\n      dateTo: new FormControl()\n    });\n  }\n\n  filter() {\n    this.onDateSelected.emit(this.fgDatePicker.value);\n    this.dateFrom = this.fgDatePicker.get('dateFrom').value;\n    this.dateTo = this.fgDatePicker.get('dateTo').value;\n  }\n\n  clearFilter() {\n    this.fgDatePicker.setValue({\n      dateFrom: null,\n      dateTo: null\n    });\n    this.onDateSelected.emit(null);\n    this.dateFrom = null;\n    this.dateTo = null;\n  }\n}\n","<div\n  dropdown\n  class=\"c8y-child-assets-selector dropdown\"\n  #datefilter=\"bs-dropdown\"\n  [insideClick]=\"true\"\n  placement=\"bottom left\"\n  [cdkTrapFocus]=\"datefilter.isOpen\"\n  >\n  <button\n    id=\"date-range\"\n    dropdownToggle\n    title=\"{{ 'Date filter' | translate }}\"\n    type=\"button\"\n    class=\"btn dropdown-toggle d-flex a-i-center c8y-dropdown\"\n  >\n    <i c8yIcon=\"calendar-o\" class=\"m-r-4 text-primary\"></i>\n    <span class=\"text-truncate\">\n      <span *ngIf=\"dateFrom\">\n        <span class=\"text-label-small\">{{ 'From`date`' | translate }}</span>\n        {{ dateFrom | date }}\n      </span>\n      <span *ngIf=\"dateTo\">\n        <span class=\"text-label-small\">{{ 'To`date`' | translate }}</span>\n        {{ dateTo | date }}\n      </span>\n      <em *ngIf=\"!dateFrom && !dateTo\" class=\"text-muted\">\n        {{ placeholder }}\n      </em>\n    </span>\n  </button>\n\n  <form [formGroup]=\"fgDatePicker\"\n    id=\"dropdown-date-range\"\n    *dropdownMenu\n    class=\"dropdown-menu\">\n    <div class=\"dropdown-form p-b-0\">\n      <c8y-form-group class=\"form-group-sm\">\n        <label for=\"dateFrom\" class=\"text-medium m-b-4\" translate>Date from</label>\n        <div class=\"form-group datepicker d-block m-b-0\">\n          <input\n            id=\"dateFrom\"\n            formControlName=\"dateFrom\"\n            class=\"form-control fit-w text-left\"\n            placeholder=\"{{ 'Date from' | translate }}\"\n            bsDatepicker\n            [maxDate]=\"dateTo\"\n            (bsValueChange)=\"dateFrom = $event\"\n            [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true }\"\n          />\n        </div>\n      </c8y-form-group>\n      <c8y-form-group class=\"form-group form-group-sm \">\n        <label for=\"dateTo\" translate>Date to</label>\n        <div class=\"form-group datepicker m-l-0 d-block m-b-0 \">\n          <input\n            name=\"dateTo\"\n            id=\"dateTo\"\n            formControlName=\"dateTo\"\n            class=\"form-control fit-w text-left\"\n            placeholder=\"{{ 'Date to' | translate }}\"\n            bsDatepicker\n            [minDate]=\"dateFrom\"\n            (bsValueChange)=\"dateTo = $event\"\n            [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true }\"\n          />\n        </div>\n      </c8y-form-group>\n    </div>\n    <div class=\"p-16 d-flex separator-top gap-8\">\n      <button\n        title=\"{{ 'Clear selection' | translate }}\"\n        type=\"button\"\n        class=\"btn btn-default btn-sm flex-grow\"\n        (click)=\"clearFilter(); datefilter.isOpen = !datefilter.isOpen\"\n      >\n        {{ 'Clear`selection`' | translate }}\n      </button>\n      <button\n        [disabled]=\"!fgDatePicker.get('dateFrom').value && !fgDatePicker.get('dateTo').value\"\n        title=\"{{ 'Apply selection' | translate }}\"\n        type=\"submit\"\n        class=\"btn btn-primary btn-sm flex-grow\"\n        (click)=\"filter(); datefilter.isOpen = !datefilter.isOpen\"\n      >\n        {{ 'Apply`selection`' | translate }}\n      </button>\n    </div>\n  </form>\n</div>\n"]}
|
|
73
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-picker.component.js","sourceRoot":"","sources":["../../../../core/date-picker/date-picker.component.ts","../../../../core/date-picker/date-picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, OnInit, Input, Output, inject } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { PickedDates } from './date-picker.model';\nimport { gettext } from '../i18n';\nimport { BsDropdownModule } from 'ngx-bootstrap/dropdown';\nimport { CdkTrapFocus } from '@angular/cdk/a11y';\nimport { IconDirective } from '../common/icon.directive';\nimport { NgIf, DatePipe } from '@angular/common';\nimport { FormGroupComponent } from '../forms/form-group.component';\nimport { C8yTranslateDirective } from '../i18n/c8y-translate.directive';\nimport { RequiredInputPlaceholderDirective } from '../forms/required-input-placeholder.directive';\nimport { BsDatepickerModule } from 'ngx-bootstrap/datepicker';\nimport { C8yTranslatePipe } from '../i18n/c8y-translate.pipe';\nimport { DateFormatService } from '../common/date-format.service';\n\n@Component({\n  selector: 'c8y-date-picker',\n  templateUrl: './date-picker.component.html',\n  standalone: true,\n  imports: [\n    BsDropdownModule,\n    CdkTrapFocus,\n    IconDirective,\n    NgIf,\n    FormsModule,\n    ReactiveFormsModule,\n    FormGroupComponent,\n    C8yTranslateDirective,\n    RequiredInputPlaceholderDirective,\n    BsDatepickerModule,\n    DatePipe,\n    C8yTranslatePipe\n  ]\n})\nexport class DatePickerComponent implements OnInit {\n  @Output() onDateSelected: EventEmitter<PickedDates> = new EventEmitter();\n  @Input()\n  placeholder = gettext('Filter by date…');\n  @Input()\n  dateInputFormat: string;\n\n  dateFrom: string;\n  dateTo: string;\n  fgDatePicker: FormGroup;\n\n  private dateFormatService = inject(DateFormatService);\n\n  ngOnInit() {\n    if (!this.dateInputFormat) {\n      this.dateInputFormat = this.dateFormatService.getDateFormat();\n    }\n    this.fgDatePicker = new FormGroup({\n      dateFrom: new FormControl(),\n      dateTo: new FormControl()\n    });\n  }\n\n  filter() {\n    this.onDateSelected.emit(this.fgDatePicker.value);\n    this.dateFrom = this.fgDatePicker.get('dateFrom').value;\n    this.dateTo = this.fgDatePicker.get('dateTo').value;\n  }\n\n  clearFilter() {\n    this.fgDatePicker.setValue({\n      dateFrom: null,\n      dateTo: null\n    });\n    this.onDateSelected.emit(null);\n    this.dateFrom = null;\n    this.dateTo = null;\n  }\n}\n","<div\n  dropdown\n  class=\"c8y-child-assets-selector dropdown\"\n  #datefilter=\"bs-dropdown\"\n  [insideClick]=\"true\"\n  placement=\"bottom left\"\n  [cdkTrapFocus]=\"datefilter.isOpen\"\n  >\n  <button\n    id=\"date-range\"\n    dropdownToggle\n    title=\"{{ 'Date filter' | translate }}\"\n    type=\"button\"\n    class=\"btn dropdown-toggle d-flex a-i-center c8y-dropdown\"\n  >\n    <i c8yIcon=\"calendar-o\" class=\"m-r-4 text-primary\"></i>\n    <span class=\"text-truncate\">\n      <span *ngIf=\"dateFrom\">\n        <span class=\"text-label-small\">{{ 'From`date`' | translate }}</span>\n        {{ dateFrom | date }}\n      </span>\n      <span *ngIf=\"dateTo\">\n        <span class=\"text-label-small\">{{ 'To`date`' | translate }}</span>\n        {{ dateTo | date }}\n      </span>\n      <em *ngIf=\"!dateFrom && !dateTo\" class=\"text-muted\">\n        {{ placeholder }}\n      </em>\n    </span>\n  </button>\n\n  <form [formGroup]=\"fgDatePicker\"\n    id=\"dropdown-date-range\"\n    *dropdownMenu\n    class=\"dropdown-menu\">\n    <div class=\"dropdown-form p-b-0\">\n      <c8y-form-group class=\"form-group-sm\">\n        <label for=\"dateFrom\" class=\"text-medium m-b-4\" translate>Date from</label>\n        <div class=\"form-group datepicker d-block m-b-0\">\n          <input\n            id=\"dateFrom\"\n            formControlName=\"dateFrom\"\n            class=\"form-control fit-w text-left\"\n            placeholder=\"{{ 'Date from' | translate }}\"\n            bsDatepicker\n            [maxDate]=\"dateTo\"\n            (bsValueChange)=\"dateFrom = $event\"\n            [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true, dateInputFormat: dateInputFormat }\"\n          />\n        </div>\n      </c8y-form-group>\n      <c8y-form-group class=\"form-group form-group-sm \">\n        <label for=\"dateTo\" translate>Date to</label>\n        <div class=\"form-group datepicker m-l-0 d-block m-b-0 \">\n          <input\n            name=\"dateTo\"\n            id=\"dateTo\"\n            formControlName=\"dateTo\"\n            class=\"form-control fit-w text-left\"\n            placeholder=\"{{ 'Date to' | translate }}\"\n            bsDatepicker\n            [minDate]=\"dateFrom\"\n            (bsValueChange)=\"dateTo = $event\"\n            [bsConfig]=\"{ customTodayClass: 'today', returnFocusToInput: true, dateInputFormat: dateInputFormat }\"\n          />\n        </div>\n      </c8y-form-group>\n    </div>\n    <div class=\"p-16 d-flex separator-top gap-8\">\n      <button\n        title=\"{{ 'Clear selection' | translate }}\"\n        type=\"button\"\n        class=\"btn btn-default btn-sm flex-grow\"\n        (click)=\"clearFilter(); datefilter.isOpen = !datefilter.isOpen\"\n      >\n        {{ 'Clear`selection`' | translate }}\n      </button>\n      <button\n        [disabled]=\"!fgDatePicker.get('dateFrom').value && !fgDatePicker.get('dateTo').value\"\n        title=\"{{ 'Apply selection' | translate }}\"\n        type=\"submit\"\n        class=\"btn btn-primary btn-sm flex-grow\"\n        (click)=\"filter(); datefilter.isOpen = !datefilter.isOpen\"\n      >\n        {{ 'Apply`selection`' | translate }}\n      </button>\n    </div>\n  </form>\n</div>\n"]}
|