@rangertechnologies/ngnxt 2.1.254 → 2.1.255
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/fesm2022/rangertechnologies-ngnxt.mjs +138 -169
- package/fesm2022/rangertechnologies-ngnxt.mjs.map +1 -1
- package/lib/components/pick-location/pick-location.component.d.ts +0 -1
- package/package.json +4 -6
- package/rangertechnologies-ngnxt-2.1.255.tgz +0 -0
- package/esm2022/environments/version.mjs +0 -15
- package/esm2022/lib/components/button/nxt-button.component.mjs +0 -154
- package/esm2022/lib/components/custom-calendar/custom-calendar.component.mjs +0 -360
- package/esm2022/lib/components/custom-dropdown/custom-dropdown.component.mjs +0 -263
- package/esm2022/lib/components/custom-model/custom-model.component.mjs +0 -53
- package/esm2022/lib/components/custom-radio/custom-radio.component.mjs +0 -158
- package/esm2022/lib/components/datatable/datatable.component.mjs +0 -1744
- package/esm2022/lib/components/file-upload/file-upload.component.mjs +0 -292
- package/esm2022/lib/components/icon-selector/icon-selector.component.mjs +0 -106
- package/esm2022/lib/components/image-cropper/component/cropper.state.mjs +0 -208
- package/esm2022/lib/components/image-cropper/component/image-cropper.component.mjs +0 -562
- package/esm2022/lib/components/image-cropper/interfaces/basic-event.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/cropper-options.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/cropper-position.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/dimensions.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/exif-transform.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/image-cropped-event.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/image-transform.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/index.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/loaded-image.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/move-start.interface.mjs +0 -8
- package/esm2022/lib/components/image-cropper/services/crop.service.mjs +0 -139
- package/esm2022/lib/components/image-cropper/services/load-image.service.mjs +0 -194
- package/esm2022/lib/components/image-cropper/utils/cropper-position.utils.mjs +0 -239
- package/esm2022/lib/components/image-cropper/utils/exif.utils.mjs +0 -79
- package/esm2022/lib/components/image-cropper/utils/keyboard.utils.mjs +0 -40
- package/esm2022/lib/components/image-cropper/utils/percentage.utils.mjs +0 -4
- package/esm2022/lib/components/image-cropper/utils/resize.utils.mjs +0 -75
- package/esm2022/lib/components/list-view-filter/list-view-filter.component.mjs +0 -392
- package/esm2022/lib/components/nxt-input/nxt-input.component.mjs +0 -2972
- package/esm2022/lib/components/pagination/pagination.component.mjs +0 -105
- package/esm2022/lib/components/pick-location/pick-location.component.mjs +0 -220
- package/esm2022/lib/components/search-box/search-box.component.mjs +0 -470
- package/esm2022/lib/country.json +0 -43237
- package/esm2022/lib/interfaces/actionMeta.mjs +0 -2
- package/esm2022/lib/interfaces/dependencyMeta.mjs +0 -2
- package/esm2022/lib/model/bookletWrapper.mjs +0 -7
- package/esm2022/lib/model/changeWrapper.mjs +0 -10
- package/esm2022/lib/model/errorWrapper.mjs +0 -6
- package/esm2022/lib/nxt-app.component.mjs +0 -22
- package/esm2022/lib/nxt-app.module.mjs +0 -115
- package/esm2022/lib/nxt-app.service.mjs +0 -14
- package/esm2022/lib/pages/booklet/booklet.component.mjs +0 -616
- package/esm2022/lib/pages/builder/element/element.component.mjs +0 -483
- package/esm2022/lib/pages/builder/form/form.component.mjs +0 -48
- package/esm2022/lib/pages/builder/properties/common-fields.constants.mjs +0 -97
- package/esm2022/lib/pages/builder/properties/properties.component.mjs +0 -1121
- package/esm2022/lib/pages/builder/templates/templates.component.mjs +0 -35
- package/esm2022/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.mjs +0 -639
- package/esm2022/lib/pages/pdfDesigner/pdf-properties/pdf-properties.component.mjs +0 -1114
- package/esm2022/lib/pages/questionbook/questionbook.component.mjs +0 -784
- package/esm2022/lib/pages/questionnaire/questionnaire.component.mjs +0 -2206
- package/esm2022/lib/pipe/button-styles.pipe.mjs +0 -26
- package/esm2022/lib/pipe/custom-translate.pipe.mjs +0 -37
- package/esm2022/lib/pipe/get-value.pipe.mjs +0 -54
- package/esm2022/lib/pipe/question-by-row.pipe.mjs +0 -51
- package/esm2022/lib/pipe/search-filter/search-filter.pipe.mjs +0 -40
- package/esm2022/lib/pipe/svg/svg.pipe.mjs +0 -26
- package/esm2022/lib/sample.mjs +0 -3715
- package/esm2022/lib/services/change/change.service.mjs +0 -46
- package/esm2022/lib/services/country/country.service.mjs +0 -135
- package/esm2022/lib/services/data/data.service.mjs +0 -100
- package/esm2022/lib/services/form-builder/form-builder.service.mjs +0 -474
- package/esm2022/lib/services/pdf-designer/pdf-designer.service.mjs +0 -395
- package/esm2022/lib/services/salesforce/salesforce.service.mjs +0 -41
- package/esm2022/lib/services/shared/shared.service.mjs +0 -100
- package/esm2022/lib/services/storage/storage.service.mjs +0 -59
- package/esm2022/lib/services/template/template.service.mjs +0 -335
- package/esm2022/lib/services/translation/translation.service.mjs +0 -121
- package/esm2022/lib/wrapper.mjs +0 -175
- package/esm2022/public-api.mjs +0 -22
- package/esm2022/rangertechnologies-ngnxt.mjs +0 -5
- package/rangertechnologies-ngnxt-2.1.254.tgz +0 -0
|
@@ -1,639 +0,0 @@
|
|
|
1
|
-
import { Component, EventEmitter, Input, Output, ViewChildren } from '@angular/core';
|
|
2
|
-
import { VERSION } from '../../../../environments/version';
|
|
3
|
-
import { PdfPropertiesComponent } from '../pdf-properties/pdf-properties.component';
|
|
4
|
-
import { CommonModule } from '@angular/common';
|
|
5
|
-
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
6
|
-
import { FormsModule } from '@angular/forms';
|
|
7
|
-
import { NxtDatatable } from '../../../components/datatable/datatable.component';
|
|
8
|
-
import { BookletComponent } from '../../booklet/booklet.component';
|
|
9
|
-
import { AppTemplatesComponent } from '../../builder/templates/templates.component';
|
|
10
|
-
import { NxtInput } from '../../../components/nxt-input/nxt-input.component';
|
|
11
|
-
import { ImageCropperComponent } from '../../../components/image-cropper/component/image-cropper.component';
|
|
12
|
-
import _ from 'lodash'; // Ensure lodash is installed
|
|
13
|
-
import { NxtCustomTranslatePipe } from '../../../pipe/custom-translate.pipe';
|
|
14
|
-
import * as i0 from "@angular/core";
|
|
15
|
-
import * as i1 from "../../../services/pdf-designer/pdf-designer.service";
|
|
16
|
-
import * as i2 from "@angular/common";
|
|
17
|
-
import * as i3 from "@angular/material/tooltip";
|
|
18
|
-
import * as i4 from "@angular/forms";
|
|
19
|
-
export class PdfDesignerComponent {
|
|
20
|
-
pdfDesignerService;
|
|
21
|
-
//@Output() elementButtonClicked = new EventEmitter<string>();
|
|
22
|
-
textareas;
|
|
23
|
-
pdfJSON;
|
|
24
|
-
bookletId;
|
|
25
|
-
isPreview = false;
|
|
26
|
-
pdfSaveHandlerEmit = new EventEmitter();
|
|
27
|
-
templateMode = new EventEmitter();
|
|
28
|
-
pdfPreviewEmit = new EventEmitter();
|
|
29
|
-
field;
|
|
30
|
-
pdfElements = [];
|
|
31
|
-
elements = [];
|
|
32
|
-
bookId;
|
|
33
|
-
sections = {
|
|
34
|
-
basic: true,
|
|
35
|
-
advanced: true
|
|
36
|
-
};
|
|
37
|
-
elementsList = [];
|
|
38
|
-
elementDisabledArray;
|
|
39
|
-
version = VERSION.version;
|
|
40
|
-
pdf;
|
|
41
|
-
selectedElement = null;
|
|
42
|
-
isSelectTablePopup = false;
|
|
43
|
-
currentType;
|
|
44
|
-
addTable = true;
|
|
45
|
-
dots = Array(6);
|
|
46
|
-
draggedIndex = null;
|
|
47
|
-
templateSelected = false;
|
|
48
|
-
isImageEdit = false;
|
|
49
|
-
selectedImageElement;
|
|
50
|
-
transform = {
|
|
51
|
-
translateUnit: 'px',
|
|
52
|
-
scale: 1,
|
|
53
|
-
rotate: 0,
|
|
54
|
-
flipH: false,
|
|
55
|
-
flipV: false,
|
|
56
|
-
translateH: 0,
|
|
57
|
-
translateV: 0
|
|
58
|
-
};
|
|
59
|
-
loading = false;
|
|
60
|
-
canvasRotation = 0;
|
|
61
|
-
cropper;
|
|
62
|
-
cropperMaxHeight = 0;
|
|
63
|
-
cropperMaxWidth = 0;
|
|
64
|
-
cropperMinHeight = 0;
|
|
65
|
-
cropperMinWidth = 0;
|
|
66
|
-
cropperStaticWidth = 0;
|
|
67
|
-
cropperStaticHeight = 0;
|
|
68
|
-
aspectRatio = 4 / 3;
|
|
69
|
-
roundCropper = false;
|
|
70
|
-
isImageHover = false;
|
|
71
|
-
alignImage = 'center';
|
|
72
|
-
showTextSettings = false;
|
|
73
|
-
selectedColumn = null;
|
|
74
|
-
//subscription: any;
|
|
75
|
-
constructor(pdfDesignerService) {
|
|
76
|
-
this.pdfDesignerService = pdfDesignerService;
|
|
77
|
-
}
|
|
78
|
-
//AP-14JUN25 - Called after view initialization to auto-resize all textareas
|
|
79
|
-
ngAfterViewInit() {
|
|
80
|
-
this.resizeAllTextareas();
|
|
81
|
-
}
|
|
82
|
-
//AP-14JUN25 - Calculates contrast text color (black or white) based on background color
|
|
83
|
-
getContrastColor(bgColor = '#000000') {
|
|
84
|
-
// Convert hex to RGB
|
|
85
|
-
const color = bgColor.replace('#', '');
|
|
86
|
-
const r = parseInt(color.substring(0, 2), 16);
|
|
87
|
-
const g = parseInt(color.substring(2, 4), 16);
|
|
88
|
-
const b = parseInt(color.substring(4, 6), 16);
|
|
89
|
-
const brightness = (r * 299 + g * 587 + b * 114) / 1000;
|
|
90
|
-
return brightness > 128 ? '#000000' : '#ffffff';
|
|
91
|
-
}
|
|
92
|
-
//AP-14JUN25 - Automatically resizes all textareas based on their content
|
|
93
|
-
resizeAllTextareas() {
|
|
94
|
-
this.textareas.forEach((textareaRef) => {
|
|
95
|
-
const textarea = textareaRef.nativeElement;
|
|
96
|
-
textarea.style.height = 'auto';
|
|
97
|
-
textarea.style.height = textarea.scrollHeight + 'px';
|
|
98
|
-
});
|
|
99
|
-
}
|
|
100
|
-
ngOnInit() {
|
|
101
|
-
this.elementsList = [
|
|
102
|
-
{ "type": "text", "img": "TextArea", "label": "Text" },
|
|
103
|
-
{ "type": "input", "img": "Text", "label": "Input" },
|
|
104
|
-
{ "type": "date", "img": "Date", "label": "Date" },
|
|
105
|
-
{ "type": "image", "img": "Image", "label": "Image" },
|
|
106
|
-
{ "type": "Line", "img": "line", "label": "Line" },
|
|
107
|
-
{ "type": "Space", "img": "space", "label": "Space" },
|
|
108
|
-
{ "type": "Table", "img": "Table", "label": "Table" },
|
|
109
|
-
{ "type": "Pdf", "img": "Search", "label": "Book" },
|
|
110
|
-
];
|
|
111
|
-
this.elementDisabledArray = {
|
|
112
|
-
Table: [
|
|
113
|
-
'Pdf', 'Calendar', "Boolean", 'List', 'Table', 'Checkbox', 'Radio',
|
|
114
|
-
'Dropdown', 'RichTextArea', 'Number', 'Label', 'image',
|
|
115
|
-
'Email', 'Date', 'Time', 'DateTime', 'Line', 'Space'
|
|
116
|
-
]
|
|
117
|
-
};
|
|
118
|
-
if (!this.isPreview) {
|
|
119
|
-
this.pdfDesignerService.pdfElements$.subscribe(elements => {
|
|
120
|
-
setTimeout(() => {
|
|
121
|
-
this.pdfElements = elements.map(field => ({ ...field }));
|
|
122
|
-
}, 0);
|
|
123
|
-
});
|
|
124
|
-
}
|
|
125
|
-
if (this.pdfJSON && (this.pdfJSON !== '')) {
|
|
126
|
-
this.initializeForm();
|
|
127
|
-
}
|
|
128
|
-
else {
|
|
129
|
-
this.pdfDesignerService.newBook();
|
|
130
|
-
}
|
|
131
|
-
// AP-12MAR25 - Added to handle bookletId
|
|
132
|
-
if (this.bookletId) {
|
|
133
|
-
localStorage.setItem('unique_id', this.bookletId);
|
|
134
|
-
}
|
|
135
|
-
this.pdf = this.pdfDesignerService.getBook();
|
|
136
|
-
if (!this.isPreview) {
|
|
137
|
-
this.pdfDesignerService.selectedElement$.subscribe(index => {
|
|
138
|
-
const elements = this.pdfDesignerService.getElements();
|
|
139
|
-
if (index >= 0) {
|
|
140
|
-
this.selectedElement = elements[index];
|
|
141
|
-
}
|
|
142
|
-
// this.pdfElements = [...elements]
|
|
143
|
-
});
|
|
144
|
-
}
|
|
145
|
-
this.pdfDesignerService.addElementWithId();
|
|
146
|
-
}
|
|
147
|
-
isSectionOpen(section) {
|
|
148
|
-
return this.sections[section];
|
|
149
|
-
}
|
|
150
|
-
ngOnChanges(changes) {
|
|
151
|
-
if (changes['pdfJSON'] && changes['pdfJSON'].currentValue) {
|
|
152
|
-
if (this.pdfJSON && (this.pdfJSON !== '')) {
|
|
153
|
-
this.initializeForm();
|
|
154
|
-
}
|
|
155
|
-
else {
|
|
156
|
-
this.pdfDesignerService.newBook();
|
|
157
|
-
}
|
|
158
|
-
}
|
|
159
|
-
}
|
|
160
|
-
onTemplateSelected(event) {
|
|
161
|
-
this.pdfDesignerService.clearElements();
|
|
162
|
-
event.elements.forEach(el => this.pdfDesignerService.addElement(el));
|
|
163
|
-
// Emit template mode to parent
|
|
164
|
-
this.templateMode.emit(true);
|
|
165
|
-
}
|
|
166
|
-
initializeForm() {
|
|
167
|
-
if (!this.pdfJSON || !this.pdfJSON) {
|
|
168
|
-
console.warn("Invalid pdfJSON structure");
|
|
169
|
-
return;
|
|
170
|
-
}
|
|
171
|
-
this.pdfElements = [];
|
|
172
|
-
this.pdfDesignerService.clearElements();
|
|
173
|
-
this.pdfDesignerService.intializeBook(this.pdfJSON);
|
|
174
|
-
this.pdfElements = this.pdfDesignerService.getElements();
|
|
175
|
-
}
|
|
176
|
-
selectElement(index) {
|
|
177
|
-
this.selectedFieldIndex = index;
|
|
178
|
-
this.pdfDesignerService.setSelectedElement(index);
|
|
179
|
-
this.pdf = this.pdfDesignerService.getBook();
|
|
180
|
-
}
|
|
181
|
-
selectedFieldIndex = null;
|
|
182
|
-
selectHeading(event) {
|
|
183
|
-
this.pdfDesignerService.selectHeading(event);
|
|
184
|
-
this.pdf = this.pdfDesignerService.getBook();
|
|
185
|
-
}
|
|
186
|
-
// AP-17APR25 generateUiId
|
|
187
|
-
generateUiId() {
|
|
188
|
-
return this.pdfDesignerService.addElementWithId();
|
|
189
|
-
}
|
|
190
|
-
addElement(type) {
|
|
191
|
-
const unique_id = this.generateUiId();
|
|
192
|
-
this.currentType = type;
|
|
193
|
-
if (this.selectedElement?.type === 'Table' && this.addTable) {
|
|
194
|
-
this.isSelectTablePopup = true;
|
|
195
|
-
}
|
|
196
|
-
else {
|
|
197
|
-
const newElement = {
|
|
198
|
-
id: unique_id,
|
|
199
|
-
type,
|
|
200
|
-
margin: [0, 0, 0, 0],
|
|
201
|
-
fontSize: 14,
|
|
202
|
-
fontWeight: '400',
|
|
203
|
-
fontStyle: [],
|
|
204
|
-
alignItems: '',
|
|
205
|
-
nxtType: type,
|
|
206
|
-
referenceField: null,
|
|
207
|
-
question: null,
|
|
208
|
-
questionNumber: this.pdfDesignerService.getElements().length + 1,
|
|
209
|
-
fieldsMeta: [],
|
|
210
|
-
pdfReference: null,
|
|
211
|
-
pdfReferenceQuestions: null,
|
|
212
|
-
style: {
|
|
213
|
-
bold: false,
|
|
214
|
-
italic: false,
|
|
215
|
-
alignment: 'left',
|
|
216
|
-
fontSize: 14,
|
|
217
|
-
margin: [0, 0, 0, 0],
|
|
218
|
-
color: '#000000',
|
|
219
|
-
},
|
|
220
|
-
imageData: null,
|
|
221
|
-
imageSize: type === 'image' ? { width: 100, height: 100 } : null,
|
|
222
|
-
tableConfig: type === 'Table' ? {
|
|
223
|
-
isNosIndicator: false,
|
|
224
|
-
addInlineRecord: true,
|
|
225
|
-
isPagination: false,
|
|
226
|
-
actionButton: false,
|
|
227
|
-
isDeleteRow: true,
|
|
228
|
-
isEditRow: false,
|
|
229
|
-
searchBar: false,
|
|
230
|
-
isButtons: false,
|
|
231
|
-
} : null,
|
|
232
|
-
width: 100,
|
|
233
|
-
styleClass: unique_id,
|
|
234
|
-
};
|
|
235
|
-
this.pdfDesignerService.addElement(newElement);
|
|
236
|
-
this.pdfElements = this.pdfDesignerService.getElements();
|
|
237
|
-
this.addTable = true;
|
|
238
|
-
}
|
|
239
|
-
}
|
|
240
|
-
//AP-14JUN25 - Handles textarea input: auto-resizes and updates the field content
|
|
241
|
-
onTextAreaInput(event, field) {
|
|
242
|
-
const textarea = event.target;
|
|
243
|
-
// 1. Auto-resize logic
|
|
244
|
-
textarea.style.height = 'auto';
|
|
245
|
-
textarea.style.height = textarea.scrollHeight + 'px';
|
|
246
|
-
// 2. Trigger value update and PDF sync
|
|
247
|
-
this.onQuestionChange(textarea.value, field);
|
|
248
|
-
}
|
|
249
|
-
onQuestionChange(value, field) {
|
|
250
|
-
let elementId;
|
|
251
|
-
// SKS19JUN25 Clone to avoid modifying the original
|
|
252
|
-
const pdfElementsClone = _.cloneDeep(this.pdfElements);
|
|
253
|
-
const updatePdfQuestion = (element, value) => {
|
|
254
|
-
if (element.type === 'Pdf') {
|
|
255
|
-
const subElements = element.pdfReferenceQuestions?.[element.pdfReference] || [];
|
|
256
|
-
for (const subElement of subElements) {
|
|
257
|
-
const found = updatePdfQuestion(subElement, value); // recursive call
|
|
258
|
-
if (found)
|
|
259
|
-
return true; // stop once match is found
|
|
260
|
-
}
|
|
261
|
-
}
|
|
262
|
-
else {
|
|
263
|
-
if (element.id === field.id) {
|
|
264
|
-
element.value = value;
|
|
265
|
-
return true;
|
|
266
|
-
}
|
|
267
|
-
}
|
|
268
|
-
return false;
|
|
269
|
-
};
|
|
270
|
-
pdfElementsClone.forEach((element, index) => {
|
|
271
|
-
const found = updatePdfQuestion(element, value);
|
|
272
|
-
if (found && elementId === undefined) {
|
|
273
|
-
elementId = index;
|
|
274
|
-
}
|
|
275
|
-
});
|
|
276
|
-
if (elementId !== undefined) {
|
|
277
|
-
const updatedElement = pdfElementsClone[elementId];
|
|
278
|
-
this.pdfDesignerService.elementUpdate(updatedElement, 'valueChange');
|
|
279
|
-
// SKS19JUN25 Optionally preview or test using updatedElement (without updating original)
|
|
280
|
-
if (this.isPreview) {
|
|
281
|
-
this.pdfPreview(this.pdfDesignerService.downloadElement());
|
|
282
|
-
}
|
|
283
|
-
}
|
|
284
|
-
}
|
|
285
|
-
pdfPreview(pdf) {
|
|
286
|
-
this.pdfPreviewEmit.emit(pdf);
|
|
287
|
-
}
|
|
288
|
-
onFieldDateChange(event) {
|
|
289
|
-
}
|
|
290
|
-
removeElement(field, index) {
|
|
291
|
-
this.pdfDesignerService.removeElementComponent(field.id);
|
|
292
|
-
this.pdfElements = this.pdfDesignerService.getElements();
|
|
293
|
-
}
|
|
294
|
-
onDragStart(event, index) {
|
|
295
|
-
this.draggedIndex = index;
|
|
296
|
-
event.dataTransfer?.setData('text/plain', index.toString());
|
|
297
|
-
}
|
|
298
|
-
onDragOver(event, index) {
|
|
299
|
-
event.preventDefault();
|
|
300
|
-
}
|
|
301
|
-
onDrop(event, dropIndex) {
|
|
302
|
-
event.preventDefault();
|
|
303
|
-
if (this.draggedIndex === null || this.draggedIndex === dropIndex)
|
|
304
|
-
return;
|
|
305
|
-
const draggedItem = this.pdfElements[this.draggedIndex];
|
|
306
|
-
// Remove dragged item from old position and insert it in new position
|
|
307
|
-
this.pdfElements.splice(this.draggedIndex, 1);
|
|
308
|
-
this.pdfElements.splice(dropIndex, 0, draggedItem);
|
|
309
|
-
//AP-28MAR25 Update questionNumber dynamically based on new order
|
|
310
|
-
this.pdfElements.forEach((element, index) => {
|
|
311
|
-
element.questionNumber = index + 1;
|
|
312
|
-
});
|
|
313
|
-
// Reset dragged index
|
|
314
|
-
this.draggedIndex = null;
|
|
315
|
-
// Notify service about the update
|
|
316
|
-
this.pdfDesignerService.updateElementsOrder(this.pdfElements);
|
|
317
|
-
}
|
|
318
|
-
toggleBoolean(field) {
|
|
319
|
-
field.boolean = !field.boolean;
|
|
320
|
-
}
|
|
321
|
-
// SKS13MAR25 table popup conformation based element add
|
|
322
|
-
onClose() {
|
|
323
|
-
this.isSelectTablePopup = false;
|
|
324
|
-
this.addTable = false;
|
|
325
|
-
this.addElement(this.currentType);
|
|
326
|
-
}
|
|
327
|
-
// SKS13MAR25 column element add inside a table
|
|
328
|
-
addOnTable() {
|
|
329
|
-
this.addTable = true;
|
|
330
|
-
this.isSelectTablePopup = false;
|
|
331
|
-
const elements = [...this.pdfDesignerService.getElements()];
|
|
332
|
-
// Find the element and update its columns
|
|
333
|
-
const index = elements.findIndex(el => el.id === this.selectedElement.id);
|
|
334
|
-
const rowNum = this.pdfElements[index].columns ? this.pdfElements[index].columns?.length : 0;
|
|
335
|
-
const unique_id = this.generateUiId();
|
|
336
|
-
const tableElement = {
|
|
337
|
-
label: `HEADER LABEL ${rowNum}`,
|
|
338
|
-
fieldName: `FIELD NAME${rowNum}`,
|
|
339
|
-
type: this.currentType,
|
|
340
|
-
id: unique_id
|
|
341
|
-
};
|
|
342
|
-
this.pdfDesignerService.addTableElement(tableElement, this.selectedFieldIndex);
|
|
343
|
-
this.pdfElements = this.pdfDesignerService.getElements();
|
|
344
|
-
// this.addTableData(unique_id, fieldName)
|
|
345
|
-
}
|
|
346
|
-
addTableData(fieldId, fieldName) {
|
|
347
|
-
// this.pdfDesignerService.addTablefieldData(tableElement);
|
|
348
|
-
}
|
|
349
|
-
columnSelected(event) {
|
|
350
|
-
// SKS19MAR25 table column update
|
|
351
|
-
if (!this.isPreview) {
|
|
352
|
-
this.pdfDesignerService.setSelectedTableElement(this.selectedFieldIndex, event);
|
|
353
|
-
}
|
|
354
|
-
}
|
|
355
|
-
removeColumn(event) {
|
|
356
|
-
this.pdfDesignerService.removeSelectedTableElement(this.selectedFieldIndex, event);
|
|
357
|
-
}
|
|
358
|
-
// SKS25MAR25 image add
|
|
359
|
-
async fileChangeEvent(field, event) {
|
|
360
|
-
const file = event.target.files[0];
|
|
361
|
-
if (!file)
|
|
362
|
-
return;
|
|
363
|
-
try {
|
|
364
|
-
const imageData = await this.readFileAsDataURL(file);
|
|
365
|
-
let elementId;
|
|
366
|
-
const updatePdfImage = (element) => {
|
|
367
|
-
if (element.type === 'Pdf') {
|
|
368
|
-
const subElements = element.pdfReferenceQuestions?.[element.pdfReference] || [];
|
|
369
|
-
for (const subElement of subElements) {
|
|
370
|
-
const found = updatePdfImage(subElement); // recursive call
|
|
371
|
-
if (found)
|
|
372
|
-
return true;
|
|
373
|
-
}
|
|
374
|
-
}
|
|
375
|
-
else {
|
|
376
|
-
if (element.id === field.id) {
|
|
377
|
-
element.imageData = imageData;
|
|
378
|
-
element.orgImageData = imageData;
|
|
379
|
-
if (!element.imageSize) {
|
|
380
|
-
element.imageSize = { width: 100, height: 100 };
|
|
381
|
-
}
|
|
382
|
-
return true;
|
|
383
|
-
}
|
|
384
|
-
}
|
|
385
|
-
return false;
|
|
386
|
-
};
|
|
387
|
-
this.pdfElements.forEach((element, index) => {
|
|
388
|
-
const found = updatePdfImage(element);
|
|
389
|
-
if (found && elementId === undefined) {
|
|
390
|
-
elementId = index; // remember top-level match
|
|
391
|
-
}
|
|
392
|
-
});
|
|
393
|
-
if (elementId !== undefined) {
|
|
394
|
-
this.pdfDesignerService.elementUpdate(this.pdfElements[elementId], 'valueChange');
|
|
395
|
-
if (this.isPreview) {
|
|
396
|
-
this.pdfPreview(this.pdfDesignerService.downloadElement());
|
|
397
|
-
}
|
|
398
|
-
}
|
|
399
|
-
}
|
|
400
|
-
catch (error) {
|
|
401
|
-
console.error("Error reading file:", error);
|
|
402
|
-
}
|
|
403
|
-
}
|
|
404
|
-
readFileAsDataURL(file) {
|
|
405
|
-
return new Promise((resolve, reject) => {
|
|
406
|
-
const reader = new FileReader();
|
|
407
|
-
reader.onload = () => resolve(reader.result);
|
|
408
|
-
reader.onerror = (error) => reject(error);
|
|
409
|
-
reader.readAsDataURL(file);
|
|
410
|
-
});
|
|
411
|
-
}
|
|
412
|
-
// SKS2APR25 disabled element
|
|
413
|
-
isElementDisabled(elementType) {
|
|
414
|
-
if (this.selectedElement) {
|
|
415
|
-
const elements = this.elementDisabledArray[this.selectedElement.type];
|
|
416
|
-
if (elements) {
|
|
417
|
-
return elements.includes(elementType);
|
|
418
|
-
}
|
|
419
|
-
}
|
|
420
|
-
return false;
|
|
421
|
-
}
|
|
422
|
-
async onImageEdit(ques) {
|
|
423
|
-
this.isImageEdit = true;
|
|
424
|
-
this.selectedImageElement = ques;
|
|
425
|
-
//Assign imageSize default value
|
|
426
|
-
if (!this.selectedImageElement.imageSize) {
|
|
427
|
-
this.selectedImageElement.imageSize = { width: 100, height: 100 };
|
|
428
|
-
}
|
|
429
|
-
}
|
|
430
|
-
async onImageDelete(ques) {
|
|
431
|
-
try {
|
|
432
|
-
ques.imageData = '';
|
|
433
|
-
ques.orgImageData = '';
|
|
434
|
-
// Initialize logo size if not already set
|
|
435
|
-
if (!ques.imageSize) {
|
|
436
|
-
ques.imageSize = { width: 100, height: 100 };
|
|
437
|
-
}
|
|
438
|
-
// await this.childEventCapture(ques.imageData, ques);
|
|
439
|
-
}
|
|
440
|
-
catch (error) {
|
|
441
|
-
console.error("Error reading file:", error);
|
|
442
|
-
}
|
|
443
|
-
}
|
|
444
|
-
// SKS25MAR25 image edit functions
|
|
445
|
-
flipHorizontal() {
|
|
446
|
-
this.transform = {
|
|
447
|
-
...this.transform,
|
|
448
|
-
flipH: !this.transform.flipH
|
|
449
|
-
};
|
|
450
|
-
}
|
|
451
|
-
flipVertical() {
|
|
452
|
-
this.transform = {
|
|
453
|
-
...this.transform,
|
|
454
|
-
flipV: !this.transform.flipV
|
|
455
|
-
};
|
|
456
|
-
}
|
|
457
|
-
resetImage() {
|
|
458
|
-
this.canvasRotation = 0;
|
|
459
|
-
this.cropper = undefined;
|
|
460
|
-
this.transform = {
|
|
461
|
-
translateUnit: 'px',
|
|
462
|
-
scale: 1,
|
|
463
|
-
rotate: 0,
|
|
464
|
-
flipH: false,
|
|
465
|
-
flipV: false,
|
|
466
|
-
translateH: 0,
|
|
467
|
-
translateV: 0
|
|
468
|
-
};
|
|
469
|
-
}
|
|
470
|
-
zoomOut() {
|
|
471
|
-
this.transform = {
|
|
472
|
-
...this.transform,
|
|
473
|
-
scale: this.transform.scale - .1
|
|
474
|
-
};
|
|
475
|
-
}
|
|
476
|
-
zoomIn() {
|
|
477
|
-
this.transform = {
|
|
478
|
-
...this.transform,
|
|
479
|
-
scale: this.transform.scale + .1
|
|
480
|
-
};
|
|
481
|
-
}
|
|
482
|
-
rotateLeft() {
|
|
483
|
-
this.loading = true;
|
|
484
|
-
setTimeout(() => {
|
|
485
|
-
this.canvasRotation--;
|
|
486
|
-
this.flipAfterRotate();
|
|
487
|
-
});
|
|
488
|
-
}
|
|
489
|
-
rotateRight() {
|
|
490
|
-
this.loading = true;
|
|
491
|
-
setTimeout(() => {
|
|
492
|
-
this.canvasRotation++;
|
|
493
|
-
this.flipAfterRotate();
|
|
494
|
-
});
|
|
495
|
-
}
|
|
496
|
-
moveLeft() {
|
|
497
|
-
this.transform = {
|
|
498
|
-
...this.transform,
|
|
499
|
-
translateH: this.transform.translateH - 1
|
|
500
|
-
};
|
|
501
|
-
}
|
|
502
|
-
moveRight() {
|
|
503
|
-
this.transform = {
|
|
504
|
-
...this.transform,
|
|
505
|
-
translateH: this.transform.translateH + 1
|
|
506
|
-
};
|
|
507
|
-
}
|
|
508
|
-
moveDown() {
|
|
509
|
-
this.transform = {
|
|
510
|
-
...this.transform,
|
|
511
|
-
translateV: this.transform.translateV + 1
|
|
512
|
-
};
|
|
513
|
-
}
|
|
514
|
-
moveUp() {
|
|
515
|
-
this.transform = {
|
|
516
|
-
...this.transform,
|
|
517
|
-
translateV: this.transform.translateV - 1
|
|
518
|
-
};
|
|
519
|
-
}
|
|
520
|
-
flipAfterRotate() {
|
|
521
|
-
const flippedH = this.transform.flipH;
|
|
522
|
-
const flippedV = this.transform.flipV;
|
|
523
|
-
this.transform = {
|
|
524
|
-
...this.transform,
|
|
525
|
-
flipH: flippedV,
|
|
526
|
-
flipV: flippedH,
|
|
527
|
-
translateH: 0,
|
|
528
|
-
translateV: 0
|
|
529
|
-
};
|
|
530
|
-
}
|
|
531
|
-
//AP-20JUN25 - Updates the image size (width and height) of the selected image element and reflects the change via the service
|
|
532
|
-
onImageSizeChange() {
|
|
533
|
-
const index = this.pdfElements.findIndex((el) => el.id === this.selectedImageElement.id);
|
|
534
|
-
if (index !== -1) {
|
|
535
|
-
this.pdfElements[index].imageSize = {
|
|
536
|
-
width: this.selectedImageElement.imageSize.width,
|
|
537
|
-
height: this.selectedImageElement.imageSize.height
|
|
538
|
-
};
|
|
539
|
-
// Update element in service
|
|
540
|
-
this.pdfDesignerService.elementUpdate(this.pdfElements[index], 'valueChange');
|
|
541
|
-
}
|
|
542
|
-
}
|
|
543
|
-
async imageCropped(event) {
|
|
544
|
-
try {
|
|
545
|
-
const base64 = await this.convertBlobToBase64(event.objectUrl);
|
|
546
|
-
this.selectedImageElement.imageData = base64;
|
|
547
|
-
// await this.childEventCapture(base64, this.selectedImageElement);
|
|
548
|
-
// set default if not present
|
|
549
|
-
if (!this.selectedImageElement.imageSize) {
|
|
550
|
-
this.selectedImageElement.imageSize = { width: 100, height: 100 };
|
|
551
|
-
}
|
|
552
|
-
}
|
|
553
|
-
catch (error) {
|
|
554
|
-
console.error("Error in imageCropped:", error);
|
|
555
|
-
}
|
|
556
|
-
}
|
|
557
|
-
cropperReady(sourceImageDimensions) {
|
|
558
|
-
this.loading = false;
|
|
559
|
-
}
|
|
560
|
-
// SKS25MAR25 blob to base 64 converter
|
|
561
|
-
convertBlobToBase64(objectUrl) {
|
|
562
|
-
return fetch(objectUrl)
|
|
563
|
-
.then(response => response.blob())
|
|
564
|
-
.then(blob => {
|
|
565
|
-
return new Promise((resolve, reject) => {
|
|
566
|
-
const reader = new FileReader();
|
|
567
|
-
reader.readAsDataURL(blob);
|
|
568
|
-
reader.onloadend = () => resolve(reader.result);
|
|
569
|
-
reader.onerror = error => reject(error);
|
|
570
|
-
});
|
|
571
|
-
});
|
|
572
|
-
}
|
|
573
|
-
closeModal() {
|
|
574
|
-
this.isImageEdit = false;
|
|
575
|
-
}
|
|
576
|
-
openTextSettings(field, event) {
|
|
577
|
-
this.selectedColumn = field;
|
|
578
|
-
this.showTextSettings = true;
|
|
579
|
-
}
|
|
580
|
-
closeTextSettings() {
|
|
581
|
-
this.showTextSettings = false;
|
|
582
|
-
}
|
|
583
|
-
//AP-14JUN25 - Toggles bold or italic style for the selected text block
|
|
584
|
-
toggleStyle(style) {
|
|
585
|
-
if (!this.selectedColumn)
|
|
586
|
-
return;
|
|
587
|
-
if (style === 'bold') {
|
|
588
|
-
this.selectedColumn.style.bold = !this.selectedColumn.style.bold;
|
|
589
|
-
}
|
|
590
|
-
else if (style === 'italic') {
|
|
591
|
-
this.selectedColumn.style.italics = !this.selectedColumn.style.italics;
|
|
592
|
-
}
|
|
593
|
-
this.updateContent();
|
|
594
|
-
}
|
|
595
|
-
//AP-14JUN25 - Sets text alignment (left, center, right) for the selected text block
|
|
596
|
-
setAlignment(alignment) {
|
|
597
|
-
if (!this.selectedColumn)
|
|
598
|
-
return;
|
|
599
|
-
this.selectedColumn.style.alignment = alignment;
|
|
600
|
-
this.updateContent();
|
|
601
|
-
}
|
|
602
|
-
//AP-14JUN25 - Updates the content in the PDF designer with the selected style changes
|
|
603
|
-
updateContent() {
|
|
604
|
-
this.pdfDesignerService.elementUpdate(this.selectedColumn, 'valueChange');
|
|
605
|
-
}
|
|
606
|
-
pdfSaveHandler(event) {
|
|
607
|
-
if (event === 'preview') {
|
|
608
|
-
this.pdfSaveHandlerEmit.emit({ "action": 'preview', "pdf": this.pdfDesignerService.downloadElement() });
|
|
609
|
-
}
|
|
610
|
-
else if (event === 'download') {
|
|
611
|
-
this.pdfSaveHandlerEmit.emit({ "action": 'download', "pdf": this.pdfDesignerService.downloadElement() });
|
|
612
|
-
}
|
|
613
|
-
else {
|
|
614
|
-
this.pdfSaveHandlerEmit.emit(event);
|
|
615
|
-
}
|
|
616
|
-
}
|
|
617
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PdfDesignerComponent, deps: [{ token: i1.PdfDesignerService }], target: i0.ɵɵFactoryTarget.Component });
|
|
618
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: PdfDesignerComponent, isStandalone: true, selector: "app-pdf-designer", inputs: { pdfJSON: "pdfJSON", bookletId: "bookletId", isPreview: "isPreview" }, outputs: { pdfSaveHandlerEmit: "pdfSaveHandlerEmit", templateMode: "templateMode", pdfPreviewEmit: "pdfPreviewEmit" }, viewQueries: [{ propertyName: "textareas", predicate: ["autoTextarea"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<!-- AP 22JAN25 - form preview and All form elements -->\n<!-- AP 25FEB25 - All elements update -->\n<div class=\"form-container\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"center-frame\" [ngClass]=\"{'isPreview': isPreview}\">\n <!-- Form Builder Section All Elements -->\n <div *ngIf=\"!isPreview\" class=\"form-builder\">\n\n <ng-container *ngFor=\"let element of elementsList\">\n <div\n class=\"element\"\n [class.disabled]=\"selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)\"\n (click)=\"!(selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)) && addElement(element.type)\"\n [draggable]=\"!(selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type))\"\n [matTooltip]=\"\n selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)\n ? ('THIS_ELEMENT_NOT_SUPPORTED' | nxtCustomTranslate : 'This element is not supported when a ') +\n (selectedElement?.type || ('CERTAIN_ELEMENT' | nxtCustomTranslate : 'certain element')) +\n (' IS_PRESENT' | nxtCustomTranslate : ' is present')\n : null\n \"\n >\n <img src=\"../assets/icons/{{ element.img }}.svg\" class=\"element-icon\">\n <div class=\"hover-label\">{{ element.label }}</div> \n <div class=\"drag-dots\">\n <div class=\"dot\" *ngFor=\"let dot of dots\"></div>\n </div>\n </div>\n </ng-container>\n<!-- \n <div class=\"template-section\">\n <app-templates (templateSelected)=\"onTemplateSelected($event)\"></app-templates>\n </div> -->\n\n <!-- SKS10MAR25 footer version show -->\n <div class=\"sticky-footer-version\">\n {{version}}\n </div>\n </div>\n <!-- AP-27MAR25 Remove CDK drag and drop replace draggable function -->\n <div class=\"form-preview\">\n <!-- AP-10MAR25 Heading -->\n <div class=\"field-container\"\n style=\"width: 100%;background-color: #EFF8FF; border: 1px solid #E6F3FF;margin-bottom:10px\"\n (click)=\"selectHeading('Header')\">\n <div class=\"label-container\" style=\"padding: 10px; display: flex;justify-content: space-between;\">\n <div *ngIf=\"pdf\">\n <div *ngIf=\"pdf.title == ''\" style=\"color:#3f4a525c\">{{ 'HEADING' | nxtCustomTranslate : 'Heading' }}</div>\n <div *ngIf=\"pdf.title !== ''\">{{pdf.title}}</div>\n </div>\n <div class=\"action-buttons\">\n <button class=\"action-btn\" (click)=\"pdfSaveHandler('preview')\">{{ 'PREVIEW' | nxtCustomTranslate : 'PREVIEW' }}</button>\n <button class=\"action-btn\" (click)=\"pdfSaveHandler('download')\">{{ 'DOWNLOAD' | nxtCustomTranslate : 'DOWNLOAD' }}</button>\n </div>\n </div>\n </div>\n\n <ng-container *ngFor=\"let field of pdfElements; let i = index\">\n\n <!-- AP-19MAR25 Line Element -->\n <div *ngIf=\"field?.type === 'Line'\" class=\"line-field\" [ngClass]=\"{'isPreview': isPreview}\"\n (click)=\"selectElement(i)\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div *ngIf=\"!isPreview\" class=\"line-element\">\n <div></div>\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n <hr class=\"custom-line\" style=\"display: inline-flex\" [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-weight': field.fontWeight || '400',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'text-align': field.textAlign || 'left',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'border-style': (field.lineStyle?.toLowerCase() || 'solid'),\n 'color': field.fontColor || '#000000',\n 'margin-top': (field.paddingTop || 0) + 'px',\n 'margin-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" />\n </div>\n\n <!--SKS25MAR25 Image Upload Element -->\n <div *ngIf=\"field?.type === 'image'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [class.highlight]=\"selectedFieldIndex === i && !isPreview\" draggable=\"true && !isPreview\"\n (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\" (drop)=\"onDrop($event, i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" (mouseenter)=\"isImageHover = true;\"\n (mouseleave)=\"isImageHover = false;\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <div>\n <div *ngIf=\"isImageHover\" style=\"display: flex; justify-content: end;\">\n <svg *ngIf=\"isPreview\" (click)=\"onImageEdit(field)\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M11.1067 6.07174L9.92833 4.8934L2.16667 12.6551V13.8334H3.345L11.1067 6.07174ZM12.285 4.8934L13.4633 3.71507L12.285 2.53674L11.1067 3.71507L12.285 4.8934ZM4.035 15.5001H0.5V11.9642L11.6958 0.768403C11.8521 0.612177 12.064 0.524414 12.285 0.524414C12.506 0.524414 12.7179 0.612177 12.8742 0.768403L15.2317 3.1259C15.3879 3.28218 15.4757 3.4941 15.4757 3.71507C15.4757 3.93604 15.3879 4.14796 15.2317 4.30424L4.03583 15.5001H4.035Z\"\n fill=\"#6C757D\" />\n </svg>\n <svg *ngIf=\"isPreview\" (click)=\"onImageDelete(field)\" width=\"16\" height=\"16\"\n viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M14 3.98726C11.78 3.76726 9.54667 3.65393 7.32 3.65393C6 3.65393 4.68 3.7206 3.36 3.85393L2 3.98726\"\n stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M5.6665 3.31362L5.81317 2.44028C5.91984 1.80695 5.99984 1.33362 7.1265 1.33362H8.87317C9.99984 1.33362 10.0865 1.83362 10.1865 2.44695L10.3332 3.31362\"\n stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M12.5667 6.09375L12.1334 12.8071C12.06 13.8537 12 14.6671 10.14 14.6671H5.86002C4.00002 14.6671 3.94002 13.8537 3.86668 12.8071L3.43335 6.09375\"\n stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M6.88647 11.0004H9.10647\" stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M6.3335 8.33325H9.66683\" stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </div>\n <div class=\"logo-container\">\n <div class=\"logo-preview\" *ngIf=\"field.imageData\">\n <img [src]=\"field.imageData\" \n [style.width.px]=\"field.imageSize?.width || 100\"\n [style.height.px]=\"field.imageSize?.height || 100\" />\n </div>\n\n <div *ngIf=\"!field.imageData\" class=\"logo-upload-placeholder\">\n <label for=\"logo-upload-{{i}}\" class=\"logo-upload-label\">\n <img src=\"../assets/icons/Image.svg\" alt=\"Upload\" />\n <span>{{ 'UPLOAD_IMAGE' | nxtCustomTranslate : 'Upload Image' }}</span>\n </label>\n <input type=\"file\" id=\"logo-upload-{{i}}\" accept=\"image/*\" (change)=\"fileChangeEvent(field, $event)\"\n style=\"display: none;\" />\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Space Element -->\n <div *ngIf=\"field?.type === 'Space'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" style=\"height:93px\" [ngClass]=\"{'isPreview': isPreview}\">\n <div *ngIf=\"!isPreview\" class=\"field-content\">\n <div class=\"label-container\">\n <div class=\"top-right\" style=\"margin: -11px -11px 0 0;\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- TextArea -->\n <div *ngIf=\"field?.type === 'text'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <textarea class=\"custom-textarea\"\n [placeholder]=\"'ENTER_DETAILED_TEXT_HERE' | nxtCustomTranslate : 'Enter detailed text here...'\"\n [value]=\"field.value? field.value : ''\"\n (input)=\"onTextAreaInput($event, field)\"\n #autoTextarea\n [style.height.px]=\"isPreview ? 40 : null\"\n (click)=\"openTextSettings(field, $event)\"\n (ngModelChange)=\"updateContent()\" \n [ngStyle]=\"{\n 'font-size': field.style?.fontSize ? field.style.fontSize + 'px' : '14px',\n 'font-weight': field.style?.bold ? 'bold' : 'normal',\n 'font-style': field.style?.italics ? 'italic' : 'normal',\n 'text-decoration': field.style?.decoration ? 'decoration' : 'none',\n 'text-align': field.style?.alignment || 'left',\n 'margin': field.style?.margin?.length === 4\n ? field.style.margin[1] + 'px ' + field.style.margin[2] + 'px ' +\n field.style.margin[3] + 'px ' + field.style.margin[0] + 'px'\n : '0',\n 'color': field.style?.color || '#000'\n }\"></textarea>\n \n </div>\n </div>\n\n <!--AP-05JUN25 Text Settings Toolbar: Allows editing font size, bold/italic styles, alignment, and closing the toolbar for the selected column -->\n <div *ngIf=\"showTextSettings && isPreview && selectedColumn === field\" class=\"text-settings-toolbar\"\n [ngStyle]=\"{\n position: 'absolute',\n background: '#fff',\n border: '1px solid #ccc',\n padding: '6px 10px',\n 'border-radius': '4px',\n 'box-shadow': '0 2px 6px rgba(0, 0, 0, 0.1)',\n 'z-index': '65535',\n display: 'flex',\n 'align-items': 'center',\n gap: '8px'\n }\">\n \n <!-- Font Size -->\n <select [(ngModel)]=\"selectedColumn.style.fontSize\" class=\"toolbar-select\">\n <option [value]=\"12\">12pt</option>\n <option [value]=\"14\">14pt</option>\n <option [value]=\"16\">16pt</option>\n <option [value]=\"18\">18pt</option>\n <option [value]=\"24\">24pt</option>\n </select>\n\n <!-- Text Color -->\n <label title=\"Text Color\" class=\"color-label\">\n <span class=\"color-box\" [style.color]=\"selectedColumn.style.color || '#000'\" (click)=\"textColorInput.click()\">A</span>\n <input #textColorInput type=\"color\" [(ngModel)]=\"selectedColumn.style.color\" (change)=\"updateContent()\" class=\"color-picker-hidden\"/>\n </label>\n\n <!-- Bold -->\n <button (click)=\"toggleStyle('bold')\" [class.active]=\"selectedColumn?.style?.bold\" class=\"toolbar-btn\">{{ 'BOLD' | nxtCustomTranslate : 'B' }}</button>\n \n <!-- Italic -->\n <button (click)=\"toggleStyle('italic')\" [class.active]=\"selectedColumn?.style?.italics\" class=\"toolbar-btn\"><i>{{ 'ITALIC' | nxtCustomTranslate : 'I' }}</i></button>\n\n <!-- Underline -->\n <!-- <button (click)=\"toggleStyle('underline')\" [class.active]=\"selectedColumn?.style?.underline\" class=\"toolbar-btn\"><u>U</u></button> -->\n\n \n <!-- Align Left -->\n <button (click)=\"setAlignment('left')\" [class.active]=\"selectedColumn?.style?.alignment === 'left'\" class=\"toolbar-btn\">\n <span class=\"material-icons\">{{ 'FORMAT_ALIGN_LEFT' | nxtCustomTranslate : 'format_align_left' }}</span>\n </button>\n \n <!-- Align Center -->\n <button (click)=\"setAlignment('center')\" [class.active]=\"selectedColumn?.style?.alignment === 'center'\" class=\"toolbar-btn\">\n <span class=\"material-icons\">{{ 'FORMAT_ALIGN_CENTER' | nxtCustomTranslate : 'format_align_center' }}</span>\n </button>\n \n <!-- Align Right -->\n <button (click)=\"setAlignment('right')\" [class.active]=\"selectedColumn?.style?.alignment === 'right'\" class=\"toolbar-btn\">\n <span class=\"material-icons\">{{ 'FORMAT_ALIGN_RIGHT' | nxtCustomTranslate : 'format_align_right' }}</span>\n </button>\n \n <!-- Close -->\n <button (click)=\"closeTextSettings()\" class=\"toolbar-btn\" style=\"margin-left: auto;\">\u2715</button>\n </div>\n \n </div>\n\n <!-- Input -->\n <div *ngIf=\"field?.type === 'input'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <!-- AP-04JUN25 - replace app-custom-input -->\n <nxt-input [type]=\"'text'\" [mode]=\"'edit'\" [value]=\"field.value\" [question]=\"\" [labelFont]=\"\"\n [label]=\"field?.questionText\" [labelColor]=\"\" [labelSize]=\"\" [inputValueSize]=\"\" [labelWeight]=\"\"\n [inputWeight]=\"\" [showLabel]=\"\" inputBorder=\"none\" svgHeight=\"20px\" svgWidth=\"20px\"\n [placeholder]=\"field.question ? field.question : 'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" [required]=\"\"\n inputBgColor=\"#F5F5F5\" [inputId]=\"\" [errorMessages]=\"{ required: 'This field is required' }\"\n [inputIconLeftSrc]=\"''\" (inputValue)=\"onQuestionChange($event, field)\">\n </nxt-input>\n </div>\n </div>\n </div>\n\n <!-- Date -->\n <div *ngIf=\"field?.type === 'date'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <!-- AP-04JUN25 - replace custom date picker -->\n <nxt-input [type]=\"'date'\" [mode]=\"'edit'\" [value]=\"field.value\" [question]=\"\" [labelFont]=\"\"\n [label]=\"field?.questionText\" [labelColor]=\"\" [labelSize]=\"\" [inputValueSize]=\"\" [labelWeight]=\"\"\n [inputWeight]=\"\" [showLabel]=\"\" inputBorder=\"none\" svgHeight=\"20px\" svgWidth=\"20px\"\n [placeholder]=\"field.question ? field.question : 'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" [required]=\"\"\n inputBgColor=\"#F5F5F5\" [inputId]=\"\" [errorMessages]=\"{ required: 'This field is required' }\"\n [inputIconLeftSrc]=\"''\" (inputValue)=\"onQuestionChange($event, field)\">\n </nxt-input>\n </div>\n </div>\n </div>\n\n <!-- Pdf -->\n <div *ngIf=\"field?.type === 'Pdf'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <!-- <lib-booklet [bookletJSON]=\"field.pdfReferenceQuestions\"></lib-booklet> -->\n <div *ngIf=\"field?.pdfReferenceQuestions\">\n <ng-container *ngFor=\"let field of field?.pdfReferenceQuestions[field?.pdfReference]; let i = index\">\n <!-- AP-19MAR25 Line Element -->\n <div *ngIf=\"field?.type === 'Line'\" class=\"line-field\">\n <hr class=\"custom-line\" style=\"display: inline-flex\" [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-weight': field.fontWeight || '400',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'text-align': field.textAlign || 'left',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'border-style': (field.lineStyle?.toLowerCase() || 'solid'),\n 'color': field.fontColor || '#000000',\n 'margin-top': (field.paddingTop || 0) + 'px',\n 'margin-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" />\n </div>\n\n <div *ngIf=\"field?.type === 'image'\" class=\"field-container\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div>\n <div class=\"logo-container\">\n <!-- Logo preview area -->\n <div class=\"logo-preview\" *ngIf=\"field.imageData\">\n <img [src]=\"field.imageData\" />\n </div>\n\n <!-- Upload button -->\n <div *ngIf=\"!field.imageData\" class=\"logo-upload-placeholder\">\n <label for=\"logo-upload-{{i}}\" class=\"logo-upload-label\">\n <img src=\"../assets/icons/Image.svg\" alt=\"Upload\" />\n <span>{{ 'UPLOAD_IMAGE' | nxtCustomTranslate : 'Upload Image' }}</span>\n </label>\n <input type=\"file\" id=\"logo-upload-{{i}}\" accept=\"image/*\"\n (change)=\"fileChangeEvent(field, $event)\" style=\"display: none;\" />\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div *ngIf=\"field?.type === 'Space'\" class=\"field-container\">\n <div class=\"field-wrapper\" style=\"height:93px\" [ngClass]=\"{'isPreview': isPreview}\">\n </div>\n </div>\n <div *ngIf=\"field?.type === 'text'\" class=\"field-container\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <textarea class=\"custom-textarea\" [placeholder]=\"'ENTER_DETAILED_TEXT_HERE' | nxtCustomTranslate : 'Enter detailed text here...'\"\n [style.height.px]=\"isPreview ? 40 : field.size || 100\" [(ngModel)]=\"field.value\"\n (ngModelChange)=\"onQuestionChange($event, field)\"></textarea>\n </div>\n </div>\n </div>\n <!-- AP-14JUN25 Added nxt-input in pdf -->\n <div *ngIf=\"field?.type === 'input'\" class=\"field-container\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\"> \n <nxt-input [type]=\"'text'\" [mode]=\"'edit'\" [(ngModel)]=\"field.value\"\n [placeholder]=\"'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" (ngModelChange)=\"onQuestionChange($event, field)\">\n </nxt-input>\n </div>\n </div>\n </div>\n <!-- AP-06MAR25 -->\n <div *ngIf=\"field?.type === 'Table'\" class=\"field-container\">\n <div class=\"field-wrapper\" style=\"overflow: hidden;\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div class=\"nxt-table-container\">\n <nxt-datatable isButtons [question]=\"field\" from=\"formBuilder\"\n [mode]=\"isPreview ? 'view' : 'edit'\" [apiMeta]=\"field?.subText\"\n [tableConfig]=\"field.tableConfig\" tableId=\"\" direction=\"ltr\" tableWidth=\"auto\"\n (valueChange)=\"onQuestionChange($event, field)\"\n [data]=\"field.value?.data\"\n isEditable=true (columnSelected)=columnSelected($event) (removeColumn)=removeColumn($event)>\n </nxt-datatable>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Table -->\n <!-- AP-06MAR25 -->\n <div *ngIf=\"field?.type === 'Table'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" style=\"overflow: hidden;\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <div class=\"nxt-table-container\">\n <nxt-datatable isButtons [question]=\"field\" from=\"formBuilder\" [mode]=\"isPreview ? 'view' : 'edit'\"\n [apiMeta]=\"field?.subText\" [tableConfig]=\"field.tableConfig\" tableId=\"\" direction=\"ltr\"\n tableWidth=\"auto\" isEditable=true (columnSelected)=columnSelected($event)\n [data]=\"field.value?.data\"\n (valueChange)=\"onQuestionChange($event, field)\"\n (removeColumn)=removeColumn($event)>\n </nxt-datatable>\n </div>\n </div>\n </div>\n </div>\n\n </ng-container>\n </div>\n </div>\n <!-- SKS13MAR25 popup conformation box -->\n <div class=\"dialog-overlay\" *ngIf=\"isSelectTablePopup\">\n <div class=\"dialog-box\">\n <button class=\"close-btn-fb\" (click)=\"onClose()\">\u2715</button>\n <p>{{ 'ADD_TABLE_CONFIRMATION' | nxtCustomTranslate : 'These element want to add a table' }}</p>\n <div class=\"button-container-fb\">\n <button class=\"yes-btn-fb\" (click)=\"addOnTable()\">{{ 'YES' | nxtCustomTranslate : 'Yes' }}</button>\n <button class=\"no-btn-fb\" (click)=\"onClose()\">{{ 'NO' | nxtCustomTranslate : 'No' }}</button>\n </div>\n </div>\n </div>\n <app-pdf-properties *ngIf=\"!isPreview\" (formButtonHandler)=\"pdfSaveHandler($event)\"></app-pdf-properties>\n</div>\n<!--SKS25MAR25 Modal Overlay -->\n<div class=\"modal-overlay\" *ngIf=\"isImageEdit\">\n <div class=\"modal-content\">\n <span class=\"close-button\" (click)=\"closeModal()\">\u00D7</span>\n <!-- Image Editor -->\n <div *ngIf=\"selectedImageElement?.type === 'image'\">\n <image-cropper *ngIf=\"selectedImageElement.imageData\" [imageBase64]=\"selectedImageElement.orgImageData\"\n [disabled]=\"false\" [alignImage]=\"alignImage\" [roundCropper]=\"roundCropper\" [backgroundColor]=\"'white'\"\n imageAltText=\"{{ 'ALTERNATIVE_IMAGE_TEXT' | nxtCustomTranslate : '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)\">\n </image-cropper>\n\n <!--AP-20JUN25 - Image size controls: Allow user to manually adjust width and height using range sliders -->\n <div *ngIf=\"selectedImageElement.imageData\" class=\"manual-size-inputs\">\n <label>{{ 'WIDTH' | nxtCustomTranslate : 'Width' }} {{ selectedImageElement.imageSize.width }}</label>\n <input type=\"range\" min=\"50\" max=\"300\" step=\"1\"\n [(ngModel)]=\"selectedImageElement.imageSize.width\"\n (input)=\"onImageSizeChange()\" />\n \n <label>{{ 'HEIGHT' | nxtCustomTranslate : 'Height' }} {{ selectedImageElement.imageSize.height }}</label>\n <input type=\"range\" min=\"50\" max=\"300\" step=\"1\"\n [(ngModel)]=\"selectedImageElement.imageSize.height\"\n (input)=\"onImageSizeChange()\" />\n </div>\n \n\n <!-- Controls -->\n <div *ngIf=\"selectedImageElement.imageData\" class=\"controls\">\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateLeft()\" title=\"{{ 'ROTATE_LEFT' | nxtCustomTranslate : 'Rotate Left' }}\">\u27F2</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateRight()\" title=\"{{ 'ROTATE_RIGHT' | nxtCustomTranslate : 'Rotate Right' }}\">\u27F3</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomOut()\" title=\"{{ 'ZOOM_OUT' | nxtCustomTranslate : 'Zoom Out' }}\">-</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomIn()\" title=\"{{ 'ZOOM_IN' | nxtCustomTranslate : 'Zoom In' }}\">+</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveLeft()\" title=\"{{ 'MOVE_LEFT' | nxtCustomTranslate : 'Move Left' }}\">\u2190</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveRight()\" title=\"{{ 'MOVE_RIGHT' | nxtCustomTranslate : 'Move Right' }}\">\u2192</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveUp()\" title=\"{{ 'MOVE_UP' | nxtCustomTranslate : 'Move Up' }}\">\u2191</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveDown()\" title=\"{{ 'MOVE_DOWN' | nxtCustomTranslate : 'Move Down' }}\">\u2193</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipHorizontal()\" [class.enabled]=\"transform.flipH\"\n title=\"{{ 'FLIP_HORIZONTALLY' | nxtCustomTranslate : 'Flip Horizontally' }}\">\u2194</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipVertical()\" [class.enabled]=\"transform.flipV\"\n title=\"{{ 'FLIP_VERTICALLY' | nxtCustomTranslate : 'Flip Vertically' }}\">\u2195</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"resetImage()\" title=\"{{ 'RESET' | nxtCustomTranslate : 'Reset' }}\">\u00D7</div>\n </div>\n </div>\n </div>\n</div>", styles: [".center-frame{display:flex;border-right-width:0}.head-elements{font-size:17px;font-weight:600}.form-builder{width:33.33%;height:calc(100vh - 20px);overflow-y:auto;background-color:#fff;padding:10px;border-right:10px solid #86A8CD;box-shadow:2px 2px 10px #0000001a}.form-builder .element{display:flex;align-items:center;gap:15px;margin-top:10px;padding:10px;border-radius:5px;background:#f8fafc;cursor:pointer;border-left:10px solid #E2F1FF;position:relative;color:#000}.form-builder .element:hover{background:#0250d9;color:#fff;border-left:10px solid #234465}.form-builder .element img{width:20px;height:20px;transition:filter .3s ease}.form-builder .element:hover img{filter:invert(1)}.form-builder .hover-label{font-size:15px;font-weight:400;color:#000;transition:color .3s ease}.form-builder .element:hover .hover-label{color:#fff}.form-builder .section-title{font-weight:700;font-size:16px;margin-top:10px;padding:5px;border-bottom:1px solid #ddd;color:#000;display:flex;justify-content:space-between;align-items:center}.form-builder .section-title:after{content:\"\\25bc\";font-size:12px;color:#555}.form-builder .section{margin-bottom:10px}.toggle-header{display:flex;align-items:center;justify-content:space-between;cursor:pointer;padding:10px}.arrow-icon{width:19px;height:23px;transition:transform .3s ease}.toggle-header:hover .arrow-icon{transform:scale(1.1)}.drag-dots{display:grid;grid-template-columns:repeat(2,1fr);grid-template-rows:repeat(3,1fr);gap:3px;position:absolute;right:15px;top:50%;transform:translateY(-50%)}.dot{width:5px;height:5px;border-radius:50%;background-color:#cbd5e1;transition:background-color .3s ease}.form-builder .element:hover .dot{background-color:#ffffffb3}.field-wrapper{background-color:#eff8ff;border:1px solid #E6F3FF;border-radius:5px;padding:10px}.field-wrapper.isPreview{background-color:transparent;border:none;border-radius:0;padding:0}.required:after{content:\"*\";color:red;margin-left:5px}.custom-input{width:100%;padding:8px;border:1px solid #DDDBDA;background-color:#fff;border-radius:5px;outline:none}.custom-input:focus{border-color:#00008b;box-shadow:0 0 5px #0000ff80}.delete-icon{width:19px;height:19px;margin-top:-5px}.field-container{padding-right:5px;padding-left:5px;transition:background .2s}.field-container:hover .top-right,.field-container.highlight .top-right,.label-container:hover .delete-icon{opacity:1;visibility:visible}.form-preview{width:100%;height:calc(100vh - 20px);overflow-y:auto;display:flex;flex-wrap:wrap;align-items:flex-start;padding:10px;height:fit-content;max-height:calc(100vh - 20px)}.field-content{display:flex;flex-direction:column;gap:5px}.label-container{display:flex;justify-content:flex-end;align-items:center}.top-right{display:flex;gap:10px;padding:6px;border:1px solid #d7edff;margin-top:-10px;background-color:#d7edff;margin-right:-10px;opacity:0;visibility:hidden;cursor:grab;transition:opacity .1s ease-in-out,visibility .1s ease-in-out}.drag-dot{height:13px}.custom-input,.custom-textarea,.dropdown,.checkbox-options-container,.radio-options-container{width:100%}.form-builder .element .drag-dots:active{cursor:grabbing}.dropdown{width:100%;padding:8px;border:1px solid #ccc;border-radius:4px;background-color:#fff;font-size:14px;color:#333;outline:none;cursor:pointer}.dropdown:focus{border-color:#007bff;box-shadow:0 0 5px #007bff80}.line-field{width:100%;margin:10px 0;background-color:#eff8ff;border:1px solid #E6F3FF;border-radius:5px;padding:3px}.line-field{background-color:transparent;border:none;border-radius:0;padding:3px}.checkbox-field-container:hover{box-shadow:0 2px 8px #0000001a}.checkbox-options-container{display:flex;flex-direction:column;gap:5px;padding:8px;min-height:38px;border:1px solid #DDDBDA;background-color:#fff;outline:none;border-radius:6px}.checkbox-option,.radio-option{display:flex;align-items:center;gap:8px;padding:8px;background-color:#fff;border-radius:4px;transition:background-color .2s ease}.checkbox-option:hover{background-color:#f1f3f5}.checkbox-input,.radio-input{width:18px;height:18px;accent-color:#4dabf7;cursor:pointer}.checkbox-label,.radio-label{font-size:14px;color:#495057;cursor:pointer;-webkit-user-select:none;user-select:none}.label-container label{font-size:15px;font-weight:400}.required:after{content:\" *\";color:red}.radio-options-container{display:flex;flex-direction:column;gap:5px;padding:8px;min-height:38px;border:1px solid #DDDBDA;background-color:#fff;outline:none;border-radius:6px}.radio-input:checked{border-color:#007bff;background-color:#007bff}.radio-input:checked:after{content:\"\";width:8px;height:8px;background:#fff;border-radius:50%;position:absolute;top:50%;left:50%;transform:translate(-50%,-50%)}.custom-textarea{width:100%;min-height:100px;border:1px solid #ccc;border-radius:4px;padding:8px;resize:vertical}.highlight{border:2px solid #5592FD!important;background-color:#eff8ff}.table-container label{font-size:14px;font-weight:700;margin-bottom:5px;display:block}.sticky-footer-version{position:fixed;bottom:0;padding:10px;text-align:center}.nxt-table-container{display:flex;justify-content:center;align-items:center;width:100%}nxt-datatable{width:100%!important;table-layout:fixed;max-width:100%}.dialog-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:#00000080;display:flex;justify-content:center;align-items:center;z-index:1000}.dialog-box{background:#fff;padding:20px;border-radius:5px;text-align:center;width:300px;position:relative}.close-btn-fb{position:absolute;top:4px;right:4px;background:#ff4242;color:#fff;border:none;border-radius:50%;width:20px;height:20px;font-size:10px;cursor:pointer}.button-container-fb{display:flex;justify-content:flex-end;gap:10px}.yes-btn-fb{background:green;color:#fff;border:none;padding:0 12px;border-radius:5px;cursor:pointer}.no-btn-fb{background:gray;color:#fff;border:none;padding:8px 15px;border-radius:5px;cursor:pointer}.line-element{width:100%;display:flex;justify-content:space-between;align-items:center}.custom-line{width:100%;border:1px solid #000}.logo-upload-placeholder{display:flex;flex-direction:column;align-items:center;justify-content:center;width:150px;height:150px;border:2px dashed #ccc;cursor:pointer;inline-size:auto}.logo-upload-label{display:flex;flex-direction:column;align-items:center;cursor:pointer;color:#666}.logo-upload-label img{width:32px;height:32px;margin-bottom:8px}.element.disabled{opacity:.5;cursor:not-allowed}.form-container{display:flex;flex-wrap:wrap;border:10px solid #86A8CD}.form-container.isPreview{border:none}.center-frame{width:74%}.center-frame.isPreview{width:100%}app-pdf-properties{border-left:10px solid #86A8CD;width:25%}:host ::ng-deep .questiondiv1{padding-left:0!important;padding-right:0!important}:host ::ng-deep .form-group.content-box{padding-bottom:0!important}:host ::ng-deep .custom-textarea{min-height:40px!important}.modal-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:100}.modal-content{background:#fff;padding:20px;width:500px;max-width:90%;border-radius:8px;position:relative;text-align:center}.close-button{position:absolute;top:6px;right:6px;font-size:25px;cursor:pointer;z-index:10}.controls{display:flex;gap:5px;justify-content:center;margin-top:10px}.logo-icon{cursor:pointer;padding:5px;border-radius:5px;background:#d3d3d3}.logo-icon:hover{background:#a9a9a9}.enabled{background:#007bff;color:#fff}.text-settings-toolbar{position:absolute;display:flex;flex-wrap:wrap;align-items:center;gap:8px;background:#fff;border:1px solid #d1d5db;padding:8px 10px;border-radius:8px;box-shadow:0 4px 12px #0000001a;z-index:65535;max-width:100%;width:auto;min-width:200px;box-sizing:border-box}.toolbar-select{padding:4px 6px;font-size:13px;border:1px solid #d1d5db;border-radius:6px;background:#fff;cursor:pointer;min-width:60px;width:auto}.toolbar-btn{background:#f3f4f6;border:1px solid #d1d5db;padding:6px 8px;border-radius:6px;cursor:pointer;font-size:13px;min-width:30px;height:30px;display:flex;align-items:center;justify-content:center}.toolbar-btn:hover{background:#e5e7eb;border-color:#9ca3af}.toolbar-btn.active{background:#005aaa;color:#fff;border-color:#005aaa}.color-label{display:flex;align-items:center}.color-box{font-weight:700;width:22px;height:22px;font-size:13px;border:1px solid #ccc;border-radius:4px;text-align:center;line-height:22px;cursor:pointer;padding:0}.color-picker-hidden{position:absolute;opacity:0;pointer-events:none;width:0;height:0}@media (max-width: 600px){.text-settings-toolbar{flex-direction:row;flex-wrap:wrap;gap:6px;padding:6px;justify-content:flex-start}.toolbar-btn,.toolbar-select{font-size:12px;padding:5px}.color-box{width:20px;height:20px;font-size:12px;line-height:20px}}.action-buttons{display:flex;align-items:center;gap:10px;flex-wrap:wrap}.action-btn{display:flex;gap:16px;padding:10px 15px;border-radius:8px;border:none}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { 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: PdfPropertiesComponent, selector: "app-pdf-properties", inputs: ["selectedElementType", "templateSelected"], outputs: ["formButtonHandler", "templateSaveHandler"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i3.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i4.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i4.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.RangeValueAccessor, selector: "input[type=range][formControlName],input[type=range][formControl],input[type=range][ngModel]" }, { kind: "directive", type: i4.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: NxtDatatable, selector: "nxt-datatable", inputs: ["data", "tableFilterData", "columns", "withCheckBox", "searchBar", "tableSaveButton", "stickyColumn", "tableWidth", "actionColumHeader", "actionButton", "title", "isButtons", "buttonArray", "tableId", "isEditRow", "isDeleteRow", "addInlineRecord", "searchConfigs", "direction", "pagination", "actionButtonArray", "multipleFilter", "isPagination", "isNosIndicator", "isEditable", "from", "question", "rowTextSize", "rowTextColor", "apiMeta", "summaryRows", "summaryColumns", "isLoading", "tableConfig", "tableParams", "listViews", "mode", "languageCode", "selectedColumn", "allIcons"], outputs: ["tableRowClick", "onEditData", "saveButtonData", "onDeleteData", "buttonEmit", "hyperLinkEmit", "sideNavEmit", "actionButtonEmit", "columnSelected", "removeColumn", "valueChange", "selectedValues", "fileEmit", "NxtTableParamsEmit", "NxtTableFilterEmit", "hadleDropDownDependent", "NxtTableEmit"] }, { kind: "component", type: NxtInput, selector: "nxt-input", inputs: ["label", "labelFont", "labelWeight", "inputWeight", "labelSize", "inputValueSize", "labelColor", "showLabel", "svgHeight", "svgWidth", "type", "inputIconRightSrc", "inputIconLeftSrc", "required", "minLength", "pattern", "errorMessages", "maxLength", "placeholder", "inputBgColor", "inputBorder", "placeholderColor", "placeholderFont", "placeholderWeight", "placeholderSize", "inputTextColor", "inputHeight", "inputWidth", "inputId", "inputBorderSize", "inputConfig", "confPassVal", "confPass", "mode", "value", "question", "showSuggestion", "ariaOwns", "ariaHasPopup", "isLoading", "options", "minDate", "maxDate", "rows", "from", "hyperLink", "size"], outputs: ["valueChange", "inputValue", "onBlur", "onFocus", "toggleEmit", "nativeInputRef", "removeValueEmit", "hyperlinkEmit"] }, { kind: "pipe", type: NxtCustomTranslatePipe, name: "nxtCustomTranslate" }] });
|
|
619
|
-
}
|
|
620
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PdfDesignerComponent, decorators: [{
|
|
621
|
-
type: Component,
|
|
622
|
-
args: [{ selector: 'app-pdf-designer', standalone: true, imports: [CommonModule, ImageCropperComponent, PdfPropertiesComponent, MatTooltipModule, FormsModule, NxtDatatable, BookletComponent, AppTemplatesComponent, NxtInput, NxtCustomTranslatePipe], template: "<!-- AP 22JAN25 - form preview and All form elements -->\n<!-- AP 25FEB25 - All elements update -->\n<div class=\"form-container\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"center-frame\" [ngClass]=\"{'isPreview': isPreview}\">\n <!-- Form Builder Section All Elements -->\n <div *ngIf=\"!isPreview\" class=\"form-builder\">\n\n <ng-container *ngFor=\"let element of elementsList\">\n <div\n class=\"element\"\n [class.disabled]=\"selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)\"\n (click)=\"!(selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)) && addElement(element.type)\"\n [draggable]=\"!(selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type))\"\n [matTooltip]=\"\n selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)\n ? ('THIS_ELEMENT_NOT_SUPPORTED' | nxtCustomTranslate : 'This element is not supported when a ') +\n (selectedElement?.type || ('CERTAIN_ELEMENT' | nxtCustomTranslate : 'certain element')) +\n (' IS_PRESENT' | nxtCustomTranslate : ' is present')\n : null\n \"\n >\n <img src=\"../assets/icons/{{ element.img }}.svg\" class=\"element-icon\">\n <div class=\"hover-label\">{{ element.label }}</div> \n <div class=\"drag-dots\">\n <div class=\"dot\" *ngFor=\"let dot of dots\"></div>\n </div>\n </div>\n </ng-container>\n<!-- \n <div class=\"template-section\">\n <app-templates (templateSelected)=\"onTemplateSelected($event)\"></app-templates>\n </div> -->\n\n <!-- SKS10MAR25 footer version show -->\n <div class=\"sticky-footer-version\">\n {{version}}\n </div>\n </div>\n <!-- AP-27MAR25 Remove CDK drag and drop replace draggable function -->\n <div class=\"form-preview\">\n <!-- AP-10MAR25 Heading -->\n <div class=\"field-container\"\n style=\"width: 100%;background-color: #EFF8FF; border: 1px solid #E6F3FF;margin-bottom:10px\"\n (click)=\"selectHeading('Header')\">\n <div class=\"label-container\" style=\"padding: 10px; display: flex;justify-content: space-between;\">\n <div *ngIf=\"pdf\">\n <div *ngIf=\"pdf.title == ''\" style=\"color:#3f4a525c\">{{ 'HEADING' | nxtCustomTranslate : 'Heading' }}</div>\n <div *ngIf=\"pdf.title !== ''\">{{pdf.title}}</div>\n </div>\n <div class=\"action-buttons\">\n <button class=\"action-btn\" (click)=\"pdfSaveHandler('preview')\">{{ 'PREVIEW' | nxtCustomTranslate : 'PREVIEW' }}</button>\n <button class=\"action-btn\" (click)=\"pdfSaveHandler('download')\">{{ 'DOWNLOAD' | nxtCustomTranslate : 'DOWNLOAD' }}</button>\n </div>\n </div>\n </div>\n\n <ng-container *ngFor=\"let field of pdfElements; let i = index\">\n\n <!-- AP-19MAR25 Line Element -->\n <div *ngIf=\"field?.type === 'Line'\" class=\"line-field\" [ngClass]=\"{'isPreview': isPreview}\"\n (click)=\"selectElement(i)\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div *ngIf=\"!isPreview\" class=\"line-element\">\n <div></div>\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n <hr class=\"custom-line\" style=\"display: inline-flex\" [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-weight': field.fontWeight || '400',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'text-align': field.textAlign || 'left',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'border-style': (field.lineStyle?.toLowerCase() || 'solid'),\n 'color': field.fontColor || '#000000',\n 'margin-top': (field.paddingTop || 0) + 'px',\n 'margin-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" />\n </div>\n\n <!--SKS25MAR25 Image Upload Element -->\n <div *ngIf=\"field?.type === 'image'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [class.highlight]=\"selectedFieldIndex === i && !isPreview\" draggable=\"true && !isPreview\"\n (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\" (drop)=\"onDrop($event, i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" (mouseenter)=\"isImageHover = true;\"\n (mouseleave)=\"isImageHover = false;\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <div>\n <div *ngIf=\"isImageHover\" style=\"display: flex; justify-content: end;\">\n <svg *ngIf=\"isPreview\" (click)=\"onImageEdit(field)\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M11.1067 6.07174L9.92833 4.8934L2.16667 12.6551V13.8334H3.345L11.1067 6.07174ZM12.285 4.8934L13.4633 3.71507L12.285 2.53674L11.1067 3.71507L12.285 4.8934ZM4.035 15.5001H0.5V11.9642L11.6958 0.768403C11.8521 0.612177 12.064 0.524414 12.285 0.524414C12.506 0.524414 12.7179 0.612177 12.8742 0.768403L15.2317 3.1259C15.3879 3.28218 15.4757 3.4941 15.4757 3.71507C15.4757 3.93604 15.3879 4.14796 15.2317 4.30424L4.03583 15.5001H4.035Z\"\n fill=\"#6C757D\" />\n </svg>\n <svg *ngIf=\"isPreview\" (click)=\"onImageDelete(field)\" width=\"16\" height=\"16\"\n viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M14 3.98726C11.78 3.76726 9.54667 3.65393 7.32 3.65393C6 3.65393 4.68 3.7206 3.36 3.85393L2 3.98726\"\n stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M5.6665 3.31362L5.81317 2.44028C5.91984 1.80695 5.99984 1.33362 7.1265 1.33362H8.87317C9.99984 1.33362 10.0865 1.83362 10.1865 2.44695L10.3332 3.31362\"\n stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M12.5667 6.09375L12.1334 12.8071C12.06 13.8537 12 14.6671 10.14 14.6671H5.86002C4.00002 14.6671 3.94002 13.8537 3.86668 12.8071L3.43335 6.09375\"\n stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M6.88647 11.0004H9.10647\" stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M6.3335 8.33325H9.66683\" stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </div>\n <div class=\"logo-container\">\n <div class=\"logo-preview\" *ngIf=\"field.imageData\">\n <img [src]=\"field.imageData\" \n [style.width.px]=\"field.imageSize?.width || 100\"\n [style.height.px]=\"field.imageSize?.height || 100\" />\n </div>\n\n <div *ngIf=\"!field.imageData\" class=\"logo-upload-placeholder\">\n <label for=\"logo-upload-{{i}}\" class=\"logo-upload-label\">\n <img src=\"../assets/icons/Image.svg\" alt=\"Upload\" />\n <span>{{ 'UPLOAD_IMAGE' | nxtCustomTranslate : 'Upload Image' }}</span>\n </label>\n <input type=\"file\" id=\"logo-upload-{{i}}\" accept=\"image/*\" (change)=\"fileChangeEvent(field, $event)\"\n style=\"display: none;\" />\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Space Element -->\n <div *ngIf=\"field?.type === 'Space'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" style=\"height:93px\" [ngClass]=\"{'isPreview': isPreview}\">\n <div *ngIf=\"!isPreview\" class=\"field-content\">\n <div class=\"label-container\">\n <div class=\"top-right\" style=\"margin: -11px -11px 0 0;\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- TextArea -->\n <div *ngIf=\"field?.type === 'text'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <textarea class=\"custom-textarea\"\n [placeholder]=\"'ENTER_DETAILED_TEXT_HERE' | nxtCustomTranslate : 'Enter detailed text here...'\"\n [value]=\"field.value? field.value : ''\"\n (input)=\"onTextAreaInput($event, field)\"\n #autoTextarea\n [style.height.px]=\"isPreview ? 40 : null\"\n (click)=\"openTextSettings(field, $event)\"\n (ngModelChange)=\"updateContent()\" \n [ngStyle]=\"{\n 'font-size': field.style?.fontSize ? field.style.fontSize + 'px' : '14px',\n 'font-weight': field.style?.bold ? 'bold' : 'normal',\n 'font-style': field.style?.italics ? 'italic' : 'normal',\n 'text-decoration': field.style?.decoration ? 'decoration' : 'none',\n 'text-align': field.style?.alignment || 'left',\n 'margin': field.style?.margin?.length === 4\n ? field.style.margin[1] + 'px ' + field.style.margin[2] + 'px ' +\n field.style.margin[3] + 'px ' + field.style.margin[0] + 'px'\n : '0',\n 'color': field.style?.color || '#000'\n }\"></textarea>\n \n </div>\n </div>\n\n <!--AP-05JUN25 Text Settings Toolbar: Allows editing font size, bold/italic styles, alignment, and closing the toolbar for the selected column -->\n <div *ngIf=\"showTextSettings && isPreview && selectedColumn === field\" class=\"text-settings-toolbar\"\n [ngStyle]=\"{\n position: 'absolute',\n background: '#fff',\n border: '1px solid #ccc',\n padding: '6px 10px',\n 'border-radius': '4px',\n 'box-shadow': '0 2px 6px rgba(0, 0, 0, 0.1)',\n 'z-index': '65535',\n display: 'flex',\n 'align-items': 'center',\n gap: '8px'\n }\">\n \n <!-- Font Size -->\n <select [(ngModel)]=\"selectedColumn.style.fontSize\" class=\"toolbar-select\">\n <option [value]=\"12\">12pt</option>\n <option [value]=\"14\">14pt</option>\n <option [value]=\"16\">16pt</option>\n <option [value]=\"18\">18pt</option>\n <option [value]=\"24\">24pt</option>\n </select>\n\n <!-- Text Color -->\n <label title=\"Text Color\" class=\"color-label\">\n <span class=\"color-box\" [style.color]=\"selectedColumn.style.color || '#000'\" (click)=\"textColorInput.click()\">A</span>\n <input #textColorInput type=\"color\" [(ngModel)]=\"selectedColumn.style.color\" (change)=\"updateContent()\" class=\"color-picker-hidden\"/>\n </label>\n\n <!-- Bold -->\n <button (click)=\"toggleStyle('bold')\" [class.active]=\"selectedColumn?.style?.bold\" class=\"toolbar-btn\">{{ 'BOLD' | nxtCustomTranslate : 'B' }}</button>\n \n <!-- Italic -->\n <button (click)=\"toggleStyle('italic')\" [class.active]=\"selectedColumn?.style?.italics\" class=\"toolbar-btn\"><i>{{ 'ITALIC' | nxtCustomTranslate : 'I' }}</i></button>\n\n <!-- Underline -->\n <!-- <button (click)=\"toggleStyle('underline')\" [class.active]=\"selectedColumn?.style?.underline\" class=\"toolbar-btn\"><u>U</u></button> -->\n\n \n <!-- Align Left -->\n <button (click)=\"setAlignment('left')\" [class.active]=\"selectedColumn?.style?.alignment === 'left'\" class=\"toolbar-btn\">\n <span class=\"material-icons\">{{ 'FORMAT_ALIGN_LEFT' | nxtCustomTranslate : 'format_align_left' }}</span>\n </button>\n \n <!-- Align Center -->\n <button (click)=\"setAlignment('center')\" [class.active]=\"selectedColumn?.style?.alignment === 'center'\" class=\"toolbar-btn\">\n <span class=\"material-icons\">{{ 'FORMAT_ALIGN_CENTER' | nxtCustomTranslate : 'format_align_center' }}</span>\n </button>\n \n <!-- Align Right -->\n <button (click)=\"setAlignment('right')\" [class.active]=\"selectedColumn?.style?.alignment === 'right'\" class=\"toolbar-btn\">\n <span class=\"material-icons\">{{ 'FORMAT_ALIGN_RIGHT' | nxtCustomTranslate : 'format_align_right' }}</span>\n </button>\n \n <!-- Close -->\n <button (click)=\"closeTextSettings()\" class=\"toolbar-btn\" style=\"margin-left: auto;\">\u2715</button>\n </div>\n \n </div>\n\n <!-- Input -->\n <div *ngIf=\"field?.type === 'input'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <!-- AP-04JUN25 - replace app-custom-input -->\n <nxt-input [type]=\"'text'\" [mode]=\"'edit'\" [value]=\"field.value\" [question]=\"\" [labelFont]=\"\"\n [label]=\"field?.questionText\" [labelColor]=\"\" [labelSize]=\"\" [inputValueSize]=\"\" [labelWeight]=\"\"\n [inputWeight]=\"\" [showLabel]=\"\" inputBorder=\"none\" svgHeight=\"20px\" svgWidth=\"20px\"\n [placeholder]=\"field.question ? field.question : 'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" [required]=\"\"\n inputBgColor=\"#F5F5F5\" [inputId]=\"\" [errorMessages]=\"{ required: 'This field is required' }\"\n [inputIconLeftSrc]=\"''\" (inputValue)=\"onQuestionChange($event, field)\">\n </nxt-input>\n </div>\n </div>\n </div>\n\n <!-- Date -->\n <div *ngIf=\"field?.type === 'date'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <!-- AP-04JUN25 - replace custom date picker -->\n <nxt-input [type]=\"'date'\" [mode]=\"'edit'\" [value]=\"field.value\" [question]=\"\" [labelFont]=\"\"\n [label]=\"field?.questionText\" [labelColor]=\"\" [labelSize]=\"\" [inputValueSize]=\"\" [labelWeight]=\"\"\n [inputWeight]=\"\" [showLabel]=\"\" inputBorder=\"none\" svgHeight=\"20px\" svgWidth=\"20px\"\n [placeholder]=\"field.question ? field.question : 'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" [required]=\"\"\n inputBgColor=\"#F5F5F5\" [inputId]=\"\" [errorMessages]=\"{ required: 'This field is required' }\"\n [inputIconLeftSrc]=\"''\" (inputValue)=\"onQuestionChange($event, field)\">\n </nxt-input>\n </div>\n </div>\n </div>\n\n <!-- Pdf -->\n <div *ngIf=\"field?.type === 'Pdf'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <!-- <lib-booklet [bookletJSON]=\"field.pdfReferenceQuestions\"></lib-booklet> -->\n <div *ngIf=\"field?.pdfReferenceQuestions\">\n <ng-container *ngFor=\"let field of field?.pdfReferenceQuestions[field?.pdfReference]; let i = index\">\n <!-- AP-19MAR25 Line Element -->\n <div *ngIf=\"field?.type === 'Line'\" class=\"line-field\">\n <hr class=\"custom-line\" style=\"display: inline-flex\" [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-weight': field.fontWeight || '400',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'text-align': field.textAlign || 'left',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'border-style': (field.lineStyle?.toLowerCase() || 'solid'),\n 'color': field.fontColor || '#000000',\n 'margin-top': (field.paddingTop || 0) + 'px',\n 'margin-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" />\n </div>\n\n <div *ngIf=\"field?.type === 'image'\" class=\"field-container\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div>\n <div class=\"logo-container\">\n <!-- Logo preview area -->\n <div class=\"logo-preview\" *ngIf=\"field.imageData\">\n <img [src]=\"field.imageData\" />\n </div>\n\n <!-- Upload button -->\n <div *ngIf=\"!field.imageData\" class=\"logo-upload-placeholder\">\n <label for=\"logo-upload-{{i}}\" class=\"logo-upload-label\">\n <img src=\"../assets/icons/Image.svg\" alt=\"Upload\" />\n <span>{{ 'UPLOAD_IMAGE' | nxtCustomTranslate : 'Upload Image' }}</span>\n </label>\n <input type=\"file\" id=\"logo-upload-{{i}}\" accept=\"image/*\"\n (change)=\"fileChangeEvent(field, $event)\" style=\"display: none;\" />\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div *ngIf=\"field?.type === 'Space'\" class=\"field-container\">\n <div class=\"field-wrapper\" style=\"height:93px\" [ngClass]=\"{'isPreview': isPreview}\">\n </div>\n </div>\n <div *ngIf=\"field?.type === 'text'\" class=\"field-container\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <textarea class=\"custom-textarea\" [placeholder]=\"'ENTER_DETAILED_TEXT_HERE' | nxtCustomTranslate : 'Enter detailed text here...'\"\n [style.height.px]=\"isPreview ? 40 : field.size || 100\" [(ngModel)]=\"field.value\"\n (ngModelChange)=\"onQuestionChange($event, field)\"></textarea>\n </div>\n </div>\n </div>\n <!-- AP-14JUN25 Added nxt-input in pdf -->\n <div *ngIf=\"field?.type === 'input'\" class=\"field-container\">\n <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\"> \n <nxt-input [type]=\"'text'\" [mode]=\"'edit'\" [(ngModel)]=\"field.value\"\n [placeholder]=\"'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" (ngModelChange)=\"onQuestionChange($event, field)\">\n </nxt-input>\n </div>\n </div>\n </div>\n <!-- AP-06MAR25 -->\n <div *ngIf=\"field?.type === 'Table'\" class=\"field-container\">\n <div class=\"field-wrapper\" style=\"overflow: hidden;\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div class=\"nxt-table-container\">\n <nxt-datatable isButtons [question]=\"field\" from=\"formBuilder\"\n [mode]=\"isPreview ? 'view' : 'edit'\" [apiMeta]=\"field?.subText\"\n [tableConfig]=\"field.tableConfig\" tableId=\"\" direction=\"ltr\" tableWidth=\"auto\"\n (valueChange)=\"onQuestionChange($event, field)\"\n [data]=\"field.value?.data\"\n isEditable=true (columnSelected)=columnSelected($event) (removeColumn)=removeColumn($event)>\n </nxt-datatable>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Table -->\n <!-- AP-06MAR25 -->\n <div *ngIf=\"field?.type === 'Table'\" class=\"field-container\" (click)=\"selectElement(i)\"\n [ngStyle]=\"{\n 'font-family': field.font || 'Helvetica Neue',\n 'font-size': (field.fontSize || 14) + 'px',\n 'width': (field.width || 100) + '%',\n 'align-items': field.alignItems || '',\n 'border-radius': '5px',\n 'border-width': (field.lineWidth || 1) + 'px',\n 'color': field.fontColor || '#000000',\n 'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n 'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n 'margin': \n (field.margin?.[0] || 0) + 'px ' +\n (field.margin?.[1] || 0) + 'px ' +\n (field.margin?.[2] || 0) + 'px ' +\n (field.margin?.[3] || 0) + 'px',\n 'padding-top': (field.paddingTop || 0) + 'px',\n 'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n (drop)=\"onDrop($event, i)\">\n <div class=\"field-wrapper\" style=\"overflow: hidden;\" [ngClass]=\"{'isPreview': isPreview}\">\n <div class=\"field-content\">\n <div *ngIf=\"!isPreview\" class=\"label-container\">\n <div class=\"top-right\">\n <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n </div>\n </div>\n <div class=\"nxt-table-container\">\n <nxt-datatable isButtons [question]=\"field\" from=\"formBuilder\" [mode]=\"isPreview ? 'view' : 'edit'\"\n [apiMeta]=\"field?.subText\" [tableConfig]=\"field.tableConfig\" tableId=\"\" direction=\"ltr\"\n tableWidth=\"auto\" isEditable=true (columnSelected)=columnSelected($event)\n [data]=\"field.value?.data\"\n (valueChange)=\"onQuestionChange($event, field)\"\n (removeColumn)=removeColumn($event)>\n </nxt-datatable>\n </div>\n </div>\n </div>\n </div>\n\n </ng-container>\n </div>\n </div>\n <!-- SKS13MAR25 popup conformation box -->\n <div class=\"dialog-overlay\" *ngIf=\"isSelectTablePopup\">\n <div class=\"dialog-box\">\n <button class=\"close-btn-fb\" (click)=\"onClose()\">\u2715</button>\n <p>{{ 'ADD_TABLE_CONFIRMATION' | nxtCustomTranslate : 'These element want to add a table' }}</p>\n <div class=\"button-container-fb\">\n <button class=\"yes-btn-fb\" (click)=\"addOnTable()\">{{ 'YES' | nxtCustomTranslate : 'Yes' }}</button>\n <button class=\"no-btn-fb\" (click)=\"onClose()\">{{ 'NO' | nxtCustomTranslate : 'No' }}</button>\n </div>\n </div>\n </div>\n <app-pdf-properties *ngIf=\"!isPreview\" (formButtonHandler)=\"pdfSaveHandler($event)\"></app-pdf-properties>\n</div>\n<!--SKS25MAR25 Modal Overlay -->\n<div class=\"modal-overlay\" *ngIf=\"isImageEdit\">\n <div class=\"modal-content\">\n <span class=\"close-button\" (click)=\"closeModal()\">\u00D7</span>\n <!-- Image Editor -->\n <div *ngIf=\"selectedImageElement?.type === 'image'\">\n <image-cropper *ngIf=\"selectedImageElement.imageData\" [imageBase64]=\"selectedImageElement.orgImageData\"\n [disabled]=\"false\" [alignImage]=\"alignImage\" [roundCropper]=\"roundCropper\" [backgroundColor]=\"'white'\"\n imageAltText=\"{{ 'ALTERNATIVE_IMAGE_TEXT' | nxtCustomTranslate : '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)\">\n </image-cropper>\n\n <!--AP-20JUN25 - Image size controls: Allow user to manually adjust width and height using range sliders -->\n <div *ngIf=\"selectedImageElement.imageData\" class=\"manual-size-inputs\">\n <label>{{ 'WIDTH' | nxtCustomTranslate : 'Width' }} {{ selectedImageElement.imageSize.width }}</label>\n <input type=\"range\" min=\"50\" max=\"300\" step=\"1\"\n [(ngModel)]=\"selectedImageElement.imageSize.width\"\n (input)=\"onImageSizeChange()\" />\n \n <label>{{ 'HEIGHT' | nxtCustomTranslate : 'Height' }} {{ selectedImageElement.imageSize.height }}</label>\n <input type=\"range\" min=\"50\" max=\"300\" step=\"1\"\n [(ngModel)]=\"selectedImageElement.imageSize.height\"\n (input)=\"onImageSizeChange()\" />\n </div>\n \n\n <!-- Controls -->\n <div *ngIf=\"selectedImageElement.imageData\" class=\"controls\">\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateLeft()\" title=\"{{ 'ROTATE_LEFT' | nxtCustomTranslate : 'Rotate Left' }}\">\u27F2</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"rotateRight()\" title=\"{{ 'ROTATE_RIGHT' | nxtCustomTranslate : 'Rotate Right' }}\">\u27F3</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomOut()\" title=\"{{ 'ZOOM_OUT' | nxtCustomTranslate : 'Zoom Out' }}\">-</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"zoomIn()\" title=\"{{ 'ZOOM_IN' | nxtCustomTranslate : 'Zoom In' }}\">+</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveLeft()\" title=\"{{ 'MOVE_LEFT' | nxtCustomTranslate : 'Move Left' }}\">\u2190</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveRight()\" title=\"{{ 'MOVE_RIGHT' | nxtCustomTranslate : 'Move Right' }}\">\u2192</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveUp()\" title=\"{{ 'MOVE_UP' | nxtCustomTranslate : 'Move Up' }}\">\u2191</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"moveDown()\" title=\"{{ 'MOVE_DOWN' | nxtCustomTranslate : 'Move Down' }}\">\u2193</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipHorizontal()\" [class.enabled]=\"transform.flipH\"\n title=\"{{ 'FLIP_HORIZONTALLY' | nxtCustomTranslate : 'Flip Horizontally' }}\">\u2194</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"flipVertical()\" [class.enabled]=\"transform.flipV\"\n title=\"{{ 'FLIP_VERTICALLY' | nxtCustomTranslate : 'Flip Vertically' }}\">\u2195</div>\n <div class=\"cursor-pointer logo-icon\" (click)=\"resetImage()\" title=\"{{ 'RESET' | nxtCustomTranslate : 'Reset' }}\">\u00D7</div>\n </div>\n </div>\n </div>\n</div>", styles: [".center-frame{display:flex;border-right-width:0}.head-elements{font-size:17px;font-weight:600}.form-builder{width:33.33%;height:calc(100vh - 20px);overflow-y:auto;background-color:#fff;padding:10px;border-right:10px solid #86A8CD;box-shadow:2px 2px 10px #0000001a}.form-builder .element{display:flex;align-items:center;gap:15px;margin-top:10px;padding:10px;border-radius:5px;background:#f8fafc;cursor:pointer;border-left:10px solid #E2F1FF;position:relative;color:#000}.form-builder .element:hover{background:#0250d9;color:#fff;border-left:10px solid #234465}.form-builder .element img{width:20px;height:20px;transition:filter .3s ease}.form-builder .element:hover img{filter:invert(1)}.form-builder .hover-label{font-size:15px;font-weight:400;color:#000;transition:color .3s ease}.form-builder .element:hover .hover-label{color:#fff}.form-builder .section-title{font-weight:700;font-size:16px;margin-top:10px;padding:5px;border-bottom:1px solid #ddd;color:#000;display:flex;justify-content:space-between;align-items:center}.form-builder .section-title:after{content:\"\\25bc\";font-size:12px;color:#555}.form-builder .section{margin-bottom:10px}.toggle-header{display:flex;align-items:center;justify-content:space-between;cursor:pointer;padding:10px}.arrow-icon{width:19px;height:23px;transition:transform .3s ease}.toggle-header:hover .arrow-icon{transform:scale(1.1)}.drag-dots{display:grid;grid-template-columns:repeat(2,1fr);grid-template-rows:repeat(3,1fr);gap:3px;position:absolute;right:15px;top:50%;transform:translateY(-50%)}.dot{width:5px;height:5px;border-radius:50%;background-color:#cbd5e1;transition:background-color .3s ease}.form-builder .element:hover .dot{background-color:#ffffffb3}.field-wrapper{background-color:#eff8ff;border:1px solid #E6F3FF;border-radius:5px;padding:10px}.field-wrapper.isPreview{background-color:transparent;border:none;border-radius:0;padding:0}.required:after{content:\"*\";color:red;margin-left:5px}.custom-input{width:100%;padding:8px;border:1px solid #DDDBDA;background-color:#fff;border-radius:5px;outline:none}.custom-input:focus{border-color:#00008b;box-shadow:0 0 5px #0000ff80}.delete-icon{width:19px;height:19px;margin-top:-5px}.field-container{padding-right:5px;padding-left:5px;transition:background .2s}.field-container:hover .top-right,.field-container.highlight .top-right,.label-container:hover .delete-icon{opacity:1;visibility:visible}.form-preview{width:100%;height:calc(100vh - 20px);overflow-y:auto;display:flex;flex-wrap:wrap;align-items:flex-start;padding:10px;height:fit-content;max-height:calc(100vh - 20px)}.field-content{display:flex;flex-direction:column;gap:5px}.label-container{display:flex;justify-content:flex-end;align-items:center}.top-right{display:flex;gap:10px;padding:6px;border:1px solid #d7edff;margin-top:-10px;background-color:#d7edff;margin-right:-10px;opacity:0;visibility:hidden;cursor:grab;transition:opacity .1s ease-in-out,visibility .1s ease-in-out}.drag-dot{height:13px}.custom-input,.custom-textarea,.dropdown,.checkbox-options-container,.radio-options-container{width:100%}.form-builder .element .drag-dots:active{cursor:grabbing}.dropdown{width:100%;padding:8px;border:1px solid #ccc;border-radius:4px;background-color:#fff;font-size:14px;color:#333;outline:none;cursor:pointer}.dropdown:focus{border-color:#007bff;box-shadow:0 0 5px #007bff80}.line-field{width:100%;margin:10px 0;background-color:#eff8ff;border:1px solid #E6F3FF;border-radius:5px;padding:3px}.line-field{background-color:transparent;border:none;border-radius:0;padding:3px}.checkbox-field-container:hover{box-shadow:0 2px 8px #0000001a}.checkbox-options-container{display:flex;flex-direction:column;gap:5px;padding:8px;min-height:38px;border:1px solid #DDDBDA;background-color:#fff;outline:none;border-radius:6px}.checkbox-option,.radio-option{display:flex;align-items:center;gap:8px;padding:8px;background-color:#fff;border-radius:4px;transition:background-color .2s ease}.checkbox-option:hover{background-color:#f1f3f5}.checkbox-input,.radio-input{width:18px;height:18px;accent-color:#4dabf7;cursor:pointer}.checkbox-label,.radio-label{font-size:14px;color:#495057;cursor:pointer;-webkit-user-select:none;user-select:none}.label-container label{font-size:15px;font-weight:400}.required:after{content:\" *\";color:red}.radio-options-container{display:flex;flex-direction:column;gap:5px;padding:8px;min-height:38px;border:1px solid #DDDBDA;background-color:#fff;outline:none;border-radius:6px}.radio-input:checked{border-color:#007bff;background-color:#007bff}.radio-input:checked:after{content:\"\";width:8px;height:8px;background:#fff;border-radius:50%;position:absolute;top:50%;left:50%;transform:translate(-50%,-50%)}.custom-textarea{width:100%;min-height:100px;border:1px solid #ccc;border-radius:4px;padding:8px;resize:vertical}.highlight{border:2px solid #5592FD!important;background-color:#eff8ff}.table-container label{font-size:14px;font-weight:700;margin-bottom:5px;display:block}.sticky-footer-version{position:fixed;bottom:0;padding:10px;text-align:center}.nxt-table-container{display:flex;justify-content:center;align-items:center;width:100%}nxt-datatable{width:100%!important;table-layout:fixed;max-width:100%}.dialog-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:#00000080;display:flex;justify-content:center;align-items:center;z-index:1000}.dialog-box{background:#fff;padding:20px;border-radius:5px;text-align:center;width:300px;position:relative}.close-btn-fb{position:absolute;top:4px;right:4px;background:#ff4242;color:#fff;border:none;border-radius:50%;width:20px;height:20px;font-size:10px;cursor:pointer}.button-container-fb{display:flex;justify-content:flex-end;gap:10px}.yes-btn-fb{background:green;color:#fff;border:none;padding:0 12px;border-radius:5px;cursor:pointer}.no-btn-fb{background:gray;color:#fff;border:none;padding:8px 15px;border-radius:5px;cursor:pointer}.line-element{width:100%;display:flex;justify-content:space-between;align-items:center}.custom-line{width:100%;border:1px solid #000}.logo-upload-placeholder{display:flex;flex-direction:column;align-items:center;justify-content:center;width:150px;height:150px;border:2px dashed #ccc;cursor:pointer;inline-size:auto}.logo-upload-label{display:flex;flex-direction:column;align-items:center;cursor:pointer;color:#666}.logo-upload-label img{width:32px;height:32px;margin-bottom:8px}.element.disabled{opacity:.5;cursor:not-allowed}.form-container{display:flex;flex-wrap:wrap;border:10px solid #86A8CD}.form-container.isPreview{border:none}.center-frame{width:74%}.center-frame.isPreview{width:100%}app-pdf-properties{border-left:10px solid #86A8CD;width:25%}:host ::ng-deep .questiondiv1{padding-left:0!important;padding-right:0!important}:host ::ng-deep .form-group.content-box{padding-bottom:0!important}:host ::ng-deep .custom-textarea{min-height:40px!important}.modal-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:100}.modal-content{background:#fff;padding:20px;width:500px;max-width:90%;border-radius:8px;position:relative;text-align:center}.close-button{position:absolute;top:6px;right:6px;font-size:25px;cursor:pointer;z-index:10}.controls{display:flex;gap:5px;justify-content:center;margin-top:10px}.logo-icon{cursor:pointer;padding:5px;border-radius:5px;background:#d3d3d3}.logo-icon:hover{background:#a9a9a9}.enabled{background:#007bff;color:#fff}.text-settings-toolbar{position:absolute;display:flex;flex-wrap:wrap;align-items:center;gap:8px;background:#fff;border:1px solid #d1d5db;padding:8px 10px;border-radius:8px;box-shadow:0 4px 12px #0000001a;z-index:65535;max-width:100%;width:auto;min-width:200px;box-sizing:border-box}.toolbar-select{padding:4px 6px;font-size:13px;border:1px solid #d1d5db;border-radius:6px;background:#fff;cursor:pointer;min-width:60px;width:auto}.toolbar-btn{background:#f3f4f6;border:1px solid #d1d5db;padding:6px 8px;border-radius:6px;cursor:pointer;font-size:13px;min-width:30px;height:30px;display:flex;align-items:center;justify-content:center}.toolbar-btn:hover{background:#e5e7eb;border-color:#9ca3af}.toolbar-btn.active{background:#005aaa;color:#fff;border-color:#005aaa}.color-label{display:flex;align-items:center}.color-box{font-weight:700;width:22px;height:22px;font-size:13px;border:1px solid #ccc;border-radius:4px;text-align:center;line-height:22px;cursor:pointer;padding:0}.color-picker-hidden{position:absolute;opacity:0;pointer-events:none;width:0;height:0}@media (max-width: 600px){.text-settings-toolbar{flex-direction:row;flex-wrap:wrap;gap:6px;padding:6px;justify-content:flex-start}.toolbar-btn,.toolbar-select{font-size:12px;padding:5px}.color-box{width:20px;height:20px;font-size:12px;line-height:20px}}.action-buttons{display:flex;align-items:center;gap:10px;flex-wrap:wrap}.action-btn{display:flex;gap:16px;padding:10px 15px;border-radius:8px;border:none}\n"] }]
|
|
623
|
-
}], ctorParameters: () => [{ type: i1.PdfDesignerService }], propDecorators: { textareas: [{
|
|
624
|
-
type: ViewChildren,
|
|
625
|
-
args: ['autoTextarea']
|
|
626
|
-
}], pdfJSON: [{
|
|
627
|
-
type: Input
|
|
628
|
-
}], bookletId: [{
|
|
629
|
-
type: Input
|
|
630
|
-
}], isPreview: [{
|
|
631
|
-
type: Input
|
|
632
|
-
}], pdfSaveHandlerEmit: [{
|
|
633
|
-
type: Output
|
|
634
|
-
}], templateMode: [{
|
|
635
|
-
type: Output
|
|
636
|
-
}], pdfPreviewEmit: [{
|
|
637
|
-
type: Output
|
|
638
|
-
}] } });
|
|
639
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pdf-designer.component.js","sourceRoot":"","sources":["../../../../../../../projects/nxt-app/src/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.ts","../../../../../../../projects/nxt-app/src/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, Input, Output, QueryList, SimpleChanges, ViewChildren } from '@angular/core';\nimport { VERSION } from '../../../../environments/version';\nimport { PdfPropertiesComponent } from '../pdf-properties/pdf-properties.component';\nimport { CommonModule } from '@angular/common';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { FormsModule } from '@angular/forms';\nimport { NxtDatatable } from '../../../components/datatable/datatable.component';\nimport { BookletComponent } from '../../booklet/booklet.component';\nimport { AppTemplatesComponent } from '../../builder/templates/templates.component';\nimport { NxtInput } from '../../../components/nxt-input/nxt-input.component';\nimport { ImageTransform } from '../../../components/image-cropper/interfaces/image-transform.interface';\nimport { Dimensions } from '../../../components/image-cropper/interfaces/dimensions.interface';\nimport { CropperPosition } from '../../../components/image-cropper/interfaces/cropper-position.interface';\nimport { ImageCropperComponent } from '../../../components/image-cropper/component/image-cropper.component';\nimport _ from 'lodash'; // Ensure lodash is installed\nimport { PdfDesignerService } from '../../../services/pdf-designer/pdf-designer.service';\nimport { NxtCustomTranslatePipe } from '../../../pipe/custom-translate.pipe';\n\n@Component({\n  selector: 'app-pdf-designer',\n  standalone: true,\n  imports: [CommonModule, ImageCropperComponent, PdfPropertiesComponent, MatTooltipModule, FormsModule, NxtDatatable, BookletComponent, AppTemplatesComponent, NxtInput, NxtCustomTranslatePipe],\n  templateUrl: './pdf-designer.component.html',\n  styleUrls: ['./pdf-designer.component.css']\n})\nexport class PdfDesignerComponent {\n  //@Output() elementButtonClicked = new EventEmitter<string>();\n  @ViewChildren('autoTextarea') textareas!: QueryList<ElementRef>;\n  @Input() pdfJSON: any;\n  @Input() bookletId: any;\n  @Input() isPreview: boolean = false;\n\n  @Output() pdfSaveHandlerEmit = new EventEmitter<any>();\n  @Output() templateMode = new EventEmitter<boolean>();\n  @Output() pdfPreviewEmit = new EventEmitter<any>();\n  field: any;\n  pdfElements: any[] = [];\n  elements: any[] = [];\n  bookId: string;\n  sections = {\n    basic: true,\n    advanced: true\n  };\n  elementsList: any[] = [];\n  elementDisabledArray: any;\n  version = VERSION.version;\n  pdf: any;\n  selectedElement: any = null;\n  isSelectTablePopup: boolean = false;\n  currentType: string;\n  addTable: boolean = true;\n  dots = Array(6);\n  draggedIndex: number | null = null;\n  templateSelected = false;\n  isImageEdit: boolean = false;\n  selectedImageElement: 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  loading = false;\n  canvasRotation = 0;\n  cropper?: CropperPosition;\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  isImageHover = false;\n  alignImage = 'center' as const;\n  showTextSettings = false;\n  selectedColumn: any = null;\n  \n\n  //subscription: any;\n  constructor(private pdfDesignerService: PdfDesignerService) {\n  }\n  //AP-14JUN25 - Called after view initialization to auto-resize all textareas\n  ngAfterViewInit() {\n    this.resizeAllTextareas();\n  }\n\n  //AP-14JUN25 - Calculates contrast text color (black or white) based on background color\n  getContrastColor(bgColor: string = '#000000'): string {\n    // Convert hex to RGB\n    const color = bgColor.replace('#', '');\n    const r = parseInt(color.substring(0, 2), 16);\n    const g = parseInt(color.substring(2, 4), 16);\n    const b = parseInt(color.substring(4, 6), 16);\n    const brightness = (r * 299 + g * 587 + b * 114) / 1000;\n    return brightness > 128 ? '#000000' : '#ffffff';\n  }\n\n  //AP-14JUN25 - Automatically resizes all textareas based on their content\n  resizeAllTextareas(): void {\n    this.textareas.forEach((textareaRef: ElementRef) => {\n      const textarea = textareaRef.nativeElement as HTMLTextAreaElement;\n      textarea.style.height = 'auto';\n      textarea.style.height = textarea.scrollHeight + 'px';\n    });\n  }\n  \n  ngOnInit() {\n    this.elementsList = [\n      { \"type\": \"text\", \"img\": \"TextArea\", \"label\": \"Text\" },\n      { \"type\": \"input\", \"img\": \"Text\", \"label\": \"Input\" },\n      { \"type\": \"date\", \"img\": \"Date\", \"label\": \"Date\" },\n      { \"type\": \"image\", \"img\": \"Image\", \"label\": \"Image\" },\n      { \"type\": \"Line\", \"img\": \"line\", \"label\": \"Line\" },\n      { \"type\": \"Space\", \"img\": \"space\", \"label\": \"Space\" },\n      { \"type\": \"Table\", \"img\": \"Table\", \"label\": \"Table\" },\n      { \"type\": \"Pdf\", \"img\": \"Search\", \"label\": \"Book\" },\n    ];\n    this.elementDisabledArray = {\n      Table: [\n        'Pdf', 'Calendar', \"Boolean\", 'List', 'Table', 'Checkbox', 'Radio',\n        'Dropdown', 'RichTextArea', 'Number', 'Label', 'image',\n        'Email', 'Date', 'Time', 'DateTime', 'Line', 'Space'\n      ]\n    };\n    if (!this.isPreview) {\n      this.pdfDesignerService.pdfElements$.subscribe(elements => {\n        setTimeout(() => {\n          this.pdfElements = elements.map(field => ({ ...field }));\n        }, 0);\n      });\n    }\n    if (this.pdfJSON && (this.pdfJSON !== '')) {\n      this.initializeForm();\n    } else {\n      this.pdfDesignerService.newBook()\n    }\n    // AP-12MAR25 - Added to handle bookletId\n    if (this.bookletId) {\n      localStorage.setItem('unique_id', this.bookletId);\n\n    }\n    this.pdf = this.pdfDesignerService.getBook();\n    if (!this.isPreview) {\n      this.pdfDesignerService.selectedElement$.subscribe(index => {\n        const elements = this.pdfDesignerService.getElements();\n        if (index >= 0) {\n          this.selectedElement = elements[index];\n        }\n        // this.pdfElements = [...elements]\n      });\n    }\n    this.pdfDesignerService.addElementWithId();\n  }\n\n  isSectionOpen(section: string): boolean {\n    return this.sections[section];\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['pdfJSON'] && changes['pdfJSON'].currentValue) {\n      if (this.pdfJSON && (this.pdfJSON !== '')) {\n        this.initializeForm();\n      } else {\n        this.pdfDesignerService.newBook()\n      }\n    }\n  }\n\n  onTemplateSelected(event: { selectedTemplate: any; elements: any[] }) {\n    this.pdfDesignerService.clearElements();\n    event.elements.forEach(el => this.pdfDesignerService.addElement(el));\n\n    // Emit template mode to parent\n    this.templateMode.emit(true);\n  }\n\n  private initializeForm() {\n    if (!this.pdfJSON || !this.pdfJSON) {\n      console.warn(\"Invalid pdfJSON structure\");\n      return;\n    }\n    this.pdfElements = []\n    this.pdfDesignerService.clearElements();\n    this.pdfDesignerService.intializeBook(this.pdfJSON);\n    this.pdfElements = this.pdfDesignerService.getElements();\n  }\n\n  selectElement(index: number) {\n    this.selectedFieldIndex = index;\n    this.pdfDesignerService.setSelectedElement(index);\n    this.pdf = this.pdfDesignerService.getBook();\n  }\n\n  selectedFieldIndex: number | null = null;\n  selectHeading(event: any) {\n    this.pdfDesignerService.selectHeading(event);\n    this.pdf = this.pdfDesignerService.getBook();\n  }\n  // AP-17APR25 generateUiId \n  generateUiId() {\n    return this.pdfDesignerService.addElementWithId();\n  }\n\n  addElement(type: string): void {\n    const unique_id = this.generateUiId();\n    this.currentType = type\n    if (this.selectedElement?.type === 'Table' && this.addTable) {\n      this.isSelectTablePopup = true\n    } else {\n      const newElement = {\n        id: unique_id,\n        type,\n        margin: [0, 0, 0, 0],\n        fontSize: 14,\n        fontWeight: '400',\n        fontStyle: [],\n        alignItems: '',\n        nxtType: type,\n        referenceField: null,\n        question: null,\n        questionNumber: this.pdfDesignerService.getElements().length + 1,\n        fieldsMeta: [],\n        pdfReference: null,\n        pdfReferenceQuestions: null,\n        style: {\n          bold: false,\n          italic: false,\n          alignment: 'left',\n          fontSize: 14,  \n          margin: [0, 0, 0, 0],\n          color: '#000000',  \n        },\n        imageData: null,\n        imageSize: type === 'image' ? { width: 100, height: 100 } : null,\n        tableConfig: type === 'Table' ? {\n          isNosIndicator: false,\n          addInlineRecord: true,\n          isPagination: false,\n          actionButton: false,\n          isDeleteRow: true,\n          isEditRow: false,\n          searchBar: false,\n          isButtons: false,\n        } : null,\n\n        width: 100,\n        styleClass: unique_id,\n      };\n      this.pdfDesignerService.addElement(newElement);\n      this.pdfElements = this.pdfDesignerService.getElements();\n      this.addTable = true\n    }\n  }\n\n  //AP-14JUN25 - Handles textarea input: auto-resizes and updates the field content\n  onTextAreaInput(event: Event, field: any): void {\n    const textarea = event.target as HTMLTextAreaElement;\n  \n    // 1. Auto-resize logic\n    textarea.style.height = 'auto';\n    textarea.style.height = textarea.scrollHeight + 'px';\n  \n    // 2. Trigger value update and PDF sync\n    this.onQuestionChange(textarea.value, field);\n  }\n\nonQuestionChange(value: string, field: any) {\n  let elementId: any | undefined;\n\n  // SKS19JUN25 Clone to avoid modifying the original\n  const pdfElementsClone = _.cloneDeep(this.pdfElements);\n\n  const updatePdfQuestion = (element: any, value: string): boolean => {\n    if (element.type === 'Pdf') {\n      const subElements = element.pdfReferenceQuestions?.[element.pdfReference] || [];\n      for (const subElement of subElements) {\n        const found = updatePdfQuestion(subElement, value); // recursive call\n        if (found) return true; // stop once match is found\n      }\n    } else {\n      if (element.id === field.id) {\n        element.value = value;\n        return true;\n      }\n    }\n    return false;\n  };\n\n  pdfElementsClone.forEach((element: any, index: number) => {\n    const found = updatePdfQuestion(element, value);\n    if (found && elementId === undefined) {\n      elementId = index;\n    }\n  });\n\n  if (elementId !== undefined) {\n    const updatedElement = pdfElementsClone[elementId];\n    this.pdfDesignerService.elementUpdate(updatedElement, 'valueChange');\n    // SKS19JUN25 Optionally preview or test using updatedElement (without updating original)\n    if (this.isPreview) {\n      this.pdfPreview(this.pdfDesignerService.downloadElement());\n    }\n  }\n}\n  \npdfPreview(pdf:any){\n  this.pdfPreviewEmit.emit(pdf)\n}\n  onFieldDateChange(event: any) {\n  }\n\n  removeElement(field: any, index: number): void {\n    this.pdfDesignerService.removeElementComponent(field.id);\n    this.pdfElements = this.pdfDesignerService.getElements();\n  }\n\n  onDragStart(event: DragEvent, index: number) {\n    this.draggedIndex = index;\n    event.dataTransfer?.setData('text/plain', index.toString());\n  }\n\n  onDragOver(event: DragEvent, index: number) {\n    event.preventDefault();\n  }\n\n  onDrop(event: DragEvent, dropIndex: number) {\n    event.preventDefault();\n\n    if (this.draggedIndex === null || this.draggedIndex === dropIndex) return;\n\n    const draggedItem = this.pdfElements[this.draggedIndex];\n\n    // Remove dragged item from old position and insert it in new position\n    this.pdfElements.splice(this.draggedIndex, 1);\n    this.pdfElements.splice(dropIndex, 0, draggedItem);\n\n    //AP-28MAR25 Update questionNumber dynamically based on new order\n    this.pdfElements.forEach((element, index) => {\n      element.questionNumber = index + 1;\n    });\n\n    // Reset dragged index\n    this.draggedIndex = null;\n\n    // Notify service about the update\n    this.pdfDesignerService.updateElementsOrder(this.pdfElements);\n  }\n\n  toggleBoolean(field: any): void {\n    field.boolean = !field.boolean;\n  }\n  // SKS13MAR25 table popup conformation based element add\n  onClose() {\n    this.isSelectTablePopup = false\n    this.addTable = false\n    this.addElement(this.currentType)\n  }\n  // SKS13MAR25 column element add inside a table\n  addOnTable() {\n    this.addTable = true\n    this.isSelectTablePopup = false\n    const elements = [...this.pdfDesignerService.getElements()];\n\n    // Find the element and update its columns\n    const index = elements.findIndex(el => el.id === this.selectedElement.id);\n    const rowNum = this.pdfElements[index].columns ? this.pdfElements[index].columns?.length : 0\n    const unique_id = this.generateUiId();\n    const tableElement = {\n      label: `HEADER LABEL ${rowNum}`,\n      fieldName: `FIELD NAME${rowNum}`,\n      type: this.currentType,\n      id: unique_id\n    }\n    this.pdfDesignerService.addTableElement(tableElement, this.selectedFieldIndex);\n    this.pdfElements = this.pdfDesignerService.getElements();\n    // this.addTableData(unique_id, fieldName)\n  }\n  addTableData(fieldId, fieldName) {\n    // this.pdfDesignerService.addTablefieldData(tableElement);\n  }\n  columnSelected(event: any) {\n    // SKS19MAR25 table column update\n    if (!this.isPreview) {\n      this.pdfDesignerService.setSelectedTableElement(this.selectedFieldIndex, event);\n    }\n  }\n  removeColumn(event: any) {\n    this.pdfDesignerService.removeSelectedTableElement(this.selectedFieldIndex, event);\n  }\n\n  // SKS25MAR25 image add\n  async fileChangeEvent(field: any, event: any): Promise<void> {\n    const file = event.target.files[0];\n    if (!file) return;\n  \n    try {\n      const imageData = await this.readFileAsDataURL(file);\n      let elementId: number | undefined;\n  \n      const updatePdfImage = (element: any): boolean => {\n        if (element.type === 'Pdf') {\n          const subElements = element.pdfReferenceQuestions?.[element.pdfReference] || [];\n          for (const subElement of subElements) {\n            const found = updatePdfImage(subElement); // recursive call\n            if (found) return true;\n          }\n        } else {\n          if (element.id === field.id) {\n            element.imageData = imageData;\n            element.orgImageData = imageData;\n            if (!element.imageSize) {\n              element.imageSize = { width: 100, height: 100 };\n            }\n            return true;\n          }\n        }\n        return false;\n      };\n  \n      this.pdfElements.forEach((element: any, index: number) => {\n        const found = updatePdfImage(element);\n        if (found && elementId === undefined) {\n          elementId = index; // remember top-level match\n        }\n      });\n  \n      if (elementId !== undefined) {\n        this.pdfDesignerService.elementUpdate(this.pdfElements[elementId], 'valueChange');\n        if(this.isPreview){\n          this.pdfPreview(this.pdfDesignerService.downloadElement())\n        }\n      }\n    } catch (error) {\n      console.error(\"Error reading file:\", error);\n    }\n  }  \n  \n  private readFileAsDataURL(file: File): Promise<string> {\n    return new Promise((resolve, reject) => {\n      const reader = new FileReader();\n      reader.onload = () => resolve(reader.result as string);\n      reader.onerror = (error) => reject(error);\n      reader.readAsDataURL(file);\n    });\n  }\n  // SKS2APR25 disabled element\n  isElementDisabled(elementType: string): boolean {\n    if (this.selectedElement) {\n      const elements = this.elementDisabledArray[this.selectedElement.type]\n      if (elements) {\n        return elements.includes(elementType);\n      }\n    }\n    return false;\n  }\n  async onImageEdit(ques: any) {\n    this.isImageEdit = true;\n    this.selectedImageElement = ques;\n\n  //Assign imageSize default value\n  if (!this.selectedImageElement.imageSize) {\n    this.selectedImageElement.imageSize = { width: 100, height: 100 };\n  }\n  }\n  async onImageDelete(ques: any) {\n    try {\n      ques.imageData = '';\n      ques.orgImageData = '';\n      // Initialize logo size if not already set\n      if (!ques.imageSize) {\n        ques.imageSize = { width: 100, height: 100 };\n      }\n      // await this.childEventCapture(ques.imageData, ques);\n    } catch (error) {\n      console.error(\"Error reading file:\", error);\n    }\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//AP-20JUN25 - Updates the image size (width and height) of the selected image element and reflects the change via the service\nonImageSizeChange() {\n  const index = this.pdfElements.findIndex((el: any) => el.id === this.selectedImageElement.id);\n  if (index !== -1) {\n    this.pdfElements[index].imageSize = { \n      width: this.selectedImageElement.imageSize.width, \n      height: this.selectedImageElement.imageSize.height \n    };\n    // Update element in service\n    this.pdfDesignerService.elementUpdate(this.pdfElements[index], 'valueChange');\n  }\n}\n  async imageCropped(event: any) {\n    try {\n      const base64 = await this.convertBlobToBase64(event.objectUrl);\n      this.selectedImageElement.imageData = base64;\n      // await this.childEventCapture(base64, this.selectedImageElement);\n       // set default if not present\n    if (!this.selectedImageElement.imageSize) {\n      this.selectedImageElement.imageSize = { width: 100, height: 100 };\n    }\n    } catch (error) {\n      console.error(\"Error in imageCropped:\", error);\n    }\n  }\n\n\n  cropperReady(sourceImageDimensions: Dimensions) {\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  closeModal() {\n    this.isImageEdit = false;\n  }\n     openTextSettings(field: any, event: MouseEvent) {\n      this.selectedColumn = field;\n      this.showTextSettings = true;\n    }\n  \n    closeTextSettings() {\n      this.showTextSettings = false;\n    }\n   \n    //AP-14JUN25 - Toggles bold or italic style for the selected text block\n    toggleStyle(style: 'bold' | 'italic') {\n      if (!this.selectedColumn) return;\n      if (style === 'bold') {\n        this.selectedColumn.style.bold = !this.selectedColumn.style.bold;\n      } else if (style === 'italic') {\n        this.selectedColumn.style.italics = !this.selectedColumn.style.italics;\n      }\n      this.updateContent();\n    }\n  \n    //AP-14JUN25 - Sets text alignment (left, center, right) for the selected text block\n    setAlignment(alignment: 'left' | 'center' | 'right') {\n      if (!this.selectedColumn) return;\n      this.selectedColumn.style.alignment = alignment;\n      this.updateContent();\n    }\n\n    //AP-14JUN25 - Updates the content in the PDF designer with the selected style changes\n    updateContent() {\n      this.pdfDesignerService.elementUpdate(this.selectedColumn, 'valueChange');\n    }\n    pdfSaveHandler(event:any){\n      if(event === 'preview'){\n        this.pdfSaveHandlerEmit.emit({\"action\":'preview',\"pdf\":this.pdfDesignerService.downloadElement()})\n      } else if(event === 'download'){\n        this.pdfSaveHandlerEmit.emit({\"action\":'download',\"pdf\":this.pdfDesignerService.downloadElement()})\n      } else {\n        this.pdfSaveHandlerEmit.emit(event)\n      }\n    }\n}","<!-- AP 22JAN25 - form preview and All form elements -->\n<!-- AP 25FEB25 - All elements update -->\n<div class=\"form-container\" [ngClass]=\"{'isPreview': isPreview}\">\n  <div class=\"center-frame\" [ngClass]=\"{'isPreview': isPreview}\">\n    <!-- Form Builder Section All Elements -->\n    <div *ngIf=\"!isPreview\" class=\"form-builder\">\n\n      <ng-container *ngFor=\"let element of elementsList\">\n        <div\n            class=\"element\"\n            [class.disabled]=\"selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)\"\n            (click)=\"!(selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)) && addElement(element.type)\"\n            [draggable]=\"!(selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type))\"\n            [matTooltip]=\"\n              selectedElement?.type && elementDisabledArray[selectedElement.type]?.includes(element.type)\n                ? ('THIS_ELEMENT_NOT_SUPPORTED' | nxtCustomTranslate : 'This element is not supported when a ') +\n                  (selectedElement?.type || ('CERTAIN_ELEMENT' | nxtCustomTranslate : 'certain element')) +\n                  (' IS_PRESENT' | nxtCustomTranslate : ' is present')\n                : null\n            \"\n          >\n          <img src=\"../assets/icons/{{ element.img }}.svg\" class=\"element-icon\">\n          <div class=\"hover-label\">{{ element.label }}</div>      \n          <div class=\"drag-dots\">\n            <div class=\"dot\" *ngFor=\"let dot of dots\"></div>\n          </div>\n        </div>\n      </ng-container>\n<!-- \n      <div class=\"template-section\">\n        <app-templates (templateSelected)=\"onTemplateSelected($event)\"></app-templates>\n      </div> -->\n\n      <!-- SKS10MAR25 footer version show -->\n      <div class=\"sticky-footer-version\">\n        {{version}}\n      </div>\n    </div>\n    <!-- AP-27MAR25 Remove CDK drag and drop replace draggable function  -->\n    <div class=\"form-preview\">\n      <!-- AP-10MAR25 Heading -->\n      <div class=\"field-container\"\n        style=\"width: 100%;background-color: #EFF8FF; border: 1px solid #E6F3FF;margin-bottom:10px\"\n        (click)=\"selectHeading('Header')\">\n        <div class=\"label-container\" style=\"padding: 10px; display: flex;justify-content: space-between;\">\n          <div *ngIf=\"pdf\">\n            <div *ngIf=\"pdf.title == ''\" style=\"color:#3f4a525c\">{{ 'HEADING' | nxtCustomTranslate : 'Heading' }}</div>\n            <div *ngIf=\"pdf.title !== ''\">{{pdf.title}}</div>\n          </div>\n          <div class=\"action-buttons\">\n            <button class=\"action-btn\" (click)=\"pdfSaveHandler('preview')\">{{ 'PREVIEW' | nxtCustomTranslate : 'PREVIEW' }}</button>\n            <button class=\"action-btn\" (click)=\"pdfSaveHandler('download')\">{{ 'DOWNLOAD' | nxtCustomTranslate : 'DOWNLOAD' }}</button>\n          </div>\n        </div>\n      </div>\n\n      <ng-container *ngFor=\"let field of pdfElements; let i = index\">\n\n        <!-- AP-19MAR25 Line Element -->\n        <div *ngIf=\"field?.type === 'Line'\" class=\"line-field\" [ngClass]=\"{'isPreview': isPreview}\"\n          (click)=\"selectElement(i)\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n          draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n          (drop)=\"onDrop($event, i)\">\n          <div *ngIf=\"!isPreview\" class=\"line-element\">\n            <div></div>\n            <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n          </div>\n          <hr class=\"custom-line\" style=\"display: inline-flex\" [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-weight': field.fontWeight || '400',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'text-align': field.textAlign || 'left',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'border-style': (field.lineStyle?.toLowerCase() || 'solid'),\n    'color': field.fontColor || '#000000',\n    'margin-top': (field.paddingTop || 0) + 'px',\n    'margin-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" />\n        </div>\n\n        <!--SKS25MAR25 Image Upload Element -->\n        <div *ngIf=\"field?.type === 'image'\" class=\"field-container\" (click)=\"selectElement(i)\"\n          [class.highlight]=\"selectedFieldIndex === i && !isPreview\" draggable=\"true && !isPreview\"\n          (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\" (drop)=\"onDrop($event, i)\"\n          [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'align-items': field.alignItems || '',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'color': field.fontColor || '#000000',\n    'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n    'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n    'margin': \n      (field.margin?.[0] || 0) + 'px ' +\n      (field.margin?.[1] || 0) + 'px ' +\n      (field.margin?.[2] || 0) + 'px ' +\n      (field.margin?.[3] || 0) + 'px',\n    'padding-top': (field.paddingTop || 0) + 'px',\n    'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" (mouseenter)=\"isImageHover = true;\"\n          (mouseleave)=\"isImageHover = false;\">\n          <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n            <div class=\"field-content\">\n              <div *ngIf=\"!isPreview\" class=\"label-container\">\n                <div class=\"top-right\">\n                  <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n                  <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n                </div>\n              </div>\n              <div>\n                  <div *ngIf=\"isImageHover\" style=\"display: flex; justify-content: end;\">\n                    <svg *ngIf=\"isPreview\" (click)=\"onImageEdit(field)\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                      fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n                      <path\n                        d=\"M11.1067 6.07174L9.92833 4.8934L2.16667 12.6551V13.8334H3.345L11.1067 6.07174ZM12.285 4.8934L13.4633 3.71507L12.285 2.53674L11.1067 3.71507L12.285 4.8934ZM4.035 15.5001H0.5V11.9642L11.6958 0.768403C11.8521 0.612177 12.064 0.524414 12.285 0.524414C12.506 0.524414 12.7179 0.612177 12.8742 0.768403L15.2317 3.1259C15.3879 3.28218 15.4757 3.4941 15.4757 3.71507C15.4757 3.93604 15.3879 4.14796 15.2317 4.30424L4.03583 15.5001H4.035Z\"\n                        fill=\"#6C757D\" />\n                    </svg>\n                    <svg *ngIf=\"isPreview\" (click)=\"onImageDelete(field)\" width=\"16\" height=\"16\"\n                      viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n                      <path\n                        d=\"M14 3.98726C11.78 3.76726 9.54667 3.65393 7.32 3.65393C6 3.65393 4.68 3.7206 3.36 3.85393L2 3.98726\"\n                        stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n                      <path\n                        d=\"M5.6665 3.31362L5.81317 2.44028C5.91984 1.80695 5.99984 1.33362 7.1265 1.33362H8.87317C9.99984 1.33362 10.0865 1.83362 10.1865 2.44695L10.3332 3.31362\"\n                        stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n                      <path\n                        d=\"M12.5667 6.09375L12.1334 12.8071C12.06 13.8537 12 14.6671 10.14 14.6671H5.86002C4.00002 14.6671 3.94002 13.8537 3.86668 12.8071L3.43335 6.09375\"\n                        stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n                      <path d=\"M6.88647 11.0004H9.10647\" stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\"\n                        stroke-linejoin=\"round\" />\n                      <path d=\"M6.3335 8.33325H9.66683\" stroke=\"#FF2C10\" stroke-width=\"1.5\" stroke-linecap=\"round\"\n                        stroke-linejoin=\"round\" />\n                    </svg>\n                  </div>\n                  <div class=\"logo-container\">\n                    <div class=\"logo-preview\" *ngIf=\"field.imageData\">\n                      <img [src]=\"field.imageData\" \n                      [style.width.px]=\"field.imageSize?.width || 100\"\n                      [style.height.px]=\"field.imageSize?.height || 100\" />\n                    </div>\n\n                    <div *ngIf=\"!field.imageData\" class=\"logo-upload-placeholder\">\n                      <label for=\"logo-upload-{{i}}\" class=\"logo-upload-label\">\n                        <img src=\"../assets/icons/Image.svg\" alt=\"Upload\" />\n                        <span>{{ 'UPLOAD_IMAGE' | nxtCustomTranslate : 'Upload Image' }}</span>\n                      </label>\n                      <input type=\"file\" id=\"logo-upload-{{i}}\" accept=\"image/*\" (change)=\"fileChangeEvent(field, $event)\"\n                        style=\"display: none;\" />\n                    </div>\n                  </div>\n              </div>\n            </div>\n          </div>\n        </div>\n\n        <!-- Space Element -->\n        <div *ngIf=\"field?.type === 'Space'\" class=\"field-container\" (click)=\"selectElement(i)\"\n          [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'align-items': field.alignItems || '',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'color': field.fontColor || '#000000',\n    'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n    'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n    'margin': \n      (field.margin?.[0] || 0) + 'px ' +\n      (field.margin?.[1] || 0) + 'px ' +\n      (field.margin?.[2] || 0) + 'px ' +\n      (field.margin?.[3] || 0) + 'px',\n    'padding-top': (field.paddingTop || 0) + 'px',\n    'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n          draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n          (drop)=\"onDrop($event, i)\">\n          <div class=\"field-wrapper\" style=\"height:93px\" [ngClass]=\"{'isPreview': isPreview}\">\n            <div *ngIf=\"!isPreview\" class=\"field-content\">\n              <div class=\"label-container\">\n                <div class=\"top-right\" style=\"margin: -11px -11px 0 0;\">\n                  <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n                  <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n                </div>\n              </div>\n            </div>\n          </div>\n        </div>\n\n        <!-- TextArea -->\n        <div *ngIf=\"field?.type === 'text'\" class=\"field-container\" (click)=\"selectElement(i)\"\n          [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'align-items': field.alignItems || '',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'color': field.fontColor || '#000000',\n    'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n    'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n    'margin': \n      (field.margin?.[0] || 0) + 'px ' +\n      (field.margin?.[1] || 0) + 'px ' +\n      (field.margin?.[2] || 0) + 'px ' +\n      (field.margin?.[3] || 0) + 'px',\n    'padding-top': (field.paddingTop || 0) + 'px',\n    'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n          draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n          (drop)=\"onDrop($event, i)\">\n          <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n            <div class=\"field-content\">\n              <div *ngIf=\"!isPreview\" class=\"label-container\">\n                <div class=\"top-right\">\n                  <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n\n                  <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n                </div>\n              </div>\n              <textarea class=\"custom-textarea\"\n              [placeholder]=\"'ENTER_DETAILED_TEXT_HERE' | nxtCustomTranslate : 'Enter detailed text here...'\"\n              [value]=\"field.value? field.value : ''\"\n              (input)=\"onTextAreaInput($event, field)\"\n              #autoTextarea\n              [style.height.px]=\"isPreview ? 40 : null\"\n              (click)=\"openTextSettings(field, $event)\"\n              (ngModelChange)=\"updateContent()\" \n              [ngStyle]=\"{\n                'font-size': field.style?.fontSize ? field.style.fontSize + 'px' : '14px',\n                'font-weight': field.style?.bold ? 'bold' : 'normal',\n                'font-style': field.style?.italics ? 'italic' : 'normal',\n                'text-decoration': field.style?.decoration ? 'decoration' : 'none',\n                'text-align': field.style?.alignment || 'left',\n                'margin': field.style?.margin?.length === 4\n                  ? field.style.margin[1] + 'px ' + field.style.margin[2] + 'px ' +\n                    field.style.margin[3] + 'px ' + field.style.margin[0] + 'px'\n                  : '0',\n                'color': field.style?.color || '#000'\n              }\"></textarea>\n            \n            </div>\n          </div>\n\n                           <!--AP-05JUN25 Text Settings Toolbar: Allows editing font size, bold/italic styles, alignment, and closing the toolbar for the selected column -->\n                           <div *ngIf=\"showTextSettings && isPreview && selectedColumn === field\" class=\"text-settings-toolbar\"\n                           [ngStyle]=\"{\n                             position: 'absolute',\n                             background: '#fff',\n                             border: '1px solid #ccc',\n                             padding: '6px 10px',\n                             'border-radius': '4px',\n                             'box-shadow': '0 2px 6px rgba(0, 0, 0, 0.1)',\n                             'z-index': '65535',\n                             display: 'flex',\n                             'align-items': 'center',\n                             gap: '8px'\n                           }\">\n                         \n                           <!-- Font Size -->\n                           <select [(ngModel)]=\"selectedColumn.style.fontSize\" class=\"toolbar-select\">\n                             <option [value]=\"12\">12pt</option>\n                             <option [value]=\"14\">14pt</option>\n                             <option [value]=\"16\">16pt</option>\n                             <option [value]=\"18\">18pt</option>\n                             <option [value]=\"24\">24pt</option>\n                           </select>\n\n                          <!-- Text Color -->\n                          <label title=\"Text Color\" class=\"color-label\">\n                          <span class=\"color-box\" [style.color]=\"selectedColumn.style.color || '#000'\" (click)=\"textColorInput.click()\">A</span>\n                          <input #textColorInput type=\"color\" [(ngModel)]=\"selectedColumn.style.color\" (change)=\"updateContent()\" class=\"color-picker-hidden\"/>\n                          </label>\n\n                           <!-- Bold -->\n                           <button (click)=\"toggleStyle('bold')\" [class.active]=\"selectedColumn?.style?.bold\" class=\"toolbar-btn\">{{ 'BOLD' | nxtCustomTranslate : 'B' }}</button>\n                         \n                           <!-- Italic -->\n                           <button (click)=\"toggleStyle('italic')\" [class.active]=\"selectedColumn?.style?.italics\" class=\"toolbar-btn\"><i>{{ 'ITALIC' | nxtCustomTranslate : 'I' }}</i></button>\n\n                           <!-- Underline -->\n                           <!-- <button (click)=\"toggleStyle('underline')\" [class.active]=\"selectedColumn?.style?.underline\" class=\"toolbar-btn\"><u>U</u></button> -->\n\n                         \n                           <!-- Align Left -->\n                           <button (click)=\"setAlignment('left')\" [class.active]=\"selectedColumn?.style?.alignment === 'left'\" class=\"toolbar-btn\">\n                             <span class=\"material-icons\">{{ 'FORMAT_ALIGN_LEFT' | nxtCustomTranslate : 'format_align_left' }}</span>\n                           </button>\n                         \n                           <!-- Align Center -->\n                           <button (click)=\"setAlignment('center')\" [class.active]=\"selectedColumn?.style?.alignment === 'center'\" class=\"toolbar-btn\">\n                             <span class=\"material-icons\">{{ 'FORMAT_ALIGN_CENTER' | nxtCustomTranslate : 'format_align_center' }}</span>\n                           </button>\n                         \n                           <!-- Align Right -->\n                           <button (click)=\"setAlignment('right')\" [class.active]=\"selectedColumn?.style?.alignment === 'right'\" class=\"toolbar-btn\">\n                             <span class=\"material-icons\">{{ 'FORMAT_ALIGN_RIGHT' | nxtCustomTranslate : 'format_align_right' }}</span>\n                           </button>\n                         \n                           <!-- Close -->\n                           <button (click)=\"closeTextSettings()\" class=\"toolbar-btn\" style=\"margin-left: auto;\">✕</button>\n                         </div>\n                          \n        </div>\n\n        <!-- Input -->\n        <div *ngIf=\"field?.type === 'input'\" class=\"field-container\" (click)=\"selectElement(i)\"\n          [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'align-items': field.alignItems || '',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'color': field.fontColor || '#000000',\n    'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n    'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n    'margin': \n      (field.margin?.[0] || 0) + 'px ' +\n      (field.margin?.[1] || 0) + 'px ' +\n      (field.margin?.[2] || 0) + 'px ' +\n      (field.margin?.[3] || 0) + 'px',\n    'padding-top': (field.paddingTop || 0) + 'px',\n    'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n          draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n          (drop)=\"onDrop($event, i)\">\n          <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n            <div class=\"field-content\">\n              <div *ngIf=\"!isPreview\" class=\"label-container\">\n                <div class=\"top-right\">\n                  <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n                  <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n                </div>\n              </div>\n              <!-- AP-04JUN25 - replace app-custom-input -->\n              <nxt-input [type]=\"'text'\" [mode]=\"'edit'\" [value]=\"field.value\" [question]=\"\" [labelFont]=\"\"\n                [label]=\"field?.questionText\" [labelColor]=\"\" [labelSize]=\"\" [inputValueSize]=\"\" [labelWeight]=\"\"\n                [inputWeight]=\"\" [showLabel]=\"\" inputBorder=\"none\" svgHeight=\"20px\" svgWidth=\"20px\"\n                [placeholder]=\"field.question ? field.question : 'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" [required]=\"\"\n                inputBgColor=\"#F5F5F5\" [inputId]=\"\" [errorMessages]=\"{ required: 'This field is required' }\"\n                [inputIconLeftSrc]=\"''\" (inputValue)=\"onQuestionChange($event, field)\">\n              </nxt-input>\n            </div>\n          </div>\n        </div>\n\n        <!-- Date -->\n        <div *ngIf=\"field?.type === 'date'\" class=\"field-container\" (click)=\"selectElement(i)\"\n          [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'align-items': field.alignItems || '',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'color': field.fontColor || '#000000',\n    'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n    'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n    'margin': \n      (field.margin?.[0] || 0) + 'px ' +\n      (field.margin?.[1] || 0) + 'px ' +\n      (field.margin?.[2] || 0) + 'px ' +\n      (field.margin?.[3] || 0) + 'px',\n    'padding-top': (field.paddingTop || 0) + 'px',\n    'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n          draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n          (drop)=\"onDrop($event, i)\">\n          <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n            <div class=\"field-content\">\n              <div *ngIf=\"!isPreview\" class=\"label-container\">\n                <div class=\"top-right\">\n                  <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n                  <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n                </div>\n              </div>\n              <!-- AP-04JUN25 - replace custom date picker -->\n              <nxt-input [type]=\"'date'\" [mode]=\"'edit'\" [value]=\"field.value\" [question]=\"\" [labelFont]=\"\"\n                [label]=\"field?.questionText\" [labelColor]=\"\" [labelSize]=\"\" [inputValueSize]=\"\" [labelWeight]=\"\"\n                [inputWeight]=\"\" [showLabel]=\"\" inputBorder=\"none\" svgHeight=\"20px\" svgWidth=\"20px\"\n                [placeholder]=\"field.question ? field.question : 'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" [required]=\"\"\n                inputBgColor=\"#F5F5F5\" [inputId]=\"\" [errorMessages]=\"{ required: 'This field is required' }\"\n                [inputIconLeftSrc]=\"''\" (inputValue)=\"onQuestionChange($event, field)\">\n              </nxt-input>\n            </div>\n          </div>\n        </div>\n\n        <!-- Pdf -->\n        <div *ngIf=\"field?.type === 'Pdf'\" class=\"field-container\" (click)=\"selectElement(i)\"\n          [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'align-items': field.alignItems || '',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'color': field.fontColor || '#000000',\n    'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n    'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n    'margin': \n      (field.margin?.[0] || 0) + 'px ' +\n      (field.margin?.[1] || 0) + 'px ' +\n      (field.margin?.[2] || 0) + 'px ' +\n      (field.margin?.[3] || 0) + 'px',\n    'padding-top': (field.paddingTop || 0) + 'px',\n    'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n          draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n          (drop)=\"onDrop($event, i)\">\n          <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n            <div class=\"field-content\">\n              <div *ngIf=\"!isPreview\" class=\"label-container\">\n                <div class=\"top-right\">\n                  <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n                  <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n                </div>\n              </div>\n              <!-- <lib-booklet [bookletJSON]=\"field.pdfReferenceQuestions\"></lib-booklet> -->\n              <div *ngIf=\"field?.pdfReferenceQuestions\">\n                <ng-container *ngFor=\"let field of field?.pdfReferenceQuestions[field?.pdfReference]; let i = index\">\n                  <!-- AP-19MAR25 Line Element -->\n                  <div *ngIf=\"field?.type === 'Line'\" class=\"line-field\">\n                    <hr class=\"custom-line\" style=\"display: inline-flex\" [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-weight': field.fontWeight || '400',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'text-align': field.textAlign || 'left',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'border-style': (field.lineStyle?.toLowerCase() || 'solid'),\n    'color': field.fontColor || '#000000',\n    'margin-top': (field.paddingTop || 0) + 'px',\n    'margin-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" />\n                  </div>\n\n                  <div *ngIf=\"field?.type === 'image'\" class=\"field-container\">\n                    <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n                      <div class=\"field-content\">\n                        <div>\n                          <div class=\"logo-container\">\n                            <!-- Logo preview area -->\n                            <div class=\"logo-preview\" *ngIf=\"field.imageData\">\n                              <img [src]=\"field.imageData\" />\n                            </div>\n\n                            <!-- Upload button -->\n                            <div *ngIf=\"!field.imageData\" class=\"logo-upload-placeholder\">\n                              <label for=\"logo-upload-{{i}}\" class=\"logo-upload-label\">\n                                <img src=\"../assets/icons/Image.svg\" alt=\"Upload\" />\n                                <span>{{ 'UPLOAD_IMAGE' | nxtCustomTranslate : 'Upload Image' }}</span>\n                              </label>\n                              <input type=\"file\" id=\"logo-upload-{{i}}\" accept=\"image/*\"\n                                (change)=\"fileChangeEvent(field, $event)\" style=\"display: none;\" />\n                            </div>\n                          </div>\n                        </div>\n                      </div>\n                    </div>\n                  </div>\n                  <div *ngIf=\"field?.type === 'Space'\" class=\"field-container\">\n                    <div class=\"field-wrapper\" style=\"height:93px\" [ngClass]=\"{'isPreview': isPreview}\">\n                    </div>\n                  </div>\n                  <div *ngIf=\"field?.type === 'text'\" class=\"field-container\">\n                    <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n                      <div class=\"field-content\">\n                        <textarea class=\"custom-textarea\" [placeholder]=\"'ENTER_DETAILED_TEXT_HERE' | nxtCustomTranslate : 'Enter detailed text here...'\"\n                          [style.height.px]=\"isPreview ? 40 : field.size || 100\" [(ngModel)]=\"field.value\"\n                         (ngModelChange)=\"onQuestionChange($event, field)\"></textarea>\n                      </div>\n                    </div>\n                  </div>\n                  <!-- AP-14JUN25 Added nxt-input in pdf -->\n                  <div *ngIf=\"field?.type === 'input'\" class=\"field-container\">\n                    <div class=\"field-wrapper\" [ngClass]=\"{'isPreview': isPreview}\">\n                      <div class=\"field-content\"> \n                        <nxt-input [type]=\"'text'\" [mode]=\"'edit'\" [(ngModel)]=\"field.value\"\n                          [placeholder]=\"'ENTER_YOUR_INPUT' | nxtCustomTranslate : 'Enter your input'\" (ngModelChange)=\"onQuestionChange($event, field)\">\n                        </nxt-input>\n                      </div>\n                    </div>\n                  </div>\n                  <!-- AP-06MAR25 -->\n                  <div *ngIf=\"field?.type === 'Table'\" class=\"field-container\">\n                    <div class=\"field-wrapper\" style=\"overflow: hidden;\" [ngClass]=\"{'isPreview': isPreview}\">\n                      <div class=\"field-content\">\n                        <div class=\"nxt-table-container\">\n                          <nxt-datatable isButtons [question]=\"field\" from=\"formBuilder\"\n                            [mode]=\"isPreview ? 'view' : 'edit'\" [apiMeta]=\"field?.subText\"\n                            [tableConfig]=\"field.tableConfig\" tableId=\"\" direction=\"ltr\" tableWidth=\"auto\"\n                            (valueChange)=\"onQuestionChange($event, field)\"\n                            [data]=\"field.value?.data\"\n                            isEditable=true (columnSelected)=columnSelected($event) (removeColumn)=removeColumn($event)>\n                          </nxt-datatable>\n                        </div>\n                      </div>\n                    </div>\n                  </div>\n                </ng-container>\n              </div>\n            </div>\n          </div>\n        </div>\n\n        <!--  Table  -->\n        <!-- AP-06MAR25 -->\n        <div *ngIf=\"field?.type === 'Table'\" class=\"field-container\" (click)=\"selectElement(i)\"\n          [ngStyle]=\"{\n    'font-family': field.font || 'Helvetica Neue',\n    'font-size': (field.fontSize || 14) + 'px',\n    'width': (field.width || 100) + '%',\n    'align-items': field.alignItems || '',\n    'border-radius': '5px',\n    'border-width': (field.lineWidth || 1) + 'px',\n    'color': field.fontColor || '#000000',\n    'font-style': (field.fontStyle || []).includes('italic') ? 'italic' : 'normal',\n    'font-weight': (field.fontStyle || []).includes('bold') ? 'bold' : (field.fontWeight || '400'),\n    'margin': \n      (field.margin?.[0] || 0) + 'px ' +\n      (field.margin?.[1] || 0) + 'px ' +\n      (field.margin?.[2] || 0) + 'px ' +\n      (field.margin?.[3] || 0) + 'px',\n    'padding-top': (field.paddingTop || 0) + 'px',\n    'padding-bottom': (field.paddingBottom ?? 10) + 'px'\n  }\" [class.highlight]=\"selectedFieldIndex === i && !isPreview\"\n          draggable=\"true && !isPreview\" (dragstart)=\"onDragStart($event, i)\" (dragover)=\"onDragOver($event, i)\"\n          (drop)=\"onDrop($event, i)\">\n          <div class=\"field-wrapper\" style=\"overflow: hidden;\" [ngClass]=\"{'isPreview': isPreview}\">\n            <div class=\"field-content\">\n              <div *ngIf=\"!isPreview\" class=\"label-container\">\n                <div class=\"top-right\">\n                  <img src=\"../assets/icons/drag-dots.svg\" alt=\"Drag\" class=\"drag-dot\" />\n                  <img src=\"../assets/icons/Trash.svg\" (click)=\"removeElement(field, i)\" class=\"delete-icon\" />\n                </div>\n              </div>\n              <div class=\"nxt-table-container\">\n                <nxt-datatable isButtons [question]=\"field\" from=\"formBuilder\" [mode]=\"isPreview ? 'view' : 'edit'\"\n                  [apiMeta]=\"field?.subText\" [tableConfig]=\"field.tableConfig\" tableId=\"\" direction=\"ltr\"\n                  tableWidth=\"auto\" isEditable=true (columnSelected)=columnSelected($event)\n                  [data]=\"field.value?.data\"\n                  (valueChange)=\"onQuestionChange($event, field)\"\n                  (removeColumn)=removeColumn($event)>\n                </nxt-datatable>\n              </div>\n            </div>\n          </div>\n        </div>\n\n      </ng-container>\n    </div>\n  </div>\n  <!-- SKS13MAR25 popup conformation box -->\n  <div class=\"dialog-overlay\" *ngIf=\"isSelectTablePopup\">\n    <div class=\"dialog-box\">\n      <button class=\"close-btn-fb\" (click)=\"onClose()\">✕</button>\n      <p>{{ 'ADD_TABLE_CONFIRMATION' | nxtCustomTranslate : 'These element want to add a table' }}</p>\n      <div class=\"button-container-fb\">\n        <button class=\"yes-btn-fb\" (click)=\"addOnTable()\">{{ 'YES' | nxtCustomTranslate : 'Yes' }}</button>\n        <button class=\"no-btn-fb\" (click)=\"onClose()\">{{ 'NO' | nxtCustomTranslate : 'No' }}</button>\n      </div>\n    </div>\n  </div>\n  <app-pdf-properties *ngIf=\"!isPreview\" (formButtonHandler)=\"pdfSaveHandler($event)\"></app-pdf-properties>\n</div>\n<!--SKS25MAR25 Modal Overlay -->\n<div class=\"modal-overlay\" *ngIf=\"isImageEdit\">\n  <div class=\"modal-content\">\n      <span class=\"close-button\" (click)=\"closeModal()\">×</span>\n      <!-- Image Editor -->\n      <div *ngIf=\"selectedImageElement?.type === 'image'\">\n          <image-cropper *ngIf=\"selectedImageElement.imageData\" [imageBase64]=\"selectedImageElement.orgImageData\"\n              [disabled]=\"false\" [alignImage]=\"alignImage\" [roundCropper]=\"roundCropper\" [backgroundColor]=\"'white'\"\n              imageAltText=\"{{ 'ALTERNATIVE_IMAGE_TEXT' | nxtCustomTranslate : '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)\">\n          </image-cropper>\n\n          <!--AP-20JUN25 - Image size controls: Allow user to manually adjust width and height using range sliders -->\n          <div *ngIf=\"selectedImageElement.imageData\" class=\"manual-size-inputs\">\n            <label>{{ 'WIDTH' | nxtCustomTranslate : 'Width' }} {{ selectedImageElement.imageSize.width }}</label>\n            <input type=\"range\" min=\"50\" max=\"300\" step=\"1\"\n              [(ngModel)]=\"selectedImageElement.imageSize.width\"\n              (input)=\"onImageSizeChange()\" />\n          \n            <label>{{ 'HEIGHT' | nxtCustomTranslate : 'Height' }} {{ selectedImageElement.imageSize.height }}</label>\n            <input type=\"range\" min=\"50\" max=\"300\" step=\"1\"\n              [(ngModel)]=\"selectedImageElement.imageSize.height\"\n              (input)=\"onImageSizeChange()\" />\n          </div>\n          \n\n          <!-- Controls -->\n          <div *ngIf=\"selectedImageElement.imageData\" class=\"controls\">\n            <div class=\"cursor-pointer logo-icon\" (click)=\"rotateLeft()\" title=\"{{ 'ROTATE_LEFT' | nxtCustomTranslate : 'Rotate Left' }}\">⟲</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"rotateRight()\" title=\"{{ 'ROTATE_RIGHT' | nxtCustomTranslate : 'Rotate Right' }}\">⟳</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"zoomOut()\" title=\"{{ 'ZOOM_OUT' | nxtCustomTranslate : 'Zoom Out' }}\">-</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"zoomIn()\" title=\"{{ 'ZOOM_IN' | nxtCustomTranslate : 'Zoom In' }}\">+</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"moveLeft()\" title=\"{{ 'MOVE_LEFT' | nxtCustomTranslate : 'Move Left' }}\">←</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"moveRight()\" title=\"{{ 'MOVE_RIGHT' | nxtCustomTranslate : 'Move Right' }}\">→</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"moveUp()\" title=\"{{ 'MOVE_UP' | nxtCustomTranslate : 'Move Up' }}\">↑</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"moveDown()\" title=\"{{ 'MOVE_DOWN' | nxtCustomTranslate : 'Move Down' }}\">↓</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"flipHorizontal()\" [class.enabled]=\"transform.flipH\"\n              title=\"{{ 'FLIP_HORIZONTALLY' | nxtCustomTranslate : 'Flip Horizontally' }}\">↔</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"flipVertical()\" [class.enabled]=\"transform.flipV\"\n              title=\"{{ 'FLIP_VERTICALLY' | nxtCustomTranslate : 'Flip Vertically' }}\">↕</div>\n            <div class=\"cursor-pointer logo-icon\" (click)=\"resetImage()\" title=\"{{ 'RESET' | nxtCustomTranslate : 'Reset' }}\">×</div>\n          </div>\n      </div>\n  </div>\n</div>"]}
|