@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.
- package/esm2020/di2mt/shared/services/api.service.mjs +3 -3
- package/esm2020/user-creation/public-api.mjs +13 -0
- package/esm2020/user-creation/src/app.constants.mjs +16 -0
- package/esm2020/user-creation/src/classes/site.mjs +53 -0
- package/esm2020/user-creation/src/shared/api/api.service.mjs +86 -0
- package/esm2020/user-creation/src/shared/material.module.mjs +76 -0
- package/esm2020/user-creation/src/shared/services/user-creation-api.service.mjs +112 -0
- package/esm2020/user-creation/src/shared/services/utils.mjs +19 -0
- package/esm2020/user-creation/src/shared/shared.module.mjs +33 -0
- package/esm2020/user-creation/src/user-creation.module.mjs +69 -0
- package/esm2020/user-creation/src/widgets/user-creation-wizard/components/user-details-step/user-details-step.component.mjs +23 -0
- package/esm2020/user-creation/src/widgets/user-creation-wizard/components/user-overview-step/user-overview-step.component.mjs +19 -0
- package/esm2020/user-creation/src/widgets/user-creation-wizard/components/user-template-step/user-template-step.component.mjs +114 -0
- package/esm2020/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.component.mjs +60 -0
- package/esm2020/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.service.mjs +273 -0
- package/esm2020/user-creation/tuki-io-tuki-widgets-user-creation.mjs +5 -0
- package/esm2020/users-list/src/components/table-toolbar/table-toolbar.component.mjs +9 -3
- package/esm2020/users-list/src/users-list.component.mjs +9 -3
- package/fesm2015/tuki-io-tuki-widgets-di2mt.mjs +2 -2
- package/fesm2015/tuki-io-tuki-widgets-di2mt.mjs.map +1 -1
- package/fesm2015/tuki-io-tuki-widgets-user-creation.mjs +919 -0
- package/fesm2015/tuki-io-tuki-widgets-user-creation.mjs.map +1 -0
- package/fesm2015/tuki-io-tuki-widgets-users-list.mjs +16 -4
- package/fesm2015/tuki-io-tuki-widgets-users-list.mjs.map +1 -1
- package/fesm2020/tuki-io-tuki-widgets-di2mt.mjs +2 -2
- package/fesm2020/tuki-io-tuki-widgets-di2mt.mjs.map +1 -1
- package/fesm2020/tuki-io-tuki-widgets-user-creation.mjs +915 -0
- package/fesm2020/tuki-io-tuki-widgets-user-creation.mjs.map +1 -0
- package/fesm2020/tuki-io-tuki-widgets-users-list.mjs +16 -4
- package/fesm2020/tuki-io-tuki-widgets-users-list.mjs.map +1 -1
- package/package.json +9 -1
- package/user-creation/index.d.ts +5 -0
- package/user-creation/public-api.d.ts +2 -0
- package/user-creation/src/app.constants.d.ts +15 -0
- package/user-creation/src/classes/site.d.ts +203 -0
- package/user-creation/src/shared/api/api.service.d.ts +20 -0
- package/user-creation/src/shared/material.module.d.ts +15 -0
- package/user-creation/src/shared/services/user-creation-api.service.d.ts +24 -0
- package/user-creation/src/shared/services/utils.d.ts +1 -0
- package/user-creation/src/shared/shared.module.d.ts +10 -0
- package/user-creation/src/user-creation.module.d.ts +17 -0
- package/user-creation/src/widgets/user-creation-wizard/components/user-details-step/user-details-step.component.d.ts +8 -0
- package/user-creation/src/widgets/user-creation-wizard/components/user-overview-step/user-overview-step.component.d.ts +11 -0
- package/user-creation/src/widgets/user-creation-wizard/components/user-template-step/user-template-step.component.d.ts +37 -0
- package/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.component.d.ts +18 -0
- package/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.service.d.ts +87 -0
- package/users-list/src/components/table-toolbar/table-toolbar.component.d.ts +3 -1
- package/users-list/src/users-list.component.d.ts +3 -1
package/esm2020/user-creation/src/widgets/user-creation-wizard/user-creation-wizard.service.mjs
ADDED
|
@@ -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
|
|
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
|
|
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,
|
|
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>"]}
|