@rangertechnologies/ngnxt 2.1.230 → 2.1.232

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.
@@ -0,0 +1,139 @@
1
+ //MSM10JUL25
2
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
3
+ import { CommonModule } from '@angular/common';
4
+ import { finalize, map, Subject, takeUntil } from 'rxjs';
5
+ import { FormsModule } from '@angular/forms';
6
+ import { MatTooltipModule } from '@angular/material/tooltip';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/common/http";
9
+ import * as i2 from "@angular/platform-browser";
10
+ import * as i3 from "@angular/common";
11
+ import * as i4 from "@angular/forms";
12
+ import * as i5 from "@angular/material/tooltip";
13
+ export class IconSelectorComponent {
14
+ http;
15
+ sanitizer;
16
+ cdr;
17
+ allIcons;
18
+ themeColor = '#ff0000'; // Default theme color
19
+ height = '300px'; // Default height
20
+ tooltipPosition = "above"; // above, below, left, right, before, after
21
+ selectedIcon; // Selected icon name
22
+ cdnIconURL = ""; // CDN URL for icons
23
+ label = ''; // Input label
24
+ labelFont = ''; // Input label font
25
+ labelWeight = ''; // Input label Weight
26
+ inputWeight = ''; // Input Weight
27
+ labelSize = ''; // Input label Size
28
+ labelColor = ''; // Input label Color
29
+ showLabel = ''; // Input label Color
30
+ required = false; // Required validation
31
+ mode = 'edit'; // New mode input
32
+ question = {};
33
+ options = []; // For radio buttons
34
+ iconSelected = new EventEmitter(); // Event emitter for selected icon
35
+ svgCache = new Map(); // Cache for better performance
36
+ destroyed = new Subject();
37
+ filteredIcons = [];
38
+ searchQuery = '';
39
+ showDropdown = false;
40
+ constructor(http, sanitizer, cdr) {
41
+ this.http = http;
42
+ this.sanitizer = sanitizer;
43
+ this.cdr = cdr;
44
+ }
45
+ ngOnInit() {
46
+ this.filteredIcons = this.allIcons != null && this.allIcons.length > 0 ? [...this.allIcons] : [];
47
+ }
48
+ filterIcons() {
49
+ if (!this.searchQuery) {
50
+ this.filteredIcons = this.allIcons != null && this.allIcons.length > 0 ? [...this.allIcons] : [];
51
+ return;
52
+ }
53
+ this.filteredIcons = this.allIcons?.filter((icon) => icon?.name?.toLowerCase().includes(this.searchQuery?.toLowerCase()));
54
+ }
55
+ selectIcon(icon) {
56
+ this.selectedIcon = icon;
57
+ this.iconSelected.emit(icon);
58
+ }
59
+ getSanitizedSvg(filename) {
60
+ // Return cached or loading SVG
61
+ return this.svgCache.get(filename) || this.loadSvg(filename);
62
+ }
63
+ loadSvg(filename) {
64
+ const loadingSvg = this.sanitizer.bypassSecurityTrustHtml(`<svg width="48" height="48" viewBox="0 0 24 24"></svg>`);
65
+ this.svgCache.set(filename, loadingSvg);
66
+ this.http.get(this.getIconUrl(filename), { responseType: 'text' })
67
+ .pipe(takeUntil(this.destroyed), map(svgData => this.processSvg(svgData, this.themeColor)), finalize(() => this.cdr.markForCheck()) // Safe change detection
68
+ )
69
+ .subscribe({
70
+ next: processedSvg => this.svgCache.set(filename, processedSvg),
71
+ error: () => this.svgCache.set(filename, loadingSvg)
72
+ });
73
+ return loadingSvg;
74
+ }
75
+ processSvg(svgData, color) {
76
+ const div = document.createElement('div');
77
+ div.innerHTML = svgData;
78
+ const svg = div.querySelector('svg');
79
+ if (svg) {
80
+ svg.querySelectorAll('[fill]:not([fill="none"]), [stroke]:not([stroke="none"])')
81
+ .forEach(el => {
82
+ if (el.hasAttribute('fill'))
83
+ el.setAttribute('fill', color);
84
+ if (el.hasAttribute('stroke'))
85
+ el.setAttribute('stroke', color);
86
+ });
87
+ }
88
+ return this.sanitizer.bypassSecurityTrustHtml(div.innerHTML);
89
+ }
90
+ getIconUrl(fileName) {
91
+ return `${this.cdnIconURL}${fileName}`;
92
+ }
93
+ toggleDropdown() {
94
+ this.showDropdown = !this.showDropdown;
95
+ }
96
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: IconSelectorComponent, deps: [{ token: i1.HttpClient }, { token: i2.DomSanitizer }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
97
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: IconSelectorComponent, isStandalone: true, selector: "nxt-icon-selector", inputs: { allIcons: "allIcons", themeColor: "themeColor", height: "height", tooltipPosition: "tooltipPosition", selectedIcon: "selectedIcon", cdnIconURL: "cdnIconURL", label: "label", labelFont: "labelFont", labelWeight: "labelWeight", inputWeight: "inputWeight", labelSize: "labelSize", labelColor: "labelColor", showLabel: "showLabel", required: "required", mode: "mode", question: "question", options: "options" }, outputs: { iconSelected: "iconSelected" }, ngImport: i0, template: "<!-- MSM10JUL25 for icon selector UI -->\n<div class=\"icon-selector\">\n <div class=\"selected-icon\" (click)=\"toggleDropdown()\">\n <span *ngIf=\"!selectedIcon\">Choose Icon</span>\n <div *ngIf=\"selectedIcon\" [innerHTML]=\"getSanitizedSvg(selectedIcon)\" class=\"svg-container\"></div><span\n class=\"material-icons\">arrow_drop_down</span>\n </div>\n\n <div class=\"icon-selector-container\" [style.height]=\"height\" *ngIf=\"showDropdown\">\n <div class=\"search-container\">\n <div class=\"search-box\">\n <input type=\"text\" [(ngModel)]=\"searchQuery\" (input)=\"filterIcons()\" placeholder=\"Search icons...\"\n class=\"search-input\">\n </div>\n </div>\n\n <div class=\"icon-grid-container\" [style.max-height]=\"'calc(' + height + ' - 120px)'\">\n <div *ngIf=\"filteredIcons.length === 0\" class=\"no-results\">\n No icons found matching \"{{searchQuery}}\"\n </div>\n\n <div class=\"icon-grid\">\n <div *ngFor=\"let icon of filteredIcons\" class=\"icon-option\" [matTooltip]=icon.name\n [matTooltipPosition]=tooltipPosition (click)=\"selectIcon(icon.file);toggleDropdown()\">\n <div [innerHTML]=\"getSanitizedSvg(icon.file)\" class=\"svg-container\"></div>\n </div>\n </div>\n </div>\n </div>\n</div>", styles: [".icon-item svg{color:red}.icon-selector-container{padding:16px;max-width:400px;margin:0 auto;display:flex;flex-direction:column;border-radius:8px;box-shadow:0 2px 10px #0000001a;background-color:#fff;overflow:hidden}.search-container{margin:8px 0 16px}.icon-grid-container{overflow-y:scroll!important;padding-right:4px}.icon-grid{display:grid;grid-template-columns:repeat(10,1fr);gap:5px}.icon-option{display:flex;align-items:center;justify-content:center;padding:6px;border-radius:8px;cursor:pointer;transition:all .2s ease;background-color:#f5f5f5;aspect-ratio:1/1}.icon-option:hover{background-color:#e0e0e0;transform:scale(1.05)}.icon-option.selected{background-color:#e3f2fd;border:2px solid #2196f3}.no-results{text-align:center;padding:20px;color:#757575;font-style:italic}::-webkit-scrollbar{width:6px}::-webkit-scrollbar-track{background:#f1f1f1;border-radius:3px}::-webkit-scrollbar-thumb{background:#888;border-radius:3px}::-webkit-scrollbar-thumb:hover{background:#555}.icon-selector{position:relative;display:inline-block}.selected-icon{display:flex;align-items:center;gap:4px;cursor:pointer;padding:3px;border:1px solid #ccc;border-radius:4px}.icon-dropdown{position:absolute;top:100%;left:0;z-index:1000;background:#fff;border:1px solid #ccc;border-radius:4px;max-height:200px;overflow-y:auto;display:grid;grid-template-columns:repeat(10,1fr);gap:5px;padding:3px}.icon-option{padding:3px;cursor:pointer;border-radius:4px;display:flex;justify-content:center}.icon-option:hover{background-color:#f0f0f0}.material-icons{font-family:Material Icons;font-weight:400;font-style:normal;font-size:24px;line-height:1}svg{width:20px!important;height:20px!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i4.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: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i5.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }] });
98
+ }
99
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: IconSelectorComponent, decorators: [{
100
+ type: Component,
101
+ args: [{ selector: 'nxt-icon-selector', imports: [CommonModule, FormsModule, MatTooltipModule], standalone: true, template: "<!-- MSM10JUL25 for icon selector UI -->\n<div class=\"icon-selector\">\n <div class=\"selected-icon\" (click)=\"toggleDropdown()\">\n <span *ngIf=\"!selectedIcon\">Choose Icon</span>\n <div *ngIf=\"selectedIcon\" [innerHTML]=\"getSanitizedSvg(selectedIcon)\" class=\"svg-container\"></div><span\n class=\"material-icons\">arrow_drop_down</span>\n </div>\n\n <div class=\"icon-selector-container\" [style.height]=\"height\" *ngIf=\"showDropdown\">\n <div class=\"search-container\">\n <div class=\"search-box\">\n <input type=\"text\" [(ngModel)]=\"searchQuery\" (input)=\"filterIcons()\" placeholder=\"Search icons...\"\n class=\"search-input\">\n </div>\n </div>\n\n <div class=\"icon-grid-container\" [style.max-height]=\"'calc(' + height + ' - 120px)'\">\n <div *ngIf=\"filteredIcons.length === 0\" class=\"no-results\">\n No icons found matching \"{{searchQuery}}\"\n </div>\n\n <div class=\"icon-grid\">\n <div *ngFor=\"let icon of filteredIcons\" class=\"icon-option\" [matTooltip]=icon.name\n [matTooltipPosition]=tooltipPosition (click)=\"selectIcon(icon.file);toggleDropdown()\">\n <div [innerHTML]=\"getSanitizedSvg(icon.file)\" class=\"svg-container\"></div>\n </div>\n </div>\n </div>\n </div>\n</div>", styles: [".icon-item svg{color:red}.icon-selector-container{padding:16px;max-width:400px;margin:0 auto;display:flex;flex-direction:column;border-radius:8px;box-shadow:0 2px 10px #0000001a;background-color:#fff;overflow:hidden}.search-container{margin:8px 0 16px}.icon-grid-container{overflow-y:scroll!important;padding-right:4px}.icon-grid{display:grid;grid-template-columns:repeat(10,1fr);gap:5px}.icon-option{display:flex;align-items:center;justify-content:center;padding:6px;border-radius:8px;cursor:pointer;transition:all .2s ease;background-color:#f5f5f5;aspect-ratio:1/1}.icon-option:hover{background-color:#e0e0e0;transform:scale(1.05)}.icon-option.selected{background-color:#e3f2fd;border:2px solid #2196f3}.no-results{text-align:center;padding:20px;color:#757575;font-style:italic}::-webkit-scrollbar{width:6px}::-webkit-scrollbar-track{background:#f1f1f1;border-radius:3px}::-webkit-scrollbar-thumb{background:#888;border-radius:3px}::-webkit-scrollbar-thumb:hover{background:#555}.icon-selector{position:relative;display:inline-block}.selected-icon{display:flex;align-items:center;gap:4px;cursor:pointer;padding:3px;border:1px solid #ccc;border-radius:4px}.icon-dropdown{position:absolute;top:100%;left:0;z-index:1000;background:#fff;border:1px solid #ccc;border-radius:4px;max-height:200px;overflow-y:auto;display:grid;grid-template-columns:repeat(10,1fr);gap:5px;padding:3px}.icon-option{padding:3px;cursor:pointer;border-radius:4px;display:flex;justify-content:center}.icon-option:hover{background-color:#f0f0f0}.material-icons{font-family:Material Icons;font-weight:400;font-style:normal;font-size:24px;line-height:1}svg{width:20px!important;height:20px!important}\n"] }]
102
+ }], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.DomSanitizer }, { type: i0.ChangeDetectorRef }], propDecorators: { allIcons: [{
103
+ type: Input
104
+ }], themeColor: [{
105
+ type: Input
106
+ }], height: [{
107
+ type: Input
108
+ }], tooltipPosition: [{
109
+ type: Input
110
+ }], selectedIcon: [{
111
+ type: Input
112
+ }], cdnIconURL: [{
113
+ type: Input
114
+ }], label: [{
115
+ type: Input
116
+ }], labelFont: [{
117
+ type: Input
118
+ }], labelWeight: [{
119
+ type: Input
120
+ }], inputWeight: [{
121
+ type: Input
122
+ }], labelSize: [{
123
+ type: Input
124
+ }], labelColor: [{
125
+ type: Input
126
+ }], showLabel: [{
127
+ type: Input
128
+ }], required: [{
129
+ type: Input
130
+ }], mode: [{
131
+ type: Input
132
+ }], question: [{
133
+ type: Input
134
+ }], options: [{
135
+ type: Input
136
+ }], iconSelected: [{
137
+ type: Output
138
+ }] } });
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"icon-selector.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/icon-selector/icon-selector.component.ts","../../../../../../projects/nxt-app/src/lib/components/icon-selector/icon-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["//MSM10JUL25\nimport { Component, EventEmitter, Input, Output, ChangeDetectorRef } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\nimport { finalize, map, Subject, takeUntil } from 'rxjs';\nimport { HttpClient } from '@angular/common/http';\nimport { FormsModule } from '@angular/forms';\nimport { MatTooltipModule } from '@angular/material/tooltip';\n\n@Component({\n  selector: 'nxt-icon-selector',\n  templateUrl: './icon-selector.component.html',\n  imports: [CommonModule, FormsModule, MatTooltipModule],\n  styleUrls: ['./icon-selector.component.css'],\n  standalone: true,\n})\nexport class IconSelectorComponent {\n  @Input() allIcons: [];\n  @Input() themeColor: string = '#ff0000'; // Default theme color\n  @Input() height: string = '300px'; // Default height\n  @Input() tooltipPosition: string = \"above\"; // above, below, left, right, before, after\n  @Input() selectedIcon: string; // Selected icon name\n  @Input() cdnIconURL: string = \"\"; // CDN URL for icons\n\n  @Input() label: string = '';  // Input label\n  @Input() labelFont: string = '';  // Input label font\n  @Input() labelWeight: string = '';  // Input label Weight\n  @Input() inputWeight: string = '';  // Input Weight\n  @Input() labelSize: string = '';  // Input label Size\n  @Input() labelColor: string = '';  // Input label Color\n  @Input() showLabel: string = '';  // Input label Color\n  @Input() required: boolean = false;  // Required validation\n  @Input() mode: 'view' | 'edit' | 'print' = 'edit'; // New mode input\n  @Input() question: any = {}\n  @Input() options: any[] = []; // For radio buttons\n\n  @Output() iconSelected = new EventEmitter<string>(); // Event emitter for selected icon\n  svgCache = new Map<string, SafeHtml>(); // Cache for better performance\n  private destroyed = new Subject<void>();\n  filteredIcons: any[] = [];\n  searchQuery = '';\n  showDropdown = false;\n  constructor(private http: HttpClient, private sanitizer: DomSanitizer, private cdr: ChangeDetectorRef) { }\n  ngOnInit() {\n    this.filteredIcons = this.allIcons != null && this.allIcons.length > 0 ? [...this.allIcons] : [];\n  }\n  filterIcons() {\n    if (!this.searchQuery) {\n      this.filteredIcons = this.allIcons != null && this.allIcons.length > 0 ? [...this.allIcons] : [];\n      return;\n    }\n    this.filteredIcons = this.allIcons?.filter((icon: any) =>\n      icon?.name?.toLowerCase().includes(this.searchQuery?.toLowerCase())\n    );\n  }\n\n  selectIcon(icon: string) {\n    this.selectedIcon = icon;\n    this.iconSelected.emit(icon);\n  }\n\n  getSanitizedSvg(filename: string): SafeHtml {\n    // Return cached or loading SVG\n    return this.svgCache.get(filename) || this.loadSvg(filename);\n  }\n\n  private loadSvg(filename: string): SafeHtml {\n    const loadingSvg = this.sanitizer.bypassSecurityTrustHtml(\n      `<svg width=\"48\" height=\"48\" viewBox=\"0 0 24 24\"></svg>`\n    );\n    this.svgCache.set(filename, loadingSvg);\n\n    this.http.get(this.getIconUrl(filename), { responseType: 'text' })\n      .pipe(\n        takeUntil(this.destroyed),\n        map(svgData => this.processSvg(svgData, this.themeColor)),\n        finalize(() => this.cdr.markForCheck()) // Safe change detection\n      )\n      .subscribe({\n        next: processedSvg => this.svgCache.set(filename, processedSvg),\n        error: () => this.svgCache.set(filename, loadingSvg)\n      });\n\n    return loadingSvg;\n  }\n\n  private processSvg(svgData: string, color: string): SafeHtml {\n    const div = document.createElement('div');\n    div.innerHTML = svgData;\n    const svg = div.querySelector('svg');\n\n    if (svg) {\n      svg.querySelectorAll('[fill]:not([fill=\"none\"]), [stroke]:not([stroke=\"none\"])')\n        .forEach(el => {\n          if (el.hasAttribute('fill')) el.setAttribute('fill', color);\n          if (el.hasAttribute('stroke')) el.setAttribute('stroke', color);\n        });\n    }\n\n    return this.sanitizer.bypassSecurityTrustHtml(div.innerHTML);\n  }\n  getIconUrl(fileName: string): string {\n    return `${this.cdnIconURL}${fileName}`;\n  }\n  toggleDropdown(): void {\n    this.showDropdown = !this.showDropdown;\n  }\n}\n","<!-- MSM10JUL25 for icon selector UI -->\n<div class=\"icon-selector\">\n  <div class=\"selected-icon\" (click)=\"toggleDropdown()\">\n    <span *ngIf=\"!selectedIcon\">Choose Icon</span>\n    <div *ngIf=\"selectedIcon\" [innerHTML]=\"getSanitizedSvg(selectedIcon)\" class=\"svg-container\"></div><span\n      class=\"material-icons\">arrow_drop_down</span>\n  </div>\n\n  <div class=\"icon-selector-container\" [style.height]=\"height\" *ngIf=\"showDropdown\">\n    <div class=\"search-container\">\n      <div class=\"search-box\">\n        <input type=\"text\" [(ngModel)]=\"searchQuery\" (input)=\"filterIcons()\" placeholder=\"Search icons...\"\n          class=\"search-input\">\n      </div>\n    </div>\n\n    <div class=\"icon-grid-container\" [style.max-height]=\"'calc(' + height + ' - 120px)'\">\n      <div *ngIf=\"filteredIcons.length === 0\" class=\"no-results\">\n        No icons found matching \"{{searchQuery}}\"\n      </div>\n\n      <div class=\"icon-grid\">\n        <div *ngFor=\"let icon of filteredIcons\" class=\"icon-option\" [matTooltip]=icon.name\n          [matTooltipPosition]=tooltipPosition (click)=\"selectIcon(icon.file);toggleDropdown()\">\n          <div [innerHTML]=\"getSanitizedSvg(icon.file)\" class=\"svg-container\"></div>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>"]}