@tekus/design-system 3.0.0 → 4.0.1

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 (55) hide show
  1. package/components/{tk-form-input-field → forms}/index.d.ts +1 -1
  2. package/components/forms/public-api.d.ts +3 -0
  3. package/components/forms/tk-form-autocomplete-field/index.d.ts +5 -0
  4. package/components/{tk-form-autocomplete-field → forms/tk-form-autocomplete-field}/tk-form-autocomplete-field.component.d.ts +6 -1
  5. package/components/forms/tk-form-chips-autocomplete-field/index.d.ts +5 -0
  6. package/components/forms/tk-form-input-field/index.d.ts +5 -0
  7. package/components/{tk-form-input-field → forms/tk-form-input-field}/tk-form-input-field.component.d.ts +6 -1
  8. package/esm2022/components/forms/public-api.mjs +4 -0
  9. package/esm2022/components/forms/tekus-design-system-components-forms.mjs +5 -0
  10. package/esm2022/components/forms/tk-form-autocomplete-field/models/option.model.mjs +2 -0
  11. package/esm2022/components/forms/tk-form-autocomplete-field/public-api.mjs +2 -0
  12. package/esm2022/components/forms/tk-form-autocomplete-field/tekus-design-system-components-forms-tk-form-autocomplete-field.mjs +5 -0
  13. package/esm2022/components/forms/tk-form-autocomplete-field/tk-form-autocomplete-field.component.mjs +245 -0
  14. package/esm2022/components/forms/tk-form-chips-autocomplete-field/models/option.model.mjs +2 -0
  15. package/esm2022/components/forms/tk-form-chips-autocomplete-field/public-api.mjs +2 -0
  16. package/esm2022/components/forms/tk-form-chips-autocomplete-field/tekus-design-system-components-forms-tk-form-chips-autocomplete-field.mjs +5 -0
  17. package/esm2022/components/forms/tk-form-chips-autocomplete-field/tk-form-chips-autocomplete-field.component.mjs +167 -0
  18. package/esm2022/components/forms/tk-form-input-field/models/validator-with-message.model.mjs +2 -0
  19. package/esm2022/components/forms/tk-form-input-field/public-api.mjs +2 -0
  20. package/esm2022/components/forms/tk-form-input-field/tekus-design-system-components-forms-tk-form-input-field.mjs +5 -0
  21. package/esm2022/components/forms/tk-form-input-field/tk-form-input-field.component.mjs +206 -0
  22. package/fesm2022/{tekus-design-system-components-tk-form-autocomplete-field.mjs → tekus-design-system-components-forms-tk-form-autocomplete-field.mjs} +10 -3
  23. package/fesm2022/tekus-design-system-components-forms-tk-form-autocomplete-field.mjs.map +1 -0
  24. package/fesm2022/tekus-design-system-components-forms-tk-form-chips-autocomplete-field.mjs +174 -0
  25. package/fesm2022/tekus-design-system-components-forms-tk-form-chips-autocomplete-field.mjs.map +1 -0
  26. package/fesm2022/{tekus-design-system-components-tk-form-input-field.mjs → tekus-design-system-components-forms-tk-form-input-field.mjs} +10 -3
  27. package/fesm2022/tekus-design-system-components-forms-tk-form-input-field.mjs.map +1 -0
  28. package/fesm2022/tekus-design-system-components-forms.mjs +604 -0
  29. package/fesm2022/tekus-design-system-components-forms.mjs.map +1 -0
  30. package/package.json +31 -22
  31. package/components/tk-form-autocomplete-field/index.d.ts +0 -5
  32. package/components/tk-form-chips-autocomplete-field/index.d.ts +0 -5
  33. package/esm2022/components/tk-form-autocomplete-field/models/option.model.mjs +0 -2
  34. package/esm2022/components/tk-form-autocomplete-field/public-api.mjs +0 -2
  35. package/esm2022/components/tk-form-autocomplete-field/tekus-design-system-components-tk-form-autocomplete-field.mjs +0 -5
  36. package/esm2022/components/tk-form-autocomplete-field/tk-form-autocomplete-field.component.mjs +0 -238
  37. package/esm2022/components/tk-form-chips-autocomplete-field/models/option.model.mjs +0 -2
  38. package/esm2022/components/tk-form-chips-autocomplete-field/public-api.mjs +0 -2
  39. package/esm2022/components/tk-form-chips-autocomplete-field/tekus-design-system-components-tk-form-chips-autocomplete-field.mjs +0 -5
  40. package/esm2022/components/tk-form-chips-autocomplete-field/tk-form-chips-autocomplete-field.component.mjs +0 -167
  41. package/esm2022/components/tk-form-input-field/models/validator-with-message.model.mjs +0 -2
  42. package/esm2022/components/tk-form-input-field/public-api.mjs +0 -2
  43. package/esm2022/components/tk-form-input-field/tekus-design-system-components-tk-form-input-field.mjs +0 -5
  44. package/esm2022/components/tk-form-input-field/tk-form-input-field.component.mjs +0 -199
  45. package/fesm2022/tekus-design-system-components-tk-form-autocomplete-field.mjs.map +0 -1
  46. package/fesm2022/tekus-design-system-components-tk-form-chips-autocomplete-field.mjs +0 -174
  47. package/fesm2022/tekus-design-system-components-tk-form-chips-autocomplete-field.mjs.map +0 -1
  48. package/fesm2022/tekus-design-system-components-tk-form-input-field.mjs.map +0 -1
  49. /package/components/{tk-form-autocomplete-field → forms/tk-form-autocomplete-field}/models/option.model.d.ts +0 -0
  50. /package/components/{tk-form-autocomplete-field → forms/tk-form-autocomplete-field}/public-api.d.ts +0 -0
  51. /package/components/{tk-form-chips-autocomplete-field → forms/tk-form-chips-autocomplete-field}/models/option.model.d.ts +0 -0
  52. /package/components/{tk-form-chips-autocomplete-field → forms/tk-form-chips-autocomplete-field}/public-api.d.ts +0 -0
  53. /package/components/{tk-form-chips-autocomplete-field → forms/tk-form-chips-autocomplete-field}/tk-form-chips-autocomplete-field.component.d.ts +0 -0
  54. /package/components/{tk-form-input-field → forms/tk-form-input-field}/models/validator-with-message.model.d.ts +0 -0
  55. /package/components/{tk-form-input-field → forms/tk-form-input-field}/public-api.d.ts +0 -0
@@ -1,5 +1,5 @@
1
1
  /**
2
2
  * Generated bundle index. Do not edit.
3
3
  */
4
- /// <amd-module name="@tekus/design-system/components/tk-form-input-field" />
4
+ /// <amd-module name="@tekus/design-system/components/forms" />
5
5
  export * from './public-api';
@@ -0,0 +1,3 @@
1
+ export * from './tk-form-input-field/tk-form-input-field.component';
2
+ export * from './tk-form-autocomplete-field/tk-form-autocomplete-field.component';
3
+ export * from './tk-form-chips-autocomplete-field/tk-form-chips-autocomplete-field.component';
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ /// <amd-module name="@tekus/design-system/components/forms/tk-form-autocomplete-field" />
5
+ export * from './public-api';
@@ -42,6 +42,11 @@ export declare class TkFormAutocompleteFieldComponent implements OnInit, OnDestr
42
42
  * @default true
43
43
  */
44
44
  autoSelectActiveOption?: boolean;
45
+ /**
46
+ * Optional field input type
47
+ * @default 'text'
48
+ */
49
+ type: string;
45
50
  /**
46
51
  * Whether the field is required
47
52
  * @default true
@@ -154,5 +159,5 @@ export declare class TkFormAutocompleteFieldComponent implements OnInit, OnDestr
154
159
  */
155
160
  ngOnDestroy(): void;
