@tuki-io/tuki-widgets 0.0.166 → 0.0.167

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/esm2020/di2mt/shared/services/api.service.mjs +3 -3
  2. package/esm2020/user-creation/public-api.mjs +13 -0
  3. package/esm2020/user-creation/src/app.constants.mjs +16 -0
  4. package/esm2020/user-creation/src/classes/site.mjs +53 -0
  5. package/esm2020/user-creation/src/shared/api/api.service.mjs +86 -0
  6. package/esm2020/user-creation/src/shared/material.module.mjs +76 -0
  7. package/esm2020/user-creation/src/shared/services/user-creation-api.service.mjs +112 -0
  8. package/esm2020/user-creation/src/shared/services/utils.mjs +19 -0
  9. package/esm2020/user-creation/src/shared/shared.module.mjs +33 -0
  10. package/esm2020/user-creation/src/user-creation.module.mjs +69 -0
  11. package/esm2020/user-creation/src/widgets/user-creation-wizard/components/user-details-step/user-details-step.component.mjs +23 -0
  12. package/esm2020/user-creation/src/widgets/user-creation-wizard/components/user-overview-step/user-overview-step.component.mjs +19 -0
  13. package/esm2020/user-creation/src/widgets/user-creation-wizard/components/user-template-step/user-template-step.component.mjs +114 -0
  14. package/esm2020/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.component.mjs +60 -0
  15. package/esm2020/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.service.mjs +273 -0
  16. package/esm2020/user-creation/tuki-io-tuki-widgets-user-creation.mjs +5 -0
  17. package/esm2020/users-list/src/components/table-toolbar/table-toolbar.component.mjs +9 -3
  18. package/esm2020/users-list/src/users-list.component.mjs +9 -3
  19. package/fesm2015/tuki-io-tuki-widgets-di2mt.mjs +2 -2
  20. package/fesm2015/tuki-io-tuki-widgets-di2mt.mjs.map +1 -1
  21. package/fesm2015/tuki-io-tuki-widgets-user-creation.mjs +919 -0
  22. package/fesm2015/tuki-io-tuki-widgets-user-creation.mjs.map +1 -0
  23. package/fesm2015/tuki-io-tuki-widgets-users-list.mjs +16 -4
  24. package/fesm2015/tuki-io-tuki-widgets-users-list.mjs.map +1 -1
  25. package/fesm2020/tuki-io-tuki-widgets-di2mt.mjs +2 -2
  26. package/fesm2020/tuki-io-tuki-widgets-di2mt.mjs.map +1 -1
  27. package/fesm2020/tuki-io-tuki-widgets-user-creation.mjs +915 -0
  28. package/fesm2020/tuki-io-tuki-widgets-user-creation.mjs.map +1 -0
  29. package/fesm2020/tuki-io-tuki-widgets-users-list.mjs +16 -4
  30. package/fesm2020/tuki-io-tuki-widgets-users-list.mjs.map +1 -1
  31. package/package.json +9 -1
  32. package/user-creation/index.d.ts +5 -0
  33. package/user-creation/public-api.d.ts +2 -0
  34. package/user-creation/src/app.constants.d.ts +15 -0
  35. package/user-creation/src/classes/site.d.ts +203 -0
  36. package/user-creation/src/shared/api/api.service.d.ts +20 -0
  37. package/user-creation/src/shared/material.module.d.ts +15 -0
  38. package/user-creation/src/shared/services/user-creation-api.service.d.ts +24 -0
  39. package/user-creation/src/shared/services/utils.d.ts +1 -0
  40. package/user-creation/src/shared/shared.module.d.ts +10 -0
  41. package/user-creation/src/user-creation.module.d.ts +17 -0
  42. package/user-creation/src/widgets/user-creation-wizard/components/user-details-step/user-details-step.component.d.ts +8 -0
  43. package/user-creation/src/widgets/user-creation-wizard/components/user-overview-step/user-overview-step.component.d.ts +11 -0
  44. package/user-creation/src/widgets/user-creation-wizard/components/user-template-step/user-template-step.component.d.ts +37 -0
  45. package/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.component.d.ts +18 -0
  46. package/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.service.d.ts +87 -0
  47. package/users-list/src/components/table-toolbar/table-toolbar.component.d.ts +3 -1
  48. package/users-list/src/users-list.component.d.ts +3 -1
