@decaf-ts/for-angular 0.0.4 → 0.0.5

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 (85) hide show
  1. package/LICENSE.md +646 -144
  2. package/README.md +37 -242
  3. package/dist/lib/README.md +92 -0
  4. package/dist/lib/assets/i18n/en.json +131 -0
  5. package/dist/lib/assets/images/angular-logo.svg +45 -0
  6. package/dist/lib/assets/images/decaf-logo-black.svg +22 -0
  7. package/dist/lib/assets/images/decaf-logo-lw.svg +50 -0
  8. package/dist/lib/assets/images/decaf-logo-white.svg +22 -0
  9. package/dist/lib/assets/images/decaf-logo.svg +54 -0
  10. package/dist/lib/components/component-renderer/component-renderer.component.d.ts +267 -0
  11. package/dist/lib/components/crud-field/crud-field.component.d.ts +447 -0
  12. package/dist/{for-angular/components/decaf-crud-form → lib/components/crud-form}/constants.d.ts +0 -0
  13. package/dist/lib/components/crud-form/crud-form.component.d.ts +102 -0
  14. package/dist/{for-angular/components/decaf-crud-form → lib/components/crud-form}/types.d.ts +0 -0
  15. package/dist/lib/components/model-renderer/model-renderer.component.d.ts +97 -0
  16. package/dist/lib/engine/DynamicModule.d.ts +17 -0
  17. package/dist/{for-angular → lib}/engine/NgxCrudFormField.d.ts +22 -14
  18. package/dist/lib/engine/NgxFormService.d.ts +167 -0
  19. package/dist/lib/engine/NgxRenderingEngine.d.ts +128 -0
  20. package/dist/lib/engine/NgxRenderingEngine2.d.ts +251 -0
  21. package/dist/lib/engine/ValidatorFactory.d.ts +15 -0
  22. package/dist/lib/engine/constants.d.ts +151 -0
  23. package/dist/lib/engine/decorators.d.ts +25 -0
  24. package/dist/lib/engine/index.d.ts +15 -0
  25. package/dist/lib/engine/types.d.ts +293 -0
  26. package/dist/lib/esm2022/components/component-renderer/component-renderer.component.mjs +309 -0
  27. package/dist/lib/esm2022/components/crud-field/crud-field.component.mjs +288 -0
  28. package/dist/lib/esm2022/components/crud-form/constants.mjs +14 -0
  29. package/dist/lib/esm2022/components/crud-form/crud-form.component.mjs +140 -0
  30. package/dist/lib/esm2022/components/crud-form/types.mjs +2 -0
  31. package/dist/lib/esm2022/components/model-renderer/model-renderer.component.mjs +137 -0
  32. package/dist/{for-angular → lib}/esm2022/decaf-ts-for-angular.mjs +0 -0
  33. package/dist/lib/esm2022/engine/DynamicModule.mjs +18 -0
  34. package/dist/lib/esm2022/engine/NgxCrudFormField.mjs +117 -0
  35. package/dist/lib/esm2022/engine/NgxFormService.mjs +315 -0
  36. package/dist/lib/esm2022/engine/NgxRenderingEngine.mjs +194 -0
  37. package/dist/lib/esm2022/engine/NgxRenderingEngine2.mjs +333 -0
  38. package/dist/lib/esm2022/engine/ValidatorFactory.mjs +102 -0
  39. package/dist/lib/esm2022/engine/constants.mjs +160 -0
  40. package/dist/lib/esm2022/engine/decorators.mjs +38 -0
  41. package/dist/lib/esm2022/engine/index.mjs +16 -0
  42. package/dist/lib/esm2022/engine/types.mjs +2 -0
  43. package/dist/lib/esm2022/for-angular.module.mjs +118 -0
  44. package/dist/lib/esm2022/interfaces.mjs +2 -0
  45. package/dist/lib/esm2022/public-apis.mjs +13 -0
  46. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs +2138 -0
  47. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs.map +1 -0
  48. package/dist/lib/for-angular.module.d.ts +44 -0
  49. package/dist/{for-angular → lib}/index.d.ts +0 -0
  50. package/dist/lib/interfaces.d.ts +28 -0
  51. package/dist/lib/public-apis.d.ts +12 -0
  52. package/package.json +73 -26
  53. package/dist/for-angular/README.md +0 -297
  54. package/dist/for-angular/assets/i18n/en.json +0 -21
  55. package/dist/for-angular/components/decaf-crud-field/decaf-crud-field.component.d.ts +0 -49
  56. package/dist/for-angular/components/decaf-crud-form/decaf-crud-form.component.d.ts +0 -28
  57. package/dist/for-angular/components/decaf-model-renderer/decaf-model-renderer.component.d.ts +0 -20
  58. package/dist/for-angular/engine/DynamicModule.d.ts +0 -2
  59. package/dist/for-angular/engine/NgxFormService.d.ts +0 -119
  60. package/dist/for-angular/engine/NgxRenderingEngine.d.ts +0 -18
  61. package/dist/for-angular/engine/ValidatorFactory.d.ts +0 -4
  62. package/dist/for-angular/engine/constants.d.ts +0 -13
  63. package/dist/for-angular/engine/decorators.d.ts +0 -1
  64. package/dist/for-angular/engine/index.d.ts +0 -5
  65. package/dist/for-angular/engine/types.d.ts +0 -44
  66. package/dist/for-angular/esm2022/components/decaf-crud-field/decaf-crud-field.component.mjs +0 -129
  67. package/dist/for-angular/esm2022/components/decaf-crud-form/constants.mjs +0 -14
  68. package/dist/for-angular/esm2022/components/decaf-crud-form/decaf-crud-form.component.mjs +0 -80
  69. package/dist/for-angular/esm2022/components/decaf-crud-form/types.mjs +0 -2
  70. package/dist/for-angular/esm2022/components/decaf-model-renderer/decaf-model-renderer.component.mjs +0 -47
  71. package/dist/for-angular/esm2022/engine/DynamicModule.mjs +0 -3
  72. package/dist/for-angular/esm2022/engine/NgxCrudFormField.mjs +0 -115
  73. package/dist/for-angular/esm2022/engine/NgxFormService.mjs +0 -235
  74. package/dist/for-angular/esm2022/engine/NgxRenderingEngine.mjs +0 -84
  75. package/dist/for-angular/esm2022/engine/ValidatorFactory.mjs +0 -48
  76. package/dist/for-angular/esm2022/engine/constants.mjs +0 -15
  77. package/dist/for-angular/esm2022/engine/decorators.mjs +0 -14
  78. package/dist/for-angular/esm2022/engine/index.mjs +0 -6
  79. package/dist/for-angular/esm2022/engine/types.mjs +0 -2
  80. package/dist/for-angular/esm2022/interfaces.mjs +0 -2
  81. package/dist/for-angular/esm2022/public-apis.mjs +0 -6
  82. package/dist/for-angular/fesm2022/decaf-ts-for-angular.mjs +0 -759
  83. package/dist/for-angular/fesm2022/decaf-ts-for-angular.mjs.map +0 -1
  84. package/dist/for-angular/interfaces.d.ts +0 -8
  85. package/dist/for-angular/public-apis.d.ts +0 -5
