UNPKG

ng2-idle-timeout

Version:

Zoneless-friendly session timeout management for Angular 16-20.

160 lines 23.1 kB
import { DestroyRef, Injectable, NgZone, inject } from '@angular/core'; import { DOCUMENT } from '@angular/common'; import { Subject } from 'rxjs'; import { DEFAULT_SESSION_TIMEOUT_CONFIG } from '../defaults'; import * as i0 from "@angular/core"; const PASSIVE_EVENT_OPTIONS = { passive: true }; const DOM_EVENT_SPECS = { mousemove: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, mousedown: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, click: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, wheel: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, scroll: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, keydown: { target: 'document', debounce: 'key' }, keyup: { target: 'document', debounce: 'key' }, touchstart: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, touchend: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, touchmove: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS }, visibilitychange: { target: 'document', debounce: 'none' } }; export class ActivityDomService { destroyRef = inject(DestroyRef); zone = inject(NgZone); document = inject(DOCUMENT, { optional: true }); eventsSubject = new Subject(); events$ = this.eventsSubject.asObservable(); config = DEFAULT_SESSION_TIMEOUT_CONFIG; listenerCleanupByEvent = new Map(); destroyHookRegistered = false; lastMouseEventAt = 0; lastKeyEventAt = 0; updateConfig(config) { this.config = config; this.syncEventListeners(); } syncEventListeners() { const doc = this.document; if (!doc || typeof window === 'undefined') { this.cleanupListeners(); return; } const win = doc.defaultView; if (!win) { this.cleanupListeners(); return; } const desired = new Set(configuredEvents(this.config.domActivityEvents)); for (const [eventName, cleanup] of this.listenerCleanupByEvent) { if (!desired.has(eventName)) { cleanup(); this.listenerCleanupByEvent.delete(eventName); } } const toAdd = []; for (const eventName of desired) { if (!this.listenerCleanupByEvent.has(eventName)) { toAdd.push(eventName); } } if (toAdd.length === 0) { this.ensureDestroyHook(); return; } this.zone.runOutsideAngular(() => { for (const eventName of toAdd) { const spec = DOM_EVENT_SPECS[eventName]; if (!spec) { continue; } const target = spec.target === 'window' ? win : doc; const handler = (event) => this.handleEvent(event, spec.debounce); target.addEventListener(eventName, handler, spec.options); this.listenerCleanupByEvent.set(eventName, () => { target.removeEventListener(eventName, handler, spec.options); }); } }); this.ensureDestroyHook(); } ensureDestroyHook() { if (this.destroyHookRegistered) { return; } this.destroyRef.onDestroy(() => { this.cleanupListeners(); }); this.destroyHookRegistered = true; } handleEvent(event, debounce) { if (!this.document) { return; } if (event.type === 'visibilitychange' && this.document.visibilityState !== 'visible') { return; } const now = Date.now(); if (debounce === 'mouse') { if (now - this.lastMouseEventAt < this.config.debounceMouseMs) { return; } this.lastMouseEventAt = now; } else if (debounce === 'key') { if (now - this.lastKeyEventAt < this.config.debounceKeyMs) { return; } this.lastKeyEventAt = now; } if (this.document.visibilityState === 'hidden' && event.type !== 'visibilitychange') { return; } const meta = { type: event.type }; if (typeof KeyboardEvent !== 'undefined' && event instanceof KeyboardEvent) { meta['key'] = event.key; meta['ctrlKey'] = event.ctrlKey; meta['shiftKey'] = event.shiftKey; meta['altKey'] = event.altKey; } else if (typeof MouseEvent !== 'undefined' && event instanceof MouseEvent) { meta['button'] = event.button; meta['clientX'] = Math.round(event.clientX); meta['clientY'] = Math.round(event.clientY); } else if (typeof TouchEvent !== 'undefined' && event instanceof TouchEvent) { meta['touches'] = event.touches?.length ?? 0; } else if (typeof InputEvent !== 'undefined' && event instanceof InputEvent) { meta['inputType'] = event.inputType; } const target = event.target; if (target instanceof Element) { meta['target'] = target.tagName.toLowerCase(); } this.eventsSubject.next({ source: 'dom', at: now, meta }); } cleanupListeners() { for (const [, cleanup] of this.listenerCleanupByEvent) { cleanup(); } this.listenerCleanupByEvent.clear(); } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.0", ngImport: i0, type: ActivityDomService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.0", ngImport: i0, type: ActivityDomService, providedIn: 'root' }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.0", ngImport: i0, type: ActivityDomService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }] }); function configuredEvents(events) { if (!events) { return DEFAULT_SESSION_TIMEOUT_CONFIG.domActivityEvents; } return events; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"activity-dom.service.js","sourceRoot":"","sources":["../../../../src/lib/services/activity-dom.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DestroyRef, Injectable, NgZone, inject } from '@angular/core';\r\nimport { DOCUMENT } from '@angular/common';\r\nimport { Subject } from 'rxjs';\r\n\r\nimport type { ActivityEvent } from '../models/activity-event';\r\nimport type { SessionTimeoutConfig, DomActivityEventName } from '../models/session-timeout-config';\r\nimport { DEFAULT_SESSION_TIMEOUT_CONFIG } from '../defaults';\r\n\r\ntype EventDebounce = 'mouse' | 'key' | 'none';\r\n\r\ninterface DomEventSpec {\r\n  target: 'document' | 'window';\r\n  debounce: EventDebounce;\r\n  options?: AddEventListenerOptions;\r\n}\r\n\r\nconst PASSIVE_EVENT_OPTIONS: AddEventListenerOptions = { passive: true };\r\n\r\nconst DOM_EVENT_SPECS: Record<DomActivityEventName, DomEventSpec> = {\r\n  mousemove: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  mousedown: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  click: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  wheel: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  scroll: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  keydown: { target: 'document', debounce: 'key' },\r\n  keyup: { target: 'document', debounce: 'key' },\r\n  touchstart: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  touchend: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  touchmove: { target: 'document', debounce: 'mouse', options: PASSIVE_EVENT_OPTIONS },\r\n  visibilitychange: { target: 'document', debounce: 'none' }\r\n} as const;\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class ActivityDomService {\r\n  private readonly destroyRef = inject(DestroyRef);\r\n  private readonly zone = inject(NgZone);\r\n  private readonly document = inject(DOCUMENT, { optional: true }) as Document | undefined;\r\n\r\n  private readonly eventsSubject = new Subject<ActivityEvent>();\r\n  readonly events$ = this.eventsSubject.asObservable();\r\n\r\n  private config: SessionTimeoutConfig = DEFAULT_SESSION_TIMEOUT_CONFIG;\r\n  private readonly listenerCleanupByEvent = new Map<DomActivityEventName, () => void>();\r\n  private destroyHookRegistered = false;\r\n  private lastMouseEventAt = 0;\r\n  private lastKeyEventAt = 0;\r\n\r\n  updateConfig(config: SessionTimeoutConfig): void {\r\n    this.config = config;\r\n    this.syncEventListeners();\r\n  }\r\n\r\n  private syncEventListeners(): void {\r\n    const doc = this.document;\r\n    if (!doc || typeof window === 'undefined') {\r\n      this.cleanupListeners();\r\n      return;\r\n    }\r\n\r\n    const win = doc.defaultView;\r\n    if (!win) {\r\n      this.cleanupListeners();\r\n      return;\r\n    }\r\n\r\n    const desired = new Set(configuredEvents(this.config.domActivityEvents));\r\n\r\n    for (const [eventName, cleanup] of this.listenerCleanupByEvent) {\r\n      if (!desired.has(eventName)) {\r\n        cleanup();\r\n        this.listenerCleanupByEvent.delete(eventName);\r\n      }\r\n    }\r\n\r\n    const toAdd: DomActivityEventName[] = [];\r\n    for (const eventName of desired) {\r\n      if (!this.listenerCleanupByEvent.has(eventName)) {\r\n        toAdd.push(eventName);\r\n      }\r\n    }\r\n\r\n    if (toAdd.length === 0) {\r\n      this.ensureDestroyHook();\r\n      return;\r\n    }\r\n\r\n    this.zone.runOutsideAngular(() => {\r\n      for (const eventName of toAdd) {\r\n        const spec = DOM_EVENT_SPECS[eventName];\r\n        if (!spec) {\r\n          continue;\r\n        }\r\n        const target = spec.target === 'window' ? win : doc;\r\n        const handler = (event: Event) => this.handleEvent(event, spec.debounce);\r\n        target.addEventListener(eventName, handler, spec.options);\r\n        this.listenerCleanupByEvent.set(eventName, () => {\r\n          target.removeEventListener(eventName, handler, spec.options);\r\n        });\r\n      }\r\n    });\r\n\r\n    this.ensureDestroyHook();\r\n  }\r\n\r\n  private ensureDestroyHook(): void {\r\n    if (this.destroyHookRegistered) {\r\n      return;\r\n    }\r\n    this.destroyRef.onDestroy(() => {\r\n      this.cleanupListeners();\r\n    });\r\n    this.destroyHookRegistered = true;\r\n  }\r\n\r\n  private handleEvent(event: Event, debounce: EventDebounce): void {\r\n    if (!this.document) {\r\n      return;\r\n    }\r\n\r\n    if (event.type === 'visibilitychange' && this.document.visibilityState !== 'visible') {\r\n      return;\r\n    }\r\n\r\n    const now = Date.now();\r\n\r\n    if (debounce === 'mouse') {\r\n      if (now - this.lastMouseEventAt < this.config.debounceMouseMs) {\r\n        return;\r\n      }\r\n      this.lastMouseEventAt = now;\r\n    } else if (debounce === 'key') {\r\n      if (now - this.lastKeyEventAt < this.config.debounceKeyMs) {\r\n        return;\r\n      }\r\n      this.lastKeyEventAt = now;\r\n    }\r\n\r\n    if (this.document.visibilityState === 'hidden' && event.type !== 'visibilitychange') {\r\n      return;\r\n    }\r\n\r\n    const meta: Record<string, unknown> = {\r\n      type: event.type\r\n    };\r\n\r\n    if (typeof KeyboardEvent !== 'undefined' && event instanceof KeyboardEvent) {\r\n      meta['key'] = event.key;\r\n      meta['ctrlKey'] = event.ctrlKey;\r\n      meta['shiftKey'] = event.shiftKey;\r\n      meta['altKey'] = event.altKey;\r\n    } else if (typeof MouseEvent !== 'undefined' && event instanceof MouseEvent) {\r\n      meta['button'] = event.button;\r\n      meta['clientX'] = Math.round(event.clientX);\r\n      meta['clientY'] = Math.round(event.clientY);\r\n    } else if (typeof TouchEvent !== 'undefined' && event instanceof TouchEvent) {\r\n      meta['touches'] = event.touches?.length ?? 0;\r\n    } else if (typeof InputEvent !== 'undefined' && event instanceof InputEvent) {\r\n      meta['inputType'] = event.inputType;\r\n    }\r\n\r\n    const target = event.target as Element | null;\r\n    if (target instanceof Element) {\r\n      meta['target'] = target.tagName.toLowerCase();\r\n    }\r\n\r\n    this.eventsSubject.next({\r\n      source: 'dom',\r\n      at: now,\r\n      meta\r\n    });\r\n  }\r\n\r\n  private cleanupListeners(): void {\r\n    for (const [, cleanup] of this.listenerCleanupByEvent) {\r\n      cleanup();\r\n    }\r\n    this.listenerCleanupByEvent.clear();\r\n  }\r\n}\r\n\r\nfunction configuredEvents(events: readonly DomActivityEventName[] | undefined): readonly DomActivityEventName[] {\r\n  if (!events) {\r\n    return DEFAULT_SESSION_TIMEOUT_CONFIG.domActivityEvents;\r\n  }\r\n  return events;\r\n}\r\n"]}