@byuhbll/components 5.1.0-beta.1 → 5.1.0

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.
@@ -1,4 +1,4 @@
1
- import { Component, computed, EventEmitter, input, Output } from '@angular/core';
1
+ import { Component, computed, EventEmitter, input, Output, effect, runInInjectionContext, inject, Injector, } from '@angular/core';
2
2
  import { HbllHeaderComponent } from '../hbll-header/hbll-header.component';
3
3
  import { ImpersonationBannerComponent } from '../impersonation-banner/impersonation-banner.component';
4
4
  import { defaultOidcBaseUri, defaultOidcDefaultIdp, ImpersonateModalComponent, } from '../impersonate-modal/impersonate-modal.component';
@@ -24,6 +24,93 @@ export class HeaderWithImpersonationComponent {
24
24
  : false);
25
25
  this.showImpersonationModal = false;
26
26
  this.isImpersonating = computed(() => !!this.parsedToken()?.['impersonator']);
27
+ // Inactivity timeout for impersonation
28
+ this.activityEvents = [
29
+ 'keydown',
30
+ 'pointerdown',
31
+ 'wheel',
32
+ 'scroll',
33
+ ];
34
+ this.inactivityTimerId = null;
35
+ this.inactivityTimeoutMs = 5 * 60 * 1000; // 5 minutes
36
+ this.debounceTimerId = null;
37
+ this.debounceDelayMs = 250;
38
+ this.trackingActive = false;
39
+ this.injector = inject(Injector);
40
+ /** Reset the inactivity countdown (no-op if not impersonating) */
41
+ this.resetInactivityTimer = () => {
42
+ if (!this.isImpersonating())
43
+ return;
44
+ if (this.inactivityTimerId)
45
+ clearTimeout(this.inactivityTimerId);
46
+ this.inactivityTimerId = window.setTimeout(() => {
47
+ this.endImpersonation.emit();
48
+ this.stopInactivityTracking();
49
+ }, this.inactivityTimeoutMs);
50
+ };
51
+ /** Debounce activity to avoid hammering resets during event storms */
52
+ this.debouncedResetTimer = () => {
53
+ if (this.debounceTimerId)
54
+ clearTimeout(this.debounceTimerId);
55
+ this.debounceTimerId = window.setTimeout(() => {
56
+ this.resetInactivityTimer();
57
+ }, this.debounceDelayMs);
58
+ };
59
+ }
60
+ ngOnInit() {
61
+ // effect can only be used within an injection context (ex: constructor)
62
+ runInInjectionContext(this.injector, () => {
63
+ effect(() => {
64
+ const impersonating = this.isImpersonating();
65
+ const token = this.parsedToken();
66
+ // when token refreshes after 5 minutes, it leaves a small moment where there is no token
67
+ // this prevents dropping inactivtiy timer during that blip
68
+ if (!token)
69
+ return;
70
+ if (impersonating) {
71
+ if (!this.trackingActive)
72
+ this.startInactivityTracking();
73
+ }
74
+ else {
75
+ if (this.trackingActive)
76
+ this.stopInactivityTracking();
77
+ }
78
+ });
79
+ });
80
+ }
81
+ ngOnDestroy() {
82
+ this.stopInactivityTracking();
83
+ }
84
+ /** Begin listening and start countdown */
85
+ startInactivityTracking() {
86
+ this.activityEvents.forEach((event) => {
87
+ // 'scroll' on document doesn't bubble; use window + capture to catch nested scrolls
88
+ const target = event === 'scroll' ? window : document;
89
+ const options = event === 'scroll'
90
+ ? { passive: true, capture: true }
91
+ : { passive: true };
92
+ target.addEventListener(event, this.debouncedResetTimer, options);
93
+ });
94
+ this.trackingActive = true;
95
+ this.resetInactivityTimer();
96
+ }
97
+ /** Remove listeners and clear timers */
98
+ stopInactivityTracking() {
99
+ this.activityEvents.forEach((event) => {
100
+ const target = event === 'scroll' ? window : document;
101
+ // IMPORTANT: capture must match how it was added for scroll
102
+ const options = event === 'scroll' ? { capture: true } : undefined;
103
+ target.removeEventListener(event, this.debouncedResetTimer, options);
104
+ });
105
+ if (this.inactivityTimerId) {
106
+ clearTimeout(this.inactivityTimerId);
107
+ this.inactivityTimerId = null;
108
+ }
109
+ if (this.debounceTimerId) {
110
+ clearTimeout(this.debounceTimerId);
111
+ this.debounceTimerId = null;
112
+ }
113
+ this.trackingActive = false;
27
114
  }
28
115
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: HeaderWithImpersonationComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
29
116
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.1.0", type: HeaderWithImpersonationComponent, isStandalone: true, selector: "lib-header-with-impersonation", inputs: { accessTokenPayload: { classPropertyName: "accessTokenPayload", publicName: "accessTokenPayload", isSignal: true, isRequired: true, transformFunction: null }, oidcBaseUri: { classPropertyName: "oidcBaseUri", publicName: "oidcBaseUri", isSignal: true, isRequired: false, transformFunction: null }, oidcDefaultIdp: { classPropertyName: "oidcDefaultIdp", publicName: "oidcDefaultIdp", isSignal: true, isRequired: false, transformFunction: null }, mainSiteBaseUrl: { classPropertyName: "mainSiteBaseUrl", publicName: "mainSiteBaseUrl", isSignal: true, isRequired: false, transformFunction: null }, personBaseUri: { classPropertyName: "personBaseUri", publicName: "personBaseUri", isSignal: true, isRequired: false, transformFunction: null }, myAccountApiBaseUri: { classPropertyName: "myAccountApiBaseUri", publicName: "myAccountApiBaseUri", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { login: "login", logout: "logout", endImpersonation: "endImpersonation" }, ngImport: i0, template: "<lib-impersonation-banner\n [accessTokenPayload]=\"accessTokenPayload()\"\n (endImpersonation)=\"endImpersonation.emit()\"\n [personBaseUri]=\"personBaseUri()\"\n [myAccountApiBaseUri]=\"myAccountApiBaseUri()\"\n></lib-impersonation-banner>\n<lib-hbll-header\n [name]=\"name()\"\n [showImpersonateButton]=\"showImpersonateButton()\"\n (openImpersonationModal)=\"showImpersonationModal = true\"\n (login)=\"login.emit()\"\n (logout)=\"logout.emit()\"\n [mainsitebaseurl]=\"mainSiteBaseUrl()\"\n/>\n<lib-impersonate-modal\n [showModal]=\"showImpersonationModal\"\n [oidcBaseUri]=\"oidcBaseUri()\"\n [oidcDefaultIdp]=\"oidcDefaultIdp()\"\n [accessTokenPayload]=\"accessTokenPayload()\"\n (dismiss)=\"showImpersonationModal = false\"\n (init)=\"showImpersonationModal = true\"\n></lib-impersonate-modal>\n", styles: [""], dependencies: [{ kind: "component", type: HbllHeaderComponent, selector: "lib-hbll-header", inputs: ["name", "mainsitebaseurl", "showImpersonateButton"], outputs: ["openImpersonationModal", "login", "logout"] }, { kind: "component", type: ImpersonationBannerComponent, selector: "lib-impersonation-banner", inputs: ["accessTokenPayload", "personBaseUri", "myAccountApiBaseUri"], outputs: ["endImpersonation"] }, { kind: "component", type: ImpersonateModalComponent, selector: "lib-impersonate-modal", inputs: ["showModal", "oidcBaseUri", "oidcDefaultIdp", "accessTokenPayload"], outputs: ["dismiss", "init"] }] }); }
@@ -38,4 +125,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
38
125
  }], endImpersonation: [{
39
126
  type: Output
40
127
  }] } });
