@rangertechnologies/ngnxt 2.1.241 → 2.1.242-beta

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 (90) hide show
  1. package/fesm2022/rangertechnologies-ngnxt.mjs +159 -193
  2. package/fesm2022/rangertechnologies-ngnxt.mjs.map +1 -1
  3. package/lib/components/image-cropper/component/image-cropper.component.d.ts +1 -1
  4. package/lib/components/pick-location/pick-location.component.d.ts +0 -1
  5. package/package.json +4 -6
  6. package/rangertechnologies-ngnxt-2.1.242-beta.tgz +0 -0
  7. package/esm2022/environments/version.mjs +0 -15
  8. package/esm2022/lib/ar.i18n.mjs +0 -29
  9. package/esm2022/lib/components/button/nxt-button.component.mjs +0 -165
  10. package/esm2022/lib/components/custom-calendar/custom-calendar.component.mjs +0 -371
  11. package/esm2022/lib/components/custom-dropdown/custom-dropdown.component.mjs +0 -274
  12. package/esm2022/lib/components/custom-model/custom-model.component.mjs +0 -51
  13. package/esm2022/lib/components/custom-radio/custom-radio.component.mjs +0 -156
  14. package/esm2022/lib/components/datatable/datatable.component.mjs +0 -1625
  15. package/esm2022/lib/components/file-upload/file-upload.component.mjs +0 -283
  16. package/esm2022/lib/components/icon-selector/icon-selector.component.mjs +0 -104
  17. package/esm2022/lib/components/image-cropper/component/cropper.state.mjs +0 -208
  18. package/esm2022/lib/components/image-cropper/component/image-cropper.component.mjs +0 -562
  19. package/esm2022/lib/components/image-cropper/interfaces/basic-event.interface.mjs +0 -2
  20. package/esm2022/lib/components/image-cropper/interfaces/cropper-options.interface.mjs +0 -2
  21. package/esm2022/lib/components/image-cropper/interfaces/cropper-position.interface.mjs +0 -2
  22. package/esm2022/lib/components/image-cropper/interfaces/dimensions.interface.mjs +0 -2
  23. package/esm2022/lib/components/image-cropper/interfaces/exif-transform.interface.mjs +0 -2
  24. package/esm2022/lib/components/image-cropper/interfaces/image-cropped-event.interface.mjs +0 -2
  25. package/esm2022/lib/components/image-cropper/interfaces/image-transform.interface.mjs +0 -2
  26. package/esm2022/lib/components/image-cropper/interfaces/index.mjs +0 -2
  27. package/esm2022/lib/components/image-cropper/interfaces/loaded-image.interface.mjs +0 -2
  28. package/esm2022/lib/components/image-cropper/interfaces/move-start.interface.mjs +0 -8
  29. package/esm2022/lib/components/image-cropper/services/crop.service.mjs +0 -139
  30. package/esm2022/lib/components/image-cropper/services/load-image.service.mjs +0 -194
  31. package/esm2022/lib/components/image-cropper/utils/cropper-position.utils.mjs +0 -239
  32. package/esm2022/lib/components/image-cropper/utils/exif.utils.mjs +0 -79
  33. package/esm2022/lib/components/image-cropper/utils/keyboard.utils.mjs +0 -40
  34. package/esm2022/lib/components/image-cropper/utils/percentage.utils.mjs +0 -4
  35. package/esm2022/lib/components/image-cropper/utils/resize.utils.mjs +0 -75
  36. package/esm2022/lib/components/list-view-filter/list-view-filter.component.mjs +0 -403
  37. package/esm2022/lib/components/nxt-input/nxt-input.component.mjs +0 -2925
  38. package/esm2022/lib/components/pagination/pagination.component.mjs +0 -101
  39. package/esm2022/lib/components/pick-location/pick-location.component.mjs +0 -234
  40. package/esm2022/lib/components/search-box/search-box.component.mjs +0 -415
  41. package/esm2022/lib/country.json +0 -43237
  42. package/esm2022/lib/en.i18n.mjs +0 -29
  43. package/esm2022/lib/i18n-config.service.mjs +0 -4
  44. package/esm2022/lib/i18n.component.mjs +0 -47
  45. package/esm2022/lib/i18n.module.mjs +0 -38
  46. package/esm2022/lib/i18n.pipe.mjs +0 -26
  47. package/esm2022/lib/i18n.service.mjs +0 -56
  48. package/esm2022/lib/interfaces/actionMeta.mjs +0 -2
  49. package/esm2022/lib/interfaces/apimeta.mjs +0 -2
  50. package/esm2022/lib/interfaces/dependencyMeta.mjs +0 -2
  51. package/esm2022/lib/model/bookletWrapper.mjs +0 -9
  52. package/esm2022/lib/model/changeWrapper.mjs +0 -11
  53. package/esm2022/lib/model/errorWrapper.mjs +0 -6
  54. package/esm2022/lib/nxt-app.component.mjs +0 -22
  55. package/esm2022/lib/nxt-app.module.mjs +0 -139
  56. package/esm2022/lib/nxt-app.service.mjs +0 -14
  57. package/esm2022/lib/pages/booklet/booklet.component.mjs +0 -675
  58. package/esm2022/lib/pages/builder/element/element.component.mjs +0 -466
  59. package/esm2022/lib/pages/builder/form/form.component.mjs +0 -45
  60. package/esm2022/lib/pages/builder/properties/common-fields.constants.mjs +0 -89
  61. package/esm2022/lib/pages/builder/properties/properties.component.mjs +0 -1021
  62. package/esm2022/lib/pages/builder/templates/templates.component.mjs +0 -34
  63. package/esm2022/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.mjs +0 -676
  64. package/esm2022/lib/pages/pdfDesigner/pdf-properties/pdf-properties.component.mjs +0 -1126
  65. package/esm2022/lib/pages/questionbook/questionbook.component.mjs +0 -841
  66. package/esm2022/lib/pages/questionnaire/questionnaire.component.mjs +0 -2405
  67. package/esm2022/lib/pipe/custom-translate.pipe.mjs +0 -27
  68. package/esm2022/lib/pipe/date/date.pipe.mjs +0 -28
  69. package/esm2022/lib/pipe/editColumnCheck/edit-column-check.pipe.mjs +0 -28
  70. package/esm2022/lib/pipe/editColumnDropdown/edit-column-dropdown.pipe.mjs +0 -20
  71. package/esm2022/lib/pipe/editColumnType/edit-column-type.pipe.mjs +0 -20
  72. package/esm2022/lib/pipe/get-value.pipe.mjs +0 -48
  73. package/esm2022/lib/pipe/search-filter/search-filter.pipe.mjs +0 -39
  74. package/esm2022/lib/pipe/time/time.pipe.mjs +0 -26
  75. package/esm2022/lib/sample.mjs +0 -3715
  76. package/esm2022/lib/services/change.service.mjs +0 -50
  77. package/esm2022/lib/services/country.service.mjs +0 -135
  78. package/esm2022/lib/services/data.service.mjs +0 -103
  79. package/esm2022/lib/services/form-builder.service.mjs +0 -375
  80. package/esm2022/lib/services/pdf-designer.service.mjs +0 -401
  81. package/esm2022/lib/services/salesforce.service.mjs +0 -46
  82. package/esm2022/lib/services/shared.service.mjs +0 -100
  83. package/esm2022/lib/services/storage.service.mjs +0 -44
  84. package/esm2022/lib/services/template.service.mjs +0 -351
  85. package/esm2022/lib/services/translation.service.mjs +0 -62
  86. package/esm2022/lib/tam.i18n.mjs +0 -29
  87. package/esm2022/lib/wrapper.mjs +0 -175
  88. package/esm2022/public-api.mjs +0 -25
  89. package/esm2022/rangertechnologies-ngnxt.mjs +0 -5
  90. package/rangertechnologies-ngnxt-2.1.241.tgz +0 -0