156
161
  static ɵfac: i0.ɵɵFactoryDeclaration<TkFormAutocompleteFieldComponent, never>;
157
- static ɵcmp: i0.ɵɵComponentDeclaration<TkFormAutocompleteFieldComponent, "lib-tk-form-autocomplete-field", never, { "fieldWidth": { "alias": "fieldWidth"; "required": false; }; "theme": { "alias": "theme"; "required": false; }; "appearance": { "alias": "appearance"; "required": false; }; "subscriptSizing": { "alias": "subscriptSizing"; "required": false; }; "autoActiveFirstOption": { "alias": "autoActiveFirstOption"; "required": false; }; "autoSelectActiveOption": { "alias": "autoSelectActiveOption"; "required": false; }; "required": { "alias": "required"; "required": false; }; "fieldTitle": { "alias": "fieldTitle"; "required": false; }; "fieldLabel": { "alias": "fieldLabel"; "required": false; }; "placeHolder": { "alias": "placeHolder"; "required": false; }; "options": { "alias": "options"; "required": false; }; }, { "optionSelected": "optionSelected"; "inputStatus": "inputStatus"; "bottomReached": "bottomReached"; }, never, never, true, never>;
162
+ static ɵcmp: i0.ɵɵComponentDeclaration<TkFormAutocompleteFieldComponent, "lib-tk-form-autocomplete-field", never, { "fieldWidth": { "alias": "fieldWidth"; "required": false; }; "theme": { "alias": "theme"; "required": false; }; "appearance": { "alias": "appearance"; "required": false; }; "subscriptSizing": { "alias": "subscriptSizing"; "required": false; }; "autoActiveFirstOption": { "alias": "autoActiveFirstOption"; "required": false; }; "autoSelectActiveOption": { "alias": "autoSelectActiveOption"; "required": false; }; "type": { "alias": "type"; "required": false; }; "required": { "alias": "required"; "required": false; }; "fieldTitle": { "alias": "fieldTitle"; "required": false; }; "fieldLabel": { "alias": "fieldLabel"; "required": false; }; "placeHolder": { "alias": "placeHolder"; "required": false; }; "options": { "alias": "options"; "required": false; }; }, { "optionSelected": "optionSelected"; "inputStatus": "inputStatus"; "bottomReached": "bottomReached"; }, never, never, true, never>;
158
163
  }
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ /// <amd-module name="@tekus/design-system/components/forms/tk-form-chips-autocomplete-field" />
5
+ export * from './public-api';
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ /// <amd-module name="@tekus/design-system/components/forms/tk-form-input-field" />
5
+ export * from './public-api';
@@ -20,6 +20,11 @@ export declare class TkFormInputFieldComponent implements OnInit, OnChanges {
20
20
  * @default 'accent'
21
21
  */
22
22
  theme?: ThemePalette;
23
+ /**
24
+ * Optional field input type
25
+ * @default 'text'
26
+ */
27
+ type: string;
23
28
  /**
24
29
  * Whether the field is required
25
30
  * @default true
@@ -128,5 +133,5 @@ export declare class TkFormInputFieldComponent implements OnInit, OnChanges {
128
133
  */
129
134
  updateErrorMessage(): void;
130
135
  static ɵfac: i0.ɵɵFactoryDeclaration<TkFormInputFieldComponent, never>;
131
- static ɵcmp: i0.ɵɵComponentDeclaration<TkFormInputFieldComponent, "lib-tk-form-input-field", never, { "fieldWidth": { "alias": "fieldWidth"; "required": false; }; "fieldTitle": { "alias": "fieldTitle"; "required": false; }; "theme": { "alias": "theme"; "required": false; }; "required": { "alias": "required"; "required": false; }; "floatLabel": { "alias": "floatLabel"; "required": false; }; "appearance": { "alias": "appearance"; "required": false; }; "subscriptSizing": { "alias": "subscriptSizing"; "required": false; }; "fieldLabel": { "alias": "fieldLabel"; "required": false; }; "placeHolder": { "alias": "placeHolder"; "required": false; }; "inputPrefix": { "alias": "inputPrefix"; "required": false; }; "inputSuffix": { "alias": "inputSuffix"; "required": false; }; "validatorsWithMessages": { "alias": "validatorsWithMessages"; "required": false; }; "asyncValidatorsWithMessages": { "alias": "asyncValidatorsWithMessages"; "required": false; }; }, { "inputStatus": "inputStatus"; }, never, never, true, never>;
136
+ static ɵcmp: i0.ɵɵComponentDeclaration<TkFormInputFieldComponent, "lib-tk-form-input-field", never, { "fieldWidth": { "alias": "fieldWidth"; "required": false; }; "fieldTitle": { "alias": "fieldTitle"; "required": false; }; "theme": { "alias": "theme"; "required": false; }; "type": { "alias": "type"; "required": false; }; "required": { "alias": "required"; "required": false; }; "floatLabel": { "alias": "floatLabel"; "required": false; }; "appearance": { "alias": "appearance"; "required": false; }; "subscriptSizing": { "alias": "subscriptSizing"; "required": false; }; "fieldLabel": { "alias": "fieldLabel"; "required": false; }; "placeHolder": { "alias": "placeHolder"; "required": false; }; "inputPrefix": { "alias": "inputPrefix"; "required": false; }; "inputSuffix": { "alias": "inputSuffix"; "required": false; }; "validatorsWithMessages": { "alias": "validatorsWithMessages"; "required": false; }; "asyncValidatorsWithMessages": { "alias": "asyncValidatorsWithMessages"; "required": false; }; }, { "inputStatus": "inputStatus"; }, never, never, true, never>;
132
137
  }
