@byuhbll/components 4.0.0-alpha.14 → 4.0.0-alpha.16

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 (30) hide show
  1. package/esm2022/lib/animations/animations.mjs +5 -5
  2. package/esm2022/lib/checkbox/checkbox.component.mjs +15 -0
  3. package/esm2022/lib/expand-collapse/expand-collapse.component.mjs +3 -3
  4. package/esm2022/lib/hbll-header/hbll-header.component.mjs +26 -7
  5. package/esm2022/lib/hbll-header/nav-bar/nav-bar.component.mjs +255 -229
  6. package/esm2022/lib/hbll-header/nav-bar-dropdown/nav-bar-dropdown.component.mjs +3 -3
  7. package/esm2022/lib/header-with-impersonation/header-with-impersonation.component.mjs +38 -0
  8. package/esm2022/lib/impersonate-modal/impersonate-modal.component.mjs +5 -3
  9. package/esm2022/lib/impersonation-banner/impersonation-banner.component.mjs +3 -4
  10. package/esm2022/lib/multi-select/multi-select.component.mjs +115 -0
  11. package/esm2022/lib/ss-search-bar/advanced-search/advanced-search.component.mjs +5 -5
  12. package/esm2022/lib/ss-search-bar/ss-search-bar.component.mjs +2 -2
  13. package/esm2022/lib/utils.mjs +2 -3
  14. package/esm2022/public-api.mjs +2 -1
  15. package/fesm2022/byuhbll-components.mjs +495 -417
  16. package/fesm2022/byuhbll-components.mjs.map +1 -1
  17. package/lib/{hbll-checkbox/hbll-checkbox.component.d.ts → checkbox/checkbox.component.d.ts} +1 -1
  18. package/lib/hbll-header/hbll-header.component.d.ts +7 -3
  19. package/lib/hbll-header/nav-bar/nav-bar.component.d.ts +5 -3
  20. package/lib/header-with-impersonation/header-with-impersonation.component.d.ts +19 -0
  21. package/lib/impersonate-modal/impersonate-modal.component.d.ts +2 -0
  22. package/lib/impersonation-banner/impersonation-banner.component.d.ts +1 -2
  23. package/lib/{hbll-multi-select/hbll-multi-select.component.d.ts → multi-select/multi-select.component.d.ts} +1 -1
  24. package/lib/ss-search-bar/advanced-search/advanced-search.component.d.ts +4 -1
  25. package/package.json +1 -1
  26. package/public-api.d.ts +1 -0
  27. package/styles/scss/_vars.scss +1 -0
  28. package/styles/scss/shared.scss +7 -0
  29. package/esm2022/lib/hbll-checkbox/hbll-checkbox.component.mjs +0 -15
  30. package/esm2022/lib/hbll-multi-select/hbll-multi-select.component.mjs +0 -115
