@qbs-origin/origin-form 0.8.5 → 0.8.7
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/esm2022/lib/formly/formly-scan-id/formly-scan-id.component.mjs +326 -122
- package/esm2022/lib/formly/formly-sign/formly-sign.component.mjs +76 -47
- package/esm2022/lib/formly/formly-view-documents/formly-view-documents.component.mjs +11 -7
- package/esm2022/lib/origin-form-auth.service.mjs +2 -3
- package/esm2022/lib/origin-form.component.mjs +15 -1
- package/esm2022/lib/origin-form.module.mjs +2 -13
- package/fesm2022/qbs-origin-origin-form.mjs +444 -241
- package/fesm2022/qbs-origin-origin-form.mjs.map +1 -1
- package/lib/formly/formly-scan-id/formly-scan-id.component.d.ts +44 -10
- package/lib/formly/formly-sign/formly-sign.component.d.ts +4 -0
- package/lib/formly/formly-view-documents/formly-view-documents.component.d.ts +1 -0
- package/lib/origin-form-auth.service.d.ts +1 -1
- package/package.json +2 -2
|
@@ -4,9 +4,12 @@ import { FieldType } from '@ngx-formly/core';
|
|
|
4
4
|
import * as i0 from "@angular/core";
|
|
5
5
|
import * as i1 from "../../services/applicationData.service";
|
|
6
6
|
import * as i2 from "@angular/common";
|
|
7
|
-
import * as i3 from "@angular/material/
|
|
8
|
-
import * as i4 from "@angular/material/
|
|
9
|
-
import * as i5 from "@angular/material/
|
|
7
|
+
import * as i3 from "@angular/material/form-field";
|
|
8
|
+
import * as i4 from "@angular/material/core";
|
|
9
|
+
import * as i5 from "@angular/material/select";
|
|
10
|
+
import * as i6 from "@angular/material/button";
|
|
11
|
+
import * as i7 from "@angular/material/icon";
|
|
12
|
+
import * as i8 from "@angular/material/progress-bar";
|
|
10
13
|
export class FormlyScanIdComponent extends FieldType {
|
|
11
14
|
constructor(appDataService, cdr) {
|
|
12
15
|
super();
|
|
@@ -26,8 +29,24 @@ export class FormlyScanIdComponent extends FieldType {
|
|
|
26
29
|
this.photoBase64Back = '';
|
|
27
30
|
this.isLoading = false;
|
|
28
31
|
this.successMessage = null;
|
|
29
|
-
this.videoStream = null;
|
|
30
32
|
this.isScanValid = true;
|
|
33
|
+
// Wizard state
|
|
34
|
+
this.currentWizardStep = 0;
|
|
35
|
+
this.wizardSteps = [];
|
|
36
|
+
// Camera management
|
|
37
|
+
this.cameraActive = false;
|
|
38
|
+
this.availableCameras = [];
|
|
39
|
+
this.selectedCameraId = '';
|
|
40
|
+
// Photo preview (full data URI for img display)
|
|
41
|
+
this.photoPreviewFront = '';
|
|
42
|
+
this.photoPreviewBack = '';
|
|
43
|
+
// Drag and drop
|
|
44
|
+
this.isDragOver = false;
|
|
45
|
+
// Flash animation
|
|
46
|
+
this.showFlash = false;
|
|
47
|
+
// Retry
|
|
48
|
+
this.canRetry = false;
|
|
49
|
+
this.videoStream = null;
|
|
31
50
|
}
|
|
32
51
|
ngOnInit() {
|
|
33
52
|
const group = this.form;
|
|
@@ -40,103 +59,245 @@ export class FormlyScanIdComponent extends FieldType {
|
|
|
40
59
|
if (this.props['config'].collected) {
|
|
41
60
|
const value = this.form.get(this.props['config'].componentCollected)?.value;
|
|
42
61
|
this.control.setValue(value);
|
|
43
|
-
this.form
|
|
62
|
+
this.valueChangesSub = this.form
|
|
44
63
|
.get(this.props['config'].componentCollected)
|
|
45
64
|
?.valueChanges.subscribe(() => {
|
|
46
65
|
this.control.setValue(this.form.get(this.props['config'].componentCollected)?.value);
|
|
47
66
|
});
|
|
48
67
|
}
|
|
68
|
+
this.initWizardSteps();
|
|
69
|
+
}
|
|
70
|
+
ngOnDestroy() {
|
|
71
|
+
this.stopCamera();
|
|
72
|
+
this.valueChangesSub?.unsubscribe();
|
|
73
|
+
}
|
|
74
|
+
get currentPhotoPreview() {
|
|
75
|
+
return this.currentWizardStep === 1
|
|
76
|
+
? this.photoPreviewFront
|
|
77
|
+
: this.photoPreviewBack;
|
|
78
|
+
}
|
|
79
|
+
get currentStepLabel() {
|
|
80
|
+
if (this.currentWizardStep === 1) {
|
|
81
|
+
return (this.props['labels']?.uploadFrontPrompt ||
|
|
82
|
+
'Please upload the front photo of your ID card.');
|
|
83
|
+
}
|
|
84
|
+
return (this.props['labels']?.uploadBackPrompt ||
|
|
85
|
+
'Please upload the back photo of your ID card.');
|
|
49
86
|
}
|
|
87
|
+
get currentStepShortLabel() {
|
|
88
|
+
if (this.currentWizardStep === 1) {
|
|
89
|
+
return this.props['labels']?.frontPhotoLabel || 'Front of ID';
|
|
90
|
+
}
|
|
91
|
+
return this.props['labels']?.backPhotoLabel || 'Back of ID';
|
|
92
|
+
}
|
|
93
|
+
// --- Wizard Step Management ---
|
|
94
|
+
initWizardSteps() {
|
|
95
|
+
this.wizardSteps = [
|
|
96
|
+
{
|
|
97
|
+
label: this.props['labels']?.cardTypeStepLabel || 'Select Card Type',
|
|
98
|
+
status: 'active',
|
|
99
|
+
},
|
|
100
|
+
];
|
|
101
|
+
}
|
|
102
|
+
updateWizardSteps() {
|
|
103
|
+
const frontLabel = this.props['labels']?.frontPhotoStepLabel || 'Front Photo';
|
|
104
|
+
const backLabel = this.props['labels']?.backPhotoStepLabel || 'Back Photo';
|
|
105
|
+
this.wizardSteps = [
|
|
106
|
+
{
|
|
107
|
+
label: this.props['labels']?.cardTypeStepLabel || 'Select Card Type',
|
|
108
|
+
status: 'completed',
|
|
109
|
+
},
|
|
110
|
+
{ label: frontLabel, status: 'active' },
|
|
111
|
+
];
|
|
112
|
+
if (!this.oldIdCard) {
|
|
113
|
+
this.wizardSteps.push({ label: backLabel, status: 'upcoming' });
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
// --- Navigation ---
|
|
117
|
+
goBack() {
|
|
118
|
+
this.closeCamera();
|
|
119
|
+
if (this.currentWizardStep === 2) {
|
|
120
|
+
// Go back to front photo step, preserve front photo
|
|
121
|
+
this.wizardSteps[2].status = 'upcoming';
|
|
122
|
+
this.wizardSteps[1].status = 'active';
|
|
123
|
+
this.currentWizardStep = 1;
|
|
124
|
+
}
|
|
125
|
+
else if (this.currentWizardStep === 1) {
|
|
126
|
+
// Go back to card type selection, reset everything
|
|
127
|
+
this.resetForm();
|
|
128
|
+
this.cardTypeSelected = false;
|
|
129
|
+
this.currentWizardStep = 0;
|
|
130
|
+
this.initWizardSteps();
|
|
131
|
+
}
|
|
132
|
+
this.cdr.detectChanges();
|
|
133
|
+
}
|
|
134
|
+
goToStep(stepIndex) {
|
|
135
|
+
if (stepIndex < this.currentWizardStep) {
|
|
136
|
+
while (this.currentWizardStep > stepIndex) {
|
|
137
|
+
this.goBack();
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
// --- Card Type Selection ---
|
|
50
142
|
selectCardType(isOldIdCard) {
|
|
51
143
|
this.oldIdCard = isOldIdCard;
|
|
52
144
|
this.cardTypeSelected = true;
|
|
53
145
|
this.resetForm();
|
|
146
|
+
this.updateWizardSteps();
|
|
147
|
+
this.currentWizardStep = 1;
|
|
148
|
+
this.cdr.detectChanges();
|
|
54
149
|
}
|
|
55
|
-
|
|
56
|
-
|
|
150
|
+
// --- File Upload ---
|
|
151
|
+
onFileSelected(event) {
|
|
152
|
+
const input = event.target;
|
|
153
|
+
const file = input?.files?.[0];
|
|
57
154
|
if (file) {
|
|
58
|
-
const
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
155
|
+
const isFront = this.currentWizardStep === 1;
|
|
156
|
+
this.processFile(file, isFront);
|
|
157
|
+
}
|
|
158
|
+
// Reset input so same file can be re-selected
|
|
159
|
+
if (input) {
|
|
160
|
+
input.value = '';
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
processFile(file, isFront) {
|
|
164
|
+
const fileExtension = '.' + file.name.split('.').pop()?.toLowerCase();
|
|
165
|
+
const fileSizeKB = file.size / 1024;
|
|
166
|
+
if (!this.validFormats.includes(fileExtension)) {
|
|
167
|
+
this.fileError = (this.props['errorMessages']?.invalidType ||
|
|
168
|
+
`Invalid file type. Allowed formats are ${this.validFormats.join(', ')}`).replace('{formats}', this.validFormats.join(', '));
|
|
169
|
+
this.control.setErrors({ invalidType: true });
|
|
170
|
+
this.cdr.detectChanges();
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
if (fileSizeKB > this.maxFileSizeKB || fileSizeKB < 1) {
|
|
174
|
+
this.fileError = (this.props['errorMessages']?.maxSize ||
|
|
175
|
+
`File size must be between 1KB and ${this.maxFileSizeKB}KB.`).replace('{size}', this.maxFileSizeKB.toString());
|
|
176
|
+
this.control.setErrors({ invalidSize: true });
|
|
177
|
+
this.cdr.detectChanges();
|
|
178
|
+
return;
|
|
179
|
+
}
|
|
180
|
+
this.fileError = null;
|
|
181
|
+
const reader = new FileReader();
|
|
182
|
+
reader.readAsDataURL(file);
|
|
183
|
+
reader.onload = () => {
|
|
184
|
+
const dataUri = reader.result?.toString() || '';
|
|
185
|
+
const base64 = dataUri.split(',')[1] || '';
|
|
186
|
+
if (isFront) {
|
|
187
|
+
this.fileNameFront = file.name;
|
|
188
|
+
this.photoBase64Front = base64;
|
|
189
|
+
this.photoPreviewFront = dataUri;
|
|
190
|
+
this.frontPhotoUploaded = true;
|
|
65
191
|
}
|
|
66
|
-
|
|
67
|
-
this.
|
|
68
|
-
|
|
69
|
-
this.
|
|
70
|
-
|
|
192
|
+
else {
|
|
193
|
+
this.fileNameBack = file.name;
|
|
194
|
+
this.photoBase64Back = base64;
|
|
195
|
+
this.photoPreviewBack = dataUri;
|
|
196
|
+
this.backPhotoUploaded = true;
|
|
71
197
|
}
|
|
72
|
-
this.
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
198
|
+
this.cdr.detectChanges();
|
|
199
|
+
};
|
|
200
|
+
}
|
|
201
|
+
// --- Drag and Drop ---
|
|
202
|
+
onDragOver(event) {
|
|
203
|
+
event.preventDefault();
|
|
204
|
+
event.stopPropagation();
|
|
205
|
+
this.isDragOver = true;
|
|
206
|
+
}
|
|
207
|
+
onDragLeave(event) {
|
|
208
|
+
event.preventDefault();
|
|
209
|
+
this.isDragOver = false;
|
|
210
|
+
}
|
|
211
|
+
onDrop(event) {
|
|
212
|
+
event.preventDefault();
|
|
213
|
+
event.stopPropagation();
|
|
214
|
+
this.isDragOver = false;
|
|
215
|
+
const files = event.dataTransfer?.files;
|
|
216
|
+
if (files && files.length > 0) {
|
|
217
|
+
const isFront = this.currentWizardStep === 1;
|
|
218
|
+
this.processFile(files[0], isFront);
|
|
90
219
|
}
|
|
91
220
|
}
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
const selectedDeviceId = prompt('Select camera', videoDevices.map((device) => device.label).join(', '));
|
|
108
|
-
const selectedDevice = videoDevices.find((device) => device.label === selectedDeviceId);
|
|
109
|
-
if (selectedDevice) {
|
|
110
|
-
this.startCamera(selectedDevice.deviceId);
|
|
111
|
-
}
|
|
112
|
-
else {
|
|
113
|
-
this.fileError = 'Invalid camera selection.';
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
})
|
|
117
|
-
.catch(() => {
|
|
118
|
-
this.fileError = 'Unable to access the camera devices.';
|
|
221
|
+
// --- Camera ---
|
|
222
|
+
async openCamera() {
|
|
223
|
+
this.fileError = null;
|
|
224
|
+
this.cdr.detectChanges();
|
|
225
|
+
if (!navigator.mediaDevices?.getUserMedia) {
|
|
226
|
+
this.fileError =
|
|
227
|
+
this.props['errorMessages']?.cameraNotSupported ||
|
|
228
|
+
'Camera not supported by this device.';
|
|
229
|
+
this.cdr.detectChanges();
|
|
230
|
+
return;
|
|
231
|
+
}
|
|
232
|
+
try {
|
|
233
|
+
// Request permission first (labels are only available after permission grant)
|
|
234
|
+
const tempStream = await navigator.mediaDevices.getUserMedia({
|
|
235
|
+
video: true,
|
|
119
236
|
});
|
|
237
|
+
tempStream.getTracks().forEach((t) => t.stop());
|
|
238
|
+
const devices = await navigator.mediaDevices.enumerateDevices();
|
|
239
|
+
this.availableCameras = devices.filter((d) => d.kind === 'videoinput');
|
|
240
|
+
if (this.availableCameras.length === 0) {
|
|
241
|
+
this.fileError =
|
|
242
|
+
this.props['errorMessages']?.noCamera ||
|
|
243
|
+
'No camera devices found.';
|
|
244
|
+
this.cdr.detectChanges();
|
|
245
|
+
return;
|
|
246
|
+
}
|
|
247
|
+
this.selectedCameraId = this.availableCameras[0].deviceId;
|
|
248
|
+
this.cameraActive = true;
|
|
249
|
+
this.cdr.detectChanges();
|
|
250
|
+
await this.startCameraStream(this.selectedCameraId);
|
|
120
251
|
}
|
|
121
|
-
|
|
122
|
-
this.fileError =
|
|
252
|
+
catch {
|
|
253
|
+
this.fileError =
|
|
254
|
+
this.props['errorMessages']?.cameraAccess ||
|
|
255
|
+
'Unable to access camera. Please check permissions.';
|
|
256
|
+
this.cdr.detectChanges();
|
|
123
257
|
}
|
|
124
258
|
}
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
259
|
+
async switchCamera(deviceId) {
|
|
260
|
+
this.stopCameraStream();
|
|
261
|
+
await this.startCameraStream(deviceId);
|
|
262
|
+
}
|
|
263
|
+
async startCameraStream(deviceId) {
|
|
264
|
+
try {
|
|
265
|
+
const stream = await navigator.mediaDevices.getUserMedia({
|
|
266
|
+
video: { deviceId: { exact: deviceId } },
|
|
267
|
+
});
|
|
129
268
|
this.videoStream = stream;
|
|
130
|
-
this.
|
|
131
|
-
this.videoElement
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
}
|
|
269
|
+
this.cdr.detectChanges();
|
|
270
|
+
if (this.videoElement?.nativeElement) {
|
|
271
|
+
this.videoElement.nativeElement.srcObject = stream;
|
|
272
|
+
await this.videoElement.nativeElement.play();
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
catch {
|
|
276
|
+
this.fileError =
|
|
277
|
+
this.props['errorMessages']?.cameraAccess ||
|
|
278
|
+
'Unable to access the selected camera.';
|
|
279
|
+
this.cameraActive = false;
|
|
280
|
+
this.cdr.detectChanges();
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
stopCameraStream() {
|
|
284
|
+
if (this.videoStream) {
|
|
285
|
+
this.videoStream.getTracks().forEach((track) => track.stop());
|
|
286
|
+
this.videoStream = null;
|
|
287
|
+
}
|
|
136
288
|
}
|
|
137
|
-
|
|
138
|
-
|
|
289
|
+
stopCamera() {
|
|
290
|
+
this.stopCameraStream();
|
|
291
|
+
this.cameraActive = false;
|
|
292
|
+
}
|
|
293
|
+
closeCamera() {
|
|
294
|
+
this.stopCamera();
|
|
295
|
+
this.cdr.detectChanges();
|
|
296
|
+
}
|
|
297
|
+
captureCurrentPhoto() {
|
|
298
|
+
if (!this.videoElement?.nativeElement) {
|
|
139
299
|
this.fileError = 'No video element available.';
|
|
300
|
+
this.cdr.detectChanges();
|
|
140
301
|
return;
|
|
141
302
|
}
|
|
142
303
|
const video = this.videoElement.nativeElement;
|
|
@@ -145,35 +306,68 @@ export class FormlyScanIdComponent extends FieldType {
|
|
|
145
306
|
canvas.height = video.videoHeight;
|
|
146
307
|
const context = canvas.getContext('2d');
|
|
147
308
|
context?.drawImage(video, 0, 0, canvas.width, canvas.height);
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
}, 'image/jpeg');
|
|
309
|
+
// Flash animation
|
|
310
|
+
this.triggerFlash();
|
|
311
|
+
const dataUri = canvas.toDataURL('image/jpeg', 0.9);
|
|
312
|
+
const base64 = dataUri.split(',')[1] || '';
|
|
313
|
+
const isFront = this.currentWizardStep === 1;
|
|
314
|
+
if (isFront) {
|
|
315
|
+
this.photoBase64Front = base64;
|
|
316
|
+
this.photoPreviewFront = dataUri;
|
|
317
|
+
this.frontPhotoUploaded = true;
|
|
318
|
+
}
|
|
319
|
+
else {
|
|
320
|
+
this.photoBase64Back = base64;
|
|
321
|
+
this.photoPreviewBack = dataUri;
|
|
322
|
+
this.backPhotoUploaded = true;
|
|
323
|
+
}
|
|
324
|
+
this.stopCamera();
|
|
325
|
+
this.cdr.detectChanges();
|
|
326
|
+
}
|
|
327
|
+
triggerFlash() {
|
|
328
|
+
this.showFlash = true;
|
|
329
|
+
this.cdr.detectChanges();
|
|
330
|
+
setTimeout(() => {
|
|
331
|
+
this.showFlash = false;
|
|
332
|
+
this.cdr.detectChanges();
|
|
333
|
+
}, 300);
|
|
174
334
|
}
|
|
335
|
+
// --- Photo Preview Actions ---
|
|
336
|
+
confirmPhoto() {
|
|
337
|
+
if (this.currentWizardStep === 1 && !this.oldIdCard) {
|
|
338
|
+
// Move to back photo step
|
|
339
|
+
this.wizardSteps[1].status = 'completed';
|
|
340
|
+
this.wizardSteps[2].status = 'active';
|
|
341
|
+
this.currentWizardStep = 2;
|
|
342
|
+
this.cdr.detectChanges();
|
|
343
|
+
}
|
|
344
|
+
else {
|
|
345
|
+
// All photos collected, trigger upload
|
|
346
|
+
this.checkUploadCondition();
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
resetCurrentPhoto() {
|
|
350
|
+
const isFront = this.currentWizardStep === 1;
|
|
351
|
+
if (isFront) {
|
|
352
|
+
this.photoBase64Front = '';
|
|
353
|
+
this.photoPreviewFront = '';
|
|
354
|
+
this.frontPhotoUploaded = false;
|
|
355
|
+
this.fileNameFront = '';
|
|
356
|
+
}
|
|
357
|
+
else {
|
|
358
|
+
this.photoBase64Back = '';
|
|
359
|
+
this.photoPreviewBack = '';
|
|
360
|
+
this.backPhotoUploaded = false;
|
|
361
|
+
this.fileNameBack = '';
|
|
362
|
+
}
|
|
363
|
+
this.fileError = null;
|
|
364
|
+
this.cdr.detectChanges();
|
|
365
|
+
}
|
|
366
|
+
// --- Upload ---
|
|
175
367
|
uploadPhoto() {
|
|
176
368
|
this.isLoading = true;
|
|
369
|
+
this.canRetry = false;
|
|
370
|
+
this.fileError = null;
|
|
177
371
|
this.cdr.detectChanges();
|
|
178
372
|
const command = {
|
|
179
373
|
appId: this.props['appId'],
|
|
@@ -182,13 +376,16 @@ export class FormlyScanIdComponent extends FieldType {
|
|
|
182
376
|
oldIdCard: this.oldIdCard,
|
|
183
377
|
photos: this.oldIdCard
|
|
184
378
|
? [{ base64: this.photoBase64Front }]
|
|
185
|
-
: [
|
|
379
|
+
: [
|
|
380
|
+
{ base64: this.photoBase64Front },
|
|
381
|
+
{ base64: this.photoBase64Back },
|
|
382
|
+
],
|
|
186
383
|
};
|
|
187
384
|
this.appDataService.scanId(command).subscribe({
|
|
188
385
|
next: async (result) => {
|
|
189
386
|
console.log('Scan ID result:', result);
|
|
190
387
|
this.isLoading = false;
|
|
191
|
-
|
|
388
|
+
const appData = await this.appDataService.getSteps(this.props['appDataId']);
|
|
192
389
|
if (appData) {
|
|
193
390
|
const status = this.extractStatusFromAppData(appData);
|
|
194
391
|
console.log('[SCAN-ID] extracted status:', status);
|
|
@@ -196,13 +393,14 @@ export class FormlyScanIdComponent extends FieldType {
|
|
|
196
393
|
this.fileError =
|
|
197
394
|
this.props['errorMessages']?.invalidStatus ||
|
|
198
395
|
'Invalid status response.';
|
|
199
|
-
this.
|
|
396
|
+
this.canRetry = true;
|
|
200
397
|
this.control.setErrors({ invalidStatus: true });
|
|
201
398
|
this.cdr.detectChanges();
|
|
202
399
|
return;
|
|
203
400
|
}
|
|
204
401
|
this.isScanValid = true;
|
|
205
402
|
this.successMessage = 'Upload successful!';
|
|
403
|
+
this.wizardSteps.forEach((s) => (s.status = 'completed'));
|
|
206
404
|
if (this.props['event']) {
|
|
207
405
|
this.props['event'](appData);
|
|
208
406
|
}
|
|
@@ -213,28 +411,37 @@ export class FormlyScanIdComponent extends FieldType {
|
|
|
213
411
|
this.fileError = 'An error occurred while processing the scan.';
|
|
214
412
|
console.error('Scan ID error:', err);
|
|
215
413
|
this.isLoading = false;
|
|
414
|
+
this.canRetry = true;
|
|
216
415
|
this.cdr.detectChanges();
|
|
217
|
-
this.resetForm();
|
|
218
416
|
},
|
|
219
417
|
});
|
|
220
418
|
}
|
|
419
|
+
retryUpload() {
|
|
420
|
+
this.uploadPhoto();
|
|
421
|
+
}
|
|
422
|
+
// --- Helpers ---
|
|
221
423
|
resetForm(preserveError = false) {
|
|
222
424
|
this.fileNameFront = '';
|
|
223
425
|
this.fileNameBack = '';
|
|
224
426
|
this.photoBase64Front = '';
|
|
225
427
|
this.photoBase64Back = '';
|
|
428
|
+
this.photoPreviewFront = '';
|
|
429
|
+
this.photoPreviewBack = '';
|
|
226
430
|
this.frontPhotoUploaded = false;
|
|
227
431
|
this.backPhotoUploaded = false;
|
|
228
432
|
if (!preserveError) {
|
|
229
433
|
this.fileError = null;
|
|
230
434
|
}
|
|
231
|
-
this.
|
|
435
|
+
this.stopCamera();
|
|
232
436
|
this.control.reset();
|
|
233
437
|
this.isLoading = false;
|
|
234
438
|
this.successMessage = null;
|
|
439
|
+
this.canRetry = false;
|
|
440
|
+
this.isDragOver = false;
|
|
235
441
|
}
|
|
236
442
|
checkUploadCondition() {
|
|
237
|
-
if (this.oldIdCard
|
|
443
|
+
if (this.oldIdCard && this.frontPhotoUploaded ||
|
|
444
|
+
!this.oldIdCard && this.frontPhotoUploaded && this.backPhotoUploaded) {
|
|
238
445
|
this.uploadPhoto();
|
|
239
446
|
}
|
|
240
447
|
}
|
|
@@ -256,29 +463,26 @@ export class FormlyScanIdComponent extends FieldType {
|
|
|
256
463
|
if (!step?.sections)
|
|
257
464
|
return null;
|
|
258
465
|
for (const section of step.sections) {
|
|
259
|
-
for (const
|
|
260
|
-
if (typeof
|
|
261
|
-
|
|
262
|
-
return
|
|
466
|
+
for (const ctrl of section.controls || []) {
|
|
467
|
+
if (typeof ctrl.fillValue === 'string' &&
|
|
468
|
+
ctrl.fillValue.trim() !== '') {
|
|
469
|
+
return ctrl.fillValue.trim();
|
|
263
470
|
}
|
|
264
471
|
}
|
|
265
472
|
}
|
|
266
473
|
return null;
|
|
267
474
|
}
|
|
268
475
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormlyScanIdComponent, deps: [{ token: i1.ApplicationDataService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
269
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: FormlyScanIdComponent, selector: "app-formly-scan-id", viewQueries: [{ propertyName: "fileInputFront", first: true, predicate: ["fileInputFront"], descendants: true }, { propertyName: "fileInputBack", first: true, predicate: ["fileInputBack"], descendants: true }, { propertyName: "videoElement", first: true, predicate: ["videoElement"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"scan-id-container\">\n <h2 class=\"title\">{{ props[\"name\"] }}</h2>\n <p class=\"description\">{{ props[\"description\"] }}</p>\n\n <!-- Selection Step -->\n <div class=\"card-selection-container\">\n <div class=\"card-options\">\n <div class=\"card-option\"\n [class.selected]=\"oldIdCard === true && cardTypeSelected\"\n (click)=\"selectCardType(true)\">\n <img src=\"assets/images/origin-form/old-id-card.png\" alt=\"Old ID Card\" />\n </div>\n <div class=\"card-option\"\n [class.selected]=\"oldIdCard === false && cardTypeSelected\"\n (click)=\"selectCardType(false)\">\n <img src=\"assets/images/origin-form/new-id-card.png\" alt=\"New ID Card\" />\n </div>\n </div>\n </div>\n\n <div *ngIf=\"isLoading\" class=\"loading-spinner\">\n <mat-spinner></mat-spinner>\n </div>\n\n <div *ngIf=\"cardTypeSelected && !isLoading\" class=\"actions\">\n <!-- Step 1: Upload Front Photo -->\n <div *ngIf=\"!frontPhotoUploaded\" class=\"upload-button-container\">\n <p>{{ props['labels'].uploadFrontPrompt || 'Please upload the front photo of your ID card.' }}</p>\n <input type=\"file\" #fileInputFront (change)=\"onFileSelected($event, true)\" hidden />\n <button mat-flat-button color=\"primary\" (click)=\"fileInputFront.click()\">\n <mat-icon>cloud_upload</mat-icon>\n {{ fileNameFront ? fileNameFront : props['labels'].uploadFileButtonTranslations || 'Upload Front Photo' }}\n </button>\n <div *ngIf=\"fileError\" class=\"error-message\">\n {{ fileError }}\n </div>\n </div>\n\n <!-- Step 2: Upload Back Photo (Only for New ID Cards) -->\n <div *ngIf=\"frontPhotoUploaded && !oldIdCard && !isLoading\" class=\"upload-button-container\">\n <p>{{ props['labels'].uploadBackPrompt || 'Please upload the back photo of your ID card.' }}</p>\n <input type=\"file\" #fileInputBack (change)=\"onFileSelected($event, false)\" hidden />\n <button mat-flat-button color=\"primary\" (click)=\"fileInputBack.click()\">\n <mat-icon>cloud_upload</mat-icon>\n {{ fileNameBack ? fileNameBack : props['labels'].uploadFileButtonTranslations || 'Upload Back Photo' }}\n </button>\n <div *ngIf=\"fileError\" class=\"error-message\">\n {{ fileError }}\n </div>\n </div> \n <!-- Take Photo Button (Optional) -->\n <div *ngIf=\"props['config'].showTakePictureButton\" class=\"take-photo-container\">\n <button mat-flat-button color=\"accent\" (click)=\"takePicture()\">\n <mat-icon>camera_alt</mat-icon>\n {{ props['labels'].takePictureButtonTranslations || 'Take a photo' }}\n </button>\n <video #videoElement *ngIf=\"videoStream\" width=\"100%\" class=\"video-preview\" autoplay></video>\n\n <!-- Take Front Photo -->\n <button mat-flat-button color=\"warn\" *ngIf=\"videoStream && !frontPhotoUploaded\" (click)=\"capturePhoto(true)\">\n <mat-icon>photo_camera</mat-icon>\n {{ this.props['errorMessages']?.frontId || 'Take a photo' }}\n </button>\n\n <!-- Take Back Photo -->\n <button mat-flat-button color=\"warn\" *ngIf=\"videoStream && frontPhotoUploaded && !oldIdCard\" (click)=\"capturePhoto(false)\">\n <mat-icon>photo_camera</mat-icon>\n {{ this.props['errorMessages']?.backId || 'Take a photo' }}\n </button>\n </div>\n </div>\n <div *ngIf=\"successMessage && isScanValid\" class=\"success-message-container\">\n <mat-icon class=\"large-icon\" color=\"primary\">check_circle</mat-icon>\n </div>\n</div>", styles: [".loading-spinner{display:flex;justify-content:center;align-items:center;margin-top:20px}.success-message-container{display:flex;align-items:center;gap:8px;color:green;font-weight:700;justify-content:center}.large-icon{font-size:36px;width:36px;height:36px}.scan-id-container{max-width:600px;margin:0 auto 16px;padding:20px;background-color:#fff;border-radius:8px;box-shadow:0 4px 8px #0000001a;text-align:center}.scan-id-container .card-selection-container{margin-bottom:20px}.scan-id-container .card-options{display:flex;justify-content:space-around}.scan-id-container .card-option{border:2px solid transparent;border-radius:10px;cursor:pointer;padding:10px;text-align:center;transition:border-color .3s,background-color .3s}.scan-id-container .card-option:hover{border-color:#3f51b5}.scan-id-container .card-option img{max-width:100%;height:auto;margin-bottom:10px}.scan-id-container .card-option p{font-weight:700}.scan-id-container .card-option.selected{border-color:#3f51b5;background-color:#e3f2fd;box-shadow:0 0 10px #0000001a}.scan-id-container .title{font-size:24px;font-weight:700;color:#333;margin-bottom:10px}.scan-id-container .description{font-size:16px;color:#666;margin-bottom:20px}.scan-id-container .actions{display:flex;flex-direction:column;gap:20px}.scan-id-container .actions .upload-button-container,.scan-id-container .actions .take-photo-container{display:flex;flex-direction:column;align-items:center}.scan-id-container .actions .upload-button-container button,.scan-id-container .actions .take-photo-container button{display:flex;align-items:center;gap:8px;padding:10px 20px;font-size:16px;font-weight:600;border-radius:50px;transition:background-color .3s}.scan-id-container .actions .upload-button-container button mat-icon,.scan-id-container .actions .take-photo-container button mat-icon{font-size:20px}.scan-id-container .actions .upload-button-container button:hover,.scan-id-container .actions .take-photo-container button:hover{background-color:#0069c0}.scan-id-container .actions .upload-button-container .error-message,.scan-id-container .actions .take-photo-container .error-message{color:#e53935;font-size:14px;margin-top:10px}.scan-id-container .actions .video-preview{margin-top:20px;border-radius:8px;box-shadow:0 2px 4px #0003}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i4.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i5.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }] }); }
|
|
476
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: FormlyScanIdComponent, selector: "app-formly-scan-id", viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }, { propertyName: "videoElement", first: true, predicate: ["videoElement"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"scan-id-container\">\n <!-- Header -->\n <h2 class=\"scan-id-title\">{{ props[\"name\"] }}</h2>\n <p class=\"scan-id-description\">{{ props[\"description\"] }}</p>\n\n <!-- Step Progress Indicator -->\n <div class=\"step-progress\" *ngIf=\"wizardSteps.length > 1\" role=\"list\">\n <div class=\"step-progress__item\" *ngFor=\"let step of wizardSteps; let i = index; let last = last\" role=\"listitem\">\n <div class=\"step-progress__circle\"\n [class.active]=\"step.status === 'active'\"\n [class.completed]=\"step.status === 'completed'\"\n [class.upcoming]=\"step.status === 'upcoming'\"\n [class.clickable]=\"step.status === 'completed'\"\n [attr.aria-current]=\"step.status === 'active' ? 'step' : null\"\n (click)=\"step.status === 'completed' ? goToStep(i) : null\">\n <mat-icon *ngIf=\"step.status === 'completed'\">check</mat-icon>\n <span *ngIf=\"step.status !== 'completed'\">{{ i + 1 }}</span>\n </div>\n <span class=\"step-progress__label\"\n [class.active]=\"step.status === 'active'\"\n [class.completed]=\"step.status === 'completed'\">\n {{ step.label }}\n </span>\n <div class=\"step-progress__connector\" *ngIf=\"!last\"\n [class.completed]=\"step.status === 'completed'\">\n </div>\n </div>\n </div>\n\n <!-- Step 1: Card Type Selection -->\n <div class=\"card-type-step\" *ngIf=\"currentWizardStep === 0\">\n <div class=\"card-type-options\" role=\"radiogroup\">\n <div class=\"card-type-option\"\n [class.selected]=\"oldIdCard === true && cardTypeSelected\"\n (click)=\"selectCardType(true)\"\n (keydown.enter)=\"selectCardType(true)\"\n (keydown.space)=\"selectCardType(true); $event.preventDefault()\"\n role=\"radio\"\n [attr.aria-checked]=\"oldIdCard === true && cardTypeSelected\"\n tabindex=\"0\">\n <div class=\"card-type-option__image-wrapper\">\n <img src=\"assets/images/origin-form/old-id-card.png\" alt=\"Old ID Card\" />\n </div>\n <div class=\"card-type-option__info\">\n <span class=\"card-type-option__label\">\n {{ props['labels']?.oldCardLabel || 'Old ID Card' }}\n </span>\n <span class=\"card-type-option__helper\">\n {{ props['labels']?.oldCardHelper || 'Single-sided card - 1 photo needed' }}\n </span>\n </div>\n </div>\n\n <div class=\"card-type-option\"\n [class.selected]=\"oldIdCard === false && cardTypeSelected\"\n (click)=\"selectCardType(false)\"\n (keydown.enter)=\"selectCardType(false)\"\n (keydown.space)=\"selectCardType(false); $event.preventDefault()\"\n role=\"radio\"\n [attr.aria-checked]=\"oldIdCard === false && cardTypeSelected\"\n tabindex=\"0\">\n <div class=\"card-type-option__image-wrapper\">\n <img src=\"assets/images/origin-form/new-id-card.png\" alt=\"New ID Card\" />\n </div>\n <div class=\"card-type-option__info\">\n <span class=\"card-type-option__label\">\n {{ props['labels']?.newCardLabel || 'New ID Card' }}\n </span>\n <span class=\"card-type-option__helper\">\n {{ props['labels']?.newCardHelper || 'Card with front and back - 2 photos needed' }}\n </span>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Steps 2-3: Photo Capture / Upload -->\n <div class=\"photo-step\" *ngIf=\"(currentWizardStep === 1 || currentWizardStep === 2) && !isLoading && !(successMessage && isScanValid)\">\n\n <!-- Back Button -->\n <button mat-button class=\"back-btn\" (click)=\"goBack()\">\n <mat-icon>arrow_back</mat-icon>\n {{ currentWizardStep === 1\n ? (props['labels']?.changeCardTypeLabel || 'Change card type')\n : (props['labels']?.backLabel || 'Back') }}\n </button>\n\n <!-- Photo Preview Mode -->\n <div class=\"photo-preview\" *ngIf=\"currentPhotoPreview && !cameraActive\">\n <p class=\"photo-preview__label\">{{ currentStepShortLabel }}</p>\n <img class=\"photo-preview__image\" [src]=\"currentPhotoPreview\" [alt]=\"currentStepShortLabel\" />\n <div class=\"photo-preview__actions\">\n <button mat-stroked-button (click)=\"resetCurrentPhoto()\">\n <mat-icon>refresh</mat-icon>\n {{ props['labels']?.retakeLabel || 'Retake' }}\n </button>\n <button mat-flat-button color=\"primary\" (click)=\"confirmPhoto()\">\n <mat-icon>check</mat-icon>\n {{ props['labels']?.usePhotoLabel || 'Use This Photo' }}\n </button>\n </div>\n </div>\n\n <!-- Capture Mode (no photo yet) -->\n <div class=\"photo-capture\" *ngIf=\"!currentPhotoPreview && !cameraActive\">\n <p class=\"photo-capture__label\">{{ currentStepLabel }}</p>\n\n <!-- Drop Zone -->\n <div class=\"photo-capture__dropzone\"\n [class.dragover]=\"isDragOver\"\n (dragover)=\"onDragOver($event)\"\n (dragleave)=\"onDragLeave($event)\"\n (drop)=\"onDrop($event)\"\n role=\"button\"\n tabindex=\"0\"\n [attr.aria-label]=\"'Drop zone for ' + currentStepShortLabel + ' upload'\">\n <mat-icon class=\"dropzone-icon\">cloud_upload</mat-icon>\n <p class=\"dropzone-text\">\n {{ props['labels']?.dragDropPrompt || 'Drag & drop your photo here, or use the buttons below' }}\n </p>\n <p class=\"dropzone-formats\">\n {{ props['labels']?.formatsLabel || 'Accepted' }}: {{ validFormats.join(', ') }} ·\n {{ props['labels']?.maxSizeLabel || 'Max' }}: {{ maxFileSizeKB }}KB\n </p>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"photo-capture__actions\">\n <input type=\"file\" #fileInput (change)=\"onFileSelected($event)\" hidden [accept]=\"validFormats.join(',')\" />\n <button mat-flat-button color=\"primary\" (click)=\"fileInput.click()\" class=\"action-btn\">\n <mat-icon>upload_file</mat-icon>\n {{ props['labels']?.uploadFileButtonTranslations || 'Upload File' }}\n </button>\n <button mat-flat-button color=\"primary\" (click)=\"openCamera()\" class=\"action-btn\"\n *ngIf=\"props['config'].showTakePictureButton !== false\">\n <mat-icon>photo_camera</mat-icon>\n {{ props['labels']?.takePictureButtonTranslations || 'Take Photo' }}\n </button>\n </div>\n </div>\n\n <!-- Camera Viewfinder -->\n <div class=\"camera-container\" *ngIf=\"cameraActive\">\n <!-- Camera selector -->\n <mat-form-field *ngIf=\"availableCameras.length > 1\" appearance=\"outline\" class=\"camera-selector\">\n <mat-label>{{ props['labels']?.selectCameraLabel || 'Select camera' }}</mat-label>\n <mat-select [(value)]=\"selectedCameraId\" (selectionChange)=\"switchCamera($event.value)\">\n <mat-option *ngFor=\"let cam of availableCameras; let i = index\" [value]=\"cam.deviceId\">\n {{ cam.label || 'Camera ' + (i + 1) }}\n </mat-option>\n </mat-select>\n </mat-form-field>\n\n <div class=\"camera-viewfinder\">\n <video #videoElement autoplay playsinline class=\"camera-video\"></video>\n <div class=\"camera-flash-overlay\" [class.flash]=\"showFlash\"></div>\n </div>\n\n <div class=\"camera-controls\">\n <button mat-stroked-button (click)=\"closeCamera()\" class=\"camera-cancel-btn\">\n {{ props['labels']?.cancelLabel || 'Cancel' }}\n </button>\n <button mat-fab color=\"primary\" (click)=\"captureCurrentPhoto()\" class=\"camera-capture-btn\"\n [attr.aria-label]=\"'Capture ' + currentStepShortLabel\">\n <mat-icon>photo_camera</mat-icon>\n </button>\n <div class=\"camera-spacer\"></div>\n </div>\n </div>\n\n <!-- Error Display -->\n <div class=\"photo-step__error\" *ngIf=\"fileError\" role=\"alert\" aria-live=\"assertive\">\n <mat-icon>error_outline</mat-icon>\n <span>{{ fileError }}</span>\n <button mat-stroked-button *ngIf=\"canRetry\" (click)=\"retryUpload()\" class=\"retry-btn\">\n <mat-icon>refresh</mat-icon>\n {{ props['labels']?.retryLabel || 'Try Again' }}\n </button>\n </div>\n </div>\n\n <!-- Loading / Processing State -->\n <div class=\"processing-overlay\" *ngIf=\"isLoading\">\n <mat-progress-bar mode=\"indeterminate\"></mat-progress-bar>\n <p class=\"processing-text\">\n {{ props['labels']?.processingMessage || 'Processing your ID scan...' }}\n </p>\n </div>\n\n <!-- Success State -->\n <div class=\"scan-success\" *ngIf=\"successMessage && isScanValid\" role=\"status\" aria-live=\"polite\">\n <mat-icon class=\"scan-success__icon\">check_circle</mat-icon>\n <h3 class=\"scan-success__title\">\n {{ props['labels']?.successTitle || 'ID scan successful' }}\n </h3>\n </div>\n</div>\n", styles: ["@keyframes fadeIn{0%{opacity:0;transform:translateY(8px)}to{opacity:1;transform:translateY(0)}}@keyframes flashCapture{0%{opacity:0}50%{opacity:.8}to{opacity:0}}.scan-id-container{max-width:600px;margin:0 auto 16px;padding:24px;background-color:#fff;border-radius:7px;box-shadow:0 2px 12px #00000014;text-align:center}@media (max-width: 600px){.scan-id-container{padding:16px}}.scan-id-title{font-size:22px;font-weight:700;color:#2a3547;margin:0 0 8px}.scan-id-description{font-size:14px;color:#5a6a85;margin:0 0 24px}.step-progress{display:flex;justify-content:center;align-items:flex-start;gap:0;margin-bottom:32px;padding:0 8px}@media (max-width: 600px){.step-progress{margin-bottom:24px}}.step-progress__item{display:flex;flex-direction:column;align-items:center;position:relative;flex:1;min-width:0}.step-progress__circle{width:36px;height:36px;border-radius:50%;display:flex;align-items:center;justify-content:center;font-size:14px;font-weight:600;transition:all .3s ease;position:relative;z-index:1;flex-shrink:0}.step-progress__circle.upcoming{background:#f2f6fa;color:#a1aab4;border:2px solid #dfe5ef}.step-progress__circle.active{background:#5d87ff;color:#fff;border:2px solid #5d87ff;box-shadow:0 2px 8px #5d87ff59}.step-progress__circle.completed{background:#13deb9;color:#fff;border:2px solid #13deb9}.step-progress__circle.completed mat-icon{font-size:18px;width:18px;height:18px}.step-progress__label{font-size:12px;color:#a1aab4;margin-top:8px;text-align:center;line-height:1.3;max-width:100px}.step-progress__label.active{color:#5d87ff;font-weight:600}.step-progress__label.completed{color:#13deb9;font-weight:500}@media (max-width: 600px){.step-progress__label{font-size:11px;max-width:80px}}.step-progress__connector{position:absolute;top:18px;left:calc(50% + 22px);right:calc(-50% + 22px);height:2px;background:#dfe5ef;transition:background .3s ease;z-index:0;width:calc(100% - 44px)}.step-progress__connector.completed{background:#13deb9}.card-type-step{animation:fadeIn .3s ease-out}.card-type-options{display:grid;grid-template-columns:1fr 1fr;gap:16px}@media (max-width: 600px){.card-type-options{grid-template-columns:1fr;gap:12px}}.card-type-option{border:2px solid #e5eaef;border-radius:7px;cursor:pointer;padding:16px;text-align:center;transition:border-color .2s,background-color .2s,box-shadow .2s;outline:none}.card-type-option:hover{border-color:#5d87ff4d;background:#5d87ff14}.card-type-option:focus-visible{border-color:#5d87ff;box-shadow:0 0 0 3px #5d87ff33}.card-type-option.selected{border-color:#5d87ff;background:#5d87ff14;box-shadow:0 2px 8px #5d87ff26}.card-type-option__image-wrapper{margin-bottom:12px}.card-type-option__image-wrapper img{max-width:100%;height:auto;max-height:140px;object-fit:contain;border-radius:4px}@media (max-width: 600px){.card-type-option__image-wrapper img{max-height:120px}}.card-type-option__info{display:flex;flex-direction:column;gap:4px}.card-type-option__label{font-size:15px;font-weight:600;color:#2a3547}.card-type-option__helper{font-size:13px;color:#5a6a85}.back-btn{display:inline-flex;align-items:center;gap:4px;margin-bottom:16px;color:#5a6a85;font-size:14px;cursor:pointer}.back-btn mat-icon{font-size:18px;width:18px;height:18px}.step-progress__circle.clickable{cursor:pointer}.step-progress__circle.clickable:hover{transform:scale(1.1)}.photo-step{animation:fadeIn .3s ease-out}.photo-capture__label{font-size:15px;color:#2a3547;font-weight:500;margin:0 0 16px}.photo-capture__dropzone{border:2px dashed #ccd5e0;border-radius:7px;padding:32px 16px;display:flex;flex-direction:column;align-items:center;gap:8px;transition:border-color .2s,background-color .2s;cursor:default;margin-bottom:16px}.photo-capture__dropzone.dragover{border-color:#5d87ff;background:#5d87ff14}@media (max-width: 600px){.photo-capture__dropzone{padding:24px 12px}}.dropzone-icon{font-size:40px;width:40px;height:40px;color:#a1aab4}.dropzone-text{font-size:14px;color:#5a6a85;margin:0}.dropzone-formats{font-size:12px;color:#a1aab4;margin:0}.photo-capture__actions{display:flex;gap:12px;justify-content:center}@media (max-width: 600px){.photo-capture__actions{flex-direction:column;align-items:stretch}}.action-btn{display:flex;align-items:center;gap:8px;padding:10px 24px;font-size:14px;font-weight:600;border-radius:7px;min-height:44px}.action-btn mat-icon{font-size:20px;width:20px;height:20px}@media (max-width: 600px){.action-btn{justify-content:center;width:100%}}.photo-preview{animation:fadeIn .3s ease-out;display:flex;flex-direction:column;align-items:center;gap:16px}.photo-preview__label{font-size:15px;font-weight:500;color:#2a3547;margin:0}.photo-preview__image{max-height:280px;max-width:100%;object-fit:contain;border-radius:7px;border:1px solid #e5eaef}@media (max-width: 600px){.photo-preview__image{max-height:220px}}.photo-preview__actions{display:flex;gap:12px;justify-content:center;flex-wrap:wrap}.photo-preview__actions button{display:flex;align-items:center;gap:6px;min-height:44px;border-radius:7px}@media (max-width: 600px){.photo-preview__actions{flex-direction:column;align-items:stretch;width:100%}.photo-preview__actions button{justify-content:center}}.camera-container{animation:fadeIn .3s ease-out;display:flex;flex-direction:column;gap:16px}.camera-selector{width:100%}.camera-viewfinder{position:relative;border-radius:7px;overflow:hidden;background:#000}.camera-video{width:100%;max-height:400px;object-fit:cover;display:block}@media (max-width: 600px){.camera-video{max-height:300px}}.camera-flash-overlay{position:absolute;inset:0;background:#fff;opacity:0;pointer-events:none}.camera-flash-overlay.flash{animation:flashCapture .3s ease-out}.camera-controls{display:flex;align-items:center;justify-content:center;gap:16px;padding:8px 0}.camera-cancel-btn{min-height:44px;border-radius:7px}.camera-capture-btn{width:56px;height:56px}.camera-spacer{width:0}@media (min-width: 600px){.camera-spacer{width:80px}}.photo-step__error{display:flex;align-items:center;gap:12px;padding:12px 16px;margin-top:16px;background:#fa896b14;border:1px solid #fa896b;border-radius:7px;color:#2a3547;text-align:left}.photo-step__error>mat-icon{color:#fa896b;flex-shrink:0}.photo-step__error>span{flex:1;font-size:14px}@media (max-width: 600px){.photo-step__error{flex-wrap:wrap}.photo-step__error .retry-btn{width:100%;justify-content:center}}.retry-btn{display:flex;align-items:center;gap:6px;min-height:36px;border-radius:7px;flex-shrink:0}.processing-overlay{display:flex;flex-direction:column;align-items:center;gap:16px;padding:32px 0}.processing-overlay mat-progress-bar{width:100%;border-radius:7px}.processing-text{font-size:14px;color:#5a6a85;margin:0}.scan-success{animation:fadeIn .3s ease-out;display:flex;flex-direction:column;align-items:center;gap:12px;padding:32px 16px;background:#13deb914;border:1px solid #13deb9;border-radius:7px}.scan-success__icon{font-size:48px;width:48px;height:48px;color:#13deb9}.scan-success__title{font-size:18px;font-weight:600;color:#2a3547;margin:0}:host-context(.dark-theme) .scan-id-container{background-color:#2a3447;box-shadow:0 2px 12px #0000004d}:host-context(.dark-theme) .scan-id-title,:host-context(.dark-theme) .card-type-option__label,:host-context(.dark-theme) .photo-capture__label,:host-context(.dark-theme) .photo-preview__label,:host-context(.dark-theme) .scan-success__title{color:#ffffffde}:host-context(.dark-theme) .scan-id-description,:host-context(.dark-theme) .card-type-option__helper,:host-context(.dark-theme) .dropzone-text,:host-context(.dark-theme) .processing-text,:host-context(.dark-theme) .back-btn{color:#fff9}:host-context(.dark-theme) .dropzone-icon,:host-context(.dark-theme) .dropzone-formats{color:#fff6}:host-context(.dark-theme) .card-type-option{border-color:#333f55}:host-context(.dark-theme) .card-type-option:hover{border-color:#5d87ff80;background:#5d87ff1a}:host-context(.dark-theme) .card-type-option.selected{border-color:#5d87ff;background:#5d87ff26}:host-context(.dark-theme) .step-progress__circle.upcoming{background:#333f55;border-color:#465670;color:#fff6}:host-context(.dark-theme) .step-progress__label{color:#fff6}:host-context(.dark-theme) .step-progress__connector{background:#465670}:host-context(.dark-theme) .photo-capture__dropzone{border-color:#465670}:host-context(.dark-theme) .photo-capture__dropzone.dragover{border-color:#5d87ff;background:#5d87ff1a}:host-context(.dark-theme) .photo-preview__image{border-color:#333f55}:host-context(.dark-theme) .photo-step__error{background:#fa896b1a;color:#ffffffde}:host-context(.dark-theme) .scan-success{background:#13deb91a}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "component", type: i4.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "component", type: i5.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i6.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i6.MatFabButton, selector: "button[mat-fab]", inputs: ["extended"], exportAs: ["matButton"] }, { kind: "component", type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i8.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }] }); }
|
|
270
477
|
}
|
|
271
478
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormlyScanIdComponent, decorators: [{
|
|
272
479
|
type: Component,
|
|
273
|
-
args: [{ selector: 'app-formly-scan-id', template: "<div class=\"scan-id-container\">\n <h2 class=\"title\">{{ props[\"name\"] }}</h2>\n <p class=\"description\">{{ props[\"description\"] }}</p>\n\n <!-- Selection Step -->\n <div class=\"card-selection-container\">\n <div class=\"card-options\">\n <div class=\"card-option\"\n [class.selected]=\"oldIdCard === true && cardTypeSelected\"\n (click)=\"selectCardType(true)\">\n <img src=\"assets/images/origin-form/old-id-card.png\" alt=\"Old ID Card\" />\n </div>\n <div class=\"card-option\"\n [class.selected]=\"oldIdCard === false && cardTypeSelected\"\n (click)=\"selectCardType(false)\">\n <img src=\"assets/images/origin-form/new-id-card.png\" alt=\"New ID Card\" />\n </div>\n </div>\n </div>\n\n <div *ngIf=\"isLoading\" class=\"loading-spinner\">\n <mat-spinner></mat-spinner>\n </div>\n\n <div *ngIf=\"cardTypeSelected && !isLoading\" class=\"actions\">\n <!-- Step 1: Upload Front Photo -->\n <div *ngIf=\"!frontPhotoUploaded\" class=\"upload-button-container\">\n <p>{{ props['labels'].uploadFrontPrompt || 'Please upload the front photo of your ID card.' }}</p>\n <input type=\"file\" #fileInputFront (change)=\"onFileSelected($event, true)\" hidden />\n <button mat-flat-button color=\"primary\" (click)=\"fileInputFront.click()\">\n <mat-icon>cloud_upload</mat-icon>\n {{ fileNameFront ? fileNameFront : props['labels'].uploadFileButtonTranslations || 'Upload Front Photo' }}\n </button>\n <div *ngIf=\"fileError\" class=\"error-message\">\n {{ fileError }}\n </div>\n </div>\n\n <!-- Step 2: Upload Back Photo (Only for New ID Cards) -->\n <div *ngIf=\"frontPhotoUploaded && !oldIdCard && !isLoading\" class=\"upload-button-container\">\n <p>{{ props['labels'].uploadBackPrompt || 'Please upload the back photo of your ID card.' }}</p>\n <input type=\"file\" #fileInputBack (change)=\"onFileSelected($event, false)\" hidden />\n <button mat-flat-button color=\"primary\" (click)=\"fileInputBack.click()\">\n <mat-icon>cloud_upload</mat-icon>\n {{ fileNameBack ? fileNameBack : props['labels'].uploadFileButtonTranslations || 'Upload Back Photo' }}\n </button>\n <div *ngIf=\"fileError\" class=\"error-message\">\n {{ fileError }}\n </div>\n </div> \n <!-- Take Photo Button (Optional) -->\n <div *ngIf=\"props['config'].showTakePictureButton\" class=\"take-photo-container\">\n <button mat-flat-button color=\"accent\" (click)=\"takePicture()\">\n <mat-icon>camera_alt</mat-icon>\n {{ props['labels'].takePictureButtonTranslations || 'Take a photo' }}\n </button>\n <video #videoElement *ngIf=\"videoStream\" width=\"100%\" class=\"video-preview\" autoplay></video>\n\n <!-- Take Front Photo -->\n <button mat-flat-button color=\"warn\" *ngIf=\"videoStream && !frontPhotoUploaded\" (click)=\"capturePhoto(true)\">\n <mat-icon>photo_camera</mat-icon>\n {{ this.props['errorMessages']?.frontId || 'Take a photo' }}\n </button>\n\n <!-- Take Back Photo -->\n <button mat-flat-button color=\"warn\" *ngIf=\"videoStream && frontPhotoUploaded && !oldIdCard\" (click)=\"capturePhoto(false)\">\n <mat-icon>photo_camera</mat-icon>\n {{ this.props['errorMessages']?.backId || 'Take a photo' }}\n </button>\n </div>\n </div>\n <div *ngIf=\"successMessage && isScanValid\" class=\"success-message-container\">\n <mat-icon class=\"large-icon\" color=\"primary\">check_circle</mat-icon>\n </div>\n</div>", styles: [".loading-spinner{display:flex;justify-content:center;align-items:center;margin-top:20px}.success-message-container{display:flex;align-items:center;gap:8px;color:green;font-weight:700;justify-content:center}.large-icon{font-size:36px;width:36px;height:36px}.scan-id-container{max-width:600px;margin:0 auto 16px;padding:20px;background-color:#fff;border-radius:8px;box-shadow:0 4px 8px #0000001a;text-align:center}.scan-id-container .card-selection-container{margin-bottom:20px}.scan-id-container .card-options{display:flex;justify-content:space-around}.scan-id-container .card-option{border:2px solid transparent;border-radius:10px;cursor:pointer;padding:10px;text-align:center;transition:border-color .3s,background-color .3s}.scan-id-container .card-option:hover{border-color:#3f51b5}.scan-id-container .card-option img{max-width:100%;height:auto;margin-bottom:10px}.scan-id-container .card-option p{font-weight:700}.scan-id-container .card-option.selected{border-color:#3f51b5;background-color:#e3f2fd;box-shadow:0 0 10px #0000001a}.scan-id-container .title{font-size:24px;font-weight:700;color:#333;margin-bottom:10px}.scan-id-container .description{font-size:16px;color:#666;margin-bottom:20px}.scan-id-container .actions{display:flex;flex-direction:column;gap:20px}.scan-id-container .actions .upload-button-container,.scan-id-container .actions .take-photo-container{display:flex;flex-direction:column;align-items:center}.scan-id-container .actions .upload-button-container button,.scan-id-container .actions .take-photo-container button{display:flex;align-items:center;gap:8px;padding:10px 20px;font-size:16px;font-weight:600;border-radius:50px;transition:background-color .3s}.scan-id-container .actions .upload-button-container button mat-icon,.scan-id-container .actions .take-photo-container button mat-icon{font-size:20px}.scan-id-container .actions .upload-button-container button:hover,.scan-id-container .actions .take-photo-container button:hover{background-color:#0069c0}.scan-id-container .actions .upload-button-container .error-message,.scan-id-container .actions .take-photo-container .error-message{color:#e53935;font-size:14px;margin-top:10px}.scan-id-container .actions .video-preview{margin-top:20px;border-radius:8px;box-shadow:0 2px 4px #0003}\n"] }]
|
|
274
|
-
}], ctorParameters: () => [{ type: i1.ApplicationDataService }, { type: i0.ChangeDetectorRef }], propDecorators: {
|
|
275
|
-
type: ViewChild,
|
|
276
|
-
args: ['fileInputFront']
|
|
277
|
-
}], fileInputBack: [{
|
|
480
|
+
args: [{ selector: 'app-formly-scan-id', template: "<div class=\"scan-id-container\">\n <!-- Header -->\n <h2 class=\"scan-id-title\">{{ props[\"name\"] }}</h2>\n <p class=\"scan-id-description\">{{ props[\"description\"] }}</p>\n\n <!-- Step Progress Indicator -->\n <div class=\"step-progress\" *ngIf=\"wizardSteps.length > 1\" role=\"list\">\n <div class=\"step-progress__item\" *ngFor=\"let step of wizardSteps; let i = index; let last = last\" role=\"listitem\">\n <div class=\"step-progress__circle\"\n [class.active]=\"step.status === 'active'\"\n [class.completed]=\"step.status === 'completed'\"\n [class.upcoming]=\"step.status === 'upcoming'\"\n [class.clickable]=\"step.status === 'completed'\"\n [attr.aria-current]=\"step.status === 'active' ? 'step' : null\"\n (click)=\"step.status === 'completed' ? goToStep(i) : null\">\n <mat-icon *ngIf=\"step.status === 'completed'\">check</mat-icon>\n <span *ngIf=\"step.status !== 'completed'\">{{ i + 1 }}</span>\n </div>\n <span class=\"step-progress__label\"\n [class.active]=\"step.status === 'active'\"\n [class.completed]=\"step.status === 'completed'\">\n {{ step.label }}\n </span>\n <div class=\"step-progress__connector\" *ngIf=\"!last\"\n [class.completed]=\"step.status === 'completed'\">\n </div>\n </div>\n </div>\n\n <!-- Step 1: Card Type Selection -->\n <div class=\"card-type-step\" *ngIf=\"currentWizardStep === 0\">\n <div class=\"card-type-options\" role=\"radiogroup\">\n <div class=\"card-type-option\"\n [class.selected]=\"oldIdCard === true && cardTypeSelected\"\n (click)=\"selectCardType(true)\"\n (keydown.enter)=\"selectCardType(true)\"\n (keydown.space)=\"selectCardType(true); $event.preventDefault()\"\n role=\"radio\"\n [attr.aria-checked]=\"oldIdCard === true && cardTypeSelected\"\n tabindex=\"0\">\n <div class=\"card-type-option__image-wrapper\">\n <img src=\"assets/images/origin-form/old-id-card.png\" alt=\"Old ID Card\" />\n </div>\n <div class=\"card-type-option__info\">\n <span class=\"card-type-option__label\">\n {{ props['labels']?.oldCardLabel || 'Old ID Card' }}\n </span>\n <span class=\"card-type-option__helper\">\n {{ props['labels']?.oldCardHelper || 'Single-sided card - 1 photo needed' }}\n </span>\n </div>\n </div>\n\n <div class=\"card-type-option\"\n [class.selected]=\"oldIdCard === false && cardTypeSelected\"\n (click)=\"selectCardType(false)\"\n (keydown.enter)=\"selectCardType(false)\"\n (keydown.space)=\"selectCardType(false); $event.preventDefault()\"\n role=\"radio\"\n [attr.aria-checked]=\"oldIdCard === false && cardTypeSelected\"\n tabindex=\"0\">\n <div class=\"card-type-option__image-wrapper\">\n <img src=\"assets/images/origin-form/new-id-card.png\" alt=\"New ID Card\" />\n </div>\n <div class=\"card-type-option__info\">\n <span class=\"card-type-option__label\">\n {{ props['labels']?.newCardLabel || 'New ID Card' }}\n </span>\n <span class=\"card-type-option__helper\">\n {{ props['labels']?.newCardHelper || 'Card with front and back - 2 photos needed' }}\n </span>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Steps 2-3: Photo Capture / Upload -->\n <div class=\"photo-step\" *ngIf=\"(currentWizardStep === 1 || currentWizardStep === 2) && !isLoading && !(successMessage && isScanValid)\">\n\n <!-- Back Button -->\n <button mat-button class=\"back-btn\" (click)=\"goBack()\">\n <mat-icon>arrow_back</mat-icon>\n {{ currentWizardStep === 1\n ? (props['labels']?.changeCardTypeLabel || 'Change card type')\n : (props['labels']?.backLabel || 'Back') }}\n </button>\n\n <!-- Photo Preview Mode -->\n <div class=\"photo-preview\" *ngIf=\"currentPhotoPreview && !cameraActive\">\n <p class=\"photo-preview__label\">{{ currentStepShortLabel }}</p>\n <img class=\"photo-preview__image\" [src]=\"currentPhotoPreview\" [alt]=\"currentStepShortLabel\" />\n <div class=\"photo-preview__actions\">\n <button mat-stroked-button (click)=\"resetCurrentPhoto()\">\n <mat-icon>refresh</mat-icon>\n {{ props['labels']?.retakeLabel || 'Retake' }}\n </button>\n <button mat-flat-button color=\"primary\" (click)=\"confirmPhoto()\">\n <mat-icon>check</mat-icon>\n {{ props['labels']?.usePhotoLabel || 'Use This Photo' }}\n </button>\n </div>\n </div>\n\n <!-- Capture Mode (no photo yet) -->\n <div class=\"photo-capture\" *ngIf=\"!currentPhotoPreview && !cameraActive\">\n <p class=\"photo-capture__label\">{{ currentStepLabel }}</p>\n\n <!-- Drop Zone -->\n <div class=\"photo-capture__dropzone\"\n [class.dragover]=\"isDragOver\"\n (dragover)=\"onDragOver($event)\"\n (dragleave)=\"onDragLeave($event)\"\n (drop)=\"onDrop($event)\"\n role=\"button\"\n tabindex=\"0\"\n [attr.aria-label]=\"'Drop zone for ' + currentStepShortLabel + ' upload'\">\n <mat-icon class=\"dropzone-icon\">cloud_upload</mat-icon>\n <p class=\"dropzone-text\">\n {{ props['labels']?.dragDropPrompt || 'Drag & drop your photo here, or use the buttons below' }}\n </p>\n <p class=\"dropzone-formats\">\n {{ props['labels']?.formatsLabel || 'Accepted' }}: {{ validFormats.join(', ') }} ·\n {{ props['labels']?.maxSizeLabel || 'Max' }}: {{ maxFileSizeKB }}KB\n </p>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"photo-capture__actions\">\n <input type=\"file\" #fileInput (change)=\"onFileSelected($event)\" hidden [accept]=\"validFormats.join(',')\" />\n <button mat-flat-button color=\"primary\" (click)=\"fileInput.click()\" class=\"action-btn\">\n <mat-icon>upload_file</mat-icon>\n {{ props['labels']?.uploadFileButtonTranslations || 'Upload File' }}\n </button>\n <button mat-flat-button color=\"primary\" (click)=\"openCamera()\" class=\"action-btn\"\n *ngIf=\"props['config'].showTakePictureButton !== false\">\n <mat-icon>photo_camera</mat-icon>\n {{ props['labels']?.takePictureButtonTranslations || 'Take Photo' }}\n </button>\n </div>\n </div>\n\n <!-- Camera Viewfinder -->\n <div class=\"camera-container\" *ngIf=\"cameraActive\">\n <!-- Camera selector -->\n <mat-form-field *ngIf=\"availableCameras.length > 1\" appearance=\"outline\" class=\"camera-selector\">\n <mat-label>{{ props['labels']?.selectCameraLabel || 'Select camera' }}</mat-label>\n <mat-select [(value)]=\"selectedCameraId\" (selectionChange)=\"switchCamera($event.value)\">\n <mat-option *ngFor=\"let cam of availableCameras; let i = index\" [value]=\"cam.deviceId\">\n {{ cam.label || 'Camera ' + (i + 1) }}\n </mat-option>\n </mat-select>\n </mat-form-field>\n\n <div class=\"camera-viewfinder\">\n <video #videoElement autoplay playsinline class=\"camera-video\"></video>\n <div class=\"camera-flash-overlay\" [class.flash]=\"showFlash\"></div>\n </div>\n\n <div class=\"camera-controls\">\n <button mat-stroked-button (click)=\"closeCamera()\" class=\"camera-cancel-btn\">\n {{ props['labels']?.cancelLabel || 'Cancel' }}\n </button>\n <button mat-fab color=\"primary\" (click)=\"captureCurrentPhoto()\" class=\"camera-capture-btn\"\n [attr.aria-label]=\"'Capture ' + currentStepShortLabel\">\n <mat-icon>photo_camera</mat-icon>\n </button>\n <div class=\"camera-spacer\"></div>\n </div>\n </div>\n\n <!-- Error Display -->\n <div class=\"photo-step__error\" *ngIf=\"fileError\" role=\"alert\" aria-live=\"assertive\">\n <mat-icon>error_outline</mat-icon>\n <span>{{ fileError }}</span>\n <button mat-stroked-button *ngIf=\"canRetry\" (click)=\"retryUpload()\" class=\"retry-btn\">\n <mat-icon>refresh</mat-icon>\n {{ props['labels']?.retryLabel || 'Try Again' }}\n </button>\n </div>\n </div>\n\n <!-- Loading / Processing State -->\n <div class=\"processing-overlay\" *ngIf=\"isLoading\">\n <mat-progress-bar mode=\"indeterminate\"></mat-progress-bar>\n <p class=\"processing-text\">\n {{ props['labels']?.processingMessage || 'Processing your ID scan...' }}\n </p>\n </div>\n\n <!-- Success State -->\n <div class=\"scan-success\" *ngIf=\"successMessage && isScanValid\" role=\"status\" aria-live=\"polite\">\n <mat-icon class=\"scan-success__icon\">check_circle</mat-icon>\n <h3 class=\"scan-success__title\">\n {{ props['labels']?.successTitle || 'ID scan successful' }}\n </h3>\n </div>\n</div>\n", styles: ["@keyframes fadeIn{0%{opacity:0;transform:translateY(8px)}to{opacity:1;transform:translateY(0)}}@keyframes flashCapture{0%{opacity:0}50%{opacity:.8}to{opacity:0}}.scan-id-container{max-width:600px;margin:0 auto 16px;padding:24px;background-color:#fff;border-radius:7px;box-shadow:0 2px 12px #00000014;text-align:center}@media (max-width: 600px){.scan-id-container{padding:16px}}.scan-id-title{font-size:22px;font-weight:700;color:#2a3547;margin:0 0 8px}.scan-id-description{font-size:14px;color:#5a6a85;margin:0 0 24px}.step-progress{display:flex;justify-content:center;align-items:flex-start;gap:0;margin-bottom:32px;padding:0 8px}@media (max-width: 600px){.step-progress{margin-bottom:24px}}.step-progress__item{display:flex;flex-direction:column;align-items:center;position:relative;flex:1;min-width:0}.step-progress__circle{width:36px;height:36px;border-radius:50%;display:flex;align-items:center;justify-content:center;font-size:14px;font-weight:600;transition:all .3s ease;position:relative;z-index:1;flex-shrink:0}.step-progress__circle.upcoming{background:#f2f6fa;color:#a1aab4;border:2px solid #dfe5ef}.step-progress__circle.active{background:#5d87ff;color:#fff;border:2px solid #5d87ff;box-shadow:0 2px 8px #5d87ff59}.step-progress__circle.completed{background:#13deb9;color:#fff;border:2px solid #13deb9}.step-progress__circle.completed mat-icon{font-size:18px;width:18px;height:18px}.step-progress__label{font-size:12px;color:#a1aab4;margin-top:8px;text-align:center;line-height:1.3;max-width:100px}.step-progress__label.active{color:#5d87ff;font-weight:600}.step-progress__label.completed{color:#13deb9;font-weight:500}@media (max-width: 600px){.step-progress__label{font-size:11px;max-width:80px}}.step-progress__connector{position:absolute;top:18px;left:calc(50% + 22px);right:calc(-50% + 22px);height:2px;background:#dfe5ef;transition:background .3s ease;z-index:0;width:calc(100% - 44px)}.step-progress__connector.completed{background:#13deb9}.card-type-step{animation:fadeIn .3s ease-out}.card-type-options{display:grid;grid-template-columns:1fr 1fr;gap:16px}@media (max-width: 600px){.card-type-options{grid-template-columns:1fr;gap:12px}}.card-type-option{border:2px solid #e5eaef;border-radius:7px;cursor:pointer;padding:16px;text-align:center;transition:border-color .2s,background-color .2s,box-shadow .2s;outline:none}.card-type-option:hover{border-color:#5d87ff4d;background:#5d87ff14}.card-type-option:focus-visible{border-color:#5d87ff;box-shadow:0 0 0 3px #5d87ff33}.card-type-option.selected{border-color:#5d87ff;background:#5d87ff14;box-shadow:0 2px 8px #5d87ff26}.card-type-option__image-wrapper{margin-bottom:12px}.card-type-option__image-wrapper img{max-width:100%;height:auto;max-height:140px;object-fit:contain;border-radius:4px}@media (max-width: 600px){.card-type-option__image-wrapper img{max-height:120px}}.card-type-option__info{display:flex;flex-direction:column;gap:4px}.card-type-option__label{font-size:15px;font-weight:600;color:#2a3547}.card-type-option__helper{font-size:13px;color:#5a6a85}.back-btn{display:inline-flex;align-items:center;gap:4px;margin-bottom:16px;color:#5a6a85;font-size:14px;cursor:pointer}.back-btn mat-icon{font-size:18px;width:18px;height:18px}.step-progress__circle.clickable{cursor:pointer}.step-progress__circle.clickable:hover{transform:scale(1.1)}.photo-step{animation:fadeIn .3s ease-out}.photo-capture__label{font-size:15px;color:#2a3547;font-weight:500;margin:0 0 16px}.photo-capture__dropzone{border:2px dashed #ccd5e0;border-radius:7px;padding:32px 16px;display:flex;flex-direction:column;align-items:center;gap:8px;transition:border-color .2s,background-color .2s;cursor:default;margin-bottom:16px}.photo-capture__dropzone.dragover{border-color:#5d87ff;background:#5d87ff14}@media (max-width: 600px){.photo-capture__dropzone{padding:24px 12px}}.dropzone-icon{font-size:40px;width:40px;height:40px;color:#a1aab4}.dropzone-text{font-size:14px;color:#5a6a85;margin:0}.dropzone-formats{font-size:12px;color:#a1aab4;margin:0}.photo-capture__actions{display:flex;gap:12px;justify-content:center}@media (max-width: 600px){.photo-capture__actions{flex-direction:column;align-items:stretch}}.action-btn{display:flex;align-items:center;gap:8px;padding:10px 24px;font-size:14px;font-weight:600;border-radius:7px;min-height:44px}.action-btn mat-icon{font-size:20px;width:20px;height:20px}@media (max-width: 600px){.action-btn{justify-content:center;width:100%}}.photo-preview{animation:fadeIn .3s ease-out;display:flex;flex-direction:column;align-items:center;gap:16px}.photo-preview__label{font-size:15px;font-weight:500;color:#2a3547;margin:0}.photo-preview__image{max-height:280px;max-width:100%;object-fit:contain;border-radius:7px;border:1px solid #e5eaef}@media (max-width: 600px){.photo-preview__image{max-height:220px}}.photo-preview__actions{display:flex;gap:12px;justify-content:center;flex-wrap:wrap}.photo-preview__actions button{display:flex;align-items:center;gap:6px;min-height:44px;border-radius:7px}@media (max-width: 600px){.photo-preview__actions{flex-direction:column;align-items:stretch;width:100%}.photo-preview__actions button{justify-content:center}}.camera-container{animation:fadeIn .3s ease-out;display:flex;flex-direction:column;gap:16px}.camera-selector{width:100%}.camera-viewfinder{position:relative;border-radius:7px;overflow:hidden;background:#000}.camera-video{width:100%;max-height:400px;object-fit:cover;display:block}@media (max-width: 600px){.camera-video{max-height:300px}}.camera-flash-overlay{position:absolute;inset:0;background:#fff;opacity:0;pointer-events:none}.camera-flash-overlay.flash{animation:flashCapture .3s ease-out}.camera-controls{display:flex;align-items:center;justify-content:center;gap:16px;padding:8px 0}.camera-cancel-btn{min-height:44px;border-radius:7px}.camera-capture-btn{width:56px;height:56px}.camera-spacer{width:0}@media (min-width: 600px){.camera-spacer{width:80px}}.photo-step__error{display:flex;align-items:center;gap:12px;padding:12px 16px;margin-top:16px;background:#fa896b14;border:1px solid #fa896b;border-radius:7px;color:#2a3547;text-align:left}.photo-step__error>mat-icon{color:#fa896b;flex-shrink:0}.photo-step__error>span{flex:1;font-size:14px}@media (max-width: 600px){.photo-step__error{flex-wrap:wrap}.photo-step__error .retry-btn{width:100%;justify-content:center}}.retry-btn{display:flex;align-items:center;gap:6px;min-height:36px;border-radius:7px;flex-shrink:0}.processing-overlay{display:flex;flex-direction:column;align-items:center;gap:16px;padding:32px 0}.processing-overlay mat-progress-bar{width:100%;border-radius:7px}.processing-text{font-size:14px;color:#5a6a85;margin:0}.scan-success{animation:fadeIn .3s ease-out;display:flex;flex-direction:column;align-items:center;gap:12px;padding:32px 16px;background:#13deb914;border:1px solid #13deb9;border-radius:7px}.scan-success__icon{font-size:48px;width:48px;height:48px;color:#13deb9}.scan-success__title{font-size:18px;font-weight:600;color:#2a3547;margin:0}:host-context(.dark-theme) .scan-id-container{background-color:#2a3447;box-shadow:0 2px 12px #0000004d}:host-context(.dark-theme) .scan-id-title,:host-context(.dark-theme) .card-type-option__label,:host-context(.dark-theme) .photo-capture__label,:host-context(.dark-theme) .photo-preview__label,:host-context(.dark-theme) .scan-success__title{color:#ffffffde}:host-context(.dark-theme) .scan-id-description,:host-context(.dark-theme) .card-type-option__helper,:host-context(.dark-theme) .dropzone-text,:host-context(.dark-theme) .processing-text,:host-context(.dark-theme) .back-btn{color:#fff9}:host-context(.dark-theme) .dropzone-icon,:host-context(.dark-theme) .dropzone-formats{color:#fff6}:host-context(.dark-theme) .card-type-option{border-color:#333f55}:host-context(.dark-theme) .card-type-option:hover{border-color:#5d87ff80;background:#5d87ff1a}:host-context(.dark-theme) .card-type-option.selected{border-color:#5d87ff;background:#5d87ff26}:host-context(.dark-theme) .step-progress__circle.upcoming{background:#333f55;border-color:#465670;color:#fff6}:host-context(.dark-theme) .step-progress__label{color:#fff6}:host-context(.dark-theme) .step-progress__connector{background:#465670}:host-context(.dark-theme) .photo-capture__dropzone{border-color:#465670}:host-context(.dark-theme) .photo-capture__dropzone.dragover{border-color:#5d87ff;background:#5d87ff1a}:host-context(.dark-theme) .photo-preview__image{border-color:#333f55}:host-context(.dark-theme) .photo-step__error{background:#fa896b1a;color:#ffffffde}:host-context(.dark-theme) .scan-success{background:#13deb91a}\n"] }]
|
|
481
|
+
}], ctorParameters: () => [{ type: i1.ApplicationDataService }, { type: i0.ChangeDetectorRef }], propDecorators: { fileInput: [{
|
|
278
482
|
type: ViewChild,
|
|
279
|
-
args: ['
|
|
483
|
+
args: ['fileInput']
|
|
280
484
|
}], videoElement: [{
|
|
281
485
|
type: ViewChild,
|
|
282
486
|
args: ['videoElement']
|
|
283
487
|
}] } });
|
|
284
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formly-scan-id.component.js","sourceRoot":"","sources":["../../../../../../projects/origin-form/src/lib/formly/formly-scan-id/formly-scan-id.component.ts","../../../../../../projects/origin-form/src/lib/formly/formly-scan-id/formly-scan-id.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  OnInit,\n  ViewChild,\n} from '@angular/core';\nimport { FormControl, Validators, FormGroup } from '@angular/forms';\nimport { FieldType } from '@ngx-formly/core';\nimport { ApplicationDataService } from '../../services/applicationData.service';\nimport { ScanIdDto } from '../../models/application.model';\nimport { AppDataFill } from '../../models/forms.model';\n\n@Component({\n  selector: 'app-formly-scan-id',\n  templateUrl: './formly-scan-id.component.html',\n  styleUrls: ['./formly-scan-id.component.scss'],\n})\nexport class FormlyScanIdComponent extends FieldType implements OnInit {\n  control: FormControl = new FormControl('', [Validators.required]);\n  fileError: string | null = null;\n  fileNameFront: string = '';\n  fileNameBack: string = '';\n  validFormats: string[] = [];\n  maxFileSizeKB: number = 0;\n  cardTypeSelected: boolean = false;\n  oldIdCard: boolean = false;\n  frontPhotoUploaded: boolean = false;\n  backPhotoUploaded: boolean = false;\n  photoBase64Front: string = '';\n  photoBase64Back: string = '';\n  isLoading: boolean = false;\n  successMessage: string | null = null;\n\n  @ViewChild('fileInputFront') fileInputFront: ElementRef;\n  @ViewChild('fileInputBack') fileInputBack: ElementRef;\n  @ViewChild('videoElement') videoElement: ElementRef;\n  videoStream: MediaStream | null = null;\n  isScanValid: boolean = true;\n\n  constructor(\n    private appDataService: ApplicationDataService,\n    private cdr: ChangeDetectorRef\n  ) {\n    super();\n  }\n\n  ngOnInit() {\n    const group = this.form as FormGroup;\n    group.addControl(this.props['identifier'], this.control);\n\n    this.validFormats = this.props['config'].allowedFormats?.split(',') || [\n      '.jpg',\n      '.png',\n    ];\n    this.maxFileSizeKB = this.props['config'].maxFileSizeKB || 2000;\n\n    if (this.props['config'].collected) {\n      const value = this.form.get(\n        this.props['config'].componentCollected\n      )?.value;\n      this.control.setValue(value);\n\n      this.form\n        .get(this.props['config'].componentCollected)\n        ?.valueChanges.subscribe(() => {\n          this.control.setValue(\n            this.form.get(this.props['config'].componentCollected)?.value\n          );\n        });\n    }\n  }\n\n  selectCardType(isOldIdCard: boolean) {\n    this.oldIdCard = isOldIdCard;\n    this.cardTypeSelected = true;\n    this.resetForm();\n  }\n\n  onFileSelected(event: any, isFront: boolean) {\n    const file: File = event.target.files[0];\n    if (file) {\n      const fileExtension = '.' + file.name.split('.').pop()?.toLowerCase();\n      const fileSizeKB = file.size / 1024;\n\n      if (!this.validFormats.includes(fileExtension)) {\n        this.fileError = (\n          this.props['errorMessages']?.invalidType ||\n          `Invalid file type. Allowed formats are ${this.validFormats.join(\n            ', '\n          )}`\n        ).replace('{formats}', this.validFormats.join(', '));\n        this.control.setErrors({ invalidType: true });\n        return;\n      }\n\n      if (fileSizeKB > this.maxFileSizeKB || fileSizeKB < 1) {\n        this.fileError = (\n          this.props['errorMessages']?.maxSize ||\n          `File size must be between 1KB and ${this.maxFileSizeKB}KB.`\n        ).replace('{size}', this.maxFileSizeKB);\n        this.control.setErrors({ invalidSize: true });\n        return;\n      }\n\n      this.fileError = null;\n      const reader = new FileReader();\n      reader.readAsDataURL(file);\n      reader.onload = () => {\n        if (isFront) {\n          this.fileNameFront = file.name;\n          this.photoBase64Front = reader.result?.toString().split(',')[1] || '';\n          this.frontPhotoUploaded = true;\n          this.cdr.detectChanges();\n        } else {\n          this.fileNameBack = file.name;\n          this.photoBase64Back = reader.result?.toString().split(',')[1] || '';\n          this.backPhotoUploaded = true;\n          this.cdr.detectChanges();\n        }\n\n        this.checkUploadCondition();\n      };\n    }\n  }\n\n  takePicture() {\n    this.resetForm();\n\n    if (navigator.mediaDevices && navigator.mediaDevices.enumerateDevices) {\n      navigator.mediaDevices\n        .enumerateDevices()\n        .then((devices) => {\n          const videoDevices = devices.filter(\n            (device) => device.kind === 'videoinput'\n          );\n\n          if (videoDevices.length === 0) {\n            this.fileError = 'No camera devices found.';\n            return;\n          }\n\n          if (videoDevices.length === 1) {\n            this.startCamera(videoDevices[0].deviceId);\n          } else {\n            const selectedDeviceId = prompt(\n              'Select camera',\n              videoDevices.map((device) => device.label).join(', ')\n            );\n            const selectedDevice = videoDevices.find(\n              (device) => device.label === selectedDeviceId\n            );\n\n            if (selectedDevice) {\n              this.startCamera(selectedDevice.deviceId);\n            } else {\n              this.fileError = 'Invalid camera selection.';\n            }\n          }\n        })\n        .catch(() => {\n          this.fileError = 'Unable to access the camera devices.';\n        });\n    } else {\n      this.fileError = 'Camera not supported by this device.';\n    }\n  }\n\n  startCamera(deviceId: string) {\n    navigator.mediaDevices\n      .getUserMedia({ video: { deviceId } })\n      .then((stream) => {\n        this.videoStream = stream;\n        this.videoElement.nativeElement.srcObject = stream;\n        this.videoElement.nativeElement.play();\n      })\n      .catch(() => {\n        this.fileError = 'Unable to access the selected camera.';\n      });\n  }\n\n  capturePhoto(isFront: boolean) {\n    if (!this.videoElement.nativeElement) {\n      this.fileError = 'No video element available.';\n      return;\n    }\n\n    const video = this.videoElement.nativeElement;\n    const canvas = document.createElement('canvas');\n    canvas.width = video.videoWidth;\n    canvas.height = video.videoHeight;\n    const context = canvas.getContext('2d');\n    context?.drawImage(video, 0, 0, canvas.width, canvas.height);\n\n    this.isLoading = true;\n\n    canvas.toBlob((blob) => {\n      if (blob) {\n        const reader = new FileReader();\n        reader.readAsDataURL(blob);\n        reader.onloadend = () => {\n          if (isFront) {\n            this.photoBase64Front =\n              reader.result?.toString().split(',')[1] || '';\n            this.frontPhotoUploaded = true;\n          } else {\n            this.photoBase64Back =\n              reader.result?.toString().split(',')[1] || '';\n            this.backPhotoUploaded = true;\n          }\n\n          this.isLoading = false;\n          this.checkUploadCondition();\n          this.cdr.detectChanges();\n        };\n      }\n\n      if (this.videoStream) {\n        this.videoStream.getTracks().forEach((track) => track.stop());\n        this.videoStream = null;\n      }\n    }, 'image/jpeg');\n  }\n\n  uploadPhoto() {\n    this.isLoading = true;\n    this.cdr.detectChanges();\n    const command: ScanIdDto = {\n      appId: this.props['appId'],\n      appDataId: this.props['appDataId'],\n      stepId: this.props['stepId'],\n      oldIdCard: this.oldIdCard,\n      photos: this.oldIdCard\n        ? [{ base64: this.photoBase64Front }]\n        : [{ base64: this.photoBase64Front }, { base64: this.photoBase64Back }],\n    };\n\n    this.appDataService.scanId(command).subscribe({\n      next: async (result) => {\n        console.log('Scan ID result:', result);\n        this.isLoading = false;\n\n        var appData = await this.appDataService.getSteps(\n          this.props['appDataId']\n        );\n\n        if (appData) {\n          const status = this.extractStatusFromAppData(appData);\n          console.log('[SCAN-ID] extracted status:', status);\n\n          if (status?.toLowerCase().includes('invalid')) {\n            this.fileError =\n              this.props['errorMessages']?.invalidStatus ||\n              'Invalid status response.';\n            this.resetForm(true);\n            this.control.setErrors({ invalidStatus: true });\n            this.cdr.detectChanges();\n            return;\n          }\n\n          this.isScanValid = true;\n          this.successMessage = 'Upload successful!';\n          if (this.props['event']) {\n            this.props['event'](appData);\n          }\n        }\n        this.cdr.detectChanges();\n      },\n      error: (err) => {\n        this.fileError = 'An error occurred while processing the scan.';\n        console.error('Scan ID error:', err);\n        this.isLoading = false;\n        this.cdr.detectChanges();\n        this.resetForm();\n      },\n    });\n  }\n\n  resetForm(preserveError: boolean = false) {\n    this.fileNameFront = '';\n    this.fileNameBack = '';\n    this.photoBase64Front = '';\n    this.photoBase64Back = '';\n    this.frontPhotoUploaded = false;\n    this.backPhotoUploaded = false;\n    if (!preserveError) {\n      this.fileError = null;\n    }\n    this.videoStream = null;\n    this.control.reset();\n    this.isLoading = false;\n    this.successMessage = null;\n  }\n\n  checkUploadCondition() {\n    if (this.oldIdCard || (this.frontPhotoUploaded && this.backPhotoUploaded)) {\n      this.uploadPhoto();\n    }\n  }\n\n  private extractStatusFromAppData(appData: any): string | null {\n    if (!appData?.fillData || !this.props['stepId']) return null;\n\n    let fillData: AppDataFill;\n    try {\n      fillData =\n        typeof appData.fillData === 'string'\n          ? (JSON.parse(appData.fillData) as AppDataFill)\n          : appData.fillData;\n    } catch (e) {\n      console.error('[extractStatusFromAppData] JSON parse error', e);\n      return null;\n    }\n\n    const step = fillData.flux.find(\n      (s: any) => s.stepId === this.props['stepId']\n    );\n    if (!step?.sections) return null;\n\n    for (const section of step.sections) {\n      for (const control of section.controls || []) {\n        if (\n          typeof control.fillValue === 'string' &&\n          control.fillValue.trim() !== ''\n        ) {\n          return control.fillValue.trim();\n        }\n      }\n    }\n\n    return null;\n  }\n}\n","<div class=\"scan-id-container\">\n  <h2 class=\"title\">{{ props[\"name\"] }}</h2>\n  <p class=\"description\">{{ props[\"description\"] }}</p>\n\n  <!-- Selection Step -->\n  <div class=\"card-selection-container\">\n    <div class=\"card-options\">\n      <div class=\"card-option\"\n           [class.selected]=\"oldIdCard === true && cardTypeSelected\"\n           (click)=\"selectCardType(true)\">\n        <img src=\"assets/images/origin-form/old-id-card.png\" alt=\"Old ID Card\" />\n      </div>\n      <div class=\"card-option\"\n           [class.selected]=\"oldIdCard === false && cardTypeSelected\"\n           (click)=\"selectCardType(false)\">\n        <img src=\"assets/images/origin-form/new-id-card.png\" alt=\"New ID Card\" />\n      </div>\n    </div>\n  </div>\n\n  <div *ngIf=\"isLoading\" class=\"loading-spinner\">\n    <mat-spinner></mat-spinner>\n  </div>\n\n  <div *ngIf=\"cardTypeSelected && !isLoading\" class=\"actions\">\n    <!-- Step 1: Upload Front Photo -->\n    <div *ngIf=\"!frontPhotoUploaded\" class=\"upload-button-container\">\n      <p>{{ props['labels'].uploadFrontPrompt || 'Please upload the front photo of your ID card.' }}</p>\n      <input type=\"file\" #fileInputFront (change)=\"onFileSelected($event, true)\" hidden />\n      <button mat-flat-button color=\"primary\" (click)=\"fileInputFront.click()\">\n        <mat-icon>cloud_upload</mat-icon>\n        {{ fileNameFront ? fileNameFront : props['labels'].uploadFileButtonTranslations || 'Upload Front Photo' }}\n      </button>\n      <div *ngIf=\"fileError\" class=\"error-message\">\n        {{ fileError }}\n      </div>\n    </div>\n\n    <!-- Step 2: Upload Back Photo (Only for New ID Cards) -->\n    <div *ngIf=\"frontPhotoUploaded && !oldIdCard && !isLoading\" class=\"upload-button-container\">\n      <p>{{ props['labels'].uploadBackPrompt || 'Please upload the back photo of your ID card.' }}</p>\n      <input type=\"file\" #fileInputBack (change)=\"onFileSelected($event, false)\" hidden />\n      <button mat-flat-button color=\"primary\" (click)=\"fileInputBack.click()\">\n        <mat-icon>cloud_upload</mat-icon>\n        {{ fileNameBack ? fileNameBack : props['labels'].uploadFileButtonTranslations || 'Upload Back Photo' }}\n      </button>\n      <div *ngIf=\"fileError\" class=\"error-message\">\n        {{ fileError }}\n      </div>\n    </div> \n    <!-- Take Photo Button (Optional) -->\n    <div *ngIf=\"props['config'].showTakePictureButton\" class=\"take-photo-container\">\n      <button mat-flat-button color=\"accent\" (click)=\"takePicture()\">\n        <mat-icon>camera_alt</mat-icon>\n        {{ props['labels'].takePictureButtonTranslations || 'Take a photo' }}\n      </button>\n      <video #videoElement *ngIf=\"videoStream\" width=\"100%\" class=\"video-preview\" autoplay></video>\n\n      <!-- Take Front Photo -->\n      <button mat-flat-button color=\"warn\" *ngIf=\"videoStream && !frontPhotoUploaded\" (click)=\"capturePhoto(true)\">\n        <mat-icon>photo_camera</mat-icon>\n        {{ this.props['errorMessages']?.frontId || 'Take a photo' }}\n      </button>\n\n      <!-- Take Back Photo -->\n      <button mat-flat-button color=\"warn\" *ngIf=\"videoStream && frontPhotoUploaded && !oldIdCard\" (click)=\"capturePhoto(false)\">\n        <mat-icon>photo_camera</mat-icon>\n        {{ this.props['errorMessages']?.backId  || 'Take a photo' }}\n      </button>\n    </div>\n  </div>\n  <div *ngIf=\"successMessage && isScanValid\" class=\"success-message-container\">\n    <mat-icon class=\"large-icon\" color=\"primary\">check_circle</mat-icon>\n  </div>\n</div>"]}
|
|
488
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formly-scan-id.component.js","sourceRoot":"","sources":["../../../../../../projects/origin-form/src/lib/formly/formly-scan-id/formly-scan-id.component.ts","../../../../../../projects/origin-form/src/lib/formly/formly-scan-id/formly-scan-id.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  OnDestroy,\n  OnInit,\n  ViewChild,\n} from '@angular/core';\nimport { FormControl, Validators, FormGroup } from '@angular/forms';\nimport { FieldType } from '@ngx-formly/core';\nimport { Subscription } from 'rxjs';\nimport { ApplicationDataService } from '../../services/applicationData.service';\nimport { ScanIdDto } from '../../models/application.model';\nimport { AppDataFill } from '../../models/forms.model';\n\nexport interface WizardStep {\n  label: string;\n  status: 'upcoming' | 'active' | 'completed';\n}\n\n@Component({\n  selector: 'app-formly-scan-id',\n  templateUrl: './formly-scan-id.component.html',\n  styleUrls: ['./formly-scan-id.component.scss'],\n})\nexport class FormlyScanIdComponent\n  extends FieldType\n  implements OnInit, OnDestroy\n{\n  control: FormControl = new FormControl('', [Validators.required]);\n  fileError: string | null = null;\n  fileNameFront: string = '';\n  fileNameBack: string = '';\n  validFormats: string[] = [];\n  maxFileSizeKB: number = 0;\n  cardTypeSelected: boolean = false;\n  oldIdCard: boolean = false;\n  frontPhotoUploaded: boolean = false;\n  backPhotoUploaded: boolean = false;\n  photoBase64Front: string = '';\n  photoBase64Back: string = '';\n  isLoading: boolean = false;\n  successMessage: string | null = null;\n  isScanValid: boolean = true;\n\n  // Wizard state\n  currentWizardStep: number = 0;\n  wizardSteps: WizardStep[] = [];\n\n  // Camera management\n  cameraActive: boolean = false;\n  availableCameras: MediaDeviceInfo[] = [];\n  selectedCameraId: string = '';\n\n  // Photo preview (full data URI for img display)\n  photoPreviewFront: string = '';\n  photoPreviewBack: string = '';\n\n  // Drag and drop\n  isDragOver: boolean = false;\n\n  // Flash animation\n  showFlash: boolean = false;\n\n  // Retry\n  canRetry: boolean = false;\n\n  @ViewChild('fileInput') fileInput!: ElementRef<HTMLInputElement>;\n  @ViewChild('videoElement') videoElement!: ElementRef<HTMLVideoElement>;\n  videoStream: MediaStream | null = null;\n\n  private valueChangesSub?: Subscription;\n\n  constructor(\n    private appDataService: ApplicationDataService,\n    private cdr: ChangeDetectorRef\n  ) {\n    super();\n  }\n\n  ngOnInit() {\n    const group = this.form as FormGroup;\n    group.addControl(this.props['identifier'], this.control);\n\n    this.validFormats = this.props['config'].allowedFormats?.split(',') || [\n      '.jpg',\n      '.png',\n    ];\n    this.maxFileSizeKB = this.props['config'].maxFileSizeKB || 2000;\n\n    if (this.props['config'].collected) {\n      const value = this.form.get(\n        this.props['config'].componentCollected\n      )?.value;\n      this.control.setValue(value);\n\n      this.valueChangesSub = this.form\n        .get(this.props['config'].componentCollected)\n        ?.valueChanges.subscribe(() => {\n          this.control.setValue(\n            this.form.get(this.props['config'].componentCollected)?.value\n          );\n        });\n    }\n\n    this.initWizardSteps();\n  }\n\n  ngOnDestroy() {\n    this.stopCamera();\n    this.valueChangesSub?.unsubscribe();\n  }\n\n  get currentPhotoPreview(): string {\n    return this.currentWizardStep === 1\n      ? this.photoPreviewFront\n      : this.photoPreviewBack;\n  }\n\n  get currentStepLabel(): string {\n    if (this.currentWizardStep === 1) {\n      return (\n        this.props['labels']?.uploadFrontPrompt ||\n        'Please upload the front photo of your ID card.'\n      );\n    }\n    return (\n      this.props['labels']?.uploadBackPrompt ||\n      'Please upload the back photo of your ID card.'\n    );\n  }\n\n  get currentStepShortLabel(): string {\n    if (this.currentWizardStep === 1) {\n      return this.props['labels']?.frontPhotoLabel || 'Front of ID';\n    }\n    return this.props['labels']?.backPhotoLabel || 'Back of ID';\n  }\n\n  // --- Wizard Step Management ---\n\n  private initWizardSteps() {\n    this.wizardSteps = [\n      {\n        label:\n          this.props['labels']?.cardTypeStepLabel || 'Select Card Type',\n        status: 'active',\n      },\n    ];\n  }\n\n  private updateWizardSteps() {\n    const frontLabel =\n      this.props['labels']?.frontPhotoStepLabel || 'Front Photo';\n    const backLabel =\n      this.props['labels']?.backPhotoStepLabel || 'Back Photo';\n\n    this.wizardSteps = [\n      {\n        label:\n          this.props['labels']?.cardTypeStepLabel || 'Select Card Type',\n        status: 'completed',\n      },\n      { label: frontLabel, status: 'active' },\n    ];\n    if (!this.oldIdCard) {\n      this.wizardSteps.push({ label: backLabel, status: 'upcoming' });\n    }\n  }\n\n  // --- Navigation ---\n\n  goBack() {\n    this.closeCamera();\n    if (this.currentWizardStep === 2) {\n      // Go back to front photo step, preserve front photo\n      this.wizardSteps[2].status = 'upcoming';\n      this.wizardSteps[1].status = 'active';\n      this.currentWizardStep = 1;\n    } else if (this.currentWizardStep === 1) {\n      // Go back to card type selection, reset everything\n      this.resetForm();\n      this.cardTypeSelected = false;\n      this.currentWizardStep = 0;\n      this.initWizardSteps();\n    }\n    this.cdr.detectChanges();\n  }\n\n  goToStep(stepIndex: number) {\n    if (stepIndex < this.currentWizardStep) {\n      while (this.currentWizardStep > stepIndex) {\n        this.goBack();\n      }\n    }\n  }\n\n  // --- Card Type Selection ---\n\n  selectCardType(isOldIdCard: boolean) {\n    this.oldIdCard = isOldIdCard;\n    this.cardTypeSelected = true;\n    this.resetForm();\n    this.updateWizardSteps();\n    this.currentWizardStep = 1;\n    this.cdr.detectChanges();\n  }\n\n  // --- File Upload ---\n\n  onFileSelected(event: Event) {\n    const input = event.target as HTMLInputElement;\n    const file = input?.files?.[0];\n    if (file) {\n      const isFront = this.currentWizardStep === 1;\n      this.processFile(file, isFront);\n    }\n    // Reset input so same file can be re-selected\n    if (input) {\n      input.value = '';\n    }\n  }\n\n  private processFile(file: File, isFront: boolean) {\n    const fileExtension = '.' + file.name.split('.').pop()?.toLowerCase();\n    const fileSizeKB = file.size / 1024;\n\n    if (!this.validFormats.includes(fileExtension)) {\n      this.fileError = (\n        this.props['errorMessages']?.invalidType ||\n        `Invalid file type. Allowed formats are ${this.validFormats.join(\n          ', '\n        )}`\n      ).replace('{formats}', this.validFormats.join(', '));\n      this.control.setErrors({ invalidType: true });\n      this.cdr.detectChanges();\n      return;\n    }\n\n    if (fileSizeKB > this.maxFileSizeKB || fileSizeKB < 1) {\n      this.fileError = (\n        this.props['errorMessages']?.maxSize ||\n        `File size must be between 1KB and ${this.maxFileSizeKB}KB.`\n      ).replace('{size}', this.maxFileSizeKB.toString());\n      this.control.setErrors({ invalidSize: true });\n      this.cdr.detectChanges();\n      return;\n    }\n\n    this.fileError = null;\n    const reader = new FileReader();\n    reader.readAsDataURL(file);\n    reader.onload = () => {\n      const dataUri = reader.result?.toString() || '';\n      const base64 = dataUri.split(',')[1] || '';\n      if (isFront) {\n        this.fileNameFront = file.name;\n        this.photoBase64Front = base64;\n        this.photoPreviewFront = dataUri;\n        this.frontPhotoUploaded = true;\n      } else {\n        this.fileNameBack = file.name;\n        this.photoBase64Back = base64;\n        this.photoPreviewBack = dataUri;\n        this.backPhotoUploaded = true;\n      }\n      this.cdr.detectChanges();\n    };\n  }\n\n  // --- Drag and Drop ---\n\n  onDragOver(event: DragEvent) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.isDragOver = true;\n  }\n\n  onDragLeave(event: DragEvent) {\n    event.preventDefault();\n    this.isDragOver = false;\n  }\n\n  onDrop(event: DragEvent) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.isDragOver = false;\n    const files = event.dataTransfer?.files;\n    if (files && files.length > 0) {\n      const isFront = this.currentWizardStep === 1;\n      this.processFile(files[0], isFront);\n    }\n  }\n\n  // --- Camera ---\n\n  async openCamera() {\n    this.fileError = null;\n    this.cdr.detectChanges();\n\n    if (!navigator.mediaDevices?.getUserMedia) {\n      this.fileError =\n        this.props['errorMessages']?.cameraNotSupported ||\n        'Camera not supported by this device.';\n      this.cdr.detectChanges();\n      return;\n    }\n\n    try {\n      // Request permission first (labels are only available after permission grant)\n      const tempStream = await navigator.mediaDevices.getUserMedia({\n        video: true,\n      });\n      tempStream.getTracks().forEach((t) => t.stop());\n\n      const devices = await navigator.mediaDevices.enumerateDevices();\n      this.availableCameras = devices.filter(\n        (d) => d.kind === 'videoinput'\n      );\n\n      if (this.availableCameras.length === 0) {\n        this.fileError =\n          this.props['errorMessages']?.noCamera ||\n          'No camera devices found.';\n        this.cdr.detectChanges();\n        return;\n      }\n\n      this.selectedCameraId = this.availableCameras[0].deviceId;\n      this.cameraActive = true;\n      this.cdr.detectChanges();\n      await this.startCameraStream(this.selectedCameraId);\n    } catch {\n      this.fileError =\n        this.props['errorMessages']?.cameraAccess ||\n        'Unable to access camera. Please check permissions.';\n      this.cdr.detectChanges();\n    }\n  }\n\n  async switchCamera(deviceId: string) {\n    this.stopCameraStream();\n    await this.startCameraStream(deviceId);\n  }\n\n  private async startCameraStream(deviceId: string) {\n    try {\n      const stream = await navigator.mediaDevices.getUserMedia({\n        video: { deviceId: { exact: deviceId } },\n      });\n      this.videoStream = stream;\n      this.cdr.detectChanges();\n      if (this.videoElement?.nativeElement) {\n        this.videoElement.nativeElement.srcObject = stream;\n        await this.videoElement.nativeElement.play();\n      }\n    } catch {\n      this.fileError =\n        this.props['errorMessages']?.cameraAccess ||\n        'Unable to access the selected camera.';\n      this.cameraActive = false;\n      this.cdr.detectChanges();\n    }\n  }\n\n  private stopCameraStream() {\n    if (this.videoStream) {\n      this.videoStream.getTracks().forEach((track) => track.stop());\n      this.videoStream = null;\n    }\n  }\n\n  private stopCamera() {\n    this.stopCameraStream();\n    this.cameraActive = false;\n  }\n\n  closeCamera() {\n    this.stopCamera();\n    this.cdr.detectChanges();\n  }\n\n  captureCurrentPhoto() {\n    if (!this.videoElement?.nativeElement) {\n      this.fileError = 'No video element available.';\n      this.cdr.detectChanges();\n      return;\n    }\n\n    const video = this.videoElement.nativeElement;\n    const canvas = document.createElement('canvas');\n    canvas.width = video.videoWidth;\n    canvas.height = video.videoHeight;\n    const context = canvas.getContext('2d');\n    context?.drawImage(video, 0, 0, canvas.width, canvas.height);\n\n    // Flash animation\n    this.triggerFlash();\n\n    const dataUri = canvas.toDataURL('image/jpeg', 0.9);\n    const base64 = dataUri.split(',')[1] || '';\n\n    const isFront = this.currentWizardStep === 1;\n    if (isFront) {\n      this.photoBase64Front = base64;\n      this.photoPreviewFront = dataUri;\n      this.frontPhotoUploaded = true;\n    } else {\n      this.photoBase64Back = base64;\n      this.photoPreviewBack = dataUri;\n      this.backPhotoUploaded = true;\n    }\n\n    this.stopCamera();\n    this.cdr.detectChanges();\n  }\n\n  private triggerFlash() {\n    this.showFlash = true;\n    this.cdr.detectChanges();\n    setTimeout(() => {\n      this.showFlash = false;\n      this.cdr.detectChanges();\n    }, 300);\n  }\n\n  // --- Photo Preview Actions ---\n\n  confirmPhoto() {\n    if (this.currentWizardStep === 1 && !this.oldIdCard) {\n      // Move to back photo step\n      this.wizardSteps[1].status = 'completed';\n      this.wizardSteps[2].status = 'active';\n      this.currentWizardStep = 2;\n      this.cdr.detectChanges();\n    } else {\n      // All photos collected, trigger upload\n      this.checkUploadCondition();\n    }\n  }\n\n  resetCurrentPhoto() {\n    const isFront = this.currentWizardStep === 1;\n    if (isFront) {\n      this.photoBase64Front = '';\n      this.photoPreviewFront = '';\n      this.frontPhotoUploaded = false;\n      this.fileNameFront = '';\n    } else {\n      this.photoBase64Back = '';\n      this.photoPreviewBack = '';\n      this.backPhotoUploaded = false;\n      this.fileNameBack = '';\n    }\n    this.fileError = null;\n    this.cdr.detectChanges();\n  }\n\n  // --- Upload ---\n\n  uploadPhoto() {\n    this.isLoading = true;\n    this.canRetry = false;\n    this.fileError = null;\n    this.cdr.detectChanges();\n\n    const command: ScanIdDto = {\n      appId: this.props['appId'],\n      appDataId: this.props['appDataId'],\n      stepId: this.props['stepId'],\n      oldIdCard: this.oldIdCard,\n      photos: this.oldIdCard\n        ? [{ base64: this.photoBase64Front }]\n        : [\n            { base64: this.photoBase64Front },\n            { base64: this.photoBase64Back },\n          ],\n    };\n\n    this.appDataService.scanId(command).subscribe({\n      next: async (result) => {\n        console.log('Scan ID result:', result);\n        this.isLoading = false;\n\n        const appData = await this.appDataService.getSteps(\n          this.props['appDataId']\n        );\n\n        if (appData) {\n          const status = this.extractStatusFromAppData(appData);\n          console.log('[SCAN-ID] extracted status:', status);\n\n          if (status?.toLowerCase().includes('invalid')) {\n            this.fileError =\n              this.props['errorMessages']?.invalidStatus ||\n              'Invalid status response.';\n            this.canRetry = true;\n            this.control.setErrors({ invalidStatus: true });\n            this.cdr.detectChanges();\n            return;\n          }\n\n          this.isScanValid = true;\n          this.successMessage = 'Upload successful!';\n          this.wizardSteps.forEach((s) => (s.status = 'completed'));\n          if (this.props['event']) {\n            this.props['event'](appData);\n          }\n        }\n        this.cdr.detectChanges();\n      },\n      error: (err) => {\n        this.fileError = 'An error occurred while processing the scan.';\n        console.error('Scan ID error:', err);\n        this.isLoading = false;\n        this.canRetry = true;\n        this.cdr.detectChanges();\n      },\n    });\n  }\n\n  retryUpload() {\n    this.uploadPhoto();\n  }\n\n  // --- Helpers ---\n\n  resetForm(preserveError: boolean = false) {\n    this.fileNameFront = '';\n    this.fileNameBack = '';\n    this.photoBase64Front = '';\n    this.photoBase64Back = '';\n    this.photoPreviewFront = '';\n    this.photoPreviewBack = '';\n    this.frontPhotoUploaded = false;\n    this.backPhotoUploaded = false;\n    if (!preserveError) {\n      this.fileError = null;\n    }\n    this.stopCamera();\n    this.control.reset();\n    this.isLoading = false;\n    this.successMessage = null;\n    this.canRetry = false;\n    this.isDragOver = false;\n  }\n\n  checkUploadCondition() {\n    if (\n      this.oldIdCard && this.frontPhotoUploaded ||\n      !this.oldIdCard && this.frontPhotoUploaded && this.backPhotoUploaded\n    ) {\n      this.uploadPhoto();\n    }\n  }\n\n  private extractStatusFromAppData(appData: any): string | null {\n    if (!appData?.fillData || !this.props['stepId']) return null;\n\n    let fillData: AppDataFill;\n    try {\n      fillData =\n        typeof appData.fillData === 'string'\n          ? (JSON.parse(appData.fillData) as AppDataFill)\n          : appData.fillData;\n    } catch (e) {\n      console.error('[extractStatusFromAppData] JSON parse error', e);\n      return null;\n    }\n\n    const step = fillData.flux.find(\n      (s: any) => s.stepId === this.props['stepId']\n    );\n    if (!step?.sections) return null;\n\n    for (const section of step.sections) {\n      for (const ctrl of section.controls || []) {\n        if (\n          typeof ctrl.fillValue === 'string' &&\n          ctrl.fillValue.trim() !== ''\n        ) {\n          return ctrl.fillValue.trim();\n        }\n      }\n    }\n\n    return null;\n  }\n}\n","<div class=\"scan-id-container\">\n  <!-- Header -->\n  <h2 class=\"scan-id-title\">{{ props[\"name\"] }}</h2>\n  <p class=\"scan-id-description\">{{ props[\"description\"] }}</p>\n\n  <!-- Step Progress Indicator -->\n  <div class=\"step-progress\" *ngIf=\"wizardSteps.length > 1\" role=\"list\">\n    <div class=\"step-progress__item\" *ngFor=\"let step of wizardSteps; let i = index; let last = last\" role=\"listitem\">\n      <div class=\"step-progress__circle\"\n           [class.active]=\"step.status === 'active'\"\n           [class.completed]=\"step.status === 'completed'\"\n           [class.upcoming]=\"step.status === 'upcoming'\"\n           [class.clickable]=\"step.status === 'completed'\"\n           [attr.aria-current]=\"step.status === 'active' ? 'step' : null\"\n           (click)=\"step.status === 'completed' ? goToStep(i) : null\">\n        <mat-icon *ngIf=\"step.status === 'completed'\">check</mat-icon>\n        <span *ngIf=\"step.status !== 'completed'\">{{ i + 1 }}</span>\n      </div>\n      <span class=\"step-progress__label\"\n            [class.active]=\"step.status === 'active'\"\n            [class.completed]=\"step.status === 'completed'\">\n        {{ step.label }}\n      </span>\n      <div class=\"step-progress__connector\" *ngIf=\"!last\"\n           [class.completed]=\"step.status === 'completed'\">\n      </div>\n    </div>\n  </div>\n\n  <!-- Step 1: Card Type Selection -->\n  <div class=\"card-type-step\" *ngIf=\"currentWizardStep === 0\">\n    <div class=\"card-type-options\" role=\"radiogroup\">\n      <div class=\"card-type-option\"\n           [class.selected]=\"oldIdCard === true && cardTypeSelected\"\n           (click)=\"selectCardType(true)\"\n           (keydown.enter)=\"selectCardType(true)\"\n           (keydown.space)=\"selectCardType(true); $event.preventDefault()\"\n           role=\"radio\"\n           [attr.aria-checked]=\"oldIdCard === true && cardTypeSelected\"\n           tabindex=\"0\">\n        <div class=\"card-type-option__image-wrapper\">\n          <img src=\"assets/images/origin-form/old-id-card.png\" alt=\"Old ID Card\" />\n        </div>\n        <div class=\"card-type-option__info\">\n          <span class=\"card-type-option__label\">\n            {{ props['labels']?.oldCardLabel || 'Old ID Card' }}\n          </span>\n          <span class=\"card-type-option__helper\">\n            {{ props['labels']?.oldCardHelper || 'Single-sided card - 1 photo needed' }}\n          </span>\n        </div>\n      </div>\n\n      <div class=\"card-type-option\"\n           [class.selected]=\"oldIdCard === false && cardTypeSelected\"\n           (click)=\"selectCardType(false)\"\n           (keydown.enter)=\"selectCardType(false)\"\n           (keydown.space)=\"selectCardType(false); $event.preventDefault()\"\n           role=\"radio\"\n           [attr.aria-checked]=\"oldIdCard === false && cardTypeSelected\"\n           tabindex=\"0\">\n        <div class=\"card-type-option__image-wrapper\">\n          <img src=\"assets/images/origin-form/new-id-card.png\" alt=\"New ID Card\" />\n        </div>\n        <div class=\"card-type-option__info\">\n          <span class=\"card-type-option__label\">\n            {{ props['labels']?.newCardLabel || 'New ID Card' }}\n          </span>\n          <span class=\"card-type-option__helper\">\n            {{ props['labels']?.newCardHelper || 'Card with front and back - 2 photos needed' }}\n          </span>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <!-- Steps 2-3: Photo Capture / Upload -->\n  <div class=\"photo-step\" *ngIf=\"(currentWizardStep === 1 || currentWizardStep === 2) && !isLoading && !(successMessage && isScanValid)\">\n\n    <!-- Back Button -->\n    <button mat-button class=\"back-btn\" (click)=\"goBack()\">\n      <mat-icon>arrow_back</mat-icon>\n      {{ currentWizardStep === 1\n         ? (props['labels']?.changeCardTypeLabel || 'Change card type')\n         : (props['labels']?.backLabel || 'Back') }}\n    </button>\n\n    <!-- Photo Preview Mode -->\n    <div class=\"photo-preview\" *ngIf=\"currentPhotoPreview && !cameraActive\">\n      <p class=\"photo-preview__label\">{{ currentStepShortLabel }}</p>\n      <img class=\"photo-preview__image\" [src]=\"currentPhotoPreview\" [alt]=\"currentStepShortLabel\" />\n      <div class=\"photo-preview__actions\">\n        <button mat-stroked-button (click)=\"resetCurrentPhoto()\">\n          <mat-icon>refresh</mat-icon>\n          {{ props['labels']?.retakeLabel || 'Retake' }}\n        </button>\n        <button mat-flat-button color=\"primary\" (click)=\"confirmPhoto()\">\n          <mat-icon>check</mat-icon>\n          {{ props['labels']?.usePhotoLabel || 'Use This Photo' }}\n        </button>\n      </div>\n    </div>\n\n    <!-- Capture Mode (no photo yet) -->\n    <div class=\"photo-capture\" *ngIf=\"!currentPhotoPreview && !cameraActive\">\n      <p class=\"photo-capture__label\">{{ currentStepLabel }}</p>\n\n      <!-- Drop Zone -->\n      <div class=\"photo-capture__dropzone\"\n           [class.dragover]=\"isDragOver\"\n           (dragover)=\"onDragOver($event)\"\n           (dragleave)=\"onDragLeave($event)\"\n           (drop)=\"onDrop($event)\"\n           role=\"button\"\n           tabindex=\"0\"\n           [attr.aria-label]=\"'Drop zone for ' + currentStepShortLabel + ' upload'\">\n        <mat-icon class=\"dropzone-icon\">cloud_upload</mat-icon>\n        <p class=\"dropzone-text\">\n          {{ props['labels']?.dragDropPrompt || 'Drag & drop your photo here, or use the buttons below' }}\n        </p>\n        <p class=\"dropzone-formats\">\n          {{ props['labels']?.formatsLabel || 'Accepted' }}: {{ validFormats.join(', ') }} &middot;\n          {{ props['labels']?.maxSizeLabel || 'Max' }}: {{ maxFileSizeKB }}KB\n        </p>\n      </div>\n\n      <!-- Action Buttons -->\n      <div class=\"photo-capture__actions\">\n        <input type=\"file\" #fileInput (change)=\"onFileSelected($event)\" hidden [accept]=\"validFormats.join(',')\" />\n        <button mat-flat-button color=\"primary\" (click)=\"fileInput.click()\" class=\"action-btn\">\n          <mat-icon>upload_file</mat-icon>\n          {{ props['labels']?.uploadFileButtonTranslations || 'Upload File' }}\n        </button>\n        <button mat-flat-button color=\"primary\" (click)=\"openCamera()\" class=\"action-btn\"\n                *ngIf=\"props['config'].showTakePictureButton !== false\">\n          <mat-icon>photo_camera</mat-icon>\n          {{ props['labels']?.takePictureButtonTranslations || 'Take Photo' }}\n        </button>\n      </div>\n    </div>\n\n    <!-- Camera Viewfinder -->\n    <div class=\"camera-container\" *ngIf=\"cameraActive\">\n      <!-- Camera selector -->\n      <mat-form-field *ngIf=\"availableCameras.length > 1\" appearance=\"outline\" class=\"camera-selector\">\n        <mat-label>{{ props['labels']?.selectCameraLabel || 'Select camera' }}</mat-label>\n        <mat-select [(value)]=\"selectedCameraId\" (selectionChange)=\"switchCamera($event.value)\">\n          <mat-option *ngFor=\"let cam of availableCameras; let i = index\" [value]=\"cam.deviceId\">\n            {{ cam.label || 'Camera ' + (i + 1) }}\n          </mat-option>\n        </mat-select>\n      </mat-form-field>\n\n      <div class=\"camera-viewfinder\">\n        <video #videoElement autoplay playsinline class=\"camera-video\"></video>\n        <div class=\"camera-flash-overlay\" [class.flash]=\"showFlash\"></div>\n      </div>\n\n      <div class=\"camera-controls\">\n        <button mat-stroked-button (click)=\"closeCamera()\" class=\"camera-cancel-btn\">\n          {{ props['labels']?.cancelLabel || 'Cancel' }}\n        </button>\n        <button mat-fab color=\"primary\" (click)=\"captureCurrentPhoto()\" class=\"camera-capture-btn\"\n                [attr.aria-label]=\"'Capture ' + currentStepShortLabel\">\n          <mat-icon>photo_camera</mat-icon>\n        </button>\n        <div class=\"camera-spacer\"></div>\n      </div>\n    </div>\n\n    <!-- Error Display -->\n    <div class=\"photo-step__error\" *ngIf=\"fileError\" role=\"alert\" aria-live=\"assertive\">\n      <mat-icon>error_outline</mat-icon>\n      <span>{{ fileError }}</span>\n      <button mat-stroked-button *ngIf=\"canRetry\" (click)=\"retryUpload()\" class=\"retry-btn\">\n        <mat-icon>refresh</mat-icon>\n        {{ props['labels']?.retryLabel || 'Try Again' }}\n      </button>\n    </div>\n  </div>\n\n  <!-- Loading / Processing State -->\n  <div class=\"processing-overlay\" *ngIf=\"isLoading\">\n    <mat-progress-bar mode=\"indeterminate\"></mat-progress-bar>\n    <p class=\"processing-text\">\n      {{ props['labels']?.processingMessage || 'Processing your ID scan...' }}\n    </p>\n  </div>\n\n  <!-- Success State -->\n  <div class=\"scan-success\" *ngIf=\"successMessage && isScanValid\" role=\"status\" aria-live=\"polite\">\n    <mat-icon class=\"scan-success__icon\">check_circle</mat-icon>\n    <h3 class=\"scan-success__title\">\n      {{ props['labels']?.successTitle || 'ID scan successful' }}\n    </h3>\n  </div>\n</div>\n"]}
|