@acorex/modules 21.0.0-next.31 → 21.0.0-next.34

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 (65) hide show
  1. package/fesm2022/{acorex-modules-ai-management-acorex-modules-ai-management-B6lnG21F.mjs → acorex-modules-ai-management-acorex-modules-ai-management-WuqoH4OL.mjs} +10 -11
  2. package/fesm2022/acorex-modules-ai-management-acorex-modules-ai-management-WuqoH4OL.mjs.map +1 -0
  3. package/fesm2022/{acorex-modules-ai-management-agent.entity-C2APPc7g.mjs → acorex-modules-ai-management-agent.entity-DK9GNMdR.mjs} +2 -2
  4. package/fesm2022/{acorex-modules-ai-management-agent.entity-C2APPc7g.mjs.map → acorex-modules-ai-management-agent.entity-DK9GNMdR.mjs.map} +1 -1
  5. package/fesm2022/{acorex-modules-ai-management-ai-delegated-agent-transcript-popup.component-D_54-MqC.mjs → acorex-modules-ai-management-ai-delegated-agent-transcript-popup.component-CWXEW9yL.mjs} +2 -2
  6. package/fesm2022/{acorex-modules-ai-management-ai-delegated-agent-transcript-popup.component-D_54-MqC.mjs.map → acorex-modules-ai-management-ai-delegated-agent-transcript-popup.component-CWXEW9yL.mjs.map} +1 -1
  7. package/fesm2022/{acorex-modules-ai-management-assist-delegated-agent-configurator-widget-edit.component-pG9y9bqO.mjs → acorex-modules-ai-management-assist-delegated-agent-configurator-widget-edit.component-CNKwIqUE.mjs} +6 -4
  8. package/fesm2022/acorex-modules-ai-management-assist-delegated-agent-configurator-widget-edit.component-CNKwIqUE.mjs.map +1 -0
  9. package/fesm2022/{acorex-modules-ai-management-assist.entity-D7sIFWTx.mjs → acorex-modules-ai-management-assist.entity-DX2U9sd5.mjs} +2 -2
  10. package/fesm2022/{acorex-modules-ai-management-assist.entity-D7sIFWTx.mjs.map → acorex-modules-ai-management-assist.entity-DX2U9sd5.mjs.map} +1 -1
  11. package/fesm2022/{acorex-modules-ai-management-index-DoOTRMvM.mjs → acorex-modules-ai-management-index-KX3TVRXr.mjs} +2 -2
  12. package/fesm2022/{acorex-modules-ai-management-index-DoOTRMvM.mjs.map → acorex-modules-ai-management-index-KX3TVRXr.mjs.map} +1 -1
  13. package/fesm2022/{acorex-modules-ai-management-model.entity-xhfG1tFZ.mjs → acorex-modules-ai-management-model.entity-BeuDUahR.mjs} +2 -2
  14. package/fesm2022/{acorex-modules-ai-management-model.entity-xhfG1tFZ.mjs.map → acorex-modules-ai-management-model.entity-BeuDUahR.mjs.map} +1 -1
  15. package/fesm2022/{acorex-modules-ai-management-open-ai-endpoint.entity-Bot8Ctq1.mjs → acorex-modules-ai-management-open-ai-endpoint.entity-D-9unl6i.mjs} +2 -2
  16. package/fesm2022/{acorex-modules-ai-management-open-ai-endpoint.entity-Bot8Ctq1.mjs.map → acorex-modules-ai-management-open-ai-endpoint.entity-D-9unl6i.mjs.map} +1 -1
  17. package/fesm2022/acorex-modules-ai-management.mjs +1 -1
  18. package/fesm2022/{acorex-modules-assessment-management-acorex-modules-assessment-management-D1DhOSYU.mjs → acorex-modules-assessment-management-acorex-modules-assessment-management-BNK6C-Ng.mjs} +25 -25
  19. package/fesm2022/{acorex-modules-assessment-management-acorex-modules-assessment-management-D1DhOSYU.mjs.map → acorex-modules-assessment-management-acorex-modules-assessment-management-BNK6C-Ng.mjs.map} +1 -1
  20. package/fesm2022/{acorex-modules-assessment-management-assessment-case.entity-BwwarDP0.mjs → acorex-modules-assessment-management-assessment-case.entity-Ds8pRzvE.mjs} +2 -2
  21. package/fesm2022/{acorex-modules-assessment-management-assessment-case.entity-BwwarDP0.mjs.map → acorex-modules-assessment-management-assessment-case.entity-Ds8pRzvE.mjs.map} +1 -1
  22. package/fesm2022/{acorex-modules-assessment-management-assessment-session.entity-CQNn4Ryb.mjs → acorex-modules-assessment-management-assessment-session.entity-tq1Q4wkS.mjs} +2 -2
  23. package/fesm2022/{acorex-modules-assessment-management-assessment-session.entity-CQNn4Ryb.mjs.map → acorex-modules-assessment-management-assessment-session.entity-tq1Q4wkS.mjs.map} +1 -1
  24. package/fesm2022/{acorex-modules-assessment-management-fill-assessment-session.command-AptlLpWh.mjs → acorex-modules-assessment-management-fill-assessment-session.command-yf0PnWC0.mjs} +2 -2
  25. package/fesm2022/{acorex-modules-assessment-management-fill-assessment-session.command-AptlLpWh.mjs.map → acorex-modules-assessment-management-fill-assessment-session.command-yf0PnWC0.mjs.map} +1 -1
  26. package/fesm2022/{acorex-modules-assessment-management-index-BBzx9W5u.mjs → acorex-modules-assessment-management-index-B2SncZAW.mjs} +6 -6
  27. package/fesm2022/acorex-modules-assessment-management-index-B2SncZAW.mjs.map +1 -0
  28. package/fesm2022/{acorex-modules-assessment-management-preview-question.command-BMoPn_op.mjs → acorex-modules-assessment-management-preview-question.command-DPzL8C44.mjs} +2 -2
  29. package/fesm2022/{acorex-modules-assessment-management-preview-question.command-BMoPn_op.mjs.map → acorex-modules-assessment-management-preview-question.command-DPzL8C44.mjs.map} +1 -1
  30. package/fesm2022/{acorex-modules-assessment-management-preview-questionnaire.command-BI7JJIbZ.mjs → acorex-modules-assessment-management-preview-questionnaire.command-BrI5gFtx.mjs} +2 -2
  31. package/fesm2022/{acorex-modules-assessment-management-preview-questionnaire.command-BI7JJIbZ.mjs.map → acorex-modules-assessment-management-preview-questionnaire.command-BrI5gFtx.mjs.map} +1 -1
  32. package/fesm2022/{acorex-modules-assessment-management-question-bank-interface-editor-widget-edit.component-BfWuSVG9.mjs → acorex-modules-assessment-management-question-bank-interface-editor-widget-edit.component-XD5IJpvW.mjs} +2 -2
  33. package/fesm2022/{acorex-modules-assessment-management-question-bank-interface-editor-widget-edit.component-BfWuSVG9.mjs.map → acorex-modules-assessment-management-question-bank-interface-editor-widget-edit.component-XD5IJpvW.mjs.map} +1 -1
  34. package/fesm2022/{acorex-modules-assessment-management-question-bank-item.entity-CcDAcPJJ.mjs → acorex-modules-assessment-management-question-bank-item.entity-CZ32sQ9M.mjs} +2 -2
  35. package/fesm2022/{acorex-modules-assessment-management-question-bank-item.entity-CcDAcPJJ.mjs.map → acorex-modules-assessment-management-question-bank-item.entity-CZ32sQ9M.mjs.map} +1 -1
  36. package/fesm2022/{acorex-modules-assessment-management-questionnaire-calculation.entity-DC7xwETF.mjs → acorex-modules-assessment-management-questionnaire-calculation.entity-C7gf4s1y.mjs} +2 -2
  37. package/fesm2022/{acorex-modules-assessment-management-questionnaire-calculation.entity-DC7xwETF.mjs.map → acorex-modules-assessment-management-questionnaire-calculation.entity-C7gf4s1y.mjs.map} +1 -1
  38. package/fesm2022/{acorex-modules-assessment-management-questionnaire-viewer-popup.component-DF7F-quh.mjs → acorex-modules-assessment-management-questionnaire-viewer-popup.component-BZyIlYWg.mjs} +2 -2
  39. package/fesm2022/{acorex-modules-assessment-management-questionnaire-viewer-popup.component-DF7F-quh.mjs.map → acorex-modules-assessment-management-questionnaire-viewer-popup.component-BZyIlYWg.mjs.map} +1 -1
  40. package/fesm2022/{acorex-modules-assessment-management-questionnaire.entity-zybar1sK.mjs → acorex-modules-assessment-management-questionnaire.entity-CmmThk2f.mjs} +3 -3
  41. package/fesm2022/{acorex-modules-assessment-management-questionnaire.entity-zybar1sK.mjs.map → acorex-modules-assessment-management-questionnaire.entity-CmmThk2f.mjs.map} +1 -1
  42. package/fesm2022/{acorex-modules-assessment-management-save-questionnaire-questions.command-DqJ-_k-4.mjs → acorex-modules-assessment-management-save-questionnaire-questions.command-GLqvK8wB.mjs} +2 -2
  43. package/fesm2022/{acorex-modules-assessment-management-save-questionnaire-questions.command-DqJ-_k-4.mjs.map → acorex-modules-assessment-management-save-questionnaire-questions.command-GLqvK8wB.mjs.map} +1 -1
  44. package/fesm2022/{acorex-modules-assessment-management-view-session-answers.command-IHxXETJQ.mjs → acorex-modules-assessment-management-view-session-answers.command-CU1TVonZ.mjs} +2 -2
  45. package/fesm2022/{acorex-modules-assessment-management-view-session-answers.command-IHxXETJQ.mjs.map → acorex-modules-assessment-management-view-session-answers.command-CU1TVonZ.mjs.map} +1 -1
  46. package/fesm2022/acorex-modules-assessment-management.mjs +1 -1
  47. package/fesm2022/acorex-modules-dashboard-management.mjs +4366 -3497
  48. package/fesm2022/acorex-modules-dashboard-management.mjs.map +1 -1
  49. package/fesm2022/acorex-modules-data-management.mjs +3 -2
  50. package/fesm2022/acorex-modules-data-management.mjs.map +1 -1
  51. package/fesm2022/acorex-modules-organization-management-org-chart.page-eRTHzLDQ.mjs +784 -0
  52. package/fesm2022/acorex-modules-organization-management-org-chart.page-eRTHzLDQ.mjs.map +1 -0
  53. package/fesm2022/acorex-modules-organization-management.mjs +3 -9
  54. package/fesm2022/acorex-modules-organization-management.mjs.map +1 -1
  55. package/fesm2022/acorex-modules-workflow-management.mjs +488 -105
  56. package/fesm2022/acorex-modules-workflow-management.mjs.map +1 -1
  57. package/package.json +2 -2
  58. package/types/acorex-modules-dashboard-management.d.ts +397 -218
  59. package/types/acorex-modules-organization-management.d.ts +3 -12
  60. package/types/acorex-modules-workflow-management.d.ts +27 -0
  61. package/fesm2022/acorex-modules-ai-management-acorex-modules-ai-management-B6lnG21F.mjs.map +0 -1
  62. package/fesm2022/acorex-modules-ai-management-assist-delegated-agent-configurator-widget-edit.component-pG9y9bqO.mjs.map +0 -1
  63. package/fesm2022/acorex-modules-assessment-management-index-BBzx9W5u.mjs.map +0 -1
  64. package/fesm2022/acorex-modules-organization-management-org-chart.page-E0hy7dXg.mjs +0 -638
  65. package/fesm2022/acorex-modules-organization-management-org-chart.page-E0hy7dXg.mjs.map +0 -1
