@byuhbll/components 4.0.0-alpha.6 → 4.0.0-alpha.7
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/copy-tooltip/copy-tooltip.component.mjs +49 -0
- package/esm2022/lib/hbll-header/impersonate-modal/impersonate-modal.component.mjs +2 -2
- package/esm2022/lib/hbll-header/models/application-access.mjs +7 -0
- package/esm2022/lib/hbll-header/models/person-summary.mjs +15 -0
- package/esm2022/lib/impersonation-banner/impersonation-banner.component.mjs +129 -0
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/byuhbll-components.mjs +174 -6
- package/fesm2022/byuhbll-components.mjs.map +1 -1
- package/lib/copy-tooltip/copy-tooltip.component.d.ts +12 -0
- package/lib/hbll-header/models/application-access.d.ts +15 -0
- package/lib/hbll-header/models/person-summary.d.ts +33 -0
- package/lib/impersonation-banner/impersonation-banner.component.d.ts +30 -0
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { Component, Input, ViewChild } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { MatTooltip, MatTooltipModule } from '@angular/material/tooltip';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/material/tooltip";
|
|
6
|
+
export class CopyTooltipComponent {
|
|
7
|
+
constructor() {
|
|
8
|
+
this.position = 'below';
|
|
9
|
+
this.clipboardCopyMessage = 'Copy';
|
|
10
|
+
this.copyToClipboard = async (text) => {
|
|
11
|
+
// Ensure the tooltip doesn't disappear when button is clicked
|
|
12
|
+
this.tooltip.show();
|
|
13
|
+
if (text) {
|
|
14
|
+
try {
|
|
15
|
+
await navigator.clipboard.writeText(text).then(() => {
|
|
16
|
+
this.clipboardCopyMessage = 'Copied!';
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
catch {
|
|
20
|
+
this.clipboardCopyMessage = 'Failed to copy';
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
else
|
|
24
|
+
this.clipboardCopyMessage = 'Nothing to copy!';
|
|
25
|
+
};
|
|
26
|
+
this.handleMouseLeave = () => {
|
|
27
|
+
// Manually hide in case the user hovers over the tooltip
|
|
28
|
+
this.tooltip.hide();
|
|
29
|
+
// Delay to account for tooltip animation
|
|
30
|
+
setTimeout(() => {
|
|
31
|
+
this.clipboardCopyMessage = 'Copy';
|
|
32
|
+
}, 100);
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CopyTooltipComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
36
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.1.0", type: CopyTooltipComponent, isStandalone: true, selector: "lib-copy-tooltip", inputs: { position: "position", copyText: "copyText" }, viewQueries: [{ propertyName: "tooltip", first: true, predicate: MatTooltip, descendants: true }], ngImport: i0, template: "<div\n (click)=\"copyToClipboard(copyText)\"\n [matTooltip]=\"clipboardCopyMessage\"\n [matTooltipPosition]=\"position\"\n (mouseleave)=\"handleMouseLeave()\"\n data-testid=\"tooltip\"\n>\n <ng-content></ng-content>\n</div>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }] }); }
|
|
37
|
+
}
|
|
38
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: CopyTooltipComponent, decorators: [{
|
|
39
|
+
type: Component,
|
|
40
|
+
args: [{ selector: 'lib-copy-tooltip', standalone: true, imports: [CommonModule, MatTooltipModule], template: "<div\n (click)=\"copyToClipboard(copyText)\"\n [matTooltip]=\"clipboardCopyMessage\"\n [matTooltipPosition]=\"position\"\n (mouseleave)=\"handleMouseLeave()\"\n data-testid=\"tooltip\"\n>\n <ng-content></ng-content>\n</div>\n" }]
|
|
41
|
+
}], propDecorators: { position: [{
|
|
42
|
+
type: Input
|
|
43
|
+
}], copyText: [{
|
|
44
|
+
type: Input
|
|
45
|
+
}], tooltip: [{
|
|
46
|
+
type: ViewChild,
|
|
47
|
+
args: [MatTooltip]
|
|
48
|
+
}] } });
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29weS10b29sdGlwLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9jb3B5LXRvb2x0aXAvY29weS10b29sdGlwLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9jb3B5LXRvb2x0aXAvY29weS10b29sdGlwLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUM1RCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFVBQVUsRUFBRSxnQkFBZ0IsRUFBbUIsTUFBTSwyQkFBMkIsQ0FBQzs7O0FBUzFGLE1BQU0sT0FBTyxvQkFBb0I7SUFQakM7UUFRYSxhQUFRLEdBQW9CLE9BQU8sQ0FBQztRQUl0Qyx5QkFBb0IsR0FBRyxNQUFNLENBQUM7UUFFOUIsb0JBQWUsR0FBRyxLQUFLLEVBQUUsSUFBYSxFQUFFLEVBQUU7WUFDN0MsOERBQThEO1lBQzlELElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDcEIsSUFBSSxJQUFJLEVBQUUsQ0FBQztnQkFDUCxJQUFJLENBQUM7b0JBQ0QsTUFBTSxTQUFTLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFO3dCQUNoRCxJQUFJLENBQUMsb0JBQW9CLEdBQUcsU0FBUyxDQUFDO29CQUMxQyxDQUFDLENBQUMsQ0FBQztnQkFDUCxDQUFDO2dCQUFDLE1BQU0sQ0FBQztvQkFDTCxJQUFJLENBQUMsb0JBQW9CLEdBQUcsZ0JBQWdCLENBQUM7Z0JBQ2pELENBQUM7WUFDTCxDQUFDOztnQkFBTSxJQUFJLENBQUMsb0JBQW9CLEdBQUcsa0JBQWtCLENBQUM7UUFDMUQsQ0FBQyxDQUFDO1FBRUsscUJBQWdCLEdBQUcsR0FBRyxFQUFFO1lBQzNCLHlEQUF5RDtZQUN6RCxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ3BCLHlDQUF5QztZQUN6QyxVQUFVLENBQUMsR0FBRyxFQUFFO2dCQUNaLElBQUksQ0FBQyxvQkFBb0IsR0FBRyxNQUFNLENBQUM7WUFDdkMsQ0FBQyxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBQ1osQ0FBQyxDQUFDO0tBQ0w7OEdBN0JZLG9CQUFvQjtrR0FBcEIsb0JBQW9CLDZLQUdsQixVQUFVLGdEQ2R6Qix5T0FTQSx5RERGYyxZQUFZLDhCQUFFLGdCQUFnQjs7MkZBSS9CLG9CQUFvQjtrQkFQaEMsU0FBUzsrQkFDSSxrQkFBa0IsY0FDaEIsSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLGdCQUFnQixDQUFDOzhCQUtoQyxRQUFRO3NCQUFoQixLQUFLO2dCQUNHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBQ2lCLE9BQU87c0JBQTdCLFNBQVM7dUJBQUMsVUFBVSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIFZpZXdDaGlsZCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IE1hdFRvb2x0aXAsIE1hdFRvb2x0aXBNb2R1bGUsIFRvb2x0aXBQb3NpdGlvbiB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL3Rvb2x0aXAnO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ2xpYi1jb3B5LXRvb2x0aXAnLFxuICAgIHN0YW5kYWxvbmU6IHRydWUsXG4gICAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgTWF0VG9vbHRpcE1vZHVsZV0sXG4gICAgdGVtcGxhdGVVcmw6ICcuL2NvcHktdG9vbHRpcC5jb21wb25lbnQuaHRtbCcsXG4gICAgc3R5bGVVcmxzOiBbJy4vY29weS10b29sdGlwLmNvbXBvbmVudC5zY3NzJ10sXG59KVxuZXhwb3J0IGNsYXNzIENvcHlUb29sdGlwQ29tcG9uZW50IHtcbiAgICBASW5wdXQoKSBwb3NpdGlvbjogVG9vbHRpcFBvc2l0aW9uID0gJ2JlbG93JztcbiAgICBASW5wdXQoKSBjb3B5VGV4dD86IHN0cmluZztcbiAgICBAVmlld0NoaWxkKE1hdFRvb2x0aXApIHRvb2x0aXAhOiBNYXRUb29sdGlwO1xuXG4gICAgcHVibGljIGNsaXBib2FyZENvcHlNZXNzYWdlID0gJ0NvcHknO1xuXG4gICAgcHVibGljIGNvcHlUb0NsaXBib2FyZCA9IGFzeW5jICh0ZXh0Pzogc3RyaW5nKSA9PiB7XG4gICAgICAgIC8vIEVuc3VyZSB0aGUgdG9vbHRpcCBkb2Vzbid0IGRpc2FwcGVhciB3aGVuIGJ1dHRvbiBpcyBjbGlja2VkXG4gICAgICAgIHRoaXMudG9vbHRpcC5zaG93KCk7XG4gICAgICAgIGlmICh0ZXh0KSB7XG4gICAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgICAgIGF3YWl0IG5hdmlnYXRvci5jbGlwYm9hcmQud3JpdGVUZXh0KHRleHQpLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICB0aGlzLmNsaXBib2FyZENvcHlNZXNzYWdlID0gJ0NvcGllZCEnO1xuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfSBjYXRjaCB7XG4gICAgICAgICAgICAgICAgdGhpcy5jbGlwYm9hcmRDb3B5TWVzc2FnZSA9ICdGYWlsZWQgdG8gY29weSc7XG4gICAgICAgICAgICB9XG4gICAgICAgIH0gZWxzZSB0aGlzLmNsaXBib2FyZENvcHlNZXNzYWdlID0gJ05vdGhpbmcgdG8gY29weSEnO1xuICAgIH07XG5cbiAgICBwdWJsaWMgaGFuZGxlTW91c2VMZWF2ZSA9ICgpID0+IHtcbiAgICAgICAgLy8gTWFudWFsbHkgaGlkZSBpbiBjYXNlIHRoZSB1c2VyIGhvdmVycyBvdmVyIHRoZSB0b29sdGlwXG4gICAgICAgIHRoaXMudG9vbHRpcC5oaWRlKCk7XG4gICAgICAgIC8vIERlbGF5IHRvIGFjY291bnQgZm9yIHRvb2x0aXAgYW5pbWF0aW9uXG4gICAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICAgICAgdGhpcy5jbGlwYm9hcmRDb3B5TWVzc2FnZSA9ICdDb3B5JztcbiAgICAgICAgfSwgMTAwKTtcbiAgICB9O1xufVxuIiwiPGRpdlxuICAoY2xpY2spPVwiY29weVRvQ2xpcGJvYXJkKGNvcHlUZXh0KVwiXG4gIFttYXRUb29sdGlwXT1cImNsaXBib2FyZENvcHlNZXNzYWdlXCJcbiAgW21hdFRvb2x0aXBQb3NpdGlvbl09XCJwb3NpdGlvblwiXG4gIChtb3VzZWxlYXZlKT1cImhhbmRsZU1vdXNlTGVhdmUoKVwiXG4gIGRhdGEtdGVzdGlkPVwidG9vbHRpcFwiXG4+XG4gIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbjwvZGl2PlxuIl19
|
|
@@ -153,7 +153,7 @@ export class ImpersonateModalComponent {
|
|
|
153
153
|
set showModal(open) {
|
|
154
154
|
this.isOpen = open;
|
|
155
155
|
if (open) {
|
|
156
|
-
// Set focus on search input shortly after
|
|
156
|
+
// Set focus on search input shortly after opening modal so user notices
|
|
157
157
|
// the input receiving focus.
|
|
158
158
|
setTimeout(() => this.eref.nativeElement.querySelector('#searchInput')?.focus(), 250);
|
|
159
159
|
}
|
|
@@ -185,4 +185,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
185
185
|
type: HostListener,
|
|
186
186
|
args: ['document:keydown', ['$event']]
|
|
187
187
|
}] } });
|
|
188
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impersonate-modal.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/hbll-header/impersonate-modal/impersonate-modal.component.ts","../../../../../../projects/components/src/lib/hbll-header/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\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 openeing modal so user notices\n            // the input receiving focus.\n            setTimeout(() => this.eref.nativeElement.querySelector('#searchInput')?.focus(), 250);\n        }\n    }\n    oidcBaseUri = input.required<string>();\n    oidcDefaultIdp = input.required<string>();\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"]}
|
|
188
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impersonate-modal.component.js","sourceRoot":"","sources":["../../../../../../projects/components/src/lib/hbll-header/impersonate-modal/impersonate-modal.component.ts","../../../../../../projects/components/src/lib/hbll-header/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\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.required<string>();\n    oidcDefaultIdp = input.required<string>();\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"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
export var AccessStatus;
|
|
2
|
+
(function (AccessStatus) {
|
|
3
|
+
AccessStatus["OK"] = "ok";
|
|
4
|
+
AccessStatus["BLOCKED"] = "blocked";
|
|
5
|
+
AccessStatus["NONE"] = "none";
|
|
6
|
+
})(AccessStatus || (AccessStatus = {}));
|
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwbGljYXRpb24tYWNjZXNzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50cy9zcmMvbGliL2hibGwtaGVhZGVyL21vZGVscy9hcHBsaWNhdGlvbi1hY2Nlc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsTUFBTSxDQUFOLElBQVksWUFJWDtBQUpELFdBQVksWUFBWTtJQUNwQix5QkFBUyxDQUFBO0lBQ1QsbUNBQW1CLENBQUE7SUFDbkIsNkJBQWEsQ0FBQTtBQUNqQixDQUFDLEVBSlcsWUFBWSxLQUFaLFlBQVksUUFJdkIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgZW51bSBBY2Nlc3NTdGF0dXMge1xuICAgIE9LID0gJ29rJyxcbiAgICBCTE9DS0VEID0gJ2Jsb2NrZWQnLFxuICAgIE5PTkUgPSAnbm9uZScsXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXBwbGljYXRpb25BY2Nlc3Mge1xuICAgIGxhYmVsOiBzdHJpbmc7XG4gICAgYWNjZXNzOiBBY2Nlc3NTdGF0dXM7XG4gICAgY29kZT86IHN0cmluZztcbiAgICBkZXRhaWw/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQWNjb3VudHNSZXNwb25zZSB7XG4gICAgbmV0SWQ6IHN0cmluZztcbiAgICBhcHBsaWNhdGlvbnM6IFJlY29yZDxzdHJpbmcsIEFwcGxpY2F0aW9uQWNjZXNzPjtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
export var EmployeePositionType;
|
|
2
|
+
(function (EmployeePositionType) {
|
|
3
|
+
EmployeePositionType["STD"] = "std";
|
|
4
|
+
EmployeePositionType["SNL"] = "snl";
|
|
5
|
+
EmployeePositionType["FAC"] = "fac";
|
|
6
|
+
EmployeePositionType["STF"] = "stf";
|
|
7
|
+
EmployeePositionType["TMP"] = "tmp";
|
|
8
|
+
EmployeePositionType["FTC"] = "ftc";
|
|
9
|
+
})(EmployeePositionType || (EmployeePositionType = {}));
|
|
10
|
+
export var UndergradGraduateStatus;
|
|
11
|
+
(function (UndergradGraduateStatus) {
|
|
12
|
+
UndergradGraduateStatus["UNDERGRAD"] = "undergrad";
|
|
13
|
+
UndergradGraduateStatus["GRADUATE"] = "graduate";
|
|
14
|
+
})(UndergradGraduateStatus || (UndergradGraduateStatus = {}));
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGVyc29uLXN1bW1hcnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzL3NyYy9saWIvaGJsbC1oZWFkZXIvbW9kZWxzL3BlcnNvbi1zdW1tYXJ5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBTixJQUFZLG9CQU9YO0FBUEQsV0FBWSxvQkFBb0I7SUFDNUIsbUNBQVcsQ0FBQTtJQUNYLG1DQUFXLENBQUE7SUFDWCxtQ0FBVyxDQUFBO0lBQ1gsbUNBQVcsQ0FBQTtJQUNYLG1DQUFXLENBQUE7SUFDWCxtQ0FBVyxDQUFBO0FBQ2YsQ0FBQyxFQVBXLG9CQUFvQixLQUFwQixvQkFBb0IsUUFPL0I7QUFFRCxNQUFNLENBQU4sSUFBWSx1QkFHWDtBQUhELFdBQVksdUJBQXVCO0lBQy9CLGtEQUF1QixDQUFBO0lBQ3ZCLGdEQUFxQixDQUFBO0FBQ3pCLENBQUMsRUFIVyx1QkFBdUIsS0FBdkIsdUJBQXVCLFFBR2xDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGVudW0gRW1wbG95ZWVQb3NpdGlvblR5cGUge1xuICAgIFNURCA9ICdzdGQnLFxuICAgIFNOTCA9ICdzbmwnLFxuICAgIEZBQyA9ICdmYWMnLFxuICAgIFNURiA9ICdzdGYnLFxuICAgIFRNUCA9ICd0bXAnLFxuICAgIEZUQyA9ICdmdGMnLFxufVxuXG5leHBvcnQgZW51bSBVbmRlcmdyYWRHcmFkdWF0ZVN0YXR1cyB7XG4gICAgVU5ERVJHUkFEID0gJ3VuZGVyZ3JhZCcsXG4gICAgR1JBRFVBVEUgPSAnZ3JhZHVhdGUnLFxufVxuXG5leHBvcnQgaW50ZXJmYWNlIEluZGVwZW5kZW50U3R1ZHlSZXNwb25zZSB7XG4gICAgbGlicmFyeV9pZDogc3RyaW5nO1xuICAgIGlzX2Vucm9sbGVkOiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBlcnNvblN1bW1hcnkge1xuICAgIGJ5dV9pZD86IHN0cmluZztcbiAgICBlbWFpbF9hZGRyZXNzPzogc3RyaW5nO1xuICAgIGZpcnN0X25hbWU6IHN0cmluZztcbiAgICBpc19hZmZpbGlhdGU6IGJvb2xlYW47XG4gICAgaXNfZW1wbG95ZWU6IGJvb2xlYW47XG4gICAgaXNfcmV0aXJlZD86IGJvb2xlYW47XG4gICAgcHJpbWFyeV9wb3NpdGlvbl90eXBlPzogRW1wbG95ZWVQb3NpdGlvblR5cGU7XG4gICAgcHJpbWFyeV9wb3NpdGlvbl90eXBlX2Rpc3BsYXk/OiBzdHJpbmc7XG4gICAgaXNfc3R1ZGVudDogYm9vbGVhbjtcbiAgICB1bmRlcmdyYWRfZ3JhZHVhdGVfc3RhdHVzPzogVW5kZXJncmFkR3JhZHVhdGVTdGF0dXM7XG4gICAgbGFzdF9uYW1lOiBzdHJpbmc7XG4gICAgbGlicmFyeV9pZDogc3RyaW5nO1xuICAgIHByZWZlcnJlZF9maXJzdF9uYW1lOiBzdHJpbmc7XG4gICAgcmVzdHJpY3RlZDogYm9vbGVhbjtcbiAgICB3b3JrZXJfaWQ/OiBzdHJpbmc7XG59XG4iXX0=
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
import { Component, computed, EventEmitter, inject, input, Output } from '@angular/core';
|
|
2
|
+
import { map, switchMap } from 'rxjs/operators';
|
|
3
|
+
import { AccessStatus, } from '../hbll-header/models/application-access';
|
|
4
|
+
import { CommonModule } from '@angular/common';
|
|
5
|
+
import { jwtDecode } from 'jwt-decode';
|
|
6
|
+
import { toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
7
|
+
import { HttpClient } from '@angular/common/http';
|
|
8
|
+
import urlcat from 'urlcat';
|
|
9
|
+
import { combineLatest, of } from 'rxjs';
|
|
10
|
+
import { CopyTooltipComponent } from '../copy-tooltip/copy-tooltip.component';
|
|
11
|
+
import * as i0 from "@angular/core";
|
|
12
|
+
import * as i1 from "@angular/common";
|
|
13
|
+
const USER_PHOTO_URL = 'https://y.byu.edu/ry/ae/prod/person/cgi/personPhoto.cgi?n=';
|
|
14
|
+
const PERSON_SUMMARY_PATH = '/summary/:libraryId/';
|
|
15
|
+
const PERSON_BASE_URI = 'https://apps.lib.byu.edu/person/v2/';
|
|
16
|
+
const INDEPENDENT_STUDY_RESPONSE_PATH = '/independent-study/:libraryId/';
|
|
17
|
+
const PATRON_ACCOUNTS_PATH = '/patron/accounts';
|
|
18
|
+
const LIBRARY_API_BASE_URI = 'https://apps.lib.byu.edu/v1';
|
|
19
|
+
export class ImpersonationBannerComponent {
|
|
20
|
+
constructor() {
|
|
21
|
+
this.http = inject(HttpClient);
|
|
22
|
+
this.accessTokenPayload = input.required();
|
|
23
|
+
this.endImpersonation = new EventEmitter();
|
|
24
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
25
|
+
this.parsedToken = computed(() => this.accessTokenPayload().token ? jwtDecode(this.accessTokenPayload().token) : {});
|
|
26
|
+
this.isImpersonating = computed(() => !!this.parsedToken()['impersonator']);
|
|
27
|
+
this.libraryId = computed(() => this.parsedToken()['library_id']);
|
|
28
|
+
this.user = toSignal(combineLatest([toObservable(this.libraryId), toObservable(this.isImpersonating)]).pipe(switchMap(([libraryId, isImpersonating]) => libraryId && isImpersonating
|
|
29
|
+
? this.http.get(urlcat(PERSON_BASE_URI, PERSON_SUMMARY_PATH, {
|
|
30
|
+
libraryId,
|
|
31
|
+
}), {
|
|
32
|
+
headers: {
|
|
33
|
+
Authorization: `Bearer ${this.accessTokenPayload().token}`,
|
|
34
|
+
},
|
|
35
|
+
})
|
|
36
|
+
: of(null))));
|
|
37
|
+
this.userFullName = computed(() => this.parsedToken()['given_name']
|
|
38
|
+
? `${this.parsedToken()['given_name']} ${this.parsedToken()['family_name']}`
|
|
39
|
+
: 'Unknown');
|
|
40
|
+
this.isRestricted = computed(() => this.user()?.restricted);
|
|
41
|
+
this.userInfoTabs = computed(() => new Map([
|
|
42
|
+
['Net ID', this.parsedToken()['net_id'] ?? ''],
|
|
43
|
+
['BYU ID', this.parsedToken()['byu_id'] ?? ''],
|
|
44
|
+
['Email', this.parsedToken()['email'] ?? ''],
|
|
45
|
+
]));
|
|
46
|
+
this.userPhotoUrl = computed(() => this.parsedToken()['net_id'] ? USER_PHOTO_URL + this.parsedToken()['net_id'] : '');
|
|
47
|
+
this.activityStatus = computed(() => {
|
|
48
|
+
if (this.user()?.is_retired)
|
|
49
|
+
return 'retired';
|
|
50
|
+
else if (this.user()?.is_employee)
|
|
51
|
+
return 'active';
|
|
52
|
+
else if (this.user()?.primary_position_type)
|
|
53
|
+
return 'inactive';
|
|
54
|
+
return null;
|
|
55
|
+
});
|
|
56
|
+
this.employeeStatusDescription = computed(() => {
|
|
57
|
+
if (!this.user()) {
|
|
58
|
+
return null;
|
|
59
|
+
}
|
|
60
|
+
let description = '';
|
|
61
|
+
if (this.user().is_retired) {
|
|
62
|
+
description = 'Retired';
|
|
63
|
+
}
|
|
64
|
+
else if (this.user().is_employee) {
|
|
65
|
+
description = 'Active';
|
|
66
|
+
}
|
|
67
|
+
else if (this.user().primary_position_type) {
|
|
68
|
+
description = 'Inactive';
|
|
69
|
+
}
|
|
70
|
+
if (this.user().primary_position_type_display) {
|
|
71
|
+
description += ` ${this.user().primary_position_type_display} Employee`;
|
|
72
|
+
}
|
|
73
|
+
return description ?? null;
|
|
74
|
+
});
|
|
75
|
+
this.independentStudyStatus = toSignal(combineLatest([toObservable(this.libraryId), toObservable(this.isImpersonating)]).pipe(switchMap(([libraryId, isImpersonating]) => libraryId && isImpersonating
|
|
76
|
+
? this.http
|
|
77
|
+
.get(urlcat(PERSON_BASE_URI, INDEPENDENT_STUDY_RESPONSE_PATH, {
|
|
78
|
+
libraryId,
|
|
79
|
+
}), {
|
|
80
|
+
headers: {
|
|
81
|
+
Authorization: `Bearer ${this.accessTokenPayload().token}`,
|
|
82
|
+
},
|
|
83
|
+
})
|
|
84
|
+
.pipe(map((response) => response?.is_enrolled))
|
|
85
|
+
: of(null))));
|
|
86
|
+
this.accountStatuses = toSignal(toObservable(this.isImpersonating).pipe(switchMap((isImpersonating) => {
|
|
87
|
+
if (!isImpersonating) {
|
|
88
|
+
return of(null);
|
|
89
|
+
}
|
|
90
|
+
return this.http
|
|
91
|
+
.get(urlcat(LIBRARY_API_BASE_URI, PATRON_ACCOUNTS_PATH), {
|
|
92
|
+
headers: {
|
|
93
|
+
Authorization: `Bearer ${this.accessTokenPayload().token}`,
|
|
94
|
+
},
|
|
95
|
+
})
|
|
96
|
+
.pipe(map((accounts) => {
|
|
97
|
+
const accountStatuses = {
|
|
98
|
+
ok: [],
|
|
99
|
+
blocked: [],
|
|
100
|
+
none: [],
|
|
101
|
+
};
|
|
102
|
+
Object.values(accounts.applications).forEach((app) => {
|
|
103
|
+
switch (app.access) {
|
|
104
|
+
case AccessStatus.OK:
|
|
105
|
+
accountStatuses.ok.push(app);
|
|
106
|
+
break;
|
|
107
|
+
case AccessStatus.BLOCKED:
|
|
108
|
+
accountStatuses.blocked.push(app);
|
|
109
|
+
break;
|
|
110
|
+
case AccessStatus.NONE:
|
|
111
|
+
accountStatuses.none.push(app);
|
|
112
|
+
break;
|
|
113
|
+
}
|
|
114
|
+
});
|
|
115
|
+
return accountStatuses;
|
|
116
|
+
}));
|
|
117
|
+
})));
|
|
118
|
+
this.STATUSES = AccessStatus;
|
|
119
|
+
}
|
|
120
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: ImpersonationBannerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
121
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.1.0", type: ImpersonationBannerComponent, isStandalone: true, selector: "lib-impersonation-banner", inputs: { accessTokenPayload: { classPropertyName: "accessTokenPayload", publicName: "accessTokenPayload", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { endImpersonation: "endImpersonation" }, ngImport: i0, template: "@if (isImpersonating()) {\n <div class=\"banner-padding\"></div>\n <div class=\"top-banner\">\n <div class=\"banner-group profile-name-container\">\n <div class=\"profile-avatar\">\n <div class=\"profile-image\">\n <img\n [src]=\"userPhotoUrl()\"\n [alt]=\"userFullName()\"\n alt=\"user photo\"\n onerror=\"this.remove()\"\n />\n </div>\n <span class=\"material-symbols-outlined profile-icon\"> person </span>\n </div>\n <div class=\"profile-name-group\">\n <span class=\"soft\">Impersonating</span>\n <div class=\"profile-name-wrapper\">\n <span class=\"profile-name\">{{ userFullName() }}</span>\n @if (accountStatuses()) {\n <div class=\"application-status-bar\">\n @if (accountStatuses()!.ok.length) {\n <div class=\"application-status-indicator\">\n <span class=\"application-status-ok application-status\">\n <span class=\"material-symbols-outlined icon-checkmark\">\n check\n </span>\n <span class=\"application-count\">{{\n accountStatuses()!.ok.length\n }}</span>\n </span>\n <div class=\"status-tooltip-container\">\n <div class=\"status-tooltip\">\n OK Applications\n <hr />\n @for (\n application of accountStatuses()!.ok;\n track application.label\n ) {\n <div class=\"application-status\">\n <span class=\"material-symbols-outlined\">\n check\n </span>\n <span>{{ application.label }}</span>\n </div>\n }\n </div>\n </div>\n </div>\n }\n @if (accountStatuses()!.blocked.length) {\n <div class=\"application-status-indicator\">\n <span class=\"application-status-blocked application-status\">\n <span class=\"material-symbols-outlined icon-warning\">\n warning\n </span>\n <span class=\"application-count\">{{\n accountStatuses()!.blocked.length\n }}</span>\n </span>\n <div class=\"status-tooltip-container\">\n <div class=\"status-tooltip\">\n Blocked Applications\n <hr />\n @for (\n application of accountStatuses()!.blocked;\n track application.code\n ) {\n <div class=\"application-status\">\n <span\n class=\"material-symbols-outlined icon-warning\"\n >\n warning\n </span>\n <span>{{ application.label }}</span>\n </div>\n }\n </div>\n </div>\n </div>\n }\n @if (accountStatuses()!.none.length) {\n <div class=\"application-status-indicator\">\n <span class=\"application-status-none application-status\">\n <span class=\"material-symbols-outlined icon-lock\">\n lock\n </span>\n <span class=\"application-count\">{{\n accountStatuses()!.none.length\n }}</span>\n </span>\n <div class=\"status-tooltip-container\">\n <div class=\"status-tooltip\">\n No Account\n <hr />\n @for (\n application of accountStatuses()!.none;\n track application.label\n ) {\n <div class=\"application-status\">\n <span class=\"material-symbols-outlined\">\n lock\n </span>\n <span>{{ application.label }}</span>\n </div>\n }\n </div>\n </div>\n </div>\n }\n </div>\n }\n </div>\n </div>\n </div>\n <div class=\"profile-details-container banner-group\">\n @for (detail of userInfoTabs() | keyvalue; track detail.key) {\n <div class=\"profile-detail\">\n <span class=\"soft label\">{{ detail.key }}</span>\n <lib-copy-tooltip [copyText]=\"detail.value\">\n <div class=\"profile-detail-tag white-tag clickable\">\n {{ detail.value || 'Unknown' }}\n </div>\n </lib-copy-tooltip>\n </div>\n }\n </div>\n <div class=\"profile-details-container banner-group\">\n <div class=\"profile-detail\">\n <span class=\"soft label\">Status</span>\n <div class=\"multiple-detail-tags\">\n <div\n class=\"profile-detail-tag color-tag\"\n title=\"{{ employeeStatusDescription() }}\"\n >\n {{ user()?.primary_position_type_display ?? 'Non-employee' }}\n @if (activityStatus()) {\n <span\n class=\"profile-status-circle\"\n [class.status-active]=\"activityStatus() === 'active'\"\n [class.status-inactive]=\"activityStatus() === 'inactive'\"\n [class.status-retired]=\"activityStatus() === 'retired'\"\n ></span>\n }\n </div>\n <div class=\"profile-detail-tag color-tag\">\n {{ (user()?.undergrad_graduate_status | titlecase) || 'Non-student' }}\n </div>\n <div class=\"profile-detail-tag color-tag\">\n {{ independentStudyStatus() ? 'Independent Study' : 'No Ind. Study' }}\n </div>\n </div>\n </div>\n </div>\n <div class=\"spacer\"></div>\n <button class=\"end-impersonation-button shadow\" (click)=\"this.endImpersonation.emit()\">\n <span class=\"material-symbols-outlined icon\"> close </span>\n <span class=\"exit-text\">Exit</span>\n </button>\n </div>\n @if (isRestricted()) {\n <div class=\"restricted-bar-padding\"></div>\n <div class=\"restricted-bar\">\n <span class=\"title\">restricted person</span>\n <span class=\"text\">\n If anyone asks about this person, you are instructed to respond,\n <span class=\"emphasize\">\"We have no records for this person.\"</span>\n </span>\n </div>\n }\n <div class=\"right-border\"></div>\n <div class=\"bottom-border\"></div>\n <div class=\"left-border\"></div>\n}\n", styles: [".top-banner,.bottom-border,.right-border,.left-border{background-color:#9070bf;box-sizing:content-box;border:none;position:fixed;z-index:6000}lib-copy-tooltip{z-index:10000;display:block}.top-banner,.bottom-border{left:0;right:0}.right-border,.left-border{width:7px;top:0;bottom:0}.top-banner{top:0;height:5em;display:flex;justify-content:space-between;width:100%;align-items:center;padding:2.5em;box-sizing:border-box;color:#f3f3f3}.bottom-border{bottom:0;height:7px}.right-border{right:0}.left-border{left:0}.banner-padding{padding-top:5em}.banner-group{display:flex;margin-right:1.6em}.profile-details-container{height:2.4em;display:flex;min-width:0}.profile-details-container .profile-detail{display:flex;flex-flow:column nowrap;align-items:flex-start;justify-content:space-between;margin-right:.6em;font-weight:600;min-width:0}.profile-details-container .profile-detail .label{font-size:.75em;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;min-width:0;max-width:100%}.profile-details-container .profile-detail .profile-detail-tag{font-size:.8em;padding:.2em .4em;border-radius:5px;display:inline-flex;align-items:center;gap:.5em;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;min-width:0;max-width:100%}.profile-details-container .profile-detail .profile-detail-tag:hover{min-width:max-content;z-index:1000}.profile-details-container .profile-detail .profile-detail-tag:last-of-type{margin-right:0}.profile-details-container .profile-detail .multiple-detail-tags{display:flex;flex-flow:row nowrap;min-width:0;max-width:100%}.profile-details-container .profile-detail .multiple-detail-tags .profile-detail-tag{margin-right:.6em}.profile-name-container{display:flex;align-items:center}.profile-name-container .profile-avatar{border-radius:50%;background-color:#fff;padding:0;display:flex;flex-direction:column;justify-content:flex-end;align-items:center;color:#a9a9a9;align-self:center;font-size:15px;overflow:hidden;height:3.5em;width:3.5em;margin-right:.6em;flex-shrink:0}.profile-name-container .profile-avatar .profile-icon{margin:-.25em}.profile-name-container .profile-avatar .profile-image{height:100%;width:100%;object-fit:cover;font-size:1em}.profile-name-container .profile-avatar .profile-image+.profile-icon{display:none;font-size:3.5em}.profile-name-container .profile-avatar .profile-image:empty{display:none}.profile-name-container .profile-avatar .profile-image:empty+.profile-icon{display:block}.profile-name-container .profile-name-group{font-size:1em;font-weight:600}.profile-name-container .profile-name-group .profile-name-wrapper{display:flex;align-items:center;white-space:nowrap}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar{margin-left:1em;opacity:1;display:flex;align-items:center}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-indicator{position:relative;margin-right:.8em}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-indicator .application-count{margin-left:.5em;vertical-align:middle}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-indicator:last-child{margin-right:0}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status:hover+.status-tooltip-container{opacity:1}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-ok{color:#87ed8f}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-ok .icon-checkmark{background-color:#87ed8f;color:#9070bf;border-radius:50%;padding:.2em;font-size:1em;vertical-align:middle}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-blocked{color:#e9ce34}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-blocked .icon-warning{font-size:1.7em;vertical-align:middle}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-blocked .application-count{margin-left:.1em}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-none{color:#f4785b}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-none .icon-lock{background-color:#f4785b;color:#9070bf;border-radius:50%;padding:.2em;font-size:1em;vertical-align:middle}.profile-status-circle{width:6px;height:6px;border-radius:50%;flex-shrink:0}.status-active{background-color:#00e732}.status-retired{background-color:#ffba38}.status-inactive{border:2px white solid}.soft{opacity:.7}.end-impersonation-button{height:2.4em;cursor:pointer;border-radius:2em;font-size:1em;padding:0 1.4em 0 1em;border:none;outline:none;flex-shrink:0;background-color:#fff;color:#c63d3d;font-weight:600;transition:box-shadow .2s;display:flex;align-items:center}.end-impersonation-button:hover{box-shadow:#0003 0 8px 10px}.end-impersonation-button .icon{font-size:1.2em;margin-right:.2em}.spacer{flex-grow:1}.shadow{box-shadow:#0003 0 2px 10px}.white-tag{cursor:pointer;background-color:#ffffff26}.white-tag:hover{background-color:#614979}.color-tag:nth-of-type(1){background-color:#26acffcc}.color-tag:nth-of-type(2){background-color:#9394ffcc}.color-tag:nth-of-type(3){background-color:#a28ceecc}.status-tooltip-container{color:#fff;background-color:#2e2e2e;opacity:0;transition:all .1s ease-in-out;position:absolute;top:calc(100% + 1em);left:50%;transform:translate(-50%);border-radius:.4em}.status-tooltip-container:before{content:\"\";position:absolute;top:-.5em;left:50%;transform:translate(-50%);border-style:solid;border-width:0 1em 1em 1em;border-color:transparent transparent rgb(46,46,46) transparent}.status-tooltip{text-align:left;font-size:1.4em;padding:.8em;line-height:2em;width:100%;font-weight:700}.status-tooltip .application-status{display:flex;align-items:center;font-weight:400}.status-tooltip .application-status .icon-checkmark{color:#87ed8f}.status-tooltip .application-status .icon-warning{color:#e9ce34}.status-tooltip .application-status .icon-lock{color:#f4785b}.restricted-bar-padding{padding-top:40px}.restricted-bar{position:fixed;top:5em;left:7px;right:7px;height:40px;display:flex;align-items:center;color:#f3f3f3;background:repeating-linear-gradient(315deg,#c23737,#c23737 35px,#c25050 35px 70px);z-index:6000;font-size:1.6em}.restricted-bar .title{text-transform:uppercase;font-weight:600;margin:0 2em}.restricted-bar .text{margin:0 1em}.restricted-bar .text .emphasize{font-weight:600;font-style:italic}@media screen and (max-width: 1100px){.profile-name-group .soft{font-size:.7em}.profile-name-group .profile-name{font-size:1.3em}.end-impersonation-button{border-radius:50%;height:auto;padding:.6em}.end-impersonation-button .icon{margin:0}.end-impersonation-button .exit-text{display:none}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "pipe", type: i1.TitleCasePipe, name: "titlecase" }, { kind: "pipe", type: i1.KeyValuePipe, name: "keyvalue" }, { kind: "component", type: CopyTooltipComponent, selector: "lib-copy-tooltip", inputs: ["position", "copyText"] }] }); }
|
|
122
|
+
}
|
|
123
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImport: i0, type: ImpersonationBannerComponent, decorators: [{
|
|
124
|
+
type: Component,
|
|
125
|
+
args: [{ standalone: true, imports: [CommonModule, CopyTooltipComponent], selector: 'lib-impersonation-banner', template: "@if (isImpersonating()) {\n <div class=\"banner-padding\"></div>\n <div class=\"top-banner\">\n <div class=\"banner-group profile-name-container\">\n <div class=\"profile-avatar\">\n <div class=\"profile-image\">\n <img\n [src]=\"userPhotoUrl()\"\n [alt]=\"userFullName()\"\n alt=\"user photo\"\n onerror=\"this.remove()\"\n />\n </div>\n <span class=\"material-symbols-outlined profile-icon\"> person </span>\n </div>\n <div class=\"profile-name-group\">\n <span class=\"soft\">Impersonating</span>\n <div class=\"profile-name-wrapper\">\n <span class=\"profile-name\">{{ userFullName() }}</span>\n @if (accountStatuses()) {\n <div class=\"application-status-bar\">\n @if (accountStatuses()!.ok.length) {\n <div class=\"application-status-indicator\">\n <span class=\"application-status-ok application-status\">\n <span class=\"material-symbols-outlined icon-checkmark\">\n check\n </span>\n <span class=\"application-count\">{{\n accountStatuses()!.ok.length\n }}</span>\n </span>\n <div class=\"status-tooltip-container\">\n <div class=\"status-tooltip\">\n OK Applications\n <hr />\n @for (\n application of accountStatuses()!.ok;\n track application.label\n ) {\n <div class=\"application-status\">\n <span class=\"material-symbols-outlined\">\n check\n </span>\n <span>{{ application.label }}</span>\n </div>\n }\n </div>\n </div>\n </div>\n }\n @if (accountStatuses()!.blocked.length) {\n <div class=\"application-status-indicator\">\n <span class=\"application-status-blocked application-status\">\n <span class=\"material-symbols-outlined icon-warning\">\n warning\n </span>\n <span class=\"application-count\">{{\n accountStatuses()!.blocked.length\n }}</span>\n </span>\n <div class=\"status-tooltip-container\">\n <div class=\"status-tooltip\">\n Blocked Applications\n <hr />\n @for (\n application of accountStatuses()!.blocked;\n track application.code\n ) {\n <div class=\"application-status\">\n <span\n class=\"material-symbols-outlined icon-warning\"\n >\n warning\n </span>\n <span>{{ application.label }}</span>\n </div>\n }\n </div>\n </div>\n </div>\n }\n @if (accountStatuses()!.none.length) {\n <div class=\"application-status-indicator\">\n <span class=\"application-status-none application-status\">\n <span class=\"material-symbols-outlined icon-lock\">\n lock\n </span>\n <span class=\"application-count\">{{\n accountStatuses()!.none.length\n }}</span>\n </span>\n <div class=\"status-tooltip-container\">\n <div class=\"status-tooltip\">\n No Account\n <hr />\n @for (\n application of accountStatuses()!.none;\n track application.label\n ) {\n <div class=\"application-status\">\n <span class=\"material-symbols-outlined\">\n lock\n </span>\n <span>{{ application.label }}</span>\n </div>\n }\n </div>\n </div>\n </div>\n }\n </div>\n }\n </div>\n </div>\n </div>\n <div class=\"profile-details-container banner-group\">\n @for (detail of userInfoTabs() | keyvalue; track detail.key) {\n <div class=\"profile-detail\">\n <span class=\"soft label\">{{ detail.key }}</span>\n <lib-copy-tooltip [copyText]=\"detail.value\">\n <div class=\"profile-detail-tag white-tag clickable\">\n {{ detail.value || 'Unknown' }}\n </div>\n </lib-copy-tooltip>\n </div>\n }\n </div>\n <div class=\"profile-details-container banner-group\">\n <div class=\"profile-detail\">\n <span class=\"soft label\">Status</span>\n <div class=\"multiple-detail-tags\">\n <div\n class=\"profile-detail-tag color-tag\"\n title=\"{{ employeeStatusDescription() }}\"\n >\n {{ user()?.primary_position_type_display ?? 'Non-employee' }}\n @if (activityStatus()) {\n <span\n class=\"profile-status-circle\"\n [class.status-active]=\"activityStatus() === 'active'\"\n [class.status-inactive]=\"activityStatus() === 'inactive'\"\n [class.status-retired]=\"activityStatus() === 'retired'\"\n ></span>\n }\n </div>\n <div class=\"profile-detail-tag color-tag\">\n {{ (user()?.undergrad_graduate_status | titlecase) || 'Non-student' }}\n </div>\n <div class=\"profile-detail-tag color-tag\">\n {{ independentStudyStatus() ? 'Independent Study' : 'No Ind. Study' }}\n </div>\n </div>\n </div>\n </div>\n <div class=\"spacer\"></div>\n <button class=\"end-impersonation-button shadow\" (click)=\"this.endImpersonation.emit()\">\n <span class=\"material-symbols-outlined icon\"> close </span>\n <span class=\"exit-text\">Exit</span>\n </button>\n </div>\n @if (isRestricted()) {\n <div class=\"restricted-bar-padding\"></div>\n <div class=\"restricted-bar\">\n <span class=\"title\">restricted person</span>\n <span class=\"text\">\n If anyone asks about this person, you are instructed to respond,\n <span class=\"emphasize\">\"We have no records for this person.\"</span>\n </span>\n </div>\n }\n <div class=\"right-border\"></div>\n <div class=\"bottom-border\"></div>\n <div class=\"left-border\"></div>\n}\n", styles: [".top-banner,.bottom-border,.right-border,.left-border{background-color:#9070bf;box-sizing:content-box;border:none;position:fixed;z-index:6000}lib-copy-tooltip{z-index:10000;display:block}.top-banner,.bottom-border{left:0;right:0}.right-border,.left-border{width:7px;top:0;bottom:0}.top-banner{top:0;height:5em;display:flex;justify-content:space-between;width:100%;align-items:center;padding:2.5em;box-sizing:border-box;color:#f3f3f3}.bottom-border{bottom:0;height:7px}.right-border{right:0}.left-border{left:0}.banner-padding{padding-top:5em}.banner-group{display:flex;margin-right:1.6em}.profile-details-container{height:2.4em;display:flex;min-width:0}.profile-details-container .profile-detail{display:flex;flex-flow:column nowrap;align-items:flex-start;justify-content:space-between;margin-right:.6em;font-weight:600;min-width:0}.profile-details-container .profile-detail .label{font-size:.75em;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;min-width:0;max-width:100%}.profile-details-container .profile-detail .profile-detail-tag{font-size:.8em;padding:.2em .4em;border-radius:5px;display:inline-flex;align-items:center;gap:.5em;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;min-width:0;max-width:100%}.profile-details-container .profile-detail .profile-detail-tag:hover{min-width:max-content;z-index:1000}.profile-details-container .profile-detail .profile-detail-tag:last-of-type{margin-right:0}.profile-details-container .profile-detail .multiple-detail-tags{display:flex;flex-flow:row nowrap;min-width:0;max-width:100%}.profile-details-container .profile-detail .multiple-detail-tags .profile-detail-tag{margin-right:.6em}.profile-name-container{display:flex;align-items:center}.profile-name-container .profile-avatar{border-radius:50%;background-color:#fff;padding:0;display:flex;flex-direction:column;justify-content:flex-end;align-items:center;color:#a9a9a9;align-self:center;font-size:15px;overflow:hidden;height:3.5em;width:3.5em;margin-right:.6em;flex-shrink:0}.profile-name-container .profile-avatar .profile-icon{margin:-.25em}.profile-name-container .profile-avatar .profile-image{height:100%;width:100%;object-fit:cover;font-size:1em}.profile-name-container .profile-avatar .profile-image+.profile-icon{display:none;font-size:3.5em}.profile-name-container .profile-avatar .profile-image:empty{display:none}.profile-name-container .profile-avatar .profile-image:empty+.profile-icon{display:block}.profile-name-container .profile-name-group{font-size:1em;font-weight:600}.profile-name-container .profile-name-group .profile-name-wrapper{display:flex;align-items:center;white-space:nowrap}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar{margin-left:1em;opacity:1;display:flex;align-items:center}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-indicator{position:relative;margin-right:.8em}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-indicator .application-count{margin-left:.5em;vertical-align:middle}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-indicator:last-child{margin-right:0}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status:hover+.status-tooltip-container{opacity:1}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-ok{color:#87ed8f}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-ok .icon-checkmark{background-color:#87ed8f;color:#9070bf;border-radius:50%;padding:.2em;font-size:1em;vertical-align:middle}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-blocked{color:#e9ce34}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-blocked .icon-warning{font-size:1.7em;vertical-align:middle}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-blocked .application-count{margin-left:.1em}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-none{color:#f4785b}.profile-name-container .profile-name-group .profile-name-wrapper .application-status-bar .application-status-none .icon-lock{background-color:#f4785b;color:#9070bf;border-radius:50%;padding:.2em;font-size:1em;vertical-align:middle}.profile-status-circle{width:6px;height:6px;border-radius:50%;flex-shrink:0}.status-active{background-color:#00e732}.status-retired{background-color:#ffba38}.status-inactive{border:2px white solid}.soft{opacity:.7}.end-impersonation-button{height:2.4em;cursor:pointer;border-radius:2em;font-size:1em;padding:0 1.4em 0 1em;border:none;outline:none;flex-shrink:0;background-color:#fff;color:#c63d3d;font-weight:600;transition:box-shadow .2s;display:flex;align-items:center}.end-impersonation-button:hover{box-shadow:#0003 0 8px 10px}.end-impersonation-button .icon{font-size:1.2em;margin-right:.2em}.spacer{flex-grow:1}.shadow{box-shadow:#0003 0 2px 10px}.white-tag{cursor:pointer;background-color:#ffffff26}.white-tag:hover{background-color:#614979}.color-tag:nth-of-type(1){background-color:#26acffcc}.color-tag:nth-of-type(2){background-color:#9394ffcc}.color-tag:nth-of-type(3){background-color:#a28ceecc}.status-tooltip-container{color:#fff;background-color:#2e2e2e;opacity:0;transition:all .1s ease-in-out;position:absolute;top:calc(100% + 1em);left:50%;transform:translate(-50%);border-radius:.4em}.status-tooltip-container:before{content:\"\";position:absolute;top:-.5em;left:50%;transform:translate(-50%);border-style:solid;border-width:0 1em 1em 1em;border-color:transparent transparent rgb(46,46,46) transparent}.status-tooltip{text-align:left;font-size:1.4em;padding:.8em;line-height:2em;width:100%;font-weight:700}.status-tooltip .application-status{display:flex;align-items:center;font-weight:400}.status-tooltip .application-status .icon-checkmark{color:#87ed8f}.status-tooltip .application-status .icon-warning{color:#e9ce34}.status-tooltip .application-status .icon-lock{color:#f4785b}.restricted-bar-padding{padding-top:40px}.restricted-bar{position:fixed;top:5em;left:7px;right:7px;height:40px;display:flex;align-items:center;color:#f3f3f3;background:repeating-linear-gradient(315deg,#c23737,#c23737 35px,#c25050 35px 70px);z-index:6000;font-size:1.6em}.restricted-bar .title{text-transform:uppercase;font-weight:600;margin:0 2em}.restricted-bar .text{margin:0 1em}.restricted-bar .text .emphasize{font-weight:600;font-style:italic}@media screen and (max-width: 1100px){.profile-name-group .soft{font-size:.7em}.profile-name-group .profile-name{font-size:1.3em}.end-impersonation-button{border-radius:50%;height:auto;padding:.6em}.end-impersonation-button .icon{margin:0}.end-impersonation-button .exit-text{display:none}}\n"] }]
|
|
126
|
+
}], propDecorators: { endImpersonation: [{
|
|
127
|
+
type: Output
|
|
128
|
+
}] } });
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impersonation-banner.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/impersonation-banner/impersonation-banner.component.ts","../../../../../projects/components/src/lib/impersonation-banner/impersonation-banner.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, computed, EventEmitter, inject, input, Output, Signal } from '@angular/core';\nimport { map, switchMap } from 'rxjs/operators';\nimport {\n    AccessStatus,\n    AccountsResponse,\n    type ApplicationAccess,\n} from '../hbll-header/models/application-access';\nimport { CommonModule } from '@angular/common';\nimport { TokenPayload } from '../hbll-header/models/token-payload';\nimport { jwtDecode, JwtPayload } from 'jwt-decode';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { HttpClient } from '@angular/common/http';\nimport { IndependentStudyResponse, PersonSummary } from '../hbll-header/models/person-summary';\nimport urlcat from 'urlcat';\nimport { combineLatest, of } from 'rxjs';\nimport { CopyTooltipComponent } from '../copy-tooltip/copy-tooltip.component';\n\nconst USER_PHOTO_URL = 'https://y.byu.edu/ry/ae/prod/person/cgi/personPhoto.cgi?n=';\nconst PERSON_SUMMARY_PATH = '/summary/:libraryId/';\nconst PERSON_BASE_URI = 'https://apps.lib.byu.edu/person/v2/';\nconst INDEPENDENT_STUDY_RESPONSE_PATH = '/independent-study/:libraryId/';\nconst PATRON_ACCOUNTS_PATH = '/patron/accounts';\nconst LIBRARY_API_BASE_URI = 'https://apps.lib.byu.edu/v1';\n\n@Component({\n    standalone: true,\n    imports: [CommonModule, CopyTooltipComponent],\n    selector: 'lib-impersonation-banner',\n    templateUrl: './impersonation-banner.component.html',\n    styleUrls: ['./impersonation-banner.component.scss'],\n})\nexport class ImpersonationBannerComponent {\n    private http = inject(HttpClient);\n    accessTokenPayload = input.required<TokenPayload>();\n    @Output() endImpersonation = new EventEmitter<void>();\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    protected parsedToken: Signal<JwtPayload & Record<string, any>> = computed(() =>\n        this.accessTokenPayload().token ? jwtDecode(this.accessTokenPayload().token) : {},\n    );\n    protected isImpersonating = computed(() => !!this.parsedToken()['impersonator']);\n    protected libraryId = computed(() => this.parsedToken()['library_id']);\n    protected user = toSignal(\n        combineLatest([toObservable(this.libraryId), toObservable(this.isImpersonating)]).pipe(\n            switchMap(([libraryId, isImpersonating]) =>\n                libraryId && isImpersonating\n                    ? this.http.get<PersonSummary>(\n                          urlcat(PERSON_BASE_URI, PERSON_SUMMARY_PATH, {\n                              libraryId,\n                          }),\n                          {\n                              headers: {\n                                  Authorization: `Bearer ${this.accessTokenPayload().token}`,\n                              },\n                          },\n                      )\n                    : of(null),\n            ),\n        ),\n    );\n    protected userFullName = computed(() =>\n        this.parsedToken()['given_name']\n            ? `${this.parsedToken()['given_name']} ${this.parsedToken()['family_name']}`\n            : 'Unknown',\n    );\n    protected isRestricted = computed(() => this.user()?.restricted);\n    protected userInfoTabs = computed(\n        () =>\n            new Map([\n                ['Net ID', this.parsedToken()['net_id'] ?? ''],\n                ['BYU ID', this.parsedToken()['byu_id'] ?? ''],\n                ['Email', this.parsedToken()['email'] ?? ''],\n            ]),\n    );\n    protected userPhotoUrl = computed(() =>\n        this.parsedToken()['net_id'] ? USER_PHOTO_URL + this.parsedToken()['net_id'] : '',\n    );\n\n    protected activityStatus = computed(() => {\n        if (this.user()?.is_retired) return 'retired';\n        else if (this.user()?.is_employee) return 'active';\n        else if (this.user()?.primary_position_type) return 'inactive';\n        return null;\n    });\n\n    protected employeeStatusDescription = computed(() => {\n        if (!this.user()) {\n            return null;\n        }\n\n        let description = '';\n        if (this.user()!.is_retired) {\n            description = 'Retired';\n        } else if (this.user()!.is_employee) {\n            description = 'Active';\n        } else if (this.user()!.primary_position_type) {\n            description = 'Inactive';\n        }\n\n        if (this.user()!.primary_position_type_display) {\n            description += ` ${this.user()!.primary_position_type_display} Employee`;\n        }\n\n        return description ?? null;\n    });\n\n    protected independentStudyStatus = toSignal(\n        combineLatest([toObservable(this.libraryId), toObservable(this.isImpersonating)]).pipe(\n            switchMap(([libraryId, isImpersonating]) =>\n                libraryId && isImpersonating\n                    ? this.http\n                          .get<IndependentStudyResponse>(\n                              urlcat(PERSON_BASE_URI, INDEPENDENT_STUDY_RESPONSE_PATH, {\n                                  libraryId,\n                              }),\n                              {\n                                  headers: {\n                                      Authorization: `Bearer ${this.accessTokenPayload().token}`,\n                                  },\n                              },\n                          )\n                          .pipe(map((response) => response?.is_enrolled))\n                    : of(null),\n            ),\n        ),\n    );\n    protected accountStatuses = toSignal(\n        toObservable(this.isImpersonating).pipe(\n            switchMap((isImpersonating) => {\n                if (!isImpersonating) {\n                    return of(null);\n                }\n                return this.http\n                    .get<AccountsResponse>(urlcat(LIBRARY_API_BASE_URI, PATRON_ACCOUNTS_PATH), {\n                        headers: {\n                            Authorization: `Bearer ${this.accessTokenPayload().token}`,\n                        },\n                    })\n                    .pipe(\n                        map((accounts) => {\n                            const accountStatuses = {\n                                ok: [] as ApplicationAccess[],\n                                blocked: [] as ApplicationAccess[],\n                                none: [] as ApplicationAccess[],\n                            };\n\n                            Object.values(accounts.applications).forEach((app) => {\n                                switch (app.access) {\n                                    case AccessStatus.OK:\n                                        accountStatuses.ok.push(app);\n                                        break;\n                                    case AccessStatus.BLOCKED:\n                                        accountStatuses.blocked.push(app);\n                                        break;\n                                    case AccessStatus.NONE:\n                                        accountStatuses.none.push(app);\n                                        break;\n                                }\n                            });\n\n                            return accountStatuses;\n                        }),\n                    );\n            }),\n        ),\n    );\n\n    protected STATUSES = AccessStatus;\n}\n","@if (isImpersonating()) {\n    <div class=\"banner-padding\"></div>\n    <div class=\"top-banner\">\n        <div class=\"banner-group profile-name-container\">\n            <div class=\"profile-avatar\">\n                <div class=\"profile-image\">\n                    <img\n                        [src]=\"userPhotoUrl()\"\n                        [alt]=\"userFullName()\"\n                        alt=\"user photo\"\n                        onerror=\"this.remove()\"\n                    />\n                </div>\n                <span class=\"material-symbols-outlined profile-icon\"> person </span>\n            </div>\n            <div class=\"profile-name-group\">\n                <span class=\"soft\">Impersonating</span>\n                <div class=\"profile-name-wrapper\">\n                    <span class=\"profile-name\">{{ userFullName() }}</span>\n                    @if (accountStatuses()) {\n                        <div class=\"application-status-bar\">\n                            @if (accountStatuses()!.ok.length) {\n                                <div class=\"application-status-indicator\">\n                                    <span class=\"application-status-ok application-status\">\n                                        <span class=\"material-symbols-outlined icon-checkmark\">\n                                            check\n                                        </span>\n                                        <span class=\"application-count\">{{\n                                            accountStatuses()!.ok.length\n                                        }}</span>\n                                    </span>\n                                    <div class=\"status-tooltip-container\">\n                                        <div class=\"status-tooltip\">\n                                            OK Applications\n                                            <hr />\n                                            @for (\n                                                application of accountStatuses()!.ok;\n                                                track application.label\n                                            ) {\n                                                <div class=\"application-status\">\n                                                    <span class=\"material-symbols-outlined\">\n                                                        check\n                                                    </span>\n                                                    <span>{{ application.label }}</span>\n                                                </div>\n                                            }\n                                        </div>\n                                    </div>\n                                </div>\n                            }\n                            @if (accountStatuses()!.blocked.length) {\n                                <div class=\"application-status-indicator\">\n                                    <span class=\"application-status-blocked application-status\">\n                                        <span class=\"material-symbols-outlined icon-warning\">\n                                            warning\n                                        </span>\n                                        <span class=\"application-count\">{{\n                                            accountStatuses()!.blocked.length\n                                        }}</span>\n                                    </span>\n                                    <div class=\"status-tooltip-container\">\n                                        <div class=\"status-tooltip\">\n                                            Blocked Applications\n                                            <hr />\n                                            @for (\n                                                application of accountStatuses()!.blocked;\n                                                track application.code\n                                            ) {\n                                                <div class=\"application-status\">\n                                                    <span\n                                                        class=\"material-symbols-outlined icon-warning\"\n                                                    >\n                                                        warning\n                                                    </span>\n                                                    <span>{{ application.label }}</span>\n                                                </div>\n                                            }\n                                        </div>\n                                    </div>\n                                </div>\n                            }\n                            @if (accountStatuses()!.none.length) {\n                                <div class=\"application-status-indicator\">\n                                    <span class=\"application-status-none application-status\">\n                                        <span class=\"material-symbols-outlined icon-lock\">\n                                            lock\n                                        </span>\n                                        <span class=\"application-count\">{{\n                                            accountStatuses()!.none.length\n                                        }}</span>\n                                    </span>\n                                    <div class=\"status-tooltip-container\">\n                                        <div class=\"status-tooltip\">\n                                            No Account\n                                            <hr />\n                                            @for (\n                                                application of accountStatuses()!.none;\n                                                track application.label\n                                            ) {\n                                                <div class=\"application-status\">\n                                                    <span class=\"material-symbols-outlined\">\n                                                        lock\n                                                    </span>\n                                                    <span>{{ application.label }}</span>\n                                                </div>\n                                            }\n                                        </div>\n                                    </div>\n                                </div>\n                            }\n                        </div>\n                    }\n                </div>\n            </div>\n        </div>\n        <div class=\"profile-details-container banner-group\">\n            @for (detail of userInfoTabs() | keyvalue; track detail.key) {\n                <div class=\"profile-detail\">\n                    <span class=\"soft label\">{{ detail.key }}</span>\n                    <lib-copy-tooltip [copyText]=\"detail.value\">\n                        <div class=\"profile-detail-tag white-tag clickable\">\n                            {{ detail.value || 'Unknown' }}\n                        </div>\n                    </lib-copy-tooltip>\n                </div>\n            }\n        </div>\n        <div class=\"profile-details-container banner-group\">\n            <div class=\"profile-detail\">\n                <span class=\"soft label\">Status</span>\n                <div class=\"multiple-detail-tags\">\n                    <div\n                        class=\"profile-detail-tag color-tag\"\n                        title=\"{{ employeeStatusDescription() }}\"\n                    >\n                        {{ user()?.primary_position_type_display ?? 'Non-employee' }}\n                        @if (activityStatus()) {\n                            <span\n                                class=\"profile-status-circle\"\n                                [class.status-active]=\"activityStatus() === 'active'\"\n                                [class.status-inactive]=\"activityStatus() === 'inactive'\"\n                                [class.status-retired]=\"activityStatus() === 'retired'\"\n                            ></span>\n                        }\n                    </div>\n                    <div class=\"profile-detail-tag color-tag\">\n                        {{ (user()?.undergrad_graduate_status | titlecase) || 'Non-student' }}\n                    </div>\n                    <div class=\"profile-detail-tag color-tag\">\n                        {{ independentStudyStatus() ? 'Independent Study' : 'No Ind. Study' }}\n                    </div>\n                </div>\n            </div>\n        </div>\n        <div class=\"spacer\"></div>\n        <button class=\"end-impersonation-button shadow\" (click)=\"this.endImpersonation.emit()\">\n            <span class=\"material-symbols-outlined icon\"> close </span>\n            <span class=\"exit-text\">Exit</span>\n        </button>\n    </div>\n    @if (isRestricted()) {\n        <div class=\"restricted-bar-padding\"></div>\n        <div class=\"restricted-bar\">\n            <span class=\"title\">restricted person</span>\n            <span class=\"text\">\n                If anyone asks about this person, you are instructed to respond,\n                <span class=\"emphasize\">\"We have no records for this person.\"</span>\n            </span>\n        </div>\n    }\n    <div class=\"right-border\"></div>\n    <div class=\"bottom-border\"></div>\n    <div class=\"left-border\"></div>\n}\n"]}
|
package/esm2022/public-api.mjs
CHANGED
|
@@ -2,9 +2,10 @@
|
|
|
2
2
|
* Public API Surface of components
|
|
3
3
|
*/
|
|
4
4
|
export * from './lib/hbll-header/hbll-header.component';
|
|
5
|
+
export * from './lib/impersonation-banner/impersonation-banner.component';
|
|
5
6
|
export * from './lib/ss-search-bar/ss-search-bar.component';
|
|
6
7
|
export * from './lib/ss-search-bar/models/advanced-search.model';
|
|
7
8
|
export * from './lib/ss-search-bar/models/search-scope.model';
|
|
8
9
|
export * from './lib/ss-search-bar/models/search-config.model';
|
|
9
10
|
export { ADVANCED_SEARCH_QUALIFIER_MAP, ADVANCED_SEARCH_FIELD_MAP, ADVANCED_SEARCH_OPTIONS, } from './lib/ss-search-bar/constants';
|
|
10
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL3B1YmxpYy1hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLHlDQUF5QyxDQUFDO0FBQ3hELGNBQWMsMkRBQTJELENBQUM7QUFDMUUsY0FBYyw2Q0FBNkMsQ0FBQztBQUM1RCxjQUFjLGtEQUFrRCxDQUFDO0FBQ2pFLGNBQWMsK0NBQStDLENBQUM7QUFDOUQsY0FBYyxnREFBZ0QsQ0FBQztBQUMvRCxPQUFPLEVBQ0gsNkJBQTZCLEVBQzdCLHlCQUF5QixFQUN6Qix1QkFBdUIsR0FDMUIsTUFBTSwrQkFBK0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgY29tcG9uZW50c1xuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vbGliL2hibGwtaGVhZGVyL2hibGwtaGVhZGVyLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9pbXBlcnNvbmF0aW9uLWJhbm5lci9pbXBlcnNvbmF0aW9uLWJhbm5lci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvc3Mtc2VhcmNoLWJhci9zcy1zZWFyY2gtYmFyLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9zcy1zZWFyY2gtYmFyL21vZGVscy9hZHZhbmNlZC1zZWFyY2gubW9kZWwnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvc3Mtc2VhcmNoLWJhci9tb2RlbHMvc2VhcmNoLXNjb3BlLm1vZGVsJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3NzLXNlYXJjaC1iYXIvbW9kZWxzL3NlYXJjaC1jb25maWcubW9kZWwnO1xuZXhwb3J0IHtcbiAgICBBRFZBTkNFRF9TRUFSQ0hfUVVBTElGSUVSX01BUCxcbiAgICBBRFZBTkNFRF9TRUFSQ0hfRklFTERfTUFQLFxuICAgIEFEVkFOQ0VEX1NFQVJDSF9PUFRJT05TLFxufSBmcm9tICcuL2xpYi9zcy1zZWFyY2gtYmFyL2NvbnN0YW50cyc7XG4iXX0=
|