@testgorilla/tgo-ui 1.2.1 → 1.2.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/components/autocomplete/autocomplete.component.d.ts +106 -0
- package/components/autocomplete/autocomplete.component.module.d.ts +19 -0
- package/components/autocomplete/autocomplete.model.d.ts +13 -0
- package/components/autocomplete/includes.pipe.d.ts +12 -0
- package/components/autocomplete/transform-Item.pipe.d.ts +11 -0
- package/directives/select-text.directive.d.ts +19 -0
- package/esm2022/components/autocomplete/autocomplete.component.mjs +288 -0
- package/esm2022/components/autocomplete/autocomplete.component.module.mjs +81 -0
- package/esm2022/components/autocomplete/autocomplete.model.mjs +16 -0
- package/esm2022/components/autocomplete/includes.pipe.mjs +21 -0
- package/esm2022/components/autocomplete/transform-Item.pipe.mjs +29 -0
- package/esm2022/directives/select-text.directive.mjs +37 -0
- package/esm2022/public-api.mjs +5 -1
- package/esm2022/utils/autocomplete-utils.mjs +61 -0
- package/fesm2022/testgorilla-tgo-ui.mjs +505 -10
- package/fesm2022/testgorilla-tgo-ui.mjs.map +1 -1
- package/package.json +2 -2
- package/public-api.d.ts +3 -0
- package/src/assets/i18n/en.json +9 -1
- package/utils/autocomplete-utils.d.ts +11 -0
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import { AfterViewInit, ElementRef, EventEmitter, OnChanges } from '@angular/core';
|
|
2
|
+
import { ControlValueAccessor } from '@angular/forms';
|
|
3
|
+
import { MatAutocompleteSelectedEvent } from '@angular/material/autocomplete';
|
|
4
|
+
import { MatFormField } from '@angular/material/form-field';
|
|
5
|
+
import { ReplaySubject } from 'rxjs';
|
|
6
|
+
import { Autocomplete, AutocompleteType } from './autocomplete.model';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
export declare class AutocompleteComponent implements ControlValueAccessor, OnChanges, AfterViewInit {
|
|
9
|
+
/**
|
|
10
|
+
* @property itemsList
|
|
11
|
+
* @description The list of items to display in the autocomplete.
|
|
12
|
+
* @memberof AutocompleteComponent
|
|
13
|
+
*/
|
|
14
|
+
itemsList: any[];
|
|
15
|
+
/**
|
|
16
|
+
* @property suggestionsList
|
|
17
|
+
* @description The list of suggestions to display in the autocomplete.
|
|
18
|
+
* @memberof AutocompleteComponent
|
|
19
|
+
*/
|
|
20
|
+
suggestionsList: any[];
|
|
21
|
+
/**
|
|
22
|
+
* @property disabled
|
|
23
|
+
* @description If `true`, the autocomplete is disabled.
|
|
24
|
+
* @memberof AutocompleteComponent
|
|
25
|
+
*/
|
|
26
|
+
disabled: boolean;
|
|
27
|
+
/**
|
|
28
|
+
* @property allowAdd
|
|
29
|
+
* @description If `true`, allows adding new values to the autocomplete.
|
|
30
|
+
* @memberof AutocompleteComponent
|
|
31
|
+
*/
|
|
32
|
+
allowAdd: boolean;
|
|
33
|
+
/**
|
|
34
|
+
* @property textField
|
|
35
|
+
* @description The field to be used as the display text in the autocomplete.
|
|
36
|
+
* @memberof AutocompleteComponent
|
|
37
|
+
*/
|
|
38
|
+
textField: string;
|
|
39
|
+
/**
|
|
40
|
+
* @property valueField
|
|
41
|
+
* @description The field to be used as the value in the autocomplete.
|
|
42
|
+
* @memberof AutocompleteComponent
|
|
43
|
+
*/
|
|
44
|
+
valueField: string;
|
|
45
|
+
/**
|
|
46
|
+
* @property label
|
|
47
|
+
* @description The label for the autocomplete input.
|
|
48
|
+
* @memberof AutocompleteComponent
|
|
49
|
+
*/
|
|
50
|
+
label: string;
|
|
51
|
+
/**
|
|
52
|
+
* @property value
|
|
53
|
+
* @description The selected value(s) in the autocomplete.
|
|
54
|
+
* @memberof AutocompleteComponent
|
|
55
|
+
*/
|
|
56
|
+
set itemValue(v: any);
|
|
57
|
+
/**
|
|
58
|
+
* @property type
|
|
59
|
+
* @description The type of the autocomplete ('single', 'multi', etc.).
|
|
60
|
+
* @memberof AutocompleteComponent
|
|
61
|
+
*/
|
|
62
|
+
type: AutocompleteType;
|
|
63
|
+
/**
|
|
64
|
+
* @property minCharactersSearch
|
|
65
|
+
* @description The minimum characters to trigger search.
|
|
66
|
+
* @memberof AutocompleteComponent
|
|
67
|
+
*/
|
|
68
|
+
minCharactersSearch: number;
|
|
69
|
+
selectionChange: EventEmitter<any>;
|
|
70
|
+
searchTextChange: EventEmitter<string>;
|
|
71
|
+
matFormFieldElement: MatFormField;
|
|
72
|
+
tagContainer: ElementRef<HTMLElement>;
|
|
73
|
+
private readonly cdr;
|
|
74
|
+
protected value: any;
|
|
75
|
+
protected inputValue: any;
|
|
76
|
+
protected isInputFocus: boolean;
|
|
77
|
+
protected autocompleteType: typeof Autocomplete;
|
|
78
|
+
protected isOverLapping: boolean;
|
|
79
|
+
protected isOverlapChecking: boolean;
|
|
80
|
+
protected overlapIndex: number;
|
|
81
|
+
protected inputValueSearch$: ReplaySubject<string>;
|
|
82
|
+
protected filteredItemsList$: import("rxjs").Observable<any>;
|
|
83
|
+
protected filteredSuggestionList$: import("rxjs").Observable<any>;
|
|
84
|
+
protected searchResult$: import("rxjs").Observable<boolean>;
|
|
85
|
+
protected translationContext: string;
|
|
86
|
+
ngOnChanges(): void;
|
|
87
|
+
ngAfterViewInit(): void;
|
|
88
|
+
onChange: (_: any) => void;
|
|
89
|
+
onTouch: () => void;
|
|
90
|
+
registerOnChange(fn: any): void;
|
|
91
|
+
registerOnTouched(fn: any): void;
|
|
92
|
+
setDisabledState(isDisabled: boolean): void;
|
|
93
|
+
writeValue(value: any): void;
|
|
94
|
+
protected onOptionSelected(event: MatAutocompleteSelectedEvent): void;
|
|
95
|
+
protected onOptionRemoved(input: any, option: any): void;
|
|
96
|
+
protected onInputChange(event: Event): void;
|
|
97
|
+
protected onClear(): void;
|
|
98
|
+
protected refillInput(): void;
|
|
99
|
+
protected onFocus(): void;
|
|
100
|
+
private checkOverlap;
|
|
101
|
+
private onUpdate;
|
|
102
|
+
private initValue;
|
|
103
|
+
private addNewValue;
|
|
104
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<AutocompleteComponent, never>;
|
|
105
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<AutocompleteComponent, "ui-autocomplete", never, { "itemsList": { "alias": "itemsList"; "required": false; }; "suggestionsList": { "alias": "suggestionsList"; "required": false; }; "disabled": { "alias": "disabled"; "required": false; }; "allowAdd": { "alias": "allowAdd"; "required": false; }; "textField": { "alias": "textField"; "required": false; }; "valueField": { "alias": "valueField"; "required": false; }; "label": { "alias": "label"; "required": false; }; "itemValue": { "alias": "itemValue"; "required": false; }; "type": { "alias": "type"; "required": false; }; "minCharactersSearch": { "alias": "minCharactersSearch"; "required": false; }; }, { "selectionChange": "selectionChange"; "searchTextChange": "searchTextChange"; }, never, never, false, never>;
|
|
106
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import * as i0 from "@angular/core";
|
|
2
|
+
import * as i1 from "./autocomplete.component";
|
|
3
|
+
import * as i2 from "./transform-Item.pipe";
|
|
4
|
+
import * as i3 from "./includes.pipe";
|
|
5
|
+
import * as i4 from "../../directives/select-text.directive";
|
|
6
|
+
import * as i5 from "@angular/common";
|
|
7
|
+
import * as i6 from "@angular/material/form-field";
|
|
8
|
+
import * as i7 from "@angular/material/autocomplete";
|
|
9
|
+
import * as i8 from "@angular/material/input";
|
|
10
|
+
import * as i9 from "@angular/forms";
|
|
11
|
+
import * as i10 from "../tag/tag.component.module";
|
|
12
|
+
import * as i11 from "@angular/material/chips";
|
|
13
|
+
import * as i12 from "../icon/icon.component.module";
|
|
14
|
+
import * as i13 from "@ngneat/transloco";
|
|
15
|
+
export declare class AutocompleteComponentModule {
|
|
16
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<AutocompleteComponentModule, never>;
|
|
17
|
+
static ɵmod: i0.ɵɵNgModuleDeclaration<AutocompleteComponentModule, [typeof i1.AutocompleteComponent, typeof i2.TransformItemPipe, typeof i3.IncludesPipe, typeof i4.SelectTextDirective], [typeof i5.CommonModule, typeof i6.MatFormFieldModule, typeof i7.MatAutocompleteModule, typeof i8.MatInputModule, typeof i9.FormsModule, typeof i10.TagComponentModule, typeof i11.MatChipsModule, typeof i12.IconComponentModule, typeof i13.TranslocoModule], [typeof i1.AutocompleteComponent]>;
|
|
18
|
+
static ɵinj: i0.ɵɵInjectorDeclaration<AutocompleteComponentModule>;
|
|
19
|
+
}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
export type AutocompleteType = 'single' | 'multi' | 'large';
|
|
2
|
+
export declare enum Autocomplete {
|
|
3
|
+
SINGLE = "single",
|
|
4
|
+
MULTI = "multi",
|
|
5
|
+
LARGE = "large"
|
|
6
|
+
}
|
|
7
|
+
export declare enum Padding {
|
|
8
|
+
AUTOCOMPLETE_CONTAINER = 80
|
|
9
|
+
}
|
|
10
|
+
export declare enum Timeout {
|
|
11
|
+
VIEW_INITIALIZED_AFTER_ACTION = 20,
|
|
12
|
+
VIEW_INITIALIZED_COMPONENT_LOAD = 500
|
|
13
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { PipeTransform } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* The purpose of this pipe is to determine if the value is present in the values array.
|
|
5
|
+
* If values is an array, the method uses the includes method to check for inclusion.
|
|
6
|
+
* If values is not an array, it checks for strict equality between value and values.
|
|
7
|
+
*/
|
|
8
|
+
export declare class IncludesPipe implements PipeTransform {
|
|
9
|
+
transform(value: any, values: any): boolean;
|
|
10
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<IncludesPipe, never>;
|
|
11
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<IncludesPipe, "includes", false>;
|
|
12
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { PipeTransform } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* the TransformItemPipe is designed to selectively extract and return values from objects based on a specified field,
|
|
5
|
+
* providing a way to transform and display data a template.
|
|
6
|
+
*/
|
|
7
|
+
export declare class TransformItemPipe implements PipeTransform {
|
|
8
|
+
transform(value: any, field: string): any;
|
|
9
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<TransformItemPipe, never>;
|
|
10
|
+
static ɵpipe: i0.ɵɵPipeDeclaration<TransformItemPipe, "transformItem", false>;
|
|
11
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { ElementRef, OnChanges } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* the SelectTextDirective provides a dynamic way to highlight or style specific text within an element.
|
|
5
|
+
*/
|
|
6
|
+
export declare class SelectTextDirective implements OnChanges {
|
|
7
|
+
private el;
|
|
8
|
+
selectText: string;
|
|
9
|
+
constructor(el: ElementRef);
|
|
10
|
+
/**
|
|
11
|
+
* this method utilizes setTimeout to defer the execution until the next tick of the event loop.
|
|
12
|
+
* The text content of the element is modified by wrapping occurrences of the specified selectText with a <span> element,
|
|
13
|
+
* applying a bold font-weight style.
|
|
14
|
+
*/
|
|
15
|
+
private setTextColor;
|
|
16
|
+
ngOnChanges(): void;
|
|
17
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<SelectTextDirective, never>;
|
|
18
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<SelectTextDirective, "[selectText]", never, { "selectText": { "alias": "selectText"; "required": false; }; }, {}, never, never, false, never>;
|
|
19
|
+
}
|
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, forwardRef, inject, Input, Output, ViewChild, } from '@angular/core';
|
|
2
|
+
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
3
|
+
import { map, ReplaySubject, startWith } from 'rxjs';
|
|
4
|
+
import { AutocompleteUtils } from '../../utils/autocomplete-utils';
|
|
5
|
+
import { Autocomplete, Padding, Timeout } from './autocomplete.model';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/common";
|
|
8
|
+
import * as i2 from "@angular/material/form-field";
|
|
9
|
+
import * as i3 from "@angular/material/autocomplete";
|
|
10
|
+
import * as i4 from "@angular/material/core";
|
|
11
|
+
import * as i5 from "@angular/material/input";
|
|
12
|
+
import * as i6 from "@angular/forms";
|
|
13
|
+
import * as i7 from "../tag/tag.component";
|
|
14
|
+
import * as i8 from "../icon/icon.component";
|
|
15
|
+
import * as i9 from "@ngneat/transloco";
|
|
16
|
+
import * as i10 from "../../directives/select-text.directive";
|
|
17
|
+
import * as i11 from "./transform-Item.pipe";
|
|
18
|
+
import * as i12 from "./includes.pipe";
|
|
19
|
+
export class AutocompleteComponent {
|
|
20
|
+
constructor() {
|
|
21
|
+
/**
|
|
22
|
+
* @property itemsList
|
|
23
|
+
* @description The list of items to display in the autocomplete.
|
|
24
|
+
* @memberof AutocompleteComponent
|
|
25
|
+
*/
|
|
26
|
+
this.itemsList = [];
|
|
27
|
+
/**
|
|
28
|
+
* @property suggestionsList
|
|
29
|
+
* @description The list of suggestions to display in the autocomplete.
|
|
30
|
+
* @memberof AutocompleteComponent
|
|
31
|
+
*/
|
|
32
|
+
this.suggestionsList = [];
|
|
33
|
+
/**
|
|
34
|
+
* @property disabled
|
|
35
|
+
* @description If `true`, the autocomplete is disabled.
|
|
36
|
+
* @memberof AutocompleteComponent
|
|
37
|
+
*/
|
|
38
|
+
this.disabled = false;
|
|
39
|
+
/**
|
|
40
|
+
* @property allowAdd
|
|
41
|
+
* @description If `true`, allows adding new values to the autocomplete.
|
|
42
|
+
* @memberof AutocompleteComponent
|
|
43
|
+
*/
|
|
44
|
+
this.allowAdd = true;
|
|
45
|
+
/**
|
|
46
|
+
* @property textField
|
|
47
|
+
* @description The field to be used as the display text in the autocomplete.
|
|
48
|
+
* @memberof AutocompleteComponent
|
|
49
|
+
*/
|
|
50
|
+
this.textField = '';
|
|
51
|
+
/**
|
|
52
|
+
* @property valueField
|
|
53
|
+
* @description The field to be used as the value in the autocomplete.
|
|
54
|
+
* @memberof AutocompleteComponent
|
|
55
|
+
*/
|
|
56
|
+
this.valueField = '';
|
|
57
|
+
/**
|
|
58
|
+
* @property label
|
|
59
|
+
* @description The label for the autocomplete input.
|
|
60
|
+
* @memberof AutocompleteComponent
|
|
61
|
+
*/
|
|
62
|
+
this.label = '';
|
|
63
|
+
/**
|
|
64
|
+
* @property type
|
|
65
|
+
* @description The type of the autocomplete ('single', 'multi', etc.).
|
|
66
|
+
* @memberof AutocompleteComponent
|
|
67
|
+
*/
|
|
68
|
+
this.type = 'multi';
|
|
69
|
+
/**
|
|
70
|
+
* @property minCharactersSearch
|
|
71
|
+
* @description The minimum characters to trigger search.
|
|
72
|
+
* @memberof AutocompleteComponent
|
|
73
|
+
*/
|
|
74
|
+
this.minCharactersSearch = 3;
|
|
75
|
+
this.selectionChange = new EventEmitter();
|
|
76
|
+
this.searchTextChange = new EventEmitter();
|
|
77
|
+
this.cdr = inject(ChangeDetectorRef);
|
|
78
|
+
this.inputValue = '';
|
|
79
|
+
this.isInputFocus = false;
|
|
80
|
+
this.autocompleteType = Autocomplete;
|
|
81
|
+
this.isOverLapping = false;
|
|
82
|
+
this.isOverlapChecking = true;
|
|
83
|
+
this.overlapIndex = 0;
|
|
84
|
+
this.inputValueSearch$ = new ReplaySubject(1);
|
|
85
|
+
this.filteredItemsList$ = this.inputValueSearch$.pipe(startWith(''), map(inputValue => inputValue.length ? AutocompleteUtils.filterItems(this.itemsList, inputValue, this.textField) : null));
|
|
86
|
+
this.filteredSuggestionList$ = this.inputValueSearch$.pipe(map(inputValue => inputValue.length ? AutocompleteUtils.filterItems(this.suggestionsList, inputValue, this.textField) : null));
|
|
87
|
+
this.searchResult$ = this.inputValueSearch$.pipe(map(inputValue => !!inputValue.trim()));
|
|
88
|
+
this.translationContext = 'AUTOCOMPLETE.';
|
|
89
|
+
this.onChange = (_) => { };
|
|
90
|
+
this.onTouch = () => { };
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* @property value
|
|
94
|
+
* @description The selected value(s) in the autocomplete.
|
|
95
|
+
* @memberof AutocompleteComponent
|
|
96
|
+
*/
|
|
97
|
+
set itemValue(v) {
|
|
98
|
+
this.value = AutocompleteUtils.isArray(v) ? v : [v];
|
|
99
|
+
}
|
|
100
|
+
ngOnChanges() {
|
|
101
|
+
this.itemsList = AutocompleteUtils.excludeSuggestions(this.itemsList, this.suggestionsList, this.valueField);
|
|
102
|
+
this.initValue();
|
|
103
|
+
}
|
|
104
|
+
ngAfterViewInit() {
|
|
105
|
+
if (this.type !== Autocomplete.LARGE) {
|
|
106
|
+
this.checkOverlap(Timeout.VIEW_INITIALIZED_COMPONENT_LOAD);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
registerOnChange(fn) {
|
|
110
|
+
this.onChange = fn;
|
|
111
|
+
}
|
|
112
|
+
registerOnTouched(fn) {
|
|
113
|
+
this.onTouch = fn;
|
|
114
|
+
}
|
|
115
|
+
setDisabledState(isDisabled) {
|
|
116
|
+
this.disabled = isDisabled;
|
|
117
|
+
}
|
|
118
|
+
writeValue(value) {
|
|
119
|
+
this.value = value;
|
|
120
|
+
}
|
|
121
|
+
onOptionSelected(event) {
|
|
122
|
+
const inputValue = event.option.value;
|
|
123
|
+
const itemValue = AutocompleteUtils.createValueItem(inputValue, this.textField, this.valueField);
|
|
124
|
+
this.addNewValue(itemValue);
|
|
125
|
+
if (this.type === this.autocompleteType.MULTI) {
|
|
126
|
+
this.checkOverlap();
|
|
127
|
+
}
|
|
128
|
+
queueMicrotask(() => {
|
|
129
|
+
if (this.type !== this.autocompleteType.SINGLE) {
|
|
130
|
+
this.inputValue = '';
|
|
131
|
+
}
|
|
132
|
+
else {
|
|
133
|
+
this.inputValue = AutocompleteUtils.transformDisplayValue(this.value, this.textField);
|
|
134
|
+
}
|
|
135
|
+
this.cdr.markForCheck();
|
|
136
|
+
});
|
|
137
|
+
this.onUpdate();
|
|
138
|
+
}
|
|
139
|
+
onOptionRemoved(input, option) {
|
|
140
|
+
setTimeout(() => (input.showPanel = false), 0);
|
|
141
|
+
const foundValue = this.value.find((v) => v === option);
|
|
142
|
+
if (foundValue) {
|
|
143
|
+
this.value = this.value.filter((v) => v !== option);
|
|
144
|
+
}
|
|
145
|
+
if (this.type === this.autocompleteType.MULTI) {
|
|
146
|
+
this.checkOverlap();
|
|
147
|
+
}
|
|
148
|
+
if (!this.value.length) {
|
|
149
|
+
this.inputValue = '';
|
|
150
|
+
}
|
|
151
|
+
this.onUpdate();
|
|
152
|
+
}
|
|
153
|
+
onInputChange(event) {
|
|
154
|
+
const inputValue = event.target.value;
|
|
155
|
+
if (inputValue.length >= this.minCharactersSearch) {
|
|
156
|
+
this.inputValueSearch$.next(inputValue);
|
|
157
|
+
this.searchTextChange.emit(inputValue);
|
|
158
|
+
}
|
|
159
|
+
else {
|
|
160
|
+
this.inputValueSearch$.next('');
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
onClear() {
|
|
164
|
+
this.inputValue = '';
|
|
165
|
+
this.value = AutocompleteUtils.isArray(this.value) ? [] : '';
|
|
166
|
+
this.isOverLapping = false;
|
|
167
|
+
this.overlapIndex = 0;
|
|
168
|
+
this.onUpdate();
|
|
169
|
+
}
|
|
170
|
+
refillInput() {
|
|
171
|
+
if (this.value && this.type === this.autocompleteType.SINGLE && !this.inputValue) {
|
|
172
|
+
this.inputValue = AutocompleteUtils.transformDisplayValue(this.value, this.textField);
|
|
173
|
+
this.inputValueSearch$.next('');
|
|
174
|
+
}
|
|
175
|
+
if (!this.inputValue && this.value?.length) {
|
|
176
|
+
this.inputValue = ' ';
|
|
177
|
+
}
|
|
178
|
+
this.isInputFocus = false;
|
|
179
|
+
}
|
|
180
|
+
onFocus() {
|
|
181
|
+
this.isInputFocus = true;
|
|
182
|
+
const inputValue = AutocompleteUtils.isArray(this.inputValue) ? this.inputValue[0] : this.inputValue;
|
|
183
|
+
if (AutocompleteUtils.isPrimitive(inputValue)) {
|
|
184
|
+
if (inputValue?.match(/^\s+/gm)) {
|
|
185
|
+
this.inputValue = inputValue.trimStart();
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
else {
|
|
189
|
+
if (inputValue[this.textField]?.match(/^\s+/gm)) {
|
|
190
|
+
this.inputValue = inputValue[this.textField].trimStart();
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
checkOverlap(delay = Timeout.VIEW_INITIALIZED_AFTER_ACTION) {
|
|
195
|
+
this.isOverLapping = false;
|
|
196
|
+
this.isOverlapChecking = true;
|
|
197
|
+
setTimeout(() => {
|
|
198
|
+
const inputWidth = this.matFormFieldElement?._textField.nativeElement.clientWidth - Padding.AUTOCOMPLETE_CONTAINER;
|
|
199
|
+
const tags = this.tagContainer?.nativeElement.querySelectorAll('.value-tag');
|
|
200
|
+
if (tags) {
|
|
201
|
+
let width = 0;
|
|
202
|
+
tags.forEach((tag, index) => {
|
|
203
|
+
width += tag.getBoundingClientRect().width;
|
|
204
|
+
if (!this.isOverLapping && width > inputWidth) {
|
|
205
|
+
this.isOverLapping = true;
|
|
206
|
+
this.overlapIndex = index;
|
|
207
|
+
}
|
|
208
|
+
});
|
|
209
|
+
this.isOverlapChecking = false;
|
|
210
|
+
this.cdr.markForCheck();
|
|
211
|
+
}
|
|
212
|
+
}, delay);
|
|
213
|
+
}
|
|
214
|
+
onUpdate() {
|
|
215
|
+
this.inputValueSearch$.next('');
|
|
216
|
+
this.selectionChange.emit(this.value);
|
|
217
|
+
this.onChange(this.value);
|
|
218
|
+
}
|
|
219
|
+
initValue() {
|
|
220
|
+
if (this.type === this.autocompleteType.SINGLE) {
|
|
221
|
+
this.inputValue = AutocompleteUtils.transformDisplayValue(this.value[0], this.textField);
|
|
222
|
+
}
|
|
223
|
+
if (this.type === this.autocompleteType.MULTI) {
|
|
224
|
+
this.checkOverlap();
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
addNewValue(newValue) {
|
|
228
|
+
if (this.type !== this.autocompleteType.SINGLE && !this.value) {
|
|
229
|
+
this.value = [];
|
|
230
|
+
}
|
|
231
|
+
const foundDuplicate = AutocompleteUtils.findDuplicate(this.value, newValue, this.textField);
|
|
232
|
+
if (this.type !== this.autocompleteType.SINGLE) {
|
|
233
|
+
this.value = foundDuplicate ? this.value.filter((v) => v !== foundDuplicate) : [...this.value, newValue];
|
|
234
|
+
}
|
|
235
|
+
else {
|
|
236
|
+
this.value = newValue;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: AutocompleteComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
240
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: AutocompleteComponent, selector: "ui-autocomplete", inputs: { itemsList: "itemsList", suggestionsList: "suggestionsList", disabled: "disabled", allowAdd: "allowAdd", textField: "textField", valueField: "valueField", label: "label", itemValue: "itemValue", type: "type", minCharactersSearch: "minCharactersSearch" }, outputs: { selectionChange: "selectionChange", searchTextChange: "searchTextChange" }, providers: [
|
|
241
|
+
{
|
|
242
|
+
provide: NG_VALUE_ACCESSOR,
|
|
243
|
+
useExisting: forwardRef(() => AutocompleteComponent),
|
|
244
|
+
multi: true,
|
|
245
|
+
},
|
|
246
|
+
], viewQueries: [{ propertyName: "matFormFieldElement", first: true, predicate: ["matFormField"], descendants: true }, { propertyName: "tagContainer", first: true, predicate: ["tagContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<ng-container *transloco=\"let t\">\n <div class=\"autocomplete-wrapper\">\n <mat-form-field\n [appearance]=\"'outline'\"\n #matFormField\n [ngClass]=\"{ 'large-size': type === autocompleteType.LARGE, disabled: disabled }\"\n >\n <mat-label *ngIf=\"label\">{{ label }}</mat-label>\n <div class=\"input-container\">\n <div\n class=\"selected-items\"\n #tagContainer\n *ngIf=\"\n (type === autocompleteType.LARGE && this.value) ||\n (!(inputValue && this.isInputFocus) && this.value && type !== autocompleteType.SINGLE)\n \"\n >\n <ui-tag\n class=\"value-tag\"\n [id]=\"'ui-tag-' + i\"\n *ngFor=\"let valueItem of value; let i = index\"\n [label]=\"valueItem | transformItem: textField\"\n [allowClose]=\"true\"\n [style.display]=\"isOverLapping && i >= overlapIndex ? 'none' : 'block'\"\n [ngClass]=\"{\n 'overlap-tag': isOverlapChecking && overlapIndex && i >= overlapIndex && type === autocompleteType.MULTI\n }\"\n (close)=\"onOptionRemoved(autocomplete, valueItem)\"\n ></ui-tag>\n <ng-container *ngIf=\"isOverLapping\">\n <div *ngIf=\"!overlapIndex; else overlapTag\" class=\"overlap-count\">\n {{ t(translationContext + 'SELECTED_AMOUNT', { numberSelected: (value | slice: overlapIndex).length }) }}\n </div>\n <ng-template #overlapTag>\n <ui-tag [readOnly]=\"true\" [label]=\"'+' + (value | slice: overlapIndex).length\"></ui-tag>\n </ng-template>\n </ng-container>\n </div>\n <input\n [disabled]=\"disabled\"\n [ngClass]=\"{ 'unset-margin': this.value && type === autocompleteType.LARGE }\"\n (blur)=\"refillInput()\"\n (focusin)=\"onFocus()\"\n matInput\n [type]=\"'text'\"\n [matAutocomplete]=\"autocomplete\"\n (input)=\"onInputChange($event)\"\n [(ngModel)]=\"inputValue\"\n />\n </div>\n\n <ui-icon\n class=\"remove-selected\"\n [ngClass]=\"{ 'large-input-icon': type === autocompleteType.LARGE }\"\n (click)=\"onClear()\"\n *ngIf=\"inputValue || value?.length\"\n [size]=\"'24'\"\n [name]=\"'Close'\"\n ></ui-icon>\n\n <mat-autocomplete\n #autocomplete\n (optionSelected)=\"onOptionSelected($event)\"\n [hideSingleSelectionIndicator]=\"false\"\n >\n <ng-container *ngIf=\"!(searchResult$ | async)\">\n <mat-optgroup [label]=\"t(translationContext + 'SUGGESTED')\" *ngIf=\"suggestionsList.length\">\n <mat-option\n [ngClass]=\"{ 'selected-option': suggested | includes: value }\"\n *ngFor=\"let suggested of suggestionsList\"\n [value]=\"suggested\"\n >\n <span [selectText]=\"inputValue\">{{ suggested | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n\n <mat-optgroup [label]=\"t(translationContext + 'SELECTED') | uppercase\" *ngIf=\"value?.length && type !== autocompleteType.SINGLE\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of value\"\n [value]=\"item\"\n >\n <span>{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n </ng-container>\n\n <ng-container *ngIf=\"filteredSuggestionList$ | async as filteredSuggestionList\">\n <ng-container *ngIf=\"filteredItemsList$ | async as filteredItemsList\">\n <mat-optgroup [label]=\"t(translationContext + 'SUGGESTED') | uppercase\" *ngIf=\"filteredSuggestionList.length\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of filteredSuggestionList\"\n [value]=\"item\"\n >\n <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n\n <ng-container *ngIf=\"!filteredSuggestionList.length; else showItemListWithGroup\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of filteredItemsList\"\n [value]=\"item\"\n >\n <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </ng-container>\n\n <ng-template #showItemListWithGroup>\n <mat-optgroup [label]=\"t(translationContext + 'ALL_ITEMS') | uppercase\" *ngIf=\"filteredItemsList.length\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of filteredItemsList\"\n [value]=\"item\"\n >\n <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n </ng-template>\n\n <ng-container *ngIf=\"!filteredItemsList.length && !filteredSuggestionList.length && inputValue\">\n <ng-container *ngIf=\"allowAdd; else notFound\">\n <mat-option [value]=\"inputValue\">\n <span>{{ t('COMMON.ADD') }}</span>\n <span class=\"add-suggestion\">\"{{ inputValue }}\"</span>\n </mat-option>\n </ng-container>\n\n <ng-template #notFound>\n <mat-option [style.pointer-events]=\"'none'\">\n <span>{{ t(translationContext + 'NO_RESULTS_FOUND') }}</span>\n </mat-option>\n </ng-template>\n </ng-container>\n </ng-container>\n </ng-container>\n </mat-autocomplete>\n </mat-form-field>\n </div>\n</ng-container>\n", styles: [".large-size input{margin:10px 0}.large-size .selected-items{margin:5px 0;flex-wrap:wrap}.large-size .input-container{display:block}.large-size ::ng-deep .mat-mdc-form-field-infix,.large-size .mat-mdc-form-field-flex{align-items:unset!important}.disabled ui-icon{opacity:.5}.input-container{width:100%;margin:auto;display:flex}.input-container .selected-items{display:flex}.input-container .selected-items .overlap-count{font-size:14px;line-height:20px;width:80px}.input-container .selected-items .overlap-tag{position:absolute;opacity:0}.input-container input{margin-left:4px}.input-container input.unset-margin{margin-top:unset}.add-suggestion{margin:0 5px;color:#000;font-weight:700}.remove-selected{cursor:pointer}.remove-selected.large-input-icon{margin-top:10px}::ng-deep .mdc-menu-surface.mat-mdc-autocomplete-panel{box-shadow:0 8px 24px 4px #00000014;max-height:312px!important}::ng-deep .mat-mdc-form-field{min-width:330px!important}::ng-deep .mat-mdc-option:hover:not(.mdc-list-item--disabled),::ng-deep .mat-mdc-option:focus:not(.mdc-list-item--disabled),::ng-deep .mat-mdc-option.mat-mdc-option-active,::ng-deep .mat-mdc-option.mdc-list-item--selected:not(.mat-mdc-option-multiple):not(.mdc-list-item--disabled){background:#E9F0F1!important}::ng-deep .mat-mdc-text-field-wrapper.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{top:28px!important}::ng-deep .mat-mdc-text-field-wrapper .mat-mdc-form-field-flex .mat-mdc-floating-label{top:50%!important}::ng-deep .mat-mdc-text-field-wrapper{padding-left:12px!important}::ng-deep .mat-focused .mat-mdc-text-field-wrapper .mat-mdc-form-field-flex .mat-mdc-floating-label{color:#276678!important}::ng-deep .mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch{border-bottom:2px solid #276678!important}::ng-deep .mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading{border-left:2px solid #276678!important;border-bottom:2px solid #276678!important;border-top:2px solid #276678!important}::ng-deep .mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border:2px solid #276678!important;border-left:unset!important}::ng-deep .mat-mdc-form-field-infix,::ng-deep .mat-mdc-form-field-flex{min-height:48px!important;max-height:184px!important;overflow-y:auto;overflow-x:hidden;padding:0!important;display:flex;justify-content:center;text-align:center;align-items:center;flex-direction:row;width:100%}::ng-deep .mdc-line-ripple{display:none!important}::ng-deep .mat-mdc-option{padding-right:32px;padding-left:12px!important}::ng-deep .mat-mdc-option ui-icon{display:none;position:absolute;right:0;bottom:calc(50% - 8px);margin:0 10px}::ng-deep .mat-mdc-option.selected-option{background:#E9F0F1}::ng-deep .mat-mdc-option.selected-option ui-icon{display:block}::ng-deep .mat-mdc-option.mdc-list-item--selected .mdc-list-item__primary-text,::ng-deep .mat-mdc-option .mdc-list-item--activated .mdc-list-item__primary-text{color:#000!important}::ng-deep .mat-mdc-option .mat-pseudo-checkbox{display:none}::ng-deep .mat-mdc-optgroup .mdc-list-item__primary-text{color:#000;font-size:12px;line-height:16px;font-weight:700}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i2.MatLabel, selector: "mat-label" }, { kind: "component", type: i3.MatAutocomplete, selector: "mat-autocomplete", inputs: ["disableRipple", "hideSingleSelectionIndicator"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i4.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "component", type: i4.MatOptgroup, selector: "mat-optgroup", inputs: ["disabled"], exportAs: ["matOptgroup"] }, { kind: "directive", type: i3.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { kind: "directive", type: i5.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: "directive", type: i6.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: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i6.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i7.TagComponent, selector: "ui-tag", inputs: ["label", "icon", "allowClose", "readOnly", "isSelected", "showIconWhenSelected", "isDisabled"], outputs: ["close", "press"] }, { kind: "component", type: i8.IconComponent, selector: "ui-icon", inputs: ["size", "cssClass", "name", "color"] }, { kind: "directive", type: i9.TranslocoDirective, selector: "[transloco]", inputs: ["transloco", "translocoParams", "translocoScope", "translocoRead", "translocoLang", "translocoLoadingTpl"] }, { kind: "directive", type: i10.SelectTextDirective, selector: "[selectText]", inputs: ["selectText"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i1.UpperCasePipe, name: "uppercase" }, { kind: "pipe", type: i1.SlicePipe, name: "slice" }, { kind: "pipe", type: i11.TransformItemPipe, name: "transformItem" }, { kind: "pipe", type: i12.IncludesPipe, name: "includes" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
247
|
+
}
|
|
248
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: AutocompleteComponent, decorators: [{
|
|
249
|
+
type: Component,
|
|
250
|
+
args: [{ selector: 'ui-autocomplete', providers: [
|
|
251
|
+
{
|
|
252
|
+
provide: NG_VALUE_ACCESSOR,
|
|
253
|
+
useExisting: forwardRef(() => AutocompleteComponent),
|
|
254
|
+
multi: true,
|
|
255
|
+
},
|
|
256
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *transloco=\"let t\">\n <div class=\"autocomplete-wrapper\">\n <mat-form-field\n [appearance]=\"'outline'\"\n #matFormField\n [ngClass]=\"{ 'large-size': type === autocompleteType.LARGE, disabled: disabled }\"\n >\n <mat-label *ngIf=\"label\">{{ label }}</mat-label>\n <div class=\"input-container\">\n <div\n class=\"selected-items\"\n #tagContainer\n *ngIf=\"\n (type === autocompleteType.LARGE && this.value) ||\n (!(inputValue && this.isInputFocus) && this.value && type !== autocompleteType.SINGLE)\n \"\n >\n <ui-tag\n class=\"value-tag\"\n [id]=\"'ui-tag-' + i\"\n *ngFor=\"let valueItem of value; let i = index\"\n [label]=\"valueItem | transformItem: textField\"\n [allowClose]=\"true\"\n [style.display]=\"isOverLapping && i >= overlapIndex ? 'none' : 'block'\"\n [ngClass]=\"{\n 'overlap-tag': isOverlapChecking && overlapIndex && i >= overlapIndex && type === autocompleteType.MULTI\n }\"\n (close)=\"onOptionRemoved(autocomplete, valueItem)\"\n ></ui-tag>\n <ng-container *ngIf=\"isOverLapping\">\n <div *ngIf=\"!overlapIndex; else overlapTag\" class=\"overlap-count\">\n {{ t(translationContext + 'SELECTED_AMOUNT', { numberSelected: (value | slice: overlapIndex).length }) }}\n </div>\n <ng-template #overlapTag>\n <ui-tag [readOnly]=\"true\" [label]=\"'+' + (value | slice: overlapIndex).length\"></ui-tag>\n </ng-template>\n </ng-container>\n </div>\n <input\n [disabled]=\"disabled\"\n [ngClass]=\"{ 'unset-margin': this.value && type === autocompleteType.LARGE }\"\n (blur)=\"refillInput()\"\n (focusin)=\"onFocus()\"\n matInput\n [type]=\"'text'\"\n [matAutocomplete]=\"autocomplete\"\n (input)=\"onInputChange($event)\"\n [(ngModel)]=\"inputValue\"\n />\n </div>\n\n <ui-icon\n class=\"remove-selected\"\n [ngClass]=\"{ 'large-input-icon': type === autocompleteType.LARGE }\"\n (click)=\"onClear()\"\n *ngIf=\"inputValue || value?.length\"\n [size]=\"'24'\"\n [name]=\"'Close'\"\n ></ui-icon>\n\n <mat-autocomplete\n #autocomplete\n (optionSelected)=\"onOptionSelected($event)\"\n [hideSingleSelectionIndicator]=\"false\"\n >\n <ng-container *ngIf=\"!(searchResult$ | async)\">\n <mat-optgroup [label]=\"t(translationContext + 'SUGGESTED')\" *ngIf=\"suggestionsList.length\">\n <mat-option\n [ngClass]=\"{ 'selected-option': suggested | includes: value }\"\n *ngFor=\"let suggested of suggestionsList\"\n [value]=\"suggested\"\n >\n <span [selectText]=\"inputValue\">{{ suggested | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n\n <mat-optgroup [label]=\"t(translationContext + 'SELECTED') | uppercase\" *ngIf=\"value?.length && type !== autocompleteType.SINGLE\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of value\"\n [value]=\"item\"\n >\n <span>{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n </ng-container>\n\n <ng-container *ngIf=\"filteredSuggestionList$ | async as filteredSuggestionList\">\n <ng-container *ngIf=\"filteredItemsList$ | async as filteredItemsList\">\n <mat-optgroup [label]=\"t(translationContext + 'SUGGESTED') | uppercase\" *ngIf=\"filteredSuggestionList.length\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of filteredSuggestionList\"\n [value]=\"item\"\n >\n <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n\n <ng-container *ngIf=\"!filteredSuggestionList.length; else showItemListWithGroup\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of filteredItemsList\"\n [value]=\"item\"\n >\n <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </ng-container>\n\n <ng-template #showItemListWithGroup>\n <mat-optgroup [label]=\"t(translationContext + 'ALL_ITEMS') | uppercase\" *ngIf=\"filteredItemsList.length\">\n <mat-option\n [ngClass]=\"{ 'selected-option': item | includes: value }\"\n *ngFor=\"let item of filteredItemsList\"\n [value]=\"item\"\n >\n <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n <ui-icon [name]=\"'Check'\"></ui-icon>\n </mat-option>\n </mat-optgroup>\n </ng-template>\n\n <ng-container *ngIf=\"!filteredItemsList.length && !filteredSuggestionList.length && inputValue\">\n <ng-container *ngIf=\"allowAdd; else notFound\">\n <mat-option [value]=\"inputValue\">\n <span>{{ t('COMMON.ADD') }}</span>\n <span class=\"add-suggestion\">\"{{ inputValue }}\"</span>\n </mat-option>\n </ng-container>\n\n <ng-template #notFound>\n <mat-option [style.pointer-events]=\"'none'\">\n <span>{{ t(translationContext + 'NO_RESULTS_FOUND') }}</span>\n </mat-option>\n </ng-template>\n </ng-container>\n </ng-container>\n </ng-container>\n </mat-autocomplete>\n </mat-form-field>\n </div>\n</ng-container>\n", styles: [".large-size input{margin:10px 0}.large-size .selected-items{margin:5px 0;flex-wrap:wrap}.large-size .input-container{display:block}.large-size ::ng-deep .mat-mdc-form-field-infix,.large-size .mat-mdc-form-field-flex{align-items:unset!important}.disabled ui-icon{opacity:.5}.input-container{width:100%;margin:auto;display:flex}.input-container .selected-items{display:flex}.input-container .selected-items .overlap-count{font-size:14px;line-height:20px;width:80px}.input-container .selected-items .overlap-tag{position:absolute;opacity:0}.input-container input{margin-left:4px}.input-container input.unset-margin{margin-top:unset}.add-suggestion{margin:0 5px;color:#000;font-weight:700}.remove-selected{cursor:pointer}.remove-selected.large-input-icon{margin-top:10px}::ng-deep .mdc-menu-surface.mat-mdc-autocomplete-panel{box-shadow:0 8px 24px 4px #00000014;max-height:312px!important}::ng-deep .mat-mdc-form-field{min-width:330px!important}::ng-deep .mat-mdc-option:hover:not(.mdc-list-item--disabled),::ng-deep .mat-mdc-option:focus:not(.mdc-list-item--disabled),::ng-deep .mat-mdc-option.mat-mdc-option-active,::ng-deep .mat-mdc-option.mdc-list-item--selected:not(.mat-mdc-option-multiple):not(.mdc-list-item--disabled){background:#E9F0F1!important}::ng-deep .mat-mdc-text-field-wrapper.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{top:28px!important}::ng-deep .mat-mdc-text-field-wrapper .mat-mdc-form-field-flex .mat-mdc-floating-label{top:50%!important}::ng-deep .mat-mdc-text-field-wrapper{padding-left:12px!important}::ng-deep .mat-focused .mat-mdc-text-field-wrapper .mat-mdc-form-field-flex .mat-mdc-floating-label{color:#276678!important}::ng-deep .mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch{border-bottom:2px solid #276678!important}::ng-deep .mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading{border-left:2px solid #276678!important;border-bottom:2px solid #276678!important;border-top:2px solid #276678!important}::ng-deep .mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border:2px solid #276678!important;border-left:unset!important}::ng-deep .mat-mdc-form-field-infix,::ng-deep .mat-mdc-form-field-flex{min-height:48px!important;max-height:184px!important;overflow-y:auto;overflow-x:hidden;padding:0!important;display:flex;justify-content:center;text-align:center;align-items:center;flex-direction:row;width:100%}::ng-deep .mdc-line-ripple{display:none!important}::ng-deep .mat-mdc-option{padding-right:32px;padding-left:12px!important}::ng-deep .mat-mdc-option ui-icon{display:none;position:absolute;right:0;bottom:calc(50% - 8px);margin:0 10px}::ng-deep .mat-mdc-option.selected-option{background:#E9F0F1}::ng-deep .mat-mdc-option.selected-option ui-icon{display:block}::ng-deep .mat-mdc-option.mdc-list-item--selected .mdc-list-item__primary-text,::ng-deep .mat-mdc-option .mdc-list-item--activated .mdc-list-item__primary-text{color:#000!important}::ng-deep .mat-mdc-option .mat-pseudo-checkbox{display:none}::ng-deep .mat-mdc-optgroup .mdc-list-item__primary-text{color:#000;font-size:12px;line-height:16px;font-weight:700}\n"] }]
|
|
257
|
+
}], propDecorators: { itemsList: [{
|
|
258
|
+
type: Input
|
|
259
|
+
}], suggestionsList: [{
|
|
260
|
+
type: Input
|
|
261
|
+
}], disabled: [{
|
|
262
|
+
type: Input
|
|
263
|
+
}], allowAdd: [{
|
|
264
|
+
type: Input
|
|
265
|
+
}], textField: [{
|
|
266
|
+
type: Input
|
|
267
|
+
}], valueField: [{
|
|
268
|
+
type: Input
|
|
269
|
+
}], label: [{
|
|
270
|
+
type: Input
|
|
271
|
+
}], itemValue: [{
|
|
272
|
+
type: Input
|
|
273
|
+
}], type: [{
|
|
274
|
+
type: Input
|
|
275
|
+
}], minCharactersSearch: [{
|
|
276
|
+
type: Input
|
|
277
|
+
}], selectionChange: [{
|
|
278
|
+
type: Output
|
|
279
|
+
}], searchTextChange: [{
|
|
280
|
+
type: Output
|
|
281
|
+
}], matFormFieldElement: [{
|
|
282
|
+
type: ViewChild,
|
|
283
|
+
args: ['matFormField']
|
|
284
|
+
}], tagContainer: [{
|
|
285
|
+
type: ViewChild,
|
|
286
|
+
args: ['tagContainer']
|
|
287
|
+
}] } });
|
|
288
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete.component.js","sourceRoot":"","sources":["../../../../../src/components/autocomplete/autocomplete.component.ts","../../../../../src/components/autocomplete/autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  inject,\n  Input,\n  OnChanges,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { MatAutocompleteSelectedEvent } from '@angular/material/autocomplete';\nimport { MatFormField } from '@angular/material/form-field';\nimport { map, ReplaySubject, startWith } from 'rxjs';\nimport { AutocompleteUtils } from '../../utils/autocomplete-utils';\nimport { Autocomplete, AutocompleteType, Padding, Timeout } from './autocomplete.model';\n\n@Component({\n  selector: 'ui-autocomplete',\n  templateUrl: './autocomplete.component.html',\n  styleUrls: ['./autocomplete.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => AutocompleteComponent),\n      multi: true,\n    },\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class AutocompleteComponent implements ControlValueAccessor, OnChanges, AfterViewInit {\n  /**\n   * @property itemsList\n   * @description The list of items to display in the autocomplete.\n   * @memberof AutocompleteComponent\n   */\n  @Input() itemsList: any[] = [];\n\n  /**\n   * @property suggestionsList\n   * @description The list of suggestions to display in the autocomplete.\n   * @memberof AutocompleteComponent\n   */\n  @Input() suggestionsList: any[] = [];\n\n  /**\n   * @property disabled\n   * @description If `true`, the autocomplete is disabled.\n   * @memberof AutocompleteComponent\n   */\n  @Input() disabled = false;\n\n  /**\n   * @property allowAdd\n   * @description If `true`, allows adding new values to the autocomplete.\n   * @memberof AutocompleteComponent\n   */\n  @Input() allowAdd = true;\n\n  /**\n   * @property textField\n   * @description The field to be used as the display text in the autocomplete.\n   * @memberof AutocompleteComponent\n   */\n  @Input() textField = '';\n\n  /**\n   * @property valueField\n   * @description The field to be used as the value in the autocomplete.\n   * @memberof AutocompleteComponent\n   */\n  @Input() valueField = '';\n\n  /**\n   * @property label\n   * @description The label for the autocomplete input.\n   * @memberof AutocompleteComponent\n   */\n  @Input() label = '';\n\n  /**\n   * @property value\n   * @description The selected value(s) in the autocomplete.\n   * @memberof AutocompleteComponent\n   */\n  @Input() set itemValue(v: any) {\n    this.value = AutocompleteUtils.isArray(v) ? v : [v];\n  }\n\n  /**\n   * @property type\n   * @description The type of the autocomplete ('single', 'multi', etc.).\n   * @memberof AutocompleteComponent\n   */\n  @Input() type: AutocompleteType = 'multi';\n\n  /**\n   * @property minCharactersSearch\n   * @description The minimum characters to trigger search.\n   * @memberof AutocompleteComponent\n   */\n  @Input() minCharactersSearch = 3;\n\n  @Output() selectionChange = new EventEmitter<any>();\n  @Output() searchTextChange = new EventEmitter<string>();\n\n  @ViewChild('matFormField') matFormFieldElement: MatFormField;\n  @ViewChild('tagContainer') tagContainer: ElementRef<HTMLElement>;\n\n  private readonly cdr = inject(ChangeDetectorRef);\n  protected value: any;\n  protected inputValue: any = '';\n  protected isInputFocus = false;\n  protected autocompleteType = Autocomplete;\n  protected isOverLapping = false;\n  protected isOverlapChecking = true;\n  protected overlapIndex = 0;\n  protected inputValueSearch$ = new ReplaySubject<string>(1);\n  protected filteredItemsList$ = this.inputValueSearch$.pipe(\n    startWith(''),\n    map(inputValue =>\n      inputValue.length ? AutocompleteUtils.filterItems(this.itemsList, inputValue, this.textField) : null\n    )\n  );\n  protected filteredSuggestionList$ = this.inputValueSearch$.pipe(\n    map(inputValue =>\n      inputValue.length ? AutocompleteUtils.filterItems(this.suggestionsList, inputValue, this.textField) : null\n    )\n  );\n  protected searchResult$ = this.inputValueSearch$.pipe(map(inputValue => !!inputValue.trim()));\n\n  protected translationContext = 'AUTOCOMPLETE.';\n\n  ngOnChanges(): void {\n    this.itemsList = AutocompleteUtils.excludeSuggestions(this.itemsList, this.suggestionsList, this.valueField);\n    this.initValue();\n  }\n\n  ngAfterViewInit(): void {\n    if (this.type !== Autocomplete.LARGE) {\n      this.checkOverlap(Timeout.VIEW_INITIALIZED_COMPONENT_LOAD);\n    }\n  }\n\n  onChange = (_: any) => {};\n  onTouch = () => {};\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouch = fn;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  writeValue(value: any): void {\n    this.value = value;\n  }\n\n  protected onOptionSelected(event: MatAutocompleteSelectedEvent): void {\n    const inputValue = event.option.value;\n    const itemValue = AutocompleteUtils.createValueItem(inputValue, this.textField, this.valueField);\n    this.addNewValue(itemValue);\n\n    if (this.type === this.autocompleteType.MULTI) {\n      this.checkOverlap();\n    }\n\n    queueMicrotask(() => {\n      if (this.type !== this.autocompleteType.SINGLE) {\n        this.inputValue = '';\n      } else {\n        this.inputValue = AutocompleteUtils.transformDisplayValue(this.value, this.textField);\n      }\n      this.cdr.markForCheck();\n    });\n\n    this.onUpdate();\n  }\n\n  protected onOptionRemoved(input: any, option: any): void {\n    setTimeout(() => (input.showPanel = false), 0);\n\n    const foundValue = this.value.find((v: any) => v === option);\n\n    if (foundValue) {\n      this.value = this.value.filter((v: any) => v !== option);\n    }\n\n    if (this.type === this.autocompleteType.MULTI) {\n      this.checkOverlap();\n    }\n\n    if (!this.value.length) {\n      this.inputValue = '';\n    }\n\n    this.onUpdate();\n  }\n\n  protected onInputChange(event: Event): void {\n    const inputValue = (event.target as HTMLInputElement).value;\n    if (inputValue.length >= this.minCharactersSearch) {\n      this.inputValueSearch$.next(inputValue);\n      this.searchTextChange.emit(inputValue);\n    } else {\n      this.inputValueSearch$.next('');\n    }\n  }\n\n  protected onClear(): void {\n    this.inputValue = '';\n    this.value = AutocompleteUtils.isArray(this.value) ? [] : '';\n    this.isOverLapping = false;\n    this.overlapIndex = 0;\n    this.onUpdate();\n  }\n\n  protected refillInput(): void {\n    if (this.value && this.type === this.autocompleteType.SINGLE && !this.inputValue) {\n      this.inputValue = AutocompleteUtils.transformDisplayValue(this.value, this.textField);\n      this.inputValueSearch$.next('');\n    }\n\n    if (!this.inputValue && this.value?.length) {\n      this.inputValue = ' ';\n    }\n\n    this.isInputFocus = false;\n  }\n\n  protected onFocus(): void {\n    this.isInputFocus = true;\n\n    const inputValue = AutocompleteUtils.isArray(this.inputValue) ? this.inputValue[0] : this.inputValue;\n\n    if (AutocompleteUtils.isPrimitive(inputValue)) {\n      if (inputValue?.match(/^\\s+/gm)) {\n        this.inputValue = inputValue.trimStart();\n      }\n    } else {\n      if (inputValue[this.textField]?.match(/^\\s+/gm)) {\n        this.inputValue = inputValue[this.textField].trimStart();\n      }\n    }\n  }\n\n  private checkOverlap(delay = Timeout.VIEW_INITIALIZED_AFTER_ACTION): void {\n    this.isOverLapping = false;\n    this.isOverlapChecking = true;\n\n    setTimeout(() => {\n      const inputWidth =\n        this.matFormFieldElement?._textField.nativeElement.clientWidth - Padding.AUTOCOMPLETE_CONTAINER;\n      const tags = this.tagContainer?.nativeElement.querySelectorAll('.value-tag');\n\n      if (tags) {\n        let width = 0;\n\n        tags.forEach((tag: any, index: number) => {\n          width += tag.getBoundingClientRect().width;\n\n          if (!this.isOverLapping && width > inputWidth) {\n            this.isOverLapping = true;\n            this.overlapIndex = index;\n          }\n        });\n        this.isOverlapChecking = false;\n        this.cdr.markForCheck();\n      }\n    }, delay);\n  }\n\n  private onUpdate(): void {\n    this.inputValueSearch$.next('');\n    this.selectionChange.emit(this.value);\n    this.onChange(this.value);\n  }\n\n  private initValue(): void {\n    if (this.type === this.autocompleteType.SINGLE) {\n      this.inputValue = AutocompleteUtils.transformDisplayValue(this.value[0], this.textField);\n    }\n    if (this.type === this.autocompleteType.MULTI) {\n      this.checkOverlap();\n    }\n  }\n\n  private addNewValue(newValue: any): void {\n    if (this.type !== this.autocompleteType.SINGLE && !this.value) {\n      this.value = [];\n    }\n\n    const foundDuplicate = AutocompleteUtils.findDuplicate(this.value, newValue, this.textField);\n\n    if (this.type !== this.autocompleteType.SINGLE) {\n      this.value = foundDuplicate ? this.value.filter((v: any) => v !== foundDuplicate) : [...this.value, newValue];\n    } else {\n      this.value = newValue;\n    }\n  }\n}\n","<ng-container *transloco=\"let t\">\n  <div class=\"autocomplete-wrapper\">\n    <mat-form-field\n      [appearance]=\"'outline'\"\n      #matFormField\n      [ngClass]=\"{ 'large-size': type === autocompleteType.LARGE, disabled: disabled }\"\n    >\n      <mat-label *ngIf=\"label\">{{ label }}</mat-label>\n      <div class=\"input-container\">\n        <div\n          class=\"selected-items\"\n          #tagContainer\n          *ngIf=\"\n            (type === autocompleteType.LARGE && this.value) ||\n            (!(inputValue && this.isInputFocus) && this.value && type !== autocompleteType.SINGLE)\n          \"\n        >\n          <ui-tag\n            class=\"value-tag\"\n            [id]=\"'ui-tag-' + i\"\n            *ngFor=\"let valueItem of value; let i = index\"\n            [label]=\"valueItem | transformItem: textField\"\n            [allowClose]=\"true\"\n            [style.display]=\"isOverLapping && i >= overlapIndex ? 'none' : 'block'\"\n            [ngClass]=\"{\n              'overlap-tag': isOverlapChecking && overlapIndex && i >= overlapIndex && type === autocompleteType.MULTI\n            }\"\n            (close)=\"onOptionRemoved(autocomplete, valueItem)\"\n          ></ui-tag>\n          <ng-container *ngIf=\"isOverLapping\">\n            <div *ngIf=\"!overlapIndex; else overlapTag\" class=\"overlap-count\">\n              {{ t(translationContext + 'SELECTED_AMOUNT', { numberSelected: (value | slice: overlapIndex).length }) }}\n            </div>\n            <ng-template #overlapTag>\n              <ui-tag [readOnly]=\"true\" [label]=\"'+' + (value | slice: overlapIndex).length\"></ui-tag>\n            </ng-template>\n          </ng-container>\n        </div>\n        <input\n          [disabled]=\"disabled\"\n          [ngClass]=\"{ 'unset-margin': this.value && type === autocompleteType.LARGE }\"\n          (blur)=\"refillInput()\"\n          (focusin)=\"onFocus()\"\n          matInput\n          [type]=\"'text'\"\n          [matAutocomplete]=\"autocomplete\"\n          (input)=\"onInputChange($event)\"\n          [(ngModel)]=\"inputValue\"\n        />\n      </div>\n\n      <ui-icon\n        class=\"remove-selected\"\n        [ngClass]=\"{ 'large-input-icon': type === autocompleteType.LARGE }\"\n        (click)=\"onClear()\"\n        *ngIf=\"inputValue || value?.length\"\n        [size]=\"'24'\"\n        [name]=\"'Close'\"\n      ></ui-icon>\n\n      <mat-autocomplete\n        #autocomplete\n        (optionSelected)=\"onOptionSelected($event)\"\n        [hideSingleSelectionIndicator]=\"false\"\n      >\n        <ng-container *ngIf=\"!(searchResult$ | async)\">\n          <mat-optgroup [label]=\"t(translationContext + 'SUGGESTED')\" *ngIf=\"suggestionsList.length\">\n            <mat-option\n              [ngClass]=\"{ 'selected-option': suggested | includes: value }\"\n              *ngFor=\"let suggested of suggestionsList\"\n              [value]=\"suggested\"\n            >\n              <span [selectText]=\"inputValue\">{{ suggested | transformItem: textField }}</span>\n              <ui-icon [name]=\"'Check'\"></ui-icon>\n            </mat-option>\n          </mat-optgroup>\n\n          <mat-optgroup [label]=\"t(translationContext + 'SELECTED') | uppercase\" *ngIf=\"value?.length && type !== autocompleteType.SINGLE\">\n            <mat-option\n              [ngClass]=\"{ 'selected-option': item | includes: value }\"\n              *ngFor=\"let item of value\"\n              [value]=\"item\"\n            >\n              <span>{{ item | transformItem: textField }}</span>\n              <ui-icon [name]=\"'Check'\"></ui-icon>\n            </mat-option>\n          </mat-optgroup>\n        </ng-container>\n\n        <ng-container *ngIf=\"filteredSuggestionList$ | async as filteredSuggestionList\">\n          <ng-container *ngIf=\"filteredItemsList$ | async as filteredItemsList\">\n            <mat-optgroup [label]=\"t(translationContext + 'SUGGESTED') | uppercase\" *ngIf=\"filteredSuggestionList.length\">\n              <mat-option\n                [ngClass]=\"{ 'selected-option': item | includes: value }\"\n                *ngFor=\"let item of filteredSuggestionList\"\n                [value]=\"item\"\n              >\n                <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n                <ui-icon [name]=\"'Check'\"></ui-icon>\n              </mat-option>\n            </mat-optgroup>\n\n            <ng-container *ngIf=\"!filteredSuggestionList.length; else showItemListWithGroup\">\n              <mat-option\n                [ngClass]=\"{ 'selected-option': item | includes: value }\"\n                *ngFor=\"let item of filteredItemsList\"\n                [value]=\"item\"\n              >\n                <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n                <ui-icon [name]=\"'Check'\"></ui-icon>\n              </mat-option>\n            </ng-container>\n\n            <ng-template #showItemListWithGroup>\n              <mat-optgroup [label]=\"t(translationContext + 'ALL_ITEMS') | uppercase\" *ngIf=\"filteredItemsList.length\">\n                <mat-option\n                  [ngClass]=\"{ 'selected-option': item | includes: value }\"\n                  *ngFor=\"let item of filteredItemsList\"\n                  [value]=\"item\"\n                >\n                  <span [selectText]=\"inputValue\">{{ item | transformItem: textField }}</span>\n                  <ui-icon [name]=\"'Check'\"></ui-icon>\n                </mat-option>\n              </mat-optgroup>\n            </ng-template>\n\n            <ng-container *ngIf=\"!filteredItemsList.length && !filteredSuggestionList.length && inputValue\">\n              <ng-container *ngIf=\"allowAdd; else notFound\">\n                <mat-option [value]=\"inputValue\">\n                  <span>{{ t('COMMON.ADD') }}</span>\n                  <span class=\"add-suggestion\">\"{{ inputValue }}\"</span>\n                </mat-option>\n              </ng-container>\n\n              <ng-template #notFound>\n                <mat-option [style.pointer-events]=\"'none'\">\n                  <span>{{ t(translationContext + 'NO_RESULTS_FOUND') }}</span>\n                </mat-option>\n              </ng-template>\n            </ng-container>\n          </ng-container>\n        </ng-container>\n      </mat-autocomplete>\n    </mat-form-field>\n  </div>\n</ng-container>\n"]}
|