UNPKG

angular-auth-oidc-client

Version:
145 lines 21.1 kB
import { DOCUMENT } from '@angular/common'; import { Injectable, inject } from '@angular/core'; import { Subject } from 'rxjs'; import { LoggerService } from '../../logging/logger.service'; import { StoragePersistenceService } from '../../storage/storage-persistence.service'; import * as i0 from "@angular/core"; export class PopUpService { constructor() { this.loggerService = inject(LoggerService); this.storagePersistenceService = inject(StoragePersistenceService); this.document = inject(DOCUMENT); this.STORAGE_IDENTIFIER = 'popupauth'; this.popUp = null; this.handle = -1; this.resultInternal$ = new Subject(); } get result$() { return this.resultInternal$.asObservable(); } get windowInternal() { return this.document.defaultView; } isCurrentlyInPopup(config) { if (this.canAccessSessionStorage()) { const popup = this.storagePersistenceService.read(this.STORAGE_IDENTIFIER, config); const windowIdentifier = this.windowInternal; if (!windowIdentifier) { return false; } return (Boolean(windowIdentifier.opener) && windowIdentifier.opener !== windowIdentifier && Boolean(popup)); } return false; } openPopUp(url, popupOptions, config) { const optionsToPass = this.getOptions(popupOptions); this.storagePersistenceService.write(this.STORAGE_IDENTIFIER, 'true', config); const windowIdentifier = this.windowInternal; if (!windowIdentifier) { return; } if (!url) { this.loggerService.logError(config, 'Could not open popup, url is empty'); return; } this.popUp = windowIdentifier.open(url, '_blank', optionsToPass); if (!this.popUp) { this.storagePersistenceService.remove(this.STORAGE_IDENTIFIER, config); this.loggerService.logError(config, 'Could not open popup'); return; } this.loggerService.logDebug(config, 'Opened popup with url ' + url); const listener = (event) => { if (!event?.data || typeof event.data !== 'string') { if (config.disableCleaningPopupOnInvalidMessage) { return; } this.cleanUp(listener, config); return; } this.loggerService.logDebug(config, 'Received message from popup with url ' + event.data); this.resultInternal$.next({ userClosed: false, receivedUrl: event.data }); this.cleanUp(listener, config); }; windowIdentifier.addEventListener('message', listener, false); this.handle = windowIdentifier.setInterval(() => { if (this.popUp?.closed) { this.resultInternal$.next({ userClosed: true, receivedUrl: '' }); this.cleanUp(listener, config); } }, 200); } sendMessageToMainWindow(url, config) { const windowIdentifier = this.windowInternal; if (!windowIdentifier) { return; } if (windowIdentifier.opener) { const href = windowIdentifier.location.href; this.sendMessage(url, href, config); } } cleanUp(listener, config) { const windowIdentifier = this.windowInternal; if (!windowIdentifier) { return; } windowIdentifier.removeEventListener('message', listener, false); windowIdentifier.clearInterval(this.handle); if (this.popUp) { this.storagePersistenceService.remove(this.STORAGE_IDENTIFIER, config); this.popUp.close(); this.popUp = null; } } sendMessage(url, href, config) { const windowIdentifier = this.windowInternal; if (!windowIdentifier) { return; } if (!url) { this.loggerService.logDebug(config, `Can not send message to parent, no url: '${url}'`); return; } windowIdentifier.opener.postMessage(url, href); } getOptions(popupOptions) { const popupDefaultOptions = { width: 500, height: 500, left: 50, top: 50, }; const options = { ...popupDefaultOptions, ...(popupOptions || {}), }; const windowIdentifier = this.windowInternal; if (!windowIdentifier) { return ''; } const width = options.width || popupDefaultOptions.width; const height = options.height || popupDefaultOptions.height; const left = windowIdentifier.screenLeft + (windowIdentifier.outerWidth - width) / 2; const top = windowIdentifier.screenTop + (windowIdentifier.outerHeight - height) / 2; options.left = left; options.top = top; return Object.entries(options) .map(([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`) .join(','); } canAccessSessionStorage() { return (typeof navigator !== 'undefined' && navigator.cookieEnabled && typeof Storage !== 'undefined'); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: PopUpService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: PopUpService, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: PopUpService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popup.service.js","sourceRoot":"","sources":["../../../../../../projects/angular-auth-oidc-client/src/lib/login/popup/popup.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\r\nimport { Injectable, inject } from '@angular/core';\r\nimport { Observable, Subject } from 'rxjs';\r\nimport { OpenIdConfiguration } from '../../config/openid-configuration';\r\nimport { LoggerService } from '../../logging/logger.service';\r\nimport { StoragePersistenceService } from '../../storage/storage-persistence.service';\r\nimport { PopupOptions } from './popup-options';\r\nimport { PopupResult } from './popup-result';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class PopUpService {\r\n  private readonly loggerService = inject(LoggerService);\r\n\r\n  private readonly storagePersistenceService = inject(\r\n    StoragePersistenceService\r\n  );\r\n\r\n  private readonly document = inject(DOCUMENT);\r\n\r\n  private readonly STORAGE_IDENTIFIER = 'popupauth';\r\n\r\n  private popUp: Window | null = null;\r\n\r\n  private handle = -1;\r\n\r\n  private readonly resultInternal$ = new Subject<PopupResult>();\r\n\r\n  get result$(): Observable<PopupResult> {\r\n    return this.resultInternal$.asObservable();\r\n  }\r\n\r\n  private get windowInternal(): Window | null {\r\n    return this.document.defaultView;\r\n  }\r\n\r\n  isCurrentlyInPopup(config: OpenIdConfiguration): boolean {\r\n    if (this.canAccessSessionStorage()) {\r\n      const popup = this.storagePersistenceService.read(\r\n        this.STORAGE_IDENTIFIER,\r\n        config\r\n      );\r\n\r\n      const windowIdentifier = this.windowInternal;\r\n\r\n      if (!windowIdentifier) {\r\n        return false;\r\n      }\r\n\r\n      return (\r\n        Boolean(windowIdentifier.opener) &&\r\n        windowIdentifier.opener !== windowIdentifier &&\r\n        Boolean(popup)\r\n      );\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  openPopUp(\r\n    url: string | null,\r\n    popupOptions: PopupOptions | undefined,\r\n    config: OpenIdConfiguration\r\n  ): void {\r\n    const optionsToPass = this.getOptions(popupOptions);\r\n\r\n    this.storagePersistenceService.write(\r\n      this.STORAGE_IDENTIFIER,\r\n      'true',\r\n      config\r\n    );\r\n\r\n    const windowIdentifier = this.windowInternal;\r\n\r\n    if (!windowIdentifier) {\r\n      return;\r\n    }\r\n\r\n    if (!url) {\r\n      this.loggerService.logError(config, 'Could not open popup, url is empty');\r\n\r\n      return;\r\n    }\r\n\r\n    this.popUp = windowIdentifier.open(url, '_blank', optionsToPass);\r\n\r\n    if (!this.popUp) {\r\n      this.storagePersistenceService.remove(this.STORAGE_IDENTIFIER, config);\r\n      this.loggerService.logError(config, 'Could not open popup');\r\n\r\n      return;\r\n    }\r\n\r\n    this.loggerService.logDebug(config, 'Opened popup with url ' + url);\r\n\r\n    const listener = (event: MessageEvent): void => {\r\n      if (!event?.data || typeof event.data !== 'string') {\r\n        if (config.disableCleaningPopupOnInvalidMessage) {\r\n          return;\r\n        }\r\n        this.cleanUp(listener, config);\r\n\r\n        return;\r\n      }\r\n\r\n      this.loggerService.logDebug(\r\n        config,\r\n        'Received message from popup with url ' + event.data\r\n      );\r\n\r\n      this.resultInternal$.next({ userClosed: false, receivedUrl: event.data });\r\n\r\n      this.cleanUp(listener, config);\r\n    };\r\n\r\n    windowIdentifier.addEventListener('message', listener, false);\r\n\r\n    this.handle = windowIdentifier.setInterval(() => {\r\n      if (this.popUp?.closed) {\r\n        this.resultInternal$.next({ userClosed: true, receivedUrl: '' });\r\n\r\n        this.cleanUp(listener, config);\r\n      }\r\n    }, 200);\r\n  }\r\n\r\n  sendMessageToMainWindow(url: string, config: OpenIdConfiguration): void {\r\n    const windowIdentifier = this.windowInternal;\r\n\r\n    if (!windowIdentifier) {\r\n      return;\r\n    }\r\n\r\n    if (windowIdentifier.opener) {\r\n      const href = windowIdentifier.location.href;\r\n\r\n      this.sendMessage(url, href, config);\r\n    }\r\n  }\r\n\r\n  private cleanUp(listener: any, config: OpenIdConfiguration): void {\r\n    const windowIdentifier = this.windowInternal;\r\n\r\n    if (!windowIdentifier) {\r\n      return;\r\n    }\r\n\r\n    windowIdentifier.removeEventListener('message', listener, false);\r\n    windowIdentifier.clearInterval(this.handle);\r\n\r\n    if (this.popUp) {\r\n      this.storagePersistenceService.remove(this.STORAGE_IDENTIFIER, config);\r\n      this.popUp.close();\r\n      this.popUp = null;\r\n    }\r\n  }\r\n\r\n  private sendMessage(\r\n    url: string,\r\n    href: string,\r\n    config: OpenIdConfiguration\r\n  ): void {\r\n    const windowIdentifier = this.windowInternal;\r\n\r\n    if (!windowIdentifier) {\r\n      return;\r\n    }\r\n\r\n    if (!url) {\r\n      this.loggerService.logDebug(\r\n        config,\r\n        `Can not send message to parent, no url: '${url}'`\r\n      );\r\n\r\n      return;\r\n    }\r\n\r\n    windowIdentifier.opener.postMessage(url, href);\r\n  }\r\n\r\n  private getOptions(popupOptions: PopupOptions | undefined): string {\r\n    const popupDefaultOptions = {\r\n      width: 500,\r\n      height: 500,\r\n      left: 50,\r\n      top: 50,\r\n    };\r\n    const options: PopupOptions = {\r\n      ...popupDefaultOptions,\r\n      ...(popupOptions || {}),\r\n    };\r\n    const windowIdentifier = this.windowInternal;\r\n\r\n    if (!windowIdentifier) {\r\n      return '';\r\n    }\r\n\r\n    const width = options.width || popupDefaultOptions.width;\r\n    const height = options.height || popupDefaultOptions.height;\r\n\r\n    const left: number =\r\n      windowIdentifier.screenLeft + (windowIdentifier.outerWidth - width) / 2;\r\n    const top: number =\r\n      windowIdentifier.screenTop + (windowIdentifier.outerHeight - height) / 2;\r\n\r\n    options.left = left;\r\n    options.top = top;\r\n\r\n    return Object.entries(options)\r\n      .map(\r\n        ([key, value]) =>\r\n          `${encodeURIComponent(key)}=${encodeURIComponent(value)}`\r\n      )\r\n      .join(',');\r\n  }\r\n\r\n  private canAccessSessionStorage(): boolean {\r\n    return (\r\n      typeof navigator !== 'undefined' &&\r\n      navigator.cookieEnabled &&\r\n      typeof Storage !== 'undefined'\r\n    );\r\n  }\r\n}\r\n"]}