@@ -0,0 +1,288 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import { Component, CUSTOM_ELEMENTS_SCHEMA, ElementRef, Input, ViewChild, } from '@angular/core';
3
+ import { FormControl } from '@angular/forms';
4
+ import { OperationKeys } from '@decaf-ts/db-decorators';
5
+ import { NgxCrudFormField } from '../../engine/NgxCrudFormField';
6
+ import { Dynamic } from '../../engine/decorators';
7
+ import { ForAngularModule } from 'src/lib/for-angular.module';
8
+ import { IonCheckbox, IonInput, IonItem, IonLabel, IonRadio, IonRadioGroup, IonSelect, IonSelectOption, IonText, IonTextarea, } from '@ionic/angular/standalone';
9
+ import { HTML5InputTypes } from '@decaf-ts/ui-decorators';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "@ionic/angular/standalone";
12
+ import * as i2 from "@angular/forms";
13
+ import * as i3 from "@ngx-translate/core";
14
+ /**
15
+ * @description A dynamic form field component for CRUD operations.
16
+ * @summary The CrudFieldComponent is a versatile form field component that adapts to different
17
+ * input types and CRUD operations. It extends NgxCrudFormField to inherit form handling capabilities
18
+ * and implements lifecycle hooks to properly initialize, render, and clean up. This component
19
+ * supports various input types (text, number, date, select, etc.), validation rules, and styling
20
+ * options, making it suitable for building dynamic forms for create, read, update, and delete
21
+ * operations.
22
+ *
23
+ * @param {CrudOperations} operation - The CRUD operation being performed (create, read, update, delete)
24
+ * @param {string} name - The field name, used as form control identifier
25
+ * @param {PossibleInputTypes} type - The input type (text, number, date, select, etc.)
26
+ * @param {string|number|Date} value - The initial value of the field
27
+ * @param {boolean} disabled - Whether the field is disabled
28
+ * @param {string} label - The display label for the field
29
+ * @param {string} placeholder - Placeholder text when field is empty
30
+ * @param {string} format - Format pattern for the field value
31
+ * @param {boolean} hidden - Whether the field should be hidden
32
+ * @param {number|Date} max - Maximum allowed value
33
+ * @param {number} maxlength - Maximum allowed length
34
+ * @param {number|Date} min - Minimum allowed value
35
+ * @param {number} minlength - Minimum allowed length
36
+ * @param {string} pattern - Validation pattern
37
+ * @param {boolean} readonly - Whether the field is read-only
38
+ * @param {boolean} required - Whether the field is required
39
+ * @param {number} step - Step value for number inputs
40
+ * @param {FormGroup} formGroup - The parent form group
41
+ * @param {StringOrBoolean} translatable - Whether field labels should be translated
42
+ *
43
+ * @component CrudFieldComponent
44
+ * @example
45
+ * <ngx-decaf-crud-field
46
+ * operation="create"
47
+ * name="firstName"
48
+ * type="text"
49
+ * label="<NAME>"
50
+ * placeholder="<NAME>"
51
+ * [value]="model.firstName"
52
+ * [disabled]="model.readOnly">
53
+ *
54
+ *
55
+ * @memberOf module:for-angular
56
+ */
57
+ let CrudFieldComponent = class CrudFieldComponent extends NgxCrudFormField {
58
+ constructor(elementRef) {
59
+ super(elementRef);
60
+ this.elementRef = elementRef;
61
+ /**
62
+ * @description The parent field path, if this field is nested.
63
+ * @summary Specifies the full dot-delimited path of the parent field. This is only set when the field is nested.
64
+ *
65
+ * @type {string}
66
+ * @memberOf CrudFieldComponent
67
+ */
68
+ this.childOf = '';
69
+ /**
70
+ * @description The initial value of the field.
71
+ * @summary Sets the initial value of the form field. This can be a string, number, or Date
72
+ * depending on the field type. For select fields, this should match one of the option values.
73
+ *
74
+ * @type {string | number | Date}
75
+ * @default ''
76
+ * @memberOf CrudFieldComponent
77
+ */
78
+ this.value = '';
79
+ /**
80
+ * @description Spellcheck attribute for text inputs.
81
+ * @summary Enables or disables spellchecking for text inputs.
82
+ * When true, the browser will check the spelling of the input text.
83
+ *
84
+ * @type {boolean}
85
+ * @default false
86
+ * @memberOf CrudFieldComponent
87
+ */
88
+ this.spellcheck = false;
89
+ /**
90
+ * @description Input mode for text inputs.
91
+ * @summary Hints at the type of data that might be entered by the user while editing the element.
92
+ * This can affect the virtual keyboard layout on mobile devices.
93
+ *
94
+ * @type {'none' | 'text' | 'tel' | 'url' | 'email' | 'numeric' | 'decimal' | 'search'}
95
+ * @default 'none'
96
+ * @memberOf CrudFieldComponent
97
+ */
98
+ this.inputmode = 'none';
99
+ /**
100
+ * @description Autocomplete behavior for the field.
101
+ * @summary Specifies whether and how the browser should automatically complete the input.
102
+ * This can improve user experience by suggesting previously entered values.
103
+ *
104
+ * @type {AutocompleteTypes}
105
+ * @default 'off'
106
+ * @memberOf CrudFieldComponent
107
+ */
108
+ this.autocomplete = 'off';
109
+ /**
110
+ * @description Fill style for the field.
111
+ * @summary Determines the fill style of the field, such as 'outline' or 'solid'.
112
+ * This affects the border and background of the field.
113
+ *
114
+ * @type {'outline' | 'solid'}
115
+ * @default 'outline'
116
+ * @memberOf CrudFieldComponent
117
+ */
118
+ this.fill = 'outline';
119
+ /**
120
+ * @description Placement of the label relative to the field.
121
+ * @summary Specifies where the label should be placed relative to the field.
122
+ * Options include 'start', 'end', 'floating', 'stacked', and 'fixed'.
123
+ *
124
+ * @type {'start' | 'end' | 'floating' | 'stacked' | 'fixed'}
125
+ * @default 'floating'
126
+ * @memberOf CrudFieldComponent
127
+ */
128
+ this.labelPlacement = 'floating';
129
+ /**
130
+ * @description Update mode for the field.
131
+ * @summary Determines when the field value should be updated in the form model.
132
+ * Options include 'change', 'blur', and 'submit'.
133
+ *
134
+ * @type {FieldUpdateMode}
135
+ * @default 'change'
136
+ * @memberOf CrudFieldComponent
137
+ */
138
+ this.updateOn = 'change';
139
+ /**
140
+ * @description Translatability of field labels.
141
+ * @summary Indicates whether the field labels should be translated based on the current language settings.
142
+ * This is useful for applications supporting multiple languages.
143
+ *
144
+ * @type {StringOrBoolean}
145
+ * @default true
146
+ * @memberOf CrudFieldComponent
147
+ */
148
+ this.translatable = true;
149
+ }
150
+ ngOnInit() {
151
+ // super.onInit(this.updateOn);
152
+ if ([OperationKeys.READ, OperationKeys.DELETE].includes(this.operation)) {
153
+ this.formGroup = undefined;
154
+ }
155
+ else {
156
+ if (this.type === HTML5InputTypes.RADIO && !this.value)
157
+ this.formGroup?.get(this.name)?.setValue(this.options[0].value); // TODO: migrate to RenderingEngine
158
+ }
159
+ }
160
+ ngAfterViewInit() {
161
+ if ([OperationKeys.READ, OperationKeys.DELETE].includes(this.operation))
162
+ super.afterViewInit();
163
+ }
164
+ ngOnDestroy() {
165
+ if ([OperationKeys.READ, OperationKeys.DELETE].includes(this.operation))
166
+ this.onDestroy();
167
+ }
168
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CrudFieldComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
169
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CrudFieldComponent, isStandalone: true, selector: "ngx-decaf-crud-field", inputs: { operation: "operation", name: "name", path: "path", childOf: "childOf", type: "type", value: "value", disabled: "disabled", label: "label", placeholder: "placeholder", format: "format", hidden: "hidden", max: "max", maxlength: "maxlength", min: "min", minlength: "minlength", pattern: "pattern", readonly: "readonly", required: "required", step: "step", equals: "equals", different: "different", lessThan: "lessThan", lessThanOrEqual: "lessThanOrEqual", greaterThan: "greaterThan", greaterThanOrEqual: "greaterThanOrEqual", cols: "cols", rows: "rows", alignment: "alignment", checked: "checked", justify: "justify", cancelText: "cancelText", interface: "interface", options: "options", mode: "mode", spellcheck: "spellcheck", inputmode: "inputmode", autocomplete: "autocomplete", fill: "fill", labelPlacement: "labelPlacement", updateOn: "updateOn", formGroup: "formGroup", formControl: "formControl", translatable: "translatable", uid: "uid" }, viewQueries: [{ propertyName: "component", first: true, predicate: ["component"], descendants: true, read: ElementRef }], usesInheritance: true, ngImport: i0, template: "@if(operation === 'read' || operation === 'delete') {\n <ng-container #component>\n <div [class]=\"'dcf-input-item ' + operation\">\n <ion-item>\n <ion-label>\n {{ label | translate }}<br />\n @if(value) {\n <ion-text class=\"dcf-display-block\" [innerHTML]=\"value\"></ion-text>\n } @else {\n <br />\n }\n </ion-label>\n </ion-item>\n </div>\n </ng-container>\n} @else {\n <ng-container #component [formGroup]=\"formGroup\">\n <div #container [class]=\"'dcf-input-item ' + (operation || 'create')\">\n @if(type === 'textarea') {\n <ion-textarea\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [required]=\"required !== undefined ? required : null\"\n [minlength]=\"minlength !== undefined ? minlength : null\"\n [maxlength]=\"maxlength !== undefined ? maxlength : null\"\n [readonly]=\"readonly !== undefined ? readonly : null\"\n [inputmode]=\"inputmode\"\n [spellcheck]=\"spellcheck\"\n [rows]=\"rows\"\n [labelPlacement]=\"labelPlacement\"\n [value]=\"value\"\n [fill]=\"fill\"\n [placeholder]=\"translatable ? (placeholder | translate) : placeholder\"\n [formControlName]=\"name\"\n [label]=\"translatable ? (label | translate) : label\"\n >\n </ion-textarea>\n }\n @else if(type === 'checkbox') {\n <ion-item>\n <ion-checkbox\n #checkboxElement\n [name]=\"path\"\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [labelPlacement]=\"labelPlacement\"\n [justify]=\"justify\"\n [value]=\"value\"\n [checked]=\"checked\"\n [readonly]=\"readonly\"\n (ionChange)=\"checked = !checked\"\n [formControlName]=\"name\"\n >\n <span [innerHTML]=\"label | translate\"></span>\n </ion-checkbox>\n </ion-item>\n }\n @else if(type === 'radio') {\n <ion-radio-group [formControlName]=\"name\" [name]=\"path\" [value]=\"value\" #component>\n <label class=\"dcf-radio-group-label\" [for]=\"path\">{{label | translate}}</label>\n @for(option of options; track option) {\n <ion-item>\n <ion-radio\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [labelPlacement]=\"labelPlacement\"\n [alignment]=\"alignment\"\n [justify]=\"justify\"\n [readonly]=\"readonly\"\n [value]=\"option.value\"\n >{{ translatable ? (option?.text | translate) : option?.text }}</ion-radio>\n </ion-item>\n }\n </ion-radio-group>\n }\n @else if(type === 'select') {\n <ion-item>\n <ion-select\n [name]=\"path\"\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [labelPlacement]=\"labelPlacement\"\n [label]=\"translatable ? (label | translate : label) : label\"\n [value]=\"value\"\n [fill]=\"fill\"\n [placeholder]=\"translatable ? (placeholder | translate) : placeholder\"\n [formControlName]=\"name\"\n [interface]=\"interface\">\n @for(option of options; track option.value) {\n <ion-select-option [value]=\"option.value\">\n {{ translatable ? (option.text | translate) : options.text }}\n </ion-select-option>\n }\n </ion-select>\n </ion-item>\n }\n @else {\n <ion-input\n [name]=\"path\"\n [type]=\"type\"\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [inputmode]=\"inputmode\"\n [labelPlacement]=\"labelPlacement\"\n [required]=\"required !== undefined ? required : false\"\n [minlength]=\"minlength !== undefined ? minlength : null\"\n [maxlength]=\"maxlength !== undefined ? maxlength : null\"\n [readonly]=\"readonly !== undefined ? readonly : null\"\n [max]=\"max !== undefined ? max : null\"\n [min]=\"min !== undefined ? min : null\"\n [pattern]=\"pattern !== undefined ? pattern : null\"\n [step]=\"step !== undefined ? step : null\"\n [fill]=\"fill\"\n [placeholder]=\"translatable ? (placeholder | translate) : placeholder\"\n [formControlName]=\"name\"\n [label]=\"translatable ? (label | translate) : label\">\n </ion-input>\n }\n @if((!formControl.pristine || formControl.touched) && !formControl.valid) {\n <div class=\"error dcf-error dcf-flex dcf-flex-top\">\n @for(error of getErrors(container); track error.key) {\n * {{ sf((\"errors.\" + error.message) | translate, this[error.key]) }}\n }\n </div>\n }\n </div>\n </ng-container>\n}\n\n", styles: [".dcf-input-item.create,.dcf-input-item.update{margin-bottom:1.8rem;margin-top:0!important}.dcf-input-item.create.checkbox+.checkbox,.dcf-input-item.update.checkbox+.checkbox{margin-top:-.25rem!important}.dcf-input-item.create ion-item,.dcf-input-item.update ion-item{--border-color: transparent}.dcf-input-item.create ion-item.dcf-text-wrap ion-label>*,.dcf-input-item.update ion-item.dcf-text-wrap ion-label>*{white-space:wrap!important;word-break:break-all!important}.dcf-input-item.read ion-label,.dcf-input-item.delete ion-label{font-weight:600}.dcf-input-item ion-item{--padding-end: 0rem;--padding-start: 0px !important;--padding-top: 0px !important;--background: transparent;--background-hover-opacity: .1;--background-hover: var(--ion-color-primary);--background-activated-opacity: .15;--background-focused: var(--ion-color-primary);--background-focused-opacity: .15}.dcf-input-item ion-item span,.dcf-input-item ion-item ion-text{font-weight:400!important;font-size:.925rem;min-height:.5rem!important}.dcf-input-item ion-item span:not(.dcf-display-block),.dcf-input-item ion-item ion-text:not(.dcf-display-block){display:inline-block}.dcf-input-item ion-item span.dcf-display-block,.dcf-input-item ion-item ion-text.dcf-display-block{display:block!important}ion-textarea textarea{scrollbar-width:thin!important;margin-bottom:.5rem!important}ion-select.dcf-select-label-placement-floating::part(label){line-height:1.2rem!important}.dcf-proccessing,.dcf-proccessing *{pointer-events:none;touch-action:none;cursor:text}ion-checkbox{--size: 1.5rem;--checkbox-background-checked: var(--ion-color-primary);--checkmark-width: 2px}ion-item{--inner-padding-start: .75rem}ion-checkbox::part(container){border-radius:50%;border:2px solid var(--ion-color-primary);padding:3px}ion-item .dcf-radio-group-label,ion-radio-group .dcf-radio-group-label{font-weight:600}ion-item .dcf-radio-group-label~ion-item,ion-radio-group .dcf-radio-group-label~ion-item{margin-top:.5rem;--inner-padding-start: .75rem}ion-item+.dcf-helper,ion-radio-group+.dcf-helper{padding-left:.75rem;position:relative}.dcf-error{position:absolute;color:var(--ion-color-danger)!important;font-size:.8rem!important;font-weight:600!important;line-height:1.1rem;box-sizing:border-box;z-index:9999;margin-top:0;animation-duration:.1s;animation-timing-function:ease-out;animation-fill-mode:both;animation-name:fadeTopSmallAnimation;display:flex;align-items:flex-start;gap:.25rem}.dcf-error .ti,.dcf-error ion-icon{position:relative;top:2px!important;min-width:20px;font-size:1rem!important;text-align:left}.dcf-helper{font-size:.875rem!important;font-weight:500;margin-top:.25rem;margin-bottom:-.75rem}.dcf-helper.dcf-has-action{cursor:pointer;color:var(--ion-color-gray-8)!important;text-decoration:underline}.dcf-error+.dcf-helper{padding-top:1rem}@keyframes fadeTopSmallAnimation{0%{opacity:0;transform:translateY(-10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeBottomSmallAnimation{0%{opacity:0;transform:translateY(10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeTopMediumAnimation{0%{opacity:0;transform:translateY(-50px)}to{opacity:1;transform:translateY(0)}}\n"], dependencies: [{ kind: "ngmodule", type: ForAngularModule }, { kind: "component", type: i1.IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i2.MinLengthValidator, selector: "[minlength][formControlName],[minlength][formControl],[minlength][ngModel]", inputs: ["minlength"] }, { kind: "directive", type: i2.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i2.PatternValidator, selector: "[pattern][formControlName],[pattern][formControl],[pattern][ngModel]", inputs: ["pattern"] }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }, { kind: "component", type: IonInput, selector: "ion-input", inputs: ["accept", "autocapitalize", "autocomplete", "autocorrect", "autofocus", "clearInput", "clearOnEdit", "color", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "max", "maxlength", "min", "minlength", "mode", "multiple", "name", "pattern", "placeholder", "readonly", "required", "shape", "size", "spellcheck", "step", "type", "value"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonCheckbox, selector: "ion-checkbox", inputs: ["checked", "color", "disabled", "errorText", "helperText", "indeterminate", "justify", "labelPlacement", "mode", "name", "value"] }, { kind: "component", type: IonRadioGroup, selector: "ion-radio-group", inputs: ["allowEmptySelection", "compareWith", "errorText", "helperText", "name", "value"] }, { kind: "component", type: IonRadio, selector: "ion-radio", inputs: ["alignment", "color", "disabled", "justify", "labelPlacement", "mode", "name", "value"] }, { kind: "component", type: IonSelect, selector: "ion-select", inputs: ["cancelText", "color", "compareWith", "disabled", "errorText", "expandedIcon", "fill", "helperText", "interface", "interfaceOptions", "justify", "label", "labelPlacement", "mode", "multiple", "name", "okText", "placeholder", "selectedText", "shape", "toggleIcon", "value"] }, { kind: "component", type: IonSelectOption, selector: "ion-select-option", inputs: ["disabled", "value"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonTextarea, selector: "ion-textarea", inputs: ["autoGrow", "autocapitalize", "autofocus", "clearOnEdit", "color", "cols", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "maxlength", "minlength", "mode", "name", "placeholder", "readonly", "required", "rows", "shape", "spellcheck", "value", "wrap"] }] }); }
170
+ };
171
+ CrudFieldComponent = __decorate([
172
+ Dynamic(),
173
+ __metadata("design:paramtypes", [ElementRef])
174
+ ], CrudFieldComponent);
175
+ export { CrudFieldComponent };
176
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CrudFieldComponent, decorators: [{
177
+ type: Component,
178
+ args: [{ standalone: true, imports: [
179
+ ForAngularModule,
180
+ IonInput,
181
+ IonItem,
182
+ IonCheckbox,
183
+ IonRadioGroup,
184
+ IonRadio,
185
+ IonSelect,
186
+ IonSelectOption,
187
+ IonLabel,
188
+ IonText,
189
+ IonTextarea,
190
+ ], selector: 'ngx-decaf-crud-field', schemas: [CUSTOM_ELEMENTS_SCHEMA], template: "@if(operation === 'read' || operation === 'delete') {\n <ng-container #component>\n <div [class]=\"'dcf-input-item ' + operation\">\n <ion-item>\n <ion-label>\n {{ label | translate }}<br />\n @if(value) {\n <ion-text class=\"dcf-display-block\" [innerHTML]=\"value\"></ion-text>\n } @else {\n <br />\n }\n </ion-label>\n </ion-item>\n </div>\n </ng-container>\n} @else {\n <ng-container #component [formGroup]=\"formGroup\">\n <div #container [class]=\"'dcf-input-item ' + (operation || 'create')\">\n @if(type === 'textarea') {\n <ion-textarea\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [required]=\"required !== undefined ? required : null\"\n [minlength]=\"minlength !== undefined ? minlength : null\"\n [maxlength]=\"maxlength !== undefined ? maxlength : null\"\n [readonly]=\"readonly !== undefined ? readonly : null\"\n [inputmode]=\"inputmode\"\n [spellcheck]=\"spellcheck\"\n [rows]=\"rows\"\n [labelPlacement]=\"labelPlacement\"\n [value]=\"value\"\n [fill]=\"fill\"\n [placeholder]=\"translatable ? (placeholder | translate) : placeholder\"\n [formControlName]=\"name\"\n [label]=\"translatable ? (label | translate) : label\"\n >\n </ion-textarea>\n }\n @else if(type === 'checkbox') {\n <ion-item>\n <ion-checkbox\n #checkboxElement\n [name]=\"path\"\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [labelPlacement]=\"labelPlacement\"\n [justify]=\"justify\"\n [value]=\"value\"\n [checked]=\"checked\"\n [readonly]=\"readonly\"\n (ionChange)=\"checked = !checked\"\n [formControlName]=\"name\"\n >\n <span [innerHTML]=\"label | translate\"></span>\n </ion-checkbox>\n </ion-item>\n }\n @else if(type === 'radio') {\n <ion-radio-group [formControlName]=\"name\" [name]=\"path\" [value]=\"value\" #component>\n <label class=\"dcf-radio-group-label\" [for]=\"path\">{{label | translate}}</label>\n @for(option of options; track option) {\n <ion-item>\n <ion-radio\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [labelPlacement]=\"labelPlacement\"\n [alignment]=\"alignment\"\n [justify]=\"justify\"\n [readonly]=\"readonly\"\n [value]=\"option.value\"\n >{{ translatable ? (option?.text | translate) : option?.text }}</ion-radio>\n </ion-item>\n }\n </ion-radio-group>\n }\n @else if(type === 'select') {\n <ion-item>\n <ion-select\n [name]=\"path\"\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [labelPlacement]=\"labelPlacement\"\n [label]=\"translatable ? (label | translate : label) : label\"\n [value]=\"value\"\n [fill]=\"fill\"\n [placeholder]=\"translatable ? (placeholder | translate) : placeholder\"\n [formControlName]=\"name\"\n [interface]=\"interface\">\n @for(option of options; track option.value) {\n <ion-select-option [value]=\"option.value\">\n {{ translatable ? (option.text | translate) : options.text }}\n </ion-select-option>\n }\n </ion-select>\n </ion-item>\n }\n @else {\n <ion-input\n [name]=\"path\"\n [type]=\"type\"\n [mode]=\"mode\"\n [hidden]=\"hidden\"\n [inputmode]=\"inputmode\"\n [labelPlacement]=\"labelPlacement\"\n [required]=\"required !== undefined ? required : false\"\n [minlength]=\"minlength !== undefined ? minlength : null\"\n [maxlength]=\"maxlength !== undefined ? maxlength : null\"\n [readonly]=\"readonly !== undefined ? readonly : null\"\n [max]=\"max !== undefined ? max : null\"\n [min]=\"min !== undefined ? min : null\"\n [pattern]=\"pattern !== undefined ? pattern : null\"\n [step]=\"step !== undefined ? step : null\"\n [fill]=\"fill\"\n [placeholder]=\"translatable ? (placeholder | translate) : placeholder\"\n [formControlName]=\"name\"\n [label]=\"translatable ? (label | translate) : label\">\n </ion-input>\n }\n @if((!formControl.pristine || formControl.touched) && !formControl.valid) {\n <div class=\"error dcf-error dcf-flex dcf-flex-top\">\n @for(error of getErrors(container); track error.key) {\n * {{ sf((\"errors.\" + error.message) | translate, this[error.key]) }}\n }\n </div>\n }\n </div>\n </ng-container>\n}\n\n", styles: [".dcf-input-item.create,.dcf-input-item.update{margin-bottom:1.8rem;margin-top:0!important}.dcf-input-item.create.checkbox+.checkbox,.dcf-input-item.update.checkbox+.checkbox{margin-top:-.25rem!important}.dcf-input-item.create ion-item,.dcf-input-item.update ion-item{--border-color: transparent}.dcf-input-item.create ion-item.dcf-text-wrap ion-label>*,.dcf-input-item.update ion-item.dcf-text-wrap ion-label>*{white-space:wrap!important;word-break:break-all!important}.dcf-input-item.read ion-label,.dcf-input-item.delete ion-label{font-weight:600}.dcf-input-item ion-item{--padding-end: 0rem;--padding-start: 0px !important;--padding-top: 0px !important;--background: transparent;--background-hover-opacity: .1;--background-hover: var(--ion-color-primary);--background-activated-opacity: .15;--background-focused: var(--ion-color-primary);--background-focused-opacity: .15}.dcf-input-item ion-item span,.dcf-input-item ion-item ion-text{font-weight:400!important;font-size:.925rem;min-height:.5rem!important}.dcf-input-item ion-item span:not(.dcf-display-block),.dcf-input-item ion-item ion-text:not(.dcf-display-block){display:inline-block}.dcf-input-item ion-item span.dcf-display-block,.dcf-input-item ion-item ion-text.dcf-display-block{display:block!important}ion-textarea textarea{scrollbar-width:thin!important;margin-bottom:.5rem!important}ion-select.dcf-select-label-placement-floating::part(label){line-height:1.2rem!important}.dcf-proccessing,.dcf-proccessing *{pointer-events:none;touch-action:none;cursor:text}ion-checkbox{--size: 1.5rem;--checkbox-background-checked: var(--ion-color-primary);--checkmark-width: 2px}ion-item{--inner-padding-start: .75rem}ion-checkbox::part(container){border-radius:50%;border:2px solid var(--ion-color-primary);padding:3px}ion-item .dcf-radio-group-label,ion-radio-group .dcf-radio-group-label{font-weight:600}ion-item .dcf-radio-group-label~ion-item,ion-radio-group .dcf-radio-group-label~ion-item{margin-top:.5rem;--inner-padding-start: .75rem}ion-item+.dcf-helper,ion-radio-group+.dcf-helper{padding-left:.75rem;position:relative}.dcf-error{position:absolute;color:var(--ion-color-danger)!important;font-size:.8rem!important;font-weight:600!important;line-height:1.1rem;box-sizing:border-box;z-index:9999;margin-top:0;animation-duration:.1s;animation-timing-function:ease-out;animation-fill-mode:both;animation-name:fadeTopSmallAnimation;display:flex;align-items:flex-start;gap:.25rem}.dcf-error .ti,.dcf-error ion-icon{position:relative;top:2px!important;min-width:20px;font-size:1rem!important;text-align:left}.dcf-helper{font-size:.875rem!important;font-weight:500;margin-top:.25rem;margin-bottom:-.75rem}.dcf-helper.dcf-has-action{cursor:pointer;color:var(--ion-color-gray-8)!important;text-decoration:underline}.dcf-error+.dcf-helper{padding-top:1rem}@keyframes fadeTopSmallAnimation{0%{opacity:0;transform:translateY(-10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeBottomSmallAnimation{0%{opacity:0;transform:translateY(10px)}to{opacity:1;transform:translateY(0)}}@keyframes fadeTopMediumAnimation{0%{opacity:0;transform:translateY(-50px)}to{opacity:1;transform:translateY(0)}}\n"] }]
191
+ }], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { operation: [{
192
+ type: Input,
193
+ args: [{ required: true }]
194
+ }], name: [{
195
+ type: Input,
196
+ args: [{ required: true }]
197
+ }], path: [{
198
+ type: Input,
199
+ args: [{ required: true }]
200
+ }], childOf: [{
201
+ type: Input
202
+ }], type: [{
203
+ type: Input,
204
+ args: [{ required: true }]
205
+ }], value: [{
206
+ type: Input
207
+ }], disabled: [{
208
+ type: Input
209
+ }], label: [{
210
+ type: Input,
211
+ args: [{ required: true }]
212
+ }], placeholder: [{
213
+ type: Input
214
+ }], format: [{
215
+ type: Input
216
+ }], hidden: [{
217
+ type: Input
218
+ }], max: [{
219
+ type: Input
220
+ }], maxlength: [{
221
+ type: Input
222
+ }], min: [{
223
+ type: Input
224
+ }], minlength: [{
225
+ type: Input
226
+ }], pattern: [{
227
+ type: Input
228
+ }], readonly: [{
229
+ type: Input
230
+ }], required: [{
231
+ type: Input
232
+ }], step: [{
233
+ type: Input
234
+ }], equals: [{
235
+ type: Input
236
+ }], different: [{
237
+ type: Input
238
+ }], lessThan: [{
239
+ type: Input
240
+ }], lessThanOrEqual: [{
241
+ type: Input
242
+ }], greaterThan: [{
243
+ type: Input
244
+ }], greaterThanOrEqual: [{
245
+ type: Input
246
+ }], cols: [{
247
+ type: Input
248
+ }], rows: [{
249
+ type: Input
250
+ }], alignment: [{
251
+ type: Input
252
+ }], checked: [{
253
+ type: Input
254
+ }], justify: [{
255
+ type: Input
256
+ }], cancelText: [{
257
+ type: Input
258
+ }], interface: [{
259
+ type: Input
260
+ }], options: [{
261
+ type: Input
262
+ }], mode: [{
263
+ type: Input
264
+ }], spellcheck: [{
265
+ type: Input
266
+ }], inputmode: [{
267
+ type: Input
268
+ }], autocomplete: [{
269
+ type: Input
270
+ }], fill: [{
271
+ type: Input
272
+ }], labelPlacement: [{
273
+ type: Input
274
+ }], updateOn: [{
275
+ type: Input
276
+ }], component: [{
277
+ type: ViewChild,
278
+ args: ['component', { read: ElementRef }]
279
+ }], formGroup: [{
280
+ type: Input
281
+ }], formControl: [{
282
+ type: Input
283
+ }], translatable: [{
284
+ type: Input
285
+ }], uid: [{
286
+ type: Input
287
+ }] } });
288
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3J1ZC1maWVsZC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvY3J1ZC1maWVsZC9jcnVkLWZpZWxkLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9jcnVkLWZpZWxkL2NydWQtZmllbGQuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFFTCxTQUFTLEVBQ1Qsc0JBQXNCLEVBQ3RCLFVBQVUsRUFDVixLQUFLLEVBR0wsU0FBUyxHQUNWLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxXQUFXLEVBQWEsTUFBTSxnQkFBZ0IsQ0FBQztBQUV4RCxPQUFPLEVBQWtCLGFBQWEsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ3hFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLCtCQUErQixDQUFDO0FBQ2pFLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUVsRCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUM5RCxPQUFPLEVBQ0wsV0FBVyxFQUNYLFFBQVEsRUFDUixPQUFPLEVBQ1AsUUFBUSxFQUNSLFFBQVEsRUFDUixhQUFhLEVBQ2IsU0FBUyxFQUNULGVBQWUsRUFDZixPQUFPLEVBQ1AsV0FBVyxHQUNaLE1BQU0sMkJBQTJCLENBQUM7QUFDbkMsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHlCQUF5QixDQUFDOzs7OztBQUUxRDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBMENHO0FBdUJJLElBQU0sa0JBQWtCLEdBQXhCLE1BQU0sa0JBQW1CLFNBQVEsZ0JBQWdCO0lBQ3RELFlBQStCLFVBQXNCO1FBQ25ELEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQztRQURXLGVBQVUsR0FBVixVQUFVLENBQVk7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O3NCQURuQixLQUFLO2dCQWFOLFlBQVk7c0JBRFgsS0FBSztnQkFXTixHQUFHO3NCQURGLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBZnRlclZpZXdJbml0LFxuICBDb21wb25lbnQsXG4gIENVU1RPTV9FTEVNRU5UU19TQ0hFTUEsXG4gIEVsZW1lbnRSZWYsXG4gIElucHV0LFxuICBPbkRlc3Ryb3ksXG4gIE9uSW5pdCxcbiAgVmlld0NoaWxkLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZvcm1Db250cm9sLCBGb3JtR3JvdXAgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBBdXRvY29tcGxldGVUeXBlcywgU2VsZWN0SW50ZXJmYWNlIH0gZnJvbSAnQGlvbmljL2NvcmUnO1xuaW1wb3J0IHsgQ3J1ZE9wZXJhdGlvbnMsIE9wZXJhdGlvbktleXMgfSBmcm9tICdAZGVjYWYtdHMvZGItZGVjb3JhdG9ycyc7XG5pbXBvcnQgeyBOZ3hDcnVkRm9ybUZpZWxkIH0gZnJvbSAnLi4vLi4vZW5naW5lL05neENydWRGb3JtRmllbGQnO1xuaW1wb3J0IHsgRHluYW1pYyB9IGZyb20gJy4uLy4uL2VuZ2luZS9kZWNvcmF0b3JzJztcbmltcG9ydCB7IEZpZWxkVXBkYXRlTW9kZSwgUG9zc2libGVJbnB1dFR5cGVzLCBSYWRpb09wdGlvbiwgU2VsZWN0T3B0aW9uLCBTdHJpbmdPckJvb2xlYW4gfSBmcm9tICcuLi8uLi9lbmdpbmUvdHlwZXMnO1xuaW1wb3J0IHsgRm9yQW5ndWxhck1vZHVsZSB9IGZyb20gJ3NyYy9saWIvZm9yLWFuZ3VsYXIubW9kdWxlJztcbmltcG9ydCB7XG4gIElvbkNoZWNrYm94LFxuICBJb25JbnB1dCxcbiAgSW9uSXRlbSxcbiAgSW9uTGFiZWwsXG4gIElvblJhZGlvLFxuICBJb25SYWRpb0dyb3VwLFxuICBJb25TZWxlY3QsXG4gIElvblNlbGVjdE9wdGlvbixcbiAgSW9uVGV4dCxcbiAgSW9uVGV4dGFyZWEsXG59IGZyb20gJ0Bpb25pYy9hbmd1bGFyL3N0YW5kYWxvbmUnO1xuaW1wb3J0IHsgSFRNTDVJbnB1dFR5cGVzIH0gZnJvbSAnQGRlY2FmLXRzL3VpLWRlY29yYXRvcnMnO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBBIGR5bmFtaWMgZm9ybSBmaWVsZCBjb21wb25lbnQgZm9yIENSVUQgb3BlcmF0aW9ucy5cbiAqIEBzdW1tYXJ5IFRoZSBDcnVkRmllbGRDb21wb25lbnQgaXMgYSB2ZXJzYXRpbGUgZm9ybSBmaWVsZCBjb21wb25lbnQgdGhhdCBhZGFwdHMgdG8gZGlmZmVyZW50XG4gKiBpbnB1dCB0eXBlcyBhbmQgQ1JVRCBvcGVyYXRpb25zLiBJdCBleHRlbmRzIE5neENydWRGb3JtRmllbGQgdG8gaW5oZXJpdCBmb3JtIGhhbmRsaW5nIGNhcGFiaWxpdGllc1xuICogYW5kIGltcGxlbWVudHMgbGlmZWN5Y2xlIGhvb2tzIHRvIHByb3Blcmx5IGluaXRpYWxpemUsIHJlbmRlciwgYW5kIGNsZWFuIHVwLiBUaGlzIGNvbXBvbmVudFxuICogc3VwcG9ydHMgdmFyaW91cyBpbnB1dCB0eXBlcyAodGV4dCwgbnVtYmVyLCBkYXRlLCBzZWxlY3QsIGV0Yy4pLCB2YWxpZGF0aW9uIHJ1bGVzLCBhbmQgc3R5bGluZ1xuICogb3B0aW9ucywgbWFraW5nIGl0IHN1aXRhYmxlIGZvciBidWlsZGluZyBkeW5hbWljIGZvcm1zIGZvciBjcmVhdGUsIHJlYWQsIHVwZGF0ZSwgYW5kIGRlbGV0ZVxuICogb3BlcmF0aW9ucy5cbiAqXG4gKiBAcGFyYW0ge0NydWRPcGVyYXRpb25zfSBvcGVyYXRpb24gLSBUaGUgQ1JVRCBvcGVyYXRpb24gYmVpbmcgcGVyZm9ybWVkIChjcmVhdGUsIHJlYWQsIHVwZGF0ZSwgZGVsZXRlKVxuICogQHBhcmFtIHtzdHJpbmd9IG5hbWUgLSBUaGUgZmllbGQgbmFtZSwgdXNlZCBhcyBmb3JtIGNvbnRyb2wgaWRlbnRpZmllclxuICogQHBhcmFtIHtQb3NzaWJsZUlucHV0VHlwZXN9IHR5cGUgLSBUaGUgaW5wdXQgdHlwZSAodGV4dCwgbnVtYmVyLCBkYXRlLCBzZWxlY3QsIGV0Yy4pXG4gKiBAcGFyYW0ge3N0cmluZ3xudW1iZXJ8RGF0ZX0gdmFsdWUgLSBUaGUgaW5pdGlhbCB2YWx1ZSBvZiB0aGUgZmllbGRcbiAqIEBwYXJhbSB7Ym9vbGVhbn0gZGlzYWJsZWQgLSBXaGV0aGVyIHRoZSBmaWVsZCBpcyBkaXNhYmxlZFxuICogQHBhcmFtIHtzdHJpbmd9IGxhYmVsIC0gVGhlIGRpc3BsYXkgbGFiZWwgZm9yIHRoZSBmaWVsZFxuICogQHBhcmFtIHtzdHJpbmd9IHBsYWNlaG9sZGVyIC0gUGxhY2Vob2xkZXIgdGV4dCB3aGVuIGZpZWxkIGlzIGVtcHR5XG4gKiBAcGFyYW0ge3N0cmluZ30gZm9ybWF0IC0gRm9ybWF0IHBhdHRlcm4gZm9yIHRoZSBmaWVsZCB2YWx1ZVxuICogQHBhcmFtIHtib29sZWFufSBoaWRkZW4gLSBXaGV0aGVyIHRoZSBmaWVsZCBzaG91bGQgYmUgaGlkZGVuXG4gKiBAcGFyYW0ge251bWJlcnxEYXRlfSBtYXggLSBNYXhpbXVtIGFsbG93ZWQgdmFsdWVcbiAqIEBwYXJhbSB7bnVtYmVyfSBtYXhsZW5ndGggLSBNYXhpbXVtIGFsbG93ZWQgbGVuZ3RoXG4gKiBAcGFyYW0ge251bWJlcnxEYXRlfSBtaW4gLSBNaW5pbXVtIGFsbG93ZWQgdmFsdWVcbiAqIEBwYXJhbSB7bnVtYmVyfSBtaW5sZW5ndGggLSBNaW5pbXVtIGFsbG93ZWQgbGVuZ3RoXG4gKiBAcGFyYW0ge3N0cmluZ30gcGF0dGVybiAtIFZhbGlkYXRpb24gcGF0dGVyblxuICogQHBhcmFtIHtib29sZWFufSByZWFkb25seSAtIFdoZXRoZXIgdGhlIGZpZWxkIGlzIHJlYWQtb25seVxuICogQHBhcmFtIHtib29sZWFufSByZXF1aXJlZCAtIFdoZXRoZXIgdGhlIGZpZWxkIGlzIHJlcXVpcmVkXG4gKiBAcGFyYW0ge251bWJlcn0gc3RlcCAtIFN0ZXAgdmFsdWUgZm9yIG51bWJlciBpbnB1dHNcbiAqIEBwYXJhbSB7Rm9ybUdyb3VwfSBmb3JtR3JvdXAgLSBUaGUgcGFyZW50IGZvcm0gZ3JvdXBcbiAqIEBwYXJhbSB7U3RyaW5nT3JCb29sZWFufSB0cmFuc2xhdGFibGUgLSBXaGV0aGVyIGZpZWxkIGxhYmVscyBzaG91bGQgYmUgdHJhbnNsYXRlZFxuICpcbiAqIEBjb21wb25lbnQgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gKiBAZXhhbXBsZVxuICogPG5neC1kZWNhZi1jcnVkLWZpZWxkXG4gKiAgIG9wZXJhdGlvbj1cImNyZWF0ZVwiXG4gKiAgIG5hbWU9XCJmaXJzdE5hbWVcIlxuICogICB0eXBlPVwidGV4dFwiXG4gKiAgIGxhYmVsPVwiPE5BTUU+XCJcbiAqICAgcGxhY2Vob2xkZXI9XCI8TkFNRT5cIlxuICogICBbdmFsdWVdPVwibW9kZWwuZmlyc3ROYW1lXCJcbiAqICAgW2Rpc2FibGVkXT1cIm1vZGVsLnJlYWRPbmx5XCI+XG4gKlxuICpcbiAqIEBtZW1iZXJPZiBtb2R1bGU6Zm9yLWFuZ3VsYXJcbiAqL1xuQER5bmFtaWMoKVxuQENvbXBvbmVudCh7XG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtcbiAgICBGb3JBbmd1bGFyTW9kdWxlLFxuICAgIElvbklucHV0LFxuICAgIElvbkl0ZW0sXG4gICAgSW9uQ2hlY2tib3gsXG4gICAgSW9uUmFkaW9Hcm91cCxcbiAgICBJb25SYWRpbyxcbiAgICBJb25TZWxlY3QsXG4gICAgSW9uU2VsZWN0T3B0aW9uLFxuICAgIElvbkxhYmVsLFxuICAgIElvblRleHQsXG4gICAgSW9uVGV4dGFyZWEsXG5cbiAgXSxcbiAgc2VsZWN0b3I6ICduZ3gtZGVjYWYtY3J1ZC1maWVsZCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9jcnVkLWZpZWxkLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmw6ICcuL2NydWQtZmllbGQuY29tcG9uZW50LnNjc3MnLFxuICBzY2hlbWFzOiBbQ1VTVE9NX0VMRU1FTlRTX1NDSEVNQV0sXG59KVxuZXhwb3J0IGNsYXNzIENydWRGaWVsZENvbXBvbmVudCBleHRlbmRzIE5neENydWRGb3JtRmllbGQgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSwgQWZ0ZXJWaWV3SW5pdCB7XG4gIGNvbnN0cnVjdG9yKHByb3RlY3RlZCBvdmVycmlkZSBlbGVtZW50UmVmOiBFbGVtZW50UmVmKSB7XG4gICAgc3VwZXIoZWxlbWVudFJlZik7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBDUlVEIG9wZXJhdGlvbiBiZWluZyBwZXJmb3JtZWQuXG4gICAqIEBzdW1tYXJ5IFNwZWNpZmllcyB3aGljaCBDUlVEIG9wZXJhdGlvbiAoQ3JlYXRlLCBSZWFkLCBVcGRhdGUsIERlbGV0ZSkgdGhlIGZpZWxkIGlzIGJlaW5nIHVzZWQgZm9yLlxuICAgKiBUaGlzIGFmZmVjdHMgaG93IHRoZSBmaWVsZCBiZWhhdmVzIGFuZCBpcyByZW5kZXJlZC4gRm9yIGV4YW1wbGUsIGZpZWxkcyBtaWdodCBiZSByZWFkLW9ubHkgaW5cbiAgICogJ3JlYWQnIG1vZGUgYnV0IGVkaXRhYmxlIGluICdjcmVhdGUnIG9yICd1cGRhdGUnIG1vZGVzLlxuICAgKlxuICAgKiBAdHlwZSB7Q3J1ZE9wZXJhdGlvbnN9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pXG4gIG92ZXJyaWRlIG9wZXJhdGlvbiE6IENydWRPcGVyYXRpb25zO1xuXG4gIC8qKlxuICAgKiBAc3VtbWFyeSAgVGhlIGZsYXQgZmllbGQgbmFtZSB1c2VkIGFzIHRoZSBmb3JtIGNvbnRyb2wgaWRlbnRpZmllciBpbiBpbW1lZGlhdGUgcGFyZW50IEZvcm1Hcm91cC5cbiAgICogQGRlc2NyaXB0aW9uXG4gICAqIFNwZWNpZmllcyB0aGUgbmFtZSBvZiB0aGUgZmllbGQsIHdoaWNoIGlzIHVzZWQgYXMgdGhlIEZvcm1Db250cm9sIGlkZW50aWZpZXIgaW4gaW1tZWRpYXRlIHBhcmVudCBGb3JtR3JvdXAuXG4gICAqIFRoaXMgdmFsdWUgbXVzdCBiZSB1bmlxdWUgd2l0aGluIHRoZSBpbW1lZGlhdGUgcGFyZW50IEZvcm1Hcm91cCBjb250ZXh0IGFuZCBzaG91bGQgbm90IGNvbnRhaW4gZG90cyBvciBuZXN0aW5nLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KVxuICBvdmVycmlkZSBuYW1lITogc3RyaW5nO1xuXG5cbiAgLyoqXG4gICAqIEBzdW1tYXJ5IFRoZSBmdWxsIGZpZWxkIHBhdGggdXNlZCBmb3IgZm9ybSBjb250cm9sIHJlc29sdXRpb24uXG4gICAqIEBkZXNjcmlwdGlvbiBTcGVjaWZpZXMgdGhlIGhpZXJhcmNoaWNhbCBwYXRoIG9mIHRoZSBmaWVsZCwgdXNlZCB0byByZXNvbHZlIGl0cyBsb2NhdGlvbiB3aXRoaW4gdGhlIHBhcmVudCBGb3JtR3JvdXAgKG9yIG5lc3RlZCBGb3JtR3JvdXBzKS5cbiAgICogSXQgaXMgdXNlZCBhcyB0aGUgaWRlbnRpZmllciBpbiB0aGUgcmVuZGVyZWQgSFRNTCwgYW5kIG1heSBpbmNsdWRlIG5lc3RpbmcgKGUuZy4sICdhZGRyZXNzLmJpbGxpbmcuc3RyZWV0JykgYW5kXG4gICAqIHNob3VsZCBtYXRjaCB0aGUgc3RydWN0dXJlIG9mIHRoZSBkYXRhIG1vZGVsXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pXG4gIG92ZXJyaWRlIHBhdGghOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUaGUgcGFyZW50IGZpZWxkIHBhdGgsIGlmIHRoaXMgZmllbGQgaXMgbmVzdGVkLlxuICAgKiBAc3VtbWFyeSBTcGVjaWZpZXMgdGhlIGZ1bGwgZG90LWRlbGltaXRlZCBwYXRoIG9mIHRoZSBwYXJlbnQgZmllbGQuIFRoaXMgaXMgb25seSBzZXQgd2hlbiB0aGUgZmllbGQgaXMgbmVzdGVkLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBjaGlsZE9mID0gJyc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUaGUgaW5wdXQgdHlwZSBvZiB0aGUgZmllbGQuXG4gICAqIEBzdW1tYXJ5IERlZmluZXMgdGhlIHR5cGUgb2YgaW5wdXQgdG8gcmVuZGVyLCBzdWNoIGFzIHRleHQsIG51bWJlciwgZGF0ZSwgc2VsZWN0LCBldGMuXG4gICAqIFRoaXMgZGV0ZXJtaW5lcyB3aGljaCBJb25pYyBmb3JtIGNvbXBvbmVudCB3aWxsIGJlIHVzZWQgdG8gcmVuZGVyIHRoZSBmaWVsZCBhbmQgaG93XG4gICAqIHRoZSBkYXRhIHdpbGwgYmUgZm9ybWF0dGVkIGFuZCB2YWxpZGF0ZWQuXG4gICAqXG4gICAqIEB0eXBlIHtQb3NzaWJsZUlucHV0VHlwZXN9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pXG4gIG92ZXJyaWRlIHR5cGUhOiBQb3NzaWJsZUlucHV0VHlwZXM7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBUaGUgaW5pdGlhbCB2YWx1ZSBvZiB0aGUgZmllbGQuXG4gICAqIEBzdW1tYXJ5IFNldHMgdGhlIGluaXRpYWwgdmFsdWUgb2YgdGhlIGZvcm0gZmllbGQuIFRoaXMgY2FuIGJlIGEgc3RyaW5nLCBudW1iZXIsIG9yIERhdGVcbiAgICogZGVwZW5kaW5nIG9uIHRoZSBmaWVsZCB0eXBlLiBGb3Igc2VsZWN0IGZpZWxkcywgdGhpcyBzaG91bGQgbWF0Y2ggb25lIG9mIHRoZSBvcHRpb24gdmFsdWVzLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nIHwgbnVtYmVyIHwgRGF0ZX1cbiAgICogQGRlZmF1bHQgJydcbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgb3ZlcnJpZGUgdmFsdWU6IHN0cmluZyB8IG51bWJlciB8IERhdGUgPSAnJztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFdoZXRoZXIgdGhlIGZpZWxkIGlzIGRpc2FibGVkLlxuICAgKiBAc3VtbWFyeSBXaGVuIHRydWUsIHRoZSBmaWVsZCB3aWxsIGJlIHJlbmRlcmVkIGluIGEgZGlzYWJsZWQgc3RhdGUsIHByZXZlbnRpbmcgdXNlciBpbnRlcmFjdGlvbi5cbiAgICogRGlzYWJsZWQgZmllbGRzIGFyZSBzdGlsbCBpbmNsdWRlZCBpbiB0aGUgZm9ybSBtb2RlbCBidXQgY2Fubm90IGJlIGVkaXRlZCBieSB0aGUgdXNlci5cbiAgICpcbiAgICogQHR5cGUge2Jvb2xlYW59XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIGRpc2FibGVkPzogYm9vbGVhbjtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRoZSBkaXNwbGF5IGxhYmVsIGZvciB0aGUgZmllbGQuXG4gICAqIEBzdW1tYXJ5IFRoZSB0ZXh0IGxhYmVsIGRpc3BsYXllZCBhbG9uZ3NpZGUgdGhlIGZpZWxkIHRvIGlkZW50aWZ5IGl0IHRvIHRoZSB1c2VyLlxuICAgKiBUaGlzIGxhYmVsIGNhbiBiZSB0cmFuc2xhdGVkIGlmIHRoZSB0cmFuc2xhdGFibGUgcHJvcGVydHkgaXMgc2V0IHRvIHRydWUuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pXG4gIGxhYmVsITogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUGxhY2Vob2xkZXIgdGV4dCB3aGVuIGZpZWxkIGlzIGVtcHR5LlxuICAgKiBAc3VtbWFyeSBUZXh0IHRoYXQgYXBwZWFycyBpbiB0aGUgaW5wdXQgd2hlbiBpdCBoYXMgbm8gdmFsdWUuIFRoaXMgcHJvdmlkZXMgYSBoaW50IHRvIHRoZSB1c2VyXG4gICAqIGFib3V0IHdoYXQga2luZCBvZiBkYXRhIGlzIGV4cGVjdGVkLiBUaGUgcGxhY2Vob2xkZXIgZGlzYXBwZWFycyB3aGVuIHRoZSB1c2VyIHN0YXJ0cyB0eXBpbmcuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIHBsYWNlaG9sZGVyITogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gRm9ybWF0IHBhdHRlcm4gZm9yIHRoZSBmaWVsZCB2YWx1ZS5cbiAgICogQHN1bW1hcnkgU3BlY2lmaWVzIGEgZm9ybWF0IHBhdHRlcm4gZm9yIHRoZSBmaWVsZCB2YWx1ZSwgd2hpY2ggY2FuIGJlIHVzZWQgZm9yIGRhdGUgZm9ybWF0dGluZyxcbiAgICogbnVtYmVyIGZvcm1hdHRpbmcsIG9yIG90aGVyIHR5cGUtc3BlY2lmaWMgZm9ybWF0dGluZyByZXF1aXJlbWVudHMuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIGZvcm1hdD86IHN0cmluZztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFdoZXRoZXIgdGhlIGZpZWxkIHNob3VsZCBiZSBoaWRkZW4uXG4gICAqIEBzdW1tYXJ5IFdoZW4gdHJ1ZSwgdGhlIGZpZWxkIHdpbGwgbm90IGJlIHZpc2libGUgaW4gdGhlIFVJIGJ1dCB3aWxsIHN0aWxsIGJlIHBhcnQgb2YgdGhlIGZvcm0gbW9kZWwuXG4gICAqIFRoaXMgaXMgdXNlZnVsIGZvciBmaWVsZHMgdGhhdCBuZWVkIHRvIGJlIGluY2x1ZGVkIGluIGZvcm0gc3VibWlzc2lvbiBidXQgc2hvdWxkIG5vdCBiZSBkaXNwbGF5ZWQgdG8gdGhlIHVzZXIuXG4gICAqXG4gICAqIEB0eXBlIHtib29sZWFufVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBoaWRkZW4/OiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gTWF4aW11bSBhbGxvd2VkIHZhbHVlIGZvciB0aGUgZmllbGQuXG4gICAqIEBzdW1tYXJ5IEZvciBudW1iZXIgaW5wdXRzLCB0aGlzIHNldHMgdGhlIG1heGltdW0gYWxsb3dlZCBudW1lcmljIHZhbHVlLlxuICAgKiBGb3IgZGF0ZSBpbnB1dHMsIHRoaXMgc2V0cyB0aGUgbGF0ZXN0IGFsbG93ZWQgZGF0ZS5cbiAgICpcbiAgICogQHR5cGUge251bWJlciB8IERhdGV9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIG1heD86IG51bWJlciB8IERhdGU7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBNYXhpbXVtIGFsbG93ZWQgbGVuZ3RoIGZvciB0ZXh0IGlucHV0LlxuICAgKiBAc3VtbWFyeSBGb3IgdGV4dCBpbnB1dHMsIHRoaXMgc2V0cyB0aGUgbWF4aW11bSBudW1iZXIgb2YgY2hhcmFjdGVycyBhbGxvd2VkLlxuICAgKiBUaGlzIGlzIHVzZWQgZm9yIHZhbGlkYXRpb24gYW5kIG1heSBhbHNvIGJlIHVzZWQgdG8gbGltaXQgaW5wdXQgaW4gdGhlIFVJLlxuICAgKlxuICAgKiBAdHlwZSB7bnVtYmVyfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBtYXhsZW5ndGg/OiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBNaW5pbXVtIGFsbG93ZWQgdmFsdWUgZm9yIHRoZSBmaWVsZC5cbiAgICogQHN1bW1hcnkgRm9yIG51bWJlciBpbnB1dHMsIHRoaXMgc2V0cyB0aGUgbWluaW11bSBhbGxvd2VkIG51bWVyaWMgdmFsdWUuXG4gICAqIEZvciBkYXRlIGlucHV0cywgdGhpcyBzZXRzIHRoZSBlYXJsaWVzdCBhbGxvd2VkIGRhdGUuXG4gICAqXG4gICAqIEB0eXBlIHtudW1iZXIgfCBEYXRlfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBtaW4/OiBudW1iZXIgfCBEYXRlO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gTWluaW11bSBhbGxvd2VkIGxlbmd0aCBmb3IgdGV4dCBpbnB1dC5cbiAgICogQHN1bW1hcnkgRm9yIHRleHQgaW5wdXRzLCB0aGlzIHNldHMgdGhlIG1pbmltdW0gbnVtYmVyIG9mIGNoYXJhY3RlcnMgcmVxdWlyZWQuXG4gICAqIFRoaXMgaXMgdXNlZCBmb3IgdmFsaWRhdGlvbiB0byBlbnN1cmUgdGhlIGlucHV0IG1lZXRzIGEgbWluaW11bSBsZW5ndGggcmVxdWlyZW1lbnQuXG4gICAqXG4gICAqIEB0eXBlIHtudW1iZXJ9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIG1pbmxlbmd0aD86IG51bWJlcjtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFZhbGlkYXRpb24gcGF0dGVybiBmb3IgdGV4dCBpbnB1dC5cbiAgICogQHN1bW1hcnkgQSByZWd1bGFyIGV4cHJlc3Npb24gcGF0dGVybiB1c2VkIHRvIHZhbGlkYXRlIHRleHQgaW5wdXQuXG4gICAqIFRoZSBpbnB1dCB2YWx1ZSBtdXN0IG1hdGNoIHRoaXMgcGF0dGVybiB0byBiZSBjb25zaWRlcmVkIHZhbGlkLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBwYXR0ZXJuPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gV2hldGhlciB0aGUgZmllbGQgaXMgcmVhZC1vbmx5LlxuICAgKiBAc3VtbWFyeSBXaGVuIHRydWUsIHRoZSBmaWVsZCB3aWxsIGJlIHJlbmRlcmVkIGluIGEgcmVhZC1vbmx5IHN0YXRlLiBVbmxpa2UgZGlzYWJsZWQgZmllbGRzLFxuICAgKiByZWFkLW9ubHkgZmllbGRzIGFyZSBzdGlsbCBmb2N1c2FibGUgYnV0IGNhbm5vdCBiZSBtb2RpZmllZCBieSB0aGUgdXNlci5cbiAgICpcbiAgICogQHR5cGUge2Jvb2xlYW59XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIHJlYWRvbmx5PzogYm9vbGVhbjtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFdoZXRoZXIgdGhlIGZpZWxkIGlzIHJlcXVpcmVkLlxuICAgKiBAc3VtbWFyeSBXaGVuIHRydWUsIHRoZSBmaWVsZCBtdXN0IGhhdmUgYSB2YWx1ZSBmb3IgdGhlIGZvcm0gdG8gYmUgdmFsaWQuXG4gICAqIFJlcXVpcmVkIGZpZWxkcyBhcmUgdHlwaWNhbGx5IG1hcmtlZCB3aXRoIGFuIGluZGljYXRvciBpbiB0aGUgVUkuXG4gICAqXG4gICAqIEB0eXBlIHtib29sZWFufVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSByZXF1aXJlZD86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBTdGVwIHZhbHVlIGZvciBudW1iZXIgaW5wdXRzLlxuICAgKiBAc3VtbWFyeSBGb3IgbnVtYmVyIGlucHV0cywgdGhpcyBzZXRzIHRoZSBpbmNyZW1lbnQvZGVjcmVtZW50IHN0ZXAgd2hlbiB1c2luZ1xuICAgKiB0aGUgdXAvZG93biBhcnJvd3Mgb3Igd2hlbiB1c2luZyBhIHJhbmdlIHNsaWRlci5cbiAgICpcbiAgICogQHR5cGUge251bWJlcn1cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgb3ZlcnJpZGUgc3RlcD86IG51bWJlcjtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEZpZWxkIG5hbWUgZm9yIGVxdWFscyBjb21wYXJpc29uLlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBlcXVhbHM/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBGaWVsZCBuYW1lIGZvciBkaWZmZXJlbnQgY29tcGFyaXNvbi5cbiAgICogQHR5cGUge3N0cmluZ31cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgb3ZlcnJpZGUgZGlmZmVyZW50Pzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gRmllbGQgbmFtZSBmb3IgbGVzcyB0aGFuIGNvbXBhcmlzb24uXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIGxlc3NUaGFuPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gRmllbGQgbmFtZSBmb3IgbGVzcyB0aGFuIG9yIGVxdWFsIGNvbXBhcmlzb24uXG4gICAqIEB0eXBlIHtzdHJpbmd9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG92ZXJyaWRlIGxlc3NUaGFuT3JFcXVhbD86IHN0cmluZztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEZpZWxkIG5hbWUgZm9yIGdyZWF0ZXIgdGhhbiBjb21wYXJpc29uLlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBncmVhdGVyVGhhbj86IHN0cmluZztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEZpZWxkIG5hbWUgZm9yIGdyZWF0ZXIgdGhhbiBvciBlcXVhbCBjb21wYXJpc29uLlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBncmVhdGVyVGhhbk9yRXF1YWw/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBOdW1iZXIgb2YgY29sdW1ucyBmb3IgdGV4dGFyZWEgaW5wdXRzLlxuICAgKiBAc3VtbWFyeSBGb3IgdGV4dGFyZWEgaW5wdXRzLCB0aGlzIHNldHMgdGhlIHZpc2libGUgd2lkdGggb2YgdGhlIHRleHQgYXJlYSBpbiBhdmVyYWdlIGNoYXJhY3RlciB3aWR0aHMuXG4gICAqIFRoaXMgaXMgdXNlZCBhbG9uZ3NpZGUgcm93cyB0byBkZWZpbmUgdGhlIGRpbWVuc2lvbnMgb2YgdGhlIHRleHRhcmVhLlxuICAgKlxuICAgKiBAdHlwZSB7bnVtYmVyfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBjb2xzPzogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gTnVtYmVyIG9mIHJvd3MgZm9yIHRleHRhcmVhIGlucHV0cy5cbiAgICogQHN1bW1hcnkgRm9yIHRleHRhcmVhIGlucHV0cywgdGhpcyBzZXRzIHRoZSB2aXNpYmxlIGhlaWdodCBvZiB0aGUgdGV4dCBhcmVhIGluIGxpbmVzIG9mIHRleHQuXG4gICAqIFRoaXMgaXMgdXNlZCBhbG9uZ3NpZGUgY29scyB0byBkZWZpbmUgdGhlIGRpbWVuc2lvbnMgb2YgdGhlIHRleHRhcmVhLlxuICAgKlxuICAgKiBAdHlwZSB7bnVtYmVyfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICByb3dzPzogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gQWxpZ25tZW50IG9mIHRoZSBmaWVsZCBjb250ZW50LlxuICAgKiBAc3VtbWFyeSBDb250cm9scyB0aGUgaG9yaXpvbnRhbCBhbGlnbm1lbnQgb2YgdGhlIGZpZWxkIGNvbnRlbnQuXG4gICAqIFRoaXMgYWZmZWN0cyBob3cgdGhlIGNvbnRlbnQgaXMgcG9zaXRpb25lZCB3aXRoaW4gdGhlIGZpZWxkIGNvbnRhaW5lci5cbiAgICpcbiAgICogQHR5cGUgeydzdGFydCcgfCAnY2VudGVyJ31cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgYWxpZ25tZW50PzogJ3N0YXJ0JyB8ICdjZW50ZXInO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gSW5pdGlhbCBjaGVja2VkIHN0YXRlIGZvciBjaGVja2JveCBvciB0b2dnbGUgaW5wdXRzLlxuICAgKiBAc3VtbWFyeSBGb3IgY2hlY2tib3ggb3IgdG9nZ2xlIGlucHV0cywgdGhpcyBzZXRzIHRoZSBpbml0aWFsIGNoZWNrZWQgc3RhdGUuXG4gICAqIFdoZW4gdHJ1ZSwgdGhlIGNoZWNrYm94IG9yIHRvZ2dsZSB3aWxsIGJlIGluaXRpYWxseSBjaGVja2VkLlxuICAgKlxuICAgKiBAdHlwZSB7Ym9vbGVhbn1cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgY2hlY2tlZD86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBKdXN0aWZpY2F0aW9uIG9mIGl0ZW1zIHdpdGhpbiB0aGUgZmllbGQuXG4gICAqIEBzdW1tYXJ5IENvbnRyb2xzIGhvdyBpdGVtcyBhcmUganVzdGlmaWVkIHdpdGhpbiB0aGUgZmllbGQgY29udGFpbmVyLlxuICAgKiBUaGlzIGlzIHBhcnRpY3VsYXJseSB1c2VmdWwgZm9yIGZpZWxkcyB3aXRoIG11bHRpcGxlIGVsZW1lbnRzLCBzdWNoIGFzIHJhZGlvIGdyb3Vwcy5cbiAgICpcbiAgICogQHR5cGUgeydzdGFydCcgfCAnZW5kJyB8ICdzcGFjZS1iZXR3ZWVuJ31cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAganVzdGlmeT86ICdzdGFydCcgfCAnZW5kJyB8ICdzcGFjZS1iZXR3ZWVuJztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFRleHQgZm9yIHRoZSBjYW5jZWwgYnV0dG9uIGluIHNlbGVjdCBpbnB1dHMuXG4gICAqIEBzdW1tYXJ5IEZvciBzZWxlY3QgaW5wdXRzIHdpdGggYSBjYW5jZWwgYnV0dG9uLCB0aGlzIHNldHMgdGhlIHRleHQgZGlzcGxheWVkIG9uIHRoZSBjYW5jZWwgYnV0dG9uLlxuICAgKiBUaGlzIGlzIHR5cGljYWxseSB1c2VkIGluIHNlbGVjdCBkaWFsb2dzIHRvIHByb3ZpZGUgYSB3YXkgZm9yIHVzZXJzIHRvIGRpc21pc3MgdGhlIHNlbGVjdGlvbiB3aXRob3V0IG1ha2luZyBhIGNoYW5nZS5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZ31cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgY2FuY2VsVGV4dD86IHN0cmluZztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEludGVyZmFjZSBzdHlsZSBmb3Igc2VsZWN0IGlucHV0cy5cbiAgICogQHN1bW1hcnkgU3BlY2lmaWVzIHRoZSBpbnRlcmZhY2Ugc3R5bGUgZm9yIHNlbGVjdCBpbnB1dHMsIHN1Y2ggYXMgJ2FsZXJ0JywgJ2FjdGlvbi1zaGVldCcsIG9yICdwb3BvdmVyJy5cbiAgICogVGhpcyBkZXRlcm1pbmVzIGhvdyB0aGUgc2VsZWN0IG9wdGlvbnMgYXJlIHByZXNlbnRlZCB0byB0aGUgdXNlci5cbiAgICpcbiAgICogQHR5cGUge1NlbGVjdEludGVyZmFjZX1cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgaW50ZXJmYWNlPzogU2VsZWN0SW50ZXJmYWNlO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gT3B0aW9ucyBmb3Igc2VsZWN0IG9yIHJhZGlvIGlucHV0cy5cbiAgICogQHN1bW1hcnkgUHJvdmlkZXMgdGhlIGxpc3Qgb2Ygb3B0aW9ucyBmb3Igc2VsZWN0IG9yIHJhZGlvIGlucHV0cy4gRWFjaCBvcHRpb24gY2FuIGhhdmUgYSB2YWx1ZSBhbmQgYSBsYWJlbC5cbiAgICogVGhpcyBpcyB1c2VkIHRvIHBvcHVsYXRlIHRoZSBkcm9wZG93biBvciByYWRpbyBncm91cCB3aXRoIGNob2ljZXMuXG4gICAqXG4gICAqIEB0eXBlIHtTZWxlY3RPcHRpb25bXSB8IFJhZGlvT3B0aW9uW119XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG9wdGlvbnMhOiBTZWxlY3RPcHRpb25bXSB8IFJhZGlvT3B0aW9uW107XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBNb2RlIG9mIHRoZSBmaWVsZC5cbiAgICogQHN1bW1hcnkgU3BlY2lmaWVzIHRoZSB2aXN1YWwgbW9kZSBvZiB0aGUgZmllbGQsIHN1Y2ggYXMgJ2lvcycgb3IgJ21kJy5cbiAgICogVGhpcyBhZmZlY3RzIHRoZSBzdHlsaW5nIGFuZCBhcHBlYXJhbmNlIG9mIHRoZSBmaWVsZCB0byBtYXRjaCB0aGUgcGxhdGZvcm0gc3R5bGUuXG4gICAqXG4gICAqIEB0eXBlIHsnaW9zJyB8ICdtZCd9XG4gICAqIEBtZW1iZXJPZiBDcnVkRmllbGRDb21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIG1vZGU/OiAnaW9zJyB8ICdtZCc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBTcGVsbGNoZWNrIGF0dHJpYnV0ZSBmb3IgdGV4dCBpbnB1dHMuXG4gICAqIEBzdW1tYXJ5IEVuYWJsZXMgb3IgZGlzYWJsZXMgc3BlbGxjaGVja2luZyBmb3IgdGV4dCBpbnB1dHMuXG4gICAqIFdoZW4gdHJ1ZSwgdGhlIGJyb3dzZXIgd2lsbCBjaGVjayB0aGUgc3BlbGxpbmcgb2YgdGhlIGlucHV0IHRleHQuXG4gICAqXG4gICAqIEB0eXBlIHtib29sZWFufVxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBzcGVsbGNoZWNrID0gZmFsc2U7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBJbnB1dCBtb2RlIGZvciB0ZXh0IGlucHV0cy5cbiAgICogQHN1bW1hcnkgSGludHMgYXQgdGhlIHR5cGUgb2YgZGF0YSB0aGF0IG1pZ2h0IGJlIGVudGVyZWQgYnkgdGhlIHVzZXIgd2hpbGUgZWRpdGluZyB0aGUgZWxlbWVudC5cbiAgICogVGhpcyBjYW4gYWZmZWN0IHRoZSB2aXJ0dWFsIGtleWJvYXJkIGxheW91dCBvbiBtb2JpbGUgZGV2aWNlcy5cbiAgICpcbiAgICogQHR5cGUgeydub25lJyB8ICd0ZXh0JyB8ICd0ZWwnIHwgJ3VybCcgfCAnZW1haWwnIHwgJ251bWVyaWMnIHwgJ2RlY2ltYWwnIHwgJ3NlYXJjaCd9XG4gICAqIEBkZWZhdWx0ICdub25lJ1xuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBpbnB1dG1vZGU6XG4gICAgfCAnbm9uZSdcbiAgICB8ICd0ZXh0J1xuICAgIHwgJ3RlbCdcbiAgICB8ICd1cmwnXG4gICAgfCAnZW1haWwnXG4gICAgfCAnbnVtZXJpYydcbiAgICB8ICdkZWNpbWFsJ1xuICAgIHwgJ3NlYXJjaCcgPSAnbm9uZSc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBBdXRvY29tcGxldGUgYmVoYXZpb3IgZm9yIHRoZSBmaWVsZC5cbiAgICogQHN1bW1hcnkgU3BlY2lmaWVzIHdoZXRoZXIgYW5kIGhvdyB0aGUgYnJvd3NlciBzaG91bGQgYXV0b21hdGljYWxseSBjb21wbGV0ZSB0aGUgaW5wdXQuXG4gICAqIFRoaXMgY2FuIGltcHJvdmUgdXNlciBleHBlcmllbmNlIGJ5IHN1Z2dlc3RpbmcgcHJldmlvdXNseSBlbnRlcmVkIHZhbHVlcy5cbiAgICpcbiAgICogQHR5cGUge0F1dG9jb21wbGV0ZVR5cGVzfVxuICAgKiBAZGVmYXVsdCAnb2ZmJ1xuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBhdXRvY29tcGxldGU6IEF1dG9jb21wbGV0ZVR5cGVzID0gJ29mZic7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBGaWxsIHN0eWxlIGZvciB0aGUgZmllbGQuXG4gICAqIEBzdW1tYXJ5IERldGVybWluZXMgdGhlIGZpbGwgc3R5bGUgb2YgdGhlIGZpZWxkLCBzdWNoIGFzICdvdXRsaW5lJyBvciAnc29saWQnLlxuICAgKiBUaGlzIGFmZmVjdHMgdGhlIGJvcmRlciBhbmQgYmFja2dyb3VuZCBvZiB0aGUgZmllbGQuXG4gICAqXG4gICAqIEB0eXBlIHsnb3V0bGluZScgfCAnc29saWQnfVxuICAgKiBAZGVmYXVsdCAnb3V0bGluZSdcbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQElucHV0KClcbiAgZmlsbDogJ291dGxpbmUnIHwgJ3NvbGlkJyA9ICdvdXRsaW5lJztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFBsYWNlbWVudCBvZiB0aGUgbGFiZWwgcmVsYXRpdmUgdG8gdGhlIGZpZWxkLlxuICAgKiBAc3VtbWFyeSBTcGVjaWZpZXMgd2hlcmUgdGhlIGxhYmVsIHNob3VsZCBiZSBwbGFjZWQgcmVsYXRpdmUgdG8gdGhlIGZpZWxkLlxuICAgKiBPcHRpb25zIGluY2x1ZGUgJ3N0YXJ0JywgJ2VuZCcsICdmbG9hdGluZycsICdzdGFja2VkJywgYW5kICdmaXhlZCcuXG4gICAqXG4gICAqIEB0eXBlIHsnc3RhcnQnIHwgJ2VuZCcgfCAnZmxvYXRpbmcnIHwgJ3N0YWNrZWQnIHwgJ2ZpeGVkJ31cbiAgICogQGRlZmF1bHQgJ2Zsb2F0aW5nJ1xuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBsYWJlbFBsYWNlbWVudDogJ3N0YXJ0JyB8ICdlbmQnIHwgJ2Zsb2F0aW5nJyB8ICdzdGFja2VkJyB8ICdmaXhlZCcgPVxuICAgICdmbG9hdGluZyc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBVcGRhdGUgbW9kZSBmb3IgdGhlIGZpZWxkLlxuICAgKiBAc3VtbWFyeSBEZXRlcm1pbmVzIHdoZW4gdGhlIGZpZWxkIHZhbHVlIHNob3VsZCBiZSB1cGRhdGVkIGluIHRoZSBmb3JtIG1vZGVsLlxuICAgKiBPcHRpb25zIGluY2x1ZGUgJ2NoYW5nZScsICdibHVyJywgYW5kICdzdWJtaXQnLlxuICAgKlxuICAgKiBAdHlwZSB7RmllbGRVcGRhdGVNb2RlfVxuICAgKiBAZGVmYXVsdCAnY2hhbmdlJ1xuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICB1cGRhdGVPbjogRmllbGRVcGRhdGVNb2RlID0gJ2NoYW5nZSc7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBSZWZlcmVuY2UgdG8gdGhlIGZpZWxkIGNvbXBvbmVudC5cbiAgICogQHN1bW1hcnkgUHJvdmlkZXMgYSByZWZlcmVuY2UgdG8gdGhlIGZpZWxkIGNvbXBvbmVudCBlbGVtZW50LCBhbGxvd2luZyBkaXJlY3QgYWNjZXNzIHRvIGl0cyBwcm9wZXJ0aWVzIGFuZCBtZXRob2RzLlxuICAgKlxuICAgKiBAdHlwZSB7RWxlbWVudFJlZn1cbiAgICogQG1lbWJlck9mIENydWRGaWVsZENvbXBvbmVudFxuICAgKi9cbiAgQFZpZXdDaGlsZCgnY29tcG9uZW50JywgeyByZWFkOiBFbGVtZW50UmVmIH0pXG4gIG92ZXJyaWRlIGNvbXBvbmVudCE6IEVsZW1lbnRSZWY7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBQYXJlbnQgZm9ybSBncm91cC5cbiAgICogQHN1bW1hcnkgUmVmZXJlbmNlcyB0aGUgcGFyZW50IGZvcm0gZ3JvdXAgdG8gd2hpY2ggdGhpcyBmaWVsZCBiZWxvbmdzLlxuICAgKiBUaGlzIGlzIG5lY2Vzc2FyeSBmb3IgaW50ZWdyYXRpbmcgdGhlIGZpZWxkIHdpdGggQW5ndWxhcidzIHJlYWN0aXZlIGZvcm1zLlxuICAgKlxuICAgKiBAdHlwZSB7Rm9ybUdyb3VwfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICBvdmVycmlkZSBmb3JtR3JvdXA6IEZvcm1Hcm91cCB8IHVuZGVmaW5lZDtcblxuICBASW5wdXQoKVxuICBvdmVycmlkZSBmb3JtQ29udHJvbCE6IEZvcm1Db250cm9sO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gVHJhbnNsYXRhYmlsaXR5IG9mIGZpZWxkIGxhYmVscy5cbiAgICogQHN1bW1hcnkgSW5kaWNhdGVzIHdoZXRoZXIgdGhlIGZpZWxkIGxhYmVscyBzaG91bGQgYmUgdHJhbnNsYXRlZCBiYXNlZCBvbiB0aGUgY3VycmVudCBsYW5ndWFnZSBzZXR0aW5ncy5cbiAgICogVGhpcyBpcyB1c2VmdWwgZm9yIGFwcGxpY2F0aW9ucyBzdXBwb3J0aW5nIG11bHRpcGxlIGxhbmd1YWdlcy5cbiAgICpcbiAgICogQHR5cGUge1N0cmluZ09yQm9vbGVhbn1cbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZpZWxkQ29tcG9uZW50XG4gICAqL1xuICBASW5wdXQoKVxuICB0cmFuc2xhdGFibGU6IFN0cmluZ09yQm9vbGVhbiA9IHRydWU7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBVbmlxdWUgaWRlbnRpZmllciBmb3IgdGhlIGN1cnJlbnQgcmVjb3JkLlxuICAgKiBAc3VtbWFyeSBBIHVuaXF1ZSBpZGVudGlmaWVyIGZvciB0aGUgY3VycmVudCByZWNvcmQgYmVpbmcgZGlzcGxheWVkIG9yIG1hbmlwdWxhdGVkLlxuICAgKiBUaGlzIGlzIHR5cGljYWxseSB1c2VkIGluIGNvbmp1bmN0aW9uIHdpdGggdGhlIHByaW1hcnkga2V5IGZvciBvcGVyYXRpb25zIG9uIHNwZWNpZmljIHJlY29yZHMuXG4gICAqXG4gICAqIEB0eXBlIHtzdHJpbmcgfCBudW1iZXJ9XG4gICAqL1xuICBASW5wdXQoKVxuICB1aWQhOiBzdHJpbmcgfCBudW1iZXIgfCB1bmRlZmluZWQ7XG5cbiAgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgLy8gc3VwZXIub25Jbml0KHRoaXMudXBkYXRlT24pO1xuICAgIGlmIChbT3BlcmF0aW9uS2V5cy5SRUFELCBPcGVyYXRpb25LZXlzLkRFTEVURV0uaW5jbHVkZXModGhpcy5vcGVyYXRpb24pKSB7XG4gICAgICB0aGlzLmZvcm1Hcm91cCA9IHVuZGVmaW5lZDtcbiAgICB9IGVsc2Uge1xuICAgICAgaWYgKHRoaXMudHlwZSA9PT0gSFRNTDVJbnB1dFR5cGVzLlJBRElPICYmICF0aGlzLnZhbHVlKVxuICAgICAgICB0aGlzLmZvcm1Hcm91cD8uZ2V0KHRoaXMubmFtZSk/LnNldFZhbHVlKHRoaXMub3B0aW9uc1swXS52YWx1ZSk7IC8vIFRPRE86IG1pZ3JhdGUgdG8gUmVuZGVyaW5nRW5naW5lXG4gICAgfVxuXG4gIH1cblxuICBuZ0FmdGVyVmlld0luaXQoKSB7XG4gICAgaWYgKFtPcGVyYXRpb25LZXlzLlJFQUQsIE9wZXJhdGlvbktleXMuREVMRVRFXS5pbmNsdWRlcyh0aGlzLm9wZXJhdGlvbikpXG4gICAgICBzdXBlci5hZnRlclZpZXdJbml0KCk7XG4gIH1cblxuICBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICBpZiAoW09wZXJhdGlvbktleXMuUkVBRCwgT3BlcmF0aW9uS2V5cy5ERUxFVEVdLmluY2x1ZGVzKHRoaXMub3BlcmF0aW9uKSlcbiAgICAgIHRoaXMub25EZXN0cm95KCk7XG4gIH1cbn1cbiIsIkBpZihvcGVyYXRpb24gPT09ICdyZWFkJyB8fCBvcGVyYXRpb24gPT09ICdkZWxldGUnKSB7XG4gIDxuZy1jb250YWluZXIgI2NvbXBvbmVudD5cbiAgICA8ZGl2IFtjbGFzc109XCInZGNmLWlucHV0LWl0ZW0gJyArIG9wZXJhdGlvblwiPlxuICAgICAgPGlvbi1pdGVtPlxuICAgICAgICA8aW9uLWxhYmVsPlxuICAgICAgICAgIHt7IGxhYmVsIHwgdHJhbnNsYXRlIH19PGJyIC8+XG4gICAgICAgICAgQGlmKHZhbHVlKSB7XG4gICAgICAgICAgICA8aW9uLXRleHQgY2xhc3M9XCJkY2YtZGlzcGxheS1ibG9ja1wiIFtpbm5lckhUTUxdPVwidmFsdWVcIj48L2lvbi10ZXh0PlxuICAgICAgICAgIH0gQGVsc2Uge1xuICAgICAgICAgICAgPGJyIC8+XG4gICAgICAgICAgfVxuICAgICAgICA8L2lvbi1sYWJlbD5cbiAgICAgIDwvaW9uLWl0ZW0+XG4gICAgPC9kaXY+XG4gIDwvbmctY29udGFpbmVyPlxufSBAZWxzZSB7XG4gIDxuZy1jb250YWluZXIgI2NvbXBvbmVudCBbZm9ybUdyb3VwXT1cImZvcm1Hcm91cFwiPlxuICAgIDxkaXYgI2NvbnRhaW5lciBbY2xhc3NdPVwiJ2RjZi1pbnB1dC1pdGVtICcgKyAob3BlcmF0aW9uIHx8ICdjcmVhdGUnKVwiPlxuICAgICAgICBAaWYodHlwZSA9PT0gJ3RleHRhcmVhJykge1xuICAgICAgICAgIDxpb24tdGV4dGFyZWFcbiAgICAgICAgICAgIFttb2RlXT1cIm1vZGVcIlxuICAgICAgICAgICAgW2hpZGRlbl09XCJoaWRkZW5cIlxuICAgICAgICAgICAgW3JlcXVpcmVkXT1cInJlcXVpcmVkICE9PSB1bmRlZmluZWQgPyByZXF1aXJlZCA6IG51bGxcIlxuICAgICAgICAgICAgW21pbmxlbmd0aF09XCJtaW5sZW5ndGggIT09IHVuZGVmaW5lZCA/IG1pbmxlbmd0aCA6IG51bGxcIlxuICAgICAgICAgICAgW21heGxlbmd0aF09XCJtYXhsZW5ndGggIT09IHVuZGVmaW5lZCA/IG1heGxlbmd0aCA6IG51bGxcIlxuICAgICAgICAgICAgW3JlYWRvbmx5XT1cInJlYWRvbmx5ICE9PSB1bmRlZmluZWQgPyByZWFkb25seSA6IG51bGxcIlxuICAgICAgICAgICAgW2lucHV0bW9kZV09XCJpbnB1dG1vZGVcIlxuICAgICAgICAgICAgW3NwZWxsY2hlY2tdPVwic3BlbGxjaGVja1wiXG4gICAgICAgICAgICBbcm93c109XCJyb3dzXCJcbiAgICAgICAgICAgIFtsYWJlbFBsYWNlbWVudF09XCJsYWJlbFBsYWNlbWVudFwiXG4gICAgICAgICAgICBbdmFsdWVdPVwidmFsdWVcIlxuICAgICAgICAgICAgW2ZpbGxdPVwiZmlsbFwiXG4gICAgICAgICAgICBbcGxhY2Vob2xkZXJdPVwidHJhbnNsYXRhYmxlID8gKHBsYWNlaG9sZGVyIHwgdHJhbnNsYXRlKSA6IHBsYWNlaG9sZGVyXCJcbiAgICAgICAgICAgIFtmb3JtQ29udHJvbE5hbWVdPVwibmFtZVwiXG4gICAgICAgICAgICBbbGFiZWxdPVwidHJhbnNsYXRhYmxlID8gKGxhYmVsIHwgdHJhbnNsYXRlKSA6IGxhYmVsXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgPC9pb24tdGV4dGFyZWE+XG4gICAgICAgIH1cbiAgICAgICAgQGVsc2UgaWYodHlwZSA9PT0gJ2NoZWNrYm94Jykge1xuICAgICAgICAgIDxpb24taXRlbT5cbiAgICAgICAgICAgIDxpb24tY2hlY2tib3hcbiAgICAgICAgICAgICAgI2NoZWNrYm94RWxlbWVudFxuICAgICAgICAgICAgICBbbmFtZV09XCJwYXRoXCJcbiAgICAgICAgICAgICAgW21vZGVdPVwibW9kZVwiXG4gICAgICAgICAgICAgIFtoaWRkZW5dPVwiaGlkZGVuXCJcbiAgICAgICAgICAgICAgW2xhYmVsUGxhY2VtZW50XT1cImxhYmVsUGxhY2VtZW50XCJcbiAgICAgICAgICAgICAgW2p1c3RpZnldPVwianVzdGlmeVwiXG4gICAgICAgICAgICAgIFt2YWx1ZV09XCJ2YWx1ZVwiXG4gICAgICAgICAgICAgIFtjaGVja2VkXT1cImNoZWNrZWRcIlxuICAgICAgICAgICAgICBbcmVhZG9ubHldPVwicmVhZG9ubHlcIlxuICAgICAgICAgICAgICAoaW9uQ2hhbmdlKT1cImNoZWNrZWQgPSAhY2hlY2tlZFwiXG4gICAgICAgICAgICAgIFtmb3JtQ29udHJvbE5hbWVdPVwibmFtZVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIDxzcGFuIFtpbm5lckhUTUxdPVwibGFiZWwgfCB0cmFuc2xhdGVcIj48L3NwYW4+XG4gICAgICAgICAgICA8L2lvbi1jaGVja2JveD5cbiAgICAgICAgICA8L2lvbi1pdGVtPlxuICAgICAgICB9XG4gICAgICAgIEBlbHNlIGlmKHR5cGUgPT09ICdyYWRpbycpIHtcbiAgICAgICAgICA8aW9uLXJhZGlvLWdyb3VwIFtmb3JtQ29udHJvbE5hbWVdPVwibmFtZVwiIFtuYW1lXT1cInBhdGhcIiBbdmFsdWVdPVwidmFsdWVcIiAjY29tcG9uZW50PlxuICAgICAgICAgICAgPGxhYmVsIGNsYXNzPVwiZGNmLXJhZGlvLWdyb3VwLWxhYmVsXCIgW2Zvcl09XCJwYXRoXCI+e3tsYWJlbCB8IHRyYW5zbGF0ZX19PC9sYWJlbD5cbiAgICAgICAgICAgIEBmb3Iob3B0aW9uIG9mIG9wdGlvbnM7IHRyYWNrIG9wdGlvbikge1xuICAgICAgICAgICAgICA8aW9uLWl0ZW0+XG4gICAgICAgICAgICAgICAgPGlvbi1yYWRpb1xuICAgICAgICAgICAgICAgICAgW21vZGVdPVwibW9kZVwiXG4gICAgICAgICAgICAgICAgICBbaGlkZGVuXT1cImhpZGRlblwiXG4gICAgICAgICAgICAgICAgICBbbGFiZWxQbGFjZW1lbnRdPVwibGFiZWxQbGFjZW1lbnRcIlxuICAgICAgICAgICAgICAgICAgW2FsaWdubWVudF09XCJhbGlnbm1lbnRcIlxuICAgICAgICAgICAgICAgICAgW2p1c3RpZnldPVwianVzdGlmeVwiXG4gICAgICAgICAgICAgICAgICBbcmVhZG9ubHldPVwicmVhZG9ubHlcIlxuICAgICAgICAgICAgICAgICAgW3ZhbHVlXT1cIm9wdGlvbi52YWx1ZVwiXG4gICAgICAgICAgICAgICAgPnt7IHRyYW5zbGF0YWJsZSA/ICAob3B0aW9uPy50ZXh0IHwgdHJhbnNsYXRlKSA6IG9wdGlvbj8udGV4dCB9fTwvaW9uLXJhZGlvPlxuICAgICAgICAgICAgICA8L2lvbi1pdGVtPlxuICAgICAgICAgICAgfVxuICAgICAgICAgIDwvaW9uLXJhZGlvLWdyb3VwPlxuICAgICAgICB9XG4gICAgICAgIEBlbHNlIGlmKHR5cGUgPT09ICdzZWxlY3QnKSB7XG4gICAgICAgICAgPGlvbi1pdGVtPlxuICAgICAgICAgICAgPGlvbi1zZWxlY3RcbiAgICAgICAgICAgICAgW25hbWVdPVwicGF0aFwiXG4gICAgICAgICAgICAgIFttb2RlXT1cIm1vZGVcIlxuICAgICAgICAgICAgICBbaGlkZGVuXT1cImhpZGRlblwiXG4gICAgICAgICAgICAgIFtsYWJlbFBsYWNlbWVudF09XCJsYWJlbFBsYWNlbWVudFwiXG4gICAgICAgICAgICAgIFtsYWJlbF09XCJ0cmFuc2xhdGFibGUgPyAobGFiZWwgfCB0cmFuc2xhdGUgOiBsYWJlbCkgOiBsYWJlbFwiXG4gICAgICAgICAgICAgIFt2YWx1ZV09XCJ2YWx1ZVwiXG4gICAgICAgICAgICAgIFtmaWxsXT1cImZpbGxcIlxuICAgICAgICAgICAgICBbcGxhY2Vob2xkZXJdPVwidHJhbnNsYXRhYmxlID8gKHBsYWNlaG9sZGVyIHwgdHJhbnNsYXRlKSA6IHBsYWNlaG9sZGVyXCJcbiAgICAgICAgICAgICAgW2Zvcm1Db250cm9sTmFtZV09XCJuYW1lXCJcbiAgICAgICAgICAgICAgW2ludGVyZmFjZV09XCJpbnRlcmZhY2VcIj5cbiAgICAgICAgICAgICAgQGZvcihvcHRpb24gb2Ygb3B0aW9uczsgdHJhY2sgb3B0aW9uLnZhbHVlKSB7XG4gICAgICAgICAgICAgICAgPGlvbi1zZWxlY3Qtb3B0aW9uIFt2YWx1ZV09XCJvcHRpb24udmFsdWVcIj5cbiAgICAgICAgICAgICAgICAgIHt7IHRyYW5zbGF0YWJsZSA/IChvcHRpb24udGV4dCB8IHRyYW5zbGF0ZSkgOiBvcHRpb25zLnRleHQgfX1cbiAgICAgICAgICAgICAgICA8L2lvbi1zZWxlY3Qtb3B0aW9uPlxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICA8L2lvbi1zZWxlY3Q+XG4gICAgICAgICAgPC9pb24taXRlbT5cbiAgICAgICAgfVxuICAgICAgICBAZWxzZSB7XG4gICAgICAgICAgPGlvbi1pbnB1dFxuICAgICAgICAgICAgW25hbWVdPVwicGF0aFwiXG4gICAgICAgICAgICBbdHlwZV09XCJ0eXBlXCJcbiAgICAgICAgICAgIFttb2RlXT1cIm1vZGVcIlxuICAgICAgICAgICAgW2hpZGRlbl09XCJoaWRkZW5cIlxuICAgICAgICAgICAgW2lucHV0bW9kZV09XCJpbnB1dG1vZGVcIlxuICAgICAgICAgICAgW2xhYmVsUGxhY2VtZW50XT1cImxhYmVsUGxhY2VtZW50XCJcbiAgICAgICAgICAgIFtyZXF1aXJlZF09XCJyZXF1aXJlZCAhPT0gdW5kZWZpbmVkID8gcmVxdWlyZWQgOiBmYWxzZVwiXG4gICAgICAgICAgICBbbWlubGVuZ3RoXT1cIm1pbmxlbmd0aCAhPT0gdW5kZWZpbmVkID8gbWlubGVuZ3RoIDogbnVsbFwiXG4gICAgICAgICAgICBbbWF4bGVuZ3RoXT1cIm1heGxlbmd0aCAhPT0gdW5kZWZpbmVkID8gbWF4bGVuZ3RoIDogbnVsbFwiXG4gICAgICAgICAgICBbcmVhZG9ubHldPVwicmVhZG9ubHkgIT09IHVuZGVmaW5lZCA/IHJlYWRvbmx5IDogbnVsbFwiXG4gICAgICAgICAgICBbbWF4XT1cIm1heCAhPT0gdW5kZWZpbmVkID8gbWF4IDogbnVsbFwiXG4gICAgICAgICAgICBbbWluXT1cIm1pbiAhPT0gdW5kZWZpbmVkID8gbWluIDogbnVsbFwiXG4gICAgICAgICAgICBbcGF0dGVybl09XCJwYXR0ZXJuICE9PSB1bmRlZmluZWQgPyBwYXR0ZXJuIDogbnVsbFwiXG4gICAgICAgICAgICBbc3RlcF09XCJzdGVwICE9PSB1bmRlZmluZWQgPyBzdGVwIDogbnVsbFwiXG4gICAgICAgICAgICBbZmlsbF09XCJmaWxsXCJcbiAgICAgICAgICAgIFtwbGFjZWhvbGRlcl09XCJ0cmFuc2xhdGFibGUgPyAocGxhY2Vob2xkZXIgfCB0cmFuc2xhdGUpIDogcGxhY2Vob2xkZXJcIlxuICAgICAgICAgICAgW2Zvcm1Db250cm9sTmFtZV09XCJuYW1lXCJcbiAgICAgICAgICAgIFtsYWJlbF09XCJ0cmFuc2xhdGFibGUgPyAobGFiZWwgfCB0cmFuc2xhdGUpIDogbGFiZWxcIj5cbiAgICAgICAgICA8L2lvbi1pbnB1dD5cbiAgICAgICAgfVxuICAgICAgICBAaWYoKCFmb3JtQ29udHJvbC5wcmlzdGluZSB8fCBmb3JtQ29udHJvbC50b3VjaGVkKSAmJiAhZm9ybUNvbnRyb2wudmFsaWQpIHtcbiAgICAgICAgICA8ZGl2IGNsYXNzPVwiZXJyb3IgZGNmLWVycm9yIGRjZi1mbGV4IGRjZi1mbGV4LXRvcFwiPlxuICAgICAgICAgICAgQGZvcihlcnJvciBvZiBnZXRFcnJvcnMoY29udGFpbmVyKTsgdHJhY2sgZXJyb3Iua2V5KSB7XG4gICAgICAgICAgICAgICoge3sgc2YoKFwiZXJyb3JzLlwiICsgZXJyb3IubWVzc2FnZSkgfCB0cmFuc2xhdGUsIHRoaXNbZXJyb3Iua2V5XSkgfX1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgIDwvZGl2PlxuICA8L25nLWNvbnRhaW5lcj5cbn1cblxuIl19
@@ -0,0 +1,14 @@
1
+ export const CssClasses = {
2
+ BUTTONS_CONTAINER: 'buttons-container',
3
+ };
4
+ export const DefaultFormReactiveOptions = {
5
+ buttons: {
6
+ submit: {
7
+ text: 'Submit',
8
+ },
9
+ clear: {
10
+ text: 'Clear',
11
+ },
12
+ },
13
+ };
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL2NydWQtZm9ybS9jb25zdGFudHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxDQUFDLE1BQU0sVUFBVSxHQUFHO0lBQ3hCLGlCQUFpQixFQUFFLG1CQUFtQjtDQUN2QyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sMEJBQTBCLEdBQW9CO0lBQ3pELE9BQU8sRUFBRTtRQUNQLE1BQU0sRUFBRTtZQUNOLElBQUksRUFBRSxRQUFRO1NBQ2Y7UUFDRCxLQUFLLEVBQUU7WUFDTCxJQUFJLEVBQUUsT0FBTztTQUNkO0tBQ0Y7Q0FDRixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ3J1ZEZvcm1PcHRpb25zIH0gZnJvbSAnLi90eXBlcyc7XG5cbmV4cG9ydCBjb25zdCBDc3NDbGFzc2VzID0ge1xuICBCVVRUT05TX0NPTlRBSU5FUjogJ2J1dHRvbnMtY29udGFpbmVyJyxcbn07XG5cbmV4cG9ydCBjb25zdCBEZWZhdWx0Rm9ybVJlYWN0aXZlT3B0aW9uczogQ3J1ZEZvcm1PcHRpb25zID0ge1xuICBidXR0b25zOiB7XG4gICAgc3VibWl0OiB7XG4gICAgICB0ZXh0OiAnU3VibWl0JyxcbiAgICB9LFxuICAgIGNsZWFyOiB7XG4gICAgICB0ZXh0OiAnQ2xlYXInLFxuICAgIH0sXG4gIH0sXG59O1xuIl19
@@ -0,0 +1,140 @@
1
+ import { __decorate } from "tslib";
2
+ import { Component, ElementRef, EventEmitter, inject, Input, Output, ViewChild, } from '@angular/core';
3
+ import { Location } from '@angular/common';
4
+ import { NgxFormService } from '../../engine/NgxFormService';
5
+ import { Dynamic, EventConstants } from '../../engine';
6
+ import { OperationKeys } from '@decaf-ts/db-decorators';
7
+ import { DefaultFormReactiveOptions } from './constants';
8
+ import { ForAngularModule, getLogger } from 'src/lib/for-angular.module';
9
+ import { IonIcon } from '@ionic/angular/standalone';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "@ionic/angular/standalone";
12
+ import * as i2 from "@angular/forms";
13
+ /**
14
+ * @component CrudFormComponent
15
+ * @example <ngx-decaf-crud-form
16
+ * action="create"
17
+ * operation="create"
18
+ * formGroup="formGroup"
19
+ * rendererId="rendererId"
20
+ * submitEvent="submitEvent"
21
+ * target="_self"
22
+ * method="event">
23
+ * </ngx-decaf-crud-form>
24
+ *
25
+ * @param {string} action - The action to be performed (create, read, update, delete)
26
+ * @param {CrudOperations} operation - The CRUD operation being performed (create, read, update, delete)
27
+ * @param {FormGroup} formGroup - The form group
28
+ * @param {string} rendererId - The renderer id
29
+ * @param {SubmitEvent} submitEvent - The submit event
30
+ * @param {string} target - The target
31
+ * @param {string} method - The method
32
+ */
33
+ let CrudFormComponent = class CrudFormComponent {
34
+ constructor() {
35
+ this.updateOn = 'change';
36
+ this.target = '_self';
37
+ this.method = 'event';
38
+ this.submitEvent = new EventEmitter();
39
+ /**
40
+ * @description Angular Location service.
41
+ * @summary Injected service that provides access to the browser's URL and history.
42
+ * This service is used for interacting with the browser's history API, allowing
43
+ * for back navigation and URL manipulation outside of Angular's router.
44
+ *
45
+ * @private
46
+ * @type {Location}
47
+ * @memberOf CrudFormComponent
48
+ */
49
+ this.location = inject(Location);
50
+ this.OperationKeys = OperationKeys;
51
+ }
52
+ // ngAfterViewInit() {
53
+ // if (![OperationKeys.READ, OperationKeys.DELETE].includes(this.operation))
54
+ // NgxFormService.formAfterViewInit(this, this.rendererId);
55
+ // }
56
+ async ngOnInit() {
57
+ if (!this.logger)
58
+ this.logger = getLogger(this);
59
+ if (this.operation === OperationKeys.READ || this.operation === OperationKeys.DELETE)
60
+ this.formGroup = undefined;
61
+ this.options = Object.assign({}, DefaultFormReactiveOptions, this.options || {});
62
+ }
63
+ ngOnDestroy() {
64
+ if (this.formGroup)
65
+ NgxFormService.unregister(this.formGroup);
66
+ }
67
+ /**
68
+ * @param {SubmitEvent} event
69
+ */
70
+ async submit(event) {
71
+ event.preventDefault();
72
+ event.stopImmediatePropagation();
73
+ event.stopPropagation();
74
+ if (!NgxFormService.validateFields(this.formGroup))
75
+ return false;
76
+ const data = NgxFormService.getFormData(this.formGroup);
77
+ console.log('Submit=', data);
78
+ this.submitEvent.emit({
79
+ data,
80
+ component: 'FormReactiveComponent',
81
+ name: this.action || EventConstants.SUBMIT_EVENT,
82
+ handlers: this.handlers,
83
+ });
84
+ }
85
+ handleReset() {
86
+ this.location.back();
87
+ // if(OperationKeys.DELETE !== this.operation)
88
+ // NgxFormService.reset(this.formGroup);
89
+ // else
90
+ // this.location.back();
91
+ }
92
+ handleDelete() {
93
+ this.submitEvent.emit({
94
+ data: this.uid,
95
+ component: 'FormReactiveComponent',
96
+ name: EventConstants.SUBMIT_EVENT,
97
+ });
98
+ }
99
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CrudFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
100
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CrudFormComponent, isStandalone: true, selector: "ngx-decaf-crud-form", inputs: { model: "model", updateOn: "updateOn", target: "target", method: "method", options: "options", action: "action", operation: "operation", handlers: "handlers", formGroup: "formGroup", childOf: "childOf", rendererId: "rendererId", uid: "uid" }, outputs: { submitEvent: "submitEvent" }, viewQueries: [{ propertyName: "component", first: true, predicate: ["reactiveForm"], descendants: true, read: ElementRef }], ngImport: i0, template: "@if(operation !== 'read' && operation !== 'delete') {\n <form #reactiveForm [id]=\"rendererId\" [formGroup]=\"formGroup\" (submit)=\"submit($event)\" novalidate [target]=\"target\">\n <ng-content #formContent></ng-content>\n <div class=\"dcf-flex dcf-flex-right\">\n <div>\n @if(options.buttons.clear) {\n <ion-button fill=\"clear\" (click)=\"handleReset()\">\n @if(options.buttons.clear?.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n }\n {{operation === 'update' ? 'Back' : options.buttons.clear?.text}}\n </ion-button>\n }\n\n <ion-button\n type=\"submit\">\n @if(options.buttons.submit.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n }\n {{ action ? action : options.buttons.submit.text}}\n </ion-button>\n </div>\n </div>\n </form>\n} @else {\n <div [class]=\"'dcf-flex dcf-flex-right ' + operation\" id=\"dcf-buttons-container\">\n <div>\n @if(options.buttons.clear) {\n <ion-button fill=\"clear\" (click)=\"handleReset()\">\n @if(options.buttons.clear?.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n }\n {{ ['delete', 'read', 'update'].includes(operation) ? 'Back' : options.buttons.clear?.text}}\n </ion-button>\n }\n @if(operation === 'delete' && uid) {\n <ion-button\n (click)=\"handleDelete()\"\n color=\"danger\"\n type=\"button\">\n @if(options.buttons.submit.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n }\n Delete\n </ion-button>\n }\n @if(operation === OperationKeys.CREATE || operation === OperationKeys.UPDATE) {\n <ion-button\n type=\"submit\">\n @if(options.buttons.submit.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n }\n {{options.buttons.submit.text}}\n </ion-button>\n }\n\n </div>\n </div>\n}\n\n", styles: ["#dcf-buttons-container{margin-top:1.8rem;margin-bottom:0}\n"], dependencies: [{ kind: "ngmodule", type: ForAngularModule }, { kind: "component", type: i1.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }] }); }
101
+ };
102
+ CrudFormComponent = __decorate([
103
+ Dynamic()
104
+ ], CrudFormComponent);
105
+ export { CrudFormComponent };
106
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CrudFormComponent, decorators: [{
107
+ type: Component,
108
+ args: [{ standalone: true, selector: 'ngx-decaf-crud-form', imports: [ForAngularModule, IonIcon], template: "@if(operation !== 'read' && operation !== 'delete') {\n <form #reactiveForm [id]=\"rendererId\" [formGroup]=\"formGroup\" (submit)=\"submit($event)\" novalidate [target]=\"target\">\n <ng-content #formContent></ng-content>\n <div class=\"dcf-flex dcf-flex-right\">\n <div>\n @if(options.buttons.clear) {\n <ion-button fill=\"clear\" (click)=\"handleReset()\">\n @if(options.buttons.clear?.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n }\n {{operation === 'update' ? 'Back' : options.buttons.clear?.text}}\n </ion-button>\n }\n\n <ion-button\n type=\"submit\">\n @if(options.buttons.submit.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n }\n {{ action ? action : options.buttons.submit.text}}\n </ion-button>\n </div>\n </div>\n </form>\n} @else {\n <div [class]=\"'dcf-flex dcf-flex-right ' + operation\" id=\"dcf-buttons-container\">\n <div>\n @if(options.buttons.clear) {\n <ion-button fill=\"clear\" (click)=\"handleReset()\">\n @if(options.buttons.clear?.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.clear?.iconSlot\" [name]=\"options.buttons.clear?.icon\"></ion-icon>\n }\n {{ ['delete', 'read', 'update'].includes(operation) ? 'Back' : options.buttons.clear?.text}}\n </ion-button>\n }\n @if(operation === 'delete' && uid) {\n <ion-button\n (click)=\"handleDelete()\"\n color=\"danger\"\n type=\"button\">\n @if(options.buttons.submit.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n }\n Delete\n </ion-button>\n }\n @if(operation === OperationKeys.CREATE || operation === OperationKeys.UPDATE) {\n <ion-button\n type=\"submit\">\n @if(options.buttons.submit.icon) {\n <ion-icon aria-hidden=\"true\" [slot]=\"options.buttons.submit.iconSlot\" [name]=\"options.buttons.submit.icon\"></ion-icon>\n }\n {{options.buttons.submit.text}}\n </ion-button>\n }\n\n </div>\n </div>\n}\n\n", styles: ["#dcf-buttons-container{margin-top:1.8rem;margin-bottom:0}\n"] }]
109
+ }], propDecorators: { model: [{
110
+ type: Input
111
+ }], updateOn: [{
112
+ type: Input
113
+ }], component: [{
114
+ type: ViewChild,
115
+ args: ['reactiveForm', { static: false, read: ElementRef }]
116
+ }], target: [{
117
+ type: Input
118
+ }], method: [{
119
+ type: Input
120
+ }], options: [{
121
+ type: Input
122
+ }], action: [{
123
+ type: Input
124
+ }], operation: [{
125
+ type: Input,
126
+ args: [{ required: true }]
127
+ }], handlers: [{
128
+ type: Input
129
+ }], formGroup: [{
130
+ type: Input
131
+ }], childOf: [{
132
+ type: Input
133
+ }], rendererId: [{
134
+ type: Input
135
+ }], uid: [{
136
+ type: Input
137
+ }], submitEvent: [{
138
+ type: Output
139
+ }] } });
140
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3J1ZC1mb3JtLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9jcnVkLWZvcm0vY3J1ZC1mb3JtLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvY29tcG9uZW50cy9jcnVkLWZvcm0vY3J1ZC1mb3JtLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPLEVBRUwsU0FBUyxFQUNULFVBQVUsRUFDVixZQUFZLEVBQ1osTUFBTSxFQUNOLEtBQUssRUFHTCxNQUFNLEVBQ04sU0FBUyxHQUNWLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUczQyxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDN0QsT0FBTyxFQUFpQixPQUFPLEVBQUUsY0FBYyxFQUFrRCxNQUFNLGNBQWMsQ0FBQztBQUV0SCxPQUFPLEVBQWtCLGFBQWEsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ3hFLE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUN6RCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsU0FBUyxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDekUsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLDJCQUEyQixDQUFDOzs7O0FBS3BEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBbUJHO0FBU0ksSUFBTSxpQkFBaUIsR0FBdkIsTUFBTSxpQkFBaUI7SUFBdkI7UUFhTCxhQUFRLEdBQW9CLFFBQVEsQ0FBQztRQU1yQyxXQUFNLEdBQW1CLE9BQU8sQ0FBQztRQUdqQyxXQUFNLEdBQTZCLE9BQU8sQ0FBQztRQTBDM0MsZ0JBQVcsR0FBRyxJQUFJLFlBQVksRUFBaUIsQ0FBQztRQWtCaEQ7Ozs7Ozs7OztXQVNHO1FBQ0ssYUFBUSxHQUFhLE1BQU0sQ0FBQyxR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UTtzQkFEUCxLQUFLO2dCQUlOLFNBQVM7c0JBRFIsS0FBSztnQkFXTixPQUFPO3NCQUROLEtBQUs7Z0JBSU4sVUFBVTtzQkFEVCxLQUFLO2dCQVdOLEdBQUc7c0JBREYsS0FBSztnQkFLTixXQUFXO3NCQURWLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBZnRlclZpZXdJbml0LFxuICBDb21wb25lbnQsXG4gIEVsZW1lbnRSZWYsXG4gIEV2ZW50RW1pdHRlcixcbiAgaW5qZWN0LFxuICBJbnB1dCxcbiAgT25EZXN0cm95LFxuICBPbkluaXQsXG4gIE91dHB1dCxcbiAgVmlld0NoaWxkLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IExvY2F0aW9uIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEZvcm1Hcm91cCB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IEZvcm1FbGVtZW50IH0gZnJvbSAnLi4vLi4vaW50ZXJmYWNlcyc7XG5pbXBvcnQgeyBOZ3hGb3JtU2VydmljZSB9IGZyb20gJy4uLy4uL2VuZ2luZS9OZ3hGb3JtU2VydmljZSc7XG5pbXBvcnQgeyBDcnVkRm9ybUV2ZW50LCBEeW5hbWljLCBFdmVudENvbnN0YW50cywgRmllbGRVcGRhdGVNb2RlLCBIVE1MRm9ybVRhcmdldCwgUmVuZGVyZWRNb2RlbCB9IGZyb20gJy4uLy4uL2VuZ2luZSc7XG5pbXBvcnQgeyBDcnVkRm9ybU9wdGlvbnMgfSBmcm9tICcuL3R5cGVzJztcbmltcG9ydCB7IENydWRPcGVyYXRpb25zLCBPcGVyYXRpb25LZXlzIH0gZnJvbSAnQGRlY2FmLXRzL2RiLWRlY29yYXRvcnMnO1xuaW1wb3J0IHsgRGVmYXVsdEZvcm1SZWFjdGl2ZU9wdGlvbnMgfSBmcm9tICcuL2NvbnN0YW50cyc7XG5pbXBvcnQgeyBGb3JBbmd1bGFyTW9kdWxlLCBnZXRMb2dnZXIgfSBmcm9tICdzcmMvbGliL2Zvci1hbmd1bGFyLm1vZHVsZSc7XG5pbXBvcnQgeyBJb25JY29uIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXIvc3RhbmRhbG9uZSc7XG5pbXBvcnQgeyBNb2RlbCB9IGZyb20gJ0BkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvbic7XG5pbXBvcnQgeyBMb2dnZXIgfSBmcm9tICdAZGVjYWYtdHMvbG9nZ2luZyc7XG5cblxuLyoqXG4gKiBAY29tcG9uZW50IENydWRGb3JtQ29tcG9uZW50XG4gKiBAZXhhbXBsZSA8bmd4LWRlY2FmLWNydWQtZm9ybVxuICogICBhY3Rpb249XCJjcmVhdGVcIlxuICogICBvcGVyYXRpb249XCJjcmVhdGVcIlxuICogICBmb3JtR3JvdXA9XCJmb3JtR3JvdXBcIlxuICogICByZW5kZXJlcklkPVwicmVuZGVyZXJJZFwiXG4gKiAgIHN1Ym1pdEV2ZW50PVwic3VibWl0RXZlbnRcIlxuICogICB0YXJnZXQ9XCJfc2VsZlwiXG4gKiAgIG1ldGhvZD1cImV2ZW50XCI+XG4gKiA8L25neC1kZWNhZi1jcnVkLWZvcm0+XG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IGFjdGlvbiAtIFRoZSBhY3Rpb24gdG8gYmUgcGVyZm9ybWVkIChjcmVhdGUsIHJlYWQsIHVwZGF0ZSwgZGVsZXRlKVxuICogQHBhcmFtIHtDcnVkT3BlcmF0aW9uc30gb3BlcmF0aW9uIC0gVGhlIENSVUQgb3BlcmF0aW9uIGJlaW5nIHBlcmZvcm1lZCAoY3JlYXRlLCByZWFkLCB1cGRhdGUsIGRlbGV0ZSlcbiAqIEBwYXJhbSB7Rm9ybUdyb3VwfSBmb3JtR3JvdXAgLSBUaGUgZm9ybSBncm91cFxuICogQHBhcmFtIHtzdHJpbmd9IHJlbmRlcmVySWQgLSBUaGUgcmVuZGVyZXIgaWRcbiAqIEBwYXJhbSB7U3VibWl0RXZlbnR9IHN1Ym1pdEV2ZW50IC0gVGhlIHN1Ym1pdCBldmVudFxuICogQHBhcmFtIHtzdHJpbmd9IHRhcmdldCAtIFRoZSB0YXJnZXRcbiAqIEBwYXJhbSB7c3RyaW5nfSBtZXRob2QgLSBUaGUgbWV0aG9kXG4gKi9cbkBEeW5hbWljKClcbkBDb21wb25lbnQoe1xuICBzdGFuZGFsb25lOiB0cnVlLFxuICBzZWxlY3RvcjogJ25neC1kZWNhZi1jcnVkLWZvcm0nLFxuICB0ZW1wbGF0ZVVybDogJy4vY3J1ZC1mb3JtLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vY3J1ZC1mb3JtLmNvbXBvbmVudC5zY3NzJ10sXG4gIGltcG9ydHM6IFtGb3JBbmd1bGFyTW9kdWxlLCBJb25JY29uXSxcbn0pXG5leHBvcnQgY2xhc3MgQ3J1ZEZvcm1Db21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQsIEZvcm1FbGVtZW50LCBPbkRlc3Ryb3ksIFJlbmRlcmVkTW9kZWwge1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gUmVwb3NpdG9yeSBtb2RlbCBmb3IgZGF0YSBvcGVyYXRpb25zLlxuICAgKiBAc3VtbWFyeSBUaGUgZGF0YSBtb2RlbCByZXBvc2l0b3J5IHRoYXQgdGhpcyBjb21wb25lbnQgd2lsbCB1c2UgZm9yIENSVUQgb3BlcmF0aW9ucy5cbiAgICogVGhpcyBwcm92aWRlcyBhIGNvbm5lY3Rpb24gdG8gdGhlIGRhdGEgbGF5ZXIgZm9yIHJldHJpZXZpbmcgYW5kIG1hbmlwdWxhdGluZyBkYXRhLlxuICAgKlxuICAgKiBAdHlwZSB7TW9kZWx8IHVuZGVmaW5lZH1cbiAgICovXG4gIEBJbnB1dCgpXG4gIG1vZGVsITogTW9kZWwgfCB1bmRlZmluZWQ7XG5cbiAgQElucHV0KClcbiAgdXBkYXRlT246IEZpZWxkVXBkYXRlTW9kZSA9ICdjaGFuZ2UnO1xuXG4gIEBWaWV3Q2hpbGQoJ3JlYWN0aXZlRm9ybScsIHsgc3RhdGljOiBmYWxzZSwgcmVhZDogRWxlbWVudFJlZiB9KVxuICBjb21wb25lbnQhOiBFbGVtZW50UmVmO1xuXG4gIEBJbnB1dCgpXG4gIHRhcmdldDogSFRNTEZvcm1UYXJnZXQgPSAnX3NlbGYnO1xuXG4gIEBJbnB1dCgpXG4gIG1ldGhvZDogJ2dldCcgfCAncG9zdCcgfCAnZXZlbnQnID0gJ2V2ZW50JztcblxuICBASW5wdXQoKVxuICBvcHRpb25zITogQ3J1ZEZvcm1PcHRpb25zO1xuXG4gIEBJbnB1dCgpXG4gIGFjdGlvbj86IHN0cmluZztcblxuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KVxuICBvcGVyYXRpb24hOiBDcnVkT3BlcmF0aW9ucztcblxuICBASW5wdXQoKVxuICBoYW5kbGVycyE6IFJlY29yZDxzdHJpbmcsICguLi5hcmdzOiBhbnlbXSkgPT4gYW55IHwgUHJvbWlzZTxhbnk+PjtcblxuICBASW5wdXQoKVxuICBmb3JtR3JvdXAhOiBGb3JtR3JvdXAgfCB1bmRlZmluZWQ7XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBQYXRoIHRvIHRoZSBwYXJlbnQgRm9ybUdyb3VwLCBpZiBuZXN0ZWQuXG4gICAqIEBzdW1tYXJ5IEZ1bGwgZG90LWRlbGltaXRlZCBwYXRoIG9mIHRoZSBwYXJlbnQgRm9ybUdyb3VwLiBTZXQgb25seSB3aGVuIGlzIHBhcnQgb2YgYSBuZXN0ZWQgc3RydWN0dXJlLlxuICAgKlxuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgQ3J1ZEZvcm1Db21wb25lbnRcbiAgICovXG4gIEBJbnB1dCgpXG4gIGNoaWxkT2Y/OiBzdHJpbmc7XG5cbiAgQElucHV0KClcbiAgcmVuZGVyZXJJZCE6IHN0cmluZztcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIFVuaXF1ZSBpZGVudGlmaWVyIGZvciB0aGUgY3VycmVudCByZWNvcmQuXG4gICAqIEBzdW1tYXJ5IEEgdW5pcXVlIGlkZW50aWZpZXIgZm9yIHRoZSBjdXJyZW50IHJlY29yZCBiZWluZyBkaXNwbGF5ZWQgb3IgbWFuaXB1bGF0ZWQuXG4gICAqIFRoaXMgaXMgdHlwaWNhbGx5IHVzZWQgaW4gY29uanVuY3Rpb24gd2l0aCB0aGUgcHJpbWFyeSBrZXkgZm9yIG9wZXJhdGlvbnMgb24gc3BlY2lmaWMgcmVjb3Jkcy5cbiAgICpcbiAgICogQHR5cGUge3N0cmluZyB8IG51bWJlcn1cbiAgICovXG4gIEBJbnB1dCgpXG4gIHVpZCE6IHN0cmluZyB8IG51bWJlciB8IHVuZGVmaW5lZDtcblxuXG4gIEBPdXRwdXQoKVxuICBzdWJtaXRFdmVudCA9IG5ldyBFdmVudEVtaXR0ZXI8Q3J1ZEZvcm1FdmVudD4oKTtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIExvZ2dlciBpbnN0YW5jZSBmb3IgdGhlIGNvbXBvbmVudC5cbiAgICogQHN1bW1hcnkgUHJvdmlkZXMgbG9nZ2luZyBjYXBhYmlsaXRpZXMgZm9yIHRoZSBjb21wb25lbnQsIGFsbG93aW5nIGZvciBjb25zaXN0ZW50XG4gICAqIGFuZCBzdHJ1Y3R1cmVkIGxvZ2dpbmcgb2YgaW5mb3JtYXRpb24sIHdhcm5pbmdzLCBhbmQgZXJyb3JzLiBUaGlzIGxvZ2dlciBpcyBpbml0aWFsaXplZFxuICAgKiBpbiB0aGUgbmdPbkluaXQgbWV0aG9kIHVzaW5nIHRoZSBnZXRMb2dnZXIgZnVuY3Rpb24gZnJvbSB0aGUgRm9yQW5ndWxhck1vZHVsZS5cbiAgICpcbiAgICogVGhlIGxvZ2dlciBpcyB1c2VkIHRocm91Z2hvdXQgdGhlIGNvbXBvbmVudCB0byByZWNvcmQgaW1wb3J0YW50IGV2ZW50cywgZGVidWcgaW5mb3JtYXRpb24sXG4gICAqIGFuZCBwb3RlbnRpYWwgaXNzdWVzLiBJdCBoZWxwcyBpbiBtb25pdG9yaW5nIHRoZSBjb21wb25lbnQncyBiZWhhdmlvciwgdHJhY2tpbmcgdGhlIGZsb3dcbiAgICogb2Ygb3BlcmF0aW9ucywgYW5kIGZhY2lsaXRhdGluZyBlYXNpZXIgZGVidWdnaW5nIGFuZCBtYWludGVuYW5jZS5cbiAgICpcbiAgICogQHR5cGUge0xvZ2dlcn1cbiAgICogQHByaXZhdGVcbiAgICogQG1lbWJlck9mIENydWRGb3JtQ29tcG9uZW50XG4gICAqL1xuICBwcml2YXRlIGxvZ2dlciE6IExvZ2dlcjtcblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uIEFuZ3VsYXIgTG9jYXRpb24gc2VydmljZS5cbiAgICogQHN1bW1hcnkgSW5qZWN0ZWQgc2VydmljZSB0aGF0IHByb3ZpZGVzIGFjY2VzcyB0byB0aGUgYnJvd3NlcidzIFVSTCBhbmQgaGlzdG9yeS5cbiAgICogVGhpcyBzZXJ2aWNlIGlzIHVzZWQgZm9yIGludGVyYWN0aW5nIHdpdGggdGhlIGJyb3dzZXIncyBoaXN0b3J5IEFQSSwgYWxsb3dpbmdcbiAgICogZm9yIGJhY2sgbmF2aWdhdGlvbiBhbmQgVVJMIG1hbmlwdWxhdGlvbiBvdXRzaWRlIG9mIEFuZ3VsYXIncyByb3V0ZXIuXG4gICAqXG4gICAqIEBwcml2YXRlXG4gICAqIEB0eXBlIHtMb2NhdGlvbn1cbiAgICogQG1lbWJlck9mIENydWRGb3JtQ29tcG9uZW50XG4gICAqL1xuICBwcml2YXRlIGxvY2F0aW9uOiBMb2NhdGlvbiA9IGluamVjdChMb2NhdGlvbik7XG5cbiAgLy8gbmdBZnRlclZpZXdJbml0KCkge1xuICAgIC8vIGlmICghW09wZXJhdGlvbktleXMuUkVBRCwgT3BlcmF0aW9uS2V5cy5ERUxFVEVdLmluY2x1ZGVzKHRoaXMub3BlcmF0aW9uKSlcbiAgICAvLyAgIE5neEZvcm1TZXJ2aWNlLmZvcm1BZnRlclZpZXdJbml0KHRoaXMsIHRoaXMucmVuZGVyZXJJZCk7XG4gIC8vIH1cblxuICBhc3luYyBuZ09uSW5pdCgpIHtcbiAgICBpZiAoIXRoaXMubG9nZ2VyKVxuICAgICAgdGhpcy5sb2dnZXIgPSBnZXRMb2dnZXIodGhpcyk7XG4gICAgaWYgKHRoaXMub3BlcmF0aW9uID09PSBPcGVyYXRpb25LZXlzLlJFQUQgfHwgdGhpcy5vcGVyYXRpb24gPT09IE9wZXJhdGlvbktleXMuREVMRVRFKVxuICAgICAgdGhpcy5mb3JtR3JvdXAgPSB1bmRlZmluZWQ7XG4gICAgdGhpcy5vcHRpb25zID0gT2JqZWN0LmFzc2lnbihcbiAgICAgIHt9LFxuICAgICAgRGVmYXVsdEZvcm1SZWFjdGl2ZU9wdGlvbnMsXG4gICAgICB0aGlzLm9wdGlvbnMgfHwge30sXG4gICAgKTtcblxuICB9XG5cbiAgbmdPbkRlc3Ryb3koKSB7XG4gICAgaWYgKHRoaXMuZm9ybUdyb3VwKVxuICAgICAgTmd4Rm9ybVNlcnZpY2UudW5yZWdpc3Rlcih0aGlzLmZvcm1Hcm91cCk7XG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtICB7U3VibWl0RXZlbnR9IGV2ZW50XG4gICAqL1xuICBhc3luYyBzdWJtaXQoZXZlbnQ6IFN1Ym1pdEV2ZW50KTogUHJvbWlzZTxib29sZWFuIHwgdm9pZD4ge1xuICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKCk7XG4gICAgZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XG5cbiAgICBpZiAoIU5neEZvcm1TZXJ2aWNlLnZhbGlkYXRlRmllbGRzKHRoaXMuZm9ybUdyb3VwIGFzIEZvcm1Hcm91cCkpXG4gICAgICByZXR1cm4gZmFsc2U7XG5cbiAgICBjb25zdCBkYXRhID0gTmd4Rm9ybVNlcnZpY2UuZ2V0Rm9ybURhdGEodGhpcy5mb3JtR3JvdXAgYXMgRm9ybUdyb3VwKTtcbiAgICBjb25zb2xlLmxvZygnU3VibWl0PScsIGRhdGEpO1xuICAgIHRoaXMuc3VibWl0RXZlbnQuZW1pdCh7XG4gICAgICBkYXRhLFxuICAgICAgY29tcG9uZW50OiAnRm9ybVJlYWN0aXZlQ29tcG9uZW50JyxcbiAgICAgIG5hbWU6IHRoaXMuYWN0aW9uIHx8IEV2ZW50Q29uc3RhbnRzLlNVQk1JVF9FVkVOVCxcbiAgICAgIGhhbmRsZXJzOiB0aGlzLmhhbmRsZXJzLFxuICAgIH0pO1xuICB9XG5cbiAgaGFuZGxlUmVzZXQoKSB7XG4gICAgdGhpcy5sb2NhdGlvbi5iYWNrKCk7XG4gICAgLy8gaWYoT3BlcmF0aW9uS2V5cy5ERUxFVEUgIT09IHRoaXMub3BlcmF0aW9uKVxuICAgIC8vICAgTmd4Rm9ybVNlcnZpY2UucmVzZXQodGhpcy5mb3JtR3JvdXApO1xuICAgIC8vIGVsc2VcbiAgICAvLyAgIHRoaXMubG9jYXRpb24uYmFjaygpO1xuICB9XG5cbiAgaGFuZGxlRGVsZXRlKCkge1xuICAgIHRoaXMuc3VibWl0RXZlbnQuZW1pdCh7XG4gICAgICBkYXRhOiB0aGlzLnVpZCxcbiAgICAgIGNvbXBvbmVudDogJ0Zvcm1SZWFjdGl2ZUNvbXBvbmVudCcsXG4gICAgICBuYW1lOiBFdmVudENvbnN0YW50cy5TVUJNSVRfRVZFTlQsXG4gICAgfSk7XG4gIH1cblxuICBwcm90ZWN0ZWQgcmVhZG9ubHkgT3BlcmF0aW9uS2V5cyA9IE9wZXJhdGlvbktleXM7XG59XG4iLCJAaWYob3BlcmF0aW9uICE9PSAncmVhZCcgJiYgb3BlcmF0aW9uICE9PSAnZGVsZXRlJykge1xuICA8Zm9ybSAjcmVhY3RpdmVGb3JtIFtpZF09XCJyZW5kZXJlcklkXCIgW2Zvcm1Hcm91cF09XCJmb3JtR3JvdXBcIiAoc3VibWl0KT1cInN1Ym1pdCgkZXZlbnQpXCIgbm92YWxpZGF0ZSBbdGFyZ2V0XT1cInRhcmdldFwiPlxuICAgIDxuZy1jb250ZW50ICNmb3JtQ29udGVudD48L25nLWNvbnRlbnQ+XG4gICAgPGRpdiBjbGFzcz1cImRjZi1mbGV4IGRjZi1mbGV4LXJpZ2h0XCI+XG4gICAgICA8ZGl2PlxuICAgICAgICBAaWYob3B0aW9ucy5idXR0b25zLmNsZWFyKSB7XG4gICAgICAgICAgPGlvbi1idXR0b24gZmlsbD1cImNsZWFyXCIgKGNsaWNrKT1cImhhbmRsZVJlc2V0KClcIj5cbiAgICAgICAgICAgIEBpZihvcHRpb25zLmJ1dHRvbnMuY2xlYXI/Lmljb24pIHtcbiAgICAgICAgICAgICAgPGlvbi1pY29uICBhcmlhLWhpZGRlbj1cInRydWVcIiBbc2xvdF09XCJvcHRpb25zLmJ1dHRvbnMuY2xlYXI/Lmljb25TbG90XCIgW25hbWVdPVwib3B0aW9ucy5idXR0b25zLmNsZWFyPy5pY29uXCI+PC9pb24taWNvbj5cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHt7b3BlcmF0aW9uID09PSAndXBkYXRlJyA/ICdCYWNrJyA6IG9wdGlvbnMuYnV0dG9ucy5jbGVhcj8udGV4dH19XG4gICAgICAgICAgPC9pb24tYnV0dG9uPlxuICAgICAgICB9XG5cbiAgICAgICAgPGlvbi1idXR0b25cbiAgICAgICAgICB0eXBlPVwic3VibWl0XCI+XG4gICAgICAgICAgQGlmKG9wdGlvbnMuYnV0dG9ucy5zdWJtaXQuaWNvbikge1xuICAgICAgICAgICAgPGlvbi1pY29uIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIFtzbG90XT1cIm9wdGlvbnMuYnV0dG9ucy5zdWJtaXQuaWNvblNsb3RcIiBbbmFtZV09XCJvcHRpb25zLmJ1dHRvbnMuc3VibWl0Lmljb25cIj48L2lvbi1pY29uPlxuICAgICAgICAgIH1cbiAgICAgICAgICB7eyBhY3Rpb24gPyBhY3Rpb24gOiBvcHRpb25zLmJ1dHRvbnMuc3VibWl0LnRleHR9fVxuICAgICAgICA8L2lvbi1idXR0b24+XG4gICAgICA8L2Rpdj5cbiAgICA8L2Rpdj5cbiAgPC9mb3JtPlxufSBAZWxzZSB7XG4gIDxkaXYgW2NsYXNzXT1cIidkY2YtZmxleCBkY2YtZmxleC1yaWdodCAnICsgb3BlcmF0aW9uXCIgaWQ9XCJkY2YtYnV0dG9ucy1jb250YWluZXJcIj5cbiAgICA8ZGl2PlxuICAgICAgQGlmKG9wdGlvbnMuYnV0dG9ucy5jbGVhcikge1xuICAgICAgICA8aW9uLWJ1dHRvbiBmaWxsPVwiY2xlYXJcIiAoY2xpY2spPVwiaGFuZGxlUmVzZXQoKVwiPlxuICAgICAgICAgIEBpZihvcHRpb25zLmJ1dHRvbnMuY2xlYXI/Lmljb24pIHtcbiAgICAgICAgICAgIDxpb24taWNvbiAgYXJpYS1oaWRkZW49XCJ0cnVlXCIgW3Nsb3RdPVwib3B0aW9ucy5idXR0b25zLmNsZWFyPy5pY29uU2xvdFwiIFtuYW1lXT1cIm9wdGlvbnMuYnV0dG9ucy5jbGVhcj8uaWNvblwiPjwvaW9uLWljb24+XG4gICAgICAgICAgfVxuICAgICAgICAgIHt7IFsnZGVsZXRlJywgJ3JlYWQnLCAndXBkYXRlJ10uaW5jbHVkZXMob3BlcmF0aW9uKSA/ICdCYWNrJyA6IG9wdGlvbnMuYnV0dG9ucy5jbGVhcj8udGV4dH19XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgIH1cbiAgICAgIEBpZihvcGVyYXRpb24gPT09ICdkZWxldGUnICYmIHVpZCkge1xuICAgICAgICA8aW9uLWJ1dHRvblxuICAgICAgICAgIChjbGljayk9XCJoYW5kbGVEZWxldGUoKVwiXG4gICAgICAgICAgY29sb3I9XCJkYW5nZXJcIlxuICAgICAgICAgIHR5cGU9XCJidXR0b25cIj5cbiAgICAgICAgICBAaWYob3B0aW9ucy5idXR0b25zLnN1Ym1pdC5pY29uKSB7XG4gICAgICAgICAgICA8aW9uLWljb24gYXJpYS1oaWRkZW49XCJ0cnVlXCIgW3Nsb3RdPVwib3B0aW9ucy5idXR0b25zLnN1Ym1pdC5pY29uU2xvdFwiIFtuYW1lXT1cIm9wdGlvbnMuYnV0dG9ucy5zdWJtaXQuaWNvblwiPjwvaW9uLWljb24+XG4gICAgICAgICAgfVxuICAgICAgICAgIERlbGV0ZVxuICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgIH1cbiAgICAgIEBpZihvcGVyYXRpb24gPT09IE9wZXJhdGlvbktleXMuQ1JFQVRFIHx8IG9wZXJhdGlvbiA9PT0gT3BlcmF0aW9uS2V5cy5VUERBVEUpIHtcbiAgICAgICAgPGlvbi1idXR0b25cbiAgICAgICAgICB0eXBlPVwic3VibWl0XCI+XG4gICAgICAgICAgQGlmKG9wdGlvbnMuYnV0dG9ucy5zdWJtaXQuaWNvbikge1xuICAgICAgICAgICAgPGlvbi1pY29uIGFyaWEtaGlkZGVuPVwidHJ1ZVwiIFtzbG90XT1cIm9wdGlvbnMuYnV0dG9ucy5zdWJtaXQuaWNvblNsb3RcIiBbbmFtZV09XCJvcHRpb25zLmJ1dHRvbnMuc3VibWl0Lmljb25cIj48L2lvbi1pY29uPlxuICAgICAgICAgIH1cbiAgICAgICAgICB7e29wdGlvbnMuYnV0dG9ucy5zdWJtaXQudGV4dH19XG4gICAgICAgIDwvaW9uLWJ1dHRvbj5cbiAgICAgIH1cblxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbn1cblxuIl19
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvY3J1ZC1mb3JtL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIEZvcm1SZWFjdGl2ZVN1Ym1pdEV2ZW50IHtcbiAgZGF0YTogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ3J1ZEZvcm1PcHRpb25zIHtcbiAgYnV0dG9uczoge1xuICAgIHN1Ym1pdDoge1xuICAgICAgaWNvbj86IHN0cmluZztcbiAgICAgIGljb25TbG90PzogJ3N0YXJ0JyB8ICdlbmQnO1xuICAgICAgdGV4dD86IHN0cmluZztcbiAgICB9O1xuICAgIGNsZWFyPzoge1xuICAgICAgaWNvbj86IHN0cmluZztcbiAgICAgIGljb25TbG90PzogJ3N0YXJ0JyB8ICdlbmQnO1xuICAgICAgdGV4dD86IHN0cmluZztcbiAgICB9O1xuICB9O1xufVxuIl19