@rangertechnologies/ngnxt 2.1.106 → 2.1.108

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (143) hide show
  1. package/esm2022/lib/ar.i18n.mjs +29 -0
  2. package/esm2022/lib/components/button/nxt-button.component.mjs +130 -0
  3. package/esm2022/lib/components/custom-button/custom-button.component.mjs +39 -0
  4. package/esm2022/lib/components/custom-calendar/custom-calendar.component.mjs +366 -0
  5. package/esm2022/lib/components/custom-date/custom-date.component.mjs +47 -0
  6. package/esm2022/lib/components/custom-date-picker/custom-date-picker.component.mjs +48 -0
  7. package/esm2022/lib/components/custom-dropdown/custom-dropdown.component.mjs +219 -0
  8. package/esm2022/lib/components/custom-image/custom-image.component.mjs +34 -0
  9. package/esm2022/lib/components/custom-input/custom-input.component.mjs +93 -0
  10. package/esm2022/lib/components/custom-label/custom-label.component.mjs +21 -0
  11. package/esm2022/lib/components/custom-model/custom-model.component.mjs +48 -0
  12. package/esm2022/lib/components/custom-radio/custom-radio.component.mjs +115 -0
  13. package/esm2022/lib/components/custom-rich-text/custom-rich-text.component.mjs +186 -0
  14. package/esm2022/lib/components/custom-table/custom-table.component.mjs +347 -0
  15. package/esm2022/lib/components/custom-text-area/custom-text-area.component.mjs +65 -0
  16. package/esm2022/lib/components/custom-time/custom-time.component.mjs +61 -0
  17. package/esm2022/lib/components/datatable/datatable.component.mjs +801 -0
  18. package/esm2022/lib/components/dependent-table/dependent-table.component.mjs +42 -0
  19. package/esm2022/lib/components/dropdown-with-flag/dropdown-with-flag.component.mjs +39 -0
  20. package/esm2022/lib/components/file-upload/file-upload.component.mjs +279 -0
  21. package/esm2022/lib/components/file-view/file-view.component.mjs +46 -0
  22. package/esm2022/lib/components/loader/loader.component.mjs +23 -0
  23. package/esm2022/lib/components/pagination/pagination.component.mjs +100 -0
  24. package/esm2022/lib/components/pick-location/pick-location.component.mjs +216 -0
  25. package/esm2022/lib/components/search-box/search-box.component.mjs +202 -0
  26. package/esm2022/lib/components/table-appendix/table-appendix.component.mjs +123 -0
  27. package/esm2022/lib/directives/componenthost/componenthost.directive.mjs +19 -0
  28. package/esm2022/lib/en.i18n.mjs +29 -0
  29. package/esm2022/lib/i18n-config.service.mjs +4 -0
  30. package/esm2022/lib/i18n.component.mjs +47 -0
  31. package/esm2022/lib/i18n.module.mjs +38 -0
  32. package/esm2022/lib/i18n.pipe.mjs +25 -0
  33. package/esm2022/lib/i18n.service.mjs +56 -0
  34. package/esm2022/lib/interfaces/actionMeta.mjs +2 -0
  35. package/esm2022/lib/interfaces/apimeta.mjs +2 -0
  36. package/esm2022/lib/interfaces/dependencyMeta.mjs +2 -0
  37. package/esm2022/lib/model/bookletWrapper.mjs +9 -0
  38. package/esm2022/lib/model/changeWrapper.mjs +11 -0
  39. package/esm2022/lib/model/errorWrapper.mjs +6 -0
  40. package/esm2022/lib/model/tableWrapper.mjs +20 -0
  41. package/esm2022/lib/nxt-app.component.mjs +22 -0
  42. package/esm2022/lib/nxt-app.module.mjs +399 -0
  43. package/esm2022/lib/nxt-app.service.mjs +14 -0
  44. package/esm2022/lib/pages/booklet/booklet.component.mjs +541 -0
  45. package/esm2022/lib/pages/builder/element/element.component.mjs +196 -0
  46. package/esm2022/lib/pages/builder/form/form.component.mjs +34 -0
  47. package/esm2022/lib/pages/builder/menu/menu.component.mjs +28 -0
  48. package/esm2022/lib/pages/builder/properties/properties.component.mjs +494 -0
  49. package/esm2022/lib/pages/questionbook/questionbook.component.mjs +458 -0
  50. package/esm2022/lib/pages/questionnaire/questionnaire.component.mjs +2387 -0
  51. package/esm2022/lib/pages/summary-page/summary-page.component.mjs +76 -0
  52. package/esm2022/lib/pipe/date/date.pipe.mjs +28 -0
  53. package/esm2022/lib/pipe/editColumnCheck/edit-column-check.pipe.mjs +28 -0
  54. package/esm2022/lib/pipe/editColumnDropdown/edit-column-dropdown.pipe.mjs +20 -0
  55. package/esm2022/lib/pipe/editColumnType/edit-column-type.pipe.mjs +20 -0
  56. package/esm2022/lib/pipe/get-value.pipe.mjs +50 -0
  57. package/esm2022/lib/pipe/search-filter/search-filter.pipe.mjs +39 -0
  58. package/esm2022/lib/pipe/time/time.pipe.mjs +26 -0
  59. package/esm2022/lib/sample.mjs +3715 -0
  60. package/esm2022/lib/services/change.service.mjs +53 -0
  61. package/esm2022/lib/services/data.service.mjs +80 -0
  62. package/esm2022/lib/services/form-builder.service.mjs +173 -0
  63. package/esm2022/lib/services/salesforce.service.mjs +46 -0
  64. package/esm2022/lib/services/shared.service.mjs +100 -0
  65. package/esm2022/lib/services/storage.service.mjs +44 -0
  66. package/esm2022/lib/tam.i18n.mjs +29 -0
  67. package/esm2022/lib/wrapper.mjs +175 -0
  68. package/esm2022/public-api.mjs +22 -0
  69. package/esm2022/rangertechnologies-ngnxt.mjs +5 -0
  70. package/fesm2022/rangertechnologies-ngnxt.mjs +12900 -0
  71. package/fesm2022/rangertechnologies-ngnxt.mjs.map +1 -0
  72. package/index.d.ts +5 -0
  73. package/lib/ar.i18n.d.ts +24 -0
  74. package/lib/components/button/nxt-button.component.d.ts +38 -0
  75. package/lib/components/custom-button/custom-button.component.d.ts +16 -0
  76. package/lib/components/custom-calendar/custom-calendar.component.d.ts +65 -0
  77. package/lib/components/custom-date/custom-date.component.d.ts +18 -0
  78. package/lib/components/custom-date-picker/custom-date-picker.component.d.ts +19 -0
  79. package/lib/components/custom-dropdown/custom-dropdown.component.d.ts +39 -0
  80. package/lib/components/custom-image/custom-image.component.d.ts +13 -0
  81. package/lib/components/custom-input/custom-input.component.d.ts +31 -0
  82. package/lib/components/custom-label/custom-label.component.d.ts +10 -0
  83. package/lib/components/custom-model/custom-model.component.d.ts +19 -0
  84. package/lib/components/custom-radio/custom-radio.component.d.ts +33 -0
  85. package/lib/components/custom-rich-text/custom-rich-text.component.d.ts +29 -0
  86. package/lib/components/custom-table/custom-table.component.d.ts +65 -0
  87. package/lib/components/custom-text-area/custom-text-area.component.d.ts +22 -0
  88. package/lib/components/custom-time/custom-time.component.d.ts +18 -0
  89. package/lib/components/datatable/datatable.component.d.ts +126 -0
  90. package/lib/components/dependent-table/dependent-table.component.d.ts +15 -0
  91. package/lib/components/dropdown-with-flag/dropdown-with-flag.component.d.ts +16 -0
  92. package/lib/components/file-upload/file-upload.component.d.ts +47 -0
  93. package/lib/components/file-view/file-view.component.d.ts +16 -0
  94. package/lib/components/loader/loader.component.d.ts +11 -0
  95. package/lib/components/pagination/pagination.component.d.ts +36 -0
  96. package/lib/components/pick-location/pick-location.component.d.ts +44 -0
  97. package/lib/components/search-box/search-box.component.d.ts +43 -0
  98. package/lib/components/table-appendix/table-appendix.component.d.ts +30 -0
  99. package/lib/directives/componenthost/componenthost.directive.d.ts +8 -0
  100. package/lib/en.i18n.d.ts +24 -0
  101. package/lib/i18n-config.service.d.ts +2 -0
  102. package/lib/i18n.component.d.ts +11 -0
  103. package/lib/i18n.module.d.ts +9 -0
  104. package/lib/i18n.pipe.d.ts +10 -0
  105. package/lib/i18n.service.d.ts +14 -0
  106. package/lib/interfaces/actionMeta.d.ts +5 -0
  107. package/lib/interfaces/apimeta.d.ts +15 -0
  108. package/lib/interfaces/dependencyMeta.d.ts +8 -0
  109. package/lib/model/bookletWrapper.d.ts +5 -0
  110. package/lib/model/changeWrapper.d.ts +10 -0
  111. package/lib/model/errorWrapper.d.ts +5 -0
  112. package/lib/model/tableWrapper.d.ts +18 -0
  113. package/lib/nxt-app.component.d.ts +8 -0
  114. package/lib/nxt-app.module.d.ts +52 -0
  115. package/lib/nxt-app.service.d.ts +6 -0
  116. package/lib/pages/booklet/booklet.component.d.ts +68 -0
  117. package/lib/pages/builder/element/element.component.d.ts +33 -0
  118. package/lib/pages/builder/form/form.component.d.ts +11 -0
  119. package/lib/pages/builder/menu/menu.component.d.ts +10 -0
  120. package/lib/pages/builder/properties/properties.component.d.ts +575 -0
  121. package/lib/pages/questionbook/questionbook.component.d.ts +68 -0
  122. package/lib/pages/questionnaire/questionnaire.component.d.ts +220 -0
  123. package/lib/pages/summary-page/summary-page.component.d.ts +20 -0
  124. package/lib/pipe/date/date.pipe.d.ts +7 -0
  125. package/lib/pipe/editColumnCheck/edit-column-check.pipe.d.ts +7 -0
  126. package/lib/pipe/editColumnDropdown/edit-column-dropdown.pipe.d.ts +7 -0
  127. package/lib/pipe/editColumnType/edit-column-type.pipe.d.ts +7 -0
  128. package/lib/pipe/get-value.pipe.d.ts +7 -0
  129. package/lib/pipe/search-filter/search-filter.pipe.d.ts +8 -0
  130. package/lib/pipe/time/time.pipe.d.ts +7 -0
  131. package/lib/sample.d.ts +11 -0
  132. package/lib/services/change.service.d.ts +22 -0
  133. package/lib/services/data.service.d.ts +13 -0
  134. package/lib/services/form-builder.service.d.ts +76 -0
  135. package/lib/services/salesforce.service.d.ts +11 -0
  136. package/lib/services/shared.service.d.ts +15 -0
  137. package/lib/services/storage.service.d.ts +13 -0
  138. package/lib/tam.i18n.d.ts +24 -0
  139. package/lib/wrapper.d.ts +203 -0
  140. package/package.json +1 -5
  141. package/public-api.d.ts +15 -0
  142. package/rangertechnologies-ngnxt-2.1.108.tgz +0 -0
  143. package/src/lib/style.css +1817 -0
