angular-auth-oidc-client
Version:
Angular Lib for OpenID Connect & OAuth2
145 lines • 21.1 kB
JavaScript
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"]}