@byuhbll/components 4.0.24 → 4.0.26-beta.1
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.
|
@@ -13,7 +13,7 @@ export class HbllFooterComponent {
|
|
|
13
13
|
this.footerSkipTargetId = 'hbllChatBtn';
|
|
14
14
|
// TODO: Convert these to signal inputs: https://github.com/angular/angular/issues/57755
|
|
15
15
|
this.mainsitebaseurl = 'https://lib.byu.edu';
|
|
16
|
-
this.libraryapibaseuri = 'https://
|
|
16
|
+
this.libraryapibaseuri = 'https://api.lib.byu.edu/v1';
|
|
17
17
|
this.date = new Date();
|
|
18
18
|
this.isSubmitted = false;
|
|
19
19
|
this.isLoading = false;
|
|
@@ -105,4 +105,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
105
105
|
}], emailstatus: [{
|
|
106
106
|
type: Input
|
|
107
107
|
}] } });
|
|
108
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hbll-footer.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/hbll-footer/hbll-footer.component.ts","../../../../../projects/components/src/lib/hbll-footer/hbll-footer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, inject, Input, ViewChild } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormBuilder, ReactiveFormsModule } from '@angular/forms';\nimport { HttpClient } from '@angular/common/http';\nimport { ContactPayload, createContactForm, submitEmail, USER_STATUSES } from '../contact-utils';\n\n@Component({\n    selector: 'lib-hbll-footer',\n    standalone: true,\n    imports: [CommonModule, ReactiveFormsModule],\n    templateUrl: './hbll-footer.component.html',\n    styleUrl: './hbll-footer.component.scss',\n})\nexport class HbllFooterComponent {\n    private readonly fb = inject(FormBuilder);\n    private readonly http = inject(HttpClient);\n    private readonly footerSkipTargetId = 'hbllChatBtn';\n    @ViewChild('emailDialog') private emailDialog!: ElementRef;\n    // TODO: Convert these to signal inputs: https://github.com/angular/angular/issues/57755\n    @Input() mainsitebaseurl = 'https://lib.byu.edu';\n    @Input() libraryapibaseuri = 'https://apps.lib.byu.edu/v1';\n    @Input() set emailname(name: string) {\n        this.emailForm.controls.name.setValue(name);\n    }\n    @Input() set emailemail(email: string) {\n        this.emailForm.controls.email.setValue(email);\n    }\n    @Input() set emailmessage(message: string) {\n        this.emailForm.controls.message.setValue(message);\n    }\n    @Input() set emailstatus(status: ContactPayload['status']) {\n        this.emailForm.controls.status.setValue(status);\n    }\n    protected date = new Date();\n    protected isSubmitted = false;\n    protected isLoading = false;\n    protected hasConnectionError = false;\n    protected isEmailSent = false;\n    protected emailForm = createContactForm(this.fb);\n    protected userStatuses = USER_STATUSES;\n\n    protected sendEmail = () => {\n        this.isSubmitted = true;\n        if (this.emailForm.invalid) return;\n        this.isLoading = true;\n        submitEmail(\n            this.emailForm.value as ContactPayload,\n            this.http,\n            this.libraryapibaseuri,\n        ).subscribe({\n            error: () => {\n                this.hasConnectionError = true;\n                this.isLoading = false;\n            },\n            complete: () => {\n                this.hasConnectionError = false;\n                this.isLoading = false;\n                this.isEmailSent = true;\n                this.emailForm.reset();\n            },\n        });\n    };\n\n    protected handleClose = () => {\n        if (this.isEmailSent) {\n            this.isEmailSent = false;\n            this.hasConnectionError = false;\n            this.isLoading = false;\n        }\n        this.isSubmitted = false;\n    };\n\n    // Expose this to be called from consuming applications.\n    openEmailForm = (formValues?: typeof this.emailForm.value) => {\n        if (formValues) {\n            if (formValues.name) this.emailForm.controls.name.setValue(formValues.name);\n            if (formValues.email) this.emailForm.controls.email.setValue(formValues.email);\n            if (formValues.message) this.emailForm.controls.message.setValue(formValues.message);\n            if (formValues.status) this.emailForm.controls.status.setValue(formValues.status);\n        }\n        this.emailDialog.nativeElement.showModal();\n    };\n\n    skipFooter(): void {\n        const skipTarget = document.getElementById(this.footerSkipTargetId) as HTMLElement;\n        if (skipTarget) {\n            skipTarget.focus();\n        } else {\n            console.warn('Skip footer target not found on the page.');\n        }\n    }\n}\n","<footer>\n    <div class=\"hbll-footer-wrapper\">\n        <button class=\"skip-link-footer\" (click)=\"skipFooter()\" >Skip footer</button>\n        <div class=\"hbll-footer-links\">\n            <section aria-labelledby=\"footerContactHeading\">\n                <h3 id=\"footerContactHeading\" >Contact</h3>\n                <ul>\n                    <li>\n                        <address>\n                            P.O. Box 26800<br />\n                            Provo, UT 84602-6800\n                        </address>\n                    </li>\n                    <li>\n                        <a href=\"tel:801-422-6061\">\n                            <span class=\"material-symbols-outlined hbll-footer-icon\">\n                                phone_in_talk </span\n                            >&nbsp; <strong>Call</strong>&nbsp;(801) 422-6061\n                        </a>\n                    </li>\n                    <li>\n                        <a href=\"sms:801-623-6838\">\n                            <span class=\"material-symbols-outlined hbll-footer-icon\"> sms </span\n                            >&nbsp; <strong>Text</strong>&nbsp;(801) 623-6838\n                        </a>\n                    </li>\n                    <li>\n                        <button (click)=\"emailDialog.showModal()\" data-testid=\"emailUsBtn\" style=\"padding: 4px;\">\n                            <span class=\"material-symbols-outlined hbll-footer-icon\"> mail </span\n                            >&nbsp;\n                            <strong>Email us</strong>\n                        </button>\n                    </li>\n                </ul>\n            </section>\n            <section aria-labelledby=\"footerResourcesHeading\">\n                <h3 id=\"footerResourcesHeading\">Resources</h3>\n                <ul>\n                    <li>\n                        <a target=\"_blank\" href=\"{{ mainsitebaseurl }}/site-index/\"\n                            >A-Z Site Index</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"{{ mainsitebaseurl }}/about/policies/\">Policies</a>\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"{{ mainsitebaseurl }}/about/location/\"\n                            >Parking & Directions</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://uac.byu.edu/accessibility-lab\"\n                            >Accessibility Resources</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://hr.lib.byu.edu/byu-library-jobs\"\n                            >Employment Opportunities</a\n                        >\n                    </li>\n                </ul>\n            </section>\n            <section aria-labelledby=\"footerFriendsHeading\">\n                <h3 id=\"footerFriendsHeading\">Friends in the Library</h3>\n                <ul>\n                    <li>\n                        <a target=\"_blank\" href=\"https://ctl.byu.edu\"\n                            >Center for Teaching and Learning</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://copyright.byu.edu\"\n                            >Copyright Licensing Office</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://facultycenter.byu.edu/aboutus\"\n                            >Faculty Center</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://dining.byu.edu/library-cafe\"\n                            >Library Cafe</a\n                        >\n                    </li>\n                </ul>\n            </section>\n            <section aria-labelledby=\"footerConnectHeading\">\n                <h3 id=\"footerConnectHeading\">Connect</h3>\n                <ul class=\"hbll-footer-social-list\">\n                    <li>\n                        <a target=\"_blank\" href=\"https://www.facebook.com/byuhbll\">\n                            <img\n                                class=\"hbll-footer-social\"\n                                src=\"https://media.lib.byu.edu/web-assets/images/1.0.0/facebook-color-round.svg\"\n                                alt=\"Facebook\"\n                                title=\"Facebook\"\n                            />\n                        </a>\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://www.instagram.com/byu_hbll/\">\n                            <img\n                                class=\"hbll-footer-social\"\n                                src=\"https://media.lib.byu.edu/web-assets/images/1.0.0/instagram-color.svg\"\n                                alt=\"Instagram\"\n                                title=\"Instagram\"\n                            />\n                        </a>\n                    </li>\n                </ul>\n            </section>\n        </div>\n        <section class=\"hbll-footer-bottom\" aria-label=\"Bottom-most part of the footer\">\n            <div class=\"hbll-footer-title\">\n                <a href=\"https://byu.edu\" target=\"_blank\">BRIGHAM YOUNG UNIVERSITY</a>\n            </div>\n            <br />\n            PROVO, UT 84602, USA | <a href=\"sms:801-422-6061\">(801) 422-6061</a>\n            <br />&copy;&nbsp;{{ date | date: 'yyyy' }}\n            ALL RIGHTS RESERVED\n            <div class=\"hbll-footer-privacy\">\n                <a href=\"https://privacy.byu.edu/\" target=\"_blank\">Privacy Notice</a>\n                |\n                <span id=\"teconsent\"></span>\n            </div>\n        </section>\n    </div>\n    <div id=\"consent-banner\"></div>\n</footer>\n\n<dialog\n    #emailDialog\n    onmousedown=\"event.target === this && this.close()\"\n    (close)=\"handleClose()\"\n    data-testid=\"dialog\"\n>\n    @if (!isEmailSent) {\n        <form [formGroup]=\"emailForm\" (submit)=\"sendEmail()\" data-testid=\"form\">\n            <div class=\"hbll-footer-email-top\">\n                <h1>Email Us</h1>\n                <button (click)=\"emailDialog.close()\" type=\"button\" data-testid=\"closeBtn\">\n                    <span class=\"material-symbols-outlined\"> close </span>\n                </button>\n            </div>\n            <label for=\"hbllFooterEmailName\">Name</label>\n            <input\n                id=\"hbllFooterEmailName\"\n                type=\"text\"\n                formControlName=\"name\"\n                data-testid=\"emailNameInput\"\n            />\n            <label for=\"hbllFooterEmailEmail\">\n                Email (please provide if you would like a response)</label\n            >\n            <input\n                id=\"hbllFooterEmailEmail\"\n                type=\"text\"\n                formControlName=\"email\"\n                data-testid=\"emailEmailInput\"\n            />\n            <label for=\"hbllFooterEmailQuestionComment\">\n                Question or comment&nbsp;<span class=\"hbll-footer-email-required\">*</span>\n            </label>\n            <textarea\n                id=\"hbllFooterEmailQuestionComment\"\n                type=\"text\"\n                formControlName=\"message\"\n                data-testid=\"emailMessageInput\"\n            ></textarea>\n            @if (isSubmitted && emailForm.controls.message.invalid) {\n                <div\n                    class=\"hbll-footer-email-required hbll-footer-email-error\"\n                    data-testid=\"commentError\"\n                >\n                    <span class=\"material-symbols-outlined hbll-footer-email-error-icon\">\n                        warning\n                    </span>\n                    <strong>Please fill out this field</strong>\n                </div>\n            }\n            <label for=\"hbllFooterEmailStatus\">Role</label>\n            <span class=\"hbll-footer-select-wrapper\">\n                <select\n                    id=\"hbllFooterEmailStatus\"\n                    formControlName=\"status\"\n                    data-testid=\"emailStatusInput\"\n                >\n                    <option selected value=\"\"></option>\n                    @for (status of userStatuses | slice: 1; track status) {\n                        <option [value]=\"status\">{{ status }}</option>\n                    }\n                </select>\n            </span>\n            <div class=\"hbll-footer-email-bottom-bar\">\n                @if (hasConnectionError) {\n                    <p class=\"hbll-footer-connection-error\" data-testid=\"connectionError\">\n                        <i>There was an error sending your email. Please try again later.</i>\n                    </p>\n                }\n                <button\n                    class=\"pill-btn--components\"\n                    type=\"submit\"\n                    [disabled]=\"isLoading\"\n                    data-testid=\"sendEmailBtn\"\n                >\n                    Send\n                </button>\n            </div>\n        </form>\n    } @else {\n        <div class=\"hbll-footer-email-success\" data-testid=\"successMessage\">\n            <p>\n                Your email was sent successfully. If you supplied your email address, we will get\n                back to you shortly.<br />Thank you!\n            </p>\n            <button class=\"pill-btn--components\" (click)=\"emailDialog.close()\" type=\"button\">\n                Close\n            </button>\n        </div>\n    }\n</dialog>\n"]}
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hbll-footer.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/hbll-footer/hbll-footer.component.ts","../../../../../projects/components/src/lib/hbll-footer/hbll-footer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, inject, Input, ViewChild } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormBuilder, ReactiveFormsModule } from '@angular/forms';\nimport { HttpClient } from '@angular/common/http';\nimport { ContactPayload, createContactForm, submitEmail, USER_STATUSES } from '../contact-utils';\n\n@Component({\n    selector: 'lib-hbll-footer',\n    standalone: true,\n    imports: [CommonModule, ReactiveFormsModule],\n    templateUrl: './hbll-footer.component.html',\n    styleUrl: './hbll-footer.component.scss',\n})\nexport class HbllFooterComponent {\n    private readonly fb = inject(FormBuilder);\n    private readonly http = inject(HttpClient);\n    private readonly footerSkipTargetId = 'hbllChatBtn';\n    @ViewChild('emailDialog') private emailDialog!: ElementRef;\n    // TODO: Convert these to signal inputs: https://github.com/angular/angular/issues/57755\n    @Input() mainsitebaseurl = 'https://lib.byu.edu';\n    @Input() libraryapibaseuri = 'https://api.lib.byu.edu/v1';\n    @Input() set emailname(name: string) {\n        this.emailForm.controls.name.setValue(name);\n    }\n    @Input() set emailemail(email: string) {\n        this.emailForm.controls.email.setValue(email);\n    }\n    @Input() set emailmessage(message: string) {\n        this.emailForm.controls.message.setValue(message);\n    }\n    @Input() set emailstatus(status: ContactPayload['status']) {\n        this.emailForm.controls.status.setValue(status);\n    }\n    protected date = new Date();\n    protected isSubmitted = false;\n    protected isLoading = false;\n    protected hasConnectionError = false;\n    protected isEmailSent = false;\n    protected emailForm = createContactForm(this.fb);\n    protected userStatuses = USER_STATUSES;\n\n    protected sendEmail = () => {\n        this.isSubmitted = true;\n        if (this.emailForm.invalid) return;\n        this.isLoading = true;\n        submitEmail(\n            this.emailForm.value as ContactPayload,\n            this.http,\n            this.libraryapibaseuri,\n        ).subscribe({\n            error: () => {\n                this.hasConnectionError = true;\n                this.isLoading = false;\n            },\n            complete: () => {\n                this.hasConnectionError = false;\n                this.isLoading = false;\n                this.isEmailSent = true;\n                this.emailForm.reset();\n            },\n        });\n    };\n\n    protected handleClose = () => {\n        if (this.isEmailSent) {\n            this.isEmailSent = false;\n            this.hasConnectionError = false;\n            this.isLoading = false;\n        }\n        this.isSubmitted = false;\n    };\n\n    // Expose this to be called from consuming applications.\n    openEmailForm = (formValues?: typeof this.emailForm.value) => {\n        if (formValues) {\n            if (formValues.name) this.emailForm.controls.name.setValue(formValues.name);\n            if (formValues.email) this.emailForm.controls.email.setValue(formValues.email);\n            if (formValues.message) this.emailForm.controls.message.setValue(formValues.message);\n            if (formValues.status) this.emailForm.controls.status.setValue(formValues.status);\n        }\n        this.emailDialog.nativeElement.showModal();\n    };\n\n    skipFooter(): void {\n        const skipTarget = document.getElementById(this.footerSkipTargetId) as HTMLElement;\n        if (skipTarget) {\n            skipTarget.focus();\n        } else {\n            console.warn('Skip footer target not found on the page.');\n        }\n    }\n}\n","<footer>\n    <div class=\"hbll-footer-wrapper\">\n        <button class=\"skip-link-footer\" (click)=\"skipFooter()\" >Skip footer</button>\n        <div class=\"hbll-footer-links\">\n            <section aria-labelledby=\"footerContactHeading\">\n                <h3 id=\"footerContactHeading\" >Contact</h3>\n                <ul>\n                    <li>\n                        <address>\n                            P.O. Box 26800<br />\n                            Provo, UT 84602-6800\n                        </address>\n                    </li>\n                    <li>\n                        <a href=\"tel:801-422-6061\">\n                            <span class=\"material-symbols-outlined hbll-footer-icon\">\n                                phone_in_talk </span\n                            >&nbsp; <strong>Call</strong>&nbsp;(801) 422-6061\n                        </a>\n                    </li>\n                    <li>\n                        <a href=\"sms:801-623-6838\">\n                            <span class=\"material-symbols-outlined hbll-footer-icon\"> sms </span\n                            >&nbsp; <strong>Text</strong>&nbsp;(801) 623-6838\n                        </a>\n                    </li>\n                    <li>\n                        <button (click)=\"emailDialog.showModal()\" data-testid=\"emailUsBtn\" style=\"padding: 4px;\">\n                            <span class=\"material-symbols-outlined hbll-footer-icon\"> mail </span\n                            >&nbsp;\n                            <strong>Email us</strong>\n                        </button>\n                    </li>\n                </ul>\n            </section>\n            <section aria-labelledby=\"footerResourcesHeading\">\n                <h3 id=\"footerResourcesHeading\">Resources</h3>\n                <ul>\n                    <li>\n                        <a target=\"_blank\" href=\"{{ mainsitebaseurl }}/site-index/\"\n                            >A-Z Site Index</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"{{ mainsitebaseurl }}/about/policies/\">Policies</a>\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"{{ mainsitebaseurl }}/about/location/\"\n                            >Parking & Directions</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://uac.byu.edu/accessibility-lab\"\n                            >Accessibility Resources</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://hr.lib.byu.edu/byu-library-jobs\"\n                            >Employment Opportunities</a\n                        >\n                    </li>\n                </ul>\n            </section>\n            <section aria-labelledby=\"footerFriendsHeading\">\n                <h3 id=\"footerFriendsHeading\">Friends in the Library</h3>\n                <ul>\n                    <li>\n                        <a target=\"_blank\" href=\"https://ctl.byu.edu\"\n                            >Center for Teaching and Learning</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://copyright.byu.edu\"\n                            >Copyright Licensing Office</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://facultycenter.byu.edu/aboutus\"\n                            >Faculty Center</a\n                        >\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://dining.byu.edu/library-cafe\"\n                            >Library Cafe</a\n                        >\n                    </li>\n                </ul>\n            </section>\n            <section aria-labelledby=\"footerConnectHeading\">\n                <h3 id=\"footerConnectHeading\">Connect</h3>\n                <ul class=\"hbll-footer-social-list\">\n                    <li>\n                        <a target=\"_blank\" href=\"https://www.facebook.com/byuhbll\">\n                            <img\n                                class=\"hbll-footer-social\"\n                                src=\"https://media.lib.byu.edu/web-assets/images/1.0.0/facebook-color-round.svg\"\n                                alt=\"Facebook\"\n                                title=\"Facebook\"\n                            />\n                        </a>\n                    </li>\n                    <li>\n                        <a target=\"_blank\" href=\"https://www.instagram.com/byu_hbll/\">\n                            <img\n                                class=\"hbll-footer-social\"\n                                src=\"https://media.lib.byu.edu/web-assets/images/1.0.0/instagram-color.svg\"\n                                alt=\"Instagram\"\n                                title=\"Instagram\"\n                            />\n                        </a>\n                    </li>\n                </ul>\n            </section>\n        </div>\n        <section class=\"hbll-footer-bottom\" aria-label=\"Bottom-most part of the footer\">\n            <div class=\"hbll-footer-title\">\n                <a href=\"https://byu.edu\" target=\"_blank\">BRIGHAM YOUNG UNIVERSITY</a>\n            </div>\n            <br />\n            PROVO, UT 84602, USA | <a href=\"sms:801-422-6061\">(801) 422-6061</a>\n            <br />&copy;&nbsp;{{ date | date: 'yyyy' }}\n            ALL RIGHTS RESERVED\n            <div class=\"hbll-footer-privacy\">\n                <a href=\"https://privacy.byu.edu/\" target=\"_blank\">Privacy Notice</a>\n                |\n                <span id=\"teconsent\"></span>\n            </div>\n        </section>\n    </div>\n    <div id=\"consent-banner\"></div>\n</footer>\n\n<dialog\n    #emailDialog\n    onmousedown=\"event.target === this && this.close()\"\n    (close)=\"handleClose()\"\n    data-testid=\"dialog\"\n>\n    @if (!isEmailSent) {\n        <form [formGroup]=\"emailForm\" (submit)=\"sendEmail()\" data-testid=\"form\">\n            <div class=\"hbll-footer-email-top\">\n                <h1>Email Us</h1>\n                <button (click)=\"emailDialog.close()\" type=\"button\" data-testid=\"closeBtn\">\n                    <span class=\"material-symbols-outlined\"> close </span>\n                </button>\n            </div>\n            <label for=\"hbllFooterEmailName\">Name</label>\n            <input\n                id=\"hbllFooterEmailName\"\n                type=\"text\"\n                formControlName=\"name\"\n                data-testid=\"emailNameInput\"\n            />\n            <label for=\"hbllFooterEmailEmail\">\n                Email (please provide if you would like a response)</label\n            >\n            <input\n                id=\"hbllFooterEmailEmail\"\n                type=\"text\"\n                formControlName=\"email\"\n                data-testid=\"emailEmailInput\"\n            />\n            <label for=\"hbllFooterEmailQuestionComment\">\n                Question or comment&nbsp;<span class=\"hbll-footer-email-required\">*</span>\n            </label>\n            <textarea\n                id=\"hbllFooterEmailQuestionComment\"\n                type=\"text\"\n                formControlName=\"message\"\n                data-testid=\"emailMessageInput\"\n            ></textarea>\n            @if (isSubmitted && emailForm.controls.message.invalid) {\n                <div\n                    class=\"hbll-footer-email-required hbll-footer-email-error\"\n                    data-testid=\"commentError\"\n                >\n                    <span class=\"material-symbols-outlined hbll-footer-email-error-icon\">\n                        warning\n                    </span>\n                    <strong>Please fill out this field</strong>\n                </div>\n            }\n            <label for=\"hbllFooterEmailStatus\">Role</label>\n            <span class=\"hbll-footer-select-wrapper\">\n                <select\n                    id=\"hbllFooterEmailStatus\"\n                    formControlName=\"status\"\n                    data-testid=\"emailStatusInput\"\n                >\n                    <option selected value=\"\"></option>\n                    @for (status of userStatuses | slice: 1; track status) {\n                        <option [value]=\"status\">{{ status }}</option>\n                    }\n                </select>\n            </span>\n            <div class=\"hbll-footer-email-bottom-bar\">\n                @if (hasConnectionError) {\n                    <p class=\"hbll-footer-connection-error\" data-testid=\"connectionError\">\n                        <i>There was an error sending your email. Please try again later.</i>\n                    </p>\n                }\n                <button\n                    class=\"pill-btn--components\"\n                    type=\"submit\"\n                    [disabled]=\"isLoading\"\n                    data-testid=\"sendEmailBtn\"\n                >\n                    Send\n                </button>\n            </div>\n        </form>\n    } @else {\n        <div class=\"hbll-footer-email-success\" data-testid=\"successMessage\">\n            <p>\n                Your email was sent successfully. If you supplied your email address, we will get\n                back to you shortly.<br />Thank you!\n            </p>\n            <button class=\"pill-btn--components\" (click)=\"emailDialog.close()\" type=\"button\">\n                Close\n            </button>\n        </div>\n    }\n</dialog>\n"]}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Component, computed, EventEmitter, inject, input, Output } from '@angular/core';
|
|
2
|
-
import { map, switchMap } from 'rxjs/operators';
|
|
2
|
+
import { catchError, map, switchMap } from 'rxjs/operators';
|
|
3
3
|
import { AccessStatus, } from './models/application-access';
|
|
4
4
|
import { CommonModule } from '@angular/common';
|
|
5
5
|
import { jwtDecode } from 'jwt-decode';
|
|
@@ -40,7 +40,7 @@ export class ImpersonationBannerComponent {
|
|
|
40
40
|
this.http = inject(HttpClient);
|
|
41
41
|
this.accessTokenPayload = input.required();
|
|
42
42
|
this.personBaseUri = input('https://apps.lib.byu.edu/person/v2/');
|
|
43
|
-
this.libraryApiBaseUri = input('https://
|
|
43
|
+
this.libraryApiBaseUri = input('https://api.lib.byu.edu/v1');
|
|
44
44
|
this.endImpersonation = new EventEmitter();
|
|
45
45
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
46
46
|
this.parsedToken = computed(() => this.accessTokenPayload().token ? jwtDecode(this.accessTokenPayload().token) : {});
|
|
@@ -134,6 +134,9 @@ export class ImpersonationBannerComponent {
|
|
|
134
134
|
}
|
|
135
135
|
});
|
|
136
136
|
return accountStatuses;
|
|
137
|
+
}), catchError((error) => {
|
|
138
|
+
console.warn('Failed to fetch patron account statuses:', error);
|
|
139
|
+
return of(null);
|
|
137
140
|
}));
|
|
138
141
|
})));
|
|
139
142
|
}
|
|
@@ -146,4 +149,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
|
|
|
146
149
|
}], propDecorators: { endImpersonation: [{
|
|
147
150
|
type: Output
|
|
148
151
|
}] } });
|
|
149
|
-
//# 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 './models/application-access';\nimport { CommonModule } from '@angular/common';\nimport { TokenPayload } from '../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 './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 INDEPENDENT_STUDY_RESPONSE_PATH = '/independent-study/:libraryId/';\nconst PATRON_ACCOUNTS_PATH = '/patron/accounts';\n\n/**\n * @todo: Refactor: my recommendation would be to allow a user to be passed into this component via an input instead of the users coming from an API call based on the JWT passed in. This might look something like:\n *  ```\n *    ＠Input() user: {\n *      primary_position_type_display?: string\n *      undergrad_graduate_status?: string\n *      restricted?: boolean\n *      is_retired?: boolean\n *      is_employee?: boolean\n *      primary_position_type?: string\n *    }\n * ```\n * Accepting all properties that are used in this component without needing to bring in the user type explicitly. Or the properties on this user object could be made even more general as to not be tied at all to any specific type.\n *\n * Something similar could be done with `accountStatuses`. I'd recommend the consuming application prepare arrays of strings to be passed into inputs of `accountsOk`, `accountsBlocked`, and `accountsNone`, that are then passed into and displayed in the template.\n *\n * The idea behind these revisions is that there is less business logic baked into this component, as it's role is just one of presentation. It will also avoid multiple API calls for the same data in our big Angular apps.\n *\n * Best of luck,\n * Paul\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    personBaseUri = input('https://apps.lib.byu.edu/person/v2/');\n    libraryApiBaseUri = input('https://apps.lib.byu.edu/v1');\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(this.personBaseUri(), 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.user()?.email_address ?? ''],\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(this.personBaseUri(), 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(this.libraryApiBaseUri(), 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","@if (isImpersonating()) {\n    <div class=\"banner-padding\"></div>\n    <div class=\"top-banner\" data-testid=\"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\n                                                        class=\"material-symbols-outlined icon-checkmark\"\n                                                    >\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\n                                                        class=\"material-symbols-outlined icon-lock\"\n                                                    >\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"]}
|
|
152
|
+
//# 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 { catchError, map, switchMap } from 'rxjs/operators';\nimport {\n    AccessStatus,\n    AccountsResponse,\n    type ApplicationAccess,\n} from './models/application-access';\nimport { CommonModule } from '@angular/common';\nimport { TokenPayload } from '../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 './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 INDEPENDENT_STUDY_RESPONSE_PATH = '/independent-study/:libraryId/';\nconst PATRON_ACCOUNTS_PATH = '/patron/accounts';\n\n/**\n * @todo: Refactor: my recommendation would be to allow a user to be passed into this component via an input instead of the users coming from an API call based on the JWT passed in. This might look something like:\n *  ```\n *    ＠Input() user: {\n *      primary_position_type_display?: string\n *      undergrad_graduate_status?: string\n *      restricted?: boolean\n *      is_retired?: boolean\n *      is_employee?: boolean\n *      primary_position_type?: string\n *    }\n * ```\n * Accepting all properties that are used in this component without needing to bring in the user type explicitly. Or the properties on this user object could be made even more general as to not be tied at all to any specific type.\n *\n * Something similar could be done with `accountStatuses`. I'd recommend the consuming application prepare arrays of strings to be passed into inputs of `accountsOk`, `accountsBlocked`, and `accountsNone`, that are then passed into and displayed in the template.\n *\n * The idea behind these revisions is that there is less business logic baked into this component, as it's role is just one of presentation. It will also avoid multiple API calls for the same data in our big Angular apps.\n *\n * Best of luck,\n * Paul\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    personBaseUri = input('https://apps.lib.byu.edu/person/v2/');\n    libraryApiBaseUri = input('https://api.lib.byu.edu/v1');\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(this.personBaseUri(), 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.user()?.email_address ?? ''],\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(this.personBaseUri(), 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(this.libraryApiBaseUri(), 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                        catchError((error) => {\n                            console.warn('Failed to fetch patron account statuses:', error);\n                            return of(null);\n                        }),\n                    );\n            }),\n        ),\n    );\n}\n","@if (isImpersonating()) {\n    <div class=\"banner-padding\"></div>\n    <div class=\"top-banner\" data-testid=\"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\n                                                        class=\"material-symbols-outlined icon-checkmark\"\n                                                    >\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\n                                                        class=\"material-symbols-outlined icon-lock\"\n                                                    >\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"]}
|
|
@@ -9,7 +9,7 @@ import { map, of, switchMap, shareReplay, combineLatest, Subject, Subscription }
|
|
|
9
9
|
import { BreakpointObserver } from '@angular/cdk/layout';
|
|
10
10
|
import * as i2 from '@angular/forms';
|
|
11
11
|
import { Validators, FormBuilder, ReactiveFormsModule, FormControl, NonNullableFormBuilder } from '@angular/forms';
|
|
12
|
-
import { switchMap as switchMap$1, map as map$1,
|
|
12
|
+
import { switchMap as switchMap$1, map as map$1, catchError, startWith, tap, filter } from 'rxjs/operators';
|
|
13
13
|
import { jwtDecode } from 'jwt-decode';
|
|
14
14
|
import urlcat from 'urlcat';
|
|
15
15
|
import * as i1$1 from '@angular/material/tooltip';
|
|
@@ -806,7 +806,7 @@ class HbllFooterComponent {
|
|
|
806
806
|
this.footerSkipTargetId = 'hbllChatBtn';
|
|
807
807
|
// TODO: Convert these to signal inputs: https://github.com/angular/angular/issues/57755
|
|
808
808
|
this.mainsitebaseurl = 'https://lib.byu.edu';
|
|
809
|
-
this.libraryapibaseuri = 'https://
|
|
809
|
+
this.libraryapibaseuri = 'https://api.lib.byu.edu/v1';
|
|
810
810
|
this.date = new Date();
|
|
811
811
|
this.isSubmitted = false;
|
|
812
812
|
this.isLoading = false;
|
|
@@ -980,7 +980,7 @@ class ImpersonationBannerComponent {
|
|
|
980
980
|
this.http = inject(HttpClient);
|
|
981
981
|
this.accessTokenPayload = input.required();
|
|
982
982
|
this.personBaseUri = input('https://apps.lib.byu.edu/person/v2/');
|
|
983
|
-
this.libraryApiBaseUri = input('https://
|
|
983
|
+
this.libraryApiBaseUri = input('https://api.lib.byu.edu/v1');
|
|
984
984
|
this.endImpersonation = new EventEmitter();
|
|
985
985
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
986
986
|
this.parsedToken = computed(() => this.accessTokenPayload().token ? jwtDecode(this.accessTokenPayload().token) : {});
|
|
@@ -1074,6 +1074,9 @@ class ImpersonationBannerComponent {
|
|
|
1074
1074
|
}
|
|
1075
1075
|
});
|
|
1076
1076
|
return accountStatuses;
|
|
1077
|
+
}), catchError((error) => {
|
|
1078
|
+
console.warn('Failed to fetch patron account statuses:', error);
|
|
1079
|
+
return of(null);
|
|
1077
1080
|
}));
|
|
1078
1081
|
})));
|
|
1079
1082
|
}
|