@qbs-origin/origin-form 0.5.0
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/README.md +24 -0
- package/esm2022/lib/action-step-handler.mjs +163 -0
- package/esm2022/lib/auth-client.service.mjs +69 -0
- package/esm2022/lib/enums/label.keys.mjs +721 -0
- package/esm2022/lib/form-css.helper.mjs +367 -0
- package/esm2022/lib/formly/baseFormlyControlComponent.mjs +52 -0
- package/esm2022/lib/formly/baseFormlyStepComponent.mjs +59 -0
- package/esm2022/lib/formly/custom-section-separator.component.mjs +32 -0
- package/esm2022/lib/formly/form-section-separator.component.mjs +36 -0
- package/esm2022/lib/formly/formly-action.mjs +56 -0
- package/esm2022/lib/formly/formly-checkbox/formly-checkbox.component.mjs +52 -0
- package/esm2022/lib/formly/formly-dictionary-dropdown-tree/formly-dictionary-dropdown-tree.component.mjs +261 -0
- package/esm2022/lib/formly/formly-download-documents/formly-download-documents.component.mjs +126 -0
- package/esm2022/lib/formly/formly-enrol-card/formly-enrol-card.component.mjs +120 -0
- package/esm2022/lib/formly/formly-field-stepper/formly-field-stepper.component.mjs +762 -0
- package/esm2022/lib/formly/formly-generate-documents/formly-generate-documents.component.mjs +57 -0
- package/esm2022/lib/formly/formly-identification.component.mjs +84 -0
- package/esm2022/lib/formly/formly-open-banking/formly-open-banking.component.mjs +590 -0
- package/esm2022/lib/formly/formly-paragraph/formly-paragraph.component.mjs +35 -0
- package/esm2022/lib/formly/formly-radio/formly-radio-component.mjs +49 -0
- package/esm2022/lib/formly/formly-row-fille.mjs +12 -0
- package/esm2022/lib/formly/formly-scan-id/formly-scan-id.component.mjs +284 -0
- package/esm2022/lib/formly/formly-sign/formly-sign.component.mjs +173 -0
- package/esm2022/lib/formly/formly-upload-documents/formly-upload-documents.component.mjs +198 -0
- package/esm2022/lib/formly/formly-validate-contact-info/formly-validate-contact-info.component.mjs +124 -0
- package/esm2022/lib/formly/formly-view-documents/formly-view-documents.component.mjs +245 -0
- package/esm2022/lib/formly/formly-view-offers/formly-view-offers.component.mjs +160 -0
- package/esm2022/lib/model-population.helper.mjs +265 -0
- package/esm2022/lib/models/application-type.model.mjs +12 -0
- package/esm2022/lib/models/application.model.mjs +30 -0
- package/esm2022/lib/models/auth/users.model.mjs +2 -0
- package/esm2022/lib/models/dictionary.model.mjs +20 -0
- package/esm2022/lib/models/flux.model.mjs +105 -0
- package/esm2022/lib/models/forms.model.mjs +572 -0
- package/esm2022/lib/models/label-info.model.mjs +2 -0
- package/esm2022/lib/models/label.model.mjs +2 -0
- package/esm2022/lib/models/language.model.mjs +3 -0
- package/esm2022/lib/models/list.model.mjs +2 -0
- package/esm2022/lib/models/partner.model.mjs +3 -0
- package/esm2022/lib/models/treeview.model.mjs +15 -0
- package/esm2022/lib/origin-form-auth.service.mjs +40 -0
- package/esm2022/lib/origin-form-config.model.mjs +2 -0
- package/esm2022/lib/origin-form-token.interceptor.mjs +35 -0
- package/esm2022/lib/origin-form.component.mjs +2391 -0
- package/esm2022/lib/origin-form.module.mjs +479 -0
- package/esm2022/lib/origin-form.service.mjs +14 -0
- package/esm2022/lib/others/check-list.database.mjs +55 -0
- package/esm2022/lib/others/config-service.mjs +42 -0
- package/esm2022/lib/others/dictionary-label-info.mjs +3 -0
- package/esm2022/lib/others/environment-type.mjs +21 -0
- package/esm2022/lib/others/external-link.directive.mjs +49 -0
- package/esm2022/lib/others/flux-helper.mjs +1397 -0
- package/esm2022/lib/others/picker.component.mjs +119 -0
- package/esm2022/lib/others/translation.pipe.mjs +21 -0
- package/esm2022/lib/others/translations-helper.mjs +258 -0
- package/esm2022/lib/others/utils.mjs +272 -0
- package/esm2022/lib/services/applicationData.service.mjs +145 -0
- package/esm2022/lib/services/auth-http.service.mjs +80 -0
- package/esm2022/lib/services/dialog.service.mjs +56 -0
- package/esm2022/lib/services/dictionary.service.mjs +198 -0
- package/esm2022/lib/services/forms.service.mjs +47 -0
- package/esm2022/lib/services/labels.service.mjs +29 -0
- package/esm2022/lib/services/language.service.mjs +24 -0
- package/esm2022/lib/services/open-banking.service.mjs +194 -0
- package/esm2022/lib/services/origin-form-signalr-handler.service.mjs +107 -0
- package/esm2022/lib/services/origin-form-signalr.service.mjs +105 -0
- package/esm2022/lib/services/otp.service.mjs +28 -0
- package/esm2022/lib/services/proxy.service.mjs +79 -0
- package/esm2022/lib/services/scroll-to-error.service.mjs +369 -0
- package/esm2022/lib/services/translation.service.mjs +27 -0
- package/esm2022/lib/shared-components/confirmation.component.mjs +34 -0
- package/esm2022/lib/shared-components/dictionaries-tree.component.mjs +301 -0
- package/esm2022/lib/shared-components/grid.component.mjs +241 -0
- package/esm2022/lib/shared-components/treeview/treeview.component.mjs +224 -0
- package/esm2022/lib/theme-css.mjs +2254 -0
- package/esm2022/lib/theme-injector.service.mjs +26 -0
- package/esm2022/public-api.mjs +4 -0
- package/esm2022/qbs-origin-origin-form.mjs +5 -0
- package/fesm2022/qbs-origin-origin-form.mjs +15215 -0
- package/fesm2022/qbs-origin-origin-form.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/action-step-handler.d.ts +49 -0
- package/lib/auth-client.service.d.ts +17 -0
- package/lib/enums/label.keys.d.ts +720 -0
- package/lib/form-css.helper.d.ts +28 -0
- package/lib/formly/baseFormlyControlComponent.d.ts +25 -0
- package/lib/formly/baseFormlyStepComponent.d.ts +29 -0
- package/lib/formly/custom-section-separator.component.d.ts +6 -0
- package/lib/formly/form-section-separator.component.d.ts +10 -0
- package/lib/formly/formly-action.d.ts +13 -0
- package/lib/formly/formly-checkbox/formly-checkbox.component.d.ts +15 -0
- package/lib/formly/formly-dictionary-dropdown-tree/formly-dictionary-dropdown-tree.component.d.ts +45 -0
- package/lib/formly/formly-download-documents/formly-download-documents.component.d.ts +22 -0
- package/lib/formly/formly-enrol-card/formly-enrol-card.component.d.ts +114 -0
- package/lib/formly/formly-field-stepper/formly-field-stepper.component.d.ts +79 -0
- package/lib/formly/formly-generate-documents/formly-generate-documents.component.d.ts +17 -0
- package/lib/formly/formly-identification.component.d.ts +19 -0
- package/lib/formly/formly-open-banking/formly-open-banking.component.d.ts +119 -0
- package/lib/formly/formly-paragraph/formly-paragraph.component.d.ts +10 -0
- package/lib/formly/formly-radio/formly-radio-component.d.ts +15 -0
- package/lib/formly/formly-row-fille.d.ts +6 -0
- package/lib/formly/formly-scan-id/formly-scan-id.component.d.ts +41 -0
- package/lib/formly/formly-sign/formly-sign.component.d.ts +36 -0
- package/lib/formly/formly-upload-documents/formly-upload-documents.component.d.ts +25 -0
- package/lib/formly/formly-validate-contact-info/formly-validate-contact-info.component.d.ts +79 -0
- package/lib/formly/formly-view-documents/formly-view-documents.component.d.ts +33 -0
- package/lib/formly/formly-view-offers/formly-view-offers.component.d.ts +23 -0
- package/lib/model-population.helper.d.ts +8 -0
- package/lib/models/application-type.model.d.ts +27 -0
- package/lib/models/application.model.d.ts +107 -0
- package/lib/models/auth/users.model.d.ts +20 -0
- package/lib/models/dictionary.model.d.ts +77 -0
- package/lib/models/flux.model.d.ts +101 -0
- package/lib/models/forms.model.d.ts +504 -0
- package/lib/models/label-info.model.d.ts +10 -0
- package/lib/models/label.model.d.ts +4 -0
- package/lib/models/language.model.d.ts +5 -0
- package/lib/models/list.model.d.ts +8 -0
- package/lib/models/partner.model.d.ts +12 -0
- package/lib/models/treeview.model.d.ts +17 -0
- package/lib/origin-form-auth.service.d.ts +15 -0
- package/lib/origin-form-config.model.d.ts +12 -0
- package/lib/origin-form-token.interceptor.d.ts +12 -0
- package/lib/origin-form.component.d.ts +231 -0
- package/lib/origin-form.module.d.ts +84 -0
- package/lib/origin-form.service.d.ts +6 -0
- package/lib/others/check-list.database.d.ts +16 -0
- package/lib/others/config-service.d.ts +22 -0
- package/lib/others/dictionary-label-info.d.ts +6 -0
- package/lib/others/environment-type.d.ts +8 -0
- package/lib/others/external-link.directive.d.ts +12 -0
- package/lib/others/flux-helper.d.ts +115 -0
- package/lib/others/picker.component.d.ts +36 -0
- package/lib/others/translation.pipe.d.ts +10 -0
- package/lib/others/translations-helper.d.ts +31 -0
- package/lib/others/utils.d.ts +37 -0
- package/lib/services/applicationData.service.d.ts +35 -0
- package/lib/services/auth-http.service.d.ts +21 -0
- package/lib/services/dialog.service.d.ts +20 -0
- package/lib/services/dictionary.service.d.ts +89 -0
- package/lib/services/forms.service.d.ts +17 -0
- package/lib/services/labels.service.d.ts +13 -0
- package/lib/services/language.service.d.ts +14 -0
- package/lib/services/open-banking.service.d.ts +137 -0
- package/lib/services/origin-form-signalr-handler.service.d.ts +29 -0
- package/lib/services/origin-form-signalr.service.d.ts +24 -0
- package/lib/services/otp.service.d.ts +22 -0
- package/lib/services/proxy.service.d.ts +29 -0
- package/lib/services/scroll-to-error.service.d.ts +54 -0
- package/lib/services/translation.service.d.ts +10 -0
- package/lib/shared-components/confirmation.component.d.ts +77 -0
- package/lib/shared-components/dictionaries-tree.component.d.ts +51 -0
- package/lib/shared-components/grid.component.d.ts +138 -0
- package/lib/shared-components/treeview/treeview.component.d.ts +121 -0
- package/lib/theme-css.d.ts +2 -0
- package/lib/theme-injector.service.d.ts +8 -0
- package/package.json +42 -0
- package/public-api.d.ts +3 -0
- package/schematics-compiled/collection.json +10 -0
- package/schematics-compiled/ng-add/index.d.ts +2 -0
- package/schematics-compiled/ng-add/index.js +67 -0
- package/schematics-compiled/ng-add/index.js.map +1 -0
- package/schematics-compiled/ng-add/schema.json +8 -0
- package/src/lib/assets/fonts/Figtree-Bold.ttf +0 -0
- package/src/lib/assets/fonts/Figtree-Light.ttf +0 -0
- package/src/lib/assets/fonts/Figtree-Regular.ttf +0 -0
- package/src/lib/assets/fonts/Sora-ExtraBold.ttf +0 -0
- package/src/lib/assets/fonts/Sora-Light.ttf +0 -0
- package/src/lib/assets/fonts/Sora-Regular.ttf +0 -0
- package/src/lib/assets/fonts/ttrounds-bold-webfont.woff +0 -0
- package/src/lib/assets/fonts/ttrounds-bold-webfont.woff2 +0 -0
- package/src/lib/assets/fonts/ttrounds-regular-webfont.woff +0 -0
- package/src/lib/assets/fonts/ttrounds-regular-webfont.woff2 +0 -0
- package/src/lib/assets/fonts/ttrounds-thin-webfont.woff +0 -0
- package/src/lib/assets/fonts/ttrounds-thin-webfont.woff2 +0 -0
- package/src/lib/assets/images/flag/icon-flag-de.svg +10 -0
- package/src/lib/assets/images/flag/icon-flag-en.svg +1 -0
- package/src/lib/assets/images/flag/icon-flag-es.svg +11 -0
- package/src/lib/assets/images/flag/icon-flag-fr.svg +1 -0
- package/src/lib/assets/images/flag/icon-flag-ro.svg +11 -0
- package/src/lib/assets/images/flag/origin-form/new-id-card.png +0 -0
- package/src/lib/assets/images/flag/origin-form/old-id-card.png +0 -0
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { BaseFormlyStepComponent } from '../baseFormlyStepComponent';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
import * as i2 from "../../others/external-link.directive";
|
|
6
|
+
export class FormlyParagraphComponent extends BaseFormlyStepComponent {
|
|
7
|
+
onInit() {
|
|
8
|
+
this.cssMargins = {
|
|
9
|
+
'margin-top': this.props?.['design']?.['.component-text-field']?.['margin-top'],
|
|
10
|
+
'margin-right': this.props?.['design']?.['.component-text-field']?.['margin-right'],
|
|
11
|
+
'margin-bottom': this.props?.['design']?.['.component-text-field']?.['margin-bottom'],
|
|
12
|
+
'margin-left': this.props?.['design']?.['.component-text-field']?.['margin-left'],
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
get formattedLabel() {
|
|
16
|
+
return this.formatDateIfNeeded(this.field.defaultValue).toString();
|
|
17
|
+
}
|
|
18
|
+
formatDateIfNeeded(value) {
|
|
19
|
+
if (!value)
|
|
20
|
+
return '';
|
|
21
|
+
let cleanValue = value.toString().trim();
|
|
22
|
+
// cast eg: "Fri Feb 14 2025 00:00:00 GMT+0200" → "2025-02-14"
|
|
23
|
+
if (/^\w{3} \w{3} \d{1,2} \d{4} \d{2}:\d{2}:\d{2} GMT[\+\-]\d{4}/.test(cleanValue)) {
|
|
24
|
+
return new Date(cleanValue).toISOString().split('T')[0];
|
|
25
|
+
}
|
|
26
|
+
return cleanValue;
|
|
27
|
+
}
|
|
28
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FormlyParagraphComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
|
|
29
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FormlyParagraphComponent, selector: "formly-paragraph", usesInheritance: true, ngImport: i0, template: "<div\n class=\"mat-mdc-text-field-wrapper formly-paragraph\"\n [class.hasLabel]=\"title\"\n [ngStyle]=\"cssMargins\"\n appExternalLink\n>\n\n <ng-container *ngIf=\"props['controlType'] === 'Paragraph'\">\n <p *ngIf=\"props['translatedParagraphTitle']\" [ngStyle]=\"props['design']['.font-label']\">\n {{ props['translatedParagraphTitle'] }}\n </p>\n <p *ngIf=\"props['translatedDescription']\" [ngStyle]=\"props['design']['.font-paragraph']\">\n {{ props['translatedDescription'] }}\n </p>\n </ng-container>\n\n <ng-container *ngIf=\"props['controlType'] !== 'Paragraph'\">\n <div\n class=\"component-text-field\"\n *ngIf=\"field?.defaultValue !== undefined && field?.defaultValue !== null\"\n >\n <p [ngStyle]=\"props['design']['.font-label']\">\n {{ field.props?.label }}\n </p>\n <p [ngStyle]=\"props['design']['.font-paragraph']\">\n {{ field.defaultValue }}\n </p>\n </div>\n </ng-container>\n</div>\n", styles: [".formly-paragraph{border:none!important;background:none!important;padding:0;display:block}.formly-paragraph label{display:block}.formly-paragraph.hasLabel p{margin-top:.5rem}.mat-mdc-text-field-wrapper.formly-paragraph{padding:8px 16px}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i2.ExternalLinkDirective, selector: "[appExternalLink]" }] }); }
|
|
30
|
+
}
|
|
31
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FormlyParagraphComponent, decorators: [{
|
|
32
|
+
type: Component,
|
|
33
|
+
args: [{ selector: 'formly-paragraph', template: "<div\n class=\"mat-mdc-text-field-wrapper formly-paragraph\"\n [class.hasLabel]=\"title\"\n [ngStyle]=\"cssMargins\"\n appExternalLink\n>\n\n <ng-container *ngIf=\"props['controlType'] === 'Paragraph'\">\n <p *ngIf=\"props['translatedParagraphTitle']\" [ngStyle]=\"props['design']['.font-label']\">\n {{ props['translatedParagraphTitle'] }}\n </p>\n <p *ngIf=\"props['translatedDescription']\" [ngStyle]=\"props['design']['.font-paragraph']\">\n {{ props['translatedDescription'] }}\n </p>\n </ng-container>\n\n <ng-container *ngIf=\"props['controlType'] !== 'Paragraph'\">\n <div\n class=\"component-text-field\"\n *ngIf=\"field?.defaultValue !== undefined && field?.defaultValue !== null\"\n >\n <p [ngStyle]=\"props['design']['.font-label']\">\n {{ field.props?.label }}\n </p>\n <p [ngStyle]=\"props['design']['.font-paragraph']\">\n {{ field.defaultValue }}\n </p>\n </div>\n </ng-container>\n</div>\n", styles: [".formly-paragraph{border:none!important;background:none!important;padding:0;display:block}.formly-paragraph label{display:block}.formly-paragraph.hasLabel p{margin-top:.5rem}.mat-mdc-text-field-wrapper.formly-paragraph{padding:8px 16px}\n"] }]
|
|
34
|
+
}] });
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybWx5LXBhcmFncmFwaC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9vcmlnaW4tZm9ybS9zcmMvbGliL2Zvcm1seS9mb3JtbHktcGFyYWdyYXBoL2Zvcm1seS1wYXJhZ3JhcGguY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvb3JpZ2luLWZvcm0vc3JjL2xpYi9mb3JtbHkvZm9ybWx5LXBhcmFncmFwaC9mb3JtbHktcGFyYWdyYXBoLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUMsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNEJBQTRCLENBQUM7Ozs7QUFPckUsTUFBTSxPQUFPLHdCQUF5QixTQUFRLHVCQUF1QjtJQUcxRCxNQUFNO1FBQ2IsSUFBSSxDQUFDLFVBQVUsR0FBRztZQUNoQixZQUFZLEVBQ1YsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsdUJBQXVCLENBQUMsRUFBRSxDQUFDLFlBQVksQ0FBQztZQUNuRSxjQUFjLEVBQ1osSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsdUJBQXVCLENBQUMsRUFBRSxDQUFDLGNBQWMsQ0FBQztZQUNyRSxlQUFlLEVBQ2IsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsdUJBQXVCLENBQUMsRUFBRSxDQUFDLGVBQWUsQ0FBQztZQUN0RSxhQUFhLEVBQ1gsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsdUJBQXVCLENBQUMsRUFBRSxDQUFDLGFBQWEsQ0FBQztTQUNyRSxDQUFDO0lBQ0osQ0FBQztJQUVELElBQUksY0FBYztRQUNoQixPQUFPLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFlBQVksQ0FBQyxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQ3JFLENBQUM7SUFFTyxrQkFBa0IsQ0FBQyxLQUFVO1FBQ25DLElBQUksQ0FBQyxLQUFLO1lBQUUsT0FBTyxFQUFFLENBQUM7UUFFdEIsSUFBSSxVQUFVLEdBQUcsS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDLElBQUksRUFBRSxDQUFDO1FBRXpDLDhEQUE4RDtRQUM5RCxJQUNFLDZEQUE2RCxDQUFDLElBQUksQ0FDaEUsVUFBVSxDQUNYLEVBQ0QsQ0FBQztZQUNELE9BQU8sSUFBSSxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUMsV0FBVyxFQUFFLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzFELENBQUM7UUFFRCxPQUFPLFVBQVUsQ0FBQztJQUNwQixDQUFDOytHQW5DVSx3QkFBd0I7bUdBQXhCLHdCQUF3QiwrRUNSckMsKzlCQThCQTs7NEZEdEJhLHdCQUF3QjtrQkFMcEMsU0FBUzsrQkFDRSxrQkFBa0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEJhc2VGb3JtbHlTdGVwQ29tcG9uZW50IH0gZnJvbSAnLi4vYmFzZUZvcm1seVN0ZXBDb21wb25lbnQnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdmb3JtbHktcGFyYWdyYXBoJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2Zvcm1seS1wYXJhZ3JhcGguY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybDogJy4vZm9ybWx5LXBhcmFncmFwaC5jb21wb25lbnQuc2NzcycsXG59KVxuZXhwb3J0IGNsYXNzIEZvcm1seVBhcmFncmFwaENvbXBvbmVudCBleHRlbmRzIEJhc2VGb3JtbHlTdGVwQ29tcG9uZW50IHtcbiAgY3NzTWFyZ2luczogYW55O1xuXG4gIG92ZXJyaWRlIG9uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLmNzc01hcmdpbnMgPSB7XG4gICAgICAnbWFyZ2luLXRvcCc6XG4gICAgICAgIHRoaXMucHJvcHM/LlsnZGVzaWduJ10/LlsnLmNvbXBvbmVudC10ZXh0LWZpZWxkJ10/LlsnbWFyZ2luLXRvcCddLFxuICAgICAgJ21hcmdpbi1yaWdodCc6XG4gICAgICAgIHRoaXMucHJvcHM/LlsnZGVzaWduJ10/LlsnLmNvbXBvbmVudC10ZXh0LWZpZWxkJ10/LlsnbWFyZ2luLXJpZ2h0J10sXG4gICAgICAnbWFyZ2luLWJvdHRvbSc6XG4gICAgICAgIHRoaXMucHJvcHM/LlsnZGVzaWduJ10/LlsnLmNvbXBvbmVudC10ZXh0LWZpZWxkJ10/LlsnbWFyZ2luLWJvdHRvbSddLFxuICAgICAgJ21hcmdpbi1sZWZ0JzpcbiAgICAgICAgdGhpcy5wcm9wcz8uWydkZXNpZ24nXT8uWycuY29tcG9uZW50LXRleHQtZmllbGQnXT8uWydtYXJnaW4tbGVmdCddLFxuICAgIH07XG4gIH1cblxuICBnZXQgZm9ybWF0dGVkTGFiZWwoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdGhpcy5mb3JtYXREYXRlSWZOZWVkZWQodGhpcy5maWVsZC5kZWZhdWx0VmFsdWUpLnRvU3RyaW5nKCk7XG4gIH1cblxuICBwcml2YXRlIGZvcm1hdERhdGVJZk5lZWRlZCh2YWx1ZTogYW55KTogc3RyaW5nIHtcbiAgICBpZiAoIXZhbHVlKSByZXR1cm4gJyc7XG5cbiAgICBsZXQgY2xlYW5WYWx1ZSA9IHZhbHVlLnRvU3RyaW5nKCkudHJpbSgpO1xuXG4gICAgLy8gY2FzdCBlZzogXCJGcmkgRmViIDE0IDIwMjUgMDA6MDA6MDAgR01UKzAyMDBcIiDihpIgXCIyMDI1LTAyLTE0XCJcbiAgICBpZiAoXG4gICAgICAvXlxcd3szfSBcXHd7M30gXFxkezEsMn0gXFxkezR9IFxcZHsyfTpcXGR7Mn06XFxkezJ9IEdNVFtcXCtcXC1dXFxkezR9Ly50ZXN0KFxuICAgICAgICBjbGVhblZhbHVlXG4gICAgICApXG4gICAgKSB7XG4gICAgICByZXR1cm4gbmV3IERhdGUoY2xlYW5WYWx1ZSkudG9JU09TdHJpbmcoKS5zcGxpdCgnVCcpWzBdO1xuICAgIH1cblxuICAgIHJldHVybiBjbGVhblZhbHVlO1xuICB9XG59XG4iLCI8ZGl2XG4gIGNsYXNzPVwibWF0LW1kYy10ZXh0LWZpZWxkLXdyYXBwZXIgZm9ybWx5LXBhcmFncmFwaFwiXG4gIFtjbGFzcy5oYXNMYWJlbF09XCJ0aXRsZVwiXG4gIFtuZ1N0eWxlXT1cImNzc01hcmdpbnNcIlxuICBhcHBFeHRlcm5hbExpbmtcbj5cblxuICA8bmctY29udGFpbmVyICpuZ0lmPVwicHJvcHNbJ2NvbnRyb2xUeXBlJ10gPT09ICdQYXJhZ3JhcGgnXCI+XG4gICAgPHAgKm5nSWY9XCJwcm9wc1sndHJhbnNsYXRlZFBhcmFncmFwaFRpdGxlJ11cIiBbbmdTdHlsZV09XCJwcm9wc1snZGVzaWduJ11bJy5mb250LWxhYmVsJ11cIj5cbiAgICAgIHt7IHByb3BzWyd0cmFuc2xhdGVkUGFyYWdyYXBoVGl0bGUnXSB9fVxuICAgIDwvcD5cbiAgICA8cCAqbmdJZj1cInByb3BzWyd0cmFuc2xhdGVkRGVzY3JpcHRpb24nXVwiIFtuZ1N0eWxlXT1cInByb3BzWydkZXNpZ24nXVsnLmZvbnQtcGFyYWdyYXBoJ11cIj5cbiAgICAgIHt7IHByb3BzWyd0cmFuc2xhdGVkRGVzY3JpcHRpb24nXSB9fVxuICAgIDwvcD5cbiAgPC9uZy1jb250YWluZXI+XG5cbiAgPG5nLWNvbnRhaW5lciAqbmdJZj1cInByb3BzWydjb250cm9sVHlwZSddICE9PSAnUGFyYWdyYXBoJ1wiPlxuICAgIDxkaXZcbiAgICAgIGNsYXNzPVwiY29tcG9uZW50LXRleHQtZmllbGRcIlxuICAgICAgKm5nSWY9XCJmaWVsZD8uZGVmYXVsdFZhbHVlICE9PSB1bmRlZmluZWQgJiYgZmllbGQ/LmRlZmF1bHRWYWx1ZSAhPT0gbnVsbFwiXG4gICAgPlxuICAgICAgPHAgW25nU3R5bGVdPVwicHJvcHNbJ2Rlc2lnbiddWycuZm9udC1sYWJlbCddXCI+XG4gICAgICAgIHt7IGZpZWxkLnByb3BzPy5sYWJlbCB9fVxuICAgICAgPC9wPlxuICAgICAgPHAgW25nU3R5bGVdPVwicHJvcHNbJ2Rlc2lnbiddWycuZm9udC1wYXJhZ3JhcGgnXVwiPlxuICAgICAgICB7eyBmaWVsZC5kZWZhdWx0VmFsdWUgfX1cbiAgICAgIDwvcD5cbiAgICA8L2Rpdj5cbiAgPC9uZy1jb250YWluZXI+XG48L2Rpdj5cbiJdfQ==
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { FieldType } from '@ngx-formly/core';
|
|
3
|
+
import { FormControl } from '@angular/forms';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/common";
|
|
6
|
+
import * as i2 from "ngx-markdown";
|
|
7
|
+
import * as i3 from "@angular/material/radio";
|
|
8
|
+
import * as i4 from "@ngx-formly/core";
|
|
9
|
+
import * as i5 from "../../others/external-link.directive";
|
|
10
|
+
export class FormlyRadioComponent extends FieldType {
|
|
11
|
+
constructor() {
|
|
12
|
+
super(...arguments);
|
|
13
|
+
this.radioOptionsList = [];
|
|
14
|
+
this.title = '';
|
|
15
|
+
this.langIso = 'en';
|
|
16
|
+
}
|
|
17
|
+
ngOnInit() {
|
|
18
|
+
this.langIso = this.props['currentLanguageIso'];
|
|
19
|
+
this.title = this.props['label'] || '';
|
|
20
|
+
if (Array.isArray(this.to.options)) {
|
|
21
|
+
this.radioOptionsList = this.to.options;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
get formControl() {
|
|
25
|
+
if (this.field.key) {
|
|
26
|
+
const keyPath = typeof this.field.key === 'string' || Array.isArray(this.field.key)
|
|
27
|
+
? this.field.key
|
|
28
|
+
: String(this.field.key);
|
|
29
|
+
return this.form.get(keyPath);
|
|
30
|
+
}
|
|
31
|
+
return new FormControl();
|
|
32
|
+
}
|
|
33
|
+
onRadioChange(e) {
|
|
34
|
+
const selected = this.radioOptionsList.find((opt) => opt.value.id === e.value);
|
|
35
|
+
if (selected) {
|
|
36
|
+
this.formControl.setValue(selected.value);
|
|
37
|
+
if (this.props['onSelectedValueChanged']) {
|
|
38
|
+
this.props['onSelectedValueChanged'](selected.value);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FormlyRadioComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
|
|
43
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FormlyRadioComponent, selector: "formly-radio", usesInheritance: true, ngImport: i0, template: "<div class=\"mat-mdc-text-field-wrapper mdc-radiobox-field\" appExternalLink>\n <label>\n <markdown [data]=\"title\"></markdown>\n </label>\n <div *ngIf=\"radioOptionsList.length > 0\">\n <mat-radio-group [value]=\"formControl.value?.id\"\n (change)=\"onRadioChange($event)\">\n <ng-container *ngFor=\"let option of radioOptionsList\">\n <div>\n <mat-radio-button [value]=\"option.value.id\">\n {{ option.label }}\n </mat-radio-button>\n </div>\n </ng-container>\n </mat-radio-group>\n </div>\n <formly-validation-message *ngIf=\"showError\" [field]=\"field\">\n </formly-validation-message>\n</div>\n", styles: ["formly-validation-message{color:red;font-size:12px}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.MarkdownComponent, selector: "markdown, [markdown]", inputs: ["data", "src", "disableSanitizer", "inline", "clipboard", "clipboardButtonComponent", "clipboardButtonTemplate", "emoji", "katex", "katexOptions", "mermaid", "mermaidOptions", "lineHighlight", "line", "lineOffset", "lineNumbers", "start", "commandLine", "filterOutput", "host", "prompt", "output", "user"], outputs: ["error", "load", "ready"] }, { kind: "directive", type: i3.MatRadioGroup, selector: "mat-radio-group", inputs: ["color", "name", "labelPosition", "value", "selected", "disabled", "required", "disabledInteractive"], outputs: ["change"], exportAs: ["matRadioGroup"] }, { kind: "component", type: i3.MatRadioButton, selector: "mat-radio-button", inputs: ["id", "name", "aria-label", "aria-labelledby", "aria-describedby", "disableRipple", "tabIndex", "checked", "value", "labelPosition", "disabled", "required", "color", "disabledInteractive"], outputs: ["change"], exportAs: ["matRadioButton"] }, { kind: "component", type: i4.ɵFormlyValidationMessage, selector: "formly-validation-message", inputs: ["field"] }, { kind: "directive", type: i5.ExternalLinkDirective, selector: "[appExternalLink]" }] }); }
|
|
44
|
+
}
|
|
45
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FormlyRadioComponent, decorators: [{
|
|
46
|
+
type: Component,
|
|
47
|
+
args: [{ selector: 'formly-radio', template: "<div class=\"mat-mdc-text-field-wrapper mdc-radiobox-field\" appExternalLink>\n <label>\n <markdown [data]=\"title\"></markdown>\n </label>\n <div *ngIf=\"radioOptionsList.length > 0\">\n <mat-radio-group [value]=\"formControl.value?.id\"\n (change)=\"onRadioChange($event)\">\n <ng-container *ngFor=\"let option of radioOptionsList\">\n <div>\n <mat-radio-button [value]=\"option.value.id\">\n {{ option.label }}\n </mat-radio-button>\n </div>\n </ng-container>\n </mat-radio-group>\n </div>\n <formly-validation-message *ngIf=\"showError\" [field]=\"field\">\n </formly-validation-message>\n</div>\n", styles: ["formly-validation-message{color:red;font-size:12px}\n"] }]
|
|
48
|
+
}] });
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybWx5LXJhZGlvLWNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL29yaWdpbi1mb3JtL3NyYy9saWIvZm9ybWx5L2Zvcm1seS1yYWRpby9mb3JtbHktcmFkaW8tY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvb3JpZ2luLWZvcm0vc3JjL2xpYi9mb3JtbHkvZm9ybWx5LXJhZGlvL2Zvcm1seS1yYWRpby1jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQ2xELE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUM3QyxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7Ozs7Ozs7QUFTN0MsTUFBTSxPQUFPLG9CQUFxQixTQUFRLFNBQVM7SUFObkQ7O1FBT0UscUJBQWdCLEdBQVUsRUFBRSxDQUFDO1FBQzdCLFVBQUssR0FBVyxFQUFFLENBQUM7UUFDbkIsWUFBTyxHQUFXLElBQUksQ0FBQztLQWdDeEI7SUE5QkMsUUFBUTtRQUNOLElBQUksQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO1FBQ2hELElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDdkMsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNuQyxJQUFJLENBQUMsZ0JBQWdCLEdBQUcsSUFBSSxDQUFDLEVBQUUsQ0FBQyxPQUFPLENBQUM7UUFDMUMsQ0FBQztJQUNILENBQUM7SUFFRCxJQUFhLFdBQVc7UUFDdEIsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsRUFBRSxDQUFDO1lBQ25CLE1BQU0sT0FBTyxHQUNYLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLEtBQUssUUFBUSxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUM7Z0JBQ2pFLENBQUMsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUc7Z0JBQ2hCLENBQUMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztZQUM3QixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBZ0IsQ0FBQztRQUMvQyxDQUFDO1FBQ0QsT0FBTyxJQUFJLFdBQVcsRUFBRSxDQUFDO0lBQzNCLENBQUM7SUFFRCxhQUFhLENBQUMsQ0FBaUI7UUFDN0IsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FDekMsQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsRUFBRSxLQUFLLENBQUMsQ0FBQyxLQUFLLENBQ2xDLENBQUM7UUFDRixJQUFJLFFBQVEsRUFBRSxDQUFDO1lBQ2IsSUFBSSxDQUFDLFdBQVcsQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQzFDLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyx3QkFBd0IsQ0FBQyxFQUFFLENBQUM7Z0JBQ3pDLElBQUksQ0FBQyxLQUFLLENBQUMsd0JBQXdCLENBQUMsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDdkQsQ0FBQztRQUNILENBQUM7SUFDSCxDQUFDOytHQWxDVSxvQkFBb0I7bUdBQXBCLG9CQUFvQiwyRUNYakMsbXJCQW1CQTs7NEZEUmEsb0JBQW9CO2tCQU5oQyxTQUFTOytCQUNFLGNBQWMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIE9uSW5pdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRmllbGRUeXBlIH0gZnJvbSAnQG5neC1mb3JtbHkvY29yZSc7XG5pbXBvcnQgeyBGb3JtQ29udHJvbCB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IE1hdFJhZGlvQ2hhbmdlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvcmFkaW8nO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdmb3JtbHktcmFkaW8nLFxuICB0ZW1wbGF0ZVVybDogJy4vZm9ybWx5LXJhZGlvLWNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vZm9ybWx5LXJhZGlvLWNvbXBvbmVudC5zY3NzJ10sXG4gIHN0eWxlczogW10sXG59KVxuZXhwb3J0IGNsYXNzIEZvcm1seVJhZGlvQ29tcG9uZW50IGV4dGVuZHMgRmllbGRUeXBlIGltcGxlbWVudHMgT25Jbml0IHtcbiAgcmFkaW9PcHRpb25zTGlzdDogYW55W10gPSBbXTtcbiAgdGl0bGU6IHN0cmluZyA9ICcnO1xuICBsYW5nSXNvOiBzdHJpbmcgPSAnZW4nO1xuXG4gIG5nT25Jbml0KCk6IHZvaWQge1xuICAgIHRoaXMubGFuZ0lzbyA9IHRoaXMucHJvcHNbJ2N1cnJlbnRMYW5ndWFnZUlzbyddO1xuICAgIHRoaXMudGl0bGUgPSB0aGlzLnByb3BzWydsYWJlbCddIHx8ICcnO1xuICAgIGlmIChBcnJheS5pc0FycmF5KHRoaXMudG8ub3B0aW9ucykpIHtcbiAgICAgIHRoaXMucmFkaW9PcHRpb25zTGlzdCA9IHRoaXMudG8ub3B0aW9ucztcbiAgICB9XG4gIH1cblxuICBvdmVycmlkZSBnZXQgZm9ybUNvbnRyb2woKTogRm9ybUNvbnRyb2wge1xuICAgIGlmICh0aGlzLmZpZWxkLmtleSkge1xuICAgICAgY29uc3Qga2V5UGF0aCA9XG4gICAgICAgIHR5cGVvZiB0aGlzLmZpZWxkLmtleSA9PT0gJ3N0cmluZycgfHwgQXJyYXkuaXNBcnJheSh0aGlzLmZpZWxkLmtleSlcbiAgICAgICAgICA/IHRoaXMuZmllbGQua2V5XG4gICAgICAgICAgOiBTdHJpbmcodGhpcy5maWVsZC5rZXkpO1xuICAgICAgcmV0dXJuIHRoaXMuZm9ybS5nZXQoa2V5UGF0aCkgYXMgRm9ybUNvbnRyb2w7XG4gICAgfVxuICAgIHJldHVybiBuZXcgRm9ybUNvbnRyb2woKTtcbiAgfVxuXG4gIG9uUmFkaW9DaGFuZ2UoZTogTWF0UmFkaW9DaGFuZ2UpOiB2b2lkIHtcbiAgICBjb25zdCBzZWxlY3RlZCA9IHRoaXMucmFkaW9PcHRpb25zTGlzdC5maW5kKFxuICAgICAgKG9wdCkgPT4gb3B0LnZhbHVlLmlkID09PSBlLnZhbHVlXG4gICAgKTtcbiAgICBpZiAoc2VsZWN0ZWQpIHtcbiAgICAgIHRoaXMuZm9ybUNvbnRyb2wuc2V0VmFsdWUoc2VsZWN0ZWQudmFsdWUpO1xuICAgICAgaWYgKHRoaXMucHJvcHNbJ29uU2VsZWN0ZWRWYWx1ZUNoYW5nZWQnXSkge1xuICAgICAgICB0aGlzLnByb3BzWydvblNlbGVjdGVkVmFsdWVDaGFuZ2VkJ10oc2VsZWN0ZWQudmFsdWUpO1xuICAgICAgfVxuICAgIH1cbiAgfVxufVxuIiwiPGRpdiBjbGFzcz1cIm1hdC1tZGMtdGV4dC1maWVsZC13cmFwcGVyIG1kYy1yYWRpb2JveC1maWVsZFwiIGFwcEV4dGVybmFsTGluaz5cbiAgPGxhYmVsPlxuICAgIDxtYXJrZG93biBbZGF0YV09XCJ0aXRsZVwiPjwvbWFya2Rvd24+XG4gIDwvbGFiZWw+XG4gIDxkaXYgKm5nSWY9XCJyYWRpb09wdGlvbnNMaXN0Lmxlbmd0aCA+IDBcIj5cbiAgICA8bWF0LXJhZGlvLWdyb3VwIFt2YWx1ZV09XCJmb3JtQ29udHJvbC52YWx1ZT8uaWRcIlxuICAgICAgICAgICAgICAgICAgICAgKGNoYW5nZSk9XCJvblJhZGlvQ2hhbmdlKCRldmVudClcIj5cbiAgICAgIDxuZy1jb250YWluZXIgKm5nRm9yPVwibGV0IG9wdGlvbiBvZiByYWRpb09wdGlvbnNMaXN0XCI+XG4gICAgICAgIDxkaXY+XG4gICAgICAgICAgPG1hdC1yYWRpby1idXR0b24gW3ZhbHVlXT1cIm9wdGlvbi52YWx1ZS5pZFwiPlxuICAgICAgICAgICAge3sgb3B0aW9uLmxhYmVsIH19XG4gICAgICAgICAgPC9tYXQtcmFkaW8tYnV0dG9uPlxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvbmctY29udGFpbmVyPlxuICAgIDwvbWF0LXJhZGlvLWdyb3VwPlxuICA8L2Rpdj5cbiAgPGZvcm1seS12YWxpZGF0aW9uLW1lc3NhZ2UgKm5nSWY9XCJzaG93RXJyb3JcIiBbZmllbGRdPVwiZmllbGRcIj5cbiAgPC9mb3JtbHktdmFsaWRhdGlvbi1tZXNzYWdlPlxuPC9kaXY+XG4iXX0=
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { FieldType } from '@ngx-formly/core';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class RowFillerComponent extends FieldType {
|
|
5
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RowFillerComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
|
|
6
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RowFillerComponent, selector: "formly-row-filler", usesInheritance: true, ngImport: i0, template: `<div class="row-filler"></div>`, isInline: true, styles: [".row-filler{width:100%}\n"] }); }
|
|
7
|
+
}
|
|
8
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RowFillerComponent, decorators: [{
|
|
9
|
+
type: Component,
|
|
10
|
+
args: [{ selector: 'formly-row-filler', template: `<div class="row-filler"></div>`, styles: [".row-filler{width:100%}\n"] }]
|
|
11
|
+
}] });
|
|
12
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybWx5LXJvdy1maWxsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL29yaWdpbi1mb3JtL3NyYy9saWIvZm9ybWx5L2Zvcm1seS1yb3ctZmlsbGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMxQyxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sa0JBQWtCLENBQUM7O0FBYTdDLE1BQU0sT0FBTyxrQkFBbUIsU0FBUSxTQUFTOytHQUFwQyxrQkFBa0I7bUdBQWxCLGtCQUFrQixnRkFUbkIsZ0NBQWdDOzs0RkFTL0Isa0JBQWtCO2tCQVg5QixTQUFTOytCQUNFLG1CQUFtQixZQUNuQixnQ0FBZ0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZpZWxkVHlwZSB9IGZyb20gJ0BuZ3gtZm9ybWx5L2NvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdmb3JtbHktcm93LWZpbGxlcicsXG4gIHRlbXBsYXRlOiBgPGRpdiBjbGFzcz1cInJvdy1maWxsZXJcIj48L2Rpdj5gLFxuICBzdHlsZXM6IFtcbiAgICBgXG4gICAgICAucm93LWZpbGxlciB7XG4gICAgICAgIHdpZHRoOiAxMDAlO1xuICAgICAgfVxuICAgIGAsXG4gIF0sXG59KVxuZXhwb3J0IGNsYXNzIFJvd0ZpbGxlckNvbXBvbmVudCBleHRlbmRzIEZpZWxkVHlwZSB7fSJdfQ==
|
|
@@ -0,0 +1,284 @@
|
|
|
1
|
+
import { Component, ViewChild, } from '@angular/core';
|
|
2
|
+
import { FormControl, Validators } from '@angular/forms';
|
|
3
|
+
import { FieldType } from '@ngx-formly/core';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "../../services/applicationData.service";
|
|
6
|
+
import * as i2 from "@angular/common";
|
|
7
|
+
import * as i3 from "@angular/material/button";
|
|
8
|
+
import * as i4 from "@angular/material/icon";
|
|
9
|
+
import * as i5 from "@angular/material/progress-spinner";
|
|
10
|
+
export class FormlyScanIdComponent extends FieldType {
|
|
11
|
+
constructor(appDataService, cdr) {
|
|
12
|
+
super();
|
|
13
|
+
this.appDataService = appDataService;
|
|
14
|
+
this.cdr = cdr;
|
|
15
|
+
this.control = new FormControl('', [Validators.required]);
|
|
16
|
+
this.fileError = null;
|
|
17
|
+
this.fileNameFront = '';
|
|
18
|
+
this.fileNameBack = '';
|
|
19
|
+
this.validFormats = [];
|
|
20
|
+
this.maxFileSizeKB = 0;
|
|
21
|
+
this.cardTypeSelected = false;
|
|
22
|
+
this.oldIdCard = false;
|
|
23
|
+
this.frontPhotoUploaded = false;
|
|
24
|
+
this.backPhotoUploaded = false;
|
|
25
|
+
this.photoBase64Front = '';
|
|
26
|
+
this.photoBase64Back = '';
|
|
27
|
+
this.isLoading = false;
|
|
28
|
+
this.successMessage = null;
|
|
29
|
+
this.videoStream = null;
|
|
30
|
+
this.isScanValid = true;
|
|
31
|
+
}
|
|
32
|
+
ngOnInit() {
|
|
33
|
+
const group = this.form;
|
|
34
|
+
group.addControl(this.props['identifier'], this.control);
|
|
35
|
+
this.validFormats = this.props['config'].allowedFormats?.split(',') || [
|
|
36
|
+
'.jpg',
|
|
37
|
+
'.png',
|
|
38
|
+
];
|
|
39
|
+
this.maxFileSizeKB = this.props['config'].maxFileSizeKB || 2000;
|
|
40
|
+
if (this.props['config'].collected) {
|
|
41
|
+
const value = this.form.get(this.props['config'].componentCollected)?.value;
|
|
42
|
+
this.control.setValue(value);
|
|
43
|
+
this.form
|
|
44
|
+
.get(this.props['config'].componentCollected)
|
|
45
|
+
?.valueChanges.subscribe(() => {
|
|
46
|
+
this.control.setValue(this.form.get(this.props['config'].componentCollected)?.value);
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
selectCardType(isOldIdCard) {
|
|
51
|
+
this.oldIdCard = isOldIdCard;
|
|
52
|
+
this.cardTypeSelected = true;
|
|
53
|
+
this.resetForm();
|
|
54
|
+
}
|
|
55
|
+
onFileSelected(event, isFront) {
|
|
56
|
+
const file = event.target.files[0];
|
|
57
|
+
if (file) {
|
|
58
|
+
const fileExtension = '.' + file.name.split('.').pop()?.toLowerCase();
|
|
59
|
+
const fileSizeKB = file.size / 1024;
|
|
60
|
+
if (!this.validFormats.includes(fileExtension)) {
|
|
61
|
+
this.fileError = (this.props['errorMessages']?.invalidType ||
|
|
62
|
+
`Invalid file type. Allowed formats are ${this.validFormats.join(', ')}`).replace('{formats}', this.validFormats.join(', '));
|
|
63
|
+
this.control.setErrors({ invalidType: true });
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
if (fileSizeKB > this.maxFileSizeKB || fileSizeKB < 1) {
|
|
67
|
+
this.fileError = (this.props['errorMessages']?.maxSize ||
|
|
68
|
+
`File size must be between 1KB and ${this.maxFileSizeKB}KB.`).replace('{size}', this.maxFileSizeKB);
|
|
69
|
+
this.control.setErrors({ invalidSize: true });
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
this.fileError = null;
|
|
73
|
+
const reader = new FileReader();
|
|
74
|
+
reader.readAsDataURL(file);
|
|
75
|
+
reader.onload = () => {
|
|
76
|
+
if (isFront) {
|
|
77
|
+
this.fileNameFront = file.name;
|
|
78
|
+
this.photoBase64Front = reader.result?.toString().split(',')[1] || '';
|
|
79
|
+
this.frontPhotoUploaded = true;
|
|
80
|
+
this.cdr.detectChanges();
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
this.fileNameBack = file.name;
|
|
84
|
+
this.photoBase64Back = reader.result?.toString().split(',')[1] || '';
|
|
85
|
+
this.backPhotoUploaded = true;
|
|
86
|
+
this.cdr.detectChanges();
|
|
87
|
+
}
|
|
88
|
+
this.checkUploadCondition();
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
takePicture() {
|
|
93
|
+
this.resetForm();
|
|
94
|
+
if (navigator.mediaDevices && navigator.mediaDevices.enumerateDevices) {
|
|
95
|
+
navigator.mediaDevices
|
|
96
|
+
.enumerateDevices()
|
|
97
|
+
.then((devices) => {
|
|
98
|
+
const videoDevices = devices.filter((device) => device.kind === 'videoinput');
|
|
99
|
+
if (videoDevices.length === 0) {
|
|
100
|
+
this.fileError = 'No camera devices found.';
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
if (videoDevices.length === 1) {
|
|
104
|
+
this.startCamera(videoDevices[0].deviceId);
|
|
105
|
+
}
|
|
106
|
+
else {
|
|
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.';
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
this.fileError = 'Camera not supported by this device.';
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
startCamera(deviceId) {
|
|
126
|
+
navigator.mediaDevices
|
|
127
|
+
.getUserMedia({ video: { deviceId } })
|
|
128
|
+
.then((stream) => {
|
|
129
|
+
this.videoStream = stream;
|
|
130
|
+
this.videoElement.nativeElement.srcObject = stream;
|
|
131
|
+
this.videoElement.nativeElement.play();
|
|
132
|
+
})
|
|
133
|
+
.catch(() => {
|
|
134
|
+
this.fileError = 'Unable to access the selected camera.';
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
capturePhoto(isFront) {
|
|
138
|
+
if (!this.videoElement.nativeElement) {
|
|
139
|
+
this.fileError = 'No video element available.';
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
const video = this.videoElement.nativeElement;
|
|
143
|
+
const canvas = document.createElement('canvas');
|
|
144
|
+
canvas.width = video.videoWidth;
|
|
145
|
+
canvas.height = video.videoHeight;
|
|
146
|
+
const context = canvas.getContext('2d');
|
|
147
|
+
context?.drawImage(video, 0, 0, canvas.width, canvas.height);
|
|
148
|
+
this.isLoading = true;
|
|
149
|
+
canvas.toBlob((blob) => {
|
|
150
|
+
if (blob) {
|
|
151
|
+
const reader = new FileReader();
|
|
152
|
+
reader.readAsDataURL(blob);
|
|
153
|
+
reader.onloadend = () => {
|
|
154
|
+
if (isFront) {
|
|
155
|
+
this.photoBase64Front =
|
|
156
|
+
reader.result?.toString().split(',')[1] || '';
|
|
157
|
+
this.frontPhotoUploaded = true;
|
|
158
|
+
}
|
|
159
|
+
else {
|
|
160
|
+
this.photoBase64Back =
|
|
161
|
+
reader.result?.toString().split(',')[1] || '';
|
|
162
|
+
this.backPhotoUploaded = true;
|
|
163
|
+
}
|
|
164
|
+
this.isLoading = false;
|
|
165
|
+
this.checkUploadCondition();
|
|
166
|
+
this.cdr.detectChanges();
|
|
167
|
+
};
|
|
168
|
+
}
|
|
169
|
+
if (this.videoStream) {
|
|
170
|
+
this.videoStream.getTracks().forEach((track) => track.stop());
|
|
171
|
+
this.videoStream = null;
|
|
172
|
+
}
|
|
173
|
+
}, 'image/jpeg');
|
|
174
|
+
}
|
|
175
|
+
uploadPhoto() {
|
|
176
|
+
this.isLoading = true;
|
|
177
|
+
this.cdr.detectChanges();
|
|
178
|
+
const command = {
|
|
179
|
+
appId: this.props['appId'],
|
|
180
|
+
appDataId: this.props['appDataId'],
|
|
181
|
+
stepId: this.props['stepId'],
|
|
182
|
+
oldIdCard: this.oldIdCard,
|
|
183
|
+
photos: this.oldIdCard
|
|
184
|
+
? [{ base64: this.photoBase64Front }]
|
|
185
|
+
: [{ base64: this.photoBase64Front }, { base64: this.photoBase64Back }],
|
|
186
|
+
};
|
|
187
|
+
this.appDataService.scanId(command).subscribe({
|
|
188
|
+
next: async (result) => {
|
|
189
|
+
console.log('Scan ID result:', result);
|
|
190
|
+
this.isLoading = false;
|
|
191
|
+
var appData = await this.appDataService.getSteps(this.props['appDataId']);
|
|
192
|
+
if (appData) {
|
|
193
|
+
const status = this.extractStatusFromAppData(appData);
|
|
194
|
+
console.log('[SCAN-ID] extracted status:', status);
|
|
195
|
+
if (status?.toLowerCase().includes('invalid')) {
|
|
196
|
+
this.fileError =
|
|
197
|
+
this.props['errorMessages']?.invalidStatus ||
|
|
198
|
+
'Invalid status response.';
|
|
199
|
+
this.resetForm(true);
|
|
200
|
+
this.control.setErrors({ invalidStatus: true });
|
|
201
|
+
this.cdr.detectChanges();
|
|
202
|
+
return;
|
|
203
|
+
}
|
|
204
|
+
this.isScanValid = true;
|
|
205
|
+
this.successMessage = 'Upload successful!';
|
|
206
|
+
if (this.props['event']) {
|
|
207
|
+
this.props['event'](appData);
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
this.cdr.detectChanges();
|
|
211
|
+
},
|
|
212
|
+
error: (err) => {
|
|
213
|
+
this.fileError = 'An error occurred while processing the scan.';
|
|
214
|
+
console.error('Scan ID error:', err);
|
|
215
|
+
this.isLoading = false;
|
|
216
|
+
this.cdr.detectChanges();
|
|
217
|
+
this.resetForm();
|
|
218
|
+
},
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
resetForm(preserveError = false) {
|
|
222
|
+
this.fileNameFront = '';
|
|
223
|
+
this.fileNameBack = '';
|
|
224
|
+
this.photoBase64Front = '';
|
|
225
|
+
this.photoBase64Back = '';
|
|
226
|
+
this.frontPhotoUploaded = false;
|
|
227
|
+
this.backPhotoUploaded = false;
|
|
228
|
+
if (!preserveError) {
|
|
229
|
+
this.fileError = null;
|
|
230
|
+
}
|
|
231
|
+
this.videoStream = null;
|
|
232
|
+
this.control.reset();
|
|
233
|
+
this.isLoading = false;
|
|
234
|
+
this.successMessage = null;
|
|
235
|
+
}
|
|
236
|
+
checkUploadCondition() {
|
|
237
|
+
if (this.oldIdCard || (this.frontPhotoUploaded && this.backPhotoUploaded)) {
|
|
238
|
+
this.uploadPhoto();
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
extractStatusFromAppData(appData) {
|
|
242
|
+
if (!appData?.fillData || !this.props['stepId'])
|
|
243
|
+
return null;
|
|
244
|
+
let fillData;
|
|
245
|
+
try {
|
|
246
|
+
fillData =
|
|
247
|
+
typeof appData.fillData === 'string'
|
|
248
|
+
? JSON.parse(appData.fillData)
|
|
249
|
+
: appData.fillData;
|
|
250
|
+
}
|
|
251
|
+
catch (e) {
|
|
252
|
+
console.error('[extractStatusFromAppData] JSON parse error', e);
|
|
253
|
+
return null;
|
|
254
|
+
}
|
|
255
|
+
const step = fillData.flux.find((s) => s.stepId === this.props['stepId']);
|
|
256
|
+
if (!step?.sections)
|
|
257
|
+
return null;
|
|
258
|
+
for (const section of step.sections) {
|
|
259
|
+
for (const control of section.controls || []) {
|
|
260
|
+
if (typeof control.fillValue === 'string' &&
|
|
261
|
+
control.fillValue.trim() !== '') {
|
|
262
|
+
return control.fillValue.trim();
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
return null;
|
|
267
|
+
}
|
|
268
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", 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.13", 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"] }] }); }
|
|
270
|
+
}
|
|
271
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FormlyScanIdComponent, decorators: [{
|
|
272
|
+
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: { fileInputFront: [{
|
|
275
|
+
type: ViewChild,
|
|
276
|
+
args: ['fileInputFront']
|
|
277
|
+
}], fileInputBack: [{
|
|
278
|
+
type: ViewChild,
|
|
279
|
+
args: ['fileInputBack']
|
|
280
|
+
}], videoElement: [{
|
|
281
|
+
type: ViewChild,
|
|
282
|
+
args: ['videoElement']
|
|
283
|
+
}] } });
|
|
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>"]}
|