41
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVhZGVyLXdpdGgtaW1wZXJzb25hdGlvbi5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzL3NyYy9saWIvaGVhZGVyLXdpdGgtaW1wZXJzb25hdGlvbi9oZWFkZXItd2l0aC1pbXBlcnNvbmF0aW9uLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9oZWFkZXItd2l0aC1pbXBlcnNvbmF0aW9uL2hlYWRlci13aXRoLWltcGVyc29uYXRpb24uY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQVUsTUFBTSxlQUFlLENBQUM7QUFDekYsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFDM0UsT0FBTyxFQUFFLDRCQUE0QixFQUFFLE1BQU0sd0RBQXdELENBQUM7QUFDdEcsT0FBTyxFQUNILGtCQUFrQixFQUNsQixxQkFBcUIsRUFDckIseUJBQXlCLEdBQzVCLE1BQU0sa0RBQWtELENBQUM7QUFFMUQsT0FBTyxFQUFFLFNBQVMsRUFBYyxNQUFNLFlBQVksQ0FBQzs7QUFTbkQsTUFBTSxPQUFPLGdDQUFnQztJQVA3QztRQVFJLHVCQUFrQixHQUFHLEtBQUssQ0FBQyxRQUFRLEVBQWdCLENBQUM7UUFDcEQsZ0JBQVcsR0FBRyxLQUFLLENBQUMsa0JBQWtCLENBQUMsQ0FBQztRQUN4QyxtQkFBYyxHQUFHLEtBQUssQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDO1FBQzlDLG9CQUFlLEdBQUcsS0FBSyxDQUFDLHFCQUFxQixDQUFDLENBQUM7UUFDL0Msa0JBQWEsR0FBRyxLQUFLLENBQUMscUNBQXFDLENBQUMsQ0FBQztRQUM3RCx3QkFBbUIsR0FBRyxLQUFLLENBQUMsNEJBQTRCLENBQUMsQ0FBQztRQUVoRCxVQUFLLEdBQUcsSUFBSSxZQUFZLEVBQVEsQ0FBQztRQUNqQyxXQUFNLEdBQUcsSUFBSSxZQUFZLEVBQVEsQ0FBQztRQUNsQyxxQkFBZ0IsR0FBRyxJQUFJLFlBQVksRUFBUSxDQUFDO1FBRXRELDhEQUE4RDtRQUNwRCxnQkFBVyxHQUFzRCxRQUFRLENBQUMsR0FBRyxFQUFFLENBQ3JGLElBQUksQ0FBQyxrQkFBa0IsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxrQkFBa0IsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQ3RGLENBQUM7UUFDUSxTQUFJLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFHLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDckYsMEJBQXFCLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUM1QyxJQUFJLENBQUMsV0FBVyxFQUFFO1lBQ2QsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUNFLENBQUMsSUFBSSxDQUFDLGVBQWUsRUFBRTtnQkFDdkIsSUFBSSxDQUFDLFdBQVcsRUFBRyxDQUFDLGlCQUFpQixDQUFDLENBQUMsa0JBQWtCLENBQUMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxFQUFFLFFBQVEsQ0FDM0UsZUFBZSxDQUNsQixDQUNKO1lBQ0gsQ0FBQyxDQUFDLEtBQUssQ0FDZCxDQUFDO1FBQ1EsMkJBQXNCLEdBQUcsS0FBSyxDQUFDO1FBQ2pDLG9CQUFlLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLEVBQUUsQ0FBQyxjQUFjLENBQUMsQ0FBQyxDQUFDO0tBQ3BGOzhHQTdCWSxnQ0FBZ0M7a0dBQWhDLGdDQUFnQyw2akNDbEI3QyxtMUJBc0JBLDBERFJjLG1CQUFtQixrTEFBRSw0QkFBNEIsNEtBQUUseUJBQXlCOzsyRkFJN0UsZ0NBQWdDO2tCQVA1QyxTQUFTOytCQUNJLCtCQUErQixjQUM3QixJQUFJLFdBQ1AsQ0FBQyxtQkFBbUIsRUFBRSw0QkFBNEIsRUFBRSx5QkFBeUIsQ0FBQzs4QkFZN0UsS0FBSztzQkFBZCxNQUFNO2dCQUNHLE1BQU07c0JBQWYsTUFBTTtnQkFDRyxnQkFBZ0I7c0JBQXpCLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIGNvbXB1dGVkLCBFdmVudEVtaXR0ZXIsIGlucHV0LCBPdXRwdXQsIFNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSGJsbEhlYWRlckNvbXBvbmVudCB9IGZyb20gJy4uL2hibGwtaGVhZGVyL2hibGwtaGVhZGVyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBJbXBlcnNvbmF0aW9uQmFubmVyQ29tcG9uZW50IH0gZnJvbSAnLi4vaW1wZXJzb25hdGlvbi1iYW5uZXIvaW1wZXJzb25hdGlvbi1iYW5uZXIuY29tcG9uZW50JztcbmltcG9ydCB7XG4gICAgZGVmYXVsdE9pZGNCYXNlVXJpLFxuICAgIGRlZmF1bHRPaWRjRGVmYXVsdElkcCxcbiAgICBJbXBlcnNvbmF0ZU1vZGFsQ29tcG9uZW50LFxufSBmcm9tICcuLi9pbXBlcnNvbmF0ZS1tb2RhbC9pbXBlcnNvbmF0ZS1tb2RhbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVG9rZW5QYXlsb2FkIH0gZnJvbSAnLi4vbW9kZWxzL3Rva2VuLXBheWxvYWQnO1xuaW1wb3J0IHsgand0RGVjb2RlLCBKd3RQYXlsb2FkIH0gZnJvbSAnand0LWRlY29kZSc7XG5cbkBDb21wb25lbnQoe1xuICAgIHNlbGVjdG9yOiAnbGliLWhlYWRlci13aXRoLWltcGVyc29uYXRpb24nLFxuICAgIHN0YW5kYWxvbmU6IHRydWUsXG4gICAgaW1wb3J0czogW0hibGxIZWFkZXJDb21wb25lbnQsIEltcGVyc29uYXRpb25CYW5uZXJDb21wb25lbnQsIEltcGVyc29uYXRlTW9kYWxDb21wb25lbnRdLFxuICAgIHRlbXBsYXRlVXJsOiAnLi9oZWFkZXItd2l0aC1pbXBlcnNvbmF0aW9uLmNvbXBvbmVudC5odG1sJyxcbiAgICBzdHlsZVVybDogJy4vaGVhZGVyLXdpdGgtaW1wZXJzb25hdGlvbi5jb21wb25lbnQuc2NzcycsXG59KVxuZXhwb3J0IGNsYXNzIEhlYWRlcldpdGhJbXBlcnNvbmF0aW9uQ29tcG9uZW50IHtcbiAgICBhY2Nlc3NUb2tlblBheWxvYWQgPSBpbnB1dC5yZXF1aXJlZDxUb2tlblBheWxvYWQ+KCk7XG4gICAgb2lkY0Jhc2VVcmkgPSBpbnB1dChkZWZhdWx0T2lkY0Jhc2VVcmkpO1xuICAgIG9pZGNEZWZhdWx0SWRwID0gaW5wdXQoZGVmYXVsdE9pZGNEZWZhdWx0SWRwKTtcbiAgICBtYWluU2l0ZUJhc2VVcmwgPSBpbnB1dCgnaHR0cHM6Ly9saWIuYnl1LmVkdScpO1xuICAgIHBlcnNvbkJhc2VVcmkgPSBpbnB1dCgnaHR0cHM6Ly9hcHBzLmxpYi5ieXUuZWR1L3BlcnNvbi92Mi8nKTtcbiAgICBteUFjY291bnRBcGlCYXNlVXJpID0gaW5wdXQoJ2h0dHBzOi8vYXBpLmxpYi5ieXUuZWR1L3YxJyk7XG5cbiAgICBAT3V0cHV0KCkgbG9naW4gPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG4gICAgQE91dHB1dCgpIGxvZ291dCA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcbiAgICBAT3V0cHV0KCkgZW5kSW1wZXJzb25hdGlvbiA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcblxuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tZXhwbGljaXQtYW55XG4gICAgcHJvdGVjdGVkIHBhcnNlZFRva2VuOiBTaWduYWw8KEp3dFBheWxvYWQgJiBSZWNvcmQ8c3RyaW5nLCBhbnk+KSB8IG51bGw+ID0gY29tcHV0ZWQoKCkgPT5cbiAgICAgICAgdGhpcy5hY2Nlc3NUb2tlblBheWxvYWQoKS50b2tlbiA/IGp3dERlY29kZSh0aGlzLmFjY2Vzc1Rva2VuUGF5bG9hZCgpLnRva2VuKSA6IG51bGwsXG4gICAgKTtcbiAgICBwcm90ZWN0ZWQgbmFtZSA9IGNvbXB1dGVkKCgpID0+ICh0aGlzLnBhcnNlZFRva2VuKCkgPyB0aGlzLnBhcnNlZFRva2VuKCkhWydnaXZlbl9uYW1lJ10gOiAnJykpO1xuICAgIHByb3RlY3RlZCBzaG93SW1wZXJzb25hdGVCdXR0b24gPSBjb21wdXRlZCgoKSA9PlxuICAgICAgICB0aGlzLnBhcnNlZFRva2VuKClcbiAgICAgICAgICAgID8gISEoXG4gICAgICAgICAgICAgICAgICAhdGhpcy5pc0ltcGVyc29uYXRpbmcoKSAmJlxuICAgICAgICAgICAgICAgICAgdGhpcy5wYXJzZWRUb2tlbigpIVsncmVzb3VyY2VfYWNjZXNzJ11bJ3JlYWxtLW1hbmFnZW1lbnQnXT8uWydyb2xlcyddPy5pbmNsdWRlcyhcbiAgICAgICAgICAgICAgICAgICAgICAnaW1wZXJzb25hdGlvbicsXG4gICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgIClcbiAgICAgICAgICAgIDogZmFsc2UsXG4gICAgKTtcbiAgICBwcm90ZWN0ZWQgc2hvd0ltcGVyc29uYXRpb25Nb2RhbCA9IGZhbHNlO1xuICAgIHByaXZhdGUgaXNJbXBlcnNvbmF0aW5nID0gY29tcHV0ZWQoKCkgPT4gISF0aGlzLnBhcnNlZFRva2VuKCk/LlsnaW1wZXJzb25hdG9yJ10pO1xufVxuIiwiPGxpYi1pbXBlcnNvbmF0aW9uLWJhbm5lclxuICAgIFthY2Nlc3NUb2tlblBheWxvYWRdPVwiYWNjZXNzVG9rZW5QYXlsb2FkKClcIlxuICAgIChlbmRJbXBlcnNvbmF0aW9uKT1cImVuZEltcGVyc29uYXRpb24uZW1pdCgpXCJcbiAgICBbcGVyc29uQmFzZVVyaV09XCJwZXJzb25CYXNlVXJpKClcIlxuICAgIFtteUFjY291bnRBcGlCYXNlVXJpXT1cIm15QWNjb3VudEFwaUJhc2VVcmkoKVwiXG4+PC9saWItaW1wZXJzb25hdGlvbi1iYW5uZXI+XG48bGliLWhibGwtaGVhZGVyXG4gICAgW25hbWVdPVwibmFtZSgpXCJcbiAgICBbc2hvd0ltcGVyc29uYXRlQnV0dG9uXT1cInNob3dJbXBlcnNvbmF0ZUJ1dHRvbigpXCJcbiAgICAob3BlbkltcGVyc29uYXRpb25Nb2RhbCk9XCJzaG93SW1wZXJzb25hdGlvbk1vZGFsID0gdHJ1ZVwiXG4gICAgKGxvZ2luKT1cImxvZ2luLmVtaXQoKVwiXG4gICAgKGxvZ291dCk9XCJsb2dvdXQuZW1pdCgpXCJcbiAgICBbbWFpbnNpdGViYXNldXJsXT1cIm1haW5TaXRlQmFzZVVybCgpXCJcbi8+XG48bGliLWltcGVyc29uYXRlLW1vZGFsXG4gICAgW3Nob3dNb2RhbF09XCJzaG93SW1wZXJzb25hdGlvbk1vZGFsXCJcbiAgICBbb2lkY0Jhc2VVcmldPVwib2lkY0Jhc2VVcmkoKVwiXG4gICAgW29pZGNEZWZhdWx0SWRwXT1cIm9pZGNEZWZhdWx0SWRwKClcIlxuICAgIFthY2Nlc3NUb2tlblBheWxvYWRdPVwiYWNjZXNzVG9rZW5QYXlsb2FkKClcIlxuICAgIChkaXNtaXNzKT1cInNob3dJbXBlcnNvbmF0aW9uTW9kYWwgPSBmYWxzZVwiXG4gICAgKGluaXQpPVwic2hvd0ltcGVyc29uYXRpb25Nb2RhbCA9IHRydWVcIlxuPjwvbGliLWltcGVyc29uYXRlLW1vZGFsPlxuIl19
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header-with-impersonation.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/header-with-impersonation/header-with-impersonation.component.ts","../../../../../projects/components/src/lib/header-with-impersonation/header-with-impersonation.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    computed,\n    EventEmitter,\n    input,\n    Output,\n    Signal,\n    OnInit,\n    OnDestroy,\n    effect,\n    runInInjectionContext,\n    inject,\n    Injector,\n} from '@angular/core';\nimport { HbllHeaderComponent } from '../hbll-header/hbll-header.component';\nimport { ImpersonationBannerComponent } from '../impersonation-banner/impersonation-banner.component';\nimport {\n    defaultOidcBaseUri,\n    defaultOidcDefaultIdp,\n    ImpersonateModalComponent,\n} from '../impersonate-modal/impersonate-modal.component';\nimport { TokenPayload } from '../models/token-payload';\nimport { jwtDecode, JwtPayload } from 'jwt-decode';\n\n@Component({\n    selector: 'lib-header-with-impersonation',\n    standalone: true,\n    imports: [HbllHeaderComponent, ImpersonationBannerComponent, ImpersonateModalComponent],\n    templateUrl: './header-with-impersonation.component.html',\n    styleUrl: './header-with-impersonation.component.scss',\n})\nexport class HeaderWithImpersonationComponent implements OnInit, OnDestroy {\n    accessTokenPayload = input.required<TokenPayload>();\n    oidcBaseUri = input(defaultOidcBaseUri);\n    oidcDefaultIdp = input(defaultOidcDefaultIdp);\n    mainSiteBaseUrl = input('https://lib.byu.edu');\n    personBaseUri = input('https://apps.lib.byu.edu/person/v2/');\n    myAccountApiBaseUri = input('https://api.lib.byu.edu/v1');\n\n    @Output() login = new EventEmitter<void>();\n    @Output() logout = new EventEmitter<void>();\n    @Output() endImpersonation = new EventEmitter<void>();\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    protected parsedToken: Signal<(JwtPayload & Record<string, any>) | null> = computed(() =>\n        this.accessTokenPayload().token ? jwtDecode(this.accessTokenPayload().token) : null,\n    );\n    protected name = computed(() => (this.parsedToken() ? this.parsedToken()!['given_name'] : ''));\n    protected showImpersonateButton = computed(() =>\n        this.parsedToken()\n            ? !!(\n                  !this.isImpersonating() &&\n                  this.parsedToken()!['resource_access']['realm-management']?.['roles']?.includes(\n                      'impersonation',\n                  )\n              )\n            : false,\n    );\n\n    protected showImpersonationModal = false;\n    private isImpersonating = computed(() => !!this.parsedToken()?.['impersonator']);\n\n    // Inactivity timeout for impersonation\n    private activityEvents: Array<'keydown' | 'pointerdown' | 'wheel' | 'scroll'> = [\n        'keydown',\n        'pointerdown',\n        'wheel',\n        'scroll',\n    ];\n    private inactivityTimerId: number | null = null;\n    private inactivityTimeoutMs = 5 * 60 * 1000; // 5 minutes\n    private debounceTimerId: number | null = null;\n    private debounceDelayMs = 250;\n\n    private trackingActive = false;\n\n    private injector = inject(Injector);\n\n    ngOnInit() {\n        // effect can only be used within an injection context (ex: constructor)\n        runInInjectionContext(this.injector, () => {\n            effect(() => {\n                const impersonating = this.isImpersonating();\n                const token = this.parsedToken();\n\n                // when token refreshes after 5 minutes, it leaves a small moment where there is no token\n                // this prevents dropping inactivtiy timer during that blip\n                if (!token) return;\n\n                if (impersonating) {\n                    if (!this.trackingActive) this.startInactivityTracking();\n                } else {\n                    if (this.trackingActive) this.stopInactivityTracking();\n                }\n            });\n        });\n    }\n\n    ngOnDestroy() {\n        this.stopInactivityTracking();\n    }\n\n    /** Begin listening and start countdown */\n    private startInactivityTracking() {\n        this.activityEvents.forEach((event) => {\n            // 'scroll' on document doesn't bubble; use window + capture to catch nested scrolls\n            const target: Window | Document = event === 'scroll' ? window : document;\n            const options =\n                event === 'scroll'\n                    ? ({ passive: true, capture: true } as AddEventListenerOptions)\n                    : ({ passive: true } as AddEventListenerOptions);\n\n            target.addEventListener(event, this.debouncedResetTimer, options);\n        });\n\n        this.trackingActive = true;\n        this.resetInactivityTimer();\n    }\n\n    /** Remove listeners and clear timers */\n    private stopInactivityTracking() {\n        this.activityEvents.forEach((event) => {\n            const target: Window | Document = event === 'scroll' ? window : document;\n            // IMPORTANT: capture must match how it was added for scroll\n            const options =\n                event === 'scroll' ? ({ capture: true } as AddEventListenerOptions) : undefined;\n\n            target.removeEventListener(event, this.debouncedResetTimer, options);\n        });\n\n        if (this.inactivityTimerId) {\n            clearTimeout(this.inactivityTimerId);\n            this.inactivityTimerId = null;\n        }\n        if (this.debounceTimerId) {\n            clearTimeout(this.debounceTimerId);\n            this.debounceTimerId = null;\n        }\n\n        this.trackingActive = false;\n    }\n\n    /** Reset the inactivity countdown (no-op if not impersonating) */\n    private resetInactivityTimer = () => {\n        if (!this.isImpersonating()) return;\n\n        if (this.inactivityTimerId) clearTimeout(this.inactivityTimerId);\n\n        this.inactivityTimerId = window.setTimeout(() => {\n            this.endImpersonation.emit();\n            this.stopInactivityTracking();\n        }, this.inactivityTimeoutMs);\n    };\n\n    /** Debounce activity to avoid hammering resets during event storms */\n    private debouncedResetTimer = () => {\n        if (this.debounceTimerId) clearTimeout(this.debounceTimerId);\n\n        this.debounceTimerId = window.setTimeout(() => {\n            this.resetInactivityTimer();\n        }, this.debounceDelayMs);\n    };\n}\n","<lib-impersonation-banner\n    [accessTokenPayload]=\"accessTokenPayload()\"\n    (endImpersonation)=\"endImpersonation.emit()\"\n    [personBaseUri]=\"personBaseUri()\"\n    [myAccountApiBaseUri]=\"myAccountApiBaseUri()\"\n></lib-impersonation-banner>\n<lib-hbll-header\n    [name]=\"name()\"\n    [showImpersonateButton]=\"showImpersonateButton()\"\n    (openImpersonationModal)=\"showImpersonationModal = true\"\n    (login)=\"login.emit()\"\n    (logout)=\"logout.emit()\"\n    [mainsitebaseurl]=\"mainSiteBaseUrl()\"\n/>\n<lib-impersonate-modal\n    [showModal]=\"showImpersonationModal\"\n    [oidcBaseUri]=\"oidcBaseUri()\"\n    [oidcDefaultIdp]=\"oidcDefaultIdp()\"\n    [accessTokenPayload]=\"accessTokenPayload()\"\n    (dismiss)=\"showImpersonationModal = false\"\n    (init)=\"showImpersonationModal = true\"\n></lib-impersonate-modal>\n"]}
@@ -10,7 +10,7 @@ import { toSignal } from '@angular/core/rxjs-interop';
10
10
  import * as i0 from "@angular/core";
11
11
  import * as i1 from "@angular/forms";
12
12
  export const defaultOidcBaseUri = 'https://keycloak.lib.byu.edu/';
13
- export const defaultOidcDefaultIdp = 'byu-realm';
13
+ export const defaultOidcDefaultIdp = 'ces';
14
14
  export class ImpersonateUserPipe {
15
15
  transform(user) {
16
16
  return `${user.name} (${user.netId || 'Unknown'})${user.restricted ? ' — Restricted' : ''}`;
@@ -187,4 +187,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
187
187
  type: HostListener,
188
188
  args: ['document:keydown', ['$event']]
189
189
  }] } });