@@ -0,0 +1,273 @@
1
+ import { inject, Injectable } from "@angular/core";
2
+ import { UserCreationApiService } from "../../shared/services/user-creation-api.service";
3
+ import * as i0 from "@angular/core";
4
+ export class UserCreationWizardService {
5
+ constructor() {
6
+ this.api = inject(UserCreationApiService);
7
+ this.ldapTableColumns = ['userId', 'firstName', 'lastName', 'email', 'add'];
8
+ this.userCreationTypes = ['CUCM', 'MT'];
9
+ this.siteOptions = [];
10
+ this.templateOptions = [];
11
+ this.lineOptions = [];
12
+ this.routePartitionOptions = [];
13
+ this.deviceRows = [];
14
+ this.ldapUsers = [];
15
+ this.overviewUsers = [];
16
+ this.overviewLines = [];
17
+ this.overviewDevices = [];
18
+ this.userDetailsForm = {
19
+ userId: '',
20
+ firstName: '',
21
+ lastName: '',
22
+ email: ''
23
+ };
24
+ this.lineSelections = [];
25
+ this.deviceEntries = [];
26
+ this.selectedSite = null;
27
+ this.selectedTemplate = null;
28
+ this.selectedUserCreationType = 'CUCM';
29
+ this.loadPlaceholderData();
30
+ }
31
+ setSelectedSite(site) {
32
+ this.selectedSite = site;
33
+ }
34
+ setSelectedTemplate(template) {
35
+ this.selectedTemplate = template;
36
+ this.rebuildOverviewUsers();
37
+ }
38
+ resetTemplateDrivenData() {
39
+ this.lineOptions = [];
40
+ this.routePartitionOptions = [];
41
+ this.lineSelections = [];
42
+ this.deviceEntries = [];
43
+ this.deviceRows = [];
44
+ this.rebuildOverviewLines();
45
+ this.rebuildOverviewDevices();
46
+ }
47
+ applyTemplateTokenPayload(payload) {
48
+ const template = this.extractTemplateFromTokenPayload(payload);
49
+ const endUser = template?.endUser || {};
50
+ const linesFromTemplate = Array.isArray(endUser.lines) ? endUser.lines : [];
51
+ const devicesFromTemplate = Array.isArray(endUser.devices) ? endUser.devices : [];
52
+ this.userDetailsForm = {
53
+ userId: this.valueToString(endUser.userid),
54
+ firstName: this.valueToString(endUser.firstName),
55
+ lastName: this.valueToString(endUser.lastName),
56
+ email: this.valueToString(endUser.email)
57
+ };
58
+ const mappedLineOptions = linesFromTemplate
59
+ .map(line => ({
60
+ number: this.valueToString(line?.directoryNumber?.directoryNumber),
61
+ did: this.valueToString(line?.didPatterns?.[0]?.pattern || line?.directoryNumber?.directoryNumber),
62
+ css: this.valueToString(typeof line?.callingSearchSpace === 'string'
63
+ ? line.callingSearchSpace
64
+ : line?.callingSearchSpace?.name),
65
+ description: this.valueToString(line?.description)
66
+ }))
67
+ .filter(line => line.number.trim().length > 0);
68
+ if (mappedLineOptions.length) {
69
+ this.lineOptions = this.uniqueByNumber(mappedLineOptions);
70
+ this.lineSelections = this.lineOptions.map(line => ({
71
+ lineNumber: line.number,
72
+ routePartitionName: linesFromTemplate.find(source => (source?.directoryNumber?.directoryNumber || '') === line.number)?.directoryNumber?.routePartitionName || null
73
+ }));
74
+ }
75
+ else {
76
+ const lineAssociations = devicesFromTemplate.flatMap(device => device.lineAssociations || []);
77
+ const associatedLineOptions = lineAssociations
78
+ .map(association => {
79
+ const number = association?.directoryNumber?.directoryNumber || '';
80
+ return {
81
+ number,
82
+ did: number,
83
+ css: '',
84
+ description: ''
85
+ };
86
+ })
87
+ .filter(line => line.number.trim().length > 0);
88
+ this.lineOptions = this.uniqueByNumber(associatedLineOptions);
89
+ this.lineSelections = this.lineOptions.map(line => ({
90
+ lineNumber: line.number,
91
+ routePartitionName: lineAssociations.find(source => (source?.directoryNumber?.directoryNumber || '') === line.number)?.directoryNumber?.routePartitionName || null
92
+ }));
93
+ }
94
+ const routePartitions = this.lineSelections
95
+ .map(line => line.routePartitionName)
96
+ .filter((value) => !!value && value.trim().length > 0);
97
+ this.routePartitionOptions = this.uniqueStrings(routePartitions);
98
+ this.deviceEntries = devicesFromTemplate.map(device => ({
99
+ deviceType: this.valueToString(device.deviceType),
100
+ protocol: this.valueToString(device.protocol),
101
+ buttonTemplate: this.valueToString(device.buttonTemplate),
102
+ name: this.valueToString(device.name)
103
+ }));
104
+ this.deviceRows = this.deviceEntries.map((_, index) => index + 1);
105
+ this.rebuildOverviewUsers();
106
+ this.rebuildOverviewLines();
107
+ this.rebuildOverviewDevices();
108
+ }
109
+ setSelectedUserCreationType(type) {
110
+ this.selectedUserCreationType = type;
111
+ }
112
+ setUserDetailsField(field, value) {
113
+ this.userDetailsForm = {
114
+ ...this.userDetailsForm,
115
+ [field]: value
116
+ };
117
+ this.rebuildOverviewUsers();
118
+ }
119
+ setLineSelection(index, lineNumber) {
120
+ if (!this.lineSelections[index]) {
121
+ return;
122
+ }
123
+ this.lineSelections[index] = {
124
+ ...this.lineSelections[index],
125
+ lineNumber
126
+ };
127
+ this.rebuildOverviewLines();
128
+ this.rebuildOverviewDevices();
129
+ }
130
+ setRoutePartitionSelection(index, routePartitionName) {
131
+ if (!this.lineSelections[index]) {
132
+ return;
133
+ }
134
+ this.lineSelections[index] = {
135
+ ...this.lineSelections[index],
136
+ routePartitionName
137
+ };
138
+ this.rebuildOverviewLines();
139
+ }
140
+ setDeviceName(index, name) {
141
+ if (!this.deviceEntries[index]) {
142
+ return;
143
+ }
144
+ this.deviceEntries[index] = {
145
+ ...this.deviceEntries[index],
146
+ name
147
+ };
148
+ this.rebuildOverviewDevices();
149
+ }
150
+ // Placeholder bootstrap until backend APIs are connected.
151
+ loadPlaceholderData() {
152
+ this.siteOptions = ['Site 1', 'Site 2', 'Site 3'];
153
+ this.templateOptions = ['User 8851 Office'];
154
+ this.lineOptions = [
155
+ {
156
+ number: '557046',
157
+ did: '+13125557046',
158
+ css: 'Atlanta_internal_css',
159
+ description: ''
160
+ }
161
+ ];
162
+ this.routePartitionOptions = ['Atlanta_internal_PT', 'Beijing_national_PT'];
163
+ this.deviceRows = [1, 2];
164
+ this.ldapUsers = [
165
+ { userId: 'John.Doe', firstName: 'John', lastName: 'Doe', email: 'john.doe@car.west.io' },
166
+ { userId: 'Jane.Smith', firstName: 'Jane', lastName: 'Smith', email: 'jane.smith@car.west.io' }
167
+ ];
168
+ this.lineSelections = [
169
+ {
170
+ lineNumber: this.lineOptions[0]?.number || null,
171
+ routePartitionName: this.routePartitionOptions[0] || null
172
+ }
173
+ ];
174
+ this.deviceEntries = [
175
+ {
176
+ deviceType: 'Cisco 8851',
177
+ protocol: 'SIP',
178
+ buttonTemplate: 'Standard 8851 SIP',
179
+ name: ''
180
+ },
181
+ {
182
+ deviceType: 'Cisco Unified Client Services Framework',
183
+ protocol: 'SIP',
184
+ buttonTemplate: 'Standard Client Services Framework',
185
+ name: ''
186
+ }
187
+ ];
188
+ this.selectedTemplate = this.selectedTemplate ?? null;
189
+ this.rebuildOverviewUsers();
190
+ this.rebuildOverviewLines();
191
+ this.rebuildOverviewDevices();
192
+ }
193
+ rebuildOverviewUsers() {
194
+ const userTemplate = this.selectedTemplate || '';
195
+ const hasUserData = Object.values(this.userDetailsForm).some(value => value.trim().length > 0);
196
+ this.overviewUsers = hasUserData
197
+ ? [{
198
+ userId: this.userDetailsForm.userId,
199
+ firstName: this.userDetailsForm.firstName,
200
+ lastName: this.userDetailsForm.lastName,
201
+ email: this.userDetailsForm.email,
202
+ userTemplate
203
+ }]
204
+ : [];
205
+ }
206
+ rebuildOverviewLines() {
207
+ this.overviewLines = this.lineSelections
208
+ .map(selection => {
209
+ if (!selection?.lineNumber) {
210
+ return null;
211
+ }
212
+ const selectedLine = this.lineOptions.find(line => line.number === selection.lineNumber);
213
+ if (!selectedLine) {
214
+ return null;
215
+ }
216
+ return {
217
+ number: selectedLine.number,
218
+ did: selectedLine.did,
219
+ routePartitionName: selection.routePartitionName || this.routePartitionOptions[0] || '',
220
+ css: selectedLine.css,
221
+ description: selectedLine.description
222
+ };
223
+ })
224
+ .filter((line) => !!line);
225
+ }
226
+ rebuildOverviewDevices() {
227
+ const selectedLineNumbers = this.overviewLines.map(line => line.number).join(', ');
228
+ this.overviewDevices = this.deviceEntries
229
+ .filter(device => device.name.trim().length > 0 || device.deviceType.trim().length > 0)
230
+ .map(device => ({
231
+ name: device.name,
232
+ deviceType: device.deviceType,
233
+ lines: selectedLineNumbers,
234
+ description: `${device.protocol} / ${device.buttonTemplate}`
235
+ }));
236
+ }
237
+ extractTemplateFromTokenPayload(payload) {
238
+ if (!payload || typeof payload !== 'object') {
239
+ return null;
240
+ }
241
+ if (Array.isArray(payload)) {
242
+ return payload[0] || null;
243
+ }
244
+ const firstKey = Object.keys(payload)[0];
245
+ return firstKey ? payload[firstKey] : null;
246
+ }
247
+ uniqueByNumber(lines) {
248
+ const seen = new Set();
249
+ return lines.filter(line => {
250
+ if (seen.has(line.number)) {
251
+ return false;
252
+ }
253
+ seen.add(line.number);
254
+ return true;
255
+ });
256
+ }
257
+ uniqueStrings(items) {
258
+ return Array.from(new Set(items));
259
+ }
260
+ valueToString(value) {
261
+ return typeof value === 'string' ? value : '';
262
+ }
263
+ // TODO: replace placeholders with real API integration.
264
+ loadDataFromApi() {
265
+ // this.api.getWizardBootstrapData().subscribe(...)
266
+ }
267
+ }
268
+ UserCreationWizardService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: UserCreationWizardService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
269
+ UserCreationWizardService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: UserCreationWizardService });
270
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: UserCreationWizardService, decorators: [{
271
+ type: Injectable
272
+ }], ctorParameters: function () { return []; } });
273
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user-creation-wizard.service.js","sourceRoot":"","sources":["../../../../../../../../projects/tuki/widgets/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from \"@angular/core\";\r\nimport { UserCreationApiService } from \"../../shared/services/user-creation-api.service\";\r\n\r\nexport interface LdapUserRow {\r\n    userId: string;\r\n    firstName: string;\r\n    lastName: string;\r\n    email: string;\r\n}\r\n\r\nexport interface OverviewUserRow extends LdapUserRow {\r\n    userTemplate: string;\r\n}\r\n\r\nexport interface OverviewLineRow {\r\n    number: string;\r\n    did: string;\r\n    routePartitionName: string;\r\n    css: string;\r\n    description: string;\r\n}\r\n\r\nexport interface OverviewDeviceRow {\r\n    name: string;\r\n    deviceType: string;\r\n    lines: string;\r\n    description: string;\r\n}\r\n\r\nexport type UserCreationType = 'CUCM' | 'MT';\r\n\r\nexport interface UserDetailsForm {\r\n    userId: string;\r\n    firstName: string;\r\n    lastName: string;\r\n    email: string;\r\n}\r\n\r\nexport interface LineOption {\r\n    number: string;\r\n    did: string;\r\n    css: string;\r\n    description: string;\r\n}\r\n\r\nexport interface SelectedLineRow {\r\n    lineNumber: string | null;\r\n    routePartitionName: string | null;\r\n}\r\n\r\nexport interface DeviceEntry {\r\n    deviceType: string;\r\n    protocol: string;\r\n    buttonTemplate: string;\r\n    name: string;\r\n}\r\n\r\ninterface TemplateLineLike {\r\n    directoryNumber?: {\r\n        directoryNumber?: string;\r\n        routePartitionName?: string | null;\r\n    };\r\n    description?: string | null;\r\n    callingSearchSpace?: string | { name?: string | null } | null;\r\n    didPatterns?: Array<{ pattern?: string | null }> | null;\r\n}\r\n\r\ninterface TemplateDeviceLike {\r\n    deviceType?: string | null;\r\n    protocol?: string | null;\r\n    buttonTemplate?: string | null;\r\n    name?: string | null;\r\n    lineAssociations?: Array<{\r\n        directoryNumber?: {\r\n            directoryNumber?: string;\r\n            routePartitionName?: string | null;\r\n        };\r\n    }>;\r\n}\r\n\r\ninterface TemplateEndUserLike {\r\n    userid?: string | null;\r\n    firstName?: string | null;\r\n    lastName?: string | null;\r\n    email?: string | null;\r\n    lines?: TemplateLineLike[] | null;\r\n    devices?: TemplateDeviceLike[] | null;\r\n}\r\n\r\n@Injectable()\r\nexport class UserCreationWizardService {\r\n    private readonly api = inject(UserCreationApiService);\r\n\r\n    readonly ldapTableColumns: string[] = ['userId', 'firstName', 'lastName', 'email', 'add'];\r\n    readonly userCreationTypes: UserCreationType[] = ['CUCM', 'MT'];\r\n\r\n    siteOptions: string[] = [];\r\n    templateOptions: string[] = [];\r\n    lineOptions: LineOption[] = [];\r\n    routePartitionOptions: string[] = [];\r\n    deviceRows: number[] = [];\r\n    ldapUsers: LdapUserRow[] = [];\r\n    overviewUsers: OverviewUserRow[] = [];\r\n    overviewLines: OverviewLineRow[] = [];\r\n    overviewDevices: OverviewDeviceRow[] = [];\r\n    userDetailsForm: UserDetailsForm = {\r\n        userId: '',\r\n        firstName: '',\r\n        lastName: '',\r\n        email: ''\r\n    };\r\n    lineSelections: SelectedLineRow[] = [];\r\n    deviceEntries: DeviceEntry[] = [];\r\n\r\n    selectedSite: string | null = null;\r\n    selectedTemplate: string | null = null;\r\n    selectedUserCreationType: UserCreationType = 'CUCM';\r\n\r\n    constructor() {\r\n        this.loadPlaceholderData();\r\n    }\r\n\r\n    setSelectedSite(site: string | null): void {\r\n        this.selectedSite = site;\r\n    }\r\n\r\n    setSelectedTemplate(template: string | null): void {\r\n        this.selectedTemplate = template;\r\n        this.rebuildOverviewUsers();\r\n    }\r\n\r\n    resetTemplateDrivenData(): void {\r\n        this.lineOptions = [];\r\n        this.routePartitionOptions = [];\r\n        this.lineSelections = [];\r\n        this.deviceEntries = [];\r\n        this.deviceRows = [];\r\n        this.rebuildOverviewLines();\r\n        this.rebuildOverviewDevices();\r\n    }\r\n\r\n    applyTemplateTokenPayload(payload: any): void {\r\n        const template = this.extractTemplateFromTokenPayload(payload);\r\n        const endUser: TemplateEndUserLike = template?.endUser || {};\r\n        const linesFromTemplate: TemplateLineLike[] = Array.isArray(endUser.lines) ? endUser.lines : [];\r\n        const devicesFromTemplate: TemplateDeviceLike[] = Array.isArray(endUser.devices) ? endUser.devices : [];\r\n\r\n        this.userDetailsForm = {\r\n            userId: this.valueToString(endUser.userid),\r\n            firstName: this.valueToString(endUser.firstName),\r\n            lastName: this.valueToString(endUser.lastName),\r\n            email: this.valueToString(endUser.email)\r\n        };\r\n\r\n        const mappedLineOptions: LineOption[] = linesFromTemplate\r\n            .map(line => ({\r\n                number: this.valueToString(line?.directoryNumber?.directoryNumber),\r\n                did: this.valueToString(line?.didPatterns?.[0]?.pattern || line?.directoryNumber?.directoryNumber),\r\n                css: this.valueToString(\r\n                    typeof line?.callingSearchSpace === 'string'\r\n                        ? line.callingSearchSpace\r\n                        : line?.callingSearchSpace?.name\r\n                ),\r\n                description: this.valueToString(line?.description)\r\n            }))\r\n            .filter(line => line.number.trim().length > 0);\r\n\r\n        if (mappedLineOptions.length) {\r\n            this.lineOptions = this.uniqueByNumber(mappedLineOptions);\r\n            this.lineSelections = this.lineOptions.map(line => ({\r\n                lineNumber: line.number,\r\n                routePartitionName: linesFromTemplate.find(\r\n                    source => (source?.directoryNumber?.directoryNumber || '') === line.number\r\n                )?.directoryNumber?.routePartitionName || null\r\n            }));\r\n        } else {\r\n            const lineAssociations = devicesFromTemplate.flatMap(device => device.lineAssociations || []);\r\n            const associatedLineOptions: LineOption[] = lineAssociations\r\n                .map(association => {\r\n                    const number = association?.directoryNumber?.directoryNumber || '';\r\n                    return {\r\n                        number,\r\n                        did: number,\r\n                        css: '',\r\n                        description: ''\r\n                    };\r\n                })\r\n                .filter(line => line.number.trim().length > 0);\r\n\r\n            this.lineOptions = this.uniqueByNumber(associatedLineOptions);\r\n            this.lineSelections = this.lineOptions.map(line => ({\r\n                lineNumber: line.number,\r\n                routePartitionName: lineAssociations.find(\r\n                    source => (source?.directoryNumber?.directoryNumber || '') === line.number\r\n                )?.directoryNumber?.routePartitionName || null\r\n            }));\r\n        }\r\n\r\n        const routePartitions = this.lineSelections\r\n            .map(line => line.routePartitionName)\r\n            .filter((value): value is string => !!value && value.trim().length > 0);\r\n        this.routePartitionOptions = this.uniqueStrings(routePartitions);\r\n\r\n        this.deviceEntries = devicesFromTemplate.map(device => ({\r\n            deviceType: this.valueToString(device.deviceType),\r\n            protocol: this.valueToString(device.protocol),\r\n            buttonTemplate: this.valueToString(device.buttonTemplate),\r\n            name: this.valueToString(device.name)\r\n        }));\r\n        this.deviceRows = this.deviceEntries.map((_, index) => index + 1);\r\n\r\n        this.rebuildOverviewUsers();\r\n        this.rebuildOverviewLines();\r\n        this.rebuildOverviewDevices();\r\n    }\r\n\r\n    setSelectedUserCreationType(type: UserCreationType): void {\r\n        this.selectedUserCreationType = type;\r\n    }\r\n\r\n    setUserDetailsField(field: keyof UserDetailsForm, value: string): void {\r\n        this.userDetailsForm = {\r\n            ...this.userDetailsForm,\r\n            [field]: value\r\n        };\r\n        this.rebuildOverviewUsers();\r\n    }\r\n\r\n    setLineSelection(index: number, lineNumber: string | null): void {\r\n        if (!this.lineSelections[index]) {\r\n            return;\r\n        }\r\n        this.lineSelections[index] = {\r\n            ...this.lineSelections[index],\r\n            lineNumber\r\n        };\r\n        this.rebuildOverviewLines();\r\n        this.rebuildOverviewDevices();\r\n    }\r\n\r\n    setRoutePartitionSelection(index: number, routePartitionName: string | null): void {\r\n        if (!this.lineSelections[index]) {\r\n            return;\r\n        }\r\n        this.lineSelections[index] = {\r\n            ...this.lineSelections[index],\r\n            routePartitionName\r\n        };\r\n        this.rebuildOverviewLines();\r\n    }\r\n\r\n    setDeviceName(index: number, name: string): void {\r\n        if (!this.deviceEntries[index]) {\r\n            return;\r\n        }\r\n        this.deviceEntries[index] = {\r\n            ...this.deviceEntries[index],\r\n            name\r\n        };\r\n        this.rebuildOverviewDevices();\r\n    }\r\n\r\n    // Placeholder bootstrap until backend APIs are connected.\r\n    loadPlaceholderData(): void {\r\n        this.siteOptions = ['Site 1', 'Site 2', 'Site 3'];\r\n        this.templateOptions = ['User 8851 Office'];\r\n        this.lineOptions = [\r\n            {\r\n                number: '557046',\r\n                did: '+13125557046',\r\n                css: 'Atlanta_internal_css',\r\n                description: ''\r\n            }\r\n        ];\r\n        this.routePartitionOptions = ['Atlanta_internal_PT', 'Beijing_national_PT'];\r\n        this.deviceRows = [1, 2];\r\n        this.ldapUsers = [\r\n            { userId: 'John.Doe', firstName: 'John', lastName: 'Doe', email: 'john.doe@car.west.io' },\r\n            { userId: 'Jane.Smith', firstName: 'Jane', lastName: 'Smith', email: 'jane.smith@car.west.io' }\r\n        ];\r\n        this.lineSelections = [\r\n            {\r\n                lineNumber: this.lineOptions[0]?.number || null,\r\n                routePartitionName: this.routePartitionOptions[0] || null\r\n            }\r\n        ];\r\n        this.deviceEntries = [\r\n            {\r\n                deviceType: 'Cisco 8851',\r\n                protocol: 'SIP',\r\n                buttonTemplate: 'Standard 8851 SIP',\r\n                name: ''\r\n            },\r\n            {\r\n                deviceType: 'Cisco Unified Client Services Framework',\r\n                protocol: 'SIP',\r\n                buttonTemplate: 'Standard Client Services Framework',\r\n                name: ''\r\n            }\r\n        ];\r\n        this.selectedTemplate = this.selectedTemplate ?? null;\r\n        this.rebuildOverviewUsers();\r\n        this.rebuildOverviewLines();\r\n        this.rebuildOverviewDevices();\r\n    }\r\n\r\n    private rebuildOverviewUsers(): void {\r\n        const userTemplate = this.selectedTemplate || '';\r\n        const hasUserData = Object.values(this.userDetailsForm).some(value => value.trim().length > 0);\r\n        this.overviewUsers = hasUserData\r\n            ? [{\r\n                userId: this.userDetailsForm.userId,\r\n                firstName: this.userDetailsForm.firstName,\r\n                lastName: this.userDetailsForm.lastName,\r\n                email: this.userDetailsForm.email,\r\n                userTemplate\r\n            }]\r\n            : [];\r\n    }\r\n\r\n    private rebuildOverviewLines(): void {\r\n        this.overviewLines = this.lineSelections\r\n            .map(selection => {\r\n                if (!selection?.lineNumber) {\r\n                    return null;\r\n                }\r\n                const selectedLine = this.lineOptions.find(line => line.number === selection.lineNumber);\r\n                if (!selectedLine) {\r\n                    return null;\r\n                }\r\n                return {\r\n                    number: selectedLine.number,\r\n                    did: selectedLine.did,\r\n                    routePartitionName: selection.routePartitionName || this.routePartitionOptions[0] || '',\r\n                    css: selectedLine.css,\r\n                    description: selectedLine.description\r\n                };\r\n            })\r\n            .filter((line): line is OverviewLineRow => !!line);\r\n    }\r\n\r\n    private rebuildOverviewDevices(): void {\r\n        const selectedLineNumbers = this.overviewLines.map(line => line.number).join(', ');\r\n        this.overviewDevices = this.deviceEntries\r\n            .filter(device => device.name.trim().length > 0 || device.deviceType.trim().length > 0)\r\n            .map(device => ({\r\n                name: device.name,\r\n                deviceType: device.deviceType,\r\n                lines: selectedLineNumbers,\r\n                description: `${device.protocol} / ${device.buttonTemplate}`\r\n            }));\r\n    }\r\n\r\n    private extractTemplateFromTokenPayload(payload: any): any {\r\n        if (!payload || typeof payload !== 'object') {\r\n            return null;\r\n        }\r\n        if (Array.isArray(payload)) {\r\n            return payload[0] || null;\r\n        }\r\n        const firstKey = Object.keys(payload)[0];\r\n        return firstKey ? payload[firstKey] : null;\r\n    }\r\n\r\n    private uniqueByNumber(lines: LineOption[]): LineOption[] {\r\n        const seen = new Set<string>();\r\n        return lines.filter(line => {\r\n            if (seen.has(line.number)) {\r\n                return false;\r\n            }\r\n            seen.add(line.number);\r\n            return true;\r\n        });\r\n    }\r\n\r\n    private uniqueStrings(items: string[]): string[] {\r\n        return Array.from(new Set(items));\r\n    }\r\n\r\n    private valueToString(value: unknown): string {\r\n        return typeof value === 'string' ? value : '';\r\n    }\r\n\r\n    // TODO: replace placeholders with real API integration.\r\n    loadDataFromApi(): void {\r\n        // this.api.getWizardBootstrapData().subscribe(...)\r\n    }\r\n}"]}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHVraS1pby10dWtpLXdpZGdldHMtdXNlci1jcmVhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R1a2kvd2lkZ2V0cy91c2VyLWNyZWF0aW9uL3R1a2ktaW8tdHVraS13aWRnZXRzLXVzZXItY3JlYXRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcbiJdfQ==
@@ -5,6 +5,7 @@ export class TableToolbarComponent {
5
5
  constructor() {
6
6
  this.searchChange = new EventEmitter();
7
7
  this.searchByType = new EventEmitter();
8
+ this.addUser = new EventEmitter();
8
9
  }
9
10
  onSearchInputChange(event) {
10
11
  const value = event.target.value.trim().toLowerCase() || '';
@@ -16,17 +17,22 @@ export class TableToolbarComponent {
16
17
  const value = event.target.value;
17
18
  this.searchByType.emit(value);
18
19
  }
20
+ onAddUser() {
21
+ this.addUser.emit();
22
+ }
19
23
  }
20
24
  TableToolbarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableToolbarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
21
- TableToolbarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: TableToolbarComponent, selector: "tk-users-table-toolbar", inputs: { totalUsersCount: "totalUsersCount" }, outputs: { searchChange: "searchChange", searchByType: "searchByType" }, ngImport: i0, template: "<header style=\"margin-bottom: 1rem;\">\r\n <div style=\"display: flex; align-items: center; justify-content: space-between;\">\r\n <div class=\"collection-header-left\">\r\n <div style=\"display: flex; align-items: center; gap: 12px;\">\r\n <!-- Search Input -->\r\n <div style=\"position: relative; width: fit-content; min-width: 200px;\">\r\n <input (input)=\"onSearchInputChange($event)\" type=\"text\" placeholder=\"Search by display name, email\" style=\"width: 100%; height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 12px 6px 44px; border: 1px solid #00000080; background: #fff; color: #222; outline: none; box-sizing: border-box; min-width: 280px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">search</i>\r\n </span>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Filter Select -->\r\n <div style=\"display: flex; align-items: center; position: relative; width: fit-content; min-width: 200px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">filter_list</i>\r\n </span>\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 44px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Filter</option>\r\n </select>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Location Select -->\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Select a Location</option>\r\n </select>\r\n\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\"\r\n (change)=\"onTypeChange($event)\">\r\n <option value=\"null\" disabled selected hidden>Select type</option>\r\n <option value=\"\">All</option>\r\n <option value=\"CUCM\">Dedicated Instance</option>\r\n <option value=\"WEBEX\">Multi-Tenant</option>\r\n <option value=\"ONPREM\">On Prem</option>\r\n </select>\r\n <span style=\"margin-left: 8px; color: #757575; font-size: 1rem; white-space: nowrap;\">{{ totalUsersCount }} users</span>\r\n </div>\r\n </div>\r\n <div style=\"display: flex; gap: 1rem;\">\r\n <button class=\"btn si\">\r\n Send invitations\r\n </button>\r\n <button class=\"btn mu\">\r\n Manage users\r\n </button>\r\n </div>\r\n </div>\r\n</header>", styles: [".btn{position:relative;display:inline-flex;gap:6px;align-items:center;justify-content:center;overflow:hidden!important;width:min-content;max-width:100%;padding:0rem .75rem;border:1px solid transparent;border-radius:6.25rem;font-weight:500;font-size:14px;text-overflow:ellipsis;white-space:nowrap;transition:all .1s cubic-bezier(.25,.1,.25,1);height:2rem;cursor:pointer}.btn:hover{background-color:#0000000d}.si{border-color:#0000004d;background-color:#0000;color:#000000f2}.mu{color:#fff;background-color:#000000f2}.mu:hover{background-color:#232323e6!important}\n"], dependencies: [{ kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }] });
25
+ TableToolbarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: TableToolbarComponent, selector: "tk-users-table-toolbar", inputs: { totalUsersCount: "totalUsersCount" }, outputs: { searchChange: "searchChange", searchByType: "searchByType", addUser: "addUser" }, ngImport: i0, template: "<header style=\"margin-bottom: 1rem;\">\r\n <div style=\"display: flex; align-items: center; justify-content: space-between;\">\r\n <div class=\"collection-header-left\">\r\n <div style=\"display: flex; align-items: center; gap: 12px;\">\r\n <!-- Search Input -->\r\n <div style=\"position: relative; width: fit-content; min-width: 200px;\">\r\n <input (input)=\"onSearchInputChange($event)\" type=\"text\" placeholder=\"Search by display name, email\" style=\"width: 100%; height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 12px 6px 44px; border: 1px solid #00000080; background: #fff; color: #222; outline: none; box-sizing: border-box; min-width: 280px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">search</i>\r\n </span>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Filter Select -->\r\n <div style=\"display: flex; align-items: center; position: relative; width: fit-content; min-width: 200px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">filter_list</i>\r\n </span>\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 44px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Filter</option>\r\n </select>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Location Select -->\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Select a Location</option>\r\n </select>\r\n\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\"\r\n (change)=\"onTypeChange($event)\">\r\n <option value=\"null\" disabled selected hidden>Select type</option>\r\n <option value=\"\">All</option>\r\n <option value=\"CUCM\">Dedicated Instance</option>\r\n <option value=\"WEBEX\">Multi-Tenant</option>\r\n <option value=\"ONPREM\">On Prem</option>\r\n </select>\r\n <span style=\"margin-left: 8px; color: #757575; font-size: 1rem; white-space: nowrap;\">{{ totalUsersCount }} users</span>\r\n </div>\r\n </div>\r\n <div style=\"display: flex; gap: 1rem;\">\r\n <button class=\"btn mu\" (click)=\"onAddUser()\">\r\n Add user\r\n </button>\r\n </div>\r\n </div>\r\n</header>", styles: [".btn{position:relative;display:inline-flex;gap:6px;align-items:center;justify-content:center;overflow:hidden!important;width:min-content;max-width:100%;padding:0rem .75rem;border:1px solid transparent;border-radius:6.25rem;font-weight:500;font-size:14px;text-overflow:ellipsis;white-space:nowrap;transition:all .1s cubic-bezier(.25,.1,.25,1);height:2rem;cursor:pointer}.btn:hover{background-color:#0000000d}.si{border-color:#0000004d;background-color:#0000;color:#000000f2}.mu{color:#fff;background-color:#000000f2}.mu:hover{background-color:#232323e6!important}\n"], dependencies: [{ kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }] });
22
26
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TableToolbarComponent, decorators: [{
23
27
  type: Component,
24
- args: [{ selector: 'tk-users-table-toolbar', template: "<header style=\"margin-bottom: 1rem;\">\r\n <div style=\"display: flex; align-items: center; justify-content: space-between;\">\r\n <div class=\"collection-header-left\">\r\n <div style=\"display: flex; align-items: center; gap: 12px;\">\r\n <!-- Search Input -->\r\n <div style=\"position: relative; width: fit-content; min-width: 200px;\">\r\n <input (input)=\"onSearchInputChange($event)\" type=\"text\" placeholder=\"Search by display name, email\" style=\"width: 100%; height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 12px 6px 44px; border: 1px solid #00000080; background: #fff; color: #222; outline: none; box-sizing: border-box; min-width: 280px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">search</i>\r\n </span>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Filter Select -->\r\n <div style=\"display: flex; align-items: center; position: relative; width: fit-content; min-width: 200px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">filter_list</i>\r\n </span>\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 44px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Filter</option>\r\n </select>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Location Select -->\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Select a Location</option>\r\n </select>\r\n\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\"\r\n (change)=\"onTypeChange($event)\">\r\n <option value=\"null\" disabled selected hidden>Select type</option>\r\n <option value=\"\">All</option>\r\n <option value=\"CUCM\">Dedicated Instance</option>\r\n <option value=\"WEBEX\">Multi-Tenant</option>\r\n <option value=\"ONPREM\">On Prem</option>\r\n </select>\r\n <span style=\"margin-left: 8px; color: #757575; font-size: 1rem; white-space: nowrap;\">{{ totalUsersCount }} users</span>\r\n </div>\r\n </div>\r\n <div style=\"display: flex; gap: 1rem;\">\r\n <button class=\"btn si\">\r\n Send invitations\r\n </button>\r\n <button class=\"btn mu\">\r\n Manage users\r\n </button>\r\n </div>\r\n </div>\r\n</header>", styles: [".btn{position:relative;display:inline-flex;gap:6px;align-items:center;justify-content:center;overflow:hidden!important;width:min-content;max-width:100%;padding:0rem .75rem;border:1px solid transparent;border-radius:6.25rem;font-weight:500;font-size:14px;text-overflow:ellipsis;white-space:nowrap;transition:all .1s cubic-bezier(.25,.1,.25,1);height:2rem;cursor:pointer}.btn:hover{background-color:#0000000d}.si{border-color:#0000004d;background-color:#0000;color:#000000f2}.mu{color:#fff;background-color:#000000f2}.mu:hover{background-color:#232323e6!important}\n"] }]
28
+ args: [{ selector: 'tk-users-table-toolbar', template: "<header style=\"margin-bottom: 1rem;\">\r\n <div style=\"display: flex; align-items: center; justify-content: space-between;\">\r\n <div class=\"collection-header-left\">\r\n <div style=\"display: flex; align-items: center; gap: 12px;\">\r\n <!-- Search Input -->\r\n <div style=\"position: relative; width: fit-content; min-width: 200px;\">\r\n <input (input)=\"onSearchInputChange($event)\" type=\"text\" placeholder=\"Search by display name, email\" style=\"width: 100%; height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 12px 6px 44px; border: 1px solid #00000080; background: #fff; color: #222; outline: none; box-sizing: border-box; min-width: 280px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">search</i>\r\n </span>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Filter Select -->\r\n <div style=\"display: flex; align-items: center; position: relative; width: fit-content; min-width: 200px;\">\r\n <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">filter_list</i>\r\n </span>\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 44px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Filter</option>\r\n </select>\r\n </div>\r\n <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n <!-- Location Select -->\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n <option>Select a Location</option>\r\n </select>\r\n\r\n <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\"\r\n (change)=\"onTypeChange($event)\">\r\n <option value=\"null\" disabled selected hidden>Select type</option>\r\n <option value=\"\">All</option>\r\n <option value=\"CUCM\">Dedicated Instance</option>\r\n <option value=\"WEBEX\">Multi-Tenant</option>\r\n <option value=\"ONPREM\">On Prem</option>\r\n </select>\r\n <span style=\"margin-left: 8px; color: #757575; font-size: 1rem; white-space: nowrap;\">{{ totalUsersCount }} users</span>\r\n </div>\r\n </div>\r\n <div style=\"display: flex; gap: 1rem;\">\r\n <button class=\"btn mu\" (click)=\"onAddUser()\">\r\n Add user\r\n </button>\r\n </div>\r\n </div>\r\n</header>", styles: [".btn{position:relative;display:inline-flex;gap:6px;align-items:center;justify-content:center;overflow:hidden!important;width:min-content;max-width:100%;padding:0rem .75rem;border:1px solid transparent;border-radius:6.25rem;font-weight:500;font-size:14px;text-overflow:ellipsis;white-space:nowrap;transition:all .1s cubic-bezier(.25,.1,.25,1);height:2rem;cursor:pointer}.btn:hover{background-color:#0000000d}.si{border-color:#0000004d;background-color:#0000;color:#000000f2}.mu{color:#fff;background-color:#000000f2}.mu:hover{background-color:#232323e6!important}\n"] }]
25
29
  }], propDecorators: { totalUsersCount: [{
26
30
  type: Input
27
31
  }], searchChange: [{
28
32
  type: Output
29
33
  }], searchByType: [{
30
34
  type: Output
35
+ }], addUser: [{
36
+ type: Output
31
37
  }] } });
