@esfaenza/forms-and-validations 15.2.55 → 15.2.57
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/esm2020/lib/forms/form-autocomplete/form-autocomplete.component.mjs +25 -13
- package/fesm2015/esfaenza-forms-and-validations.mjs +24 -12
- package/fesm2015/esfaenza-forms-and-validations.mjs.map +1 -1
- package/fesm2020/esfaenza-forms-and-validations.mjs +24 -12
- package/fesm2020/esfaenza-forms-and-validations.mjs.map +1 -1
- package/package.json +6 -6
|
@@ -98,18 +98,22 @@ export class FormAutocompleteComponent extends BaseFormControl {
|
|
|
98
98
|
onFailure?.();
|
|
99
99
|
return;
|
|
100
100
|
}
|
|
101
|
-
this.
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
if (
|
|
106
|
-
|
|
101
|
+
this.log("filtersource enqueue");
|
|
102
|
+
this.throttla("filtersource", () => {
|
|
103
|
+
this.log("filtersource fire");
|
|
104
|
+
this.doSearchProtected(tmpModel, true, this.SearchFunctionContext, (t) => {
|
|
105
|
+
if (t && t.length > 0) {
|
|
106
|
+
this.Source = t;
|
|
107
|
+
this.tryBindSourceDisplay();
|
|
108
|
+
if (!this.Multi)
|
|
109
|
+
setTimeout(() => { this.finalizeValue(tmpModel); });
|
|
110
|
+
else
|
|
111
|
+
this.cdr.markForCheck();
|
|
112
|
+
}
|
|
107
113
|
else
|
|
108
|
-
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
onFailure?.();
|
|
112
|
-
});
|
|
114
|
+
onFailure?.();
|
|
115
|
+
});
|
|
116
|
+
}, 400);
|
|
113
117
|
}
|
|
114
118
|
/**
|
|
115
119
|
* Evento di filtro della sorgente dati in base all'input utente
|
|
@@ -138,6 +142,7 @@ export class FormAutocompleteComponent extends BaseFormControl {
|
|
|
138
142
|
return;
|
|
139
143
|
}
|
|
140
144
|
if (!event || event.length < this.MinChars) {
|
|
145
|
+
this.log("search event ignored: " + event);
|
|
141
146
|
this.FilteredBoundSource = [];
|
|
142
147
|
this.cdr.markForCheck();
|
|
143
148
|
return;
|
|
@@ -145,8 +150,11 @@ export class FormAutocompleteComponent extends BaseFormControl {
|
|
|
145
150
|
if (!this.SearchFunction && (!this.Source || this.Source.length == 0))
|
|
146
151
|
throw "Impossibile filtrare gli elementi senza una funzione di ricerca che restituisca una lista di { id: string, description: string }";
|
|
147
152
|
if (this.SearchFunction) {
|
|
153
|
+
this.log("filtersource enqueue");
|
|
148
154
|
this.throttla("filtersource", () => {
|
|
155
|
+
this.log("filtersource fire");
|
|
149
156
|
this.doSearchProtected(event, false, this.SearchFunctionContext, (t) => {
|
|
157
|
+
this.log("filtersource received API response");
|
|
150
158
|
this.Source = t;
|
|
151
159
|
this.tryBindSourceDisplay();
|
|
152
160
|
// In questo caso è già filtrata dalla SearchFunction
|
|
@@ -158,6 +166,7 @@ export class FormAutocompleteComponent extends BaseFormControl {
|
|
|
158
166
|
}
|
|
159
167
|
else {
|
|
160
168
|
this.throttla("filtersource", () => {
|
|
169
|
+
this.log("filtersource fire");
|
|
161
170
|
// In questo caso devo filtrare io in memoria
|
|
162
171
|
this.FilteredBoundSource = this.BoundSource.filter(t => (this.CaseSensitive && t.description.includes(event)) || (!this.CaseSensitive && t.description.toLowerCase().includes(event.toLowerCase())));
|
|
163
172
|
this.removeFilteredSourceOnDescriptionSelection();
|
|
@@ -174,9 +183,12 @@ export class FormAutocompleteComponent extends BaseFormControl {
|
|
|
174
183
|
this.timer = null;
|
|
175
184
|
}, 3000);
|
|
176
185
|
}
|
|
177
|
-
if (this.callsDoneThisThreeSeconds >= this.maxCallsPerSecond)
|
|
186
|
+
if (this.callsDoneThisThreeSeconds >= this.maxCallsPerSecond) {
|
|
187
|
+
this.log("Ignoring excessive calls");
|
|
178
188
|
return;
|
|
189
|
+
}
|
|
179
190
|
this.callsDoneThisThreeSeconds++;
|
|
191
|
+
this.log("Search function being called...");
|
|
180
192
|
this.SearchFunction(search, byid, context).subscribe(t => {
|
|
181
193
|
onRes(t);
|
|
182
194
|
});
|
|
@@ -274,4 +286,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
|
|
|
274
286
|
}], UnboundMode: [{
|
|
275
287
|
type: Input
|
|
276
288
|
}] } });
|
|
277
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-autocomplete.component.js","sourceRoot":"","sources":["../../../../../../projects/forms-and-validations/src/lib/forms/form-autocomplete/form-autocomplete.component.ts","../../../../../../projects/forms-and-validations/src/lib/forms/form-autocomplete/form-autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["// Angular\r\nimport { ChangeDetectionStrategy, Component, Inject, Input, Optional, ChangeDetectorRef, Self, SimpleChanges, Host } from \"@angular/core\";\r\nimport { ControlValueAccessor, NG_VALIDATORS, NgControl } from \"@angular/forms\";\r\n\r\n// Applicazione\r\nimport { BaseFormControl } from \"../base-form-control\";\r\n\r\n// Configurazioni\r\nimport { AccessControlService, ComponentContext } from '@esfaenza/access-control';\r\nimport { LocalizationService } from \"@esfaenza/localizations\";\r\nimport { ACO_CUSTOMKEY, FAV_DEBUG_MODE } from '../../tokens';\r\n\r\nimport { Observable } from \"rxjs\";\r\n\r\n//Localizzazione\r\nimport { FormAutocompleteComponentLoc } from \"./form-autocomplete.component.loc\";\r\n\r\n/** Componente di input che si auto-completa in base al valore attuale */\r\n@Component({\r\n    selector: \"form-autocomplete\",\r\n    templateUrl: \"form-autocomplete.component.html\",\r\n    providers: [{ provide: LocalizationService, useClass: FormAutocompleteComponentLoc }],\r\n    changeDetection: ChangeDetectionStrategy.OnPush\r\n})\r\nexport class FormAutocompleteComponent extends BaseFormControl implements ControlValueAccessor {\r\n\r\n    /** Permette di specificare il testo della Label flottante material-style */\r\n    @Input() SelectLabel: string = \"\";\r\n\r\n    /** Contesto per la funzione di ricerca */\r\n    @Input() SearchFunctionContext: any;\r\n\r\n    /** Funzione di ricerca che verrà chiamata dal componente */\r\n    @Input() SearchFunction: (search: string, byid: boolean, context?: any) => Observable<{ id: string, description: string }[]> = null;\r\n\r\n    /** Numero minimo di caratteri con cui cercare */\r\n    @Input() MinChars: number = 3;\r\n\r\n    /** Definisce che ad ogni selezione deve chiudere la tendina */\r\n    @Input() HideChoicesOnSelection: boolean = false;\r\n\r\n    /** Override del placeholder per select requried */\r\n    @Input() RequiredPlaceholder: string = null;\r\n\r\n    /** Indica se i controlli devono essere effettuati tenendo conto del Case o meno. Vale solo qualora la **Source** fosse fornita */\r\n    @Input() CaseSensitive: boolean = false;\r\n\r\n    /** Indica se usare la modalità multiselezione o no */\r\n    @Input() Multi: boolean = false;\r\n\r\n    /** Indica se usare la modalità multiselezione o no */\r\n    @Input() MultiElementsThreshold: number = 10;\r\n\r\n    /** Indica il fatto che non c'è una Source da cui scegliere */\r\n    @Input() UnboundMode: boolean = false;\r\n\r\n    /**\r\n     * Indica se ignorare il prossimo evento writeValue che normalmente dovrebbe richiedere la nuova source. Serve per quando l'utente seleziona un elemento:\r\n     * Subito dopo partirebbe un altro evento modelChange che ricaricherebbe nuovamente la source\r\n     */\r\n    private ignoreNextWriteValue: boolean = false;\r\n\r\n    /** Sorgente Bindata Filtrata in base al contenuto della casella di testo */\r\n    public FilteredBoundSource: { id: string, description: string }[] = [];\r\n\r\n    /** @ignore Costruttore  */\r\n    constructor(cdr: ChangeDetectorRef, public lc: LocalizationService, @Optional() @Self() ngControl: NgControl, @Optional() @Inject(NG_VALIDATORS) _validators: Array<any>, @Optional() ac: AccessControlService, @Optional() AppContext: ComponentContext, @Optional() @Inject(ACO_CUSTOMKEY) ACO_CUSTOMKEY: string, @Optional() @Inject(FAV_DEBUG_MODE) FAV_DEBUG_MODE: boolean) {\r\n        super(cdr, ngControl, _validators, ac, AppContext, ACO_CUSTOMKEY, FAV_DEBUG_MODE);\r\n    }\r\n\r\n    /** @ignore */\r\n    writeValue(value: any): void {\r\n        if (!value) return;\r\n\r\n        if (this.SearchFunction && !this.UnboundMode) {\r\n\r\n            this.doSearchProtected(value, true, this.SearchFunctionContext, (t) => {\r\n                this.Source = t;\r\n                this.tryBindSourceDisplay();\r\n                setTimeout(() => { this.finalizeValue(value); });\r\n            });\r\n            return;\r\n        }\r\n\r\n        if (!this.Multi)\r\n            this.finalizeValue(value);\r\n    }\r\n\r\n    /**\r\n     * Dato un valore verifica se può restituire le informazioni trovate in **Source** con id uguale a **value** o se deve restituire il valore in se\r\n     * \r\n     * @param {any} value Valore scritto nell'input di testo\r\n     */\r\n    private finalizeValue(value: any) {\r\n        var val = this.UnboundMode ? null : this.BoundSource.find(t => t.id == value);\r\n        this.propagateChange(val ? val.id : value);\r\n\r\n        // Alor... praticamente qui potrebbero arrivare degli switcheroni in cui il Model è uguale all'inizio e alla fine dello stack, per qualche motivo\r\n        // quindi per assicurarmi che il valore venga preso correttamente faccio il classico metti a null poi metti su\r\n        // Notare che mettere il modello a null scatena i writevalue ma tutti i writevalue che si collegao a questo giro dovrebbero essere NOOP su null\r\n        this.Model = null;\r\n        setTimeout(() => {\r\n            this.Model = val ? val.description : value;\r\n            this.EvaluatedModel = this.Model;\r\n            this.cdr.markForCheck();\r\n        });\r\n    }\r\n\r\n    /**\r\n     * Quando viene scritto un nuovo valore bisogna controllare se per botta di culo fosse un id, in tal caso dev'essere rieffettuato il bind\r\n     * per mostrare solo la descrizione nell'autocomplete\r\n     */\r\n    public evaluateIdResearch(onFailure: Function) {\r\n        let tmpModel = this.Model;\r\n        \r\n        if(this.UnboundMode)\r\n            return;\r\n\r\n        if (!tmpModel || !this.SearchFunction) {\r\n            onFailure?.();\r\n            return;\r\n        }\r\n\r\n        this.doSearchProtected(tmpModel, true, this.SearchFunctionContext, (t) => {\r\n            if (t && t.length > 0) {\r\n                this.Source = t;\r\n                this.tryBindSourceDisplay();\r\n                if (!this.Multi)\r\n                    setTimeout(() => { this.finalizeValue(tmpModel); });\r\n                else\r\n                    this.cdr.markForCheck();\r\n            }\r\n            else\r\n                onFailure?.();\r\n        });\r\n    }\r\n\r\n    /**\r\n     * Evento di filtro della sorgente dati in base all'input utente\r\n     * \r\n     * @param {string} event Input utente\r\n     */\r\n    public filterSource(event: string) {\r\n\r\n        // Contorllo prima di tutto se è un ID. in caso contrario effettuo la ricerca\r\n        this.evaluateIdResearch(() => {\r\n            this.doFilterSource(event);\r\n        })\r\n    }\r\n\r\n    doFilterSource(event: string){\r\n        if (this.UnboundMode)\r\n            return;\r\n\r\n        if (this.ignoreNextWriteValue) {\r\n            this.ignoreNextWriteValue = false;\r\n            return;\r\n        }\r\n\r\n        // Quando filtro la source, se non devo ignorare l'evento devo comunque assicurarmi di impostare il valore selezionato a null\r\n        if (!this.Multi)\r\n            super.changed(null);\r\n\r\n        if (!event && this.MinChars == 0 && !this.SearchFunction) {\r\n            this.FilteredBoundSource = this.BoundSource;\r\n            this.cdr.markForCheck();\r\n            return;\r\n        }\r\n\r\n        if (!event || event.length < this.MinChars) {\r\n            this.FilteredBoundSource = [];\r\n            this.cdr.markForCheck();\r\n            return;\r\n        }\r\n\r\n        if (!this.SearchFunction && (!this.Source || this.Source.length == 0))\r\n            throw \"Impossibile filtrare gli elementi senza una funzione di ricerca che restituisca una lista di { id: string, description: string }\";\r\n\r\n        if (this.SearchFunction) {\r\n            this.throttla(\"filtersource\", () => {\r\n                this.doSearchProtected(event, false, this.SearchFunctionContext, (t) => {\r\n                    this.Source = t;\r\n                    this.tryBindSourceDisplay();\r\n                    // In questo caso è già filtrata dalla SearchFunction\r\n                    this.FilteredBoundSource = this.BoundSource;\r\n                    this.removeFilteredSourceOnDescriptionSelection();\r\n                    this.cdr.markForCheck();\r\n                })\r\n            }, 400);\r\n        }\r\n        else {\r\n            this.throttla(\"filtersource\", () => {\r\n                // In questo caso devo filtrare io in memoria\r\n                this.FilteredBoundSource = this.BoundSource.filter(t => (this.CaseSensitive && t.description.includes(event)) || (!this.CaseSensitive && t.description.toLowerCase().includes(event.toLowerCase())));\r\n                this.removeFilteredSourceOnDescriptionSelection();\r\n                this.cdr.markForCheck();\r\n            }, 100);\r\n        }\r\n    }\r\n\r\n    timer = null;\r\n    maxCallsPerSecond = 4;\r\n    callsDoneThisThreeSeconds = 0;\r\n\r\n    // Permetto al massimo 4 chiamate ogni 3 secondi, se nell'arco di 3 secondi ci sono più di 4 chiamate, ignoro le eccedenti\r\n    private doSearchProtected(search: string, byid: boolean, context?: any, onRes?: Function) {\r\n        if (!this.timer) {\r\n            this.timer = setTimeout(() => {\r\n                this.callsDoneThisThreeSeconds = 0;\r\n                clearTimeout(this.timer);\r\n                this.timer = null;\r\n            }, 3000);\r\n        }\r\n\r\n        if (this.callsDoneThisThreeSeconds >= this.maxCallsPerSecond)\r\n            return;\r\n\r\n        this.callsDoneThisThreeSeconds++;\r\n        this.SearchFunction(search, byid, context).subscribe(t => {\r\n            onRes(t);\r\n        });\r\n    }\r\n\r\n    /** @ignore */\r\n    private removeFilteredSourceOnDescriptionSelection() {\r\n        if (this.FilteredBoundSource.length == 1 && (this.FilteredBoundSource[0].description === this.Model) && !this.Multi) {\r\n            let model = this.BoundSource[0].id\r\n            this.Model = model;\r\n            this.EvaluatedModel = this.BoundSource[0].description;\r\n            this.FilteredBoundSource = [];\r\n            setTimeout(() => { this.finalizeValue(model); });\r\n        }\r\n    }\r\n\r\n    /** @ignore */\r\n    public ngOnChanges(changes: SimpleChanges) {\r\n        let newSource = changes[\"Source\"];\r\n        if (newSource) {\r\n            this.tryBindSourceDisplay();\r\n\r\n            // Prima assegnazione se cambia la source sotto (quindi se viene bindata direttamente da HTML)\r\n            if ((this.Model?.length || 0) >= this.MinChars && !this.SearchFunction)\r\n                this.FilteredBoundSource = this.BoundSource;\r\n            else\r\n                this.FilteredBoundSource = [];\r\n\r\n            this.cdr.markForCheck();\r\n        }\r\n    }\r\n\r\n    /** @ignore Override che marca anche il prossimo evento di filterSource da ignorare */\r\n    changed(forcedValue: any = null, markForCheck: boolean = false) {\r\n        if (!this.Multi)\r\n            this.ignoreNextWriteValue = true;\r\n\r\n        if (forcedValue == \"\" && this.Multi)\r\n            this.Model = \"\";\r\n\r\n        var selecteds: string[] = forcedValue ? [forcedValue] : this.Model ? (this.Model.Multi ? this.Model.split(',') : [this.Model]) : [];\r\n        this.EvaluatedModel = selecteds.map(s => this.BoundSource && this.BoundSource.length > 0 ? this.BoundSource.find(t => t.id == s)?.description : this.UnboundMode ? s : \"\").join(', ');\r\n\r\n        super.changed(forcedValue, markForCheck);\r\n    }\r\n\r\n    /** @ignore */\r\n    onNotNullValueSet(): void { }\r\n\r\n    //******************** Funzione di throttling per non spammare richieste in caso di animazioni attivate\r\n    //TODO: spostarla in un metodo di utilità (esfaenza/extensions)\r\n    /** @ignore */\r\n    private executionTimers: { [index: string]: any } = {};\r\n    /** @ignore */\r\n    private throttla(id: string, func: Function, throttleTime: number): void {\r\n        //Se ho la funzione che vuole eseguire ripulisco quel timeout\r\n        if (this.executionTimers[id])\r\n            clearTimeout(this.executionTimers[id]);\r\n\r\n        //Ricreo il timeout per eseguire quella funzione dopo throttleTime millisecondi\r\n        this.executionTimers[id] = setTimeout(() => { func(); this.executionTimers[id] = null; }, throttleTime);\r\n    }\r\n    //*************************\r\n}","<!-- Uguale in tutti i componenti --------------------------------------------------------------------------->\r\n<ng-container *ngIf=\"!FormLayout && (!DisplayMode || (DisplayLayout != 'hidden' && DisplayCondition))\">\r\n    <ng-container *ngIf=\"DisplayMode && !DisplayModeTemplate\">\r\n        <ng-container *ngIf=\"DisplayLayout == 'form'\">{{ EvaluatedModel }}</ng-container>\r\n        <div *ngIf=\"DisplayLayout == 'inline'\" class=\"app-inline\">{{ EvaluatedModel }}</div>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"DisplayMode && DisplayModeTemplate\"><ng-container *ngTemplateOutlet=\"DisplayModeTemplate, context: { $implicit: EvaluatedModel }\"></ng-container></ng-container>\r\n    <div [hidden]=\"DisplayMode\"><ng-container *ngTemplateOutlet=\"controlTemplate\"></ng-container></div>\r\n</ng-container>\r\n\r\n<div *ngIf=\"FormLayout && (!DisplayMode || (DisplayLayout != 'hidden' && DisplayCondition))\" class=\"{{FormGroupClass + (Last ? ' app-margin-bottom-0 app-margin-right-0 ' : '') + (DisplayLayout == 'inline' && DisplayMode ? (' app-inline-block ' + (!Last ? 'app-margin-right-10' : '')) : ' form-group row')}}\">\r\n\r\n    <label class=\"col-md-{{(DisplayMode && DisplayLayout == 'inline' ? 'none app-bold app-margin-bottom-0' : LabelColWidth) + (DisplayMode ? ' app-bold' : ' m-t-5') }}\">{{Label}}{{Required && !DisplayMode ? '*' : ''}}{{Label ? \":\" : \"\"}}</label>\r\n    <span *ngIf=\"DisplayMode && DisplayLayout == 'inline' && InlineSeparator != ''\">{{InlineSeparator}}</span>\r\n    <div class=\"col-md-{{DisplayMode && DisplayLayout == 'inline' ? 'none app-inline-block' : InputColWidth}}\">\r\n\r\n        <ng-container *ngIf=\"DisplayMode && !DisplayModeTemplate\">{{ EvaluatedModel }}</ng-container>\r\n        <ng-container *ngIf=\"DisplayMode && DisplayModeTemplate\"><ng-container *ngTemplateOutlet=\"DisplayModeTemplate, context: { $implicit: EvaluatedModel }\"></ng-container></ng-container>\r\n        <div [hidden]=\"DisplayMode\"><ng-container *ngTemplateOutlet=\"controlTemplate\"></ng-container></div>\r\n    </div>\r\n    <div class=\"clearfix\"></div>\r\n</div>\r\n<!----------------------------------------------------------------------------------------------------------->\r\n<!-- Form-Model: {{Model}} -->\r\n<ng-template #controlTemplate>\r\n    <val-autocomplete *ngIf=\"!Multi\" [FocusSubject]=\"FocusSubject\" [SetValidationSubject]=\"SetValidationSubject\" [FieldAppearence]=\"FieldAppearence\" [noValidate]=\"!Validation\"\r\n               [submitted]=\"Form?.submitted\"\r\n               [Readonly]=\"Readonly\"\r\n               [label]=\"SelectLabel\"\r\n               type=\"text\"\r\n               [(ngModel)]=\"Model\"\r\n               [id]=\"GeneratedName\"\r\n               name=\"{{GeneratedName}}\"\r\n               #validationControl=\"ngModel\"\r\n               (inputChange)=\"filterSource($event);\"\r\n               (inputFocus)=\"focused($event);\"\r\n               (inputFinalized)=\"finalized()\"\r\n               (optionChange)=\"changed($event);\"\r\n               [placeholder]=\"Required ? ((RequiredPlaceholder != null ? RequiredPlaceholder : (('Seleziona' | localize : lc) + '...'))) : Placeholder\"\r\n               [validationFailed]=\"FailedValidationMessage\"\r\n               [FilteredSource]=\"FilteredBoundSource\">\r\n    </val-autocomplete>\r\n\r\n    <val-autocomplete-multi *ngIf=\"Multi\" [FocusSubject]=\"FocusSubject\" [SetValidationSubject]=\"SetValidationSubject\" [FieldAppearence]=\"FieldAppearence\" [noValidate]=\"!Validation\"\r\n               [submitted]=\"Form?.submitted\"\r\n               [Readonly]=\"Readonly\"\r\n               [label]=\"SelectLabel\"\r\n               [ChipThreshold]=\"MultiElementsThreshold\"\r\n               [HideChoicesOnSelection]=\"HideChoicesOnSelection\"\r\n               type=\"text\"\r\n               [(ngModel)]=\"Model\"\r\n               [id]=\"GeneratedName\"\r\n               name=\"{{GeneratedName}}\"\r\n               #validationControl=\"ngModel\"\r\n               (inputChange)=\"filterSource($event);\"\r\n               (inputFocus)=\"focused($event);\"\r\n               (inputFinalized)=\"finalized()\"\r\n               (optionChange)=\"changed($event);\"\r\n               [placeholder]=\"Required ? ((RequiredPlaceholder != null ? RequiredPlaceholder : (('Seleziona' | localize : lc) + '...'))) : Placeholder\"\r\n               [validationFailed]=\"FailedValidationMessage\"\r\n               [FilteredSource]=\"FilteredBoundSource\"\r\n               [UnboundMode]=\"UnboundMode\">\r\n\r\n    </val-autocomplete-multi>\r\n</ng-template>"]}
|
|
289
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-autocomplete.component.js","sourceRoot":"","sources":["../../../../../../projects/forms-and-validations/src/lib/forms/form-autocomplete/form-autocomplete.component.ts","../../../../../../projects/forms-and-validations/src/lib/forms/form-autocomplete/form-autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["// Angular\r\nimport { ChangeDetectionStrategy, Component, Inject, Input, Optional, ChangeDetectorRef, Self, SimpleChanges, Host } from \"@angular/core\";\r\nimport { ControlValueAccessor, NG_VALIDATORS, NgControl } from \"@angular/forms\";\r\n\r\n// Applicazione\r\nimport { BaseFormControl } from \"../base-form-control\";\r\n\r\n// Configurazioni\r\nimport { AccessControlService, ComponentContext } from '@esfaenza/access-control';\r\nimport { LocalizationService } from \"@esfaenza/localizations\";\r\nimport { ACO_CUSTOMKEY, FAV_DEBUG_MODE } from '../../tokens';\r\n\r\nimport { Observable } from \"rxjs\";\r\n\r\n//Localizzazione\r\nimport { FormAutocompleteComponentLoc } from \"./form-autocomplete.component.loc\";\r\n\r\n/** Componente di input che si auto-completa in base al valore attuale */\r\n@Component({\r\n    selector: \"form-autocomplete\",\r\n    templateUrl: \"form-autocomplete.component.html\",\r\n    providers: [{ provide: LocalizationService, useClass: FormAutocompleteComponentLoc }],\r\n    changeDetection: ChangeDetectionStrategy.OnPush\r\n})\r\nexport class FormAutocompleteComponent extends BaseFormControl implements ControlValueAccessor {\r\n\r\n    /** Permette di specificare il testo della Label flottante material-style */\r\n    @Input() SelectLabel: string = \"\";\r\n\r\n    /** Contesto per la funzione di ricerca */\r\n    @Input() SearchFunctionContext: any;\r\n\r\n    /** Funzione di ricerca che verrà chiamata dal componente */\r\n    @Input() SearchFunction: (search: string, byid: boolean, context?: any) => Observable<{ id: string, description: string }[]> = null;\r\n\r\n    /** Numero minimo di caratteri con cui cercare */\r\n    @Input() MinChars: number = 3;\r\n\r\n    /** Definisce che ad ogni selezione deve chiudere la tendina */\r\n    @Input() HideChoicesOnSelection: boolean = false;\r\n\r\n    /** Override del placeholder per select requried */\r\n    @Input() RequiredPlaceholder: string = null;\r\n\r\n    /** Indica se i controlli devono essere effettuati tenendo conto del Case o meno. Vale solo qualora la **Source** fosse fornita */\r\n    @Input() CaseSensitive: boolean = false;\r\n\r\n    /** Indica se usare la modalità multiselezione o no */\r\n    @Input() Multi: boolean = false;\r\n\r\n    /** Indica se usare la modalità multiselezione o no */\r\n    @Input() MultiElementsThreshold: number = 10;\r\n\r\n    /** Indica il fatto che non c'è una Source da cui scegliere */\r\n    @Input() UnboundMode: boolean = false;\r\n\r\n    /**\r\n     * Indica se ignorare il prossimo evento writeValue che normalmente dovrebbe richiedere la nuova source. Serve per quando l'utente seleziona un elemento:\r\n     * Subito dopo partirebbe un altro evento modelChange che ricaricherebbe nuovamente la source\r\n     */\r\n    private ignoreNextWriteValue: boolean = false;\r\n\r\n    /** Sorgente Bindata Filtrata in base al contenuto della casella di testo */\r\n    public FilteredBoundSource: { id: string, description: string }[] = [];\r\n\r\n    /** @ignore Costruttore  */\r\n    constructor(cdr: ChangeDetectorRef, public lc: LocalizationService, @Optional() @Self() ngControl: NgControl, @Optional() @Inject(NG_VALIDATORS) _validators: Array<any>, @Optional() ac: AccessControlService, @Optional() AppContext: ComponentContext, @Optional() @Inject(ACO_CUSTOMKEY) ACO_CUSTOMKEY: string, @Optional() @Inject(FAV_DEBUG_MODE) FAV_DEBUG_MODE: boolean) {\r\n        super(cdr, ngControl, _validators, ac, AppContext, ACO_CUSTOMKEY, FAV_DEBUG_MODE);\r\n    }\r\n\r\n    /** @ignore */\r\n    writeValue(value: any): void {\r\n        if (!value) return;\r\n\r\n        if (this.SearchFunction && !this.UnboundMode) {\r\n\r\n            this.doSearchProtected(value, true, this.SearchFunctionContext, (t) => {\r\n                this.Source = t;\r\n                this.tryBindSourceDisplay();\r\n                setTimeout(() => { this.finalizeValue(value); });\r\n            });\r\n            return;\r\n        }\r\n\r\n        if (!this.Multi)\r\n            this.finalizeValue(value);\r\n    }\r\n\r\n    /**\r\n     * Dato un valore verifica se può restituire le informazioni trovate in **Source** con id uguale a **value** o se deve restituire il valore in se\r\n     * \r\n     * @param {any} value Valore scritto nell'input di testo\r\n     */\r\n    private finalizeValue(value: any) {\r\n        var val = this.UnboundMode ? null : this.BoundSource.find(t => t.id == value);\r\n        this.propagateChange(val ? val.id : value);\r\n\r\n        // Alor... praticamente qui potrebbero arrivare degli switcheroni in cui il Model è uguale all'inizio e alla fine dello stack, per qualche motivo\r\n        // quindi per assicurarmi che il valore venga preso correttamente faccio il classico metti a null poi metti su\r\n        // Notare che mettere il modello a null scatena i writevalue ma tutti i writevalue che si collegao a questo giro dovrebbero essere NOOP su null\r\n        this.Model = null;\r\n        setTimeout(() => {\r\n            this.Model = val ? val.description : value;\r\n            this.EvaluatedModel = this.Model;\r\n            this.cdr.markForCheck();\r\n        });\r\n    }\r\n\r\n    /**\r\n     * Quando viene scritto un nuovo valore bisogna controllare se per botta di culo fosse un id, in tal caso dev'essere rieffettuato il bind\r\n     * per mostrare solo la descrizione nell'autocomplete\r\n     */\r\n    public evaluateIdResearch(onFailure: Function) {\r\n        let tmpModel = this.Model;\r\n        \r\n        if(this.UnboundMode)\r\n            return;\r\n\r\n        if (!tmpModel || !this.SearchFunction) {\r\n            onFailure?.();\r\n            return;\r\n        }\r\n\r\n        this.log(\"filtersource enqueue\");\r\n        this.throttla(\"filtersource\", () => {\r\n            this.log(\"filtersource fire\");\r\n            this.doSearchProtected(tmpModel, true, this.SearchFunctionContext, (t) => {\r\n                if (t && t.length > 0) {\r\n                    this.Source = t;\r\n                    this.tryBindSourceDisplay();\r\n                    if (!this.Multi)\r\n                        setTimeout(() => { this.finalizeValue(tmpModel); });\r\n                    else\r\n                        this.cdr.markForCheck();\r\n                }\r\n                else\r\n                    onFailure?.();\r\n            });\r\n        }, 400);\r\n    }\r\n\r\n    /**\r\n     * Evento di filtro della sorgente dati in base all'input utente\r\n     * \r\n     * @param {string} event Input utente\r\n     */\r\n    public filterSource(event: string) {\r\n\r\n        // Contorllo prima di tutto se è un ID. in caso contrario effettuo la ricerca\r\n        this.evaluateIdResearch(() => {\r\n            this.doFilterSource(event);\r\n        })\r\n    }\r\n\r\n    doFilterSource(event: string){\r\n        if (this.UnboundMode)\r\n            return;\r\n\r\n        if (this.ignoreNextWriteValue) {\r\n            this.ignoreNextWriteValue = false;\r\n            return;\r\n        }\r\n\r\n        // Quando filtro la source, se non devo ignorare l'evento devo comunque assicurarmi di impostare il valore selezionato a null\r\n        if (!this.Multi)\r\n            super.changed(null);\r\n\r\n        if (!event && this.MinChars == 0 && !this.SearchFunction) {\r\n            this.FilteredBoundSource = this.BoundSource;\r\n            this.cdr.markForCheck();\r\n            return;\r\n        }\r\n\r\n        if (!event || event.length < this.MinChars) {\r\n            this.log(\"search event ignored: \" + event);\r\n            this.FilteredBoundSource = [];\r\n            this.cdr.markForCheck();\r\n            return;\r\n        }\r\n\r\n        if (!this.SearchFunction && (!this.Source || this.Source.length == 0))\r\n            throw \"Impossibile filtrare gli elementi senza una funzione di ricerca che restituisca una lista di { id: string, description: string }\";\r\n\r\n        if (this.SearchFunction) {\r\n            this.log(\"filtersource enqueue\");\r\n            this.throttla(\"filtersource\", () => {\r\n                this.log(\"filtersource fire\");\r\n                this.doSearchProtected(event, false, this.SearchFunctionContext, (t) => {\r\n                    this.log(\"filtersource received API response\");\r\n                    this.Source = t;\r\n                    this.tryBindSourceDisplay();\r\n                    // In questo caso è già filtrata dalla SearchFunction\r\n                    this.FilteredBoundSource = this.BoundSource;\r\n                    this.removeFilteredSourceOnDescriptionSelection();\r\n                    this.cdr.markForCheck();\r\n                })\r\n            }, 400);\r\n        }\r\n        else {\r\n            this.throttla(\"filtersource\", () => {\r\n                this.log(\"filtersource fire\");\r\n                // In questo caso devo filtrare io in memoria\r\n                this.FilteredBoundSource = this.BoundSource.filter(t => (this.CaseSensitive && t.description.includes(event)) || (!this.CaseSensitive && t.description.toLowerCase().includes(event.toLowerCase())));\r\n                this.removeFilteredSourceOnDescriptionSelection();\r\n                this.cdr.markForCheck();\r\n            }, 100);\r\n        }\r\n    }\r\n\r\n    timer = null;\r\n    maxCallsPerSecond = 4;\r\n    callsDoneThisThreeSeconds = 0;\r\n\r\n    // Permetto al massimo 4 chiamate ogni 3 secondi, se nell'arco di 3 secondi ci sono più di 4 chiamate, ignoro le eccedenti\r\n    private doSearchProtected(search: string, byid: boolean, context?: any, onRes?: Function) {\r\n        if (!this.timer) {\r\n            this.timer = setTimeout(() => {\r\n                this.callsDoneThisThreeSeconds = 0;\r\n                clearTimeout(this.timer);\r\n                this.timer = null;\r\n            }, 3000);\r\n        }\r\n\r\n        if (this.callsDoneThisThreeSeconds >= this.maxCallsPerSecond) {\r\n            this.log(\"Ignoring excessive calls\");\r\n            return;\r\n        }\r\n\r\n        this.callsDoneThisThreeSeconds++;\r\n        this.log(\"Search function being called...\");\r\n        this.SearchFunction(search, byid, context).subscribe(t => {\r\n            onRes(t);\r\n        });\r\n    }\r\n\r\n    /** @ignore */\r\n    private removeFilteredSourceOnDescriptionSelection() {\r\n        if (this.FilteredBoundSource.length == 1 && (this.FilteredBoundSource[0].description === this.Model) && !this.Multi) {\r\n            let model = this.BoundSource[0].id\r\n            this.Model = model;\r\n            this.EvaluatedModel = this.BoundSource[0].description;\r\n            this.FilteredBoundSource = [];\r\n            setTimeout(() => { this.finalizeValue(model); });\r\n        }\r\n    }\r\n\r\n    /** @ignore */\r\n    public ngOnChanges(changes: SimpleChanges) {\r\n        let newSource = changes[\"Source\"];\r\n        if (newSource) {\r\n            this.tryBindSourceDisplay();\r\n\r\n            // Prima assegnazione se cambia la source sotto (quindi se viene bindata direttamente da HTML)\r\n            if ((this.Model?.length || 0) >= this.MinChars && !this.SearchFunction)\r\n                this.FilteredBoundSource = this.BoundSource;\r\n            else\r\n                this.FilteredBoundSource = [];\r\n\r\n            this.cdr.markForCheck();\r\n        }\r\n    }\r\n\r\n    /** @ignore Override che marca anche il prossimo evento di filterSource da ignorare */\r\n    changed(forcedValue: any = null, markForCheck: boolean = false) {\r\n        if (!this.Multi)\r\n            this.ignoreNextWriteValue = true;\r\n\r\n        if (forcedValue == \"\" && this.Multi)\r\n            this.Model = \"\";\r\n\r\n        var selecteds: string[] = forcedValue ? [forcedValue] : this.Model ? (this.Model.Multi ? this.Model.split(',') : [this.Model]) : [];\r\n        this.EvaluatedModel = selecteds.map(s => this.BoundSource && this.BoundSource.length > 0 ? this.BoundSource.find(t => t.id == s)?.description : this.UnboundMode ? s : \"\").join(', ');\r\n\r\n        super.changed(forcedValue, markForCheck);\r\n    }\r\n\r\n    /** @ignore */\r\n    onNotNullValueSet(): void { }\r\n\r\n    //******************** Funzione di throttling per non spammare richieste in caso di animazioni attivate\r\n    //TODO: spostarla in un metodo di utilità (esfaenza/extensions)\r\n    /** @ignore */\r\n    private executionTimers: { [index: string]: any } = {};\r\n    /** @ignore */\r\n    private throttla(id: string, func: Function, throttleTime: number): void {\r\n        //Se ho la funzione che vuole eseguire ripulisco quel timeout\r\n        if (this.executionTimers[id])\r\n            clearTimeout(this.executionTimers[id]);\r\n\r\n        //Ricreo il timeout per eseguire quella funzione dopo throttleTime millisecondi\r\n        this.executionTimers[id] = setTimeout(() => { func(); this.executionTimers[id] = null; }, throttleTime);\r\n    }\r\n    //*************************\r\n}","<!-- Uguale in tutti i componenti --------------------------------------------------------------------------->\r\n<ng-container *ngIf=\"!FormLayout && (!DisplayMode || (DisplayLayout != 'hidden' && DisplayCondition))\">\r\n    <ng-container *ngIf=\"DisplayMode && !DisplayModeTemplate\">\r\n        <ng-container *ngIf=\"DisplayLayout == 'form'\">{{ EvaluatedModel }}</ng-container>\r\n        <div *ngIf=\"DisplayLayout == 'inline'\" class=\"app-inline\">{{ EvaluatedModel }}</div>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"DisplayMode && DisplayModeTemplate\"><ng-container *ngTemplateOutlet=\"DisplayModeTemplate, context: { $implicit: EvaluatedModel }\"></ng-container></ng-container>\r\n    <div [hidden]=\"DisplayMode\"><ng-container *ngTemplateOutlet=\"controlTemplate\"></ng-container></div>\r\n</ng-container>\r\n\r\n<div *ngIf=\"FormLayout && (!DisplayMode || (DisplayLayout != 'hidden' && DisplayCondition))\" class=\"{{FormGroupClass + (Last ? ' app-margin-bottom-0 app-margin-right-0 ' : '') + (DisplayLayout == 'inline' && DisplayMode ? (' app-inline-block ' + (!Last ? 'app-margin-right-10' : '')) : ' form-group row')}}\">\r\n\r\n    <label class=\"col-md-{{(DisplayMode && DisplayLayout == 'inline' ? 'none app-bold app-margin-bottom-0' : LabelColWidth) + (DisplayMode ? ' app-bold' : ' m-t-5') }}\">{{Label}}{{Required && !DisplayMode ? '*' : ''}}{{Label ? \":\" : \"\"}}</label>\r\n    <span *ngIf=\"DisplayMode && DisplayLayout == 'inline' && InlineSeparator != ''\">{{InlineSeparator}}</span>\r\n    <div class=\"col-md-{{DisplayMode && DisplayLayout == 'inline' ? 'none app-inline-block' : InputColWidth}}\">\r\n\r\n        <ng-container *ngIf=\"DisplayMode && !DisplayModeTemplate\">{{ EvaluatedModel }}</ng-container>\r\n        <ng-container *ngIf=\"DisplayMode && DisplayModeTemplate\"><ng-container *ngTemplateOutlet=\"DisplayModeTemplate, context: { $implicit: EvaluatedModel }\"></ng-container></ng-container>\r\n        <div [hidden]=\"DisplayMode\"><ng-container *ngTemplateOutlet=\"controlTemplate\"></ng-container></div>\r\n    </div>\r\n    <div class=\"clearfix\"></div>\r\n</div>\r\n<!----------------------------------------------------------------------------------------------------------->\r\n<!-- Form-Model: {{Model}} -->\r\n<ng-template #controlTemplate>\r\n    <val-autocomplete *ngIf=\"!Multi\" [FocusSubject]=\"FocusSubject\" [SetValidationSubject]=\"SetValidationSubject\" [FieldAppearence]=\"FieldAppearence\" [noValidate]=\"!Validation\"\r\n               [submitted]=\"Form?.submitted\"\r\n               [Readonly]=\"Readonly\"\r\n               [label]=\"SelectLabel\"\r\n               type=\"text\"\r\n               [(ngModel)]=\"Model\"\r\n               [id]=\"GeneratedName\"\r\n               name=\"{{GeneratedName}}\"\r\n               #validationControl=\"ngModel\"\r\n               (inputChange)=\"filterSource($event);\"\r\n               (inputFocus)=\"focused($event);\"\r\n               (inputFinalized)=\"finalized()\"\r\n               (optionChange)=\"changed($event);\"\r\n               [placeholder]=\"Required ? ((RequiredPlaceholder != null ? RequiredPlaceholder : (('Seleziona' | localize : lc) + '...'))) : Placeholder\"\r\n               [validationFailed]=\"FailedValidationMessage\"\r\n               [FilteredSource]=\"FilteredBoundSource\">\r\n    </val-autocomplete>\r\n\r\n    <val-autocomplete-multi *ngIf=\"Multi\" [FocusSubject]=\"FocusSubject\" [SetValidationSubject]=\"SetValidationSubject\" [FieldAppearence]=\"FieldAppearence\" [noValidate]=\"!Validation\"\r\n               [submitted]=\"Form?.submitted\"\r\n               [Readonly]=\"Readonly\"\r\n               [label]=\"SelectLabel\"\r\n               [ChipThreshold]=\"MultiElementsThreshold\"\r\n               [HideChoicesOnSelection]=\"HideChoicesOnSelection\"\r\n               type=\"text\"\r\n               [(ngModel)]=\"Model\"\r\n               [id]=\"GeneratedName\"\r\n               name=\"{{GeneratedName}}\"\r\n               #validationControl=\"ngModel\"\r\n               (inputChange)=\"filterSource($event);\"\r\n               (inputFocus)=\"focused($event);\"\r\n               (inputFinalized)=\"finalized()\"\r\n               (optionChange)=\"changed($event);\"\r\n               [placeholder]=\"Required ? ((RequiredPlaceholder != null ? RequiredPlaceholder : (('Seleziona' | localize : lc) + '...'))) : Placeholder\"\r\n               [validationFailed]=\"FailedValidationMessage\"\r\n               [FilteredSource]=\"FilteredBoundSource\"\r\n               [UnboundMode]=\"UnboundMode\">\r\n\r\n    </val-autocomplete-multi>\r\n</ng-template>"]}
|
|
@@ -3844,18 +3844,22 @@ class FormAutocompleteComponent extends BaseFormControl {
|
|
|
3844
3844
|
onFailure === null || onFailure === void 0 ? void 0 : onFailure();
|
|
3845
3845
|
return;
|
|
3846
3846
|
}
|
|
3847
|
-
this.
|
|
3848
|
-
|
|
3849
|
-
|
|
3850
|
-
|
|
3851
|
-
if (
|
|
3852
|
-
|
|
3847
|
+
this.log("filtersource enqueue");
|
|
3848
|
+
this.throttla("filtersource", () => {
|
|
3849
|
+
this.log("filtersource fire");
|
|
3850
|
+
this.doSearchProtected(tmpModel, true, this.SearchFunctionContext, (t) => {
|
|
3851
|
+
if (t && t.length > 0) {
|
|
3852
|
+
this.Source = t;
|
|
3853
|
+
this.tryBindSourceDisplay();
|
|
3854
|
+
if (!this.Multi)
|
|
3855
|
+
setTimeout(() => { this.finalizeValue(tmpModel); });
|
|
3856
|
+
else
|
|
3857
|
+
this.cdr.markForCheck();
|
|
3858
|
+
}
|
|
3853
3859
|
else
|
|
3854
|
-
|
|
3855
|
-
}
|
|
3856
|
-
|
|
3857
|
-
onFailure === null || onFailure === void 0 ? void 0 : onFailure();
|
|
3858
|
-
});
|
|
3860
|
+
onFailure === null || onFailure === void 0 ? void 0 : onFailure();
|
|
3861
|
+
});
|
|
3862
|
+
}, 400);
|
|
3859
3863
|
}
|
|
3860
3864
|
/**
|
|
3861
3865
|
* Evento di filtro della sorgente dati in base all'input utente
|
|
@@ -3884,6 +3888,7 @@ class FormAutocompleteComponent extends BaseFormControl {
|
|
|
3884
3888
|
return;
|
|
3885
3889
|
}
|
|
3886
3890
|
if (!event || event.length < this.MinChars) {
|
|
3891
|
+
this.log("search event ignored: " + event);
|
|
3887
3892
|
this.FilteredBoundSource = [];
|
|
3888
3893
|
this.cdr.markForCheck();
|
|
3889
3894
|
return;
|
|
@@ -3891,8 +3896,11 @@ class FormAutocompleteComponent extends BaseFormControl {
|
|
|
3891
3896
|
if (!this.SearchFunction && (!this.Source || this.Source.length == 0))
|
|
3892
3897
|
throw "Impossibile filtrare gli elementi senza una funzione di ricerca che restituisca una lista di { id: string, description: string }";
|
|
3893
3898
|
if (this.SearchFunction) {
|
|
3899
|
+
this.log("filtersource enqueue");
|
|
3894
3900
|
this.throttla("filtersource", () => {
|
|
3901
|
+
this.log("filtersource fire");
|
|
3895
3902
|
this.doSearchProtected(event, false, this.SearchFunctionContext, (t) => {
|
|
3903
|
+
this.log("filtersource received API response");
|
|
3896
3904
|
this.Source = t;
|
|
3897
3905
|
this.tryBindSourceDisplay();
|
|
3898
3906
|
// In questo caso è già filtrata dalla SearchFunction
|
|
@@ -3904,6 +3912,7 @@ class FormAutocompleteComponent extends BaseFormControl {
|
|
|
3904
3912
|
}
|
|
3905
3913
|
else {
|
|
3906
3914
|
this.throttla("filtersource", () => {
|
|
3915
|
+
this.log("filtersource fire");
|
|
3907
3916
|
// In questo caso devo filtrare io in memoria
|
|
3908
3917
|
this.FilteredBoundSource = this.BoundSource.filter(t => (this.CaseSensitive && t.description.includes(event)) || (!this.CaseSensitive && t.description.toLowerCase().includes(event.toLowerCase())));
|
|
3909
3918
|
this.removeFilteredSourceOnDescriptionSelection();
|
|
@@ -3920,9 +3929,12 @@ class FormAutocompleteComponent extends BaseFormControl {
|
|
|
3920
3929
|
this.timer = null;
|
|
3921
3930
|
}, 3000);
|
|
3922
3931
|
}
|
|
3923
|
-
if (this.callsDoneThisThreeSeconds >= this.maxCallsPerSecond)
|
|
3932
|
+
if (this.callsDoneThisThreeSeconds >= this.maxCallsPerSecond) {
|
|
3933
|
+
this.log("Ignoring excessive calls");
|
|
3924
3934
|
return;
|
|
3935
|
+
}
|
|
3925
3936
|
this.callsDoneThisThreeSeconds++;
|
|
3937
|
+
this.log("Search function being called...");
|
|
3926
3938
|
this.SearchFunction(search, byid, context).subscribe(t => {
|
|
3927
3939
|
onRes(t);
|
|
3928
3940
|
});
|