190
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impersonate-modal.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/impersonate-modal/impersonate-modal.component.ts","../../../../../projects/components/src/lib/impersonate-modal/impersonate-modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostListener,\n    Input,\n    OnDestroy,\n    Output,\n    Pipe,\n    PipeTransform,\n    inject,\n    input,\n} from '@angular/core';\nimport { FormBuilder, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { Subject, Subscription, of } from 'rxjs';\nimport { catchError, startWith, switchMap, tap } from 'rxjs/operators';\nimport { libHbllFadeInOut } from '../animations/animations';\nimport urlcat from 'urlcat';\nimport { HttpClient } from '@angular/common/http';\nimport { CommonModule } from '@angular/common';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { TokenPayload } from '../models/token-payload';\n\nexport const defaultOidcBaseUri = 'https://keycloak.lib.byu.edu/';\nexport const defaultOidcDefaultIdp = 'byu-realm';\n\n@Pipe({\n    name: 'impersonateUser',\n    standalone: true,\n})\nexport class ImpersonateUserPipe implements PipeTransform {\n    transform(user: ImpersonateSearchResult): string {\n        return `${user.name} (${user.netId || 'Unknown'})${user.restricted ? ' — Restricted' : ''}`;\n    }\n}\nexport interface ImpersonateSearchResult {\n    netId: string;\n    institution: string;\n    username: string;\n    name: string;\n    preferredName: string;\n    givenName: string;\n    surname: string;\n    restricted: boolean;\n}\n\nconst SEARCH_USERS_PATH = '/impersonate/api/search/';\nconst START_IMPERSONATE_PATH = '/impersonate/:username';\n\n@Component({\n    standalone: true,\n    imports: [CommonModule, ReactiveFormsModule, ImpersonateUserPipe],\n    selector: 'lib-impersonate-modal',\n    templateUrl: './impersonate-modal.component.html',\n    styleUrls: ['./impersonate-modal.component.scss'],\n    animations: [libHbllFadeInOut],\n})\nexport class ImpersonateModalComponent implements OnDestroy {\n    private readonly http = inject(HttpClient);\n    private readonly fb = inject(FormBuilder);\n    private readonly eref = inject(ElementRef);\n\n    @Input() set showModal(open: boolean) {\n        this.isOpen = open;\n        if (open) {\n            // Set focus on search input shortly after opening modal so user notices\n            // the input receiving focus.\n            setTimeout(() => this.eref.nativeElement.querySelector('#searchInput')?.focus(), 250);\n        }\n    }\n    oidcBaseUri = input(defaultOidcBaseUri);\n    oidcDefaultIdp = input(defaultOidcDefaultIdp);\n    // Require an object here so that access tokens are not visible/extractable from the DOM in consuming applications. Instead they are only stored in memory.\n    accessTokenPayload = input.required<TokenPayload>();\n    @Output() dismiss = new EventEmitter<void>();\n    @Output() init = new EventEmitter<void>();\n\n    protected isOpen = false;\n    protected hasError = false;\n    protected form = this.fb.nonNullable.group({\n        search: this.fb.control('', { validators: Validators.minLength(3) }),\n    });\n    protected selectedUsername?: string;\n\n    protected loading = false;\n    protected handleSearchSubject = new Subject<boolean>();\n    protected results = toSignal(\n        this.form.controls.search.valueChanges.pipe(\n            switchMap((search) =>\n                this.handleSearchSubject.pipe(\n                    startWith(false),\n                    switchMap((handleSearch) => {\n                        this.selectedUsername = undefined;\n                        this.hasError = false;\n\n                        if (!search || !handleSearch) {\n                            return of(null);\n                        }\n\n                        this.loading = true;\n                        const results = this.searchUsers(search);\n\n                        return results.pipe(\n                            catchError((e) => {\n                                this.hasError = true;\n                                console.error(e);\n                                return of(null);\n                            }),\n                        );\n                    }),\n                ),\n            ),\n            tap(() => {\n                this.loading = false;\n            }),\n        ),\n    );\n\n    private subs = new Subscription();\n\n    @HostListener('document:mousedown', ['$event']) outsideClick(event: MouseEvent) {\n        if ((event.target as HTMLElement)?.id === 'modalBackdrop') {\n            this.close();\n        }\n    }\n\n    @HostListener('document:keydown', ['$event']) handleKeyDown = (event: KeyboardEvent) => {\n        switch (event.key) {\n            case 'Esc':\n            case 'Escape': {\n                this.close();\n                break;\n            }\n            case 'I':\n            case 'i': {\n                if (event.ctrlKey || event.metaKey) {\n                    if (this.isOpen) {\n                        this.close();\n                    } else {\n                        this.init.emit();\n                    }\n                    event.preventDefault();\n                }\n                break;\n            }\n            default:\n                break;\n        }\n    };\n\n    ngOnDestroy() {\n        this.subs.unsubscribe();\n    }\n\n    /** Redirect to Keycloak impersonate page, which will redirect back\n     * after impersonation begins.\n     */\n    protected startImpersonation = (username?: string) => {\n        const _username = username ?? this.selectedUsername;\n        if (!_username) {\n            return;\n        }\n        const url = urlcat(this.oidcBaseUri(), START_IMPERSONATE_PATH, {\n            username,\n            returnUri: window.location.href,\n            defaultIdp: this.oidcDefaultIdp(),\n        });\n        this.replaceUrl(url);\n    };\n\n    protected handleSelectUser = (event: Event) => {\n        this.selectedUsername = (event.target as HTMLInputElement).value;\n    };\n\n    protected clearSearch = () => {\n        this.form.reset();\n    };\n\n    protected close = () => {\n        this.dismiss.emit();\n        this.clearSearch();\n    };\n\n    protected handleFormSubmit = (event: SubmitEvent) => {\n        event.preventDefault();\n        if (this.form.valid) {\n            this.handleSearchSubject.next(true);\n        }\n    };\n\n    protected handleSearchKeyPress = (event: KeyboardEvent) => {\n        if (!['ArrowDown', 'Down'].includes(event.key)) {\n            return;\n        }\n        event.preventDefault();\n        if (this.results()?.length) {\n            const firstResult = this.eref.nativeElement.querySelector(`#result_0`);\n            firstResult?.click();\n            firstResult?.focus();\n        }\n    };\n\n    protected handleResultKeyPress = (event: KeyboardEvent) => {\n        if (\n            ['ArrowUp', 'Up'].includes(event.key) &&\n            (event.target as HTMLInputElement)?.id === 'result_0'\n        ) {\n            event.preventDefault();\n            this.eref.nativeElement.querySelector('#searchInput')?.focus();\n        } else if (event.key === 'Enter') {\n            event.preventDefault();\n            this.startImpersonation();\n        }\n    };\n\n    /** Search Keycloak users using a generic search query. */\n    private searchUsers = (query: string) => {\n        const uri = urlcat(this.oidcBaseUri(), SEARCH_USERS_PATH, {\n            query,\n        });\n\n        return this.http.get<ImpersonateSearchResult[]>(uri, {\n            headers: {\n                Authorization: `Bearer ${this.accessTokenPayload().token}`,\n            },\n        });\n    };\n\n    private replaceUrl = (url: string) => window.location.replace(url);\n}\n","@if (isOpen) {\n    <div @libHbllFadeInOut class=\"modal-wrapper\" id=\"modalBackdrop\" data-testid=\"backdrop\">\n        <div class=\"modal-container\" data-testid=\"modal\">\n            <div class=\"modal-header\">\n                <h2>Impersonate</h2>\n                <button type=\"button\" (click)=\"close()\" aria-label=\"Close\" data-testid=\"close\">\n                    <span class=\"material-symbols-outlined icon-close\"> close </span>\n                </button>\n            </div>\n            <form [formGroup]=\"form\" (submit)=\"handleFormSubmit($event)\" data-testid=\"searchForm\">\n                <div class=\"search-header\">\n                    <div class=\"secondary\" [class.disabled]=\"!form.valid\">\n                        <span class=\"keyboard-key\">Enter</span> to search\n                    </div>\n                </div>\n                <label\n                    for=\"searchInput\"\n                    class=\"search-wrapper\"\n                    [class.invalid]=\"form.invalid && form.dirty\"\n                >\n                    <span class=\"material-symbols-outlined icon-search\"> search </span>\n                    <input\n                        id=\"searchInput\"\n                        type=\"text\"\n                        autocomplete=\"off\"\n                        formControlName=\"search\"\n                        placeholder=\"Search patrons...\"\n                        (keydown)=\"handleSearchKeyPress($event)\"\n                        #searchBox\n                        data-testid=\"searchInput\"\n                    />\n                    @if (!!searchBox.value && form.valid) {\n                        <span class=\"material-symbols-outlined icon-checkmark\"> check </span>\n                    }\n                    @if (searchBox.value.length) {\n                        <span\n                            (click)=\"clearSearch()\"\n                            @libHbllFadeInOut\n                            class=\"material-symbols-outlined icon-close\"\n                        >\n                            close\n                        </span>\n                    }\n                </label>\n            </form>\n            <fieldset\n                class=\"search-results-wrapper\"\n                id=\"resultsScrollContainer\"\n                (change)=\"handleSelectUser($event)\"\n            >\n                @if (!loading && results()) {\n                    @for (user of results(); track user.netId; let idx = $index) {\n                        <div\n                            class=\"result-field result\"\n                            [class.focus]=\"user.username === selectedUsername\"\n                            data-testid=\"result\"\n                        >\n                            <label\n                                [for]=\"'result_' + idx\"\n                                [class.warning]=\"user.restricted\"\n                                (mouseover)=\"selectedUsername = user.username\"\n                            >\n                                @if (user.restricted) {\n                                    <span class=\"material-symbols-outlined icon\"> warning </span>\n                                } @else {\n                                    <span class=\"material-symbols-outlined icon\"> person </span>\n                                }\n                                &nbsp; &nbsp;\n                                <span [title]=\"user\" data-testid=\"resultText\">{{\n                                    user | impersonateUser\n                                }}</span>\n                                <input\n                                    type=\"radio\"\n                                    [value]=\"user.username\"\n                                    class=\"hidden\"\n                                    [id]=\"'result_' + idx\"\n                                    name=\"resultSelect\"\n                                    (keydown)=\"handleResultKeyPress($event)\"\n                                />\n                            </label>\n                            <button\n                                class=\"impersonate-button\"\n                                data-testid=\"impersonateBtn\"\n                                (click)=\"startImpersonation(user.username)\"\n                            >\n                                Impersonate\n                            </button>\n                        </div>\n                    } @empty {\n                        <div class=\"result-field\">\n                            No results. Try searching by Net ID or BYU ID.\n                        </div>\n                    }\n                }\n                @if (loading) {\n                    <div class=\"result-field\">\n                        <div class=\"lib-spinner\"></div>\n                    </div>\n                }\n                @if (hasError) {\n                    <div class=\"result-field\">Something went wrong. We'll keep trying.</div>\n                }\n            </fieldset>\n        </div>\n    </div>\n}\n"]}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impersonate-modal.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/impersonate-modal/impersonate-modal.component.ts","../../../../../projects/components/src/lib/impersonate-modal/impersonate-modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostListener,\n    Input,\n    OnDestroy,\n    Output,\n    Pipe,\n    PipeTransform,\n    inject,\n    input,\n} from '@angular/core';\nimport { FormBuilder, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { Subject, Subscription, of } from 'rxjs';\nimport { catchError, startWith, switchMap, tap } from 'rxjs/operators';\nimport { libHbllFadeInOut } from '../animations/animations';\nimport urlcat from 'urlcat';\nimport { HttpClient } from '@angular/common/http';\nimport { CommonModule } from '@angular/common';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { TokenPayload } from '../models/token-payload';\n\nexport const defaultOidcBaseUri = 'https://keycloak.lib.byu.edu/';\nexport const defaultOidcDefaultIdp = 'ces';\n\n@Pipe({\n    name: 'impersonateUser',\n    standalone: true,\n})\nexport class ImpersonateUserPipe implements PipeTransform {\n    transform(user: ImpersonateSearchResult): string {\n        return `${user.name} (${user.netId || 'Unknown'})${user.restricted ? ' — Restricted' : ''}`;\n    }\n}\nexport interface ImpersonateSearchResult {\n    netId: string;\n    institution: string;\n    username: string;\n    name: string;\n    preferredName: string;\n    givenName: string;\n    surname: string;\n    restricted: boolean;\n}\n\nconst SEARCH_USERS_PATH = '/impersonate/api/search/';\nconst START_IMPERSONATE_PATH = '/impersonate/:username';\n\n@Component({\n    standalone: true,\n    imports: [CommonModule, ReactiveFormsModule, ImpersonateUserPipe],\n    selector: 'lib-impersonate-modal',\n    templateUrl: './impersonate-modal.component.html',\n    styleUrls: ['./impersonate-modal.component.scss'],\n    animations: [libHbllFadeInOut],\n})\nexport class ImpersonateModalComponent implements OnDestroy {\n    private readonly http = inject(HttpClient);\n    private readonly fb = inject(FormBuilder);\n    private readonly eref = inject(ElementRef);\n\n    @Input() set showModal(open: boolean) {\n        this.isOpen = open;\n        if (open) {\n            // Set focus on search input shortly after opening modal so user notices\n            // the input receiving focus.\n            setTimeout(() => this.eref.nativeElement.querySelector('#searchInput')?.focus(), 250);\n        }\n    }\n    oidcBaseUri = input(defaultOidcBaseUri);\n    oidcDefaultIdp = input(defaultOidcDefaultIdp);\n    // Require an object here so that access tokens are not visible/extractable from the DOM in consuming applications. Instead they are only stored in memory.\n    accessTokenPayload = input.required<TokenPayload>();\n    @Output() dismiss = new EventEmitter<void>();\n    @Output() init = new EventEmitter<void>();\n\n    protected isOpen = false;\n    protected hasError = false;\n    protected form = this.fb.nonNullable.group({\n        search: this.fb.control('', { validators: Validators.minLength(3) }),\n    });\n    protected selectedUsername?: string;\n\n    protected loading = false;\n    protected handleSearchSubject = new Subject<boolean>();\n    protected results = toSignal(\n        this.form.controls.search.valueChanges.pipe(\n            switchMap((search) =>\n                this.handleSearchSubject.pipe(\n                    startWith(false),\n                    switchMap((handleSearch) => {\n                        this.selectedUsername = undefined;\n                        this.hasError = false;\n\n                        if (!search || !handleSearch) {\n                            return of(null);\n                        }\n\n                        this.loading = true;\n                        const results = this.searchUsers(search);\n\n                        return results.pipe(\n                            catchError((e) => {\n                                this.hasError = true;\n                                console.error(e);\n                                return of(null);\n                            }),\n                        );\n                    }),\n                ),\n            ),\n            tap(() => {\n                this.loading = false;\n            }),\n        ),\n    );\n\n    private subs = new Subscription();\n\n    @HostListener('document:mousedown', ['$event']) outsideClick(event: MouseEvent) {\n        if ((event.target as HTMLElement)?.id === 'modalBackdrop') {\n            this.close();\n        }\n    }\n\n    @HostListener('document:keydown', ['$event']) handleKeyDown = (event: KeyboardEvent) => {\n        switch (event.key) {\n            case 'Esc':\n            case 'Escape': {\n                this.close();\n                break;\n            }\n            case 'I':\n            case 'i': {\n                if (event.ctrlKey || event.metaKey) {\n                    if (this.isOpen) {\n                        this.close();\n                    } else {\n                        this.init.emit();\n                    }\n                    event.preventDefault();\n                }\n                break;\n            }\n            default:\n                break;\n        }\n    };\n\n    ngOnDestroy() {\n        this.subs.unsubscribe();\n    }\n\n    /** Redirect to Keycloak impersonate page, which will redirect back\n     * after impersonation begins.\n     */\n    protected startImpersonation = (username?: string) => {\n        const _username = username ?? this.selectedUsername;\n        if (!_username) {\n            return;\n        }\n        const url = urlcat(this.oidcBaseUri(), START_IMPERSONATE_PATH, {\n            username,\n            returnUri: window.location.href,\n            defaultIdp: this.oidcDefaultIdp(),\n        });\n        this.replaceUrl(url);\n    };\n\n    protected handleSelectUser = (event: Event) => {\n        this.selectedUsername = (event.target as HTMLInputElement).value;\n    };\n\n    protected clearSearch = () => {\n        this.form.reset();\n    };\n\n    protected close = () => {\n        this.dismiss.emit();\n        this.clearSearch();\n    };\n\n    protected handleFormSubmit = (event: SubmitEvent) => {\n        event.preventDefault();\n        if (this.form.valid) {\n            this.handleSearchSubject.next(true);\n        }\n    };\n\n    protected handleSearchKeyPress = (event: KeyboardEvent) => {\n        if (!['ArrowDown', 'Down'].includes(event.key)) {\n            return;\n        }\n        event.preventDefault();\n        if (this.results()?.length) {\n            const firstResult = this.eref.nativeElement.querySelector(`#result_0`);\n            firstResult?.click();\n            firstResult?.focus();\n        }\n    };\n\n    protected handleResultKeyPress = (event: KeyboardEvent) => {\n        if (\n            ['ArrowUp', 'Up'].includes(event.key) &&\n            (event.target as HTMLInputElement)?.id === 'result_0'\n        ) {\n            event.preventDefault();\n            this.eref.nativeElement.querySelector('#searchInput')?.focus();\n        } else if (event.key === 'Enter') {\n            event.preventDefault();\n            this.startImpersonation();\n        }\n    };\n\n    /** Search Keycloak users using a generic search query. */\n    private searchUsers = (query: string) => {\n        const uri = urlcat(this.oidcBaseUri(), SEARCH_USERS_PATH, {\n            query,\n        });\n\n        return this.http.get<ImpersonateSearchResult[]>(uri, {\n            headers: {\n                Authorization: `Bearer ${this.accessTokenPayload().token}`,\n            },\n        });\n    };\n\n    private replaceUrl = (url: string) => window.location.replace(url);\n}\n","@if (isOpen) {\n    <div @libHbllFadeInOut class=\"modal-wrapper\" id=\"modalBackdrop\" data-testid=\"backdrop\">\n        <div class=\"modal-container\" data-testid=\"modal\">\n            <div class=\"modal-header\">\n                <h2>Impersonate</h2>\n                <button type=\"button\" (click)=\"close()\" aria-label=\"Close\" data-testid=\"close\">\n                    <span class=\"material-symbols-outlined icon-close\"> close </span>\n                </button>\n            </div>\n            <form [formGroup]=\"form\" (submit)=\"handleFormSubmit($event)\" data-testid=\"searchForm\">\n                <div class=\"search-header\">\n                    <div class=\"secondary\" [class.disabled]=\"!form.valid\">\n                        <span class=\"keyboard-key\">Enter</span> to search\n                    </div>\n                </div>\n                <label\n                    for=\"searchInput\"\n                    class=\"search-wrapper\"\n                    [class.invalid]=\"form.invalid && form.dirty\"\n                >\n                    <span class=\"material-symbols-outlined icon-search\"> search </span>\n                    <input\n                        id=\"searchInput\"\n                        type=\"text\"\n                        autocomplete=\"off\"\n                        formControlName=\"search\"\n                        placeholder=\"Search patrons...\"\n                        (keydown)=\"handleSearchKeyPress($event)\"\n                        #searchBox\n                        data-testid=\"searchInput\"\n                    />\n                    @if (!!searchBox.value && form.valid) {\n                        <span class=\"material-symbols-outlined icon-checkmark\"> check </span>\n                    }\n                    @if (searchBox.value.length) {\n                        <span\n                            (click)=\"clearSearch()\"\n                            @libHbllFadeInOut\n                            class=\"material-symbols-outlined icon-close\"\n                        >\n                            close\n                        </span>\n                    }\n                </label>\n            </form>\n            <fieldset\n                class=\"search-results-wrapper\"\n                id=\"resultsScrollContainer\"\n                (change)=\"handleSelectUser($event)\"\n            >\n                @if (!loading && results()) {\n                    @for (user of results(); track user.netId; let idx = $index) {\n                        <div\n                            class=\"result-field result\"\n                            [class.focus]=\"user.username === selectedUsername\"\n                            data-testid=\"result\"\n                        >\n                            <label\n                                [for]=\"'result_' + idx\"\n                                [class.warning]=\"user.restricted\"\n                                (mouseover)=\"selectedUsername = user.username\"\n                            >\n                                @if (user.restricted) {\n                                    <span class=\"material-symbols-outlined icon\"> warning </span>\n                                } @else {\n                                    <span class=\"material-symbols-outlined icon\"> person </span>\n                                }\n                                &nbsp; &nbsp;\n                                <span [title]=\"user\" data-testid=\"resultText\">{{\n                                    user | impersonateUser\n                                }}</span>\n                                <input\n                                    type=\"radio\"\n                                    [value]=\"user.username\"\n                                    class=\"hidden\"\n                                    [id]=\"'result_' + idx\"\n                                    name=\"resultSelect\"\n                                    (keydown)=\"handleResultKeyPress($event)\"\n                                />\n                            </label>\n                            <button\n                                class=\"impersonate-button\"\n                                data-testid=\"impersonateBtn\"\n                                (click)=\"startImpersonation(user.username)\"\n                            >\n                                Impersonate\n                            </button>\n                        </div>\n                    } @empty {\n                        <div class=\"result-field\">\n                            No results. Try searching by Net ID or BYU ID.\n                        </div>\n                    }\n                }\n                @if (loading) {\n                    <div class=\"result-field\">\n                        <div class=\"lib-spinner\"></div>\n                    </div>\n                }\n                @if (hasError) {\n                    <div class=\"result-field\">Something went wrong. We'll keep trying.</div>\n                }\n            </fieldset>\n        </div>\n    </div>\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VhcmNoLWNvbmZpZy5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9zcy1zZWFyY2gtYmFyL21vZGVscy9zZWFyY2gtY29uZmlnLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBZHZhbmNlZFNlYXJjaFF1ZXJ5Um93IH0gZnJvbSAnLi9hZHZhbmNlZC1zZWFyY2gubW9kZWwnO1xuaW1wb3J0IHsgU2VhcmNoU2NvcGUgfSBmcm9tICcuL3NlYXJjaC1zY29wZS5tb2RlbCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VhcmNoQ29uZmlnIHtcbiAgICBpbnN0aXR1dGlvbjogJ2J5dScgfCAnbGF3JyB8ICdlbnNpZ24nO1xuICAgIHNob3dBZHZhbmNlZFNlYXJjaDogYm9vbGVhbjtcbiAgICBzaG93QWR2YW5jZWRTZWFyY2hBc1RleHQ6IGJvb2xlYW47XG4gICAgc2NvcGU6IFNlYXJjaFNjb3BlO1xuICAgIHE6IHN0cmluZztcbiAgICBpc1N1Z2dlc3Rpb246IGJvb2xlYW47XG4gICAgYWR2YW5jZWRTZWFyY2hRdWVyeVJvd3M6IEFkdmFuY2VkU2VhcmNoUXVlcnlSb3dbXTtcbiAgICBsb2NhbEFkdmFuY2VkU2VhcmNoOiB7XG4gICAgICAgIGNyZWF0aW9uRGF0ZTogRGF0ZTtcbiAgICAgICAgc2VsZWN0ZWRMYW5ndWFnZXM6IHN0cmluZ1tdO1xuICAgICAgICBzZWxlY3RlZFJlc291cmNlVHlwZXM6IHN0cmluZ1tdO1xuICAgICAgICBzZWxlY3RlZENvbGxlY3Rpb25zOiBzdHJpbmdbXTtcbiAgICB9O1xuICAgIGV4dGVybmFsQWR2YW5jZWRTZWFyY2g6IHtcbiAgICAgICAgZGF0ZVB1Ymxpc2hlZDogRGF0ZTtcbiAgICAgICAgc2VsZWN0ZWRMYW5ndWFnZXM6IHN0cmluZ1tdO1xuICAgICAgICBsaW1pdFJlc3VsdHM6IHtcbiAgICAgICAgICAgIHBlZXJSZXZpZXdlZDogYm9vbGVhbjtcbiAgICAgICAgfTtcbiAgICAgICAgZXhwYW5kUmVzdWx0czoge1xuICAgICAgICAgICAgYXBwbHlFcXVpdmFsZW50U3ViamVjdHM6IGJvb2xlYW47XG4gICAgICAgICAgICBmdWxsVGV4dDogYm9vbGVhbjtcbiAgICAgICAgfTtcbiAgICB9O1xufVxuXG5pbnRlcmZhY2UgRGF0ZSB7XG4gICAgZnJvbTogc3RyaW5nO1xuICAgIHRvOiBzdHJpbmc7XG59XG4iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VhcmNoLWNvbmZpZy5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9zcy1zZWFyY2gtYmFyL21vZGVscy9zZWFyY2gtY29uZmlnLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBZHZhbmNlZFNlYXJjaFF1ZXJ5Um93IH0gZnJvbSAnLi9hZHZhbmNlZC1zZWFyY2gubW9kZWwnO1xuaW1wb3J0IHsgU2VhcmNoU2NvcGUgfSBmcm9tICcuL3NlYXJjaC1zY29wZS5tb2RlbCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VhcmNoQ29uZmlnIHtcbiAgICBpbnN0aXR1dGlvbjogJ2J5dScgfCAnbGF3JyB8ICdlbnNpZ24nO1xuICAgIHNob3dBZHZhbmNlZFNlYXJjaDogYm9vbGVhbjtcbiAgICBzaG93QWR2YW5jZWRTZWFyY2hBc1RleHQ6IGJvb2xlYW47XG4gICAgc2NvcGU6IFNlYXJjaFNjb3BlO1xuICAgIHE6IHN0cmluZztcbiAgICBhZHZhbmNlZFNlYXJjaFF1ZXJ5Um93czogQWR2YW5jZWRTZWFyY2hRdWVyeVJvd1tdO1xuICAgIGxvY2FsQWR2YW5jZWRTZWFyY2g6IHtcbiAgICAgICAgY3JlYXRpb25EYXRlOiBEYXRlO1xuICAgICAgICBzZWxlY3RlZExhbmd1YWdlczogc3RyaW5nW107XG4gICAgICAgIHNlbGVjdGVkUmVzb3VyY2VUeXBlczogc3RyaW5nW107XG4gICAgICAgIHNlbGVjdGVkQ29sbGVjdGlvbnM6IHN0cmluZ1tdO1xuICAgIH07XG4gICAgZXh0ZXJuYWxBZHZhbmNlZFNlYXJjaDoge1xuICAgICAgICBkYXRlUHVibGlzaGVkOiBEYXRlO1xuICAgICAgICBzZWxlY3RlZExhbmd1YWdlczogc3RyaW5nW107XG4gICAgICAgIGxpbWl0UmVzdWx0czoge1xuICAgICAgICAgICAgcGVlclJldmlld2VkOiBib29sZWFuO1xuICAgICAgICB9O1xuICAgICAgICBleHBhbmRSZXN1bHRzOiB7XG4gICAgICAgICAgICBhcHBseUVxdWl2YWxlbnRTdWJqZWN0czogYm9vbGVhbjtcbiAgICAgICAgICAgIGZ1bGxUZXh0OiBib29sZWFuO1xuICAgICAgICB9O1xuICAgIH07XG59XG5cbmludGVyZmFjZSBEYXRlIHtcbiAgICBmcm9tOiBzdHJpbmc7XG4gICAgdG86IHN0cmluZztcbn1cbiJdfQ==
@@ -1,8 +1,8 @@
1
- import { Component, EventEmitter, Input, Output, ViewChild, inject, HostListener, } from '@angular/core';
1
+ import { Component, EventEmitter, Input, Output, ViewChild, inject, } from '@angular/core';
2
2
  import { FormBuilder, ReactiveFormsModule, Validators } from '@angular/forms';
3
3
  import { MatTooltip, MatTooltipModule } from '@angular/material/tooltip';
4
4
  import { Subscription } from 'rxjs';
5
- import { startWith, filter, debounceTime, distinctUntilChanged } from 'rxjs/operators';
5
+ import { startWith, filter } from 'rxjs/operators';
6
6
  import { MatIconModule } from '@angular/material/icon';
7
7
  import { CommonModule } from '@angular/common';
8
8
  import * as i0 from "@angular/core";
@@ -13,65 +13,25 @@ export class SimpleSearchComponent {
13
13
  constructor() {
14
14
  this.fb = inject(FormBuilder);
15
15
  this.subscription = new Subscription();
16
- this.suggestions = [];
17
16
  this.simpleSearch = new EventEmitter();
18
17
  this.clearSimpleSearch = new EventEmitter();
19
- this.suggest = new EventEmitter();
20
18
  this.isSubmitted = false;
21
- this.showSuggestions = false;
22
- this.selectedSuggestionIndex = -1;
23
- this.skipNextSuggest = false;
24
19
  this.searchForm = this.fb.nonNullable.group({
25
20
  simpleQuery: ['', Validators.required],
26
21
  });
27
- this.emitSimpleSearch = (isSuggestion = false) => {
22
+ this.emitSimpleSearch = () => {
28
23
  this.isSubmitted = true;
29
24
  // Don't perform a search unless there is a query
30
25
  if (this.simpleQuery.invalid) {
31
26
  this.showSearchValidationToolTip();
32
27
  return;
33
28
  }
34
- this.simpleSearch.emit({
35
- ...this.config,
36
- q: this.simpleQuery.value,
37
- isSuggestion: isSuggestion,
38
- });
29
+ this.simpleSearch.emit({ ...this.config, q: this.simpleQuery.value });
39
30
  this.isSubmitted = false;
40
31
  };
41
32
  this.clearQuery = () => {
42
33
  this.simpleQuery.setValue('');
43
34
  this.clearSimpleSearch.emit();
44
- this.hideSuggestions();
45
- };
46
- this.selectSuggestion = (suggestion) => {
47
- this.skipNextSuggest = true;
48
- this.simpleQuery.setValue(suggestion);
49
- this.hideSuggestions();
50
- this.emitSimpleSearch(true);
51
- };
52
- this.onInputKeydown = (event) => {
53
- if (!this.showSuggestions || this.suggestions.length === 0) {
54
- return;
55
- }
56
- switch (event.key) {
57
- case 'ArrowDown':
58
- event.preventDefault();
59
- this.selectedSuggestionIndex = Math.min(this.selectedSuggestionIndex + 1, this.suggestions.length - 1);
60
- break;
61
- case 'ArrowUp':
62
- event.preventDefault();
63
- this.selectedSuggestionIndex = Math.max(this.selectedSuggestionIndex - 1, -1);
64
- break;
65
- case 'Enter':
66
- if (this.selectedSuggestionIndex >= 0) {
67
- event.preventDefault();
68
- this.selectSuggestion(this.suggestions[this.selectedSuggestionIndex]);
69
- }
70
- break;
71
- case 'Escape':
72
- this.hideSuggestions();
73
- break;
74
- }
75
35
  };
76
36
  this.setupForm = () => {
77
37
  this.simpleQuery.setValue(this.config.q || this.config.advancedSearchQueryRows[0]?.query || '');
@@ -81,10 +41,6 @@ export class SimpleSearchComponent {
81
41
  this.inputTooltip.disabled = false;
82
42
  this.inputTooltip.show();
83
43
  };
84
- this.hideSuggestions = () => {
85
- this.showSuggestions = false;
86
- this.selectedSuggestionIndex = -1;
87
- };
88
44
  }
89
45
  set config(config) {
90
46
  this._config = config;
@@ -104,34 +60,16 @@ export class SimpleSearchComponent {
104
60
  .subscribe(() => {
105
61
  this.inputTooltip.disabled = true;
106
62
  }));
107
- // Emit suggestion requests with debounce
108
- this.subscription.add(this.simpleQuery.valueChanges
109
- .pipe(debounceTime(300), distinctUntilChanged(), filter((value) => value.trim().length > 0))
110
- .subscribe((value) => {
111
- if (this.skipNextSuggest) {
112
- this.skipNextSuggest = false;
113
- return;
114
- }
115
- this.suggest.emit({ ...this.config, q: value });
116
- this.showSuggestions = true;
117
- this.selectedSuggestionIndex = -1;
118
- }));
119
63
  }
120
64
  ngOnDestroy() {
121
65
  this.subscription.unsubscribe();
122
66
  }
123
- onDocumentClick(event) {
124
- const target = event.target;
125
- if (!target.closest('.ss-container')) {
126
- this.hideSuggestions();
127
- }
128
- }
129
67
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: SimpleSearchComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
130
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: SimpleSearchComponent, isStandalone: true, selector: "lib-ss-simple-search", inputs: { config: "config", suggestions: "suggestions" }, outputs: { simpleSearch: "simpleSearch", clearSimpleSearch: "clearSimpleSearch", suggest: "suggest" }, host: { listeners: { "document:click": "onDocumentClick($event)" } }, viewQueries: [{ propertyName: "searchInput", first: true, predicate: ["searchInput"], descendants: true }, { propertyName: "inputTooltip", first: true, predicate: MatTooltip, descendants: true }], ngImport: i0, template: "<div class=\"ss-container\">\n <form [formGroup]=\"searchForm\" (submit)=\"emitSimpleSearch()\" data-testid=\"searchForm\">\n <input\n #searchInput\n data-testid=\"searchInput\"\n id=\"q\"\n name=\"q\"\n type=\"text\"\n required\n aria-required=\"true\"\n aria-label=\"search input\"\n autocapitalize=\"off\"\n formControlName=\"simpleQuery\"\n matTooltip=\"Fill out this field\"\n [matTooltipPosition]=\"'above'\"\n [matTooltipDisabled]=\"true\"\n [attr.aria-invalid]=\"isSubmitted && simpleQuery.invalid\"\n (keydown)=\"onInputKeydown($event)\"\n />\n <label for=\"q\" data-testid=\"label\">{{\n config.scope === 'local'\n ? 'Books, media, special collections and more'\n : 'Ebooks, articles, journals, databases, streaming media and more'\n }}</label>\n <button\n type=\"submit\"\n aria-label=\"search\"\n data-testid=\"searchBtn\"\n [ngClass]=\"{ ensign: config.institution === 'ensign' }\"\n >\n <span class=\"material-symbols-outlined ss-icon\"> search </span>\n </button>\n </form>\n\n @if (simpleQuery.value) {\n <button id=\"clear\" (click)=\"clearQuery()\" data-testid=\"clearBtn\">\n <span class=\"material-symbols-outlined ss-icon\"> cancel </span>\n </button>\n }\n\n @if (showSuggestions && suggestions.length > 0) {\n <ul class=\"suggestions-list\" role=\"listbox\" data-testid=\"suggestionsList\">\n @for (suggestion of suggestions; track suggestion; let i = $index) {\n <li\n class=\"suggestion-item\"\n [class.selected]=\"i === selectedSuggestionIndex\"\n (click)=\"selectSuggestion(suggestion)\"\n (keydown.enter)=\"selectSuggestion(suggestion)\"\n (keydown.space)=\"selectSuggestion(suggestion); $event.preventDefault()\"\n role=\"option\"\n tabindex=\"0\"\n [attr.aria-selected]=\"i === selectedSuggestionIndex\"\n data-testid=\"suggestionItem\"\n >\n {{ suggestion }}\n </li>\n }\n </ul>\n }\n</div>\n", styles: ["a,button{border:none;background:none;font-family:inherit;padding:0;margin:0;font-size:inherit;color:#1c7ec9;text-decoration:none;cursor:pointer}a:hover,button:hover{color:#8ab6f0}.ss-container{position:relative}form{display:flex}.ss-icon{font-size:1em}button[type=submit]{background:#fff;border-radius:0 4px 4px 0;margin:0;width:3em;cursor:pointer;display:flex;justify-content:center;align-items:center;transition:color .2s,background-color .2s}button[type=submit] .ss-icon{color:#0047ba;font-size:1.7em}button[type=submit]:hover .ss-icon{color:#6892ca}button[type=submit].ensign .ss-icon{color:#2b6042}button[type=submit].ensign .ss-icon:hover{color:#357551}#clear{position:absolute;right:3em;top:0%;height:100%;display:flex;justify-content:center;align-items:center;padding:0 0 0 .375rem}#clear .ss-icon{height:auto;color:#acacac;transition:color ease-in-out .05s}#clear:hover .ss-icon{color:#666}input[type=text]{background-color:#fff;color:#000;font-family:inherit;border:none;font-weight:600;margin:0;overflow:hidden;cursor:text;width:calc(100% - 3em);font-size:1em;border-radius:4px 0 0 4px;padding:.56em 2em .56em .56em}input[type=text]:focus{outline:none}input[type=text]:valid+label,input[type=text]:focus+label{font-size:.75em;top:-.8em;padding-top:0;padding-bottom:0;pointer-events:none;margin-top:0;margin-left:.56em;cursor:default}input[type=text]:valid+label:before,input[type=text]:focus+label:before{opacity:1}label{cursor:text;transition:all .1s ease-in-out;position:absolute;padding:.5em .28em;margin-left:.28em;left:0;color:#707070;z-index:1;max-width:calc(100% - 3.4em);white-space:nowrap;line-height:normal;overflow:hidden;text-overflow:ellipsis;border-radius:4px}label:before{transition:all .1s ease-in-out;background-color:#fff;content:\"\";position:absolute;inset:0;opacity:0;z-index:-1}.suggestions-list{position:absolute;top:100%;left:0;right:0;background:#fff;border:1px solid #d0d0d0;border-top:none;border-radius:0 0 4px 4px;list-style:none;margin:0;padding:0;max-height:300px;overflow-y:auto;box-shadow:0 4px 6px #0000001a;z-index:1000}.suggestion-item{padding:.75em 1em;cursor:pointer;transition:background-color .15s ease;color:#333;font-size:1em}.suggestion-item:hover,.suggestion-item.selected{background-color:#f0f0f0}.suggestion-item.selected{background-color:#e6f2ff}.suggestion-item:last-child{border-radius:0 0 4px 4px}\n"], dependencies: [{ kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.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: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] }); }
68
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: SimpleSearchComponent, isStandalone: true, selector: "lib-ss-simple-search", inputs: { config: "config" }, outputs: { simpleSearch: "simpleSearch", clearSimpleSearch: "clearSimpleSearch" }, viewQueries: [{ propertyName: "searchInput", first: true, predicate: ["searchInput"], descendants: true }, { propertyName: "inputTooltip", first: true, predicate: MatTooltip, descendants: true }], ngImport: i0, template: "<div class=\"ss-container\">\n <form [formGroup]=\"searchForm\" (submit)=\"emitSimpleSearch()\" data-testid=\"searchForm\">\n <input\n #searchInput\n data-testid=\"searchInput\"\n id=\"q\"\n name=\"q\"\n type=\"text\"\n required\n aria-required=\"true\"\n aria-label=\"search input\"\n autocapitalize=\"off\"\n formControlName=\"simpleQuery\"\n matTooltip=\"Fill out this field\"\n [matTooltipPosition]=\"'above'\"\n [matTooltipDisabled]=\"true\"\n [attr.aria-invalid]=\"isSubmitted && simpleQuery.invalid\"\n />\n <label for=\"q\" data-testid=\"label\">{{\n config.scope === 'local'\n ? 'Books, media, special collections and more'\n : 'Ebooks, articles, journals, databases, streaming media and more'\n }}</label>\n <button\n type=\"submit\"\n aria-label=\"search\"\n data-testid=\"searchBtn\"\n [ngClass]=\"{ ensign: config.institution === 'ensign' }\"\n >\n <span class=\"material-symbols-outlined ss-icon\"> search </span>\n </button>\n </form>\n\n @if (simpleQuery.value) {\n <button id=\"clear\" (click)=\"clearQuery()\" data-testid=\"clearBtn\">\n <span class=\"material-symbols-outlined ss-icon\"> cancel </span>\n </button>\n }\n</div>\n", styles: ["a,button{border:none;background:none;font-family:inherit;padding:0;margin:0;font-size:inherit;color:#1c7ec9;text-decoration:none;cursor:pointer}a:hover,button:hover{color:#8ab6f0}.ss-container{position:relative}form{display:flex}.ss-icon{font-size:1em}button[type=submit]{background:#fff;border-radius:0 4px 4px 0;margin:0;width:3em;cursor:pointer;display:flex;justify-content:center;align-items:center;transition:color .2s,background-color .2s}button[type=submit] .ss-icon{color:#0047ba;font-size:1.7em}button[type=submit]:hover .ss-icon{color:#6892ca}button[type=submit].ensign .ss-icon{color:#2b6042}button[type=submit].ensign .ss-icon:hover{color:#357551}#clear{position:absolute;right:3em;top:0%;height:100%;display:flex;justify-content:center;align-items:center;padding:0 0 0 .375rem}#clear .ss-icon{height:auto;color:#acacac;transition:color ease-in-out .05s}#clear:hover .ss-icon{color:#666}input[type=text]{background-color:#fff;color:#000;font-family:inherit;border:none;font-weight:600;margin:0;overflow:hidden;cursor:text;width:calc(100% - 3em);font-size:1em;border-radius:4px 0 0 4px;padding:.56em 2em .56em .56em}input[type=text]:focus{outline:none}input[type=text]:valid+label,input[type=text]:focus+label{font-size:.75em;top:-.8em;padding-top:0;padding-bottom:0;pointer-events:none;margin-top:0;margin-left:.56em;cursor:default}input[type=text]:valid+label:before,input[type=text]:focus+label:before{opacity:1}label{cursor:text;transition:all .1s ease-in-out;position:absolute;padding:.5em .28em;margin-left:.28em;left:0;color:#707070;z-index:1;max-width:calc(100% - 3.4em);white-space:nowrap;line-height:normal;overflow:hidden;text-overflow:ellipsis;border-radius:4px}label:before{transition:all .1s ease-in-out;background-color:#fff;content:\"\";position:absolute;inset:0;opacity:0;z-index:-1}\n"], dependencies: [{ kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.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: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] }); }
131
69
  }
132
70
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: SimpleSearchComponent, decorators: [{
133
71
  type: Component,
134
- args: [{ selector: 'lib-ss-simple-search', standalone: true, imports: [MatTooltipModule, MatIconModule, ReactiveFormsModule, CommonModule], template: "<div class=\"ss-container\">\n <form [formGroup]=\"searchForm\" (submit)=\"emitSimpleSearch()\" data-testid=\"searchForm\">\n <input\n #searchInput\n data-testid=\"searchInput\"\n id=\"q\"\n name=\"q\"\n type=\"text\"\n required\n aria-required=\"true\"\n aria-label=\"search input\"\n autocapitalize=\"off\"\n formControlName=\"simpleQuery\"\n matTooltip=\"Fill out this field\"\n [matTooltipPosition]=\"'above'\"\n [matTooltipDisabled]=\"true\"\n [attr.aria-invalid]=\"isSubmitted && simpleQuery.invalid\"\n (keydown)=\"onInputKeydown($event)\"\n />\n <label for=\"q\" data-testid=\"label\">{{\n config.scope === 'local'\n ? 'Books, media, special collections and more'\n : 'Ebooks, articles, journals, databases, streaming media and more'\n }}</label>\n <button\n type=\"submit\"\n aria-label=\"search\"\n data-testid=\"searchBtn\"\n [ngClass]=\"{ ensign: config.institution === 'ensign' }\"\n >\n <span class=\"material-symbols-outlined ss-icon\"> search </span>\n </button>\n </form>\n\n @if (simpleQuery.value) {\n <button id=\"clear\" (click)=\"clearQuery()\" data-testid=\"clearBtn\">\n <span class=\"material-symbols-outlined ss-icon\"> cancel </span>\n </button>\n }\n\n @if (showSuggestions && suggestions.length > 0) {\n <ul class=\"suggestions-list\" role=\"listbox\" data-testid=\"suggestionsList\">\n @for (suggestion of suggestions; track suggestion; let i = $index) {\n <li\n class=\"suggestion-item\"\n [class.selected]=\"i === selectedSuggestionIndex\"\n (click)=\"selectSuggestion(suggestion)\"\n (keydown.enter)=\"selectSuggestion(suggestion)\"\n (keydown.space)=\"selectSuggestion(suggestion); $event.preventDefault()\"\n role=\"option\"\n tabindex=\"0\"\n [attr.aria-selected]=\"i === selectedSuggestionIndex\"\n data-testid=\"suggestionItem\"\n >\n {{ suggestion }}\n </li>\n }\n </ul>\n }\n</div>\n", styles: ["a,button{border:none;background:none;font-family:inherit;padding:0;margin:0;font-size:inherit;color:#1c7ec9;text-decoration:none;cursor:pointer}a:hover,button:hover{color:#8ab6f0}.ss-container{position:relative}form{display:flex}.ss-icon{font-size:1em}button[type=submit]{background:#fff;border-radius:0 4px 4px 0;margin:0;width:3em;cursor:pointer;display:flex;justify-content:center;align-items:center;transition:color .2s,background-color .2s}button[type=submit] .ss-icon{color:#0047ba;font-size:1.7em}button[type=submit]:hover .ss-icon{color:#6892ca}button[type=submit].ensign .ss-icon{color:#2b6042}button[type=submit].ensign .ss-icon:hover{color:#357551}#clear{position:absolute;right:3em;top:0%;height:100%;display:flex;justify-content:center;align-items:center;padding:0 0 0 .375rem}#clear .ss-icon{height:auto;color:#acacac;transition:color ease-in-out .05s}#clear:hover .ss-icon{color:#666}input[type=text]{background-color:#fff;color:#000;font-family:inherit;border:none;font-weight:600;margin:0;overflow:hidden;cursor:text;width:calc(100% - 3em);font-size:1em;border-radius:4px 0 0 4px;padding:.56em 2em .56em .56em}input[type=text]:focus{outline:none}input[type=text]:valid+label,input[type=text]:focus+label{font-size:.75em;top:-.8em;padding-top:0;padding-bottom:0;pointer-events:none;margin-top:0;margin-left:.56em;cursor:default}input[type=text]:valid+label:before,input[type=text]:focus+label:before{opacity:1}label{cursor:text;transition:all .1s ease-in-out;position:absolute;padding:.5em .28em;margin-left:.28em;left:0;color:#707070;z-index:1;max-width:calc(100% - 3.4em);white-space:nowrap;line-height:normal;overflow:hidden;text-overflow:ellipsis;border-radius:4px}label:before{transition:all .1s ease-in-out;background-color:#fff;content:\"\";position:absolute;inset:0;opacity:0;z-index:-1}.suggestions-list{position:absolute;top:100%;left:0;right:0;background:#fff;border:1px solid #d0d0d0;border-top:none;border-radius:0 0 4px 4px;list-style:none;margin:0;padding:0;max-height:300px;overflow-y:auto;box-shadow:0 4px 6px #0000001a;z-index:1000}.suggestion-item{padding:.75em 1em;cursor:pointer;transition:background-color .15s ease;color:#333;font-size:1em}.suggestion-item:hover,.suggestion-item.selected{background-color:#f0f0f0}.suggestion-item.selected{background-color:#e6f2ff}.suggestion-item:last-child{border-radius:0 0 4px 4px}\n"] }]
72
+ args: [{ selector: 'lib-ss-simple-search', standalone: true, imports: [MatTooltipModule, MatIconModule, ReactiveFormsModule, CommonModule], template: "<div class=\"ss-container\">\n <form [formGroup]=\"searchForm\" (submit)=\"emitSimpleSearch()\" data-testid=\"searchForm\">\n <input\n #searchInput\n data-testid=\"searchInput\"\n id=\"q\"\n name=\"q\"\n type=\"text\"\n required\n aria-required=\"true\"\n aria-label=\"search input\"\n autocapitalize=\"off\"\n formControlName=\"simpleQuery\"\n matTooltip=\"Fill out this field\"\n [matTooltipPosition]=\"'above'\"\n [matTooltipDisabled]=\"true\"\n [attr.aria-invalid]=\"isSubmitted && simpleQuery.invalid\"\n />\n <label for=\"q\" data-testid=\"label\">{{\n config.scope === 'local'\n ? 'Books, media, special collections and more'\n : 'Ebooks, articles, journals, databases, streaming media and more'\n }}</label>\n <button\n type=\"submit\"\n aria-label=\"search\"\n data-testid=\"searchBtn\"\n [ngClass]=\"{ ensign: config.institution === 'ensign' }\"\n >\n <span class=\"material-symbols-outlined ss-icon\"> search </span>\n </button>\n </form>\n\n @if (simpleQuery.value) {\n <button id=\"clear\" (click)=\"clearQuery()\" data-testid=\"clearBtn\">\n <span class=\"material-symbols-outlined ss-icon\"> cancel </span>\n </button>\n }\n</div>\n", styles: ["a,button{border:none;background:none;font-family:inherit;padding:0;margin:0;font-size:inherit;color:#1c7ec9;text-decoration:none;cursor:pointer}a:hover,button:hover{color:#8ab6f0}.ss-container{position:relative}form{display:flex}.ss-icon{font-size:1em}button[type=submit]{background:#fff;border-radius:0 4px 4px 0;margin:0;width:3em;cursor:pointer;display:flex;justify-content:center;align-items:center;transition:color .2s,background-color .2s}button[type=submit] .ss-icon{color:#0047ba;font-size:1.7em}button[type=submit]:hover .ss-icon{color:#6892ca}button[type=submit].ensign .ss-icon{color:#2b6042}button[type=submit].ensign .ss-icon:hover{color:#357551}#clear{position:absolute;right:3em;top:0%;height:100%;display:flex;justify-content:center;align-items:center;padding:0 0 0 .375rem}#clear .ss-icon{height:auto;color:#acacac;transition:color ease-in-out .05s}#clear:hover .ss-icon{color:#666}input[type=text]{background-color:#fff;color:#000;font-family:inherit;border:none;font-weight:600;margin:0;overflow:hidden;cursor:text;width:calc(100% - 3em);font-size:1em;border-radius:4px 0 0 4px;padding:.56em 2em .56em .56em}input[type=text]:focus{outline:none}input[type=text]:valid+label,input[type=text]:focus+label{font-size:.75em;top:-.8em;padding-top:0;padding-bottom:0;pointer-events:none;margin-top:0;margin-left:.56em;cursor:default}input[type=text]:valid+label:before,input[type=text]:focus+label:before{opacity:1}label{cursor:text;transition:all .1s ease-in-out;position:absolute;padding:.5em .28em;margin-left:.28em;left:0;color:#707070;z-index:1;max-width:calc(100% - 3.4em);white-space:nowrap;line-height:normal;overflow:hidden;text-overflow:ellipsis;border-radius:4px}label:before{transition:all .1s ease-in-out;background-color:#fff;content:\"\";position:absolute;inset:0;opacity:0;z-index:-1}\n"] }]
135
73
  }], propDecorators: { searchInput: [{
136
74
  type: ViewChild,
137
75
  args: ['searchInput']
@@ -141,16 +79,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
141
79
  }], config: [{
142
80
  type: Input,
143
81
  args: [{ required: true }]
144
- }], suggestions: [{
145
- type: Input
146
82
  }], simpleSearch: [{
147
83
  type: Output
148
84
  }], clearSimpleSearch: [{
149
85
  type: Output
150
- }], suggest: [{
151
- type: Output
152
- }], onDocumentClick: [{
153
- type: HostListener,
154
- args: ['document:click', ['$event']]
155
86
  }] } });