@@ -2,5 +2,5 @@ import * as i0 from "@angular/core";
2
2
  export declare class HbllCheckboxComponent {
3
3
  isChecked: import("@angular/core").InputSignal<boolean>;
4
4
  static ɵfac: i0.ɵɵFactoryDeclaration<HbllCheckboxComponent, never>;
5
- static ɵcmp: i0.ɵɵComponentDeclaration<HbllCheckboxComponent, "lib-hbll-checkbox", never, { "isChecked": { "alias": "isChecked"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
5
+ static ɵcmp: i0.ɵɵComponentDeclaration<HbllCheckboxComponent, "lib-checkbox", never, { "isChecked": { "alias": "isChecked"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
6
6
  }
@@ -6,8 +6,10 @@ export declare class HbllHeaderComponent implements AfterViewInit {
6
6
  private readonly r2;
7
7
  private readonly http;
8
8
  private readonly bo;
9
+ private doc;
9
10
  header: ElementRef;
10
- name: import("@angular/core").InputSignal<string>;
11
+ name: string;
12
+ mainsitebaseurl: string;
11
13
  showImpersonateButton: import("@angular/core").InputSignal<boolean>;
12
14
  openImpersonationModal: EventEmitter<void>;
13
15
  login: EventEmitter<void>;
@@ -15,7 +17,7 @@ export declare class HbllHeaderComponent implements AfterViewInit {
15
17
  private formatDateForHours;
16
18
  private accountInfoEl;
17
19
  private hoursEl;
18
- protected isLoggedIn: import("@angular/core").Signal<boolean>;
20
+ protected get isLoggedIn(): boolean;
19
21
  protected libraryHours: import("@angular/core").Signal<LibraryHours | undefined>;
20
22
  protected hoursExceptions$: import("rxjs").Observable<LibraryHours[]>;
21
23
  protected showAccountDropdown: boolean;
@@ -24,8 +26,10 @@ export declare class HbllHeaderComponent implements AfterViewInit {
24
26
  protected showNavBar: boolean;
25
27
  protected isScreenSmall: import("@angular/core").Signal<boolean | undefined>;
26
28
  ngAfterViewInit(): void;
29
+ protected openSidebarNav: () => void;
30
+ protected closeSidebarNav: () => void;
27
31
  private onResize;
28
32
  private setMobileSidebarHeight;
29
33
  static ɵfac: i0.ɵɵFactoryDeclaration<HbllHeaderComponent, never>;
30
- static ɵcmp: i0.ɵɵComponentDeclaration<HbllHeaderComponent, "lib-hbll-header", never, { "name": { "alias": "name"; "required": false; "isSignal": true; }; "showImpersonateButton": { "alias": "showImpersonateButton"; "required": false; "isSignal": true; }; }, { "openImpersonationModal": "openImpersonationModal"; "login": "login"; "logout": "logout"; }, never, never, true, never>;
34
+ static ɵcmp: i0.ɵɵComponentDeclaration<HbllHeaderComponent, "lib-hbll-header", never, { "name": { "alias": "name"; "required": false; }; "mainsitebaseurl": { "alias": "mainsitebaseurl"; "required": false; }; "showImpersonateButton": { "alias": "showImpersonateButton"; "required": false; "isSignal": true; }; }, { "openImpersonationModal": "openImpersonationModal"; "login": "login"; "logout": "logout"; }, never, never, true, never>;
31
35
  }
@@ -1,3 +1,4 @@
1
+ import { Signal } from '@angular/core';
1
2
  import { NavBarDropdownComponent } from '../nav-bar-dropdown/nav-bar-dropdown.component';
2
3
  import * as i0 from "@angular/core";
3
4
  type NavInfo = {
@@ -17,12 +18,13 @@ type Link = {
17
18
  };
18
19
  export declare class NavBarComponent {
19
20
  private bo;
21
+ mainSiteBaseUrl: import("@angular/core").InputSignal<string>;
20
22
  height: import("@angular/core").InputSignal<number | null>;
21
23
  dropdownComps: NavBarDropdownComponent[];
22
24
  protected handleDropdownOpenEvent: (title: string) => void;
23
- protected isScreenSmall: import("@angular/core").Signal<boolean | undefined>;
24
- protected navInfos: NavInfo[];
25
+ protected isScreenSmall: Signal<boolean | undefined>;
26
+ protected navInfos: Signal<NavInfo[]>;
25
27
  static ɵfac: i0.ɵɵFactoryDeclaration<NavBarComponent, never>;
26
- static ɵcmp: i0.ɵɵComponentDeclaration<NavBarComponent, "lib-nav-bar", never, { "height": { "alias": "height"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
28
+ static ɵcmp: i0.ɵɵComponentDeclaration<NavBarComponent, "lib-nav-bar", never, { "mainSiteBaseUrl": { "alias": "mainSiteBaseUrl"; "required": true; "isSignal": true; }; "height": { "alias": "height"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
27
29
  }
28
30
  export {};
@@ -0,0 +1,19 @@
1
+ import { EventEmitter, Signal } from '@angular/core';
2
+ import { TokenPayload } from '../models/token-payload';
3
+ import { JwtPayload } from 'jwt-decode';
4
+ import * as i0 from "@angular/core";
5
+ export declare class HeaderWithImpersonationComponent {
6
+ accessTokenPayload: import("@angular/core").InputSignal<TokenPayload>;
7
+ oidcBaseUri: import("@angular/core").InputSignal<string>;
8
+ oidcDefaultIdp: import("@angular/core").InputSignal<string>;
9
+ login: EventEmitter<void>;
10
+ logout: EventEmitter<void>;
11
+ endImpersonation: EventEmitter<void>;
12
+ protected parsedToken: Signal<(JwtPayload & Record<string, any>) | null>;
13
+ protected name: Signal<any>;
14
+ protected showImpersonateButton: Signal<boolean>;
15
+ protected showImpersonationModal: boolean;
16
+ private isImpersonating;
17
+ static ɵfac: i0.ɵɵFactoryDeclaration<HeaderWithImpersonationComponent, never>;
18
+ static ɵcmp: i0.ɵɵComponentDeclaration<HeaderWithImpersonationComponent, "lib-header-with-impersonation", never, { "accessTokenPayload": { "alias": "accessTokenPayload"; "required": true; "isSignal": true; }; "oidcBaseUri": { "alias": "oidcBaseUri"; "required": false; "isSignal": true; }; "oidcDefaultIdp": { "alias": "oidcDefaultIdp"; "required": false; "isSignal": true; }; }, { "login": "login"; "logout": "logout"; "endImpersonation": "endImpersonation"; }, never, never, true, never>;
19
+ }
@@ -2,6 +2,8 @@ import { EventEmitter, OnDestroy, PipeTransform } from '@angular/core';
2
2
  import { Subject } from 'rxjs';
3
3
  import { TokenPayload } from '../models/token-payload';
4
4
  import * as i0 from "@angular/core";
5
+ export declare const defaultOidcBaseUri = "https://keycloak.lib.byu.edu/";
6
+ export declare const defaultOidcDefaultIdp = "byu-realm";
5
7
  export declare class ImpersonateUserPipe implements PipeTransform {
6
8
  transform(user: ImpersonateSearchResult): string;
7
9
  static ɵfac: i0.ɵɵFactoryDeclaration<ImpersonateUserPipe, never>;
@@ -1,5 +1,5 @@
1
1
  import { EventEmitter, Signal } from '@angular/core';
2
- import { AccessStatus, type ApplicationAccess } from './models/application-access';
2
+ import { type ApplicationAccess } from './models/application-access';
3
3
  import { TokenPayload } from '../models/token-payload';
4
4
  import { JwtPayload } from 'jwt-decode';
5
5
  import { PersonSummary } from './models/person-summary';
@@ -26,7 +26,6 @@ export declare class ImpersonationBannerComponent {
26
26
  blocked: ApplicationAccess[];
27
27
  none: ApplicationAccess[];
28
28
  } | null | undefined>;
29
- protected STATUSES: typeof AccessStatus;
30
29
  static ɵfac: i0.ɵɵFactoryDeclaration<ImpersonationBannerComponent, never>;
31
30
  static ɵcmp: i0.ɵɵComponentDeclaration<ImpersonationBannerComponent, "lib-impersonation-banner", never, { "accessTokenPayload": { "alias": "accessTokenPayload"; "required": true; "isSignal": true; }; "personBaseUri": { "alias": "personBaseUri"; "required": false; "isSignal": true; }; "libraryApiBaseUri": { "alias": "libraryApiBaseUri"; "required": false; "isSignal": true; }; }, { "endImpersonation": "endImpersonation"; }, never, never, true, never>;
32
31
  }
@@ -48,5 +48,5 @@ export declare class HbllMultiSelectComponent {
48
48
  */
49
49
  private addOptionToSelectedOptions;
50
50
  static ɵfac: i0.ɵɵFactoryDeclaration<HbllMultiSelectComponent, never>;
51
- static ɵcmp: i0.ɵɵComponentDeclaration<HbllMultiSelectComponent, "lib-hbll-multi-select", never, { "allOptions": { "alias": "allOptions"; "required": false; }; "label": { "alias": "label"; "required": false; }; "selectedKeys": { "alias": "selectedKeys"; "required": false; }; }, { "selectedKeysChange": "selectedKeysChange"; }, never, never, true, never>;
51
+ static ɵcmp: i0.ɵɵComponentDeclaration<HbllMultiSelectComponent, "lib-multi-select", never, { "allOptions": { "alias": "allOptions"; "required": false; }; "label": { "alias": "label"; "required": false; }; "selectedKeys": { "alias": "selectedKeys"; "required": false; }; }, { "selectedKeysChange": "selectedKeysChange"; }, never, never, true, never>;
52
52
  }
@@ -105,7 +105,10 @@ export declare class AdvancedSearchComponent implements OnDestroy {
105
105
  readonly 'law -- iclrs': "Law & Religion Studies";
106
106
  readonly 'law -- main': "Main Collection";
107
107
  readonly 'law -- media': "Media";
108
- readonly 'law -- reserve': "Reserve";
108
+ readonly 'law -- reserve': "Reserve"; /**
109
+ * Takes a `FormGroup` and syncs up the `field` and `qualifier` values.
110
+ * After syncing, when the `field` value changes, the `qualifier` value will update appropriately.
111
+ */
109
112
  readonly 'law -- self_help': "Self Help";
110
113
  readonly 'law -- study_guides': "Study Guides";
111
114
  };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@byuhbll/components",
3
- "version": "4.0.0-alpha.14",
3
+ "version": "4.0.0-alpha.16",
4
4
  "peerDependencies": {
5
5
  "@angular/common": "^18.0.0",
6
6
  "@angular/core": "^18.0.0"
package/public-api.d.ts CHANGED
@@ -1,4 +1,5 @@
1
1
  export * from './lib/hbll-header/hbll-header.component';
2
+ export * from './lib/header-with-impersonation/header-with-impersonation.component';
2
3
  export * from './lib/impersonate-modal/impersonate-modal.component';
3
4
  export * from './lib/impersonation-banner/impersonation-banner.component';
4
5
  export * from './lib/ss-search-bar/ss-search-bar.component';
@@ -1,5 +1,6 @@
1
1
  $navy-blue: #002e5d;
2
2
  $royal-blue: #0047ba;
3
+ $royal-blue--hover: #1967e5;
3
4
 
4
5
  $primary-blue: #3a6093;
5
6
 
@@ -0,0 +1,7 @@
1
+ * {
2
+ box-sizing: border-box;
3
+ text-decoration: none;
4
+ &:not(.material-symbols-outlined) {
5
+ font-family: inherit;
6
+ }
7
+ }
@@ -1,15 +0,0 @@
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 = input(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: { classPropertyName: "isChecked", publicName: "isChecked", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<span class=\"components-checkbox-container\" [class.components-checked]=\"isChecked()\">\n @if (isChecked()) {\n <span class=\"material-symbols-outlined components-icon\"> check </span>\n }\n</span>\n", styles: [".components-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}.components-checkbox-container.components-checked{border-color:#3a6093;background-color:#3a6093}.components-checkbox-container.components-checked .components-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=\"components-checkbox-container\" [class.components-checked]=\"isChecked()\">\n @if (isChecked()) {\n <span class=\"material-symbols-outlined components-icon\"> check </span>\n }\n</span>\n", styles: [".components-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}.components-checkbox-container.components-checked{border-color:#3a6093;background-color:#3a6093}.components-checkbox-container.components-checked .components-icon{position:absolute;font-size:1.1em}\n"] }]
14
- }] });
15
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGJsbC1jaGVja2JveC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzL3NyYy9saWIvaGJsbC1jaGVja2JveC9oYmxsLWNoZWNrYm94LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9oYmxsLWNoZWNrYm94L2hibGwtY2hlY2tib3guY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQVNqRCxNQUFNLE9BQU8scUJBQXFCO0lBUGxDO1FBUUksY0FBUyxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQztLQUM1Qjs4R0FGWSxxQkFBcUI7a0dBQXJCLHFCQUFxQiw2TkNWbEMsd05BS0EsbWVER2MsWUFBWTs7MkZBRWIscUJBQXFCO2tCQVBqQyxTQUFTOytCQUNJLG1CQUFtQixjQUdqQixJQUFJLFdBQ1AsQ0FBQyxZQUFZLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQ29tcG9uZW50LCBpbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ2xpYi1oYmxsLWNoZWNrYm94JyxcbiAgICB0ZW1wbGF0ZVVybDogJy4vaGJsbC1jaGVja2JveC5jb21wb25lbnQuaHRtbCcsXG4gICAgc3R5bGVVcmxzOiBbJy4vaGJsbC1jaGVja2JveC5jb21wb25lbnQuc2NzcyddLFxuICAgIHN0YW5kYWxvbmU6IHRydWUsXG4gICAgaW1wb3J0czogW0NvbW1vbk1vZHVsZV0sXG59KVxuZXhwb3J0IGNsYXNzIEhibGxDaGVja2JveENvbXBvbmVudCB7XG4gICAgaXNDaGVja2VkID0gaW5wdXQoZmFsc2UpO1xufVxuIiwiPHNwYW4gY2xhc3M9XCJjb21wb25lbnRzLWNoZWNrYm94LWNvbnRhaW5lclwiIFtjbGFzcy5jb21wb25lbnRzLWNoZWNrZWRdPVwiaXNDaGVja2VkKClcIj5cbiAgICBAaWYgKGlzQ2hlY2tlZCgpKSB7XG4gICAgICAgIDxzcGFuIGNsYXNzPVwibWF0ZXJpYWwtc3ltYm9scy1vdXRsaW5lZCBjb21wb25lbnRzLWljb25cIj4gY2hlY2sgPC9zcGFuPlxuICAgIH1cbjwvc3Bhbj5cbiJdfQ==
@@ -1,115 +0,0 @@
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 value as well as the currently selected values.
49
- * @param {string} value value to filter by.
50
- * @returns {string[]} the filtered keys
51
- */
52
- this.filterOptions = (value) => {
53
- return Object.keys(this.allOptions).filter((key) => this.allOptions[key].toLowerCase().includes(value.toLowerCase()) &&
54
- !this.selectedKeys.includes(key));
55
- };
56
- /**
57
- * Adds a key to the selected keys array if the option is truthy and not already in the array.
58
- * The input is also cleared (if available), and the new selected keys are emitted.
59
- * @param {key} option key to add
60
- */
61
- this.addOptionToSelectedOptions = (key) => {
62
- if (key && !this.selectedKeys.includes(key) && this.allOptions[key]) {
63
- this.selectedKeys.push(key);
64
- this.inputControl.setValue('');
65
- this.selectedKeysChange.emit(this.selectedKeys);
66
- if (this.inputRef)
67
- this.inputRef.nativeElement.value = '';
68
- }
69
- };
70
- }
71
- /**
72
- * Adds a key from the input to the array of selected key.
73
- * The value from the input must match a key from the record of all options.
74
- * @param {MatChipInputEvent} event MatChipInputEvent
75
- */
76
- addOption(event) {
77
- this.addOptionToSelectedOptions(event.value);
78
- }
79
- /**
80
- * Removes a key from the array of selected options and emits the new selected keys.
81
- * @param {MultiSelectKeyValPair} key key to remove
82
- */
83
- removeOption(key) {
84
- const index = this.selectedKeys.indexOf(key);
85
- if (index >= 0) {
86
- this.selectedKeys.splice(index, 1);
87
- this.selectedKeysChange.emit(this.selectedKeys);
88
- }
89
- }
90
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllMultiSelectComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
91
- 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=\"components-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 @if (allOptions[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 components-icon\"> cancel </span>\n </button>\n </mat-chip-row>\n }\n }\n </mat-chip-grid>\n <label class=\"components-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: [".components-multi-select{font:inherit}.components-multi-select mat-form-field{font:inherit;background-color:#fff;width:100%}.components-multi-select mat-form-field .mat-mdc-form-field-infix{padding:.35em 0;min-height:0}.components-multi-select mat-form-field .mat-mdc-chip-input{font:inherit;margin-left:0}.components-multi-select mat-form-field .mat-mdc-chip-input::placeholder{opacity:.75}.components-multi-select mat-form-field .mat-mdc-text-field-wrapper{padding:0 .5em}.components-multi-select .components-icon{font-size:1em}.components-multi-select .mat-mdc-chip.mdc-evolution-chip--with-trailing-action .mat-mdc-chip-action-label{font:inherit}.components-multi-select .components-hidden{display:none}.mat-mdc-option.mdc-list-item{background-color:#fff;font: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 }); }
92
- }
93
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllMultiSelectComponent, decorators: [{
94
- type: Component,
95
- args: [{ selector: 'lib-hbll-multi-select', standalone: true, imports: [
96
- CommonModule,
97
- MatChipsModule,
98
- MatFormFieldModule,
99
- MatAutocompleteModule,
100
- ReactiveFormsModule,
101
- MatIconModule,
102
- ], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"components-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 @if (allOptions[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 components-icon\"> cancel </span>\n </button>\n </mat-chip-row>\n }\n }\n </mat-chip-grid>\n <label class=\"components-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: [".components-multi-select{font:inherit}.components-multi-select mat-form-field{font:inherit;background-color:#fff;width:100%}.components-multi-select mat-form-field .mat-mdc-form-field-infix{padding:.35em 0;min-height:0}.components-multi-select mat-form-field .mat-mdc-chip-input{font:inherit;margin-left:0}.components-multi-select mat-form-field .mat-mdc-chip-input::placeholder{opacity:.75}.components-multi-select mat-form-field .mat-mdc-text-field-wrapper{padding:0 .5em}.components-multi-select .components-icon{font-size:1em}.components-multi-select .mat-mdc-chip.mdc-evolution-chip--with-trailing-action .mat-mdc-chip-action-label{font:inherit}.components-multi-select .components-hidden{display:none}.mat-mdc-option.mdc-list-item{background-color:#fff;font:inherit}.mat-mdc-autocomplete-panel{background-color:#fff!important}\n"] }]
103
- }], propDecorators: { inputRef: [{
104
- type: ViewChild,
105
- args: ['input']
106
- }], allOptions: [{
107
- type: Input
108
- }], label: [{
109
- type: Input
110
- }], selectedKeys: [{
111
- type: Input
112
- }], selectedKeysChange: [{
113
- type: Output
114
- }] } });
115
- //# 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 value as well as the currently selected values.\n     * @param {string} value value to filter by.\n     * @returns {string[]} the filtered keys\n     */\n    private filterOptions = (value: string): string[] => {\n        return Object.keys(this.allOptions).filter(\n            (key: string) =>\n                this.allOptions[key].toLowerCase().includes(value.toLowerCase()) &&\n                !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=\"components-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                @if (allOptions[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 components-icon\"> cancel </span>\n                        </button>\n                    </mat-chip-row>\n                }\n            }\n        </mat-chip-grid>\n        <label class=\"components-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"]}