bitwarden-estensione-browser/src/background/main.background.ts

637 lines
23 KiB
TypeScript
Raw Normal View History

2018-01-09 20:26:20 +01:00
import { CipherType } from 'jslib/enums';
import {
ApiService,
AppIdService,
AuditService,
2018-01-10 05:05:46 +01:00
CipherService,
2018-01-10 05:12:14 +01:00
CollectionService,
2018-01-10 04:22:49 +01:00
ConstantsService,
2018-01-10 04:28:21 +01:00
ContainerService,
2018-01-09 20:26:20 +01:00
CryptoService,
2018-01-10 02:20:45 +01:00
EnvironmentService,
2018-01-10 04:39:38 +01:00
FolderService,
2018-01-10 05:18:55 +01:00
LockService,
2018-01-09 23:55:28 +01:00
PasswordGenerationService,
2018-01-10 04:47:53 +01:00
SettingsService,
2018-01-10 05:27:15 +01:00
SyncService,
TokenService,
2018-01-09 23:45:17 +01:00
TotpService,
2018-01-09 23:37:40 +01:00
UserService,
2018-01-09 20:26:20 +01:00
} from 'jslib/services';
2018-04-22 05:23:34 +02:00
import { WebCryptoFunctionService } from 'jslib/services/webCryptoFunction.service';
2018-01-09 20:26:20 +01:00
import {
ApiService as ApiServiceAbstraction,
AppIdService as AppIdServiceAbstraction,
AuditService as AuditServiceAbstraction,
2018-01-10 05:05:46 +01:00
CipherService as CipherServiceAbstraction,
2018-01-10 05:12:14 +01:00
CollectionService as CollectionServiceAbstraction,
2018-01-09 20:26:20 +01:00
CryptoService as CryptoServiceAbstraction,
2018-01-10 02:20:45 +01:00
EnvironmentService as EnvironmentServiceAbstraction,
2018-01-10 04:39:38 +01:00
FolderService as FolderServiceAbstraction,
2018-01-27 04:38:54 +01:00
I18nService as I18nServiceAbstraction,
2018-01-10 05:18:55 +01:00
LockService as LockServiceAbstraction,
2018-01-09 20:26:20 +01:00
MessagingService as MessagingServiceAbstraction,
2018-01-09 23:55:28 +01:00
PasswordGenerationService as PasswordGenerationServiceAbstraction,
2018-01-09 20:26:20 +01:00
PlatformUtilsService as PlatformUtilsServiceAbstraction,
2018-01-10 04:47:53 +01:00
SettingsService as SettingsServiceAbstraction,
2018-01-09 20:26:20 +01:00
StorageService as StorageServiceAbstraction,
2018-01-10 05:27:15 +01:00
SyncService as SyncServiceAbstraction,
TokenService as TokenServiceAbstraction,
2018-01-09 23:45:17 +01:00
TotpService as TotpServiceAbstraction,
2018-01-09 23:37:40 +01:00
UserService as UserServiceAbstraction,
2018-01-09 20:26:20 +01:00
} from 'jslib/abstractions';
2018-04-22 05:23:34 +02:00
import { CryptoFunctionService as CryptoFunctionServiceAbstraction } from 'jslib/abstractions/cryptoFunction.service';
2017-12-05 23:04:30 +01:00
2018-01-26 16:48:32 +01:00
import { Analytics } from 'jslib/misc';
2018-01-12 16:05:30 +01:00
2018-01-12 17:09:30 +01:00
import { BrowserApi } from '../browser/browserApi';
2017-12-07 21:36:24 +01:00
import CommandsBackground from './commands.background';
2017-12-07 22:02:15 +01:00
import ContextMenusBackground from './contextMenus.background';
import IdleBackground from './idle.background';
import RuntimeBackground from './runtime.background';
import TabsBackground from './tabs.background';
import WebRequestBackground from './webRequest.background';
import WindowsBackground from './windows.background';
2017-12-05 23:04:30 +01:00
import AutofillService from '../services/autofill.service';
2018-01-04 22:06:00 +01:00
import BrowserMessagingService from '../services/browserMessaging.service';
2018-01-05 22:38:50 +01:00
import BrowserPlatformUtilsService from '../services/browserPlatformUtils.service';
2018-01-04 22:15:06 +01:00
import BrowserStorageService from '../services/browserStorage.service';
2018-04-11 20:52:49 +02:00
import I18nService from '../services/i18n.service';
2018-01-04 16:51:08 +01:00
2018-01-11 20:45:27 +01:00
import { AutofillService as AutofillServiceAbstraction } from '../services/abstractions/autofill.service';
2017-12-05 23:04:30 +01:00
export default class MainBackground {
2018-01-09 20:26:20 +01:00
messagingService: MessagingServiceAbstraction;
storageService: StorageServiceAbstraction;
secureStorageService: StorageServiceAbstraction;
2018-04-11 20:52:49 +02:00
i18nService: I18nServiceAbstraction;
2018-01-09 20:26:20 +01:00
platformUtilsService: PlatformUtilsServiceAbstraction;
2017-12-05 23:04:30 +01:00
constantsService: ConstantsService;
2018-01-09 20:26:20 +01:00
cryptoService: CryptoServiceAbstraction;
tokenService: TokenServiceAbstraction;
appIdService: AppIdServiceAbstraction;
apiService: ApiServiceAbstraction;
2018-01-10 02:20:45 +01:00
environmentService: EnvironmentServiceAbstraction;
2018-01-09 23:37:40 +01:00
userService: UserServiceAbstraction;
2018-01-10 04:47:53 +01:00
settingsService: SettingsServiceAbstraction;
2018-01-10 05:05:46 +01:00
cipherService: CipherServiceAbstraction;
2018-01-10 04:39:38 +01:00
folderService: FolderServiceAbstraction;
2018-01-10 05:12:14 +01:00
collectionService: CollectionServiceAbstraction;
2018-01-10 05:18:55 +01:00
lockService: LockServiceAbstraction;
2018-01-10 05:27:15 +01:00
syncService: SyncServiceAbstraction;
2018-01-09 23:55:28 +01:00
passwordGenerationService: PasswordGenerationServiceAbstraction;
2018-01-09 23:45:17 +01:00
totpService: TotpServiceAbstraction;
2018-01-11 20:45:27 +01:00
autofillService: AutofillServiceAbstraction;
containerService: ContainerService;
auditService: AuditServiceAbstraction;
2018-01-19 22:19:24 +01:00
analytics: Analytics;
2017-12-05 23:04:30 +01:00
onUpdatedRan: boolean;
onReplacedRan: boolean;
2017-12-07 21:36:24 +01:00
loginToAutoFill: any = null;
loginsToAdd: any[] = [];
private commandsBackground: CommandsBackground;
2017-12-07 22:02:15 +01:00
private contextMenusBackground: ContextMenusBackground;
private idleBackground: IdleBackground;
private runtimeBackground: RuntimeBackground;
private tabsBackground: TabsBackground;
private webRequestBackground: WebRequestBackground;
private windowsBackground: WindowsBackground;
2017-12-05 23:04:30 +01:00
private sidebarAction: any;
private buildingContextMenu: boolean;
private menuOptionsLoaded: any[] = [];
2018-04-06 17:48:45 +02:00
private syncTimeout: any;
2018-01-12 04:36:22 +01:00
private isSafari: boolean;
2017-12-05 23:04:30 +01:00
constructor() {
2017-12-05 23:04:30 +01:00
// Services
2018-04-10 20:20:03 +02:00
this.messagingService = new BrowserMessagingService();
this.platformUtilsService = new BrowserPlatformUtilsService(this.messagingService);
this.storageService = new BrowserStorageService(this.platformUtilsService, false);
this.secureStorageService = new BrowserStorageService(this.platformUtilsService, true);
2018-04-11 20:52:49 +02:00
this.i18nService = new I18nService(BrowserApi.getUILanguage(window),
BrowserApi.isSafariApi ? './_locales/' : null);
2018-04-22 05:23:34 +02:00
const cryptoFunctionService = new WebCryptoFunctionService(window, this.platformUtilsService);
this.cryptoService = new CryptoService(this.storageService, this.secureStorageService, cryptoFunctionService);
2018-01-04 16:51:08 +01:00
this.tokenService = new TokenService(this.storageService);
this.appIdService = new AppIdService(this.storageService);
2018-01-05 22:30:15 +01:00
this.apiService = new ApiService(this.tokenService, this.platformUtilsService,
2017-12-15 15:58:26 +01:00
(expired: boolean) => this.logout(expired));
2018-01-04 16:51:08 +01:00
this.environmentService = new EnvironmentService(this.apiService, this.storageService);
this.userService = new UserService(this.tokenService, this.storageService);
this.settingsService = new SettingsService(this.userService, this.storageService);
this.cipherService = new CipherService(this.cryptoService, this.userService, this.settingsService,
2018-04-23 19:04:11 +02:00
this.apiService, this.storageService, this.i18nService, this.platformUtilsService);
2018-01-13 03:51:07 +01:00
this.folderService = new FolderService(this.cryptoService, this.userService,
2018-04-11 20:52:49 +02:00
() => this.i18nService.t('noneFolder'), this.apiService, this.storageService, this.i18nService);
2018-01-27 04:38:54 +01:00
this.collectionService = new CollectionService(this.cryptoService, this.userService, this.storageService,
2018-04-11 20:52:49 +02:00
this.i18nService);
2017-12-05 23:04:30 +01:00
this.lockService = new LockService(this.cipherService, this.folderService, this.collectionService,
2018-03-04 04:48:38 +01:00
this.cryptoService, this.platformUtilsService, this.storageService, this.messagingService, async () => {
await this.setIcon();
await this.refreshBadgeAndMenu(true);
});
2017-12-05 23:04:30 +01:00
this.syncService = new SyncService(this.userService, this.apiService, this.settingsService,
2017-12-07 04:28:33 +01:00
this.folderService, this.cipherService, this.cryptoService, this.collectionService,
2018-01-04 22:06:00 +01:00
this.storageService, this.messagingService, (expired: boolean) => this.logout(expired));
2018-01-04 16:51:08 +01:00
this.passwordGenerationService = new PasswordGenerationService(this.cryptoService, this.storageService);
2018-04-22 05:55:34 +02:00
this.totpService = new TotpService(this.storageService, cryptoFunctionService);
2017-12-05 23:04:30 +01:00
this.autofillService = new AutofillService(this.cipherService, this.tokenService,
this.totpService);
this.containerService = new ContainerService(this.cryptoService, this.platformUtilsService);
2018-04-22 05:55:34 +02:00
this.auditService = new AuditService(cryptoFunctionService);
2018-01-26 16:48:32 +01:00
this.analytics = new Analytics(window, () => BrowserApi.gaFilter(), this.platformUtilsService,
this.storageService, this.appIdService);
2017-12-05 23:04:30 +01:00
// Other fields
2018-01-12 04:36:22 +01:00
this.isSafari = this.platformUtilsService.isSafari();
this.sidebarAction = this.isSafari ? null : (typeof opr !== 'undefined') && opr.sidebarAction ?
2017-12-05 23:18:20 +01:00
opr.sidebarAction : (window as any).chrome.sidebarAction;
// Background
2018-01-12 18:24:34 +01:00
this.runtimeBackground = new RuntimeBackground(this, this.autofillService, this.cipherService,
2018-04-10 20:20:03 +02:00
this.platformUtilsService as BrowserPlatformUtilsService, this.storageService, this.i18nService,
this.analytics);
2018-01-13 20:56:38 +01:00
this.tabsBackground = new TabsBackground(this, this.platformUtilsService);
this.commandsBackground = new CommandsBackground(this, this.passwordGenerationService,
2018-01-19 22:19:24 +01:00
this.platformUtilsService, this.analytics);
2018-01-12 18:24:34 +01:00
2018-01-12 05:39:16 +01:00
if (!this.isSafari) {
this.contextMenusBackground = new ContextMenusBackground(this, this.cipherService,
2018-04-23 19:04:11 +02:00
this.passwordGenerationService, this.analytics, this.platformUtilsService);
2018-01-12 05:39:16 +01:00
this.idleBackground = new IdleBackground(this, this.lockService, this.storageService);
this.webRequestBackground = new WebRequestBackground(this.platformUtilsService, this.cipherService);
this.windowsBackground = new WindowsBackground(this);
}
2017-12-05 23:04:30 +01:00
}
2017-12-06 05:37:32 +01:00
async bootstrap() {
2018-01-19 22:19:24 +01:00
this.analytics.ga('send', 'pageview', '/background.html');
this.containerService.attachToWindow(window);
2018-04-11 21:59:39 +02:00
await (this.i18nService as I18nService).init();
await this.runtimeBackground.init();
2018-01-13 20:56:38 +01:00
await this.tabsBackground.init();
await this.commandsBackground.init();
2018-01-13 21:09:05 +01:00
2018-01-12 04:36:22 +01:00
if (!this.isSafari) {
await this.contextMenusBackground.init();
await this.idleBackground.init();
await this.webRequestBackground.init();
await this.windowsBackground.init();
}
return new Promise((resolve) => {
setTimeout(async () => {
await this.environmentService.setUrlsFromStorage();
await this.setIcon();
this.cleanupLoginsToAdd();
await this.fullSync(true);
resolve();
}, 500);
});
}
2017-12-07 22:02:15 +01:00
async setIcon() {
2018-01-12 04:36:22 +01:00
if (this.isSafari || (!chrome.browserAction && !this.sidebarAction)) {
2017-12-07 22:02:15 +01:00
return;
}
const isAuthenticated = await this.userService.isAuthenticated();
const key = await this.cryptoService.getKey();
let suffix = '';
if (!isAuthenticated) {
suffix = '_gray';
} else if (!key) {
suffix = '_locked';
}
await this.actionSetIcon(chrome.browserAction, suffix);
await this.actionSetIcon(this.sidebarAction, suffix);
}
2018-03-04 04:48:38 +01:00
async refreshBadgeAndMenu(forLocked: boolean = false) {
2018-01-12 04:36:22 +01:00
if (this.isSafari || !chrome.windows || !chrome.contextMenus) {
2017-12-07 22:02:15 +01:00
return;
}
2018-03-04 04:48:38 +01:00
const menuDisabled = await this.storageService.get<boolean>(ConstantsService.disableContextMenuItemKey);
if (!menuDisabled) {
2017-12-07 22:02:15 +01:00
await this.buildContextMenu();
} else {
await this.contextMenusRemoveAll();
2018-03-04 04:48:38 +01:00
}
if (forLocked) {
await this.loadMenuAndUpdateBadgeForLockedState(!menuDisabled);
this.onUpdatedRan = this.onReplacedRan = false;
return;
}
const tab = await BrowserApi.getTabFromCurrentWindow();
if (tab) {
await this.contextMenuReady(tab, !menuDisabled);
2017-12-07 22:02:15 +01:00
}
}
async logout(expired: boolean) {
const userId = await this.userService.getUserId();
await Promise.all([
this.syncService.setLastSync(new Date(0)),
this.tokenService.clearToken(),
this.cryptoService.clearKeys(),
this.userService.clear(),
this.settingsService.clear(userId),
this.cipherService.clear(userId),
this.folderService.clear(userId),
this.passwordGenerationService.clear(),
]);
2018-01-04 22:06:00 +01:00
this.messagingService.send('doneLoggingOut', { expired: expired });
2017-12-07 22:02:15 +01:00
await this.setIcon();
await this.refreshBadgeAndMenu();
}
collectPageDetailsForContentScript(tab: any, sender: string, frameId: number = null) {
if (tab == null || !tab.id) {
return;
}
const options: any = {};
if (frameId != null) {
options.frameId = frameId;
}
BrowserApi.tabSendMessage(tab, {
2017-12-07 22:02:15 +01:00
command: 'collectPageDetails',
tab: tab,
sender: sender,
}, options);
2017-12-07 22:02:15 +01:00
}
async checkLoginsToAdd(tab: any = null): Promise<any> {
if (!this.loginsToAdd.length) {
return;
}
if (tab != null) {
this.doCheck(tab);
return;
}
const currentTab = await BrowserApi.getTabFromCurrentWindow();
if (currentTab != null) {
this.doCheck(currentTab);
}
}
2018-01-14 00:16:19 +01:00
sendInternalRuntimeMessage(message: any) {
if (!this.isSafari) {
throw new Error('Only safari can send internal runtime messages.');
}
this.runtimeBackground.processMessage(message, { tab: null }, () => { /* No response needed. */ });
}
2018-01-18 22:17:58 +01:00
async openPopup() {
// Chrome APIs cannot open popup
if (!this.isSafari || !safari.extension.toolbarItems || !safari.extension.toolbarItems.length) {
return;
}
const activeToolBars = safari.extension.toolbarItems.filter((tb: any) => {
return tb.browserWindow === safari.application.activeBrowserWindow;
});
if (activeToolBars && activeToolBars.length) {
activeToolBars[0].showPopover();
}
}
2017-12-05 23:04:30 +01:00
private async buildContextMenu() {
2018-01-12 04:36:22 +01:00
if (this.isSafari || !chrome.contextMenus || this.buildingContextMenu) {
2017-12-05 23:04:30 +01:00
return;
}
this.buildingContextMenu = true;
await this.contextMenusRemoveAll();
await this.contextMenusCreate({
type: 'normal',
id: 'root',
contexts: ['all'],
2018-02-27 19:37:57 +01:00
title: 'Bitwarden',
2017-12-05 23:04:30 +01:00
});
await this.contextMenusCreate({
type: 'normal',
id: 'autofill',
parentId: 'root',
contexts: ['all'],
2018-04-11 20:52:49 +02:00
title: this.i18nService.t('autoFill'),
2017-12-05 23:04:30 +01:00
});
// Firefox & Edge do not support writing to the clipboard from background
2018-01-05 22:30:15 +01:00
if (!this.platformUtilsService.isFirefox() && !this.platformUtilsService.isEdge()) {
2017-12-05 23:04:30 +01:00
await this.contextMenusCreate({
type: 'normal',
id: 'copy-username',
parentId: 'root',
contexts: ['all'],
2018-04-11 20:52:49 +02:00
title: this.i18nService.t('copyUsername'),
2017-12-05 23:04:30 +01:00
});
await this.contextMenusCreate({
type: 'normal',
id: 'copy-password',
parentId: 'root',
contexts: ['all'],
2018-04-11 20:52:49 +02:00
title: this.i18nService.t('copyPassword'),
2017-12-05 23:04:30 +01:00
});
await this.contextMenusCreate({
type: 'separator',
parentId: 'root',
2017-12-05 23:04:30 +01:00
});
await this.contextMenusCreate({
type: 'normal',
id: 'generate-password',
parentId: 'root',
contexts: ['all'],
2018-04-11 20:52:49 +02:00
title: this.i18nService.t('generatePasswordCopied'),
2017-12-05 23:04:30 +01:00
});
}
this.buildingContextMenu = false;
}
private async contextMenuReady(tab: any, contextMenuEnabled: boolean) {
await this.loadMenuAndUpdateBadge(tab.url, tab.id, contextMenuEnabled);
this.onUpdatedRan = this.onReplacedRan = false;
}
private async loadMenuAndUpdateBadge(url: string, tabId: number, contextMenuEnabled: boolean) {
if (!url || (!chrome.browserAction && !this.sidebarAction)) {
return;
}
this.actionSetBadgeBackgroundColor(chrome.browserAction);
this.actionSetBadgeBackgroundColor(this.sidebarAction);
this.menuOptionsLoaded = [];
try {
const ciphers = await this.cipherService.getAllDecryptedForUrl(url);
2018-03-02 18:43:12 +01:00
ciphers.sort((a, b) => this.cipherService.sortCiphersByLastUsedThenName(a, b));
2017-12-05 23:04:30 +01:00
if (contextMenuEnabled) {
ciphers.forEach((cipher) => {
this.loadLoginContextMenuOptions(cipher);
2017-12-05 23:04:30 +01:00
});
}
let theText = '';
2017-12-05 23:04:30 +01:00
if (ciphers.length > 0 && ciphers.length < 9) {
theText = ciphers.length.toString();
} else if (ciphers.length > 0) {
theText = '9+';
} else {
if (contextMenuEnabled) {
2018-04-11 20:52:49 +02:00
await this.loadNoLoginsContextMenuOptions(this.i18nService.t('noMatchingLogins'));
2017-12-05 23:04:30 +01:00
}
}
this.browserActionSetBadgeText(theText, tabId);
this.sidebarActionSetBadgeText(theText, tabId);
} catch (e) {
2018-03-04 04:48:38 +01:00
await this.loadMenuAndUpdateBadgeForLockedState(contextMenuEnabled);
}
}
private async loadMenuAndUpdateBadgeForLockedState(contextMenuEnabled: boolean) {
if (contextMenuEnabled) {
2018-04-11 20:52:49 +02:00
await this.loadNoLoginsContextMenuOptions(this.i18nService.t('vaultLocked'));
2018-03-04 04:48:38 +01:00
}
const tabs = await BrowserApi.getActiveTabs();
if (tabs != null) {
tabs.forEach((tab) => {
if (tab.id != null) {
this.browserActionSetBadgeText('', tab.id);
this.sidebarActionSetBadgeText('', tab.id);
}
});
2017-12-05 23:04:30 +01:00
}
}
private async loadLoginContextMenuOptions(cipher: any) {
2018-01-09 20:26:20 +01:00
if (cipher == null || cipher.type !== CipherType.Login) {
2017-12-05 23:04:30 +01:00
return;
}
let title = cipher.name;
if (cipher.login.username && cipher.login.username !== '') {
title += (' (' + cipher.login.username + ')');
}
2017-12-05 23:04:30 +01:00
await this.loadContextMenuOptions(title, cipher.id, cipher);
}
private async loadNoLoginsContextMenuOptions(noLoginsMessage: string) {
await this.loadContextMenuOptions(noLoginsMessage, 'noop', null);
}
private async loadContextMenuOptions(title: string, idSuffix: string, cipher: any) {
if (!chrome.contextMenus || this.menuOptionsLoaded.indexOf(idSuffix) > -1 ||
2018-01-09 20:26:20 +01:00
(cipher != null && cipher.type !== CipherType.Login)) {
2017-12-05 23:04:30 +01:00
return;
}
this.menuOptionsLoaded.push(idSuffix);
2017-12-07 04:10:02 +01:00
if (cipher == null || (cipher.login.password && cipher.login.password !== '')) {
2017-12-05 23:04:30 +01:00
await this.contextMenusCreate({
type: 'normal',
id: 'autofill_' + idSuffix,
parentId: 'autofill',
contexts: ['all'],
title: title,
2017-12-05 23:04:30 +01:00
});
}
2018-01-05 22:30:15 +01:00
if (this.platformUtilsService.isFirefox()) {
2017-12-05 23:04:30 +01:00
// Firefox does not support writing to the clipboard from background
return;
}
2017-12-07 04:10:02 +01:00
if (cipher == null || (cipher.login.username && cipher.login.username !== '')) {
2017-12-05 23:04:30 +01:00
await this.contextMenusCreate({
type: 'normal',
id: 'copy-username_' + idSuffix,
parentId: 'copy-username',
contexts: ['all'],
title: title,
2017-12-05 23:04:30 +01:00
});
}
2017-12-07 04:10:02 +01:00
if (cipher == null || (cipher.login.password && cipher.login.password !== '')) {
2017-12-05 23:04:30 +01:00
await this.contextMenusCreate({
type: 'normal',
id: 'copy-password_' + idSuffix,
parentId: 'copy-password',
contexts: ['all'],
title: title,
2017-12-05 23:04:30 +01:00
});
}
}
private cleanupLoginsToAdd() {
for (let i = this.loginsToAdd.length - 1; i >= 0; i--) {
if (this.loginsToAdd[i].expires < new Date()) {
this.loginsToAdd.splice(i, 1);
}
}
setTimeout(() => this.cleanupLoginsToAdd(), 2 * 60 * 1000); // check every 2 minutes
}
private doCheck(tab: any) {
if (tab == null) {
return;
}
2018-01-05 22:30:15 +01:00
const tabDomain = this.platformUtilsService.getDomain(tab.url);
if (tabDomain == null) {
return;
}
for (let i = 0; i < this.loginsToAdd.length; i++) {
if (this.loginsToAdd[i].tabId !== tab.id || this.loginsToAdd[i].domain !== tabDomain) {
continue;
}
BrowserApi.tabSendMessageData(tab, 'openNotificationBar', {
type: 'add',
});
break;
}
}
private async fullSync(override: boolean = false) {
const syncInternal = 6 * 60 * 60 * 1000; // 6 hours
const lastSync = await this.syncService.getLastSync();
let lastSyncAgo = syncInternal + 1;
if (lastSync != null) {
lastSyncAgo = new Date().getTime() - lastSync.getTime();
}
if (override || lastSyncAgo >= syncInternal) {
await this.syncService.fullSync(override);
this.scheduleNextSync();
} else {
this.scheduleNextSync();
}
}
private scheduleNextSync() {
if (this.syncTimeout) {
clearTimeout(this.syncTimeout);
}
this.syncTimeout = setTimeout(async () => await this.fullSync(), 5 * 60 * 1000); // check every 5 minutes
}
// Browser API Helpers
2017-12-05 23:04:30 +01:00
private contextMenusRemoveAll() {
return new Promise((resolve) => {
chrome.contextMenus.removeAll(() => {
2017-12-05 23:04:30 +01:00
resolve();
if (chrome.runtime.lastError) {
return;
}
});
});
}
private contextMenusCreate(options: any) {
return new Promise((resolve) => {
chrome.contextMenus.create(options, () => {
2017-12-05 23:04:30 +01:00
resolve();
if (chrome.runtime.lastError) {
return;
}
});
});
}
private async actionSetIcon(theAction: any, suffix: string): Promise<any> {
2017-12-05 23:04:30 +01:00
if (!theAction || !theAction.setIcon) {
return;
2017-12-05 23:04:30 +01:00
}
const options = {
path: {
19: 'images/icon19' + suffix + '.png',
38: 'images/icon38' + suffix + '.png',
},
};
2018-01-05 22:30:15 +01:00
if (this.platformUtilsService.isFirefox()) {
await theAction.setIcon(options);
} else {
return new Promise((resolve) => {
theAction.setIcon(options, () => resolve());
2017-12-05 23:04:30 +01:00
});
}
2017-12-05 23:04:30 +01:00
}
private actionSetBadgeBackgroundColor(action: any) {
if (action && action.setBadgeBackgroundColor) {
action.setBadgeBackgroundColor({ color: '#294e5f' });
}
}
private browserActionSetBadgeText(text: string, tabId: number) {
if (chrome.browserAction && chrome.browserAction.setBadgeText) {
chrome.browserAction.setBadgeText({
text: text,
tabId: tabId,
2017-12-05 23:04:30 +01:00
});
}
}
private sidebarActionSetBadgeText(text: string, tabId: number) {
if (!this.sidebarAction) {
return;
}
if (this.sidebarAction.setBadgeText) {
this.sidebarAction.setBadgeText({
text: text,
tabId: tabId,
2017-12-05 23:04:30 +01:00
});
} else if (this.sidebarAction.setTitle) {
2018-02-27 19:37:57 +01:00
let title = 'Bitwarden';
2017-12-05 23:04:30 +01:00
if (text && text !== '') {
title += (' [' + text + ']');
}
this.sidebarAction.setTitle({
title: title,
tabId: tabId,
2017-12-05 23:04:30 +01:00
});
}
}
}