156
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"simple-search.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/ss-search-bar/simple-search/simple-search.component.ts","../../../../../../projects/components/src/lib/ss-search-bar/simple-search/simple-search.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    ViewChild,\n    inject,\n    HostListener,\n} from '@angular/core';\nimport { FormBuilder, FormControl, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { MatTooltip, MatTooltipModule } from '@angular/material/tooltip';\nimport { Subscription } from 'rxjs';\nimport { startWith, filter, debounceTime, distinctUntilChanged } from 'rxjs/operators';\nimport { MatIconModule } from '@angular/material/icon';\nimport { CommonModule } from '@angular/common';\nimport { SearchConfig } from '../models/search-config.model';\n\n@Component({\n    selector: 'lib-ss-simple-search',\n    standalone: true,\n    imports: [MatTooltipModule, MatIconModule, ReactiveFormsModule, CommonModule],\n    templateUrl: './simple-search.component.html',\n    styleUrls: ['./simple-search.component.scss'],\n})\nexport class SimpleSearchComponent implements OnInit, OnDestroy {\n    private readonly fb = inject(FormBuilder);\n    @ViewChild('searchInput') private searchInput!: ElementRef;\n    // The tooltip is disabled in the template.\n    // It is enabled and displayed if a user searches against an empty query (simple or q1).\n    // Any change to the search field disables the tooltip.\n    @ViewChild(MatTooltip) private inputTooltip!: MatTooltip;\n    private subscription = new Subscription();\n    private _config!: SearchConfig;\n    @Input({ required: true }) set config(config: SearchConfig) {\n        this._config = config;\n        this.setupForm();\n    }\n    get config() {\n        return this._config;\n    }\n    @Input() suggestions: string[] = [];\n    @Output() simpleSearch = new EventEmitter<SearchConfig>();\n    @Output() clearSimpleSearch = new EventEmitter<void>();\n    @Output() suggest = new EventEmitter<SearchConfig>();\n    protected isSubmitted = false;\n    protected showSuggestions = false;\n    protected selectedSuggestionIndex = -1;\n    private skipNextSuggest = false;\n    protected searchForm = this.fb.nonNullable.group({\n        simpleQuery: ['', Validators.required],\n    });\n\n    get simpleQuery() {\n        return this.searchForm.get('simpleQuery') as FormControl;\n    }\n\n    ngOnInit() {\n        // Ensure validation tooltip closes when uses enters a query\n        this.subscription.add(\n            this.simpleQuery.valueChanges\n                .pipe(startWith(''))\n                .pipe(filter(() => !!this.inputTooltip))\n                .subscribe(() => {\n                    this.inputTooltip.disabled = true;\n                }),\n        );\n\n        // Emit suggestion requests with debounce\n        this.subscription.add(\n            this.simpleQuery.valueChanges\n                .pipe(\n                    debounceTime(300),\n                    distinctUntilChanged(),\n                    filter((value) => value.trim().length > 0),\n                )\n                .subscribe((value) => {\n                    if (this.skipNextSuggest) {\n                        this.skipNextSuggest = false;\n                        return;\n                    }\n                    this.suggest.emit({ ...this.config, q: value });\n                    this.showSuggestions = true;\n                    this.selectedSuggestionIndex = -1;\n                }),\n        );\n    }\n\n    ngOnDestroy(): void {\n        this.subscription.unsubscribe();\n    }\n\n    protected emitSimpleSearch = (isSuggestion: boolean = false) => {\n        this.isSubmitted = true;\n        // Don't perform a search unless there is a query\n        if (this.simpleQuery.invalid) {\n            this.showSearchValidationToolTip();\n            return;\n        }\n        this.simpleSearch.emit({\n            ...this.config,\n            q: this.simpleQuery.value,\n            isSuggestion: isSuggestion,\n        });\n        this.isSubmitted = false;\n    };\n\n    protected clearQuery = () => {\n        this.simpleQuery.setValue('');\n        this.clearSimpleSearch.emit();\n        this.hideSuggestions();\n    };\n\n    protected selectSuggestion = (suggestion: string) => {\n        this.skipNextSuggest = true;\n        this.simpleQuery.setValue(suggestion);\n        this.hideSuggestions();\n        this.emitSimpleSearch(true);\n    };\n\n    protected onInputKeydown = (event: KeyboardEvent) => {\n        if (!this.showSuggestions || this.suggestions.length === 0) {\n            return;\n        }\n\n        switch (event.key) {\n            case 'ArrowDown':\n                event.preventDefault();\n                this.selectedSuggestionIndex = Math.min(\n                    this.selectedSuggestionIndex + 1,\n                    this.suggestions.length - 1,\n                );\n                break;\n            case 'ArrowUp':\n                event.preventDefault();\n                this.selectedSuggestionIndex = Math.max(this.selectedSuggestionIndex - 1, -1);\n                break;\n            case 'Enter':\n                if (this.selectedSuggestionIndex >= 0) {\n                    event.preventDefault();\n                    this.selectSuggestion(this.suggestions[this.selectedSuggestionIndex]);\n                }\n                break;\n            case 'Escape':\n                this.hideSuggestions();\n                break;\n        }\n    };\n\n    @HostListener('document:click', ['$event'])\n    onDocumentClick(event: MouseEvent) {\n        const target = event.target as HTMLElement;\n        if (!target.closest('.ss-container')) {\n            this.hideSuggestions();\n        }\n    }\n\n    private setupForm = () => {\n        this.simpleQuery.setValue(\n            this.config.q || this.config.advancedSearchQueryRows[0]?.query || '',\n        );\n    };\n\n    private showSearchValidationToolTip = () => {\n        this.searchInput.nativeElement.focus();\n        this.inputTooltip.disabled = false;\n        this.inputTooltip.show();\n    };\n\n    private hideSuggestions = () => {\n        this.showSuggestions = false;\n        this.selectedSuggestionIndex = -1;\n    };\n}\n","<div class=\"ss-container\">\n    <form [formGroup]=\"searchForm\" (submit)=\"emitSimpleSearch()\" data-testid=\"searchForm\">\n        <input\n            #searchInput\n            data-testid=\"searchInput\"\n            id=\"q\"\n            name=\"q\"\n            type=\"text\"\n            required\n            aria-required=\"true\"\n            aria-label=\"search input\"\n            autocapitalize=\"off\"\n            formControlName=\"simpleQuery\"\n            matTooltip=\"Fill out this field\"\n            [matTooltipPosition]=\"'above'\"\n            [matTooltipDisabled]=\"true\"\n            [attr.aria-invalid]=\"isSubmitted && simpleQuery.invalid\"\n            (keydown)=\"onInputKeydown($event)\"\n        />\n        <label for=\"q\" data-testid=\"label\">{{\n            config.scope === 'local'\n                ? 'Books, media, special collections and more'\n                : 'Ebooks, articles, journals, databases, streaming media and more'\n        }}</label>\n        <button\n            type=\"submit\"\n            aria-label=\"search\"\n            data-testid=\"searchBtn\"\n            [ngClass]=\"{ ensign: config.institution === 'ensign' }\"\n        >\n            <span class=\"material-symbols-outlined ss-icon\"> search </span>\n        </button>\n    </form>\n\n    @if (simpleQuery.value) {\n        <button id=\"clear\" (click)=\"clearQuery()\" data-testid=\"clearBtn\">\n            <span class=\"material-symbols-outlined ss-icon\"> cancel </span>\n        </button>\n    }\n\n    @if (showSuggestions && suggestions.length > 0) {\n        <ul class=\"suggestions-list\" role=\"listbox\" data-testid=\"suggestionsList\">\n            @for (suggestion of suggestions; track suggestion; let i = $index) {\n                <li\n                    class=\"suggestion-item\"\n                    [class.selected]=\"i === selectedSuggestionIndex\"\n                    (click)=\"selectSuggestion(suggestion)\"\n                    (keydown.enter)=\"selectSuggestion(suggestion)\"\n                    (keydown.space)=\"selectSuggestion(suggestion); $event.preventDefault()\"\n                    role=\"option\"\n                    tabindex=\"0\"\n                    [attr.aria-selected]=\"i === selectedSuggestionIndex\"\n                    data-testid=\"suggestionItem\"\n                >\n                    {{ suggestion }}\n                </li>\n            }\n        </ul>\n    }\n</div>\n"]}
87
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"simple-search.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/ss-search-bar/simple-search/simple-search.component.ts","../../../../../../projects/components/src/lib/ss-search-bar/simple-search/simple-search.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    ViewChild,\n    inject,\n} from '@angular/core';\nimport { FormBuilder, FormControl, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { MatTooltip, MatTooltipModule } from '@angular/material/tooltip';\nimport { Subscription } from 'rxjs';\nimport { startWith, filter } from 'rxjs/operators';\nimport { MatIconModule } from '@angular/material/icon';\nimport { CommonModule } from '@angular/common';\nimport { SearchConfig } from '../models/search-config.model';\n\n@Component({\n    selector: 'lib-ss-simple-search',\n    standalone: true,\n    imports: [MatTooltipModule, MatIconModule, ReactiveFormsModule, CommonModule],\n    templateUrl: './simple-search.component.html',\n    styleUrls: ['./simple-search.component.scss'],\n})\nexport class SimpleSearchComponent implements OnInit, OnDestroy {\n    private readonly fb = inject(FormBuilder);\n    @ViewChild('searchInput') private searchInput!: ElementRef;\n    // The tooltip is disabled in the template.\n    // It is enabled and displayed if a user searches against an empty query (simple or q1).\n    // Any change to the search field disables the tooltip.\n    @ViewChild(MatTooltip) private inputTooltip!: MatTooltip;\n    private subscription = new Subscription();\n    private _config!: SearchConfig;\n    @Input({ required: true }) set config(config: SearchConfig) {\n        this._config = config;\n        this.setupForm();\n    }\n    get config() {\n        return this._config;\n    }\n    @Output() simpleSearch = new EventEmitter<SearchConfig>();\n    @Output() clearSimpleSearch = new EventEmitter<void>();\n    protected isSubmitted = false;\n    protected searchForm = this.fb.nonNullable.group({\n        simpleQuery: ['', Validators.required],\n    });\n\n    get simpleQuery() {\n        return this.searchForm.get('simpleQuery') as FormControl;\n    }\n\n    ngOnInit() {\n        // Ensure validation tooltip closes when uses enters a query\n        this.subscription.add(\n            this.simpleQuery.valueChanges\n                .pipe(startWith(''))\n                .pipe(filter(() => !!this.inputTooltip))\n                .subscribe(() => {\n                    this.inputTooltip.disabled = true;\n                }),\n        );\n    }\n\n    ngOnDestroy(): void {\n        this.subscription.unsubscribe();\n    }\n\n    protected emitSimpleSearch = () => {\n        this.isSubmitted = true;\n        // Don't perform a search unless there is a query\n        if (this.simpleQuery.invalid) {\n            this.showSearchValidationToolTip();\n            return;\n        }\n        this.simpleSearch.emit({ ...this.config, q: this.simpleQuery.value });\n        this.isSubmitted = false;\n    };\n\n    protected clearQuery = () => {\n        this.simpleQuery.setValue('');\n        this.clearSimpleSearch.emit();\n    };\n\n    private setupForm = () => {\n        this.simpleQuery.setValue(\n            this.config.q || this.config.advancedSearchQueryRows[0]?.query || '',\n        );\n    };\n\n    private showSearchValidationToolTip = () => {\n        this.searchInput.nativeElement.focus();\n        this.inputTooltip.disabled = false;\n        this.inputTooltip.show();\n    };\n}\n","<div class=\"ss-container\">\n    <form [formGroup]=\"searchForm\" (submit)=\"emitSimpleSearch()\" data-testid=\"searchForm\">\n        <input\n            #searchInput\n            data-testid=\"searchInput\"\n            id=\"q\"\n            name=\"q\"\n            type=\"text\"\n            required\n            aria-required=\"true\"\n            aria-label=\"search input\"\n            autocapitalize=\"off\"\n            formControlName=\"simpleQuery\"\n            matTooltip=\"Fill out this field\"\n            [matTooltipPosition]=\"'above'\"\n            [matTooltipDisabled]=\"true\"\n            [attr.aria-invalid]=\"isSubmitted && simpleQuery.invalid\"\n        />\n        <label for=\"q\" data-testid=\"label\">{{\n            config.scope === 'local'\n                ? 'Books, media, special collections and more'\n                : 'Ebooks, articles, journals, databases, streaming media and more'\n        }}</label>\n        <button\n            type=\"submit\"\n            aria-label=\"search\"\n            data-testid=\"searchBtn\"\n            [ngClass]=\"{ ensign: config.institution === 'ensign' }\"\n        >\n            <span class=\"material-symbols-outlined ss-icon\"> search </span>\n        </button>\n    </form>\n\n    @if (simpleQuery.value) {\n        <button id=\"clear\" (click)=\"clearQuery()\" data-testid=\"clearBtn\">\n            <span class=\"material-symbols-outlined ss-icon\"> cancel </span>\n        </button>\n    }\n</div>\n"]}