@byuhbll/components 5.0.2 → 5.0.3

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.
@@ -61,7 +61,13 @@ export class HeaderWithImpersonationComponent {
61
61
  // effect can only be used within an injection context (ex: constructor)
62
62
  runInInjectionContext(this.injector, () => {
63
63
  effect(() => {
64
- if (this.isImpersonating()) {
64
+ const impersonating = this.isImpersonating();
65
+ const token = this.parsedToken();
66
+ // when token refreshes after 5 minutes, it leaves a small moment where there is no token
67
+ // this prevents dropping inactivtiy timer during that blip
68
+ if (!token)
69
+ return;
70
+ if (impersonating) {
65
71
  if (!this.trackingActive)
66
72
  this.startInactivityTracking();
67
73
  }
@@ -92,7 +98,9 @@ export class HeaderWithImpersonationComponent {
92
98
  stopInactivityTracking() {
93
99
  this.activityEvents.forEach((event) => {
94
100
  const target = event === 'scroll' ? window : document;
95
- target.removeEventListener(event, this.debouncedResetTimer);
101
+ // IMPORTANT: capture must match how it was added for scroll
102
+ const options = event === 'scroll' ? { capture: true } : undefined;
103
+ target.removeEventListener(event, this.debouncedResetTimer, options);
96
104
  });
97
105
  if (this.inactivityTimerId) {
98
106
  clearTimeout(this.inactivityTimerId);
@@ -117,4 +125,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.0", ngImpor
117
125
  }], endImpersonation: [{
118
126
  type: Output
119
127
  }] } });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header-with-impersonation.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/header-with-impersonation/header-with-impersonation.component.ts","../../../../../projects/components/src/lib/header-with-impersonation/header-with-impersonation.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,SAAS,EACT,QAAQ,EACR,YAAY,EACZ,KAAK,EACL,MAAM,EAIN,MAAM,EACN,qBAAqB,EACrB,MAAM,EACN,QAAQ,GACX,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAC3E,OAAO,EAAE,4BAA4B,EAAE,MAAM,wDAAwD,CAAC;AACtG,OAAO,EACH,kBAAkB,EAClB,qBAAqB,EACrB,yBAAyB,GAC5B,MAAM,kDAAkD,CAAC;AAE1D,OAAO,EAAE,SAAS,EAAc,MAAM,YAAY,CAAC;;AASnD,MAAM,OAAO,gCAAgC;IAP7C;QAQI,uBAAkB,GAAG,KAAK,CAAC,QAAQ,EAAgB,CAAC;QACpD,gBAAW,GAAG,KAAK,CAAC,kBAAkB,CAAC,CAAC;QACxC,mBAAc,GAAG,KAAK,CAAC,qBAAqB,CAAC,CAAC;QAC9C,oBAAe,GAAG,KAAK,CAAC,qBAAqB,CAAC,CAAC;QAC/C,kBAAa,GAAG,KAAK,CAAC,qCAAqC,CAAC,CAAC;QAC7D,wBAAmB,GAAG,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAEhD,UAAK,GAAG,IAAI,YAAY,EAAQ,CAAC;QACjC,WAAM,GAAG,IAAI,YAAY,EAAQ,CAAC;QAClC,qBAAgB,GAAG,IAAI,YAAY,EAAQ,CAAC;QAEtD,8DAA8D;QACpD,gBAAW,GAAsD,QAAQ,CAAC,GAAG,EAAE,CACrF,IAAI,CAAC,kBAAkB,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CACtF,CAAC;QACQ,SAAI,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAG,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACrF,0BAAqB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAC5C,IAAI,CAAC,WAAW,EAAE;YACd,CAAC,CAAC,CAAC,CAAC,CACE,CAAC,IAAI,CAAC,eAAe,EAAE;gBACvB,IAAI,CAAC,WAAW,EAAG,CAAC,iBAAiB,CAAC,CAAC,kBAAkB,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,QAAQ,CAC3E,eAAe,CAClB,CACJ;YACH,CAAC,CAAC,KAAK,CACd,CAAC;QAEQ,2BAAsB,GAAG,KAAK,CAAC;QACjC,oBAAe,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC,cAAc,CAAC,CAAC,CAAC;QAEjF,uCAAuC;QAC/B,mBAAc,GAA0D;YAC5E,SAAS;YACT,aAAa;YACb,OAAO;YACP,QAAQ;SACX,CAAC;QACM,sBAAiB,GAAkB,IAAI,CAAC;QACxC,wBAAmB,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,YAAY;QACjD,oBAAe,GAAkB,IAAI,CAAC;QACtC,oBAAe,GAAG,GAAG,CAAC;QAEtB,mBAAc,GAAG,KAAK,CAAC;QAEvB,aAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;QAuDpC,kEAAkE;QAC1D,yBAAoB,GAAG,GAAG,EAAE;YAChC,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE;gBAAE,OAAO;YAEpC,IAAI,IAAI,CAAC,iBAAiB;gBAAE,YAAY,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAEjE,IAAI,CAAC,iBAAiB,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;gBAC5C,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,CAAC;gBAC7B,IAAI,CAAC,sBAAsB,EAAE,CAAC;YAClC,CAAC,EAAE,IAAI,CAAC,mBAAmB,CAAC,CAAC;QACjC,CAAC,CAAC;QAEF,sEAAsE;QAC9D,wBAAmB,GAAG,GAAG,EAAE;YAC/B,IAAI,IAAI,CAAC,eAAe;gBAAE,YAAY,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YAE7D,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;gBAC1C,IAAI,CAAC,oBAAoB,EAAE,CAAC;YAChC,CAAC,EAAE,IAAI,CAAC,eAAe,CAAC,CAAC;QAC7B,CAAC,CAAC;KACL;IAzEG,QAAQ;QACJ,wEAAwE;QACxE,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE;YACtC,MAAM,CAAC,GAAG,EAAE;gBACR,IAAI,IAAI,CAAC,eAAe,EAAE,EAAE,CAAC;oBACzB,IAAI,CAAC,IAAI,CAAC,cAAc;wBAAE,IAAI,CAAC,uBAAuB,EAAE,CAAC;gBAC7D,CAAC;qBAAM,CAAC;oBACJ,IAAI,IAAI,CAAC,cAAc;wBAAE,IAAI,CAAC,sBAAsB,EAAE,CAAC;gBAC3D,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACP,CAAC;IAED,WAAW;QACP,IAAI,CAAC,sBAAsB,EAAE,CAAC;IAClC,CAAC;IAED,0CAA0C;IAClC,uBAAuB;QAC3B,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;YAClC,oFAAoF;YACpF,MAAM,MAAM,GAAsB,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC;YACzE,MAAM,OAAO,GACT,KAAK,KAAK,QAAQ;gBACd,CAAC,CAAE,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAA8B;gBAC/D,CAAC,CAAE,EAAE,OAAO,EAAE,IAAI,EAA8B,CAAC;YAEzD,MAAM,CAAC,gBAAgB,CAAC,KAAK,EAAE,IAAI,CAAC,mBAAmB,EAAE,OAAO,CAAC,CAAC;QACtE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAC3B,IAAI,CAAC,oBAAoB,EAAE,CAAC;IAChC,CAAC;IAED,wCAAwC;IAChC,sBAAsB;QAC1B,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;YAClC,MAAM,MAAM,GAAsB,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC;YACzE,MAAM,CAAC,mBAAmB,CAAC,KAAK,EAAE,IAAI,CAAC,mBAAmB,CAAC,CAAC;QAChE,CAAC,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,YAAY,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YACrC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;QAClC,CAAC;QACD,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,YAAY,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YACnC,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;QAChC,CAAC;QAED,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;IAChC,CAAC;8GAlGQ,gCAAgC;kGAAhC,gCAAgC,6jCC/B7C,m1BAsBA,0DDKc,mBAAmB,kLAAE,4BAA4B,4KAAE,yBAAyB;;2FAI7E,gCAAgC;kBAP5C,SAAS;+BACI,+BAA+B,cAC7B,IAAI,WACP,CAAC,mBAAmB,EAAE,4BAA4B,EAAE,yBAAyB,CAAC;8BAY7E,KAAK;sBAAd,MAAM;gBACG,MAAM;sBAAf,MAAM;gBACG,gBAAgB;sBAAzB,MAAM","sourcesContent":["import {\n    Component,\n    computed,\n    EventEmitter,\n    input,\n    Output,\n    Signal,\n    OnInit,\n    OnDestroy,\n    effect,\n    runInInjectionContext,\n    inject,\n    Injector,\n} from '@angular/core';\nimport { HbllHeaderComponent } from '../hbll-header/hbll-header.component';\nimport { ImpersonationBannerComponent } from '../impersonation-banner/impersonation-banner.component';\nimport {\n    defaultOidcBaseUri,\n    defaultOidcDefaultIdp,\n    ImpersonateModalComponent,\n} from '../impersonate-modal/impersonate-modal.component';\nimport { TokenPayload } from '../models/token-payload';\nimport { jwtDecode, JwtPayload } from 'jwt-decode';\n\n@Component({\n    selector: 'lib-header-with-impersonation',\n    standalone: true,\n    imports: [HbllHeaderComponent, ImpersonationBannerComponent, ImpersonateModalComponent],\n    templateUrl: './header-with-impersonation.component.html',\n    styleUrl: './header-with-impersonation.component.scss',\n})\nexport class HeaderWithImpersonationComponent implements OnInit, OnDestroy {\n    accessTokenPayload = input.required<TokenPayload>();\n    oidcBaseUri = input(defaultOidcBaseUri);\n    oidcDefaultIdp = input(defaultOidcDefaultIdp);\n    mainSiteBaseUrl = input('https://lib.byu.edu');\n    personBaseUri = input('https://apps.lib.byu.edu/person/v2/');\n    myAccountApiBaseUri = input('https://api.lib.byu.edu/v1');\n\n    @Output() login = new EventEmitter<void>();\n    @Output() logout = new EventEmitter<void>();\n    @Output() endImpersonation = new EventEmitter<void>();\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    protected parsedToken: Signal<(JwtPayload & Record<string, any>) | null> = computed(() =>\n        this.accessTokenPayload().token ? jwtDecode(this.accessTokenPayload().token) : null,\n    );\n    protected name = computed(() => (this.parsedToken() ? this.parsedToken()!['given_name'] : ''));\n    protected showImpersonateButton = computed(() =>\n        this.parsedToken()\n            ? !!(\n                  !this.isImpersonating() &&\n                  this.parsedToken()!['resource_access']['realm-management']?.['roles']?.includes(\n                      'impersonation',\n                  )\n              )\n            : false,\n    );\n\n    protected showImpersonationModal = false;\n    private isImpersonating = computed(() => !!this.parsedToken()?.['impersonator']);\n\n    // Inactivity timeout for impersonation\n    private activityEvents: Array<'keydown' | 'pointerdown' | 'wheel' | 'scroll'> = [\n        'keydown',\n        'pointerdown',\n        'wheel',\n        'scroll',\n    ];\n    private inactivityTimerId: number | null = null;\n    private inactivityTimeoutMs = 5 * 60 * 1000; // 5 minutes\n    private debounceTimerId: number | null = null;\n    private debounceDelayMs = 250;\n\n    private trackingActive = false;\n\n    private injector = inject(Injector);\n\n    ngOnInit() {\n        // effect can only be used within an injection context (ex: constructor)\n        runInInjectionContext(this.injector, () => {\n            effect(() => {\n                if (this.isImpersonating()) {\n                    if (!this.trackingActive) this.startInactivityTracking();\n                } else {\n                    if (this.trackingActive) this.stopInactivityTracking();\n                }\n            });\n        });\n    }\n\n    ngOnDestroy() {\n        this.stopInactivityTracking();\n    }\n\n    /** Begin listening and start countdown */\n    private startInactivityTracking() {\n        this.activityEvents.forEach((event) => {\n            // 'scroll' on document doesn't bubble; use window + capture to catch nested scrolls\n            const target: Window | Document = event === 'scroll' ? window : document;\n            const options =\n                event === 'scroll'\n                    ? ({ passive: true, capture: true } as AddEventListenerOptions)\n                    : ({ passive: true } as AddEventListenerOptions);\n\n            target.addEventListener(event, this.debouncedResetTimer, options);\n        });\n\n        this.trackingActive = true;\n        this.resetInactivityTimer();\n    }\n\n    /** Remove listeners and clear timers */\n    private stopInactivityTracking() {\n        this.activityEvents.forEach((event) => {\n            const target: Window | Document = event === 'scroll' ? window : document;\n            target.removeEventListener(event, this.debouncedResetTimer);\n        });\n\n        if (this.inactivityTimerId) {\n            clearTimeout(this.inactivityTimerId);\n            this.inactivityTimerId = null;\n        }\n        if (this.debounceTimerId) {\n            clearTimeout(this.debounceTimerId);\n            this.debounceTimerId = null;\n        }\n\n        this.trackingActive = false;\n    }\n\n    /** Reset the inactivity countdown (no-op if not impersonating) */\n    private resetInactivityTimer = () => {\n        if (!this.isImpersonating()) return;\n\n        if (this.inactivityTimerId) clearTimeout(this.inactivityTimerId);\n\n        this.inactivityTimerId = window.setTimeout(() => {\n            this.endImpersonation.emit();\n            this.stopInactivityTracking();\n        }, this.inactivityTimeoutMs);\n    };\n\n    /** Debounce activity to avoid hammering resets during event storms */\n    private debouncedResetTimer = () => {\n        if (this.debounceTimerId) clearTimeout(this.debounceTimerId);\n\n        this.debounceTimerId = window.setTimeout(() => {\n            this.resetInactivityTimer();\n        }, this.debounceDelayMs);\n    };\n}\n","<lib-impersonation-banner\n    [accessTokenPayload]=\"accessTokenPayload()\"\n    (endImpersonation)=\"endImpersonation.emit()\"\n    [personBaseUri]=\"personBaseUri()\"\n    [myAccountApiBaseUri]=\"myAccountApiBaseUri()\"\n></lib-impersonation-banner>\n<lib-hbll-header\n    [name]=\"name()\"\n    [showImpersonateButton]=\"showImpersonateButton()\"\n    (openImpersonationModal)=\"showImpersonationModal = true\"\n    (login)=\"login.emit()\"\n    (logout)=\"logout.emit()\"\n    [mainsitebaseurl]=\"mainSiteBaseUrl()\"\n/>\n<lib-impersonate-modal\n    [showModal]=\"showImpersonationModal\"\n    [oidcBaseUri]=\"oidcBaseUri()\"\n    [oidcDefaultIdp]=\"oidcDefaultIdp()\"\n    [accessTokenPayload]=\"accessTokenPayload()\"\n    (dismiss)=\"showImpersonationModal = false\"\n    (init)=\"showImpersonationModal = true\"\n></lib-impersonate-modal>\n"]}
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header-with-impersonation.component.js","sourceRoot":"","sources":["../../../../../projects/components/src/lib/header-with-impersonation/header-with-impersonation.component.ts","../../../../../projects/components/src/lib/header-with-impersonation/header-with-impersonation.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    computed,\n    EventEmitter,\n    input,\n    Output,\n    Signal,\n    OnInit,\n    OnDestroy,\n    effect,\n    runInInjectionContext,\n    inject,\n    Injector,\n} from '@angular/core';\nimport { HbllHeaderComponent } from '../hbll-header/hbll-header.component';\nimport { ImpersonationBannerComponent } from '../impersonation-banner/impersonation-banner.component';\nimport {\n    defaultOidcBaseUri,\n    defaultOidcDefaultIdp,\n    ImpersonateModalComponent,\n} from '../impersonate-modal/impersonate-modal.component';\nimport { TokenPayload } from '../models/token-payload';\nimport { jwtDecode, JwtPayload } from 'jwt-decode';\n\n@Component({\n    selector: 'lib-header-with-impersonation',\n    standalone: true,\n    imports: [HbllHeaderComponent, ImpersonationBannerComponent, ImpersonateModalComponent],\n    templateUrl: './header-with-impersonation.component.html',\n    styleUrl: './header-with-impersonation.component.scss',\n})\nexport class HeaderWithImpersonationComponent implements OnInit, OnDestroy {\n    accessTokenPayload = input.required<TokenPayload>();\n    oidcBaseUri = input(defaultOidcBaseUri);\n    oidcDefaultIdp = input(defaultOidcDefaultIdp);\n    mainSiteBaseUrl = input('https://lib.byu.edu');\n    personBaseUri = input('https://apps.lib.byu.edu/person/v2/');\n    myAccountApiBaseUri = input('https://api.lib.byu.edu/v1');\n\n    @Output() login = new EventEmitter<void>();\n    @Output() logout = new EventEmitter<void>();\n    @Output() endImpersonation = new EventEmitter<void>();\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    protected parsedToken: Signal<(JwtPayload & Record<string, any>) | null> = computed(() =>\n        this.accessTokenPayload().token ? jwtDecode(this.accessTokenPayload().token) : null,\n    );\n    protected name = computed(() => (this.parsedToken() ? this.parsedToken()!['given_name'] : ''));\n    protected showImpersonateButton = computed(() =>\n        this.parsedToken()\n            ? !!(\n                  !this.isImpersonating() &&\n                  this.parsedToken()!['resource_access']['realm-management']?.['roles']?.includes(\n                      'impersonation',\n                  )\n              )\n            : false,\n    );\n\n    protected showImpersonationModal = false;\n    private isImpersonating = computed(() => !!this.parsedToken()?.['impersonator']);\n\n    // Inactivity timeout for impersonation\n    private activityEvents: Array<'keydown' | 'pointerdown' | 'wheel' | 'scroll'> = [\n        'keydown',\n        'pointerdown',\n        'wheel',\n        'scroll',\n    ];\n    private inactivityTimerId: number | null = null;\n    private inactivityTimeoutMs = 5 * 60 * 1000; // 5 minutes\n    private debounceTimerId: number | null = null;\n    private debounceDelayMs = 250;\n\n    private trackingActive = false;\n\n    private injector = inject(Injector);\n\n    ngOnInit() {\n        // effect can only be used within an injection context (ex: constructor)\n        runInInjectionContext(this.injector, () => {\n            effect(() => {\n                const impersonating = this.isImpersonating();\n                const token = this.parsedToken();\n\n                // when token refreshes after 5 minutes, it leaves a small moment where there is no token\n                // this prevents dropping inactivtiy timer during that blip\n                if (!token) return;\n\n                if (impersonating) {\n                    if (!this.trackingActive) this.startInactivityTracking();\n                } else {\n                    if (this.trackingActive) this.stopInactivityTracking();\n                }\n            });\n        });\n    }\n\n    ngOnDestroy() {\n        this.stopInactivityTracking();\n    }\n\n    /** Begin listening and start countdown */\n    private startInactivityTracking() {\n        this.activityEvents.forEach((event) => {\n            // 'scroll' on document doesn't bubble; use window + capture to catch nested scrolls\n            const target: Window | Document = event === 'scroll' ? window : document;\n            const options =\n                event === 'scroll'\n                    ? ({ passive: true, capture: true } as AddEventListenerOptions)\n                    : ({ passive: true } as AddEventListenerOptions);\n\n            target.addEventListener(event, this.debouncedResetTimer, options);\n        });\n\n        this.trackingActive = true;\n        this.resetInactivityTimer();\n    }\n\n    /** Remove listeners and clear timers */\n    private stopInactivityTracking() {\n        this.activityEvents.forEach((event) => {\n            const target: Window | Document = event === 'scroll' ? window : document;\n            // IMPORTANT: capture must match how it was added for scroll\n            const options =\n                event === 'scroll' ? ({ capture: true } as AddEventListenerOptions) : undefined;\n\n            target.removeEventListener(event, this.debouncedResetTimer, options);\n        });\n\n        if (this.inactivityTimerId) {\n            clearTimeout(this.inactivityTimerId);\n            this.inactivityTimerId = null;\n        }\n        if (this.debounceTimerId) {\n            clearTimeout(this.debounceTimerId);\n            this.debounceTimerId = null;\n        }\n\n        this.trackingActive = false;\n    }\n\n    /** Reset the inactivity countdown (no-op if not impersonating) */\n    private resetInactivityTimer = () => {\n        if (!this.isImpersonating()) return;\n\n        if (this.inactivityTimerId) clearTimeout(this.inactivityTimerId);\n\n        this.inactivityTimerId = window.setTimeout(() => {\n            this.endImpersonation.emit();\n            this.stopInactivityTracking();\n        }, this.inactivityTimeoutMs);\n    };\n\n    /** Debounce activity to avoid hammering resets during event storms */\n    private debouncedResetTimer = () => {\n        if (this.debounceTimerId) clearTimeout(this.debounceTimerId);\n\n        this.debounceTimerId = window.setTimeout(() => {\n            this.resetInactivityTimer();\n        }, this.debounceDelayMs);\n    };\n}\n","<lib-impersonation-banner\n    [accessTokenPayload]=\"accessTokenPayload()\"\n    (endImpersonation)=\"endImpersonation.emit()\"\n    [personBaseUri]=\"personBaseUri()\"\n    [myAccountApiBaseUri]=\"myAccountApiBaseUri()\"\n></lib-impersonation-banner>\n<lib-hbll-header\n    [name]=\"name()\"\n    [showImpersonateButton]=\"showImpersonateButton()\"\n    (openImpersonationModal)=\"showImpersonationModal = true\"\n    (login)=\"login.emit()\"\n    (logout)=\"logout.emit()\"\n    [mainsitebaseurl]=\"mainSiteBaseUrl()\"\n/>\n<lib-impersonate-modal\n    [showModal]=\"showImpersonationModal\"\n    [oidcBaseUri]=\"oidcBaseUri()\"\n    [oidcDefaultIdp]=\"oidcDefaultIdp()\"\n    [accessTokenPayload]=\"accessTokenPayload()\"\n    (dismiss)=\"showImpersonationModal = false\"\n    (init)=\"showImpersonationModal = true\"\n></lib-impersonate-modal>\n"]}
@@ -1339,7 +1339,13 @@ class HeaderWithImpersonationComponent {
1339
1339
  // effect can only be used within an injection context (ex: constructor)
1340
1340
  runInInjectionContext(this.injector, () => {
1341
1341
  effect(() => {
1342
- if (this.isImpersonating()) {
1342
+ const impersonating = this.isImpersonating();
1343
+ const token = this.parsedToken();
1344
+ // when token refreshes after 5 minutes, it leaves a small moment where there is no token
1345
+ // this prevents dropping inactivtiy timer during that blip
1346
+ if (!token)
1347
+ return;
1348
+ if (impersonating) {
1343
1349
  if (!this.trackingActive)
1344
1350
  this.startInactivityTracking();
1345
1351
  }
@@ -1370,7 +1376,9 @@ class HeaderWithImpersonationComponent {
1370
1376
  stopInactivityTracking() {
1371
1377
  this.activityEvents.forEach((event) => {
1372
1378
  const target = event === 'scroll' ? window : document;
1373
- target.removeEventListener(event, this.debouncedResetTimer);
1379
+ // IMPORTANT: capture must match how it was added for scroll
1380
+ const options = event === 'scroll' ? { capture: true } : undefined;
1381
+ target.removeEventListener(event, this.debouncedResetTimer, options);
1374
1382
  });
1375
1383
  if (this.inactivityTimerId) {
1376
1384
  clearTimeout(this.inactivityTimerId);