32
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-toolbar.component.js","sourceRoot":"","sources":["../../../../../../../../projects/tuki/widgets/users-list/src/components/table-toolbar/table-toolbar.component.ts","../../../../../../../../projects/tuki/widgets/users-list/src/components/table-toolbar/table-toolbar.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;;;AAS/E,MAAM,OAAO,qBAAqB;IALlC;QAOuB,iBAAY,GAAG,IAAI,YAAY,EAAU,CAAC;QAC1C,iBAAY,GAAG,IAAI,YAAY,EAAmB,CAAC;KAiBzE;IAbG,mBAAmB,CAAC,KAAY;QAC5B,MAAM,KAAK,GAAI,KAAK,CAAC,MAA2B,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,IAAI,EAAE,CAAC;QAElF,IAAI,IAAI,CAAC,OAAO;YAAE,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAEpD,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,IAAI,CAAC,CAAC;IAChF,CAAC;IAEM,YAAY,CAAC,KAAY;QAC5B,MAAM,KAAK,GAAI,KAAK,CAAC,MAA4B,CAAC,KAAwB,CAAC;QAE3E,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAClC,CAAC;;mHAnBQ,qBAAqB;uGAArB,qBAAqB,uLCTlC,i3IA+CS;4FDtCI,qBAAqB;kBALjC,SAAS;+BACI,wBAAwB;8BAKzB,eAAe;sBAAvB,KAAK;gBACa,YAAY;sBAA9B,MAAM;gBACY,YAAY;sBAA9B,MAAM","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from \"@angular/core\";\r\nimport { UserService } from \"../../services/user.service\";\r\nimport { USER_TYPES_TYPE } from \"../../app.constants\";\r\n\r\n@Component({\r\n    selector: 'tk-users-table-toolbar',\r\n    templateUrl: './table-toolbar.component.html',\r\n    styleUrls: ['./table-toolbar.component.scss']\r\n})\r\nexport class TableToolbarComponent {\r\n    @Input() totalUsersCount!: number;\r\n    @Output() readonly searchChange = new EventEmitter<string>();\r\n    @Output() readonly searchByType = new EventEmitter<USER_TYPES_TYPE>();\r\n\r\n    private timerId!: number; \r\n\r\n    onSearchInputChange(event: Event): void {\r\n        const value = (event.target as HTMLInputElement).value.trim().toLowerCase() || '';\r\n\r\n        if (this.timerId) window.clearTimeout(this.timerId);\r\n\r\n        this.timerId = window.setTimeout(() => this.searchChange.emit(value), 1500);\r\n    }\r\n\r\n    public onTypeChange(event: Event): void {\r\n        const value = (event.target as HTMLSelectElement).value as USER_TYPES_TYPE;\r\n\r\n        this.searchByType.emit(value);\r\n    }\r\n}","<header style=\"margin-bottom: 1rem;\">\r\n    <div style=\"display: flex; align-items: center; justify-content: space-between;\">\r\n      <div class=\"collection-header-left\">\r\n        <div style=\"display: flex; align-items: center; gap: 12px;\">\r\n          <!-- Search Input -->\r\n          <div style=\"position: relative; width: fit-content; min-width: 200px;\">\r\n            <input (input)=\"onSearchInputChange($event)\" type=\"text\" placeholder=\"Search by display name, email\" style=\"width: 100%; height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 12px 6px 44px; border: 1px solid #00000080; background: #fff; color: #222; outline: none; box-sizing: border-box; min-width: 280px;\">\r\n            <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n              <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">search</i>\r\n            </span>\r\n          </div>\r\n          <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n          <!-- Filter Select -->\r\n          <div style=\"display: flex; align-items: center; position: relative; width: fit-content; min-width: 200px;\">\r\n            <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n              <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">filter_list</i>\r\n            </span>\r\n            <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 44px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n              <option>Filter</option>\r\n            </select>\r\n          </div>\r\n          <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n          <!-- Location Select -->\r\n          <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n            <option>Select a Location</option>\r\n          </select>\r\n\r\n          <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\"\r\n            (change)=\"onTypeChange($event)\">\r\n            <option value=\"null\" disabled selected hidden>Select type</option>\r\n            <option value=\"\">All</option>\r\n            <option value=\"CUCM\">Dedicated Instance</option>\r\n            <option value=\"WEBEX\">Multi-Tenant</option>\r\n            <option value=\"ONPREM\">On Prem</option>\r\n          </select>\r\n          <span style=\"margin-left: 8px; color: #757575; font-size: 1rem; white-space: nowrap;\">{{ totalUsersCount }} users</span>\r\n        </div>\r\n      </div>\r\n      <div style=\"display: flex; gap: 1rem;\">\r\n        <button class=\"btn si\">\r\n          Send invitations\r\n        </button>\r\n        <button class=\"btn mu\">\r\n          Manage users\r\n        </button>\r\n      </div>\r\n    </div>\r\n</header>"]}
38
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-toolbar.component.js","sourceRoot":"","sources":["../../../../../../../../projects/tuki/widgets/users-list/src/components/table-toolbar/table-toolbar.component.ts","../../../../../../../../projects/tuki/widgets/users-list/src/components/table-toolbar/table-toolbar.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;;;AAS/E,MAAM,OAAO,qBAAqB;IALlC;QAOuB,iBAAY,GAAG,IAAI,YAAY,EAAU,CAAC;QAC1C,iBAAY,GAAG,IAAI,YAAY,EAAmB,CAAC;QACnD,YAAO,GAAG,IAAI,YAAY,EAAQ,CAAC;KAqBzD;IAjBG,mBAAmB,CAAC,KAAY;QAC5B,MAAM,KAAK,GAAI,KAAK,CAAC,MAA2B,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,IAAI,EAAE,CAAC;QAElF,IAAI,IAAI,CAAC,OAAO;YAAE,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAEpD,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,IAAI,CAAC,CAAC;IAChF,CAAC;IAEM,YAAY,CAAC,KAAY;QAC5B,MAAM,KAAK,GAAI,KAAK,CAAC,MAA4B,CAAC,KAAwB,CAAC;QAE3E,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAClC,CAAC;IAED,SAAS;QACL,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;IACxB,CAAC;;mHAxBQ,qBAAqB;uGAArB,qBAAqB,2MCTlC,6yIA4CS;4FDnCI,qBAAqB;kBALjC,SAAS;+BACI,wBAAwB;8BAKzB,eAAe;sBAAvB,KAAK;gBACa,YAAY;sBAA9B,MAAM;gBACY,YAAY;sBAA9B,MAAM;gBACY,OAAO;sBAAzB,MAAM","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from \"@angular/core\";\r\nimport { UserService } from \"../../services/user.service\";\r\nimport { USER_TYPES_TYPE } from \"../../app.constants\";\r\n\r\n@Component({\r\n    selector: 'tk-users-table-toolbar',\r\n    templateUrl: './table-toolbar.component.html',\r\n    styleUrls: ['./table-toolbar.component.scss']\r\n})\r\nexport class TableToolbarComponent {\r\n    @Input() totalUsersCount!: number;\r\n    @Output() readonly searchChange = new EventEmitter<string>();\r\n    @Output() readonly searchByType = new EventEmitter<USER_TYPES_TYPE>();\r\n    @Output() readonly addUser = new EventEmitter<void>();\r\n\r\n    private timerId!: number; \r\n\r\n    onSearchInputChange(event: Event): void {\r\n        const value = (event.target as HTMLInputElement).value.trim().toLowerCase() || '';\r\n\r\n        if (this.timerId) window.clearTimeout(this.timerId);\r\n\r\n        this.timerId = window.setTimeout(() => this.searchChange.emit(value), 1500);\r\n    }\r\n\r\n    public onTypeChange(event: Event): void {\r\n        const value = (event.target as HTMLSelectElement).value as USER_TYPES_TYPE;\r\n\r\n        this.searchByType.emit(value);\r\n    }\r\n\r\n    onAddUser(): void {\r\n        this.addUser.emit();\r\n    }\r\n}","<header style=\"margin-bottom: 1rem;\">\r\n    <div style=\"display: flex; align-items: center; justify-content: space-between;\">\r\n      <div class=\"collection-header-left\">\r\n        <div style=\"display: flex; align-items: center; gap: 12px;\">\r\n          <!-- Search Input -->\r\n          <div style=\"position: relative; width: fit-content; min-width: 200px;\">\r\n            <input (input)=\"onSearchInputChange($event)\" type=\"text\" placeholder=\"Search by display name, email\" style=\"width: 100%; height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 12px 6px 44px; border: 1px solid #00000080; background: #fff; color: #222; outline: none; box-sizing: border-box; min-width: 280px;\">\r\n            <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n              <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">search</i>\r\n            </span>\r\n          </div>\r\n          <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n          <!-- Filter Select -->\r\n          <div style=\"display: flex; align-items: center; position: relative; width: fit-content; min-width: 200px;\">\r\n            <span style=\"position: absolute; left: 12px; top: 50%; transform: translateY(-50%); color: #757575; font-size: 18px; font-weight: normal; pointer-events: none;\">\r\n              <i class=\"material-icons\" style=\"font-size: 18px; font-weight: normal;\">filter_list</i>\r\n            </span>\r\n            <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 44px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n              <option>Filter</option>\r\n            </select>\r\n          </div>\r\n          <span style=\"margin: 0 4px; color: #757575; font-size: 1rem;\">or</span>\r\n          <!-- Location Select -->\r\n          <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\">\r\n            <option>Select a Location</option>\r\n          </select>\r\n\r\n          <select style=\"height: 34px; border-radius: .5rem; font-size: 16px; padding: 6px 32px 6px 12px; border: 1px solid #00000080; background: #fff; min-width: 200px; width: fit-content; color: #222; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml;utf8,<svg fill=\\'%23757575\\' height=\\'20\\' viewBox=\\'0 0 24 24\\' width=\\'20\\' xmlns=\\'http://www.w3.org/2000/svg\\'><path d=\\'M7 10l5 5 5-5z\\'/></svg>'); background-repeat: no-repeat; background-position: right 12px center; background-size: 20px 20px;\"\r\n            (change)=\"onTypeChange($event)\">\r\n            <option value=\"null\" disabled selected hidden>Select type</option>\r\n            <option value=\"\">All</option>\r\n            <option value=\"CUCM\">Dedicated Instance</option>\r\n            <option value=\"WEBEX\">Multi-Tenant</option>\r\n            <option value=\"ONPREM\">On Prem</option>\r\n          </select>\r\n          <span style=\"margin-left: 8px; color: #757575; font-size: 1rem; white-space: nowrap;\">{{ totalUsersCount }} users</span>\r\n        </div>\r\n      </div>\r\n      <div style=\"display: flex; gap: 1rem;\">\r\n        <button class=\"btn mu\" (click)=\"onAddUser()\">\r\n          Add user\r\n        </button>\r\n      </div>\r\n    </div>\r\n</header>"]}