@esfaenza/forms-and-validations 16.2.30 → 17.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (83) hide show
  1. package/esm2022/esfaenza-forms-and-validations.mjs +4 -4
  2. package/esm2022/lib/forms/base-form-control.mjs +558 -558
  3. package/esm2022/lib/forms/form-adaptive/form-adaptive.component.loc.mjs +21 -21
  4. package/esm2022/lib/forms/form-adaptive/form-adaptive.component.mjs +420 -420
  5. package/esm2022/lib/forms/form-autocomplete/form-autocomplete.component.loc.mjs +20 -20
  6. package/esm2022/lib/forms/form-autocomplete/form-autocomplete.component.mjs +289 -289
  7. package/esm2022/lib/forms/form-checkbox/form-checkbox.component.mjs +74 -74
  8. package/esm2022/lib/forms/form-date/form-date.component.mjs +69 -69
  9. package/esm2022/lib/forms/form-datetime/form-datetime.component.mjs +67 -67
  10. package/esm2022/lib/forms/form-empty/form-empty.component.mjs +29 -29
  11. package/esm2022/lib/forms/form-error/form-error.component.mjs +41 -41
  12. package/esm2022/lib/forms/form-file/form-file.component.loc.mjs +21 -21
  13. package/esm2022/lib/forms/form-file/form-file.component.mjs +175 -175
  14. package/esm2022/lib/forms/form-info/form-info.component.mjs +41 -41
  15. package/esm2022/lib/forms/form-input/form-input.component.mjs +83 -83
  16. package/esm2022/lib/forms/form-multiselect/form-multiselect.component.loc.mjs +22 -22
  17. package/esm2022/lib/forms/form-multiselect/form-multiselect.component.mjs +163 -163
  18. package/esm2022/lib/forms/form-select/form-select.component.loc.mjs +20 -20
  19. package/esm2022/lib/forms/form-select/form-select.component.mjs +165 -165
  20. package/esm2022/lib/forms/form-template/form-template.component.mjs +76 -76
  21. package/esm2022/lib/forms/form-textarea/form-textarea.component.mjs +57 -57
  22. package/esm2022/lib/forms/form-time/form-time.component.mjs +115 -115
  23. package/esm2022/lib/forms-and-validations.module.mjs +199 -199
  24. package/esm2022/lib/models/AppFile.mjs +13 -13
  25. package/esm2022/lib/models/ChangeEvent.mjs +6 -6
  26. package/esm2022/lib/models/FormsAndValidationsModuleConfig.mjs +5 -5
  27. package/esm2022/lib/models/dayjs-adapter/dayjs-date-adapter.mjs +89 -89
  28. package/esm2022/lib/tokens.mjs +7 -7
  29. package/esm2022/lib/validations/base-validation.loc.mjs +22 -22
  30. package/esm2022/lib/validations/base-validation.mjs +328 -328
  31. package/esm2022/lib/validations/customValidators/CustomRequiredDirective.mjs +40 -40
  32. package/esm2022/lib/validations/validation-autocomplete/validation-autocomplete.component.mjs +147 -147
  33. package/esm2022/lib/validations/validation-autocomplete-multi/validation-autocomplete-multi.component.mjs +240 -240
  34. package/esm2022/lib/validations/validation-currency/validation-currency.component.mjs +75 -75
  35. package/esm2022/lib/validations/validation-date/validation-date.component.mjs +183 -183
  36. package/esm2022/lib/validations/validation-datetime/validation-datetime.component.mjs +212 -212
  37. package/esm2022/lib/validations/validation-input/validation-input.component.mjs +222 -222
  38. package/esm2022/lib/validations/validation-select/validation-select.component.mjs +183 -183
  39. package/esm2022/lib/validations/validation-text-area/validation-text-area.component.mjs +78 -78
  40. package/esm2022/public-api.mjs +39 -39
  41. package/fesm2022/esfaenza-forms-and-validations.mjs +4152 -4152
  42. package/fesm2022/esfaenza-forms-and-validations.mjs.map +1 -1
  43. package/index.d.ts +5 -5
  44. package/lib/forms/base-form-control.d.ts +276 -276
  45. package/lib/forms/form-adaptive/form-adaptive.component.d.ts +126 -126
  46. package/lib/forms/form-adaptive/form-adaptive.component.loc.d.ts +14 -14
  47. package/lib/forms/form-autocomplete/form-autocomplete.component.d.ts +84 -84
  48. package/lib/forms/form-autocomplete/form-autocomplete.component.loc.d.ts +14 -14
  49. package/lib/forms/form-checkbox/form-checkbox.component.d.ts +23 -23
  50. package/lib/forms/form-date/form-date.component.d.ts +21 -21
  51. package/lib/forms/form-datetime/form-datetime.component.d.ts +23 -23
  52. package/lib/forms/form-empty/form-empty.component.d.ts +15 -15
  53. package/lib/forms/form-error/form-error.component.d.ts +19 -19
  54. package/lib/forms/form-file/form-file.component.d.ts +46 -46
  55. package/lib/forms/form-file/form-file.component.loc.d.ts +14 -14
  56. package/lib/forms/form-info/form-info.component.d.ts +19 -19
  57. package/lib/forms/form-input/form-input.component.d.ts +33 -33
  58. package/lib/forms/form-multiselect/form-multiselect.component.d.ts +67 -67
  59. package/lib/forms/form-multiselect/form-multiselect.component.loc.d.ts +14 -14
  60. package/lib/forms/form-select/form-select.component.d.ts +43 -43
  61. package/lib/forms/form-select/form-select.component.loc.d.ts +14 -14
  62. package/lib/forms/form-template/form-template.component.d.ts +38 -38
  63. package/lib/forms/form-textarea/form-textarea.component.d.ts +18 -18
  64. package/lib/forms/form-time/form-time.component.d.ts +40 -40
  65. package/lib/forms-and-validations.module.d.ts +47 -47
  66. package/lib/models/AppFile.d.ts +21 -21
  67. package/lib/models/ChangeEvent.d.ts +5 -5
  68. package/lib/models/FormsAndValidationsModuleConfig.d.ts +11 -11
  69. package/lib/models/dayjs-adapter/dayjs-date-adapter.d.ts +49 -49
  70. package/lib/tokens.d.ts +7 -7
  71. package/lib/validations/base-validation.d.ts +219 -219
  72. package/lib/validations/base-validation.loc.d.ts +14 -14
  73. package/lib/validations/customValidators/CustomRequiredDirective.d.ts +24 -24
  74. package/lib/validations/validation-autocomplete/validation-autocomplete.component.d.ts +58 -58
  75. package/lib/validations/validation-autocomplete-multi/validation-autocomplete-multi.component.d.ts +88 -88
  76. package/lib/validations/validation-currency/validation-currency.component.d.ts +28 -28
  77. package/lib/validations/validation-date/validation-date.component.d.ts +79 -79
  78. package/lib/validations/validation-datetime/validation-datetime.component.d.ts +109 -109
  79. package/lib/validations/validation-input/validation-input.component.d.ts +134 -134
  80. package/lib/validations/validation-select/validation-select.component.d.ts +95 -95
  81. package/lib/validations/validation-text-area/validation-text-area.component.d.ts +28 -28
  82. package/package.json +13 -13
  83. package/public-api.d.ts +31 -31