@@ -1,638 +0,0 @@
1
- import { AXPanViewDirective } from '@acorex/cdk/pan-view';
2
- import { AXBadgeModule } from '@acorex/components/badge';
3
- import { AXBreadcrumbsModule } from '@acorex/components/breadcrumbs';
4
- import * as i1 from '@acorex/components/button';
5
- import { AXButtonModule } from '@acorex/components/button';
6
- import * as i2 from '@acorex/components/decorators';
7
- import { AXDecoratorModule } from '@acorex/components/decorators';
8
- import { AXDropdownButtonModule } from '@acorex/components/dropdown-button';
9
- import { AXLoadingModule } from '@acorex/components/loading';
10
- import { AXContextMenuComponent, AXMenuModule } from '@acorex/components/menu';
11
- import * as i4 from '@acorex/core/translation';
12
- import { AXTranslationService, AXTranslationModule } from '@acorex/core/translation';
13
- import { AXPPageLayoutBaseComponent, AXPPageLayoutComponent, AXPPageLayoutBase } from '@acorex/platform/layout/views';
14
- import { AXPThemeLayoutBlockComponent } from '@acorex/platform/layout/components';
15
- import * as i3 from '@angular/common';
16
- import { CommonModule } from '@angular/common';
17
- import * as i0 from '@angular/core';
18
- import { inject, signal, Injectable, viewChild, NgZone, ViewContainerRef, afterNextRender, ViewEncapsulation, ChangeDetectionStrategy, Component } from '@angular/core';
19
- import { RouterModule } from '@angular/router';
20
- import set from 'lodash-es/set';
21
- import { AXMOrganizationManagementChartEntityService, RootConfig, AXMOrgChartNodeType, AXMOrgChartCategory } from './acorex-modules-organization-management.mjs';
22
- import { AXLoadingDialogService } from '@acorex/components/loading-dialog';
23
- import { AXPExportService } from '@acorex/platform/common';
24
- import { AXPCommandService } from '@acorex/platform/runtime';
25
- import { AXPLayoutBuilderService } from '@acorex/platform/layout/builder';
26
- import { createMultiLanguageString, AXPDataGenerator, getActionButton, AXPResolveMultiLanguageStringPipe } from '@acorex/platform/core';
27
-
28
- class AXMOrgChartService {
29
- constructor() {
30
- // Employee moved to HCM; resolve via runtime query mapping instead of direct service
31
- this.chartService = inject(AXMOrganizationManagementChartEntityService);
32
- this.loadingDialog = inject(AXLoadingDialogService);
33
- this.exportService = inject(AXPExportService);
34
- this.translationService = inject(AXTranslationService);
35
- this.commandService = inject(AXPCommandService);
36
- this.layoutBuilder = inject(AXPLayoutBuilderService);
37
- this._rootNode = signal({
38
- id: AXPDataGenerator.uuid(),
39
- /** Until `load()`; keep in sync with `organization-management.organization-chart.nodes.company`. */
40
- title: createMultiLanguageString('Company', 'شرکت'),
41
- }, ...(ngDevMode ? [{ debugName: "_rootNode" }] : /* istanbul ignore next */ []));
42
- this.rootNode = this._rootNode.asReadonly();
43
- }
44
- async load() {
45
- const first = (await this.chartService.query({
46
- skip: 0,
47
- take: 1,
48
- sort: [{ field: 'version', dir: 'desc' }],
49
- })).items[0];
50
- //
51
- if (first && first.data) {
52
- const data = first.data;
53
- this._rootNode.set(data);
54
- }
55
- else {
56
- await this.generate();
57
- await this.load();
58
- }
59
- }
60
- async generate() {
61
- const dialogResult = await this.showGenerateDialog();
62
- if (dialogResult.cancelled) {
63
- return;
64
- }
65
- const input = dialogResult.input || {
66
- includeEmployees: true,
67
- preferPrimary: true,
68
- mode: 'structure',
69
- };
70
- // Prefer generating a fresh chart via mock command when available
71
- await this.commandService.execute('OrganizationManagement.OrgChart:Generate', input);
72
- }
73
- /**
74
- * Show dialog to get OrgChart generation parameters from user
75
- */
76
- async showGenerateDialog() {
77
- const modeLabel = await this.translationService.translateAsync('@organization-management:organization-chart.components.generate-dialog.fields.mode.label');
78
- const includeEmployeesLabel = await this.translationService.translateAsync('@organization-management:organization-chart.components.generate-dialog.fields.include-employees.label');
79
- const preferPrimaryLabel = await this.translationService.translateAsync('@organization-management:organization-chart.components.generate-dialog.fields.prefer-primary.label');
80
- return this.layoutBuilder
81
- .create()
82
- .dialog(dialog => {
83
- dialog
84
- .setTitle('@organization-management:organization-chart.components.generate-dialog.title')
85
- .setMessage('@organization-management:organization-chart.components.generate-dialog.message')
86
- .setContext({
87
- mode: 'structure',
88
- includeEmployees: true,
89
- preferPrimary: false
90
- })
91
- .content(flex => {
92
- flex.formField(modeLabel, (field) => {
93
- field.path('mode');
94
- field.selectionList({
95
- valueField: 'value',
96
- textField: 'text',
97
- dataSource: [
98
- { value: 'structure', text: '@organization-management:organization-chart.options.mode.structure' },
99
- { value: 'location', text: '@organization-management:organization-chart.options.mode.location' },
100
- { value: 'reporting', text: '@organization-management:organization-chart.options.mode.reporting' }
101
- ]
102
- });
103
- })
104
- .formField(includeEmployeesLabel, (field) => {
105
- field.path('includeEmployees');
106
- field.toggleSwitch();
107
- })
108
- .formField(preferPrimaryLabel, (field) => {
109
- field.path('preferPrimary');
110
- field.toggleSwitch();
111
- });
112
- })
113
- .setActions(actions => {
114
- actions
115
- .cancel()
116
- .submit('@general:generate');
117
- })
118
- .onAction(async (dialogRef) => {
119
- const context = dialogRef.context();
120
- const action = dialogRef.action();
121
- if (!context || action === 'cancel') {
122
- return { cancelled: true };
123
- }
124
- // dialogRef.setLoading(true);
125
- // await new Promise((r) => setTimeout(r, 800));
126
- // dialogRef.setLoading(false);
127
- // if (1 == 1) {
128
- // throw new Error('test');
129
- // }
130
- return {
131
- cancelled: false,
132
- input: context
133
- };
134
- });
135
- })
136
- .show();
137
- }
138
- //#region ---- Printing ----
139
- async printChart(element) {
140
- const d = this.loadingDialog.show({
141
- title: 'Printing Organization Chart...',
142
- mode: 'determinate',
143
- progressColor: 'primary',
144
- progressValue: 0,
145
- status: '0/10',
146
- text: 'Initializing print process...',
147
- buttons: [
148
- { text: 'Cancel', color: 'danger', onClick: () => d.close() }
149
- ],
150
- });
151
- try {
152
- await this.updateStep(d, await this.translate('@organization-management:organization-chart.states.preparing-document'), 10, '1/10');
153
- const chartElement = element;
154
- const { originalStyles, modifiedStyles } = this.expandChart(chartElement);
155
- await this.updateStep(d, await this.translate('@organization-management:organization-chart.states.rendering-chart-image'), 25, '2.5/10');
156
- const blob = await this.exportService.generateBlobFromElement(chartElement, {
157
- popup: true,
158
- filter: (node) => !(node instanceof Element && node.classList.contains('--toggle')),
159
- bgcolor: 'transparent',
160
- scale: 2,
161
- quality: 1,
162
- height: Number.parseInt(modifiedStyles.modifiedHeight),
163
- width: Number.parseInt(modifiedStyles.modifiedWidth),
164
- });
165
- await this.updateStep(d, await this.translate('@organization-management:organization-chart.states.finalizing-image-processing'), 50, '5/10');
166
- this.restoreChartStyles(chartElement, originalStyles);
167
- await this.updateStep(d, await this.translate('@organization-management:organization-chart.states.downloading-image'), 75, '7.5/10');
168
- this.exportService.download(blob, 'org-chart.png');
169
- await this.updateStep(d, await this.translate('@organization-management:organization-chart.states.process-complete'), 100, '10/10');
170
- setTimeout(() => d.close(), 1500);
171
- }
172
- catch (err) {
173
- console.error('Error capturing chart:', err);
174
- await this.updateStep(d, await this.translate('@organization-management:organization-chart.states.failed-to-generate-chart-image'), 100, 'Error', 'danger');
175
- d.close();
176
- }
177
- }
178
- expandChart(chartElement) {
179
- const originalStyles = {
180
- originalOverflow: chartElement.style.overflow,
181
- originalWidth: chartElement.style.width,
182
- originalHeight: chartElement.style.height,
183
- };
184
- const modifiedStyles = {
185
- modifiedOverflow: 'visible',
186
- modifiedWidth: `${chartElement.scrollWidth}px`,
187
- modifiedHeight: `${chartElement.scrollHeight}px`,
188
- };
189
- chartElement.style.overflow = modifiedStyles.modifiedOverflow;
190
- chartElement.style.width = modifiedStyles.modifiedWidth;
191
- chartElement.style.height = modifiedStyles.modifiedHeight;
192
- return { originalStyles, modifiedStyles };
193
- }
194
- restoreChartStyles(chartElement, originalStyles) {
195
- chartElement.style.overflow = originalStyles.originalOverflow;
196
- chartElement.style.width = originalStyles.originalWidth;
197
- chartElement.style.height = originalStyles.originalHeight;
198
- }
199
- async updateStep(dialog, text, progress, status, color = 'primary') {
200
- return new Promise((resolve) => {
201
- dialog.setProgressColor(color);
202
- dialog.setProgressText(text);
203
- dialog.setProgressValue(progress);
204
- dialog.setProgressStatus(status);
205
- resolve();
206
- });
207
- }
208
- async translate(key) {
209
- return (await this.translationService.translateAsync(key)) + '...';
210
- }
211
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: AXMOrgChartService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
212
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: AXMOrgChartService }); }
213
- }
214
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: AXMOrgChartService, decorators: [{
215
- type: Injectable
216
- }] });
217
-
218
- /**
219
- * Component for displaying and interacting with an organizational chart.
220
- */
221
- class AXMOrgChartPage extends AXPPageLayoutBaseComponent {
222
- constructor() {
223
- super(...arguments);
224
- this.rootConfig = RootConfig;
225
- this.nodeTemplate = viewChild.required('nodeTemplate');
226
- this.treeContainer = viewChild.required('chart');
227
- this.parent = viewChild.required('parent');
228
- this.contextMenu = viewChild(AXContextMenuComponent, ...(ngDevMode ? [{ debugName: "contextMenu" }] : /* istanbul ignore next */ []));
229
- this.panView = viewChild(AXPanViewDirective, ...(ngDevMode ? [{ debugName: "panView" }] : /* istanbul ignore next */ []));
230
- this.ngzone = inject(NgZone);
231
- this.viewContainerRef = inject(ViewContainerRef);
232
- // printing is handled by service itself
233
- this.chartService = inject(AXMOrgChartService);
234
- this.NODE_STROKE_COLOR = 'rgba(var(--ax-sys-color-primary-400))';
235
- this.LINK_STROKE_WIDTH = 1;
236
- this.NodeTypes = AXMOrgChartNodeType;
237
- this.isFullscreen = signal(false, ...(ngDevMode ? [{ debugName: "isFullscreen" }] : /* istanbul ignore next */ []));
238
- this.panX = signal(0, ...(ngDevMode ? [{ debugName: "panX" }] : /* istanbul ignore next */ []));
239
- this.panY = signal(0, ...(ngDevMode ? [{ debugName: "panY" }] : /* istanbul ignore next */ []));
240
- this.panZoomLevel = signal(100, ...(ngDevMode ? [{ debugName: "panZoomLevel" }] : /* istanbul ignore next */ []));
241
- this.isChartRendered = false;
242
- this.nodeTypes = [
243
- { title: 'Company', color: 'primary', icon: 'fal fa-building', category: AXMOrgChartCategory.Company },
244
- { title: 'Location', color: 'accent1', icon: 'fal fa-globe', category: AXMOrgChartCategory.Location },
245
- { title: 'Business Unit', color: 'accent2', icon: 'fal fa-sitemap', category: AXMOrgChartCategory.BusinessUnit },
246
- { title: 'Position', color: 'accent3', icon: 'fal fa-user-tie', category: AXMOrgChartCategory.Position },
247
- { title: 'Employee', color: 'surface', icon: 'fal fa-address-card', category: AXMOrgChartCategory.Employee },
248
- ];
249
- /**
250
- * UI chrome for chart cards; align copy with `organization-chart.organization-chart.nodes` / mock `org-chart-labels`.
251
- */
252
- this.chartML = {
253
- vacant: createMultiLanguageString('Vacant', 'خالی'),
254
- directReports: createMultiLanguageString('Direct reports', 'گزارش مستقیم'),
255
- teamMembers: createMultiLanguageString('Team members', 'اعضای تیم'),
256
- };
257
- this.#initialize = afterNextRender(async () => {
258
- this.d3 = await import('d3');
259
- await this.refresh();
260
- });
261
- }
262
- chartInitials(name) {
263
- if (!name?.trim()) {
264
- return '?';
265
- }
266
- const parts = name
267
- .trim()
268
- .split(/\s+/)
269
- .filter(Boolean);
270
- if (parts.length === 1) {
271
- return parts[0].slice(0, 2).toUpperCase();
272
- }
273
- return (parts[0][0] + parts[parts.length - 1][0]).toUpperCase();
274
- }
275
- #initialize;
276
- /**
277
- * Zooms in the chart.
278
- */
279
- zoomIn() {
280
- this.panView()?.zoomIn();
281
- }
282
- /**
283
- * Zooms out the chart.
284
- */
285
- zoomOut() {
286
- this.panView()?.zoomOut();
287
- }
288
- /**
289
- * Resets the zoom level and pan position of the chart.
290
- */
291
- zoomReset() {
292
- this.panView()?.resetPosition();
293
- }
294
- /**
295
- * Toggles fullscreen mode for the chart.
296
- */
297
- toggleFullscreen() {
298
- if (this.isFullscreen()) {
299
- document.exitFullscreen();
300
- this.isFullscreen.set(false);
301
- }
302
- else {
303
- this.parent().nativeElement.requestFullscreen();
304
- this.isFullscreen.set(true);
305
- }
306
- document.querySelector(`foreignObject[node="node-${this.chartService.rootNode().id}"]`)?.scrollIntoView({
307
- behavior: 'smooth',
308
- block: 'start',
309
- inline: 'center',
310
- });
311
- }
312
- /**
313
- * Calculates the maximum depth of the organizational chart.
314
- * @param node - The root node of the chart.
315
- * @returns The maximum depth of the chart.
316
- */
317
- calculateMaxDepth(node) {
318
- if (!node.children || node.children.length === 0 || node.isExpanded === false) {
319
- return 1;
320
- }
321
- return 1 + Math.max(...node.children.map((child) => this.calculateMaxDepth(child)));
322
- }
323
- /**
324
- * Calculates the maximum breadth of the organizational chart.
325
- * @param node - The root node of the chart.
326
- * @returns The maximum breadth of the chart.
327
- */
328
- calculateMaxBreadth(node) {
329
- if (!node.children || node.children.length === 0 || node.isExpanded === false) {
330
- return 1;
331
- }
332
- return node.children.reduce((acc, child) => acc + this.calculateMaxBreadth(child), 0);
333
- }
334
- /**
335
- * Creates and renders the organizational chart.
336
- * @param el - The container element for the chart.
337
- * @param data - The root node of the organizational chart.
338
- */
339
- createChart(el, data) {
340
- if (!this.d3) {
341
- return;
342
- }
343
- this.ngzone.runOutsideAngular(() => {
344
- const NODE_WIDTH_BASE = 300;
345
- const NODE_HEIGHT_BASE = 220;
346
- const HORIZONTAL_SPACING = 20;
347
- const VERTICAL_SPACING = 40;
348
- this.clearChart(el);
349
- const newSvg = this.d3.select(el).append('svg');
350
- const root = this.d3.hierarchy(data);
351
- root.descendants().forEach((node) => {
352
- if (node.data.isExpanded === false) {
353
- node.children = undefined;
354
- }
355
- });
356
- const treeLayout = this.d3
357
- .tree()
358
- .nodeSize([NODE_WIDTH_BASE + HORIZONTAL_SPACING, NODE_HEIGHT_BASE + VERTICAL_SPACING]);
359
- treeLayout(root);
360
- let minX = Infinity, maxX = -Infinity, minY = Infinity, maxY = -Infinity;
361
- root.descendants().forEach((d) => {
362
- if (d.x < minX)
363
- minX = d.x;
364
- if (d.x > maxX)
365
- maxX = d.x;
366
- if (d.y < minY)
367
- minY = d.y;
368
- if (d.y > maxY)
369
- maxY = d.y;
370
- });
371
- const CONTAINER_WIDTH = maxX - minX + NODE_WIDTH_BASE + 2 * HORIZONTAL_SPACING;
372
- const CONTAINER_HEIGHT = maxY - minY + NODE_HEIGHT_BASE + 2 * VERTICAL_SPACING;
373
- newSvg.attr('width', CONTAINER_WIDTH).attr('height', CONTAINER_HEIGHT);
374
- const g = newSvg
375
- .append('g')
376
- .attr('transform', `translate(${-minX + NODE_WIDTH_BASE / 2 + HORIZONTAL_SPACING / 2}, ${-minY + NODE_HEIGHT_BASE / 2 + VERTICAL_SPACING / 2})`);
377
- const link = g
378
- .selectAll('.link')
379
- .data(root.links())
380
- .enter()
381
- .append('path')
382
- .attr('class', 'link')
383
- .attr('d', (d) => {
384
- const source = d.source;
385
- const target = d.target;
386
- return `M${source.x},${source.y}V${(source.y + target.y) / 2}H${target.x}V${target.y - 10}`;
387
- })
388
- .attr('fill', 'none')
389
- .attr('stroke', this.NODE_STROKE_COLOR)
390
- .attr('stroke-width', this.LINK_STROKE_WIDTH)
391
- .attr('from', (d) => `node-${d.source.data.id}`)
392
- .attr('to', (d) => `node-${d.target.data.id}`);
393
- const node = g
394
- .selectAll('.axp-chart-node')
395
- .data(root.descendants())
396
- .enter()
397
- .append('foreignObject')
398
- .attr('class', 'axp-chart-node')
399
- .attr('node', (d) => `node-${d.data.id}`)
400
- .attr('x', (d) => d.x - NODE_WIDTH_BASE / 2)
401
- .attr('y', (d) => d.y - NODE_HEIGHT_BASE / 2 + 10)
402
- .attr('width', NODE_WIDTH_BASE)
403
- .attr('height', NODE_HEIGHT_BASE)
404
- .each((d) => this.renderNodeTemplate(d.data, d.x, d.y));
405
- this.contextMenu()?.refresh();
406
- });
407
- if (!this.isChartRendered) {
408
- // document.querySelector(`foreignObject[node="node-${data.id}"] .--heading-container`)?.scrollIntoView({
409
- // behavior: 'smooth',
410
- // block: 'center',
411
- // inline: 'center',
412
- // });
413
- this.isChartRendered = true;
414
- this.panView()?.resetPosition();
415
- }
416
- }
417
- /**
418
- * Renders the node template for a given organizational node.
419
- * @param data - The node data.
420
- * @param x - The x-coordinate of the node.
421
- * @param y - The y-coordinate of the node.
422
- */
423
- renderNodeTemplate(data, x, y) {
424
- // Find the key that matches the node type
425
- // category is the same as type in simplified structure
426
- set(data, 'category', data.type);
427
- const viewRef = this.viewContainerRef.createEmbeddedView(this.nodeTemplate(), {
428
- $implicit: data,
429
- });
430
- const nodeElement = this.d3.select(this.treeContainer().nativeElement).select(`[node="node-${data.id}"]`);
431
- const element = nodeElement.node();
432
- if (element) {
433
- element.appendChild(viewRef.rootNodes[0]);
434
- }
435
- }
436
- /**
437
- * Toggles the expansion state of a node.
438
- * @param data - The node data.
439
- * @param el - The HTML element representing the node.
440
- */
441
- toggleNode(data, el) {
442
- data.isExpanded = data.isExpanded === undefined ? false : !data.isExpanded;
443
- this.createChart(this.treeContainer().nativeElement, this.chartService.rootNode());
444
- // el.scrollIntoView({
445
- // behavior: 'smooth',
446
- // });
447
- }
448
- /**
449
- * Clears the chart from the container element.
450
- * @param el - The container element.
451
- */
452
- clearChart(el) {
453
- if (this.d3) {
454
- this.d3.select(el).select('svg').remove();
455
- this.viewContainerRef.clear();
456
- }
457
- }
458
- async refresh() {
459
- if (this.d3) {
460
- await this.chartService.load();
461
- this.createChart(this.treeContainer().nativeElement, this.chartService.rootNode());
462
- }
463
- }
464
- //#region Context Menu
465
- // protected async handleContextMenuOnOpening(e: AXContextMenuOpeningEvent) {
466
- // const node = get(e.targetElement, '__data__.data') as any as AXMOrganizationNode;
467
- // const allowedAddItems = await this.chartService.getAllowedAddMenuItems(node);
468
- // const allowDelete = node.type != AXMOrganizationNodeType.Company;
469
- // const isPasteAvailable = await this.chartService.isPasteAvailable();
470
- // const allowPaste = await this.chartService.allowPaste(node);
471
- // //
472
- // if (allowedAddItems.length) {
473
- // e.items.push({
474
- // text: 'Add New',
475
- // icon: 'fa-light fa-plus',
476
- // items: allowedAddItems,
477
- // break: allowDelete,
478
- // });
479
- // }
480
- // //
481
- // if (allowDelete) {
482
- // e.items.push({
483
- // name: 'cut',
484
- // text: 'Cut',
485
- // icon: 'fa-light fa-cut',
486
- // data: node,
487
- // });
488
- // e.items.push({
489
- // name: 'copy',
490
- // text: 'Copy',
491
- // icon: 'fa-light fa-copy',
492
- // data: node,
493
- // break: !isPasteAvailable,
494
- // });
495
- // }
496
- // if (isPasteAvailable) {
497
- // e.items.push({
498
- // name: 'paste',
499
- // text: 'Paste',
500
- // icon: 'fa-light fa-paste',
501
- // data: node,
502
- // break: true,
503
- // disabled: !allowPaste,
504
- // });
505
- // }
506
- // //
507
- // if (allowDelete) {
508
- // e.items.push({
509
- // name: 'remove',
510
- // text: 'Remove',
511
- // color: 'danger',
512
- // icon: 'fa-light fa-trash-can',
513
- // data: node,
514
- // });
515
- // }
516
- // //
517
- // if (e.items.length == 0) {
518
- // e.canceled = true;
519
- // }
520
- // }
521
- async handleContextMenuItemClick(e) {
522
- }
523
- //#endregion
524
- /**
525
- * Prints the organizational chart.
526
- */
527
- async printChart() {
528
- let [panX, panY, panZoomLevel] = [this.panX(), this.panY(), this.panZoomLevel()];
529
- this.panX.set(0), this.panY.set(0), this.panZoomLevel.set(100);
530
- await this.chartService.printChart(this.treeContainer().nativeElement);
531
- this.panX.set(panX), this.panY.set(panY), this.panZoomLevel.set(panZoomLevel);
532
- }
533
- /**
534
- * Gets the node colors based on the provided color.
535
- * @param color - The color to use for the node.
536
- * @returns An object containing the node's background, text, and border colors.
537
- */
538
- getNodeColors(color) {
539
- if (color.includes('surface')) {
540
- return {
541
- '--node-bg-color': `var(--ax-sys-color-${color})`,
542
- '--node-text-color': `var(--ax-sys-color-on-${color})`,
543
- '--node-border-color': `var(--ax-sys-color-border-${color})`,
544
- };
545
- }
546
- return {
547
- '--node-bg-color': `var(--ax-sys-color-${color}-surface)`,
548
- '--node-text-color': `var(--ax-sys-color-on-${color}-surface)`,
549
- '--node-border-color': `var(--ax-sys-color-border-${color}-surface)`,
550
- };
551
- }
552
- updateVar(oldValue, newValue) {
553
- oldValue.cat = newValue;
554
- }
555
- async getPageTitle() {
556
- return await this.translateService.translateAsync('@organization-management:organization-chart.components.chart.title');
557
- }
558
- async getPageDescription() {
559
- return await this.translateService.translateAsync('@organization-management:organization-chart.components.chart.description');
560
- }
561
- async getSecondaryMenuItems() {
562
- return [
563
- {
564
- ...getActionButton('print'),
565
- command: {
566
- name: 'print',
567
- },
568
- },
569
- {
570
- ...getActionButton('refresh'),
571
- command: {
572
- name: 'refresh',
573
- },
574
- }
575
- ];
576
- }
577
- async getPageBreadcrumbs() {
578
- return [
579
- {
580
- title: await this.translateService.translateAsync('@organization-management:module.menus.root.title')
581
- },
582
- ];
583
- }
584
- async execute(command) {
585
- if (command.name == 'print') {
586
- await this.printChart();
587
- }
588
- if (command.name == 'refresh') {
589
- await this.chartService.generate();
590
- await this.refresh();
591
- }
592
- }
593
- ngOnDestroy() {
594
- this.clearChart(this.treeContainer().nativeElement);
595
- }
596
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: AXMOrgChartPage, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
597
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.9", type: AXMOrgChartPage, isStandalone: true, selector: "ng-component", providers: [
598
- AXMOrgChartService,
599
- {
600
- provide: AXPPageLayoutBase,
601
- useExisting: AXMOrgChartPage
602
- }
603
- ], viewQueries: [{ propertyName: "nodeTemplate", first: true, predicate: ["nodeTemplate"], descendants: true, isSignal: true }, { propertyName: "treeContainer", first: true, predicate: ["chart"], descendants: true, isSignal: true }, { propertyName: "parent", first: true, predicate: ["parent"], descendants: true, isSignal: true }, { propertyName: "contextMenu", first: true, predicate: AXContextMenuComponent, descendants: true, isSignal: true }, { propertyName: "panView", first: true, predicate: AXPanViewDirective, descendants: true, isSignal: true }], usesInheritance: true, ngImport: i0, template: "<axp-page-layout *translate=\"let t\">\n <axp-page-content class=\"ax-overflow-hidden\">\n <div class=\"ax-h-[calc(100vh-290px)]\" #parent id=\"parent\">\n <div #chart axPanView [(panX)]=\"panX\" [(panY)]=\"panY\" [freeMode]=\"true\" [(zoom)]=\"panZoomLevel\"\n [fitContent]=\"true\" wrapperClasses=\"h\" class=\"ax-light\"></div>\n </div>\n </axp-page-content>\n <axp-page-footer>\n <axp-layout-prefix>\n <div class=\"ax-flex ax-gap-1 ax-light ax-flex-wrap ax-justify-center\">\n @let v = {cat:'Company'}; @for (item of nodeTypes; track $index) {\n <div class=\"axp-node-hint\" [style]=\"getNodeColors(item.color)\" [class.ax-ms-4]=\"v.cat!=item.category\">\n <div class=\"--badge\"></div>\n {{item.title}}\n </div>\n {{updateVar(v, item.category)}} }\n </div>\n </axp-layout-prefix>\n <axp-layout-suffix>\n <div class=\"ax-flex ax-gap-2\">\n <ax-button (onClick)=\"zoomOut()\">\n <ax-icon class=\"fa-solid fa-minus\"> </ax-icon>\n </ax-button>\n <ax-button [text]=\"panZoomLevel()\" (onClick)=\"zoomReset()\"></ax-button>\n <ax-button (onClick)=\"zoomIn()\">\n <ax-icon class=\"fa-solid fa-add\"> </ax-icon>\n </ax-button>\n <ax-button (onClick)=\"toggleFullscreen()\">\n <ax-icon>\n <i class=\"fa-solid\" [class]=\"isFullscreen()? 'fa-compress' : 'fa-expand'\"></i>\n </ax-icon>\n </ax-button>\n </div>\n </axp-layout-suffix>\n </axp-page-footer>\n</axp-page-layout>\n\n<ng-template #nodeTemplate let-data>\n <div>\n @switch (data.category) {\n @case (NodeTypes.Company) {\n @if (data.metadata?.chartTemplate === 'merged-executive') {\n <ng-container *ngTemplateOutlet=\"mergedExecutiveTemplate; context: { data, color: 'primary' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'primary' }\"></ng-container>\n } }\n @case (NodeTypes.Country) {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent1' }\"></ng-container>\n } @case (NodeTypes.State) {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent1' }\"></ng-container>\n } @case (NodeTypes.City) {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent1' }\"></ng-container>\n }\n @case (NodeTypes.Division) {\n @if (data.metadata?.chartTemplate === 'merged-unit') {\n <ng-container *ngTemplateOutlet=\"mergedUnitTemplate; context: { data, color: 'accent2' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent2' }\"></ng-container>\n }\n } @case (NodeTypes.Department) {\n @if (data.metadata?.chartTemplate === 'merged-unit') {\n <ng-container *ngTemplateOutlet=\"mergedUnitTemplate; context: { data, color: 'accent2' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent2' }\"></ng-container>\n }\n } @case (NodeTypes.Branch) {\n @if (data.metadata?.chartTemplate === 'merged-unit') {\n <ng-container *ngTemplateOutlet=\"mergedUnitTemplate; context: { data, color: 'accent2' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent2' }\"></ng-container>\n }\n }\n @case (NodeTypes.Position) {\n @if (data.metadata?.chartTemplate === 'merged-manager') {\n <ng-container *ngTemplateOutlet=\"mergedManagerTemplate; context: { data, color: 'accent3' }\"></ng-container>\n } @else if (data.metadata?.chartTemplate === 'position-ic') {\n <ng-container *ngTemplateOutlet=\"positionIcTemplate; context: { data, color: 'accent3' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent3' }\"></ng-container>\n }\n }\n @case (NodeTypes.Employee) {\n @if (data.metadata?.chartTemplate === 'employee-card') {\n <ng-container *ngTemplateOutlet=\"employeeCardTemplate; context: { data, color: 'surface' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'surface' }\"></ng-container>\n } } }\n </div>\n</ng-template>\n\n<!--- Merged company (CEO / top role) -->\n<ng-template #mergedExecutiveTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --merged-card\" [class.--vacant]=\"data.metadata?.topRole?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--merged-top\">\n <span class=\"--merged-entity-title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.description) {\n <span class=\"--merged-entity-desc\">{{ data.description | axpResolveMultiLanguageString }}</span>\n }\n </div>\n @if (data.metadata?.topRole) {\n <div class=\"--merged-role-block\">\n <div class=\"--merged-avatar\"\n [class.--placeholder]=\"data.metadata.topRole.vacant || !data.metadata.topRole.employeeName\">\n {{ chartInitials(data.metadata.topRole.employeeName) }}\n </div>\n <div class=\"--merged-role-text\">\n @if (!data.metadata.topRole.vacant && data.metadata.topRole.employeeName) {\n <span class=\"--merged-name\">{{ data.metadata.topRole.employeeName }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.topRole.positionTitle | axpResolveMultiLanguageString\n }}</span>\n } @else {\n <span class=\"--merged-vacant\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.topRole.positionTitle | axpResolveMultiLanguageString\n }}</span>\n }\n </div>\n </div>\n }\n @if (data.metadata?.stats?.directChildren != null) {\n <div class=\"--merged-stats\">\n <span class=\"--merged-stat-n\">{{ data.metadata.stats.directChildren }}</span>\n <span class=\"--merged-stat-label\">{{ chartML.teamMembers | axpResolveMultiLanguageString }}</span>\n </div>\n }\n\n </div>\n </div>\n</ng-template>\n\n<!--- Merged business unit (head role) -->\n<ng-template #mergedUnitTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --merged-card --merged-unit\" [class.--vacant]=\"data.metadata?.unitHead?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--merged-top\">\n <span class=\"--merged-entity-title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.description) {\n <span class=\"--merged-entity-desc\">{{ data.description | axpResolveMultiLanguageString }}</span>\n }\n </div>\n @if (data.metadata?.unitHead) {\n <div class=\"--merged-role-block\">\n <div class=\"--merged-avatar\"\n [class.--placeholder]=\"data.metadata.unitHead.vacant || !data.metadata.unitHead.employeeName\">\n {{ chartInitials(data.metadata.unitHead.employeeName) }}\n </div>\n <div class=\"--merged-role-text\">\n @if (!data.metadata.unitHead.vacant && data.metadata.unitHead.employeeName) {\n <span class=\"--merged-name\">{{ data.metadata.unitHead.employeeName }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.unitHead.positionTitle | axpResolveMultiLanguageString\n }}</span>\n } @else {\n <span class=\"--merged-vacant\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.unitHead.positionTitle | axpResolveMultiLanguageString\n }}</span>\n }\n </div>\n </div>\n }\n @if (data.metadata?.stats?.directChildren != null) {\n <div class=\"--merged-stats\">\n <span class=\"--merged-stat-n\">{{ data.metadata.stats.directChildren }}</span>\n <span class=\"--merged-stat-label\">{{ chartML.teamMembers | axpResolveMultiLanguageString }}</span>\n </div>\n }\n @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>\n\n<!--- Manager position: seat + occupant merged -->\n<ng-template #mergedManagerTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --merged-card --merged-manager\" [class.--vacant]=\"data.metadata?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--merged-top --merged-top-compact\">\n <i class=\"fa-light fa-user-tie fa-fw --merged-type-icon\"></i>\n <span class=\"--merged-entity-title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n </div>\n <div class=\"--merged-role-block\">\n <div class=\"--merged-avatar\"\n [class.--placeholder]=\"data.metadata?.vacant || !data.metadata?.occupant?.displayName\">\n {{ chartInitials(data.metadata?.occupant?.displayName) }}\n </div>\n <div class=\"--merged-role-text\">\n @if (!data.metadata?.vacant && data.metadata?.occupant?.displayName) {\n <span class=\"--merged-name\">{{ data.metadata.occupant.displayName }}</span>\n @if (data.description) {\n <span class=\"--merged-position-sub\">{{ data.description | axpResolveMultiLanguageString }}</span>\n }\n } @else {\n <span class=\"--merged-vacant\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n }\n </div>\n </div>\n @if (data.metadata?.stats?.directReports != null && data.metadata.stats.directReports > 0) {\n <div class=\"--merged-stats\">\n <span class=\"--merged-stat-n\">{{ data.metadata.stats.directReports }}</span>\n <span class=\"--merged-stat-label\">{{ chartML.directReports | axpResolveMultiLanguageString }}</span>\n </div>\n }\n @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>\n\n<!--- IC position: seat box (employee child rendered separately) -->\n<ng-template #positionIcTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --ic-position\" [class.--vacant]=\"data.metadata?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--heading-container\">\n <div class=\"--content\">\n <div class=\"--icon\">\n <i class=\"fa-light fa-xl fa-fw\" [class]=\"data.icon\"></i>\n </div>\n <div class=\"--titles\">\n <span class=\"--title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.metadata?.vacant) {\n <span class=\"--sub-title\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n }\n </div>\n @if (data?.children?.length) {\n <div class=\"--badge\">\n <span>{{ data?.children?.length ?? 0 }}</span>\n </div>\n }\n </div>\n </div>\n @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>\n\n<!--- Employee under IC position -->\n<ng-template #employeeCardTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --employee-card\" [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--employee-row\">\n <div class=\"--merged-avatar --employee-avatar\">\n {{ chartInitials(data.title | axpResolveMultiLanguageString) }}\n </div>\n <div class=\"--employee-text\">\n <span class=\"--merged-name\">{{ data.title | axpResolveMultiLanguageString }}</span>\n @if (data.subtitle) {\n <span class=\"--merged-position-sub\">{{ data.subtitle | axpResolveMultiLanguageString }}</span>\n }\n </div>\n </div>\n </div>\n </div>\n</ng-template>\n\n<ng-template #generalTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container\" [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--heading-container\">\n <div class=\"--content\">\n <div class=\"--icon\">\n <i class=\"fa-light fa-xl fa-fw\" [class]=\"data.icon\"></i>\n </div>\n <div class=\"--titles\">\n <span class=\"--title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.subtitle) {\n <span class=\"--sub-title\" [title]=\"data.subtitle | axpResolveMultiLanguageString\">{{\n data.subtitle | axpResolveMultiLanguageString\n }}</span>\n }\n </div>\n @if (data?.children?.length) {\n <div class=\"--badge\">\n <span>{{ data?.children?.length ?? 0 }}</span>\n </div>\n }\n </div>\n </div>\n @if (data.description) {\n <div class=\"--body-container\">\n <div class=\"--description\" [class.ax-pb-3]=\"data.children && data.children.length > 0\">\n <span>{{ data.description | axpResolveMultiLanguageString }}</span>\n </div>\n </div>\n } @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>", styles: [".axp-node-hint{display:flex;align-items:center;justify-content:center;gap:.25rem;font-size:.75rem;line-height:1rem}.axp-node-hint>div.--badge{background-color:rgba(var(--node-bg-color));color:rgba(var(--node-text-color));display:flex;width:.75rem;height:.75rem;align-items:center;justify-content:center;border-radius:.125rem}.axp-chart-node{display:flex;align-items:center;justify-content:center}.axp-chart-node .--node-container{position:relative;cursor:pointer;--node-bg-color: var(--ax-sys-color-primary-surface);--node-text-color: var(--ax-sys-color-on-primary-surface);--node-border-color: var(--ax-sys-color-border-primary-surface)}.axp-chart-node .--node-container .--container{display:flex;max-height:100%;width:100%;flex-direction:column;overflow:hidden;border-radius:.375rem;border-width:1px;--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / .1), 0 2px 4px -2px rgb(0 0 0 / .1);--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow);border-color:rgba(var(--node-border-color))}.axp-chart-node .--node-container .--container .--heading-container{background-color:rgba(var(--node-bg-color));color:rgba(var(--node-text-color));width:100%;padding:.5rem .75rem}.axp-chart-node .--node-container .--container .--heading-container .--content{display:flex;width:100%;align-items:center;justify-content:center;gap:.5rem;overflow:hidden}.axp-chart-node .--node-container .--container .--heading-container .--content .--icon{background-color:color-mix(in srgb,rgba(var(--node-bg-color),1) 85%,black);display:flex;width:2.75rem;height:2.75rem;flex:none;align-items:center;justify-content:center;border-radius:.5rem}.axp-chart-node .--node-container .--container .--heading-container .--content .--badge{background-color:color-mix(in srgb,rgba(var(--node-bg-color),1) 85%,black);display:flex;width:1.5rem;height:1.5rem;flex:none;align-items:center;justify-content:center;border-radius:.375rem;line-height:1}.axp-chart-node .--node-container .--container .--heading-container .--content .--badge>span{font-size:.75rem;line-height:1rem}.axp-chart-node .--node-container .--container .--heading-container .--content .--titles{display:flex;width:100%;flex:1 1 0%;flex-direction:column;overflow:hidden}.axp-chart-node .--node-container .--container .--heading-container .--content .--titles .--title{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:1.125rem;line-height:1.75rem;font-weight:600}.axp-chart-node .--node-container .--container .--heading-container .--content .--titles .--sub-title{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.75rem;line-height:1rem;opacity:.85}.axp-chart-node .--node-container .--container .--body-container{width:100%;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.5rem .75rem;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container .--container .--body-container .--description{text-align:center;font-size:.875rem;line-height:1rem;opacity:.85}.axp-chart-node .--node-container .--container .--toggle{position:absolute;left:0;right:0;bottom:-.75rem;display:flex;width:100%;-webkit-user-select:none;user-select:none;align-items:center;justify-content:center}.axp-chart-node .--node-container.--merged-card .--container{min-width:280px}.axp-chart-node .--node-container.--merged-card.--vacant .--container{border-style:dashed;opacity:.95}.axp-chart-node .--node-container.--merged-card .--merged-top{background-color:rgba(var(--node-bg-color));color:rgba(var(--node-text-color));display:flex;flex-direction:column;gap:.125rem;padding:.5rem .75rem}.axp-chart-node .--node-container.--merged-card .--merged-top-compact{flex-direction:row;align-items:center;gap:.5rem}.axp-chart-node .--node-container.--merged-card .--merged-type-icon{opacity:.9}.axp-chart-node .--node-container.--merged-card .--merged-entity-title{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:1rem;line-height:1.5rem;font-weight:600;line-height:1.25}.axp-chart-node .--node-container.--merged-card .--merged-entity-desc{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:2;font-size:.75rem;line-height:1rem;opacity:.8}.axp-chart-node .--node-container.--merged-card .--merged-role-block{display:flex;align-items:center;gap:.5rem;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.5rem .75rem;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container.--merged-card .--merged-avatar{display:flex;width:2.5rem;height:2.5rem;flex:none;align-items:center;justify-content:center;border-radius:9999px;font-size:.75rem;line-height:1rem;font-weight:600;background-color:color-mix(in srgb,rgba(var(--ax-sys-color-primary-500),1) 18%,transparent);color:rgba(var(--ax-sys-color-on-lightest))}.axp-chart-node .--node-container.--merged-card .--merged-avatar.--placeholder{border-width:1px;border-style:dashed;border-color:rgba(var(--ax-sys-color-border-neutral));background-color:transparent}.axp-chart-node .--node-container.--merged-card .--merged-role-text{display:flex;min-width:0px;flex:1 1 0%;flex-direction:column;gap:.125rem}.axp-chart-node .--node-container.--merged-card .--merged-name{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.875rem;line-height:1.25rem;font-weight:600;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container.--merged-card .--merged-position{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.75rem;line-height:1rem;opacity:.8}.axp-chart-node .--node-container.--merged-card .--merged-position-sub{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.75rem;line-height:1rem;opacity:.7}.axp-chart-node .--node-container.--merged-card .--merged-vacant{font-size:.75rem;line-height:1rem;font-weight:500;text-transform:uppercase;letter-spacing:.025em;opacity:.75}.axp-chart-node .--node-container.--merged-card .--merged-actions{margin-top:.25rem;display:flex;flex-wrap:wrap;gap:.25rem}.axp-chart-node .--node-container.--merged-card .--merged-stats{display:flex;align-items:baseline;gap:.5rem;border-top-width:1px;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.375rem .75rem;border-color:rgba(var(--ax-sys-color-border-neutral))}.axp-chart-node .--node-container.--merged-card .--merged-stat-n{font-size:.875rem;line-height:1.25rem;font-weight:600}.axp-chart-node .--node-container.--merged-card .--merged-stat-label{font-size:.75rem;line-height:1rem;opacity:.75}.axp-chart-node .--node-container.--ic-position.--vacant .--container{border-style:dashed}.axp-chart-node .--node-container.--ic-position .--ic-actions{justify-content:flex-end;padding-left:.5rem;padding-right:.5rem;padding-bottom:.5rem}.axp-chart-node .--node-container.--employee-card .--container{min-width:220px}.axp-chart-node .--node-container.--employee-card .--employee-row{display:flex;align-items:center;gap:.5rem;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.5rem;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container.--employee-card .--employee-avatar{width:2.25rem;height:2.25rem;font-size:10px}.axp-chart-node .--node-container.--employee-card .--employee-text{display:flex;min-width:0px;flex-direction:column}#parent:fullscreen{background-color:rgba(var(--ax-sys-color-primary-lightest-surface))}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: RouterModule }, { kind: "ngmodule", type:
604
- //
605
- AXMenuModule }, { kind: "ngmodule", type: AXButtonModule }, { kind: "component", type: i1.AXButtonComponent, selector: "ax-button", inputs: ["disabled", "size", "tabIndex", "color", "look", "text", "toggleable", "selected", "iconOnly", "type", "loadingText"], outputs: ["onBlur", "onFocus", "onClick", "selectedChange", "toggleableChange", "lookChange", "colorChange", "disabledChange", "loadingTextChange"] }, { kind: "ngmodule", type: AXDropdownButtonModule }, { kind: "ngmodule", type: AXTranslationModule }, { kind: "directive", type: i4.AXTranslatorDirective, selector: "[translate]" }, { kind: "ngmodule", type: AXDecoratorModule }, { kind: "component", type: i2.AXDecoratorIconComponent, selector: "ax-icon", inputs: ["icon"] }, { kind: "directive", type: AXPanViewDirective, selector: "[axPanView]", inputs: ["zoomStep", "minZoom", "maxZoom", "freeMode", "fitContent", "disablePan", "disableZoom", "wrapperClasses", "panX", "panY", "zoom"], outputs: ["panXChange", "panYChange", "zoomChange", "positionChange"], exportAs: ["axPanView"] }, { kind: "ngmodule", type: AXLoadingModule }, { kind: "ngmodule", type: AXBreadcrumbsModule }, { kind: "ngmodule", type: AXBadgeModule }, { kind: "component", type:
606
- //
607
- AXPThemeLayoutBlockComponent, selector: " axp-page-content, axp-page-footer-container, axp-page-footer, axp-page-header, axp-page-header-container, axp-page-toolbar, axp-layout-content, axp-layout-page-content, axp-layout-sections, axp-layout-body, axp-layout-page-body, axp-layout-prefix, axp-layout-suffix, axp-layout-title-bar, axp-layout-title, axp-layout-title-actions, axp-layout-nav-button, axp-layout-description, axp-layout-breadcrumbs, axp-layout-list-action, " }, { kind: "component", type: AXPPageLayoutComponent, selector: "axp-page-layout" }, { kind: "pipe", type: AXPResolveMultiLanguageStringPipe, name: "axpResolveMultiLanguageString" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
608
- }
609
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: AXMOrgChartPage, decorators: [{
610
- type: Component,
611
- args: [{ imports: [
612
- CommonModule,
613
- RouterModule,
614
- //
615
- AXMenuModule,
616
- AXButtonModule,
617
- AXDropdownButtonModule,
618
- AXTranslationModule,
619
- AXDecoratorModule,
620
- AXPanViewDirective,
621
- AXLoadingModule,
622
- AXBreadcrumbsModule,
623
- AXBadgeModule,
624
- //
625
- AXPThemeLayoutBlockComponent,
626
- AXPPageLayoutComponent,
627
- AXPResolveMultiLanguageStringPipe,
628
- ], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, providers: [
629
- AXMOrgChartService,
630
- {
631
- provide: AXPPageLayoutBase,
632
- useExisting: AXMOrgChartPage
633
- }
634
- ], template: "<axp-page-layout *translate=\"let t\">\n <axp-page-content class=\"ax-overflow-hidden\">\n <div class=\"ax-h-[calc(100vh-290px)]\" #parent id=\"parent\">\n <div #chart axPanView [(panX)]=\"panX\" [(panY)]=\"panY\" [freeMode]=\"true\" [(zoom)]=\"panZoomLevel\"\n [fitContent]=\"true\" wrapperClasses=\"h\" class=\"ax-light\"></div>\n </div>\n </axp-page-content>\n <axp-page-footer>\n <axp-layout-prefix>\n <div class=\"ax-flex ax-gap-1 ax-light ax-flex-wrap ax-justify-center\">\n @let v = {cat:'Company'}; @for (item of nodeTypes; track $index) {\n <div class=\"axp-node-hint\" [style]=\"getNodeColors(item.color)\" [class.ax-ms-4]=\"v.cat!=item.category\">\n <div class=\"--badge\"></div>\n {{item.title}}\n </div>\n {{updateVar(v, item.category)}} }\n </div>\n </axp-layout-prefix>\n <axp-layout-suffix>\n <div class=\"ax-flex ax-gap-2\">\n <ax-button (onClick)=\"zoomOut()\">\n <ax-icon class=\"fa-solid fa-minus\"> </ax-icon>\n </ax-button>\n <ax-button [text]=\"panZoomLevel()\" (onClick)=\"zoomReset()\"></ax-button>\n <ax-button (onClick)=\"zoomIn()\">\n <ax-icon class=\"fa-solid fa-add\"> </ax-icon>\n </ax-button>\n <ax-button (onClick)=\"toggleFullscreen()\">\n <ax-icon>\n <i class=\"fa-solid\" [class]=\"isFullscreen()? 'fa-compress' : 'fa-expand'\"></i>\n </ax-icon>\n </ax-button>\n </div>\n </axp-layout-suffix>\n </axp-page-footer>\n</axp-page-layout>\n\n<ng-template #nodeTemplate let-data>\n <div>\n @switch (data.category) {\n @case (NodeTypes.Company) {\n @if (data.metadata?.chartTemplate === 'merged-executive') {\n <ng-container *ngTemplateOutlet=\"mergedExecutiveTemplate; context: { data, color: 'primary' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'primary' }\"></ng-container>\n } }\n @case (NodeTypes.Country) {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent1' }\"></ng-container>\n } @case (NodeTypes.State) {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent1' }\"></ng-container>\n } @case (NodeTypes.City) {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent1' }\"></ng-container>\n }\n @case (NodeTypes.Division) {\n @if (data.metadata?.chartTemplate === 'merged-unit') {\n <ng-container *ngTemplateOutlet=\"mergedUnitTemplate; context: { data, color: 'accent2' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent2' }\"></ng-container>\n }\n } @case (NodeTypes.Department) {\n @if (data.metadata?.chartTemplate === 'merged-unit') {\n <ng-container *ngTemplateOutlet=\"mergedUnitTemplate; context: { data, color: 'accent2' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent2' }\"></ng-container>\n }\n } @case (NodeTypes.Branch) {\n @if (data.metadata?.chartTemplate === 'merged-unit') {\n <ng-container *ngTemplateOutlet=\"mergedUnitTemplate; context: { data, color: 'accent2' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent2' }\"></ng-container>\n }\n }\n @case (NodeTypes.Position) {\n @if (data.metadata?.chartTemplate === 'merged-manager') {\n <ng-container *ngTemplateOutlet=\"mergedManagerTemplate; context: { data, color: 'accent3' }\"></ng-container>\n } @else if (data.metadata?.chartTemplate === 'position-ic') {\n <ng-container *ngTemplateOutlet=\"positionIcTemplate; context: { data, color: 'accent3' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'accent3' }\"></ng-container>\n }\n }\n @case (NodeTypes.Employee) {\n @if (data.metadata?.chartTemplate === 'employee-card') {\n <ng-container *ngTemplateOutlet=\"employeeCardTemplate; context: { data, color: 'surface' }\"></ng-container>\n } @else {\n <ng-container *ngTemplateOutlet=\"generalTemplate; context: { data, color: 'surface' }\"></ng-container>\n } } }\n </div>\n</ng-template>\n\n<!--- Merged company (CEO / top role) -->\n<ng-template #mergedExecutiveTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --merged-card\" [class.--vacant]=\"data.metadata?.topRole?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--merged-top\">\n <span class=\"--merged-entity-title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.description) {\n <span class=\"--merged-entity-desc\">{{ data.description | axpResolveMultiLanguageString }}</span>\n }\n </div>\n @if (data.metadata?.topRole) {\n <div class=\"--merged-role-block\">\n <div class=\"--merged-avatar\"\n [class.--placeholder]=\"data.metadata.topRole.vacant || !data.metadata.topRole.employeeName\">\n {{ chartInitials(data.metadata.topRole.employeeName) }}\n </div>\n <div class=\"--merged-role-text\">\n @if (!data.metadata.topRole.vacant && data.metadata.topRole.employeeName) {\n <span class=\"--merged-name\">{{ data.metadata.topRole.employeeName }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.topRole.positionTitle | axpResolveMultiLanguageString\n }}</span>\n } @else {\n <span class=\"--merged-vacant\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.topRole.positionTitle | axpResolveMultiLanguageString\n }}</span>\n }\n </div>\n </div>\n }\n @if (data.metadata?.stats?.directChildren != null) {\n <div class=\"--merged-stats\">\n <span class=\"--merged-stat-n\">{{ data.metadata.stats.directChildren }}</span>\n <span class=\"--merged-stat-label\">{{ chartML.teamMembers | axpResolveMultiLanguageString }}</span>\n </div>\n }\n\n </div>\n </div>\n</ng-template>\n\n<!--- Merged business unit (head role) -->\n<ng-template #mergedUnitTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --merged-card --merged-unit\" [class.--vacant]=\"data.metadata?.unitHead?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--merged-top\">\n <span class=\"--merged-entity-title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.description) {\n <span class=\"--merged-entity-desc\">{{ data.description | axpResolveMultiLanguageString }}</span>\n }\n </div>\n @if (data.metadata?.unitHead) {\n <div class=\"--merged-role-block\">\n <div class=\"--merged-avatar\"\n [class.--placeholder]=\"data.metadata.unitHead.vacant || !data.metadata.unitHead.employeeName\">\n {{ chartInitials(data.metadata.unitHead.employeeName) }}\n </div>\n <div class=\"--merged-role-text\">\n @if (!data.metadata.unitHead.vacant && data.metadata.unitHead.employeeName) {\n <span class=\"--merged-name\">{{ data.metadata.unitHead.employeeName }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.unitHead.positionTitle | axpResolveMultiLanguageString\n }}</span>\n } @else {\n <span class=\"--merged-vacant\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n <span class=\"--merged-position\">{{\n data.metadata.unitHead.positionTitle | axpResolveMultiLanguageString\n }}</span>\n }\n </div>\n </div>\n }\n @if (data.metadata?.stats?.directChildren != null) {\n <div class=\"--merged-stats\">\n <span class=\"--merged-stat-n\">{{ data.metadata.stats.directChildren }}</span>\n <span class=\"--merged-stat-label\">{{ chartML.teamMembers | axpResolveMultiLanguageString }}</span>\n </div>\n }\n @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>\n\n<!--- Manager position: seat + occupant merged -->\n<ng-template #mergedManagerTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --merged-card --merged-manager\" [class.--vacant]=\"data.metadata?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--merged-top --merged-top-compact\">\n <i class=\"fa-light fa-user-tie fa-fw --merged-type-icon\"></i>\n <span class=\"--merged-entity-title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n </div>\n <div class=\"--merged-role-block\">\n <div class=\"--merged-avatar\"\n [class.--placeholder]=\"data.metadata?.vacant || !data.metadata?.occupant?.displayName\">\n {{ chartInitials(data.metadata?.occupant?.displayName) }}\n </div>\n <div class=\"--merged-role-text\">\n @if (!data.metadata?.vacant && data.metadata?.occupant?.displayName) {\n <span class=\"--merged-name\">{{ data.metadata.occupant.displayName }}</span>\n @if (data.description) {\n <span class=\"--merged-position-sub\">{{ data.description | axpResolveMultiLanguageString }}</span>\n }\n } @else {\n <span class=\"--merged-vacant\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n }\n </div>\n </div>\n @if (data.metadata?.stats?.directReports != null && data.metadata.stats.directReports > 0) {\n <div class=\"--merged-stats\">\n <span class=\"--merged-stat-n\">{{ data.metadata.stats.directReports }}</span>\n <span class=\"--merged-stat-label\">{{ chartML.directReports | axpResolveMultiLanguageString }}</span>\n </div>\n }\n @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>\n\n<!--- IC position: seat box (employee child rendered separately) -->\n<ng-template #positionIcTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --ic-position\" [class.--vacant]=\"data.metadata?.vacant\"\n [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--heading-container\">\n <div class=\"--content\">\n <div class=\"--icon\">\n <i class=\"fa-light fa-xl fa-fw\" [class]=\"data.icon\"></i>\n </div>\n <div class=\"--titles\">\n <span class=\"--title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.metadata?.vacant) {\n <span class=\"--sub-title\">{{ chartML.vacant | axpResolveMultiLanguageString }}</span>\n }\n </div>\n @if (data?.children?.length) {\n <div class=\"--badge\">\n <span>{{ data?.children?.length ?? 0 }}</span>\n </div>\n }\n </div>\n </div>\n @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>\n\n<!--- Employee under IC position -->\n<ng-template #employeeCardTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container --employee-card\" [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--employee-row\">\n <div class=\"--merged-avatar --employee-avatar\">\n {{ chartInitials(data.title | axpResolveMultiLanguageString) }}\n </div>\n <div class=\"--employee-text\">\n <span class=\"--merged-name\">{{ data.title | axpResolveMultiLanguageString }}</span>\n @if (data.subtitle) {\n <span class=\"--merged-position-sub\">{{ data.subtitle | axpResolveMultiLanguageString }}</span>\n }\n </div>\n </div>\n </div>\n </div>\n</ng-template>\n\n<ng-template #generalTemplate let-data=\"data\" let-color=\"color\">\n <div #node class=\"--node-container\" [style]=\"getNodeColors(color)\">\n <div class=\"--container\">\n <div class=\"--heading-container\">\n <div class=\"--content\">\n <div class=\"--icon\">\n <i class=\"fa-light fa-xl fa-fw\" [class]=\"data.icon\"></i>\n </div>\n <div class=\"--titles\">\n <span class=\"--title\" [title]=\"data.title | axpResolveMultiLanguageString\">{{\n data.title | axpResolveMultiLanguageString\n }}</span>\n @if (data.subtitle) {\n <span class=\"--sub-title\" [title]=\"data.subtitle | axpResolveMultiLanguageString\">{{\n data.subtitle | axpResolveMultiLanguageString\n }}</span>\n }\n </div>\n @if (data?.children?.length) {\n <div class=\"--badge\">\n <span>{{ data?.children?.length ?? 0 }}</span>\n </div>\n }\n </div>\n </div>\n @if (data.description) {\n <div class=\"--body-container\">\n <div class=\"--description\" [class.ax-pb-3]=\"data.children && data.children.length > 0\">\n <span>{{ data.description | axpResolveMultiLanguageString }}</span>\n </div>\n </div>\n } @if (data.children && data.children.length > 0) {\n <div class=\"--toggle\">\n <ax-button class=\"ax-xs ax-rounded-sm\" (onClick)=\"toggleNode(data, node)\">\n <ax-icon icon=\"fa-solid\" [class]=\"data.isExpanded === false ? 'fa-chevron-down' : 'fa-chevron-up'\"></ax-icon>\n </ax-button>\n </div>\n }\n </div>\n </div>\n</ng-template>", styles: [".axp-node-hint{display:flex;align-items:center;justify-content:center;gap:.25rem;font-size:.75rem;line-height:1rem}.axp-node-hint>div.--badge{background-color:rgba(var(--node-bg-color));color:rgba(var(--node-text-color));display:flex;width:.75rem;height:.75rem;align-items:center;justify-content:center;border-radius:.125rem}.axp-chart-node{display:flex;align-items:center;justify-content:center}.axp-chart-node .--node-container{position:relative;cursor:pointer;--node-bg-color: var(--ax-sys-color-primary-surface);--node-text-color: var(--ax-sys-color-on-primary-surface);--node-border-color: var(--ax-sys-color-border-primary-surface)}.axp-chart-node .--node-container .--container{display:flex;max-height:100%;width:100%;flex-direction:column;overflow:hidden;border-radius:.375rem;border-width:1px;--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / .1), 0 2px 4px -2px rgb(0 0 0 / .1);--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow);border-color:rgba(var(--node-border-color))}.axp-chart-node .--node-container .--container .--heading-container{background-color:rgba(var(--node-bg-color));color:rgba(var(--node-text-color));width:100%;padding:.5rem .75rem}.axp-chart-node .--node-container .--container .--heading-container .--content{display:flex;width:100%;align-items:center;justify-content:center;gap:.5rem;overflow:hidden}.axp-chart-node .--node-container .--container .--heading-container .--content .--icon{background-color:color-mix(in srgb,rgba(var(--node-bg-color),1) 85%,black);display:flex;width:2.75rem;height:2.75rem;flex:none;align-items:center;justify-content:center;border-radius:.5rem}.axp-chart-node .--node-container .--container .--heading-container .--content .--badge{background-color:color-mix(in srgb,rgba(var(--node-bg-color),1) 85%,black);display:flex;width:1.5rem;height:1.5rem;flex:none;align-items:center;justify-content:center;border-radius:.375rem;line-height:1}.axp-chart-node .--node-container .--container .--heading-container .--content .--badge>span{font-size:.75rem;line-height:1rem}.axp-chart-node .--node-container .--container .--heading-container .--content .--titles{display:flex;width:100%;flex:1 1 0%;flex-direction:column;overflow:hidden}.axp-chart-node .--node-container .--container .--heading-container .--content .--titles .--title{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:1.125rem;line-height:1.75rem;font-weight:600}.axp-chart-node .--node-container .--container .--heading-container .--content .--titles .--sub-title{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.75rem;line-height:1rem;opacity:.85}.axp-chart-node .--node-container .--container .--body-container{width:100%;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.5rem .75rem;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container .--container .--body-container .--description{text-align:center;font-size:.875rem;line-height:1rem;opacity:.85}.axp-chart-node .--node-container .--container .--toggle{position:absolute;left:0;right:0;bottom:-.75rem;display:flex;width:100%;-webkit-user-select:none;user-select:none;align-items:center;justify-content:center}.axp-chart-node .--node-container.--merged-card .--container{min-width:280px}.axp-chart-node .--node-container.--merged-card.--vacant .--container{border-style:dashed;opacity:.95}.axp-chart-node .--node-container.--merged-card .--merged-top{background-color:rgba(var(--node-bg-color));color:rgba(var(--node-text-color));display:flex;flex-direction:column;gap:.125rem;padding:.5rem .75rem}.axp-chart-node .--node-container.--merged-card .--merged-top-compact{flex-direction:row;align-items:center;gap:.5rem}.axp-chart-node .--node-container.--merged-card .--merged-type-icon{opacity:.9}.axp-chart-node .--node-container.--merged-card .--merged-entity-title{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:1rem;line-height:1.5rem;font-weight:600;line-height:1.25}.axp-chart-node .--node-container.--merged-card .--merged-entity-desc{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:2;font-size:.75rem;line-height:1rem;opacity:.8}.axp-chart-node .--node-container.--merged-card .--merged-role-block{display:flex;align-items:center;gap:.5rem;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.5rem .75rem;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container.--merged-card .--merged-avatar{display:flex;width:2.5rem;height:2.5rem;flex:none;align-items:center;justify-content:center;border-radius:9999px;font-size:.75rem;line-height:1rem;font-weight:600;background-color:color-mix(in srgb,rgba(var(--ax-sys-color-primary-500),1) 18%,transparent);color:rgba(var(--ax-sys-color-on-lightest))}.axp-chart-node .--node-container.--merged-card .--merged-avatar.--placeholder{border-width:1px;border-style:dashed;border-color:rgba(var(--ax-sys-color-border-neutral));background-color:transparent}.axp-chart-node .--node-container.--merged-card .--merged-role-text{display:flex;min-width:0px;flex:1 1 0%;flex-direction:column;gap:.125rem}.axp-chart-node .--node-container.--merged-card .--merged-name{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.875rem;line-height:1.25rem;font-weight:600;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container.--merged-card .--merged-position{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.75rem;line-height:1rem;opacity:.8}.axp-chart-node .--node-container.--merged-card .--merged-position-sub{overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.75rem;line-height:1rem;opacity:.7}.axp-chart-node .--node-container.--merged-card .--merged-vacant{font-size:.75rem;line-height:1rem;font-weight:500;text-transform:uppercase;letter-spacing:.025em;opacity:.75}.axp-chart-node .--node-container.--merged-card .--merged-actions{margin-top:.25rem;display:flex;flex-wrap:wrap;gap:.25rem}.axp-chart-node .--node-container.--merged-card .--merged-stats{display:flex;align-items:baseline;gap:.5rem;border-top-width:1px;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.375rem .75rem;border-color:rgba(var(--ax-sys-color-border-neutral))}.axp-chart-node .--node-container.--merged-card .--merged-stat-n{font-size:.875rem;line-height:1.25rem;font-weight:600}.axp-chart-node .--node-container.--merged-card .--merged-stat-label{font-size:.75rem;line-height:1rem;opacity:.75}.axp-chart-node .--node-container.--ic-position.--vacant .--container{border-style:dashed}.axp-chart-node .--node-container.--ic-position .--ic-actions{justify-content:flex-end;padding-left:.5rem;padding-right:.5rem;padding-bottom:.5rem}.axp-chart-node .--node-container.--employee-card .--container{min-width:220px}.axp-chart-node .--node-container.--employee-card .--employee-row{display:flex;align-items:center;gap:.5rem;--tw-bg-opacity: 1;background-color:rgba(var(--ax-sys-color-lightest-surface),var(--tw-bg-opacity, 1));padding:.5rem;--tw-text-opacity: 1;color:rgba(var(--ax-sys-color-on-lightest-surface),var(--tw-text-opacity, 1))}.axp-chart-node .--node-container.--employee-card .--employee-avatar{width:2.25rem;height:2.25rem;font-size:10px}.axp-chart-node .--node-container.--employee-card .--employee-text{display:flex;min-width:0px;flex-direction:column}#parent:fullscreen{background-color:rgba(var(--ax-sys-color-primary-lightest-surface))}\n"] }]
635
- }], propDecorators: { nodeTemplate: [{ type: i0.ViewChild, args: ['nodeTemplate', { isSignal: true }] }], treeContainer: [{ type: i0.ViewChild, args: ['chart', { isSignal: true }] }], parent: [{ type: i0.ViewChild, args: ['parent', { isSignal: true }] }], contextMenu: [{ type: i0.ViewChild, args: [i0.forwardRef(() => AXContextMenuComponent), { isSignal: true }] }], panView: [{ type: i0.ViewChild, args: [i0.forwardRef(() => AXPanViewDirective), { isSignal: true }] }] } });
636
-
637
- export { AXMOrgChartPage };
638
- //# sourceMappingURL=acorex-modules-organization-management-org-chart.page-E0hy7dXg.mjs.map