@byuhbll/components 5.1.0-beta.1 → 5.2.0-beta.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.
- package/esm2022/lib/header-with-impersonation/header-with-impersonation.component.mjs +89 -2
- package/esm2022/lib/ss-search-bar/models/search-config.model.mjs +1 -1
- package/esm2022/lib/ss-search-bar/simple-search/simple-search.component.mjs +34 -75
- package/esm2022/lib/ss-search-bar/ss-search-bar.component.mjs +3 -9
- package/fesm2022/byuhbll-components.mjs +122 -82
- package/fesm2022/byuhbll-components.mjs.map +1 -1
- package/lib/header-with-impersonation/header-with-impersonation.component.d.ts +19 -2
- package/lib/ss-search-bar/models/search-config.model.d.ts +0 -1
- package/lib/ss-search-bar/simple-search/simple-search.component.d.ts +4 -11
- package/lib/ss-search-bar/ss-search-bar.component.d.ts +1 -3
- package/package.json +1 -1
|
@@ -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"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VhcmNoLWNvbmZpZy5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9zcy1zZWFyY2gtYmFyL21vZGVscy9zZWFyY2gtY29uZmlnLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBZHZhbmNlZFNlYXJjaFF1ZXJ5Um93IH0gZnJvbSAnLi9hZHZhbmNlZC1zZWFyY2gubW9kZWwnO1xuaW1wb3J0IHsgU2VhcmNoU2NvcGUgfSBmcm9tICcuL3NlYXJjaC1zY29wZS5tb2RlbCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VhcmNoQ29uZmlnIHtcbiAgICBpbnN0aXR1dGlvbjogJ2J5dScgfCAnbGF3JyB8ICdlbnNpZ24nO1xuICAgIHNob3dBZHZhbmNlZFNlYXJjaDogYm9vbGVhbjtcbiAgICBzaG93QWR2YW5jZWRTZWFyY2hBc1RleHQ6IGJvb2xlYW47XG4gICAgc2NvcGU6IFNlYXJjaFNjb3BlO1xuICAgIHE6IHN0cmluZztcbiAgICBhZHZhbmNlZFNlYXJjaFF1ZXJ5Um93czogQWR2YW5jZWRTZWFyY2hRdWVyeVJvd1tdO1xuICAgIGxvY2FsQWR2YW5jZWRTZWFyY2g6IHtcbiAgICAgICAgY3JlYXRpb25EYXRlOiBEYXRlO1xuICAgICAgICBzZWxlY3RlZExhbmd1YWdlczogc3RyaW5nW107XG4gICAgICAgIHNlbGVjdGVkUmVzb3VyY2VUeXBlczogc3RyaW5nW107XG4gICAgICAgIHNlbGVjdGVkQ29sbGVjdGlvbnM6IHN0cmluZ1tdO1xuICAgIH07XG4gICAgZXh0ZXJuYWxBZHZhbmNlZFNlYXJjaDoge1xuICAgICAgICBkYXRlUHVibGlzaGVkOiBEYXRlO1xuICAgICAgICBzZWxlY3RlZExhbmd1YWdlczogc3RyaW5nW107XG4gICAgICAgIGxpbWl0UmVzdWx0czoge1xuICAgICAgICAgICAgcGVlclJldmlld2VkOiBib29sZWFuO1xuICAgICAgICB9O1xuICAgICAgICBleHBhbmRSZXN1bHRzOiB7XG4gICAgICAgICAgICBhcHBseUVxdWl2YWxlbnRTdWJqZWN0czogYm9vbGVhbjtcbiAgICAgICAgICAgIGZ1bGxUZXh0OiBib29sZWFuO1xuICAgICAgICB9O1xuICAgIH07XG59XG5cbmludGVyZmFjZSBEYXRlIHtcbiAgICBmcm9tOiBzdHJpbmc7XG4gICAgdG86IHN0cmluZztcbn1cbiJdfQ==
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import { Component, EventEmitter, Input, Output, ViewChild, inject,
|
|
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
|
|
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,78 +13,57 @@ export class SimpleSearchComponent {
|
|
|
13
13
|
constructor() {
|
|
14
14
|
this.fb = inject(FormBuilder);
|
|
15
15
|
this.subscription = new Subscription();
|
|
16
|
-
this.
|
|
16
|
+
this.draftQuery = '';
|
|
17
|
+
this.isPatching = false;
|
|
17
18
|
this.simpleSearch = new EventEmitter();
|
|
18
19
|
this.clearSimpleSearch = new EventEmitter();
|
|
19
|
-
this.suggest = new EventEmitter();
|
|
20
20
|
this.isSubmitted = false;
|
|
21
|
-
this.showSuggestions = false;
|
|
22
|
-
this.selectedSuggestionIndex = -1;
|
|
23
|
-
this.skipNextSuggest = false;
|
|
24
21
|
this.searchForm = this.fb.nonNullable.group({
|
|
25
22
|
simpleQuery: ['', Validators.required],
|
|
26
23
|
});
|
|
27
|
-
this.emitSimpleSearch = (
|
|
24
|
+
this.emitSimpleSearch = () => {
|
|
28
25
|
this.isSubmitted = true;
|
|
29
26
|
// Don't perform a search unless there is a query
|
|
30
27
|
if (this.simpleQuery.invalid) {
|
|
31
28
|
this.showSearchValidationToolTip();
|
|
32
29
|
return;
|
|
33
30
|
}
|
|
34
|
-
this.simpleSearch.emit({
|
|
35
|
-
...this.config,
|
|
36
|
-
q: this.simpleQuery.value,
|
|
37
|
-
isSuggestion: isSuggestion,
|
|
38
|
-
});
|
|
31
|
+
this.simpleSearch.emit({ ...this.config, q: this.simpleQuery.value });
|
|
39
32
|
this.isSubmitted = false;
|
|
40
33
|
};
|
|
41
34
|
this.clearQuery = () => {
|
|
35
|
+
this.isPatching = true;
|
|
42
36
|
this.simpleQuery.setValue('');
|
|
37
|
+
this.isPatching = false;
|
|
38
|
+
this.draftQuery = '';
|
|
43
39
|
this.clearSimpleSearch.emit();
|
|
44
|
-
this.hideSuggestions();
|
|
45
40
|
};
|
|
46
|
-
this.
|
|
47
|
-
this.
|
|
48
|
-
this.simpleQuery.
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
41
|
+
this.setupForm = () => {
|
|
42
|
+
const incoming = this.config.q || this.config.advancedSearchQueryRows[0]?.query || '';
|
|
43
|
+
const current = this.simpleQuery.value ?? '';
|
|
44
|
+
// If we received a real query (submitted/synced), it should win.
|
|
45
|
+
if (incoming) {
|
|
46
|
+
if (incoming !== current) {
|
|
47
|
+
this.isPatching = true;
|
|
48
|
+
this.simpleQuery.setValue(incoming);
|
|
49
|
+
this.isPatching = false;
|
|
50
|
+
}
|
|
51
|
+
this.draftQuery = incoming; // keep draft aligned
|
|
54
52
|
return;
|
|
55
53
|
}
|
|
56
|
-
switch
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
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;
|
|
54
|
+
// No incoming query (tab switch before submit).
|
|
55
|
+
// Preserve what the user already typed (current/draft). Do NOT clear.
|
|
56
|
+
if (!current && this.draftQuery) {
|
|
57
|
+
this.isPatching = true;
|
|
58
|
+
this.simpleQuery.setValue(this.draftQuery);
|
|
59
|
+
this.isPatching = false;
|
|
74
60
|
}
|
|
75
61
|
};
|
|
76
|
-
this.setupForm = () => {
|
|
77
|
-
this.simpleQuery.setValue(this.config.q || this.config.advancedSearchQueryRows[0]?.query || '');
|
|
78
|
-
};
|
|
79
62
|
this.showSearchValidationToolTip = () => {
|
|
80
63
|
this.searchInput.nativeElement.focus();
|
|
81
64
|
this.inputTooltip.disabled = false;
|
|
82
65
|
this.inputTooltip.show();
|
|
83
66
|
};
|
|
84
|
-
this.hideSuggestions = () => {
|
|
85
|
-
this.showSuggestions = false;
|
|
86
|
-
this.selectedSuggestionIndex = -1;
|
|
87
|
-
};
|
|
88
67
|
}
|
|
89
68
|
set config(config) {
|
|
90
69
|
this._config = config;
|
|
@@ -98,40 +77,27 @@ export class SimpleSearchComponent {
|
|
|
98
77
|
}
|
|
99
78
|
ngOnInit() {
|
|
100
79
|
// Ensure validation tooltip closes when uses enters a query
|
|
80
|
+
console.log('>>> Initializing');
|
|
101
81
|
this.subscription.add(this.simpleQuery.valueChanges
|
|
102
82
|
.pipe(startWith(''))
|
|
103
83
|
.pipe(filter(() => !!this.inputTooltip))
|
|
104
|
-
.subscribe(() => {
|
|
84
|
+
.subscribe((v) => {
|
|
105
85
|
this.inputTooltip.disabled = true;
|
|
106
|
-
|
|
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;
|
|
86
|
+
if (this.isPatching)
|
|
113
87
|
return;
|
|
114
|
-
|
|
115
|
-
this.suggest.emit({ ...this.config, q: value });
|
|
116
|
-
this.showSuggestions = true;
|
|
117
|
-
this.selectedSuggestionIndex = -1;
|
|
88
|
+
this.draftQuery = v ?? '';
|
|
118
89
|
}));
|
|
119
90
|
}
|
|
120
91
|
ngOnDestroy() {
|
|
92
|
+
console.log('>>> Destroying');
|
|
121
93
|
this.subscription.unsubscribe();
|
|
122
94
|
}
|
|
123
|
-
onDocumentClick(event) {
|
|
124
|
-
const target = event.target;
|
|
125
|
-
if (!target.closest('.ss-container')) {
|
|
126
|
-
this.hideSuggestions();
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
95
|
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"
|
|
96
|
+
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
97
|
}
|
|
132
98
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: SimpleSearchComponent, decorators: [{
|
|
133
99
|
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
|
|
100
|
+
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
101
|
}], propDecorators: { searchInput: [{
|
|
136
102
|
type: ViewChild,
|
|
137
103
|
args: ['searchInput']
|
|
@@ -141,16 +107,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
141
107
|
}], config: [{
|
|
142
108
|
type: Input,
|
|
143
109
|
args: [{ required: true }]
|
|
144
|
-
}], suggestions: [{
|
|
145
|
-
type: Input
|
|
146
110
|
}], simpleSearch: [{
|
|
147
111
|
type: Output
|
|
148
112
|
}], clearSimpleSearch: [{
|
|
149
113
|
type: Output
|
|
150
|
-
}], suggest: [{
|
|
151
|
-
type: Output
|
|
152
|
-
}], onDocumentClick: [{
|
|
153
|
-
type: HostListener,
|
|
154
|
-
args: ['document:click', ['$event']]
|
|
155
114
|
}] } });
|
|
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"]}
|
|
115
|
+
//# 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    private draftQuery = '';\n    private isPatching = false;\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        console.log('>>> Initializing');\n        this.subscription.add(\n            this.simpleQuery.valueChanges\n                .pipe(startWith(''))\n                .pipe(filter(() => !!this.inputTooltip))\n                .subscribe((v) => {\n                    this.inputTooltip.disabled = true;\n                    if (this.isPatching) return;\n                    this.draftQuery = v ?? '';\n                }),\n        );\n    }\n\n    ngOnDestroy(): void {\n        console.log('>>> Destroying');\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.isPatching = true;\n        this.simpleQuery.setValue('');\n        this.isPatching = false;\n\n        this.draftQuery = '';\n        this.clearSimpleSearch.emit();\n    };\n\n    private setupForm = () => {\n        const incoming = this.config.q || this.config.advancedSearchQueryRows[0]?.query || '';\n        const current = this.simpleQuery.value ?? '';\n\n        // If we received a real query (submitted/synced), it should win.\n        if (incoming) {\n            if (incoming !== current) {\n                this.isPatching = true;\n                this.simpleQuery.setValue(incoming);\n                this.isPatching = false;\n            }\n            this.draftQuery = incoming; // keep draft aligned\n            return;\n        }\n\n        // No incoming query (tab switch before submit).\n        // Preserve what the user already typed (current/draft). Do NOT clear.\n        if (!current && this.draftQuery) {\n            this.isPatching = true;\n            this.simpleQuery.setValue(this.draftQuery);\n            this.isPatching = false;\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"]}
|