@@ -1,1021 +0,0 @@
1
- import { Component, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
2
- import { v4 as uuidv4 } from 'uuid'; //AP-28MAR25 import uuid
3
- import { CommonModule } from '@angular/common';
4
- import { FormsModule } from '@angular/forms';
5
- import { ImageCropperComponent } from "../../../components/image-cropper/component/image-cropper.component";
6
- import { NxtSearchBox } from "../../../components/search-box/search-box.component";
7
- //AP-25JUN25-Importing common constants for element, field, and appearance properties
8
- import { COMMON_ELEMENT_PROPS, COMMON_OPTIONS_FIELD, COMMON_FIELD_PROPS, COMMON_APPEARANCE_PROPS } from '../properties/common-fields.constants';
9
- import * as i0 from "@angular/core";
10
- import * as i1 from "@angular/common/http";
11
- import * as i2 from "../../../services/form-builder.service";
12
- import * as i3 from "../../../services/template.service";
13
- import * as i4 from "@angular/common";
14
- import * as i5 from "@angular/forms";
15
- export class PropertiesComponent {
16
- http;
17
- formBuilderService;
18
- templateService;
19
- element;
20
- from;
21
- columns;
22
- createElementConfig(config) {
23
- return {
24
- elementProps: [...COMMON_ELEMENT_PROPS, ...(config?.customElementProps || [])],
25
- fieldProps: [...COMMON_FIELD_PROPS, ...(config?.customFieldProps || [])],
26
- appearance: [...COMMON_APPEARANCE_PROPS, ...(config?.customAppearanceProps || [])]
27
- };
28
- }
29
- formButtonHandler = new EventEmitter();
30
- templateSaveHandler = new EventEmitter();
31
- selectedOption = '';
32
- selectedElementIndex = -1;
33
- selectedElementType = '';
34
- selectedAlign = 'align-left'; // Add this property
35
- selectedStyles = []; // Using array since multiple styles can be selected
36
- errorMessage = '';
37
- selectedElement = null; // Receive the selected element
38
- headerSelect = false;
39
- bookId;
40
- book;
41
- activeTab = 'attributes';
42
- selectColumn;
43
- transform = {
44
- translateUnit: 'px',
45
- scale: 1,
46
- rotate: 0,
47
- flipH: false,
48
- flipV: false,
49
- translateH: 0,
50
- translateV: 0
51
- };
52
- canvasRotation = 0;
53
- cropper;
54
- loading = false;
55
- cropperMaxHeight = 0;
56
- cropperMaxWidth = 0;
57
- cropperMinHeight = 0;
58
- cropperMinWidth = 0;
59
- cropperStaticWidth = 0;
60
- cropperStaticHeight = 0;
61
- aspectRatio = 4 / 3;
62
- roundCropper = false;
63
- imageChangedEvent = null;
64
- alignImage = 'center';
65
- // "endpoint": "https://dev-api.valarhr.com/nxt",
66
- // SKS28MAR25 default book search endpoint
67
- bookSubtext = {
68
- "endpoint": "https://dev-api.valarhr.com/nxt",
69
- "variable": null,
70
- "field": ['label', 'name'],
71
- "defaultField": "label"
72
- };
73
- isLinkDropdownOpen = false;
74
- templateSelected = false;
75
- dropdown;
76
- draggedId = null;
77
- formElements = [];
78
- unique_id = 'book-1234'; // should be consistent
79
- selectedTemplate;
80
- elementProperties = {
81
- // AP-10MAR25 - Defines the Header element with a text input field
82
- // AP-25MAR25 Add subtext all variables
83
- 'Header': {
84
- elementProps: [
85
- { label: 'Label', placeholder: 'Enter Text', type: 'text', key: 'title' }
86
- ]
87
- },
88
- // AP-19MAR25 Add Line Property
89
- 'Line': {
90
- elementProps: [
91
- { label: 'Line Width', type: 'fieldSize', key: 'size' },
92
- { label: 'Padding Top', type: 'number', key: 'paddingTop', placeholder: 'Enter padding top' },
93
- { label: 'Padding Bottom', type: 'number', key: 'paddingBottom', placeholder: 'Enter padding bottom' },
94
- {
95
- label: 'Line Style', type: 'select', key: 'lineStyle',
96
- options: ['Solid', 'Dashed', 'Dotted']
97
- }
98
- ],
99
- fieldProps: []
100
- },
101
- 'Space': {
102
- elementProps: [
103
- { label: 'Element', type: 'number', key: 'questionNumber' },
104
- { label: 'Field Size', type: 'fieldSize', key: 'size' },
105
- ],
106
- fieldProps: []
107
- },
108
- "Table": {
109
- "elementProps": [
110
- { label: 'Is Label', placeholder: 'Enter Text ', type: 'checkbox', key: 'style.showLabel' },
111
- { key: 'primaryKey' },
112
- { "label": "tableName", "placeholder": "Employee Details", "type": "text", "key": "questionText" },
113
- { "label": "inputTextAlignment", "placeholder": "Left", "type": "text", "key": "inputTextAlignment" },
114
- { "label": "tableScaleSize", "placeholder": "06 - Full Scale", "type": "text", "key": "tableScaleSize" },
115
- { "label": "isNosIndicator", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.isNosIndicator" },
116
- { "label": "isPagination", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.isPagination" },
117
- { "label": "addInlineRecord", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.addInlineRecord" },
118
- { "label": "actionButton", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.actionButton" },
119
- { "label": "searchBar", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.searchBar" },
120
- { "label": "isDeleteRow", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.isDeleteRow" },
121
- { "label": "isEditRow", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.isEditRow" },
122
- { "label": "isButtons", "placeholder": "06 - Full Scale", "type": "checkbox", "key": "tableConfig.isButtons" },
123
- { "label": "rowChoice", "placeholder": "", "type": "text", "key": "rowChoice" },
124
- { label: 'questionNumber', type: 'number', key: 'questionNumber' },
125
- { label: 'Field Size', type: 'fieldSize', key: 'size' },
126
- ],
127
- "fieldProps": [
128
- { "label": "tableId", "placeholder": "Emp_Table_01", "type": "text", "key": "tableId" },
129
- { "label": "helpText", "placeholder": "Enter message", "type": "text", "key": "helpText" },
130
- { "label": "defaultValue", "placeholder": "Default Value", "type": "text", "key": "defaultValue" },
131
- { label: 'Reference', placeholder: 'Reference Field', type: 'text', key: 'referenceField' },
132
- {
133
- "label": "", "placeholder": "Enter Text", "type": "subQuestion", "key": "isSubText",
134
- "subQuestion": [
135
- { "label": "End Point", "placeholder": "endpoint", "type": "text", "key": "subText.endpoint" },
136
- { "label": "Variable", "placeholder": "variable", "type": "text", "key": "subText.variable" },
137
- { "label": "Field", "placeholder": "field", "type": "text", "key": "subText.field" },
138
- { "label": "Default Field", "placeholder": "defaultField", "type": "text", "key": "subText.defaultField" },
139
- { "label": "Label Field", "placeholder": "labelField", "type": "text", "key": "subText.labelField" },
140
- { "label": "Value Field", "placeholder": "valueField", "type": "text", "key": "subText.valueField" },
141
- { "label": "Source Question Id", "placeholder": "sourceQuestionId", "type": "text", "key": "subText.sourceQuestionId" },
142
- { "label": "Dependent Value", "placeholder": "dependentValue", "type": "text", "key": "subText.dependentValue" },
143
- { "label": "Dependent Field", "placeholder": "isDependentField", "type": "boolean", "key": "subText.isDependentField" },
144
- { "label": "Query Field", "placeholder": "queryField", "type": "text", "key": "subText.queryField" },
145
- { "label": "Query Value", "placeholder": "queryValue", "type": "text", "key": "subText.queryValue" },
146
- { "label": "Query Value Ref", "placeholder": "queryValueRef", "type": "text", "key": "subText.queryValueReference" },
147
- { "label": "Unique Key", "placeholder": "uniqueKey", "type": "text", "key": "subText.uniqueKey" },
148
- ]
149
- },
150
- ],
151
- "appearance": [
152
- {
153
- label: 'Select Font', type: 'select', key: 'font', placeholder: 'Left', required: true,
154
- options: ['Helvetica Neue', 'Arial', 'Times New Roman', 'Roboto']
155
- },
156
- { label: 'Font Color', type: 'color', key: 'fontColor', defaultValue: '', required: true },
157
- {
158
- label: 'Font Size', type: 'select', key: 'fontSize', defaultValue: '', required: true,
159
- options: [
160
- { label: 'Small', value: '12px' },
161
- { label: 'Medium', value: '14px' },
162
- { label: 'Large', value: '16px' }
163
- ]
164
- },
165
- {
166
- label: 'Font Width', type: 'select', key: 'fontWeight', defaultValue: '', required: true,
167
- options: [
168
- { value: '400', label: '400-Normal' },
169
- { value: '500', label: '500-Medium' },
170
- { value: '600', label: '600-Semi Bold' },
171
- { value: '700', label: '700-Bold' }
172
- ]
173
- },
174
- { label: 'Duplicate Field', type: 'button-toggle', key: 'duplicateField', defaultValue: false }
175
- ]
176
- },
177
- "TableColumn": {
178
- "elementProps": [
179
- { "label": "headerLabels", "placeholder": "Employee Details", "type": "text", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "label" },
180
- { "label": "apiName", "placeholder": "Employee Details", "type": "text", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "apiName" },
181
- {
182
- "label": "Summary Column", "placeholder": "Summary Column", "type": "subQuestion", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "summaryColumn",
183
- "subQuestion": [
184
- { "label": "Type", "placeholder": "Employee Details", "type": "text", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "fldType" },
185
- { "label": "Operation", "placeholder": "Employee Details", "type": "text", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "operation" },
186
- { "label": "Operands", "placeholder": "Employee Details", "type": "array", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "operands" },
187
- ]
188
- },
189
- {
190
- "label": "Summary Row", "placeholder": "Summary Row", "type": "subQuestion", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "summaryRow", "subQuestion": [
191
- { "label": "Type", "placeholder": "Employee Details", "type": "text", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "fldType" },
192
- { "label": "operation", "placeholder": "operation", "type": "text", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "operation" },
193
- { "label": "column", "placeholder": "Employee Details", "type": "radio", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "column" },
194
- { "label": "operands", "placeholder": "Employee Details", "type": "array", "targetArray": "fieldsMeta", "targetArrayKey": "uniqueIdentifier", "key": "operands" },
195
- ]
196
- },
197
- { "label": "inputTextAlignment", "placeholder": "Left", "type": "text", "key": "inputTextAlignment" },
198
- { "label": "tableScaleSize", "placeholder": "06 - Full Scale", "type": "text", "key": "tableScaleSize" },
199
- { "label": "rowChoice", "placeholder": "", "type": "text", "key": "rowChoice" },
200
- ],
201
- "fieldProps": [
202
- { "label": "tableId", "placeholder": "Emp_Table_01", "type": "text", "key": "tableId" },
203
- { "label": "inputTextAlignment", "placeholder": "Left", "type": "text", "key": "inputTextAlignment" },
204
- { "label": "helpText", "placeholder": "Enter help message here", "type": "text", "key": "helpText" },
205
- { "label": "defaultValue", "placeholder": "Default Value", "type": "text", "key": "defaultValue" },
206
- { "label": "referenceAnchor", "placeholder": "http://source/rangernxt_bio_data/74/d.doc", "type": "text", "key": "referenceAnchor" },
207
- ],
208
- "appearance": [
209
- {
210
- label: 'Select Font', type: 'select', key: 'font', placeholder: 'Left', required: true,
211
- options: ['Helvetica Neue', 'Arial', 'Times New Roman', 'Roboto']
212
- },
213
- { label: 'Font Color', type: 'color', key: 'fontColor', defaultValue: '', required: true },
214
- {
215
- label: 'Font Size', type: 'select', key: 'fontSize', defaultValue: '', required: true,
216
- options: [
217
- { label: 'Small', value: '12px' },
218
- { label: 'Medium', value: '14px' },
219
- { label: 'Large', value: '16px' }
220
- ]
221
- },
222
- {
223
- label: 'Font Width', type: 'select', key: 'fontWeight', defaultValue: '', required: true,
224
- options: [
225
- { value: '400', label: '400-Normal' },
226
- { value: '500', label: '500-Medium' },
227
- { value: '600', label: '600-Semi Bold' },
228
- { value: '700', label: '700-Bold' }
229
- ]
230
- },
231
- { label: 'Duplicate Field', type: 'button-toggle', key: 'duplicateField', defaultValue: false }
232
- ]
233
- },
234
- ListColumn: this.createElementConfig({
235
- customFieldProps: [
236
- ...COMMON_OPTIONS_FIELD.map(opt => ({
237
- type: 'list'
238
- })),
239
- ],
240
- customElementProps: [
241
- { label: "headerLabels", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "label" },
242
- { label: "apiName", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "apiName" },
243
- {
244
- label: "Summary Column", placeholder: "Summary Column", type: "subQuestion", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "summaryColumn",
245
- subQuestion: [
246
- { label: "Type", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "fldType" },
247
- { label: "Operation", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operation" },
248
- { label: "Operands", placeholder: "Employee Details", type: "array", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operands" },
249
- ]
250
- },
251
- {
252
- label: "Summary Row", placeholder: "Summary Row", type: "subQuestion", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "summaryRow",
253
- subQuestion: [
254
- { label: "Type", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "fldType" },
255
- { label: "operation", placeholder: "operation", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operation" },
256
- { label: "column", placeholder: "Employee Details", type: "radio", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "column" },
257
- { label: "operands", placeholder: "Employee Details", type: "array", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operands" },
258
- ]
259
- },
260
- ]
261
- }),
262
- DropdownColumn: this.createElementConfig({
263
- customFieldProps: [
264
- ...COMMON_OPTIONS_FIELD.map(opt => ({
265
- ...opt,
266
- type: 'dropdown'
267
- })),
268
- ],
269
- customElementProps: [
270
- { label: "headerLabels", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "label" },
271
- { label: "apiName", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "apiName" },
272
- {
273
- label: "Summary Column", placeholder: "Summary Column", type: "subQuestion", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "summaryColumn",
274
- subQuestion: [
275
- { label: "Type", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "fldType" },
276
- { label: "Operation", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operation" },
277
- { label: "Operands", placeholder: "Employee Details", type: "array", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operands" },
278
- ]
279
- },
280
- {
281
- label: "Summary Row", placeholder: "Summary Row", type: "subQuestion", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "summaryRow",
282
- subQuestion: [
283
- { label: "Type", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "fldType" },
284
- { label: "operation", placeholder: "operation", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operation" },
285
- { label: "column", placeholder: "Employee Details", type: "radio", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "column" },
286
- { label: "operands", placeholder: "Employee Details", type: "array", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operands" },
287
- ]
288
- },
289
- ]
290
- }),
291
- DateColumn: this.createElementConfig({
292
- customFieldProps: [
293
- ...COMMON_OPTIONS_FIELD.map(opt => ({
294
- type: 'date'
295
- })),
296
- ],
297
- customElementProps: [
298
- { label: "headerLabels", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "label" },
299
- { label: "apiName", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "apiName" },
300
- {
301
- label: "Summary Column", placeholder: "Summary Column", type: "subQuestion", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "summaryColumn",
302
- subQuestion: [
303
- { label: "Type", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "fldType" },
304
- { label: "Operation", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operation" },
305
- { label: "Operands", placeholder: "Employee Details", type: "array", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operands" },
306
- ]
307
- },
308
- {
309
- label: "Summary Row", placeholder: "Summary Row", type: "subQuestion", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "summaryRow",
310
- subQuestion: [
311
- { label: "Type", placeholder: "Employee Details", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "fldType" },
312
- { label: "operation", placeholder: "operation", type: "text", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operation" },
313
- { label: "column", placeholder: "Employee Details", type: "radio", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "column" },
314
- { label: "operands", placeholder: "Employee Details", type: "array", targetArray: "fieldsMeta", targetArrayKey: "uniqueIdentifier", key: "operands" },
315
- ]
316
- },
317
- ]
318
- }),
319
- Checkbox: this.createElementConfig({
320
- customFieldProps: COMMON_OPTIONS_FIELD.map(opt => ({
321
- ...opt,
322
- type: 'checkbox'
323
- }))
324
- }),
325
- Dropdown: this.createElementConfig({
326
- customFieldProps: COMMON_OPTIONS_FIELD.map(opt => ({
327
- ...opt,
328
- type: 'dropdown'
329
- }))
330
- }),
331
- Radio: this.createElementConfig({
332
- customFieldProps: COMMON_OPTIONS_FIELD.map(opt => ({
333
- ...opt,
334
- type: 'checkbox'
335
- }))
336
- }),
337
- Text: this.createElementConfig(),
338
- Calendar: this.createElementConfig(),
339
- Date: this.createElementConfig(),
340
- Time: this.createElementConfig(),
341
- DateTime: this.createElementConfig(),
342
- Email: this.createElementConfig(),
343
- Book: this.createElementConfig(),
344
- List: this.createElementConfig(),
345
- TextArea: this.createElementConfig(),
346
- RichTextArea: this.createElementConfig(),
347
- Number: this.createElementConfig(),
348
- Image: this.createElementConfig(),
349
- Label: this.createElementConfig(),
350
- Boolean: this.createElementConfig(),
351
- File: this.createElementConfig(),
352
- Icon: this.createElementConfig(), //MSM10JUL25 icon selector element
353
- };
354
- constructor(http, formBuilderService, templateService) {
355
- this.http = http;
356
- this.formBuilderService = formBuilderService;
357
- this.templateService = templateService;
358
- }
359
- handleTemplateSelection(event) {
360
- this.formElements = event.elements;
361
- this.selectedTemplate = event.selectedTemplate;
362
- this.templateService.formElements = this.formElements;
363
- this.templateService.book = this.book;
364
- this.templateService.unique_id = this.unique_id;
365
- }
366
- duplicateField(element) {
367
- const elements = this.formBuilderService.getElements();
368
- const copy = JSON.parse(JSON.stringify(element));
369
- //AP-18APR25 Generate a new ID using the service's method
370
- copy.id = this.formBuilderService.addElementWithId();
371
- //copy.uuid = copy.id;
372
- // AP-18APR25 Set the questionNumber to the next available number
373
- copy.questionNumber = elements.length + 1;
374
- this.formBuilderService.addElement(copy);
375
- }
376
- saveData() {
377
- this.templateService.formElements = this.formElements;
378
- this.templateService.book = this.book;
379
- this.templateService.unique_id = this.unique_id;
380
- const updatedBook = this.templateService.download();
381
- }
382
- ngOnInit() {
383
- //AP-10MAR25 Subscribes to header selection changes and updates the selected element
384
- this.formBuilderService.selectHeaderSubject$.subscribe(header => {
385
- if (header) {
386
- if (!this.selectedElement) {
387
- this.selectedElement = {};
388
- }
389
- this.selectedElement['type'] = header;
390
- this.headerSelect = true;
391
- this.book = this.formBuilderService.getBook();
392
- this.bookId = this.book.records[0].id;
393
- }
394
- });
395
- this.formBuilderService.selectedElement$.subscribe(index => {
396
- this.selectedElementIndex = index;
397
- this.headerSelect = false;
398
- if (index >= 0) {
399
- const elements = this.formBuilderService.getElements();
400
- this.selectedElement = elements[index];
401
- if (this.selectedElement?.type === 'Table') {
402
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement.fieldsMeta === 'object' ? this.selectedElement.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);
403
- this.selectColumn = this.formBuilderService.getSelectTableColumn() === undefined ? null : this.formBuilderService.getSelectTableColumn();
404
- }
405
- else {
406
- this.selectColumn = null;
407
- }
408
- }
409
- this.getProperties();
410
- });
411
- this.formBuilderService.selectHeaderSubject$.subscribe(action => {
412
- if (action) {
413
- this.book = this.formBuilderService.getBook();
414
- }
415
- });
416
- if (this.from === 'formBuilder') {
417
- const fieldsMeta = typeof this.element.fieldsMeta === 'string'
418
- ? JSON.parse(this.element.fieldsMeta)
419
- : this.element.fieldsMeta;
420
- this.columns = fieldsMeta.map(column => ({
421
- ...column,
422
- type: column.fldType || 'Text' // Default to 'Text' if fldType not available
423
- }));
424
- }
425
- }
426
- // AP 23MAY25 - Add a new button with default styling
427
- addNewButton() {
428
- const lastPercent = this.book?.questionbook?.action?.slice(-1)[0]?.positionPercent || 0;
429
- const buttons = {
430
- name: '',
431
- eventtoemit: '',
432
- alt: '',
433
- endpoint: '',
434
- method: '',
435
- id: this.formBuilderService.addElementWithId(),
436
- positionPercent: lastPercent + 10,
437
- width: 100,
438
- borderRadius: 6,
439
- alignment: 'flex-start'
440
- };
441
- this.formBuilderService.updateQuestionBookActionButtons(buttons, 'add');
442
- }
443
- //AP 23MAY25 - Remove a button from the action list
444
- removeButton(btn) {
445
- this.formBuilderService.updateQuestionBookActionButtons(btn, 'remove');
446
- }
447
- //AP 23MAY25 - Update a specific property of a button
448
- onButtonPropertyChange(index, key, value) {
449
- const buttons = this.getValueByPath('action') || [];
450
- if (buttons[index]) {
451
- buttons[index][key] = value;
452
- this.setValueByPath('action', buttons);
453
- }
454
- }
455
- //AP-10MAR25 Updates the title using the form builder service
456
- updateTitle(event) {
457
- this.formBuilderService.updateTitle(event);
458
- }
459
- validateInput(value, type) {
460
- if (type === 'label' || type === 'placeholder') {
461
- const regex = /^[a-zA-Z0-9\s]*$/;
462
- if (!regex.test(value)) {
463
- this.errorMessage = 'Only letters and numbers are allowed';
464
- return false;
465
- }
466
- }
467
- this.errorMessage = '';
468
- return true;
469
- }
470
- get fieldAsString() {
471
- return this.bookSubtext.field.join(', '); //SKS28MAR25 Convert array to comma-separated string
472
- }
473
- updateField(value) {
474
- this.bookSubtext.field = value.split(',').map(item => item.trim()); //SKS28MAR25 Convert string back to array
475
- }
476
- // SKS20MAR25 Getter to filter columns, excluding the one with currentUniqueIdentifier
477
- get filteredColumns() {
478
- const targetArray = this.getDataByPath('fieldsMeta');
479
- return targetArray.filter(column => column.uniqueIdentifier !== this.selectColumn);
480
- }
481
- // SKS20MAR25 Method to handle checkbox changes
482
- onCheckboxChange(targetArrayPath, targetArrayKey, key, apiName, isChecked) {
483
- const targetArray = this.getDataByPath(targetArrayPath);
484
- const foundItem = targetArray.find(item => item[targetArrayKey] === this.selectColumn);
485
- foundItem[key] = foundItem[key] ? foundItem[key] : [];
486
- if (isChecked) {
487
- // Add apiName to selectedApiNames if not already present
488
- if (!foundItem[key].includes(apiName)) {
489
- foundItem[key].push(apiName);
490
- }
491
- }
492
- else {
493
- // Remove apiName from selectedApiNames if present
494
- const index = foundItem[key].indexOf(apiName);
495
- if (index > -1) {
496
- foundItem[key].splice(index, 1);
497
- }
498
- }
499
- // this.updateElement(this.selectedElement)
500
- }
501
- updateProperty(key, value) {
502
- if (this.selectedElementIndex >= 0) {
503
- if (this.selectedElement.type === 'Table') {
504
- this.setValueByPath(key, value);
505
- }
506
- if (key === 'questionText' || key === 'question') {
507
- if (!this.validateInput(value, key)) {
508
- return;
509
- }
510
- }
511
- if (key === 'questionNumber') {
512
- //AP-10MAR25 Convert the value to an integer
513
- const newOrder = parseInt(value, 10);
514
- if (isNaN(newOrder))
515
- return;
516
- //AP-10MAR25 Update the element's questionNumber in the form builder service
517
- this.formBuilderService.updateElement(this.selectedElementIndex, { questionNumber: newOrder });
518
- //AP-10MAR25 Sort elements based on the updated questionNumber
519
- this.formBuilderService.sortElementsByOrder();
520
- }
521
- if (key === 'fontWeight') {
522
- this.selectedElement[key] = value;
523
- }
524
- const update = { [key]: value };
525
- this.formBuilderService.updateElement(this.selectedElementIndex, update);
526
- // Special handling for font and font weight
527
- if (key === 'font') {
528
- this.selectedElement.font = value; // Directly update the selected element's font property
529
- }
530
- }
531
- }
532
- // SKS21MAR25 radio button click handler
533
- onRadioChange(targetArrayPath, targetArrayKey, key, value) {
534
- if (this.selectedElement.type === 'Table') {
535
- // this.updateValueByArrayPath(targetArray, targetArrayKey, this.selectColumn, key, value);
536
- if (this.selectedElement?.type === 'Table') {
537
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);
538
- }
539
- const targetArray = this.getDataByPath(targetArrayPath);
540
- // console.log("targetArray", targetArray, "targetArrayKey", targetArrayKey, "selectColumn", this.selectColumn, "keyPath", key, "event", value)
541
- if (!Array.isArray(targetArray))
542
- return;
543
- // Find the object that matches selectColumn
544
- let foundItem = targetArray.find(item => item[targetArrayKey] === this.selectColumn);
545
- // If the item does not exist, create and add it
546
- if (!foundItem) {
547
- foundItem = { [targetArrayKey]: this.selectColumn }; // Ensure the key exists
548
- targetArray.push(foundItem);
549
- }
550
- // Traverse and set the value
551
- const keys = key.split('.');
552
- let obj = foundItem;
553
- for (let i = 0; i < keys.length - 1; i++) {
554
- const key = keys[i];
555
- if (!obj[key])
556
- obj[key] = {}; // Initialize missing objects
557
- obj = obj[key];
558
- }
559
- // Assign the value based on input type
560
- const lastKey = keys[keys.length - 1];
561
- obj[lastKey] = value;
562
- if (this.selectedElement?.type === 'Table') {
563
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? JSON.stringify(this.selectedElement?.fieldsMeta) : this.selectedElement['fieldsMeta'] || [];
564
- }
565
- this.updateElement(this.selectedElement);
566
- }
567
- else {
568
- this.setValueByPath(key, value); // Fallback for non-Table elements, if applicable
569
- }
570
- }
571
- onRequiredChange(value) {
572
- if (this.selectedElement) {
573
- let label = this.selectedElement.label ? this.selectedElement.label.replace(/\s*\*+$/, '') : 'Label';
574
- if (value) {
575
- label = `${label} *`;
576
- }
577
- this.updateProperty('label', label);
578
- this.updateProperty('isOptional', value);
579
- }
580
- }
581
- // Toggle for Sub Questions (Each prop has its own state)
582
- toggleSubQuestion(prop) {
583
- prop.isExpanded = !prop.isExpanded;
584
- }
585
- // Separate Toggle for Style Section
586
- isStyleExpanded = false;
587
- toggleStyleSection() {
588
- this.isStyleExpanded = !this.isStyleExpanded;
589
- }
590
- getProperties() {
591
- if (!this.selectedElement)
592
- return null;
593
- // AP-10MAR25 Retrieve the latest book data
594
- this.book = this.formBuilderService.getBook();
595
- // AP-10MAR25 If the selected element is a 'Header', return its properties
596
- if (this.selectedElement?.type === 'Header') {
597
- return this.elementProperties['Header'];
598
- }
599
- if (this.selectedElement?.type === 'Table') {
600
- if (this.selectColumn !== null) {
601
- // Get the column metadata
602
- const fieldsMeta = Array.isArray(this.selectedElement.fieldsMeta)
603
- ? this.selectedElement.fieldsMeta
604
- : JSON.parse(this.selectedElement.fieldsMeta || '[]');
605
- const column = fieldsMeta.find((c) => c.uniqueIdentifier === this.selectColumn);
606
- if (column) {
607
- // Return specific properties based on column type
608
- switch (column.fldType) {
609
- case 'List':
610
- return this.elementProperties['ListColumn'];
611
- case 'Dropdown':
612
- return this.elementProperties['DropdownColumn'];
613
- case 'Date':
614
- return this.elementProperties['DateColumn'];
615
- default:
616
- return this.elementProperties['TableColumn'];
617
- }
618
- }
619
- return this.elementProperties[this.selectedElement?.type] || this.elementProperties['DefaultColumn'];
620
- }
621
- }
622
- return this.elementProperties[this.selectedElement?.type];
623
- }
624
- // AP-28MAR25 Add an option with a unique UUID
625
- addOption(options) {
626
- const unique_id = uuidv4();
627
- options.push({ id: unique_id, value: '' });
628
- }
629
- // AP-28MAR25 Function to remove an option based on its unique UUID
630
- removeOption(options, id) {
631
- const index = options.findIndex(option => option.id === id);
632
- if (index !== -1) {
633
- options.splice(index, 1);
634
- }
635
- }
636
- handleButtonClick() {
637
- this.formButtonHandler.emit(this.formBuilderService.downloadElement());
638
- }
639
- handleTemplateSave() {
640
- this.templateSaveHandler.emit(this.templateService.download());
641
- }
642
- // SKS13MAR25 active tab select
643
- setActiveTab(tab) {
644
- this.activeTab = tab;
645
- }
646
- // SKS13MAR25 table property save
647
- onSave() {
648
- // console.log('Saving data:');
649
- }
650
- // SKS13MAR25 table property reset
651
- onCancel() {
652
- // console.log('Operation cancelled');
653
- }
654
- updateProperties(elementType) {
655
- // Reset alignment and styles if switching elements
656
- this.selectedAlign = 'align-left';
657
- this.selectedStyles = [];
658
- // Fetch the properties of the selected element type
659
- const properties = this.getProperties();
660
- if (properties) {
661
- // Make sure the properties are dynamically updated
662
- }
663
- }
664
- onAlignSelect(value) {
665
- if (this.selectedElement) {
666
- this.selectedElement.textAlign = value;
667
- }
668
- }
669
- onStyleSelect(value) {
670
- if (!this.selectedElement)
671
- return;
672
- // Initialize styles array if it doesn't exist
673
- this.selectedElement.styles = this.selectedElement.styles || [];
674
- // Ensure only one style is selected at a time
675
- this.selectedElement.styles = [value];
676
- }
677
- isStyleActive(value) {
678
- return this.selectedElement?.styles?.includes(value) || false;
679
- }
680
- getValueByPath(path) {
681
- if (this.selectedElement?.type === 'Table') {
682
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);
683
- }
684
- this.selectedElement['subText'] =
685
- typeof this.selectedElement?.subText === 'object'
686
- ? this.selectedElement.subText
687
- : (typeof this.selectedElement?.subText === 'string' && this.selectedElement.subText.trim() !== ''
688
- ? JSON.parse(this.selectedElement.subText)
689
- : {});
690
- if (this.selectedElement['subText'] && Array.isArray(this.selectedElement['subText'].field)) {
691
- // Convert array to a comma-separated string
692
- this.selectedElement['subText'].field = this.selectedElement['subText'].field.join(', ');
693
- }
694
- return path.split('.').reduce((obj, key) => obj?.[key] ?? '', this.selectedElement);
695
- }
696
- isAnotherIdSelected() {
697
- return this.formBuilderService.getElements()
698
- .some((el, i) => i !== this.selectedElementIndex && el.primaryKey);
699
- }
700
- onToggleChange(path, event) {
701
- const checked = event.target.checked;
702
- this.setValueByPath(path, checked);
703
- if (this.selectedElement?.type === 'Table') {
704
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? JSON.stringify(this.selectedElement?.fieldsMeta) : this.selectedElement['fieldsMeta'] || [];
705
- }
706
- this.updateElement(this.selectedElement);
707
- }
708
- setValueByPath(path, value) {
709
- if (this.selectedElementIndex >= 0) {
710
- if (path === 'questionText' || value === 'question') {
711
- if (!this.validateInput(value, value)) {
712
- }
713
- }
714
- if (path === 'questionNumber') {
715
- //AP-10MAR25 Convert the value to an integer
716
- const newOrder = parseInt(value, 10);
717
- if (isNaN(newOrder))
718
- return;
719
- //AP-10MAR25 Update the element's questionNumber in the form builder service
720
- this.formBuilderService.updateElement(this.selectedElementIndex, { questionNumber: newOrder });
721
- //AP-10MAR25 Sort elements based on the updated questionNumber
722
- this.formBuilderService.sortElementsByOrder();
723
- }
724
- if (path === 'fontWeight') {
725
- // console.log('Font weight selected:', value);
726
- this.selectedElement[path] = value;
727
- }
728
- // Special handling for font and font weight
729
- if (path === 'font') {
730
- this.selectedElement.font = value; // Directly update the selected element's font property
731
- }
732
- }
733
- const keys = path.split('.');
734
- let obj = this.selectedElement;
735
- if (this.selectedElement?.type === 'Table') {
736
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);
737
- }
738
- this.selectedElement['subText'] = typeof this.selectedElement?.subText === 'object' ? this.selectedElement?.subText : JSON.parse(this.selectedElement['subText']);
739
- keys.forEach((key, index) => {
740
- if (!obj[key]) {
741
- // Check if the next key is a number (array index)
742
- const nextKey = keys[index + 1];
743
- obj[key] = isNaN(Number(nextKey)) ? {} : [];
744
- }
745
- if (index === keys.length - 1) {
746
- obj[key] = value; // Assign value to last key
747
- }
748
- else {
749
- obj = obj[key]; // Move deeper
750
- }
751
- });
752
- this.formBuilderService.elementUpdate(this.selectedElementIndex, this.selectedElement);
753
- }
754
- getStyleKeys() {
755
- if (!this.selectedElement || !this.selectedElement.style || typeof this.selectedElement.style !== 'object') {
756
- return [];
757
- }
758
- return Object.keys(this.selectedElement.style);
759
- }
760
- // SKS19MAR25 update element in servies
761
- updateElement(element) {
762
- this.formBuilderService.elementUpdate(this.selectedElementIndex, element);
763
- if (this.selectedElement?.type === 'Table' && this.selectColumn) {
764
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement.fieldsMeta || []);
765
- const foundItem = this.selectedElement['fieldsMeta'].find(item => item['uniqueIdentifier'] === this.selectColumn);
766
- this.formBuilderService.setSelectedTableElement(this.selectedElementIndex, { column: foundItem?.apiName || null });
767
- }
768
- }
769
- // SKS19MAR25 get value from array
770
- getValueByArrayPath(targetArrayPath, targetArrayKey, selectColumn, keyPath) {
771
- if (this.selectedElement?.type === 'Table') {
772
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);
773
- }
774
- const targetArray = this.getDataByPath(targetArrayPath);
775
- if (!Array.isArray(targetArray))
776
- return '';
777
- const foundItem = targetArray.find(item => item[targetArrayKey] === selectColumn);
778
- return foundItem ? keyPath.split('.').reduce((obj, key) => obj?.[key], foundItem) : '';
779
- }
780
- //SKS19MAR25 Function to update a value in an array dynamically
781
- updateValueByArrayPath(targetArrayPath, targetArrayKey, selectColumn, keyPath, event) {
782
- if (this.selectedElement?.type === 'Table') {
783
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);
784
- }
785
- const targetArray = this.getDataByPath(targetArrayPath);
786
- // console.log("targetArray", targetArray, "targetArrayKey", targetArrayKey, "selectColumn", selectColumn, "keyPath", keyPath, "event", event)
787
- if (!Array.isArray(targetArray))
788
- return;
789
- // Find the object that matches selectColumn
790
- let foundItem = targetArray.find(item => item[targetArrayKey] === selectColumn);
791
- // If the item does not exist, create and add it
792
- if (!foundItem) {
793
- foundItem = { [targetArrayKey]: selectColumn }; // Ensure the key exists
794
- targetArray.push(foundItem);
795
- }
796
- // Traverse and set the value
797
- const keys = keyPath.split('.');
798
- let obj = foundItem;
799
- for (let i = 0; i < keys.length - 1; i++) {
800
- const key = keys[i];
801
- if (!obj[key])
802
- obj[key] = {}; // Initialize missing objects
803
- obj = obj[key];
804
- }
805
- // Assign the value based on input type
806
- const lastKey = keys[keys.length - 1];
807
- const inputElement = event.target;
808
- obj[lastKey] = inputElement?.type === 'checkbox' ? inputElement.checked : inputElement.value;
809
- if (keyPath === 'summaryColumn') {
810
- if (obj['summaryRow'] === true) {
811
- obj['summaryRow'] = false;
812
- }
813
- if (inputElement.checked === true) {
814
- obj['fldType'] = 'calculation';
815
- }
816
- else {
817
- obj['fldType'] = 'Text';
818
- }
819
- }
820
- if (keyPath === 'summaryRow') {
821
- if (obj['summaryColumn'] === true) {
822
- obj['summaryColumn'] = false;
823
- }
824
- if (inputElement.checked === true) {
825
- obj['fldType'] = 'calculation';
826
- }
827
- else {
828
- obj['fldType'] = 'Text';
829
- }
830
- }
831
- if (this.selectedElement?.type === 'Table') {
832
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? JSON.stringify(this.selectedElement?.fieldsMeta) : this.selectedElement['fieldsMeta'] || [];
833
- }
834
- this.updateElement(this.selectedElement);
835
- }
836
- getDataByPath(path) {
837
- if (this.selectedElement?.type === 'Table') {
838
- this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);
839
- }
840
- return path.split('.').reduce((obj, key) => obj?.[key], this.selectedElement);
841
- }
842
- // SKS25MAR25 image edit functions
843
- flipHorizontal() {
844
- this.transform = {
845
- ...this.transform,
846
- flipH: !this.transform.flipH
847
- };
848
- }
849
- flipVertical() {
850
- this.transform = {
851
- ...this.transform,
852
- flipV: !this.transform.flipV
853
- };
854
- }
855
- resetImage() {
856
- this.canvasRotation = 0;
857
- this.cropper = undefined;
858
- this.transform = {
859
- translateUnit: 'px',
860
- scale: 1,
861
- rotate: 0,
862
- flipH: false,
863
- flipV: false,
864
- translateH: 0,
865
- translateV: 0
866
- };
867
- }
868
- zoomOut() {
869
- this.transform = {
870
- ...this.transform,
871
- scale: this.transform.scale - .1
872
- };
873
- }
874
- zoomIn() {
875
- this.transform = {
876
- ...this.transform,
877
- scale: this.transform.scale + .1
878
- };
879
- }
880
- rotateLeft() {
881
- this.loading = true;
882
- setTimeout(() => {
883
- this.canvasRotation--;
884
- this.flipAfterRotate();
885
- });
886
- }
887
- rotateRight() {
888
- this.loading = true;
889
- setTimeout(() => {
890
- this.canvasRotation++;
891
- this.flipAfterRotate();
892
- });
893
- }
894
- moveLeft() {
895
- this.transform = {
896
- ...this.transform,
897
- translateH: this.transform.translateH - 1
898
- };
899
- }
900
- moveRight() {
901
- this.transform = {
902
- ...this.transform,
903
- translateH: this.transform.translateH + 1
904
- };
905
- }
906
- moveDown() {
907
- this.transform = {
908
- ...this.transform,
909
- translateV: this.transform.translateV + 1
910
- };
911
- }
912
- moveUp() {
913
- this.transform = {
914
- ...this.transform,
915
- translateV: this.transform.translateV - 1
916
- };
917
- }
918
- flipAfterRotate() {
919
- const flippedH = this.transform.flipH;
920
- const flippedV = this.transform.flipV;
921
- this.transform = {
922
- ...this.transform,
923
- flipH: flippedV,
924
- flipV: flippedH,
925
- translateH: 0,
926
- translateV: 0
927
- };
928
- }
929
- async imageCropped(event) {
930
- try {
931
- const base64 = await this.convertBlobToBase64(event.objectUrl);
932
- this.selectedElement.imageData = base64;
933
- await this.formBuilderService.elementUpdate(this.selectedElementIndex, this.selectedElement);
934
- // console.log('CROPPED', event);
935
- }
936
- catch (error) {
937
- console.error("Error in imageCropped:", error);
938
- }
939
- }
940
- cropperReady(sourceImageDimensions) {
941
- // console.log('Cropper ready', sourceImageDimensions);
942
- this.loading = false;
943
- }
944
- // SKS25MAR25 blob to base 64 converter
945
- convertBlobToBase64(objectUrl) {
946
- return fetch(objectUrl)
947
- .then(response => response.blob())
948
- .then(blob => {
949
- return new Promise((resolve, reject) => {
950
- const reader = new FileReader();
951
- reader.readAsDataURL(blob);
952
- reader.onloadend = () => resolve(reader.result);
953
- reader.onerror = error => reject(error);
954
- });
955
- });
956
- }
957
- // SKS28MAR25 qb book update in book element
958
- childEventCapture(event) {
959
- this.selectedElement.qbReference = event.valueObj?.name;
960
- this.selectedElement.qbReferenceQuestions = event.valueObj?.jsonBody;
961
- this.formBuilderService.elementUpdate(this.selectedElementIndex, this.selectedElement);
962
- }
963
- // SKS28MAR25 book search dropdown emit
964
- linkToggleDropdown(event) {
965
- event.stopPropagation();
966
- this.isLinkDropdownOpen = !this.isLinkDropdownOpen;
967
- }
968
- // SKS28MAR25 book search dropdown close
969
- onClickOutside(event) {
970
- if (this.dropdown && !this.dropdown.nativeElement.contains(event.target)) {
971
- this.isLinkDropdownOpen = false;
972
- }
973
- }
974
- // AP-28MAR25 When drag starts, store the index
975
- onDragStart(event, id) {
976
- this.draggedId = id;
977
- event.dataTransfer.effectAllowed = "move";
978
- }
979
- // AP-28MAR25 Prevent default behavior to allow drop
980
- onDragOver(event) {
981
- event.preventDefault();
982
- }
983
- // AP-28MAR25 Swap the dragged item with the dropped position
984
- onDrop(event, key) {
985
- event.preventDefault();
986
- const targetId = event.target.closest(".option-items")?.getAttribute("data-id");
987
- if (this.draggedId && targetId && this.draggedId !== targetId) {
988
- const options = this.selectedElement[key];
989
- [options[this.draggedId], options[targetId]] = [options[targetId], options[this.draggedId]];
990
- // Find the indexes of the dragged and target options
991
- const draggedIndex = options.findIndex(option => option.id === this.draggedId);
992
- const targetIndex = options.findIndex(option => option.id === targetId);
993
- // Swap the items
994
- if (draggedIndex !== -1 && targetIndex !== -1) {
995
- [options[draggedIndex], options[targetIndex]] = [options[targetIndex], options[draggedIndex]];
996
- }
997
- }
998
- this.draggedId = null;
999
- }
1000
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PropertiesComponent, deps: [{ token: i1.HttpClient }, { token: i2.FormBuilderService }, { token: i3.TemplateService }], target: i0.ɵɵFactoryTarget.Component });
1001
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: PropertiesComponent, isStandalone: true, selector: "app-properties", inputs: { selectedElementType: "selectedElementType", templateSelected: "templateSelected" }, outputs: { formButtonHandler: "formButtonHandler", templateSaveHandler: "templateSaveHandler" }, host: { listeners: { "document:click": "onClickOutside($event)" } }, viewQueries: [{ propertyName: "dropdown", first: true, predicate: ["dropdown"], descendants: true }], ngImport: i0, template: "<!-- AP 22JAN25 - Field and Element Properties -->\n<div class=\"container\">\n <div class=\"tabs\">\n <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'attributes'}\" (click)=\"setActiveTab('attributes')\">\n Attributes\n </div>\n <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'property'}\" (click)=\"setActiveTab('property')\">\n Property\n </div>\n <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'appearance'}\" (click)=\"setActiveTab('appearance')\">\n Appearance\n </div>\n </div>\n\n <div class=\"tab-content\" *ngIf=\"activeTab === 'attributes'\">\n <!-- Element Properrties -->\n <!-- AP-06MAR25 Select element type show -->\n <div *ngIf=\"getProperties()\">\n <!-- SKS25MAR25 this is for image edit -->\n <div *ngIf=\"selectedElement?.type === 'Image'\">\n <image-cropper *ngIf=\"selectedElement.imageData \" [imageBase64]=\"selectedElement.orgImageData\"\n [disabled]=\"false\" [alignImage]=\"alignImage\" [roundCropper]=\"roundCropper\" [backgroundColor]=\"'white'\"\n imageAltText=\"Alternative image text\" [allowMoveImage]=\"false\" [hideResizeSquares]=\"false\"\n [canvasRotation]=\"canvasRotation\" [aspectRatio]=\"aspectRatio\" [containWithinAspectRatio]=\"false\"\n [maintainAspectRatio]=\"false\" [cropperStaticWidth]=\"cropperStaticWidth\"\n [cropperStaticHeight]=\"cropperStaticHeight\" [cropperMinWidth]=\"cropperMinWidth\"\n [cropperMinHeight]=\"cropperMinHeight\" [cropperMaxWidth]=\"cropperMaxWidth\"\n [cropperMaxHeight]=\"cropperMaxHeight\" [resetCropOnAspectRatioChange]='true' [(cropper)]=\"cropper\"\n [(transform)]=\"transform\" [onlyScaleDown]=\"true\" output=\"blob\" format=\"png\"\n (imageCropped)=\"imageCropped($event)\" (cropperReady)=\"cropperReady($event)\"></image-cropper>\n <div *ngIf=\"selectedElement.imageData \" style=\"display: flex; gap: 2px;\">\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateLeft()\" title=\"Rotate Left\">\u27F2</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateRight()\" title=\"Rotate Right\">\u27F3</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomOut()\" title=\"Zoom Out\">-</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomIn()\" title=\"Zoom In\">+</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveLeft()\" title=\"Move Left\">\u2190</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveRight()\" title=\"Move Right\">\u2192</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveUp()\" title=\"Move Up\">\u2191</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveDown()\" title=\"Move Down\">\u2193</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipHorizontal()\" [class.enabled]=\"transform.flipH\"\n title=\"Flip Horizontally\">\u2194</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipVertical()\" [class.enabled]=\"transform.flipV\"\n title=\"Flip Vertically\">\u2195</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"resetImage()\" title=\"Reset\">\u00D7</div>\n </div>\n </div>\n <!-- SKS28MAR25 search book specific for book element -->\n <div *ngIf=\"selectedElement?.type === 'Book'\">\n <label class=\"text-sm\">Search Book</label>\n <div style=\"display: flex; gap: 2px; align-items: center; justify-content: center;\">\n <nxt-search-box [question]=\"selectedElement\" [apiMeta]=\"bookSubtext\" [placeHolderText]=\"'Search...'\"\n (searchValueChange)=\"childEventCapture($event.value)\">\n </nxt-search-box>\n <div class=\"link-icon\">\n <svg (click)=\"linkToggleDropdown($event)\" fill=\"#000000\" version=\"1.1\" id=\"Capa_1\"\n xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" width=\"24px\" height=\"24px\"\n viewBox=\"0 0 450 450\" xml:space=\"preserve\">\n <g>\n <g>\n <g>\n <path d=\"M318.15,230.195l77.934-77.937c31.894-31.892,31.894-83.782-0.004-115.674l-12.66-12.66\n c-31.893-31.896-83.78-31.896-115.674-0.004l-77.937,77.934c-17.588,17.588-25.457,41.264-23.646,64.311\n c-23.045-1.813-46.722,6.056-64.308,23.647L23.92,267.748c-31.894,31.889-31.894,83.779,0,115.674l12.664,12.662\n c31.893,31.893,83.783,31.893,115.674,0l77.935-77.936c17.592-17.59,25.459-41.266,23.647-64.309\n C276.884,255.654,300.56,247.783,318.15,230.195z M202.653,290.605l-77.936,77.938c-16.705,16.703-43.889,16.703-60.59,0\n l-12.666-12.666c-16.705-16.701-16.703-43.885,0-60.594l77.936-77.932c14.14-14.141,35.779-16.306,52.226-6.516l-32.302,32.307\n c-7.606,7.604-7.606,19.938,0,27.541c7.605,7.607,19.937,7.607,27.541,0l32.306-32.303\n C218.959,254.828,216.795,276.469,202.653,290.605z M238.382,209.169l32.299-32.306c7.608-7.602,7.608-19.935,0-27.538\n c-7.604-7.61-19.936-7.61-27.541-0.004l-32.303,32.303c-9.791-16.446-7.627-38.087,6.514-52.226l77.935-77.935\n c16.707-16.707,43.89-16.707,60.594,0l12.664,12.664c16.705,16.705,16.705,43.886,0,60.591l-77.936,77.937\n C276.468,216.797,254.828,218.959,238.382,209.169z\" />\n <path d=\"M343.466,261.465c-45.287,0-82,36.713-82,82s36.713,82,82,82c45.286,0,82-36.713,82-82S388.753,261.465,343.466,261.465z\n M372.505,333.564l-56.046,56.104c-0.239,0.238-0.536,0.41-0.862,0.496l-22.315,5.85c-0.649,0.168-1.347-0.02-1.822-0.494\n c-0.477-0.479-0.666-1.172-0.496-1.824l5.826-22.318c0.084-0.326,0.256-0.627,0.494-0.863l56.047-56.104\n c0.742-0.742,1.945-0.744,2.688-0.002l4.548,4.541c0.739,0.74,0.741,1.943,0,2.688l-37.433,37.471l4.709,4.703l37.435-37.471\n c0.739-0.742,1.94-0.742,2.682-0.002l4.55,4.541C373.25,331.617,373.25,332.822,372.505,333.564z M395.472,310.574l-17,17.018\n c-0.739,0.744-1.942,0.744-2.685,0.002l-16.489-16.475c-0.744-0.74-0.744-1.943-0.002-2.688l17-17.02\n c0.741-0.74,1.944-0.74,2.688-0.002l16.487,16.477C396.216,308.629,396.216,309.832,395.472,310.574z\" />\n </g>\n </g>\n </g>\n </svg>\n <div class=\"link-dropdown-menu\" *ngIf=\"isLinkDropdownOpen\" #dropdown>\n <label>Endpoint:</label>\n <input type=\"text\" [(ngModel)]=\"bookSubtext.endpoint\" />\n\n <label>Variable:</label>\n <input type=\"text\" [(ngModel)]=\"bookSubtext.variable\" />\n\n <label>Field:</label>\n <input type=\"text\" [ngModel]=\"fieldAsString\" (ngModelChange)=\"updateField($event)\" />\n\n <label>Default Field:</label>\n <input type=\"text\" [(ngModel)]=\"bookSubtext.defaultField\" />\n </div>\n </div>\n </div>\n </div>\n <ng-container *ngFor=\"let prop of getProperties().elementProps\">\n <div class=\"form-group\">\n <label *ngIf=\"prop.type !== 'checkbox' && prop.type !== 'subQuestion'\" class=\"text-sm\">{{ prop.label\n }}</label>\n\n <!-- Text Input -->\n <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n [value]=\"headerSelect ? book.records[0].title : (selectedElement?.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ) : getValueByPath(prop.key))\"\n (input)=\"headerSelect ? updateTitle($event.target.value) : (selectedElement?.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : setValueByPath(prop.key, $event.target.value) ) : setValueByPath(prop.key, $event.target.value))\"\n [class.read-only]=\"selectedElement.readOnly\" [readonly]=\"selectedElement.readOnly\" />\n\n <!-- help text -->\n <div *ngIf=\"prop.key === 'helpText' && selectedElement.helpText\">{{\n selectedElement.helpText }}</div>\n\n <!-- questionNumber -->\n <input *ngIf=\"prop.key === 'questionNumber'\" type=\"number\" [value]=\"selectedElement.questionNumber\"\n (input)=\"setValueByPath('questionNumber', $event.target.value)\" />\n\n <!-- file -->\n <!-- Add this inside the elementProps loop where other inputs are rendered -->\n <select *ngIf=\"prop.type === 'select' && prop.key === 'supportType'\" [value]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option value=\"\">Select file category</option>\n <option *ngFor=\"let option of prop.options\" [value]=\"option.value\"> {{ option.label }} </option>\n </select>\n\n <div *ngIf=\"prop.type === 'checkbox'\">\n\n <input type=\"checkbox\"\n [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key) \"\n (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n <span class=\"toggle-label\" style=\"padding-left: 10px;\">{{ prop.label }}</span>\n\n </div>\n\n <!--AP-11JUN25 - Show this section only if the property key is 'primaryKey' -->\n <div *ngIf=\"prop.key === 'primaryKey'\">\n <input type=\"checkbox\" [checked]=\"getValueByPath('primaryKey')\" [disabled]=\"!getValueByPath('primaryKey') && isAnotherIdSelected()\" (change)=\"onToggleChange('primaryKey', $event)\"/>\n <span class=\"toggle-label\" style=\"padding-left: 10px;\">primaryKey</span>\n </div>\n\n <!-- SKS20MAR25 Subquestion Type -->\n <div *ngIf=\"prop.type === 'subQuestion'\">\n <div style=\"display: flex; flex-direction: row; gap: 10px; align-items: center;\">\n <div>{{ prop.label }}</div>\n <input type=\"checkbox\"\n [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key) \"\n (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n </div>\n <!-- SKS20MAR25 Render subquestions when checkbox is checked -->\n <div\n *ngIf=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key)\">\n <div *ngFor=\"let subProp of prop.subQuestion\"\n style=\"background-color: #e7f2ff; padding: 8px; border-radius: 4px;\">\n <div class=\"form-group\">\n <label>{{ subProp.label }}</label>\n <div *ngIf=\"subProp.type === 'array'\">\n <!-- Iterate over filtered columns to display checkboxes -->\n <div *ngFor=\"let column of filteredColumns\">\n <label>\n <input type=\"checkbox\"\n [checked]=\"subProp.operands ? subProp.operands.includes(column.apiName) : false\"\n (change)=\"onCheckboxChange(subProp.targetArray,subProp.targetArrayKey,subProp.key, column.apiName, $event.target.checked)\">\n {{ column.apiName }}\n </label>\n </div>\n </div>\n <!-- SKS21MAR25 New radio type -->\n <div *ngIf=\"subProp.type === 'radio'\">\n <div *ngFor=\"let column of filteredColumns\">\n <label>\n <input type=\"radio\" [name]=\"subProp.key\" [value]=\"column.apiName\"\n [checked]=\"getValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key) === column.apiName\"\n (change)=\"onRadioChange(subProp.targetArray, subProp.targetArrayKey, subProp.key, column.apiName)\">\n {{ column.apiName }}\n </label>\n </div>\n </div>\n <!-- SKS20MAR25 Add more subproperty types as needed -->\n <!-- Inside the subProp ngFor loop -->\n <input *ngIf=\"subProp.type === 'text'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n [value]=\"selectedElement.type === 'Table' ? (subProp.targetArray ? getValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key ) : getValueByPath(subProp.key) ) : getValueByPath(subProp.key)\"\n (input)=\"selectedElement.type === 'Table' ? (subProp.targetArray ? updateValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key, $event) : setValueByPath(subProp.key, $event.target.value) ) : setValueByPath(subProp.key, $event.target.value)\" />\n\n <input *ngIf=\"subProp.type === 'boolean'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n <select *ngIf=\"subProp.type === 'select'\" [value]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.value)\">\n <option *ngFor=\"let option of subProp.options\" [value]=\"option\">{{ option }}</option>\n </select>\n <input *ngIf=\"subProp.type === 'checkbox'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n </div>\n </div>\n </div>\n </div>\n\n <!-- SKS20MAR25 Text Align Buttons -->\n <div *ngIf=\"prop.type === 'align'\">\n <button *ngFor=\"let option of prop.options\" (click)=\"onAlignSelect(option.value)\"\n [class.active]=\"selectedElement?.textAlign === option.value\" [title]=\"option.value\">\n <img [src]=\"'../assets/icons/' + option.icon + '.svg'\" [alt]=\"option.value\" class=\"icon-size\" />\n </button>\n </div>\n\n <div *ngIf=\"prop.type === 'style'\">\n <button *ngFor=\"let option of prop.options\" (click)=\"onStyleSelect(option.value)\"\n [class.active]=\"isStyleActive(option.value)\" [title]=\"option.value\">\n <img [src]=\"'../assets/icons/' + option.icon + '.svg'\" [alt]=\"option.value\" class=\"icon-size\" />\n </button>\n </div>\n\n <!-- Field Size Controls -->\n <!-- AP 25FEB25 - Change key size -->\n <div *ngIf=\"prop.key === 'size'\">\n <input type=\"number\" [value]=\"selectedElement?.size\" (input)=\"setValueByPath('size', $event.target.value)\"\n class=\"size-input\" />\n </div>\n\n <!-- AP-19MAR25 Line Properties -->\n <!-- Padding Top -->\n <div *ngIf=\"prop.key === 'paddingTop'\">\n <input type=\"number\" [placeholder]=\"prop.placeholder\" [value]=\"selectedElement?.paddingTop\"\n (input)=\"setValueByPath('paddingTop', $event.target.value)\" />\n </div>\n\n <!-- Padding Bottom -->\n <div *ngIf=\"prop.key === 'paddingBottom'\">\n <input type=\"number\" [placeholder]=\"prop.placeholder\" [value]=\"selectedElement?.paddingBottom\"\n (input)=\"setValueByPath('paddingBottom', $event.target.value)\" />\n </div>\n\n <!-- Line Style -->\n <div *ngIf=\"prop.key === 'lineStyle'\">\n <select [value]=\"selectedElement?.lineStyle\" (change)=\"setValueByPath('lineStyle', $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option\">{{ option }}</option>\n </select>\n </div>\n\n <!-- Color -->\n <div *ngIf=\"prop.key === 'color'\">\n <input type=\"color\" [value]=\"selectedElement?.color\"\n (input)=\"setValueByPath('color', $event.target.value)\" />\n </div>\n\n </div>\n \n <!-- AP 23MAY25 - Action Button -->\n <div *ngIf=\"prop.key === 'title'\" style=\"margin-top: 15px; border: 1px solid #ddd; padding: 20px; border-radius: 12px; background-color: #f9f9f9;\">\n <div class=\"form-header\">\n <label class=\"form-label\">{{ prop.label }}</label>\n <button (click)=\"addNewButton()\" class=\"add-button\">+ Add Button</button>\n </div>\n \n <div *ngFor=\"let btn of book?.questionbook?.action || []; let i = index\" class=\"button-config-card\">\n <div class=\"form-group\">\n <label>Button Name</label>\n <input type=\"text\" [(ngModel)]=\"btn.name\" (ngModelChange)=\"onButtonPropertyChange(i, 'name', $event)\" placeholder=\"Enter button name\">\n </div>\n \n <div class=\"form-group\">\n <label>Event Name</label>\n <input type=\"text\" [(ngModel)]=\"btn.eventtoemit\" (ngModelChange)=\"onButtonPropertyChange(i, 'eventtoemit', $event)\" placeholder=\"Event to emit\">\n </div>\n \n <div class=\"form-group\">\n <label>Alt</label>\n <input type=\"text\" [(ngModel)]=\"btn.alt\" (ngModelChange)=\"onButtonPropertyChange(i, 'alt', $event)\" placeholder=\"Button alt text\">\n </div>\n \n <div class=\"form-group\">\n <label>Endpoint</label>\n <input type=\"text\" [(ngModel)]=\"btn.endpoint\" (ngModelChange)=\"onButtonPropertyChange(i, 'endpoint', $event)\" placeholder=\"API endpoint\">\n </div>\n \n <div class=\"form-group\">\n <label>Method</label>\n <select [(ngModel)]=\"btn.method\" (ngModelChange)=\"onButtonPropertyChange(i, 'method', $event)\">\n <option [value]=\"'GET'\">GET</option>\n <option [value]=\"'POST'\">POST</option>\n <option [value]=\"'PUT'\">PUT</option>\n <option [value]=\"'DELETE'\">DELETE</option>\n </select>\n </div>\n \n <!-- AP 23MAY25 - Action Style Fields -->\n <div class=\"form-group\">\n <label>Background Color</label>\n <input type=\"color\" [(ngModel)]=\"btn.bgColor\" (ngModelChange)=\"onButtonPropertyChange(i, 'bgColor', $event)\">\n </div>\n \n <div class=\"form-group\">\n <label>Border Radius</label>\n <input type=\"range\" min=\"0\" max=\"50\" [(ngModel)]=\"btn.borderRadius\" \n (ngModelChange)=\"onButtonPropertyChange(i, 'borderRadius', $event)\">\n <span>{{ btn.borderRadius }}px</span>\n </div> \n \n <div class=\"form-group\">\n <label>Button Width (px)</label>\n <input type=\"number\" [(ngModel)]=\"btn.width\"\n (ngModelChange)=\"onButtonPropertyChange(i, 'width', $event)\" min=\"50\" placeholder=\"Enter width in px\">\n </div>\n \n <div class=\"form-group\">\n <label>Text Color</label>\n <input type=\"color\" [(ngModel)]=\"btn.textColor\"\n (ngModelChange)=\"onButtonPropertyChange(i, 'textColor', $event)\">\n </div>\n \n <div class=\"form-group\">\n <label>Position (%)</label>\n <input type=\"range\" min=\"0\" max=\"100\" [(ngModel)]=\"btn.positionPercent\"\n (ngModelChange)=\"onButtonPropertyChange(i, 'positionPercent', $event)\">\n <span>{{ btn.positionPercent }}%</span>\n </div>\n <button (click)=\"removeButton(btn)\" class=\"remove-button\">\u00D7 Remove</button>\n </div>\n </div>\n \n </ng-container>\n </div>\n </div>\n <div class=\"tab-content\" *ngIf=\"activeTab === 'property'\">\n <!-- Field Elements Properties -->\n <!-- AP-06MAR25 Show elements ID -->\n <div *ngIf=\"getProperties()\">\n <label>Label Id</label>\n <div\n style=\"font-size: 13px; padding: 11px; border-radius: 5px; background-color: #f8f8f8; border: 1px solid #ddd;\">\n {{ headerSelect ? bookId : selectedElement.id }}</div>\n <ng-container *ngFor=\"let prop of getProperties().fieldProps\">\n <div class=\"form-group\">\n <label class=\"text-sm\">{{ prop.label }}</label>\n\n <!-- SKS21MAR25 Toggle Group -->\n <div *ngIf=\"prop.type === 'toggleGroup'\" class=\"toggle-group\">\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" [checked]=\"selectedElement?.isOptional\"\n (change)=\"onRequiredChange($event.target.checked)\" />\n Required\n </label>\n </div>\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" [checked]=\"selectedElement?.isReadOnly\"\n (change)=\"onToggleChange('isReadOnly', $event)\" />\n Read Only\n </label>\n </div>\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" />\n Disabled\n </label>\n </div>\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" [checked]=\"selectedElement?.isHidden\"\n (change)=\"onToggleChange('isHidden', $event)\" />\n Is Hide\n </label>\n </div>\n </div>\n\n <!-- AP 28FEB25 - handled options with UUID -->\n <div\n *ngIf=\"prop.type === 'dropdown' || prop.type === 'checkbox' || prop.type === 'radio' && prop.key === 'options'\"\n class=\"options-container\">\n\n <div class=\"option-list\" (dragover)=\"onDragOver($event)\" (drop)=\"onDrop($event, prop.key)\">\n <div *ngFor=\"let option of selectedElement[prop.key]\" class=\"option-items\" [attr.data-id]=\"option.id\"\n draggable=\"true\" (dragstart)=\"onDragStart($event, option.id)\">\n\n <input type=\"text\" [(ngModel)]=\"option.value\" placeholder=\"Option\" class=\"options\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeOption(selectedElement[prop.key], option.id)\">\n <span class=\"drag-handle\">\u2630</span>\n </div>\n </div>\n\n <button (click)=\"addOption(selectedElement[prop.key])\">\n <div class=\"add-varient\">\n <span class=\"text-lg\">+</span>\n <span>Add</span>\n </div>\n </button>\n </div>\n\n <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n [value]=\"headerSelect ? book.records[0].title : (selectedElement?.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ) : getValueByPath(prop.key))\"\n (input)=\"headerSelect ? updateTitle($event.target.value) : (selectedElement?.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : setValueByPath(prop.key, $event.target.value) ) : setValueByPath(prop.key, $event.target.value))\"\n [class.read-only]=\"selectedElement.readOnly\" [readonly]=\"selectedElement.readOnly\" />\n\n <!-- Sub Questions Toggle -->\n <div *ngIf=\"prop.type === 'subQuestion'\">\n <div class=\"style-toggle-header\" (click)=\"toggleSubQuestion(prop)\">\n <div class=\"head-elements\">Sub Text</div>\n <img [src]=\"prop.isExpanded ? '../assets/icons/arrow-down.svg' : '../assets/icons/arrow-right.svg'\"\n alt=\"Toggle Arrow\" class=\"arrow-icon\">\n </div>\n\n <!-- Render subquestions when arrow is down -->\n <div *ngIf=\"prop.isExpanded\" style=\"border: 1px solid #ddd; padding: 8px; border-radius: 4px;\">\n <div *ngFor=\"let subProp of prop.subQuestion\" class=\"sub-question-container\">\n <div class=\"form-group\">\n <label>{{ subProp.label }}</label>\n\n <input *ngIf=\"subProp.type === 'text'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n [value]=\"getValueByPath(subProp.key)\" (input)=\"setValueByPath(subProp.key, $event.target.value)\" />\n \n <!-- AP-02APR25 Render input field only if subProp.type is 'array' -->\n <input *ngIf=\"subProp.type === 'array'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n [value]=\"getValueByPath(subProp.key)\" (input)=\"setValueByPath(subProp.key, $event.target.value)\" />\n <!-- <input type=\"text\" [ngModel]=\"fieldAsString\" (ngModelChange)=\"updateField($event)\" /> -->\n\n <input *ngIf=\"subProp.type === 'boolean'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"prop.type === 'checkbox'\">\n <span class=\"toggle-label\" style=\"padding-right: 10px;\">{{ prop.label }}</span>\n <input type=\"checkbox\"\n [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key) \"\n (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n\n </div>\n\n </div>\n </ng-container>\n\n <!-- SKS21MAR25 Move Style Outside the Loop -->\n <!-- Style Toggle -->\n <div class=\"style-toggle-header\" (click)=\"toggleStyleSection()\">\n <div class=\"head-elements\">Style</div>\n <img [src]=\"isStyleExpanded ? '../assets/icons/arrow-down.svg' : '../assets/icons/arrow-right.svg'\"\n alt=\"Toggle Arrow\" class=\"arrow-icon\">\n </div>\n\n <div *ngIf=\"isStyleExpanded\" style=\"border: 1px solid #ddd; padding: 8px; border-radius: 4px;\">\n <div class=\"form-group\">\n <div *ngFor=\"let key of getStyleKeys()\">\n <label>{{ key }}</label>\n <input type=\"text\" [(ngModel)]=\"selectedElement.style[key]\"\n (input)=\"setValueByPath('style.' + key, $event.target.value)\" placeholder=\"Enter {{ key }}\" />\n </div>\n </div>\n </div>\n\n </div>\n </div>\n <div class=\"tab-content\" *ngIf=\"activeTab === 'appearance'\">\n <div *ngIf=\"getProperties()\">\n <ng-container *ngFor=\"let prop of getProperties().appearance\">\n <div class=\"form-group\">\n <label class=\"text-sm\">{{ prop.label }}</label>\n <!-- Font Selection -->\n <div *ngIf=\"prop.key === 'font'\">\n <select *ngIf=\"prop.type === 'select'\" class=\"select-container\" [(ngModel)]=\"selectedElement.font\"\n (change)=\"setValueByPath('font', $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option\">\n {{ option }}\n </option>\n </select>\n </div>\n <!-- Input Box -->\n <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n [value]=\"selectedElement[prop.key]\" (input)=\"setValueByPath(prop.key, $event.target.value)\" />\n\n <!-- Font Size Selection -->\n <div *ngIf=\"prop.key === 'fontSize'\">\n <select *ngIf=\"prop.type === 'select'\" [(ngModel)]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option.value\">\n {{ option.label }}\n </option>\n </select>\n </div>\n <!-- Font Weight Selection -->\n <div *ngIf=\"prop.key === 'fontWeight'\">\n <select *ngIf=\"prop.type === 'select'\" [(ngModel)]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option.value\">\n {{ option.label }}\n </option>\n </select>\n </div>\n\n <div class=\"flex-container\">\n <!-- Dropdown Box -->\n <div *ngIf=\"prop.type === 'dropdown'\" class=\"input-box-field\">\n <select [(ngModel)]=\"selectedElement[prop.key]\" (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option\">\n {{ option }}\n </option>\n </select>\n </div>\n\n <!-- Color Picker -->\n <div *ngIf=\"prop.type === 'color'\" class=\"color-selector\">\n <input type=\"color\" [(ngModel)]=\"selectedElement.fontColor\"\n (change)=\"setValueByPath('fontColor', $event.target.value)\">\n </div>\n\n <!-- HEX Input Box -->\n <div *ngIf=\"prop.type === 'color'\" class=\"hex-input-container\">\n <span>HEX Code</span>\n <input type=\"text\" [(ngModel)]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\" />\n </div>\n </div>\n \n <!-- AP-17APR25 Button for Duplicate Field -->\n<div *ngIf=\"prop.type === 'button-toggle'\" class=\"button-toggle-wrapper\">\n <button type=\"button\"\n class=\"toggle-button\"\n (click)=\"duplicateField(selectedElement)\">\n Add Duplicate\n </button>\n</div>\n\n\n </div>\n </ng-container>\n </div>\n </div>\n\n<!-- AP-08APR25 Default Save Button -->\n<div class=\"button-container\" *ngIf=\"!templateSelected\">\n <button class=\"cancel-btn\" (click)=\"onCancel()\">Cancel</button>\n <button class=\"save-btn\" (click)=\"handleButtonClick()\">Save</button>\n</div>\n\n<!-- AP-08APR25 Template Save Button -->\n<div class=\"button-container\" *ngIf=\"templateSelected\" style=\"margin-top: 20px;\">\n <button class=\"save-btn\" (click)=\"handleTemplateSave()\">Save Template</button>\n</div>\n\n</div>", styles: ["*{margin:0;padding:0;box-sizing:border-box;font-family:Roboto,sans-serif}.properties{height:calc(100vh - 20px);overflow-y:auto}.container{width:100%;max-width:500px;margin:0 auto;background:#fff;box-shadow:0 2px 10px #0000001a;overflow:hidden;font-family:Arial,sans-serif}.design-header{display:flex;justify-content:center;align-items:center;padding:15px 20px;background:#fff;border-radius:8px;box-shadow:0 2px 10px #0000001a;margin-bottom:20px}.header-title{font-size:20px;font-weight:400;color:#222}.toggle-header,.style-toggle-header{display:flex;align-items:center;justify-content:space-between;cursor:pointer;padding:10px;background:#f8f8f8;border:1px solid #ddd;border-radius:6px;margin-bottom:6px}.tabs{display:flex;background:#f0f2f5;border-bottom:2px solid #0052cc}.tab{flex:1;padding:15px 10px;text-align:center;cursor:pointer}.tab.active{background:#0052cc;color:#fff;font-weight:700}.tab-content{padding:20px;max-height:80vh;overflow-y:auto;overflow-x:hidden}.form-group{margin-bottom:15px}label{display:block;font-size:14px;font-weight:500;color:#444;margin-bottom:8px}.required:before{content:\"*\";color:red;margin-right:3px}input[type=text],input[type=number],select,textarea{width:100%;padding:10px;font-size:14px;border:1px solid #ccc;border-radius:6px;outline:none;transition:all .3s;background:#f8f8f8}input:focus,select:focus,textarea:focus{border-color:#007bff;background:#fff;box-shadow:0 0 5px #007bff4d}textarea{min-height:55px;resize:vertical}button,.btn{padding:10px 15px;border:none;border-radius:6px;background:#007bff;color:#fff;font-size:15px;cursor:pointer;transition:all .3s}button:hover,.btn:hover{background:#0056b3;transform:translateY(-2px)}.add-button{padding:8px 14px;background:#0954c5;border-radius:8px}.remove-button{background:#e74c3c;padding:6px 12px;margin-top:10px}.remove-button:hover{background:#c0392b}.cancel-btn{background:#fff;color:#666;border:1px solid #ddd;margin-right:10px}.save-btn{background:#0052cc}.toggle-group{display:grid;grid-template-columns:1fr 1fr;gap:10px}.toggle-item{display:flex;align-items:center;gap:10px;padding:8px}.switch{position:relative;width:42px;height:22px}.switch input{opacity:0;width:0;height:0}.slider{position:absolute;cursor:pointer;inset:0;background:#ccc;transition:.4s;border-radius:24px}.slider:before{position:absolute;content:\"\";height:20px;width:20px;left:1px;bottom:1px;background:#fff;transition:.4s;border-radius:50%}input:checked+.slider{background:#2196f3}input:checked+.slider:before{transform:translate(18px)}.radio-item{display:flex;align-items:center;gap:20px}.radio-item input{accent-color:#007bff}.checkbox-row{display:flex;margin-bottom:10px}.checkbox-group{display:flex;align-items:center;margin-right:20px;min-width:120px}.checkbox-group input[type=checkbox]{margin-right:5px}.color-picker-row,.color-picker-container{display:flex;align-items:center}.color-picker-container{border:1px solid #ddd;border-radius:4px;margin-right:10px}.color-box,.color-selector input[type=color]{width:78px;height:40px;border:none;border-radius:4px;cursor:pointer}.hex-input-container{display:flex;gap:20px;align-items:center;font-size:12px;color:#b0b0b0}.hex-input-container span{font-size:14px;margin-top:10px}.hex-input,.hex-input-container input[type=text]{width:120px;padding:10px 12px;border:1px solid #d1d1d1;color:#28343e}.hex-label{color:#999;margin-right:5px}.select-container,.input-box-field{position:relative;width:100%}.dropdown-arrow,.input-box-field:after{position:absolute;right:10px;top:50%;transform:translateY(-50%);pointer-events:none;color:#666;font-size:12px}.input-box-field:after{content:\"\\25bc\";color:#1c1b1f;font-size:10px}.input-box-field select{background:#28343e;color:#fff;border:none;appearance:none;-webkit-appearance:none;-moz-appearance:none}.input-box-field select::-ms-expand{display:none}.all-properties details{background:#fff;border:1px solid #ddd;border-radius:8px;margin-bottom:12px;padding:12px;box-shadow:0 2px 8px #0000000d}.all-properties summary{font-size:15px;font-weight:400;cursor:pointer;padding:6px;transition:color .3s}.all-properties summary:hover{color:#007bff}.inner-content{padding:12px 0}.head-elements{font-size:14px;font-weight:500;color:#444}.options-container{display:flex;flex-direction:column;gap:12px}.field-size-controls,.flex-container{display:flex;align-items:center;gap:12px}.flex-container{margin-bottom:1rem}.size-input{width:110px;text-align:center}.input-container{display:flex;flex-direction:column;align-items:flex-start;gap:10px;width:100%}.subtext-textarea{width:100%;margin-left:auto}.arrow-icon{width:19px;height:23px;transition:transform .3s ease}.icon{display:inline-block;width:18px;height:18px;border-radius:50%;text-align:center;line-height:18px;color:#fff;font-size:12px;margin-left:5px}.edit-icon{background:#4caf50}.view-icon{background:#2196f3}.delete-icon{background:#f44336}.logo-icon{width:20px;height:20px;display:flex;justify-content:center;background:#d0d9ff;border-radius:4px}.link-icon{background:#e7f2ff;padding:5px;border-radius:5px;margin:5px;display:inline-block;cursor:pointer;position:relative;transition:all .3s}.link-icon:hover{background:#d0e5ff}.link-icon:active{background:#a8d0ff}.link-dropdown-menu{position:absolute;top:100%;right:0;background:#fff;border:1px solid #ccc;padding:10px;width:200px;box-shadow:0 4px 6px #0000001a}.link-dropdown-menu input{width:100%;margin-bottom:5px;padding:5px;border:1px solid #ccc;border-radius:3px}.option-items{display:flex;align-items:center;padding:5px;border:1px solid #ccc;margin-bottom:10px;cursor:grab;background:#fff}.option-items:active{opacity:.5}.drag-handle{margin-left:10px;cursor:grab}.button-toggle-wrapper{margin-top:8px}.toggle-button{padding:8px 16px;border:1px solid #cbd2d9;border-radius:6px;font-size:14px;cursor:pointer;transition:all .2s ease-in-out}.toggle-button.active:hover{background:#2c6dd5}.button-config-card{background:#fff;padding:15px;border-radius:12px;border:1px solid #ccc;margin-top:20px;box-shadow:0 2px 8px #0000000d}.design-footer{display:flex;justify-content:space-between;margin-top:20px}.button-container{display:flex;padding:15px;background:#f9f9f9;border-top:1px solid #eee}.button-container .cancel-btn,.button-container .save-btn{flex:1;padding:12px;font-weight:700}.divider{border-top:1px dashed #ddd;margin:20px 0}input{width:auto}@media screen and (max-width: 768px){.container{height:calc(100vh - 20px);min-height:calc(100vh - 20px)}.tabs{flex-direction:column}.tab{padding:1rem;font-size:.9rem}.form-group,.tab-content[aria-label=attributes] .form-group,.tab-content[aria-label=property] .form-group,.tab-content[aria-label=appearance] .form-group{flex-direction:column;grid-template-columns:1fr}.toggle-group{flex-direction:column;align-items:flex-start}}@media screen and (max-width: 480px){.container{padding:0 10px}.tabs{flex-direction:column}.tab{padding:1rem;border-bottom:1px solid #dee2e6}input[type=text],input[type=number],select,textarea{font-size:.9rem}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i5.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i5.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i5.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i5.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i5.RangeValueAccessor, selector: "input[type=range][formControlName],input[type=range][formControl],input[type=range][ngModel]" }, { kind: "directive", type: i5.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "directive", type: i5.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: ImageCropperComponent, selector: "image-cropper", inputs: ["imageChangedEvent", "imageURL", "imageBase64", "imageFile", "imageAltText", "options", "cropperFrameAriaLabel", "output", "format", "autoCrop", "cropper", "transform", "maintainAspectRatio", "aspectRatio", "resetCropOnAspectRatioChange", "resizeToWidth", "resizeToHeight", "cropperMinWidth", "cropperMinHeight", "cropperMaxHeight", "cropperMaxWidth", "cropperStaticWidth", "cropperStaticHeight", "canvasRotation", "initialStepSize", "roundCropper", "onlyScaleDown", "imageQuality", "backgroundColor", "containWithinAspectRatio", "hideResizeSquares", "allowMoveImage", "checkImageType", "alignImage", "disabled", "hidden"], outputs: ["imageCropped", "startCropImage", "imageLoaded", "cropperReady", "loadImageFailed", "transformChange", "cropperChange"] }, { kind: "component", type: NxtSearchBox, selector: "nxt-search-box", inputs: ["placeHolderText", "question", "apiMeta", "id", "readOnly", "mode"], outputs: ["searchValueChange"] }] });
1002
- }
1003
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PropertiesComponent, decorators: [{
1004
- type: Component,
1005
- args: [{ selector: 'app-properties', standalone: true, imports: [CommonModule, FormsModule, ImageCropperComponent, NxtSearchBox], template: "<!-- AP 22JAN25 - Field and Element Properties -->\n<div class=\"container\">\n <div class=\"tabs\">\n <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'attributes'}\" (click)=\"setActiveTab('attributes')\">\n Attributes\n </div>\n <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'property'}\" (click)=\"setActiveTab('property')\">\n Property\n </div>\n <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'appearance'}\" (click)=\"setActiveTab('appearance')\">\n Appearance\n </div>\n </div>\n\n <div class=\"tab-content\" *ngIf=\"activeTab === 'attributes'\">\n <!-- Element Properrties -->\n <!-- AP-06MAR25 Select element type show -->\n <div *ngIf=\"getProperties()\">\n <!-- SKS25MAR25 this is for image edit -->\n <div *ngIf=\"selectedElement?.type === 'Image'\">\n <image-cropper *ngIf=\"selectedElement.imageData \" [imageBase64]=\"selectedElement.orgImageData\"\n [disabled]=\"false\" [alignImage]=\"alignImage\" [roundCropper]=\"roundCropper\" [backgroundColor]=\"'white'\"\n imageAltText=\"Alternative image text\" [allowMoveImage]=\"false\" [hideResizeSquares]=\"false\"\n [canvasRotation]=\"canvasRotation\" [aspectRatio]=\"aspectRatio\" [containWithinAspectRatio]=\"false\"\n [maintainAspectRatio]=\"false\" [cropperStaticWidth]=\"cropperStaticWidth\"\n [cropperStaticHeight]=\"cropperStaticHeight\" [cropperMinWidth]=\"cropperMinWidth\"\n [cropperMinHeight]=\"cropperMinHeight\" [cropperMaxWidth]=\"cropperMaxWidth\"\n [cropperMaxHeight]=\"cropperMaxHeight\" [resetCropOnAspectRatioChange]='true' [(cropper)]=\"cropper\"\n [(transform)]=\"transform\" [onlyScaleDown]=\"true\" output=\"blob\" format=\"png\"\n (imageCropped)=\"imageCropped($event)\" (cropperReady)=\"cropperReady($event)\"></image-cropper>\n <div *ngIf=\"selectedElement.imageData \" style=\"display: flex; gap: 2px;\">\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateLeft()\" title=\"Rotate Left\">\u27F2</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateRight()\" title=\"Rotate Right\">\u27F3</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomOut()\" title=\"Zoom Out\">-</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomIn()\" title=\"Zoom In\">+</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveLeft()\" title=\"Move Left\">\u2190</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveRight()\" title=\"Move Right\">\u2192</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveUp()\" title=\"Move Up\">\u2191</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveDown()\" title=\"Move Down\">\u2193</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipHorizontal()\" [class.enabled]=\"transform.flipH\"\n title=\"Flip Horizontally\">\u2194</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipVertical()\" [class.enabled]=\"transform.flipV\"\n title=\"Flip Vertically\">\u2195</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"resetImage()\" title=\"Reset\">\u00D7</div>\n </div>\n </div>\n <!-- SKS28MAR25 search book specific for book element -->\n <div *ngIf=\"selectedElement?.type === 'Book'\">\n <label class=\"text-sm\">Search Book</label>\n <div style=\"display: flex; gap: 2px; align-items: center; justify-content: center;\">\n <nxt-search-box [question]=\"selectedElement\" [apiMeta]=\"bookSubtext\" [placeHolderText]=\"'Search...'\"\n (searchValueChange)=\"childEventCapture($event.value)\">\n </nxt-search-box>\n <div class=\"link-icon\">\n <svg (click)=\"linkToggleDropdown($event)\" fill=\"#000000\" version=\"1.1\" id=\"Capa_1\"\n xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" width=\"24px\" height=\"24px\"\n viewBox=\"0 0 450 450\" xml:space=\"preserve\">\n <g>\n <g>\n <g>\n <path d=\"M318.15,230.195l77.934-77.937c31.894-31.892,31.894-83.782-0.004-115.674l-12.66-12.66\n c-31.893-31.896-83.78-31.896-115.674-0.004l-77.937,77.934c-17.588,17.588-25.457,41.264-23.646,64.311\n c-23.045-1.813-46.722,6.056-64.308,23.647L23.92,267.748c-31.894,31.889-31.894,83.779,0,115.674l12.664,12.662\n c31.893,31.893,83.783,31.893,115.674,0l77.935-77.936c17.592-17.59,25.459-41.266,23.647-64.309\n C276.884,255.654,300.56,247.783,318.15,230.195z M202.653,290.605l-77.936,77.938c-16.705,16.703-43.889,16.703-60.59,0\n l-12.666-12.666c-16.705-16.701-16.703-43.885,0-60.594l77.936-77.932c14.14-14.141,35.779-16.306,52.226-6.516l-32.302,32.307\n c-7.606,7.604-7.606,19.938,0,27.541c7.605,7.607,19.937,7.607,27.541,0l32.306-32.303\n C218.959,254.828,216.795,276.469,202.653,290.605z M238.382,209.169l32.299-32.306c7.608-7.602,7.608-19.935,0-27.538\n c-7.604-7.61-19.936-7.61-27.541-0.004l-32.303,32.303c-9.791-16.446-7.627-38.087,6.514-52.226l77.935-77.935\n c16.707-16.707,43.89-16.707,60.594,0l12.664,12.664c16.705,16.705,16.705,43.886,0,60.591l-77.936,77.937\n C276.468,216.797,254.828,218.959,238.382,209.169z\" />\n <path d=\"M343.466,261.465c-45.287,0-82,36.713-82,82s36.713,82,82,82c45.286,0,82-36.713,82-82S388.753,261.465,343.466,261.465z\n M372.505,333.564l-56.046,56.104c-0.239,0.238-0.536,0.41-0.862,0.496l-22.315,5.85c-0.649,0.168-1.347-0.02-1.822-0.494\n c-0.477-0.479-0.666-1.172-0.496-1.824l5.826-22.318c0.084-0.326,0.256-0.627,0.494-0.863l56.047-56.104\n c0.742-0.742,1.945-0.744,2.688-0.002l4.548,4.541c0.739,0.74,0.741,1.943,0,2.688l-37.433,37.471l4.709,4.703l37.435-37.471\n c0.739-0.742,1.94-0.742,2.682-0.002l4.55,4.541C373.25,331.617,373.25,332.822,372.505,333.564z M395.472,310.574l-17,17.018\n c-0.739,0.744-1.942,0.744-2.685,0.002l-16.489-16.475c-0.744-0.74-0.744-1.943-0.002-2.688l17-17.02\n c0.741-0.74,1.944-0.74,2.688-0.002l16.487,16.477C396.216,308.629,396.216,309.832,395.472,310.574z\" />\n </g>\n </g>\n </g>\n </svg>\n <div class=\"link-dropdown-menu\" *ngIf=\"isLinkDropdownOpen\" #dropdown>\n <label>Endpoint:</label>\n <input type=\"text\" [(ngModel)]=\"bookSubtext.endpoint\" />\n\n <label>Variable:</label>\n <input type=\"text\" [(ngModel)]=\"bookSubtext.variable\" />\n\n <label>Field:</label>\n <input type=\"text\" [ngModel]=\"fieldAsString\" (ngModelChange)=\"updateField($event)\" />\n\n <label>Default Field:</label>\n <input type=\"text\" [(ngModel)]=\"bookSubtext.defaultField\" />\n </div>\n </div>\n </div>\n </div>\n <ng-container *ngFor=\"let prop of getProperties().elementProps\">\n <div class=\"form-group\">\n <label *ngIf=\"prop.type !== 'checkbox' && prop.type !== 'subQuestion'\" class=\"text-sm\">{{ prop.label\n }}</label>\n\n <!-- Text Input -->\n <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n [value]=\"headerSelect ? book.records[0].title : (selectedElement?.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ) : getValueByPath(prop.key))\"\n (input)=\"headerSelect ? updateTitle($event.target.value) : (selectedElement?.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : setValueByPath(prop.key, $event.target.value) ) : setValueByPath(prop.key, $event.target.value))\"\n [class.read-only]=\"selectedElement.readOnly\" [readonly]=\"selectedElement.readOnly\" />\n\n <!-- help text -->\n <div *ngIf=\"prop.key === 'helpText' && selectedElement.helpText\">{{\n selectedElement.helpText }}</div>\n\n <!-- questionNumber -->\n <input *ngIf=\"prop.key === 'questionNumber'\" type=\"number\" [value]=\"selectedElement.questionNumber\"\n (input)=\"setValueByPath('questionNumber', $event.target.value)\" />\n\n <!-- file -->\n <!-- Add this inside the elementProps loop where other inputs are rendered -->\n <select *ngIf=\"prop.type === 'select' && prop.key === 'supportType'\" [value]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option value=\"\">Select file category</option>\n <option *ngFor=\"let option of prop.options\" [value]=\"option.value\"> {{ option.label }} </option>\n </select>\n\n <div *ngIf=\"prop.type === 'checkbox'\">\n\n <input type=\"checkbox\"\n [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key) \"\n (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n <span class=\"toggle-label\" style=\"padding-left: 10px;\">{{ prop.label }}</span>\n\n </div>\n\n <!--AP-11JUN25 - Show this section only if the property key is 'primaryKey' -->\n <div *ngIf=\"prop.key === 'primaryKey'\">\n <input type=\"checkbox\" [checked]=\"getValueByPath('primaryKey')\" [disabled]=\"!getValueByPath('primaryKey') && isAnotherIdSelected()\" (change)=\"onToggleChange('primaryKey', $event)\"/>\n <span class=\"toggle-label\" style=\"padding-left: 10px;\">primaryKey</span>\n </div>\n\n <!-- SKS20MAR25 Subquestion Type -->\n <div *ngIf=\"prop.type === 'subQuestion'\">\n <div style=\"display: flex; flex-direction: row; gap: 10px; align-items: center;\">\n <div>{{ prop.label }}</div>\n <input type=\"checkbox\"\n [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key) \"\n (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n </div>\n <!-- SKS20MAR25 Render subquestions when checkbox is checked -->\n <div\n *ngIf=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key)\">\n <div *ngFor=\"let subProp of prop.subQuestion\"\n style=\"background-color: #e7f2ff; padding: 8px; border-radius: 4px;\">\n <div class=\"form-group\">\n <label>{{ subProp.label }}</label>\n <div *ngIf=\"subProp.type === 'array'\">\n <!-- Iterate over filtered columns to display checkboxes -->\n <div *ngFor=\"let column of filteredColumns\">\n <label>\n <input type=\"checkbox\"\n [checked]=\"subProp.operands ? subProp.operands.includes(column.apiName) : false\"\n (change)=\"onCheckboxChange(subProp.targetArray,subProp.targetArrayKey,subProp.key, column.apiName, $event.target.checked)\">\n {{ column.apiName }}\n </label>\n </div>\n </div>\n <!-- SKS21MAR25 New radio type -->\n <div *ngIf=\"subProp.type === 'radio'\">\n <div *ngFor=\"let column of filteredColumns\">\n <label>\n <input type=\"radio\" [name]=\"subProp.key\" [value]=\"column.apiName\"\n [checked]=\"getValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key) === column.apiName\"\n (change)=\"onRadioChange(subProp.targetArray, subProp.targetArrayKey, subProp.key, column.apiName)\">\n {{ column.apiName }}\n </label>\n </div>\n </div>\n <!-- SKS20MAR25 Add more subproperty types as needed -->\n <!-- Inside the subProp ngFor loop -->\n <input *ngIf=\"subProp.type === 'text'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n [value]=\"selectedElement.type === 'Table' ? (subProp.targetArray ? getValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key ) : getValueByPath(subProp.key) ) : getValueByPath(subProp.key)\"\n (input)=\"selectedElement.type === 'Table' ? (subProp.targetArray ? updateValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key, $event) : setValueByPath(subProp.key, $event.target.value) ) : setValueByPath(subProp.key, $event.target.value)\" />\n\n <input *ngIf=\"subProp.type === 'boolean'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n <select *ngIf=\"subProp.type === 'select'\" [value]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.value)\">\n <option *ngFor=\"let option of subProp.options\" [value]=\"option\">{{ option }}</option>\n </select>\n <input *ngIf=\"subProp.type === 'checkbox'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n </div>\n </div>\n </div>\n </div>\n\n <!-- SKS20MAR25 Text Align Buttons -->\n <div *ngIf=\"prop.type === 'align'\">\n <button *ngFor=\"let option of prop.options\" (click)=\"onAlignSelect(option.value)\"\n [class.active]=\"selectedElement?.textAlign === option.value\" [title]=\"option.value\">\n <img [src]=\"'../assets/icons/' + option.icon + '.svg'\" [alt]=\"option.value\" class=\"icon-size\" />\n </button>\n </div>\n\n <div *ngIf=\"prop.type === 'style'\">\n <button *ngFor=\"let option of prop.options\" (click)=\"onStyleSelect(option.value)\"\n [class.active]=\"isStyleActive(option.value)\" [title]=\"option.value\">\n <img [src]=\"'../assets/icons/' + option.icon + '.svg'\" [alt]=\"option.value\" class=\"icon-size\" />\n </button>\n </div>\n\n <!-- Field Size Controls -->\n <!-- AP 25FEB25 - Change key size -->\n <div *ngIf=\"prop.key === 'size'\">\n <input type=\"number\" [value]=\"selectedElement?.size\" (input)=\"setValueByPath('size', $event.target.value)\"\n class=\"size-input\" />\n </div>\n\n <!-- AP-19MAR25 Line Properties -->\n <!-- Padding Top -->\n <div *ngIf=\"prop.key === 'paddingTop'\">\n <input type=\"number\" [placeholder]=\"prop.placeholder\" [value]=\"selectedElement?.paddingTop\"\n (input)=\"setValueByPath('paddingTop', $event.target.value)\" />\n </div>\n\n <!-- Padding Bottom -->\n <div *ngIf=\"prop.key === 'paddingBottom'\">\n <input type=\"number\" [placeholder]=\"prop.placeholder\" [value]=\"selectedElement?.paddingBottom\"\n (input)=\"setValueByPath('paddingBottom', $event.target.value)\" />\n </div>\n\n <!-- Line Style -->\n <div *ngIf=\"prop.key === 'lineStyle'\">\n <select [value]=\"selectedElement?.lineStyle\" (change)=\"setValueByPath('lineStyle', $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option\">{{ option }}</option>\n </select>\n </div>\n\n <!-- Color -->\n <div *ngIf=\"prop.key === 'color'\">\n <input type=\"color\" [value]=\"selectedElement?.color\"\n (input)=\"setValueByPath('color', $event.target.value)\" />\n </div>\n\n </div>\n \n <!-- AP 23MAY25 - Action Button -->\n <div *ngIf=\"prop.key === 'title'\" style=\"margin-top: 15px; border: 1px solid #ddd; padding: 20px; border-radius: 12px; background-color: #f9f9f9;\">\n <div class=\"form-header\">\n <label class=\"form-label\">{{ prop.label }}</label>\n <button (click)=\"addNewButton()\" class=\"add-button\">+ Add Button</button>\n </div>\n \n <div *ngFor=\"let btn of book?.questionbook?.action || []; let i = index\" class=\"button-config-card\">\n <div class=\"form-group\">\n <label>Button Name</label>\n <input type=\"text\" [(ngModel)]=\"btn.name\" (ngModelChange)=\"onButtonPropertyChange(i, 'name', $event)\" placeholder=\"Enter button name\">\n </div>\n \n <div class=\"form-group\">\n <label>Event Name</label>\n <input type=\"text\" [(ngModel)]=\"btn.eventtoemit\" (ngModelChange)=\"onButtonPropertyChange(i, 'eventtoemit', $event)\" placeholder=\"Event to emit\">\n </div>\n \n <div class=\"form-group\">\n <label>Alt</label>\n <input type=\"text\" [(ngModel)]=\"btn.alt\" (ngModelChange)=\"onButtonPropertyChange(i, 'alt', $event)\" placeholder=\"Button alt text\">\n </div>\n \n <div class=\"form-group\">\n <label>Endpoint</label>\n <input type=\"text\" [(ngModel)]=\"btn.endpoint\" (ngModelChange)=\"onButtonPropertyChange(i, 'endpoint', $event)\" placeholder=\"API endpoint\">\n </div>\n \n <div class=\"form-group\">\n <label>Method</label>\n <select [(ngModel)]=\"btn.method\" (ngModelChange)=\"onButtonPropertyChange(i, 'method', $event)\">\n <option [value]=\"'GET'\">GET</option>\n <option [value]=\"'POST'\">POST</option>\n <option [value]=\"'PUT'\">PUT</option>\n <option [value]=\"'DELETE'\">DELETE</option>\n </select>\n </div>\n \n <!-- AP 23MAY25 - Action Style Fields -->\n <div class=\"form-group\">\n <label>Background Color</label>\n <input type=\"color\" [(ngModel)]=\"btn.bgColor\" (ngModelChange)=\"onButtonPropertyChange(i, 'bgColor', $event)\">\n </div>\n \n <div class=\"form-group\">\n <label>Border Radius</label>\n <input type=\"range\" min=\"0\" max=\"50\" [(ngModel)]=\"btn.borderRadius\" \n (ngModelChange)=\"onButtonPropertyChange(i, 'borderRadius', $event)\">\n <span>{{ btn.borderRadius }}px</span>\n </div> \n \n <div class=\"form-group\">\n <label>Button Width (px)</label>\n <input type=\"number\" [(ngModel)]=\"btn.width\"\n (ngModelChange)=\"onButtonPropertyChange(i, 'width', $event)\" min=\"50\" placeholder=\"Enter width in px\">\n </div>\n \n <div class=\"form-group\">\n <label>Text Color</label>\n <input type=\"color\" [(ngModel)]=\"btn.textColor\"\n (ngModelChange)=\"onButtonPropertyChange(i, 'textColor', $event)\">\n </div>\n \n <div class=\"form-group\">\n <label>Position (%)</label>\n <input type=\"range\" min=\"0\" max=\"100\" [(ngModel)]=\"btn.positionPercent\"\n (ngModelChange)=\"onButtonPropertyChange(i, 'positionPercent', $event)\">\n <span>{{ btn.positionPercent }}%</span>\n </div>\n <button (click)=\"removeButton(btn)\" class=\"remove-button\">\u00D7 Remove</button>\n </div>\n </div>\n \n </ng-container>\n </div>\n </div>\n <div class=\"tab-content\" *ngIf=\"activeTab === 'property'\">\n <!-- Field Elements Properties -->\n <!-- AP-06MAR25 Show elements ID -->\n <div *ngIf=\"getProperties()\">\n <label>Label Id</label>\n <div\n style=\"font-size: 13px; padding: 11px; border-radius: 5px; background-color: #f8f8f8; border: 1px solid #ddd;\">\n {{ headerSelect ? bookId : selectedElement.id }}</div>\n <ng-container *ngFor=\"let prop of getProperties().fieldProps\">\n <div class=\"form-group\">\n <label class=\"text-sm\">{{ prop.label }}</label>\n\n <!-- SKS21MAR25 Toggle Group -->\n <div *ngIf=\"prop.type === 'toggleGroup'\" class=\"toggle-group\">\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" [checked]=\"selectedElement?.isOptional\"\n (change)=\"onRequiredChange($event.target.checked)\" />\n Required\n </label>\n </div>\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" [checked]=\"selectedElement?.isReadOnly\"\n (change)=\"onToggleChange('isReadOnly', $event)\" />\n Read Only\n </label>\n </div>\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" />\n Disabled\n </label>\n </div>\n <div class=\"toggle-item\">\n <label class=\"toggle-label\">\n <input type=\"checkbox\" [checked]=\"selectedElement?.isHidden\"\n (change)=\"onToggleChange('isHidden', $event)\" />\n Is Hide\n </label>\n </div>\n </div>\n\n <!-- AP 28FEB25 - handled options with UUID -->\n <div\n *ngIf=\"prop.type === 'dropdown' || prop.type === 'checkbox' || prop.type === 'radio' && prop.key === 'options'\"\n class=\"options-container\">\n\n <div class=\"option-list\" (dragover)=\"onDragOver($event)\" (drop)=\"onDrop($event, prop.key)\">\n <div *ngFor=\"let option of selectedElement[prop.key]\" class=\"option-items\" [attr.data-id]=\"option.id\"\n draggable=\"true\" (dragstart)=\"onDragStart($event, option.id)\">\n\n <input type=\"text\" [(ngModel)]=\"option.value\" placeholder=\"Option\" class=\"options\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeOption(selectedElement[prop.key], option.id)\">\n <span class=\"drag-handle\">\u2630</span>\n </div>\n </div>\n\n <button (click)=\"addOption(selectedElement[prop.key])\">\n <div class=\"add-varient\">\n <span class=\"text-lg\">+</span>\n <span>Add</span>\n </div>\n </button>\n </div>\n\n <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n [value]=\"headerSelect ? book.records[0].title : (selectedElement?.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ) : getValueByPath(prop.key))\"\n (input)=\"headerSelect ? updateTitle($event.target.value) : (selectedElement?.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : setValueByPath(prop.key, $event.target.value) ) : setValueByPath(prop.key, $event.target.value))\"\n [class.read-only]=\"selectedElement.readOnly\" [readonly]=\"selectedElement.readOnly\" />\n\n <!-- Sub Questions Toggle -->\n <div *ngIf=\"prop.type === 'subQuestion'\">\n <div class=\"style-toggle-header\" (click)=\"toggleSubQuestion(prop)\">\n <div class=\"head-elements\">Sub Text</div>\n <img [src]=\"prop.isExpanded ? '../assets/icons/arrow-down.svg' : '../assets/icons/arrow-right.svg'\"\n alt=\"Toggle Arrow\" class=\"arrow-icon\">\n </div>\n\n <!-- Render subquestions when arrow is down -->\n <div *ngIf=\"prop.isExpanded\" style=\"border: 1px solid #ddd; padding: 8px; border-radius: 4px;\">\n <div *ngFor=\"let subProp of prop.subQuestion\" class=\"sub-question-container\">\n <div class=\"form-group\">\n <label>{{ subProp.label }}</label>\n\n <input *ngIf=\"subProp.type === 'text'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n [value]=\"getValueByPath(subProp.key)\" (input)=\"setValueByPath(subProp.key, $event.target.value)\" />\n \n <!-- AP-02APR25 Render input field only if subProp.type is 'array' -->\n <input *ngIf=\"subProp.type === 'array'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n [value]=\"getValueByPath(subProp.key)\" (input)=\"setValueByPath(subProp.key, $event.target.value)\" />\n <!-- <input type=\"text\" [ngModel]=\"fieldAsString\" (ngModelChange)=\"updateField($event)\" /> -->\n\n <input *ngIf=\"subProp.type === 'boolean'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"prop.type === 'checkbox'\">\n <span class=\"toggle-label\" style=\"padding-right: 10px;\">{{ prop.label }}</span>\n <input type=\"checkbox\"\n [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ): getValueByPath(prop.key) \"\n (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n\n </div>\n\n </div>\n </ng-container>\n\n <!-- SKS21MAR25 Move Style Outside the Loop -->\n <!-- Style Toggle -->\n <div class=\"style-toggle-header\" (click)=\"toggleStyleSection()\">\n <div class=\"head-elements\">Style</div>\n <img [src]=\"isStyleExpanded ? '../assets/icons/arrow-down.svg' : '../assets/icons/arrow-right.svg'\"\n alt=\"Toggle Arrow\" class=\"arrow-icon\">\n </div>\n\n <div *ngIf=\"isStyleExpanded\" style=\"border: 1px solid #ddd; padding: 8px; border-radius: 4px;\">\n <div class=\"form-group\">\n <div *ngFor=\"let key of getStyleKeys()\">\n <label>{{ key }}</label>\n <input type=\"text\" [(ngModel)]=\"selectedElement.style[key]\"\n (input)=\"setValueByPath('style.' + key, $event.target.value)\" placeholder=\"Enter {{ key }}\" />\n </div>\n </div>\n </div>\n\n </div>\n </div>\n <div class=\"tab-content\" *ngIf=\"activeTab === 'appearance'\">\n <div *ngIf=\"getProperties()\">\n <ng-container *ngFor=\"let prop of getProperties().appearance\">\n <div class=\"form-group\">\n <label class=\"text-sm\">{{ prop.label }}</label>\n <!-- Font Selection -->\n <div *ngIf=\"prop.key === 'font'\">\n <select *ngIf=\"prop.type === 'select'\" class=\"select-container\" [(ngModel)]=\"selectedElement.font\"\n (change)=\"setValueByPath('font', $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option\">\n {{ option }}\n </option>\n </select>\n </div>\n <!-- Input Box -->\n <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n [value]=\"selectedElement[prop.key]\" (input)=\"setValueByPath(prop.key, $event.target.value)\" />\n\n <!-- Font Size Selection -->\n <div *ngIf=\"prop.key === 'fontSize'\">\n <select *ngIf=\"prop.type === 'select'\" [(ngModel)]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option.value\">\n {{ option.label }}\n </option>\n </select>\n </div>\n <!-- Font Weight Selection -->\n <div *ngIf=\"prop.key === 'fontWeight'\">\n <select *ngIf=\"prop.type === 'select'\" [(ngModel)]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option.value\">\n {{ option.label }}\n </option>\n </select>\n </div>\n\n <div class=\"flex-container\">\n <!-- Dropdown Box -->\n <div *ngIf=\"prop.type === 'dropdown'\" class=\"input-box-field\">\n <select [(ngModel)]=\"selectedElement[prop.key]\" (change)=\"setValueByPath(prop.key, $event.target.value)\">\n <option *ngFor=\"let option of prop.options\" [value]=\"option\">\n {{ option }}\n </option>\n </select>\n </div>\n\n <!-- Color Picker -->\n <div *ngIf=\"prop.type === 'color'\" class=\"color-selector\">\n <input type=\"color\" [(ngModel)]=\"selectedElement.fontColor\"\n (change)=\"setValueByPath('fontColor', $event.target.value)\">\n </div>\n\n <!-- HEX Input Box -->\n <div *ngIf=\"prop.type === 'color'\" class=\"hex-input-container\">\n <span>HEX Code</span>\n <input type=\"text\" [(ngModel)]=\"selectedElement[prop.key]\"\n (change)=\"setValueByPath(prop.key, $event.target.value)\" />\n </div>\n </div>\n \n <!-- AP-17APR25 Button for Duplicate Field -->\n<div *ngIf=\"prop.type === 'button-toggle'\" class=\"button-toggle-wrapper\">\n <button type=\"button\"\n class=\"toggle-button\"\n (click)=\"duplicateField(selectedElement)\">\n Add Duplicate\n </button>\n</div>\n\n\n </div>\n </ng-container>\n </div>\n </div>\n\n<!-- AP-08APR25 Default Save Button -->\n<div class=\"button-container\" *ngIf=\"!templateSelected\">\n <button class=\"cancel-btn\" (click)=\"onCancel()\">Cancel</button>\n <button class=\"save-btn\" (click)=\"handleButtonClick()\">Save</button>\n</div>\n\n<!-- AP-08APR25 Template Save Button -->\n<div class=\"button-container\" *ngIf=\"templateSelected\" style=\"margin-top: 20px;\">\n <button class=\"save-btn\" (click)=\"handleTemplateSave()\">Save Template</button>\n</div>\n\n</div>", styles: ["*{margin:0;padding:0;box-sizing:border-box;font-family:Roboto,sans-serif}.properties{height:calc(100vh - 20px);overflow-y:auto}.container{width:100%;max-width:500px;margin:0 auto;background:#fff;box-shadow:0 2px 10px #0000001a;overflow:hidden;font-family:Arial,sans-serif}.design-header{display:flex;justify-content:center;align-items:center;padding:15px 20px;background:#fff;border-radius:8px;box-shadow:0 2px 10px #0000001a;margin-bottom:20px}.header-title{font-size:20px;font-weight:400;color:#222}.toggle-header,.style-toggle-header{display:flex;align-items:center;justify-content:space-between;cursor:pointer;padding:10px;background:#f8f8f8;border:1px solid #ddd;border-radius:6px;margin-bottom:6px}.tabs{display:flex;background:#f0f2f5;border-bottom:2px solid #0052cc}.tab{flex:1;padding:15px 10px;text-align:center;cursor:pointer}.tab.active{background:#0052cc;color:#fff;font-weight:700}.tab-content{padding:20px;max-height:80vh;overflow-y:auto;overflow-x:hidden}.form-group{margin-bottom:15px}label{display:block;font-size:14px;font-weight:500;color:#444;margin-bottom:8px}.required:before{content:\"*\";color:red;margin-right:3px}input[type=text],input[type=number],select,textarea{width:100%;padding:10px;font-size:14px;border:1px solid #ccc;border-radius:6px;outline:none;transition:all .3s;background:#f8f8f8}input:focus,select:focus,textarea:focus{border-color:#007bff;background:#fff;box-shadow:0 0 5px #007bff4d}textarea{min-height:55px;resize:vertical}button,.btn{padding:10px 15px;border:none;border-radius:6px;background:#007bff;color:#fff;font-size:15px;cursor:pointer;transition:all .3s}button:hover,.btn:hover{background:#0056b3;transform:translateY(-2px)}.add-button{padding:8px 14px;background:#0954c5;border-radius:8px}.remove-button{background:#e74c3c;padding:6px 12px;margin-top:10px}.remove-button:hover{background:#c0392b}.cancel-btn{background:#fff;color:#666;border:1px solid #ddd;margin-right:10px}.save-btn{background:#0052cc}.toggle-group{display:grid;grid-template-columns:1fr 1fr;gap:10px}.toggle-item{display:flex;align-items:center;gap:10px;padding:8px}.switch{position:relative;width:42px;height:22px}.switch input{opacity:0;width:0;height:0}.slider{position:absolute;cursor:pointer;inset:0;background:#ccc;transition:.4s;border-radius:24px}.slider:before{position:absolute;content:\"\";height:20px;width:20px;left:1px;bottom:1px;background:#fff;transition:.4s;border-radius:50%}input:checked+.slider{background:#2196f3}input:checked+.slider:before{transform:translate(18px)}.radio-item{display:flex;align-items:center;gap:20px}.radio-item input{accent-color:#007bff}.checkbox-row{display:flex;margin-bottom:10px}.checkbox-group{display:flex;align-items:center;margin-right:20px;min-width:120px}.checkbox-group input[type=checkbox]{margin-right:5px}.color-picker-row,.color-picker-container{display:flex;align-items:center}.color-picker-container{border:1px solid #ddd;border-radius:4px;margin-right:10px}.color-box,.color-selector input[type=color]{width:78px;height:40px;border:none;border-radius:4px;cursor:pointer}.hex-input-container{display:flex;gap:20px;align-items:center;font-size:12px;color:#b0b0b0}.hex-input-container span{font-size:14px;margin-top:10px}.hex-input,.hex-input-container input[type=text]{width:120px;padding:10px 12px;border:1px solid #d1d1d1;color:#28343e}.hex-label{color:#999;margin-right:5px}.select-container,.input-box-field{position:relative;width:100%}.dropdown-arrow,.input-box-field:after{position:absolute;right:10px;top:50%;transform:translateY(-50%);pointer-events:none;color:#666;font-size:12px}.input-box-field:after{content:\"\\25bc\";color:#1c1b1f;font-size:10px}.input-box-field select{background:#28343e;color:#fff;border:none;appearance:none;-webkit-appearance:none;-moz-appearance:none}.input-box-field select::-ms-expand{display:none}.all-properties details{background:#fff;border:1px solid #ddd;border-radius:8px;margin-bottom:12px;padding:12px;box-shadow:0 2px 8px #0000000d}.all-properties summary{font-size:15px;font-weight:400;cursor:pointer;padding:6px;transition:color .3s}.all-properties summary:hover{color:#007bff}.inner-content{padding:12px 0}.head-elements{font-size:14px;font-weight:500;color:#444}.options-container{display:flex;flex-direction:column;gap:12px}.field-size-controls,.flex-container{display:flex;align-items:center;gap:12px}.flex-container{margin-bottom:1rem}.size-input{width:110px;text-align:center}.input-container{display:flex;flex-direction:column;align-items:flex-start;gap:10px;width:100%}.subtext-textarea{width:100%;margin-left:auto}.arrow-icon{width:19px;height:23px;transition:transform .3s ease}.icon{display:inline-block;width:18px;height:18px;border-radius:50%;text-align:center;line-height:18px;color:#fff;font-size:12px;margin-left:5px}.edit-icon{background:#4caf50}.view-icon{background:#2196f3}.delete-icon{background:#f44336}.logo-icon{width:20px;height:20px;display:flex;justify-content:center;background:#d0d9ff;border-radius:4px}.link-icon{background:#e7f2ff;padding:5px;border-radius:5px;margin:5px;display:inline-block;cursor:pointer;position:relative;transition:all .3s}.link-icon:hover{background:#d0e5ff}.link-icon:active{background:#a8d0ff}.link-dropdown-menu{position:absolute;top:100%;right:0;background:#fff;border:1px solid #ccc;padding:10px;width:200px;box-shadow:0 4px 6px #0000001a}.link-dropdown-menu input{width:100%;margin-bottom:5px;padding:5px;border:1px solid #ccc;border-radius:3px}.option-items{display:flex;align-items:center;padding:5px;border:1px solid #ccc;margin-bottom:10px;cursor:grab;background:#fff}.option-items:active{opacity:.5}.drag-handle{margin-left:10px;cursor:grab}.button-toggle-wrapper{margin-top:8px}.toggle-button{padding:8px 16px;border:1px solid #cbd2d9;border-radius:6px;font-size:14px;cursor:pointer;transition:all .2s ease-in-out}.toggle-button.active:hover{background:#2c6dd5}.button-config-card{background:#fff;padding:15px;border-radius:12px;border:1px solid #ccc;margin-top:20px;box-shadow:0 2px 8px #0000000d}.design-footer{display:flex;justify-content:space-between;margin-top:20px}.button-container{display:flex;padding:15px;background:#f9f9f9;border-top:1px solid #eee}.button-container .cancel-btn,.button-container .save-btn{flex:1;padding:12px;font-weight:700}.divider{border-top:1px dashed #ddd;margin:20px 0}input{width:auto}@media screen and (max-width: 768px){.container{height:calc(100vh - 20px);min-height:calc(100vh - 20px)}.tabs{flex-direction:column}.tab{padding:1rem;font-size:.9rem}.form-group,.tab-content[aria-label=attributes] .form-group,.tab-content[aria-label=property] .form-group,.tab-content[aria-label=appearance] .form-group{flex-direction:column;grid-template-columns:1fr}.toggle-group{flex-direction:column;align-items:flex-start}}@media screen and (max-width: 480px){.container{padding:0 10px}.tabs{flex-direction:column}.tab{padding:1rem;border-bottom:1px solid #dee2e6}input[type=text],input[type=number],select,textarea{font-size:.9rem}}\n"] }]
1006
- }], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.FormBuilderService }, { type: i3.TemplateService }], propDecorators: { formButtonHandler: [{
1007
- type: Output
1008
- }], templateSaveHandler: [{
1009
- type: Output
1010
- }], selectedElementType: [{
1011
- type: Input
1012
- }], templateSelected: [{
1013
- type: Input
1014
- }], dropdown: [{
1015
- type: ViewChild,
1016
- args: ['dropdown', { static: false }]
1017
- }], onClickOutside: [{
1018
- type: HostListener,
1019
- args: ['document:click', ['$event']]
1020
- }] } });
1021
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"properties.component.js","sourceRoot":"","sources":["../../../../../../../projects/nxt-app/src/lib/pages/builder/properties/properties.component.ts","../../../../../../../projects/nxt-app/src/lib/pages/builder/properties/properties.component.html"],"names":[],"mappings":"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","sourcesContent":["// AP 22JAN25\nimport { HttpClient } from '@angular/common/http';\nimport { Component, ElementRef, EventEmitter, HostListener, Input, OnInit, Output, SimpleChanges, ViewChild } from '@angular/core';\nimport { FormBuilderService } from '../../../services/form-builder.service';\nimport { CropperPosition, Dimensions, ImageTransform } from '../../../components/image-cropper/interfaces';\nimport { v4 as uuidv4 } from 'uuid';//AP-28MAR25 import uuid\n// AP-11APR25 import template service\nimport { TemplateService } from '../../../services/template.service';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { ImageCropperComponent } from \"../../../components/image-cropper/component/image-cropper.component\";\nimport { NxtSearchBox } from \"../../../components/search-box/search-box.component\";\n//AP-25JUN25-Importing common constants for element, field, and appearance properties\nimport { COMMON_ELEMENT_PROPS, COMMON_OPTIONS_FIELD, COMMON_FIELD_PROPS, COMMON_APPEARANCE_PROPS, COMMON_COLUMN_FIELD} from '../properties/common-fields.constants';\n@Component({\n  selector: 'app-properties',\n  standalone: true,\n  imports: [CommonModule, FormsModule, ImageCropperComponent, NxtSearchBox],\n  templateUrl: './properties.component.html',\n  styleUrls: ['./properties.component.css']\n})\nexport class PropertiesComponent implements OnInit {\n  element: any;\n  from: string;\n  columns: any;\n\n  createElementConfig(config?: { customElementProps?: any[], customFieldProps?: any[], customAppearanceProps?: any[]}) {\n    return {\n      elementProps: [...COMMON_ELEMENT_PROPS, ...(config?.customElementProps || [])],\n      fieldProps: [...COMMON_FIELD_PROPS, ...(config?.customFieldProps || [])],\n      appearance: [...COMMON_APPEARANCE_PROPS, ...(config?.customAppearanceProps || [])]\n    };\n  }\n\n  @Output() formButtonHandler = new EventEmitter<any>();\n  @Output() templateSaveHandler = new EventEmitter<any>();\n  selectedOption: string = '';\n  selectedElementIndex: number = -1;\n  @Input() selectedElementType: string = '';\n  selectedAlign: string = 'align-left';  // Add this property\n  selectedStyles: string[] = [];  // Using array since multiple styles can be selected\n  errorMessage: string = '';\n  selectedElement: any = null;  // Receive the selected element\n  headerSelect: boolean = false;\n  bookId: any;\n  book: any;\n  activeTab: string = 'attributes';\n  selectColumn: any\n  transform: ImageTransform = {\n    translateUnit: 'px',\n    scale: 1,\n    rotate: 0,\n    flipH: false,\n    flipV: false,\n    translateH: 0,\n    translateV: 0\n  };\n  canvasRotation = 0;\n  cropper?: CropperPosition;\n  loading = false;\n  cropperMaxHeight = 0;\n  cropperMaxWidth = 0;\n  cropperMinHeight = 0;\n  cropperMinWidth = 0;\n  cropperStaticWidth = 0;\n  cropperStaticHeight = 0;\n  aspectRatio = 4 / 3;\n  roundCropper = false;\n  imageChangedEvent: Event | null = null;\n  alignImage = 'center' as const;\n  //  \"endpoint\": \"https://dev-api.valarhr.com/nxt\",\n  // SKS28MAR25 default book search endpoint\n  bookSubtext = {\n    \"endpoint\": \"https://dev-api.valarhr.com/nxt\",\n    \"variable\": null,\n    \"field\": ['label', 'name'],\n    \"defaultField\": \"label\"\n  }\n  isLinkDropdownOpen = false;\n  @Input() templateSelected: boolean = false;\n\n  @ViewChild('dropdown', { static: false }) dropdown!: ElementRef;\n\n\n  draggedId: string | null = null;\n  formElements: any[] = [];\n \n  unique_id: string = 'book-1234'; // should be consistent\n  selectedTemplate: any;\n\n  elementProperties = {\n    // AP-10MAR25 - Defines the Header element with a text input field   \n    // AP-25MAR25 Add subtext all variables\n    'Header': {\n      elementProps: [\n        { label: 'Label', placeholder: 'Enter Text', type: 'text', key: 'title' }]\n    },\n    // AP-19MAR25 Add Line Property\n    'Line': {\n      elementProps: [\n        { label: 'Line Width', type: 'fieldSize', key: 'size' },\n        { label: 'Padding Top', type: 'number', key: 'paddingTop', placeholder: 'Enter padding top' },\n        { label: 'Padding Bottom', type: 'number', key: 'paddingBottom', placeholder: 'Enter padding bottom' },\n        {\n          label: 'Line Style', type: 'select', key: 'lineStyle',\n          options: ['Solid', 'Dashed', 'Dotted']\n        }\n      ],\n      fieldProps: []\n    },\n\n    'Space': {\n      elementProps: [\n        { label: 'Element', type: 'number', key: 'questionNumber' },\n        { label: 'Field Size', type: 'fieldSize', key: 'size' },\n      ],\n      fieldProps: []\n    },\n\n    \"Table\": {\n      \"elementProps\": [\n        {label: 'Is Label', placeholder: 'Enter Text ', type: 'checkbox', key: 'style.showLabel' },\n        { key: 'primaryKey' },\n        { \"label\": \"tableName\", \"placeholder\": \"Employee Details\", \"type\": \"text\", \"key\": \"questionText\" },\n        { \"label\": \"inputTextAlignment\", \"placeholder\": \"Left\", \"type\": \"text\", \"key\": \"inputTextAlignment\" },\n        { \"label\": \"tableScaleSize\", \"placeholder\": \"06 - Full Scale\", \"type\": \"text\", \"key\": \"tableScaleSize\" },\n        { \"label\": \"isNosIndicator\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.isNosIndicator\" },\n        { \"label\": \"isPagination\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.isPagination\" },\n        { \"label\": \"addInlineRecord\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.addInlineRecord\" },\n        { \"label\": \"actionButton\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.actionButton\" },\n        { \"label\": \"searchBar\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.searchBar\" },\n        { \"label\": \"isDeleteRow\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.isDeleteRow\" },\n        { \"label\": \"isEditRow\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.isEditRow\" },\n        { \"label\": \"isButtons\", \"placeholder\": \"06 - Full Scale\", \"type\": \"checkbox\", \"key\": \"tableConfig.isButtons\" },\n\n\n        { \"label\": \"rowChoice\", \"placeholder\": \"\", \"type\": \"text\", \"key\": \"rowChoice\" },\n        { label: 'questionNumber', type: 'number', key: 'questionNumber' },\n        { label: 'Field Size', type: 'fieldSize', key: 'size' },\n      ],\n      \"fieldProps\": [\n        { \"label\": \"tableId\", \"placeholder\": \"Emp_Table_01\", \"type\": \"text\", \"key\": \"tableId\" },\n        { \"label\": \"helpText\", \"placeholder\": \"Enter message\", \"type\": \"text\", \"key\": \"helpText\" },\n        { \"label\": \"defaultValue\", \"placeholder\": \"Default Value\", \"type\": \"text\", \"key\": \"defaultValue\" },\n        { label: 'Reference', placeholder: 'Reference Field', type: 'text', key: 'referenceField' },\n        {\n          \"label\": \"\", \"placeholder\": \"Enter Text\", \"type\": \"subQuestion\", \"key\": \"isSubText\",\n          \"subQuestion\": [\n            { \"label\": \"End Point\", \"placeholder\": \"endpoint\", \"type\": \"text\", \"key\": \"subText.endpoint\" },\n            { \"label\": \"Variable\", \"placeholder\": \"variable\", \"type\": \"text\", \"key\": \"subText.variable\" },\n            { \"label\": \"Field\", \"placeholder\": \"field\", \"type\": \"text\", \"key\": \"subText.field\" },\n\n            { \"label\": \"Default Field\", \"placeholder\": \"defaultField\", \"type\": \"text\", \"key\": \"subText.defaultField\" },\n            { \"label\": \"Label Field\", \"placeholder\": \"labelField\", \"type\": \"text\", \"key\": \"subText.labelField\" },\n            { \"label\": \"Value Field\", \"placeholder\": \"valueField\", \"type\": \"text\", \"key\": \"subText.valueField\" },\n            { \"label\": \"Source Question Id\", \"placeholder\": \"sourceQuestionId\", \"type\": \"text\", \"key\": \"subText.sourceQuestionId\" },\n            { \"label\": \"Dependent Value\", \"placeholder\": \"dependentValue\", \"type\": \"text\", \"key\": \"subText.dependentValue\" },\n            { \"label\": \"Dependent Field\", \"placeholder\": \"isDependentField\", \"type\": \"boolean\", \"key\": \"subText.isDependentField\" },\n\n            { \"label\": \"Query Field\", \"placeholder\": \"queryField\", \"type\": \"text\", \"key\": \"subText.queryField\" },\n            { \"label\": \"Query Value\", \"placeholder\": \"queryValue\", \"type\": \"text\", \"key\": \"subText.queryValue\" },\n            { \"label\": \"Query Value Ref\", \"placeholder\": \"queryValueRef\", \"type\": \"text\", \"key\": \"subText.queryValueReference\" },\n            { \"label\": \"Unique Key\", \"placeholder\": \"uniqueKey\", \"type\": \"text\", \"key\": \"subText.uniqueKey\" },\n          ]\n        },\n      ],\n      \"appearance\": [\n          {\n            label: 'Select Font', type: 'select', key: 'font', placeholder: 'Left', required: true,\n            options: ['Helvetica Neue', 'Arial', 'Times New Roman', 'Roboto']\n          },\n          { label: 'Font Color', type: 'color', key: 'fontColor', defaultValue: '', required: true },\n          {\n            label: 'Font Size', type: 'select', key: 'fontSize', defaultValue: '', required: true,\n            options: [\n              { label: 'Small', value: '12px' },\n              { label: 'Medium', value: '14px' },\n              { label: 'Large', value: '16px' }\n            ]\n          },\n  \n          {\n            label: 'Font Width', type: 'select', key: 'fontWeight', defaultValue: '', required: true,\n            options: [\n              { value: '400', label: '400-Normal' },\n              { value: '500', label: '500-Medium' },\n              { value: '600', label: '600-Semi Bold' },\n              { value: '700', label: '700-Bold' }\n            ]\n          },\n          { label: 'Duplicate Field', type: 'button-toggle', key: 'duplicateField', defaultValue: false }\n        ]\n    },\n    \"TableColumn\": {\n      \"elementProps\": [\n\n        { \"label\": \"headerLabels\", \"placeholder\": \"Employee Details\", \"type\": \"text\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"label\" },\n        { \"label\": \"apiName\", \"placeholder\": \"Employee Details\", \"type\": \"text\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"apiName\" },\n\n        {\n          \"label\": \"Summary Column\", \"placeholder\": \"Summary Column\", \"type\": \"subQuestion\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"summaryColumn\",\n          \"subQuestion\": [\n            { \"label\": \"Type\", \"placeholder\": \"Employee Details\", \"type\": \"text\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"fldType\" },\n            { \"label\": \"Operation\", \"placeholder\": \"Employee Details\", \"type\": \"text\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"operation\" },\n            { \"label\": \"Operands\", \"placeholder\": \"Employee Details\", \"type\": \"array\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"operands\" },\n          ]\n        },\n        {\n          \"label\": \"Summary Row\", \"placeholder\": \"Summary Row\", \"type\": \"subQuestion\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"summaryRow\", \"subQuestion\": [\n            { \"label\": \"Type\", \"placeholder\": \"Employee Details\", \"type\": \"text\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"fldType\" },\n            { \"label\": \"operation\", \"placeholder\": \"operation\", \"type\": \"text\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"operation\" },\n            { \"label\": \"column\", \"placeholder\": \"Employee Details\", \"type\": \"radio\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"column\" },\n            { \"label\": \"operands\", \"placeholder\": \"Employee Details\", \"type\": \"array\", \"targetArray\": \"fieldsMeta\", \"targetArrayKey\": \"uniqueIdentifier\", \"key\": \"operands\" },\n          ]\n        },\n\n        { \"label\": \"inputTextAlignment\", \"placeholder\": \"Left\", \"type\": \"text\", \"key\": \"inputTextAlignment\" },\n        { \"label\": \"tableScaleSize\", \"placeholder\": \"06 - Full Scale\", \"type\": \"text\", \"key\": \"tableScaleSize\" },\n        { \"label\": \"rowChoice\", \"placeholder\": \"\", \"type\": \"text\", \"key\": \"rowChoice\" },\n      ],\n      \"fieldProps\": [\n        { \"label\": \"tableId\", \"placeholder\": \"Emp_Table_01\", \"type\": \"text\", \"key\": \"tableId\" },\n        { \"label\": \"inputTextAlignment\", \"placeholder\": \"Left\", \"type\": \"text\", \"key\": \"inputTextAlignment\" },\n        { \"label\": \"helpText\", \"placeholder\": \"Enter help message here\", \"type\": \"text\", \"key\": \"helpText\" },\n        { \"label\": \"defaultValue\", \"placeholder\": \"Default Value\", \"type\": \"text\", \"key\": \"defaultValue\" },\n        { \"label\": \"referenceAnchor\", \"placeholder\": \"http://source/rangernxt_bio_data/74/d.doc\", \"type\": \"text\", \"key\": \"referenceAnchor\" },\n      ],\n      \"appearance\": [\n        {\n          label: 'Select Font', type: 'select', key: 'font', placeholder: 'Left', required: true,\n          options: ['Helvetica Neue', 'Arial', 'Times New Roman', 'Roboto']\n        },\n        { label: 'Font Color', type: 'color', key: 'fontColor', defaultValue: '', required: true },\n        {\n          label: 'Font Size', type: 'select', key: 'fontSize', defaultValue: '', required: true,\n          options: [\n            { label: 'Small', value: '12px' },\n            { label: 'Medium', value: '14px' },\n            { label: 'Large', value: '16px' }\n          ]\n        },\n\n        {\n          label: 'Font Width', type: 'select', key: 'fontWeight', defaultValue: '', required: true,\n          options: [\n            { value: '400', label: '400-Normal' },\n            { value: '500', label: '500-Medium' },\n            { value: '600', label: '600-Semi Bold' },\n            { value: '700', label: '700-Bold' }\n          ]\n        },\n        { label: 'Duplicate Field', type: 'button-toggle', key: 'duplicateField', defaultValue: false }\n      ]\n    },\n\n    ListColumn: this.createElementConfig({\n      customFieldProps: [\n        ...COMMON_OPTIONS_FIELD.map(opt => ({\n          type: 'list'\n        })),\n      ],\n      customElementProps:[\n        { label: \"headerLabels\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"label\" },\n        { label: \"apiName\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"apiName\" },\n    \n        {\n          label: \"Summary Column\", placeholder: \"Summary Column\", type: \"subQuestion\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"summaryColumn\",\n          subQuestion: [\n            { label: \"Type\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"fldType\" },\n            { label: \"Operation\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operation\" },\n            { label: \"Operands\", placeholder: \"Employee Details\", type: \"array\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operands\" },\n          ]\n        },\n    \n        {\n          label: \"Summary Row\", placeholder: \"Summary Row\", type: \"subQuestion\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"summaryRow\",\n          subQuestion: [\n            { label: \"Type\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"fldType\" },\n            { label: \"operation\", placeholder: \"operation\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operation\" },\n            { label: \"column\", placeholder: \"Employee Details\", type: \"radio\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"column\" },\n            { label: \"operands\", placeholder: \"Employee Details\", type: \"array\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operands\" },\n          ]\n        },\n      ]\n    }),    \n\n    DropdownColumn: this.createElementConfig({\n      customFieldProps: [\n\n        ...COMMON_OPTIONS_FIELD.map(opt => ({\n          ...opt,\n          type: 'dropdown'\n        })),\n      ],\n        customElementProps:[\n        { label: \"headerLabels\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"label\" },\n        { label: \"apiName\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"apiName\" },\n    \n        {\n          label: \"Summary Column\", placeholder: \"Summary Column\", type: \"subQuestion\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"summaryColumn\",\n          subQuestion: [\n            { label: \"Type\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"fldType\" },\n            { label: \"Operation\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operation\" },\n            { label: \"Operands\", placeholder: \"Employee Details\", type: \"array\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operands\" },\n          ]\n        },\n    \n        {\n          label: \"Summary Row\", placeholder: \"Summary Row\", type: \"subQuestion\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"summaryRow\",\n          subQuestion: [\n            { label: \"Type\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"fldType\" },\n            { label: \"operation\", placeholder: \"operation\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operation\" },\n            { label: \"column\", placeholder: \"Employee Details\", type: \"radio\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"column\" },\n            { label: \"operands\", placeholder: \"Employee Details\", type: \"array\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operands\" },\n          ]\n        },\n      ]\n    }),    \n\n    DateColumn: this.createElementConfig({\n      customFieldProps: [\n        ...COMMON_OPTIONS_FIELD.map(opt => ({\n          type: 'date'\n        })),\n      ],\n      customElementProps:[\n        { label: \"headerLabels\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"label\" },\n        { label: \"apiName\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"apiName\" },\n    \n        {\n          label: \"Summary Column\", placeholder: \"Summary Column\", type: \"subQuestion\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"summaryColumn\",\n          subQuestion: [\n            { label: \"Type\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"fldType\" },\n            { label: \"Operation\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operation\" },\n            { label: \"Operands\", placeholder: \"Employee Details\", type: \"array\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operands\" },\n          ]\n        },\n    \n        {\n          label: \"Summary Row\", placeholder: \"Summary Row\", type: \"subQuestion\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"summaryRow\",\n          subQuestion: [\n            { label: \"Type\", placeholder: \"Employee Details\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"fldType\" },\n            { label: \"operation\", placeholder: \"operation\", type: \"text\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operation\" },\n            { label: \"column\", placeholder: \"Employee Details\", type: \"radio\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"column\" },\n            { label: \"operands\", placeholder: \"Employee Details\", type: \"array\", targetArray: \"fieldsMeta\", targetArrayKey: \"uniqueIdentifier\", key: \"operands\" },\n          ]\n        },\n      ]\n    }),    \n\n    Checkbox: this.createElementConfig({\n      customFieldProps: COMMON_OPTIONS_FIELD.map(opt => ({\n        ...opt,\n        type: 'checkbox'\n      }))\n    }),\n  \n    Dropdown: this.createElementConfig({\n      customFieldProps: COMMON_OPTIONS_FIELD.map(opt => ({\n        ...opt,\n        type: 'dropdown'\n      }))\n    }),\n  \n    Radio: this.createElementConfig({\n      customFieldProps: COMMON_OPTIONS_FIELD.map(opt => ({\n        ...opt,\n        type: 'checkbox'\n      }))\n    }),\n\n    Text: this.createElementConfig(),\n\n    Calendar: this.createElementConfig(),\n\n    Date: this.createElementConfig(),\n\n    Time: this.createElementConfig(),\n\n    DateTime: this.createElementConfig(),\n\n    Email: this.createElementConfig(),\n\n    Book: this.createElementConfig(),\n    \n    List: this.createElementConfig(),\n\n    TextArea: this.createElementConfig(),\n\n    RichTextArea: this.createElementConfig(),\n    \n    Number: this.createElementConfig(),\n\n    Image: this.createElementConfig(),\n\n    Label: this.createElementConfig(),\n\n    Boolean: this.createElementConfig(),\n\n    File: this.createElementConfig(),\n\n    Icon: this.createElementConfig(), //MSM10JUL25 icon selector element\n  };\n  \n\n  constructor(private http: HttpClient, private formBuilderService: FormBuilderService,private templateService: TemplateService) { }\n\n  handleTemplateSelection(event: any) {\n    this.formElements = event.elements; \n    this.selectedTemplate = event.selectedTemplate;\n    this.templateService.formElements = this.formElements;\n    this.templateService.book = this.book;\n    this.templateService.unique_id = this.unique_id;\n  }\n  duplicateField(element: any) {\n    const elements = this.formBuilderService.getElements();\n    const copy = JSON.parse(JSON.stringify(element));\n    //AP-18APR25 Generate a new ID using the service's method\n    copy.id = this.formBuilderService.addElementWithId();\n    //copy.uuid = copy.id; \n    // AP-18APR25 Set the questionNumber to the next available number\n    copy.questionNumber = elements.length + 1;\n    this.formBuilderService.addElement(copy);\n  }\n  \n  saveData() {\n    this.templateService.formElements = this.formElements;\n    this.templateService.book = this.book;\n    this.templateService.unique_id = this.unique_id;\n    const updatedBook = this.templateService.download();\n  }\n\n  ngOnInit() {\n    //AP-10MAR25 Subscribes to header selection changes and updates the selected element  \n    this.formBuilderService.selectHeaderSubject$.subscribe(header => {\n      if (header) {\n        if (!this.selectedElement) {\n          this.selectedElement = {};\n        }\n        this.selectedElement['type'] = header;\n        this.headerSelect = true;\n        this.book = this.formBuilderService.getBook();\n        this.bookId = this.book.records[0].id;\n      }\n\n    });\n    this.formBuilderService.selectedElement$.subscribe(index => {\n      this.selectedElementIndex = index;\n      this.headerSelect = false;\n      if (index >= 0) {\n        const elements = this.formBuilderService.getElements();\n        this.selectedElement = elements[index];\n        if (this.selectedElement?.type === 'Table') {\n          this.selectedElement['fieldsMeta'] = typeof this.selectedElement.fieldsMeta === 'object' ? this.selectedElement.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);\n          this.selectColumn = this.formBuilderService.getSelectTableColumn() === undefined ? null : this.formBuilderService.getSelectTableColumn();\n        } else {\n          this.selectColumn = null\n        }\n      }\n      this.getProperties()\n    });\n    this.formBuilderService.selectHeaderSubject$.subscribe(action => {\n      if (action) {\n        this.book = this.formBuilderService.getBook();\n      }\n\n    });\n     if (this.from === 'formBuilder') {\n    const fieldsMeta = typeof this.element.fieldsMeta === 'string'\n      ? JSON.parse(this.element.fieldsMeta)\n      : this.element.fieldsMeta;\n\n    this.columns = fieldsMeta.map(column => ({\n      ...column,\n      type: column.fldType || 'Text' // Default to 'Text' if fldType not available\n    }));\n  }\n  }\n\n  // AP 23MAY25 - Add a new button with default styling\n  addNewButton(): void {\n    const lastPercent = this.book?.questionbook?.action?.slice(-1)[0]?.positionPercent || 0;\n    const buttons = {\n      name: '',\n      eventtoemit: '',\n      alt: '',\n      endpoint: '',\n      method: '',\n      id: this.formBuilderService.addElementWithId(),\n      positionPercent: lastPercent + 10,  \n      width: 100,\n      borderRadius: 6,\n      alignment: 'flex-start'\n    };\n    this.formBuilderService.updateQuestionBookActionButtons(buttons, 'add');\n  }\n  \n //AP 23MAY25 - Remove a button from the action list\n  removeButton(btn: any): void {\n    this.formBuilderService.updateQuestionBookActionButtons(btn,'remove');\n  }\n  //AP 23MAY25 - Update a specific property of a button\n  onButtonPropertyChange(index: number, key: string, value: any): void {\n    const buttons = this.getValueByPath('action') || [];\n    if (buttons[index]) {\n      buttons[index][key] = value;\n      this.setValueByPath('action', buttons); \n    }\n  }\n  //AP-10MAR25 Updates the title using the form builder service  \n  updateTitle(event: any) {\n    this.formBuilderService.updateTitle(event);\n  }\n  validateInput(value: string, type: string): boolean {\n    if (type === 'label' || type === 'placeholder') {\n      const regex = /^[a-zA-Z0-9\\s]*$/;\n      if (!regex.test(value)) {\n        this.errorMessage = 'Only letters and numbers are allowed';\n        return false;\n      }\n    }\n    this.errorMessage = '';\n    return true;\n  }\n\n  get fieldAsString(): string {\n    return this.bookSubtext.field.join(', '); //SKS28MAR25 Convert array to comma-separated string\n  }\n\n  updateField(value: string) {\n    this.bookSubtext.field = value.split(',').map(item => item.trim()); //SKS28MAR25 Convert string back to array\n  }\n  // SKS20MAR25 Getter to filter columns, excluding the one with currentUniqueIdentifier\n  get filteredColumns(): any[] {\n    const targetArray = this.getDataByPath('fieldsMeta');\n    return targetArray.filter(column => column.uniqueIdentifier !== this.selectColumn);\n  }\n\n  // SKS20MAR25 Method to handle checkbox changes\n  onCheckboxChange(targetArrayPath: any, targetArrayKey: any, key: any, apiName: string, isChecked: boolean) {\n    const targetArray = this.getDataByPath(targetArrayPath);\n    const foundItem = targetArray.find(item => item[targetArrayKey] === this.selectColumn);\n    foundItem[key] = foundItem[key] ? foundItem[key] : [];\n    if (isChecked) {\n      // Add apiName to selectedApiNames if not already present\n      if (!foundItem[key].includes(apiName)) {\n        foundItem[key].push(apiName);\n      }\n    } else {\n      // Remove apiName from selectedApiNames if present\n      const index = foundItem[key].indexOf(apiName);\n      if (index > -1) {\n        foundItem[key].splice(index, 1);\n      }\n    }\n    // this.updateElement(this.selectedElement)\n  }\n  updateProperty(key: string, value: any) {\n    if (this.selectedElementIndex >= 0) {\n      if (this.selectedElement.type === 'Table') {\n        this.setValueByPath(key, value);\n      }\n      if (key === 'questionText' || key === 'question') {\n        if (!this.validateInput(value, key)) {\n          return;\n        }\n      }\n      if (key === 'questionNumber') {\n        //AP-10MAR25 Convert the value to an integer  \n        const newOrder = parseInt(value, 10);\n        if (isNaN(newOrder)) return;\n\n        //AP-10MAR25 Update the element's questionNumber in the form builder service  \n        this.formBuilderService.updateElement(this.selectedElementIndex, { questionNumber: newOrder });\n\n        //AP-10MAR25 Sort elements based on the updated questionNumber  \n        this.formBuilderService.sortElementsByOrder();\n      }\n\n\n      if (key === 'fontWeight') {\n        this.selectedElement[key] = value;\n      }\n\n      const update = { [key]: value };\n      this.formBuilderService.updateElement(this.selectedElementIndex, update);\n\n      // Special handling for font and font weight\n      if (key === 'font') {\n        this.selectedElement.font = value; // Directly update the selected element's font property\n      }\n    }\n  }\n  // SKS21MAR25 radio button click handler\n  onRadioChange(targetArrayPath: any, targetArrayKey: any, key: any, value: any) {\n    if (this.selectedElement.type === 'Table') {\n      // this.updateValueByArrayPath(targetArray, targetArrayKey, this.selectColumn, key, value);\n      if (this.selectedElement?.type === 'Table') {\n        this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);\n      }\n      const targetArray = this.getDataByPath(targetArrayPath);\n      // console.log(\"targetArray\", targetArray, \"targetArrayKey\", targetArrayKey, \"selectColumn\", this.selectColumn, \"keyPath\", key, \"event\", value)\n      if (!Array.isArray(targetArray)) return;\n\n      // Find the object that matches selectColumn\n      let foundItem = targetArray.find(item => item[targetArrayKey] === this.selectColumn);\n\n      // If the item does not exist, create and add it\n      if (!foundItem) {\n        foundItem = { [targetArrayKey]: this.selectColumn }; // Ensure the key exists\n        targetArray.push(foundItem);\n      }\n\n      // Traverse and set the value\n      const keys = key.split('.');\n      let obj = foundItem;\n\n      for (let i = 0; i < keys.length - 1; i++) {\n        const key = keys[i];\n        if (!obj[key]) obj[key] = {}; // Initialize missing objects\n        obj = obj[key];\n      }\n\n      // Assign the value based on input type\n      const lastKey = keys[keys.length - 1];\n\n      obj[lastKey] = value;\n      if (this.selectedElement?.type === 'Table') {\n        this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? JSON.stringify(this.selectedElement?.fieldsMeta) : this.selectedElement['fieldsMeta'] || [];\n      }\n      this.updateElement(this.selectedElement)\n    } else {\n      this.setValueByPath(key, value); // Fallback for non-Table elements, if applicable\n    }\n  }\n  onRequiredChange(value: boolean) {\n    if (this.selectedElement) {\n      let label = this.selectedElement.label ? this.selectedElement.label.replace(/\\s*\\*+$/, '') : 'Label';\n\n      if (value) {\n        label = `${label} *`;\n      }\n\n      this.updateProperty('label', label);\n      this.updateProperty('isOptional', value);\n    }\n  }\n\n  // Toggle for Sub Questions (Each prop has its own state)\n  toggleSubQuestion(prop: any) {\n    prop.isExpanded = !prop.isExpanded;\n  }\n\n  // Separate Toggle for Style Section\n  isStyleExpanded = false;\n\n  toggleStyleSection() {\n    this.isStyleExpanded = !this.isStyleExpanded;\n  }\n\n  getProperties() {\n    if (!this.selectedElement) return null;\n\n    // AP-10MAR25 Retrieve the latest book data\n    this.book = this.formBuilderService.getBook();\n\n    // AP-10MAR25 If the selected element is a 'Header', return its properties\n    if (this.selectedElement?.type === 'Header') {\n      return this.elementProperties['Header'];\n    }\n    if (this.selectedElement?.type === 'Table') {\n      if (this.selectColumn !== null) {\n        // Get the column metadata\n        const fieldsMeta = Array.isArray(this.selectedElement.fieldsMeta) \n          ? this.selectedElement.fieldsMeta \n          : JSON.parse(this.selectedElement.fieldsMeta || '[]');\n        \n        const column = fieldsMeta.find((c: any) => c.uniqueIdentifier === this.selectColumn);\n        \n        if (column) {\n          // Return specific properties based on column type\n          switch (column.fldType) {\n            case 'List':\n              return this.elementProperties['ListColumn'];\n            case 'Dropdown':\n              return this.elementProperties['DropdownColumn'];\n            case 'Date':\n              return this.elementProperties['DateColumn'];\n            default:\n              return this.elementProperties['TableColumn'];\n          }\n        }\n        return this.elementProperties[this.selectedElement?.type] || this.elementProperties['DefaultColumn'];\n      }\n    }\n    return this.elementProperties[this.selectedElement?.type];\n  }\n  // AP-28MAR25 Add an option with a unique UUID\n  addOption(options: any[]) {\n    const unique_id = uuidv4();\n    options.push({ id: unique_id,value: '' });\n  }\n\n  // AP-28MAR25 Function to remove an option based on its unique UUID\n  removeOption(options: any[], id: string) {\n    const index = options.findIndex(option => option.id === id);\n    if (index !== -1) {\n      options.splice(index, 1);\n    }\n  }\n\n  handleButtonClick() {\n    this.formButtonHandler.emit(this.formBuilderService.downloadElement());\n  }\n  handleTemplateSave() {\n    this.templateSaveHandler.emit(this.templateService.download());\n  }\n  // SKS13MAR25 active tab select\n  setActiveTab(tab: string): void {\n    this.activeTab = tab;\n  }\n  // SKS13MAR25 table property save\n  onSave(): void {\n    // console.log('Saving data:');\n  }\n  // SKS13MAR25 table property reset\n  onCancel(): void {\n    // console.log('Operation cancelled');\n  }\n\n  updateProperties(elementType: string): void {\n    // Reset alignment and styles if switching elements\n    this.selectedAlign = 'align-left';\n    this.selectedStyles = [];\n\n    // Fetch the properties of the selected element type\n\n    const properties = this.getProperties();\n    if (properties) {\n      // Make sure the properties are dynamically updated\n    }\n  }\n\n  onAlignSelect(value: string): void {\n    if (this.selectedElement) {\n      this.selectedElement.textAlign = value;\n    }\n  }\n\n  onStyleSelect(value: string): void {\n    if (!this.selectedElement) return;\n\n    // Initialize styles array if it doesn't exist\n    this.selectedElement.styles = this.selectedElement.styles || [];\n\n    // Ensure only one style is selected at a time\n    this.selectedElement.styles = [value];\n\n  }\n\n  isStyleActive(value: string): boolean {\n    return this.selectedElement?.styles?.includes(value) || false;\n  }\n  getValueByPath(path: string): any {\n    if (this.selectedElement?.type === 'Table') {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);\n    }\n\n    this.selectedElement['subText'] = \n        typeof this.selectedElement?.subText === 'object'\n            ? this.selectedElement.subText\n            : (typeof this.selectedElement?.subText === 'string' && this.selectedElement.subText.trim() !== '' \n                ? JSON.parse(this.selectedElement.subText) \n                : {});\n\n    if (this.selectedElement['subText'] && Array.isArray(this.selectedElement['subText'].field)) {\n        // Convert array to a comma-separated string\n        this.selectedElement['subText'].field = this.selectedElement['subText'].field.join(', ');\n    }  \n    return path.split('.').reduce((obj, key) => obj?.[key] ?? '', this.selectedElement);\n  }\nisAnotherIdSelected(): boolean {\n  return this.formBuilderService.getElements()\n    .some((el, i) => i !== this.selectedElementIndex && el.primaryKey);\n}\n  onToggleChange(path: string, event: Event): void {\n    const checked = (event.target as HTMLInputElement).checked;\n    this.setValueByPath(path, checked);\n    if (this.selectedElement?.type === 'Table') {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? JSON.stringify(this.selectedElement?.fieldsMeta) : this.selectedElement['fieldsMeta'] || [];\n    }\n    this.updateElement(this.selectedElement)\n  }\n\n  setValueByPath(path: string, value: any): void {\n\n    if (this.selectedElementIndex >= 0) {\n      if (path === 'questionText' || value === 'question') {\n        if (!this.validateInput(value, value)) {\n        }\n      }\n      if (path === 'questionNumber') {\n        //AP-10MAR25 Convert the value to an integer  \n        const newOrder = parseInt(value, 10);\n        if (isNaN(newOrder)) return;\n\n        //AP-10MAR25 Update the element's questionNumber in the form builder service  \n        this.formBuilderService.updateElement(this.selectedElementIndex, { questionNumber: newOrder });\n\n        //AP-10MAR25 Sort elements based on the updated questionNumber  \n        this.formBuilderService.sortElementsByOrder();\n      }\n      if (path === 'fontWeight') {\n        // console.log('Font weight selected:', value);\n        this.selectedElement[path] = value;\n      }\n      // Special handling for font and font weight\n      if (path === 'font') {\n        this.selectedElement.font = value; // Directly update the selected element's font property\n      }\n    }\n\n    const keys = path.split('.');\n    let obj = this.selectedElement;\n    if (this.selectedElement?.type === 'Table') {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);\n    }\n    this.selectedElement['subText'] = typeof this.selectedElement?.subText === 'object' ? this.selectedElement?.subText : JSON.parse(this.selectedElement['subText']);\n\n    keys.forEach((key, index) => {\n      if (!obj[key]) {\n        // Check if the next key is a number (array index)\n        const nextKey = keys[index + 1];\n        obj[key] = isNaN(Number(nextKey)) ? {} : [];\n      }\n      if (index === keys.length - 1) {\n        obj[key] = value; // Assign value to last key\n      } else {\n        obj = obj[key]; // Move deeper\n      }\n    });\n    this.formBuilderService.elementUpdate(this.selectedElementIndex, this.selectedElement);\n  }\n  getStyleKeys(): string[] {\n    if (!this.selectedElement || !this.selectedElement.style || typeof this.selectedElement.style !== 'object') {\n      return [];\n    }\n    return Object.keys(this.selectedElement.style);\n  }  \n\n  // SKS19MAR25 update element in servies\n  updateElement(element: any) {\n    this.formBuilderService.elementUpdate(this.selectedElementIndex, element)\n    if (this.selectedElement?.type === 'Table' && this.selectColumn) {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement.fieldsMeta || []);\n      const foundItem = this.selectedElement['fieldsMeta'].find(item => item['uniqueIdentifier'] === this.selectColumn);\n      this.formBuilderService.setSelectedTableElement(this.selectedElementIndex, { column: foundItem?.apiName || null });\n    }\n  }\n  // SKS19MAR25 get value from array\n  getValueByArrayPath(targetArrayPath: any, targetArrayKey: string, selectColumn: string, keyPath: string): any {\n    if (this.selectedElement?.type === 'Table') {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);\n    }\n    const targetArray = this.getDataByPath(targetArrayPath);\n    if (!Array.isArray(targetArray)) return '';\n\n    const foundItem = targetArray.find(item => item[targetArrayKey] === selectColumn);\n\n    return foundItem ? keyPath.split('.').reduce((obj, key) => obj?.[key], foundItem) : '';\n  }\n\n  //SKS19MAR25 Function to update a value in an array dynamically\n  updateValueByArrayPath(targetArrayPath: any, targetArrayKey: string, selectColumn: string, keyPath: string, event: Event): void {\n    if (this.selectedElement?.type === 'Table') {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);\n    }\n    const targetArray = this.getDataByPath(targetArrayPath);\n    // console.log(\"targetArray\", targetArray, \"targetArrayKey\", targetArrayKey, \"selectColumn\", selectColumn, \"keyPath\", keyPath, \"event\", event)\n    if (!Array.isArray(targetArray)) return;\n\n    // Find the object that matches selectColumn\n    let foundItem = targetArray.find(item => item[targetArrayKey] === selectColumn);\n\n    // If the item does not exist, create and add it\n    if (!foundItem) {\n      foundItem = { [targetArrayKey]: selectColumn }; // Ensure the key exists\n      targetArray.push(foundItem);\n    }\n\n    // Traverse and set the value\n    const keys = keyPath.split('.');\n    let obj = foundItem;\n\n    for (let i = 0; i < keys.length - 1; i++) {\n      const key = keys[i];\n      if (!obj[key]) obj[key] = {}; // Initialize missing objects\n      obj = obj[key];\n    }\n\n    // Assign the value based on input type\n    const lastKey = keys[keys.length - 1];\n    const inputElement = event.target as HTMLInputElement;\n\n    obj[lastKey] = inputElement?.type === 'checkbox' ? inputElement.checked : inputElement.value;\n    if (keyPath === 'summaryColumn') {\n      if (obj['summaryRow'] === true) {\n        obj['summaryRow'] = false;\n      }\n      if (inputElement.checked === true) {\n        obj['fldType'] = 'calculation'\n      } else {\n        obj['fldType'] = 'Text'\n      }\n    }\n    if (keyPath === 'summaryRow') {\n      if (obj['summaryColumn'] === true) {\n        obj['summaryColumn'] = false;\n      }\n      if (inputElement.checked === true) {\n        obj['fldType'] = 'calculation'\n      } else {\n        obj['fldType'] = 'Text'\n      }\n    }\n    if (this.selectedElement?.type === 'Table') {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? JSON.stringify(this.selectedElement?.fieldsMeta) : this.selectedElement['fieldsMeta'] || [];\n    }\n    this.updateElement(this.selectedElement)\n  }\n  getDataByPath(path: string): any {\n    if (this.selectedElement?.type === 'Table') {\n      this.selectedElement['fieldsMeta'] = typeof this.selectedElement?.fieldsMeta === 'object' ? this.selectedElement?.fieldsMeta : JSON.parse(this.selectedElement['fieldsMeta'] || []);\n    }\n    return path.split('.').reduce((obj, key) => obj?.[key], this.selectedElement);\n  }\n\n  // SKS25MAR25 image edit functions\n  flipHorizontal() {\n    this.transform = {\n      ...this.transform,\n      flipH: !this.transform.flipH\n    };\n  }\n\n  flipVertical() {\n    this.transform = {\n      ...this.transform,\n      flipV: !this.transform.flipV\n    };\n  }\n\n  resetImage() {\n    this.canvasRotation = 0;\n    this.cropper = undefined;\n    this.transform = {\n      translateUnit: 'px',\n      scale: 1,\n      rotate: 0,\n      flipH: false,\n      flipV: false,\n      translateH: 0,\n      translateV: 0\n    };\n  }\n\n  zoomOut() {\n    this.transform = {\n      ...this.transform,\n      scale: this.transform.scale! - .1\n    };\n  }\n\n  zoomIn() {\n    this.transform = {\n      ...this.transform,\n      scale: this.transform.scale! + .1\n    };\n  }\n  rotateLeft() {\n    this.loading = true;\n    setTimeout(() => { // Use timeout because rotating image is a heavy operation and will block the ui thread\n      this.canvasRotation--;\n      this.flipAfterRotate();\n    });\n  }\n\n  rotateRight() {\n    this.loading = true;\n    setTimeout(() => {\n      this.canvasRotation++;\n      this.flipAfterRotate();\n    });\n  }\n\n  moveLeft() {\n    this.transform = {\n      ...this.transform,\n      translateH: this.transform.translateH! - 1\n    };\n  }\n\n  moveRight() {\n    this.transform = {\n      ...this.transform,\n      translateH: this.transform.translateH! + 1\n    };\n  }\n\n  moveDown() {\n    this.transform = {\n      ...this.transform,\n      translateV: this.transform.translateV! + 1\n    };\n  }\n\n  moveUp() {\n    this.transform = {\n      ...this.transform,\n      translateV: this.transform.translateV! - 1\n    };\n  }\n  private flipAfterRotate() {\n    const flippedH = this.transform.flipH;\n    const flippedV = this.transform.flipV;\n    this.transform = {\n      ...this.transform,\n      flipH: flippedV,\n      flipV: flippedH,\n      translateH: 0,\n      translateV: 0\n    };\n  }\n  async imageCropped(event: any) {\n    try {\n      const base64 = await this.convertBlobToBase64(event.objectUrl);\n      this.selectedElement.imageData = base64;\n      await this.formBuilderService.elementUpdate(this.selectedElementIndex, this.selectedElement);\n      // console.log('CROPPED', event);\n    } catch (error) {\n      console.error(\"Error in imageCropped:\", error);\n    }\n  }\n\n\n  cropperReady(sourceImageDimensions: Dimensions) {\n    // console.log('Cropper ready', sourceImageDimensions);\n    this.loading = false;\n  }\n  // SKS25MAR25 blob to base 64 converter\n  convertBlobToBase64(objectUrl: string): Promise<string> {\n    return fetch(objectUrl)\n      .then(response => response.blob())\n      .then(blob => {\n        return new Promise((resolve, reject) => {\n          const reader = new FileReader();\n          reader.readAsDataURL(blob);\n          reader.onloadend = () => resolve(reader.result as string);\n          reader.onerror = error => reject(error);\n        });\n      });\n  }\n  // SKS28MAR25 qb book update in book element\n  childEventCapture(event: any) {\n    this.selectedElement.qbReference = event.valueObj?.name\n    this.selectedElement.qbReferenceQuestions = event.valueObj?.jsonBody\n    this.formBuilderService.elementUpdate(this.selectedElementIndex, this.selectedElement);\n  }\n\n  // SKS28MAR25 book search dropdown emit\n  linkToggleDropdown(event: Event) {\n    event.stopPropagation();\n    this.isLinkDropdownOpen = !this.isLinkDropdownOpen;\n  }\n\n  // SKS28MAR25 book search dropdown close\n  @HostListener('document:click', ['$event'])\n  onClickOutside(event: Event) {\n    if (this.dropdown && !this.dropdown.nativeElement.contains(event.target)) {\n      this.isLinkDropdownOpen = false;\n    }\n  }\n  // AP-28MAR25 When drag starts, store the index\n  onDragStart(event: DragEvent, id: string) {\n    this.draggedId = id;\n    event.dataTransfer!.effectAllowed = \"move\";\n  }\n\n  // AP-28MAR25 Prevent default behavior to allow drop\n  onDragOver(event: DragEvent) {\n    event.preventDefault();\n  }\n\n  // AP-28MAR25 Swap the dragged item with the dropped position\n  onDrop(event: DragEvent, key: string) {\n    event.preventDefault();\n    const targetId = (event.target as HTMLElement).closest(\".option-items\")?.getAttribute(\"data-id\");\n\n    if (this.draggedId && targetId && this.draggedId !== targetId) {\n      const options = this.selectedElement[key];\n\n      [options[this.draggedId], options[targetId]] = [options[targetId], options[this.draggedId]];\n      // Find the indexes of the dragged and target options\n      const draggedIndex = options.findIndex(option => option.id === this.draggedId);\n      const targetIndex = options.findIndex(option => option.id === targetId);\n\n      // Swap the items\n      if (draggedIndex !== -1 && targetIndex !== -1) {\n        [options[draggedIndex], options[targetIndex]] = [options[targetIndex], options[draggedIndex]];\n      }\n    }\n\n    this.draggedId = null;\n  }\n}\n","<!-- AP 22JAN25 - Field and Element Properties -->\n<div class=\"container\">\n  <div class=\"tabs\">\n    <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'attributes'}\" (click)=\"setActiveTab('attributes')\">\n      Attributes\n    </div>\n    <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'property'}\" (click)=\"setActiveTab('property')\">\n      Property\n    </div>\n    <div class=\"tab\" [ngClass]=\"{'active': activeTab === 'appearance'}\" (click)=\"setActiveTab('appearance')\">\n      Appearance\n    </div>\n  </div>\n\n  <div class=\"tab-content\" *ngIf=\"activeTab === 'attributes'\">\n    <!-- Element Properrties -->\n    <!-- AP-06MAR25 Select element type show -->\n    <div *ngIf=\"getProperties()\">\n      <!-- SKS25MAR25 this is for image edit -->\n      <div *ngIf=\"selectedElement?.type === 'Image'\">\n        <image-cropper *ngIf=\"selectedElement.imageData \" [imageBase64]=\"selectedElement.orgImageData\"\n          [disabled]=\"false\" [alignImage]=\"alignImage\" [roundCropper]=\"roundCropper\" [backgroundColor]=\"'white'\"\n          imageAltText=\"Alternative image text\" [allowMoveImage]=\"false\" [hideResizeSquares]=\"false\"\n          [canvasRotation]=\"canvasRotation\" [aspectRatio]=\"aspectRatio\" [containWithinAspectRatio]=\"false\"\n          [maintainAspectRatio]=\"false\" [cropperStaticWidth]=\"cropperStaticWidth\"\n          [cropperStaticHeight]=\"cropperStaticHeight\" [cropperMinWidth]=\"cropperMinWidth\"\n          [cropperMinHeight]=\"cropperMinHeight\" [cropperMaxWidth]=\"cropperMaxWidth\"\n          [cropperMaxHeight]=\"cropperMaxHeight\" [resetCropOnAspectRatioChange]='true' [(cropper)]=\"cropper\"\n          [(transform)]=\"transform\" [onlyScaleDown]=\"true\" output=\"blob\" format=\"png\"\n          (imageCropped)=\"imageCropped($event)\" (cropperReady)=\"cropperReady($event)\"></image-cropper>\n        <div *ngIf=\"selectedElement.imageData \" style=\"display: flex; gap: 2px;\">\n          <div class=\"cursor-pointer logo-icon\" (click)=\"rotateLeft()\" title=\"Rotate Left\">⟲</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"rotateRight()\" title=\"Rotate Right\">⟳</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"zoomOut()\" title=\"Zoom Out\">-</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"zoomIn()\" title=\"Zoom In\">+</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"moveLeft()\" title=\"Move Left\">←</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"moveRight()\" title=\"Move Right\">→</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"moveUp()\" title=\"Move Up\">↑</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"moveDown()\" title=\"Move Down\">↓</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"flipHorizontal()\" [class.enabled]=\"transform.flipH\"\n            title=\"Flip Horizontally\">↔</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"flipVertical()\" [class.enabled]=\"transform.flipV\"\n            title=\"Flip Vertically\">↕</div>\n          <div class=\"cursor-pointer logo-icon\" (click)=\"resetImage()\" title=\"Reset\">×</div>\n        </div>\n      </div>\n      <!-- SKS28MAR25 search book specific for book element -->\n      <div *ngIf=\"selectedElement?.type === 'Book'\">\n        <label class=\"text-sm\">Search Book</label>\n        <div style=\"display: flex; gap: 2px; align-items: center; justify-content: center;\">\n          <nxt-search-box [question]=\"selectedElement\" [apiMeta]=\"bookSubtext\" [placeHolderText]=\"'Search...'\"\n            (searchValueChange)=\"childEventCapture($event.value)\">\n          </nxt-search-box>\n          <div class=\"link-icon\">\n            <svg (click)=\"linkToggleDropdown($event)\" fill=\"#000000\" version=\"1.1\" id=\"Capa_1\"\n              xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" width=\"24px\" height=\"24px\"\n              viewBox=\"0 0 450 450\" xml:space=\"preserve\">\n              <g>\n                <g>\n                  <g>\n                    <path d=\"M318.15,230.195l77.934-77.937c31.894-31.892,31.894-83.782-0.004-115.674l-12.66-12.66\n          c-31.893-31.896-83.78-31.896-115.674-0.004l-77.937,77.934c-17.588,17.588-25.457,41.264-23.646,64.311\n          c-23.045-1.813-46.722,6.056-64.308,23.647L23.92,267.748c-31.894,31.889-31.894,83.779,0,115.674l12.664,12.662\n          c31.893,31.893,83.783,31.893,115.674,0l77.935-77.936c17.592-17.59,25.459-41.266,23.647-64.309\n          C276.884,255.654,300.56,247.783,318.15,230.195z M202.653,290.605l-77.936,77.938c-16.705,16.703-43.889,16.703-60.59,0\n          l-12.666-12.666c-16.705-16.701-16.703-43.885,0-60.594l77.936-77.932c14.14-14.141,35.779-16.306,52.226-6.516l-32.302,32.307\n          c-7.606,7.604-7.606,19.938,0,27.541c7.605,7.607,19.937,7.607,27.541,0l32.306-32.303\n          C218.959,254.828,216.795,276.469,202.653,290.605z M238.382,209.169l32.299-32.306c7.608-7.602,7.608-19.935,0-27.538\n          c-7.604-7.61-19.936-7.61-27.541-0.004l-32.303,32.303c-9.791-16.446-7.627-38.087,6.514-52.226l77.935-77.935\n          c16.707-16.707,43.89-16.707,60.594,0l12.664,12.664c16.705,16.705,16.705,43.886,0,60.591l-77.936,77.937\n          C276.468,216.797,254.828,218.959,238.382,209.169z\" />\n                    <path d=\"M343.466,261.465c-45.287,0-82,36.713-82,82s36.713,82,82,82c45.286,0,82-36.713,82-82S388.753,261.465,343.466,261.465z\n           M372.505,333.564l-56.046,56.104c-0.239,0.238-0.536,0.41-0.862,0.496l-22.315,5.85c-0.649,0.168-1.347-0.02-1.822-0.494\n          c-0.477-0.479-0.666-1.172-0.496-1.824l5.826-22.318c0.084-0.326,0.256-0.627,0.494-0.863l56.047-56.104\n          c0.742-0.742,1.945-0.744,2.688-0.002l4.548,4.541c0.739,0.74,0.741,1.943,0,2.688l-37.433,37.471l4.709,4.703l37.435-37.471\n          c0.739-0.742,1.94-0.742,2.682-0.002l4.55,4.541C373.25,331.617,373.25,332.822,372.505,333.564z M395.472,310.574l-17,17.018\n          c-0.739,0.744-1.942,0.744-2.685,0.002l-16.489-16.475c-0.744-0.74-0.744-1.943-0.002-2.688l17-17.02\n          c0.741-0.74,1.944-0.74,2.688-0.002l16.487,16.477C396.216,308.629,396.216,309.832,395.472,310.574z\" />\n                  </g>\n                </g>\n              </g>\n            </svg>\n            <div class=\"link-dropdown-menu\" *ngIf=\"isLinkDropdownOpen\" #dropdown>\n              <label>Endpoint:</label>\n              <input type=\"text\" [(ngModel)]=\"bookSubtext.endpoint\" />\n\n              <label>Variable:</label>\n              <input type=\"text\" [(ngModel)]=\"bookSubtext.variable\" />\n\n              <label>Field:</label>\n              <input type=\"text\" [ngModel]=\"fieldAsString\" (ngModelChange)=\"updateField($event)\" />\n\n              <label>Default Field:</label>\n              <input type=\"text\" [(ngModel)]=\"bookSubtext.defaultField\" />\n            </div>\n          </div>\n        </div>\n      </div>\n      <ng-container *ngFor=\"let prop of getProperties().elementProps\">\n        <div class=\"form-group\">\n          <label *ngIf=\"prop.type !== 'checkbox' && prop.type !== 'subQuestion'\" class=\"text-sm\">{{ prop.label\n            }}</label>\n\n          <!-- Text Input -->\n          <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n            [value]=\"headerSelect ? book.records[0].title : (selectedElement?.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ) :  getValueByPath(prop.key))\"\n            (input)=\"headerSelect ? updateTitle($event.target.value) : (selectedElement?.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : setValueByPath(prop.key, $event.target.value) ) : setValueByPath(prop.key, $event.target.value))\"\n            [class.read-only]=\"selectedElement.readOnly\" [readonly]=\"selectedElement.readOnly\" />\n\n          <!-- help text -->\n          <div *ngIf=\"prop.key === 'helpText' && selectedElement.helpText\">{{\n            selectedElement.helpText }}</div>\n\n          <!-- questionNumber -->\n          <input *ngIf=\"prop.key === 'questionNumber'\" type=\"number\" [value]=\"selectedElement.questionNumber\"\n            (input)=\"setValueByPath('questionNumber', $event.target.value)\" />\n\n          <!-- file -->\n          <!-- Add this inside the elementProps loop where other inputs are rendered -->\n          <select *ngIf=\"prop.type === 'select' && prop.key === 'supportType'\" [value]=\"selectedElement[prop.key]\"\n            (change)=\"setValueByPath(prop.key, $event.target.value)\">\n            <option value=\"\">Select file category</option>\n            <option *ngFor=\"let option of prop.options\" [value]=\"option.value\"> {{ option.label }} </option>\n          </select>\n\n          <div *ngIf=\"prop.type === 'checkbox'\">\n\n            <input type=\"checkbox\"\n              [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ):  getValueByPath(prop.key) \"\n              (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n              <span class=\"toggle-label\" style=\"padding-left: 10px;\">{{ prop.label }}</span>\n\n          </div>\n\n          <!--AP-11JUN25 - Show this section only if the property key is 'primaryKey' -->\n          <div *ngIf=\"prop.key === 'primaryKey'\">\n            <input type=\"checkbox\" [checked]=\"getValueByPath('primaryKey')\" [disabled]=\"!getValueByPath('primaryKey') && isAnotherIdSelected()\" (change)=\"onToggleChange('primaryKey', $event)\"/>\n            <span class=\"toggle-label\" style=\"padding-left: 10px;\">primaryKey</span>\n          </div>\n\n          <!-- SKS20MAR25 Subquestion Type -->\n          <div *ngIf=\"prop.type === 'subQuestion'\">\n            <div style=\"display: flex; flex-direction: row; gap: 10px; align-items: center;\">\n              <div>{{ prop.label }}</div>\n              <input type=\"checkbox\"\n                [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ):  getValueByPath(prop.key) \"\n                (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n            </div>\n            <!-- SKS20MAR25 Render subquestions when checkbox is checked -->\n            <div\n              *ngIf=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ):  getValueByPath(prop.key)\">\n              <div *ngFor=\"let subProp of prop.subQuestion\"\n                style=\"background-color: #e7f2ff; padding: 8px; border-radius: 4px;\">\n                <div class=\"form-group\">\n                  <label>{{ subProp.label }}</label>\n                  <div *ngIf=\"subProp.type === 'array'\">\n                    <!-- Iterate over filtered columns to display checkboxes -->\n                    <div *ngFor=\"let column of filteredColumns\">\n                      <label>\n                        <input type=\"checkbox\"\n                          [checked]=\"subProp.operands ? subProp.operands.includes(column.apiName) : false\"\n                          (change)=\"onCheckboxChange(subProp.targetArray,subProp.targetArrayKey,subProp.key, column.apiName, $event.target.checked)\">\n                        {{ column.apiName }}\n                      </label>\n                    </div>\n                  </div>\n                  <!-- SKS21MAR25 New radio type -->\n                  <div *ngIf=\"subProp.type === 'radio'\">\n                    <div *ngFor=\"let column of filteredColumns\">\n                      <label>\n                        <input type=\"radio\" [name]=\"subProp.key\" [value]=\"column.apiName\"\n                          [checked]=\"getValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key) === column.apiName\"\n                          (change)=\"onRadioChange(subProp.targetArray, subProp.targetArrayKey, subProp.key, column.apiName)\">\n                        {{ column.apiName }}\n                      </label>\n                    </div>\n                  </div>\n                  <!-- SKS20MAR25 Add more subproperty types as needed -->\n                  <!-- Inside the subProp ngFor loop -->\n                  <input *ngIf=\"subProp.type === 'text'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n                    [value]=\"selectedElement.type === 'Table' ? (subProp.targetArray ? getValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key ) : getValueByPath(subProp.key) ) : getValueByPath(subProp.key)\"\n                    (input)=\"selectedElement.type === 'Table' ? (subProp.targetArray ? updateValueByArrayPath(subProp.targetArray, subProp.targetArrayKey, selectColumn, subProp.key, $event) : setValueByPath(subProp.key, $event.target.value) ) : setValueByPath(subProp.key, $event.target.value)\" />\n\n                  <input *ngIf=\"subProp.type === 'boolean'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n                    (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n                  <select *ngIf=\"subProp.type === 'select'\" [value]=\"getValueByPath(subProp.key)\"\n                    (change)=\"setValueByPath(subProp.key, $event.target.value)\">\n                    <option *ngFor=\"let option of subProp.options\" [value]=\"option\">{{ option }}</option>\n                  </select>\n                  <input *ngIf=\"subProp.type === 'checkbox'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n                    (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n                </div>\n              </div>\n            </div>\n          </div>\n\n          <!-- SKS20MAR25 Text Align Buttons -->\n          <div *ngIf=\"prop.type === 'align'\">\n            <button *ngFor=\"let option of prop.options\" (click)=\"onAlignSelect(option.value)\"\n              [class.active]=\"selectedElement?.textAlign === option.value\" [title]=\"option.value\">\n              <img [src]=\"'../assets/icons/' + option.icon + '.svg'\" [alt]=\"option.value\" class=\"icon-size\" />\n            </button>\n          </div>\n\n          <div *ngIf=\"prop.type === 'style'\">\n            <button *ngFor=\"let option of prop.options\" (click)=\"onStyleSelect(option.value)\"\n              [class.active]=\"isStyleActive(option.value)\" [title]=\"option.value\">\n              <img [src]=\"'../assets/icons/' + option.icon + '.svg'\" [alt]=\"option.value\" class=\"icon-size\" />\n            </button>\n          </div>\n\n          <!-- Field Size Controls -->\n          <!-- AP 25FEB25 - Change key size -->\n          <div *ngIf=\"prop.key === 'size'\">\n            <input type=\"number\" [value]=\"selectedElement?.size\" (input)=\"setValueByPath('size', $event.target.value)\"\n              class=\"size-input\" />\n          </div>\n\n          <!-- AP-19MAR25 Line Properties -->\n          <!-- Padding Top -->\n          <div *ngIf=\"prop.key === 'paddingTop'\">\n            <input type=\"number\" [placeholder]=\"prop.placeholder\" [value]=\"selectedElement?.paddingTop\"\n              (input)=\"setValueByPath('paddingTop', $event.target.value)\" />\n          </div>\n\n          <!-- Padding Bottom -->\n          <div *ngIf=\"prop.key === 'paddingBottom'\">\n            <input type=\"number\" [placeholder]=\"prop.placeholder\" [value]=\"selectedElement?.paddingBottom\"\n              (input)=\"setValueByPath('paddingBottom', $event.target.value)\" />\n          </div>\n\n          <!-- Line Style -->\n          <div *ngIf=\"prop.key === 'lineStyle'\">\n            <select [value]=\"selectedElement?.lineStyle\" (change)=\"setValueByPath('lineStyle', $event.target.value)\">\n              <option *ngFor=\"let option of prop.options\" [value]=\"option\">{{ option }}</option>\n            </select>\n          </div>\n\n          <!-- Color -->\n          <div *ngIf=\"prop.key === 'color'\">\n            <input type=\"color\" [value]=\"selectedElement?.color\"\n              (input)=\"setValueByPath('color', $event.target.value)\" />\n          </div>\n\n        </div>\n        \n         <!-- AP 23MAY25 - Action Button -->\n        <div *ngIf=\"prop.key === 'title'\" style=\"margin-top: 15px; border: 1px solid #ddd; padding: 20px; border-radius: 12px; background-color: #f9f9f9;\">\n          <div class=\"form-header\">\n            <label class=\"form-label\">{{ prop.label }}</label>\n            <button (click)=\"addNewButton()\" class=\"add-button\">+ Add Button</button>\n          </div>\n        \n          <div *ngFor=\"let btn of book?.questionbook?.action || []; let i = index\" class=\"button-config-card\">\n            <div class=\"form-group\">\n              <label>Button Name</label>\n              <input type=\"text\" [(ngModel)]=\"btn.name\" (ngModelChange)=\"onButtonPropertyChange(i, 'name', $event)\" placeholder=\"Enter button name\">\n            </div>\n        \n            <div class=\"form-group\">\n              <label>Event Name</label>\n              <input type=\"text\" [(ngModel)]=\"btn.eventtoemit\" (ngModelChange)=\"onButtonPropertyChange(i, 'eventtoemit', $event)\" placeholder=\"Event to emit\">\n            </div>\n        \n            <div class=\"form-group\">\n              <label>Alt</label>\n              <input type=\"text\" [(ngModel)]=\"btn.alt\" (ngModelChange)=\"onButtonPropertyChange(i, 'alt', $event)\" placeholder=\"Button alt text\">\n            </div>\n        \n            <div class=\"form-group\">\n              <label>Endpoint</label>\n              <input type=\"text\" [(ngModel)]=\"btn.endpoint\" (ngModelChange)=\"onButtonPropertyChange(i, 'endpoint', $event)\" placeholder=\"API endpoint\">\n            </div>\n        \n            <div class=\"form-group\">\n              <label>Method</label>\n              <select [(ngModel)]=\"btn.method\" (ngModelChange)=\"onButtonPropertyChange(i, 'method', $event)\">\n                <option [value]=\"'GET'\">GET</option>\n                <option [value]=\"'POST'\">POST</option>\n                <option [value]=\"'PUT'\">PUT</option>\n                <option [value]=\"'DELETE'\">DELETE</option>\n              </select>\n            </div>\n        \n            <!-- AP 23MAY25 - Action Style Fields -->\n            <div class=\"form-group\">\n              <label>Background Color</label>\n              <input type=\"color\" [(ngModel)]=\"btn.bgColor\" (ngModelChange)=\"onButtonPropertyChange(i, 'bgColor', $event)\">\n            </div>\n        \n            <div class=\"form-group\">\n              <label>Border Radius</label>\n              <input type=\"range\" min=\"0\" max=\"50\" [(ngModel)]=\"btn.borderRadius\" \n                     (ngModelChange)=\"onButtonPropertyChange(i, 'borderRadius', $event)\">\n              <span>{{ btn.borderRadius }}px</span>\n            </div>            \n            \n            <div class=\"form-group\">\n              <label>Button Width (px)</label>\n              <input type=\"number\" [(ngModel)]=\"btn.width\"\n                     (ngModelChange)=\"onButtonPropertyChange(i, 'width', $event)\" min=\"50\" placeholder=\"Enter width in px\">\n            </div>\n            \n            <div class=\"form-group\">\n              <label>Text Color</label>\n              <input type=\"color\" [(ngModel)]=\"btn.textColor\"\n                     (ngModelChange)=\"onButtonPropertyChange(i, 'textColor', $event)\">\n            </div>\n            \n            <div class=\"form-group\">\n              <label>Position (%)</label>\n              <input type=\"range\" min=\"0\" max=\"100\" [(ngModel)]=\"btn.positionPercent\"\n                     (ngModelChange)=\"onButtonPropertyChange(i, 'positionPercent', $event)\">\n              <span>{{ btn.positionPercent }}%</span>\n            </div>\n            <button (click)=\"removeButton(btn)\" class=\"remove-button\">× Remove</button>\n          </div>\n        </div>\n        \n      </ng-container>\n    </div>\n  </div>\n  <div class=\"tab-content\" *ngIf=\"activeTab === 'property'\">\n    <!-- Field Elements Properties -->\n    <!-- AP-06MAR25 Show elements ID  -->\n    <div *ngIf=\"getProperties()\">\n      <label>Label Id</label>\n      <div\n        style=\"font-size: 13px; padding: 11px; border-radius: 5px; background-color: #f8f8f8; border: 1px solid #ddd;\">\n        {{ headerSelect ? bookId : selectedElement.id }}</div>\n      <ng-container *ngFor=\"let prop of getProperties().fieldProps\">\n        <div class=\"form-group\">\n          <label class=\"text-sm\">{{ prop.label }}</label>\n\n          <!-- SKS21MAR25 Toggle Group -->\n          <div *ngIf=\"prop.type === 'toggleGroup'\" class=\"toggle-group\">\n            <div class=\"toggle-item\">\n              <label class=\"toggle-label\">\n                <input type=\"checkbox\" [checked]=\"selectedElement?.isOptional\"\n                  (change)=\"onRequiredChange($event.target.checked)\" />\n                Required\n              </label>\n            </div>\n            <div class=\"toggle-item\">\n              <label class=\"toggle-label\">\n                <input type=\"checkbox\" [checked]=\"selectedElement?.isReadOnly\"\n                  (change)=\"onToggleChange('isReadOnly', $event)\" />\n                Read Only\n              </label>\n            </div>\n            <div class=\"toggle-item\">\n              <label class=\"toggle-label\">\n                <input type=\"checkbox\" />\n                Disabled\n              </label>\n            </div>\n            <div class=\"toggle-item\">\n              <label class=\"toggle-label\">\n                <input type=\"checkbox\" [checked]=\"selectedElement?.isHidden\"\n                  (change)=\"onToggleChange('isHidden', $event)\" />\n                Is Hide\n              </label>\n            </div>\n          </div>\n\n          <!-- AP 28FEB25 - handled options with UUID -->\n          <div\n            *ngIf=\"prop.type === 'dropdown' || prop.type === 'checkbox' || prop.type === 'radio' && prop.key === 'options'\"\n            class=\"options-container\">\n\n            <div class=\"option-list\" (dragover)=\"onDragOver($event)\" (drop)=\"onDrop($event, prop.key)\">\n              <div *ngFor=\"let option of selectedElement[prop.key]\" class=\"option-items\" [attr.data-id]=\"option.id\"\n                draggable=\"true\" (dragstart)=\"onDragStart($event, option.id)\">\n\n                <input type=\"text\" [(ngModel)]=\"option.value\" placeholder=\"Option\" class=\"options\" />\n                <img src=\"../assets/icons/Trash.svg\" (click)=\"removeOption(selectedElement[prop.key], option.id)\">\n                <span class=\"drag-handle\">☰</span>\n              </div>\n            </div>\n\n            <button (click)=\"addOption(selectedElement[prop.key])\">\n              <div class=\"add-varient\">\n                <span class=\"text-lg\">+</span>\n                <span>Add</span>\n              </div>\n            </button>\n          </div>\n\n          <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n            [value]=\"headerSelect ? book.records[0].title : (selectedElement?.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ) :  getValueByPath(prop.key))\"\n            (input)=\"headerSelect ? updateTitle($event.target.value) : (selectedElement?.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : setValueByPath(prop.key, $event.target.value) ) : setValueByPath(prop.key, $event.target.value))\"\n            [class.read-only]=\"selectedElement.readOnly\" [readonly]=\"selectedElement.readOnly\" />\n\n          <!-- Sub Questions Toggle -->\n          <div *ngIf=\"prop.type === 'subQuestion'\">\n            <div class=\"style-toggle-header\" (click)=\"toggleSubQuestion(prop)\">\n              <div class=\"head-elements\">Sub Text</div>\n              <img [src]=\"prop.isExpanded ? '../assets/icons/arrow-down.svg' : '../assets/icons/arrow-right.svg'\"\n                alt=\"Toggle Arrow\" class=\"arrow-icon\">\n            </div>\n\n            <!-- Render subquestions when arrow is down -->\n            <div *ngIf=\"prop.isExpanded\" style=\"border: 1px solid #ddd; padding: 8px; border-radius: 4px;\">\n              <div *ngFor=\"let subProp of prop.subQuestion\" class=\"sub-question-container\">\n                <div class=\"form-group\">\n                  <label>{{ subProp.label }}</label>\n\n                  <input *ngIf=\"subProp.type === 'text'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n                    [value]=\"getValueByPath(subProp.key)\" (input)=\"setValueByPath(subProp.key, $event.target.value)\" />\n                    \n                  <!-- AP-02APR25 Render input field only if subProp.type is 'array' -->\n                  <input *ngIf=\"subProp.type === 'array'\" type=\"text\" [placeholder]=\"subProp.placeholder\"\n                    [value]=\"getValueByPath(subProp.key)\" (input)=\"setValueByPath(subProp.key, $event.target.value)\" />\n                    <!-- <input type=\"text\" [ngModel]=\"fieldAsString\" (ngModelChange)=\"updateField($event)\" /> -->\n\n                  <input *ngIf=\"subProp.type === 'boolean'\" type=\"checkbox\" [checked]=\"getValueByPath(subProp.key)\"\n                    (change)=\"setValueByPath(subProp.key, $event.target.checked)\" />\n                </div>\n              </div>\n            </div>\n          </div>\n\n          <div *ngIf=\"prop.type === 'checkbox'\">\n            <span class=\"toggle-label\" style=\"padding-right: 10px;\">{{ prop.label }}</span>\n            <input type=\"checkbox\"\n              [checked]=\"selectedElement.type === 'Table' ? (prop.targetArray ? getValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key ) : getValueByPath(prop.key) ):  getValueByPath(prop.key) \"\n              (change)=\" selectedElement.type === 'Table' ? (prop.targetArray ? updateValueByArrayPath(prop.targetArray, prop.targetArrayKey, selectColumn, prop.key, $event) : onToggleChange(prop.key, $event) ) : onToggleChange(prop.key, $event)\" />\n\n          </div>\n\n        </div>\n      </ng-container>\n\n      <!-- SKS21MAR25 Move Style Outside the Loop -->\n      <!-- Style Toggle -->\n      <div class=\"style-toggle-header\" (click)=\"toggleStyleSection()\">\n        <div class=\"head-elements\">Style</div>\n        <img [src]=\"isStyleExpanded ? '../assets/icons/arrow-down.svg' : '../assets/icons/arrow-right.svg'\"\n          alt=\"Toggle Arrow\" class=\"arrow-icon\">\n      </div>\n\n      <div *ngIf=\"isStyleExpanded\" style=\"border: 1px solid #ddd; padding: 8px; border-radius: 4px;\">\n        <div class=\"form-group\">\n          <div *ngFor=\"let key of getStyleKeys()\">\n            <label>{{ key }}</label>\n            <input type=\"text\" [(ngModel)]=\"selectedElement.style[key]\"\n              (input)=\"setValueByPath('style.' + key, $event.target.value)\" placeholder=\"Enter {{ key }}\" />\n          </div>\n        </div>\n      </div>\n\n    </div>\n  </div>\n  <div class=\"tab-content\" *ngIf=\"activeTab === 'appearance'\">\n    <div *ngIf=\"getProperties()\">\n      <ng-container *ngFor=\"let prop of getProperties().appearance\">\n        <div class=\"form-group\">\n          <label class=\"text-sm\">{{ prop.label }}</label>\n          <!-- Font Selection -->\n          <div *ngIf=\"prop.key === 'font'\">\n            <select *ngIf=\"prop.type === 'select'\" class=\"select-container\" [(ngModel)]=\"selectedElement.font\"\n              (change)=\"setValueByPath('font', $event.target.value)\">\n              <option *ngFor=\"let option of prop.options\" [value]=\"option\">\n                {{ option }}\n              </option>\n            </select>\n          </div>\n          <!-- Input Box -->\n          <input *ngIf=\"prop.type === 'text'\" type=\"text\" [placeholder]=\"prop.placeholder\"\n            [value]=\"selectedElement[prop.key]\" (input)=\"setValueByPath(prop.key, $event.target.value)\" />\n\n          <!-- Font Size Selection -->\n          <div *ngIf=\"prop.key === 'fontSize'\">\n            <select *ngIf=\"prop.type === 'select'\" [(ngModel)]=\"selectedElement[prop.key]\"\n              (change)=\"setValueByPath(prop.key, $event.target.value)\">\n              <option *ngFor=\"let option of prop.options\" [value]=\"option.value\">\n                {{ option.label }}\n              </option>\n            </select>\n          </div>\n          <!-- Font Weight Selection -->\n          <div *ngIf=\"prop.key === 'fontWeight'\">\n            <select *ngIf=\"prop.type === 'select'\" [(ngModel)]=\"selectedElement[prop.key]\"\n              (change)=\"setValueByPath(prop.key, $event.target.value)\">\n              <option *ngFor=\"let option of prop.options\" [value]=\"option.value\">\n                {{ option.label }}\n              </option>\n            </select>\n          </div>\n\n          <div class=\"flex-container\">\n            <!-- Dropdown Box -->\n            <div *ngIf=\"prop.type === 'dropdown'\" class=\"input-box-field\">\n              <select [(ngModel)]=\"selectedElement[prop.key]\" (change)=\"setValueByPath(prop.key, $event.target.value)\">\n                <option *ngFor=\"let option of prop.options\" [value]=\"option\">\n                  {{ option }}\n                </option>\n              </select>\n            </div>\n\n            <!-- Color Picker -->\n            <div *ngIf=\"prop.type === 'color'\" class=\"color-selector\">\n              <input type=\"color\" [(ngModel)]=\"selectedElement.fontColor\"\n                (change)=\"setValueByPath('fontColor', $event.target.value)\">\n            </div>\n\n            <!-- HEX Input Box -->\n            <div *ngIf=\"prop.type === 'color'\" class=\"hex-input-container\">\n              <span>HEX Code</span>\n              <input type=\"text\" [(ngModel)]=\"selectedElement[prop.key]\"\n                (change)=\"setValueByPath(prop.key, $event.target.value)\" />\n            </div>\n          </div>\n          \n          <!-- AP-17APR25 Button for Duplicate Field  -->\n<div *ngIf=\"prop.type === 'button-toggle'\" class=\"button-toggle-wrapper\">\n  <button type=\"button\"\n    class=\"toggle-button\"\n    (click)=\"duplicateField(selectedElement)\">\n    Add Duplicate\n  </button>\n</div>\n\n\n        </div>\n      </ng-container>\n    </div>\n  </div>\n\n<!-- AP-08APR25 Default Save Button -->\n<div class=\"button-container\" *ngIf=\"!templateSelected\">\n  <button class=\"cancel-btn\" (click)=\"onCancel()\">Cancel</button>\n  <button class=\"save-btn\" (click)=\"handleButtonClick()\">Save</button>\n</div>\n\n<!-- AP-08APR25 Template Save Button -->\n<div class=\"button-container\" *ngIf=\"templateSelected\" style=\"margin-top: 20px;\">\n  <button class=\"save-btn\" (click)=\"handleTemplateSave()\">Save Template</button>\n</div>\n\n</div>"]}