@rangertechnologies/ngnxt 2.1.80 → 2.1.82

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.
@@ -82,11 +82,11 @@ export class CustomRadioComponent {
82
82
  }
83
83
  }
84
84
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomRadioComponent, deps: [{ token: i1.ChangeService }, { token: i2.DataService }, { token: i3.I18nService }], target: i0.ɵɵFactoryTarget.Component });
85
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CustomRadioComponent, selector: "app-custom-radio", inputs: { options: "options", apiMeta: "apiMeta", selectedValue: "selectedValue", progressBar: "progressBar", id: "id", readOnly: "readOnly", errorMessage: "errorMessage", error: "error", fromShengel: "fromShengel", referenceField: "referenceField", token: "token" }, outputs: { valueChange: "valueChange" }, ngImport: i0, template: "<!-- RS 09DEC24 Changed keys--> \n<div class=\"custom-radio-container\">\n <div\n *ngFor=\"let option of options\"\n [class]=\"invalidFieldIds.includes(id) || error ? 'custom-radio-option invalid' : 'custom-radio-option'\"\n >\n <input\n type=\"radio\"\n [id]=\"apiMeta !== undefined ? option[labelField] : option.value \"\n [checked] = \"selectedValue == option.value \"\n [name]=\"id\"\n [value]=\"selectedValue\"\n (change)=\"radioChange($event)\"\n [disabled]=\"readOnly\"\n />\n <label class=\"nxt-radio-label\" [for]=\"apiMeta !== undefined ? option[labelField] : option.value\">{{ apiMeta !== undefined ? option[labelField] : option.value }}</label>\n </div>\n <span *ngIf=\"error || invalidFieldIds.includes(id)\" class=\"error-msg\">{{ errorMessage }}</span>\n </div>\n ", styles: [".custom-radio-option{display:flex;flex-direction:row;margin-bottom:5px}input[type=radio]{width:auto}.nxt-radio-label{margin-left:15px;margin-bottom:0}.custom-radio-option.invalid label{color:red}.error-msg{color:red;font-size:12px;margin-top:5px}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
85
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CustomRadioComponent, selector: "app-custom-radio", inputs: { options: "options", apiMeta: "apiMeta", selectedValue: "selectedValue", progressBar: "progressBar", id: "id", readOnly: "readOnly", errorMessage: "errorMessage", error: "error", fromShengel: "fromShengel", referenceField: "referenceField", token: "token" }, outputs: { valueChange: "valueChange" }, ngImport: i0, template: "<!-- RS 09DEC24 Changed keys-->\n <!-- RS 019JAN25 -->\n <!-- validate for NULL -->\n<div class=\"custom-radio-container\">\n <div\n *ngFor=\"let option of options\"\n [class]=\"invalidFieldIds.includes(id) || error ? 'custom-radio-option invalid' : 'custom-radio-option'\"\n >\n <input\n type=\"radio\"\n [id]=\"apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value\"\n [checked] = \"selectedValue == option.value\"\n [name]=\"id\"\n [value]=\"selectedValue\"\n (change)=\"radioChange($event)\"\n [disabled]=\"readOnly\"\n />\n <label class=\"nxt-radio-label\" [for]=\"apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value\">{{ apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value}}</label>\n </div>\n <span *ngIf=\"error || invalidFieldIds.includes(id)\" class=\"error-msg\">{{ errorMessage }}</span>\n</div>", styles: [".custom-radio-option{display:flex;flex-direction:row;margin-bottom:5px}input[type=radio]{width:auto}.nxt-radio-label{margin-left:15px;margin-bottom:0}.custom-radio-option.invalid label{color:red}.error-msg{color:red;font-size:12px;margin-top:5px}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
86
86
  }
87
87
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomRadioComponent, decorators: [{
88
88
  type: Component,
89
- args: [{ selector: 'app-custom-radio', template: "<!-- RS 09DEC24 Changed keys--> \n<div class=\"custom-radio-container\">\n <div\n *ngFor=\"let option of options\"\n [class]=\"invalidFieldIds.includes(id) || error ? 'custom-radio-option invalid' : 'custom-radio-option'\"\n >\n <input\n type=\"radio\"\n [id]=\"apiMeta !== undefined ? option[labelField] : option.value \"\n [checked] = \"selectedValue == option.value \"\n [name]=\"id\"\n [value]=\"selectedValue\"\n (change)=\"radioChange($event)\"\n [disabled]=\"readOnly\"\n />\n <label class=\"nxt-radio-label\" [for]=\"apiMeta !== undefined ? option[labelField] : option.value\">{{ apiMeta !== undefined ? option[labelField] : option.value }}</label>\n </div>\n <span *ngIf=\"error || invalidFieldIds.includes(id)\" class=\"error-msg\">{{ errorMessage }}</span>\n </div>\n ", styles: [".custom-radio-option{display:flex;flex-direction:row;margin-bottom:5px}input[type=radio]{width:auto}.nxt-radio-label{margin-left:15px;margin-bottom:0}.custom-radio-option.invalid label{color:red}.error-msg{color:red;font-size:12px;margin-top:5px}\n"] }]
89
+ args: [{ selector: 'app-custom-radio', template: "<!-- RS 09DEC24 Changed keys-->\n <!-- RS 019JAN25 -->\n <!-- validate for NULL -->\n<div class=\"custom-radio-container\">\n <div\n *ngFor=\"let option of options\"\n [class]=\"invalidFieldIds.includes(id) || error ? 'custom-radio-option invalid' : 'custom-radio-option'\"\n >\n <input\n type=\"radio\"\n [id]=\"apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value\"\n [checked] = \"selectedValue == option.value\"\n [name]=\"id\"\n [value]=\"selectedValue\"\n (change)=\"radioChange($event)\"\n [disabled]=\"readOnly\"\n />\n <label class=\"nxt-radio-label\" [for]=\"apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value\">{{ apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value}}</label>\n </div>\n <span *ngIf=\"error || invalidFieldIds.includes(id)\" class=\"error-msg\">{{ errorMessage }}</span>\n</div>", styles: [".custom-radio-option{display:flex;flex-direction:row;margin-bottom:5px}input[type=radio]{width:auto}.nxt-radio-label{margin-left:15px;margin-bottom:0}.custom-radio-option.invalid label{color:red}.error-msg{color:red;font-size:12px;margin-top:5px}\n"] }]
90
90
  }], ctorParameters: () => [{ type: i1.ChangeService }, { type: i2.DataService }, { type: i3.I18nService }], propDecorators: { options: [{
91
91
  type: Input
92
92
  }], apiMeta: [{
@@ -112,4 +112,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
112
112
  }], valueChange: [{
113
113
  type: Output
114
114
  }] } });
