@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.
Files changed (32) hide show
  1. package/README.md +25 -0
  2. package/esm2022/byuhbll-components.mjs +5 -0
  3. package/esm2022/lib/animations/animations.mjs +22 -0
  4. package/esm2022/lib/directives/hbll-pill-btn/hbll-pill-btn.directive.mjs +121 -0
  5. package/esm2022/lib/hbll-checkbox/hbll-checkbox.component.mjs +17 -0
  6. package/esm2022/lib/hbll-header/hbll-header.component.mjs +60 -0
  7. package/esm2022/lib/hbll-multi-select/hbll-multi-select.component.mjs +114 -0
  8. package/esm2022/lib/ss-search-bar/advanced-search/advanced-search.component.mjs +270 -0
  9. package/esm2022/lib/ss-search-bar/constants.mjs +41 -0
  10. package/esm2022/lib/ss-search-bar/date-range/date-range.component.mjs +71 -0
  11. package/esm2022/lib/ss-search-bar/pipes/advanced-queries.pipe.mjs +27 -0
  12. package/esm2022/lib/ss-search-bar/simple-search/simple-search.component.mjs +85 -0
  13. package/esm2022/lib/ss-search-bar/ss-search-bar.component.mjs +199 -0
  14. package/esm2022/lib/ss-search-bar/utils.mjs +16 -0
  15. package/esm2022/public-api.mjs +6 -0
  16. package/fesm2022/byuhbll-components.mjs +1001 -0
  17. package/fesm2022/byuhbll-components.mjs.map +1 -0
  18. package/index.d.ts +5 -0
  19. package/lib/animations/animations.d.ts +4 -0
  20. package/lib/directives/hbll-pill-btn/hbll-pill-btn.directive.d.ts +17 -0
  21. package/lib/hbll-checkbox/hbll-checkbox.component.d.ts +6 -0
  22. package/lib/hbll-header/hbll-header.component.d.ts +40 -0
  23. package/lib/hbll-multi-select/hbll-multi-select.component.d.ts +52 -0
  24. package/lib/ss-search-bar/advanced-search/advanced-search.component.d.ts +122 -0
  25. package/lib/ss-search-bar/constants.d.ts +40 -0
  26. package/lib/ss-search-bar/date-range/date-range.component.d.ts +25 -0
  27. package/lib/ss-search-bar/pipes/advanced-queries.pipe.d.ts +8 -0
  28. package/lib/ss-search-bar/simple-search/simple-search.component.d.ts +28 -0
  29. package/lib/ss-search-bar/ss-search-bar.component.d.ts +121 -0
  30. package/lib/ss-search-bar/utils.d.ts +7 -0
  31. package/package.json +25 -0
  32. 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"]}