@byuhbll/components 0.0.8
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/README.md +25 -0
- package/esm2022/byuhbll-components.mjs +5 -0
- package/esm2022/lib/animations/animations.mjs +22 -0
- package/esm2022/lib/directives/hbll-pill-btn/hbll-pill-btn.directive.mjs +121 -0
- package/esm2022/lib/hbll-checkbox/hbll-checkbox.component.mjs +17 -0
- package/esm2022/lib/hbll-header/hbll-header.component.mjs +60 -0
- package/esm2022/lib/hbll-multi-select/hbll-multi-select.component.mjs +114 -0
- package/esm2022/lib/ss-search-bar/advanced-search/advanced-search.component.mjs +270 -0
- package/esm2022/lib/ss-search-bar/constants.mjs +41 -0
- package/esm2022/lib/ss-search-bar/date-range/date-range.component.mjs +71 -0
- package/esm2022/lib/ss-search-bar/pipes/advanced-queries.pipe.mjs +27 -0
- package/esm2022/lib/ss-search-bar/simple-search/simple-search.component.mjs +85 -0
- package/esm2022/lib/ss-search-bar/ss-search-bar.component.mjs +199 -0
- package/esm2022/lib/ss-search-bar/utils.mjs +16 -0
- package/esm2022/public-api.mjs +6 -0
- package/fesm2022/byuhbll-components.mjs +1001 -0
- package/fesm2022/byuhbll-components.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/animations/animations.d.ts +4 -0
- package/lib/directives/hbll-pill-btn/hbll-pill-btn.directive.d.ts +17 -0
- package/lib/hbll-checkbox/hbll-checkbox.component.d.ts +6 -0
- package/lib/hbll-header/hbll-header.component.d.ts +40 -0
- package/lib/hbll-multi-select/hbll-multi-select.component.d.ts +52 -0
- package/lib/ss-search-bar/advanced-search/advanced-search.component.d.ts +122 -0
- package/lib/ss-search-bar/constants.d.ts +40 -0
- package/lib/ss-search-bar/date-range/date-range.component.d.ts +25 -0
- package/lib/ss-search-bar/pipes/advanced-queries.pipe.d.ts +8 -0
- package/lib/ss-search-bar/simple-search/simple-search.component.d.ts +28 -0
- package/lib/ss-search-bar/ss-search-bar.component.d.ts +121 -0
- package/lib/ss-search-bar/utils.d.ts +7 -0
- package/package.json +25 -0
- package/public-api.d.ts +2 -0
package/README.md
ADDED
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
# Components
|
|
2
|
+
|
|
3
|
+
This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 18.0.0.
|
|
4
|
+
|
|
5
|
+
## Code scaffolding
|
|
6
|
+
|
|
7
|
+
Run `ng generate component component-name --project components` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module --project components`.
|
|
8
|
+
|
|
9
|
+
> Note: Don't forget to add `--project components` or else it will be added to the default project in your `angular.json` file.
|
|
10
|
+
|
|
11
|
+
## Build
|
|
12
|
+
|
|
13
|
+
Run `ng build components` to build the project. The build artifacts will be stored in the `dist/` directory.
|
|
14
|
+
|
|
15
|
+
## Publishing
|
|
16
|
+
|
|
17
|
+
After building your library with `ng build components`, go to the dist folder `cd dist/components` and run `npm publish`.
|
|
18
|
+
|
|
19
|
+
## Running unit tests
|
|
20
|
+
|
|
21
|
+
Run `ng test components` to execute the unit tests via [Karma](https://karma-runner.github.io).
|
|
22
|
+
|
|
23
|
+
## Further help
|
|
24
|
+
|
|
25
|
+
To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI Overview and Command Reference](https://angular.dev/tools/cli) page.
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './public-api';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnl1aGJsbC1jb21wb25lbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50cy9zcmMvYnl1aGJsbC1jb21wb25lbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vcHVibGljLWFwaSc7XG4iXX0=
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { trigger, transition, animate, style, group, query, animateChild, } from '@angular/animations';
|
|
2
|
+
export const libHbllExpandCollapse = trigger('libHbllExpandCollapse', [
|
|
3
|
+
transition('void <=> *', []),
|
|
4
|
+
transition('* <=> *', [
|
|
5
|
+
group([
|
|
6
|
+
style({ height: '{{startHeight}}px' }),
|
|
7
|
+
query('@*', [animateChild()], { optional: true }),
|
|
8
|
+
animate('.15s ease-in-out'),
|
|
9
|
+
]),
|
|
10
|
+
], { params: { startHeight: '0px' } }),
|
|
11
|
+
]);
|
|
12
|
+
export const libHbllFadeInOut = trigger('libHbllFadeInOut', [
|
|
13
|
+
transition(':enter', [style({ opacity: '0' }), animate('.15s ease-out')]),
|
|
14
|
+
transition(':leave', [animate('.15s ease-out', style({ opacity: '0' }))]),
|
|
15
|
+
]);
|
|
16
|
+
export const libHbllFadeIn = trigger('libHbllFadeIn', [
|
|
17
|
+
transition(':enter', [style({ opacity: '0' }), animate('.15s ease-out')]),
|
|
18
|
+
]);
|
|
19
|
+
export const libHbllFadeOut = trigger('libHbllFadeOut', [
|
|
20
|
+
transition(':leave', [animate('.15s ease-out', style({ opacity: '0' }))]),
|
|
21
|
+
]);
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW5pbWF0aW9ucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9hbmltYXRpb25zL2FuaW1hdGlvbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNILE9BQU8sRUFDUCxVQUFVLEVBQ1YsT0FBTyxFQUNQLEtBQUssRUFDTCxLQUFLLEVBQ0wsS0FBSyxFQUNMLFlBQVksR0FDZixNQUFNLHFCQUFxQixDQUFDO0FBRTdCLE1BQU0sQ0FBQyxNQUFNLHFCQUFxQixHQUFHLE9BQU8sQ0FBQyx1QkFBdUIsRUFBRTtJQUNsRSxVQUFVLENBQUMsWUFBWSxFQUFFLEVBQUUsQ0FBQztJQUM1QixVQUFVLENBQ04sU0FBUyxFQUNUO1FBQ0ksS0FBSyxDQUFDO1lBQ0YsS0FBSyxDQUFDLEVBQUUsTUFBTSxFQUFFLG1CQUFtQixFQUFFLENBQUM7WUFDdEMsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDLFlBQVksRUFBRSxDQUFDLEVBQUUsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLENBQUM7WUFDakQsT0FBTyxDQUFDLGtCQUFrQixDQUFDO1NBQzlCLENBQUM7S0FDTCxFQUNELEVBQUUsTUFBTSxFQUFFLEVBQUUsV0FBVyxFQUFFLEtBQUssRUFBRSxFQUFFLENBQ3JDO0NBQ0osQ0FBQyxDQUFDO0FBRUgsTUFBTSxDQUFDLE1BQU0sZ0JBQWdCLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFO0lBQ3hELFVBQVUsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxLQUFLLENBQUMsRUFBRSxPQUFPLEVBQUUsR0FBRyxFQUFFLENBQUMsRUFBRSxPQUFPLENBQUMsZUFBZSxDQUFDLENBQUMsQ0FBQztJQUN6RSxVQUFVLENBQUMsUUFBUSxFQUFFLENBQUMsT0FBTyxDQUFDLGVBQWUsRUFBRSxLQUFLLENBQUMsRUFBRSxPQUFPLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7Q0FDNUUsQ0FBQyxDQUFDO0FBRUgsTUFBTSxDQUFDLE1BQU0sYUFBYSxHQUFHLE9BQU8sQ0FBQyxlQUFlLEVBQUU7SUFDbEQsVUFBVSxDQUFDLFFBQVEsRUFBRSxDQUFDLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxHQUFHLEVBQUUsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxlQUFlLENBQUMsQ0FBQyxDQUFDO0NBQzVFLENBQUMsQ0FBQztBQUVILE1BQU0sQ0FBQyxNQUFNLGNBQWMsR0FBRyxPQUFPLENBQUMsZ0JBQWdCLEVBQUU7SUFDcEQsVUFBVSxDQUFDLFFBQVEsRUFBRSxDQUFDLE9BQU8sQ0FBQyxlQUFlLEVBQUUsS0FBSyxDQUFDLEVBQUUsT0FBTyxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO0NBQzVFLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gICAgdHJpZ2dlcixcbiAgICB0cmFuc2l0aW9uLFxuICAgIGFuaW1hdGUsXG4gICAgc3R5bGUsXG4gICAgZ3JvdXAsXG4gICAgcXVlcnksXG4gICAgYW5pbWF0ZUNoaWxkLFxufSBmcm9tICdAYW5ndWxhci9hbmltYXRpb25zJztcblxuZXhwb3J0IGNvbnN0IGxpYkhibGxFeHBhbmRDb2xsYXBzZSA9IHRyaWdnZXIoJ2xpYkhibGxFeHBhbmRDb2xsYXBzZScsIFtcbiAgICB0cmFuc2l0aW9uKCd2b2lkIDw9PiAqJywgW10pLFxuICAgIHRyYW5zaXRpb24oXG4gICAgICAgICcqIDw9PiAqJyxcbiAgICAgICAgW1xuICAgICAgICAgICAgZ3JvdXAoW1xuICAgICAgICAgICAgICAgIHN0eWxlKHsgaGVpZ2h0OiAne3tzdGFydEhlaWdodH19cHgnIH0pLFxuICAgICAgICAgICAgICAgIHF1ZXJ5KCdAKicsIFthbmltYXRlQ2hpbGQoKV0sIHsgb3B0aW9uYWw6IHRydWUgfSksXG4gICAgICAgICAgICAgICAgYW5pbWF0ZSgnLjE1cyBlYXNlLWluLW91dCcpLFxuICAgICAgICAgICAgXSksXG4gICAgICAgIF0sXG4gICAgICAgIHsgcGFyYW1zOiB7IHN0YXJ0SGVpZ2h0OiAnMHB4JyB9IH0sXG4gICAgKSxcbl0pO1xuXG5leHBvcnQgY29uc3QgbGliSGJsbEZhZGVJbk91dCA9IHRyaWdnZXIoJ2xpYkhibGxGYWRlSW5PdXQnLCBbXG4gICAgdHJhbnNpdGlvbignOmVudGVyJywgW3N0eWxlKHsgb3BhY2l0eTogJzAnIH0pLCBhbmltYXRlKCcuMTVzIGVhc2Utb3V0JyldKSxcbiAgICB0cmFuc2l0aW9uKCc6bGVhdmUnLCBbYW5pbWF0ZSgnLjE1cyBlYXNlLW91dCcsIHN0eWxlKHsgb3BhY2l0eTogJzAnIH0pKV0pLFxuXSk7XG5cbmV4cG9ydCBjb25zdCBsaWJIYmxsRmFkZUluID0gdHJpZ2dlcignbGliSGJsbEZhZGVJbicsIFtcbiAgICB0cmFuc2l0aW9uKCc6ZW50ZXInLCBbc3R5bGUoeyBvcGFjaXR5OiAnMCcgfSksIGFuaW1hdGUoJy4xNXMgZWFzZS1vdXQnKV0pLFxuXSk7XG5cbmV4cG9ydCBjb25zdCBsaWJIYmxsRmFkZU91dCA9IHRyaWdnZXIoJ2xpYkhibGxGYWRlT3V0JywgW1xuICAgIHRyYW5zaXRpb24oJzpsZWF2ZScsIFthbmltYXRlKCcuMTVzIGVhc2Utb3V0Jywgc3R5bGUoeyBvcGFjaXR5OiAnMCcgfSkpXSksXG5dKTtcbiJdfQ==
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
import { Directive, ElementRef, HostListener, Input, inject, } from '@angular/core';
|
|
2
|
+
import { ANIMATION_LENGTH_STD, DESTRUCTIVE_RED, DESTRUCTIVE_RED_HOVER, LIGHT_GRAY, PRIMARY_BLUE, PRIMARY_BLUE_HOVER, PRIMARY_PURPLE, PRIMARY_PURPLE_HOVER, SECONDARY_TEXT_GRAY, TEXT_SEMIBOLD, } from '../../ss-search-bar/constants';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
const BG_COLORS = {
|
|
5
|
+
blue: PRIMARY_BLUE,
|
|
6
|
+
purple: PRIMARY_PURPLE,
|
|
7
|
+
};
|
|
8
|
+
const BG_COLORS_HOVER = {
|
|
9
|
+
blue: PRIMARY_BLUE_HOVER,
|
|
10
|
+
purple: PRIMARY_PURPLE_HOVER,
|
|
11
|
+
};
|
|
12
|
+
export class HbllPillBtnDirective {
|
|
13
|
+
constructor() {
|
|
14
|
+
this.el = inject(ElementRef);
|
|
15
|
+
this.isDestructiveStyle = false;
|
|
16
|
+
this.isDisabled = false;
|
|
17
|
+
this.isFullWidth = false;
|
|
18
|
+
this.color = 'purple';
|
|
19
|
+
}
|
|
20
|
+
onMouseEnter() {
|
|
21
|
+
if (this.isDisabled) {
|
|
22
|
+
return;
|
|
23
|
+
}
|
|
24
|
+
else {
|
|
25
|
+
if (!this.isDestructiveStyle) {
|
|
26
|
+
this.el.nativeElement.style.backgroundColor = BG_COLORS_HOVER[this.color];
|
|
27
|
+
}
|
|
28
|
+
else {
|
|
29
|
+
this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED_HOVER;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
this.el.nativeElement.style.cursor = 'pointer';
|
|
33
|
+
}
|
|
34
|
+
onMouseLeave() {
|
|
35
|
+
if (this.isDisabled) {
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
else {
|
|
39
|
+
if (!this.isDestructiveStyle) {
|
|
40
|
+
this.el.nativeElement.style.backgroundColor = BG_COLORS[this.color];
|
|
41
|
+
}
|
|
42
|
+
else {
|
|
43
|
+
this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
this.el.nativeElement.style.cursor = 'pointer';
|
|
47
|
+
}
|
|
48
|
+
onClick() {
|
|
49
|
+
if (this.isDisabled) {
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
else {
|
|
53
|
+
if (!this.isDestructiveStyle) {
|
|
54
|
+
this.el.nativeElement.style.backgroundColor = BG_COLORS[this.color];
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
this.el.nativeElement.style.cursor = 'pointer';
|
|
61
|
+
}
|
|
62
|
+
ngOnInit() {
|
|
63
|
+
this.setupButton();
|
|
64
|
+
}
|
|
65
|
+
ngOnChanges() {
|
|
66
|
+
this.setupButton();
|
|
67
|
+
}
|
|
68
|
+
setupButton() {
|
|
69
|
+
if (this.isDisabled) {
|
|
70
|
+
this.el.nativeElement.style.color = SECONDARY_TEXT_GRAY;
|
|
71
|
+
this.el.nativeElement.style.backgroundColor = LIGHT_GRAY;
|
|
72
|
+
}
|
|
73
|
+
else {
|
|
74
|
+
if (!this.isDestructiveStyle) {
|
|
75
|
+
this.el.nativeElement.style.backgroundColor = BG_COLORS[this.color];
|
|
76
|
+
}
|
|
77
|
+
else {
|
|
78
|
+
this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED;
|
|
79
|
+
}
|
|
80
|
+
this.el.nativeElement.style.color = 'white';
|
|
81
|
+
}
|
|
82
|
+
this.el.nativeElement.style.transition = 'all ' + ANIMATION_LENGTH_STD;
|
|
83
|
+
this.el.nativeElement.style.fontSize = '1em';
|
|
84
|
+
this.el.nativeElement.style.fontWeight = TEXT_SEMIBOLD;
|
|
85
|
+
this.el.nativeElement.style.textAlign = 'center';
|
|
86
|
+
this.el.nativeElement.style.display = this.isFullWidth ? 'flex' : 'inline-flex';
|
|
87
|
+
this.el.nativeElement.style.justifyContent = 'center';
|
|
88
|
+
this.el.nativeElement.style.alignItems = 'center';
|
|
89
|
+
this.el.nativeElement.style.width = this.isFullWidth ? '100%' : 'auto';
|
|
90
|
+
this.el.nativeElement.style.borderRadius = '100em';
|
|
91
|
+
this.el.nativeElement.style.padding = '0.4em 1.3em';
|
|
92
|
+
this.el.nativeElement.style.boxSizing = 'border-box';
|
|
93
|
+
}
|
|
94
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllPillBtnDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
95
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.1.0", type: HbllPillBtnDirective, isStandalone: true, selector: "[libHbllPillBtn]", inputs: { isDestructiveStyle: "isDestructiveStyle", isDisabled: "isDisabled", isFullWidth: "isFullWidth", color: "color" }, host: { listeners: { "mouseenter": "onMouseEnter()", "mouseleave": "onMouseLeave()", "click": "onClick()" } }, usesOnChanges: true, ngImport: i0 }); }
|
|
96
|
+
}
|
|
97
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllPillBtnDirective, decorators: [{
|
|
98
|
+
type: Directive,
|
|
99
|
+
args: [{
|
|
100
|
+
selector: '[libHbllPillBtn]',
|
|
101
|
+
standalone: true,
|
|
102
|
+
}]
|
|
103
|
+
}], propDecorators: { isDestructiveStyle: [{
|
|
104
|
+
type: Input
|
|
105
|
+
}], isDisabled: [{
|
|
106
|
+
type: Input
|
|
107
|
+
}], isFullWidth: [{
|
|
108
|
+
type: Input
|
|
109
|
+
}], color: [{
|
|
110
|
+
type: Input
|
|
111
|
+
}], onMouseEnter: [{
|
|
112
|
+
type: HostListener,
|
|
113
|
+
args: ['mouseenter']
|
|
114
|
+
}], onMouseLeave: [{
|
|
115
|
+
type: HostListener,
|
|
116
|
+
args: ['mouseleave']
|
|
117
|
+
}], onClick: [{
|
|
118
|
+
type: HostListener,
|
|
119
|
+
args: ['click']
|
|
120
|
+
}] } });
|
|
121
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hbll-pill-btn.directive.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/directives/hbll-pill-btn/hbll-pill-btn.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Directive,\n    ElementRef,\n    HostListener,\n    Input,\n    inject,\n    type OnChanges,\n    type OnInit,\n} from '@angular/core';\nimport {\n    ANIMATION_LENGTH_STD,\n    DESTRUCTIVE_RED,\n    DESTRUCTIVE_RED_HOVER,\n    LIGHT_GRAY,\n    PRIMARY_BLUE,\n    PRIMARY_BLUE_HOVER,\n    PRIMARY_PURPLE,\n    PRIMARY_PURPLE_HOVER,\n    SECONDARY_TEXT_GRAY,\n    TEXT_SEMIBOLD,\n} from '../../ss-search-bar/constants';\n\nconst BG_COLORS = {\n    blue: PRIMARY_BLUE,\n    purple: PRIMARY_PURPLE,\n};\n\nconst BG_COLORS_HOVER = {\n    blue: PRIMARY_BLUE_HOVER,\n    purple: PRIMARY_PURPLE_HOVER,\n};\n\n@Directive({\n    selector: '[libHbllPillBtn]',\n    standalone: true,\n})\nexport class HbllPillBtnDirective implements OnInit, OnChanges {\n    private readonly el = inject(ElementRef);\n    @Input() isDestructiveStyle = false;\n    @Input() isDisabled = false;\n    @Input() isFullWidth = false;\n    @Input() color: 'blue' | 'purple' = 'purple';\n\n    @HostListener('mouseenter') onMouseEnter() {\n        if (this.isDisabled) {\n            return;\n        } else {\n            if (!this.isDestructiveStyle) {\n                this.el.nativeElement.style.backgroundColor = BG_COLORS_HOVER[this.color];\n            } else {\n                this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED_HOVER;\n            }\n        }\n        this.el.nativeElement.style.cursor = 'pointer';\n    }\n\n    @HostListener('mouseleave') onMouseLeave() {\n        if (this.isDisabled) {\n            return;\n        } else {\n            if (!this.isDestructiveStyle) {\n                this.el.nativeElement.style.backgroundColor = BG_COLORS[this.color];\n            } else {\n                this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED;\n            }\n        }\n        this.el.nativeElement.style.cursor = 'pointer';\n    }\n\n    @HostListener('click') onClick() {\n        if (this.isDisabled) {\n            return;\n        } else {\n            if (!this.isDestructiveStyle) {\n                this.el.nativeElement.style.backgroundColor = BG_COLORS[this.color];\n            } else {\n                this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED;\n            }\n        }\n        this.el.nativeElement.style.cursor = 'pointer';\n    }\n\n    ngOnInit() {\n        this.setupButton();\n    }\n\n    ngOnChanges() {\n        this.setupButton();\n    }\n\n    setupButton() {\n        if (this.isDisabled) {\n            this.el.nativeElement.style.color = SECONDARY_TEXT_GRAY;\n            this.el.nativeElement.style.backgroundColor = LIGHT_GRAY;\n        } else {\n            if (!this.isDestructiveStyle) {\n                this.el.nativeElement.style.backgroundColor = BG_COLORS[this.color];\n            } else {\n                this.el.nativeElement.style.backgroundColor = DESTRUCTIVE_RED;\n            }\n            this.el.nativeElement.style.color = 'white';\n        }\n        this.el.nativeElement.style.transition = 'all ' + ANIMATION_LENGTH_STD;\n        this.el.nativeElement.style.fontSize = '1em';\n        this.el.nativeElement.style.fontWeight = TEXT_SEMIBOLD;\n        this.el.nativeElement.style.textAlign = 'center';\n        this.el.nativeElement.style.display = this.isFullWidth ? 'flex' : 'inline-flex';\n        this.el.nativeElement.style.justifyContent = 'center';\n        this.el.nativeElement.style.alignItems = 'center';\n        this.el.nativeElement.style.width = this.isFullWidth ? '100%' : 'auto';\n        this.el.nativeElement.style.borderRadius = '100em';\n        this.el.nativeElement.style.padding = '0.4em 1.3em';\n        this.el.nativeElement.style.boxSizing = 'border-box';\n    }\n}\n"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, Input } from '@angular/core';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class HbllCheckboxComponent {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.isChecked = false;
|
|
7
|
+
}
|
|
8
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllCheckboxComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
9
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: HbllCheckboxComponent, isStandalone: true, selector: "lib-hbll-checkbox", inputs: { isChecked: "isChecked" }, ngImport: i0, template: "<span class=\"checkbox-container\" [class.checked]=\"isChecked\">\n @if (isChecked) {\n <span class=\"material-symbols-outlined icon\"> check </span>\n }\n</span>\n", styles: [".checkbox-container{transition:.15s;height:1.13em;aspect-ratio:1/1;display:flex;align-items:center;justify-content:center;border-radius:4px;border:solid 1px #707070;color:#fff;box-sizing:border-box;position:relative}.checkbox-container.checked{border-color:#3a6093;background-color:#3a6093}.checkbox-container.checked .icon{position:absolute;font-size:1.1em}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
|
|
10
|
+
}
|
|
11
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllCheckboxComponent, decorators: [{
|
|
12
|
+
type: Component,
|
|
13
|
+
args: [{ selector: 'lib-hbll-checkbox', standalone: true, imports: [CommonModule], template: "<span class=\"checkbox-container\" [class.checked]=\"isChecked\">\n @if (isChecked) {\n <span class=\"material-symbols-outlined icon\"> check </span>\n }\n</span>\n", styles: [".checkbox-container{transition:.15s;height:1.13em;aspect-ratio:1/1;display:flex;align-items:center;justify-content:center;border-radius:4px;border:solid 1px #707070;color:#fff;box-sizing:border-box;position:relative}.checkbox-container.checked{border-color:#3a6093;background-color:#3a6093}.checkbox-container.checked .icon{position:absolute;font-size:1.1em}\n"] }]
|
|
14
|
+
}], propDecorators: { isChecked: [{
|
|
15
|
+
type: Input
|
|
16
|
+
}] } });
|
|
17
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGJsbC1jaGVja2JveC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzL3NyYy9saWIvaGJsbC1jaGVja2JveC9oYmxsLWNoZWNrYm94LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9oYmxsLWNoZWNrYm94L2hibGwtY2hlY2tib3guY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQVNqRCxNQUFNLE9BQU8scUJBQXFCO0lBUGxDO1FBUWEsY0FBUyxHQUFHLEtBQUssQ0FBQztLQUM5Qjs4R0FGWSxxQkFBcUI7a0dBQXJCLHFCQUFxQixpSENWbEMsbUxBS0EsaWFER2MsWUFBWTs7MkZBRWIscUJBQXFCO2tCQVBqQyxTQUFTOytCQUNJLG1CQUFtQixjQUdqQixJQUFJLFdBQ1AsQ0FBQyxZQUFZLENBQUM7OEJBR2QsU0FBUztzQkFBakIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBDb21wb25lbnQoe1xuICAgIHNlbGVjdG9yOiAnbGliLWhibGwtY2hlY2tib3gnLFxuICAgIHRlbXBsYXRlVXJsOiAnLi9oYmxsLWNoZWNrYm94LmNvbXBvbmVudC5odG1sJyxcbiAgICBzdHlsZVVybHM6IFsnLi9oYmxsLWNoZWNrYm94LmNvbXBvbmVudC5zY3NzJ10sXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcbn0pXG5leHBvcnQgY2xhc3MgSGJsbENoZWNrYm94Q29tcG9uZW50IHtcbiAgICBASW5wdXQoKSBpc0NoZWNrZWQgPSBmYWxzZTtcbn1cbiIsIjxzcGFuIGNsYXNzPVwiY2hlY2tib3gtY29udGFpbmVyXCIgW2NsYXNzLmNoZWNrZWRdPVwiaXNDaGVja2VkXCI+XG4gICAgQGlmIChpc0NoZWNrZWQpIHtcbiAgICAgICAgPHNwYW4gY2xhc3M9XCJtYXRlcmlhbC1zeW1ib2xzLW91dGxpbmVkIGljb25cIj4gY2hlY2sgPC9zcGFuPlxuICAgIH1cbjwvc3Bhbj5cbiJdfQ==
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import { DatePipe, LowerCasePipe } from '@angular/common';
|
|
2
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
|
3
|
+
import { HttpClient } from '@angular/common/http';
|
|
4
|
+
import { Component, EventEmitter, Output, Pipe, Renderer2, computed, inject, input, viewChild, } from '@angular/core';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
const LIBRARY_HOURS_API_URL = 'https://apps.lib.byu.edu/libraryhours/api/hours';
|
|
7
|
+
export class LibraryHoursDatePipe {
|
|
8
|
+
transform(date, time) {
|
|
9
|
+
return new Date(date + 'T' + time);
|
|
10
|
+
}
|
|
11
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: LibraryHoursDatePipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
12
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.1.0", ngImport: i0, type: LibraryHoursDatePipe, isStandalone: true, name: "libraryHoursDate" }); }
|
|
13
|
+
}
|
|
14
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: LibraryHoursDatePipe, decorators: [{
|
|
15
|
+
type: Pipe,
|
|
16
|
+
args: [{
|
|
17
|
+
name: 'libraryHoursDate',
|
|
18
|
+
standalone: true,
|
|
19
|
+
}]
|
|
20
|
+
}] });
|
|
21
|
+
/**
|
|
22
|
+
* Header component built to be exported as a custom element.
|
|
23
|
+
* This component uses icons provided by Google Material.
|
|
24
|
+
* The link to these icons should be included once in the \<head> of your base html.
|
|
25
|
+
*
|
|
26
|
+
* When a non-empty string `name` is provided to this component, the user is considered logged in.
|
|
27
|
+
* Two outputs/events are accessible: `login` and `logout`.
|
|
28
|
+
*/
|
|
29
|
+
export class HbllHeaderComponent {
|
|
30
|
+
constructor() {
|
|
31
|
+
this.r2 = inject(Renderer2);
|
|
32
|
+
this.http = inject(HttpClient);
|
|
33
|
+
// --- API ---
|
|
34
|
+
this.name = input('');
|
|
35
|
+
this.login = new EventEmitter();
|
|
36
|
+
this.logout = new EventEmitter();
|
|
37
|
+
// -----------
|
|
38
|
+
this.accountInfoEl = viewChild('accountInfo');
|
|
39
|
+
this.isLoggedIn = computed(() => !!this.name());
|
|
40
|
+
this.libraryHours = toSignal(this.http.get(LIBRARY_HOURS_API_URL));
|
|
41
|
+
this.showAccountDropdown = false;
|
|
42
|
+
}
|
|
43
|
+
ngAfterViewInit() {
|
|
44
|
+
this.r2.listen('window', 'click', (e) => {
|
|
45
|
+
if (!this.accountInfoEl()?.nativeElement.contains(e.target))
|
|
46
|
+
this.showAccountDropdown = false;
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllHeaderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
50
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: HbllHeaderComponent, isStandalone: true, selector: "lib-hbll-header", inputs: { name: { classPropertyName: "name", publicName: "name", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { login: "login", logout: "logout" }, viewQueries: [{ propertyName: "accountInfoEl", first: true, predicate: ["accountInfo"], descendants: true, isSignal: true }], ngImport: i0, template: "<header role=\"banner\" class=\"wrapper\">\n <h1>\n <a href=\"https://lib.byu.edu/\">\n <img\n src=\"https://media.lib.byu.edu/web-assets/images/1.0.0/byu-library-logo-full.svg\"\n alt=\"BYU Library Logo\"\n />\n </a>\n </h1>\n <div id=\"libraryInfo\" class=\"wrapper\">\n @if (libraryHours()) {\n <div id=\"libraryHours\">\n <div class=\"wrapper\">\n <span class=\"material-symbols-outlined left-icon\"> schedule </span>\n {{\n libraryHours()?.is_closed\n ? \"CLOSED\"\n : \"Today's hours: \" +\n (libraryHours()!.date\n | libraryHoursDate : libraryHours()!.open_time\n | date : \"ha\"\n | lowercase) +\n \" - \" +\n (libraryHours()!.date\n | libraryHoursDate : libraryHours()!.close_time\n | date : \"ha\"\n | lowercase)\n }}\n </div>\n </div>\n }\n <div id=\"accountInfo\" #accountInfo>\n <button\n (click)=\"\n isLoggedIn()\n ? (showAccountDropdown = !showAccountDropdown)\n : login.emit()\n \"\n class=\"wrapper\"\n id=\"accountBtn\"\n >\n <span class=\"material-symbols-outlined left-icon\"> person </span>\n {{ isLoggedIn() ? name : \"Login\" }}\n @if (isLoggedIn()) {\n <span class=\"material-symbols-outlined\"> arrow_drop_down </span>\n }\n </button>\n @if (isLoggedIn() && showAccountDropdown) {\n <div id=\"accountDropdown\">\n <a class=\"item\">My Account</a>\n <a class=\"item\">Preferences</a>\n <button class=\"item\" (click)=\"logout.emit()\">Logout</button>\n </div>\n }\n </div>\n </div>\n</header>\n<nav></nav>\n", styles: [":host{font-family:Source Sans Pro}.wrapper{display:flex;align-items:center}header{background-color:#002e5d;display:flex;align-items:center;padding:.6rem 1rem}h1{height:2rem}#libraryInfo{margin-left:auto}#libraryHours,#accountInfo{color:#fff;margin:.6rem}.left-icon{margin-right:.4rem}img{height:100%}#accountInfo{position:relative}#accountInfo #accountBtn:hover{opacity:.6}#accountInfo #accountDropdown{font-size:1.1rem;text-wrap:nowrap;position:absolute;background-color:#002e5d;top:100%;right:0;padding:1.2rem}#accountInfo #accountDropdown .item{display:block;padding:.4rem 0;width:100%;text-align:left}#accountInfo #accountDropdown .item:first-child{padding-top:0}#accountInfo #accountDropdown .item:last-child{padding-bottom:0}button{background-color:transparent;border:none;cursor:pointer;font-family:inherit;font-size:inherit;color:inherit;padding:0}\n"], dependencies: [{ kind: "pipe", type: DatePipe, name: "date" }, { kind: "pipe", type: LowerCasePipe, name: "lowercase" }, { kind: "pipe", type: LibraryHoursDatePipe, name: "libraryHoursDate" }] }); }
|
|
51
|
+
}
|
|
52
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllHeaderComponent, decorators: [{
|
|
53
|
+
type: Component,
|
|
54
|
+
args: [{ selector: 'lib-hbll-header', standalone: true, imports: [DatePipe, LowerCasePipe, LibraryHoursDatePipe], template: "<header role=\"banner\" class=\"wrapper\">\n <h1>\n <a href=\"https://lib.byu.edu/\">\n <img\n src=\"https://media.lib.byu.edu/web-assets/images/1.0.0/byu-library-logo-full.svg\"\n alt=\"BYU Library Logo\"\n />\n </a>\n </h1>\n <div id=\"libraryInfo\" class=\"wrapper\">\n @if (libraryHours()) {\n <div id=\"libraryHours\">\n <div class=\"wrapper\">\n <span class=\"material-symbols-outlined left-icon\"> schedule </span>\n {{\n libraryHours()?.is_closed\n ? \"CLOSED\"\n : \"Today's hours: \" +\n (libraryHours()!.date\n | libraryHoursDate : libraryHours()!.open_time\n | date : \"ha\"\n | lowercase) +\n \" - \" +\n (libraryHours()!.date\n | libraryHoursDate : libraryHours()!.close_time\n | date : \"ha\"\n | lowercase)\n }}\n </div>\n </div>\n }\n <div id=\"accountInfo\" #accountInfo>\n <button\n (click)=\"\n isLoggedIn()\n ? (showAccountDropdown = !showAccountDropdown)\n : login.emit()\n \"\n class=\"wrapper\"\n id=\"accountBtn\"\n >\n <span class=\"material-symbols-outlined left-icon\"> person </span>\n {{ isLoggedIn() ? name : \"Login\" }}\n @if (isLoggedIn()) {\n <span class=\"material-symbols-outlined\"> arrow_drop_down </span>\n }\n </button>\n @if (isLoggedIn() && showAccountDropdown) {\n <div id=\"accountDropdown\">\n <a class=\"item\">My Account</a>\n <a class=\"item\">Preferences</a>\n <button class=\"item\" (click)=\"logout.emit()\">Logout</button>\n </div>\n }\n </div>\n </div>\n</header>\n<nav></nav>\n", styles: [":host{font-family:Source Sans Pro}.wrapper{display:flex;align-items:center}header{background-color:#002e5d;display:flex;align-items:center;padding:.6rem 1rem}h1{height:2rem}#libraryInfo{margin-left:auto}#libraryHours,#accountInfo{color:#fff;margin:.6rem}.left-icon{margin-right:.4rem}img{height:100%}#accountInfo{position:relative}#accountInfo #accountBtn:hover{opacity:.6}#accountInfo #accountDropdown{font-size:1.1rem;text-wrap:nowrap;position:absolute;background-color:#002e5d;top:100%;right:0;padding:1.2rem}#accountInfo #accountDropdown .item{display:block;padding:.4rem 0;width:100%;text-align:left}#accountInfo #accountDropdown .item:first-child{padding-top:0}#accountInfo #accountDropdown .item:last-child{padding-bottom:0}button{background-color:transparent;border:none;cursor:pointer;font-family:inherit;font-size:inherit;color:inherit;padding:0}\n"] }]
|
|
55
|
+
}], propDecorators: { login: [{
|
|
56
|
+
type: Output
|
|
57
|
+
}], logout: [{
|
|
58
|
+
type: Output
|
|
59
|
+
}] } });
|
|
60
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hbll-header.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/hbll-header/hbll-header.component.ts","../../../../../projects/components/src/lib/hbll-header/hbll-header.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAC1D,OAAO,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AACtD,OAAO,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAClD,OAAO,EAEH,SAAS,EAET,YAAY,EACZ,MAAM,EACN,IAAI,EAEJ,SAAS,EACT,QAAQ,EACR,MAAM,EACN,KAAK,EACL,SAAS,GACZ,MAAM,eAAe,CAAC;;AAEvB,MAAM,qBAAqB,GAAG,iDAAiD,CAAC;AAiBhF,MAAM,OAAO,oBAAoB;IAC7B,SAAS,CAAC,IAAY,EAAE,IAAY;QAChC,OAAO,IAAI,IAAI,CAAC,IAAI,GAAG,GAAG,GAAG,IAAI,CAAC,CAAC;IACvC,CAAC;8GAHQ,oBAAoB;4GAApB,oBAAoB;;2FAApB,oBAAoB;kBAJhC,IAAI;mBAAC;oBACF,IAAI,EAAE,kBAAkB;oBACxB,UAAU,EAAE,IAAI;iBACnB;;AAOD;;;;;;;GAOG;AAQH,MAAM,OAAO,mBAAmB;IAPhC;QAQqB,OAAE,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC;QACvB,SAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAE3C,cAAc;QACJ,SAAI,GAAG,KAAK,CAAS,EAAE,CAAC,CAAC;QACzB,UAAK,GAAG,IAAI,YAAY,EAAQ,CAAC;QACjC,WAAM,GAAG,IAAI,YAAY,EAAQ,CAAC;QAC5C,cAAc;QAEN,kBAAa,GAAG,SAAS,CAAa,aAAa,CAAC,CAAC;QAEnD,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAC3C,iBAAY,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAe,qBAAqB,CAAC,CAAC,CAAC;QAC5E,wBAAmB,GAAG,KAAK,CAAC;KAQzC;IANG,eAAe;QACX,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAQ,EAAE,EAAE;YAC3C,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,EAAE,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC;gBACvD,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;QACzC,CAAC,CAAC,CAAC;IACP,CAAC;8GArBQ,mBAAmB;kGAAnB,mBAAmB,yXCxDhC,myDA0DA,+4BDNc,QAAQ,wCAAE,aAAa,6CAjBxB,oBAAoB;;2FAqBpB,mBAAmB;kBAP/B,SAAS;+BACI,iBAAiB,cACf,IAAI,WACP,CAAC,QAAQ,EAAE,aAAa,EAAE,oBAAoB,CAAC;8BAU9C,KAAK;sBAAd,MAAM;gBACG,MAAM;sBAAf,MAAM","sourcesContent":["import { DatePipe, LowerCasePipe } from '@angular/common';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { HttpClient } from '@angular/common/http';\nimport {\n    AfterViewInit,\n    Component,\n    ElementRef,\n    EventEmitter,\n    Output,\n    Pipe,\n    PipeTransform,\n    Renderer2,\n    computed,\n    inject,\n    input,\n    viewChild,\n} from '@angular/core';\n\nconst LIBRARY_HOURS_API_URL = 'https://apps.lib.byu.edu/libraryhours/api/hours';\n\ninterface LibraryHours {\n    date: string;\n    is_closed: boolean;\n    open_time: string;\n    close_time: string;\n    has_exception: boolean;\n    exception_title: string;\n    exception_message: string;\n    is_currently_open: boolean;\n}\n\n@Pipe({\n    name: 'libraryHoursDate',\n    standalone: true,\n})\nexport class LibraryHoursDatePipe implements PipeTransform {\n    transform(date: string, time: string): Date {\n        return new Date(date + 'T' + time);\n    }\n}\n\n/**\n * Header component built to be exported as a custom element.\n * This component uses icons provided by Google Material.\n * The link to these icons should be included once in the \\<head> of your base html.\n *\n * When a non-empty string `name` is provided to this component, the user is considered logged in.\n * Two outputs/events are accessible: `login` and `logout`.\n */\n@Component({\n    selector: 'lib-hbll-header',\n    standalone: true,\n    imports: [DatePipe, LowerCasePipe, LibraryHoursDatePipe],\n    templateUrl: './hbll-header.component.html',\n    styleUrl: './hbll-header.component.scss',\n})\nexport class HbllHeaderComponent implements AfterViewInit {\n    private readonly r2 = inject(Renderer2);\n    private readonly http = inject(HttpClient);\n\n    // --- API ---\n    protected name = input<string>('');\n    @Output() login = new EventEmitter<void>();\n    @Output() logout = new EventEmitter<void>();\n    // -----------\n\n    private accountInfoEl = viewChild<ElementRef>('accountInfo');\n\n    protected isLoggedIn = computed(() => !!this.name());\n    protected libraryHours = toSignal(this.http.get<LibraryHours>(LIBRARY_HOURS_API_URL));\n    protected showAccountDropdown = false;\n\n    ngAfterViewInit(): void {\n        this.r2.listen('window', 'click', (e: Event) => {\n            if (!this.accountInfoEl()?.nativeElement.contains(e.target))\n                this.showAccountDropdown = false;\n        });\n    }\n}\n","<header role=\"banner\" class=\"wrapper\">\n  <h1>\n    <a href=\"https://lib.byu.edu/\">\n      <img\n        src=\"https://media.lib.byu.edu/web-assets/images/1.0.0/byu-library-logo-full.svg\"\n        alt=\"BYU Library Logo\"\n      />\n    </a>\n  </h1>\n  <div id=\"libraryInfo\" class=\"wrapper\">\n    @if (libraryHours()) {\n    <div id=\"libraryHours\">\n      <div class=\"wrapper\">\n        <span class=\"material-symbols-outlined left-icon\"> schedule </span>\n        {{\n          libraryHours()?.is_closed\n            ? \"CLOSED\"\n            : \"Today's hours: \" +\n              (libraryHours()!.date\n                | libraryHoursDate : libraryHours()!.open_time\n                | date : \"ha\"\n                | lowercase) +\n              \" - \" +\n              (libraryHours()!.date\n                | libraryHoursDate : libraryHours()!.close_time\n                | date : \"ha\"\n                | lowercase)\n        }}\n      </div>\n    </div>\n    }\n    <div id=\"accountInfo\" #accountInfo>\n      <button\n        (click)=\"\n          isLoggedIn()\n            ? (showAccountDropdown = !showAccountDropdown)\n            : login.emit()\n        \"\n        class=\"wrapper\"\n        id=\"accountBtn\"\n      >\n        <span class=\"material-symbols-outlined left-icon\"> person </span>\n        {{ isLoggedIn() ? name : \"Login\" }}\n        @if (isLoggedIn()) {\n        <span class=\"material-symbols-outlined\"> arrow_drop_down </span>\n        }\n      </button>\n      @if (isLoggedIn() && showAccountDropdown) {\n      <div id=\"accountDropdown\">\n        <a class=\"item\">My Account</a>\n        <a class=\"item\">Preferences</a>\n        <button class=\"item\" (click)=\"logout.emit()\">Logout</button>\n      </div>\n      }\n    </div>\n  </div>\n</header>\n<nav></nav>\n"]}
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, ViewChild, ViewEncapsulation, } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { MatAutocompleteModule, } from '@angular/material/autocomplete';
|
|
4
|
+
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
5
|
+
import { MatChipsModule } from '@angular/material/chips';
|
|
6
|
+
import { COMMA, ENTER } from '@angular/cdk/keycodes';
|
|
7
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
8
|
+
import { FormControl, ReactiveFormsModule } from '@angular/forms';
|
|
9
|
+
import { combineLatest } from 'rxjs';
|
|
10
|
+
import { map, startWith } from 'rxjs/operators';
|
|
11
|
+
import * as i0 from "@angular/core";
|
|
12
|
+
import * as i1 from "@angular/common";
|
|
13
|
+
import * as i2 from "@angular/material/chips";
|
|
14
|
+
import * as i3 from "@angular/material/form-field";
|
|
15
|
+
import * as i4 from "@angular/material/autocomplete";
|
|
16
|
+
import * as i5 from "@angular/material/core";
|
|
17
|
+
import * as i6 from "@angular/forms";
|
|
18
|
+
export class HbllMultiSelectComponent {
|
|
19
|
+
constructor() {
|
|
20
|
+
this.allOptions = {};
|
|
21
|
+
this.label = '';
|
|
22
|
+
/**
|
|
23
|
+
* An array that indicates which keys are selected.
|
|
24
|
+
*/
|
|
25
|
+
this.selectedKeys = [];
|
|
26
|
+
/**
|
|
27
|
+
* An EventEmitter that emits an array of keys indicating which options are currently selected.
|
|
28
|
+
*/
|
|
29
|
+
this.selectedKeysChange = new EventEmitter();
|
|
30
|
+
this.inputControl = new FormControl('');
|
|
31
|
+
this.filteredOptions$ = combineLatest([
|
|
32
|
+
this.inputControl.valueChanges.pipe(startWith('')),
|
|
33
|
+
this.selectedKeysChange.asObservable().pipe(startWith([])),
|
|
34
|
+
]).pipe(map(([key]) =>
|
|
35
|
+
// Display filtered options if there is a value, else display all options currently not selected.
|
|
36
|
+
key
|
|
37
|
+
? this.filterOptions(key)
|
|
38
|
+
: Object.keys(this.allOptions).filter((key) => !this.selectedKeys.find((selectedOption) => key === selectedOption))));
|
|
39
|
+
this.separatorKeysCodes = [ENTER, COMMA];
|
|
40
|
+
/**
|
|
41
|
+
* Adds an option to the array of selected keys when a user selects from the autocomplete.
|
|
42
|
+
* @param {MatAutocompleteSelectedEvent} event MatAutocompleteSelectedEvent
|
|
43
|
+
*/
|
|
44
|
+
this.selectOption = (event) => {
|
|
45
|
+
this.addOptionToSelectedOptions(event.option.value);
|
|
46
|
+
};
|
|
47
|
+
/**
|
|
48
|
+
* Filters the options by the supplied key as well as the currently selected key.
|
|
49
|
+
* @param {string} key key to filter by.
|
|
50
|
+
* @returns {string[]} the filtered keys
|
|
51
|
+
*/
|
|
52
|
+
this.filterOptions = (newKey) => {
|
|
53
|
+
return Object.keys(this.allOptions).filter((key) => this.allOptions[key] !== newKey && !this.selectedKeys.includes(key));
|
|
54
|
+
};
|
|
55
|
+
/**
|
|
56
|
+
* Adds a key to the selected keys array if the option is truthy and not already in the array.
|
|
57
|
+
* The input is also cleared (if available), and the new selected keys are emitted.
|
|
58
|
+
* @param {key} option key to add
|
|
59
|
+
*/
|
|
60
|
+
this.addOptionToSelectedOptions = (key) => {
|
|
61
|
+
if (key && !this.selectedKeys.includes(key) && this.allOptions[key]) {
|
|
62
|
+
this.selectedKeys.push(key);
|
|
63
|
+
this.inputControl.setValue('');
|
|
64
|
+
this.selectedKeysChange.emit(this.selectedKeys);
|
|
65
|
+
if (this.inputRef)
|
|
66
|
+
this.inputRef.nativeElement.value = '';
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Adds a key from the input to the array of selected key.
|
|
72
|
+
* The value from the input must match a key from the record of all options.
|
|
73
|
+
* @param {MatChipInputEvent} event MatChipInputEvent
|
|
74
|
+
*/
|
|
75
|
+
addOption(event) {
|
|
76
|
+
this.addOptionToSelectedOptions(event.value);
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Removes a key from the array of selected options and emits the new selected keys.
|
|
80
|
+
* @param {MultiSelectKeyValPair} key key to remove
|
|
81
|
+
*/
|
|
82
|
+
removeOption(key) {
|
|
83
|
+
const index = this.selectedKeys.indexOf(key);
|
|
84
|
+
if (index >= 0) {
|
|
85
|
+
this.selectedKeys.splice(index, 1);
|
|
86
|
+
this.selectedKeysChange.emit(this.selectedKeys);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllMultiSelectComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
90
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: HbllMultiSelectComponent, isStandalone: true, selector: "lib-hbll-multi-select", inputs: { allOptions: "allOptions", label: "label", selectedKeys: "selectedKeys" }, outputs: { selectedKeysChange: "selectedKeysChange" }, viewQueries: [{ propertyName: "inputRef", first: true, predicate: ["input"], descendants: true }], ngImport: i0, template: "<div class=\"hbll-multi-select\">\n <mat-form-field appearance=\"outline\">\n <mat-chip-grid #chipGrid [attr.aria-label]=\"label + ' selection'\">\n @for (key of selectedKeys; track key) {\n <mat-chip-row (removed)=\"removeOption(key)\" data-testid=\"matChipRow\">\n {{ allOptions[key] }}\n <button\n matChipRemove\n [attr.aria-label]=\"'remove ' + allOptions[key]\"\n [attr.data-testid]=\"'remove' + key\"\n >\n <span class=\"material-symbols-outlined icon\"> cancel </span>\n </button>\n </mat-chip-row>\n }\n </mat-chip-grid>\n <label class=\"hidden\" for=\"input\">{{ label }}</label>\n <input\n [placeholder]=\"label | titlecase\"\n #input\n id=\"input\"\n [formControl]=\"inputControl\"\n [matChipInputFor]=\"chipGrid\"\n [matAutocomplete]=\"auto\"\n [matChipInputSeparatorKeyCodes]=\"separatorKeysCodes\"\n (matChipInputTokenEnd)=\"addOption($event)\"\n data-testid=\"input\"\n />\n <mat-autocomplete\n #auto=\"matAutocomplete\"\n (optionSelected)=\"selectOption($event)\"\n data-testid=\"autocomplete\"\n >\n @for (key of filteredOptions$ | async; track key) {\n <mat-option [value]=\"key\" data-testid=\"autocompleteOption\">\n {{ allOptions[key] }}\n </mat-option>\n }\n </mat-autocomplete>\n </mat-form-field>\n</div>\n", styles: [".hbll-multi-select mat-form-field{width:100%}.hbll-multi-select mat-form-field .mat-mdc-form-field-infix{padding:.35em 0;min-height:0}.hbll-multi-select mat-form-field .mat-mdc-chip-input{margin-left:0}.hbll-multi-select mat-form-field .mat-mdc-chip-input::placeholder{opacity:.75}.hbll-multi-select mat-form-field .mat-mdc-text-field-wrapper{padding:0 .5em}.hbll-multi-select .icon{font-size:1em}.hbll-multi-select mat-autocomplete{font-family:inherit}.hbll-multi-select .hidden{display:none}.mat-mdc-option.mdc-list-item{background-color:#fff;font-family:inherit}.mat-mdc-autocomplete-panel{background-color:#fff!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i1.TitleCasePipe, name: "titlecase" }, { kind: "ngmodule", type: MatChipsModule }, { kind: "component", type: i2.MatChipGrid, selector: "mat-chip-grid", inputs: ["disabled", "placeholder", "required", "value", "errorStateMatcher"], outputs: ["change", "valueChange"] }, { kind: "directive", type: i2.MatChipInput, selector: "input[matChipInputFor]", inputs: ["matChipInputFor", "matChipInputAddOnBlur", "matChipInputSeparatorKeyCodes", "placeholder", "id", "disabled"], outputs: ["matChipInputTokenEnd"], exportAs: ["matChipInput", "matChipInputFor"] }, { kind: "directive", type: i2.MatChipRemove, selector: "[matChipRemove]" }, { kind: "component", type: i2.MatChipRow, selector: "mat-chip-row, [mat-chip-row], mat-basic-chip-row, [mat-basic-chip-row]", inputs: ["editable"], outputs: ["edited"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i4.MatAutocomplete, selector: "mat-autocomplete", inputs: ["aria-label", "aria-labelledby", "displayWith", "autoActiveFirstOption", "autoSelectActiveOption", "requireSelection", "panelWidth", "disableRipple", "class", "hideSingleSelectionIndicator"], outputs: ["optionSelected", "opened", "closed", "optionActivated"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i5.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "directive", type: i4.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", inputs: ["matAutocomplete", "matAutocompletePosition", "matAutocompleteConnectedTo", "autocomplete", "matAutocompleteDisabled"], exportAs: ["matAutocompleteTrigger"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { 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.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatIconModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
91
|
+
}
|
|
92
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllMultiSelectComponent, decorators: [{
|
|
93
|
+
type: Component,
|
|
94
|
+
args: [{ selector: 'lib-hbll-multi-select', standalone: true, imports: [
|
|
95
|
+
CommonModule,
|
|
96
|
+
MatChipsModule,
|
|
97
|
+
MatFormFieldModule,
|
|
98
|
+
MatAutocompleteModule,
|
|
99
|
+
ReactiveFormsModule,
|
|
100
|
+
MatIconModule,
|
|
101
|
+
], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"hbll-multi-select\">\n <mat-form-field appearance=\"outline\">\n <mat-chip-grid #chipGrid [attr.aria-label]=\"label + ' selection'\">\n @for (key of selectedKeys; track key) {\n <mat-chip-row (removed)=\"removeOption(key)\" data-testid=\"matChipRow\">\n {{ allOptions[key] }}\n <button\n matChipRemove\n [attr.aria-label]=\"'remove ' + allOptions[key]\"\n [attr.data-testid]=\"'remove' + key\"\n >\n <span class=\"material-symbols-outlined icon\"> cancel </span>\n </button>\n </mat-chip-row>\n }\n </mat-chip-grid>\n <label class=\"hidden\" for=\"input\">{{ label }}</label>\n <input\n [placeholder]=\"label | titlecase\"\n #input\n id=\"input\"\n [formControl]=\"inputControl\"\n [matChipInputFor]=\"chipGrid\"\n [matAutocomplete]=\"auto\"\n [matChipInputSeparatorKeyCodes]=\"separatorKeysCodes\"\n (matChipInputTokenEnd)=\"addOption($event)\"\n data-testid=\"input\"\n />\n <mat-autocomplete\n #auto=\"matAutocomplete\"\n (optionSelected)=\"selectOption($event)\"\n data-testid=\"autocomplete\"\n >\n @for (key of filteredOptions$ | async; track key) {\n <mat-option [value]=\"key\" data-testid=\"autocompleteOption\">\n {{ allOptions[key] }}\n </mat-option>\n }\n </mat-autocomplete>\n </mat-form-field>\n</div>\n", styles: [".hbll-multi-select mat-form-field{width:100%}.hbll-multi-select mat-form-field .mat-mdc-form-field-infix{padding:.35em 0;min-height:0}.hbll-multi-select mat-form-field .mat-mdc-chip-input{margin-left:0}.hbll-multi-select mat-form-field .mat-mdc-chip-input::placeholder{opacity:.75}.hbll-multi-select mat-form-field .mat-mdc-text-field-wrapper{padding:0 .5em}.hbll-multi-select .icon{font-size:1em}.hbll-multi-select mat-autocomplete{font-family:inherit}.hbll-multi-select .hidden{display:none}.mat-mdc-option.mdc-list-item{background-color:#fff;font-family:inherit}.mat-mdc-autocomplete-panel{background-color:#fff!important}\n"] }]
|
|
102
|
+
}], propDecorators: { inputRef: [{
|
|
103
|
+
type: ViewChild,
|
|
104
|
+
args: ['input']
|
|
105
|
+
}], allOptions: [{
|
|
106
|
+
type: Input
|
|
107
|
+
}], label: [{
|
|
108
|
+
type: Input
|
|
109
|
+
}], selectedKeys: [{
|
|
110
|
+
type: Input
|
|
111
|
+
}], selectedKeysChange: [{
|
|
112
|
+
type: Output
|
|
113
|
+
}] } });
|
|
114
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hbll-multi-select.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/hbll-multi-select/hbll-multi-select.component.ts","../../../../../projects/components/src/lib/hbll-multi-select/hbll-multi-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    Output,\n    ViewChild,\n    ViewEncapsulation,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport {\n    MatAutocompleteModule,\n    MatAutocompleteSelectedEvent,\n} from '@angular/material/autocomplete';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatChipInputEvent, MatChipsModule } from '@angular/material/chips';\nimport { COMMA, ENTER } from '@angular/cdk/keycodes';\nimport { MatIconModule } from '@angular/material/icon';\nimport { FormControl, ReactiveFormsModule } from '@angular/forms';\nimport { Observable, combineLatest } from 'rxjs';\nimport { map, startWith } from 'rxjs/operators';\n\n@Component({\n    selector: 'lib-hbll-multi-select',\n    standalone: true,\n    imports: [\n        CommonModule,\n        MatChipsModule,\n        MatFormFieldModule,\n        MatAutocompleteModule,\n        ReactiveFormsModule,\n        MatIconModule,\n    ],\n    // Necessary to override material design styles.\n    // IMPORTANT: Tightly scope classes and ids to this component if they are necessary.\n    encapsulation: ViewEncapsulation.None,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    templateUrl: './hbll-multi-select.component.html',\n    styleUrls: ['./hbll-multi-select.component.scss'],\n})\nexport class HbllMultiSelectComponent {\n    @ViewChild('input') private inputRef!: ElementRef<HTMLInputElement>;\n    @Input() allOptions: Record<string, string> = {};\n    @Input() label = '';\n    /**\n     * An array that indicates which keys are selected.\n     */\n    @Input() selectedKeys: string[] = [];\n    /**\n     * An EventEmitter that emits an array of keys indicating which options are currently selected.\n     */\n    @Output() selectedKeysChange = new EventEmitter<string[]>();\n\n    protected inputControl = new FormControl('');\n    protected filteredOptions$: Observable<string[]> = combineLatest([\n        this.inputControl.valueChanges.pipe(startWith('')),\n        this.selectedKeysChange.asObservable().pipe(startWith([])),\n    ]).pipe(\n        map(([key]: [string | null, unknown]) =>\n            // Display filtered options if there is a value, else display all options currently not selected.\n            key\n                ? this.filterOptions(key)\n                : Object.keys(this.allOptions).filter(\n                      (key) => !this.selectedKeys.find((selectedOption) => key === selectedOption),\n                  ),\n        ),\n    );\n    protected separatorKeysCodes: number[] = [ENTER, COMMA];\n\n    /**\n     * Adds a key from the input to the array of selected key.\n     * The value from the input must match a key from the record of all options.\n     * @param {MatChipInputEvent} event MatChipInputEvent\n     */\n    protected addOption(event: MatChipInputEvent): void {\n        this.addOptionToSelectedOptions(event.value);\n    }\n\n    /**\n     * Removes a key from the array of selected options and emits the new selected keys.\n     * @param {MultiSelectKeyValPair} key key to remove\n     */\n    protected removeOption(key: string): void {\n        const index = this.selectedKeys.indexOf(key);\n        if (index >= 0) {\n            this.selectedKeys.splice(index, 1);\n            this.selectedKeysChange.emit(this.selectedKeys);\n        }\n    }\n\n    /**\n     * Adds an option to the array of selected keys when a user selects from the autocomplete.\n     * @param {MatAutocompleteSelectedEvent} event MatAutocompleteSelectedEvent\n     */\n    protected selectOption = (event: MatAutocompleteSelectedEvent): void => {\n        this.addOptionToSelectedOptions(event.option.value);\n    };\n\n    /**\n     * Filters the options by the supplied key as well as the currently selected key.\n     * @param {string} key key to filter by.\n     * @returns {string[]} the filtered keys\n     */\n    private filterOptions = (newKey: string): string[] => {\n        return Object.keys(this.allOptions).filter(\n            (key: string) => this.allOptions[key] !== newKey && !this.selectedKeys.includes(key),\n        );\n    };\n\n    /**\n     * Adds a key to the selected keys array if the option is truthy and not already in the array.\n     * The input is also cleared (if available), and the new selected keys are emitted.\n     * @param {key} option key to add\n     */\n    private addOptionToSelectedOptions = (key: string): void => {\n        if (key && !this.selectedKeys.includes(key) && this.allOptions[key]) {\n            this.selectedKeys.push(key);\n            this.inputControl.setValue('');\n            this.selectedKeysChange.emit(this.selectedKeys);\n            if (this.inputRef) this.inputRef.nativeElement.value = '';\n        }\n    };\n}\n","<div class=\"hbll-multi-select\">\n    <mat-form-field appearance=\"outline\">\n        <mat-chip-grid #chipGrid [attr.aria-label]=\"label + ' selection'\">\n            @for (key of selectedKeys; track key) {\n                <mat-chip-row (removed)=\"removeOption(key)\" data-testid=\"matChipRow\">\n                    {{ allOptions[key] }}\n                    <button\n                        matChipRemove\n                        [attr.aria-label]=\"'remove ' + allOptions[key]\"\n                        [attr.data-testid]=\"'remove' + key\"\n                    >\n                        <span class=\"material-symbols-outlined icon\"> cancel </span>\n                    </button>\n                </mat-chip-row>\n            }\n        </mat-chip-grid>\n        <label class=\"hidden\" for=\"input\">{{ label }}</label>\n        <input\n            [placeholder]=\"label | titlecase\"\n            #input\n            id=\"input\"\n            [formControl]=\"inputControl\"\n            [matChipInputFor]=\"chipGrid\"\n            [matAutocomplete]=\"auto\"\n            [matChipInputSeparatorKeyCodes]=\"separatorKeysCodes\"\n            (matChipInputTokenEnd)=\"addOption($event)\"\n            data-testid=\"input\"\n        />\n        <mat-autocomplete\n            #auto=\"matAutocomplete\"\n            (optionSelected)=\"selectOption($event)\"\n            data-testid=\"autocomplete\"\n        >\n            @for (key of filteredOptions$ | async; track key) {\n                <mat-option [value]=\"key\" data-testid=\"autocompleteOption\">\n                    {{ allOptions[key] }}\n                </mat-option>\n            }\n        </mat-autocomplete>\n    </mat-form-field>\n</div>\n"]}
|