115
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-radio.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/custom-radio/custom-radio.component.ts","../../../../../../projects/nxt-app/src/lib/components/custom-radio/custom-radio.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;;;;;;AAY1D,MAAM,OAAO,oBAAoB;IAqBrB;IACA;IACD;IArBA,OAAO,GAAU,EAAE,CAAC;IACpB,OAAO,CAAS;IAChB,aAAa,CAAS;IACtB,WAAW,CAAU;IACrB,EAAE,CAAS;IACX,QAAQ,GAAG,KAAK,CAAC;IACjB,YAAY,CAAS;IACrB,KAAK,CAAM;IACX,WAAW,GAAY,KAAK,CAAC;IAC7B,cAAc,CAAS;IACvB,KAAK,CAAS;IACb,WAAW,GAAgC,IAAI,YAAY,EAAiB,CAAC;IACvF,eAAe,GAAa,EAAE,CAAC;IAExB,UAAU,CAAS;IACnB,UAAU,CAAS;IAC1B,YAAY,CAAe;IAE3B,YACU,aAA4B,EAC5B,WAAwB,EACzB,WAAwB;QAFvB,kBAAa,GAAb,aAAa,CAAe;QAC5B,gBAAW,GAAX,WAAW,CAAa;QACzB,gBAAW,GAAX,WAAW,CAAa;QAE/B,IAAI,CAAC,aAAa,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE;YACpD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;IACL,CAAC;IAED,QAAQ;QACN,wBAAwB;QACxB,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,IAAI,MAAM,GAAY,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC/C,IAAI,CAAC,UAAU,GAAG,MAAM,CAAC,KAAK,CAAC;YAE/B,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,SAAS,CAAC,CAAC,WAAW,EAAE,EAAE;gBACvE,IAAI,SAAS,CAAC;gBACd,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;oBACpB,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;oBACpE,IAAI,OAAO,GAAG,EAAE,CAAC;oBACjB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,SAAS,EAAE,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;wBAC3C,IAAI,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;wBACxB,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBACrB,CAAC;oBACD,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;gBACzB,CAAC;qBAAM,CAAC;oBACN,SAAS,GAAG,WAAW,CAAC;oBACxB,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;gBAC3B,CAAC;YACH,CAAC,CAAC,CAAC;YAEH,IAAI,QAAQ,GAAG,MAAM,CAAC,gBAAgB,CAAC;YACvC,IAAI,QAAQ,EAAE,CAAC;gBACb,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC,SAAS,CAC/D,CAAC,WAAW,EAAE,EAAE;oBACd,IAAI,WAAW,IAAI,SAAS,EAAE,CAAC;wBAC7B,IAAI,WAAW,CAAC,QAAQ,IAAI,SAAS,IAAI,WAAW,CAAC,cAAc,IAAI,MAAM,CAAC,gBAAgB,EAAE,CAAC;4BAC/F,IAAI,CAAC,aAAa,GAAG,WAAW,CAAC,QAAQ,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;4BAC7D,IAAI,KAAK,GAAG,EAAE,CAAC;4BACf,KAAK,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC;4BACnC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;wBAC1B,CAAC;wBACD,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;oBAC5D,CAAC;gBACH,CAAC,CACF,CAAC;YACJ,CAAC;QACH,CAAC;IACH,CAAC;IAED,WAAW,CAAC,KAAU;QACpB,IAAI,MAAM,GAAG,IAAI,aAAa,EAAE,CAAC;QACjC,MAAM,CAAC,cAAc,GAAG,IAAI,CAAC,EAAE,CAAC;QAChC,MAAM,CAAC,QAAQ,GAAG,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QAC3D,MAAM,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;QAC5C,MAAM,CAAC,WAAW,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACzD,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC9B,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC;QAC5B,CAAC;IACH,CAAC;wGAjFU,oBAAoB;4FAApB,oBAAoB,6WCbjC,s2BAmBE;;4FDNW,oBAAoB;kBALhC,SAAS;+BACE,kBAAkB;sIAMnB,OAAO;sBAAf,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,EAAE;sBAAV,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACI,WAAW;sBAApB,MAAM","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { ChangeWrapper } from '../../model/changeWrapper';\nimport { ChangeService } from '../../services/change.service';\nimport { I18nService } from '../../i18n.service';\nimport { DataService } from '../../services/data.service';\nimport { Subscription } from 'rxjs';\nimport { APIMeta } from '../../interfaces/apimeta';\n\n@Component({\n  selector: 'app-custom-radio',\n  templateUrl: './custom-radio.component.html',\n  styleUrls: ['./custom-radio.component.css']\n})\nexport class CustomRadioComponent implements OnInit {\n\n  @Input() options: any[] = [];\n  @Input() apiMeta: string;\n  @Input() selectedValue: string;\n  @Input() progressBar: boolean;\n  @Input() id: string;\n  @Input() readOnly = false;\n  @Input() errorMessage: string;\n  @Input() error: any;\n  @Input() fromShengel: boolean = false;\n  @Input() referenceField: string;\n  @Input() token: string;\n  @Output() valueChange: EventEmitter<ChangeWrapper> = new EventEmitter<ChangeWrapper>();\n  invalidFieldIds: string[] = [];\n\n  public labelField: string;\n  public valueField: string;\n  subscription: Subscription;\n\n  constructor(\n    private changeService: ChangeService,\n    private dataService: DataService,\n    public i18nService: I18nService\n  ) {\n    this.changeService.submitValidate$.subscribe((data) => {\n      this.invalidFieldIds.push(data);\n    });\n  }\n\n  ngOnInit(): void {\n    // VD 31NOV24 null check\n    if (this.apiMeta) {\n      let apiObj: APIMeta = JSON.parse(this.apiMeta);\n      this.labelField = apiObj.field;\n\n      this.dataService.apiResponse(apiObj.endpoint)?.subscribe((apiResponse) => {\n        let responses;\n        if (apiObj.variable) {\n          responses = this.dataService.getValue(apiResponse, apiObj.variable);\n          let results = [];\n          for (let i = 0; i < responses?.length; i++) {\n            var resp = responses[i];\n            results.push(resp);\n          }\n          this.options = results;\n        } else {\n          responses = apiResponse;\n          this.options = responses;\n        }\n      });\n\n      let sourceId = apiObj.sourceQuestionId;\n      if (sourceId) {\n        this.subscription = this.changeService.changeAnnounced$.subscribe(\n          (changeValue) => {\n            if (changeValue != undefined) {\n              if (changeValue.valueObj != undefined && changeValue.fromQuestionId == apiObj.sourceQuestionId) {\n                this.selectedValue = changeValue.valueObj[apiObj.valueField];\n                let value = {};\n                value['name'] = this.selectedValue;\n                this.radioChange(value);\n              }\n              this.changeService.confirmChange(apiObj.sourceQuestionId);\n            }\n          }\n        );\n      }\n    }\n  }\n\n  radioChange(event: any) {\n    let change = new ChangeWrapper();\n    change.fromQuestionId = this.id;\n    change.valueObj = event.target?.id ? event.target?.id : '';\n    change.referenceField = this.referenceField;\n    change.selectedObj = event ? event[this.labelField] : '';\n    this.valueChange.emit(change);\n    if (event) {\n      this.invalidFieldIds = [];\n    }\n  }\n\n}\n","<!-- RS 09DEC24 Changed keys--> \n<div class=\"custom-radio-container\">\n    <div\n      *ngFor=\"let option of options\"\n      [class]=\"invalidFieldIds.includes(id) || error ? 'custom-radio-option invalid' : 'custom-radio-option'\"\n    >\n      <input\n        type=\"radio\"\n        [id]=\"apiMeta !== undefined ? option[labelField] : option.value \"\n        [checked] = \"selectedValue == option.value \"\n        [name]=\"id\"\n        [value]=\"selectedValue\"\n        (change)=\"radioChange($event)\"\n        [disabled]=\"readOnly\"\n      />\n      <label class=\"nxt-radio-label\" [for]=\"apiMeta !== undefined ? option[labelField] : option.value\">{{ apiMeta !== undefined ? option[labelField] : option.value  }}</label>\n    </div>\n    <span *ngIf=\"error || invalidFieldIds.includes(id)\" class=\"error-msg\">{{ errorMessage }}</span>\n  </div>\n  "]}
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-radio.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/custom-radio/custom-radio.component.ts","../../../../../../projects/nxt-app/src/lib/components/custom-radio/custom-radio.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { ChangeWrapper } from '../../model/changeWrapper';\nimport { ChangeService } from '../../services/change.service';\nimport { I18nService } from '../../i18n.service';\nimport { DataService } from '../../services/data.service';\nimport { Subscription } from 'rxjs';\nimport { APIMeta } from '../../interfaces/apimeta';\n\n@Component({\n  selector: 'app-custom-radio',\n  templateUrl: './custom-radio.component.html',\n  styleUrls: ['./custom-radio.component.css']\n})\nexport class CustomRadioComponent implements OnInit {\n\n  @Input() options: any[] = [];\n  @Input() apiMeta: string;\n  @Input() selectedValue: string;\n  @Input() progressBar: boolean;\n  @Input() id: string;\n  @Input() readOnly = false;\n  @Input() errorMessage: string;\n  @Input() error: any;\n  @Input() fromShengel: boolean = false;\n  @Input() referenceField: string;\n  @Input() token: string;\n  @Output() valueChange: EventEmitter<ChangeWrapper> = new EventEmitter<ChangeWrapper>();\n  invalidFieldIds: string[] = [];\n\n  public labelField: string;\n  public valueField: string;\n  subscription: Subscription;\n\n  constructor(\n    private changeService: ChangeService,\n    private dataService: DataService,\n    public i18nService: I18nService\n  ) {\n    this.changeService.submitValidate$.subscribe((data) => {\n      this.invalidFieldIds.push(data);\n    });\n  }\n\n  ngOnInit(): void {\n    // VD 31NOV24 null check\n    if (this.apiMeta) {\n      let apiObj: APIMeta = JSON.parse(this.apiMeta);\n      this.labelField = apiObj.field;\n\n      this.dataService.apiResponse(apiObj.endpoint)?.subscribe((apiResponse) => {\n        let responses;\n        if (apiObj.variable) {\n          responses = this.dataService.getValue(apiResponse, apiObj.variable);\n          let results = [];\n          for (let i = 0; i < responses?.length; i++) {\n            var resp = responses[i];\n            results.push(resp);\n          }\n          this.options = results;\n        } else {\n          responses = apiResponse;\n          this.options = responses;\n        }\n      });\n\n      let sourceId = apiObj.sourceQuestionId;\n      if (sourceId) {\n        this.subscription = this.changeService.changeAnnounced$.subscribe(\n          (changeValue) => {\n            if (changeValue != undefined) {\n              if (changeValue.valueObj != undefined && changeValue.fromQuestionId == apiObj.sourceQuestionId) {\n                this.selectedValue = changeValue.valueObj[apiObj.valueField];\n                let value = {};\n                value['name'] = this.selectedValue;\n                this.radioChange(value);\n              }\n              this.changeService.confirmChange(apiObj.sourceQuestionId);\n            }\n          }\n        );\n      }\n    }\n  }\n\n  radioChange(event: any) {\n    let change = new ChangeWrapper();\n    change.fromQuestionId = this.id;\n    change.valueObj = event.target?.id ? event.target?.id : '';\n    change.referenceField = this.referenceField;\n    change.selectedObj = event ? event[this.labelField] : '';\n    this.valueChange.emit(change);\n    if (event) {\n      this.invalidFieldIds = [];\n    }\n  }\n\n}\n","<!-- RS 09DEC24 Changed keys-->\n <!-- RS 019JAN25 -->\n <!-- validate for NULL  -->\n<div class=\"custom-radio-container\">\n  <div\n    *ngFor=\"let option of options\"\n    [class]=\"invalidFieldIds.includes(id) || error ? 'custom-radio-option invalid' : 'custom-radio-option'\"\n  >\n    <input\n      type=\"radio\"\n      [id]=\"apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value\"\n      [checked] = \"selectedValue == option.value\"\n      [name]=\"id\"\n      [value]=\"selectedValue\"\n      (change)=\"radioChange($event)\"\n      [disabled]=\"readOnly\"\n    />\n    <label class=\"nxt-radio-label\" [for]=\"apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value\">{{ apiMeta !== undefined && apiMeta !== null ? option[labelField] : option.value}}</label>\n  </div>\n  <span *ngIf=\"error || invalidFieldIds.includes(id)\" class=\"error-msg\">{{ errorMessage }}</span>\n</div>"]}
@@ -6,18 +6,32 @@ import { FormsModule } from '@angular/forms';
6
6
  import Quill from 'quill';
7
7
  import { Mention } from 'quill-mention';
8
8
  import ImageResizor from 'quill-image-resizor';
9
+ import { CommonModule } from '@angular/common';
9
10
  import * as i0 from "@angular/core";
10
11
  import * as i1 from "../../i18n.service";
11
12
  import * as i2 from "../../services/change.service";
12
13
  import * as i3 from "@angular/forms";
14
+ import * as i4 from "@angular/common";
15
+ // RS 17JAN2025
16
+ // An array fontFamilyArr is created containing a list of font family names as strings.
17
+ const fontFamilyArr = ["Roboto", "Roboto Condensed", "Arial", "Verdana", "Tahoma", "Trebuchet MS",
18
+ "Georgia", "Times New Roman", "Courier New", "Palatino Linotype",
19
+ "Segoe UI", "Calibri", "Calibri Light", "Sans-Serif", "Helvetica",
20
+ "Impact", "Garamond", "Comic Sans MS", "Lucida Console", "Franklin Gothic Medium"];
13
21
  // Register Quill modules
14
22
  Quill.register('modules/mention', Mention);
15
23
  ImageResizor.Quill = Quill;
16
24
  Quill.register('modules/imageResizor', ImageResizor);
25
+ // RS 17JAN2025
26
+ //The Quill font style attributor is imported and it is registered with Quil
27
+ let fonts = Quill.import("attributors/style/font");
28
+ fonts.whitelist = fontFamilyArr;
29
+ Quill.register(fonts, true);
17
30
  export const QuillConfiguration = {
18
31
  imageResizor: {},
19
32
  toolbar: {
20
33
  container: [
34
+ [{ 'font': fontFamilyArr }], //RS 17JAN2025 Added font family dropdown
21
35
  ['bold', 'italic', 'underline', 'strike'],
22
36
  ['blockquote', 'code-block'],
23
37
  [{ header: [1, 2, 3, 4, 5, 6, false] }],
@@ -129,11 +143,11 @@ export class CustomRichTextComponent {
129
143
  }
130
144
  }
131
145
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomRichTextComponent, deps: [{ token: i1.I18nService }, { token: i2.ChangeService }], target: i0.ɵɵFactoryTarget.Component });
132
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CustomRichTextComponent, isStandalone: true, selector: "app-custom-rich-text", inputs: { value: "value", placeholder: "placeholder", error: "error", question: "question", rows: "rows", readOnly: "readOnly", minLength: "minLength", maxLength: "maxLength" }, outputs: { textValueChange: "textValueChange" }, ngImport: i0, template: "<!-- RS 06JAN25 -->\n<div class=\"rich-text-container\">\n <quill-editor\n [(ngModel)]=\"value\"\n [placeholder]=\"placeholder\"\n [modules]=\"quillConfiguration\"\n [readOnly]=\"readOnly\"\n (onContentChanged)=\"onEditorChange($event)\"\n [class.error]=\"error\">\n </quill-editor>\n <div *ngIf=\"error\" class=\"error-message\">\n {{ error }}\n </div>\n</div>", styles: [".rich-text-container{width:100%;margin:10px 0}.error{border:1px solid red}:is() .ql-editor img{cursor:pointer}:is() .image-resizer{display:block!important;visibility:visible!important}:is() .ql-editor .image-resizer{border:1px dashed #000;position:absolute}:is() .ql-editor .image-resizer .handle{background-color:#000;border:1px solid #fff;border-radius:50%;height:12px;width:12px;position:absolute}\n"], dependencies: [{ kind: "component", type: QuillEditorComponent, selector: "quill-editor" }, { kind: "ngmodule", type: FormsModule }, { 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"] }] });
146
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CustomRichTextComponent, isStandalone: true, selector: "app-custom-rich-text", inputs: { value: "value", placeholder: "placeholder", error: "error", question: "question", rows: "rows", readOnly: "readOnly", minLength: "minLength", maxLength: "maxLength" }, outputs: { textValueChange: "textValueChange" }, ngImport: i0, template: "<!-- RS 06JAN25 -->\n<div class=\"rich-text-container\">\n <quill-editor\n [(ngModel)]=\"value\"\n [placeholder]=\"placeholder\"\n [modules]=\"quillConfiguration\"\n [readOnly]=\"readOnly\"\n (onContentChanged)=\"onEditorChange($event)\"\n [class.error]=\"error\">\n </quill-editor>\n <div *ngIf=\"error\" class=\"error-message\">\n {{ error }}\n </div>\n</div>", styles: [".rich-text-container{width:100%;margin:10px 0}.error{border:1px solid red}:is() .ql-editor img{cursor:pointer}:is() .image-resizer{display:block!important;visibility:visible!important}:is() .ql-editor .image-resizer{border:1px dashed #000;position:absolute}:is() .ql-editor .image-resizer .handle{background-color:#000;border:1px solid #fff;border-radius:50%;height:12px;width:12px;position:absolute}\n"], dependencies: [{ kind: "component", type: QuillEditorComponent, selector: "quill-editor" }, { kind: "ngmodule", type: FormsModule }, { 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: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
133
147
  }
134
148
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomRichTextComponent, decorators: [{
135
149
  type: Component,
136
- args: [{ imports: [QuillEditorComponent, FormsModule], selector: 'app-custom-rich-text', standalone: true, template: "<!-- RS 06JAN25 -->\n<div class=\"rich-text-container\">\n <quill-editor\n [(ngModel)]=\"value\"\n [placeholder]=\"placeholder\"\n [modules]=\"quillConfiguration\"\n [readOnly]=\"readOnly\"\n (onContentChanged)=\"onEditorChange($event)\"\n [class.error]=\"error\">\n </quill-editor>\n <div *ngIf=\"error\" class=\"error-message\">\n {{ error }}\n </div>\n</div>", styles: [".rich-text-container{width:100%;margin:10px 0}.error{border:1px solid red}:is() .ql-editor img{cursor:pointer}:is() .image-resizer{display:block!important;visibility:visible!important}:is() .ql-editor .image-resizer{border:1px dashed #000;position:absolute}:is() .ql-editor .image-resizer .handle{background-color:#000;border:1px solid #fff;border-radius:50%;height:12px;width:12px;position:absolute}\n"] }]
150
+ args: [{ imports: [QuillEditorComponent, FormsModule, CommonModule], selector: 'app-custom-rich-text', standalone: true, template: "<!-- RS 06JAN25 -->\n<div class=\"rich-text-container\">\n <quill-editor\n [(ngModel)]=\"value\"\n [placeholder]=\"placeholder\"\n [modules]=\"quillConfiguration\"\n [readOnly]=\"readOnly\"\n (onContentChanged)=\"onEditorChange($event)\"\n [class.error]=\"error\">\n </quill-editor>\n <div *ngIf=\"error\" class=\"error-message\">\n {{ error }}\n </div>\n</div>", styles: [".rich-text-container{width:100%;margin:10px 0}.error{border:1px solid red}:is() .ql-editor img{cursor:pointer}:is() .image-resizer{display:block!important;visibility:visible!important}:is() .ql-editor .image-resizer{border:1px dashed #000;position:absolute}:is() .ql-editor .image-resizer .handle{background-color:#000;border:1px solid #fff;border-radius:50%;height:12px;width:12px;position:absolute}\n"] }]
137
151
  }], ctorParameters: () => [{ type: i1.I18nService }, { type: i2.ChangeService }], propDecorators: { value: [{
138
152
  type: Input
139
153
  }], placeholder: [{
@@ -153,4 +167,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
153
167
  }], maxLength: [{
154
168
  type: Input
155
169
  }] } });
156
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-rich-text.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/custom-rich-text/custom-rich-text.component.ts","../../../../../../projects/nxt-app/src/lib/components/custom-rich-text/custom-rich-text.component.html"],"names":[],"mappings":"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","sourcesContent":["// Changes commented out due to Angular version compatibility; will apply after upgrade.\n// // RS 06JAN2025\nimport { Component, EventEmitter, Input, OnInit, Output, OnDestroy, CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';\nimport { I18nService } from '../../i18n.service';\nimport { Subscription } from 'rxjs';\nimport { ChangeService } from '../../services/change.service';\nimport { DependencyMeta } from '../../interfaces/dependencyMeta';\nimport { QuillEditorComponent, QuillModules } from 'ngx-quill';\nimport { FormsModule } from '@angular/forms'\nimport Quill from 'quill';\nimport { Mention } from 'quill-mention';\nimport ImageResizor from 'quill-image-resizor'\n\n// Register Quill modules\nQuill.register('modules/mention', Mention);\nImageResizor.Quill = Quill\nQuill.register('modules/imageResizor', ImageResizor)\n\nexport const QuillConfiguration: QuillModules = {\n    imageResizor: {},\n  toolbar: {\n    container: [\n      ['bold', 'italic', 'underline', 'strike'],\n      ['blockquote', 'code-block'],\n      [{ header: [1, 2, 3, 4, 5, 6, false] }],\n      [{ list: 'ordered' }, { list: 'bullet' }],\n      [{ color: [] }, { background: [] }],\n      [{ align: [] }],\n      ['link', 'image', 'video'],\n      ['formula'],\n      ['clean'],\n    ],\n  },\n  mention: {\n    allowedChars: /^[A-Za-z\\sÅÄÖåäö]*$/,\n    mentionDenotationChars: ['@', '#'],\n    source: function (searchTerm: string, renderList: any) {\n      const values = [\n        { id: 1, value: 'User 1' },\n        { id: 2, value: 'User 2' },\n      ];\n\n      if (searchTerm.length === 0) {\n        renderList(values, searchTerm);\n      } else {\n        const matches = values.filter((item) =>\n          item.value.toLowerCase().includes(searchTerm.toLowerCase())\n        );\n        renderList(matches, searchTerm);\n      }\n    },\n  },\n};\n\n// resize: {\n//     displaySize: true,\n//     modules: ['Resize', 'DisplaySize', 'Toolbar'],\n//     toolbarStyles: {\n//         backgroundColor: 'black',\n//         border: 'none',\n//         color: 'white'\n//     },\n//     handleStyles: {\n//         backgroundColor: 'black',\n//         border: 'none',\n//         color: 'white'\n//     }\n// },\n@Component({\n    imports: [QuillEditorComponent, FormsModule],\n    selector: 'app-custom-rich-text',\n    standalone: true,\n    templateUrl: './custom-rich-text.component.html',\n    styleUrls: ['./custom-rich-text.component.css']\n})\n\nexport class CustomRichTextComponent implements OnInit, OnDestroy { \n  @Input() value: any | any[] = '';\n  @Input() placeholder: string;\n  @Input() error: any;\n  @Input() question: any;\n  @Input() rows:Number; //The number of visible text lines for the control\n  @Input() readOnly = false;\n  @Output() textValueChange: EventEmitter<string> = new EventEmitter<string>();\n  @Input() minLength?: number;\n  @Input() maxLength?: number;\n  // @Input() value: any = ''; // Set default value\n\n  quillConfiguration: QuillModules = QuillConfiguration;\n  subscription: Subscription;  \n\n  constructor(\n    public i18nService: I18nService,\n    private changeService: ChangeService\n  ) { }\n\n  ngOnInit(): void {\n    console.log('Rich Text Init:', {\n      value: this.value,\n      question: this.question\n    });\n    if (this.value === undefined || this.value === null) {\n      this.value = '';\n      this.textValueChange.emit('');\n\n    }\n    this.initializeDependency();\n  }\n\n  // Separated dependency initialization for better organization\n  private initializeDependency(): void {\n    if (this.question?.subText) {\n        console.log('subText:', this.question.subText);\n        try {\n            const dependencyObj: DependencyMeta = JSON.parse(this.question.subText);\n            if (dependencyObj?.sourceQuestionId) {\n                this.subscription = this.changeService.changeAnnounced$.subscribe(\n                    (changeValue) => {\n                        console.log('Change Value:', changeValue);\n                        if (changeValue && changeValue.valueObj &&\n                            changeValue.fromQuestionId === dependencyObj.sourceQuestionId) {\n                            this.value = changeValue.valueObj[dependencyObj.valueField];\n                            this.textValueChange.emit(this.value);\n                        }\n                        this.changeService.confirmChange(dependencyObj.sourceQuestionId);\n                    }\n                );\n            }\n        } catch (error) {\n            console.error('Error parsing subText:', error);\n        }\n    }\n}\n\n\nonEditorChange(event: any): void {\n  if (event && event.html !== undefined) {\n    this.textValueChange.emit(event.html);\n  } else {\n    this.textValueChange.emit('');\n  }\n}\n  // Added ngOnDestroy to prevent memory leaks\n  ngOnDestroy(): void {\n    if (this.subscription) {\n      this.subscription.unsubscribe();\n    }\n  }\n}\n\n","<!-- RS 06JAN25 -->\n<div class=\"rich-text-container\">\n  <quill-editor\n    [(ngModel)]=\"value\"\n    [placeholder]=\"placeholder\"\n    [modules]=\"quillConfiguration\"\n    [readOnly]=\"readOnly\"\n    (onContentChanged)=\"onEditorChange($event)\"\n    [class.error]=\"error\">\n  </quill-editor>\n  <div *ngIf=\"error\" class=\"error-message\">\n    {{ error }}\n  </div>\n</div>"]}
170
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-rich-text.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/custom-rich-text/custom-rich-text.component.ts","../../../../../../projects/nxt-app/src/lib/components/custom-rich-text/custom-rich-text.component.html"],"names":[],"mappings":"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","sourcesContent":["// Changes commented out due to Angular version compatibility; will apply after upgrade.\n// // RS 06JAN2025\nimport { Component, EventEmitter, Input, OnInit, Output, OnDestroy, CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';\nimport { I18nService } from '../../i18n.service';\nimport { Subscription } from 'rxjs';\nimport { ChangeService } from '../../services/change.service';\nimport { DependencyMeta } from '../../interfaces/dependencyMeta';\nimport { QuillEditorComponent, QuillModules } from 'ngx-quill';\nimport { FormsModule } from '@angular/forms'\nimport Quill from 'quill';\nimport { Mention } from 'quill-mention';\nimport ImageResizor from 'quill-image-resizor'\nimport { CommonModule } from '@angular/common';\n\n// RS 17JAN2025 \n// An array fontFamilyArr is created containing a list of font family names as strings.\nconst fontFamilyArr = [\"Roboto\", \"Roboto Condensed\", \"Arial\", \"Verdana\", \"Tahoma\", \"Trebuchet MS\",\n  \"Georgia\", \"Times New Roman\", \"Courier New\", \"Palatino Linotype\",\n  \"Segoe UI\", \"Calibri\", \"Calibri Light\", \"Sans-Serif\", \"Helvetica\",\n  \"Impact\", \"Garamond\", \"Comic Sans MS\", \"Lucida Console\", \"Franklin Gothic Medium\"];\n\n// Register Quill modules\nQuill.register('modules/mention', Mention);\nImageResizor.Quill = Quill\nQuill.register('modules/imageResizor', ImageResizor)\n\n// RS 17JAN2025 \n//The Quill font style attributor is imported and it is registered with Quil\nlet fonts: any = Quill.import(\"attributors/style/font\");\nfonts.whitelist = fontFamilyArr;\nQuill.register(fonts, true);\n\nexport const QuillConfiguration: QuillModules = {\n    imageResizor: {},\n  toolbar: {\n    container: [\n      [{ 'font': fontFamilyArr }],  //RS 17JAN2025  Added font family dropdown\n      ['bold', 'italic', 'underline', 'strike'],\n      ['blockquote', 'code-block'],\n      [{ header: [1, 2, 3, 4, 5, 6, false] }],\n      [{ list: 'ordered' }, { list: 'bullet' }],\n      [{ color: [] }, { background: [] }],\n      [{ align: [] }],\n      ['link', 'image', 'video'],\n      ['formula'],\n      ['clean'],\n    ],\n  },\n  mention: {\n    allowedChars: /^[A-Za-z\\sÅÄÖåäö]*$/,\n    mentionDenotationChars: ['@', '#'],\n    source: function (searchTerm: string, renderList: any) {\n      const values = [\n        { id: 1, value: 'User 1' },\n        { id: 2, value: 'User 2' },\n      ];\n\n      if (searchTerm.length === 0) {\n        renderList(values, searchTerm);\n      } else {\n        const matches = values.filter((item) =>\n          item.value.toLowerCase().includes(searchTerm.toLowerCase())\n        );\n        renderList(matches, searchTerm);\n      }\n    },\n  },\n};\n\n// resize: {\n//     displaySize: true,\n//     modules: ['Resize', 'DisplaySize', 'Toolbar'],\n//     toolbarStyles: {\n//         backgroundColor: 'black',\n//         border: 'none',\n//         color: 'white'\n//     },\n//     handleStyles: {\n//         backgroundColor: 'black',\n//         border: 'none',\n//         color: 'white'\n//     }\n// },\n@Component({\n    imports: [QuillEditorComponent, FormsModule ,CommonModule],\n    selector: 'app-custom-rich-text',\n    standalone: true,\n    templateUrl: './custom-rich-text.component.html',\n    styleUrls: ['./custom-rich-text.component.css']\n})\n\nexport class CustomRichTextComponent implements OnInit, OnDestroy { \n  @Input() value: any | any[] = '';\n  @Input() placeholder: string;\n  @Input() error: any;\n  @Input() question: any;\n  @Input() rows:Number; //The number of visible text lines for the control\n  @Input() readOnly = false;\n  @Output() textValueChange: EventEmitter<string> = new EventEmitter<string>();\n  @Input() minLength?: number;\n  @Input() maxLength?: number;\n  // @Input() value: any = ''; // Set default value\n\n  quillConfiguration: QuillModules = QuillConfiguration;\n  subscription: Subscription;  \n\n  constructor(\n    public i18nService: I18nService,\n    private changeService: ChangeService\n  ) { }\n\n  ngOnInit(): void {\n    console.log('Rich Text Init:', {\n      value: this.value,\n      question: this.question\n    });\n    if (this.value === undefined || this.value === null) {\n      this.value = '';\n      this.textValueChange.emit('');\n\n    }\n    this.initializeDependency();\n  }\n\n  // Separated dependency initialization for better organization\n  private initializeDependency(): void {\n    if (this.question?.subText) {\n        console.log('subText:', this.question.subText);\n        try {\n            const dependencyObj: DependencyMeta = JSON.parse(this.question.subText);\n            if (dependencyObj?.sourceQuestionId) {\n                this.subscription = this.changeService.changeAnnounced$.subscribe(\n                    (changeValue) => {\n                        console.log('Change Value:', changeValue);\n                        if (changeValue && changeValue.valueObj &&\n                            changeValue.fromQuestionId === dependencyObj.sourceQuestionId) {\n                            this.value = changeValue.valueObj[dependencyObj.valueField];\n                            this.textValueChange.emit(this.value);\n                        }\n                        this.changeService.confirmChange(dependencyObj.sourceQuestionId);\n                    }\n                );\n            }\n        } catch (error) {\n            console.error('Error parsing subText:', error);\n        }\n    }\n}\n\n\nonEditorChange(event: any): void {\n  if (event && event.html !== undefined) {\n    this.textValueChange.emit(event.html);\n  } else {\n    this.textValueChange.emit('');\n  }\n}\n  // Added ngOnDestroy to prevent memory leaks\n  ngOnDestroy(): void {\n    if (this.subscription) {\n      this.subscription.unsubscribe();\n    }\n  }\n}\n\n","<!-- RS 06JAN25 -->\n<div class=\"rich-text-container\">\n  <quill-editor\n    [(ngModel)]=\"value\"\n    [placeholder]=\"placeholder\"\n    [modules]=\"quillConfiguration\"\n    [readOnly]=\"readOnly\"\n    (onContentChanged)=\"onEditorChange($event)\"\n    [class.error]=\"error\">\n  </quill-editor>\n  <div *ngIf=\"error\" class=\"error-message\">\n    {{ error }}\n  </div>\n</div>"]}
@@ -20,6 +20,11 @@ export class FileUploadComponent {
20
20
  tableFile; // file Upload from table
21
21
  question;
22
22
  error;
23
+ //RS 09JAN2025
24
+ allowedFileTypes = [];
25
+ maxFileSize = 0; // in bytes(Binary)
26
+ maxFiles = 4; // default limit
27
+ validationErrors = [];
23
28
  selectedFileNameArray = [];
24
29
  copyOfInputAllFiles;
25
30
  copyOfFileUploadingLimit;
@@ -34,7 +39,10 @@ export class FileUploadComponent {
34
39
  this.sanitizer = sanitizer;
35
40
  this.i18nService = i18nService;
36
41
  }
37
- ngOnInit() { }
42
+ //RS 09JAN2025
43
+ ngOnInit() {
44
+ this.initializeFileConfigs();
45
+ }
38
46
  // VD 03May24 file upload fix
39
47
  // onFileUpload(event: any) {
40
48
  // const fileUploaderElement: HTMLElement = document.getElementById(
@@ -51,17 +59,51 @@ export class FileUploadComponent {
51
59
  this.copyOfFileUploadingLimit = simpleChanges.limitFileUploading?.currentValue;
52
60
  }
53
61
  }
62
+ //RS 09JAN2025 Initializes file upload settings (allowed types, max size, and max files) by parsing metadata from fieldsMeta
63
+ initializeFileConfigs() {
64
+ if (this.question?.fieldsMeta) {
65
+ try {
66
+ const metaData = JSON.parse(this.question.fieldsMeta);
67
+ const fileConfig = metaData[0]; // Get first config object
68
+ if (fileConfig) {
69
+ this.allowedFileTypes = fileConfig.allowedTypes || [];
70
+ this.maxFileSize = fileConfig.maxFileSize || 0;
71
+ this.maxFiles = fileConfig.maxFiles || 5;
72
+ }
73
+ }
74
+ catch (error) {
75
+ console.error('Error parsing fieldsMeta:', error);
76
+ }
77
+ }
78
+ }
54
79
  uploadMultipleFiles(event) {
55
80
  console.log('File uploader initiated');
81
+ this.validationErrors = []; // RS 17JAN2025 Clear previous errors
56
82
  let inputFiles = this.copyOfInputAllFiles ? this.copyOfInputAllFiles : [];
57
83
  const selectedFileData = [];
58
84
  const uploadedFiles = event.target.files;
59
- if (uploadedFiles.length + inputFiles.length <= this.copyOfFileUploadingLimit) {
60
- const readFilesPromises = [];
61
- for (const eachUploadedFile of uploadedFiles) {
62
- const file = eachUploadedFile;
63
- const format = file.name.split('.')[1];
64
- // Create a promise for each file read
85
+ // RS 09JAN2025 Check file count limit
86
+ if (uploadedFiles.length + inputFiles.length > this.maxFiles) {
87
+ this.validationErrors.push(`Maximum ${this.maxFiles} files allowed`);
88
+ event.target.value = '';
89
+ return;
90
+ }
91
+ const readFilesPromises = [];
92
+ for (const eachUploadedFile of uploadedFiles) {
93
+ const file = eachUploadedFile;
94
+ //RS 09JAN2025 Check file type
95
+ const fileExt = '.' + file.name.split('.').pop().toLowerCase();
96
+ if (this.allowedFileTypes.length > 0 && !this.allowedFileTypes.includes(fileExt)) {
97
+ this.validationErrors.push(`File type ${fileExt} not allowed. Allowed types: ${this.allowedFileTypes.join(', ')}`);
98
+ continue;
99
+ }
100
+ //RS 09JAN2025 Check file size
101
+ if (this.maxFileSize > 0 && file.size > this.maxFileSize) {
102
+ this.validationErrors.push(`File "${file.name}" exceeds size limit of ${this.maxFileSize / 1048576}MB`);
103
+ continue;
104
+ }
105
+ if (!this.validationErrors.length) {
106
+ const format = fileExt.substring(1);
65
107
  const fileReadPromise = new Promise((resolve, reject) => {
66
108
  const reader = new FileReader();
67
109
  reader.readAsDataURL(file);
@@ -71,6 +113,7 @@ export class FileUploadComponent {
71
113
  name: file.name,
72
114
  type: file.type,
73
115
  format,
116
+ size: file.size, // RS 17JAN2025
74
117
  id: null,
75
118
  };
76
119
  selectedFileData.push(fileData);
@@ -79,10 +122,13 @@ export class FileUploadComponent {
79
122
  };
80
123
  reader.onerror = () => {
81
124
  reject();
125
+ this.validationErrors.push(`Error reading file "${file.name}"`); // RS 17JAN2025
82
126
  };
83
127
  });
84
128
  readFilesPromises.push(fileReadPromise);
85
129
  }
130
+ }
131
+ if (readFilesPromises.length > 0) {
86
132
  // Wait for all promises to resolve
87
133
  Promise.all(readFilesPromises).then(() => {
88
134
  this.copyOfInputAllFiles = inputFiles;
@@ -206,11 +252,11 @@ export class FileUploadComponent {
206
252
  }
207
253
  }
208
254
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FileUploadComponent, deps: [{ token: i1.SharedService }, { token: i2.DataService }, { token: i3.DomSanitizer }, { token: i4.I18nService }], target: i0.ɵɵFactoryTarget.Component });
209
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FileUploadComponent, selector: "app-file-upload", inputs: { allFiles: "allFiles", limitFileUploading: "limitFileUploading", isDeleteFileButtonVisible: "isDeleteFileButtonVisible", isShowNoFileIcon: "isShowNoFileIcon", tableFile: "tableFile", question: "question", error: "error" }, outputs: { selectedFileData: "selectedFileData", deletedFileData: "deletedFileData" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"row\" style=\"margin: 0;\">\n <div class=\"col-md-12\" *ngIf=\"!isDeleteFileButtonVisible\" style=\"text-align: left;\">\n <!-- HA 19DEC23 For translation -->\n <label class=\"she-label\">{{ 'attachment' | i18n:i18nService.currentLanguage }}</label>\n </div>\n <div class=\"col-lg-5 document-cnt m-t-10 m-b-10\"\n *ngFor=\"let eachFile of copyOfInputAllFiles; let currentFileIndex = index\">\n <div *ngIf=\"!tableFile\"> <!-- (click)=\"viewFile(eachFile)\" -->\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.name)\">\n </div>\n <div class=\"col-lg-9 document_name\">\n {{eachFile?.name}}\n </div>\n <div class=\"document_delete\" (click)=\"deleteFile(currentFileIndex);$event.stopPropagation()\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/bin.svg\">\n </div>\n <!-- VD 20May24 - preview changes-->\n <div class=\"preview-icon\" (click)=\"viewFile(eachFile)\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/icons8-eye-24.png\">\n </div>\n </div>\n <div *ngIf=\"tableFile\" class=\"row\" style=\"cursor: pointer;\">\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.doc)\" style=\"margin-right: 10px;height: 40px;\">\n </div>\n <div class=\"col-lg-9 document_name\" style=\"overflow:hidden;height:35px;\">\n {{eachFile?.name}}\n </div>\n </div>\n </div>\n <div class=\"col-md-12 m-b-30 text-center\" *ngIf=\"copyOfInputAllFiles?.length === 0 && isShowNoFileIcon\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/ic_no_attachments.svg\" style=\"height: 140px;\">\n </div>\n</div>\n\n<div class=\"col-lg-6\" style=\"padding:0;margin-top: 10px;\">\n <label class=\"custom-file\" *ngIf=\"isDeleteFileButtonVisible && !tableFile\">\n <!-- HA 19DEC23 For translation -->\n <!-- VD 03May24 file upload fix-->\n <button [class]=\"error ? 'she-btn-primary-bordered Invalid' : 'she-btn-primary-bordered'\" style=\"width:275px;\" (click)=\"fileInput.click()\"\n [ngClass]=\"{ 'btn-disabled': copyOfInputAllFiles?.length >= copyOfFileUploadingLimit }\" [disabled]=\"copyOfInputAllFiles?.length >= copyOfFileUploadingLimit\"\n >Choose Files</button>\n\n <input #fileInput type=\"file\" name=\"fileUpload\" multiple=\"multiple\" accept=\"*\" style=\"display:none;\"\n (change) = uploadMultipleFiles($event)\n />\n </label>\n<!-- HA 19DEC23 For translation -->\n <label *ngIf=\"tableFile\" class=\"btn btn-primary\">\n {{ 'uploadFile' | i18n:i18nService.currentLanguage }}\n <input id=\"file\" style=\"display: none;\" type=\"file\" class=\"form-control\" (change)= uploadMultipleFiles($event) />\n </label>\n <!-- <span class=\"error-msg\" *ngIf=\"(formControl?.documents?.touched || formControl?.documents?.dirty) &&\n formControl?.documents?.errors?.required\">\n {{LM_POLICY_VALIDATION_MESSAGE.REQUIRED}}\n </span> -->\n <!-- RS 09DEC24 Changed keys--> \n</div>\n<div *ngIf=\"error\" class=\"error-msg\" style=\"margin-top: 10px;\">{{question?.errorMessage}}</div>\n<div class=\"nxt-file-overlay\" *ngIf=\"showFile\">\n <div class=\"nxt-file-map-modal\">\n <div class=\"nxt-file-model-content\">\n <div class=\"nxt-file-model-header\">\n <h4>{{fileName}}</h4>\n <button class=\"close-button\" (click)=\"close()\">X</button>\n </div>\n <ng-container *ngIf=\"isImage; else otherFile\">\n <img [src]=\"fileUrl\" class=\"img-fluid\" alt=\"File Preview\" />\n </ng-container>\n <ng-template #otherFile>\n <iframe [src]=\"fileUrl\" width=\"100%\" height=\"500px\"></iframe>\n </ng-template>\n </div>\n </div>\n</div>\n\n\n", styles: [".document-cnt{padding:10px;border:1px solid #ccc;border-radius:5px;background-color:#f9f9f9;margin-bottom:10px;cursor:pointer;transition:background-color .3s ease;position:relative}.document-cnt:hover{background-color:#e9ecef}.document_image img{width:30px;height:30px}.document_name{font-size:14px;font-weight:700;color:#333;line-height:30px}.label{margin-bottom:0}.document_delete img,.preview-icon img{width:20px;height:20px;cursor:pointer;transition:transform .3s ease}.document_delete img{filter:brightness(0) saturate(100%) invert(20%) sepia(97%) saturate(7481%) hue-rotate(357deg) brightness(98%) contrast(119%)}.document_delete img:hover,.preview-icon img:hover{transform:scale(1.1)}.document_delete,.preview-icon{position:absolute;top:50%;transform:translateY(-50%)}.document_delete{right:10px}.preview-icon{right:40px}@media (min-width: 992px){.document-cnt{margin-left:10px}}.btn-disabled{background:#e1e1e1;color:#fff;border:none;border-radius:5px;height:50px}.form-control[disabled]{border-radius:5px}.custom-file{color:#9a9a9a;font-size:14px;font-weight:400;display:inline-block;width:auto;margin-bottom:5px}.she-btn-primary-bordered{background:#fff;color:#48b7ff;border:1px solid #48B7FF;border-radius:5px;height:50px;outline:none!important}.btn-primary{background-color:#03a9f4!important;border:1px solid #03a9f4!important;color:#fff!important}.btn.btn-primary{border-radius:2px;padding:6px 14px;font-size:14px}.nxt-file-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;display:flex;justify-content:center;align-items:center;z-index:1000}.nxt-file-map-modal{background-color:#fff;border-radius:8px;overflow:hidden;width:80%;max-width:800px;max-height:90%;display:flex;flex-direction:column}.nxt-file-model-content{display:flex;flex-direction:column;height:100%}.nxt-file-model-header{display:flex;justify-content:space-between;align-items:center;padding:10px 20px;background-color:#f5f5f5;border-bottom:1px solid #ddd}.nxt-file-model-header h4{margin:0;font-size:1.25em}.close-button{background:none;border:none;font-size:1.5em;cursor:pointer}.img-fluid{max-width:100%;height:auto;flex:1}iframe{border:none}.Invalid{border:1px solid red!important}\n"], dependencies: [{ kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i6.I18nPipe, name: "i18n" }] });
255
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FileUploadComponent, selector: "app-file-upload", inputs: { allFiles: "allFiles", limitFileUploading: "limitFileUploading", isDeleteFileButtonVisible: "isDeleteFileButtonVisible", isShowNoFileIcon: "isShowNoFileIcon", tableFile: "tableFile", question: "question", error: "error" }, outputs: { selectedFileData: "selectedFileData", deletedFileData: "deletedFileData" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"row\" style=\"margin: 0;\">\n <div class=\"col-md-12\" *ngIf=\"!isDeleteFileButtonVisible\" style=\"text-align: left;\">\n <!-- HA 19DEC23 For translation -->\n <label class=\"she-label\">{{ 'attachment' | i18n:i18nService.currentLanguage }}</label>\n </div>\n\n <div class=\"col-lg-5 document-cnt m-t-10 m-b-10\"\n *ngFor=\"let eachFile of copyOfInputAllFiles; let currentFileIndex = index\">\n <div *ngIf=\"!tableFile\"> <!-- (click)=\"viewFile(eachFile)\" -->\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.name)\">\n </div>\n <div class=\"col-lg-9 document_name\">\n {{eachFile?.name}}\n </div>\n <div class=\"document_delete\" (click)=\"deleteFile(currentFileIndex);$event.stopPropagation()\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/bin.svg\">\n </div>\n <!-- VD 20May24 - preview changes-->\n <div class=\"preview-icon\" (click)=\"viewFile(eachFile)\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/icons8-eye-24.png\">\n </div>\n </div>\n <div *ngIf=\"tableFile\" class=\"row\" style=\"cursor: pointer;\">\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.doc)\" style=\"margin-right: 10px;height: 40px;\">\n </div>\n <div class=\"col-lg-9 document_name\" style=\"overflow:hidden;height:35px;\">\n {{eachFile?.name}}\n </div>\n </div>\n </div>\n <div class=\"col-md-12 m-b-30 text-center\" *ngIf=\"copyOfInputAllFiles?.length === 0 && isShowNoFileIcon\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/ic_no_attachments.svg\" style=\"height: 140px;\">\n </div>\n</div>\n\n<div class=\"col-lg-6\" style=\"padding:0;margin-top: 10px;\">\n <label class=\"custom-file\" *ngIf=\"isDeleteFileButtonVisible && !tableFile\">\n <!-- HA 19DEC23 For translation -->\n <!-- VD 03May24 file upload fix-->\n <!-- RS 09JAN2025 -->\n <button [class]=\"error ? 'she-btn-primary-bordered Invalid' : 'she-btn-primary-bordered'\" style=\"width:275px;\" (click)=\"fileInput.click()\"[ngClass]=\"{ 'btn-disabled': copyOfInputAllFiles?.length >= maxFiles }\" [disabled]=\"copyOfInputAllFiles?.length >= maxFiles\">Choose Files</button>\n <input #fileInput type=\"file\" name=\"fileUpload\" multiple=\"multiple\" [accept]=\"allowedFileTypes.join(',')\"\n style=\"display:none;\"\n (change) = uploadMultipleFiles($event)\n />\n </label>\n\n <!-- RS 09JAN2025 -->\n <!-- Error messages section -->\n <div class=\"file-constraints\" *ngIf=\"validationErrors.length > 0\">\n <small class=\"text-danger\">\n <div *ngFor=\"let error of validationErrors\">{{ error }}</div>\n </small>\n </div>\n\n<!-- HA 19DEC23 For translation -->\n <label *ngIf=\"tableFile\" class=\"btn btn-primary\">\n {{ 'uploadFile' | i18n:i18nService.currentLanguage }}\n <input id=\"file\" style=\"display: none;\" type=\"file\" class=\"form-control\" (change)= uploadMultipleFiles($event) />\n </label>\n <!-- <span class=\"error-msg\" *ngIf=\"(formControl?.documents?.touched || formControl?.documents?.dirty) &&\n formControl?.documents?.errors?.required\">\n {{LM_POLICY_VALIDATION_MESSAGE.REQUIRED}}\n </span> -->\n <!-- RS 09DEC24 Changed keys--> \n</div>\n<div *ngIf=\"error\" class=\"error-msg\" style=\"margin-top: 10px;\">{{question?.errorMessage}}</div>\n<div class=\"nxt-file-overlay\" *ngIf=\"showFile\">\n <div class=\"nxt-file-map-modal\">\n <div class=\"nxt-file-model-content\">\n <div class=\"nxt-file-model-header\">\n <h4>{{fileName}}</h4>\n <button class=\"close-button\" (click)=\"close()\">X</button>\n </div>\n <ng-container *ngIf=\"isImage; else otherFile\">\n <img [src]=\"fileUrl\" class=\"img-fluid\" alt=\"File Preview\" />\n </ng-container>\n <ng-template #otherFile>\n <iframe [src]=\"fileUrl\" width=\"100%\" height=\"500px\"></iframe>\n </ng-template>\n </div>\n </div>\n</div>\n\n\n", styles: [".document-cnt{padding:10px;border:1px solid #ccc;border-radius:5px;background-color:#f9f9f9;margin-bottom:10px;cursor:pointer;transition:background-color .3s ease;position:relative}.document-cnt:hover{background-color:#e9ecef}.document_image img{width:30px;height:30px}.document_name{font-size:14px;font-weight:700;color:#333;line-height:30px}.label{margin-bottom:0}.document_delete img,.preview-icon img{width:20px;height:20px;cursor:pointer;transition:transform .3s ease}.document_delete img{filter:brightness(0) saturate(100%) invert(20%) sepia(97%) saturate(7481%) hue-rotate(357deg) brightness(98%) contrast(119%)}.document_delete img:hover,.preview-icon img:hover{transform:scale(1.1)}.document_delete,.preview-icon{position:absolute;top:50%;transform:translateY(-50%)}.document_delete{right:10px}.preview-icon{right:40px}@media (min-width: 992px){.document-cnt{margin-left:10px}}.btn-disabled{background:#e1e1e1;color:#fff;border:none;border-radius:5px;height:50px}.form-control[disabled]{border-radius:5px}.custom-file{color:#9a9a9a;font-size:14px;font-weight:400;display:inline-block;width:auto;margin-bottom:5px}.she-btn-primary-bordered{background:#fff;color:#48b7ff;border:1px solid #48B7FF;border-radius:5px;height:50px;outline:none!important}.btn-primary{background-color:#03a9f4!important;border:1px solid #03a9f4!important;color:#fff!important}.btn.btn-primary{border-radius:2px;padding:6px 14px;font-size:14px}.nxt-file-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;display:flex;justify-content:center;align-items:center;z-index:1000}.nxt-file-map-modal{background-color:#fff;border-radius:8px;overflow:hidden;width:80%;max-width:800px;max-height:90%;display:flex;flex-direction:column}.nxt-file-model-content{display:flex;flex-direction:column;height:100%}.nxt-file-model-header{display:flex;justify-content:space-between;align-items:center;padding:10px 20px;background-color:#f5f5f5;border-bottom:1px solid #ddd}.nxt-file-model-header h4{margin:0;font-size:1.25em}.close-button{background:none;border:none;font-size:1.5em;cursor:pointer}.img-fluid{max-width:100%;height:auto;flex:1}iframe{border:none}.Invalid{border:1px solid red!important}\n"], dependencies: [{ kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i6.I18nPipe, name: "i18n" }] });
210
256
  }
211
257
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FileUploadComponent, decorators: [{
212
258
  type: Component,
213
- args: [{ selector: 'app-file-upload', template: "<div class=\"row\" style=\"margin: 0;\">\n <div class=\"col-md-12\" *ngIf=\"!isDeleteFileButtonVisible\" style=\"text-align: left;\">\n <!-- HA 19DEC23 For translation -->\n <label class=\"she-label\">{{ 'attachment' | i18n:i18nService.currentLanguage }}</label>\n </div>\n <div class=\"col-lg-5 document-cnt m-t-10 m-b-10\"\n *ngFor=\"let eachFile of copyOfInputAllFiles; let currentFileIndex = index\">\n <div *ngIf=\"!tableFile\"> <!-- (click)=\"viewFile(eachFile)\" -->\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.name)\">\n </div>\n <div class=\"col-lg-9 document_name\">\n {{eachFile?.name}}\n </div>\n <div class=\"document_delete\" (click)=\"deleteFile(currentFileIndex);$event.stopPropagation()\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/bin.svg\">\n </div>\n <!-- VD 20May24 - preview changes-->\n <div class=\"preview-icon\" (click)=\"viewFile(eachFile)\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/icons8-eye-24.png\">\n </div>\n </div>\n <div *ngIf=\"tableFile\" class=\"row\" style=\"cursor: pointer;\">\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.doc)\" style=\"margin-right: 10px;height: 40px;\">\n </div>\n <div class=\"col-lg-9 document_name\" style=\"overflow:hidden;height:35px;\">\n {{eachFile?.name}}\n </div>\n </div>\n </div>\n <div class=\"col-md-12 m-b-30 text-center\" *ngIf=\"copyOfInputAllFiles?.length === 0 && isShowNoFileIcon\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/ic_no_attachments.svg\" style=\"height: 140px;\">\n </div>\n</div>\n\n<div class=\"col-lg-6\" style=\"padding:0;margin-top: 10px;\">\n <label class=\"custom-file\" *ngIf=\"isDeleteFileButtonVisible && !tableFile\">\n <!-- HA 19DEC23 For translation -->\n <!-- VD 03May24 file upload fix-->\n <button [class]=\"error ? 'she-btn-primary-bordered Invalid' : 'she-btn-primary-bordered'\" style=\"width:275px;\" (click)=\"fileInput.click()\"\n [ngClass]=\"{ 'btn-disabled': copyOfInputAllFiles?.length >= copyOfFileUploadingLimit }\" [disabled]=\"copyOfInputAllFiles?.length >= copyOfFileUploadingLimit\"\n >Choose Files</button>\n\n <input #fileInput type=\"file\" name=\"fileUpload\" multiple=\"multiple\" accept=\"*\" style=\"display:none;\"\n (change) = uploadMultipleFiles($event)\n />\n </label>\n<!-- HA 19DEC23 For translation -->\n <label *ngIf=\"tableFile\" class=\"btn btn-primary\">\n {{ 'uploadFile' | i18n:i18nService.currentLanguage }}\n <input id=\"file\" style=\"display: none;\" type=\"file\" class=\"form-control\" (change)= uploadMultipleFiles($event) />\n </label>\n <!-- <span class=\"error-msg\" *ngIf=\"(formControl?.documents?.touched || formControl?.documents?.dirty) &&\n formControl?.documents?.errors?.required\">\n {{LM_POLICY_VALIDATION_MESSAGE.REQUIRED}}\n </span> -->\n <!-- RS 09DEC24 Changed keys--> \n</div>\n<div *ngIf=\"error\" class=\"error-msg\" style=\"margin-top: 10px;\">{{question?.errorMessage}}</div>\n<div class=\"nxt-file-overlay\" *ngIf=\"showFile\">\n <div class=\"nxt-file-map-modal\">\n <div class=\"nxt-file-model-content\">\n <div class=\"nxt-file-model-header\">\n <h4>{{fileName}}</h4>\n <button class=\"close-button\" (click)=\"close()\">X</button>\n </div>\n <ng-container *ngIf=\"isImage; else otherFile\">\n <img [src]=\"fileUrl\" class=\"img-fluid\" alt=\"File Preview\" />\n </ng-container>\n <ng-template #otherFile>\n <iframe [src]=\"fileUrl\" width=\"100%\" height=\"500px\"></iframe>\n </ng-template>\n </div>\n </div>\n</div>\n\n\n", styles: [".document-cnt{padding:10px;border:1px solid #ccc;border-radius:5px;background-color:#f9f9f9;margin-bottom:10px;cursor:pointer;transition:background-color .3s ease;position:relative}.document-cnt:hover{background-color:#e9ecef}.document_image img{width:30px;height:30px}.document_name{font-size:14px;font-weight:700;color:#333;line-height:30px}.label{margin-bottom:0}.document_delete img,.preview-icon img{width:20px;height:20px;cursor:pointer;transition:transform .3s ease}.document_delete img{filter:brightness(0) saturate(100%) invert(20%) sepia(97%) saturate(7481%) hue-rotate(357deg) brightness(98%) contrast(119%)}.document_delete img:hover,.preview-icon img:hover{transform:scale(1.1)}.document_delete,.preview-icon{position:absolute;top:50%;transform:translateY(-50%)}.document_delete{right:10px}.preview-icon{right:40px}@media (min-width: 992px){.document-cnt{margin-left:10px}}.btn-disabled{background:#e1e1e1;color:#fff;border:none;border-radius:5px;height:50px}.form-control[disabled]{border-radius:5px}.custom-file{color:#9a9a9a;font-size:14px;font-weight:400;display:inline-block;width:auto;margin-bottom:5px}.she-btn-primary-bordered{background:#fff;color:#48b7ff;border:1px solid #48B7FF;border-radius:5px;height:50px;outline:none!important}.btn-primary{background-color:#03a9f4!important;border:1px solid #03a9f4!important;color:#fff!important}.btn.btn-primary{border-radius:2px;padding:6px 14px;font-size:14px}.nxt-file-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;display:flex;justify-content:center;align-items:center;z-index:1000}.nxt-file-map-modal{background-color:#fff;border-radius:8px;overflow:hidden;width:80%;max-width:800px;max-height:90%;display:flex;flex-direction:column}.nxt-file-model-content{display:flex;flex-direction:column;height:100%}.nxt-file-model-header{display:flex;justify-content:space-between;align-items:center;padding:10px 20px;background-color:#f5f5f5;border-bottom:1px solid #ddd}.nxt-file-model-header h4{margin:0;font-size:1.25em}.close-button{background:none;border:none;font-size:1.5em;cursor:pointer}.img-fluid{max-width:100%;height:auto;flex:1}iframe{border:none}.Invalid{border:1px solid red!important}\n"] }]
259
+ args: [{ selector: 'app-file-upload', template: "<div class=\"row\" style=\"margin: 0;\">\n <div class=\"col-md-12\" *ngIf=\"!isDeleteFileButtonVisible\" style=\"text-align: left;\">\n <!-- HA 19DEC23 For translation -->\n <label class=\"she-label\">{{ 'attachment' | i18n:i18nService.currentLanguage }}</label>\n </div>\n\n <div class=\"col-lg-5 document-cnt m-t-10 m-b-10\"\n *ngFor=\"let eachFile of copyOfInputAllFiles; let currentFileIndex = index\">\n <div *ngIf=\"!tableFile\"> <!-- (click)=\"viewFile(eachFile)\" -->\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.name)\">\n </div>\n <div class=\"col-lg-9 document_name\">\n {{eachFile?.name}}\n </div>\n <div class=\"document_delete\" (click)=\"deleteFile(currentFileIndex);$event.stopPropagation()\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/bin.svg\">\n </div>\n <!-- VD 20May24 - preview changes-->\n <div class=\"preview-icon\" (click)=\"viewFile(eachFile)\" *ngIf=\"isDeleteFileButtonVisible\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/icons8-eye-24.png\">\n </div>\n </div>\n <div *ngIf=\"tableFile\" class=\"row\" style=\"cursor: pointer;\">\n <div class=\"col-lg-3 document_image\">\n <img [src]=\"getDocIcon(eachFile?.doc)\" style=\"margin-right: 10px;height: 40px;\">\n </div>\n <div class=\"col-lg-9 document_name\" style=\"overflow:hidden;height:35px;\">\n {{eachFile?.name}}\n </div>\n </div>\n </div>\n <div class=\"col-md-12 m-b-30 text-center\" *ngIf=\"copyOfInputAllFiles?.length === 0 && isShowNoFileIcon\">\n <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/ic_no_attachments.svg\" style=\"height: 140px;\">\n </div>\n</div>\n\n<div class=\"col-lg-6\" style=\"padding:0;margin-top: 10px;\">\n <label class=\"custom-file\" *ngIf=\"isDeleteFileButtonVisible && !tableFile\">\n <!-- HA 19DEC23 For translation -->\n <!-- VD 03May24 file upload fix-->\n <!-- RS 09JAN2025 -->\n <button [class]=\"error ? 'she-btn-primary-bordered Invalid' : 'she-btn-primary-bordered'\" style=\"width:275px;\" (click)=\"fileInput.click()\"[ngClass]=\"{ 'btn-disabled': copyOfInputAllFiles?.length >= maxFiles }\" [disabled]=\"copyOfInputAllFiles?.length >= maxFiles\">Choose Files</button>\n <input #fileInput type=\"file\" name=\"fileUpload\" multiple=\"multiple\" [accept]=\"allowedFileTypes.join(',')\"\n style=\"display:none;\"\n (change) = uploadMultipleFiles($event)\n />\n </label>\n\n <!-- RS 09JAN2025 -->\n <!-- Error messages section -->\n <div class=\"file-constraints\" *ngIf=\"validationErrors.length > 0\">\n <small class=\"text-danger\">\n <div *ngFor=\"let error of validationErrors\">{{ error }}</div>\n </small>\n </div>\n\n<!-- HA 19DEC23 For translation -->\n <label *ngIf=\"tableFile\" class=\"btn btn-primary\">\n {{ 'uploadFile' | i18n:i18nService.currentLanguage }}\n <input id=\"file\" style=\"display: none;\" type=\"file\" class=\"form-control\" (change)= uploadMultipleFiles($event) />\n </label>\n <!-- <span class=\"error-msg\" *ngIf=\"(formControl?.documents?.touched || formControl?.documents?.dirty) &&\n formControl?.documents?.errors?.required\">\n {{LM_POLICY_VALIDATION_MESSAGE.REQUIRED}}\n </span> -->\n <!-- RS 09DEC24 Changed keys--> \n</div>\n<div *ngIf=\"error\" class=\"error-msg\" style=\"margin-top: 10px;\">{{question?.errorMessage}}</div>\n<div class=\"nxt-file-overlay\" *ngIf=\"showFile\">\n <div class=\"nxt-file-map-modal\">\n <div class=\"nxt-file-model-content\">\n <div class=\"nxt-file-model-header\">\n <h4>{{fileName}}</h4>\n <button class=\"close-button\" (click)=\"close()\">X</button>\n </div>\n <ng-container *ngIf=\"isImage; else otherFile\">\n <img [src]=\"fileUrl\" class=\"img-fluid\" alt=\"File Preview\" />\n </ng-container>\n <ng-template #otherFile>\n <iframe [src]=\"fileUrl\" width=\"100%\" height=\"500px\"></iframe>\n </ng-template>\n </div>\n </div>\n</div>\n\n\n", styles: [".document-cnt{padding:10px;border:1px solid #ccc;border-radius:5px;background-color:#f9f9f9;margin-bottom:10px;cursor:pointer;transition:background-color .3s ease;position:relative}.document-cnt:hover{background-color:#e9ecef}.document_image img{width:30px;height:30px}.document_name{font-size:14px;font-weight:700;color:#333;line-height:30px}.label{margin-bottom:0}.document_delete img,.preview-icon img{width:20px;height:20px;cursor:pointer;transition:transform .3s ease}.document_delete img{filter:brightness(0) saturate(100%) invert(20%) sepia(97%) saturate(7481%) hue-rotate(357deg) brightness(98%) contrast(119%)}.document_delete img:hover,.preview-icon img:hover{transform:scale(1.1)}.document_delete,.preview-icon{position:absolute;top:50%;transform:translateY(-50%)}.document_delete{right:10px}.preview-icon{right:40px}@media (min-width: 992px){.document-cnt{margin-left:10px}}.btn-disabled{background:#e1e1e1;color:#fff;border:none;border-radius:5px;height:50px}.form-control[disabled]{border-radius:5px}.custom-file{color:#9a9a9a;font-size:14px;font-weight:400;display:inline-block;width:auto;margin-bottom:5px}.she-btn-primary-bordered{background:#fff;color:#48b7ff;border:1px solid #48B7FF;border-radius:5px;height:50px;outline:none!important}.btn-primary{background-color:#03a9f4!important;border:1px solid #03a9f4!important;color:#fff!important}.btn.btn-primary{border-radius:2px;padding:6px 14px;font-size:14px}.nxt-file-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;display:flex;justify-content:center;align-items:center;z-index:1000}.nxt-file-map-modal{background-color:#fff;border-radius:8px;overflow:hidden;width:80%;max-width:800px;max-height:90%;display:flex;flex-direction:column}.nxt-file-model-content{display:flex;flex-direction:column;height:100%}.nxt-file-model-header{display:flex;justify-content:space-between;align-items:center;padding:10px 20px;background-color:#f5f5f5;border-bottom:1px solid #ddd}.nxt-file-model-header h4{margin:0;font-size:1.25em}.close-button{background:none;border:none;font-size:1.5em;cursor:pointer}.img-fluid{max-width:100%;height:auto;flex:1}iframe{border:none}.Invalid{border:1px solid red!important}\n"] }]
214
260
  }], ctorParameters: () => [{ type: i1.SharedService }, { type: i2.DataService }, { type: i3.DomSanitizer }, { type: i4.I18nService }], propDecorators: { selectedFileData: [{
215
261
  type: Output
216
262
  }], deletedFileData: [{
@@ -230,4 +276,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
230
276
  }], error: [{
231
277
  type: Input
232
278
  }] } });
233
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/file-upload/file-upload.component.ts","../../../../../../projects/nxt-app/src/lib/components/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit,Output,EventEmitter, Input ,ElementRef, OnChanges,SimpleChanges} from '@angular/core';\n// HA 19DEC23 imported translation service\nimport { I18nService } from '../../i18n.service';\nimport { SharedService } from '../../services/shared.service';\nimport { DataService } from '../../services/data.service';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\n@Component({\n  selector: 'app-file-upload',\n  templateUrl: './file-upload.component.html',\n  styleUrls: ['./file-upload.component.css']\n})\nexport class FileUploadComponent implements OnInit,OnChanges {\n  @Output() selectedFileData = new EventEmitter<any[]>();\n  @Output() deletedFileData = new EventEmitter<any>();\n  @Input() allFiles;\n  @Input() limitFileUploading;\n  @Input() isDeleteFileButtonVisible;\n  @Input() isShowNoFileIcon;\n  @Input() tableFile;  // file Upload from table\n  @Input() question;\n  @Input() error:any;\n\n  public selectedFileNameArray: string[] = [];\n  public copyOfInputAllFiles;\n  public copyOfFileUploadingLimit;\n  public fileUrl;\n  public showFile = false;\n  public isImage = false;\n  public fileName = '';\n  currentFile: any;\n  constructor(private SharedService: SharedService,  private dataService: DataService,private sanitizer: DomSanitizer,\n    public i18nService: I18nService\n  ) { }\n\n  ngOnInit() { }\n  // VD 03May24 file upload fix\n  // onFileUpload(event: any) {\n  //   const fileUploaderElement: HTMLElement = document.getElementById(\n  //     'fileUpload'\n  //   ) as HTMLElement;\n  //   fileUploaderElement.click();\n  // }\n\n  ngOnChanges(simpleChanges: SimpleChanges) {\n    console.log('simple changes', simpleChanges);\n    if (simpleChanges.allFiles) {  //SKS5NOV25\n      this.copyOfInputAllFiles = simpleChanges.allFiles.currentValue || [];\n    }\n\n    if (simpleChanges.limitFileUploading) {\n      this.copyOfFileUploadingLimit = simpleChanges.limitFileUploading?.currentValue;\n    }\n\n  }\n\n  uploadMultipleFiles(event: any) {\n    console.log('File uploader initiated');\n    let inputFiles: any[] = this.copyOfInputAllFiles ? this.copyOfInputAllFiles : [];\n    const selectedFileData = [];\n    const uploadedFiles = event.target.files;\n    if (uploadedFiles.length + inputFiles.length <= this.copyOfFileUploadingLimit) {\n      const readFilesPromises = [];\n  \n      for (const eachUploadedFile of uploadedFiles) {\n        const file: File = eachUploadedFile;\n        const format = file.name.split('.')[1];\n        \n        // Create a promise for each file read\n        const fileReadPromise = new Promise<void>((resolve, reject) => {\n          const reader = new FileReader();\n          reader.readAsDataURL(file);\n  \n          reader.onload = () => {\n            const fileData = {\n              doc: reader.result,\n              name: file.name,\n              type: file.type,\n              format,\n              id: null,\n            };\n  \n            selectedFileData.push(fileData);\n            inputFiles.push(fileData);\n            resolve();\n          };\n  \n          reader.onerror = () => {\n            reject();\n          };\n        });\n  \n        readFilesPromises.push(fileReadPromise);\n      }\n  \n      // Wait for all promises to resolve\n      Promise.all(readFilesPromises).then(() => {\n        this.copyOfInputAllFiles = inputFiles;\n        console.log('uploadedFiles', inputFiles);\n        this.selectedFileData.emit(inputFiles);\n      }).catch(() => {\n        console.error('Error reading files');\n      });\n    } else {\n      console.warn('You can upload max 5 files');\n      // this.toastr.warning(TOASTER_MESSAGES.MAX_FIVE_FILES, TOASTER_MESSAGES.WARNING_TITLE);\n    }\n  \n    event.target.value = '';\n  }\n  \n\n  // uploadMultipleFiles(event: any) {\n  //   console.log('File uploader initiated');\n  //   let inputFiles: any[] = this.copyOfInputAllFiles ? this.copyOfInputAllFiles : [];\n  //   const selectedFileData = [];\n  //   const uploadedFiles = event.target.files;\n  //   if (uploadedFiles.length + inputFiles.length <= this.copyOfFileUploadingLimit) {\n  //     for (const eachUploadedFile of uploadedFiles) {\n  //       const reader = new FileReader();\n  //       const file: File = eachUploadedFile;\n  //       const format = file.name.split('.')[1];\n  //       reader.readAsDataURL(file);\n  //       reader.onload = () => {\n  //         selectedFileData.push({\n  //           doc: reader.result,\n  //           name: file.name,\n  //           type: file.type,\n  //           format,\n  //           id: null,\n  //         });\n  //         inputFiles.push({\n  //           doc: reader.result,\n  //           name: file.name,\n  //           type: file.type,\n  //           format,\n  //           id: null,\n  //         });\n  //       };\n  //     }\n  //   } else {\n  //     console.warn('You can upload max 5 files');\n  //     //this.toastr.warning(TOASTER_MESSAGES.MAX_FIVE_FILES, TOASTER_MESSAGES.WARNING_TITLE);\n  //   }\n  //   event.target.value = '';\n  //   this.copyOfInputAllFiles = inputFiles;\n  //   console.log('uploadedFiles',inputFiles);\n  //   this.selectedFileData.emit(inputFiles);\n  // }\n\n\n  deleteFile(currentFileIndex) {\n        const deletedFileName = this.selectedFileNameArray.splice(currentFileIndex, 1);\n        const deletedFile = this.copyOfInputAllFiles.splice(currentFileIndex, 1);\n        console.log('emit', deletedFile[0]);\n        this.deletedFileData.emit(deletedFile[0]);\n        this.selectedFileData.emit(this.copyOfInputAllFiles);\n  }\n\n\n// VD 20May24 - preview changes\n// RS 09DEC24 Changed keys\n  viewFile(currentFile) {\n    this.currentFile = currentFile;\n    this.showFile = true;\n    console.log(currentFile);\n    if(this.question.subText != undefined) {\n      let fileMeta = JSON.parse(this.question.subText);\n      let endpoint = fileMeta.endpoint;\n      let queryParameter = fileMeta.queryParameter;\n        let file = currentFile;\n          if(endpoint){\n            const fullEndPoint = endpoint + '?'+ queryParameter + '=' + file[queryParameter];\n            if(fullEndPoint){\n              this.dataService.apiResponse(fullEndPoint).subscribe((apiResponse) => {\n                if(apiResponse){\n                  this.handleFileContent(apiResponse);\n                }\n              })\n         }\n       }\n    }else {\n        this.viewLocalFile()\n    }\n  }\n\n// // VD 03Aug24 process local preview \n  viewLocalFile(){\n    if(this.currentFile){\n      this.setFileUrl(this.currentFile?.doc);\n      this.fileName = this.currentFile.name;\n      this.isImage = this.currentFile.type.startsWith('image/');\n    }\n\n  }\n\n \n  handleFileContent(fileResponse: any) {\n    const byteArray = new Uint8Array(fileResponse.content.data);\n    const blob = new Blob([byteArray], { type: fileResponse.type });\n    const url = window.URL.createObjectURL(blob);\n    this.setFileUrl(url);\n    this.isImage = false;\n    if(fileResponse.name){\n      this.fileName = fileResponse.name;\n      const imageExtensions = ['jpg', 'jpeg', 'png', 'gif', 'bmp', 'webp'];\n      const fileExtension = fileResponse.name.split('.').pop()?.toLowerCase();\n      this.isImage = imageExtensions.includes(fileExtension || '');\n    }\n  }\n\n  setFileUrl(url: string) {\n    this.fileUrl = this.sanitizer.bypassSecurityTrustResourceUrl(url);\n  }\n\n  getDocIcon(docName) {\n    const ext = docName.split('.').pop(-1);\n    return this.SharedService.docIcon(ext);\n  }\n\n  close(){\n    this.showFile = false;\n    this.fileName = '';\n    this.fileUrl = '';\n    this.isImage = false;\n    if (this.fileUrl) {\n      URL.revokeObjectURL(this.fileUrl);\n    }\n  }\n\n}\n","<div class=\"row\" style=\"margin: 0;\">\n  <div class=\"col-md-12\" *ngIf=\"!isDeleteFileButtonVisible\" style=\"text-align: left;\">\n    <!-- HA 19DEC23 For translation -->\n    <label class=\"she-label\">{{ 'attachment' | i18n:i18nService.currentLanguage }}</label>\n  </div>\n  <div class=\"col-lg-5 document-cnt m-t-10 m-b-10\"\n    *ngFor=\"let eachFile of copyOfInputAllFiles; let currentFileIndex = index\">\n     <div *ngIf=\"!tableFile\"> <!-- (click)=\"viewFile(eachFile)\"  -->\n      <div class=\"col-lg-3 document_image\">\n        <img [src]=\"getDocIcon(eachFile?.name)\">\n    </div>\n      <div class=\"col-lg-9 document_name\">\n        {{eachFile?.name}}\n      </div>\n      <div class=\"document_delete\" (click)=\"deleteFile(currentFileIndex);$event.stopPropagation()\" *ngIf=\"isDeleteFileButtonVisible\">\n        <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/bin.svg\">\n      </div>\n      <!-- VD 20May24 - preview changes-->\n      <div class=\"preview-icon\" (click)=\"viewFile(eachFile)\" *ngIf=\"isDeleteFileButtonVisible\">\n        <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/icons8-eye-24.png\">\n      </div>\n    </div>\n    <div *ngIf=\"tableFile\" class=\"row\" style=\"cursor: pointer;\">\n      <div class=\"col-lg-3 document_image\">\n          <img [src]=\"getDocIcon(eachFile?.doc)\" style=\"margin-right: 10px;height: 40px;\">\n      </div>\n      <div class=\"col-lg-9 document_name\" style=\"overflow:hidden;height:35px;\">\n          {{eachFile?.name}}\n      </div>\n  </div>\n  </div>\n  <div class=\"col-md-12 m-b-30 text-center\" *ngIf=\"copyOfInputAllFiles?.length === 0 && isShowNoFileIcon\">\n    <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/ic_no_attachments.svg\" style=\"height: 140px;\">\n  </div>\n</div>\n\n<div class=\"col-lg-6\" style=\"padding:0;margin-top: 10px;\">\n  <label class=\"custom-file\" *ngIf=\"isDeleteFileButtonVisible && !tableFile\">\n    <!-- HA 19DEC23 For translation -->\n    <!-- VD 03May24 file upload fix-->\n    <button [class]=\"error ? 'she-btn-primary-bordered Invalid' : 'she-btn-primary-bordered'\" style=\"width:275px;\" (click)=\"fileInput.click()\"\n    [ngClass]=\"{ 'btn-disabled': copyOfInputAllFiles?.length >= copyOfFileUploadingLimit }\" [disabled]=\"copyOfInputAllFiles?.length >= copyOfFileUploadingLimit\"\n    >Choose Files</button>\n\n    <input #fileInput type=\"file\"  name=\"fileUpload\" multiple=\"multiple\" accept=\"*\" style=\"display:none;\"\n    (change) = uploadMultipleFiles($event)\n    />\n  </label>\n<!-- HA 19DEC23 For translation -->\n  <label *ngIf=\"tableFile\" class=\"btn btn-primary\">\n    {{ 'uploadFile' | i18n:i18nService.currentLanguage }}\n    <input id=\"file\" style=\"display: none;\" type=\"file\" class=\"form-control\" (change)= uploadMultipleFiles($event) />\n  </label>\n  <!-- <span class=\"error-msg\" *ngIf=\"(formControl?.documents?.touched ||        formControl?.documents?.dirty) &&\n  formControl?.documents?.errors?.required\">\n    {{LM_POLICY_VALIDATION_MESSAGE.REQUIRED}}\n  </span> -->\n  <!-- RS 09DEC24 Changed keys--> \n</div>\n<div *ngIf=\"error\" class=\"error-msg\" style=\"margin-top: 10px;\">{{question?.errorMessage}}</div>\n<div class=\"nxt-file-overlay\" *ngIf=\"showFile\">\n  <div class=\"nxt-file-map-modal\">\n    <div class=\"nxt-file-model-content\">\n      <div class=\"nxt-file-model-header\">\n        <h4>{{fileName}}</h4>\n        <button class=\"close-button\" (click)=\"close()\">X</button>\n      </div>\n      <ng-container *ngIf=\"isImage; else otherFile\">\n        <img [src]=\"fileUrl\" class=\"img-fluid\" alt=\"File Preview\" />\n      </ng-container>\n      <ng-template #otherFile>\n        <iframe [src]=\"fileUrl\" width=\"100%\" height=\"500px\"></iframe>\n      </ng-template>\n    </div>\n  </div>\n</div>\n\n\n"]}
279
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/file-upload/file-upload.component.ts","../../../../../../projects/nxt-app/src/lib/components/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit,Output,EventEmitter, Input ,ElementRef, OnChanges,SimpleChanges} from '@angular/core';\n// HA 19DEC23 imported translation service\nimport { I18nService } from '../../i18n.service';\nimport { SharedService } from '../../services/shared.service';\nimport { DataService } from '../../services/data.service';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\n@Component({\n  selector: 'app-file-upload',\n  templateUrl: './file-upload.component.html',\n  styleUrls: ['./file-upload.component.css']\n})\nexport class FileUploadComponent implements OnInit,OnChanges {\n  @Output() selectedFileData = new EventEmitter<any[]>();\n  @Output() deletedFileData = new EventEmitter<any>();\n  @Input() allFiles;\n  @Input() limitFileUploading;\n  @Input() isDeleteFileButtonVisible;\n  @Input() isShowNoFileIcon;\n  @Input() tableFile;  // file Upload from table\n  @Input() question;\n  @Input() error:any;\n//RS 09JAN2025\n  public allowedFileTypes: string[] = [];\n  public maxFileSize: number = 0;  // in bytes(Binary)\n  public maxFiles: number = 4;     // default limit\n  public validationErrors: string[] = [];\n \n\n  public selectedFileNameArray: string[] = [];\n  public copyOfInputAllFiles;\n  public copyOfFileUploadingLimit;\n  public fileUrl;\n  public showFile = false;\n  public isImage = false;\n  public fileName = '';\n  currentFile: any;\n  constructor(private SharedService: SharedService,  private dataService: DataService,private sanitizer: DomSanitizer,\n    public i18nService: I18nService\n  ) { }\n  //RS 09JAN2025\n  ngOnInit() {\n    this.initializeFileConfigs();\n   }\n  // VD 03May24 file upload fix\n  // onFileUpload(event: any) {\n  //   const fileUploaderElement: HTMLElement = document.getElementById(\n  //     'fileUpload'\n  //   ) as HTMLElement;\n  //   fileUploaderElement.click();\n  // }\n\n  ngOnChanges(simpleChanges: SimpleChanges) {\n    console.log('simple changes', simpleChanges);\n    if (simpleChanges.allFiles) {  //SKS5NOV25\n      this.copyOfInputAllFiles = simpleChanges.allFiles.currentValue || [];\n    }\n\n    if (simpleChanges.limitFileUploading) {\n      this.copyOfFileUploadingLimit = simpleChanges.limitFileUploading?.currentValue;\n    }\n\n  }\n\n  //RS 09JAN2025 Initializes file upload settings (allowed types, max size, and max files) by parsing metadata from fieldsMeta\n   initializeFileConfigs() {\n    if (this.question?.fieldsMeta) {\n      try {\n        const metaData = JSON.parse(this.question.fieldsMeta);\n        const fileConfig = metaData[0];  // Get first config object\n        if (fileConfig) {\n          this.allowedFileTypes = fileConfig.allowedTypes || [];\n          this.maxFileSize = fileConfig.maxFileSize || 0;\n          this.maxFiles = fileConfig.maxFiles || 5;\n        }\n      } catch (error) {\n        console.error('Error parsing fieldsMeta:', error);\n      }\n    }\n  }\n\n  uploadMultipleFiles(event: any) {\n    console.log('File uploader initiated');\n    this.validationErrors = []; // RS 17JAN2025 Clear previous errors\n    let inputFiles: any[] = this.copyOfInputAllFiles ? this.copyOfInputAllFiles : [];\n    const selectedFileData = [];\n    const uploadedFiles = event.target.files;\n\n    // RS 09JAN2025 Check file count limit\n    if (uploadedFiles.length + inputFiles.length > this.maxFiles) {\n      this.validationErrors.push(`Maximum ${this.maxFiles} files allowed`);\n      event.target.value = '';\n      return;\n    }\n\n    const readFilesPromises = [];\n\n    for (const eachUploadedFile of uploadedFiles) {\n      const file: File = eachUploadedFile;\n      \n      //RS 09JAN2025 Check file type\n      const fileExt = '.' + file.name.split('.').pop().toLowerCase();\n      if (this.allowedFileTypes.length > 0 && !this.allowedFileTypes.includes(fileExt)) {\n        this.validationErrors.push(`File type ${fileExt} not allowed. Allowed types: ${this.allowedFileTypes.join(', ')}`);\n        continue;\n      }\n\n      //RS 09JAN2025 Check file size\n      if (this.maxFileSize > 0 && file.size > this.maxFileSize) {\n        this.validationErrors.push(`File \"${file.name}\" exceeds size limit of ${this.maxFileSize / 1048576}MB`);\n        continue;\n      }\n\n      if (!this.validationErrors.length) {\n        const format = fileExt.substring(1);\n        const fileReadPromise = new Promise<void>((resolve, reject) => {\n          const reader = new FileReader();\n          reader.readAsDataURL(file);\n  \n          reader.onload = () => {\n            const fileData = {\n              doc: reader.result,\n              name: file.name,\n              type: file.type,\n              format,\n              size: file.size, // RS 17JAN2025 \n              id: null,\n            };\n  \n            selectedFileData.push(fileData);\n            inputFiles.push(fileData);\n            resolve();\n          };\n  \n          reader.onerror = () => {\n            reject();\n            this.validationErrors.push(`Error reading file \"${file.name}\"`);// RS 17JAN2025 \n          };\n        });\n        readFilesPromises.push(fileReadPromise);     \n    }\n  }\n    if (readFilesPromises.length > 0) {\n      // Wait for all promises to resolve\n      Promise.all(readFilesPromises).then(() => {\n        this.copyOfInputAllFiles = inputFiles;\n        console.log('uploadedFiles', inputFiles);\n        this.selectedFileData.emit(inputFiles);\n      }).catch(() => {\n        console.error('Error reading files');\n      });\n    } else {\n      console.warn('You can upload max 5 files');\n      // this.toastr.warning(TOASTER_MESSAGES.MAX_FIVE_FILES, TOASTER_MESSAGES.WARNING_TITLE);\n    }\n  \n    event.target.value = '';\n  }\n  \n\n  // uploadMultipleFiles(event: any) {\n  //   console.log('File uploader initiated');\n  //   let inputFiles: any[] = this.copyOfInputAllFiles ? this.copyOfInputAllFiles : [];\n  //   const selectedFileData = [];\n  //   const uploadedFiles = event.target.files;\n  //   if (uploadedFiles.length + inputFiles.length <= this.copyOfFileUploadingLimit) {\n  //     for (const eachUploadedFile of uploadedFiles) {\n  //       const reader = new FileReader();\n  //       const file: File = eachUploadedFile;\n  //       const format = file.name.split('.')[1];\n  //       reader.readAsDataURL(file);\n  //       reader.onload = () => {\n  //         selectedFileData.push({\n  //           doc: reader.result,\n  //           name: file.name,\n  //           type: file.type,\n  //           format,\n  //           id: null,\n  //         });\n  //         inputFiles.push({\n  //           doc: reader.result,\n  //           name: file.name,\n  //           type: file.type,\n  //           format,\n  //           id: null,\n  //         });\n  //       };\n  //     }\n  //   } else {\n  //     console.warn('You can upload max 5 files');\n  //     //this.toastr.warning(TOASTER_MESSAGES.MAX_FIVE_FILES, TOASTER_MESSAGES.WARNING_TITLE);\n  //   }\n  //   event.target.value = '';\n  //   this.copyOfInputAllFiles = inputFiles;\n  //   console.log('uploadedFiles',inputFiles);\n  //   this.selectedFileData.emit(inputFiles);\n  // }\n\n\n  deleteFile(currentFileIndex) {\n        const deletedFileName = this.selectedFileNameArray.splice(currentFileIndex, 1);\n        const deletedFile = this.copyOfInputAllFiles.splice(currentFileIndex, 1);\n        console.log('emit', deletedFile[0]);\n        this.deletedFileData.emit(deletedFile[0]);\n        this.selectedFileData.emit(this.copyOfInputAllFiles);\n  }\n\n\n// VD 20May24 - preview changes\n// RS 09DEC24 Changed keys\n  viewFile(currentFile) {\n    this.currentFile = currentFile;\n    this.showFile = true;\n    console.log(currentFile);\n    if(this.question.subText != undefined) {\n      let fileMeta = JSON.parse(this.question.subText);\n      let endpoint = fileMeta.endpoint;\n      let queryParameter = fileMeta.queryParameter;\n        let file = currentFile;\n          if(endpoint){\n            const fullEndPoint = endpoint + '?'+ queryParameter + '=' + file[queryParameter];\n            if(fullEndPoint){\n              this.dataService.apiResponse(fullEndPoint).subscribe((apiResponse) => {\n                if(apiResponse){\n                  this.handleFileContent(apiResponse);\n                }\n              })\n         }\n       }\n    }else {\n        this.viewLocalFile()\n    }\n  }\n\n// // VD 03Aug24 process local preview \n  viewLocalFile(){\n    if(this.currentFile){\n      this.setFileUrl(this.currentFile?.doc);\n      this.fileName = this.currentFile.name;\n      this.isImage = this.currentFile.type.startsWith('image/');\n    }\n\n  }\n\n \n  handleFileContent(fileResponse: any) {\n    const byteArray = new Uint8Array(fileResponse.content.data);\n    const blob = new Blob([byteArray], { type: fileResponse.type });\n    const url = window.URL.createObjectURL(blob);\n    this.setFileUrl(url);\n    this.isImage = false;\n    if(fileResponse.name){\n      this.fileName = fileResponse.name;\n      const imageExtensions = ['jpg', 'jpeg', 'png', 'gif', 'bmp', 'webp'];\n      const fileExtension = fileResponse.name.split('.').pop()?.toLowerCase();\n      this.isImage = imageExtensions.includes(fileExtension || '');\n    }\n  }\n\n  setFileUrl(url: string) {\n    this.fileUrl = this.sanitizer.bypassSecurityTrustResourceUrl(url);\n  }\n\n  getDocIcon(docName) {\n    const ext = docName.split('.').pop(-1);\n    return this.SharedService.docIcon(ext);\n  }\n\n  close(){\n    this.showFile = false;\n    this.fileName = '';\n    this.fileUrl = '';\n    this.isImage = false;\n    if (this.fileUrl) {\n      URL.revokeObjectURL(this.fileUrl);\n    }\n  }\n\n}\n","<div class=\"row\" style=\"margin: 0;\">\n  <div class=\"col-md-12\" *ngIf=\"!isDeleteFileButtonVisible\" style=\"text-align: left;\">\n    <!-- HA 19DEC23 For translation -->\n    <label class=\"she-label\">{{ 'attachment' | i18n:i18nService.currentLanguage }}</label>\n  </div>\n\n  <div class=\"col-lg-5 document-cnt m-t-10 m-b-10\"\n    *ngFor=\"let eachFile of copyOfInputAllFiles; let currentFileIndex = index\">\n     <div *ngIf=\"!tableFile\"> <!-- (click)=\"viewFile(eachFile)\"  -->\n      <div class=\"col-lg-3 document_image\">\n        <img [src]=\"getDocIcon(eachFile?.name)\">\n    </div>\n      <div class=\"col-lg-9 document_name\">\n        {{eachFile?.name}}\n      </div>\n      <div class=\"document_delete\" (click)=\"deleteFile(currentFileIndex);$event.stopPropagation()\" *ngIf=\"isDeleteFileButtonVisible\">\n        <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/bin.svg\">\n      </div>\n      <!-- VD 20May24 - preview changes-->\n      <div class=\"preview-icon\" (click)=\"viewFile(eachFile)\" *ngIf=\"isDeleteFileButtonVisible\">\n        <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/icons8-eye-24.png\">\n      </div>\n    </div>\n    <div *ngIf=\"tableFile\" class=\"row\" style=\"cursor: pointer;\">\n      <div class=\"col-lg-3 document_image\">\n          <img [src]=\"getDocIcon(eachFile?.doc)\" style=\"margin-right: 10px;height: 40px;\">\n      </div>\n      <div class=\"col-lg-9 document_name\" style=\"overflow:hidden;height:35px;\">\n          {{eachFile?.name}}\n      </div>\n  </div>\n  </div>\n  <div class=\"col-md-12 m-b-30 text-center\" *ngIf=\"copyOfInputAllFiles?.length === 0 && isShowNoFileIcon\">\n    <img src=\"https://rnxt.s3.amazonaws.com/Icons/fileTypeIcons/ic_no_attachments.svg\" style=\"height: 140px;\">\n  </div>\n</div>\n\n<div class=\"col-lg-6\" style=\"padding:0;margin-top: 10px;\">\n  <label class=\"custom-file\" *ngIf=\"isDeleteFileButtonVisible && !tableFile\">\n    <!-- HA 19DEC23 For translation -->\n    <!-- VD 03May24 file upload fix-->\n    <!-- RS 09JAN2025 -->\n    <button [class]=\"error ? 'she-btn-primary-bordered Invalid' : 'she-btn-primary-bordered'\" style=\"width:275px;\" (click)=\"fileInput.click()\"[ngClass]=\"{ 'btn-disabled': copyOfInputAllFiles?.length >= maxFiles }\" [disabled]=\"copyOfInputAllFiles?.length >= maxFiles\">Choose Files</button>\n    <input #fileInput type=\"file\"  name=\"fileUpload\" multiple=\"multiple\"  [accept]=\"allowedFileTypes.join(',')\"\n    style=\"display:none;\"\n    (change) = uploadMultipleFiles($event)\n    />\n  </label>\n\n  <!-- RS 09JAN2025 -->\n  <!-- Error messages section -->\n  <div class=\"file-constraints\" *ngIf=\"validationErrors.length > 0\">\n    <small class=\"text-danger\">\n      <div *ngFor=\"let error of validationErrors\">{{ error }}</div>\n    </small>\n  </div>\n\n<!-- HA 19DEC23 For translation -->\n  <label *ngIf=\"tableFile\" class=\"btn btn-primary\">\n    {{ 'uploadFile' | i18n:i18nService.currentLanguage }}\n    <input id=\"file\" style=\"display: none;\" type=\"file\" class=\"form-control\" (change)= uploadMultipleFiles($event) />\n  </label>\n  <!-- <span class=\"error-msg\" *ngIf=\"(formControl?.documents?.touched ||        formControl?.documents?.dirty) &&\n  formControl?.documents?.errors?.required\">\n    {{LM_POLICY_VALIDATION_MESSAGE.REQUIRED}}\n  </span> -->\n  <!-- RS 09DEC24 Changed keys--> \n</div>\n<div *ngIf=\"error\" class=\"error-msg\" style=\"margin-top: 10px;\">{{question?.errorMessage}}</div>\n<div class=\"nxt-file-overlay\" *ngIf=\"showFile\">\n  <div class=\"nxt-file-map-modal\">\n    <div class=\"nxt-file-model-content\">\n      <div class=\"nxt-file-model-header\">\n        <h4>{{fileName}}</h4>\n        <button class=\"close-button\" (click)=\"close()\">X</button>\n      </div>\n      <ng-container *ngIf=\"isImage; else otherFile\">\n        <img [src]=\"fileUrl\" class=\"img-fluid\" alt=\"File Preview\" />\n      </ng-container>\n      <ng-template #otherFile>\n        <iframe [src]=\"fileUrl\" width=\"100%\" height=\"500px\"></iframe>\n      </ng-template>\n    </div>\n  </div>\n</div>\n\n\n"]}