@ruc-lib/org-chart 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,202 @@
1
+ # ruclib-org-chart
2
+
3
+ A component for creating a org chart. There are various features like Downloading, Collapseing and Expanding available in it.
4
+
5
+ # Features
6
+
7
+ - User is allowed to input the data source in the correct JSON format (specific to organizational structure); otherwise, print console for that specific missing properties.
8
+ - User is allowed to select different templates for displaying the organizational chart, including various node shapes such as Portrait, Landscape, and Triangle.
9
+ - User is allowed to search by title or subtitle in the organizational chart, with the ability to toggle the visibility of the search bar based on configuration.
10
+ - User is allowed to see the searched node highlighted, while other nodes will be faded when a search is performed.
11
+ - User is allowed to access a hamburger menu (three horizontal lines) that displays a list of options such as Expand All, Collapse All, Download PNG, Download SVG, and Download PDF.
12
+ - User is allowed to expand all nodes by clicking the "Expand All" button and collapse all nodes by clicking the "Collapse All" button.
13
+ - User is allowed to download the organizational chart in PNG format by clicking the "Download PNG" button, in SVG format with the "Download SVG" option, and in PDF format using the "Download PDF" button.
14
+ - User is allowed to hover over any node in the chart, which will highlight the parent and child nodes while fading other nodes, along with displaying a tooltip containing the node's description.
15
+ - User is allowed to click the expand icon on any node to toggle between expanding or collapsing the node, provided it has child data.
16
+ - User is allowed to configure custom CSS for the organizational chart, including styling options like node color, background color.
17
+
18
+
19
+ # Installation guide
20
+
21
+ # Install complete library
22
+
23
+ `npm install @uxpractice/ruc-lib`
24
+
25
+ # Install individual component
26
+
27
+ If users only need the dataflow component, they can install it separately
28
+ `npm install @ruc-lib/org-chart`
29
+
30
+ # Usage
31
+
32
+ After installing the the required package successfully, we have to import the CSS in our style.scss file in following way.
33
+
34
+ `@import "../../../node_modules/primeng/resources/themes/lara-light-indigo/theme.css";`
35
+ `@import "../../../node_modules/primeng/resources/primeng.min.css";`
36
+
37
+ here path of the scss file is subject to change as per choice of the installation
38
+
39
+ for library
40
+ `@import "../node_modules/primeng/resources/themes/lara-light-indigo/theme.css";`
41
+ `@import "../node_modules/primeng/resources/primeng.min.css";`
42
+
43
+ for seperate package
44
+ `@import "../node_modules/primeng/resources/themes/lara-light-indigo/theme.css";`
45
+ `@import "../node_modules/primeng/resources/primeng.min.css";`
46
+
47
+
48
+ # import required modules
49
+
50
+ for library
51
+ `import { RuclibOrgChartModule } from '@uxpractice/ruc-lib/org-chart';`
52
+
53
+ for seperate package
54
+ `import { RuclibOrgChartModule } from '@ruc-lib/org-chart';`
55
+
56
+ # use component selector
57
+
58
+ ```
59
+ <uxp-org-chart [rucInputData]="inputOrgData" [customTheme]="customTheme"></uxp-org-chart>
60
+ ```
61
+
62
+
63
+ # Input and Output
64
+
65
+ # Inputs
66
+ rucInputData -> It is the configuration input to configure the org chart
67
+
68
+ customTheme -> It is the name of the theme.
69
+ # Output
70
+
71
+ # rucInputData (sample object)
72
+
73
+ # Detail definition of the each property can be found in type definition file.
74
+
75
+ ```
76
+ const inputOrgData = {
77
+ isDisplayHambergerMenu: true,
78
+ isDisplaySearchBar : true,
79
+ nodeTemplate: 'triangle', // portrait , triangle, landscape
80
+ hambergerMenuList : [
81
+ { label: 'Download pdf',
82
+ id: 1
83
+ },
84
+ { label: 'Download png',
85
+ id: 2
86
+ },
87
+ { label: 'Download jpeg',
88
+ id: 3
89
+ },
90
+ {
91
+ label: 'Expand All',
92
+ id: 4,
93
+ },
94
+ {
95
+ label: 'Collapse All',
96
+ id: 5,
97
+ }
98
+ ],
99
+ greyNodeStyle : {
100
+ backgroundColor: '#d3d3d3',
101
+ color: '#808080',
102
+ },
103
+ orgData: [
104
+ {
105
+ label: 'ceo',
106
+ expanded: false,
107
+ description: 'ceo description',
108
+ customNodeStyle: {
109
+ backgroundColor: '#3498db',
110
+ color: 'white',
111
+ padding: '1.5em',
112
+ borderRadius: '0px',
113
+ },
114
+
115
+ data: {
116
+ image:'https://primefaces.org/cdn/primeng/images/demo/avatar/amyelsner.png',
117
+ name: 'Amy Elsner',
118
+ title: 'CEO',
119
+ },
120
+ children: [
121
+ {
122
+ expanded: false,
123
+ label: 'cmo',
124
+ description: 'cmo description',
125
+ customNodeStyle: {
126
+ backgroundColor: 'rgb(152, 1, 4)',
127
+ color: 'white',
128
+ padding: '1.5em',
129
+ borderRadius: '0px',
130
+ },
131
+ data: {
132
+ image:
133
+ 'https://primefaces.org/cdn/primeng/images/demo/avatar/annafali.png',
134
+ name: 'Ahaanna Fali',
135
+ title: 'CMO',
136
+ }
137
+ },
138
+ {
139
+ expanded: false,
140
+ label: 'CTO',
141
+ description: 'CTO description',
142
+ customNodeStyle: {
143
+ backgroundColor: 'rgb(152, 1, 4)',
144
+ color: 'white',
145
+ padding: '1.5em',
146
+ borderRadius: '0px',
147
+ },
148
+ data: {
149
+ image:
150
+ 'https://primefaces.org/cdn/primeng/images/demo/avatar/stephenshaw.png',
151
+ name: 'Stephen Shaw',
152
+ title: 'CTO',
153
+ },
154
+ children: [
155
+ {
156
+ expanded: false,
157
+ label: 'Architect',
158
+ description: 'CTO description',
159
+ customNodeStyle: {
160
+ backgroundColor: 'rgb(137, 152, 1)',
161
+ color: 'white',
162
+ padding: '1.5em',
163
+ borderRadius: '0px',
164
+ },
165
+ data: {
166
+ image:
167
+ 'https://primefaces.org/cdn/primeng/images/demo/avatar/stephenshaw.png',
168
+ name: 'Thomas Shaw',
169
+ title: 'Architect',
170
+ }
171
+ },
172
+ {
173
+ expanded: false,
174
+ label: 'Manager',
175
+ description: 'CTO description',
176
+ customNodeStyle: {
177
+ backgroundColor: 'rgb(137, 152, 1)',
178
+ color: 'white',
179
+ padding: '1.5em',
180
+ borderRadius: '0px',
181
+ },
182
+ data: {
183
+ image:
184
+ 'https://primefaces.org/cdn/primeng/images/demo/avatar/stephenshaw.png',
185
+ name: 'Steve Shaw',
186
+ title: 'Manager',
187
+ }
188
+ }],
189
+ },
190
+ ],
191
+ },
192
+ ]
193
+ }
194
+
195
+ ```
196
+
197
+
198
+ # Contribution
199
+ Contributions are welcome! Feel free to open issues or pull requests for any enhancements or fixes.
200
+
201
+ # Acknowledgements
202
+ Thank you for choosing the Data flow Component Library. If you have any feedback or suggestions, please let us know!
@@ -0,0 +1,4 @@
1
+ export * from './lib/ruclib-org-chart.module';
2
+ export * from './lib/org-chart/org-chart.component';
3
+ // export * from './services/org-chart.service';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLHFDQUFxQyxDQUFDO0FBQ3BELGdEQUFnRCIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vbGliL3J1Y2xpYi1vcmctY2hhcnQubW9kdWxlJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvb3JnLWNoYXJ0L29yZy1jaGFydC5jb21wb25lbnQnO1xyXG4vLyBleHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzL29yZy1jaGFydC5zZXJ2aWNlJzsiXX0=
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3JnLWNoYXJ0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2ludGVyZmFjZXMvb3JnLWNoYXJ0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUcmVlTm9kZSB9IGZyb20gJ3ByaW1lbmcvYXBpJztcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgQ3VzdG9tVHJlZU5vZGUgZXh0ZW5kcyBUcmVlTm9kZSB7XHJcbiAgICBjdXN0b21Ob2RlU3R5bGU/OiB7IFtrZXk6IHN0cmluZ106IHN0cmluZyB8IG51bWJlciB9OyAvLyBTdHlsaW5nIGZvciBvcmcgY2hhcnQgbm9kZVxyXG4gICAgb3JpZ2luYWxTdHlsZT86IHsgW2tleTogc3RyaW5nXTogc3RyaW5nIHwgbnVtYmVyIH07IC8vIGlnbm9yZSBpdCBhcyBpdHMgYXNzaWduaW5nIGluc2lkZSBsaWJyYXJ5XHJcbiAgICBkZXNjcmlwdGlvbj86IHN0cmluZzsgLy8gZGVzY3JpcHRpb24gaW5zaWRlIG5vZGUgd2hlbiB3ZSBob3ZlciBvbiBpdFxyXG59XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIEN1c3RvbU5vZGVTdHlsZSB7XHJcbiAgYmFja2dyb3VuZENvbG9yPzogc3RyaW5nO1xyXG4gIGNvbG9yPzogc3RyaW5nO1xyXG4gIHBhZGRpbmc/OiBzdHJpbmc7XHJcbiAgYm9yZGVyUmFkaXVzPzogc3RyaW5nO1xyXG59XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIE9yZ0RhdGFJdGVtIHtcclxuICBsYWJlbD86IHN0cmluZztcclxuICBleHBhbmRlZD86IGJvb2xlYW47IFxyXG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xyXG4gIGN1c3RvbU5vZGVTdHlsZT86IEN1c3RvbU5vZGVTdHlsZTsgLy8gQ3VzdG9tIHN0eWxpbmcgZm9yIG5vZGVcclxuICB0eXBlPzogc3RyaW5nOyAvLyBpZ25vcmUgaXQgYXMgaXQgaXMgYXNzaWduaW5nIGluc2lkZSBsaWJyYXJ5IFxyXG4gIGRhdGE/OiB7XHJcbiAgICBpbWFnZT86IHN0cmluZztcclxuICAgIG5hbWU/OiBzdHJpbmc7XHJcbiAgICB0aXRsZTogc3RyaW5nO1xyXG4gIH07XHJcbiAgY2hpbGRyZW4/OiBPcmdEYXRhSXRlbVtdO1xyXG4gIG9yaWdpbmFsU3R5bGU/OiBDdXN0b21Ob2RlU3R5bGU7IC8vIEN1c3RvbSBzdHlsaW5nIGZvciBub2RlXHJcbn1cclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgT3JnRGF0YSB7XHJcbiAgaXNEaXNwbGF5SGFtYmVyZ2VyTWVudT86IGJvb2xlYW47IC8vIFRvIGRpc3BsYXkgU2VhcmNoIGJhci4gSXQgY2FuIGJlIHRydWUgfCBmYWxzZVxyXG4gIGlzRGlzcGxheVNlYXJjaEJhcj86IGJvb2xlYW47IC8vIFRvIGRpc3BsYXkgU2VhcmNoIGJhci4gSXQgY2FuIGJlIHRydWUgfCBmYWxzZVxyXG4gIG5vZGVUZW1wbGF0ZTogc3RyaW5nOyAgLy8gVGVtcGxhdGUgaGF2ZSB0aHJlZSBvcHRpb25zICdwb3J0cmFpdCcgfCAndHJpYW5nbGUnIHwgJ2xhbmRzY2FwZSc7XHJcbiAgaGFtYmVyZ2VyTWVudUxpc3Q6IHsgbGFiZWw6IHN0cmluZzsgaWQ6IG51bWJlciB9W107XHJcbiAgZ3JleU5vZGVTdHlsZTogQ3VzdG9tTm9kZVN0eWxlO1xyXG4gIG9yZ0RhdGE6IE9yZ0RhdGFJdGVtW107XHJcbn1cclxuICAiXX0=
@@ -0,0 +1,209 @@
1
+ import { Component, ViewChild, Input } from '@angular/core';
2
+ import { checkPropsType, downloadChart } from '../../utils/chart-download.util';
3
+ import { OrganizationChart } from 'primeng/organizationchart';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/common";
6
+ import * as i2 from "primeng/api";
7
+ import * as i3 from "@angular/forms";
8
+ import * as i4 from "primeng/organizationchart";
9
+ import * as i5 from "@angular/material/input";
10
+ import * as i6 from "@angular/material/form-field";
11
+ import * as i7 from "@angular/material/menu";
12
+ import * as i8 from "@angular/material/button";
13
+ import * as i9 from "@angular/material/icon";
14
+ /** Displays an organizational chart with nodes and relationships (Org Chart Component) */
15
+ export class OrgChartComponent {
16
+ constructor() {
17
+ this.searchText = '';
18
+ this.hoveredNode = null;
19
+ }
20
+ /** Initializes the component */
21
+ ngOnInit() {
22
+ let updatedJson = this.addDefaultKeysIfNotAvailable(this.rucInputData);
23
+ this.rucInputData.orgData = this.updateRucInputData(updatedJson.orgData);
24
+ checkPropsType(this.rucInputData); // To check rucInputData is in correct format or not
25
+ this.expandAllNodes();
26
+ }
27
+ /** add default property for grey nodes if user does not provided */
28
+ addDefaultKeysIfNotAvailable(orgData) {
29
+ if (!orgData.hasOwnProperty('greyNodeStyle')) {
30
+ orgData.greyNodeStyle = {
31
+ backgroundColor: '#d3d3d3',
32
+ color: '#808080',
33
+ };
34
+ }
35
+ return orgData;
36
+ }
37
+ /** manipulate input json as If customNodeStyle exists, create originalStyle for node and its children */
38
+ updateRucInputData(data) {
39
+ return data.map((node) => {
40
+ node['type'] = 'person';
41
+ if (node.customNodeStyle) {
42
+ node.originalStyle = { ...node.customNodeStyle };
43
+ }
44
+ if (node.children && node.children.length > 0) {
45
+ node['type'] = 'person';
46
+ node.children = this.updateRucInputData(node.children); // Recursive call
47
+ }
48
+ return node;
49
+ });
50
+ }
51
+ /** Expands all nodes in the org chart */
52
+ expandAllNodes() {
53
+ this.rucInputData.orgData = this.expandNodes(this.rucInputData.orgData);
54
+ }
55
+ /** Recursively expands all nodes in the given array
56
+ * @param nodes Nodes to expand @returns Expanded nodes */
57
+ expandNodes(nodes) {
58
+ return nodes.map((node) => ({
59
+ ...node,
60
+ expanded: true,
61
+ children: node.children ? this.expandNodes(node.children) : [],
62
+ }));
63
+ }
64
+ /** Collapses all nodes in the org chart */
65
+ collapseAllNodes() {
66
+ this.rucInputData.orgData = this.collapseNodes(this.rucInputData.orgData);
67
+ }
68
+ /** Recursively collapses all nodes in the given array
69
+ * @param nodes Nodes to collapse @returns Collapsed nodes */
70
+ collapseNodes(nodes) {
71
+ return nodes.map((node) => ({
72
+ ...node,
73
+ expanded: false,
74
+ children: node.children ? this.collapseNodes(node.children) : [],
75
+ }));
76
+ }
77
+ /** Gets the background color of a node based on search text
78
+ @param node Node to check @returns Whether the node matches the search text */
79
+ getBackgroungColorOfNode(node) {
80
+ return (!this.searchText ||
81
+ node.data.title.toLowerCase().includes(this.searchText.toLowerCase()) ||
82
+ node.data.name.toLowerCase().includes(this.searchText.toLowerCase()) ||
83
+ node.label?.toLowerCase().includes(this.searchText.toLowerCase()));
84
+ }
85
+ /** Handles node hover event @param node Hovered node */
86
+ onNodeHover(node) {
87
+ if (this.searchText)
88
+ return;
89
+ this.hoveredNode = node;
90
+ this.updateNodeStyles();
91
+ }
92
+ /** Handles node leave event */
93
+ onNodeLeave() {
94
+ this.hoveredNode = null;
95
+ this.resetNodeStyles();
96
+ }
97
+ /** Updates node styles based on the hovered node */
98
+ updateNodeStyles() {
99
+ this.applyStyles(this.rucInputData.orgData, true);
100
+ }
101
+ /** Resets node styles to their original state */
102
+ resetNodeStyles() {
103
+ this.applyStyles(this.rucInputData.orgData, false);
104
+ }
105
+ /** Applies grey node style to non-matching nodes @param nodes Nodes to apply styles to */
106
+ applyGreyNodes(nodes) {
107
+ nodes.forEach((node) => {
108
+ const isMatching = this.searchText
109
+ ? node.data?.title.includes(this.searchText)
110
+ : false;
111
+ node.customNodeStyle = isMatching
112
+ ? node.customNodeStyle || node.originalStyle
113
+ : {
114
+ ...node.originalStyle,
115
+ backgroundColor: this.rucInputData.greyNodeStyle.backgroundColor,
116
+ color: this.rucInputData.greyNodeStyle.color,
117
+ };
118
+ if (node.children) {
119
+ this.applyGreyNodes(node.children);
120
+ }
121
+ });
122
+ }
123
+ /** Applies styles to nodes based on the hovered node
124
+ * @param nodes Nodes to apply styles @param isHovered Whether the node is hovered */
125
+ applyStyles(nodes, isHovered) {
126
+ nodes.forEach((node) => {
127
+ if (isHovered) {
128
+ node.customNodeStyle =
129
+ node === this.hoveredNode ||
130
+ this.isParent(node, this.hoveredNode) ||
131
+ this.isChild(node, this.hoveredNode)
132
+ ? node.customNodeStyle || node.originalStyle
133
+ : {
134
+ ...node.originalStyle,
135
+ backgroundColor: this.rucInputData.greyNodeStyle.backgroundColor,
136
+ color: this.rucInputData.greyNodeStyle.color,
137
+ };
138
+ }
139
+ else {
140
+ node.customNodeStyle = node.originalStyle;
141
+ }
142
+ if (node.children) {
143
+ this.applyStyles(node.children, isHovered);
144
+ }
145
+ });
146
+ }
147
+ /** Use the downloadChart utility method */
148
+ downloadChart(format) {
149
+ const chartContainer = this.orgChart.el.nativeElement;
150
+ downloadChart(chartContainer, format); // Calls the utility function
151
+ }
152
+ /** Handles menu click event @param id Menu item ID */
153
+ onMenuClick(id, event) {
154
+ event.preventDefault();
155
+ event.stopPropagation();
156
+ switch (id) {
157
+ case 1:
158
+ this.downloadChart('pdf');
159
+ break;
160
+ case 2:
161
+ this.downloadChart('png');
162
+ break;
163
+ case 3:
164
+ this.downloadChart('jpeg');
165
+ break;
166
+ case 4:
167
+ this.expandAllNodes();
168
+ break;
169
+ case 5:
170
+ this.collapseAllNodes();
171
+ break;
172
+ default:
173
+ console.warn(`Unknown menu item ID: ${id}. Please enter valid id.`);
174
+ break;
175
+ }
176
+ }
177
+ /** Gets tooltip data for a node @param node Node to get tooltip data */
178
+ getTooltipData(node) {
179
+ return node.description || '';
180
+ }
181
+ /** Checks if a node is a parent of another node.
182
+ * @param currentNode The current node to check.
183
+ * @param hoveredNode The node to check for parentage.
184
+ * @returns Whether the current node is a parent of the hovered node. */
185
+ isParent(currentNode, hoveredNode) {
186
+ return hoveredNode && currentNode.children?.includes(hoveredNode);
187
+ }
188
+ /** Checks if a node is a child of another node.
189
+ * @param node The node to check for child status
190
+ * @param target The node to check for parentage.
191
+ * @returns Whether the node is a child of the target node.*/
192
+ isChild(node, target) {
193
+ return target && target.children?.includes(node);
194
+ }
195
+ }
196
+ OrgChartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: OrgChartComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
197
+ OrgChartComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: OrgChartComponent, selector: "uxp-org-chart", inputs: { rucInputData: "rucInputData", customTheme: "customTheme" }, viewQueries: [{ propertyName: "orgChart", first: true, predicate: ["orgChart"], descendants: true }], ngImport: i0, template: "<div class={{customTheme}}>\r\n <div class=\"container\">\r\n <!-- Hamburger Menu -->\r\n <button *ngIf=\"rucInputData.isDisplayHambergerMenu === true\" mat-icon-button [matMenuTriggerFor]=\"menu\" \r\n class=\"left hamburger-button\">\r\n <mat-icon>menu</mat-icon>\r\n </button>\r\n \r\n <!-- Menu -->\r\n <mat-menu #menu=\"matMenu\">\r\n <button *ngFor=\"let menu of rucInputData.hambergerMenuList\" mat-menu-item (click)=\"onMenuClick(menu.id, $event);\" \r\n [id]=\"'menu-item-' + menu.id\">\r\n {{ menu.label }}\r\n </button>\r\n </mat-menu>\r\n \r\n <!-- Search Bar -->\r\n <mat-form-field *ngIf=\"rucInputData.isDisplaySearchBar === true\" class=\"search-box right\" \r\n >\r\n <mat-label>Search</mat-label>\r\n <input matInput placeholder=\"Type to search...\" [(ngModel)]=\"searchText\" />\r\n <mat-icon matSuffix>search</mat-icon>\r\n </mat-form-field>\r\n </div>\r\n \r\n \r\n <div class=\"org-chart-container \" id=\"org-chart-data\">\r\n <p-organizationChart #orgChart class=\"org-chart\" [value]=\"this.rucInputData.orgData\" selectionMode=\"multiple\"\r\n [(selection)]=\"selectedNodes\">\r\n <ng-template let-node pTemplate=\"person\">\r\n <div *ngIf=\"this.rucInputData.nodeTemplate === 'portrait'\" class=\"flex flex-col nodeContent\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\" (mouseenter)=\"onNodeHover(node)\"\r\n [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\"> \r\n <div class=\"flex flex-col items-center\" >\r\n <span *ngIf=\"node.data.image\"><img [src]=\"node.data.image\" class=\"portraitImageDimension\" /></span>\r\n <div class=\"font-bold mb-2\">{{ node.data.name }}</div>\r\n <div>{{ node.data.title }}\r\n </div>\r\n </div>\r\n </div>\r\n \r\n <div *ngIf=\"this.rucInputData.nodeTemplate === 'triangle'\" class=\"flex trianlge-up flex-col nodeContent\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\" (mouseenter)=\"onNodeHover(node)\"\r\n [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\"> \r\n <div class=\"flex flex-col items-center\" *ngIf=\"node.data.image\" >\r\n <img [src]=\"node.data.image\" class=\"triangleImageDimension\" />\r\n <div class=\"font-bold mb-2\">{{ node.data.name }}</div>\r\n <div>{{ node.data.title }}\r\n </div>\r\n </div>\r\n </div>\r\n <div *ngIf=\"this.rucInputData.nodeTemplate === 'landscape'\">\r\n <div class=\"card borderSolidGrey\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\"\r\n (mouseenter)=\"onNodeHover(node)\"\r\n [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\">\r\n <div class=\"body\" >\r\n <div class=\"icon\" *ngIf=\"node.data.image\">\r\n <img [src]=\"node.data.image\" class=\"landscapeImageDimension\" />\r\n </div>\r\n <div class=\"content\">\r\n <div>{{node.data.name}}</div>\r\n <div>{{node.data.title}}</div>\r\n </div>\r\n </div> \r\n </div>\r\n </div>\r\n </ng-template>\r\n </p-organizationChart>\r\n </div>\r\n \r\n</div>", styles: [".org-chart-container{overflow-x:auto}.nodeContent{border:none;padding:1.5em;border-radius:\"8px\";background-color:none;color:#000}.card{display:flex;width:250px;height:100px;padding:24px;align-self:stretch;align-items:flex-start}.card .body{width:100%;display:flex}.card .body .icon{width:64px;height:64px}.trianlge-up{width:220px;background-color:#980104;clip-path:polygon(50% 0%,0% 100%,100% 100%)}mat-form-field{margin:10px}::ng-deep .p-organizationchart .p-organizationchart-node-content{border:none;background:none;color:#495057;padding:0!important}::ng-deep .p-organizationchart .p-organizationchart-node-content .p-node-toggler{background-color:#dee2e6;cursor:pointer}::ng-deep .p-organizationchart .p-organizationchart-node-content .p-node-toggler .p-node-toggler-icon{top:.2rem}.container{display:flex;justify-content:space-between;width:100%}.container .left{margin-right:auto}.container .right{margin-left:auto}mat-icon-button{margin-right:10px}.mat-form-field{margin-left:10px}.borderSolidGrey{border:1px solid lightgray}.landscapeImageDimension{width:3em;height:3em}.portraitImageDimension,.triangleImageDimension{margin-bottom:4px;width:3em;height:3em}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i2.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i4.OrganizationChart, selector: "p-organizationChart", inputs: ["value", "style", "styleClass", "selectionMode", "preserveSpace", "selection"], outputs: ["selectionChange", "onNodeSelect", "onNodeUnselect", "onNodeExpand", "onNodeCollapse"] }, { kind: "directive", type: i5.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i6.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i6.MatLabel, selector: "mat-label" }, { kind: "directive", type: i6.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "component", type: i7.MatMenu, selector: "mat-menu", exportAs: ["matMenu"] }, { kind: "component", type: i7.MatMenuItem, selector: "[mat-menu-item]", inputs: ["disabled", "disableRipple", "role"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i7.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", exportAs: ["matMenuTrigger"] }, { kind: "component", type: i8.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i9.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }] });
198
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: OrgChartComponent, decorators: [{
199
+ type: Component,
200
+ args: [{ selector: 'uxp-org-chart', template: "<div class={{customTheme}}>\r\n <div class=\"container\">\r\n <!-- Hamburger Menu -->\r\n <button *ngIf=\"rucInputData.isDisplayHambergerMenu === true\" mat-icon-button [matMenuTriggerFor]=\"menu\" \r\n class=\"left hamburger-button\">\r\n <mat-icon>menu</mat-icon>\r\n </button>\r\n \r\n <!-- Menu -->\r\n <mat-menu #menu=\"matMenu\">\r\n <button *ngFor=\"let menu of rucInputData.hambergerMenuList\" mat-menu-item (click)=\"onMenuClick(menu.id, $event);\" \r\n [id]=\"'menu-item-' + menu.id\">\r\n {{ menu.label }}\r\n </button>\r\n </mat-menu>\r\n \r\n <!-- Search Bar -->\r\n <mat-form-field *ngIf=\"rucInputData.isDisplaySearchBar === true\" class=\"search-box right\" \r\n >\r\n <mat-label>Search</mat-label>\r\n <input matInput placeholder=\"Type to search...\" [(ngModel)]=\"searchText\" />\r\n <mat-icon matSuffix>search</mat-icon>\r\n </mat-form-field>\r\n </div>\r\n \r\n \r\n <div class=\"org-chart-container \" id=\"org-chart-data\">\r\n <p-organizationChart #orgChart class=\"org-chart\" [value]=\"this.rucInputData.orgData\" selectionMode=\"multiple\"\r\n [(selection)]=\"selectedNodes\">\r\n <ng-template let-node pTemplate=\"person\">\r\n <div *ngIf=\"this.rucInputData.nodeTemplate === 'portrait'\" class=\"flex flex-col nodeContent\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\" (mouseenter)=\"onNodeHover(node)\"\r\n [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\"> \r\n <div class=\"flex flex-col items-center\" >\r\n <span *ngIf=\"node.data.image\"><img [src]=\"node.data.image\" class=\"portraitImageDimension\" /></span>\r\n <div class=\"font-bold mb-2\">{{ node.data.name }}</div>\r\n <div>{{ node.data.title }}\r\n </div>\r\n </div>\r\n </div>\r\n \r\n <div *ngIf=\"this.rucInputData.nodeTemplate === 'triangle'\" class=\"flex trianlge-up flex-col nodeContent\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\" (mouseenter)=\"onNodeHover(node)\"\r\n [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\"> \r\n <div class=\"flex flex-col items-center\" *ngIf=\"node.data.image\" >\r\n <img [src]=\"node.data.image\" class=\"triangleImageDimension\" />\r\n <div class=\"font-bold mb-2\">{{ node.data.name }}</div>\r\n <div>{{ node.data.title }}\r\n </div>\r\n </div>\r\n </div>\r\n <div *ngIf=\"this.rucInputData.nodeTemplate === 'landscape'\">\r\n <div class=\"card borderSolidGrey\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\"\r\n (mouseenter)=\"onNodeHover(node)\"\r\n [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\">\r\n <div class=\"body\" >\r\n <div class=\"icon\" *ngIf=\"node.data.image\">\r\n <img [src]=\"node.data.image\" class=\"landscapeImageDimension\" />\r\n </div>\r\n <div class=\"content\">\r\n <div>{{node.data.name}}</div>\r\n <div>{{node.data.title}}</div>\r\n </div>\r\n </div> \r\n </div>\r\n </div>\r\n </ng-template>\r\n </p-organizationChart>\r\n </div>\r\n \r\n</div>", styles: [".org-chart-container{overflow-x:auto}.nodeContent{border:none;padding:1.5em;border-radius:\"8px\";background-color:none;color:#000}.card{display:flex;width:250px;height:100px;padding:24px;align-self:stretch;align-items:flex-start}.card .body{width:100%;display:flex}.card .body .icon{width:64px;height:64px}.trianlge-up{width:220px;background-color:#980104;clip-path:polygon(50% 0%,0% 100%,100% 100%)}mat-form-field{margin:10px}::ng-deep .p-organizationchart .p-organizationchart-node-content{border:none;background:none;color:#495057;padding:0!important}::ng-deep .p-organizationchart .p-organizationchart-node-content .p-node-toggler{background-color:#dee2e6;cursor:pointer}::ng-deep .p-organizationchart .p-organizationchart-node-content .p-node-toggler .p-node-toggler-icon{top:.2rem}.container{display:flex;justify-content:space-between;width:100%}.container .left{margin-right:auto}.container .right{margin-left:auto}mat-icon-button{margin-right:10px}.mat-form-field{margin-left:10px}.borderSolidGrey{border:1px solid lightgray}.landscapeImageDimension{width:3em;height:3em}.portraitImageDimension,.triangleImageDimension{margin-bottom:4px;width:3em;height:3em}\n"] }]
201
+ }], propDecorators: { rucInputData: [{
202
+ type: Input
203
+ }], customTheme: [{
204
+ type: Input
205
+ }], orgChart: [{
206
+ type: ViewChild,
207
+ args: ['orgChart', { static: false }]
208
+ }] } });
209
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"org-chart.component.js","sourceRoot":"","sources":["../../../../src/lib/org-chart/org-chart.component.ts","../../../../src/lib/org-chart/org-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, ViewChild, Input } from '@angular/core';\r\nimport { CustomTreeNode, OrgData, OrgDataItem } from '../../interfaces/org-chart';\r\nimport { TreeNode } from 'primeng/api';\r\nimport { checkPropsType, downloadChart } from '../../utils/chart-download.util';\r\nimport { OrganizationChart } from 'primeng/organizationchart';\r\n\r\n/** Displays an organizational chart with nodes and relationships (Org Chart Component)  */\r\n@Component({\r\n  selector: 'uxp-org-chart',\r\n  templateUrl: './org-chart.component.html',\r\n  styleUrls: ['./org-chart.component.scss'],\r\n})\r\nexport class OrgChartComponent implements OnInit {\r\n  @Input() rucInputData!: OrgData;\r\n  @Input() customTheme: string | undefined;\r\n  selectedNodes!: CustomTreeNode[];\r\n  searchText = '';\r\n  hoveredNode: CustomTreeNode | null = null;\r\n\r\n  @ViewChild('orgChart', { static: false }) orgChart!: OrganizationChart;\r\n\r\n  /** Initializes the component  */\r\n  ngOnInit(): void {\r\n    let updatedJson = this.addDefaultKeysIfNotAvailable(this.rucInputData);\r\n    this.rucInputData.orgData = this.updateRucInputData(updatedJson.orgData);\r\n    checkPropsType(this.rucInputData); // To check rucInputData is in correct format or not \r\n    this.expandAllNodes();\r\n  } \r\n\r\n  /** add default property for grey nodes if user does not provided */\r\n  addDefaultKeysIfNotAvailable(orgData: OrgData) {\r\n    if (!orgData.hasOwnProperty('greyNodeStyle')) {\r\n      orgData.greyNodeStyle = {\r\n        backgroundColor: '#d3d3d3',\r\n        color: '#808080',\r\n      };\r\n    }\r\n    return orgData;\r\n  }\r\n\r\n  /** manipulate input json as If customNodeStyle exists, create originalStyle for node and its children  */\r\n  updateRucInputData(data: OrgDataItem[]): OrgDataItem[] {\r\n    return data.map((node: OrgDataItem) => {\r\n      node['type'] = 'person';\r\n      if (node.customNodeStyle) {\r\n        node.originalStyle = { ...node.customNodeStyle };\r\n      }\r\n      if (node.children && node.children.length > 0) {\r\n        node['type'] = 'person';\r\n        node.children = this.updateRucInputData(node.children); // Recursive call\r\n      }\r\n      return node;\r\n    });\r\n  }\r\n\r\n  /** Expands all nodes in the org chart */\r\n  expandAllNodes(): void {\r\n    this.rucInputData.orgData = this.expandNodes(this.rucInputData.orgData);\r\n  }\r\n\r\n  /** Recursively expands all nodes in the given array\r\n   * @param nodes Nodes to expand @returns Expanded nodes */\r\n  expandNodes(nodes: OrgDataItem[]): OrgDataItem[] {\r\n    return nodes.map((node) => ({\r\n      ...node,\r\n      expanded: true,\r\n      children: node.children ? this.expandNodes(node.children) : [],\r\n    }));\r\n  }\r\n\r\n  /** Collapses all nodes in the org chart   */\r\n  collapseAllNodes(): void {\r\n    this.rucInputData.orgData = this.collapseNodes(this.rucInputData.orgData);\r\n  }\r\n\r\n  /** Recursively collapses all nodes in the given array\r\n   * @param nodes Nodes to collapse @returns Collapsed nodes  */\r\n  collapseNodes(nodes: OrgDataItem[]): OrgDataItem[] {\r\n    return nodes.map((node) => ({\r\n      ...node,\r\n      expanded: false,\r\n      children: node.children ? this.collapseNodes(node.children) : [],\r\n    }));\r\n  }\r\n\r\n  /** Gets the background color of a node based on search text\r\n    @param node Node to check @returns Whether the node matches the search text */\r\n  getBackgroungColorOfNode(node: CustomTreeNode): boolean {\r\n    return (\r\n      !this.searchText ||\r\n      node.data.title.toLowerCase().includes(this.searchText.toLowerCase()) ||\r\n      node.data.name.toLowerCase().includes(this.searchText.toLowerCase()) ||\r\n      node.label?.toLowerCase().includes(this.searchText.toLowerCase())\r\n    );\r\n  }\r\n\r\n  /** Handles node hover event @param node Hovered node */\r\n  onNodeHover(node: CustomTreeNode): void {\r\n    if (this.searchText) return;\r\n    this.hoveredNode = node;\r\n    this.updateNodeStyles();\r\n  }\r\n\r\n  /** Handles node leave event */\r\n  onNodeLeave(): void {\r\n    this.hoveredNode = null;\r\n    this.resetNodeStyles();\r\n  }\r\n\r\n  /**   Updates node styles based on the hovered node */\r\n  updateNodeStyles(): void {\r\n    this.applyStyles(this.rucInputData.orgData, true);\r\n  }\r\n\r\n  /**   Resets node styles to their original state */\r\n  resetNodeStyles(): void {\r\n    this.applyStyles(this.rucInputData.orgData, false);\r\n  }\r\n\r\n  /** Applies grey node style to non-matching nodes @param nodes Nodes to apply styles to */\r\n  applyGreyNodes(nodes:OrgDataItem[]): void {\r\n    nodes.forEach((node: OrgDataItem) => {\r\n      const isMatching = this.searchText\r\n        ? node.data?.title.includes(this.searchText)\r\n        : false;\r\n      node.customNodeStyle = isMatching\r\n        ? node.customNodeStyle || node.originalStyle\r\n        : {\r\n            ...node.originalStyle,\r\n            backgroundColor: this.rucInputData.greyNodeStyle.backgroundColor,\r\n            color: this.rucInputData.greyNodeStyle.color,\r\n          };\r\n\r\n      if (node.children) {\r\n        this.applyGreyNodes(node.children);\r\n      }\r\n    });\r\n  }\r\n\r\n  /** Applies styles to nodes based on the hovered node\r\n   * @param nodes Nodes to apply styles @param isHovered Whether the node is hovered */\r\n  applyStyles(nodes: OrgDataItem[], isHovered: boolean): void {\r\n    nodes.forEach((node) => {\r\n      if (isHovered) {\r\n        node.customNodeStyle =\r\n          node === this.hoveredNode ||\r\n          this.isParent(node, this.hoveredNode) ||\r\n          this.isChild(node, this.hoveredNode)\r\n            ? node.customNodeStyle || node.originalStyle\r\n            : {\r\n                ...node.originalStyle,\r\n                backgroundColor:\r\n                  this.rucInputData.greyNodeStyle.backgroundColor,\r\n                color: this.rucInputData.greyNodeStyle.color,\r\n              };\r\n      } else {\r\n        node.customNodeStyle = node.originalStyle;\r\n      }\r\n\r\n      if (node.children) {\r\n        this.applyStyles(node.children, isHovered);\r\n      }\r\n    });\r\n  }\r\n\r\n  /**  Use the downloadChart utility method */\r\n  downloadChart(format: 'png' | 'jpeg' | 'pdf'): void {\r\n    const chartContainer = this.orgChart.el.nativeElement;\r\n    downloadChart(chartContainer, format); // Calls the utility function\r\n  }\r\n\r\n  /** Handles menu click event @param id Menu item ID  */\r\n  onMenuClick(id: number,event: MouseEvent): void {\r\n    event.preventDefault();\r\n    event.stopPropagation();\r\n    switch (id) {\r\n      case 1:\r\n        this.downloadChart('pdf');\r\n        break;\r\n      case 2:\r\n        this.downloadChart('png');\r\n        break;\r\n      case 3:\r\n        this.downloadChart('jpeg');\r\n        break;\r\n      case 4:\r\n        this.expandAllNodes();\r\n        break;\r\n      case 5:\r\n        this.collapseAllNodes();\r\n        break;\r\n      default:\r\n        console.warn(`Unknown menu item ID: ${id}. Please enter valid id.`);\r\n        break;\r\n    }\r\n  }\r\n\r\n  /** Gets tooltip data for a node @param node Node to get tooltip data */\r\n  getTooltipData(node: CustomTreeNode): string | undefined {\r\n    return node.description || '';\r\n  }\r\n\r\n  /** Checks if a node is a parent of another node.\r\n   * @param currentNode The current node to check.\r\n   * @param hoveredNode The node to check for parentage.\r\n   * @returns Whether the current node is a parent of the hovered node. */\r\n  isParent(\r\n    currentNode: TreeNode,\r\n    hoveredNode: TreeNode | null\r\n  ): boolean | null | undefined {\r\n    return hoveredNode && currentNode.children?.includes(hoveredNode);\r\n  }\r\n\r\n  /** Checks if a node is a child of another node.\r\n   * @param node The node to check for child status\r\n   * @param target The node to check for parentage.\r\n   * @returns Whether the node is a child of the target node.*/\r\n  isChild(node: TreeNode, target: TreeNode | null): boolean | null | undefined {\r\n    return target && target.children?.includes(node);\r\n  }\r\n}\r\n","<div class={{customTheme}}>\r\n  <div class=\"container\">\r\n    <!-- Hamburger Menu -->\r\n    <button *ngIf=\"rucInputData.isDisplayHambergerMenu === true\" mat-icon-button [matMenuTriggerFor]=\"menu\" \r\n      class=\"left hamburger-button\">\r\n      <mat-icon>menu</mat-icon>\r\n    </button>\r\n  \r\n    <!-- Menu -->\r\n    <mat-menu #menu=\"matMenu\">\r\n      <button *ngFor=\"let menu of rucInputData.hambergerMenuList\" mat-menu-item (click)=\"onMenuClick(menu.id, $event);\" \r\n      [id]=\"'menu-item-' + menu.id\">\r\n        {{ menu.label }}\r\n      </button>\r\n    </mat-menu>\r\n  \r\n    <!-- Search Bar -->\r\n    <mat-form-field *ngIf=\"rucInputData.isDisplaySearchBar === true\" class=\"search-box right\" \r\n      >\r\n      <mat-label>Search</mat-label>\r\n      <input matInput placeholder=\"Type to search...\" [(ngModel)]=\"searchText\" />\r\n      <mat-icon matSuffix>search</mat-icon>\r\n    </mat-form-field>\r\n  </div>\r\n  \r\n  \r\n  <div class=\"org-chart-container \" id=\"org-chart-data\">\r\n  <p-organizationChart #orgChart class=\"org-chart\" [value]=\"this.rucInputData.orgData\" selectionMode=\"multiple\"\r\n    [(selection)]=\"selectedNodes\">\r\n    <ng-template let-node pTemplate=\"person\">\r\n      <div *ngIf=\"this.rucInputData.nodeTemplate === 'portrait'\"  class=\"flex flex-col nodeContent\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\" (mouseenter)=\"onNodeHover(node)\"\r\n        [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\"> \r\n        <div class=\"flex flex-col items-center\"  >\r\n          <span *ngIf=\"node.data.image\"><img   [src]=\"node.data.image\" class=\"portraitImageDimension\" /></span>\r\n          <div class=\"font-bold mb-2\">{{ node.data.name }}</div>\r\n          <div>{{ node.data.title }}\r\n          </div>\r\n        </div>\r\n      </div>\r\n  \r\n      <div *ngIf=\"this.rucInputData.nodeTemplate === 'triangle'\"  class=\"flex trianlge-up flex-col nodeContent\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\" (mouseenter)=\"onNodeHover(node)\"\r\n        [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\"> \r\n        <div class=\"flex flex-col items-center\" *ngIf=\"node.data.image\" >\r\n          <img  [src]=\"node.data.image\" class=\"triangleImageDimension\" />\r\n          <div class=\"font-bold mb-2\">{{ node.data.name }}</div>\r\n          <div>{{ node.data.title }}\r\n          </div>\r\n        </div>\r\n      </div>\r\n    <div *ngIf=\"this.rucInputData.nodeTemplate === 'landscape'\">\r\n      <div class=\"card borderSolidGrey\" [ngStyle]=\"getBackgroungColorOfNode(node) ? node.customNodeStyle : rucInputData.greyNodeStyle\"\r\n      (mouseenter)=\"onNodeHover(node)\"\r\n      [title]=\"getTooltipData(node)\" (mouseleave)=\"onNodeLeave()\">\r\n        <div class=\"body\" >\r\n          <div class=\"icon\" *ngIf=\"node.data.image\">\r\n            <img  [src]=\"node.data.image\" class=\"landscapeImageDimension\" />\r\n          </div>\r\n          <div class=\"content\">\r\n              <div>{{node.data.name}}</div>\r\n              <div>{{node.data.title}}</div>\r\n          </div>\r\n        </div> \r\n      </div>\r\n    </div>\r\n    </ng-template>\r\n  </p-organizationChart>\r\n  </div>\r\n  \r\n</div>"]}
@@ -0,0 +1,26 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ButtonModule } from 'primeng/button';
4
+ import { OrganizationChartModule } from 'primeng/organizationchart';
5
+ import { MatInputModule } from '@angular/material/input';
6
+ import { FormsModule } from '@angular/forms';
7
+ import { MatMenuModule } from '@angular/material/menu';
8
+ import { MatButtonModule } from '@angular/material/button';
9
+ import { MatIconModule } from '@angular/material/icon';
10
+ import { OrgChartComponent } from './org-chart/org-chart.component';
11
+ import { BrowserModule } from '@angular/platform-browser';
12
+ import * as i0 from "@angular/core";
13
+ export class RuclibOrgChartModule {
14
+ }
15
+ RuclibOrgChartModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RuclibOrgChartModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
16
+ RuclibOrgChartModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.10", ngImport: i0, type: RuclibOrgChartModule, declarations: [OrgChartComponent], imports: [CommonModule, ButtonModule, BrowserModule, FormsModule, OrganizationChartModule, MatInputModule, MatMenuModule, MatButtonModule, MatIconModule], exports: [OrgChartComponent] });
17
+ RuclibOrgChartModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RuclibOrgChartModule, imports: [CommonModule, ButtonModule, BrowserModule, FormsModule, OrganizationChartModule, MatInputModule, MatMenuModule, MatButtonModule, MatIconModule] });
18
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RuclibOrgChartModule, decorators: [{
19
+ type: NgModule,
20
+ args: [{
21
+ imports: [CommonModule, ButtonModule, BrowserModule, FormsModule, OrganizationChartModule, MatInputModule, MatMenuModule, MatButtonModule, MatIconModule],
22
+ declarations: [OrgChartComponent],
23
+ exports: [OrgChartComponent]
24
+ }]
25
+ }] });
26
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVjbGliLW9yZy1jaGFydC5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL3J1Y2xpYi1vcmctY2hhcnQubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUM5QyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDekQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzdDLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUN2RCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDM0QsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3ZELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQ3BFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQzs7QUFPMUQsTUFBTSxPQUFPLG9CQUFvQjs7a0hBQXBCLG9CQUFvQjttSEFBcEIsb0JBQW9CLGlCQUhoQixpQkFBaUIsYUFEdEIsWUFBWSxFQUFFLFlBQVksRUFBRSxhQUFhLEVBQUUsV0FBVyxFQUFFLHVCQUF1QixFQUFFLGNBQWMsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLGFBQWEsYUFFOUksaUJBQWlCO21IQUVoQixvQkFBb0IsWUFKckIsWUFBWSxFQUFFLFlBQVksRUFBRSxhQUFhLEVBQUUsV0FBVyxFQUFFLHVCQUF1QixFQUFFLGNBQWMsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLGFBQWE7NEZBSTdJLG9CQUFvQjtrQkFMaEMsUUFBUTttQkFBQztvQkFDUixPQUFPLEVBQUUsQ0FBQyxZQUFZLEVBQUUsWUFBWSxFQUFFLGFBQWEsRUFBRSxXQUFXLEVBQUUsdUJBQXVCLEVBQUUsY0FBYyxFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsYUFBYSxDQUFDO29CQUN6SixZQUFZLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztvQkFDakMsT0FBTyxFQUFFLENBQUMsaUJBQWlCLENBQUM7aUJBQzdCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgQnV0dG9uTW9kdWxlIH0gZnJvbSAncHJpbWVuZy9idXR0b24nO1xyXG5pbXBvcnQgeyBPcmdhbml6YXRpb25DaGFydE1vZHVsZSB9IGZyb20gJ3ByaW1lbmcvb3JnYW5pemF0aW9uY2hhcnQnO1xyXG5pbXBvcnQgeyBNYXRJbnB1dE1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2lucHV0JztcclxuaW1wb3J0IHsgRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XHJcbmltcG9ydCB7IE1hdE1lbnVNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9tZW51JztcclxuaW1wb3J0IHsgTWF0QnV0dG9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvYnV0dG9uJztcclxuaW1wb3J0IHsgTWF0SWNvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2ljb24nO1xyXG5pbXBvcnQgeyBPcmdDaGFydENvbXBvbmVudCB9IGZyb20gJy4vb3JnLWNoYXJ0L29yZy1jaGFydC5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBCcm93c2VyTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvcGxhdGZvcm0tYnJvd3Nlcic7XHJcblxyXG5ATmdNb2R1bGUoe1xyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEJ1dHRvbk1vZHVsZSwgQnJvd3Nlck1vZHVsZSwgRm9ybXNNb2R1bGUsIE9yZ2FuaXphdGlvbkNoYXJ0TW9kdWxlLCBNYXRJbnB1dE1vZHVsZSwgTWF0TWVudU1vZHVsZSwgTWF0QnV0dG9uTW9kdWxlLCBNYXRJY29uTW9kdWxlXSxcclxuICBkZWNsYXJhdGlvbnM6IFtPcmdDaGFydENvbXBvbmVudF0sXHJcbiAgZXhwb3J0czogW09yZ0NoYXJ0Q29tcG9uZW50XVxyXG59KVxyXG5leHBvcnQgY2xhc3MgUnVjbGliT3JnQ2hhcnRNb2R1bGUge31cclxuXHJcbiJdfQ==
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './index';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVjLWxpYi1vcmctY2hhcnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcnVjLWxpYi1vcmctY2hhcnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLFNBQVMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9pbmRleCc7XG4iXX0=