@@ -0,0 +1,4 @@
1
+ export * from './tk-form-input-field/tk-form-input-field.component';
2
+ export * from './tk-form-autocomplete-field/tk-form-autocomplete-field.component';
3
+ export * from './tk-form-chips-autocomplete-field/tk-form-chips-autocomplete-field.component';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Rlc2lnbi1zeXN0ZW0vY29tcG9uZW50cy9mb3Jtcy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMscURBQXFELENBQUM7QUFDcEUsY0FBYyxtRUFBbUUsQ0FBQztBQUNsRixjQUFjLCtFQUErRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi90ay1mb3JtLWlucHV0LWZpZWxkL3RrLWZvcm0taW5wdXQtZmllbGQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGstZm9ybS1hdXRvY29tcGxldGUtZmllbGQvdGstZm9ybS1hdXRvY29tcGxldGUtZmllbGQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGstZm9ybS1jaGlwcy1hdXRvY29tcGxldGUtZmllbGQvdGstZm9ybS1jaGlwcy1hdXRvY29tcGxldGUtZmllbGQuY29tcG9uZW50JztcbiJdfQ==
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVrdXMtZGVzaWduLXN5c3RlbS1jb21wb25lbnRzLWZvcm1zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZGVzaWduLXN5c3RlbS9jb21wb25lbnRzL2Zvcm1zL3Rla3VzLWRlc2lnbi1zeXN0ZW0tY29tcG9uZW50cy1mb3Jtcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3B0aW9uLm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZGVzaWduLXN5c3RlbS9jb21wb25lbnRzL2Zvcm1zL3RrLWZvcm0tYXV0b2NvbXBsZXRlLWZpZWxkL21vZGVscy9vcHRpb24ubW9kZWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgT3B0aW9uIHtcbiAgVmlld1ZhbHVlOiBzdHJpbmc7XG4gIFZhbHVlOiBzdHJpbmcgfCBudW1iZXI7XG59XG4iXX0=
@@ -0,0 +1,2 @@
1
+ export * from './tk-form-autocomplete-field.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Rlc2lnbi1zeXN0ZW0vY29tcG9uZW50cy9mb3Jtcy90ay1mb3JtLWF1dG9jb21wbGV0ZS1maWVsZC9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsd0NBQXdDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3RrLWZvcm0tYXV0b2NvbXBsZXRlLWZpZWxkLmNvbXBvbmVudCc7XG4iXX0=
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVrdXMtZGVzaWduLXN5c3RlbS1jb21wb25lbnRzLWZvcm1zLXRrLWZvcm0tYXV0b2NvbXBsZXRlLWZpZWxkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZGVzaWduLXN5c3RlbS9jb21wb25lbnRzL2Zvcm1zL3RrLWZvcm0tYXV0b2NvbXBsZXRlLWZpZWxkL3Rla3VzLWRlc2lnbi1zeXN0ZW0tY29tcG9uZW50cy1mb3Jtcy10ay1mb3JtLWF1dG9jb21wbGV0ZS1maWVsZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -0,0 +1,245 @@
1
+ import { Input, Output, Component, ViewChild, EventEmitter, } from '@angular/core';
2
+ import { MatFormFieldModule, } from '@angular/material/form-field';
3
+ import { MatAutocompleteModule, } from '@angular/material/autocomplete';
4
+ import { AsyncPipe, NgStyle } from '@angular/common';
5
+ import { MatInputModule } from '@angular/material/input';
6
+ import { map, startWith, Subscription } from 'rxjs';
7
+ import { FormControl, FormsModule, ReactiveFormsModule } from '@angular/forms';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/forms";
10
+ import * as i2 from "@angular/material/input";
11
+ import * as i3 from "@angular/material/form-field";
12
+ import * as i4 from "@angular/material/autocomplete";
13
+ import * as i5 from "@angular/material/core";
14
+ export class TkFormAutocompleteFieldComponent {
15
+ constructor() {
16
+ /**
17
+ * Input field control
18
+ * @ignore
19
+ */
20
+ this.formControl = new FormControl('');
21
+ /**
22
+ * What color palette to use
23
+ * @default 'accent'
24
+ */
25
+ this.theme = 'accent';
26
+ /**
27
+ * Form field appearance
28
+ * @default 'fill'
29
+ */
30
+ this.appearance = 'fill';
31
+ /**
32
+ * Form field reserved space for one line by default.
33
+ * @default 'dynamic'
34
+ */
35
+ this.subscriptSizing = 'dynamic';
36
+ /**
37
+ * Whether the first option should be highlighted when the autocomplete panel is opened
38
+ * @default true
39
+ */
40
+ this.autoActiveFirstOption = true;
41
+ /**
42
+ * Whether the active option should be selected as the user is navigating.
43
+ * @default true
44
+ */
45
+ this.autoSelectActiveOption = true;
46
+ /**
47
+ * Optional field input type
48
+ * @default 'text'
49
+ */
50
+ this.type = 'text';
51
+ /**
52
+ * Whether the field is required
53
+ * @default true
54
+ */
55
+ this.required = true;
56
+ /**
57
+ * Input field label
58
+ */
59
+ this.fieldLabel = '';
60
+ /**
61
+ * Input field placeholder
62
+ * @default ''
63
+ */
64
+ this.placeHolder = '';
65
+ /**
66
+ * Options to be displayed
67
+ * @ignore
68
+ */
69
+ this.options = [];
70
+ /**
71
+ * Option changes handler
72
+ */
73
+ this.optionSelected = new EventEmitter();
74
+ /**
75
+ * Input field changes handler
76
+ */
77
+ this.inputStatus = new EventEmitter();
78
+ /**
79
+ * Autocomplete bottom reached handler
80
+ */
81
+ this.bottomReached = new EventEmitter();
82
+ /**
83
+ * Subscriptions property to handle all subscriptions
84
+ * @ignore
85
+ */
86
+ this.subscriptions = new Subscription();
87
+ }
88
+ /**
89
+ * Component lifecycle method
90
+ * @ignore
91
+ */
92
+ ngOnInit() {
93
+ this.filteredOptions = this.formControl.valueChanges.pipe(startWith(''), map(value => {
94
+ const option = typeof value === 'string' ? value : value?.ViewValue;
95
+ return option ? this._filterOptions(option) : this.options.slice();
96
+ }));
97
+ }
98
+ /**
99
+ * Component lifecycle method
100
+ * @ignore
101
+ */
102
+ ngAfterViewInit() {
103
+ this.initAutocompleteListener();
104
+ }
105
+ /**
106
+ * Starts the autocomplete subscriptions
107
+ * @ignore
108
+ */
109
+ initAutocompleteListener() {
110
+ const openedSubscription = this.onAutocompleteOpened();
111
+ const closedSubscription = this.onAutocompleteClosed();
112
+ this.subscriptions.add(openedSubscription);
113
+ this.subscriptions.add(closedSubscription);
114
+ }
115
+ /**
116
+ * Handles the autocomplete opened event
117
+ * @ignore
118
+ */
119
+ onAutocompleteOpened() {
120
+ return this.matAutocomplete.opened.subscribe(() => {
121
+ setTimeout(() => {
122
+ const panel = this.matAutocomplete.panel?.nativeElement;
123
+ if (panel) {
124
+ panel.addEventListener('scroll', this.onScroll.bind(this));
125
+ }
126
+ });
127
+ });
128
+ }
129
+ /**
130
+ * Handles the autocomplete closed event
131
+ * @ignore
132
+ */
133
+ onAutocompleteClosed() {
134
+ return this.matAutocomplete.closed.subscribe(() => {
135
+ const panel = this.matAutocomplete.panel?.nativeElement;
136
+ if (panel) {
137
+ panel.removeEventListener('scroll', this.onScroll.bind(this));
138
+ }
139
+ });
140
+ }
141
+ /**
142
+ * Handles the scrolling event
143
+ * @ignore
144
+ */
145
+ onScroll(event) {
146
+ const panel = event.target;
147
+ const scrollTop = panel.scrollTop;
148
+ const scrollHeight = panel.scrollHeight;
149
+ const offsetHeight = panel.offsetHeight;
150
+ if (scrollTop + offsetHeight >= scrollHeight) {
151
+ this.onScrolledToEnd();
152
+ }
153
+ }
154
+ /**
155
+ * Emits the bottomReached when the user reaches the end of the autocomplete panel
156
+ * @ignore
157
+ */
158
+ onScrolledToEnd() {
159
+ this.bottomReached.emit();
160
+ }
161
+ /**
162
+ * Conditions the options' view to display the view value
163
+ * @ignore
164
+ */
165
+ displayFn(option) {
166
+ return option?.ViewValue ? option.ViewValue : '';
167
+ }
168
+ /**
169
+ * Filters the options according to the input value
170
+ * @ignore
171
+ */
172
+ _filterOptions(value) {
173
+ const filterValue = value.toLowerCase();
174
+ return this.options.filter(option => option.ViewValue.toLowerCase().includes(filterValue));
175
+ }
176
+ /**
177
+ * Emits the selected option every time it changes
178
+ * @ignore
179
+ */
180
+ onSelectionChanged(option) {
181
+ this.optionSelected.emit(option.Value);
182
+ }
183
+ /**
184
+ * Emits the formControl variable every time the input value changes
185
+ * @ignore
186
+ */
187
+ onInputChanged() {
188
+ this.inputStatus.emit(this.formControl);
189
+ }
190
+ /**
191
+ * Unsubscribe from all subscriptions when the component is destroyed
192
+ * @ignore
193
+ */
194
+ ngOnDestroy() {
195
+ this.subscriptions?.unsubscribe();
196
+ }
197
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.10", ngImport: i0, type: TkFormAutocompleteFieldComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
198
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.10", type: TkFormAutocompleteFieldComponent, isStandalone: true, selector: "lib-tk-form-autocomplete-field", inputs: { fieldWidth: "fieldWidth", theme: "theme", appearance: "appearance", subscriptSizing: "subscriptSizing", autoActiveFirstOption: "autoActiveFirstOption", autoSelectActiveOption: "autoSelectActiveOption", type: "type", required: "required", fieldTitle: "fieldTitle", fieldLabel: "fieldLabel", placeHolder: "placeHolder", options: "options" }, outputs: { optionSelected: "optionSelected", inputStatus: "inputStatus", bottomReached: "bottomReached" }, viewQueries: [{ propertyName: "matAutocomplete", first: true, predicate: ["auto"], descendants: true }], ngImport: i0, template: "<section\n class=\"section\"\n [ngStyle]=\"{ width: fieldWidth ? 'fit-content' : '100%' }\">\n @if (fieldTitle) {\n <h1 class=\"field__title\">{{ fieldTitle }}</h1>\n }\n <mat-form-field\n class=\"form__field\"\n [color]=\"theme\"\n [appearance]=\"appearance\"\n [subscriptSizing]=\"subscriptSizing\"\n [ngStyle]=\"{ width: fieldWidth ? fieldWidth : '100%' }\">\n <mat-label class=\"form__field__label\">{{ fieldLabel }}</mat-label>\n <input\n matInput\n [type]=\"type\"\n aria-label=\"Option\"\n [required]=\"required\"\n [matAutocomplete]=\"auto\"\n (input)=\"onInputChanged()\"\n [formControl]=\"formControl\"\n [placeholder]=\"placeHolder\" />\n <mat-autocomplete\n #auto=\"matAutocomplete\"\n [displayWith]=\"displayFn\"\n [autoActiveFirstOption]=\"autoActiveFirstOption\"\n [autoSelectActiveOption]=\"autoSelectActiveOption\"\n (optionSelected)=\"onSelectionChanged($event.option.value)\">\n @for (option of filteredOptions | async; track option.Value) {\n <mat-option [value]=\"option\">\n <span>{{ option.ViewValue }}</span>\n </mat-option>\n }\n </mat-autocomplete>\n </mat-form-field>\n</section>\n", styles: [".section{gap:.5rem;display:flex;flex-wrap:wrap;max-width:100%;align-items:center;justify-content:flex-start}.form__field{flex:1;height:100%;width:20rem;display:flex;max-width:100%;min-width:10rem;font-size:clamp(.8rem,.657rem + .381vw,1rem)}.form__field__label{font-weight:400;font-style:normal;line-height:.98438em;font-size:clamp(.688rem,.554rem + .357vw,.875rem)}.field__title{margin:0;padding:0;max-width:100%;font-weight:500;overflow:hidden;font-style:normal;align-self:center;white-space:nowrap;text-overflow:ellipsis;font-size:clamp(.9rem,.852rem + .276vw,1.1rem)}\n"], dependencies: [{ kind: "directive", type: NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i2.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i4.MatAutocomplete, selector: "mat-autocomplete", inputs: ["aria-label", "aria-labelledby", "displayWith", "autoActiveFirstOption", "autoSelectActiveOption", "requireSelection", "panelWidth", "disableRipple", "class", "hideSingleSelectionIndicator"], outputs: ["optionSelected", "opened", "closed", "optionActivated"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i5.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "directive", type: i4.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", inputs: ["matAutocomplete", "matAutocompletePosition", "matAutocompleteConnectedTo", "autocomplete", "matAutocompleteDisabled"], exportAs: ["matAutocompleteTrigger"] }] }); }
199
+ }
200
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.10", ngImport: i0, type: TkFormAutocompleteFieldComponent, decorators: [{
201
+ type: Component,
202
+ args: [{ selector: 'lib-tk-form-autocomplete-field', standalone: true, imports: [
203
+ NgStyle,
204
+ AsyncPipe,
205
+ FormsModule,
206
+ MatInputModule,
207
+ MatFormFieldModule,
208
+ ReactiveFormsModule,
209
+ MatAutocompleteModule,
210
+ ], template: "<section\n class=\"section\"\n [ngStyle]=\"{ width: fieldWidth ? 'fit-content' : '100%' }\">\n @if (fieldTitle) {\n <h1 class=\"field__title\">{{ fieldTitle }}</h1>\n }\n <mat-form-field\n class=\"form__field\"\n [color]=\"theme\"\n [appearance]=\"appearance\"\n [subscriptSizing]=\"subscriptSizing\"\n [ngStyle]=\"{ width: fieldWidth ? fieldWidth : '100%' }\">\n <mat-label class=\"form__field__label\">{{ fieldLabel }}</mat-label>\n <input\n matInput\n [type]=\"type\"\n aria-label=\"Option\"\n [required]=\"required\"\n [matAutocomplete]=\"auto\"\n (input)=\"onInputChanged()\"\n [formControl]=\"formControl\"\n [placeholder]=\"placeHolder\" />\n <mat-autocomplete\n #auto=\"matAutocomplete\"\n [displayWith]=\"displayFn\"\n [autoActiveFirstOption]=\"autoActiveFirstOption\"\n [autoSelectActiveOption]=\"autoSelectActiveOption\"\n (optionSelected)=\"onSelectionChanged($event.option.value)\">\n @for (option of filteredOptions | async; track option.Value) {\n <mat-option [value]=\"option\">\n <span>{{ option.ViewValue }}</span>\n </mat-option>\n }\n </mat-autocomplete>\n </mat-form-field>\n</section>\n", styles: [".section{gap:.5rem;display:flex;flex-wrap:wrap;max-width:100%;align-items:center;justify-content:flex-start}.form__field{flex:1;height:100%;width:20rem;display:flex;max-width:100%;min-width:10rem;font-size:clamp(.8rem,.657rem + .381vw,1rem)}.form__field__label{font-weight:400;font-style:normal;line-height:.98438em;font-size:clamp(.688rem,.554rem + .357vw,.875rem)}.field__title{margin:0;padding:0;max-width:100%;font-weight:500;overflow:hidden;font-style:normal;align-self:center;white-space:nowrap;text-overflow:ellipsis;font-size:clamp(.9rem,.852rem + .276vw,1.1rem)}\n"] }]
211
+ }], propDecorators: { fieldWidth: [{
212
+ type: Input
213
+ }], theme: [{
214
+ type: Input
215
+ }], appearance: [{
216
+ type: Input
217
+ }], subscriptSizing: [{
218
+ type: Input
219
+ }], autoActiveFirstOption: [{
220
+ type: Input
221
+ }], autoSelectActiveOption: [{
222
+ type: Input
223
+ }], type: [{
224
+ type: Input
225
+ }], required: [{
226
+ type: Input
227
+ }], fieldTitle: [{
228
+ type: Input
229
+ }], fieldLabel: [{
230
+ type: Input
231
+ }], placeHolder: [{
232
+ type: Input
233
+ }], options: [{
234
+ type: Input
235
+ }], optionSelected: [{
236
+ type: Output
237
+ }], inputStatus: [{
238
+ type: Output
239
+ }], bottomReached: [{
240
+ type: Output
241
+ }], matAutocomplete: [{
242
+ type: ViewChild,
243
+ args: ['auto']
244
+ }] } });
245
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tk-form-autocomplete-field.component.js","sourceRoot":"","sources":["../../../../../../projects/design-system/components/forms/tk-form-autocomplete-field/tk-form-autocomplete-field.component.ts","../../../../../../projects/design-system/components/forms/tk-form-autocomplete-field/tk-form-autocomplete-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Input,\n  Output,\n  OnInit,\n  Component,\n  ViewChild,\n  OnDestroy,\n  EventEmitter,\n  AfterViewInit,\n} from '@angular/core';\nimport {\n  SubscriptSizing,\n  MatFormFieldModule,\n  MatFormFieldAppearance,\n} from '@angular/material/form-field';\nimport {\n  MatAutocomplete,\n  MatAutocompleteModule,\n} from '@angular/material/autocomplete';\nimport { Option } from './models/option.model';\nimport { AsyncPipe, NgStyle } from '@angular/common';\nimport { ThemePalette } from '@angular/material/core';\nimport { MatInputModule } from '@angular/material/input';\nimport { map, Observable, startWith, Subscription } from 'rxjs';\nimport { FormControl, FormsModule, ReactiveFormsModule } from '@angular/forms';\n\n@Component({\n  selector: 'lib-tk-form-autocomplete-field',\n  standalone: true,\n  imports: [\n    NgStyle,\n    AsyncPipe,\n    FormsModule,\n    MatInputModule,\n    MatFormFieldModule,\n    ReactiveFormsModule,\n    MatAutocompleteModule,\n  ],\n  templateUrl: './tk-form-autocomplete-field.component.html',\n  styleUrl: './tk-form-autocomplete-field.component.scss',\n})\nexport class TkFormAutocompleteFieldComponent\n  implements OnInit, OnDestroy, AfterViewInit\n{\n  /**\n   * Input field control\n   * @ignore\n   */\n  public formControl = new FormControl<string | Option>('');\n\n  /**\n   * Optional input field width\n   * @default 100%\n   */\n  @Input()\n  public fieldWidth?: string;\n\n  /**\n   * What color palette to use\n   * @default 'accent'\n   */\n  @Input()\n  public theme?: ThemePalette = 'accent';\n\n  /**\n   * Form field appearance\n   * @default 'fill'\n   */\n  @Input()\n  public appearance: MatFormFieldAppearance = 'fill';\n\n  /**\n   * Form field reserved space for one line by default.\n   * @default 'dynamic'\n   */\n  @Input()\n  public subscriptSizing: SubscriptSizing = 'dynamic';\n\n  /**\n   * Whether the first option should be highlighted when the autocomplete panel is opened\n   * @default true\n   */\n  @Input()\n  public autoActiveFirstOption?: boolean = true;\n\n  /**\n   * Whether the active option should be selected as the user is navigating.\n   * @default true\n   */\n  @Input()\n  public autoSelectActiveOption?: boolean = true;\n\n  /**\n   * Optional field input type\n   * @default 'text'\n   */\n  @Input()\n  public type = 'text';\n\n  /**\n   * Whether the field is required\n   * @default true\n   */\n  @Input()\n  public required = true;\n\n  /**\n   * Optional input field title\n   * @default null\n   */\n  @Input()\n  public fieldTitle?: string;\n\n  /**\n   * Input field label\n   */\n  @Input()\n  public fieldLabel = '';\n\n  /**\n   * Input field placeholder\n   * @default ''\n   */\n  @Input()\n  public placeHolder = '';\n\n  /**\n   * Filtered options\n   * @ignore\n   */\n  public filteredOptions!: Observable<Option[]>;\n\n  /**\n   * Options to be displayed\n   * @ignore\n   */\n  @Input()\n  public options: Option[] = [];\n\n  /**\n   * Option changes handler\n   */\n  @Output()\n  public optionSelected = new EventEmitter<string | number>();\n\n  /**\n   * Input field changes handler\n   */\n  @Output()\n  public inputStatus = new EventEmitter<FormControl>();\n\n  /**\n   * Autocomplete bottom reached handler\n   */\n  @Output()\n  public bottomReached = new EventEmitter<void>();\n\n  /**\n   * Subscriptions property to handle all subscriptions\n   * @ignore\n   */\n  private subscriptions: Subscription = new Subscription();\n\n  /**\n   * Mat-autocomplete reference\n   * @ignore\n   */\n  @ViewChild('auto') matAutocomplete!: MatAutocomplete;\n\n  /**\n   * Component lifecycle method\n   * @ignore\n   */\n  ngOnInit(): void {\n    this.filteredOptions = this.formControl.valueChanges.pipe(\n      startWith(''),\n      map(value => {\n        const option = typeof value === 'string' ? value : value?.ViewValue;\n        return option ? this._filterOptions(option) : this.options.slice();\n      })\n    );\n  }\n\n  /**\n   * Component lifecycle method\n   * @ignore\n   */\n  ngAfterViewInit(): void {\n    this.initAutocompleteListener();\n  }\n\n  /**\n   * Starts the autocomplete subscriptions\n   * @ignore\n   */\n  public initAutocompleteListener(): void {\n    const openedSubscription = this.onAutocompleteOpened();\n    const closedSubscription = this.onAutocompleteClosed();\n\n    this.subscriptions.add(openedSubscription);\n    this.subscriptions.add(closedSubscription);\n  }\n\n  /**\n   * Handles the autocomplete opened event\n   * @ignore\n   */\n  private onAutocompleteOpened(): Subscription {\n    return this.matAutocomplete.opened.subscribe(() => {\n      setTimeout(() => {\n        const panel = this.matAutocomplete.panel?.nativeElement;\n        if (panel) {\n          panel.addEventListener('scroll', this.onScroll.bind(this));\n        }\n      });\n    });\n  }\n\n  /**\n   * Handles the autocomplete closed event\n   * @ignore\n   */\n  private onAutocompleteClosed(): Subscription {\n    return this.matAutocomplete.closed.subscribe(() => {\n      const panel = this.matAutocomplete.panel?.nativeElement;\n      if (panel) {\n        panel.removeEventListener('scroll', this.onScroll.bind(this));\n      }\n    });\n  }\n\n  /**\n   * Handles the scrolling event\n   * @ignore\n   */\n  private onScroll(event: Event) {\n    const panel = event.target as HTMLElement;\n    const scrollTop = panel.scrollTop;\n    const scrollHeight = panel.scrollHeight;\n    const offsetHeight = panel.offsetHeight;\n\n    if (scrollTop + offsetHeight >= scrollHeight) {\n      this.onScrolledToEnd();\n    }\n  }\n\n  /**\n   * Emits the bottomReached when the user reaches the end of the autocomplete panel\n   * @ignore\n   */\n  private onScrolledToEnd(): void {\n    this.bottomReached.emit();\n  }\n\n  /**\n   * Conditions the options' view to display the view value\n   * @ignore\n   */\n  public displayFn(option: Option | null): string {\n    return option?.ViewValue ? option.ViewValue : '';\n  }\n\n  /**\n   * Filters the options according to the input value\n   * @ignore\n   */\n  private _filterOptions(value: string): Option[] {\n    const filterValue = value.toLowerCase();\n\n    return this.options.filter(option =>\n      option.ViewValue.toLowerCase().includes(filterValue)\n    );\n  }\n\n  /**\n   * Emits the selected option every time it changes\n   * @ignore\n   */\n  public onSelectionChanged(option: Option): void {\n    this.optionSelected.emit(option.Value);\n  }\n\n  /**\n   * Emits the formControl variable every time the input value changes\n   * @ignore\n   */\n  public onInputChanged(): void {\n    this.inputStatus.emit(this.formControl);\n  }\n\n  /**\n   * Unsubscribe from all subscriptions when the component is destroyed\n   * @ignore\n   */\n  ngOnDestroy() {\n    this.subscriptions?.unsubscribe();\n  }\n}\n","<section\n  class=\"section\"\n  [ngStyle]=\"{ width: fieldWidth ? 'fit-content' : '100%' }\">\n  @if (fieldTitle) {\n  <h1 class=\"field__title\">{{ fieldTitle }}</h1>\n  }\n  <mat-form-field\n    class=\"form__field\"\n    [color]=\"theme\"\n    [appearance]=\"appearance\"\n    [subscriptSizing]=\"subscriptSizing\"\n    [ngStyle]=\"{ width: fieldWidth ? fieldWidth : '100%' }\">\n    <mat-label class=\"form__field__label\">{{ fieldLabel }}</mat-label>\n    <input\n      matInput\n      [type]=\"type\"\n      aria-label=\"Option\"\n      [required]=\"required\"\n      [matAutocomplete]=\"auto\"\n      (input)=\"onInputChanged()\"\n      [formControl]=\"formControl\"\n      [placeholder]=\"placeHolder\" />\n    <mat-autocomplete\n      #auto=\"matAutocomplete\"\n      [displayWith]=\"displayFn\"\n      [autoActiveFirstOption]=\"autoActiveFirstOption\"\n      [autoSelectActiveOption]=\"autoSelectActiveOption\"\n      (optionSelected)=\"onSelectionChanged($event.option.value)\">\n      @for (option of filteredOptions | async; track option.Value) {\n      <mat-option [value]=\"option\">\n        <span>{{ option.ViewValue }}</span>\n      </mat-option>\n      }\n    </mat-autocomplete>\n  </mat-form-field>\n</section>\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3B0aW9uLm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZGVzaWduLXN5c3RlbS9jb21wb25lbnRzL2Zvcm1zL3RrLWZvcm0tY2hpcHMtYXV0b2NvbXBsZXRlLWZpZWxkL21vZGVscy9vcHRpb24ubW9kZWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgT3B0aW9uIHtcbiAgVmlld1ZhbHVlOiBzdHJpbmc7XG4gIFZhbHVlOiBzdHJpbmcgfCBudW1iZXI7XG59XG4iXX0=
@@ -0,0 +1,2 @@
1
+ export * from './tk-form-chips-autocomplete-field.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Rlc2lnbi1zeXN0ZW0vY29tcG9uZW50cy9mb3Jtcy90ay1mb3JtLWNoaXBzLWF1dG9jb21wbGV0ZS1maWVsZC9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsOENBQThDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3RrLWZvcm0tY2hpcHMtYXV0b2NvbXBsZXRlLWZpZWxkLmNvbXBvbmVudCc7XG4iXX0=
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVrdXMtZGVzaWduLXN5c3RlbS1jb21wb25lbnRzLWZvcm1zLXRrLWZvcm0tY2hpcHMtYXV0b2NvbXBsZXRlLWZpZWxkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZGVzaWduLXN5c3RlbS9jb21wb25lbnRzL2Zvcm1zL3RrLWZvcm0tY2hpcHMtYXV0b2NvbXBsZXRlLWZpZWxkL3Rla3VzLWRlc2lnbi1zeXN0ZW0tY29tcG9uZW50cy1mb3Jtcy10ay1mb3JtLWNoaXBzLWF1dG9jb21wbGV0ZS1maWVsZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -0,0 +1,167 @@
1
+ import { Input, Output, Component, ViewChild, EventEmitter, ChangeDetectionStrategy, } from '@angular/core';
2
+ import { MatFormFieldModule, } from '@angular/material/form-field';
3
+ import { MatAutocompleteModule, } from '@angular/material/autocomplete';
4
+ import { map, startWith } from 'rxjs';
5
+ import { AsyncPipe, NgStyle } from '@angular/common';
6
+ import { COMMA, ENTER } from '@angular/cdk/keycodes';
7
+ import { MatIconModule } from '@angular/material/icon';
8
+ import { MatChipsModule } from '@angular/material/chips';
9
+ import { MatButtonModule } from '@angular/material/button';
10
+ import { FormControl, FormsModule, ReactiveFormsModule } from '@angular/forms';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "@angular/forms";
13
+ import * as i2 from "@angular/material/icon";
14
+ import * as i3 from "@angular/material/chips";
15
+ import * as i4 from "@angular/material/form-field";
16
+ import * as i5 from "@angular/material/autocomplete";
17
+ import * as i6 from "@angular/material/core";
18
+ export class TkFormChipsAutocompleteFieldComponent {
19
+ /**
20
+ * @ignore
21
+ * Class constructor
22
+ */
23
+ constructor() {
24
+ /**
25
+ * Input field label
26
+ */
27
+ this.fieldLabel = '';
28
+ /**
29
+ * Input field placeholder
30
+ * @default ''
31
+ */
32
+ this.placeHolder = '';
33
+ /**
34
+ * What color palette to use
35
+ * @default 'accent'
36
+ */
37
+ this.theme = 'accent';
38
+ /**
39
+ * Form field reserved space for one line by default.
40
+ * @default 'dynamic'
41
+ */
42
+ this.subscriptSizing = 'dynamic';
43
+ /**
44
+ * Form field appearance
45
+ * @default 'fill'
46
+ */
47
+ this.appearance = 'fill';
48
+ /**
49
+ * Input field control
50
+ * @ignore
51
+ */
52
+ this.formControl = new FormControl('');
53
+ /**
54
+ * Selected keywords
55
+ * @ignore
56
+ */
57
+ this.keywords = [];
58
+ /**
59
+ * Input field changes handler
60
+ */
61
+ this.inputStatus = new EventEmitter();
62
+ /**
63
+ * Keywords changes handler
64
+ */
65
+ this.selectedKeywords = new EventEmitter();
66
+ /**
67
+ * Separator constants
68
+ * @ignore
69
+ */
70
+ this.separatorKeysCodes = [ENTER, COMMA];
71
+ /**
72
+ * Options to be displayed
73
+ * @ignore
74
+ */
75
+ this.allKeywords = [];
76
+ this.filteredKeywords = this.formControl.valueChanges.pipe(startWith(null), map((keyword) => keyword ? this._filter(String(keyword)) : this.allKeywords.slice()));
77
+ }
78
+ /**
79
+ * Handles every keyword deletion
80
+ * @ignore
81
+ */
82
+ remove(keyword) {
83
+ const index = this.keywords.indexOf(keyword);
84
+ if (index >= 0) {
85
+ this.keywords.splice(index, 1);
86
+ this.onKeywordsChanged();
87
+ }
88
+ }
89
+ /**
90
+ * Emits the formControl variable every time the field value changes
91
+ * @ignore
92
+ */
93
+ onInputChanged() {
94
+ this.inputStatus.emit(this.formControl);
95
+ }
96
+ /**
97
+ * Emits the selected keywords every time they changed
98
+ * @ignore
99
+ */
100
+ onKeywordsChanged() {
101
+ this.selectedKeywords.emit(this.keywords.map(keyword => keyword.Value));
102
+ }
103
+ /**
104
+ * Handles the autocomplete selection
105
+ * @ignore
106
+ */
107
+ selected(event) {
108
+ const value = event.option.value;
109
+ const selectedKeyword = this.allKeywords.find(option => option.Value === value);
110
+ if (selectedKeyword &&
111
+ !this.keywords.some(keyword => keyword.Value === selectedKeyword.Value)) {
112
+ this.keywords.push(selectedKeyword);
113
+ this.onKeywordsChanged();
114
+ }
115
+ this.keywordInput.nativeElement.value = '';
116
+ this.formControl.setValue(null);
117
+ }
118
+ /**
119
+ * Filters the keywords based on the input
120
+ * @ignore
121
+ */
122
+ _filter(value) {
123
+ this.onInputChanged();
124
+ const filterValue = value.toLowerCase();
125
+ return this.allKeywords.filter(keyword => keyword.ViewValue.toLowerCase().includes(filterValue));
126
+ }
127
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.10", ngImport: i0, type: TkFormChipsAutocompleteFieldComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
128
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.10", type: TkFormChipsAutocompleteFieldComponent, isStandalone: true, selector: "lib-tk-form-chips-autocomplete-field", inputs: { fieldLabel: "fieldLabel", placeHolder: "placeHolder", fieldTitle: "fieldTitle", theme: "theme", subscriptSizing: "subscriptSizing", fieldWidth: "fieldWidth", appearance: "appearance", allKeywords: "allKeywords" }, outputs: { inputStatus: "inputStatus", selectedKeywords: "selectedKeywords" }, viewQueries: [{ propertyName: "keywordInput", first: true, predicate: ["keywordInput"], descendants: true }], ngImport: i0, template: "<section\n class=\"section\"\n [ngStyle]=\"{ width: fieldWidth ? 'fit-content' : '100%' }\">\n @if (fieldTitle) {\n <h1 class=\"field__title\">{{ fieldTitle }}</h1>\n }\n <mat-form-field\n class=\"form__field\"\n [color]=\"theme\"\n [appearance]=\"appearance\"\n [subscriptSizing]=\"subscriptSizing\"\n [ngStyle]=\"{ width: fieldWidth ? fieldWidth : '100%' }\">\n <mat-label class=\"form__field__label\">{{ fieldLabel }}</mat-label>\n <mat-chip-grid #chipGrid aria-label=\"Keyword selection\">\n @for (keyword of keywords; track keyword.Value) {\n <mat-chip-row (removed)=\"remove(keyword)\">\n {{ keyword.ViewValue }}\n <button matChipRemove [attr.aria-label]=\"'remove ' + keyword.ViewValue\">\n <mat-icon>cancel</mat-icon>\n </button>\n </mat-chip-row>\n }\n </mat-chip-grid>\n <input\n #keywordInput\n [matAutocomplete]=\"auto\"\n [formControl]=\"formControl\"\n [placeholder]=\"placeHolder\"\n [matChipInputFor]=\"chipGrid\"\n [matChipInputSeparatorKeyCodes]=\"separatorKeysCodes\" />\n <mat-autocomplete\n #auto=\"matAutocomplete\"\n (optionSelected)=\"selected($event)\">\n @for (keyword of filteredKeywords | async; track keyword.Value) {\n <mat-option [value]=\"keyword.Value\">{{ keyword.ViewValue }}</mat-option>\n }\n </mat-autocomplete>\n </mat-form-field>\n</section>\n", styles: [".section{gap:.5rem;display:flex;flex-wrap:wrap;max-width:100%;align-items:center;justify-content:flex-start}.form__field{flex:1;height:100%;width:20rem;display:flex;max-width:100%;min-width:10rem;font-size:clamp(.8rem,.657rem + .381vw,1rem)}.form__field__label{font-weight:400;font-style:normal;line-height:.98438em;font-size:clamp(.688rem,.554rem + .357vw,.875rem)}.field__title{margin:0;padding:0;max-width:100%;font-weight:500;overflow:hidden;font-style:normal;align-self:center;white-space:nowrap;text-overflow:ellipsis;font-size:clamp(.9rem,.852rem + .276vw,1.1rem)}\n"], dependencies: [{ kind: "directive", type: NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatChipsModule }, { kind: "component", type: i3.MatChipGrid, selector: "mat-chip-grid", inputs: ["disabled", "placeholder", "required", "value", "errorStateMatcher"], outputs: ["change", "valueChange"] }, { kind: "directive", type: i3.MatChipInput, selector: "input[matChipInputFor]", inputs: ["matChipInputFor", "matChipInputAddOnBlur", "matChipInputSeparatorKeyCodes", "placeholder", "id", "disabled"], outputs: ["matChipInputTokenEnd"], exportAs: ["matChipInput", "matChipInputFor"] }, { kind: "directive", type: i3.MatChipRemove, selector: "[matChipRemove]" }, { kind: "component", type: i3.MatChipRow, selector: "mat-chip-row, [mat-chip-row], mat-basic-chip-row, [mat-basic-chip-row]", inputs: ["editable"], outputs: ["edited"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i5.MatAutocomplete, selector: "mat-autocomplete", inputs: ["aria-label", "aria-labelledby", "displayWith", "autoActiveFirstOption", "autoSelectActiveOption", "requireSelection", "panelWidth", "disableRipple", "class", "hideSingleSelectionIndicator"], outputs: ["optionSelected", "opened", "closed", "optionActivated"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i6.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "directive", type: i5.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", inputs: ["matAutocomplete", "matAutocompletePosition", "matAutocompleteConnectedTo", "autocomplete", "matAutocompleteDisabled"], exportAs: ["matAutocompleteTrigger"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
129
+ }
130
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.10", ngImport: i0, type: TkFormChipsAutocompleteFieldComponent, decorators: [{
131
+ type: Component,
132
+ args: [{ selector: 'lib-tk-form-chips-autocomplete-field', standalone: true, imports: [
133
+ NgStyle,
134
+ AsyncPipe,
135
+ FormsModule,
136
+ MatIconModule,
137
+ MatChipsModule,
138
+ MatButtonModule,
139
+ MatFormFieldModule,
140
+ ReactiveFormsModule,
141
+ MatAutocompleteModule,
142
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<section\n class=\"section\"\n [ngStyle]=\"{ width: fieldWidth ? 'fit-content' : '100%' }\">\n @if (fieldTitle) {\n <h1 class=\"field__title\">{{ fieldTitle }}</h1>\n }\n <mat-form-field\n class=\"form__field\"\n [color]=\"theme\"\n [appearance]=\"appearance\"\n [subscriptSizing]=\"subscriptSizing\"\n [ngStyle]=\"{ width: fieldWidth ? fieldWidth : '100%' }\">\n <mat-label class=\"form__field__label\">{{ fieldLabel }}</mat-label>\n <mat-chip-grid #chipGrid aria-label=\"Keyword selection\">\n @for (keyword of keywords; track keyword.Value) {\n <mat-chip-row (removed)=\"remove(keyword)\">\n {{ keyword.ViewValue }}\n <button matChipRemove [attr.aria-label]=\"'remove ' + keyword.ViewValue\">\n <mat-icon>cancel</mat-icon>\n </button>\n </mat-chip-row>\n }\n </mat-chip-grid>\n <input\n #keywordInput\n [matAutocomplete]=\"auto\"\n [formControl]=\"formControl\"\n [placeholder]=\"placeHolder\"\n [matChipInputFor]=\"chipGrid\"\n [matChipInputSeparatorKeyCodes]=\"separatorKeysCodes\" />\n <mat-autocomplete\n #auto=\"matAutocomplete\"\n (optionSelected)=\"selected($event)\">\n @for (keyword of filteredKeywords | async; track keyword.Value) {\n <mat-option [value]=\"keyword.Value\">{{ keyword.ViewValue }}</mat-option>\n }\n </mat-autocomplete>\n </mat-form-field>\n</section>\n", styles: [".section{gap:.5rem;display:flex;flex-wrap:wrap;max-width:100%;align-items:center;justify-content:flex-start}.form__field{flex:1;height:100%;width:20rem;display:flex;max-width:100%;min-width:10rem;font-size:clamp(.8rem,.657rem + .381vw,1rem)}.form__field__label{font-weight:400;font-style:normal;line-height:.98438em;font-size:clamp(.688rem,.554rem + .357vw,.875rem)}.field__title{margin:0;padding:0;max-width:100%;font-weight:500;overflow:hidden;font-style:normal;align-self:center;white-space:nowrap;text-overflow:ellipsis;font-size:clamp(.9rem,.852rem + .276vw,1.1rem)}\n"] }]
143
+ }], ctorParameters: () => [], propDecorators: { fieldLabel: [{
144
+ type: Input
145
+ }], placeHolder: [{
146
+ type: Input
147
+ }], fieldTitle: [{
148
+ type: Input
149
+ }], theme: [{
150
+ type: Input
151
+ }], subscriptSizing: [{
152
+ type: Input
153
+ }], fieldWidth: [{
154
+ type: Input
155
+ }], appearance: [{
156
+ type: Input
157
+ }], inputStatus: [{
158
+ type: Output
159
+ }], selectedKeywords: [{
160
+ type: Output
161
+ }], allKeywords: [{
162
+ type: Input
163
+ }], keywordInput: [{
164
+ type: ViewChild,
165
+ args: ['keywordInput']
166
+ }] } });
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tk-form-chips-autocomplete-field.component.js","sourceRoot":"","sources":["../../../../../../projects/design-system/components/forms/tk-form-chips-autocomplete-field/tk-form-chips-autocomplete-field.component.ts","../../../../../../projects/design-system/components/forms/tk-form-chips-autocomplete-field/tk-form-chips-autocomplete-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Input,\n  Output,\n  Component,\n  ViewChild,\n  ElementRef,\n  EventEmitter,\n  ChangeDetectionStrategy,\n} from '@angular/core';\nimport {\n  SubscriptSizing,\n  MatFormFieldModule,\n  MatFormFieldAppearance,\n} from '@angular/material/form-field';\nimport {\n  MatAutocompleteModule,\n  MatAutocompleteSelectedEvent,\n} from '@angular/material/autocomplete';\nimport { Option } from './models/option.model';\nimport { map, Observable, startWith } from 'rxjs';\nimport { AsyncPipe, NgStyle } from '@angular/common';\nimport { COMMA, ENTER } from '@angular/cdk/keycodes';\nimport { ThemePalette } from '@angular/material/core';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatChipsModule } from '@angular/material/chips';\nimport { MatButtonModule } from '@angular/material/button';\nimport { FormControl, FormsModule, ReactiveFormsModule } from '@angular/forms';\n\n@Component({\n  selector: 'lib-tk-form-chips-autocomplete-field',\n  standalone: true,\n  imports: [\n    NgStyle,\n    AsyncPipe,\n    FormsModule,\n    MatIconModule,\n    MatChipsModule,\n    MatButtonModule,\n    MatFormFieldModule,\n    ReactiveFormsModule,\n    MatAutocompleteModule,\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  templateUrl: './tk-form-chips-autocomplete-field.component.html',\n  styleUrl: './tk-form-chips-autocomplete-field.component.scss',\n})\nexport class TkFormChipsAutocompleteFieldComponent {\n  /**\n   * Input field label\n   */\n  @Input()\n  public fieldLabel = '';\n\n  /**\n   * Input field placeholder\n   * @default ''\n   */\n  @Input()\n  public placeHolder = '';\n\n  /**\n   * Optional input field title\n   * @default null\n   */\n  @Input()\n  public fieldTitle?: string;\n\n  /**\n   * What color palette to use\n   * @default 'accent'\n   */\n  @Input()\n  public theme?: ThemePalette = 'accent';\n\n  /**\n   * Form field reserved space for one line by default.\n   * @default 'dynamic'\n   */\n  @Input()\n  public subscriptSizing: SubscriptSizing = 'dynamic';\n\n  /**\n   * Optional input field width\n   * @default 100%\n   */\n  @Input()\n  public fieldWidth?: string;\n\n  /**\n   * Form field appearance\n   * @default 'fill'\n   */\n  @Input()\n  public appearance: MatFormFieldAppearance = 'fill';\n\n  /**\n   * Input field control\n   * @ignore\n   */\n  public formControl = new FormControl('');\n  /**\n   * Selected keywords\n   * @ignore\n   */\n  public keywords: Option[] = [];\n\n  /**\n   * Input field changes handler\n   */\n  @Output()\n  public inputStatus = new EventEmitter<FormControl>();\n\n  /**\n   * Keywords changes handler\n   */\n  @Output()\n  public selectedKeywords = new EventEmitter<(string | number)[]>();\n\n  /**\n   * Separator constants\n   * @ignore\n   */\n  public readonly separatorKeysCodes: number[] = [ENTER, COMMA];\n\n  /**\n   * Options to be displayed\n   * @ignore\n   */\n  @Input()\n  public allKeywords: Option[] = [];\n\n  /**\n   * Filtered keywords after the input field changes\n   * @ignore\n   */\n  public readonly filteredKeywords: Observable<Option[]>;\n\n  /**\n   * Keyword Input element reference\n   * @ignore\n   */\n  @ViewChild('keywordInput') keywordInput!: ElementRef<HTMLInputElement>;\n\n  /**\n   * @ignore\n   * Class constructor\n   */\n  constructor() {\n    this.filteredKeywords = this.formControl.valueChanges.pipe(\n      startWith(null),\n      map((keyword: string | null) =>\n        keyword ? this._filter(String(keyword)) : this.allKeywords.slice()\n      )\n    );\n  }\n\n  /**\n   * Handles every keyword deletion\n   * @ignore\n   */\n  public remove(keyword: Option): void {\n    const index = this.keywords.indexOf(keyword);\n\n    if (index >= 0) {\n      this.keywords.splice(index, 1);\n      this.onKeywordsChanged();\n    }\n  }\n\n  /**\n   * Emits the formControl variable every time the field value changes\n   * @ignore\n   */\n  public onInputChanged(): void {\n    this.inputStatus.emit(this.formControl);\n  }\n\n  /**\n   * Emits the selected keywords every time they changed\n   * @ignore\n   */\n  public onKeywordsChanged(): void {\n    this.selectedKeywords.emit(this.keywords.map(keyword => keyword.Value));\n  }\n\n  /**\n   * Handles the autocomplete selection\n   * @ignore\n   */\n  public selected(event: MatAutocompleteSelectedEvent): void {\n    const value = event.option.value;\n    const selectedKeyword = this.allKeywords.find(\n      option => option.Value === value\n    );\n\n    if (\n      selectedKeyword &&\n      !this.keywords.some(keyword => keyword.Value === selectedKeyword.Value)\n    ) {\n      this.keywords.push(selectedKeyword);\n      this.onKeywordsChanged();\n    }\n\n    this.keywordInput.nativeElement.value = '';\n    this.formControl.setValue(null);\n  }\n\n  /**\n   * Filters the keywords based on the input\n   * @ignore\n   */\n  private _filter(value: string): Option[] {\n    this.onInputChanged();\n    const filterValue = value.toLowerCase();\n\n    return this.allKeywords.filter(keyword =>\n      keyword.ViewValue.toLowerCase().includes(filterValue)\n    );\n  }\n}\n","<section\n  class=\"section\"\n  [ngStyle]=\"{ width: fieldWidth ? 'fit-content' : '100%' }\">\n  @if (fieldTitle) {\n  <h1 class=\"field__title\">{{ fieldTitle }}</h1>\n  }\n  <mat-form-field\n    class=\"form__field\"\n    [color]=\"theme\"\n    [appearance]=\"appearance\"\n    [subscriptSizing]=\"subscriptSizing\"\n    [ngStyle]=\"{ width: fieldWidth ? fieldWidth : '100%' }\">\n    <mat-label class=\"form__field__label\">{{ fieldLabel }}</mat-label>\n    <mat-chip-grid #chipGrid aria-label=\"Keyword selection\">\n      @for (keyword of keywords; track keyword.Value) {\n      <mat-chip-row (removed)=\"remove(keyword)\">\n        {{ keyword.ViewValue }}\n        <button matChipRemove [attr.aria-label]=\"'remove ' + keyword.ViewValue\">\n          <mat-icon>cancel</mat-icon>\n        </button>\n      </mat-chip-row>\n      }\n    </mat-chip-grid>\n    <input\n      #keywordInput\n      [matAutocomplete]=\"auto\"\n      [formControl]=\"formControl\"\n      [placeholder]=\"placeHolder\"\n      [matChipInputFor]=\"chipGrid\"\n      [matChipInputSeparatorKeyCodes]=\"separatorKeysCodes\" />\n    <mat-autocomplete\n      #auto=\"matAutocomplete\"\n      (optionSelected)=\"selected($event)\">\n      @for (keyword of filteredKeywords | async; track keyword.Value) {\n      <mat-option [value]=\"keyword.Value\">{{ keyword.ViewValue }}</mat-option>\n      }\n    </mat-autocomplete>\n  </mat-form-field>\n</section>\n"]}