@flywheel-io/vision 2.8.0 → 2.9.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.
- package/components/button-toggle/button-toggle-item/button-toggle-item.component.d.ts +1 -1
- package/components/chip-list/chip-list.component.d.ts +2 -1
- package/components/radio/radio-group.component.d.ts +9 -11
- package/components/radio/radio.component.d.ts +3 -3
- package/components/typeahead/typeahead.component.d.ts +3 -1
- package/esm2022/components/button-toggle/button-toggle-item/button-toggle-item.component.mjs +6 -4
- package/esm2022/components/chip/chip.component.mjs +2 -2
- package/esm2022/components/chip-list/chip-list.component.mjs +14 -6
- package/esm2022/components/icon/icon.component.mjs +2 -2
- package/esm2022/components/menu/menu-item/menu-item.component.mjs +1 -1
- package/esm2022/components/radio/radio-group.component.mjs +50 -50
- package/esm2022/components/radio/radio.component.mjs +7 -10
- package/esm2022/components/typeahead/typeahead.component.mjs +6 -4
- package/fesm2022/flywheel-io-vision.mjs +81 -72
- package/fesm2022/flywheel-io-vision.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -1,67 +1,69 @@
|
|
|
1
|
-
import { Component,
|
|
1
|
+
import { Component, contentChildren, DestroyRef, effect, EventEmitter, HostBinding, inject, Input, Output, signal, } from '@angular/core';
|
|
2
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
2
3
|
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
4
|
+
import { DomSanitizer } from '@angular/platform-browser';
|
|
3
5
|
import { FwRadioComponent } from './radio.component';
|
|
4
6
|
import * as i0 from "@angular/core";
|
|
5
|
-
import * as i1 from "@angular/platform-browser";
|
|
6
7
|
export class FwRadioGroupComponent {
|
|
7
8
|
get style() {
|
|
8
|
-
|
|
9
|
-
if (this.direction === 'vertical') {
|
|
10
|
-
direction = 'column';
|
|
11
|
-
}
|
|
9
|
+
const dir = this.direction === 'vertical' ? 'column' : 'row';
|
|
12
10
|
return this.sanitizer.bypassSecurityTrustStyle(`
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
11
|
+
display: flex;
|
|
12
|
+
flex-direction: ${dir};
|
|
13
|
+
`);
|
|
16
14
|
}
|
|
17
|
-
constructor(
|
|
18
|
-
this.sanitizer = sanitizer;
|
|
15
|
+
constructor() {
|
|
19
16
|
this.direction = 'horizontal';
|
|
20
17
|
// eslint-disable-next-line @angular-eslint/no-output-native
|
|
21
18
|
this.change = new EventEmitter();
|
|
22
|
-
this.
|
|
19
|
+
this.sanitizer = inject(DomSanitizer);
|
|
20
|
+
this.dr = inject(DestroyRef);
|
|
21
|
+
this.radioButtons = contentChildren(FwRadioComponent);
|
|
22
|
+
this._value = signal(null);
|
|
23
|
+
this.radioSubs = new Map();
|
|
23
24
|
this.onTouch = () => {
|
|
24
25
|
};
|
|
25
26
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
26
27
|
this.onChange = (value) => {
|
|
27
28
|
};
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
this.updateLayout();
|
|
45
|
-
}
|
|
46
|
-
updateLayout() {
|
|
47
|
-
if (this.radioButtons) {
|
|
48
|
-
this.radioButtons.forEach(rb => {
|
|
49
|
-
rb.group = this.group;
|
|
50
|
-
rb.checked = rb.value === this.value;
|
|
51
|
-
if (this.color) {
|
|
52
|
-
rb.color = this.color;
|
|
29
|
+
effect(() => {
|
|
30
|
+
const radios = this.radioButtons();
|
|
31
|
+
const value = this._value();
|
|
32
|
+
const disabled = this.disabled;
|
|
33
|
+
const group = this.group;
|
|
34
|
+
const color = this.color;
|
|
35
|
+
const size = this.size;
|
|
36
|
+
radios.forEach(rb => {
|
|
37
|
+
rb.group = group;
|
|
38
|
+
rb.checked.set(rb.value() === value);
|
|
39
|
+
rb.disabled = Boolean(disabled);
|
|
40
|
+
if (color) {
|
|
41
|
+
rb.color = color;
|
|
42
|
+
}
|
|
43
|
+
if (size) {
|
|
44
|
+
rb.size = size;
|
|
53
45
|
}
|
|
54
|
-
|
|
55
|
-
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
effect(() => {
|
|
49
|
+
const radios = this.radioButtons();
|
|
50
|
+
this.radioSubs.forEach((sub, rb) => {
|
|
51
|
+
if (!radios.includes(rb)) {
|
|
52
|
+
sub.unsubscribe();
|
|
53
|
+
this.radioSubs.delete(rb);
|
|
56
54
|
}
|
|
57
|
-
|
|
58
|
-
|
|
55
|
+
});
|
|
56
|
+
radios.forEach(rb => {
|
|
57
|
+
if (!this.radioSubs.has(rb)) {
|
|
58
|
+
const sub = rb.change.pipe(takeUntilDestroyed(this.dr)).subscribe(value => this.handleChange(value));
|
|
59
|
+
this.radioSubs.set(rb, sub);
|
|
59
60
|
}
|
|
60
61
|
});
|
|
61
|
-
}
|
|
62
|
+
});
|
|
62
63
|
}
|
|
63
64
|
writeValue(value) {
|
|
64
65
|
this.value = value;
|
|
66
|
+
this._value.set(value);
|
|
65
67
|
}
|
|
66
68
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
67
69
|
registerOnChange(fn) {
|
|
@@ -76,15 +78,16 @@ export class FwRadioGroupComponent {
|
|
|
76
78
|
}
|
|
77
79
|
handleChange(value) {
|
|
78
80
|
this.value = value;
|
|
81
|
+
this._value.set(value);
|
|
79
82
|
this.onChange(value);
|
|
80
83
|
this.change.emit(this.value);
|
|
81
84
|
}
|
|
82
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FwRadioGroupComponent, deps: [
|
|
83
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
85
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FwRadioGroupComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
86
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "17.3.12", type: FwRadioGroupComponent, selector: "fw-radio-group", inputs: { value: "value", group: "group", color: "color", size: "size", disabled: "disabled", direction: "direction" }, outputs: { change: "change" }, host: { properties: { "style": "this.style" } }, providers: [{
|
|
84
87
|
provide: NG_VALUE_ACCESSOR,
|
|
85
88
|
useExisting: FwRadioGroupComponent,
|
|
86
89
|
multi: true,
|
|
87
|
-
}], queries: [{ propertyName: "radioButtons", predicate: FwRadioComponent
|
|
90
|
+
}], queries: [{ propertyName: "radioButtons", predicate: FwRadioComponent, isSignal: true }], ngImport: i0, template: '<ng-content select="fw-radio-button"></ng-content>', isInline: true }); }
|
|
88
91
|
}
|
|
89
92
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FwRadioGroupComponent, decorators: [{
|
|
90
93
|
type: Component,
|
|
@@ -93,7 +96,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
93
96
|
useExisting: FwRadioGroupComponent,
|
|
94
97
|
multi: true,
|
|
95
98
|
}] }]
|
|
96
|
-
}], ctorParameters: () => [
|
|
99
|
+
}], ctorParameters: () => [], propDecorators: { value: [{
|
|
97
100
|
type: Input
|
|
98
101
|
}], group: [{
|
|
99
102
|
type: Input
|
|
@@ -107,11 +110,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
107
110
|
type: Input
|
|
108
111
|
}], change: [{
|
|
109
112
|
type: Output
|
|
110
|
-
}], radioButtons: [{
|
|
111
|
-
type: ContentChildren,
|
|
112
|
-
args: [FwRadioComponent]
|
|
113
113
|
}], style: [{
|
|
114
114
|
type: HostBinding,
|
|
115
115
|
args: ['style']
|
|
116
116
|
}] } });
|
|
117
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
117
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radio-group.component.js","sourceRoot":"","sources":["../../../../../src/components/radio/radio-group.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  contentChildren,\n  DestroyRef,\n  effect,\n  EventEmitter,\n  HostBinding,\n  inject,\n  Input,\n  Output,\n  signal,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { DomSanitizer, SafeStyle } from '@angular/platform-browser';\nimport { Subscription } from 'rxjs';\n\nimport { FwRadioComponent } from './radio.component';\n\n@Component({\n    selector: 'fw-radio-group',\n    styleUrls: [],\n    template: '<ng-content select=\"fw-radio-button\"></ng-content>',\n    providers: [{\n        provide: NG_VALUE_ACCESSOR,\n        useExisting: FwRadioGroupComponent,\n        multi: true,\n    }],\n})\nexport class FwRadioGroupComponent implements ControlValueAccessor {\n    @Input() value: string;\n    @Input() group: string;\n    @Input() color?: 'primary' | 'secondary' | 'default';\n    @Input() size?: 'medium' | 'compact';\n    @Input() disabled?: boolean;\n    @Input() direction?: 'horizontal' | 'vertical' = 'horizontal';\n    // eslint-disable-next-line @angular-eslint/no-output-native\n    @Output() change = new EventEmitter<string>();\n\n    private sanitizer = inject(DomSanitizer);\n    private dr = inject(DestroyRef);\n\n    protected radioButtons = contentChildren(FwRadioComponent);\n    private _value = signal<string | null>(null);\n    private radioSubs = new Map<FwRadioComponent, Subscription>();\n\n    @HostBinding('style') get style(): SafeStyle {\n      const dir = this.direction === 'vertical' ? 'column' : 'row';\n      return this.sanitizer.bypassSecurityTrustStyle(`\n        display: flex;\n        flex-direction: ${dir};\n      `);\n    }\n\n    constructor() {\n      effect(() => {\n        const radios = this.radioButtons();\n        const value = this._value();\n        const disabled = this.disabled;\n        const group = this.group;\n        const color = this.color;\n        const size = this.size;\n\n        radios.forEach(rb => {\n          rb.group = group;\n          rb.checked.set(rb.value() === value);\n          rb.disabled = Boolean(disabled);\n\n          if (color) {\n            rb.color = color;\n          }\n\n          if (size) {\n            rb.size = size;\n          }\n        });\n      });\n\n    effect(() => {\n      const radios = this.radioButtons();\n\n      this.radioSubs.forEach((sub, rb) => {\n        if (!radios.includes(rb)) {\n          sub.unsubscribe();\n          this.radioSubs.delete(rb);\n        }\n      });\n\n      radios.forEach(rb => {\n        if (!this.radioSubs.has(rb)) {\n          const sub = rb.change.pipe(takeUntilDestroyed(this.dr)).subscribe(value => this.handleChange(value));\n          this.radioSubs.set(rb, sub);\n        }\n      });\n    });\n  }\n\n    public onTouch = (): void => {\n    };\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    public onChange = (value: string): void => {\n    };\n\n    writeValue(value: string): void {\n      this.value = value;\n      this._value.set(value);\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    registerOnChange(fn: any): void {\n      this.onChange = fn;\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    registerOnTouched(fn: any): void {\n      this.onTouch = fn;\n    }\n\n    setDisabledState(isDisabled: boolean): void {\n      this.disabled = isDisabled;\n    }\n\n    handleChange(value: string): void {\n      this.value = value;\n      this._value.set(value);\n      this.onChange(value);\n      this.change.emit(this.value);\n    }\n}\n"]}
|
|
@@ -1,9 +1,10 @@
|
|
|
1
|
-
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
1
|
+
import { Component, EventEmitter, Input, input, model, Output } from '@angular/core';
|
|
2
2
|
import * as i0 from "@angular/core";
|
|
3
3
|
import * as i1 from "@angular/common";
|
|
4
4
|
export class FwRadioComponent {
|
|
5
5
|
constructor() {
|
|
6
|
-
this.
|
|
6
|
+
this.value = input();
|
|
7
|
+
this.checked = model(false);
|
|
7
8
|
this.disabled = false;
|
|
8
9
|
this.size = 'medium';
|
|
9
10
|
this.color = 'default';
|
|
@@ -24,16 +25,12 @@ export class FwRadioComponent {
|
|
|
24
25
|
this.change.emit(event.target.value);
|
|
25
26
|
}
|
|
26
27
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FwRadioComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
27
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
28
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: FwRadioComponent, selector: "fw-radio-button", inputs: { value: { classPropertyName: "value", publicName: "value", isSignal: true, isRequired: false, transformFunction: null }, checked: { classPropertyName: "checked", publicName: "checked", isSignal: true, isRequired: false, transformFunction: null }, group: { classPropertyName: "group", publicName: "group", isSignal: false, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: false, isRequired: false, transformFunction: null }, size: { classPropertyName: "size", publicName: "size", isSignal: false, isRequired: false, transformFunction: null }, color: { classPropertyName: "color", publicName: "color", isSignal: false, isRequired: false, transformFunction: null }, title: { classPropertyName: "title", publicName: "title", isSignal: false, isRequired: false, transformFunction: null }, focused: { classPropertyName: "focused", publicName: "focused", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { checked: "checkedChange", change: "change" }, ngImport: i0, template: "<div [ngClass]=\"['fw-radio-wrapper', disabled?'disabled':'']\">\n <div [ngClass]=\"radioStyles\">\n <input\n type=\"radio\"\n class=\"fw-radio\"\n [id]=\"group+value()\"\n [value]=\"value()\"\n [checked]=\"checked()\"\n [name]=\"group\"\n [disabled]=\"disabled\"\n (change)=\"handleChange($event)\"\n (focusout)=\"focused=false\"\n (focusin)=\"focused=true\"\n />\n </div>\n\n @if (title) {\n <label [htmlFor]=\"group+value()\" class=\"vision-h4\">{{ title }}\n <ng-content></ng-content>\n </label>\n }\n</div>\n", styles: [":host .fw-radio-wrapper{display:flex;align-items:center;width:fit-content;cursor:pointer}:host .fw-radio-wrapper label{margin:0 6px 0 0;color:var(--typography-base)}:host .fw-radio-wrapper label:empty{display:none}:host .fw-radio-wrapper .fw-radio{height:36px;width:36px;display:flex;align-items:center;justify-content:center;border-radius:50%}:host .fw-radio-wrapper .fw-radio:hover{background-color:var(--slate-hover)}:host .fw-radio-wrapper .fw-radio.focused{background-color:var(--slate-focus)}:host .fw-radio-wrapper .fw-radio input[type=radio]{appearance:none;display:inline-block;width:20px;height:20px;margin:6px;background-clip:content-box;border:2px solid var(--typography-light);background-color:transparent;border-radius:50%;cursor:pointer}:host .fw-radio-wrapper .fw-radio input[type=radio]:checked{width:16px;height:16px;background-color:var(--typography-light);outline:2px solid var(--typography-light);border:2px solid white;margin:8px}:host .fw-radio-wrapper .fw-radio.primary.focused{background-color:var(--primary-focus)}:host .fw-radio-wrapper .fw-radio.primary.focused input[type=radio]{border:2px solid var(--primary-base)}:host .fw-radio-wrapper .fw-radio.primary input[type=radio]:checked{border:2px solid white;background-color:var(--primary-base);outline:2px solid var(--primary-base)}:host .fw-radio-wrapper .fw-radio.secondary.focused{background-color:var(--secondary-focus)}:host .fw-radio-wrapper .fw-radio.secondary.focused input[type=radio]{border:2px solid var(--secondary-base)}:host .fw-radio-wrapper .fw-radio.secondary input[type=radio]:checked{border:2px solid white;background-color:var(--secondary-base);outline:2px solid var(--secondary-base)}:host .fw-radio-wrapper .fw-radio.disabled{cursor:not-allowed}:host .fw-radio-wrapper .fw-radio.disabled input[type=radio]:checked{border:2px solid white;background-color:var(--typography-light);outline:2px solid var(--typography-light)}:host .fw-radio-wrapper .fw-radio.compact{width:30px;height:30px}:host .fw-radio-wrapper .fw-radio.compact input[type=radio]{width:18px;height:18px;margin:5px}:host .fw-radio-wrapper .fw-radio.compact input[type=radio]:checked{width:14px;height:14px;margin:7px}:host .fw-radio-wrapper.disabled{cursor:not-allowed;opacity:.4}:host .fw-radio-wrapper.disabled label{color:var(--typography-light)}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] }); }
|
|
28
29
|
}
|
|
29
30
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FwRadioComponent, decorators: [{
|
|
30
31
|
type: Component,
|
|
31
|
-
args: [{ selector: 'fw-radio-button', template: "<div [ngClass]=\"['fw-radio-wrapper', disabled?'disabled':'']\">\n <div [ngClass]=\"radioStyles\">\n <input\n type=\"radio\"\n class=\"fw-radio\"\n [id]=\"group+value\"\n [value]=\"value\"\n [checked]=\"checked\"\n [name]=\"group\"\n [disabled]=\"disabled\"\n (change)=\"handleChange($event)\"\n (focusout)=\"focused=false\"\n (focusin)=\"focused=true\"\n />\n </div>\n
|
|
32
|
-
}], propDecorators: {
|
|
33
|
-
type: Input
|
|
34
|
-
}], value: [{
|
|
35
|
-
type: Input
|
|
36
|
-
}], group: [{
|
|
32
|
+
args: [{ selector: 'fw-radio-button', template: "<div [ngClass]=\"['fw-radio-wrapper', disabled?'disabled':'']\">\n <div [ngClass]=\"radioStyles\">\n <input\n type=\"radio\"\n class=\"fw-radio\"\n [id]=\"group+value()\"\n [value]=\"value()\"\n [checked]=\"checked()\"\n [name]=\"group\"\n [disabled]=\"disabled\"\n (change)=\"handleChange($event)\"\n (focusout)=\"focused=false\"\n (focusin)=\"focused=true\"\n />\n </div>\n\n @if (title) {\n <label [htmlFor]=\"group+value()\" class=\"vision-h4\">{{ title }}\n <ng-content></ng-content>\n </label>\n }\n</div>\n", styles: [":host .fw-radio-wrapper{display:flex;align-items:center;width:fit-content;cursor:pointer}:host .fw-radio-wrapper label{margin:0 6px 0 0;color:var(--typography-base)}:host .fw-radio-wrapper label:empty{display:none}:host .fw-radio-wrapper .fw-radio{height:36px;width:36px;display:flex;align-items:center;justify-content:center;border-radius:50%}:host .fw-radio-wrapper .fw-radio:hover{background-color:var(--slate-hover)}:host .fw-radio-wrapper .fw-radio.focused{background-color:var(--slate-focus)}:host .fw-radio-wrapper .fw-radio input[type=radio]{appearance:none;display:inline-block;width:20px;height:20px;margin:6px;background-clip:content-box;border:2px solid var(--typography-light);background-color:transparent;border-radius:50%;cursor:pointer}:host .fw-radio-wrapper .fw-radio input[type=radio]:checked{width:16px;height:16px;background-color:var(--typography-light);outline:2px solid var(--typography-light);border:2px solid white;margin:8px}:host .fw-radio-wrapper .fw-radio.primary.focused{background-color:var(--primary-focus)}:host .fw-radio-wrapper .fw-radio.primary.focused input[type=radio]{border:2px solid var(--primary-base)}:host .fw-radio-wrapper .fw-radio.primary input[type=radio]:checked{border:2px solid white;background-color:var(--primary-base);outline:2px solid var(--primary-base)}:host .fw-radio-wrapper .fw-radio.secondary.focused{background-color:var(--secondary-focus)}:host .fw-radio-wrapper .fw-radio.secondary.focused input[type=radio]{border:2px solid var(--secondary-base)}:host .fw-radio-wrapper .fw-radio.secondary input[type=radio]:checked{border:2px solid white;background-color:var(--secondary-base);outline:2px solid var(--secondary-base)}:host .fw-radio-wrapper .fw-radio.disabled{cursor:not-allowed}:host .fw-radio-wrapper .fw-radio.disabled input[type=radio]:checked{border:2px solid white;background-color:var(--typography-light);outline:2px solid var(--typography-light)}:host .fw-radio-wrapper .fw-radio.compact{width:30px;height:30px}:host .fw-radio-wrapper .fw-radio.compact input[type=radio]{width:18px;height:18px;margin:5px}:host .fw-radio-wrapper .fw-radio.compact input[type=radio]:checked{width:14px;height:14px;margin:7px}:host .fw-radio-wrapper.disabled{cursor:not-allowed;opacity:.4}:host .fw-radio-wrapper.disabled label{color:var(--typography-light)}\n"] }]
|
|
33
|
+
}], propDecorators: { group: [{
|
|
37
34
|
type: Input
|
|
38
35
|
}], disabled: [{
|
|
39
36
|
type: Input
|
|
@@ -48,4 +45,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
48
45
|
}], change: [{
|
|
49
46
|
type: Output
|
|
50
47
|
}] } });
|
|
51
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
48
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFkaW8uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvcmFkaW8vcmFkaW8uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvcmFkaW8vcmFkaW8uY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBQyxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFPcEYsTUFBTSxPQUFPLGdCQUFnQjtJQUw3QjtRQU1XLFVBQUssR0FBRyxLQUFLLEVBQVUsQ0FBQztRQUN4QixZQUFPLEdBQUcsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBR3JCLGFBQVEsR0FBWSxLQUFLLENBQUM7UUFDMUIsU0FBSSxHQUF5QixRQUFRLENBQUM7UUFDdEMsVUFBSyxHQUF3QyxTQUFTLENBQUM7UUFFdkQsWUFBTyxHQUFZLEtBQUssQ0FBQztRQUNsQyw0REFBNEQ7UUFDbEQsV0FBTSxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7S0FlakQ7SUFiRyxJQUFJLFdBQVc7UUFDWCxPQUFPO1lBQ0gsVUFBVTtZQUNWLElBQUksQ0FBQyxLQUFLO1lBQ1YsSUFBSSxDQUFDLElBQUk7WUFDVCxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLEVBQUU7WUFDN0IsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxFQUFFO1NBQ2xDLENBQUM7SUFDTixDQUFDO0lBRUQsWUFBWSxDQUFDLEtBQUs7UUFDZCxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3pDLENBQUM7K0dBekJRLGdCQUFnQjttR0FBaEIsZ0JBQWdCLG9sQ0NQN0IsMGtCQXNCQTs7NEZEZmEsZ0JBQWdCO2tCQUw1QixTQUFTOytCQUNJLGlCQUFpQjs4QkFRbEIsS0FBSztzQkFBYixLQUFLO2dCQUNHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBQ0csSUFBSTtzQkFBWixLQUFLO2dCQUNHLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxLQUFLO3NCQUFiLEtBQUs7Z0JBQ0csT0FBTztzQkFBZixLQUFLO2dCQUVJLE1BQU07c0JBQWYsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBJbnB1dCwgaW5wdXQsbW9kZWwsIE91dHB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ2Z3LXJhZGlvLWJ1dHRvbicsXG4gICAgdGVtcGxhdGVVcmw6ICcuL3JhZGlvLmNvbXBvbmVudC5odG1sJyxcbiAgICBzdHlsZVVybHM6IFsnLi9yYWRpby5jb21wb25lbnQuc2NzcyddLFxufSlcbmV4cG9ydCBjbGFzcyBGd1JhZGlvQ29tcG9uZW50IHtcbiAgICBwdWJsaWMgdmFsdWUgPSBpbnB1dDxzdHJpbmc+KCk7XG4gICAgcHVibGljIGNoZWNrZWQgPSBtb2RlbChmYWxzZSk7XG5cbiAgICBASW5wdXQoKSBncm91cDogc3RyaW5nO1xuICAgIEBJbnB1dCgpIGRpc2FibGVkOiBib29sZWFuID0gZmFsc2U7XG4gICAgQElucHV0KCkgc2l6ZTogJ21lZGl1bScgfCAnY29tcGFjdCcgPSAnbWVkaXVtJztcbiAgICBASW5wdXQoKSBjb2xvcjogJ2RlZmF1bHQnIHwgJ3ByaW1hcnknIHwgJ3NlY29uZGFyeScgPSAnZGVmYXVsdCc7XG4gICAgQElucHV0KCkgdGl0bGU6IHN0cmluZztcbiAgICBASW5wdXQoKSBmb2N1c2VkOiBib29sZWFuID0gZmFsc2U7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEBhbmd1bGFyLWVzbGludC9uby1vdXRwdXQtbmF0aXZlXG4gICAgQE91dHB1dCgpIGNoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8c3RyaW5nPigpO1xuXG4gICAgZ2V0IHJhZGlvU3R5bGVzKCk6IHN0cmluZ1tdIHtcbiAgICAgICAgcmV0dXJuIFtcbiAgICAgICAgICAgICdmdy1yYWRpbycsXG4gICAgICAgICAgICB0aGlzLmNvbG9yLFxuICAgICAgICAgICAgdGhpcy5zaXplLFxuICAgICAgICAgICAgdGhpcy5mb2N1c2VkID8gJ2ZvY3VzZWQnIDogJycsXG4gICAgICAgICAgICB0aGlzLmRpc2FibGVkID8gJ2Rpc2FibGVkJyA6ICcnLFxuICAgICAgICBdO1xuICAgIH1cblxuICAgIGhhbmRsZUNoYW5nZShldmVudCk6IHZvaWQge1xuICAgICAgICB0aGlzLmNoYW5nZS5lbWl0KGV2ZW50LnRhcmdldC52YWx1ZSk7XG4gICAgfVxufVxuIiwiPGRpdiBbbmdDbGFzc109XCJbJ2Z3LXJhZGlvLXdyYXBwZXInLCBkaXNhYmxlZD8nZGlzYWJsZWQnOicnXVwiPlxuICA8ZGl2IFtuZ0NsYXNzXT1cInJhZGlvU3R5bGVzXCI+XG4gICAgPGlucHV0XG4gICAgICB0eXBlPVwicmFkaW9cIlxuICAgICAgY2xhc3M9XCJmdy1yYWRpb1wiXG4gICAgICBbaWRdPVwiZ3JvdXArdmFsdWUoKVwiXG4gICAgICBbdmFsdWVdPVwidmFsdWUoKVwiXG4gICAgICBbY2hlY2tlZF09XCJjaGVja2VkKClcIlxuICAgICAgW25hbWVdPVwiZ3JvdXBcIlxuICAgICAgW2Rpc2FibGVkXT1cImRpc2FibGVkXCJcbiAgICAgIChjaGFuZ2UpPVwiaGFuZGxlQ2hhbmdlKCRldmVudClcIlxuICAgICAgKGZvY3Vzb3V0KT1cImZvY3VzZWQ9ZmFsc2VcIlxuICAgICAgKGZvY3VzaW4pPVwiZm9jdXNlZD10cnVlXCJcbiAgICAvPlxuICA8L2Rpdj5cblxuICBAaWYgKHRpdGxlKSB7XG4gICAgPGxhYmVsIFtodG1sRm9yXT1cImdyb3VwK3ZhbHVlKClcIiBjbGFzcz1cInZpc2lvbi1oNFwiPnt7IHRpdGxlIH19XG4gICAgICA8bmctY29udGVudD48L25nLWNvbnRlbnQ+XG4gICAgPC9sYWJlbD5cbiAgfVxuPC9kaXY+XG4iXX0=
|
|
@@ -39,6 +39,7 @@ export class FwTypeaheadComponent {
|
|
|
39
39
|
this.maxOptionsHeight = input('400px');
|
|
40
40
|
this.minOptionsHeight = input('0px');
|
|
41
41
|
this.optionsWidth = input('');
|
|
42
|
+
this.maxHeight = input('');
|
|
42
43
|
this.notifyOnValueChanges = effect(() => {
|
|
43
44
|
this.onChange(this.value());
|
|
44
45
|
});
|
|
@@ -46,6 +47,7 @@ export class FwTypeaheadComponent {
|
|
|
46
47
|
this.searchValueChanges$ = toObservable(this.searchValue);
|
|
47
48
|
this.allowNew = input(true);
|
|
48
49
|
this.placeholder = input('Enter tags...');
|
|
50
|
+
this.showPlaceholder = computed(() => this.value().length === 0);
|
|
49
51
|
/**
|
|
50
52
|
* Options after they've been both filtered for matching the search and already selected
|
|
51
53
|
*/
|
|
@@ -225,13 +227,13 @@ export class FwTypeaheadComponent {
|
|
|
225
227
|
newlyFocused.scrollIntoView();
|
|
226
228
|
}
|
|
227
229
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FwTypeaheadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
228
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: FwTypeaheadComponent, isStandalone: true, selector: "fw-typeahead", inputs: { loading: { classPropertyName: "loading", publicName: "loading", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, value: { classPropertyName: "value", publicName: "value", isSignal: true, isRequired: false, transformFunction: null }, optionsInput: { classPropertyName: "optionsInput", publicName: "options", isSignal: true, isRequired: false, transformFunction: null }, maxOptionsHeight: { classPropertyName: "maxOptionsHeight", publicName: "maxOptionsHeight", isSignal: true, isRequired: false, transformFunction: null }, minOptionsHeight: { classPropertyName: "minOptionsHeight", publicName: "minOptionsHeight", isSignal: true, isRequired: false, transformFunction: null }, optionsWidth: { classPropertyName: "optionsWidth", publicName: "optionsWidth", isSignal: true, isRequired: false, transformFunction: null }, searchValue: { classPropertyName: "searchValue", publicName: "searchValue", isSignal: true, isRequired: false, transformFunction: null }, allowNew: { classPropertyName: "allowNew", publicName: "allowNew", isSignal: true, isRequired: false, transformFunction: null }, placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { loading: "loadingChange", disabled: "disabledChange", value: "valueChange", searchValue: "searchValueChange" }, host: { listeners: { "document:click": "outsideClick()", "click": "onClick($event)" } }, providers: [
|
|
230
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: FwTypeaheadComponent, isStandalone: true, selector: "fw-typeahead", inputs: { loading: { classPropertyName: "loading", publicName: "loading", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, value: { classPropertyName: "value", publicName: "value", isSignal: true, isRequired: false, transformFunction: null }, optionsInput: { classPropertyName: "optionsInput", publicName: "options", isSignal: true, isRequired: false, transformFunction: null }, maxOptionsHeight: { classPropertyName: "maxOptionsHeight", publicName: "maxOptionsHeight", isSignal: true, isRequired: false, transformFunction: null }, minOptionsHeight: { classPropertyName: "minOptionsHeight", publicName: "minOptionsHeight", isSignal: true, isRequired: false, transformFunction: null }, optionsWidth: { classPropertyName: "optionsWidth", publicName: "optionsWidth", isSignal: true, isRequired: false, transformFunction: null }, maxHeight: { classPropertyName: "maxHeight", publicName: "maxHeight", isSignal: true, isRequired: false, transformFunction: null }, searchValue: { classPropertyName: "searchValue", publicName: "searchValue", isSignal: true, isRequired: false, transformFunction: null }, allowNew: { classPropertyName: "allowNew", publicName: "allowNew", isSignal: true, isRequired: false, transformFunction: null }, placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { loading: "loadingChange", disabled: "disabledChange", value: "valueChange", searchValue: "searchValueChange" }, host: { listeners: { "document:click": "outsideClick()", "click": "onClick($event)" } }, providers: [
|
|
229
231
|
{
|
|
230
232
|
provide: NG_VALUE_ACCESSOR,
|
|
231
233
|
useExisting: forwardRef(() => FwTypeaheadComponent),
|
|
232
234
|
multi: true,
|
|
233
235
|
},
|
|
234
|
-
], viewQueries: [{ propertyName: "trigger", first: true, predicate: CdkMenuTrigger, descendants: true, isSignal: true }, { propertyName: "displayedOptions", predicate: FwMenuItemComponent, descendants: true, isSignal: true }, { propertyName: "inputRef", first: true, predicate: ["input"], descendants: true, isSignal: true }], ngImport: i0, template: "<div\n (click)=\"focusInput($event)\"\n class=\"input-container\"\n [ngClass]=\"{ 'disabled': disabled() }\"\n [cdkMenuTriggerFor]=\"menuContent\"\n fwMenuRegister\n #inputContainer>\n <fw-chip-list class=\"chips\">\n @for(val of value(); track val) {\n <fw-chip\n color=\"primary\"\n [showClose]=\"true\"\n [title]=\"val\"\n [selectable]=\"false\"\n (close)=\"closeChip(val)\"\n />\n }\n </fw-chip-list>\n <input\n test-id=\"typeahead-input\"\n [placeholder]=\"placeholder()\"\n [disabled]=\"disabled()\"\n [value]=\"searchValue()\"\n (keyup)=\"handleSearchChange($event)\"\n (keydown)=\"handleKey($event)\"\n (focusout)=\"onFocusLoss($event)\"\n #input\n type=\"text\"\n />\n @if(loading()) {\n <fw-progress-spinner size=\"small\"/>\n }\n</div>\n<ng-template #menuContent>\n <fw-menu-container\n [maxHeight]=\"maxOptionsHeight()\"\n [minHeight]=\"minOptionsHeight()\"\n [width]=\"optionsWidth() || inputContainer.offsetWidth - 2 + 'px'\"\n >\n <fw-menu>\n @if(loading() && !displayNewOption()) {\n <fw-menu-item title=\"Searching...\" [disabled]=\"true\"/>\n } @else if(!loading()) {\n @for (option of filteredOptions(); track option) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(option)\"\n [title]=\"option\"\n [focused]=\"focusedOption() === option\"\n [value]=\"option\"\n />\n }\n @empty {\n @if (!displayNewOption()) {\n <fw-menu-item title=\"No tag suggestions\" [disabled]=\"true\"/>\n }\n }\n }\n @if(displayNewOption()) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(searchValue())\"\n [title]=\"searchValue()\"\n [value]=\"searchValue()\"\n [focused]=\"focusedOption() === searchValue()\">\n <p class=\"new-tag\">New</p>\n </fw-menu-item>\n }\n </fw-menu>\n </fw-menu-container>\n</ng-template>\n", styles: [".new-tag{margin:0;color:var(--typography-light)}:host{display:flex;flex-direction:column;width:100%;line-height:21px}:host.disabled{cursor:not-allowed}:host.disabled fw-icon{cursor:not-allowed!important}:host .chips,:host fw-progress-spinner{margin:-4px 0}:host .input-container{width:100%;box-sizing:border-box;color:var(--typography-light);background:var(--page-light);display:flex;padding:8px;align-items:center;border-radius:6px;border:1px solid var(--separations-input);font-family:Inter,sans-serif}:host .input-container:focus-within{border:1px solid var(--primary-base)}:host .input-container input{min-width:80px;font-size:14px;flex:1 1 80px;color:var(--typography-base);background:var(--page-light);border:none}:host .input-container input:focus{outline:none;border:none}:host .input-container input::placeholder{color:var(--typography-light)}:host .input-container .context{color:var(--typography-light)}:host.errored .input-container,:host.ng-touched.ng-invalid .input-container{border:1px solid var(--red-base)}.disabled{opacity:.4;pointer-events:none}\n"], dependencies: [{ kind: "ngmodule", type: FwTextInputModule }, { kind: "ngmodule", type: FwChipModule }, { kind: "component", type: i1.FwChipComponent, selector: "fw-chip", inputs: ["value", "variant", "color", "icon", "title", "description", "showClose", "disabled", "selected", "textWrap", "selectable"], outputs: ["close", "select"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "ngmodule", type: FwMenuModule }, { kind: "component", type: i2.FwMenuComponent, selector: "fw-menu", inputs: ["disabled", "size", "multiSelect", "useCheckbox", "value"], outputs: ["change"] }, { kind: "component", type: i3.FwMenuContainerComponent, selector: "fw-menu-container, fw-menu-filter", inputs: ["width", "maxHeight", "minHeight", "border", "shadow", "showFilter", "filterText", "focusFilterOnMount", "offset", "emptyText", "filterFn", "additionalMenuItems", "keyHandler"], outputs: ["filteredMenuItemChange", "filterChanged"] }, { kind: "component", type: i4.FwMenuItemComponent, selector: "fw-menu-item", inputs: ["value", "size", "title", "description", "icon", "iconColor", "disabled", "showCheckbox", "checkboxColor", "multiSelect", "hidden", "collapsed", "href", "target", "subItemsOpen", "mouseEnterHandler", "focused", "selected"], outputs: ["mouseEnterHandlerChange", "click"] }, { kind: "ngmodule", type: CdkMenuModule }, { kind: "directive", type: i5.CdkMenuTrigger, selector: "[cdkMenuTriggerFor]", inputs: ["cdkMenuTriggerFor", "cdkMenuPosition", "cdkMenuTriggerData"], outputs: ["cdkMenuOpened", "cdkMenuClosed"], exportAs: ["cdkMenuTriggerFor"] }, { kind: "directive", type: MenuRegisterDirective, selector: "[fwMenuRegister]" }, { kind: "ngmodule", type: FwProgressModule }, { kind: "component", type: i6.FwProgressSpinnerComponent, selector: "fw-progress-spinner", inputs: ["mode", "size", "color", "showValue", "value"] }, { kind: "component", type: FwChipListComponent, selector: "fw-chip-list", inputs: ["resizeDebounceMs"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
236
|
+
], viewQueries: [{ propertyName: "trigger", first: true, predicate: CdkMenuTrigger, descendants: true, isSignal: true }, { propertyName: "displayedOptions", predicate: FwMenuItemComponent, descendants: true, isSignal: true }, { propertyName: "inputRef", first: true, predicate: ["input"], descendants: true, isSignal: true }], ngImport: i0, template: "<div\n (click)=\"focusInput($event)\"\n class=\"input-container\"\n [ngClass]=\"{ 'disabled': disabled() }\"\n [style.max-height]=\"maxHeight()\"\n [cdkMenuTriggerFor]=\"menuContent\"\n fwMenuRegister\n #inputContainer>\n <fw-chip-list class=\"chips\" [disableOverflow]=\"!!maxHeight()\">\n @for(val of value(); track val) {\n <fw-chip\n color=\"primary\"\n [showClose]=\"true\"\n [title]=\"val\"\n [selectable]=\"false\"\n (close)=\"closeChip(val)\"\n />\n }\n </fw-chip-list>\n <input\n test-id=\"typeahead-input\"\n [placeholder]=\"showPlaceholder() ? placeholder() : ''\"\n [disabled]=\"disabled()\"\n [value]=\"searchValue()\"\n (keyup)=\"handleSearchChange($event)\"\n (keydown)=\"handleKey($event)\"\n (focusout)=\"onFocusLoss($event)\"\n #input\n type=\"text\"\n />\n @if(loading()) {\n <fw-progress-spinner size=\"small\"/>\n }\n</div>\n<ng-template #menuContent>\n <fw-menu-container\n [maxHeight]=\"maxOptionsHeight()\"\n [minHeight]=\"minOptionsHeight()\"\n [width]=\"optionsWidth() || inputContainer.offsetWidth - 2 + 'px'\"\n >\n <fw-menu>\n @if(loading() && !displayNewOption()) {\n <fw-menu-item title=\"Searching...\" [disabled]=\"true\"/>\n } @else if(!loading()) {\n @for (option of filteredOptions(); track option) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(option)\"\n [title]=\"option\"\n [focused]=\"focusedOption() === option\"\n [value]=\"option\"\n />\n }\n @empty {\n @if (!displayNewOption()) {\n <fw-menu-item title=\"No tag suggestions\" [disabled]=\"true\"/>\n }\n }\n }\n @if(displayNewOption()) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(searchValue())\"\n [title]=\"searchValue()\"\n [value]=\"searchValue()\"\n [focused]=\"focusedOption() === searchValue()\">\n <p class=\"new-tag\">New</p>\n </fw-menu-item>\n }\n </fw-menu>\n </fw-menu-container>\n</ng-template>\n", styles: [".new-tag{margin:0;color:var(--typography-light)}:host{display:flex;flex-direction:column;width:100%;line-height:21px}:host.disabled{cursor:not-allowed}:host.disabled fw-icon{cursor:not-allowed!important}:host .chips,:host fw-progress-spinner{margin:-4px 0}:host .input-container{width:100%;box-sizing:border-box;color:var(--typography-light);background:var(--page-light);display:flex;padding:8px;row-gap:8px;align-items:center;border-radius:6px;border:1px solid var(--separations-input);font-family:Inter,sans-serif;flex-wrap:wrap;overflow-y:auto}:host .input-container:focus-within{border:1px solid var(--primary-base)}:host .input-container input{min-width:80px;font-size:14px;flex:1 1 80px;color:var(--typography-base);background:var(--page-light);border:none}:host .input-container input:focus{outline:none;border:none}:host .input-container input::placeholder{color:var(--typography-light)}:host .input-container .context{color:var(--typography-light)}:host.errored .input-container,:host.ng-touched.ng-invalid .input-container{border:1px solid var(--red-base)}.disabled{opacity:.4;pointer-events:none}\n"], dependencies: [{ kind: "ngmodule", type: FwTextInputModule }, { kind: "ngmodule", type: FwChipModule }, { kind: "component", type: i1.FwChipComponent, selector: "fw-chip", inputs: ["value", "variant", "color", "icon", "title", "description", "showClose", "disabled", "selected", "textWrap", "selectable"], outputs: ["close", "select"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "ngmodule", type: FwMenuModule }, { kind: "component", type: i2.FwMenuComponent, selector: "fw-menu", inputs: ["disabled", "size", "multiSelect", "useCheckbox", "value"], outputs: ["change"] }, { kind: "component", type: i3.FwMenuContainerComponent, selector: "fw-menu-container, fw-menu-filter", inputs: ["width", "maxHeight", "minHeight", "border", "shadow", "showFilter", "filterText", "focusFilterOnMount", "offset", "emptyText", "filterFn", "additionalMenuItems", "keyHandler"], outputs: ["filteredMenuItemChange", "filterChanged"] }, { kind: "component", type: i4.FwMenuItemComponent, selector: "fw-menu-item", inputs: ["value", "size", "title", "description", "icon", "iconColor", "disabled", "showCheckbox", "checkboxColor", "multiSelect", "hidden", "collapsed", "href", "target", "subItemsOpen", "mouseEnterHandler", "focused", "selected"], outputs: ["mouseEnterHandlerChange", "click"] }, { kind: "ngmodule", type: CdkMenuModule }, { kind: "directive", type: i5.CdkMenuTrigger, selector: "[cdkMenuTriggerFor]", inputs: ["cdkMenuTriggerFor", "cdkMenuPosition", "cdkMenuTriggerData"], outputs: ["cdkMenuOpened", "cdkMenuClosed"], exportAs: ["cdkMenuTriggerFor"] }, { kind: "directive", type: MenuRegisterDirective, selector: "[fwMenuRegister]" }, { kind: "ngmodule", type: FwProgressModule }, { kind: "component", type: i6.FwProgressSpinnerComponent, selector: "fw-progress-spinner", inputs: ["mode", "size", "color", "showValue", "value"] }, { kind: "component", type: FwChipListComponent, selector: "fw-chip-list", inputs: ["resizeDebounceMs", "disableOverflow"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
235
237
|
}
|
|
236
238
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FwTypeaheadComponent, decorators: [{
|
|
237
239
|
type: Component,
|
|
@@ -251,7 +253,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
251
253
|
useExisting: forwardRef(() => FwTypeaheadComponent),
|
|
252
254
|
multi: true,
|
|
253
255
|
},
|
|
254
|
-
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n (click)=\"focusInput($event)\"\n class=\"input-container\"\n [ngClass]=\"{ 'disabled': disabled() }\"\n [cdkMenuTriggerFor]=\"menuContent\"\n fwMenuRegister\n #inputContainer>\n <fw-chip-list class=\"chips\">\n @for(val of value(); track val) {\n <fw-chip\n color=\"primary\"\n [showClose]=\"true\"\n [title]=\"val\"\n [selectable]=\"false\"\n (close)=\"closeChip(val)\"\n />\n }\n </fw-chip-list>\n <input\n test-id=\"typeahead-input\"\n [placeholder]=\"placeholder()\"\n [disabled]=\"disabled()\"\n [value]=\"searchValue()\"\n (keyup)=\"handleSearchChange($event)\"\n (keydown)=\"handleKey($event)\"\n (focusout)=\"onFocusLoss($event)\"\n #input\n type=\"text\"\n />\n @if(loading()) {\n <fw-progress-spinner size=\"small\"/>\n }\n</div>\n<ng-template #menuContent>\n <fw-menu-container\n [maxHeight]=\"maxOptionsHeight()\"\n [minHeight]=\"minOptionsHeight()\"\n [width]=\"optionsWidth() || inputContainer.offsetWidth - 2 + 'px'\"\n >\n <fw-menu>\n @if(loading() && !displayNewOption()) {\n <fw-menu-item title=\"Searching...\" [disabled]=\"true\"/>\n } @else if(!loading()) {\n @for (option of filteredOptions(); track option) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(option)\"\n [title]=\"option\"\n [focused]=\"focusedOption() === option\"\n [value]=\"option\"\n />\n }\n @empty {\n @if (!displayNewOption()) {\n <fw-menu-item title=\"No tag suggestions\" [disabled]=\"true\"/>\n }\n }\n }\n @if(displayNewOption()) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(searchValue())\"\n [title]=\"searchValue()\"\n [value]=\"searchValue()\"\n [focused]=\"focusedOption() === searchValue()\">\n <p class=\"new-tag\">New</p>\n </fw-menu-item>\n }\n </fw-menu>\n </fw-menu-container>\n</ng-template>\n", styles: [".new-tag{margin:0;color:var(--typography-light)}:host{display:flex;flex-direction:column;width:100%;line-height:21px}:host.disabled{cursor:not-allowed}:host.disabled fw-icon{cursor:not-allowed!important}:host .chips,:host fw-progress-spinner{margin:-4px 0}:host .input-container{width:100%;box-sizing:border-box;color:var(--typography-light);background:var(--page-light);display:flex;padding:8px;align-items:center;border-radius:6px;border:1px solid var(--separations-input);font-family:Inter,sans-serif}:host .input-container:focus-within{border:1px solid var(--primary-base)}:host .input-container input{min-width:80px;font-size:14px;flex:1 1 80px;color:var(--typography-base);background:var(--page-light);border:none}:host .input-container input:focus{outline:none;border:none}:host .input-container input::placeholder{color:var(--typography-light)}:host .input-container .context{color:var(--typography-light)}:host.errored .input-container,:host.ng-touched.ng-invalid .input-container{border:1px solid var(--red-base)}.disabled{opacity:.4;pointer-events:none}\n"] }]
|
|
256
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n (click)=\"focusInput($event)\"\n class=\"input-container\"\n [ngClass]=\"{ 'disabled': disabled() }\"\n [style.max-height]=\"maxHeight()\"\n [cdkMenuTriggerFor]=\"menuContent\"\n fwMenuRegister\n #inputContainer>\n <fw-chip-list class=\"chips\" [disableOverflow]=\"!!maxHeight()\">\n @for(val of value(); track val) {\n <fw-chip\n color=\"primary\"\n [showClose]=\"true\"\n [title]=\"val\"\n [selectable]=\"false\"\n (close)=\"closeChip(val)\"\n />\n }\n </fw-chip-list>\n <input\n test-id=\"typeahead-input\"\n [placeholder]=\"showPlaceholder() ? placeholder() : ''\"\n [disabled]=\"disabled()\"\n [value]=\"searchValue()\"\n (keyup)=\"handleSearchChange($event)\"\n (keydown)=\"handleKey($event)\"\n (focusout)=\"onFocusLoss($event)\"\n #input\n type=\"text\"\n />\n @if(loading()) {\n <fw-progress-spinner size=\"small\"/>\n }\n</div>\n<ng-template #menuContent>\n <fw-menu-container\n [maxHeight]=\"maxOptionsHeight()\"\n [minHeight]=\"minOptionsHeight()\"\n [width]=\"optionsWidth() || inputContainer.offsetWidth - 2 + 'px'\"\n >\n <fw-menu>\n @if(loading() && !displayNewOption()) {\n <fw-menu-item title=\"Searching...\" [disabled]=\"true\"/>\n } @else if(!loading()) {\n @for (option of filteredOptions(); track option) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(option)\"\n [title]=\"option\"\n [focused]=\"focusedOption() === option\"\n [value]=\"option\"\n />\n }\n @empty {\n @if (!displayNewOption()) {\n <fw-menu-item title=\"No tag suggestions\" [disabled]=\"true\"/>\n }\n }\n }\n @if(displayNewOption()) {\n <fw-menu-item\n (click)=\"handleOptionClick($event)\"\n (mouseenter)=\"setFocusByValue(searchValue())\"\n [title]=\"searchValue()\"\n [value]=\"searchValue()\"\n [focused]=\"focusedOption() === searchValue()\">\n <p class=\"new-tag\">New</p>\n </fw-menu-item>\n }\n </fw-menu>\n </fw-menu-container>\n</ng-template>\n", styles: [".new-tag{margin:0;color:var(--typography-light)}:host{display:flex;flex-direction:column;width:100%;line-height:21px}:host.disabled{cursor:not-allowed}:host.disabled fw-icon{cursor:not-allowed!important}:host .chips,:host fw-progress-spinner{margin:-4px 0}:host .input-container{width:100%;box-sizing:border-box;color:var(--typography-light);background:var(--page-light);display:flex;padding:8px;row-gap:8px;align-items:center;border-radius:6px;border:1px solid var(--separations-input);font-family:Inter,sans-serif;flex-wrap:wrap;overflow-y:auto}:host .input-container:focus-within{border:1px solid var(--primary-base)}:host .input-container input{min-width:80px;font-size:14px;flex:1 1 80px;color:var(--typography-base);background:var(--page-light);border:none}:host .input-container input:focus{outline:none;border:none}:host .input-container input::placeholder{color:var(--typography-light)}:host .input-container .context{color:var(--typography-light)}:host.errored .input-container,:host.ng-touched.ng-invalid .input-container{border:1px solid var(--red-base)}.disabled{opacity:.4;pointer-events:none}\n"] }]
|
|
255
257
|
}], propDecorators: { outsideClick: [{
|
|
256
258
|
type: HostListener,
|
|
257
259
|
args: ['document:click']
|
|
@@ -259,4 +261,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
259
261
|
type: HostListener,
|
|
260
262
|
args: ['click', ['$event']]
|
|
261
263
|
}] } });
|
|
262
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typeahead.component.js","sourceRoot":"","sources":["../../../../../src/components/typeahead/typeahead.component.ts","../../../../../src/components/typeahead/typeahead.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CdkMenuModule, CdkMenuTrigger } from '@angular/cdk/menu';\nimport { NgClass } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  computed,\n  effect,\n  ElementRef,\n  forwardRef,\n  HostListener,\n  inject,\n  input,\n  model,\n  signal,\n  viewChild,\n  viewChildren,\n} from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\nimport { combineLatest, map, Observable, of, switchMap, tap } from 'rxjs';\n\nimport { MenuRegisterDirective } from '../../directives/menu-register.directive';\nimport { FwChipModule } from '../chip/chip.module';\nimport { FwChipListComponent } from '../chip-list/chip-list.component';\nimport { FwMenuModule } from '../menu/menu.module';\nimport { FwMenuItemComponent } from '../menu/menu-item/menu-item.component';\nimport { FwProgressModule } from '../progress/progress.module';\nimport { FwTextInputModule } from '../text-input/text-input.module';\n\ntype OptionFetchingFn = (searchString: string) => Observable<string[]>;\n\n/**\n * Component for having a user select multiple values while also allowing arbitrary values to be added\n * @see [Vision Docs](https://cdn.flywheel.io/docs/vision/master/?path=/docs/form-controls-typeahead--docs)\n */\n@Component({\n  selector: 'fw-typeahead',\n  standalone: true,\n  imports: [\n    FwTextInputModule,\n    FwChipModule,\n    ReactiveFormsModule,\n    FwMenuModule,\n    CdkMenuModule,\n    MenuRegisterDirective,\n    FwProgressModule,\n    FwChipListComponent,\n    NgClass,\n  ],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => FwTypeaheadComponent),\n      multi: true,\n    },\n  ],\n  templateUrl: './typeahead.component.html',\n  styleUrl: './typeahead.component.scss',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FwTypeaheadComponent implements ControlValueAccessor {\n  @HostListener('document:click') outsideClick(): void {\n      this.trigger().close();\n  }\n\n  trigger = viewChild(CdkMenuTrigger);\n  changeDetector = inject(ChangeDetectorRef);\n  loading = model(false);\n  disabled = model(false);\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled.set(isDisabled);\n  }\n\n  writeValue(incomingValue: string[]): void {\n    this.value.set(incomingValue);\n  }\n\n  onChange = (_: string[]): void => {\n  };\n\n  registerOnChange(onChangeFn): void {\n    this.onChange = onChangeFn;\n  }\n\n  onTouched = (): void => {\n  };\n\n  registerOnTouched(onTouchedFn): void {\n    this.onTouched = onTouchedFn;\n  }\n\n  @HostListener('click', ['$event']) onClick(event: MouseEvent): void {\n    event.preventDefault();\n    event.stopPropagation();\n  }\n\n  value = model<string[]>([]);\n  value$ = toObservable(this.value);\n  optionsInput = input<string[] | OptionFetchingFn>([], { alias: 'options' });\n  optionsInput$ = toObservable(this.optionsInput);\n\n  maxOptionsHeight = input('400px');\n  minOptionsHeight = input('0px');\n  optionsWidth = input('');\n\n  notifyOnValueChanges = effect(() => {\n    this.onChange(this.value());\n  });\n\n  searchValue = model('');\n  searchValueChanges$ = toObservable(this.searchValue);\n\n  handleSearchChange(event: KeyboardEvent): void {\n    if (this.trigger().closed) {\n      this.trigger().open();\n    }\n    const value = (event.target as HTMLInputElement).value;\n    this.searchValue.set(value);\n  }\n\n  allowNew = input(true);\n  placeholder = input('Enter tags...');\n\n  /**\n   * Options after they've been both filtered for matching the search and already selected\n   */\n  filteredOptions$: Observable<string[]> = combineLatest([\n    this.searchValueChanges$,\n    this.optionsInput$,\n  ]).pipe(\n    tap(() => this.loading.set(true)),\n\n    // first determine if it's a function or static options\n    switchMap(([searchString, optionInput]) => {\n      if (typeof optionInput === 'function') {\n        const fetchingFunction = optionInput as OptionFetchingFn;\n        return fetchingFunction(searchString);\n      } else {\n        // if they're static options we need to do the search filtering\n        const filteredOptionInput = optionInput?.filter(opt => opt.toLowerCase().includes(searchString.toLowerCase()));\n        return of(filteredOptionInput);\n      }\n\n    }),\n\n    // next filter out already selected values\n    switchMap(options => combineLatest([of(options), this.value$])),\n    map(([options, currentValue]) => {\n      return options?.filter(option => {\n        const alreadySelected = currentValue.find(val => val.toLowerCase() === option.toLowerCase());\n        return !alreadySelected;\n      });\n    }),\n\n    tap(() => this.loading.set(false)),\n  );\n  /**\n   * Options after they've been both filtered for matching the search and already selected\n   */\n  filteredOptions = toSignal(this.filteredOptions$);\n\n\n  closeChip(chipValue: string): void {\n    const currentValue = this.value();\n    const newValue = currentValue.filter(val => val !== chipValue);\n    this.value.set(newValue);\n    this.focusInput();\n    // eslint-disable-next-line @rx-angular/no-explicit-change-detection-apis\n    this.changeDetector.detectChanges();\n  }\n\n  displayNewOption = computed(() => {\n    const newValue = this.searchValue();\n\n    if (!newValue) {\n      return false;\n    }\n\n    const directMatch = this.filteredOptions()?.includes?.(newValue);\n    const loading = this.loading();\n    const alreadySelected = Boolean(this.value().find(val => val.toLowerCase() === newValue.toLowerCase()));\n\n\n    return this.allowNew() && !alreadySelected && (!directMatch || loading);\n  });\n\n  addValue = (newValue: string): void => {\n\n    const isInOptions = this.filteredOptions().includes(newValue);\n    const isNewWhileDisallowed = !this.allowNew() && !isInOptions;\n\n    const isDuplicate = this.value().includes(newValue.toLowerCase());\n\n    if (isNewWhileDisallowed || isDuplicate) {\n      return;\n    }\n\n    this.value.update(prevValue => [\n      ...prevValue || [],\n      newValue,\n    ]);\n  };\n\n  handleOptionClick = (optionValue): void => {\n    this.onTouched();\n    this.addValue(optionValue);\n    this.focusInput();\n    this.searchValue.set('');\n  };\n\n  resetFocusOnOptionsChange = effect(() => {\n    if (this.displayedOptions() || this.searchValue()) {\n      this.focusedIndex.set(0);\n    }\n  }, { allowSignalWrites: true });\n\n  displayedOptions = viewChildren(FwMenuItemComponent);\n  focusedIndex = signal(0);\n  focusedOption = computed(() => {\n    const options = this.displayedOptions();\n    const focused = this.focusedIndex();\n\n    const onlyOne = options.length === 1;\n    return onlyOne ? this.searchValue() : options[focused]?.value;\n  });\n\n  inputRef = viewChild.required<ElementRef<HTMLInputElement>>('input');\n\n  focusInput(e: MouseEvent = null): void {\n    e?.preventDefault();\n    e?.stopPropagation();\n    e?.stopImmediatePropagation();\n    // eslint-disable-next-line @rx-angular/prefer-no-layout-sensitive-apis\n    this.inputRef().nativeElement.focus();\n  }\n\n  onFocusLoss(_: Event): void {\n    this.onTouched();\n  }\n\n  /* eslint-disable @typescript-eslint/naming-convention */\n  keyHandlerMap: { [keycode: string]: (e: KeyboardEvent) => void } = {\n    'Enter': () => {\n      this.onTouched();\n      const newValue = this.focusedOption() || this.searchValue();\n      const duplicate = this.value().find(val => val.toLowerCase() === newValue.toLowerCase());\n\n      if (!newValue || duplicate) {\n        return;\n      }\n\n      this.addValue(newValue);\n      this.searchValue.set('');\n    },\n    'ArrowDown': () => {\n      this.moveFocused('down');\n    },\n    'ArrowUp': () => {\n      this.moveFocused('up');\n    },\n    'Backspace': () => {\n      const searchIsEmpty = !this.searchValue(); // any falsy value is empty\n      if (searchIsEmpty && this.value()?.length > 0) {\n        this.closeChip(this.value()[this.value().length - 1]);\n      }\n      return !searchIsEmpty;\n    },\n    'Tab': () => {\n      if (this.trigger().isOpen) {\n        this.trigger().close();\n      }\n    },\n  };\n\n  /* eslint-enable */\n\n  handleKey(e: KeyboardEvent): void {\n    const handler = this.keyHandlerMap[e.key];\n    handler?.(e);\n  }\n\n  setFocusByValue(value: string): void {\n    const options = this.displayedOptions();\n    const foundIndex = options.findIndex(opt => opt.value === value);\n    const finalIndex = foundIndex === -1 ? 0 : foundIndex;\n    this.focusedIndex.set(finalIndex);\n  }\n\n  private moveFocused(direction: 'up' | 'down'): void {\n    const options = this.displayedOptions();\n    const index = this.focusedIndex();\n\n    const clampIndex = (n: number): number => {\n      if (n < 0) {\n        return options.length - 1;\n      } else if (n >= options.length) {\n        return 0;\n      }\n\n      return n;\n    };\n\n    const indexChange = direction === 'up' ? -1 : 1;\n\n    const newIndex = clampIndex(index + indexChange);\n    this.focusedIndex.set(newIndex);\n\n    const newlyFocused = this.displayedOptions()[this.focusedIndex()];\n\n    // eslint-disable-next-line @rx-angular/prefer-no-layout-sensitive-apis\n    newlyFocused.scrollIntoView();\n  }\n\n}\n","<div\n  (click)=\"focusInput($event)\"\n  class=\"input-container\"\n  [ngClass]=\"{ 'disabled': disabled() }\"\n  [cdkMenuTriggerFor]=\"menuContent\"\n  fwMenuRegister\n  #inputContainer>\n  <fw-chip-list class=\"chips\">\n    @for(val of value(); track val) {\n      <fw-chip\n        color=\"primary\"\n        [showClose]=\"true\"\n        [title]=\"val\"\n        [selectable]=\"false\"\n        (close)=\"closeChip(val)\"\n      />\n    }\n  </fw-chip-list>\n  <input\n    test-id=\"typeahead-input\"\n    [placeholder]=\"placeholder()\"\n    [disabled]=\"disabled()\"\n    [value]=\"searchValue()\"\n    (keyup)=\"handleSearchChange($event)\"\n    (keydown)=\"handleKey($event)\"\n    (focusout)=\"onFocusLoss($event)\"\n    #input\n    type=\"text\"\n  />\n  @if(loading()) {\n    <fw-progress-spinner size=\"small\"/>\n  }\n</div>\n<ng-template #menuContent>\n    <fw-menu-container\n      [maxHeight]=\"maxOptionsHeight()\"\n      [minHeight]=\"minOptionsHeight()\"\n      [width]=\"optionsWidth() || inputContainer.offsetWidth - 2 + 'px'\"\n    >\n      <fw-menu>\n        @if(loading() && !displayNewOption()) {\n          <fw-menu-item title=\"Searching...\" [disabled]=\"true\"/>\n        } @else if(!loading()) {\n          @for (option of filteredOptions(); track option) {\n            <fw-menu-item\n              (click)=\"handleOptionClick($event)\"\n              (mouseenter)=\"setFocusByValue(option)\"\n              [title]=\"option\"\n              [focused]=\"focusedOption() === option\"\n              [value]=\"option\"\n            />\n          }\n          @empty {\n            @if (!displayNewOption()) {\n              <fw-menu-item title=\"No tag suggestions\" [disabled]=\"true\"/>\n            }\n          }\n        }\n        @if(displayNewOption()) {\n          <fw-menu-item\n            (click)=\"handleOptionClick($event)\"\n            (mouseenter)=\"setFocusByValue(searchValue())\"\n            [title]=\"searchValue()\"\n            [value]=\"searchValue()\"\n            [focused]=\"focusedOption() === searchValue()\">\n            <p class=\"new-tag\">New</p>\n          </fw-menu-item>\n        }\n      </fw-menu>\n    </fw-menu-container>\n</ng-template>\n"]}
|
|
264
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typeahead.component.js","sourceRoot":"","sources":["../../../../../src/components/typeahead/typeahead.component.ts","../../../../../src/components/typeahead/typeahead.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CdkMenuModule, CdkMenuTrigger } from '@angular/cdk/menu';\nimport { NgClass } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  computed,\n  effect,\n  ElementRef,\n  forwardRef,\n  HostListener,\n  inject,\n  input,\n  model,\n  signal,\n  viewChild,\n  viewChildren,\n} from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\nimport { combineLatest, map, Observable, of, switchMap, tap } from 'rxjs';\n\nimport { MenuRegisterDirective } from '../../directives/menu-register.directive';\nimport { FwChipModule } from '../chip/chip.module';\nimport { FwChipListComponent } from '../chip-list/chip-list.component';\nimport { FwMenuModule } from '../menu/menu.module';\nimport { FwMenuItemComponent } from '../menu/menu-item/menu-item.component';\nimport { FwProgressModule } from '../progress/progress.module';\nimport { FwTextInputModule } from '../text-input/text-input.module';\n\ntype OptionFetchingFn = (searchString: string) => Observable<string[]>;\n\n/**\n * Component for having a user select multiple values while also allowing arbitrary values to be added\n * @see [Vision Docs](https://cdn.flywheel.io/docs/vision/master/?path=/docs/form-controls-typeahead--docs)\n */\n@Component({\n  selector: 'fw-typeahead',\n  standalone: true,\n  imports: [\n    FwTextInputModule,\n    FwChipModule,\n    ReactiveFormsModule,\n    FwMenuModule,\n    CdkMenuModule,\n    MenuRegisterDirective,\n    FwProgressModule,\n    FwChipListComponent,\n    NgClass,\n  ],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => FwTypeaheadComponent),\n      multi: true,\n    },\n  ],\n  templateUrl: './typeahead.component.html',\n  styleUrl: './typeahead.component.scss',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FwTypeaheadComponent implements ControlValueAccessor {\n  @HostListener('document:click') outsideClick(): void {\n      this.trigger().close();\n  }\n\n  trigger = viewChild(CdkMenuTrigger);\n  changeDetector = inject(ChangeDetectorRef);\n  loading = model(false);\n  disabled = model(false);\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled.set(isDisabled);\n  }\n\n  writeValue(incomingValue: string[]): void {\n    this.value.set(incomingValue);\n  }\n\n  onChange = (_: string[]): void => {\n  };\n\n  registerOnChange(onChangeFn): void {\n    this.onChange = onChangeFn;\n  }\n\n  onTouched = (): void => {\n  };\n\n  registerOnTouched(onTouchedFn): void {\n    this.onTouched = onTouchedFn;\n  }\n\n  @HostListener('click', ['$event']) onClick(event: MouseEvent): void {\n    event.preventDefault();\n    event.stopPropagation();\n  }\n\n  value = model<string[]>([]);\n  value$ = toObservable(this.value);\n  optionsInput = input<string[] | OptionFetchingFn>([], { alias: 'options' });\n  optionsInput$ = toObservable(this.optionsInput);\n\n  maxOptionsHeight = input('400px');\n  minOptionsHeight = input('0px');\n  optionsWidth = input('');\n  maxHeight = input('');\n\n  notifyOnValueChanges = effect(() => {\n    this.onChange(this.value());\n  });\n\n  searchValue = model('');\n  searchValueChanges$ = toObservable(this.searchValue);\n\n  handleSearchChange(event: KeyboardEvent): void {\n    if (this.trigger().closed) {\n      this.trigger().open();\n    }\n    const value = (event.target as HTMLInputElement).value;\n    this.searchValue.set(value);\n  }\n\n  allowNew = input(true);\n  placeholder = input('Enter tags...');\n  showPlaceholder = computed(() => this.value().length === 0);\n\n  /**\n   * Options after they've been both filtered for matching the search and already selected\n   */\n  filteredOptions$: Observable<string[]> = combineLatest([\n    this.searchValueChanges$,\n    this.optionsInput$,\n  ]).pipe(\n    tap(() => this.loading.set(true)),\n\n    // first determine if it's a function or static options\n    switchMap(([searchString, optionInput]) => {\n      if (typeof optionInput === 'function') {\n        const fetchingFunction = optionInput as OptionFetchingFn;\n        return fetchingFunction(searchString);\n      } else {\n        // if they're static options we need to do the search filtering\n        const filteredOptionInput = optionInput?.filter(opt => opt.toLowerCase().includes(searchString.toLowerCase()));\n        return of(filteredOptionInput);\n      }\n\n    }),\n\n    // next filter out already selected values\n    switchMap(options => combineLatest([of(options), this.value$])),\n    map(([options, currentValue]) => {\n      return options?.filter(option => {\n        const alreadySelected = currentValue.find(val => val.toLowerCase() === option.toLowerCase());\n        return !alreadySelected;\n      });\n    }),\n\n    tap(() => this.loading.set(false)),\n  );\n  /**\n   * Options after they've been both filtered for matching the search and already selected\n   */\n  filteredOptions = toSignal(this.filteredOptions$);\n\n\n  closeChip(chipValue: string): void {\n    const currentValue = this.value();\n    const newValue = currentValue.filter(val => val !== chipValue);\n    this.value.set(newValue);\n    this.focusInput();\n    // eslint-disable-next-line @rx-angular/no-explicit-change-detection-apis\n    this.changeDetector.detectChanges();\n  }\n\n  displayNewOption = computed(() => {\n    const newValue = this.searchValue();\n\n    if (!newValue) {\n      return false;\n    }\n\n    const directMatch = this.filteredOptions()?.includes?.(newValue);\n    const loading = this.loading();\n    const alreadySelected = Boolean(this.value().find(val => val.toLowerCase() === newValue.toLowerCase()));\n\n\n    return this.allowNew() && !alreadySelected && (!directMatch || loading);\n  });\n\n  addValue = (newValue: string): void => {\n\n    const isInOptions = this.filteredOptions().includes(newValue);\n    const isNewWhileDisallowed = !this.allowNew() && !isInOptions;\n\n    const isDuplicate = this.value().includes(newValue.toLowerCase());\n\n    if (isNewWhileDisallowed || isDuplicate) {\n      return;\n    }\n\n    this.value.update(prevValue => [\n      ...prevValue || [],\n      newValue,\n    ]);\n  };\n\n  handleOptionClick = (optionValue): void => {\n    this.onTouched();\n    this.addValue(optionValue);\n    this.focusInput();\n    this.searchValue.set('');\n  };\n\n  resetFocusOnOptionsChange = effect(() => {\n    if (this.displayedOptions() || this.searchValue()) {\n      this.focusedIndex.set(0);\n    }\n  }, { allowSignalWrites: true });\n\n  displayedOptions = viewChildren(FwMenuItemComponent);\n  focusedIndex = signal(0);\n  focusedOption = computed(() => {\n    const options = this.displayedOptions();\n    const focused = this.focusedIndex();\n\n    const onlyOne = options.length === 1;\n    return onlyOne ? this.searchValue() : options[focused]?.value;\n  });\n\n  inputRef = viewChild.required<ElementRef<HTMLInputElement>>('input');\n\n  focusInput(e: MouseEvent = null): void {\n    e?.preventDefault();\n    e?.stopPropagation();\n    e?.stopImmediatePropagation();\n    // eslint-disable-next-line @rx-angular/prefer-no-layout-sensitive-apis\n    this.inputRef().nativeElement.focus();\n  }\n\n  onFocusLoss(_: Event): void {\n    this.onTouched();\n  }\n\n  /* eslint-disable @typescript-eslint/naming-convention */\n  keyHandlerMap: { [keycode: string]: (e: KeyboardEvent) => void } = {\n    'Enter': () => {\n      this.onTouched();\n      const newValue = this.focusedOption() || this.searchValue();\n      const duplicate = this.value().find(val => val.toLowerCase() === newValue.toLowerCase());\n\n      if (!newValue || duplicate) {\n        return;\n      }\n\n      this.addValue(newValue);\n      this.searchValue.set('');\n    },\n    'ArrowDown': () => {\n      this.moveFocused('down');\n    },\n    'ArrowUp': () => {\n      this.moveFocused('up');\n    },\n    'Backspace': () => {\n      const searchIsEmpty = !this.searchValue(); // any falsy value is empty\n      if (searchIsEmpty && this.value()?.length > 0) {\n        this.closeChip(this.value()[this.value().length - 1]);\n      }\n      return !searchIsEmpty;\n    },\n    'Tab': () => {\n      if (this.trigger().isOpen) {\n        this.trigger().close();\n      }\n    },\n  };\n\n  /* eslint-enable */\n\n  handleKey(e: KeyboardEvent): void {\n    const handler = this.keyHandlerMap[e.key];\n    handler?.(e);\n  }\n\n  setFocusByValue(value: string): void {\n    const options = this.displayedOptions();\n    const foundIndex = options.findIndex(opt => opt.value === value);\n    const finalIndex = foundIndex === -1 ? 0 : foundIndex;\n    this.focusedIndex.set(finalIndex);\n  }\n\n  private moveFocused(direction: 'up' | 'down'): void {\n    const options = this.displayedOptions();\n    const index = this.focusedIndex();\n\n    const clampIndex = (n: number): number => {\n      if (n < 0) {\n        return options.length - 1;\n      } else if (n >= options.length) {\n        return 0;\n      }\n\n      return n;\n    };\n\n    const indexChange = direction === 'up' ? -1 : 1;\n\n    const newIndex = clampIndex(index + indexChange);\n    this.focusedIndex.set(newIndex);\n\n    const newlyFocused = this.displayedOptions()[this.focusedIndex()];\n\n    // eslint-disable-next-line @rx-angular/prefer-no-layout-sensitive-apis\n    newlyFocused.scrollIntoView();\n  }\n\n}\n","<div\n  (click)=\"focusInput($event)\"\n  class=\"input-container\"\n  [ngClass]=\"{ 'disabled': disabled() }\"\n  [style.max-height]=\"maxHeight()\"\n  [cdkMenuTriggerFor]=\"menuContent\"\n  fwMenuRegister\n  #inputContainer>\n  <fw-chip-list class=\"chips\" [disableOverflow]=\"!!maxHeight()\">\n    @for(val of value(); track val) {\n      <fw-chip\n        color=\"primary\"\n        [showClose]=\"true\"\n        [title]=\"val\"\n        [selectable]=\"false\"\n        (close)=\"closeChip(val)\"\n      />\n    }\n  </fw-chip-list>\n  <input\n    test-id=\"typeahead-input\"\n    [placeholder]=\"showPlaceholder() ? placeholder() : ''\"\n    [disabled]=\"disabled()\"\n    [value]=\"searchValue()\"\n    (keyup)=\"handleSearchChange($event)\"\n    (keydown)=\"handleKey($event)\"\n    (focusout)=\"onFocusLoss($event)\"\n    #input\n    type=\"text\"\n  />\n  @if(loading()) {\n    <fw-progress-spinner size=\"small\"/>\n  }\n</div>\n<ng-template #menuContent>\n    <fw-menu-container\n      [maxHeight]=\"maxOptionsHeight()\"\n      [minHeight]=\"minOptionsHeight()\"\n      [width]=\"optionsWidth() || inputContainer.offsetWidth - 2 + 'px'\"\n    >\n      <fw-menu>\n        @if(loading() && !displayNewOption()) {\n          <fw-menu-item title=\"Searching...\" [disabled]=\"true\"/>\n        } @else if(!loading()) {\n          @for (option of filteredOptions(); track option) {\n            <fw-menu-item\n              (click)=\"handleOptionClick($event)\"\n              (mouseenter)=\"setFocusByValue(option)\"\n              [title]=\"option\"\n              [focused]=\"focusedOption() === option\"\n              [value]=\"option\"\n            />\n          }\n          @empty {\n            @if (!displayNewOption()) {\n              <fw-menu-item title=\"No tag suggestions\" [disabled]=\"true\"/>\n            }\n          }\n        }\n        @if(displayNewOption()) {\n          <fw-menu-item\n            (click)=\"handleOptionClick($event)\"\n            (mouseenter)=\"setFocusByValue(searchValue())\"\n            [title]=\"searchValue()\"\n            [value]=\"searchValue()\"\n            [focused]=\"focusedOption() === searchValue()\">\n            <p class=\"new-tag\">New</p>\n          </fw-menu-item>\n        }\n      </fw-menu>\n    </fw-menu-container>\n</ng-template>\n"]}
|