2018-01-13 03:31:46 +01:00
|
|
|
import { BrowserApi } from "../browser/browserApi";
|
2019-08-19 17:58:43 +02:00
|
|
|
import { SafariApp } from "../browser/safariApp";
|
2018-01-13 03:31:46 +01:00
|
|
|
|
2021-06-07 19:25:37 +02:00
|
|
|
import { DeviceType } from "jslib-common/enums/deviceType";
|
2021-10-04 22:30:31 +02:00
|
|
|
import { ThemeType } from "jslib-common/enums/themeType";
|
2018-01-09 20:26:20 +01:00
|
|
|
|
2021-06-07 19:25:37 +02:00
|
|
|
import { MessagingService } from "jslib-common/abstractions/messaging.service";
|
|
|
|
import { PlatformUtilsService } from "jslib-common/abstractions/platformUtils.service";
|
2021-10-04 22:30:31 +02:00
|
|
|
import { StorageService } from "jslib-common/abstractions/storage.service";
|
|
|
|
|
|
|
|
import { ConstantsService } from "jslib-common/services/constants.service";
|
2018-01-04 18:32:10 +01:00
|
|
|
|
2018-04-14 15:39:47 +02:00
|
|
|
const DialogPromiseExpiration = 600000; // 10 minutes
|
2018-04-10 20:20:03 +02:00
|
|
|
|
2018-01-09 20:26:20 +01:00
|
|
|
export default class BrowserPlatformUtilsService implements PlatformUtilsService {
|
2018-03-21 16:21:42 +01:00
|
|
|
identityClientId: string = "browser";
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-04-10 20:20:03 +02:00
|
|
|
private showDialogResolves = new Map<number, { resolve: (value: boolean) => void; date: Date }>();
|
2021-05-03 20:56:38 +02:00
|
|
|
private passwordDialogResolves = new Map<
|
|
|
|
number,
|
|
|
|
{ tryResolve: (canceled: boolean, password: string) => Promise<boolean>; date: Date }
|
|
|
|
>();
|
2018-01-09 20:26:20 +01:00
|
|
|
private deviceCache: DeviceType = null;
|
2020-08-31 22:33:01 +02:00
|
|
|
private prefersColorSchemeDark = window.matchMedia("(prefers-color-scheme: dark)");
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2021-10-04 22:30:31 +02:00
|
|
|
constructor(
|
|
|
|
private messagingService: MessagingService,
|
|
|
|
private storageService: StorageService,
|
2020-10-09 17:16:15 +02:00
|
|
|
private clipboardWriteCallback: (clipboardValue: string, clearMs: number) => void,
|
2020-10-12 18:01:34 +02:00
|
|
|
private biometricCallback: () => Promise<boolean>
|
|
|
|
) {}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-01-09 20:26:20 +01:00
|
|
|
getDevice(): DeviceType {
|
2018-07-09 15:12:41 +02:00
|
|
|
if (this.deviceCache) {
|
2018-01-05 17:13:24 +01:00
|
|
|
return this.deviceCache;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
2021-02-03 20:36:05 +01:00
|
|
|
navigator.userAgent.indexOf(" Firefox/") !== -1 ||
|
|
|
|
navigator.userAgent.indexOf(" Gecko/") !== -1
|
2021-12-21 15:43:35 +01:00
|
|
|
) {
|
2018-07-09 15:12:41 +02:00
|
|
|
this.deviceCache = DeviceType.FirefoxExtension;
|
2021-02-03 20:36:05 +01:00
|
|
|
} else if (
|
2018-01-04 18:32:10 +01:00
|
|
|
(!!(window as any).opr && !!opr.addons) ||
|
|
|
|
!!(window as any).opera ||
|
|
|
|
navigator.userAgent.indexOf(" OPR/") >= 0
|
2021-12-21 15:43:35 +01:00
|
|
|
) {
|
2018-07-09 15:12:41 +02:00
|
|
|
this.deviceCache = DeviceType.OperaExtension;
|
2018-01-05 17:13:24 +01:00
|
|
|
} else if (navigator.userAgent.indexOf(" Edg/") !== -1) {
|
2018-07-09 15:12:41 +02:00
|
|
|
this.deviceCache = DeviceType.EdgeExtension;
|
2018-01-04 18:32:10 +01:00
|
|
|
} else if (navigator.userAgent.indexOf(" Vivaldi/") !== -1) {
|
2018-07-09 15:12:41 +02:00
|
|
|
this.deviceCache = DeviceType.VivaldiExtension;
|
2018-01-16 03:40:42 +01:00
|
|
|
} else if ((window as any).chrome && navigator.userAgent.indexOf(" Chrome/") !== -1) {
|
2018-07-09 15:12:41 +02:00
|
|
|
this.deviceCache = DeviceType.ChromeExtension;
|
2021-02-03 20:36:05 +01:00
|
|
|
} else if (navigator.userAgent.indexOf(" Safari/") !== -1) {
|
2018-01-05 17:13:24 +01:00
|
|
|
this.deviceCache = DeviceType.SafariExtension;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-05 17:13:24 +01:00
|
|
|
return this.deviceCache;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-05 17:13:24 +01:00
|
|
|
getDeviceString(): string {
|
2018-07-09 15:30:37 +02:00
|
|
|
const device = DeviceType[this.getDevice()].toLowerCase();
|
|
|
|
return device.replace("extension", "");
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 18:32:10 +01:00
|
|
|
isFirefox(): boolean {
|
2018-07-09 15:12:41 +02:00
|
|
|
return this.getDevice() === DeviceType.FirefoxExtension;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 18:32:10 +01:00
|
|
|
isChrome(): boolean {
|
2018-07-09 15:12:41 +02:00
|
|
|
return this.getDevice() === DeviceType.ChromeExtension;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 18:32:10 +01:00
|
|
|
isEdge(): boolean {
|
2018-07-09 15:12:41 +02:00
|
|
|
return this.getDevice() === DeviceType.EdgeExtension;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 18:32:10 +01:00
|
|
|
isOpera(): boolean {
|
2018-07-09 15:12:41 +02:00
|
|
|
return this.getDevice() === DeviceType.OperaExtension;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 18:32:10 +01:00
|
|
|
isVivaldi(): boolean {
|
2018-07-09 15:12:41 +02:00
|
|
|
return this.getDevice() === DeviceType.VivaldiExtension;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 18:32:10 +01:00
|
|
|
isSafari(): boolean {
|
2018-07-09 15:12:41 +02:00
|
|
|
return this.getDevice() === DeviceType.SafariExtension;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-06-08 05:36:58 +02:00
|
|
|
isIE(): boolean {
|
2019-08-20 19:54:10 +02:00
|
|
|
return false;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-02-27 05:51:17 +01:00
|
|
|
isMacAppStore(): boolean {
|
2019-08-20 19:54:10 +02:00
|
|
|
return false;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2019-08-20 19:54:10 +02:00
|
|
|
async isViewOpen(): Promise<boolean> {
|
|
|
|
if (await BrowserApi.isPopupOpen()) {
|
|
|
|
return true;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-13 03:31:46 +01:00
|
|
|
if (this.isSafari()) {
|
2019-08-20 19:54:10 +02:00
|
|
|
return false;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 22:49:58 +01:00
|
|
|
const sidebarView = this.sidebarViewName();
|
|
|
|
const sidebarOpen =
|
|
|
|
sidebarView != null && chrome.extension.getViews({ type: sidebarView }).length > 0;
|
2018-01-05 17:13:24 +01:00
|
|
|
if (sidebarOpen) {
|
2019-08-20 19:54:10 +02:00
|
|
|
return true;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-05 17:13:24 +01:00
|
|
|
const tabOpen = chrome.extension.getViews({ type: "tab" }).length > 0;
|
2019-08-20 19:54:10 +02:00
|
|
|
return tabOpen;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-06-09 20:50:29 +02:00
|
|
|
lockTimeout(): number {
|
|
|
|
return null;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-25 20:31:54 +01:00
|
|
|
launchUri(uri: string, options?: any): void {
|
|
|
|
BrowserApi.createNewTab(uri, options && options.extensionPage === true);
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-25 20:31:54 +01:00
|
|
|
saveFile(win: Window, blobData: any, blobOptions: any, fileName: string): void {
|
|
|
|
BrowserApi.downloadFile(win, blobData, blobOptions, fileName);
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2021-04-07 20:43:07 +02:00
|
|
|
getApplicationVersion(): Promise<string> {
|
|
|
|
return Promise.resolve(BrowserApi.getApplicationVersion());
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2021-03-17 22:14:26 +01:00
|
|
|
supportsWebAuthn(win: Window): boolean {
|
|
|
|
return typeof PublicKeyCredential !== "undefined";
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-05-16 21:30:36 +02:00
|
|
|
supportsDuo(): boolean {
|
|
|
|
return true;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-10-03 15:46:00 +02:00
|
|
|
showToast(
|
|
|
|
type: "error" | "success" | "warning" | "info",
|
|
|
|
title: string,
|
|
|
|
text: string | string[],
|
|
|
|
options?: any
|
2021-12-21 15:43:35 +01:00
|
|
|
): void {
|
2018-10-03 05:33:56 +02:00
|
|
|
this.messagingService.send("showToast", {
|
|
|
|
text: text,
|
2018-04-10 20:20:03 +02:00
|
|
|
title: title,
|
|
|
|
type: type,
|
2018-10-03 05:33:56 +02:00
|
|
|
options: options,
|
2021-12-21 15:43:35 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-03-04 22:06:02 +01:00
|
|
|
showDialog(
|
|
|
|
body: string,
|
|
|
|
title?: string,
|
|
|
|
confirmText?: string,
|
|
|
|
cancelText?: string,
|
|
|
|
type?: string,
|
|
|
|
bodyIsHtml: boolean = false
|
2021-12-21 15:43:35 +01:00
|
|
|
) {
|
2018-04-10 20:20:03 +02:00
|
|
|
const dialogId = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER);
|
|
|
|
this.messagingService.send("showDialog", {
|
2021-03-04 22:06:02 +01:00
|
|
|
text: bodyIsHtml ? null : body,
|
|
|
|
html: bodyIsHtml ? body : null,
|
2018-04-10 20:20:03 +02:00
|
|
|
title: title,
|
|
|
|
confirmText: confirmText,
|
|
|
|
cancelText: cancelText,
|
|
|
|
type: type,
|
|
|
|
dialogId: dialogId,
|
2021-12-21 15:43:35 +01:00
|
|
|
});
|
2021-02-10 16:40:15 +01:00
|
|
|
return new Promise<boolean>((resolve) => {
|
2018-04-10 20:20:03 +02:00
|
|
|
this.showDialogResolves.set(dialogId, { resolve: resolve, date: new Date() });
|
2021-12-21 15:43:35 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-02-13 23:23:30 +01:00
|
|
|
isDev(): boolean {
|
2018-04-14 19:32:56 +02:00
|
|
|
return process.env.ENV === "development";
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-06-30 19:51:20 +02:00
|
|
|
isSelfHost(): boolean {
|
|
|
|
return false;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-02-16 20:00:41 +01:00
|
|
|
copyToClipboard(text: string, options?: any): void {
|
2018-08-13 15:44:59 +02:00
|
|
|
let win = window;
|
|
|
|
let doc = window.document;
|
2018-08-17 18:25:09 +02:00
|
|
|
if (options && (options.window || options.win)) {
|
|
|
|
win = options.window || options.win;
|
2018-08-13 15:44:59 +02:00
|
|
|
doc = win.document;
|
|
|
|
} else if (options && options.doc) {
|
|
|
|
doc = options.doc;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
2019-05-30 15:37:09 +02:00
|
|
|
const clearing = options ? !!options.clearing : false;
|
2019-02-27 15:28:16 +01:00
|
|
|
const clearMs: number = options && options.clearMs ? options.clearMs : null;
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2021-02-04 23:34:10 +01:00
|
|
|
if (this.isSafari()) {
|
|
|
|
SafariApp.sendMessageToApp("copyToClipboard", text).then(() => {
|
2019-05-30 15:37:09 +02:00
|
|
|
if (!clearing && this.clipboardWriteCallback != null) {
|
2019-02-27 15:28:16 +01:00
|
|
|
this.clipboardWriteCallback(text, clearMs);
|
2018-01-04 18:32:10 +01:00
|
|
|
}
|
2018-07-09 15:12:41 +02:00
|
|
|
});
|
2018-01-04 18:32:10 +01:00
|
|
|
} else if (
|
|
|
|
this.isFirefox() &&
|
|
|
|
(win as any).navigator.clipboard &&
|
2020-09-15 16:50:45 +02:00
|
|
|
(win as any).navigator.clipboard.writeText
|
2018-01-04 18:32:10 +01:00
|
|
|
) {
|
2018-01-16 03:40:42 +01:00
|
|
|
(win as any).navigator.clipboard.writeText(text).then(() => {
|
2021-02-03 20:36:05 +01:00
|
|
|
if (!clearing && this.clipboardWriteCallback != null) {
|
|
|
|
this.clipboardWriteCallback(text, clearMs);
|
2018-01-04 18:32:10 +01:00
|
|
|
}
|
2018-01-05 17:13:24 +01:00
|
|
|
});
|
|
|
|
} else if ((win as any).clipboardData && (win as any).clipboardData.setData) {
|
2018-07-09 15:12:41 +02:00
|
|
|
// IE specific code path to prevent textarea being shown while dialog is visible.
|
|
|
|
(win as any).clipboardData.setData("Text", text);
|
2018-01-04 18:32:10 +01:00
|
|
|
if (!clearing && this.clipboardWriteCallback != null) {
|
2018-07-09 15:12:41 +02:00
|
|
|
this.clipboardWriteCallback(text, clearMs);
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
2018-07-09 15:12:41 +02:00
|
|
|
} else if (doc.queryCommandSupported && doc.queryCommandSupported("copy")) {
|
|
|
|
if (this.isChrome() && text === "") {
|
2020-07-09 19:25:46 +02:00
|
|
|
text = "\u0000";
|
2018-01-04 18:32:10 +01:00
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-07-09 15:12:41 +02:00
|
|
|
const textarea = doc.createElement("textarea");
|
|
|
|
textarea.textContent = text == null || text === "" ? " " : text;
|
2018-01-04 18:32:10 +01:00
|
|
|
// Prevent scrolling to bottom of page in MS Edge.
|
2018-07-09 15:12:41 +02:00
|
|
|
textarea.style.position = "fixed";
|
2018-02-27 05:51:17 +01:00
|
|
|
doc.body.appendChild(textarea);
|
2019-08-20 19:54:10 +02:00
|
|
|
textarea.select();
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2019-08-20 19:54:10 +02:00
|
|
|
try {
|
|
|
|
// Security exception may be thrown by some browsers.
|
2018-01-13 03:31:46 +01:00
|
|
|
if (doc.execCommand("copy") && !clearing && this.clipboardWriteCallback != null) {
|
|
|
|
this.clipboardWriteCallback(text, clearMs);
|
2018-01-05 17:13:24 +01:00
|
|
|
}
|
2019-02-27 04:37:21 +01:00
|
|
|
} catch (e) {
|
|
|
|
// tslint:disable-next-line
|
2018-04-23 19:04:11 +02:00
|
|
|
console.warn("Copy to clipboard failed.", e);
|
2018-01-04 22:49:58 +01:00
|
|
|
} finally {
|
2019-02-27 04:37:21 +01:00
|
|
|
doc.body.removeChild(textarea);
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-27 04:37:21 +01:00
|
|
|
async readFromClipboard(options?: any): Promise<string> {
|
|
|
|
let win = window;
|
|
|
|
let doc = window.document;
|
2018-01-04 22:49:58 +01:00
|
|
|
if (options && (options.window || options.win)) {
|
2019-02-27 04:37:21 +01:00
|
|
|
win = options.window || options.win;
|
|
|
|
doc = win.document;
|
|
|
|
} else if (options && options.doc) {
|
|
|
|
doc = options.doc;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2018-01-04 22:49:58 +01:00
|
|
|
if (this.isSafari()) {
|
|
|
|
return await SafariApp.sendMessageToApp("readFromClipboard");
|
2019-02-27 04:37:21 +01:00
|
|
|
} else if (
|
2018-01-04 22:49:58 +01:00
|
|
|
this.isFirefox() &&
|
|
|
|
(win as any).navigator.clipboard &&
|
|
|
|
(win as any).navigator.clipboard.readText
|
2021-12-21 15:43:35 +01:00
|
|
|
) {
|
2019-02-27 04:37:21 +01:00
|
|
|
return await (win as any).navigator.clipboard.readText();
|
|
|
|
} else if (doc.queryCommandSupported && doc.queryCommandSupported("paste")) {
|
2018-01-04 22:49:58 +01:00
|
|
|
const textarea = doc.createElement("textarea");
|
|
|
|
// Prevent scrolling to bottom of page in MS Edge.
|
|
|
|
textarea.style.position = "fixed";
|
|
|
|
doc.body.appendChild(textarea);
|
|
|
|
textarea.focus();
|
|
|
|
try {
|
|
|
|
// Security exception may be thrown by some browsers.
|
2018-01-05 17:13:24 +01:00
|
|
|
if (doc.execCommand("paste")) {
|
2019-08-20 19:54:10 +02:00
|
|
|
return textarea.value;
|
2018-01-05 17:13:24 +01:00
|
|
|
}
|
|
|
|
} catch (e) {
|
2018-01-25 20:31:54 +01:00
|
|
|
// tslint:disable-next-line
|
|
|
|
console.warn("Read from clipboard failed.", e);
|
|
|
|
} finally {
|
2021-04-07 20:43:07 +02:00
|
|
|
doc.body.removeChild(textarea);
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
}
|
2021-04-07 20:43:07 +02:00
|
|
|
return null;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2021-04-07 20:43:07 +02:00
|
|
|
resolveDialogPromise(dialogId: number, confirmed: boolean) {
|
2018-10-03 05:33:56 +02:00
|
|
|
if (this.showDialogResolves.has(dialogId)) {
|
2021-03-04 22:06:02 +01:00
|
|
|
const resolveObj = this.showDialogResolves.get(dialogId);
|
|
|
|
resolveObj.resolve(confirmed);
|
|
|
|
this.showDialogResolves.delete(dialogId);
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
|
|
|
|
2021-03-04 22:06:02 +01:00
|
|
|
// Clean up old promises
|
2018-04-10 20:20:03 +02:00
|
|
|
this.showDialogResolves.forEach((val, key) => {
|
|
|
|
const age = new Date().getTime() - val.date.getTime();
|
|
|
|
if (age > DialogPromiseExpiration) {
|
|
|
|
this.showDialogResolves.delete(key);
|
|
|
|
}
|
|
|
|
});
|
2018-02-03 05:46:30 +01:00
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-04-14 19:32:56 +02:00
|
|
|
async resolvePasswordDialogPromise(
|
2018-04-10 20:20:03 +02:00
|
|
|
dialogId: number,
|
2018-06-30 19:51:20 +02:00
|
|
|
canceled: boolean,
|
2018-02-16 20:00:41 +01:00
|
|
|
password: string
|
|
|
|
): Promise<boolean> {
|
2019-05-30 15:37:09 +02:00
|
|
|
let result = false;
|
2021-02-04 23:34:10 +01:00
|
|
|
if (this.passwordDialogResolves.has(dialogId)) {
|
2018-04-23 19:04:11 +02:00
|
|
|
const resolveObj = this.passwordDialogResolves.get(dialogId);
|
2019-05-30 15:37:09 +02:00
|
|
|
if (await resolveObj.tryResolve(canceled, password)) {
|
2019-02-27 15:28:16 +01:00
|
|
|
this.passwordDialogResolves.delete(dialogId);
|
2018-04-23 19:04:11 +02:00
|
|
|
result = true;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
2018-04-23 19:04:11 +02:00
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-04-23 19:04:11 +02:00
|
|
|
// Clean up old promises
|
2020-07-09 21:59:47 +02:00
|
|
|
this.passwordDialogResolves.forEach((val, key) => {
|
2018-04-23 19:04:11 +02:00
|
|
|
const age = new Date().getTime() - val.date.getTime();
|
2019-08-13 17:57:20 +02:00
|
|
|
if (age > DialogPromiseExpiration) {
|
2019-02-27 15:28:16 +01:00
|
|
|
this.passwordDialogResolves.delete(key);
|
|
|
|
}
|
2018-04-23 19:04:11 +02:00
|
|
|
});
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-04-23 19:04:11 +02:00
|
|
|
return result;
|
2018-02-16 20:00:41 +01:00
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2019-02-27 04:37:21 +01:00
|
|
|
async supportsBiometric() {
|
2019-08-19 17:58:43 +02:00
|
|
|
const platformInfo = await BrowserApi.getPlatformInfo();
|
|
|
|
if (platformInfo.os === "android") {
|
|
|
|
return false;
|
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2019-08-19 17:58:43 +02:00
|
|
|
if (this.isFirefox()) {
|
2019-02-27 04:37:21 +01:00
|
|
|
return parseInt((await browser.runtime.getBrowserInfo()).version.split(".")[0], 10) >= 87;
|
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2019-02-27 04:37:21 +01:00
|
|
|
return true;
|
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-04-10 20:20:03 +02:00
|
|
|
authenticateBiometric() {
|
|
|
|
return this.biometricCallback();
|
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2021-05-03 20:56:38 +02:00
|
|
|
sidebarViewName(): string {
|
|
|
|
if ((window as any).chrome.sidebarAction && this.isFirefox()) {
|
|
|
|
return "sidebar";
|
|
|
|
} else if (this.isOpera() && typeof opr !== "undefined" && opr.sidebarAction) {
|
|
|
|
return "sidebar_panel";
|
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2021-05-03 20:56:38 +02:00
|
|
|
return null;
|
2018-04-10 20:20:03 +02:00
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2021-03-10 21:27:05 +01:00
|
|
|
supportsSecureStorage(): boolean {
|
2021-07-16 10:11:21 +02:00
|
|
|
return false;
|
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2021-04-07 17:39:59 +02:00
|
|
|
getDefaultSystemTheme(): Promise<ThemeType.Light | ThemeType.Dark> {
|
|
|
|
return Promise.resolve(this.prefersColorSchemeDark.matches ? ThemeType.Dark : ThemeType.Light);
|
2020-07-09 20:14:51 +02:00
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2018-01-04 22:49:58 +01:00
|
|
|
onDefaultSystemThemeChange(callback: (theme: ThemeType.Light | ThemeType.Dark) => unknown) {
|
|
|
|
this.prefersColorSchemeDark.addEventListener("change", ({ matches }) => {
|
2021-10-04 22:30:31 +02:00
|
|
|
callback(matches ? ThemeType.Dark : ThemeType.Light);
|
|
|
|
});
|
2020-08-31 22:33:01 +02:00
|
|
|
}
|
2021-12-21 15:43:35 +01:00
|
|
|
|
2021-10-04 22:30:31 +02:00
|
|
|
async getEffectiveTheme() {
|
|
|
|
const theme = await this.storageService.get<ThemeType>(ConstantsService.themeKey);
|
|
|
|
if (theme == null || theme === ThemeType.System) {
|
|
|
|
return this.getDefaultSystemTheme();
|
|
|
|
} else {
|
|
|
|
return theme;
|
2021-12-21 15:43:35 +01:00
|
|
|
}
|
2021-10-04 22:30:31 +02:00
|
|
|
}
|
2018-01-04 18:32:10 +01:00
|
|
|
}
|