@byuhbll/components 4.0.0-alpha.3 → 4.0.0-alpha.30
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/esm2022/lib/animations/animations.mjs +19 -1
- package/esm2022/lib/checkbox/checkbox.component.mjs +15 -0
- package/esm2022/lib/contact-utils.mjs +41 -0
- package/esm2022/lib/copy-tooltip/copy-tooltip.component.mjs +49 -0
- package/esm2022/lib/expand-collapse/expand-collapse.component.mjs +31 -0
- package/esm2022/lib/hbll-footer/hbll-footer.component.mjs +81 -0
- package/esm2022/lib/hbll-header/hbll-header.component.mjs +81 -31
- package/esm2022/lib/hbll-header/models/library-hours.mjs +2 -0
- package/esm2022/lib/hbll-header/nav-bar/nav-bar.component.mjs +333 -0
- package/esm2022/lib/hbll-header/nav-bar-dropdown/nav-bar-dropdown.component.mjs +29 -0
- package/esm2022/lib/hbll-header/pipes/library-hours.pipe.mjs +31 -0
- package/esm2022/lib/header-with-impersonation/header-with-impersonation.component.mjs +38 -0
- package/esm2022/lib/impersonate-modal/impersonate-modal.component.mjs +190 -0
- package/esm2022/lib/impersonation-banner/impersonation-banner.component.mjs +128 -0
- package/esm2022/lib/impersonation-banner/models/application-access.mjs +7 -0
- package/esm2022/lib/impersonation-banner/models/person-summary.mjs +15 -0
- package/esm2022/lib/models/token-payload.mjs +2 -0
- package/esm2022/lib/multi-select/multi-select.component.mjs +115 -0
- package/esm2022/lib/pipes/hbll-item-type-icon.pipe.mjs +128 -0
- package/esm2022/lib/ss-search-bar/advanced-search/advanced-search.component.mjs +5 -5
- package/esm2022/lib/ss-search-bar/date-range/date-range.component.mjs +3 -3
- package/esm2022/lib/ss-search-bar/ss-search-bar.component.mjs +3 -3
- package/esm2022/lib/utils.mjs +7 -0
- package/esm2022/public-api.mjs +7 -1
- package/fesm2022/byuhbll-components.mjs +1153 -66
- package/fesm2022/byuhbll-components.mjs.map +1 -1
- package/lib/animations/animations.d.ts +1 -0
- package/lib/{hbll-checkbox/hbll-checkbox.component.d.ts → checkbox/checkbox.component.d.ts} +2 -2
- package/lib/contact-utils.d.ts +19 -0
- package/lib/copy-tooltip/copy-tooltip.component.d.ts +12 -0
- package/lib/expand-collapse/expand-collapse.component.d.ts +10 -0
- package/lib/hbll-footer/hbll-footer.component.d.ts +28 -0
- package/lib/hbll-header/hbll-header.component.d.ts +23 -28
- package/lib/hbll-header/models/library-hours.d.ts +10 -0
- package/lib/hbll-header/nav-bar/nav-bar.component.d.ts +30 -0
- package/lib/hbll-header/nav-bar-dropdown/nav-bar-dropdown.component.d.ts +12 -0
- package/lib/hbll-header/pipes/library-hours.pipe.d.ts +7 -0
- package/lib/header-with-impersonation/header-with-impersonation.component.d.ts +19 -0
- package/lib/impersonate-modal/impersonate-modal.component.d.ts +60 -0
- package/lib/impersonation-banner/impersonation-banner.component.d.ts +31 -0
- package/lib/impersonation-banner/models/application-access.d.ts +15 -0
- package/lib/impersonation-banner/models/person-summary.d.ts +33 -0
- package/lib/models/token-payload.d.ts +3 -0
- package/lib/{hbll-multi-select/hbll-multi-select.component.d.ts → multi-select/multi-select.component.d.ts} +1 -1
- package/lib/pipes/hbll-item-type-icon.pipe.d.ts +17 -0
- package/lib/ss-search-bar/advanced-search/advanced-search.component.d.ts +4 -1
- package/lib/utils.d.ts +5 -0
- package/package.json +4 -1
- package/public-api.d.ts +6 -0
- package/styles/scss/_mixins.scss +2 -2
- package/styles/scss/_vars.scss +7 -0
- package/styles/scss/base.scss +39 -7
- package/styles/scss/cta-btn.scss +2 -2
- package/styles/scss/pill-btn.scss +2 -2
- package/styles/scss/shared.scss +8 -0
- package/styles/scss/spinner.scss +20 -0
- package/esm2022/lib/hbll-checkbox/hbll-checkbox.component.mjs +0 -17
- package/esm2022/lib/hbll-multi-select/hbll-multi-select.component.mjs +0 -115
|
@@ -0,0 +1,115 @@
|
|
|
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-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-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":"multi-select.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/multi-select/multi-select.component.ts","../../../../../projects/components/src/lib/multi-select/multi-select.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,uBAAuB,EACvB,SAAS,EAET,YAAY,EACZ,KAAK,EACL,MAAM,EACN,SAAS,EACT,iBAAiB,GACpB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACH,qBAAqB,GAExB,MAAM,gCAAgC,CAAC;AACxC,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAqB,cAAc,EAAE,MAAM,yBAAyB,CAAC;AAC5E,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAClE,OAAO,EAAc,aAAa,EAAE,MAAM,MAAM,CAAC;AACjD,OAAO,EAAE,GAAG,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;;;;;;;;AAoBhD,MAAM,OAAO,wBAAwB;IAlBrC;QAoBa,eAAU,GAA2B,EAAE,CAAC;QACxC,UAAK,GAAG,EAAE,CAAC;QACpB;;WAEG;QACM,iBAAY,GAAa,EAAE,CAAC;QACrC;;WAEG;QACO,uBAAkB,GAAG,IAAI,YAAY,EAAY,CAAC;QAElD,iBAAY,GAAG,IAAI,WAAW,CAAC,EAAE,CAAC,CAAC;QACnC,qBAAgB,GAAyB,aAAa,CAAC;YAC7D,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;YAClD,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;SAC7D,CAAC,CAAC,IAAI,CACH,GAAG,CAAC,CAAC,CAAC,GAAG,CAA2B,EAAE,EAAE;QACpC,iGAAiG;QACjG,GAAG;YACC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC;YACzB,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,CAC/B,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,cAAc,EAAE,EAAE,CAAC,GAAG,KAAK,cAAc,CAAC,CAC/E,CACV,CACJ,CAAC;QACQ,uBAAkB,GAAa,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;QAuBxD;;;WAGG;QACO,iBAAY,GAAG,CAAC,KAAmC,EAAQ,EAAE;YACnE,IAAI,CAAC,0BAA0B,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACxD,CAAC,CAAC;QAEF;;;;WAIG;QACK,kBAAa,GAAG,CAAC,KAAa,EAAY,EAAE;YAChD,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,CACtC,CAAC,GAAW,EAAE,EAAE,CACZ,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;gBAChE,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,GAAG,CAAC,CACvC,CAAC;QACN,CAAC,CAAC;QAEF;;;;WAIG;QACK,+BAA0B,GAAG,CAAC,GAAW,EAAQ,EAAE;YACvD,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;gBAClE,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAC5B,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;gBAC/B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBAChD,IAAI,IAAI,CAAC,QAAQ;oBAAE,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,KAAK,GAAG,EAAE,CAAC;YAC9D,CAAC;QACL,CAAC,CAAC;KACL;IAvDG;;;;OAIG;IACO,SAAS,CAAC,KAAwB;QACxC,IAAI,CAAC,0BAA0B,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACjD,CAAC;IAED;;;OAGG;IACO,YAAY,CAAC,GAAW;QAC9B,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAC7C,IAAI,KAAK,IAAI,CAAC,EAAE,CAAC;YACb,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;YACnC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACpD,CAAC;IACL,CAAC;8GAhDQ,wBAAwB;kGAAxB,wBAAwB,0TCzCrC,qzDA2CA,23BDhBQ,YAAY,gJACZ,cAAc,4uBACd,kBAAkB,yOAClB,qBAAqB,w1BACrB,mBAAmB,ykBACnB,aAAa;;2FASR,wBAAwB;kBAlBpC,SAAS;+BACI,kBAAkB,cAChB,IAAI,WACP;wBACL,YAAY;wBACZ,cAAc;wBACd,kBAAkB;wBAClB,qBAAqB;wBACrB,mBAAmB;wBACnB,aAAa;qBAChB,iBAGc,iBAAiB,CAAC,IAAI,mBACpB,uBAAuB,CAAC,MAAM;8BAKnB,QAAQ;sBAAnC,SAAS;uBAAC,OAAO;gBACT,UAAU;sBAAlB,KAAK;gBACG,KAAK;sBAAb,KAAK;gBAIG,YAAY;sBAApB,KAAK;gBAII,kBAAkB;sBAA3B,MAAM","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-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: './multi-select.component.html',\n    styleUrls: ['./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"]}
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
import { Pipe } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
const typeToIconMap = {
|
|
4
|
+
archive: 'inventory_2',
|
|
5
|
+
article: 'article',
|
|
6
|
+
audio: 'mic',
|
|
7
|
+
audiobook: 'mic',
|
|
8
|
+
'blu-ray': 'album',
|
|
9
|
+
book: 'book_4',
|
|
10
|
+
'book-chapter': 'book_6',
|
|
11
|
+
'book-review': 'reviews',
|
|
12
|
+
broadside: 'image',
|
|
13
|
+
calculator: 'calculate',
|
|
14
|
+
cassette: 'album',
|
|
15
|
+
cd: 'album',
|
|
16
|
+
'cd-rom': 'album',
|
|
17
|
+
clothing: 'album',
|
|
18
|
+
'clothing-pattern': 'album',
|
|
19
|
+
collection: 'library_books',
|
|
20
|
+
conference: 'podium',
|
|
21
|
+
'conference-proceeding': 'podium',
|
|
22
|
+
cuneiform: 'aod_tablet',
|
|
23
|
+
curricula: 'aod_tablet',
|
|
24
|
+
database: 'database',
|
|
25
|
+
dataset: 'dataset',
|
|
26
|
+
diary: 'book_2',
|
|
27
|
+
dissertation: 'school',
|
|
28
|
+
dvd: 'album',
|
|
29
|
+
ebook: 'two_pager',
|
|
30
|
+
ebookreader: 'chrome_reader_mode',
|
|
31
|
+
ejournal: 'two_pager',
|
|
32
|
+
'electronic-resource': 'description',
|
|
33
|
+
equipment: 'service_toolbox',
|
|
34
|
+
filmstrip: 'theaters',
|
|
35
|
+
'finding-aid': 'travel_explore',
|
|
36
|
+
'government-document': 'travel_explore',
|
|
37
|
+
'government-novel': 'book_4',
|
|
38
|
+
guide: 'travel_explore',
|
|
39
|
+
image: 'image',
|
|
40
|
+
index: 'list',
|
|
41
|
+
journal: 'book_2',
|
|
42
|
+
'juv-book': 'book_4',
|
|
43
|
+
kit: 'home_repair_service',
|
|
44
|
+
laserdisc: 'album',
|
|
45
|
+
'legal-document': 'gavel',
|
|
46
|
+
letter: 'mail',
|
|
47
|
+
magazine: 'newsmode',
|
|
48
|
+
manuscript: 'article',
|
|
49
|
+
'manuscript-archive': 'inventory_2',
|
|
50
|
+
map: 'map',
|
|
51
|
+
media: 'play_circle',
|
|
52
|
+
microfiche: 'theaters',
|
|
53
|
+
microfilm: 'theaters',
|
|
54
|
+
microform: 'theaters',
|
|
55
|
+
model: 'modeling',
|
|
56
|
+
movie: 'movie',
|
|
57
|
+
multimedia: 'play_circle',
|
|
58
|
+
'music-recording': 'album',
|
|
59
|
+
'musical-recording': 'play_circle',
|
|
60
|
+
newsletter: 'article',
|
|
61
|
+
newspaper: 'newspaper',
|
|
62
|
+
'newspaper-article': 'newspaper',
|
|
63
|
+
ostracon: 'description',
|
|
64
|
+
other: 'description',
|
|
65
|
+
painting: 'palette',
|
|
66
|
+
palmleaf: 'eco',
|
|
67
|
+
pamphlet: 'article',
|
|
68
|
+
papyrus: 'article',
|
|
69
|
+
performance: 'music_note',
|
|
70
|
+
periodical: 'article',
|
|
71
|
+
photograph: 'image',
|
|
72
|
+
picture: 'image',
|
|
73
|
+
poem: 'article',
|
|
74
|
+
poster: 'image',
|
|
75
|
+
presentation: 'slideshow',
|
|
76
|
+
'primary-source': 'inventory_2',
|
|
77
|
+
'primary-source-document': 'article',
|
|
78
|
+
record: 'album',
|
|
79
|
+
'reel-to-reel': 'video_camera_back',
|
|
80
|
+
'reference-entry': 'article',
|
|
81
|
+
report: 'summarize',
|
|
82
|
+
'research-data-set': 'dataset',
|
|
83
|
+
review: 'summarize',
|
|
84
|
+
score: 'music_note',
|
|
85
|
+
script: 'article',
|
|
86
|
+
serial: 'article',
|
|
87
|
+
slide: 'slideshow',
|
|
88
|
+
software: 'terminal',
|
|
89
|
+
'sound-recording': 'mic',
|
|
90
|
+
standards: 'article',
|
|
91
|
+
'statistical-data-set': 'dataset',
|
|
92
|
+
'streaming-audio': 'play_circle',
|
|
93
|
+
'streaming-media': 'play_circle',
|
|
94
|
+
'streaming-video': 'play_circle',
|
|
95
|
+
'student-project': 'article',
|
|
96
|
+
'technical-report': 'summarize',
|
|
97
|
+
'text-resource': 'summarize',
|
|
98
|
+
unknown: 'article',
|
|
99
|
+
vhs: 'theaters',
|
|
100
|
+
video: 'movie',
|
|
101
|
+
website: 'globe',
|
|
102
|
+
};
|
|
103
|
+
/**
|
|
104
|
+
* Converts an item type to an icon type to be used with Google Material Icons
|
|
105
|
+
*
|
|
106
|
+
* Intended to be used with an imported font, such as: `<link
|
|
107
|
+
rel="stylesheet"
|
|
108
|
+
href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@24,400,1,0"
|
|
109
|
+
/>`
|
|
110
|
+
*
|
|
111
|
+
* Reference: https://fonts.google.com/icons?selected=Material+Symbols+Outlined:inventory_2:FILL@0;wght@400;GRAD@0;opsz@24&icon.query=invent&icon.size=24&icon.color=%235f6368
|
|
112
|
+
*/
|
|
113
|
+
export class HbllItemTypeIconPipe {
|
|
114
|
+
transform(itemType) {
|
|
115
|
+
const type = itemType.replace(/_/g, '-').toLowerCase();
|
|
116
|
+
return typeToIconMap[type] ?? 'article';
|
|
117
|
+
}
|
|
118
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllItemTypeIconPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
119
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.1.0", ngImport: i0, type: HbllItemTypeIconPipe, isStandalone: true, name: "libByuItemTypeIcon" }); }
|
|
120
|
+
}
|
|
121
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HbllItemTypeIconPipe, decorators: [{
|
|
122
|
+
type: Pipe,
|
|
123
|
+
args: [{
|
|
124
|
+
name: 'libByuItemTypeIcon',
|
|
125
|
+
standalone: true,
|
|
126
|
+
}]
|
|
127
|
+
}] });
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hbll-item-type-icon.pipe.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/pipes/hbll-item-type-icon.pipe.ts"],"names":[],"mappings":"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","sourcesContent":["import { Pipe, type PipeTransform } from '@angular/core';\n\nconst typeToIconMap = {\n    archive: 'inventory_2',\n    article: 'article',\n    audio: 'mic',\n    audiobook: 'mic',\n    'blu-ray': 'album',\n    book: 'book_4',\n    'book-chapter': 'book_6',\n    'book-review': 'reviews',\n    broadside: 'image',\n    calculator: 'calculate',\n    cassette: 'album',\n    cd: 'album',\n    'cd-rom': 'album',\n    clothing: 'album',\n    'clothing-pattern': 'album',\n    collection: 'library_books',\n    conference: 'podium',\n    'conference-proceeding': 'podium',\n    cuneiform: 'aod_tablet',\n    curricula: 'aod_tablet',\n    database: 'database',\n    dataset: 'dataset',\n    diary: 'book_2',\n    dissertation: 'school',\n    dvd: 'album',\n    ebook: 'two_pager',\n    ebookreader: 'chrome_reader_mode',\n    ejournal: 'two_pager',\n    'electronic-resource': 'description',\n    equipment: 'service_toolbox',\n    filmstrip: 'theaters',\n    'finding-aid': 'travel_explore',\n    'government-document': 'travel_explore',\n    'government-novel': 'book_4',\n    guide: 'travel_explore',\n    image: 'image',\n    index: 'list',\n    journal: 'book_2',\n    'juv-book': 'book_4',\n    kit: 'home_repair_service',\n    laserdisc: 'album',\n    'legal-document': 'gavel',\n    letter: 'mail',\n    magazine: 'newsmode',\n    manuscript: 'article',\n    'manuscript-archive': 'inventory_2',\n    map: 'map',\n    media: 'play_circle',\n    microfiche: 'theaters',\n    microfilm: 'theaters',\n    microform: 'theaters',\n    model: 'modeling',\n    movie: 'movie',\n    multimedia: 'play_circle',\n    'music-recording': 'album',\n    'musical-recording': 'play_circle',\n    newsletter: 'article',\n    newspaper: 'newspaper',\n    'newspaper-article': 'newspaper',\n    ostracon: 'description',\n    other: 'description',\n    painting: 'palette',\n    palmleaf: 'eco',\n    pamphlet: 'article',\n    papyrus: 'article',\n    performance: 'music_note',\n    periodical: 'article',\n    photograph: 'image',\n    picture: 'image',\n    poem: 'article',\n    poster: 'image',\n    presentation: 'slideshow',\n    'primary-source': 'inventory_2',\n    'primary-source-document': 'article',\n    record: 'album',\n    'reel-to-reel': 'video_camera_back',\n    'reference-entry': 'article',\n    report: 'summarize',\n    'research-data-set': 'dataset',\n    review: 'summarize',\n    score: 'music_note',\n    script: 'article',\n    serial: 'article',\n    slide: 'slideshow',\n    software: 'terminal',\n    'sound-recording': 'mic',\n    standards: 'article',\n    'statistical-data-set': 'dataset',\n    'streaming-audio': 'play_circle',\n    'streaming-media': 'play_circle',\n    'streaming-video': 'play_circle',\n    'student-project': 'article',\n    'technical-report': 'summarize',\n    'text-resource': 'summarize',\n    unknown: 'article',\n    vhs: 'theaters',\n    video: 'movie',\n    website: 'globe',\n} as const;\n\n/**\n * Converts an item type to an icon type to be used with Google Material Icons\n *\n * Intended to be used with an imported font, such as: `<link\n            rel=\"stylesheet\"\n            href=\"https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL,GRAD@24,400,1,0\"\n        />`\n *\n * Reference: https://fonts.google.com/icons?selected=Material+Symbols+Outlined:inventory_2:FILL@0;wght@400;GRAD@0;opsz@24&icon.query=invent&icon.size=24&icon.color=%235f6368\n */\n@Pipe({\n    name: 'libByuItemTypeIcon',\n    standalone: true,\n})\nexport class HbllItemTypeIconPipe implements PipeTransform {\n    transform(itemType: string): string {\n        const type = itemType.replace(/_/g, '-').toLowerCase();\n        return typeToIconMap[type as keyof typeof typeToIconMap] ?? 'article';\n    }\n}\n"]}
|