@dsivd/prestations-ng 15.3.1-beta5 → 15.3.1-beta7
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/CHANGELOG.md +21 -0
- package/directives/currency-formatter.directive.d.ts +1 -0
- package/dsivd-prestations-ng-v15.3.1-beta7.tgz +0 -0
- package/esm2020/directives/currency-formatter.directive.mjs +15 -2
- package/esm2020/foehn-input/foehn-input-number.component.mjs +24 -27
- package/esm2020/foehn-upload/foehn-picture-upload/foehn-picture-upload.component.mjs +35 -6
- package/esm2020/sdk-dictionary/default-dictionary.mjs +2 -1
- package/fesm2015/dsivd-prestations-ng.mjs +75 -34
- package/fesm2015/dsivd-prestations-ng.mjs.map +1 -1
- package/fesm2020/dsivd-prestations-ng.mjs +72 -32
- package/fesm2020/dsivd-prestations-ng.mjs.map +1 -1
- package/foehn-input/foehn-input-number.component.d.ts +0 -2
- package/foehn-upload/foehn-picture-upload/foehn-picture-upload.component.d.ts +2 -0
- package/package.json +1 -1
- package/dsivd-prestations-ng-v15.3.1-beta5.tgz +0 -0
package/CHANGELOG.md
CHANGED
|
@@ -30,6 +30,23 @@ A change is considered **breaking** if you have to change your code or update yo
|
|
|
30
30
|
|
|
31
31
|
### Updated
|
|
32
32
|
|
|
33
|
+
- [foehn-input-number.component.ts](projects/prestations-ng/src/foehn-input/foehn-input-number.component.ts)
|
|
34
|
+
- As in the other components, we don't want to force the user input, when `[allowFreeInput]="false"` (which is the default) :
|
|
35
|
+
- It is still done for :
|
|
36
|
+
- maxLength (to avoid a number overflow in the backend)
|
|
37
|
+
- allowDecimal (to avoid a number conversion error in the backend)
|
|
38
|
+
- allowNegative
|
|
39
|
+
- `min` and `max` are not used to force the user input anymore
|
|
40
|
+
- `min` and `max` are only used to display a standard help text
|
|
41
|
+
- the standard help text is only displayed if `min` and `max` are set
|
|
42
|
+
- to better understand how to use the `foehn-input-number`, please check the explanations on https://dsi-vd.github.io/prestations-ng/input-number
|
|
43
|
+
|
|
44
|
+
- explication sur la page de démo
|
|
45
|
+
sauf cas exceptionels, les foehn-input-number sont a utliser ainsi :
|
|
46
|
+
- min max si on veut le help text
|
|
47
|
+
- maxLength selon son type backend
|
|
48
|
+
- allow decimal et allownegative si pertinent
|
|
49
|
+
|
|
33
50
|
- [foehn-input.component.ts](projects/prestations-ng/src/foehn-input/foehn-input.component.ts)
|
|
34
51
|
- removed `@Input() showHelpModal = false` show an help-modal related to an foehn-input
|
|
35
52
|
- removed `@Input() helpModalTitle: string` The title of the modal
|
|
@@ -51,6 +68,10 @@ A change is considered **breaking** if you have to change your code or update yo
|
|
|
51
68
|
- [foehn-boolean-checkbox.component.html](projects/prestations-ng/src/foehn-boolean/foehn-boolean-checkbox.component.html)
|
|
52
69
|
- handle `isLabelSrOnly` input
|
|
53
70
|
|
|
71
|
+
- [foehn-picture-upload.component.ts](projects/prestations-ng/src/foehn-upload/foehn-picture-upload/foehn-picture-upload.component.ts)
|
|
72
|
+
- check file format before trying to display the image for cropping
|
|
73
|
+
- when file has wrong extension, display error with key `foehn-picture-upload.invalid-file-type`
|
|
74
|
+
|
|
54
75
|
## [15.3.0]
|
|
55
76
|
|
|
56
77
|
### Added
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { NgZone, OnInit } from '@angular/core';
|
|
2
2
|
import { FoehnInputNumberComponent } from '../foehn-input/foehn-input-number.component';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
|
+
export declare function formatNumberAsGiven(value: string): string;
|
|
4
5
|
export declare function formatDecimalCurrency(value: string, maxDecimalCount: number, maxLength: number, allowFreeInput: boolean): string;
|
|
5
6
|
export declare function formatNonDecimalCurrency(value: string): string;
|
|
6
7
|
export declare class NumberCurrencyFormatterDirective implements OnInit {
|
|
Binary file
|
|
@@ -20,6 +20,19 @@ function formatWithThousandSeparators(value) {
|
|
|
20
20
|
return value.replace(CURRENCY_REGEXP, THOUSANDS_SEPARATOR);
|
|
21
21
|
}
|
|
22
22
|
// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc
|
|
23
|
+
export function formatNumberAsGiven(value) {
|
|
24
|
+
if (value === undefined || value === null || value === '') {
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
const [intg, dec] = removeFormatting(value).split(DECIMALS_SEPARATOR);
|
|
28
|
+
if (!!dec?.length && dec !== undefined) {
|
|
29
|
+
return formatWithThousandSeparators(intg) + DECIMALS_SEPARATOR + dec;
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
return formatWithThousandSeparators(intg);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc
|
|
23
36
|
export function formatDecimalCurrency(value, maxDecimalCount, maxLength, allowFreeInput) {
|
|
24
37
|
if (value === undefined || value === null || value === '') {
|
|
25
38
|
return null;
|
|
@@ -84,7 +97,7 @@ export class NumberCurrencyFormatterDirective {
|
|
|
84
97
|
}
|
|
85
98
|
setFormatting(value) {
|
|
86
99
|
// Hooks on the inputs of the component to customize the behavior
|
|
87
|
-
const formattedValue = this.host.allowDecimal
|
|
100
|
+
const formattedValue = this.host.allowDecimal || this.host.allowFreeInput
|
|
88
101
|
? formatDecimalCurrency(value, this.host.maxDecimalCount, this.host.maxlength, this.host.allowFreeInput)
|
|
89
102
|
: formatNonDecimalCurrency(value);
|
|
90
103
|
const newValueWithoutFormatting = removeFormatting(formattedValue);
|
|
@@ -118,4 +131,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImpo
|
|
|
118
131
|
type: HostListener,
|
|
119
132
|
args: ['modelChange', ['$event']]
|
|
120
133
|
}] } });
|
|
121
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"currency-formatter.directive.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/directives/currency-formatter.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, HostListener, NgZone, OnInit } from '@angular/core';\n// eslint-disable-next-line import/no-cycle\nimport { FoehnInputNumberComponent } from '../foehn-input/foehn-input-number.component';\nimport { first } from 'rxjs/operators';\nimport {\n    CURRENCY_REGEXP,\n    DECIMALS_SEPARATOR,\n    THOUSANDS_SEPARATOR\n} from '../sdk-currency/currency.helper';\n\n// eslint-disable max-classes-per-file\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nfunction removeFormatting(value: string): string {\n    if (value === undefined || value === null) {\n        // It's important to return null as the backend will consider this as no value.\n        return null;\n    }\n\n    return value\n        .toString()\n        .trim()\n        .replace(new RegExp(THOUSANDS_SEPARATOR, 'g'), '');\n}\n\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nfunction formatWithThousandSeparators(value: string): string {\n    return value.replace(CURRENCY_REGEXP, THOUSANDS_SEPARATOR);\n}\n\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nexport function formatDecimalCurrency(\n    value: string,\n    maxDecimalCount: number,\n    maxLength: number,\n    allowFreeInput: boolean\n): string {\n    if (value === undefined || value === null || value === '') {\n        return null;\n    }\n\n    const [intg, dec] = removeFormatting(value).split(DECIMALS_SEPARATOR);\n\n    if (allowFreeInput) {\n        return (\n            formatWithThousandSeparators(intg) +\n            DECIMALS_SEPARATOR +\n            (dec || '0').padEnd(maxDecimalCount, '0')\n        );\n    }\n\n    if (maxDecimalCount === 0 || !maxDecimalCount) {\n        return formatWithThousandSeparators(intg);\n    }\n    const validEmptyEnd = '0'.repeat(maxDecimalCount);\n\n    let newEnd = dec ? dec + validEmptyEnd : validEmptyEnd;\n    if (newEnd && newEnd.length >= maxDecimalCount) {\n        newEnd = newEnd.substring(0, maxDecimalCount);\n    }\n\n    let newDecimal = intg + DECIMALS_SEPARATOR + newEnd;\n    if (newDecimal.length > maxLength) {\n        newDecimal = newDecimal.substring(0, maxLength);\n    }\n\n    [, newEnd] = newDecimal.split(DECIMALS_SEPARATOR);\n    if (!newEnd) {\n        return formatWithThousandSeparators(intg);\n    }\n\n    return formatWithThousandSeparators(intg) + DECIMALS_SEPARATOR + newEnd;\n}\n\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nexport function formatNonDecimalCurrency(value: string): string {\n    if (value === undefined || value === null) {\n        return null;\n    }\n\n    return formatWithThousandSeparators(removeFormatting(value));\n}\n\n// eslint-disable-next-line @angular-eslint/directive-selector\n@Directive({ selector: '[numberCurrencyFormatter]' })\nexport class NumberCurrencyFormatterDirective implements OnInit {\n    private hasFocus = false;\n\n    constructor(\n        private host: FoehnInputNumberComponent,\n        private ngZone: NgZone\n    ) {}\n\n    @HostListener('blur', ['$event.target.value'])\n    onBlur(value: string): void {\n        this.hasFocus = false;\n        this.setFormatting(value);\n    }\n\n    @HostListener('focus', ['$event.target.value'])\n    onFocus(value: string): void {\n        this.hasFocus = true;\n        this.host.forceUpdateNgModel(removeFormatting(value));\n    }\n\n    @HostListener('modelChange', ['$event'])\n    onModelChange(value: string): void {\n        if (!!value && !this.hasFocus) {\n            this.setFormatting(value);\n        }\n    }\n\n    ngOnInit(): void {\n        // sometimes, when the view is \"simple\" and doesn't require lot a processing\n        // we arrive in this method with this.host.model not already initialized, so the formatting is not working\n        this.ngZone.onMicrotaskEmpty.pipe(first()).subscribe(() => {\n            this.setFormatting(this.host.model);\n        });\n    }\n\n    private setFormatting(value: string): void {\n        // Hooks on the inputs of the component to customize the behavior\n        const formattedValue = this.host.allowDecimal\n            ? formatDecimalCurrency(\n                  value,\n                  this.host.maxDecimalCount,\n                  this.host.maxlength,\n                  this.host.allowFreeInput\n              )\n            : formatNonDecimalCurrency(value);\n\n        const newValueWithoutFormatting = removeFormatting(formattedValue);\n        if (this.hasValueChanged(newValueWithoutFormatting)) {\n            // Since we have altered the model (i.e. by removing some decimals), we have to update it\n            this.host.updateNgModel(newValueWithoutFormatting);\n        }\n\n        // Due to ngModel being possibly updated, we need to wait for a micro task empty\n        this.ngZone.onMicrotaskEmpty.pipe(first()).subscribe(() => {\n            // We're injection non-decimal characters, hence the force.\n            this.host.forceUpdateNgModel(formattedValue);\n        });\n    }\n\n    private hasValueChanged(newValueWithoutFormatting: string): boolean {\n        return (\n            !this.host.model ||\n            this.host.model.toString() !== newValueWithoutFormatting\n        );\n    }\n}\n"]}
|
|
134
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"currency-formatter.directive.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/directives/currency-formatter.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAkB,MAAM,eAAe,CAAC;AAGxE,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACvC,OAAO,EACH,eAAe,EACf,kBAAkB,EAClB,mBAAmB,EACtB,MAAM,iCAAiC,CAAC;;;AAEzC,sCAAsC;AACtC,mFAAmF;AACnF,SAAS,gBAAgB,CAAC,KAAa;IACnC,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,EAAE;QACvC,+EAA+E;QAC/E,OAAO,IAAI,CAAC;KACf;IAED,OAAO,KAAK;SACP,QAAQ,EAAE;SACV,IAAI,EAAE;SACN,OAAO,CAAC,IAAI,MAAM,CAAC,mBAAmB,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC;AAC3D,CAAC;AAED,mFAAmF;AACnF,SAAS,4BAA4B,CAAC,KAAa;IAC/C,OAAO,KAAK,CAAC,OAAO,CAAC,eAAe,EAAE,mBAAmB,CAAC,CAAC;AAC/D,CAAC;AAED,mFAAmF;AACnF,MAAM,UAAU,mBAAmB,CAAC,KAAa;IAC7C,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,EAAE,EAAE;QACvD,OAAO,IAAI,CAAC;KACf;IAED,MAAM,CAAC,IAAI,EAAE,GAAG,CAAC,GAAG,gBAAgB,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAC;IAEtE,IAAI,CAAC,CAAC,GAAG,EAAE,MAAM,IAAI,GAAG,KAAK,SAAS,EAAE;QACpC,OAAO,4BAA4B,CAAC,IAAI,CAAC,GAAG,kBAAkB,GAAG,GAAG,CAAC;KACxE;SAAM;QACH,OAAO,4BAA4B,CAAC,IAAI,CAAC,CAAC;KAC7C;AACL,CAAC;AAED,mFAAmF;AACnF,MAAM,UAAU,qBAAqB,CACjC,KAAa,EACb,eAAuB,EACvB,SAAiB,EACjB,cAAuB;IAEvB,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,EAAE,EAAE;QACvD,OAAO,IAAI,CAAC;KACf;IAED,MAAM,CAAC,IAAI,EAAE,GAAG,CAAC,GAAG,gBAAgB,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAC;IAEtE,IAAI,cAAc,EAAE;QAChB,OAAO,CACH,4BAA4B,CAAC,IAAI,CAAC;YAClC,kBAAkB;YAClB,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC,MAAM,CAAC,eAAe,EAAE,GAAG,CAAC,CAC5C,CAAC;KACL;IAED,IAAI,eAAe,KAAK,CAAC,IAAI,CAAC,eAAe,EAAE;QAC3C,OAAO,4BAA4B,CAAC,IAAI,CAAC,CAAC;KAC7C;IACD,MAAM,aAAa,GAAG,GAAG,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;IAElD,IAAI,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC;IACvD,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,IAAI,eAAe,EAAE;QAC5C,MAAM,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC,EAAE,eAAe,CAAC,CAAC;KACjD;IAED,IAAI,UAAU,GAAG,IAAI,GAAG,kBAAkB,GAAG,MAAM,CAAC;IACpD,IAAI,UAAU,CAAC,MAAM,GAAG,SAAS,EAAE;QAC/B,UAAU,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;KACnD;IAED,CAAC,EAAE,MAAM,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAC;IAClD,IAAI,CAAC,MAAM,EAAE;QACT,OAAO,4BAA4B,CAAC,IAAI,CAAC,CAAC;KAC7C;IAED,OAAO,4BAA4B,CAAC,IAAI,CAAC,GAAG,kBAAkB,GAAG,MAAM,CAAC;AAC5E,CAAC;AAED,mFAAmF;AACnF,MAAM,UAAU,wBAAwB,CAAC,KAAa;IAClD,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,EAAE;QACvC,OAAO,IAAI,CAAC;KACf;IAED,OAAO,4BAA4B,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,CAAC;AACjE,CAAC;AAED,8DAA8D;AAE9D,MAAM,OAAO,gCAAgC;IAGzC,YACY,IAA+B,EAC/B,MAAc;QADd,SAAI,GAAJ,IAAI,CAA2B;QAC/B,WAAM,GAAN,MAAM,CAAQ;QAJlB,aAAQ,GAAG,KAAK,CAAC;IAKtB,CAAC;IAGJ,MAAM,CAAC,KAAa;QAChB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QACtB,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;IAC9B,CAAC;IAGD,OAAO,CAAC,KAAa;QACjB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;QACrB,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1D,CAAC;IAGD,aAAa,CAAC,KAAa;QACvB,IAAI,CAAC,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YAC3B,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;SAC7B;IACL,CAAC;IAED,QAAQ;QACJ,4EAA4E;QAC5E,0GAA0G;QAC1G,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YACtD,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,aAAa,CAAC,KAAa;QAC/B,iEAAiE;QACjE,MAAM,cAAc,GAChB,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,IAAI,CAAC,cAAc;YAC9C,CAAC,CAAC,qBAAqB,CACjB,KAAK,EACL,IAAI,CAAC,IAAI,CAAC,eAAe,EACzB,IAAI,CAAC,IAAI,CAAC,SAAS,EACnB,IAAI,CAAC,IAAI,CAAC,cAAc,CAC3B;YACH,CAAC,CAAC,wBAAwB,CAAC,KAAK,CAAC,CAAC;QAE1C,MAAM,yBAAyB,GAAG,gBAAgB,CAAC,cAAc,CAAC,CAAC;QACnE,IAAI,IAAI,CAAC,eAAe,CAAC,yBAAyB,CAAC,EAAE;YACjD,yFAAyF;YACzF,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,yBAAyB,CAAC,CAAC;SACtD;QAED,gFAAgF;QAChF,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YACtD,2DAA2D;YAC3D,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,cAAc,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,eAAe,CAAC,yBAAiC;QACrD,OAAO,CACH,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK;YAChB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,KAAK,yBAAyB,CAC3D,CAAC;IACN,CAAC;;8HAjEQ,gCAAgC;kHAAhC,gCAAgC;4FAAhC,gCAAgC;kBAD5C,SAAS;mBAAC,EAAE,QAAQ,EAAE,2BAA2B,EAAE;qIAUhD,MAAM;sBADL,YAAY;uBAAC,MAAM,EAAE,CAAC,qBAAqB,CAAC;gBAO7C,OAAO;sBADN,YAAY;uBAAC,OAAO,EAAE,CAAC,qBAAqB,CAAC;gBAO9C,aAAa;sBADZ,YAAY;uBAAC,aAAa,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import { Directive, HostListener, NgZone, OnInit } from '@angular/core';\n// eslint-disable-next-line import/no-cycle\nimport { FoehnInputNumberComponent } from '../foehn-input/foehn-input-number.component';\nimport { first } from 'rxjs/operators';\nimport {\n    CURRENCY_REGEXP,\n    DECIMALS_SEPARATOR,\n    THOUSANDS_SEPARATOR\n} from '../sdk-currency/currency.helper';\n\n// eslint-disable max-classes-per-file\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nfunction removeFormatting(value: string): string {\n    if (value === undefined || value === null) {\n        // It's important to return null as the backend will consider this as no value.\n        return null;\n    }\n\n    return value\n        .toString()\n        .trim()\n        .replace(new RegExp(THOUSANDS_SEPARATOR, 'g'), '');\n}\n\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nfunction formatWithThousandSeparators(value: string): string {\n    return value.replace(CURRENCY_REGEXP, THOUSANDS_SEPARATOR);\n}\n\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nexport function formatNumberAsGiven(value: string): string {\n    if (value === undefined || value === null || value === '') {\n        return null;\n    }\n\n    const [intg, dec] = removeFormatting(value).split(DECIMALS_SEPARATOR);\n\n    if (!!dec?.length && dec !== undefined) {\n        return formatWithThousandSeparators(intg) + DECIMALS_SEPARATOR + dec;\n    } else {\n        return formatWithThousandSeparators(intg);\n    }\n}\n\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nexport function formatDecimalCurrency(\n    value: string,\n    maxDecimalCount: number,\n    maxLength: number,\n    allowFreeInput: boolean\n): string {\n    if (value === undefined || value === null || value === '') {\n        return null;\n    }\n\n    const [intg, dec] = removeFormatting(value).split(DECIMALS_SEPARATOR);\n\n    if (allowFreeInput) {\n        return (\n            formatWithThousandSeparators(intg) +\n            DECIMALS_SEPARATOR +\n            (dec || '0').padEnd(maxDecimalCount, '0')\n        );\n    }\n\n    if (maxDecimalCount === 0 || !maxDecimalCount) {\n        return formatWithThousandSeparators(intg);\n    }\n    const validEmptyEnd = '0'.repeat(maxDecimalCount);\n\n    let newEnd = dec ? dec + validEmptyEnd : validEmptyEnd;\n    if (newEnd && newEnd.length >= maxDecimalCount) {\n        newEnd = newEnd.substring(0, maxDecimalCount);\n    }\n\n    let newDecimal = intg + DECIMALS_SEPARATOR + newEnd;\n    if (newDecimal.length > maxLength) {\n        newDecimal = newDecimal.substring(0, maxLength);\n    }\n\n    [, newEnd] = newDecimal.split(DECIMALS_SEPARATOR);\n    if (!newEnd) {\n        return formatWithThousandSeparators(intg);\n    }\n\n    return formatWithThousandSeparators(intg) + DECIMALS_SEPARATOR + newEnd;\n}\n\n// eslint-disable-next-line prefer-arrow/prefer-arrow-functions,jsdoc/require-jsdoc\nexport function formatNonDecimalCurrency(value: string): string {\n    if (value === undefined || value === null) {\n        return null;\n    }\n\n    return formatWithThousandSeparators(removeFormatting(value));\n}\n\n// eslint-disable-next-line @angular-eslint/directive-selector\n@Directive({ selector: '[numberCurrencyFormatter]' })\nexport class NumberCurrencyFormatterDirective implements OnInit {\n    private hasFocus = false;\n\n    constructor(\n        private host: FoehnInputNumberComponent,\n        private ngZone: NgZone\n    ) {}\n\n    @HostListener('blur', ['$event.target.value'])\n    onBlur(value: string): void {\n        this.hasFocus = false;\n        this.setFormatting(value);\n    }\n\n    @HostListener('focus', ['$event.target.value'])\n    onFocus(value: string): void {\n        this.hasFocus = true;\n        this.host.forceUpdateNgModel(removeFormatting(value));\n    }\n\n    @HostListener('modelChange', ['$event'])\n    onModelChange(value: string): void {\n        if (!!value && !this.hasFocus) {\n            this.setFormatting(value);\n        }\n    }\n\n    ngOnInit(): void {\n        // sometimes, when the view is \"simple\" and doesn't require lot a processing\n        // we arrive in this method with this.host.model not already initialized, so the formatting is not working\n        this.ngZone.onMicrotaskEmpty.pipe(first()).subscribe(() => {\n            this.setFormatting(this.host.model);\n        });\n    }\n\n    private setFormatting(value: string): void {\n        // Hooks on the inputs of the component to customize the behavior\n        const formattedValue =\n            this.host.allowDecimal || this.host.allowFreeInput\n                ? formatDecimalCurrency(\n                      value,\n                      this.host.maxDecimalCount,\n                      this.host.maxlength,\n                      this.host.allowFreeInput\n                  )\n                : formatNonDecimalCurrency(value);\n\n        const newValueWithoutFormatting = removeFormatting(formattedValue);\n        if (this.hasValueChanged(newValueWithoutFormatting)) {\n            // Since we have altered the model (i.e. by removing some decimals), we have to update it\n            this.host.updateNgModel(newValueWithoutFormatting);\n        }\n\n        // Due to ngModel being possibly updated, we need to wait for a micro task empty\n        this.ngZone.onMicrotaskEmpty.pipe(first()).subscribe(() => {\n            // We're injection non-decimal characters, hence the force.\n            this.host.forceUpdateNgModel(formattedValue);\n        });\n    }\n\n    private hasValueChanged(newValueWithoutFormatting: string): boolean {\n        return (\n            !this.host.model ||\n            this.host.model.toString() !== newValueWithoutFormatting\n        );\n    }\n}\n"]}
|
|
@@ -4,7 +4,7 @@ import { FoehnInputStringComponent } from './foehn-input-string.component';
|
|
|
4
4
|
import { FoehnInputComponent } from './foehn-input.component';
|
|
5
5
|
import { DECIMALS_SEPARATOR } from '../sdk-currency/currency.helper';
|
|
6
6
|
// eslint-disable-next-line import/no-cycle
|
|
7
|
-
import { formatDecimalCurrency, formatNonDecimalCurrency } from '../directives/currency-formatter.directive';
|
|
7
|
+
import { formatDecimalCurrency, formatNonDecimalCurrency, formatNumberAsGiven } from '../directives/currency-formatter.directive';
|
|
8
8
|
import * as i0 from "@angular/core";
|
|
9
9
|
import * as i1 from "../sdk-currency/currency.helper";
|
|
10
10
|
import * as i2 from "../sdk-dictionary/sdk-dictionary.service";
|
|
@@ -14,6 +14,7 @@ import * as i5 from "../foehn-icons/foehn-icon-times.component";
|
|
|
14
14
|
import * as i6 from "@angular/common";
|
|
15
15
|
import * as i7 from "@angular/forms";
|
|
16
16
|
import * as i8 from "../sdk-dictionary/sdk-dictionary.pipe";
|
|
17
|
+
const isDefined = (value) => value === 0 || !!value;
|
|
17
18
|
export class FoehnInputNumberComponent extends FoehnInputStringComponent {
|
|
18
19
|
constructor(currencyHelper, dictionaryService) {
|
|
19
20
|
super();
|
|
@@ -71,6 +72,11 @@ export class FoehnInputNumberComponent extends FoehnInputStringComponent {
|
|
|
71
72
|
noCommaString && typeof noCommaString === 'string'
|
|
72
73
|
? noCommaString.replace(CLEAN_NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN, '')
|
|
73
74
|
: noCommaString;
|
|
75
|
+
// check that it is a number in any of its possible form
|
|
76
|
+
// to avoid -3.14-23--534
|
|
77
|
+
if (!NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN.test(cleanNumber)) {
|
|
78
|
+
cleanNumber = this.model;
|
|
79
|
+
}
|
|
74
80
|
}
|
|
75
81
|
super.updateNgModel(cleanNumber);
|
|
76
82
|
// Need to update HTML input value because we let user input what he wants and then we clean model
|
|
@@ -85,35 +91,11 @@ export class FoehnInputNumberComponent extends FoehnInputStringComponent {
|
|
|
85
91
|
this.model_ = value;
|
|
86
92
|
this.updateInputElementValue(value);
|
|
87
93
|
}
|
|
88
|
-
isValueInRange(value) {
|
|
89
|
-
if (this.allowNegative && value === '-') {
|
|
90
|
-
return true;
|
|
91
|
-
}
|
|
92
|
-
const valueAsNumber = this.toNumber(value);
|
|
93
|
-
if (!this.isEmpty(this.min) || !this.isEmpty(this.max)) {
|
|
94
|
-
if (isNaN(valueAsNumber)) {
|
|
95
|
-
return false;
|
|
96
|
-
}
|
|
97
|
-
if (!this.isEmpty(this.min) && !this.isEmpty(this.max)) {
|
|
98
|
-
return valueAsNumber >= this.min && valueAsNumber <= this.max;
|
|
99
|
-
}
|
|
100
|
-
if (!this.isEmpty(this.min)) {
|
|
101
|
-
return valueAsNumber >= this.min;
|
|
102
|
-
}
|
|
103
|
-
if (!this.isEmpty(this.max)) {
|
|
104
|
-
return valueAsNumber <= this.max;
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
return true;
|
|
108
|
-
}
|
|
109
|
-
toNumber(value) {
|
|
110
|
-
return this.allowDecimal ? parseFloat(value) : parseInt(value, 10);
|
|
111
|
-
}
|
|
112
94
|
isStringAValidNumber(value) {
|
|
113
95
|
return this.getValidationRegex().test(value);
|
|
114
96
|
}
|
|
115
97
|
isModelValid(value) {
|
|
116
|
-
return this.isStringAValidNumber(value)
|
|
98
|
+
return this.isStringAValidNumber(value);
|
|
117
99
|
}
|
|
118
100
|
getValidationRegex(isForClean = false) {
|
|
119
101
|
if (this.allowDecimal || this.allowNegative) {
|
|
@@ -186,6 +168,21 @@ export class FoehnInputNumberComponent extends FoehnInputStringComponent {
|
|
|
186
168
|
if (!!this.hideStandardHelpText) {
|
|
187
169
|
return;
|
|
188
170
|
}
|
|
171
|
+
if (!this.allowFreeInput) {
|
|
172
|
+
if (isDefined(this.min) && isDefined(this.max)) {
|
|
173
|
+
if (!!this.helpText?.length) {
|
|
174
|
+
this.helpText += '<br/>';
|
|
175
|
+
}
|
|
176
|
+
else {
|
|
177
|
+
this.helpText = '';
|
|
178
|
+
}
|
|
179
|
+
this.helpText += this.dictionaryService.getKeySync('foehn-input-number.standard-help-text.label', {
|
|
180
|
+
minValue: formatNumberAsGiven(this.min.toString()),
|
|
181
|
+
maxValue: formatNumberAsGiven(this.max.toString())
|
|
182
|
+
});
|
|
183
|
+
}
|
|
184
|
+
return;
|
|
185
|
+
}
|
|
189
186
|
if (!!this.helpText?.length) {
|
|
190
187
|
this.helpText += '<br/>';
|
|
191
188
|
}
|
|
@@ -261,4 +258,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImpo
|
|
|
261
258
|
}], model: [{
|
|
262
259
|
type: Input
|
|
263
260
|
}] } });
|
|
264
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"foehn-input-number.component.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/foehn-input/foehn-input-number.component.ts","../../../../projects/prestations-ng/src/foehn-input/foehn-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, forwardRef, Input, OnInit } from '@angular/core';\nimport {\n    CLEAN_DECIMAL_NUMBERS_PATTERN,\n    CLEAN_NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN,\n    CLEAN_NEGATIVE_NUMBERS_PATTERN,\n    CLEAN_NUMBERS_PATTERN,\n    DECIMAL_NUMBERS_PATTERN,\n    HTML_DECIMAL_PATTERN,\n    HTML_NO_DECIMAL_PATTERN,\n    NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN,\n    NEGATIVE_NUMBERS_PATTERN,\n    NUMBERS_PATTERN\n} from './pattern.const';\nimport { FoehnInputStringComponent } from './foehn-input-string.component';\nimport { FoehnInputComponent } from './foehn-input.component';\nimport {\n    CurrencyHelper,\n    DECIMALS_SEPARATOR\n} from '../sdk-currency/currency.helper';\nimport { SdkDictionaryService } from '../sdk-dictionary/sdk-dictionary.service';\n// eslint-disable-next-line import/no-cycle\nimport {\n    formatDecimalCurrency,\n    formatNonDecimalCurrency\n} from '../directives/currency-formatter.directive';\n\n@Component({\n    selector: 'foehn-input-number',\n    templateUrl: './foehn-input.component.html',\n    styleUrls: ['./foehn-input.component.css'],\n    providers: [\n        {\n            provide: FoehnInputComponent,\n            useExisting: forwardRef(() => FoehnInputNumberComponent),\n            multi: true\n        }\n    ]\n})\nexport class FoehnInputNumberComponent\n    extends FoehnInputStringComponent\n    implements OnInit {\n    @Input()\n    hideStandardHelpText = false;\n\n    @Input()\n    allowDecimal: boolean;\n\n    @Input()\n    allowNegative: boolean;\n\n    @Input()\n    maxDecimalCount = 2;\n\n    @Input()\n    allowFreeInput = false;\n\n    constructor(\n        private currencyHelper: CurrencyHelper,\n        private dictionaryService: SdkDictionaryService\n    ) {\n        super();\n    }\n\n    get model(): string {\n        // If the setter is overridden, it's critical to override the getter too.\n        return this.model_;\n    }\n\n    @Input()\n    set model(value: string) {\n        // This doesn't trigger the component's version of updateNgModel strait away as this might\n        // be called before allowDecimal or allowNegative are properly set (before the init lifecycle\n        // of the component). This is critical to not re-format the input before the OnInit, otherwise\n        // we might end up with invalid value displayed to the user. this.updateNgModel is called implicitly\n        // as super.updateNgModel will emit a change on this.modelChange.\n        // FIXME this is a hack and should be refactored.\n        super.updateNgModel(value);\n    }\n\n    ngOnInit(): void {\n        super.ngOnInit();\n\n        this.type = 'text';\n\n        // Pattern used only for numpad trigger on mobile phones\n        this.pattern = this.allowDecimal\n            ? HTML_DECIMAL_PATTERN\n            : HTML_NO_DECIMAL_PATTERN;\n\n        this.maxlength = !!this.maxlength ? this.maxlength : 9;\n\n        this.manageHelpText();\n    }\n\n    getMaxLength(): number {\n        if (this.allowFreeInput) {\n            return null;\n        }\n        return super.getMaxLength();\n    }\n\n    updateNgModel(value: string): void {\n        if (this.isEmpty(value)) {\n            this.setToEmpty(value);\n            return;\n        }\n\n        let cleanNumber;\n        if (!this.allowFreeInput) {\n            cleanNumber = this.getCleanNumber(value.toString());\n            const isModelValid = this.isModelValid(cleanNumber);\n            if (!isModelValid) {\n                cleanNumber = this.model;\n            }\n        } else {\n            const noCommaString = this.replaceCommaWithDot(value);\n            cleanNumber =\n                noCommaString && typeof noCommaString === 'string'\n                    ? noCommaString.replace(\n                          CLEAN_NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN,\n                          ''\n                      )\n                    : noCommaString;\n        }\n\n        super.updateNgModel(cleanNumber);\n\n        // Need to update HTML input value because we let user input what he wants and then we clean model\n        this.updateInputElementValue(cleanNumber);\n    }\n\n    /**\n     * To be used for post-formatting by directives.\n     *\n     * @param value of model\n     */\n    forceUpdateNgModel(value: string): void {\n        this.model_ = value;\n        this.updateInputElementValue(value);\n    }\n\n    private isValueInRange(value: string): boolean {\n        if (this.allowNegative && value === '-') {\n            return true;\n        }\n\n        const valueAsNumber = this.toNumber(value);\n        if (!this.isEmpty(this.min) || !this.isEmpty(this.max)) {\n            if (isNaN(valueAsNumber)) {\n                return false;\n            }\n\n            if (!this.isEmpty(this.min) && !this.isEmpty(this.max)) {\n                return valueAsNumber >= this.min && valueAsNumber <= this.max;\n            }\n\n            if (!this.isEmpty(this.min)) {\n                return valueAsNumber >= this.min;\n            }\n\n            if (!this.isEmpty(this.max)) {\n                return valueAsNumber <= this.max;\n            }\n        }\n        return true;\n    }\n\n    private toNumber(value: string): number {\n        return this.allowDecimal ? parseFloat(value) : parseInt(value, 10);\n    }\n\n    private isStringAValidNumber(value: string): boolean {\n        return this.getValidationRegex().test(value);\n    }\n\n    private isModelValid(value: string): boolean {\n        return this.isStringAValidNumber(value) && this.isValueInRange(value);\n    }\n\n    private getValidationRegex(isForClean: boolean = false): RegExp {\n        if (this.allowDecimal || this.allowNegative) {\n            // Allow decimal and negative\n            if (this.allowDecimal && this.allowNegative) {\n                return !isForClean\n                    ? NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN\n                    : CLEAN_NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN;\n            }\n            // Allow decimal\n            if (this.allowDecimal) {\n                return !isForClean\n                    ? DECIMAL_NUMBERS_PATTERN\n                    : CLEAN_DECIMAL_NUMBERS_PATTERN;\n            }\n            // Allow negative\n            return !isForClean\n                ? NEGATIVE_NUMBERS_PATTERN\n                : CLEAN_NEGATIVE_NUMBERS_PATTERN;\n        }\n        return !isForClean ? NUMBERS_PATTERN : CLEAN_NUMBERS_PATTERN;\n    }\n\n    private replaceCommaWithDot(value: string): string {\n        return value && typeof value === 'string'\n            ? value.replace(new RegExp(',', 'g'), DECIMALS_SEPARATOR)\n            : value;\n    }\n\n    private removeNonNumericCharacters(value: string): string {\n        return value && typeof value === 'string'\n            ? value.replace(this.getValidationRegex(true), '')\n            : value;\n    }\n\n    private cleanDecimals(\n        value: string,\n        separator: string = DECIMALS_SEPARATOR\n    ): string {\n        if (!value || !this.allowDecimal || !value.includes(separator)) {\n            return value;\n        }\n\n        const [intg, dec] = value.split(separator);\n        const cleanDec = dec.substring(0, this.maxDecimalCount);\n\n        return intg + separator + cleanDec;\n    }\n\n    private getCleanNumber(value: string): string {\n        const noCommaString = this.replaceCommaWithDot(value);\n        const cleanDecimals = this.cleanDecimals(noCommaString);\n        const cleanNumber = this.removeNonNumericCharacters(cleanDecimals);\n\n        if (cleanNumber && cleanNumber.length) {\n            return cleanNumber.substring(0, this.getMaxLength());\n        }\n\n        return cleanNumber;\n    }\n\n    private updateInputElementValue(cleanNumber: string): void {\n        if (!this.inputElement) {\n            // No need to update the native element if it doesn't yet exist.\n            return;\n        }\n\n        const inputElemValue = this.inputElement.nativeElement.value;\n        if (inputElemValue !== cleanNumber) {\n            // Need to update HTML input value because we let user input what he wants and then we clean model\n            this.inputElement.nativeElement.value = cleanNumber;\n        }\n    }\n\n    private setToEmpty(value: string): void {\n        // Empty means null, otherwise the backend doesn't handle numbers properly. We have to keep 'undefined'\n        // if 'undefined' otherwise Angular complains about the value of the model being changed without updating\n        // the view.\n        const emptyValue = value === '' ? null : value;\n        super.updateNgModel(emptyValue);\n        this.updateInputElementValue(emptyValue);\n    }\n\n    private manageHelpText(): void {\n        if (!!this.hideStandardHelpText) {\n            return;\n        }\n\n        if (!!this.helpText?.length) {\n            this.helpText += '<br/>';\n        } else {\n            this.helpText = '';\n        }\n\n        const maxIntegerCount = this.allowDecimal\n            ? this.maxlength - this.maxDecimalCount - 1\n            : this.maxlength;\n\n        let maxAsString;\n        if (!!this.max) {\n            maxAsString = this.max;\n        } else {\n            maxAsString = '9'.repeat(maxIntegerCount);\n            if (this.allowDecimal) {\n                maxAsString += `.${'9'.repeat(this.maxDecimalCount)}`;\n            }\n        }\n\n        let minAsString;\n        if (this.min) {\n            minAsString = this.min;\n        } else {\n            minAsString = this.allowNegative\n                ? `-${'9'.repeat(maxIntegerCount - 1)}`\n                : '0';\n            if (this.allowDecimal) {\n                minAsString += this.allowNegative\n                    ? `.${'9'.repeat(this.maxDecimalCount || 0)}`\n                    : `.${'0'.repeat(this.maxDecimalCount || 0)}`;\n            }\n        }\n\n        this.helpText += this.dictionaryService.getKeySync(\n            'foehn-input-number.standard-help-text.label',\n            {\n                minValue: this.formatValue(minAsString),\n                maxValue: this.formatValue(maxAsString)\n            }\n        );\n    }\n\n    private formatValue(value: string | number): string {\n        if (this.allowDecimal) {\n            return formatDecimalCurrency(\n                value?.toString(),\n                this.maxDecimalCount,\n                this.maxlength,\n                this.allowFreeInput\n            );\n        }\n        return formatNonDecimalCurrency(value?.toString());\n    }\n}\n","<div\n    class=\"form-group clearable-input-form-group\"\n    [class.has-danger]=\"hasErrorsToDisplay()\"\n    [class.vd-form-group-danger]=\"hasErrorsToDisplay()\"\n    [attr.id]=\"buildId('Container')\"\n    tabindex=\"-1\"\n>\n    <div class=\"d-flex justify-content-between\">\n        <label\n            style=\"display: inline-block\"\n            [attr.for]=\"buildChildId()\"\n            *ngIf=\"label && type !== 'hidden'\"\n            [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n        >\n            <span [innerHTML]=\"label\"></span>\n            <span\n                *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n                aria-hidden=\"true\"\n            >\n                {{ 'foehn-input.optional' | fromDictionary }}\n            </span>\n        </label>\n\n        <foehn-help-modal\n            class=\"removePaddingButton\"\n            *ngIf=\"!!helpModal\"\n            [modalContent]=\"helpModal\"\n        ></foehn-help-modal>\n    </div>\n\n    <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n    <small\n        *ngIf=\"helpText && type !== 'hidden'\"\n        [attr.id]=\"buildChildId() + 'Help'\"\n        class=\"form-text text-secondary\"\n        [innerHTML]=\"helpText\"\n    ></small>\n\n    <ng-content></ng-content>\n\n    <input\n        [class.is-invalid]=\"hasErrorsToDisplay() || hasInheritErrorFromParent()\"\n        [class.clearable-input]=\"displayClearButton() | async\"\n        class=\"form-control\"\n        [name]=\"name || label\"\n        [attr.maxlength]=\"getMaxLength()\"\n        [attr.autocomplete]=\"getAutoComplete()\"\n        [attr.autocapitalize]=\"autocapitalize\"\n        [attr.disabled]=\"disabled ? 'disabled' : null\"\n        [attr.type]=\"type\"\n        [attr.pattern]=\"pattern\"\n        [attr.min]=\"min\"\n        [attr.max]=\"max\"\n        [attr.id]=\"buildChildId()\"\n        [attr.aria-describedby]=\"getDescribedBy()\"\n        [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n        [attr.aria-required]=\"required || null\"\n        [ngModel]=\"model\"\n        (paste)=\"onPaste($event)\"\n        (ngModelChange)=\"updateNgModel($event)\"\n        (input)=\"handleChange(entryComponent.value)\"\n        (keydown)=\"onKeydown($event)\"\n        (blur)=\"onBlur($event)\"\n        (focus)=\"onFocus($event)\"\n        #entryComponent\n    />\n\n    <button\n        type=\"button\"\n        [id]=\"buildChildId() + 'ClearButton'\"\n        *ngIf=\"displayClearButton() | async\"\n        class=\"clearable-input-clear-button btn\"\n        (click)=\"onClear()\"\n    >\n        <foehn-icon-times\n            class=\"clearable-input-clear-button-icon\"\n            title=\"Réinitialiser le champs\"\n        ></foehn-icon-times>\n    </button>\n</div>\n"]}
|
|
261
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"foehn-input-number.component.js","sourceRoot":"","sources":["../../../../projects/prestations-ng/src/foehn-input/foehn-input-number.component.ts","../../../../projects/prestations-ng/src/foehn-input/foehn-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, forwardRef, Input, OnInit } from '@angular/core';\nimport {\n    CLEAN_DECIMAL_NUMBERS_PATTERN,\n    CLEAN_NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN,\n    CLEAN_NEGATIVE_NUMBERS_PATTERN,\n    CLEAN_NUMBERS_PATTERN,\n    DECIMAL_NUMBERS_PATTERN,\n    HTML_DECIMAL_PATTERN,\n    HTML_NO_DECIMAL_PATTERN,\n    NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN,\n    NEGATIVE_NUMBERS_PATTERN,\n    NUMBERS_PATTERN\n} from './pattern.const';\nimport { FoehnInputStringComponent } from './foehn-input-string.component';\nimport { FoehnInputComponent } from './foehn-input.component';\nimport {\n    CurrencyHelper,\n    DECIMALS_SEPARATOR\n} from '../sdk-currency/currency.helper';\nimport { SdkDictionaryService } from '../sdk-dictionary/sdk-dictionary.service';\n// eslint-disable-next-line import/no-cycle\nimport {\n    formatDecimalCurrency,\n    formatNonDecimalCurrency,\n    formatNumberAsGiven\n} from '../directives/currency-formatter.directive';\n\nconst isDefined = (value: number): boolean => value === 0 || !!value;\n\n@Component({\n    selector: 'foehn-input-number',\n    templateUrl: './foehn-input.component.html',\n    styleUrls: ['./foehn-input.component.css'],\n    providers: [\n        {\n            provide: FoehnInputComponent,\n            useExisting: forwardRef(() => FoehnInputNumberComponent),\n            multi: true\n        }\n    ]\n})\nexport class FoehnInputNumberComponent\n    extends FoehnInputStringComponent\n    implements OnInit {\n    @Input()\n    hideStandardHelpText = false;\n\n    @Input()\n    allowDecimal: boolean;\n\n    @Input()\n    allowNegative: boolean;\n\n    @Input()\n    maxDecimalCount = 2;\n\n    @Input()\n    allowFreeInput = false;\n\n    constructor(\n        private currencyHelper: CurrencyHelper,\n        private dictionaryService: SdkDictionaryService\n    ) {\n        super();\n    }\n\n    get model(): string {\n        // If the setter is overridden, it's critical to override the getter too.\n        return this.model_;\n    }\n\n    @Input()\n    set model(value: string) {\n        // This doesn't trigger the component's version of updateNgModel strait away as this might\n        // be called before allowDecimal or allowNegative are properly set (before the init lifecycle\n        // of the component). This is critical to not re-format the input before the OnInit, otherwise\n        // we might end up with invalid value displayed to the user. this.updateNgModel is called implicitly\n        // as super.updateNgModel will emit a change on this.modelChange.\n        // FIXME this is a hack and should be refactored.\n        super.updateNgModel(value);\n    }\n\n    ngOnInit(): void {\n        super.ngOnInit();\n\n        this.type = 'text';\n\n        // Pattern used only for numpad trigger on mobile phones\n        this.pattern = this.allowDecimal\n            ? HTML_DECIMAL_PATTERN\n            : HTML_NO_DECIMAL_PATTERN;\n\n        this.maxlength = !!this.maxlength ? this.maxlength : 9;\n\n        this.manageHelpText();\n    }\n\n    getMaxLength(): number {\n        if (this.allowFreeInput) {\n            return null;\n        }\n        return super.getMaxLength();\n    }\n\n    updateNgModel(value: string): void {\n        if (this.isEmpty(value)) {\n            this.setToEmpty(value);\n            return;\n        }\n\n        let cleanNumber;\n        if (!this.allowFreeInput) {\n            cleanNumber = this.getCleanNumber(value.toString());\n            const isModelValid = this.isModelValid(cleanNumber);\n            if (!isModelValid) {\n                cleanNumber = this.model;\n            }\n        } else {\n            const noCommaString = this.replaceCommaWithDot(value);\n            cleanNumber =\n                noCommaString && typeof noCommaString === 'string'\n                    ? noCommaString.replace(\n                          CLEAN_NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN,\n                          ''\n                      )\n                    : noCommaString;\n            // check that it is a number in any of its possible form\n            // to avoid -3.14-23--534\n            if (!NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN.test(cleanNumber)) {\n                cleanNumber = this.model;\n            }\n        }\n\n        super.updateNgModel(cleanNumber);\n\n        // Need to update HTML input value because we let user input what he wants and then we clean model\n        this.updateInputElementValue(cleanNumber);\n    }\n\n    /**\n     * To be used for post-formatting by directives.\n     *\n     * @param value of model\n     */\n    forceUpdateNgModel(value: string): void {\n        this.model_ = value;\n        this.updateInputElementValue(value);\n    }\n\n    private isStringAValidNumber(value: string): boolean {\n        return this.getValidationRegex().test(value);\n    }\n\n    private isModelValid(value: string): boolean {\n        return this.isStringAValidNumber(value);\n    }\n\n    private getValidationRegex(isForClean: boolean = false): RegExp {\n        if (this.allowDecimal || this.allowNegative) {\n            // Allow decimal and negative\n            if (this.allowDecimal && this.allowNegative) {\n                return !isForClean\n                    ? NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN\n                    : CLEAN_NEGATIVE_AND_DECIMAL_NUMBERS_PATTERN;\n            }\n            // Allow decimal\n            if (this.allowDecimal) {\n                return !isForClean\n                    ? DECIMAL_NUMBERS_PATTERN\n                    : CLEAN_DECIMAL_NUMBERS_PATTERN;\n            }\n            // Allow negative\n            return !isForClean\n                ? NEGATIVE_NUMBERS_PATTERN\n                : CLEAN_NEGATIVE_NUMBERS_PATTERN;\n        }\n        return !isForClean ? NUMBERS_PATTERN : CLEAN_NUMBERS_PATTERN;\n    }\n\n    private replaceCommaWithDot(value: string): string {\n        return value && typeof value === 'string'\n            ? value.replace(new RegExp(',', 'g'), DECIMALS_SEPARATOR)\n            : value;\n    }\n\n    private removeNonNumericCharacters(value: string): string {\n        return value && typeof value === 'string'\n            ? value.replace(this.getValidationRegex(true), '')\n            : value;\n    }\n\n    private cleanDecimals(\n        value: string,\n        separator: string = DECIMALS_SEPARATOR\n    ): string {\n        if (!value || !this.allowDecimal || !value.includes(separator)) {\n            return value;\n        }\n\n        const [intg, dec] = value.split(separator);\n        const cleanDec = dec.substring(0, this.maxDecimalCount);\n\n        return intg + separator + cleanDec;\n    }\n\n    private getCleanNumber(value: string): string {\n        const noCommaString = this.replaceCommaWithDot(value);\n        const cleanDecimals = this.cleanDecimals(noCommaString);\n        const cleanNumber = this.removeNonNumericCharacters(cleanDecimals);\n\n        if (cleanNumber && cleanNumber.length) {\n            return cleanNumber.substring(0, this.getMaxLength());\n        }\n\n        return cleanNumber;\n    }\n\n    private updateInputElementValue(cleanNumber: string): void {\n        if (!this.inputElement) {\n            // No need to update the native element if it doesn't yet exist.\n            return;\n        }\n\n        const inputElemValue = this.inputElement.nativeElement.value;\n        if (inputElemValue !== cleanNumber) {\n            // Need to update HTML input value because we let user input what he wants and then we clean model\n            this.inputElement.nativeElement.value = cleanNumber;\n        }\n    }\n\n    private setToEmpty(value: string): void {\n        // Empty means null, otherwise the backend doesn't handle numbers properly. We have to keep 'undefined'\n        // if 'undefined' otherwise Angular complains about the value of the model being changed without updating\n        // the view.\n        const emptyValue = value === '' ? null : value;\n        super.updateNgModel(emptyValue);\n        this.updateInputElementValue(emptyValue);\n    }\n\n    private manageHelpText(): void {\n        if (!!this.hideStandardHelpText) {\n            return;\n        }\n\n        if (!this.allowFreeInput) {\n            if (isDefined(this.min) && isDefined(this.max)) {\n                if (!!this.helpText?.length) {\n                    this.helpText += '<br/>';\n                } else {\n                    this.helpText = '';\n                }\n                this.helpText += this.dictionaryService.getKeySync(\n                    'foehn-input-number.standard-help-text.label',\n                    {\n                        minValue: formatNumberAsGiven(this.min.toString()),\n                        maxValue: formatNumberAsGiven(this.max.toString())\n                    }\n                );\n            }\n            return;\n        }\n\n        if (!!this.helpText?.length) {\n            this.helpText += '<br/>';\n        } else {\n            this.helpText = '';\n        }\n\n        const maxIntegerCount = this.allowDecimal\n            ? this.maxlength - this.maxDecimalCount - 1\n            : this.maxlength;\n\n        let maxAsString;\n        if (!!this.max) {\n            maxAsString = this.max;\n        } else {\n            maxAsString = '9'.repeat(maxIntegerCount);\n            if (this.allowDecimal) {\n                maxAsString += `.${'9'.repeat(this.maxDecimalCount)}`;\n            }\n        }\n\n        let minAsString;\n        if (this.min) {\n            minAsString = this.min;\n        } else {\n            minAsString = this.allowNegative\n                ? `-${'9'.repeat(maxIntegerCount - 1)}`\n                : '0';\n            if (this.allowDecimal) {\n                minAsString += this.allowNegative\n                    ? `.${'9'.repeat(this.maxDecimalCount || 0)}`\n                    : `.${'0'.repeat(this.maxDecimalCount || 0)}`;\n            }\n        }\n\n        this.helpText += this.dictionaryService.getKeySync(\n            'foehn-input-number.standard-help-text.label',\n            {\n                minValue: this.formatValue(minAsString),\n                maxValue: this.formatValue(maxAsString)\n            }\n        );\n    }\n\n    private formatValue(value: string | number): string {\n        if (this.allowDecimal) {\n            return formatDecimalCurrency(\n                value?.toString(),\n                this.maxDecimalCount,\n                this.maxlength,\n                this.allowFreeInput\n            );\n        }\n        return formatNonDecimalCurrency(value?.toString());\n    }\n}\n","<div\n    class=\"form-group clearable-input-form-group\"\n    [class.has-danger]=\"hasErrorsToDisplay()\"\n    [class.vd-form-group-danger]=\"hasErrorsToDisplay()\"\n    [attr.id]=\"buildId('Container')\"\n    tabindex=\"-1\"\n>\n    <div class=\"d-flex justify-content-between\">\n        <label\n            style=\"display: inline-block\"\n            [attr.for]=\"buildChildId()\"\n            *ngIf=\"label && type !== 'hidden'\"\n            [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n        >\n            <span [innerHTML]=\"label\"></span>\n            <span\n                *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n                aria-hidden=\"true\"\n            >\n                {{ 'foehn-input.optional' | fromDictionary }}\n            </span>\n        </label>\n\n        <foehn-help-modal\n            class=\"removePaddingButton\"\n            *ngIf=\"!!helpModal\"\n            [modalContent]=\"helpModal\"\n        ></foehn-help-modal>\n    </div>\n\n    <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n    <small\n        *ngIf=\"helpText && type !== 'hidden'\"\n        [attr.id]=\"buildChildId() + 'Help'\"\n        class=\"form-text text-secondary\"\n        [innerHTML]=\"helpText\"\n    ></small>\n\n    <ng-content></ng-content>\n\n    <input\n        [class.is-invalid]=\"hasErrorsToDisplay() || hasInheritErrorFromParent()\"\n        [class.clearable-input]=\"displayClearButton() | async\"\n        class=\"form-control\"\n        [name]=\"name || label\"\n        [attr.maxlength]=\"getMaxLength()\"\n        [attr.autocomplete]=\"getAutoComplete()\"\n        [attr.autocapitalize]=\"autocapitalize\"\n        [attr.disabled]=\"disabled ? 'disabled' : null\"\n        [attr.type]=\"type\"\n        [attr.pattern]=\"pattern\"\n        [attr.min]=\"min\"\n        [attr.max]=\"max\"\n        [attr.id]=\"buildChildId()\"\n        [attr.aria-describedby]=\"getDescribedBy()\"\n        [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n        [attr.aria-required]=\"required || null\"\n        [ngModel]=\"model\"\n        (paste)=\"onPaste($event)\"\n        (ngModelChange)=\"updateNgModel($event)\"\n        (input)=\"handleChange(entryComponent.value)\"\n        (keydown)=\"onKeydown($event)\"\n        (blur)=\"onBlur($event)\"\n        (focus)=\"onFocus($event)\"\n        #entryComponent\n    />\n\n    <button\n        type=\"button\"\n        [id]=\"buildChildId() + 'ClearButton'\"\n        *ngIf=\"displayClearButton() | async\"\n        class=\"clearable-input-clear-button btn\"\n        (click)=\"onClear()\"\n    >\n        <foehn-icon-times\n            class=\"clearable-input-clear-button-icon\"\n            title=\"Réinitialiser le champs\"\n        ></foehn-icon-times>\n    </button>\n</div>\n"]}
|
|
@@ -45,6 +45,7 @@ export class FoehnPictureUploadComponent extends FoehnInputComponent {
|
|
|
45
45
|
this.resizeToWidth = 300;
|
|
46
46
|
this.resizeToHeight = 0;
|
|
47
47
|
this.shouldDisplayFileSavedConfirmation = true;
|
|
48
|
+
this.acceptFormats = '.jpeg,.jpg,.png';
|
|
48
49
|
this.loading = false;
|
|
49
50
|
}
|
|
50
51
|
get hasPicture() {
|
|
@@ -60,9 +61,30 @@ export class FoehnPictureUploadComponent extends FoehnInputComponent {
|
|
|
60
61
|
});
|
|
61
62
|
}
|
|
62
63
|
onPictureSelection(imgEvent) {
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
64
|
+
const targetInput = imgEvent.target;
|
|
65
|
+
if (!targetInput?.files?.length || !targetInput.files[0]) {
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
const file = targetInput.files[0];
|
|
69
|
+
if (this.isValidFileType(file.name)) {
|
|
70
|
+
this.refreshErrors([[], true]);
|
|
71
|
+
this.pictureToCrop = imgEvent;
|
|
72
|
+
this.loading = true;
|
|
73
|
+
}
|
|
74
|
+
else {
|
|
75
|
+
const errors = [
|
|
76
|
+
{
|
|
77
|
+
name: this.name,
|
|
78
|
+
code: 'InvalidFileType',
|
|
79
|
+
message: this.dictionaryService.getKeySync('foehn-picture-upload.invalid-file-type', {
|
|
80
|
+
filename: file.name,
|
|
81
|
+
acceptFormats: this.acceptFormats
|
|
82
|
+
})
|
|
83
|
+
}
|
|
84
|
+
];
|
|
85
|
+
this.refreshErrors([errors, true]);
|
|
86
|
+
this.resetPictureSelection();
|
|
87
|
+
}
|
|
66
88
|
}
|
|
67
89
|
onPictureSelectionFailed() {
|
|
68
90
|
const errors = [
|
|
@@ -125,6 +147,13 @@ export class FoehnPictureUploadComponent extends FoehnInputComponent {
|
|
|
125
147
|
}
|
|
126
148
|
super.refreshErrors(results);
|
|
127
149
|
}
|
|
150
|
+
isValidFileType(filename) {
|
|
151
|
+
const acceptedFormats = this.acceptFormats
|
|
152
|
+
.split(',')
|
|
153
|
+
.map(ext => ext.toLowerCase());
|
|
154
|
+
const fileExtension = `.${filename.split('.').pop()}`.toLowerCase();
|
|
155
|
+
return acceptedFormats.includes(fileExtension);
|
|
156
|
+
}
|
|
128
157
|
}
|
|
129
158
|
FoehnPictureUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: FoehnPictureUploadComponent, deps: [{ token: i1.GrowlBrokerService }, { token: i2.MultiUploadService }, { token: i3.SdkDictionaryService }], target: i0.ɵɵFactoryTarget.Component });
|
|
130
159
|
FoehnPictureUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.11", type: FoehnPictureUploadComponent, selector: "foehn-picture-upload", inputs: { name: "name", key: "key", label: "label", baseUrl: "baseUrl", croppedPictureFilename: "croppedPictureFilename", resizeToWidth: "resizeToWidth", resizeToHeight: "resizeToHeight", shouldDisplayFileSavedConfirmation: "shouldDisplayFileSavedConfirmation" }, providers: [
|
|
@@ -133,7 +162,7 @@ FoehnPictureUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.
|
|
|
133
162
|
useExisting: forwardRef(() => FoehnPictureUploadComponent),
|
|
134
163
|
multi: true
|
|
135
164
|
}
|
|
136
|
-
], usesInheritance: true, ngImport: i0, template: "<div\n [attr.id]=\"buildId('Container')\"\n [class.has-danger]=\"hasErrors()\"\n [class.vd-form-group-danger]=\"hasErrors()\"\n class=\"form-group clearable-input-form-group\"\n tabindex=\"-1\"\n>\n <label\n [attr.for]=\"buildChildId()\"\n *ngIf=\"!!label\"\n [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n >\n <span [innerHTML]=\"label\"></span>\n <span\n *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n aria-hidden=\"true\"\n >\n {{ 'foehn-input.optional' | fromDictionary }}\n </span>\n </label>\n\n <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n <small\n *ngIf=\"helpText\"\n [attr.id]=\"buildChildId() + 'Help'\"\n class=\"form-text text-secondary\"\n [innerHTML]=\"helpText\"\n ></small>\n\n <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n <ng-content></ng-content>\n\n <div *ngIf=\"hasPicture\" class=\"row\">\n <div class=\"col-md-6\">\n <img\n [id]=\"buildChildId('Picture')\"\n [attr.alt]=\"label\"\n [src]=\"getDownloadUrl()\"\n />\n </div>\n <div class=\"col-md-4\">\n <button\n type=\"button\"\n class=\"btn btn-danger w-100\"\n [attr.id]=\"buildChildId('DeleteButton')\"\n (click)=\"deleteFile()\"\n >\n {{\n 'foehn-picture-upload.delete-picture-label' | fromDictionary\n }}\n </button>\n </div>\n </div>\n\n <div *ngIf=\"!hasPicture && !croppedPictureAsBase64Url\">\n <input\n #inputFile\n type=\"file\"\n class=\"form-control-file d-none\"\n [attr.id]=\"name\"\n [attr.accept]=\"
|
|
165
|
+
], usesInheritance: true, ngImport: i0, template: "<div\n [attr.id]=\"buildId('Container')\"\n [class.has-danger]=\"hasErrors()\"\n [class.vd-form-group-danger]=\"hasErrors()\"\n class=\"form-group clearable-input-form-group\"\n tabindex=\"-1\"\n>\n <label\n [attr.for]=\"buildChildId()\"\n *ngIf=\"!!label\"\n [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n >\n <span [innerHTML]=\"label\"></span>\n <span\n *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n aria-hidden=\"true\"\n >\n {{ 'foehn-input.optional' | fromDictionary }}\n </span>\n </label>\n\n <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n <small\n *ngIf=\"helpText\"\n [attr.id]=\"buildChildId() + 'Help'\"\n class=\"form-text text-secondary\"\n [innerHTML]=\"helpText\"\n ></small>\n\n <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n <ng-content></ng-content>\n\n <div *ngIf=\"hasPicture\" class=\"row\">\n <div class=\"col-md-6\">\n <img\n [id]=\"buildChildId('Picture')\"\n [attr.alt]=\"label\"\n [src]=\"getDownloadUrl()\"\n />\n </div>\n <div class=\"col-md-4\">\n <button\n type=\"button\"\n class=\"btn btn-danger w-100\"\n [attr.id]=\"buildChildId('DeleteButton')\"\n (click)=\"deleteFile()\"\n >\n {{\n 'foehn-picture-upload.delete-picture-label' | fromDictionary\n }}\n </button>\n </div>\n </div>\n\n <div *ngIf=\"!hasPicture && !croppedPictureAsBase64Url\">\n <input\n #inputFile\n type=\"file\"\n class=\"form-control-file d-none\"\n [attr.id]=\"name\"\n [attr.accept]=\"acceptFormats\"\n [multiple]=\"false\"\n [name]=\"name\"\n (change)=\"onPictureSelection($event)\"\n />\n\n <button\n type=\"button\"\n class=\"btn btn-primary my-2\"\n [attr.id]=\"buildChildId('ChooseButton')\"\n [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n [attr.aria-describedby]=\"buildId('ErrorsContainer')\"\n (click)=\"inputFile.click()\"\n [attr.disabled]=\"loading ? 'disabled' : null\"\n >\n {{ 'foehn-picture-upload.choose-button-label' | fromDictionary }}\n </button>\n </div>\n\n <div *ngIf=\"loading\">\n {{ 'foehn-picture-upload.loading-label' | fromDictionary }}\n </div>\n\n <div *ngIf=\"!hasPicture\" class=\"row\">\n <div class=\"col-md-6\">\n <image-cropper\n [id]=\"buildChildId('Cropper')\"\n (cropperReady)=\"loading = false\"\n (imageCropped)=\"onImageCropped($event)\"\n (loadImageFailed)=\"onPictureSelectionFailed()\"\n [aspectRatio]=\"3 / 4\"\n [imageChangedEvent]=\"pictureToCrop\"\n [maintainAspectRatio]=\"true\"\n [resizeToWidth]=\"resizeToWidth\"\n [resizeToHeight]=\"resizeToHeight\"\n [onlyScaleDown]=\"true\"\n format=\"png\"\n ></image-cropper>\n </div>\n\n <div *ngIf=\"!hasPicture\" class=\"col-md-4\">\n <div class=\"row mb-3\">\n <button\n *ngIf=\"croppedPictureAsBase64Url\"\n type=\"button\"\n class=\"btn btn-primary w-100\"\n [attr.id]=\"buildChildId('SaveButton')\"\n (click)=\"savePicture()\"\n >\n {{\n 'foehn-picture-upload.validate-selection-label'\n | fromDictionary\n }}\n </button>\n </div>\n <div class=\"row\">\n <button\n *ngIf=\"croppedPictureAsBase64Url\"\n type=\"button\"\n class=\"btn btn-secondary w-100\"\n [attr.id]=\"buildChildId('CancelButton')\"\n (click)=\"resetPictureSelection()\"\n >\n {{\n 'foehn-picture-upload.cancel-selection-label'\n | fromDictionary\n }}\n </button>\n </div>\n </div>\n </div>\n</div>\n", components: [{ type: i4.FoehnValidationAlertsComponent, selector: "foehn-validation-alerts", inputs: ["component", "shouldErrorsBeLive"] }, { type: i5.ImageCropperComponent, selector: "image-cropper", inputs: ["imageChangedEvent", "imageURL", "imageBase64", "imageFile", "format", "transform", "maintainAspectRatio", "aspectRatio", "resizeToWidth", "resizeToHeight", "cropperMinWidth", "cropperMinHeight", "cropperMaxHeight", "cropperMaxWidth", "cropperStaticWidth", "cropperStaticHeight", "canvasRotation", "initialStepSize", "roundCropper", "onlyScaleDown", "imageQuality", "autoCrop", "backgroundColor", "containWithinAspectRatio", "hideResizeSquares", "cropper", "alignImage", "disabled"], outputs: ["imageCropped", "startCropImage", "imageLoaded", "cropperReady", "loadImageFailed"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], pipes: { "fromDictionary": i8.SdkDictionaryPipe } });
|
|
137
166
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: FoehnPictureUploadComponent, decorators: [{
|
|
138
167
|
type: Component,
|
|
139
168
|
args: [{ selector: 'foehn-picture-upload', providers: [
|
|
@@ -142,7 +171,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImpo
|
|
|
142
171
|
useExisting: forwardRef(() => FoehnPictureUploadComponent),
|
|
143
172
|
multi: true
|
|
144
173
|
}
|
|
145
|
-
], template: "<div\n [attr.id]=\"buildId('Container')\"\n [class.has-danger]=\"hasErrors()\"\n [class.vd-form-group-danger]=\"hasErrors()\"\n class=\"form-group clearable-input-form-group\"\n tabindex=\"-1\"\n>\n <label\n [attr.for]=\"buildChildId()\"\n *ngIf=\"!!label\"\n [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n >\n <span [innerHTML]=\"label\"></span>\n <span\n *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n aria-hidden=\"true\"\n >\n {{ 'foehn-input.optional' | fromDictionary }}\n </span>\n </label>\n\n <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n <small\n *ngIf=\"helpText\"\n [attr.id]=\"buildChildId() + 'Help'\"\n class=\"form-text text-secondary\"\n [innerHTML]=\"helpText\"\n ></small>\n\n <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n <ng-content></ng-content>\n\n <div *ngIf=\"hasPicture\" class=\"row\">\n <div class=\"col-md-6\">\n <img\n [id]=\"buildChildId('Picture')\"\n [attr.alt]=\"label\"\n [src]=\"getDownloadUrl()\"\n />\n </div>\n <div class=\"col-md-4\">\n <button\n type=\"button\"\n class=\"btn btn-danger w-100\"\n [attr.id]=\"buildChildId('DeleteButton')\"\n (click)=\"deleteFile()\"\n >\n {{\n 'foehn-picture-upload.delete-picture-label' | fromDictionary\n }}\n </button>\n </div>\n </div>\n\n <div *ngIf=\"!hasPicture && !croppedPictureAsBase64Url\">\n <input\n #inputFile\n type=\"file\"\n class=\"form-control-file d-none\"\n [attr.id]=\"name\"\n [attr.accept]=\"
|
|
174
|
+
], template: "<div\n [attr.id]=\"buildId('Container')\"\n [class.has-danger]=\"hasErrors()\"\n [class.vd-form-group-danger]=\"hasErrors()\"\n class=\"form-group clearable-input-form-group\"\n tabindex=\"-1\"\n>\n <label\n [attr.for]=\"buildChildId()\"\n *ngIf=\"!!label\"\n [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n >\n <span [innerHTML]=\"label\"></span>\n <span\n *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n aria-hidden=\"true\"\n >\n {{ 'foehn-input.optional' | fromDictionary }}\n </span>\n </label>\n\n <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n <small\n *ngIf=\"helpText\"\n [attr.id]=\"buildChildId() + 'Help'\"\n class=\"form-text text-secondary\"\n [innerHTML]=\"helpText\"\n ></small>\n\n <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n <ng-content></ng-content>\n\n <div *ngIf=\"hasPicture\" class=\"row\">\n <div class=\"col-md-6\">\n <img\n [id]=\"buildChildId('Picture')\"\n [attr.alt]=\"label\"\n [src]=\"getDownloadUrl()\"\n />\n </div>\n <div class=\"col-md-4\">\n <button\n type=\"button\"\n class=\"btn btn-danger w-100\"\n [attr.id]=\"buildChildId('DeleteButton')\"\n (click)=\"deleteFile()\"\n >\n {{\n 'foehn-picture-upload.delete-picture-label' | fromDictionary\n }}\n </button>\n </div>\n </div>\n\n <div *ngIf=\"!hasPicture && !croppedPictureAsBase64Url\">\n <input\n #inputFile\n type=\"file\"\n class=\"form-control-file d-none\"\n [attr.id]=\"name\"\n [attr.accept]=\"acceptFormats\"\n [multiple]=\"false\"\n [name]=\"name\"\n (change)=\"onPictureSelection($event)\"\n />\n\n <button\n type=\"button\"\n class=\"btn btn-primary my-2\"\n [attr.id]=\"buildChildId('ChooseButton')\"\n [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n [attr.aria-describedby]=\"buildId('ErrorsContainer')\"\n (click)=\"inputFile.click()\"\n [attr.disabled]=\"loading ? 'disabled' : null\"\n >\n {{ 'foehn-picture-upload.choose-button-label' | fromDictionary }}\n </button>\n </div>\n\n <div *ngIf=\"loading\">\n {{ 'foehn-picture-upload.loading-label' | fromDictionary }}\n </div>\n\n <div *ngIf=\"!hasPicture\" class=\"row\">\n <div class=\"col-md-6\">\n <image-cropper\n [id]=\"buildChildId('Cropper')\"\n (cropperReady)=\"loading = false\"\n (imageCropped)=\"onImageCropped($event)\"\n (loadImageFailed)=\"onPictureSelectionFailed()\"\n [aspectRatio]=\"3 / 4\"\n [imageChangedEvent]=\"pictureToCrop\"\n [maintainAspectRatio]=\"true\"\n [resizeToWidth]=\"resizeToWidth\"\n [resizeToHeight]=\"resizeToHeight\"\n [onlyScaleDown]=\"true\"\n format=\"png\"\n ></image-cropper>\n </div>\n\n <div *ngIf=\"!hasPicture\" class=\"col-md-4\">\n <div class=\"row mb-3\">\n <button\n *ngIf=\"croppedPictureAsBase64Url\"\n type=\"button\"\n class=\"btn btn-primary w-100\"\n [attr.id]=\"buildChildId('SaveButton')\"\n (click)=\"savePicture()\"\n >\n {{\n 'foehn-picture-upload.validate-selection-label'\n | fromDictionary\n }}\n </button>\n </div>\n <div class=\"row\">\n <button\n *ngIf=\"croppedPictureAsBase64Url\"\n type=\"button\"\n class=\"btn btn-secondary w-100\"\n [attr.id]=\"buildChildId('CancelButton')\"\n (click)=\"resetPictureSelection()\"\n >\n {{\n 'foehn-picture-upload.cancel-selection-label'\n | fromDictionary\n }}\n </button>\n </div>\n </div>\n </div>\n</div>\n" }]
|
|
146
175
|
}], ctorParameters: function () { return [{ type: i1.GrowlBrokerService }, { type: i2.MultiUploadService }, { type: i3.SdkDictionaryService }]; }, propDecorators: { name: [{
|
|
147
176
|
type: Input
|
|
148
177
|
}], key: [{
|
|
@@ -160,4 +189,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImpo
|
|
|
160
189
|
}], shouldDisplayFileSavedConfirmation: [{
|
|
161
190
|
type: Input
|
|
162
191
|
}] } });
|
|
163
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"foehn-picture-upload.component.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-picture-upload/foehn-picture-upload.component.ts","../../../../../projects/prestations-ng/src/foehn-upload/foehn-picture-upload/foehn-picture-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, forwardRef, Input, OnInit } from '@angular/core';\nimport { distinctUntilChanged, filter } from 'rxjs/operators';\n\nimport { Subscription } from 'rxjs';\nimport { FoehnInputComponent } from '../../foehn-input/foehn-input.component';\nimport { DocumentReference } from '../../foehn-upload/document-reference';\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { MultiUploadService } from '../../foehn-upload/foehn-multi-upload/multi-upload.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { FormError } from '../../form-error';\nimport { ImageCroppedEvent } from 'ngx-image-cropper';\n\nconst blobToFile = (blobData: Blob, fileName: string): File => {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const blob: any = blobData;\n    blob.name = fileName;\n    return blob as File;\n};\n\nconst convertBase64UrlToFile = (\n    croppedImageasBase64Url: string,\n    filename: string\n): File => {\n    const arr = croppedImageasBase64Url.split(',');\n    const mime = arr[0].match(/:(.*?);/)[1];\n    const bstr = atob(arr[1]);\n    let n = bstr.length;\n    const u8arr = new Uint8Array(n);\n    while (n--) {\n        u8arr[n] = bstr.charCodeAt(n);\n    }\n\n    if (!navigator.userAgent.includes('Edge')) {\n        // detect if not Edge\n        return new File([u8arr], filename, { type: mime });\n    } else {\n        // Edge support the FileAPI but not the File constructor.\n        return blobToFile(new Blob([u8arr], { type: mime }), filename);\n    }\n};\n\n@Component({\n    selector: 'foehn-picture-upload',\n    templateUrl: './foehn-picture-upload.component.html',\n    providers: [\n        {\n            provide: FoehnInputComponent,\n            useExisting: forwardRef(() => FoehnPictureUploadComponent),\n            multi: true\n        }\n    ]\n})\nexport class FoehnPictureUploadComponent\n    extends FoehnInputComponent<DocumentReference[]>\n    implements OnInit {\n    @Input()\n    name: string;\n\n    @Input()\n    key: string;\n\n    @Input()\n    label: string;\n\n    @Input()\n    baseUrl = 'api/document';\n\n    @Input()\n    croppedPictureFilename = 'photo.png';\n\n    @Input()\n    resizeToWidth = 300;\n\n    @Input()\n    resizeToHeight = 0;\n\n    @Input()\n    shouldDisplayFileSavedConfirmation = true;\n\n    pictureToCrop: Event;\n    croppedPictureAsBase64Url: string;\n    loading = false;\n\n    protected currentLanguageSubscription: Subscription;\n    protected currentLanguage: string;\n\n    constructor(\n        private growlService: GrowlBrokerService,\n        private uploadService: MultiUploadService,\n        private dictionaryService: SdkDictionaryService\n    ) {\n        super();\n    }\n\n    get hasPicture(): boolean {\n        return this.model && this.model.length && !!this.model[0].reference;\n    }\n\n    ngOnInit(): void {\n        super.ngOnInit();\n        this.currentLanguageSubscription = this.dictionaryService\n            .getCurrentLanguageCode()\n            .pipe(distinctUntilChanged())\n            .subscribe(language => {\n                this.currentLanguage = language;\n            });\n    }\n\n    onPictureSelection(imgEvent: Event): void {\n        this.refreshErrors([[], true]);\n        this.pictureToCrop = imgEvent;\n        this.loading = true;\n    }\n\n    onPictureSelectionFailed(): void {\n        const errors = [\n            {\n                name: this.name,\n                code: 'LoadingFailure',\n                message: this.dictionaryService.getKeySync(\n                    'foehn-picture-upload.loading-failure-label'\n                )\n            }\n        ];\n        this.refreshErrors([errors, true]);\n        this.resetPictureSelection();\n    }\n\n    onImageCropped(croppedEvent: ImageCroppedEvent): void {\n        this.croppedPictureAsBase64Url = croppedEvent.base64;\n    }\n\n    resetPictureSelection(): void {\n        this.croppedPictureAsBase64Url = null;\n        this.pictureToCrop = null;\n        this.loading = false;\n    }\n\n    savePicture(): void {\n        this.markAsPristine();\n\n        const photo = convertBase64UrlToFile(\n            this.croppedPictureAsBase64Url,\n            this.croppedPictureFilename\n        );\n\n        this.uploadService\n            .uploadDocuments(\n                this.baseUrl,\n                this.name,\n                this.label,\n                [photo],\n                this.key,\n                false,\n                this.currentLanguage,\n                this.shouldDisplayFileSavedConfirmation\n            )\n            .pipe(filter(result => !!result))\n            .subscribe(({ documents, errors }) => {\n                this.resetPictureSelection();\n                if (!!documents.length) {\n                    this.markAsDirty();\n                    this.model = documents;\n                    this.refreshErrors([[], true]);\n                } else {\n                    this.model = [];\n                    this.refreshErrors([errors, true]);\n                }\n            });\n    }\n\n    deleteFile(): void {\n        this.uploadService\n            .deleteDocument(this.baseUrl, this.model[0])\n            .subscribe(() => {\n                this.model = [];\n                this.resetPictureSelection();\n            });\n    }\n\n    getDownloadUrl(): string {\n        return this.uploadService.getDownloadUrl(this.baseUrl, this.model[0]);\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    refreshErrors(results: any[]): void {\n        const errors: FormError[] = results[0];\n\n        const isInSelectionMode =\n            !this.hasPicture && !!this.croppedPictureAsBase64Url;\n        if (isInSelectionMode) {\n            const notNullError = errors.find(\n                e => e.name === this.name && e.code === 'NotEmpty'\n            );\n            if (notNullError) {\n                notNullError.message = this.dictionaryService.getKeySync(\n                    'foehn-picture-upload.selection-not-validated-label'\n                );\n            }\n        }\n\n        super.refreshErrors(results);\n    }\n}\n","<div\n    [attr.id]=\"buildId('Container')\"\n    [class.has-danger]=\"hasErrors()\"\n    [class.vd-form-group-danger]=\"hasErrors()\"\n    class=\"form-group clearable-input-form-group\"\n    tabindex=\"-1\"\n>\n    <label\n        [attr.for]=\"buildChildId()\"\n        *ngIf=\"!!label\"\n        [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n    >\n        <span [innerHTML]=\"label\"></span>\n        <span\n            *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n            aria-hidden=\"true\"\n        >\n            {{ 'foehn-input.optional' | fromDictionary }}\n        </span>\n    </label>\n\n    <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n    <small\n        *ngIf=\"helpText\"\n        [attr.id]=\"buildChildId() + 'Help'\"\n        class=\"form-text text-secondary\"\n        [innerHTML]=\"helpText\"\n    ></small>\n\n    <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n    <ng-content></ng-content>\n\n    <div *ngIf=\"hasPicture\" class=\"row\">\n        <div class=\"col-md-6\">\n            <img\n                [id]=\"buildChildId('Picture')\"\n                [attr.alt]=\"label\"\n                [src]=\"getDownloadUrl()\"\n            />\n        </div>\n        <div class=\"col-md-4\">\n            <button\n                type=\"button\"\n                class=\"btn btn-danger w-100\"\n                [attr.id]=\"buildChildId('DeleteButton')\"\n                (click)=\"deleteFile()\"\n            >\n                {{\n                    'foehn-picture-upload.delete-picture-label' | fromDictionary\n                }}\n            </button>\n        </div>\n    </div>\n\n    <div *ngIf=\"!hasPicture && !croppedPictureAsBase64Url\">\n        <input\n            #inputFile\n            type=\"file\"\n            class=\"form-control-file d-none\"\n            [attr.id]=\"name\"\n            [attr.accept]=\"'.jpeg,.jpg,.png'\"\n            [multiple]=\"false\"\n            [name]=\"name\"\n            (change)=\"onPictureSelection($event)\"\n        />\n\n        <button\n            type=\"button\"\n            class=\"btn btn-primary my-2\"\n            [attr.id]=\"buildChildId('ChooseButton')\"\n            [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n            [attr.aria-describedby]=\"buildId('ErrorsContainer')\"\n            (click)=\"inputFile.click()\"\n            [attr.disabled]=\"loading ? 'disabled' : null\"\n        >\n            {{ 'foehn-picture-upload.choose-button-label' | fromDictionary }}\n        </button>\n    </div>\n\n    <div *ngIf=\"loading\">\n        {{ 'foehn-picture-upload.loading-label' | fromDictionary }}\n    </div>\n\n    <div *ngIf=\"!hasPicture\" class=\"row\">\n        <div class=\"col-md-6\">\n            <image-cropper\n                [id]=\"buildChildId('Cropper')\"\n                (cropperReady)=\"loading = false\"\n                (imageCropped)=\"onImageCropped($event)\"\n                (loadImageFailed)=\"onPictureSelectionFailed()\"\n                [aspectRatio]=\"3 / 4\"\n                [imageChangedEvent]=\"pictureToCrop\"\n                [maintainAspectRatio]=\"true\"\n                [resizeToWidth]=\"resizeToWidth\"\n                [resizeToHeight]=\"resizeToHeight\"\n                [onlyScaleDown]=\"true\"\n                format=\"png\"\n            ></image-cropper>\n        </div>\n\n        <div *ngIf=\"!hasPicture\" class=\"col-md-4\">\n            <div class=\"row mb-3\">\n                <button\n                    *ngIf=\"croppedPictureAsBase64Url\"\n                    type=\"button\"\n                    class=\"btn btn-primary w-100\"\n                    [attr.id]=\"buildChildId('SaveButton')\"\n                    (click)=\"savePicture()\"\n                >\n                    {{\n                        'foehn-picture-upload.validate-selection-label'\n                            | fromDictionary\n                    }}\n                </button>\n            </div>\n            <div class=\"row\">\n                <button\n                    *ngIf=\"croppedPictureAsBase64Url\"\n                    type=\"button\"\n                    class=\"btn btn-secondary w-100\"\n                    [attr.id]=\"buildChildId('CancelButton')\"\n                    (click)=\"resetPictureSelection()\"\n                >\n                    {{\n                        'foehn-picture-upload.cancel-selection-label'\n                            | fromDictionary\n                    }}\n                </button>\n            </div>\n        </div>\n    </div>\n</div>\n"]}
|
|
192
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"foehn-picture-upload.component.js","sourceRoot":"","sources":["../../../../../projects/prestations-ng/src/foehn-upload/foehn-picture-upload/foehn-picture-upload.component.ts","../../../../../projects/prestations-ng/src/foehn-upload/foehn-picture-upload/foehn-picture-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, forwardRef, Input, OnInit } from '@angular/core';\nimport { distinctUntilChanged, filter } from 'rxjs/operators';\n\nimport { Subscription } from 'rxjs';\nimport { FoehnInputComponent } from '../../foehn-input/foehn-input.component';\nimport { DocumentReference } from '../../foehn-upload/document-reference';\nimport { GrowlBrokerService } from '../../foehn-growl/growl-broker.service';\nimport { MultiUploadService } from '../../foehn-upload/foehn-multi-upload/multi-upload.service';\nimport { SdkDictionaryService } from '../../sdk-dictionary/sdk-dictionary.service';\nimport { FormError } from '../../form-error';\nimport { ImageCroppedEvent } from 'ngx-image-cropper';\n\nconst blobToFile = (blobData: Blob, fileName: string): File => {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const blob: any = blobData;\n    blob.name = fileName;\n    return blob as File;\n};\n\nconst convertBase64UrlToFile = (\n    croppedImageasBase64Url: string,\n    filename: string\n): File => {\n    const arr = croppedImageasBase64Url.split(',');\n    const mime = arr[0].match(/:(.*?);/)[1];\n    const bstr = atob(arr[1]);\n    let n = bstr.length;\n    const u8arr = new Uint8Array(n);\n    while (n--) {\n        u8arr[n] = bstr.charCodeAt(n);\n    }\n\n    if (!navigator.userAgent.includes('Edge')) {\n        // detect if not Edge\n        return new File([u8arr], filename, { type: mime });\n    } else {\n        // Edge support the FileAPI but not the File constructor.\n        return blobToFile(new Blob([u8arr], { type: mime }), filename);\n    }\n};\n\n@Component({\n    selector: 'foehn-picture-upload',\n    templateUrl: './foehn-picture-upload.component.html',\n    providers: [\n        {\n            provide: FoehnInputComponent,\n            useExisting: forwardRef(() => FoehnPictureUploadComponent),\n            multi: true\n        }\n    ]\n})\nexport class FoehnPictureUploadComponent\n    extends FoehnInputComponent<DocumentReference[]>\n    implements OnInit {\n    @Input()\n    name: string;\n\n    @Input()\n    key: string;\n\n    @Input()\n    label: string;\n\n    @Input()\n    baseUrl = 'api/document';\n\n    @Input()\n    croppedPictureFilename = 'photo.png';\n\n    @Input()\n    resizeToWidth = 300;\n\n    @Input()\n    resizeToHeight = 0;\n\n    @Input()\n    shouldDisplayFileSavedConfirmation = true;\n\n    readonly acceptFormats = '.jpeg,.jpg,.png';\n\n    pictureToCrop: Event;\n    croppedPictureAsBase64Url: string;\n    loading = false;\n\n    protected currentLanguageSubscription: Subscription;\n    protected currentLanguage: string;\n\n    constructor(\n        private growlService: GrowlBrokerService,\n        private uploadService: MultiUploadService,\n        private dictionaryService: SdkDictionaryService\n    ) {\n        super();\n    }\n\n    get hasPicture(): boolean {\n        return this.model && this.model.length && !!this.model[0].reference;\n    }\n\n    ngOnInit(): void {\n        super.ngOnInit();\n        this.currentLanguageSubscription = this.dictionaryService\n            .getCurrentLanguageCode()\n            .pipe(distinctUntilChanged())\n            .subscribe(language => {\n                this.currentLanguage = language;\n            });\n    }\n\n    onPictureSelection(imgEvent: Event): void {\n        const targetInput = imgEvent.target as HTMLInputElement;\n        if (!targetInput?.files?.length || !targetInput.files[0]) {\n            return;\n        }\n        const file = targetInput.files[0];\n\n        if (this.isValidFileType(file.name)) {\n            this.refreshErrors([[], true]);\n            this.pictureToCrop = imgEvent;\n            this.loading = true;\n        } else {\n            const errors = [\n                {\n                    name: this.name,\n                    code: 'InvalidFileType',\n                    message: this.dictionaryService.getKeySync(\n                        'foehn-picture-upload.invalid-file-type',\n                        {\n                            filename: file.name,\n                            acceptFormats: this.acceptFormats\n                        }\n                    )\n                }\n            ];\n            this.refreshErrors([errors, true]);\n            this.resetPictureSelection();\n        }\n    }\n\n    onPictureSelectionFailed(): void {\n        const errors = [\n            {\n                name: this.name,\n                code: 'LoadingFailure',\n                message: this.dictionaryService.getKeySync(\n                    'foehn-picture-upload.loading-failure-label'\n                )\n            }\n        ];\n        this.refreshErrors([errors, true]);\n        this.resetPictureSelection();\n    }\n\n    onImageCropped(croppedEvent: ImageCroppedEvent): void {\n        this.croppedPictureAsBase64Url = croppedEvent.base64;\n    }\n\n    resetPictureSelection(): void {\n        this.croppedPictureAsBase64Url = null;\n        this.pictureToCrop = null;\n        this.loading = false;\n    }\n\n    savePicture(): void {\n        this.markAsPristine();\n\n        const photo = convertBase64UrlToFile(\n            this.croppedPictureAsBase64Url,\n            this.croppedPictureFilename\n        );\n\n        this.uploadService\n            .uploadDocuments(\n                this.baseUrl,\n                this.name,\n                this.label,\n                [photo],\n                this.key,\n                false,\n                this.currentLanguage,\n                this.shouldDisplayFileSavedConfirmation\n            )\n            .pipe(filter(result => !!result))\n            .subscribe(({ documents, errors }) => {\n                this.resetPictureSelection();\n                if (!!documents.length) {\n                    this.markAsDirty();\n                    this.model = documents;\n                    this.refreshErrors([[], true]);\n                } else {\n                    this.model = [];\n                    this.refreshErrors([errors, true]);\n                }\n            });\n    }\n\n    deleteFile(): void {\n        this.uploadService\n            .deleteDocument(this.baseUrl, this.model[0])\n            .subscribe(() => {\n                this.model = [];\n                this.resetPictureSelection();\n            });\n    }\n\n    getDownloadUrl(): string {\n        return this.uploadService.getDownloadUrl(this.baseUrl, this.model[0]);\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    refreshErrors(results: any[]): void {\n        const errors: FormError[] = results[0];\n        const isInSelectionMode =\n            !this.hasPicture && !!this.croppedPictureAsBase64Url;\n        if (isInSelectionMode) {\n            const notNullError = errors.find(\n                e => e.name === this.name && e.code === 'NotEmpty'\n            );\n            if (notNullError) {\n                notNullError.message = this.dictionaryService.getKeySync(\n                    'foehn-picture-upload.selection-not-validated-label'\n                );\n            }\n        }\n\n        super.refreshErrors(results);\n    }\n\n    private isValidFileType(filename: string): boolean {\n        const acceptedFormats = this.acceptFormats\n            .split(',')\n            .map(ext => ext.toLowerCase());\n        const fileExtension = `.${filename.split('.').pop()}`.toLowerCase();\n        return acceptedFormats.includes(fileExtension);\n    }\n}\n","<div\n    [attr.id]=\"buildId('Container')\"\n    [class.has-danger]=\"hasErrors()\"\n    [class.vd-form-group-danger]=\"hasErrors()\"\n    class=\"form-group clearable-input-form-group\"\n    tabindex=\"-1\"\n>\n    <label\n        [attr.for]=\"buildChildId()\"\n        *ngIf=\"!!label\"\n        [ngClass]=\"isLabelSrOnly ? 'sr-only' : labelStyleModifier\"\n    >\n        <span [innerHTML]=\"label\"></span>\n        <span\n            *ngIf=\"!required && !hideNotRequiredExtraLabel\"\n            aria-hidden=\"true\"\n        >\n            {{ 'foehn-input.optional' | fromDictionary }}\n        </span>\n    </label>\n\n    <foehn-validation-alerts [component]=\"this\"></foehn-validation-alerts>\n\n    <small\n        *ngIf=\"helpText\"\n        [attr.id]=\"buildChildId() + 'Help'\"\n        class=\"form-text text-secondary\"\n        [innerHTML]=\"helpText\"\n    ></small>\n\n    <input type=\"hidden\" [name]=\"name || label\" [ngModel]=\"model\" />\n\n    <ng-content></ng-content>\n\n    <div *ngIf=\"hasPicture\" class=\"row\">\n        <div class=\"col-md-6\">\n            <img\n                [id]=\"buildChildId('Picture')\"\n                [attr.alt]=\"label\"\n                [src]=\"getDownloadUrl()\"\n            />\n        </div>\n        <div class=\"col-md-4\">\n            <button\n                type=\"button\"\n                class=\"btn btn-danger w-100\"\n                [attr.id]=\"buildChildId('DeleteButton')\"\n                (click)=\"deleteFile()\"\n            >\n                {{\n                    'foehn-picture-upload.delete-picture-label' | fromDictionary\n                }}\n            </button>\n        </div>\n    </div>\n\n    <div *ngIf=\"!hasPicture && !croppedPictureAsBase64Url\">\n        <input\n            #inputFile\n            type=\"file\"\n            class=\"form-control-file d-none\"\n            [attr.id]=\"name\"\n            [attr.accept]=\"acceptFormats\"\n            [multiple]=\"false\"\n            [name]=\"name\"\n            (change)=\"onPictureSelection($event)\"\n        />\n\n        <button\n            type=\"button\"\n            class=\"btn btn-primary my-2\"\n            [attr.id]=\"buildChildId('ChooseButton')\"\n            [attr.aria-invalid]=\"hasErrorsToDisplay() || null\"\n            [attr.aria-describedby]=\"buildId('ErrorsContainer')\"\n            (click)=\"inputFile.click()\"\n            [attr.disabled]=\"loading ? 'disabled' : null\"\n        >\n            {{ 'foehn-picture-upload.choose-button-label' | fromDictionary }}\n        </button>\n    </div>\n\n    <div *ngIf=\"loading\">\n        {{ 'foehn-picture-upload.loading-label' | fromDictionary }}\n    </div>\n\n    <div *ngIf=\"!hasPicture\" class=\"row\">\n        <div class=\"col-md-6\">\n            <image-cropper\n                [id]=\"buildChildId('Cropper')\"\n                (cropperReady)=\"loading = false\"\n                (imageCropped)=\"onImageCropped($event)\"\n                (loadImageFailed)=\"onPictureSelectionFailed()\"\n                [aspectRatio]=\"3 / 4\"\n                [imageChangedEvent]=\"pictureToCrop\"\n                [maintainAspectRatio]=\"true\"\n                [resizeToWidth]=\"resizeToWidth\"\n                [resizeToHeight]=\"resizeToHeight\"\n                [onlyScaleDown]=\"true\"\n                format=\"png\"\n            ></image-cropper>\n        </div>\n\n        <div *ngIf=\"!hasPicture\" class=\"col-md-4\">\n            <div class=\"row mb-3\">\n                <button\n                    *ngIf=\"croppedPictureAsBase64Url\"\n                    type=\"button\"\n                    class=\"btn btn-primary w-100\"\n                    [attr.id]=\"buildChildId('SaveButton')\"\n                    (click)=\"savePicture()\"\n                >\n                    {{\n                        'foehn-picture-upload.validate-selection-label'\n                            | fromDictionary\n                    }}\n                </button>\n            </div>\n            <div class=\"row\">\n                <button\n                    *ngIf=\"croppedPictureAsBase64Url\"\n                    type=\"button\"\n                    class=\"btn btn-secondary w-100\"\n                    [attr.id]=\"buildChildId('CancelButton')\"\n                    (click)=\"resetPictureSelection()\"\n                >\n                    {{\n                        'foehn-picture-upload.cancel-selection-label'\n                            | fromDictionary\n                    }}\n                </button>\n            </div>\n        </div>\n    </div>\n</div>\n"]}
|