@@ -1,558 +1,558 @@
1
- // Angular
2
- import { RequiredValidator } from "@angular/forms";
3
- import { EventEmitter, Input, Output, ViewChild, Optional, Inject, Directive } from "@angular/core";
4
- import { ACO_CUSTOMKEY, FAV_DEBUG_MODE } from '../tokens';
5
- import { Subject } from "rxjs";
6
- import { ChangeEvent } from "../models/ChangeEvent";
7
- import { takeUntil } from "rxjs/operators";
8
- import * as i0 from "@angular/core";
9
- import * as i1 from "@angular/forms";
10
- import * as i2 from "@esfaenza/access-control";
11
- /** Componente base da cui tutti i componenti Form implementano */
12
- export class BaseFormControl {
13
- /**
14
- * Ottiene il valore della variabile **_readonly**
15
- *
16
- * @returns {boolean} **true** se sola lettura, **false** altrimenti
17
- */
18
- get Readonly() {
19
- return this._readonly;
20
- }
21
- /**
22
- * Imposta il valore della variabile **_readonly** gestendo anche il Contesto applicativo
23
- *
24
- * @param {boolean} value Valore da impostare
25
- */
26
- set Readonly(value) {
27
- if (this.AppContextOwnership)
28
- this._readonly = value;
29
- }
30
- /**
31
- * Permette di settare in una volta solo gli input LabelColWidth e InputColWidth, basta scriverlo con la sintassi "X Y"
32
- *
33
- * @param {string} input Input nel formato "X Y"
34
- */
35
- set LabelInputRatio(input) {
36
- var ratio = input.split(/\s/);
37
- this.LabelColWidth = parseInt(ratio[0]);
38
- this.InputColWidth = parseInt(ratio[1]);
39
- }
40
- /**
41
- * Query sull'elemento 'validationControl' che funziona in ambiente statico
42
- *
43
- * @param {NgModel} comp Elemento HTML marcato con "#validationControl"
44
- */
45
- set _validationControl(comp) { this.registerValComp(comp); }
46
- /**
47
- * Query sull'elemento 'validationControl' che funziona in ambiente dinamico
48
- *
49
- * @param {NgModel} comp Elemento HTML marcato con "#validationControl"
50
- */
51
- set validationControl_static(comp) { this.registerValComp(comp); }
52
- ;
53
- /** @ignore Costruttore */
54
- constructor(cdr, ngControl, _validators, ac, AppContext, ACO_CUSTOMKEY, FAV_DEBUG_MODE, nativeInput = false) {
55
- this.cdr = cdr;
56
- this.ngControl = ngControl;
57
- this._validators = _validators;
58
- this.ac = ac;
59
- this.AppContext = AppContext;
60
- this.ACO_CUSTOMKEY = ACO_CUSTOMKEY;
61
- this.FAV_DEBUG_MODE = FAV_DEBUG_MODE;
62
- this.nativeInput = nativeInput;
63
- /**
64
- * https://github.com/angular/angular/issues/14988
65
- * TL:DR; Il primo valore passato al writeValue è nullo perché quando i cagnacci fanno new FormControl()
66
- * scrivono un writeValue nullo... quindi al primo binding salto l'inizializzazione se ho la coppia null-firstBind
67
- * poi mettendo firstBind a false di lì in poi funziono come devo. Vista la casistica evito essenzialmente metà detectChanges
68
- */
69
- this.firstBind = true;
70
- /**
71
- * Indica di utilizzare i validatori che l'utente piazza sul componente.
72
- * Da non usare nel caso di form-adaptive dato che i validatori vengono valutati in base al tipo ecc
73
- */
74
- this.UseUserValidators = true;
75
- /** Indica se il campo è obbligatorio i opzionale */
76
- this.Required = false;
77
- /** Indica se il Form che contiene questo campo è già stato registrato */
78
- this.formHasBennBound = false;
79
- /**
80
- * Indica se questo campo è staccato dal Form, ad esempio nel caso sia in un ng-template proiettato nel form.
81
- *
82
- * In quel caso la variabile **Form** del componente dev'essere valorizzata
83
- */
84
- this.detatchedFromform = false;
85
- /** @ignore */
86
- this.propagateChange = (_) => { };
87
- /** @ignore */
88
- this.onTouched = () => { };
89
- /** @ignore Vedi getter e setter */
90
- this._readonly = false;
91
- /**
92
- * Qualora ci fosse un contesto collegato questa variabile identifica se Sono l'Owner o meno.
93
- *
94
- * Se non lo sono, il campo risulterà in sola lettura
95
- */
96
- this.AppContextOwnership = true;
97
- /** Modalità in cui viene mostrato il campo, 1-1 rispetto alle definizioni di Angular Material */
98
- this.FieldAppearence = "outline";
99
- /** Utilizza o meno il Layout di un form (Label con input di fianco), se false mostra solo l'input */
100
- this.FormLayout = true;
101
- /** Definisce se richiamare l'EventEmitter **inputChange** in maniera classica (true) o solo per la change considerate definitive come tasto invio / blur / ecc... (false) */
102
- this.EmitPendingChanges = true;
103
- /** Attiva o disattiva la validazione per questo componente */
104
- this.Validation = true;
105
- /**
106
- * Permette di scrivere o meno nel campo di Input. Si differenzia dal Readonly in quanto fisicamente l'Input è scrivibile e vengono mantenuti i temi della validazione
107
- * ma interagire direttamente con la casella di Input per l'utente non è possibile
108
- */
109
- this.Frozen = false;
110
- /** Placeholder per quando l'input è vuoto */
111
- this.Placeholder = "";
112
- /** Classe extra per il form-group in cui viene wrappato l'input */
113
- this.FormGroupClass = "";
114
- /** Messaggio da mostrare quando la validazione fallisce per questo componente. Di default viene proposta dalla libreria */
115
- this.FailedValidationMessage = "";
116
- /** Input che forza l'invalidità del componente */
117
- this.ForcedError = false;
118
- /** Mostra un testo a sinistra dell'input (FormLayout permettendo) */
119
- this.Label = "";
120
- /** col-md-X per la label */
121
- this.LabelColWidth = 4;
122
- /** col-md-X per l'input */
123
- this.InputColWidth = 8;
124
- /** Indica se è l'ultimo componente di un form (serve per rimuovere il margine finale) */
125
- this.Last = false;
126
- /** Associa il componente Form a questo input */
127
- this.Form = null;
128
- /** Nome della proprietà che contiene l'Id degli oggetti bindati nella **Source** */
129
- this.IdField = 'id';
130
- /**
131
- * Indica se il **Placeholder** dev'essere usato come una floating label di material o solo come un placeholder standard
132
- *
133
- * Ignorata lato autocomplete, autocomplete-multi e select in quanto hanno gestioni particolari di placeholder ecc...
134
- */
135
- this.FloatingLabel = true;
136
- /** Indica che la parte input del controllo dev'essere una label readonly e la parte label di titolo dev'essere in bold. Per utilizzare un solo componente sia per la modifica che per il dettaglio */
137
- this.DisplayMode = false;
138
- /** Indica la condizione per cui un elemento in Modalità Display può essere visibile in base a ulteriori controlli */
139
- this.DisplayCondition = true;
140
- /** Layout del Form quando è in modalità Display. Di default mantiene la formattazione a form */
141
- this.DisplayLayout = 'form';
142
- /** Separatore da mettere in automatico fra un elemento Form e l'altro quando sono in **DisplayMode** con **DisplayLayout** inline */
143
- this.InlineSeparator = ' ';
144
- /**
145
- * Espressione simil-Angular per cambiare il testo dei componenti che scelgono il proprio modello da una **Source**
146
- *
147
- * Il funzionamento è identico al binder di angular, solo con una graffa invece che 2.
148
- * È inoltre possibile aggiungere pezzi di descrizione in maniera condizionale con la sintassi **:prop?(Prop vale {prop})**
149
- */
150
- this.Display = '{description}';
151
- /** Risultato dell'unione fra le informazioni degli oggetti in **Source** e le impostazioni specificate nell'Input **Display** */
152
- this.BoundSource = [];
153
- /** Indica se è il primo Bind della Source */
154
- this.SourceFirstBind = true;
155
- /** Evento chiamato alla modifica del valore collegato a questo campo */
156
- this.inputChange = new EventEmitter();
157
- /** Evento chiamato al focus del campo di testo/combo/quelcheè, riemitta l'evento originale Javascript */
158
- this.inputFocus = new EventEmitter();
159
- /** Evento chiamato al finalize del campo di testo/combo/quelcheè */
160
- this.inputFinalized = new EventEmitter();
161
- /** Cache delle condizioni scritte tipo :prop?(Roba con {prop}) */
162
- this.BindCheckingGroups = [];
163
- /** Cache delle proprietà scritte tipo --> {prop} */
164
- this.BindProperties = [];
165
- this.destroyed$ = new Subject();
166
- if (ngControl == null) {
167
- if (!this.handleNullNgControl())
168
- console.error("ngControl nullo per qualche motivo! Il 90% delle funzionalità di questo input saranno disabilitate");
169
- return;
170
- }
171
- //Se mi arriva l'injectable del contesto posso capire se questo campo deve essere in readonly o no, solo se ho già inizializzato i controller custom dell'access control
172
- if (this.AppContext && this.AppContext.name && this.ACO_CUSTOMKEY && this.ac && this.ac.InitializedCustom) {
173
- this.AppContextOwnership = !this.ac.has(this.ACO_CUSTOMKEY + ":" + this.AppContext.name, true);
174
- this._readonly = !this.AppContextOwnership;
175
- }
176
- ngControl.valueAccessor = this;
177
- //Se ho già un parent impostato vuol dire che sotto al tag <form> ho direttamente un tag <form-*>..
178
- //.. quindi posso impostare direttamente il riferimento del Form da usare in HTML per il submitted
179
- //Se non c'è vuol dire che dentro al tag <form> c'è uno splendido <ng-container *ngTemplateOutlet="template; context: ....">..
180
- //.. il quale a sua volta contiene il controllo <form-*> quindi avrò il form nell'input "Form"
181
- if (ngControl._parent)
182
- this.Form = ngControl._parent;
183
- else
184
- this.detatchedFromform = true;
185
- //Questa serve per impostare me stesso come si deve, il validatore invece lo passo al validation-input quando viene settato il validationControl
186
- if (this.ngControl) {
187
- //Per tutti i validatori: rigenero la funzione di validazione. Per qualche motivo a volte ho validatori senza...
188
- //... questo è un workaround per essere sicuro di averla sempre.
189
- //Solo i validatori con parametri hanno bisogno della _createValidator (es. "maxlength" che ha 1 parametro), quindi no validatori tipo "required"
190
- if (this._validators) {
191
- this._validators.forEach(v => {
192
- if (v._createValidator)
193
- v._createValidator();
194
- });
195
- }
196
- // Hard kill di validatori che vengono iniettati nel punto sbagliato....
197
- //(<any>this.ngControl.valueAccessor)._validators = []; TODO: Revert @Ema
198
- if (this._validators)
199
- this.ngControl.control.addValidators(this._validators.filter(v => !v.ngOnInit));
200
- this.ngControl.control.updateValueAndValidity();
201
- }
202
- }
203
- /**
204
- * Effettua il Bind/Parse delle varie **Source** unendo le informazioni specificate in **Display**
205
- *
206
- * Eventualmente rieffettua un bind al modello in base al valore della variabile **RebindModelAfterSource**
207
- */
208
- tryBindSourceDisplay() {
209
- if (!this.Source)
210
- return;
211
- if (this.Source.length == 0 && !this.SourceFirstBind)
212
- this.BoundSource = [];
213
- this.SourceFirstBind = false;
214
- // Cache locale per evitare di rifare dei regex.match ogni santa volta
215
- if (this.Source.length > 0) {
216
- this.BoundSource = [];
217
- if (this.BindCheckingGroups.length == 0)
218
- this.evaluateBindCheckingGroups();
219
- if (this.BindProperties.length == 0)
220
- this.evaluateBindProperties();
221
- // Blocco per generare la descrizione finale di un elemento
222
- this.Source.forEach(s => {
223
- // Aggiungo alla BoundSource in formato standard KeyValue
224
- this.BoundSource.push(this.transformSourceItem(s));
225
- });
226
- // Post trybind se c'è un modello associato ricalcolo l'**EvaluatedModel**
227
- if (this.Model)
228
- this.EvaluatedModel = this.BoundSource.find(t => t.id == this.Model)?.description;
229
- }
230
- }
231
- /** Valuta il contenuto della variabile BindCheckingGroups */
232
- evaluateBindCheckingGroups() {
233
- // Blocco per tirare fuori le condizioni scritte tipo --> :prop?(Roba con {prop})
234
- var iffedMatches = this.Display.match(/:([a-z]+)\?\(([^\(\)]+)\)/g);
235
- if (iffedMatches) {
236
- iffedMatches.forEach(m => {
237
- // Stessa regex di sopra ma il tag "i" serve per tirare fuori i singoli capturing group e per qualche motivo
238
- // new RegExp(baseRegex, "g") non funziona quindi non ho potuto razionalizzarlo
239
- let groups = m.match(/:([a-z]+)\?\(([^\(\)]+)\)/i);
240
- this.BindCheckingGroups.push({ global: m, prop: groups[1], whenexists: groups[2] });
241
- });
242
- }
243
- }
244
- /** Valuta il contenuto della variabile BindProperties */
245
- evaluateBindProperties() {
246
- // Blocco per tirare fuori le proprietà scritte tipo --> {prop}
247
- var matches = this.Display.match(/{[a-z]+}/gi);
248
- if (matches)
249
- matches.forEach(m => { this.BindProperties.push({ global: m, prop: m.substring(1, m.length - 1) }); });
250
- }
251
- /**
252
- * Trasforma un oggetto della Source alla sua versione "nuova" basandosi sulle informazioni su gruppi e proprietà
253
- *
254
- * @param {any} item Oggetto da trasformare
255
- *
256
- * @returns {{id: string, description: string}} Oggetto finale trasformato
257
- */
258
- transformSourceItem(s) {
259
- // Parto sempre dalla variabile di Display, poi sostituisco pezzo per pezzo
260
- let desc = this.Display;
261
- // Taglio o mantengo le condizioni in base alla proprietà su cui fare check
262
- // Dopodiché scrivo tutte le proprietà
263
- for (let i = 0; i < this.BindCheckingGroups.length; i++) {
264
- let t = this.BindCheckingGroups[i];
265
- desc = desc.replace(t.global, (s[t.prop] !== null && s[t.prop] !== undefined && s[t.prop] !== "") ? t.whenexists : "");
266
- }
267
- for (let i = 0; i < this.BindProperties.length; i++) {
268
- let t = this.BindProperties[i];
269
- desc = desc.replace(t.global, s[t.prop]);
270
- }
271
- return { id: s[this.IdField], description: desc };
272
- }
273
- /**
274
- * Indica se il comopnente in questione è in grado di gestire ngControl nulli.
275
- * Di default è **false**
276
- *
277
- * @returns {boolean} **true** se gestisco ngControl nulli, **false** altrimenti
278
- */
279
- handleNullNgControl() {
280
- return false;
281
- }
282
- /** @ignore */
283
- ngOnInit() {
284
- // A volte nell'ngOnInit non ci passa quindi lo metto sia qui sia nell'afterviewinit, penso che valga per i componenti di terze parti
285
- // che si collegano per i cazzi loro al form ecc, quindi la parte sopra non serve... BOH
286
- this.checkRequiredness();
287
- //Il nome non c'è se metto standalone : true (CHE NON é DA METTERE)
288
- //Dato che anche il controllo esterno viene considerato come componente del form, non posso basarmi solo sul nome ma devo aggiungere un tag. In questo caso "_internal"
289
- // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA
290
- if (this.ngControl.name)
291
- this.GeneratedName = this.ngControl.name.toString() + "_internal";
292
- // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA
293
- if (this.Form) {
294
- if (!this.formHasBennBound && this.validationControl) {
295
- this.Form.addControl(this.validationControl);
296
- this.formHasBennBound = true;
297
- }
298
- // Quando un form viene submittato la variabile Form.submitted viene assegnata all'input [submitted] del val-control
299
- // Essendo che però la CD è OnPush per dire ad Angular che l'Input è cambiato e permettere i sideffects di assegnare [submitted]
300
- // devo attaccarmi al submit attuale del form e prima di eseguirlo devo marcare il componente per il check
301
- let origFunc = this.Form.onSubmit;
302
- var formOutsideRef = this.Form;
303
- this.Form.onSubmit = (ev) => {
304
- this.cdr.markForCheck();
305
- return origFunc.apply(formOutsideRef);
306
- };
307
- }
308
- if (this.SetValidationSubject) {
309
- this.SetValidationSubject.pipe(takeUntil(this.destroyed$)).subscribe(v => {
310
- if (v.fieldName + "_internal" == this.GeneratedName) {
311
- let control = this.Form.getControl(this.validationControl);
312
- // Tentativo di pezza pezzente
313
- if (control) {
314
- control.setErrors({ forcedtoinvalid: true });
315
- control.markAsTouched();
316
- }
317
- this.ngControl.control.setErrors({ forcedtoinvalid: true });
318
- this.ngControl.control.markAsTouched();
319
- }
320
- });
321
- }
322
- }
323
- /**
324
- * Se sono in modalità detatched devo rimuovere manualmente questo controllo quando entro nella dispose, altrimenti se poi ne andassi a riaggiungere
325
- * un altro con lo stesso "name" ma magari resettato, Angular lo re-idraterebbe con i valori precedenti nonostante il modello fosse vuoto
326
- */
327
- ngOnDestroy() {
328
- if (this.Form && this.validationControl && !!this.Form.getControl(this.validationControl))
329
- this.Form.removeControl(this.validationControl);
330
- this.destroyed$.next();
331
- this.destroyed$.complete();
332
- }
333
- /** Elabora i validatori iniettati e capisce se il valore è obbligatorio o meno */
334
- /** Fratello della funzione con lo stesso nome lato validations */
335
- checkRequiredness() {
336
- // controllo se è settato un required per decidere in maniera condizionale se utilizzare il validatore required nel componente interno
337
- // Lo faccio qui e non in un eventuale ngOnInit o costruttore perché ngOnInit non sempre viene chiamato e da costruttore il validatore non ha ancora il valore giusto
338
- if (this._validators && this._validators.some(elem => elem instanceof RequiredValidator && (elem.required === true || (!elem.required && elem.required !== false))))
339
- this.Required = true;
340
- }
341
- /** @ignore */
342
- ngAfterViewInit() {
343
- if (!this.ngControl)
344
- return;
345
- this.checkRequiredness();
346
- }
347
- /** Helper che collega la funzione di reset del controllo form al controllo di validazione sottostante */
348
- bindResetFunction() {
349
- //faccio in modo che se chiamo il reset del controllo (o della form) dall'esterno si resetti anche il controllo nativo
350
- const origFunc = this.ngControl.control.__original_reset__ || this.ngControl.control.reset;
351
- var ngControl_outsideReference = this.ngControl.control;
352
- var validationControl_outsideReference = this.validationControl.control;
353
- this.ngControl.control.reset = () => {
354
- origFunc.apply(ngControl_outsideReference);
355
- validationControl_outsideReference.reset();
356
- };
357
- //Per consentire richiami multipli di questa funzione prendendo sempre la fiunzione di reset base
358
- this.ngControl.control.__original_reset__ = origFunc;
359
- }
360
- /** @ignore */
361
- writeValue(obj) {
362
- if (obj == null && this.firstBind) {
363
- this.firstBind = false;
364
- return;
365
- }
366
- this.Model = obj;
367
- //Per svegliare Angular. Il valore che entra potrebbe essere valido/non valido e questo lo binda a tutto, form compreso
368
- if (this.ngControl)
369
- this.ngControl.control.updateValueAndValidity();
370
- if (this.Model)
371
- this.onNotNullValueSet();
372
- // markForCheck non va bene in quanto arrivano possibilmente 2 update su un solo ciclo di stack
373
- // setTime*ut che wrappa tutto non va bene sennò spamma delle change detection a suon di interval che rallentano tantissimo l'app
374
- this.cdr.detectChanges();
375
- }
376
- /** @ignore */
377
- registerOnChange(fn) {
378
- this.propagateChange = fn;
379
- }
380
- /** @ignore */
381
- registerOnTouched(fn) {
382
- this.onTouched = fn;
383
- }
384
- /**
385
- * Registra internamente il componente di validazione, ne imposta i validatori e collega il Form, se presente
386
- *
387
- * @param {NgModel} comp Elemento HTML marcato con "#validationControl"
388
- */
389
- registerValComp(comp) {
390
- if (comp) {
391
- // È arrivato un nuovo validationContorl (succede solo nel form-select quando cambia la source da 0 elementi / null ad N elementi, quindi al massimo una volta)
392
- if (this.validationControl && this.Form)
393
- this.Form.removeControl(this.validationControl);
394
- this.validationControl = comp;
395
- //Questa serve per passare avanti i validatori
396
- this.validationControl.control.setValidators(this._validators);
397
- //Se sono staccato dal Form (vuol dire che sono ad esempio dentro un ng-template) aggiungo il controllo al Form passato come input (context del template)
398
- if (this.detatchedFromform && this.Form && !this.formHasBennBound) {
399
- this.Form.addControl(this.validationControl);
400
- this.formHasBennBound = true;
401
- }
402
- if (this.ngControl)
403
- this.bindResetFunction();
404
- //I val-roba questo lo fanno automaticamente dentro... gli altri no (es. semplice input checkbox)
405
- this.updateValidityForNativeInput();
406
- }
407
- }
408
- registerForm(ngForm) {
409
- this.Form = ngForm;
410
- this.Form.addControl(this.validationControl);
411
- this.formHasBennBound = true;
412
- }
413
- /**
414
- * Aggiorna il valore e la validità dell'input sottostante. Non viene chiamato per input nativi come le checkbox
415
- *
416
- * @param {boolean} forcedValue Forza l'aggiornamento anche se non è un input nativo (input nativo = checkbox, ad esempio)
417
- */
418
- updateValidityForNativeInput(forcedValue = false) {
419
- if (forcedValue || this.nativeInput)
420
- this.validationControl.control.updateValueAndValidity();
421
- }
422
- /**
423
- * @ignore
424
- * Classico "spara fuori". Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.
425
- *
426
- * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html
427
- */
428
- changed(forcedValue = null, markForCheck = false, modelEvaluationHandled = false, internal = false) {
429
- let toEmit = forcedValue == null || forcedValue == "" ? this.Model ?? "" : forcedValue;
430
- if (!modelEvaluationHandled)
431
- this.EvaluatedModel = toEmit;
432
- this.propagateChange(toEmit);
433
- this.inputChange.emit(new ChangeEvent(internal, toEmit));
434
- if (markForCheck)
435
- this.cdr.markForCheck();
436
- }
437
- /**
438
- * @ignore
439
- * Classico "spara fuori" sul focus. Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.
440
- *
441
- * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html
442
- */
443
- focused(event) {
444
- this.inputFocus.emit(event);
445
- this.log("Control Focused - " + this.GeneratedName);
446
- }
447
- /**
448
- * @ignore
449
- * Classico "spara fuori" sulla finalizzazione del valore dell'input (blur / tasto invio, ecc...). Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.
450
- *
451
- * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html
452
- */
453
- finalized() {
454
- this.inputFinalized.emit(this.ngControl?.name?.toString());
455
- this.log("Control Value Finalized - " + this.GeneratedName);
456
- }
457
- /**
458
- * Stampa un log a console
459
- *
460
- * @param {string} message Messaggio da stampare
461
- */
462
- log(message) {
463
- if (this.FAV_DEBUG_MODE)
464
- console.log("@forms-and-validations: " + message);
465
- }
466
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: BaseFormControl, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive }); }
467
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: BaseFormControl, inputs: { FocusSubject: "FocusSubject", SetValidationSubject: "SetValidationSubject", FieldAppearence: "FieldAppearence", FormLayout: "FormLayout", EmitPendingChanges: "EmitPendingChanges", Validation: "Validation", Frozen: "Frozen", Placeholder: "Placeholder", FormGroupClass: "FormGroupClass", FailedValidationMessage: "FailedValidationMessage", ForcedError: "ForcedError", Label: "Label", LabelColWidth: "LabelColWidth", InputColWidth: "InputColWidth", Last: "Last", Form: "Form", Source: "Source", IdField: "IdField", FloatingLabel: "FloatingLabel", DisplayMode: "DisplayMode", DisplayCondition: "DisplayCondition", DisplayModeTemplate: "DisplayModeTemplate", DisplayLayout: "DisplayLayout", InlineSeparator: "InlineSeparator", Type: "Type", Display: "Display", maxlength: "maxlength", minlength: "minlength", Readonly: "Readonly", LabelInputRatio: "LabelInputRatio" }, outputs: { inputChange: "inputChange", inputFocus: "inputFocus", inputFinalized: "inputFinalized" }, viewQueries: [{ propertyName: "_validationControl", first: true, predicate: ["validationControl"], descendants: true }, { propertyName: "validationControl_static", first: true, predicate: ["validationControl"], descendants: true, static: true }], ngImport: i0 }); }
468
- }
469
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: BaseFormControl, decorators: [{
470
- type: Directive
471
- }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.NgControl }, { type: Array }, { type: i2.AccessControlService, decorators: [{
472
- type: Optional
473
- }] }, { type: i2.ComponentContext, decorators: [{
474
- type: Optional
475
- }] }, { type: undefined, decorators: [{
476
- type: Optional
477
- }, {
478
- type: Inject,
479
- args: [ACO_CUSTOMKEY]
480
- }] }, { type: undefined, decorators: [{
481
- type: Optional
482
- }, {
483
- type: Inject,
484
- args: [FAV_DEBUG_MODE]
485
- }] }, { type: undefined }]; }, propDecorators: { FocusSubject: [{
486
- type: Input
487
- }], SetValidationSubject: [{
488
- type: Input
489
- }], FieldAppearence: [{
490
- type: Input
491
- }], FormLayout: [{
492
- type: Input
493
- }], EmitPendingChanges: [{
494
- type: Input
495
- }], Validation: [{
496
- type: Input
497
- }], Frozen: [{
498
- type: Input
499
- }], Placeholder: [{
500
- type: Input
501
- }], FormGroupClass: [{
502
- type: Input
503
- }], FailedValidationMessage: [{
504
- type: Input
505
- }], ForcedError: [{
506
- type: Input
507
- }], Label: [{
508
- type: Input
509
- }], LabelColWidth: [{
510
- type: Input
511
- }], InputColWidth: [{
512
- type: Input
513
- }], Last: [{
514
- type: Input
515
- }], Form: [{
516
- type: Input
517
- }], Source: [{
518
- type: Input
519
- }], IdField: [{
520
- type: Input
521
- }], FloatingLabel: [{
522
- type: Input
523
- }], DisplayMode: [{
524
- type: Input
525
- }], DisplayCondition: [{
526
- type: Input
527
- }], DisplayModeTemplate: [{
528
- type: Input
529
- }], DisplayLayout: [{
530
- type: Input
531
- }], InlineSeparator: [{
532
- type: Input
533
- }], Type: [{
534
- type: Input
535
- }], Display: [{
536
- type: Input
537
- }], maxlength: [{
538
- type: Input
539
- }], minlength: [{
540
- type: Input
541
- }], Readonly: [{
542
- type: Input
543
- }], LabelInputRatio: [{
544
- type: Input
545
- }], _validationControl: [{
546
- type: ViewChild,
547
- args: ["validationControl", { static: false }]
548
- }], validationControl_static: [{
549
- type: ViewChild,
550
- args: ["validationControl", { static: true }]
551
- }], inputChange: [{
552
- type: Output
553
- }], inputFocus: [{
554
- type: Output
555
- }], inputFinalized: [{
556
- type: Output
557
- }] } });
558
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-form-control.js","sourceRoot":"","sources":["../../../../../projects/forms-and-validations/src/lib/forms/base-form-control.ts"],"names":[],"mappings":"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","sourcesContent":["// Angular\r\nimport { ControlValueAccessor, NgControl, NgForm, NgModel, RequiredValidator } from \"@angular/forms\";\r\nimport { EventEmitter, Input, Output, ViewChild, Optional, Inject, Directive, ChangeDetectorRef } from \"@angular/core\";\r\n\r\n// // Configurazioni\r\nimport { AccessControlService, ComponentContext } from \"@esfaenza/access-control\";\r\nimport { ACO_CUSTOMKEY, FAV_DEBUG_MODE } from '../tokens';\r\nimport { Subject } from \"rxjs\";\r\nimport { TemplateRef } from \"@angular/core\";\r\nimport { ChangeEvent } from \"../models/ChangeEvent\";\r\nimport { takeUntil } from \"rxjs/operators\";\r\n\r\n/** Componente base da cui tutti i componenti Form implementano */\r\n@Directive()\r\nexport abstract class BaseFormControl implements ControlValueAccessor {\r\n\r\n    /**\r\n     * https://github.com/angular/angular/issues/14988\r\n     * TL:DR; Il primo valore passato al writeValue è nullo perché quando i cagnacci fanno new FormControl()\r\n     * scrivono un writeValue nullo... quindi al primo binding salto l'inizializzazione se ho la coppia null-firstBind\r\n     * poi mettendo firstBind a false di lì in poi funziono come devo. Vista la casistica evito essenzialmente metà detectChanges\r\n     */\r\n    private firstBind: boolean = true;\r\n\r\n    /**\r\n    * Indica di utilizzare i validatori che l'utente piazza sul componente. \r\n    * Da non usare nel caso di form-adaptive dato che i validatori vengono valutati in base al tipo ecc\r\n    */\r\n    protected UseUserValidators:boolean = true;\r\n\r\n    /** Indica se il campo è obbligatorio i opzionale */\r\n    public Required: boolean = false;\r\n\r\n    /** Modello rappresentato da questo campo */\r\n    public Model: any;\r\n\r\n    /** Valore del Modello */\r\n    public EvaluatedModel: string;\r\n\r\n    /** Nome generato per il controllo interno. Di default è il nome del controllo esterno con l'aggiunta di \"_internal\". Serve per evitare cross-bind */\r\n    public GeneratedName: string;\r\n\r\n    /** Indica se il Form che contiene questo campo è già stato registrato */\r\n    protected formHasBennBound: boolean = false;\r\n\r\n    /**\r\n     * Indica se questo campo è staccato dal Form, ad esempio nel caso sia in un ng-template proiettato nel form.\r\n     * \r\n     * In quel caso la variabile **Form** del componente dev'essere valorizzata\r\n     */\r\n    protected detatchedFromform: boolean = false;\r\n\r\n    /** @ignore */\r\n    protected propagateChange = (_: any) => { };\r\n\r\n    /** @ignore */\r\n    protected onTouched = () => { };\r\n\r\n    /** @ignore Vedi getter e setter */\r\n    public _readonly: boolean = false;\r\n\r\n    /**\r\n     * Qualora ci fosse un contesto collegato questa variabile identifica se Sono l'Owner o meno.\r\n     * \r\n     * Se non lo sono, il campo risulterà in sola lettura \r\n     */\r\n    private AppContextOwnership: boolean = true;\r\n\r\n    /** Subject a cui l'oggetto interno si collega per effettuare il focus dell'elemento input sottostante */\r\n    @Input() FocusSubject: Subject<void>;\r\n    /**\r\n    * @EXPERIMENTAL\r\n    */\r\n    @Input() SetValidationSubject: Subject<{ fieldName: string, error: string }>;\r\n\r\n    /** Modalità in cui viene mostrato il campo, 1-1 rispetto alle definizioni di Angular Material  */\r\n    @Input() FieldAppearence: \"legacy\" | \"standard\" | \"fill\" | \"outline\" = \"outline\";\r\n\r\n    /** Utilizza o meno il Layout di un form (Label con input di fianco), se false mostra solo l'input */\r\n    @Input() FormLayout: boolean = true;\r\n\r\n    /** Definisce se richiamare l'EventEmitter **inputChange** in maniera classica (true) o solo per la change considerate definitive come tasto invio / blur / ecc... (false) */\r\n    @Input() EmitPendingChanges: boolean = true;\r\n\r\n    /** Attiva o disattiva la validazione per questo componente */\r\n    @Input() Validation: boolean = true;\r\n\r\n    /** \r\n     * Permette di scrivere o meno nel campo di Input. Si differenzia dal Readonly in quanto fisicamente l'Input è scrivibile e vengono mantenuti i temi della validazione\r\n     * ma interagire direttamente con la casella di Input per l'utente non è possibile\r\n     */\r\n    @Input() Frozen: boolean = false;\r\n\r\n    /** Placeholder per quando l'input è vuoto */\r\n    @Input() Placeholder: string = \"\";\r\n\r\n    /** Classe extra per il form-group in cui viene wrappato l'input */\r\n    @Input() FormGroupClass: string = \"\";\r\n\r\n    /** Messaggio da mostrare quando la validazione fallisce per questo componente. Di default viene proposta dalla libreria */\r\n    @Input() FailedValidationMessage: string = \"\";\r\n\r\n    /** Input che forza l'invalidità del componente */\r\n    @Input() ForcedError: boolean = false;\r\n\r\n    /** Mostra un testo a sinistra dell'input (FormLayout permettendo) */\r\n    @Input() Label: string = \"\";\r\n\r\n    /** col-md-X per la label */\r\n    @Input() LabelColWidth: number = 4;\r\n\r\n    /** col-md-X per l'input */\r\n    @Input() InputColWidth: number = 8;\r\n\r\n    /** Indica se è l'ultimo componente di un form (serve per rimuovere il margine finale) */\r\n    @Input() Last: boolean = false;\r\n\r\n    /** Associa il componente Form a questo input */\r\n    @Input() Form: NgForm = null;\r\n\r\n    /** Sorgente da cui scegliere valori per i componenti Form che lo consentono */\r\n    @Input() Source: any[];\r\n\r\n    /** Nome della proprietà che contiene l'Id degli oggetti bindati nella **Source** */\r\n    @Input() IdField: string = 'id';\r\n\r\n    /**\r\n     * Indica se il **Placeholder** dev'essere usato come una floating label di material o solo come un placeholder standard \r\n     * \r\n     * Ignorata lato autocomplete, autocomplete-multi e select in quanto hanno gestioni particolari di placeholder ecc...\r\n     */\r\n    @Input() FloatingLabel: boolean = true;\r\n\r\n    /** Indica che la parte input del controllo dev'essere una label readonly e la parte label di titolo dev'essere in bold. Per utilizzare un solo componente sia per la modifica che per il dettaglio */\r\n    @Input() DisplayMode: boolean = false;\r\n\r\n    /** Indica la condizione per cui un elemento in Modalità Display può essere visibile in base a ulteriori controlli */\r\n    @Input() DisplayCondition: boolean = true;\r\n\r\n    /** Template per visualizzare la modalità **DisplayMode** come vuole il programmatore frocio di turno */\r\n    @Input() DisplayModeTemplate: TemplateRef<any>;\r\n\r\n    /** Layout del Form quando è in modalità Display. Di default mantiene la formattazione a form */\r\n    @Input() DisplayLayout: 'form' | 'inline' | 'hidden' = 'form';\r\n\r\n    /** Separatore da mettere in automatico fra un elemento Form e l'altro quando sono in **DisplayMode** con **DisplayLayout** inline */\r\n    @Input() InlineSeparator: string = ' ';\r\n    \r\n    /** Type HTML (password/text/etc...) */\r\n    @Input() Type: string;\r\n\r\n    /**\r\n     * Espressione simil-Angular per cambiare il testo dei componenti che scelgono il proprio modello da una **Source**\r\n     * \r\n     * Il funzionamento è identico al binder di angular, solo con una graffa invece che 2.\r\n     * È inoltre possibile aggiungere pezzi di descrizione in maniera condizionale con la sintassi **:prop?(Prop vale {prop})**\r\n     */\r\n    @Input() Display: string = '{description}';\r\n\r\n    /** Risultato dell'unione fra le informazioni degli oggetti in **Source** e le impostazioni specificate nell'Input **Display** */\r\n    public BoundSource: { id: string, description: string }[] = [];\r\n\r\n    /** Indica se è il primo Bind della Source */\r\n    protected SourceFirstBind: boolean = true;\r\n\r\n    /**\r\n     * Max length\r\n     */\r\n    @Input() maxlength: number;\r\n\r\n    /**\r\n     * Max length\r\n     */\r\n    @Input() minlength: number;\r\n\r\n    /**\r\n     * Ottiene il valore della variabile **_readonly**\r\n     * \r\n     * @returns {boolean} **true** se sola lettura, **false** altrimenti\r\n     */\r\n    get Readonly(): boolean { \r\n        return this._readonly;\r\n    }\r\n\r\n    /**\r\n     * Imposta il valore della variabile **_readonly** gestendo anche il Contesto applicativo\r\n     * \r\n     * @param {boolean} value Valore da impostare\r\n     */\r\n    @Input() set Readonly(value: boolean) {\r\n        if (this.AppContextOwnership) \r\n            this._readonly = value;\r\n    }\r\n\r\n    /**\r\n     * Permette di settare in una volta solo gli input LabelColWidth e InputColWidth, basta scriverlo con la sintassi \"X Y\"\r\n     * \r\n     * @param {string} input Input nel formato \"X Y\"\r\n     */\r\n    @Input() set LabelInputRatio(input: string) {\r\n        var ratio = input.split(/\\s/);\r\n        this.LabelColWidth = parseInt(ratio[0]);\r\n        this.InputColWidth = parseInt(ratio[1]);\r\n    }\r\n\r\n    /** val-qualcosa che fa effettivamente da input, o eventualmente il singolo input in casi particolari (es checkbox) */\r\n    protected validationControl: NgModel;\r\n\r\n    /**\r\n     * Query sull'elemento 'validationControl' che funziona in ambiente statico\r\n     * \r\n     * @param {NgModel} comp Elemento HTML marcato con \"#validationControl\"\r\n     */\r\n    @ViewChild(\"validationControl\", { static: false }) set _validationControl(comp: NgModel) { this.registerValComp(comp); }\r\n\r\n    /**\r\n     * Query sull'elemento 'validationControl' che funziona in ambiente dinamico\r\n     * \r\n     * @param {NgModel} comp Elemento HTML marcato con \"#validationControl\"\r\n     */\r\n    @ViewChild(\"validationControl\", { static: true }) set validationControl_static(comp: NgModel) { this.registerValComp(comp); };\r\n\r\n    /** Evento chiamato alla modifica del valore collegato a questo campo */\r\n    @Output() inputChange: EventEmitter<ChangeEvent> = new EventEmitter<ChangeEvent>();\r\n\r\n    /** Evento chiamato al focus del campo di testo/combo/quelcheè, riemitta l'evento originale Javascript */\r\n    @Output() inputFocus: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n    /** Evento chiamato al finalize del campo di testo/combo/quelcheè */\r\n    @Output() inputFinalized: EventEmitter<string> = new EventEmitter<string>();\r\n\r\n    /** Cache delle condizioni scritte tipo :prop?(Roba con {prop}) */\r\n    private BindCheckingGroups: { global: string, prop: string, whenexists: string }[] = []\r\n\r\n    /** Cache delle proprietà scritte tipo --> {prop} */\r\n    private BindProperties: { global: string, prop: string }[] = [];\r\n\r\n    /** @ignore Costruttore  */\r\n    constructor(protected cdr: ChangeDetectorRef, public ngControl: NgControl, protected _validators: Array<any>, @Optional() private ac: AccessControlService, @Optional() public AppContext: ComponentContext, @Optional() @Inject(ACO_CUSTOMKEY) private ACO_CUSTOMKEY: string, @Optional() @Inject(FAV_DEBUG_MODE) private FAV_DEBUG_MODE: boolean, private nativeInput: boolean = false) {\r\n\r\n        if (ngControl == null) {\r\n            if (!this.handleNullNgControl())\r\n                console.error(\"ngControl nullo per qualche motivo! Il 90% delle funzionalità di questo input saranno disabilitate\");\r\n\r\n            return;\r\n        }\r\n\r\n        //Se mi arriva l'injectable del contesto posso capire se questo campo deve essere in readonly o no, solo se ho già inizializzato i controller custom dell'access control\r\n        if (this.AppContext && this.AppContext.name && this.ACO_CUSTOMKEY && this.ac && this.ac.InitializedCustom) {\r\n            this.AppContextOwnership = !this.ac.has(this.ACO_CUSTOMKEY + \":\" + this.AppContext.name, true);\r\n            this._readonly = !this.AppContextOwnership\r\n        }\r\n\r\n        ngControl.valueAccessor = this;\r\n\r\n        //Se ho già un parent impostato vuol dire che sotto al tag <form> ho direttamente un tag <form-*>..\r\n        //.. quindi posso impostare direttamente il riferimento del Form da usare in HTML per il submitted\r\n\r\n        //Se non c'è vuol dire che dentro al tag <form> c'è uno splendido <ng-container *ngTemplateOutlet=\"template; context: ....\">..\r\n        //.. il quale a sua volta contiene il controllo <form-*> quindi avrò il form nell'input \"Form\"\r\n        if ((<any>ngControl)._parent)\r\n            this.Form = <NgForm>(<any>ngControl)._parent;\r\n        else\r\n            this.detatchedFromform = true;\r\n\r\n        //Questa serve per impostare me stesso come si deve, il validatore invece lo passo al validation-input quando viene settato il validationControl\r\n        if (this.ngControl) {\r\n            //Per tutti i validatori: rigenero la funzione di validazione. Per qualche motivo a volte ho validatori senza...\r\n            //... questo è un workaround per essere sicuro di averla sempre.\r\n            //Solo i validatori con parametri hanno bisogno della _createValidator (es. \"maxlength\" che ha 1 parametro), quindi no validatori tipo \"required\"\r\n            if (this._validators) {\r\n                this._validators.forEach(v => {\r\n                    if (v._createValidator)\r\n                        v._createValidator();\r\n                });\r\n            }\r\n\r\n            // Hard kill di validatori che vengono iniettati nel punto sbagliato....\r\n            //(<any>this.ngControl.valueAccessor)._validators = []; TODO: Revert @Ema\r\n\r\n            if (this._validators)\r\n                this.ngControl.control.addValidators(this._validators.filter(v => !v.ngOnInit));\r\n\r\n            this.ngControl.control.updateValueAndValidity();\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Effettua il Bind/Parse delle varie **Source** unendo le informazioni specificate in **Display**\r\n     * \r\n     * Eventualmente rieffettua un bind al modello in base al valore della variabile **RebindModelAfterSource**\r\n     */\r\n    protected tryBindSourceDisplay(): void {\r\n        if (!this.Source)\r\n            return;\r\n\r\n        if (this.Source.length == 0 && !this.SourceFirstBind)\r\n            this.BoundSource = [];\r\n\r\n        this.SourceFirstBind = false;\r\n\r\n        // Cache locale per evitare di rifare dei regex.match ogni santa volta\r\n        if (this.Source.length > 0) {\r\n            this.BoundSource = [];\r\n\r\n            if (this.BindCheckingGroups.length == 0)\r\n                this.evaluateBindCheckingGroups();\r\n\r\n            if (this.BindProperties.length == 0)\r\n                this.evaluateBindProperties();\r\n\r\n            // Blocco per generare la descrizione finale di un elemento\r\n            this.Source.forEach(s => {\r\n                // Aggiungo alla BoundSource in formato standard KeyValue\r\n                this.BoundSource.push(this.transformSourceItem(s));\r\n            });\r\n\r\n            // Post trybind se c'è un modello associato ricalcolo l'**EvaluatedModel**\r\n            if (this.Model)\r\n                this.EvaluatedModel = this.BoundSource.find(t => t.id == this.Model)?.description;\r\n        }\r\n    }\r\n\r\n    /** Valuta il contenuto della variabile BindCheckingGroups */\r\n    private evaluateBindCheckingGroups() {\r\n        // Blocco per tirare fuori le condizioni scritte tipo --> :prop?(Roba con {prop})\r\n        var iffedMatches = this.Display.match(/:([a-z]+)\\?\\(([^\\(\\)]+)\\)/g);\r\n        if (iffedMatches) {\r\n            iffedMatches.forEach(m => {\r\n                // Stessa regex di sopra ma il tag \"i\" serve per tirare fuori i singoli capturing group e per qualche motivo \r\n                // new RegExp(baseRegex, \"g\") non funziona quindi non ho potuto razionalizzarlo\r\n                let groups = m.match(/:([a-z]+)\\?\\(([^\\(\\)]+)\\)/i);\r\n                this.BindCheckingGroups.push({ global: m, prop: groups[1], whenexists: groups[2] });\r\n            });\r\n        }\r\n    }\r\n\r\n    /** Valuta il contenuto della variabile BindProperties */\r\n    private evaluateBindProperties() {\r\n        // Blocco per tirare fuori le proprietà scritte tipo --> {prop}\r\n        var matches = this.Display.match(/{[a-z]+}/gi);\r\n        if (matches)\r\n            matches.forEach(m => { this.BindProperties.push({ global: m, prop: m.substring(1, m.length - 1) }); });\r\n    }\r\n\r\n    /**\r\n     * Trasforma un oggetto della Source alla sua versione \"nuova\" basandosi sulle informazioni su gruppi e proprietà\r\n     * \r\n     * @param {any} item Oggetto da trasformare\r\n     * \r\n     * @returns {{id: string, description: string}} Oggetto finale trasformato\r\n     */\r\n    private transformSourceItem(s: any): {id: string, description: string} {\r\n        // Parto sempre dalla variabile di Display, poi sostituisco pezzo per pezzo\r\n        let desc = this.Display;\r\n\r\n        // Taglio o mantengo le condizioni in base alla proprietà su cui fare check\r\n        // Dopodiché scrivo tutte le proprietà\r\n        for (let i = 0; i < this.BindCheckingGroups.length; i++) {\r\n            let t = this.BindCheckingGroups[i];\r\n            desc = desc.replace(t.global, (s[t.prop] !== null && s[t.prop] !== undefined && s[t.prop] !== \"\") ? t.whenexists : \"\");\r\n        }\r\n\r\n        for (let i = 0; i < this.BindProperties.length; i++) {\r\n            let t = this.BindProperties[i];\r\n            desc = desc.replace(t.global, s[t.prop]);\r\n        }\r\n\r\n        return { id: s[this.IdField], description: desc };\r\n    }\r\n\r\n    /**\r\n     * Indica se il comopnente in questione è in grado di gestire ngControl nulli. \r\n     * Di default è **false**\r\n     * \r\n     * @returns {boolean} **true** se gestisco ngControl nulli, **false** altrimenti\r\n     */\r\n    handleNullNgControl(): boolean {\r\n        return false;\r\n    }\r\n\r\n    /** @ignore */\r\n    ngOnInit() {\r\n        // A volte nell'ngOnInit non ci passa quindi lo metto sia qui sia nell'afterviewinit, penso che valga per i componenti di terze parti\r\n        // che si collegano per i cazzi loro al form ecc, quindi la parte sopra non serve... BOH\r\n        this.checkRequiredness();\r\n\r\n        //Il nome non c'è se metto standalone : true (CHE NON é DA METTERE)\r\n        //Dato che anche il controllo esterno viene considerato come componente del form, non posso basarmi solo sul nome ma devo aggiungere un tag. In questo caso \"_internal\"\r\n        // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA\r\n        if (this.ngControl.name)\r\n            this.GeneratedName = this.ngControl.name.toString() + \"_internal\";\r\n\r\n        // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA\r\n        if (this.Form) {\r\n            if (!this.formHasBennBound && this.validationControl) {\r\n                this.Form.addControl(this.validationControl);\r\n                this.formHasBennBound = true;\r\n            }\r\n\r\n            // Quando un form viene submittato la variabile Form.submitted viene assegnata all'input [submitted] del val-control\r\n            // Essendo che però la CD è OnPush per dire ad Angular che l'Input è cambiato e permettere i sideffects di assegnare [submitted]\r\n            // devo attaccarmi al submit attuale del form e prima di eseguirlo devo marcare il componente per il check\r\n            let origFunc = this.Form.onSubmit;\r\n            var formOutsideRef = this.Form;\r\n            this.Form.onSubmit = (ev): boolean => {\r\n                this.cdr.markForCheck();\r\n                return origFunc.apply(formOutsideRef);\r\n            };\r\n        }\r\n\r\n        if (this.SetValidationSubject) {\r\n            this.SetValidationSubject.pipe(takeUntil(this.destroyed$)).subscribe(v => {\r\n                if (v.fieldName + \"_internal\" == this.GeneratedName) {\r\n                    let control = this.Form.getControl(this.validationControl);\r\n                    \r\n                    // Tentativo di pezza pezzente\r\n                    if (control) {\r\n                        control.setErrors({ forcedtoinvalid: true });\r\n                        control.markAsTouched();\r\n                    }\r\n\r\n                    this.ngControl.control.setErrors({ forcedtoinvalid: true });\r\n                    this.ngControl.control.markAsTouched();\r\n                }\r\n            })\r\n        }\r\n    }\r\n\r\n    destroyed$: Subject<void> = new Subject<void>();\r\n\r\n    /**\r\n     * Se sono in modalità detatched devo rimuovere manualmente questo controllo quando entro nella dispose, altrimenti se poi ne andassi a riaggiungere\r\n     * un altro con lo stesso \"name\" ma magari resettato, Angular lo re-idraterebbe con i valori precedenti nonostante il modello fosse vuoto\r\n     */\r\n    ngOnDestroy() {\r\n        if (this.Form && this.validationControl && !!this.Form.getControl(this.validationControl))\r\n            this.Form.removeControl(this.validationControl);\r\n\r\n\r\n        this.destroyed$.next();\r\n        this.destroyed$.complete();\r\n    }\r\n\r\n    /** Elabora i validatori iniettati e capisce se il valore è obbligatorio o meno */\r\n    /** Fratello della funzione con lo stesso nome lato validations */\r\n    checkRequiredness() {\r\n        // controllo se è settato un required per decidere in maniera condizionale se utilizzare il validatore required nel componente interno\r\n        // Lo faccio qui e non in un eventuale ngOnInit o costruttore perché ngOnInit non sempre viene chiamato e da costruttore il validatore non ha ancora il valore giusto\r\n        if (this._validators && this._validators.some(elem => elem instanceof RequiredValidator && (elem.required === true || (!elem.required && elem.required !== false))))\r\n            this.Required = true;\r\n    }\r\n\r\n    /** @ignore */\r\n    ngAfterViewInit() {\r\n        if (!this.ngControl)\r\n            return;\r\n\r\n        this.checkRequiredness();\r\n    }\r\n\r\n    /** Helper che collega la funzione di reset del controllo form al controllo di validazione sottostante */\r\n    private bindResetFunction() {\r\n        //faccio in modo che se chiamo il reset del controllo (o della form) dall'esterno si resetti anche il controllo nativo\r\n        const origFunc = (<any>this.ngControl.control).__original_reset__ || this.ngControl.control.reset;\r\n        var ngControl_outsideReference = this.ngControl.control;\r\n        var validationControl_outsideReference = this.validationControl.control;\r\n\r\n        this.ngControl.control.reset = () => {\r\n            origFunc.apply(ngControl_outsideReference);\r\n            validationControl_outsideReference.reset();\r\n        };\r\n\r\n        //Per consentire richiami multipli di questa funzione prendendo sempre la fiunzione di reset base\r\n        (<any>this.ngControl.control).__original_reset__ = origFunc\r\n    }\r\n\r\n    /** @ignore */\r\n    writeValue(obj: any): void {\r\n        if (obj == null && this.firstBind) {\r\n            this.firstBind = false;\r\n            return;\r\n        }\r\n\r\n        this.Model = obj;\r\n\r\n        //Per svegliare Angular. Il valore che entra potrebbe essere valido/non valido e questo lo binda a tutto, form compreso\r\n        if (this.ngControl)\r\n            this.ngControl.control.updateValueAndValidity();\r\n\r\n        if (this.Model)\r\n            this.onNotNullValueSet();\r\n\r\n        // markForCheck non va bene in quanto arrivano possibilmente 2 update su un solo ciclo di stack\r\n        // setTime*ut che wrappa tutto non va bene sennò spamma delle change detection a suon di interval che rallentano tantissimo l'app\r\n        this.cdr.detectChanges();\r\n    }\r\n\r\n    /** @ignore */\r\n    registerOnChange(fn: any): void {\r\n        this.propagateChange = fn;\r\n    }\r\n\r\n    /** @ignore */\r\n    registerOnTouched(fn: any): void {\r\n        this.onTouched = fn;\r\n    }\r\n\r\n    /**\r\n     * Registra internamente il componente di validazione, ne imposta i validatori e collega il Form, se presente\r\n     * \r\n     * @param {NgModel} comp Elemento HTML marcato con \"#validationControl\"\r\n     */\r\n    registerValComp(comp: NgModel) {\r\n        if (comp) {\r\n            // È arrivato un nuovo validationContorl (succede solo nel form-select quando cambia la source da 0 elementi / null ad N elementi, quindi al massimo una volta) \r\n            if (this.validationControl && this.Form)\r\n                this.Form.removeControl(this.validationControl);\r\n\r\n            this.validationControl = comp;\r\n\r\n            //Questa serve per passare avanti i validatori\r\n            this.validationControl.control.setValidators(this._validators);\r\n\r\n            //Se sono staccato dal Form (vuol dire che sono ad esempio dentro un ng-template) aggiungo il controllo al Form passato come input (context del template)\r\n            if (this.detatchedFromform && this.Form && !this.formHasBennBound) {\r\n                this.Form.addControl(this.validationControl)\r\n                this.formHasBennBound = true;\r\n            }\r\n\r\n            if (this.ngControl)\r\n                this.bindResetFunction();\r\n\r\n            //I val-roba questo lo fanno automaticamente dentro... gli altri no (es. semplice input checkbox)\r\n            this.updateValidityForNativeInput();\r\n        }\r\n    }\r\n\r\n    public registerForm(ngForm: NgForm) {\r\n        this.Form = ngForm;\r\n        this.Form.addControl(this.validationControl)\r\n        this.formHasBennBound = true;\r\n    }\r\n\r\n    /**\r\n     * Aggiorna il valore e la validità dell'input sottostante. Non viene chiamato per input nativi come le checkbox\r\n     * \r\n     * @param {boolean} forcedValue Forza l'aggiornamento anche se non è un input nativo (input nativo = checkbox, ad esempio)\r\n     */\r\n    updateValidityForNativeInput(forcedValue: boolean = false) {\r\n        if (forcedValue || this.nativeInput)\r\n            this.validationControl.control.updateValueAndValidity();\r\n    }\r\n\r\n    /**\r\n     * @ignore\r\n     * Classico \"spara fuori\". Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.\r\n     * \r\n     * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html\r\n     */\r\n    changed(forcedValue: any = null, markForCheck: boolean = false, modelEvaluationHandled: boolean = false, internal: boolean = false) {\r\n        let toEmit = forcedValue == null || forcedValue == \"\" ? this.Model ?? \"\" : forcedValue;\r\n\r\n        if(!modelEvaluationHandled)\r\n            this.EvaluatedModel = toEmit;\r\n        \r\n        this.propagateChange(toEmit);\r\n        this.inputChange.emit(new ChangeEvent(internal, toEmit));\r\n        if (markForCheck)\r\n            this.cdr.markForCheck();\r\n    }\r\n\r\n    /**\r\n     * @ignore\r\n     * Classico \"spara fuori\" sul focus. Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.\r\n     * \r\n     * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html\r\n     */\r\n    focused(event: any) {\r\n        this.inputFocus.emit(event);\r\n        this.log(\"Control Focused - \" + this.GeneratedName);\r\n    }\r\n\r\n    /**\r\n     * @ignore\r\n     * Classico \"spara fuori\" sulla finalizzazione del valore dell'input (blur / tasto invio, ecc...). Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.\r\n     * \r\n     * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html\r\n     */\r\n    finalized() {\r\n        this.inputFinalized.emit(this.ngControl?.name?.toString());\r\n        this.log(\"Control Value Finalized - \" + this.GeneratedName);\r\n    }\r\n\r\n    /**\r\n     * Stampa un log a console\r\n     * \r\n     * @param {string} message Messaggio da stampare\r\n     */\r\n    public log(message: string) {\r\n        if (this.FAV_DEBUG_MODE)\r\n            console.log(\"@forms-and-validations: \" + message)\r\n    }\r\n\r\n    /** @ignore */\r\n    abstract onNotNullValueSet(): void;\r\n}"]}
1
+ // Angular
2
+ import { RequiredValidator } from "@angular/forms";
3
+ import { EventEmitter, Input, Output, ViewChild, Optional, Inject, Directive } from "@angular/core";
4
+ import { ACO_CUSTOMKEY, FAV_DEBUG_MODE } from '../tokens';
5
+ import { Subject } from "rxjs";
6
+ import { ChangeEvent } from "../models/ChangeEvent";
7
+ import { takeUntil } from "rxjs/operators";
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/forms";
10
+ import * as i2 from "@esfaenza/access-control";
11
+ /** Componente base da cui tutti i componenti Form implementano */
12
+ export class BaseFormControl {
13
+ /**
14
+ * Ottiene il valore della variabile **_readonly**
15
+ *
16
+ * @returns {boolean} **true** se sola lettura, **false** altrimenti
17
+ */
18
+ get Readonly() {
19
+ return this._readonly;
20
+ }
21
+ /**
22
+ * Imposta il valore della variabile **_readonly** gestendo anche il Contesto applicativo
23
+ *
24
+ * @param {boolean} value Valore da impostare
25
+ */
26
+ set Readonly(value) {
27
+ if (this.AppContextOwnership)
28
+ this._readonly = value;
29
+ }
30
+ /**
31
+ * Permette di settare in una volta solo gli input LabelColWidth e InputColWidth, basta scriverlo con la sintassi "X Y"
32
+ *
33
+ * @param {string} input Input nel formato "X Y"
34
+ */
35
+ set LabelInputRatio(input) {
36
+ var ratio = input.split(/\s/);
37
+ this.LabelColWidth = parseInt(ratio[0]);
38
+ this.InputColWidth = parseInt(ratio[1]);
39
+ }
40
+ /**
41
+ * Query sull'elemento 'validationControl' che funziona in ambiente statico
42
+ *
43
+ * @param {NgModel} comp Elemento HTML marcato con "#validationControl"
44
+ */
45
+ set _validationControl(comp) { this.registerValComp(comp); }
46
+ /**
47
+ * Query sull'elemento 'validationControl' che funziona in ambiente dinamico
48
+ *
49
+ * @param {NgModel} comp Elemento HTML marcato con "#validationControl"
50
+ */
51
+ set validationControl_static(comp) { this.registerValComp(comp); }
52
+ ;
53
+ /** @ignore Costruttore */
54
+ constructor(cdr, ngControl, _validators, ac, AppContext, ACO_CUSTOMKEY, FAV_DEBUG_MODE, nativeInput = false) {
55
+ this.cdr = cdr;
56
+ this.ngControl = ngControl;
57
+ this._validators = _validators;
58
+ this.ac = ac;
59
+ this.AppContext = AppContext;
60
+ this.ACO_CUSTOMKEY = ACO_CUSTOMKEY;
61
+ this.FAV_DEBUG_MODE = FAV_DEBUG_MODE;
62
+ this.nativeInput = nativeInput;
63
+ /**
64
+ * https://github.com/angular/angular/issues/14988
65
+ * TL:DR; Il primo valore passato al writeValue è nullo perché quando i cagnacci fanno new FormControl()
66
+ * scrivono un writeValue nullo... quindi al primo binding salto l'inizializzazione se ho la coppia null-firstBind
67
+ * poi mettendo firstBind a false di lì in poi funziono come devo. Vista la casistica evito essenzialmente metà detectChanges
68
+ */
69
+ this.firstBind = true;
70
+ /**
71
+ * Indica di utilizzare i validatori che l'utente piazza sul componente.
72
+ * Da non usare nel caso di form-adaptive dato che i validatori vengono valutati in base al tipo ecc
73
+ */
74
+ this.UseUserValidators = true;
75
+ /** Indica se il campo è obbligatorio i opzionale */
76
+ this.Required = false;
77
+ /** Indica se il Form che contiene questo campo è già stato registrato */
78
+ this.formHasBennBound = false;
79
+ /**
80
+ * Indica se questo campo è staccato dal Form, ad esempio nel caso sia in un ng-template proiettato nel form.
81
+ *
82
+ * In quel caso la variabile **Form** del componente dev'essere valorizzata
83
+ */
84
+ this.detatchedFromform = false;
85
+ /** @ignore */
86
+ this.propagateChange = (_) => { };
87
+ /** @ignore */
88
+ this.onTouched = () => { };
89
+ /** @ignore Vedi getter e setter */
90
+ this._readonly = false;
91
+ /**
92
+ * Qualora ci fosse un contesto collegato questa variabile identifica se Sono l'Owner o meno.
93
+ *
94
+ * Se non lo sono, il campo risulterà in sola lettura
95
+ */
96
+ this.AppContextOwnership = true;
97
+ /** Modalità in cui viene mostrato il campo, 1-1 rispetto alle definizioni di Angular Material */
98
+ this.FieldAppearence = "outline";
99
+ /** Utilizza o meno il Layout di un form (Label con input di fianco), se false mostra solo l'input */
100
+ this.FormLayout = true;
101
+ /** Definisce se richiamare l'EventEmitter **inputChange** in maniera classica (true) o solo per la change considerate definitive come tasto invio / blur / ecc... (false) */
102
+ this.EmitPendingChanges = true;
103
+ /** Attiva o disattiva la validazione per questo componente */
104
+ this.Validation = true;
105
+ /**
106
+ * Permette di scrivere o meno nel campo di Input. Si differenzia dal Readonly in quanto fisicamente l'Input è scrivibile e vengono mantenuti i temi della validazione
107
+ * ma interagire direttamente con la casella di Input per l'utente non è possibile
108
+ */
109
+ this.Frozen = false;
110
+ /** Placeholder per quando l'input è vuoto */
111
+ this.Placeholder = "";
112
+ /** Classe extra per il form-group in cui viene wrappato l'input */
113
+ this.FormGroupClass = "";
114
+ /** Messaggio da mostrare quando la validazione fallisce per questo componente. Di default viene proposta dalla libreria */
115
+ this.FailedValidationMessage = "";
116
+ /** Input che forza l'invalidità del componente */
117
+ this.ForcedError = false;
118
+ /** Mostra un testo a sinistra dell'input (FormLayout permettendo) */
119
+ this.Label = "";
120
+ /** col-md-X per la label */
121
+ this.LabelColWidth = 4;
122
+ /** col-md-X per l'input */
123
+ this.InputColWidth = 8;
124
+ /** Indica se è l'ultimo componente di un form (serve per rimuovere il margine finale) */
125
+ this.Last = false;
126
+ /** Associa il componente Form a questo input */
127
+ this.Form = null;
128
+ /** Nome della proprietà che contiene l'Id degli oggetti bindati nella **Source** */
129
+ this.IdField = 'id';
130
+ /**
131
+ * Indica se il **Placeholder** dev'essere usato come una floating label di material o solo come un placeholder standard
132
+ *
133
+ * Ignorata lato autocomplete, autocomplete-multi e select in quanto hanno gestioni particolari di placeholder ecc...
134
+ */
135
+ this.FloatingLabel = true;
136
+ /** Indica che la parte input del controllo dev'essere una label readonly e la parte label di titolo dev'essere in bold. Per utilizzare un solo componente sia per la modifica che per il dettaglio */
137
+ this.DisplayMode = false;
138
+ /** Indica la condizione per cui un elemento in Modalità Display può essere visibile in base a ulteriori controlli */
139
+ this.DisplayCondition = true;
140
+ /** Layout del Form quando è in modalità Display. Di default mantiene la formattazione a form */
141
+ this.DisplayLayout = 'form';
142
+ /** Separatore da mettere in automatico fra un elemento Form e l'altro quando sono in **DisplayMode** con **DisplayLayout** inline */
143
+ this.InlineSeparator = ' ';
144
+ /**
145
+ * Espressione simil-Angular per cambiare il testo dei componenti che scelgono il proprio modello da una **Source**
146
+ *
147
+ * Il funzionamento è identico al binder di angular, solo con una graffa invece che 2.
148
+ * È inoltre possibile aggiungere pezzi di descrizione in maniera condizionale con la sintassi **:prop?(Prop vale {prop})**
149
+ */
150
+ this.Display = '{description}';
151
+ /** Risultato dell'unione fra le informazioni degli oggetti in **Source** e le impostazioni specificate nell'Input **Display** */
152
+ this.BoundSource = [];
153
+ /** Indica se è il primo Bind della Source */
154
+ this.SourceFirstBind = true;
155
+ /** Evento chiamato alla modifica del valore collegato a questo campo */
156
+ this.inputChange = new EventEmitter();
157
+ /** Evento chiamato al focus del campo di testo/combo/quelcheè, riemitta l'evento originale Javascript */
158
+ this.inputFocus = new EventEmitter();
159
+ /** Evento chiamato al finalize del campo di testo/combo/quelcheè */
160
+ this.inputFinalized = new EventEmitter();
161
+ /** Cache delle condizioni scritte tipo :prop?(Roba con {prop}) */
162
+ this.BindCheckingGroups = [];
163
+ /** Cache delle proprietà scritte tipo --> {prop} */
164
+ this.BindProperties = [];
165
+ this.destroyed$ = new Subject();
166
+ if (ngControl == null) {
167
+ if (!this.handleNullNgControl())
168
+ console.error("ngControl nullo per qualche motivo! Il 90% delle funzionalità di questo input saranno disabilitate");
169
+ return;
170
+ }
171
+ //Se mi arriva l'injectable del contesto posso capire se questo campo deve essere in readonly o no, solo se ho già inizializzato i controller custom dell'access control
172
+ if (this.AppContext && this.AppContext.name && this.ACO_CUSTOMKEY && this.ac && this.ac.InitializedCustom) {
173
+ this.AppContextOwnership = !this.ac.has(this.ACO_CUSTOMKEY + ":" + this.AppContext.name, true);
174
+ this._readonly = !this.AppContextOwnership;
175
+ }
176
+ ngControl.valueAccessor = this;
177
+ //Se ho già un parent impostato vuol dire che sotto al tag <form> ho direttamente un tag <form-*>..
178
+ //.. quindi posso impostare direttamente il riferimento del Form da usare in HTML per il submitted
179
+ //Se non c'è vuol dire che dentro al tag <form> c'è uno splendido <ng-container *ngTemplateOutlet="template; context: ....">..
180
+ //.. il quale a sua volta contiene il controllo <form-*> quindi avrò il form nell'input "Form"
181
+ if (ngControl._parent)
182
+ this.Form = ngControl._parent;
183
+ else
184
+ this.detatchedFromform = true;
185
+ //Questa serve per impostare me stesso come si deve, il validatore invece lo passo al validation-input quando viene settato il validationControl
186
+ if (this.ngControl) {
187
+ //Per tutti i validatori: rigenero la funzione di validazione. Per qualche motivo a volte ho validatori senza...
188
+ //... questo è un workaround per essere sicuro di averla sempre.
189
+ //Solo i validatori con parametri hanno bisogno della _createValidator (es. "maxlength" che ha 1 parametro), quindi no validatori tipo "required"
190
+ if (this._validators) {
191
+ this._validators.forEach(v => {
192
+ if (v._createValidator)
193
+ v._createValidator();
194
+ });
195
+ }
196
+ // Hard kill di validatori che vengono iniettati nel punto sbagliato....
197
+ //(<any>this.ngControl.valueAccessor)._validators = []; TODO: Revert @Ema
198
+ if (this._validators)
199
+ this.ngControl.control.addValidators(this._validators.filter(v => !v.ngOnInit));
200
+ this.ngControl.control.updateValueAndValidity();
201
+ }
202
+ }
203
+ /**
204
+ * Effettua il Bind/Parse delle varie **Source** unendo le informazioni specificate in **Display**
205
+ *
206
+ * Eventualmente rieffettua un bind al modello in base al valore della variabile **RebindModelAfterSource**
207
+ */
208
+ tryBindSourceDisplay() {
209
+ if (!this.Source)
210
+ return;
211
+ if (this.Source.length == 0 && !this.SourceFirstBind)
212
+ this.BoundSource = [];
213
+ this.SourceFirstBind = false;
214
+ // Cache locale per evitare di rifare dei regex.match ogni santa volta
215
+ if (this.Source.length > 0) {
216
+ this.BoundSource = [];
217
+ if (this.BindCheckingGroups.length == 0)
218
+ this.evaluateBindCheckingGroups();
219
+ if (this.BindProperties.length == 0)
220
+ this.evaluateBindProperties();
221
+ // Blocco per generare la descrizione finale di un elemento
222
+ this.Source.forEach(s => {
223
+ // Aggiungo alla BoundSource in formato standard KeyValue
224
+ this.BoundSource.push(this.transformSourceItem(s));
225
+ });
226
+ // Post trybind se c'è un modello associato ricalcolo l'**EvaluatedModel**
227
+ if (this.Model)
228
+ this.EvaluatedModel = this.BoundSource.find(t => t.id == this.Model)?.description;
229
+ }
230
+ }
231
+ /** Valuta il contenuto della variabile BindCheckingGroups */
232
+ evaluateBindCheckingGroups() {
233
+ // Blocco per tirare fuori le condizioni scritte tipo --> :prop?(Roba con {prop})
234
+ var iffedMatches = this.Display.match(/:([a-z]+)\?\(([^\(\)]+)\)/g);
235
+ if (iffedMatches) {
236
+ iffedMatches.forEach(m => {
237
+ // Stessa regex di sopra ma il tag "i" serve per tirare fuori i singoli capturing group e per qualche motivo
238
+ // new RegExp(baseRegex, "g") non funziona quindi non ho potuto razionalizzarlo
239
+ let groups = m.match(/:([a-z]+)\?\(([^\(\)]+)\)/i);
240
+ this.BindCheckingGroups.push({ global: m, prop: groups[1], whenexists: groups[2] });
241
+ });
242
+ }
243
+ }
244
+ /** Valuta il contenuto della variabile BindProperties */
245
+ evaluateBindProperties() {
246
+ // Blocco per tirare fuori le proprietà scritte tipo --> {prop}
247
+ var matches = this.Display.match(/{[a-z]+}/gi);
248
+ if (matches)
249
+ matches.forEach(m => { this.BindProperties.push({ global: m, prop: m.substring(1, m.length - 1) }); });
250
+ }
251
+ /**
252
+ * Trasforma un oggetto della Source alla sua versione "nuova" basandosi sulle informazioni su gruppi e proprietà
253
+ *
254
+ * @param {any} item Oggetto da trasformare
255
+ *
256
+ * @returns {{id: string, description: string}} Oggetto finale trasformato
257
+ */
258
+ transformSourceItem(s) {
259
+ // Parto sempre dalla variabile di Display, poi sostituisco pezzo per pezzo
260
+ let desc = this.Display;
261
+ // Taglio o mantengo le condizioni in base alla proprietà su cui fare check
262
+ // Dopodiché scrivo tutte le proprietà
263
+ for (let i = 0; i < this.BindCheckingGroups.length; i++) {
264
+ let t = this.BindCheckingGroups[i];
265
+ desc = desc.replace(t.global, (s[t.prop] !== null && s[t.prop] !== undefined && s[t.prop] !== "") ? t.whenexists : "");
266
+ }
267
+ for (let i = 0; i < this.BindProperties.length; i++) {
268
+ let t = this.BindProperties[i];
269
+ desc = desc.replace(t.global, s[t.prop]);
270
+ }
271
+ return { id: s[this.IdField], description: desc };
272
+ }
273
+ /**
274
+ * Indica se il comopnente in questione è in grado di gestire ngControl nulli.
275
+ * Di default è **false**
276
+ *
277
+ * @returns {boolean} **true** se gestisco ngControl nulli, **false** altrimenti
278
+ */
279
+ handleNullNgControl() {
280
+ return false;
281
+ }
282
+ /** @ignore */
283
+ ngOnInit() {
284
+ // A volte nell'ngOnInit non ci passa quindi lo metto sia qui sia nell'afterviewinit, penso che valga per i componenti di terze parti
285
+ // che si collegano per i cazzi loro al form ecc, quindi la parte sopra non serve... BOH
286
+ this.checkRequiredness();
287
+ //Il nome non c'è se metto standalone : true (CHE NON é DA METTERE)
288
+ //Dato che anche il controllo esterno viene considerato come componente del form, non posso basarmi solo sul nome ma devo aggiungere un tag. In questo caso "_internal"
289
+ // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA
290
+ if (this.ngControl.name)
291
+ this.GeneratedName = this.ngControl.name.toString() + "_internal";
292
+ // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA
293
+ if (this.Form) {
294
+ if (!this.formHasBennBound && this.validationControl) {
295
+ this.Form.addControl(this.validationControl);
296
+ this.formHasBennBound = true;
297
+ }
298
+ // Quando un form viene submittato la variabile Form.submitted viene assegnata all'input [submitted] del val-control
299
+ // Essendo che però la CD è OnPush per dire ad Angular che l'Input è cambiato e permettere i sideffects di assegnare [submitted]
300
+ // devo attaccarmi al submit attuale del form e prima di eseguirlo devo marcare il componente per il check
301
+ let origFunc = this.Form.onSubmit;
302
+ var formOutsideRef = this.Form;
303
+ this.Form.onSubmit = (ev) => {
304
+ this.cdr.markForCheck();
305
+ return origFunc.apply(formOutsideRef);
306
+ };
307
+ }
308
+ if (this.SetValidationSubject) {
309
+ this.SetValidationSubject.pipe(takeUntil(this.destroyed$)).subscribe(v => {
310
+ if (v.fieldName + "_internal" == this.GeneratedName) {
311
+ let control = this.Form.getControl(this.validationControl);
312
+ // Tentativo di pezza pezzente
313
+ if (control) {
314
+ control.setErrors({ forcedtoinvalid: true });
315
+ control.markAsTouched();
316
+ }
317
+ this.ngControl.control.setErrors({ forcedtoinvalid: true });
318
+ this.ngControl.control.markAsTouched();
319
+ }
320
+ });
321
+ }
322
+ }
323
+ /**
324
+ * Se sono in modalità detatched devo rimuovere manualmente questo controllo quando entro nella dispose, altrimenti se poi ne andassi a riaggiungere
325
+ * un altro con lo stesso "name" ma magari resettato, Angular lo re-idraterebbe con i valori precedenti nonostante il modello fosse vuoto
326
+ */
327
+ ngOnDestroy() {
328
+ if (this.Form && this.validationControl && !!this.Form.getControl(this.validationControl))
329
+ this.Form.removeControl(this.validationControl);
330
+ this.destroyed$.next();
331
+ this.destroyed$.complete();
332
+ }
333
+ /** Elabora i validatori iniettati e capisce se il valore è obbligatorio o meno */
334
+ /** Fratello della funzione con lo stesso nome lato validations */
335
+ checkRequiredness() {
336
+ // controllo se è settato un required per decidere in maniera condizionale se utilizzare il validatore required nel componente interno
337
+ // Lo faccio qui e non in un eventuale ngOnInit o costruttore perché ngOnInit non sempre viene chiamato e da costruttore il validatore non ha ancora il valore giusto
338
+ if (this._validators && this._validators.some(elem => elem instanceof RequiredValidator && (elem.required === true || (!elem.required && elem.required !== false))))
339
+ this.Required = true;
340
+ }
341
+ /** @ignore */
342
+ ngAfterViewInit() {
343
+ if (!this.ngControl)
344
+ return;
345
+ this.checkRequiredness();
346
+ }
347
+ /** Helper che collega la funzione di reset del controllo form al controllo di validazione sottostante */
348
+ bindResetFunction() {
349
+ //faccio in modo che se chiamo il reset del controllo (o della form) dall'esterno si resetti anche il controllo nativo
350
+ const origFunc = this.ngControl.control.__original_reset__ || this.ngControl.control.reset;
351
+ var ngControl_outsideReference = this.ngControl.control;
352
+ var validationControl_outsideReference = this.validationControl.control;
353
+ this.ngControl.control.reset = () => {
354
+ origFunc.apply(ngControl_outsideReference);
355
+ validationControl_outsideReference.reset();
356
+ };
357
+ //Per consentire richiami multipli di questa funzione prendendo sempre la fiunzione di reset base
358
+ this.ngControl.control.__original_reset__ = origFunc;
359
+ }
360
+ /** @ignore */
361
+ writeValue(obj) {
362
+ if (obj == null && this.firstBind) {
363
+ this.firstBind = false;
364
+ return;
365
+ }
366
+ this.Model = obj;
367
+ //Per svegliare Angular. Il valore che entra potrebbe essere valido/non valido e questo lo binda a tutto, form compreso
368
+ if (this.ngControl)
369
+ this.ngControl.control.updateValueAndValidity();
370
+ if (this.Model)
371
+ this.onNotNullValueSet();
372
+ // markForCheck non va bene in quanto arrivano possibilmente 2 update su un solo ciclo di stack
373
+ // setTime*ut che wrappa tutto non va bene sennò spamma delle change detection a suon di interval che rallentano tantissimo l'app
374
+ this.cdr.detectChanges();
375
+ }
376
+ /** @ignore */
377
+ registerOnChange(fn) {
378
+ this.propagateChange = fn;
379
+ }
380
+ /** @ignore */
381
+ registerOnTouched(fn) {
382
+ this.onTouched = fn;
383
+ }
384
+ /**
385
+ * Registra internamente il componente di validazione, ne imposta i validatori e collega il Form, se presente
386
+ *
387
+ * @param {NgModel} comp Elemento HTML marcato con "#validationControl"
388
+ */
389
+ registerValComp(comp) {
390
+ if (comp) {
391
+ // È arrivato un nuovo validationContorl (succede solo nel form-select quando cambia la source da 0 elementi / null ad N elementi, quindi al massimo una volta)
392
+ if (this.validationControl && this.Form)
393
+ this.Form.removeControl(this.validationControl);
394
+ this.validationControl = comp;
395
+ //Questa serve per passare avanti i validatori
396
+ this.validationControl.control.setValidators(this._validators);
397
+ //Se sono staccato dal Form (vuol dire che sono ad esempio dentro un ng-template) aggiungo il controllo al Form passato come input (context del template)
398
+ if (this.detatchedFromform && this.Form && !this.formHasBennBound) {
399
+ this.Form.addControl(this.validationControl);
400
+ this.formHasBennBound = true;
401
+ }
402
+ if (this.ngControl)
403
+ this.bindResetFunction();
404
+ //I val-roba questo lo fanno automaticamente dentro... gli altri no (es. semplice input checkbox)
405
+ this.updateValidityForNativeInput();
406
+ }
407
+ }
408
+ registerForm(ngForm) {
409
+ this.Form = ngForm;
410
+ this.Form.addControl(this.validationControl);
411
+ this.formHasBennBound = true;
412
+ }
413
+ /**
414
+ * Aggiorna il valore e la validità dell'input sottostante. Non viene chiamato per input nativi come le checkbox
415
+ *
416
+ * @param {boolean} forcedValue Forza l'aggiornamento anche se non è un input nativo (input nativo = checkbox, ad esempio)
417
+ */
418
+ updateValidityForNativeInput(forcedValue = false) {
419
+ if (forcedValue || this.nativeInput)
420
+ this.validationControl.control.updateValueAndValidity();
421
+ }
422
+ /**
423
+ * @ignore
424
+ * Classico "spara fuori". Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.
425
+ *
426
+ * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html
427
+ */
428
+ changed(forcedValue = null, markForCheck = false, modelEvaluationHandled = false, internal = false) {
429
+ let toEmit = forcedValue == null || forcedValue == "" ? this.Model ?? "" : forcedValue;
430
+ if (!modelEvaluationHandled)
431
+ this.EvaluatedModel = toEmit;
432
+ this.propagateChange(toEmit);
433
+ this.inputChange.emit(new ChangeEvent(internal, toEmit));
434
+ if (markForCheck)
435
+ this.cdr.markForCheck();
436
+ }
437
+ /**
438
+ * @ignore
439
+ * Classico "spara fuori" sul focus. Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.
440
+ *
441
+ * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html
442
+ */
443
+ focused(event) {
444
+ this.inputFocus.emit(event);
445
+ this.log("Control Focused - " + this.GeneratedName);
446
+ }
447
+ /**
448
+ * @ignore
449
+ * Classico "spara fuori" sulla finalizzazione del valore dell'input (blur / tasto invio, ecc...). Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.
450
+ *
451
+ * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html
452
+ */
453
+ finalized() {
454
+ this.inputFinalized.emit(this.ngControl?.name?.toString());
455
+ this.log("Control Value Finalized - " + this.GeneratedName);
456
+ }
457
+ /**
458
+ * Stampa un log a console
459
+ *
460
+ * @param {string} message Messaggio da stampare
461
+ */
462
+ log(message) {
463
+ if (this.FAV_DEBUG_MODE)
464
+ console.log("@forms-and-validations: " + message);
465
+ }
466
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BaseFormControl, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive }); }
467
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.12", type: BaseFormControl, inputs: { FocusSubject: "FocusSubject", SetValidationSubject: "SetValidationSubject", FieldAppearence: "FieldAppearence", FormLayout: "FormLayout", EmitPendingChanges: "EmitPendingChanges", Validation: "Validation", Frozen: "Frozen", Placeholder: "Placeholder", FormGroupClass: "FormGroupClass", FailedValidationMessage: "FailedValidationMessage", ForcedError: "ForcedError", Label: "Label", LabelColWidth: "LabelColWidth", InputColWidth: "InputColWidth", Last: "Last", Form: "Form", Source: "Source", IdField: "IdField", FloatingLabel: "FloatingLabel", DisplayMode: "DisplayMode", DisplayCondition: "DisplayCondition", DisplayModeTemplate: "DisplayModeTemplate", DisplayLayout: "DisplayLayout", InlineSeparator: "InlineSeparator", Type: "Type", Display: "Display", maxlength: "maxlength", minlength: "minlength", Readonly: "Readonly", LabelInputRatio: "LabelInputRatio" }, outputs: { inputChange: "inputChange", inputFocus: "inputFocus", inputFinalized: "inputFinalized" }, viewQueries: [{ propertyName: "_validationControl", first: true, predicate: ["validationControl"], descendants: true }, { propertyName: "validationControl_static", first: true, predicate: ["validationControl"], descendants: true, static: true }], ngImport: i0 }); }
468
+ }
469
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BaseFormControl, decorators: [{
470
+ type: Directive
471
+ }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.NgControl }, { type: Array }, { type: i2.AccessControlService, decorators: [{
472
+ type: Optional
473
+ }] }, { type: i2.ComponentContext, decorators: [{
474
+ type: Optional
475
+ }] }, { type: undefined, decorators: [{
476
+ type: Optional
477
+ }, {
478
+ type: Inject,
479
+ args: [ACO_CUSTOMKEY]
480
+ }] }, { type: undefined, decorators: [{
481
+ type: Optional
482
+ }, {
483
+ type: Inject,
484
+ args: [FAV_DEBUG_MODE]
485
+ }] }, { type: undefined }], propDecorators: { FocusSubject: [{
486
+ type: Input
487
+ }], SetValidationSubject: [{
488
+ type: Input
489
+ }], FieldAppearence: [{
490
+ type: Input
491
+ }], FormLayout: [{
492
+ type: Input
493
+ }], EmitPendingChanges: [{
494
+ type: Input
495
+ }], Validation: [{
496
+ type: Input
497
+ }], Frozen: [{
498
+ type: Input
499
+ }], Placeholder: [{
500
+ type: Input
501
+ }], FormGroupClass: [{
502
+ type: Input
503
+ }], FailedValidationMessage: [{
504
+ type: Input
505
+ }], ForcedError: [{
506
+ type: Input
507
+ }], Label: [{
508
+ type: Input
509
+ }], LabelColWidth: [{
510
+ type: Input
511
+ }], InputColWidth: [{
512
+ type: Input
513
+ }], Last: [{
514
+ type: Input
515
+ }], Form: [{
516
+ type: Input
517
+ }], Source: [{
518
+ type: Input
519
+ }], IdField: [{
520
+ type: Input
521
+ }], FloatingLabel: [{
522
+ type: Input
523
+ }], DisplayMode: [{
524
+ type: Input
525
+ }], DisplayCondition: [{
526
+ type: Input
527
+ }], DisplayModeTemplate: [{
528
+ type: Input
529
+ }], DisplayLayout: [{
530
+ type: Input
531
+ }], InlineSeparator: [{
532
+ type: Input
533
+ }], Type: [{
534
+ type: Input
535
+ }], Display: [{
536
+ type: Input
537
+ }], maxlength: [{
538
+ type: Input
539
+ }], minlength: [{
540
+ type: Input
541
+ }], Readonly: [{
542
+ type: Input
543
+ }], LabelInputRatio: [{
544
+ type: Input
545
+ }], _validationControl: [{
546
+ type: ViewChild,
547
+ args: ["validationControl", { static: false }]
548
+ }], validationControl_static: [{
549
+ type: ViewChild,
550
+ args: ["validationControl", { static: true }]
551
+ }], inputChange: [{
552
+ type: Output
553
+ }], inputFocus: [{
554
+ type: Output
555
+ }], inputFinalized: [{
556
+ type: Output
557
+ }] } });
558
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-form-control.js","sourceRoot":"","sources":["../../../../../projects/forms-and-validations/src/lib/forms/base-form-control.ts"],"names":[],"mappings":"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","sourcesContent":["// Angular\r\nimport { ControlValueAccessor, NgControl, NgForm, NgModel, RequiredValidator } from \"@angular/forms\";\r\nimport { EventEmitter, Input, Output, ViewChild, Optional, Inject, Directive, ChangeDetectorRef } from \"@angular/core\";\r\n\r\n// // Configurazioni\r\nimport { AccessControlService, ComponentContext } from \"@esfaenza/access-control\";\r\nimport { ACO_CUSTOMKEY, FAV_DEBUG_MODE } from '../tokens';\r\nimport { Subject } from \"rxjs\";\r\nimport { TemplateRef } from \"@angular/core\";\r\nimport { ChangeEvent } from \"../models/ChangeEvent\";\r\nimport { takeUntil } from \"rxjs/operators\";\r\n\r\n/** Componente base da cui tutti i componenti Form implementano */\r\n@Directive()\r\nexport abstract class BaseFormControl implements ControlValueAccessor {\r\n\r\n    /**\r\n     * https://github.com/angular/angular/issues/14988\r\n     * TL:DR; Il primo valore passato al writeValue è nullo perché quando i cagnacci fanno new FormControl()\r\n     * scrivono un writeValue nullo... quindi al primo binding salto l'inizializzazione se ho la coppia null-firstBind\r\n     * poi mettendo firstBind a false di lì in poi funziono come devo. Vista la casistica evito essenzialmente metà detectChanges\r\n     */\r\n    private firstBind: boolean = true;\r\n\r\n    /**\r\n    * Indica di utilizzare i validatori che l'utente piazza sul componente. \r\n    * Da non usare nel caso di form-adaptive dato che i validatori vengono valutati in base al tipo ecc\r\n    */\r\n    protected UseUserValidators:boolean = true;\r\n\r\n    /** Indica se il campo è obbligatorio i opzionale */\r\n    public Required: boolean = false;\r\n\r\n    /** Modello rappresentato da questo campo */\r\n    public Model: any;\r\n\r\n    /** Valore del Modello */\r\n    public EvaluatedModel: string;\r\n\r\n    /** Nome generato per il controllo interno. Di default è il nome del controllo esterno con l'aggiunta di \"_internal\". Serve per evitare cross-bind */\r\n    public GeneratedName: string;\r\n\r\n    /** Indica se il Form che contiene questo campo è già stato registrato */\r\n    protected formHasBennBound: boolean = false;\r\n\r\n    /**\r\n     * Indica se questo campo è staccato dal Form, ad esempio nel caso sia in un ng-template proiettato nel form.\r\n     * \r\n     * In quel caso la variabile **Form** del componente dev'essere valorizzata\r\n     */\r\n    protected detatchedFromform: boolean = false;\r\n\r\n    /** @ignore */\r\n    protected propagateChange = (_: any) => { };\r\n\r\n    /** @ignore */\r\n    protected onTouched = () => { };\r\n\r\n    /** @ignore Vedi getter e setter */\r\n    public _readonly: boolean = false;\r\n\r\n    /**\r\n     * Qualora ci fosse un contesto collegato questa variabile identifica se Sono l'Owner o meno.\r\n     * \r\n     * Se non lo sono, il campo risulterà in sola lettura \r\n     */\r\n    private AppContextOwnership: boolean = true;\r\n\r\n    /** Subject a cui l'oggetto interno si collega per effettuare il focus dell'elemento input sottostante */\r\n    @Input() FocusSubject: Subject<void>;\r\n    /**\r\n    * @EXPERIMENTAL\r\n    */\r\n    @Input() SetValidationSubject: Subject<{ fieldName: string, error: string }>;\r\n\r\n    /** Modalità in cui viene mostrato il campo, 1-1 rispetto alle definizioni di Angular Material  */\r\n    @Input() FieldAppearence: \"legacy\" | \"standard\" | \"fill\" | \"outline\" = \"outline\";\r\n\r\n    /** Utilizza o meno il Layout di un form (Label con input di fianco), se false mostra solo l'input */\r\n    @Input() FormLayout: boolean = true;\r\n\r\n    /** Definisce se richiamare l'EventEmitter **inputChange** in maniera classica (true) o solo per la change considerate definitive come tasto invio / blur / ecc... (false) */\r\n    @Input() EmitPendingChanges: boolean = true;\r\n\r\n    /** Attiva o disattiva la validazione per questo componente */\r\n    @Input() Validation: boolean = true;\r\n\r\n    /** \r\n     * Permette di scrivere o meno nel campo di Input. Si differenzia dal Readonly in quanto fisicamente l'Input è scrivibile e vengono mantenuti i temi della validazione\r\n     * ma interagire direttamente con la casella di Input per l'utente non è possibile\r\n     */\r\n    @Input() Frozen: boolean = false;\r\n\r\n    /** Placeholder per quando l'input è vuoto */\r\n    @Input() Placeholder: string = \"\";\r\n\r\n    /** Classe extra per il form-group in cui viene wrappato l'input */\r\n    @Input() FormGroupClass: string = \"\";\r\n\r\n    /** Messaggio da mostrare quando la validazione fallisce per questo componente. Di default viene proposta dalla libreria */\r\n    @Input() FailedValidationMessage: string = \"\";\r\n\r\n    /** Input che forza l'invalidità del componente */\r\n    @Input() ForcedError: boolean = false;\r\n\r\n    /** Mostra un testo a sinistra dell'input (FormLayout permettendo) */\r\n    @Input() Label: string = \"\";\r\n\r\n    /** col-md-X per la label */\r\n    @Input() LabelColWidth: number = 4;\r\n\r\n    /** col-md-X per l'input */\r\n    @Input() InputColWidth: number = 8;\r\n\r\n    /** Indica se è l'ultimo componente di un form (serve per rimuovere il margine finale) */\r\n    @Input() Last: boolean = false;\r\n\r\n    /** Associa il componente Form a questo input */\r\n    @Input() Form: NgForm = null;\r\n\r\n    /** Sorgente da cui scegliere valori per i componenti Form che lo consentono */\r\n    @Input() Source: any[];\r\n\r\n    /** Nome della proprietà che contiene l'Id degli oggetti bindati nella **Source** */\r\n    @Input() IdField: string = 'id';\r\n\r\n    /**\r\n     * Indica se il **Placeholder** dev'essere usato come una floating label di material o solo come un placeholder standard \r\n     * \r\n     * Ignorata lato autocomplete, autocomplete-multi e select in quanto hanno gestioni particolari di placeholder ecc...\r\n     */\r\n    @Input() FloatingLabel: boolean = true;\r\n\r\n    /** Indica che la parte input del controllo dev'essere una label readonly e la parte label di titolo dev'essere in bold. Per utilizzare un solo componente sia per la modifica che per il dettaglio */\r\n    @Input() DisplayMode: boolean = false;\r\n\r\n    /** Indica la condizione per cui un elemento in Modalità Display può essere visibile in base a ulteriori controlli */\r\n    @Input() DisplayCondition: boolean = true;\r\n\r\n    /** Template per visualizzare la modalità **DisplayMode** come vuole il programmatore frocio di turno */\r\n    @Input() DisplayModeTemplate: TemplateRef<any>;\r\n\r\n    /** Layout del Form quando è in modalità Display. Di default mantiene la formattazione a form */\r\n    @Input() DisplayLayout: 'form' | 'inline' | 'hidden' = 'form';\r\n\r\n    /** Separatore da mettere in automatico fra un elemento Form e l'altro quando sono in **DisplayMode** con **DisplayLayout** inline */\r\n    @Input() InlineSeparator: string = ' ';\r\n    \r\n    /** Type HTML (password/text/etc...) */\r\n    @Input() Type: string;\r\n\r\n    /**\r\n     * Espressione simil-Angular per cambiare il testo dei componenti che scelgono il proprio modello da una **Source**\r\n     * \r\n     * Il funzionamento è identico al binder di angular, solo con una graffa invece che 2.\r\n     * È inoltre possibile aggiungere pezzi di descrizione in maniera condizionale con la sintassi **:prop?(Prop vale {prop})**\r\n     */\r\n    @Input() Display: string = '{description}';\r\n\r\n    /** Risultato dell'unione fra le informazioni degli oggetti in **Source** e le impostazioni specificate nell'Input **Display** */\r\n    public BoundSource: { id: string, description: string }[] = [];\r\n\r\n    /** Indica se è il primo Bind della Source */\r\n    protected SourceFirstBind: boolean = true;\r\n\r\n    /**\r\n     * Max length\r\n     */\r\n    @Input() maxlength: number;\r\n\r\n    /**\r\n     * Max length\r\n     */\r\n    @Input() minlength: number;\r\n\r\n    /**\r\n     * Ottiene il valore della variabile **_readonly**\r\n     * \r\n     * @returns {boolean} **true** se sola lettura, **false** altrimenti\r\n     */\r\n    get Readonly(): boolean { \r\n        return this._readonly;\r\n    }\r\n\r\n    /**\r\n     * Imposta il valore della variabile **_readonly** gestendo anche il Contesto applicativo\r\n     * \r\n     * @param {boolean} value Valore da impostare\r\n     */\r\n    @Input() set Readonly(value: boolean) {\r\n        if (this.AppContextOwnership) \r\n            this._readonly = value;\r\n    }\r\n\r\n    /**\r\n     * Permette di settare in una volta solo gli input LabelColWidth e InputColWidth, basta scriverlo con la sintassi \"X Y\"\r\n     * \r\n     * @param {string} input Input nel formato \"X Y\"\r\n     */\r\n    @Input() set LabelInputRatio(input: string) {\r\n        var ratio = input.split(/\\s/);\r\n        this.LabelColWidth = parseInt(ratio[0]);\r\n        this.InputColWidth = parseInt(ratio[1]);\r\n    }\r\n\r\n    /** val-qualcosa che fa effettivamente da input, o eventualmente il singolo input in casi particolari (es checkbox) */\r\n    protected validationControl: NgModel;\r\n\r\n    /**\r\n     * Query sull'elemento 'validationControl' che funziona in ambiente statico\r\n     * \r\n     * @param {NgModel} comp Elemento HTML marcato con \"#validationControl\"\r\n     */\r\n    @ViewChild(\"validationControl\", { static: false }) set _validationControl(comp: NgModel) { this.registerValComp(comp); }\r\n\r\n    /**\r\n     * Query sull'elemento 'validationControl' che funziona in ambiente dinamico\r\n     * \r\n     * @param {NgModel} comp Elemento HTML marcato con \"#validationControl\"\r\n     */\r\n    @ViewChild(\"validationControl\", { static: true }) set validationControl_static(comp: NgModel) { this.registerValComp(comp); };\r\n\r\n    /** Evento chiamato alla modifica del valore collegato a questo campo */\r\n    @Output() inputChange: EventEmitter<ChangeEvent> = new EventEmitter<ChangeEvent>();\r\n\r\n    /** Evento chiamato al focus del campo di testo/combo/quelcheè, riemitta l'evento originale Javascript */\r\n    @Output() inputFocus: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n    /** Evento chiamato al finalize del campo di testo/combo/quelcheè */\r\n    @Output() inputFinalized: EventEmitter<string> = new EventEmitter<string>();\r\n\r\n    /** Cache delle condizioni scritte tipo :prop?(Roba con {prop}) */\r\n    private BindCheckingGroups: { global: string, prop: string, whenexists: string }[] = []\r\n\r\n    /** Cache delle proprietà scritte tipo --> {prop} */\r\n    private BindProperties: { global: string, prop: string }[] = [];\r\n\r\n    /** @ignore Costruttore  */\r\n    constructor(protected cdr: ChangeDetectorRef, public ngControl: NgControl, protected _validators: Array<any>, @Optional() private ac: AccessControlService, @Optional() public AppContext: ComponentContext, @Optional() @Inject(ACO_CUSTOMKEY) private ACO_CUSTOMKEY: string, @Optional() @Inject(FAV_DEBUG_MODE) private FAV_DEBUG_MODE: boolean, private nativeInput: boolean = false) {\r\n\r\n        if (ngControl == null) {\r\n            if (!this.handleNullNgControl())\r\n                console.error(\"ngControl nullo per qualche motivo! Il 90% delle funzionalità di questo input saranno disabilitate\");\r\n\r\n            return;\r\n        }\r\n\r\n        //Se mi arriva l'injectable del contesto posso capire se questo campo deve essere in readonly o no, solo se ho già inizializzato i controller custom dell'access control\r\n        if (this.AppContext && this.AppContext.name && this.ACO_CUSTOMKEY && this.ac && this.ac.InitializedCustom) {\r\n            this.AppContextOwnership = !this.ac.has(this.ACO_CUSTOMKEY + \":\" + this.AppContext.name, true);\r\n            this._readonly = !this.AppContextOwnership\r\n        }\r\n\r\n        ngControl.valueAccessor = this;\r\n\r\n        //Se ho già un parent impostato vuol dire che sotto al tag <form> ho direttamente un tag <form-*>..\r\n        //.. quindi posso impostare direttamente il riferimento del Form da usare in HTML per il submitted\r\n\r\n        //Se non c'è vuol dire che dentro al tag <form> c'è uno splendido <ng-container *ngTemplateOutlet=\"template; context: ....\">..\r\n        //.. il quale a sua volta contiene il controllo <form-*> quindi avrò il form nell'input \"Form\"\r\n        if ((<any>ngControl)._parent)\r\n            this.Form = <NgForm>(<any>ngControl)._parent;\r\n        else\r\n            this.detatchedFromform = true;\r\n\r\n        //Questa serve per impostare me stesso come si deve, il validatore invece lo passo al validation-input quando viene settato il validationControl\r\n        if (this.ngControl) {\r\n            //Per tutti i validatori: rigenero la funzione di validazione. Per qualche motivo a volte ho validatori senza...\r\n            //... questo è un workaround per essere sicuro di averla sempre.\r\n            //Solo i validatori con parametri hanno bisogno della _createValidator (es. \"maxlength\" che ha 1 parametro), quindi no validatori tipo \"required\"\r\n            if (this._validators) {\r\n                this._validators.forEach(v => {\r\n                    if (v._createValidator)\r\n                        v._createValidator();\r\n                });\r\n            }\r\n\r\n            // Hard kill di validatori che vengono iniettati nel punto sbagliato....\r\n            //(<any>this.ngControl.valueAccessor)._validators = []; TODO: Revert @Ema\r\n\r\n            if (this._validators)\r\n                this.ngControl.control.addValidators(this._validators.filter(v => !v.ngOnInit));\r\n\r\n            this.ngControl.control.updateValueAndValidity();\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Effettua il Bind/Parse delle varie **Source** unendo le informazioni specificate in **Display**\r\n     * \r\n     * Eventualmente rieffettua un bind al modello in base al valore della variabile **RebindModelAfterSource**\r\n     */\r\n    protected tryBindSourceDisplay(): void {\r\n        if (!this.Source)\r\n            return;\r\n\r\n        if (this.Source.length == 0 && !this.SourceFirstBind)\r\n            this.BoundSource = [];\r\n\r\n        this.SourceFirstBind = false;\r\n\r\n        // Cache locale per evitare di rifare dei regex.match ogni santa volta\r\n        if (this.Source.length > 0) {\r\n            this.BoundSource = [];\r\n\r\n            if (this.BindCheckingGroups.length == 0)\r\n                this.evaluateBindCheckingGroups();\r\n\r\n            if (this.BindProperties.length == 0)\r\n                this.evaluateBindProperties();\r\n\r\n            // Blocco per generare la descrizione finale di un elemento\r\n            this.Source.forEach(s => {\r\n                // Aggiungo alla BoundSource in formato standard KeyValue\r\n                this.BoundSource.push(this.transformSourceItem(s));\r\n            });\r\n\r\n            // Post trybind se c'è un modello associato ricalcolo l'**EvaluatedModel**\r\n            if (this.Model)\r\n                this.EvaluatedModel = this.BoundSource.find(t => t.id == this.Model)?.description;\r\n        }\r\n    }\r\n\r\n    /** Valuta il contenuto della variabile BindCheckingGroups */\r\n    private evaluateBindCheckingGroups() {\r\n        // Blocco per tirare fuori le condizioni scritte tipo --> :prop?(Roba con {prop})\r\n        var iffedMatches = this.Display.match(/:([a-z]+)\\?\\(([^\\(\\)]+)\\)/g);\r\n        if (iffedMatches) {\r\n            iffedMatches.forEach(m => {\r\n                // Stessa regex di sopra ma il tag \"i\" serve per tirare fuori i singoli capturing group e per qualche motivo \r\n                // new RegExp(baseRegex, \"g\") non funziona quindi non ho potuto razionalizzarlo\r\n                let groups = m.match(/:([a-z]+)\\?\\(([^\\(\\)]+)\\)/i);\r\n                this.BindCheckingGroups.push({ global: m, prop: groups[1], whenexists: groups[2] });\r\n            });\r\n        }\r\n    }\r\n\r\n    /** Valuta il contenuto della variabile BindProperties */\r\n    private evaluateBindProperties() {\r\n        // Blocco per tirare fuori le proprietà scritte tipo --> {prop}\r\n        var matches = this.Display.match(/{[a-z]+}/gi);\r\n        if (matches)\r\n            matches.forEach(m => { this.BindProperties.push({ global: m, prop: m.substring(1, m.length - 1) }); });\r\n    }\r\n\r\n    /**\r\n     * Trasforma un oggetto della Source alla sua versione \"nuova\" basandosi sulle informazioni su gruppi e proprietà\r\n     * \r\n     * @param {any} item Oggetto da trasformare\r\n     * \r\n     * @returns {{id: string, description: string}} Oggetto finale trasformato\r\n     */\r\n    private transformSourceItem(s: any): {id: string, description: string} {\r\n        // Parto sempre dalla variabile di Display, poi sostituisco pezzo per pezzo\r\n        let desc = this.Display;\r\n\r\n        // Taglio o mantengo le condizioni in base alla proprietà su cui fare check\r\n        // Dopodiché scrivo tutte le proprietà\r\n        for (let i = 0; i < this.BindCheckingGroups.length; i++) {\r\n            let t = this.BindCheckingGroups[i];\r\n            desc = desc.replace(t.global, (s[t.prop] !== null && s[t.prop] !== undefined && s[t.prop] !== \"\") ? t.whenexists : \"\");\r\n        }\r\n\r\n        for (let i = 0; i < this.BindProperties.length; i++) {\r\n            let t = this.BindProperties[i];\r\n            desc = desc.replace(t.global, s[t.prop]);\r\n        }\r\n\r\n        return { id: s[this.IdField], description: desc };\r\n    }\r\n\r\n    /**\r\n     * Indica se il comopnente in questione è in grado di gestire ngControl nulli. \r\n     * Di default è **false**\r\n     * \r\n     * @returns {boolean} **true** se gestisco ngControl nulli, **false** altrimenti\r\n     */\r\n    handleNullNgControl(): boolean {\r\n        return false;\r\n    }\r\n\r\n    /** @ignore */\r\n    ngOnInit() {\r\n        // A volte nell'ngOnInit non ci passa quindi lo metto sia qui sia nell'afterviewinit, penso che valga per i componenti di terze parti\r\n        // che si collegano per i cazzi loro al form ecc, quindi la parte sopra non serve... BOH\r\n        this.checkRequiredness();\r\n\r\n        //Il nome non c'è se metto standalone : true (CHE NON é DA METTERE)\r\n        //Dato che anche il controllo esterno viene considerato come componente del form, non posso basarmi solo sul nome ma devo aggiungere un tag. In questo caso \"_internal\"\r\n        // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA\r\n        if (this.ngControl.name)\r\n            this.GeneratedName = this.ngControl.name.toString() + \"_internal\";\r\n\r\n        // ATTENZIONE: QUALSIASI SIA IL MOTIVO, SE STO COSO VIENE SPOSTATO DALL ngOnInit NON FUNZIONA PIU NULLA\r\n        if (this.Form) {\r\n            if (!this.formHasBennBound && this.validationControl) {\r\n                this.Form.addControl(this.validationControl);\r\n                this.formHasBennBound = true;\r\n            }\r\n\r\n            // Quando un form viene submittato la variabile Form.submitted viene assegnata all'input [submitted] del val-control\r\n            // Essendo che però la CD è OnPush per dire ad Angular che l'Input è cambiato e permettere i sideffects di assegnare [submitted]\r\n            // devo attaccarmi al submit attuale del form e prima di eseguirlo devo marcare il componente per il check\r\n            let origFunc = this.Form.onSubmit;\r\n            var formOutsideRef = this.Form;\r\n            this.Form.onSubmit = (ev): boolean => {\r\n                this.cdr.markForCheck();\r\n                return origFunc.apply(formOutsideRef);\r\n            };\r\n        }\r\n\r\n        if (this.SetValidationSubject) {\r\n            this.SetValidationSubject.pipe(takeUntil(this.destroyed$)).subscribe(v => {\r\n                if (v.fieldName + \"_internal\" == this.GeneratedName) {\r\n                    let control = this.Form.getControl(this.validationControl);\r\n                    \r\n                    // Tentativo di pezza pezzente\r\n                    if (control) {\r\n                        control.setErrors({ forcedtoinvalid: true });\r\n                        control.markAsTouched();\r\n                    }\r\n\r\n                    this.ngControl.control.setErrors({ forcedtoinvalid: true });\r\n                    this.ngControl.control.markAsTouched();\r\n                }\r\n            })\r\n        }\r\n    }\r\n\r\n    destroyed$: Subject<void> = new Subject<void>();\r\n\r\n    /**\r\n     * Se sono in modalità detatched devo rimuovere manualmente questo controllo quando entro nella dispose, altrimenti se poi ne andassi a riaggiungere\r\n     * un altro con lo stesso \"name\" ma magari resettato, Angular lo re-idraterebbe con i valori precedenti nonostante il modello fosse vuoto\r\n     */\r\n    ngOnDestroy() {\r\n        if (this.Form && this.validationControl && !!this.Form.getControl(this.validationControl))\r\n            this.Form.removeControl(this.validationControl);\r\n\r\n\r\n        this.destroyed$.next();\r\n        this.destroyed$.complete();\r\n    }\r\n\r\n    /** Elabora i validatori iniettati e capisce se il valore è obbligatorio o meno */\r\n    /** Fratello della funzione con lo stesso nome lato validations */\r\n    checkRequiredness() {\r\n        // controllo se è settato un required per decidere in maniera condizionale se utilizzare il validatore required nel componente interno\r\n        // Lo faccio qui e non in un eventuale ngOnInit o costruttore perché ngOnInit non sempre viene chiamato e da costruttore il validatore non ha ancora il valore giusto\r\n        if (this._validators && this._validators.some(elem => elem instanceof RequiredValidator && (elem.required === true || (!elem.required && elem.required !== false))))\r\n            this.Required = true;\r\n    }\r\n\r\n    /** @ignore */\r\n    ngAfterViewInit() {\r\n        if (!this.ngControl)\r\n            return;\r\n\r\n        this.checkRequiredness();\r\n    }\r\n\r\n    /** Helper che collega la funzione di reset del controllo form al controllo di validazione sottostante */\r\n    private bindResetFunction() {\r\n        //faccio in modo che se chiamo il reset del controllo (o della form) dall'esterno si resetti anche il controllo nativo\r\n        const origFunc = (<any>this.ngControl.control).__original_reset__ || this.ngControl.control.reset;\r\n        var ngControl_outsideReference = this.ngControl.control;\r\n        var validationControl_outsideReference = this.validationControl.control;\r\n\r\n        this.ngControl.control.reset = () => {\r\n            origFunc.apply(ngControl_outsideReference);\r\n            validationControl_outsideReference.reset();\r\n        };\r\n\r\n        //Per consentire richiami multipli di questa funzione prendendo sempre la fiunzione di reset base\r\n        (<any>this.ngControl.control).__original_reset__ = origFunc\r\n    }\r\n\r\n    /** @ignore */\r\n    writeValue(obj: any): void {\r\n        if (obj == null && this.firstBind) {\r\n            this.firstBind = false;\r\n            return;\r\n        }\r\n\r\n        this.Model = obj;\r\n\r\n        //Per svegliare Angular. Il valore che entra potrebbe essere valido/non valido e questo lo binda a tutto, form compreso\r\n        if (this.ngControl)\r\n            this.ngControl.control.updateValueAndValidity();\r\n\r\n        if (this.Model)\r\n            this.onNotNullValueSet();\r\n\r\n        // markForCheck non va bene in quanto arrivano possibilmente 2 update su un solo ciclo di stack\r\n        // setTime*ut che wrappa tutto non va bene sennò spamma delle change detection a suon di interval che rallentano tantissimo l'app\r\n        this.cdr.detectChanges();\r\n    }\r\n\r\n    /** @ignore */\r\n    registerOnChange(fn: any): void {\r\n        this.propagateChange = fn;\r\n    }\r\n\r\n    /** @ignore */\r\n    registerOnTouched(fn: any): void {\r\n        this.onTouched = fn;\r\n    }\r\n\r\n    /**\r\n     * Registra internamente il componente di validazione, ne imposta i validatori e collega il Form, se presente\r\n     * \r\n     * @param {NgModel} comp Elemento HTML marcato con \"#validationControl\"\r\n     */\r\n    registerValComp(comp: NgModel) {\r\n        if (comp) {\r\n            // È arrivato un nuovo validationContorl (succede solo nel form-select quando cambia la source da 0 elementi / null ad N elementi, quindi al massimo una volta) \r\n            if (this.validationControl && this.Form)\r\n                this.Form.removeControl(this.validationControl);\r\n\r\n            this.validationControl = comp;\r\n\r\n            //Questa serve per passare avanti i validatori\r\n            this.validationControl.control.setValidators(this._validators);\r\n\r\n            //Se sono staccato dal Form (vuol dire che sono ad esempio dentro un ng-template) aggiungo il controllo al Form passato come input (context del template)\r\n            if (this.detatchedFromform && this.Form && !this.formHasBennBound) {\r\n                this.Form.addControl(this.validationControl)\r\n                this.formHasBennBound = true;\r\n            }\r\n\r\n            if (this.ngControl)\r\n                this.bindResetFunction();\r\n\r\n            //I val-roba questo lo fanno automaticamente dentro... gli altri no (es. semplice input checkbox)\r\n            this.updateValidityForNativeInput();\r\n        }\r\n    }\r\n\r\n    public registerForm(ngForm: NgForm) {\r\n        this.Form = ngForm;\r\n        this.Form.addControl(this.validationControl)\r\n        this.formHasBennBound = true;\r\n    }\r\n\r\n    /**\r\n     * Aggiorna il valore e la validità dell'input sottostante. Non viene chiamato per input nativi come le checkbox\r\n     * \r\n     * @param {boolean} forcedValue Forza l'aggiornamento anche se non è un input nativo (input nativo = checkbox, ad esempio)\r\n     */\r\n    updateValidityForNativeInput(forcedValue: boolean = false) {\r\n        if (forcedValue || this.nativeInput)\r\n            this.validationControl.control.updateValueAndValidity();\r\n    }\r\n\r\n    /**\r\n     * @ignore\r\n     * Classico \"spara fuori\". Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.\r\n     * \r\n     * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html\r\n     */\r\n    changed(forcedValue: any = null, markForCheck: boolean = false, modelEvaluationHandled: boolean = false, internal: boolean = false) {\r\n        let toEmit = forcedValue == null || forcedValue == \"\" ? this.Model ?? \"\" : forcedValue;\r\n\r\n        if(!modelEvaluationHandled)\r\n            this.EvaluatedModel = toEmit;\r\n        \r\n        this.propagateChange(toEmit);\r\n        this.inputChange.emit(new ChangeEvent(internal, toEmit));\r\n        if (markForCheck)\r\n            this.cdr.markForCheck();\r\n    }\r\n\r\n    /**\r\n     * @ignore\r\n     * Classico \"spara fuori\" sul focus. Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.\r\n     * \r\n     * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html\r\n     */\r\n    focused(event: any) {\r\n        this.inputFocus.emit(event);\r\n        this.log(\"Control Focused - \" + this.GeneratedName);\r\n    }\r\n\r\n    /**\r\n     * @ignore\r\n     * Classico \"spara fuori\" sulla finalizzazione del valore dell'input (blur / tasto invio, ecc...). Mantengo lo stesso nome dell'evento dei <val-*> (inputChange) per retrocompatibilità.\r\n     * \r\n     * Non posso tenerlo protected altrimenti posso eseguirlo solo dal .ts e non dall' .html\r\n     */\r\n    finalized() {\r\n        this.inputFinalized.emit(this.ngControl?.name?.toString());\r\n        this.log(\"Control Value Finalized - \" + this.GeneratedName);\r\n    }\r\n\r\n    /**\r\n     * Stampa un log a console\r\n     * \r\n     * @param {string} message Messaggio da stampare\r\n     */\r\n    public log(message: string) {\r\n        if (this.FAV_DEBUG_MODE)\r\n            console.log(\"@forms-and-validations: \" + message)\r\n    }\r\n\r\n    /** @ignore */\r\n    abstract onNotNullValueSet(): void;\r\n}"]}