@@ -0,0 +1,115 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { ChangeWrapper } from '../../model/changeWrapper';
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "../../services/change.service";
5
+ import * as i2 from "../../services/data.service";
6
+ import * as i3 from "../../i18n.service";
7
+ import * as i4 from "@angular/common";
8
+ export class CustomRadioComponent {
9
+ changeService;
10
+ dataService;
11
+ i18nService;
12
+ options = [];
13
+ apiMeta;
14
+ selectedValue;
15
+ progressBar;
16
+ id;
17
+ readOnly = false;
18
+ errorMessage;
19
+ error;
20
+ fromShengel = false;
21
+ referenceField;
22
+ token;
23
+ valueChange = new EventEmitter();
24
+ invalidFieldIds = [];
25
+ labelField;
26
+ valueField;
27
+ subscription;
28
+ constructor(changeService, dataService, i18nService) {
29
+ this.changeService = changeService;
30
+ this.dataService = dataService;
31
+ this.i18nService = i18nService;
32
+ this.changeService.submitValidate$.subscribe((data) => {
33
+ this.invalidFieldIds.push(data);
34
+ });
35
+ }
36
+ ngOnInit() {
37
+ // VD 31NOV24 null check
38
+ if (this.apiMeta) {
39
+ let apiObj = JSON.parse(this.apiMeta);
40
+ this.labelField = apiObj.field;
41
+ this.dataService.apiResponse(apiObj.endpoint)?.subscribe((apiResponse) => {
42
+ let responses;
43
+ if (apiObj.variable) {
44
+ responses = this.dataService.getValue(apiResponse, apiObj.variable);
45
+ let results = [];
46
+ for (let i = 0; i < responses?.length; i++) {
47
+ var resp = responses[i];
48
+ results.push(resp);
49
+ }
50
+ this.options = results;
51
+ }
52
+ else {
53
+ responses = apiResponse;
54
+ this.options = responses;
55
+ }
56
+ });
57
+ let sourceId = apiObj.sourceQuestionId;
58
+ if (sourceId) {
59
+ this.subscription = this.changeService.changeAnnounced$.subscribe((changeValue) => {
60
+ if (changeValue != undefined) {
61
+ if (changeValue.valueObj != undefined && changeValue.fromQuestionId == apiObj.sourceQuestionId) {
62
+ this.selectedValue = changeValue.valueObj[apiObj.valueField];
63
+ let value = {};
64
+ value['name'] = this.selectedValue;
65
+ this.radioChange(value);
66
+ }
67
+ this.changeService.confirmChange(apiObj.sourceQuestionId);
68
+ }
69
+ });
70
+ }
71
+ }
72
+ }
73
+ radioChange(event) {
74
+ let change = new ChangeWrapper();
75
+ change.fromQuestionId = this.id;
76
+ change.valueObj = event.target?.id ? event.target?.id : '';
77
+ change.referenceField = this.referenceField;
78
+ change.selectedObj = event ? event[this.labelField] : '';
79
+ this.valueChange.emit(change);
80
+ if (event) {
81
+ this.invalidFieldIds = [];
82
+ }
83
+ }
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 <!-- 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
+ }
87
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomRadioComponent, decorators: [{
88
+ type: Component,
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
+ }], ctorParameters: () => [{ type: i1.ChangeService }, { type: i2.DataService }, { type: i3.I18nService }], propDecorators: { options: [{
91
+ type: Input
92
+ }], apiMeta: [{
93
+ type: Input
94
+ }], selectedValue: [{
95
+ type: Input
96
+ }], progressBar: [{
97
+ type: Input
98
+ }], id: [{
99
+ type: Input
100
+ }], readOnly: [{
101
+ type: Input
102
+ }], errorMessage: [{
103
+ type: Input
104
+ }], error: [{
105
+ type: Input
106
+ }], fromShengel: [{
107
+ type: Input
108
+ }], referenceField: [{
109
+ type: Input
110
+ }], token: [{
111
+ type: Input
112
+ }], valueChange: [{
113
+ type: Output
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":"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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>"]}
@@ -0,0 +1,186 @@
1
+ // Changes commented out due to Angular version compatibility; will apply after upgrade.
2
+ // // RS 06JAN2025
3
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
4
+ import { QuillEditorComponent } from 'ngx-quill';
5
+ import { FormsModule } from '@angular/forms';
6
+ import Quill from 'quill';
7
+ import { Mention } from 'quill-mention';
8
+ import ImageResizor from 'quill-image-resizor';
9
+ import { CommonModule } from '@angular/common';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "../../i18n.service";
12
+ import * as i2 from "../../services/change.service";
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"];
21
+ // Register Quill modules
22
+ Quill.register('modules/mention', Mention);
23
+ ImageResizor.Quill = Quill;
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);
30
+ export const QuillConfiguration = {
31
+ imageResizor: {},
32
+ toolbar: {
33
+ container: [
34
+ [{ 'font': fontFamilyArr }], //RS 17JAN2025 Added font family dropdown
35
+ ['bold', 'italic', 'underline', 'strike'],
36
+ ['blockquote', 'code-block'],
37
+ [{ header: [1, 2, 3, 4, 5, 6, false] }],
38
+ [{ list: 'ordered' }, { list: 'bullet' }],
39
+ [{ color: [] }, { background: [] }],
40
+ [{ align: [] }],
41
+ ['link', 'image', 'video'],
42
+ ['formula'],
43
+ ['clean'],
44
+ ],
45
+ },
46
+ mention: {
47
+ allowedChars: /^[A-Za-z\sÅÄÖåäö]*$/,
48
+ mentionDenotationChars: ['@', '#'],
49
+ source: function (searchTerm, renderList) {
50
+ const values = [
51
+ { id: 1, value: 'User 1' },
52
+ { id: 2, value: 'User 2' },
53
+ ];
54
+ if (searchTerm.length === 0) {
55
+ renderList(values, searchTerm);
56
+ }
57
+ else {
58
+ const matches = values.filter((item) => item.value.toLowerCase().includes(searchTerm.toLowerCase()));
59
+ renderList(matches, searchTerm);
60
+ }
61
+ },
62
+ },
63
+ };
64
+ // resize: {
65
+ // displaySize: true,
66
+ // modules: ['Resize', 'DisplaySize', 'Toolbar'],
67
+ // toolbarStyles: {
68
+ // backgroundColor: 'black',
69
+ // border: 'none',
70
+ // color: 'white'
71
+ // },
72
+ // handleStyles: {
73
+ // backgroundColor: 'black',
74
+ // border: 'none',
75
+ // color: 'white'
76
+ // }
77
+ // },
78
+ export class CustomRichTextComponent {
79
+ i18nService;
80
+ changeService;
81
+ value = '';
82
+ placeholder;
83
+ error;
84
+ question;
85
+ rows; //The number of visible text lines for the control
86
+ readOnly = false;
87
+ textValueChange = new EventEmitter();
88
+ minLength;
89
+ maxLength;
90
+ // @Input() value: any = ''; // Set default value
91
+ quillConfiguration = QuillConfiguration;
92
+ subscription;
93
+ constructor(i18nService, changeService) {
94
+ this.i18nService = i18nService;
95
+ this.changeService = changeService;
96
+ }
97
+ ngOnInit() {
98
+ console.log('Rich Text Init:', {
99
+ value: this.value,
100
+ question: this.question
101
+ });
102
+ if (this.value === undefined || this.value === null) {
103
+ this.value = '';
104
+ this.textValueChange.emit('');
105
+ }
106
+ this.initializeDependency();
107
+ }
108
+ // Separated dependency initialization for better organization
109
+ initializeDependency() {
110
+ if (this.question?.subText) {
111
+ console.log('subText:', this.question.subText);
112
+ try {
113
+ const dependencyObj = JSON.parse(this.question.subText);
114
+ if (dependencyObj?.sourceQuestionId) {
115
+ this.subscription = this.changeService.changeAnnounced$.subscribe((changeValue) => {
116
+ console.log('Change Value:', changeValue);
117
+ if (changeValue && changeValue.valueObj &&
118
+ changeValue.fromQuestionId === dependencyObj.sourceQuestionId) {
119
+ this.value = changeValue.valueObj[dependencyObj.valueField];
120
+ this.textValueChange.emit(this.value);
121
+ }
122
+ this.changeService.confirmChange(dependencyObj.sourceQuestionId);
123
+ });
124
+ }
125
+ }
126
+ catch (error) {
127
+ console.error('Error parsing subText:', error);
128
+ }
129
+ }
130
+ }
131
+ // onEditorChange(event: any): void {
132
+ // if (event && event.html !== undefined) {
133
+ // this.textValueChange.emit(event.html);
134
+ // } else {
135
+ // this.textValueChange.emit('');
136
+ // }
137
+ // }
138
+ // onEditorChange(event: any): void {
139
+ // console.log('Typing detected...', event.html);
140
+ // if (this.typingTimer) {
141
+ // clearTimeout(this.typingTimer); // Clear previous timer
142
+ // }
143
+ // this.typingTimer = setTimeout(() => {
144
+ // console.log('Debounced event fired:', event.html);
145
+ // const newValue = event?.html ? event.html : ''; // Ensure empty values are handled
146
+ // this.textValueChange.emit(newValue);
147
+ // }, this.doneTypingInterval);
148
+ // }
149
+ // RS 28JAN2015
150
+ onEditorFocusOut() {
151
+ const currentValue = this.value || '';
152
+ this.textValueChange.emit(currentValue);
153
+ console.log('Rich Text Editor Focus Out - Emitting Value:', currentValue);
154
+ }
155
+ // Added ngOnDestroy to prevent memory leaks
156
+ ngOnDestroy() {
157
+ if (this.subscription) {
158
+ this.subscription.unsubscribe();
159
+ }
160
+ }
161
+ 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 });
162
+ 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 (focusout)=\"onEditorFocusOut()\"\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}quill-editor{width:100%}\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"] }] });
163
+ }
164
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomRichTextComponent, decorators: [{
165
+ type: Component,
166
+ 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 (focusout)=\"onEditorFocusOut()\"\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}quill-editor{width:100%}\n"] }]
167
+ }], ctorParameters: () => [{ type: i1.I18nService }, { type: i2.ChangeService }], propDecorators: { value: [{
168
+ type: Input
169
+ }], placeholder: [{
170
+ type: Input
171
+ }], error: [{
172
+ type: Input
173
+ }], question: [{
174
+ type: Input
175
+ }], rows: [{
176
+ type: Input
177
+ }], readOnly: [{
178
+ type: Input
179
+ }], textValueChange: [{
180
+ type: Output
181
+ }], minLength: [{
182
+ type: Input
183
+ }], maxLength: [{
184
+ type: Input
185
+ }] } });
186
+ //# 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\n// onEditorChange(event: any): void {\n//   if (event && event.html !== undefined) {\n//     this.textValueChange.emit(event.html);\n//   } else {\n//     this.textValueChange.emit('');\n//   }\n// }\n// onEditorChange(event: any): void {\n//   console.log('Typing detected...', event.html);\n\n//   if (this.typingTimer) {\n//     clearTimeout(this.typingTimer); // Clear previous timer\n//   }\n//   this.typingTimer = setTimeout(() => {\n//     console.log('Debounced event fired:', event.html);\n\n//     const newValue = event?.html ? event.html : ''; // Ensure empty values are handled\n//     this.textValueChange.emit(newValue);\n//   }, this.doneTypingInterval);\n// }\n\n// RS 28JAN2015\nonEditorFocusOut(): void {\n  const currentValue = this.value || '';\n  this.textValueChange.emit(currentValue);\n  console.log('Rich Text Editor Focus Out - Emitting Value:', currentValue);\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\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    (focusout)=\"onEditorFocusOut()\"\n    [class.error]=\"error\">\n  </quill-editor>\n  <div *ngIf=\"error\" class=\"error-message\">\n    {{ error }}\n  </div>\n</div>"]}