mirror of
https://github.com/bitwarden/browser
synced 2025-12-11 13:53:34 +00:00
context menus background
This commit is contained in:
97
src/background/contextMenus.background.ts
Normal file
97
src/background/contextMenus.background.ts
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
import BrowserApi from '../browser/browserApi';
|
||||||
|
|
||||||
|
import MainBackground from './main.background';
|
||||||
|
|
||||||
|
import CipherService from '../services/cipher.service';
|
||||||
|
import PasswordGenerationService from '../services/passwordGeneration.service';
|
||||||
|
import UtilsService from '../services/utils.service';
|
||||||
|
|
||||||
|
export default class ContextMenusBackground {
|
||||||
|
private contextMenus: any;
|
||||||
|
|
||||||
|
constructor(private main: MainBackground, private cipherService: CipherService,
|
||||||
|
private passwordGenerationService: PasswordGenerationService) {
|
||||||
|
this.contextMenus = chrome.contextMenus;
|
||||||
|
}
|
||||||
|
|
||||||
|
async init() {
|
||||||
|
if (!this.contextMenus) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.contextMenus.onClicked.addListener(async (info: any, tab: any) => {
|
||||||
|
if (info.menuItemId === 'generate-password') {
|
||||||
|
await this.generatePasswordToClipboard();
|
||||||
|
} else if (info.parentMenuItemId === 'autofill' || info.parentMenuItemId === 'copy-username' ||
|
||||||
|
info.parentMenuItemId === 'copy-password') {
|
||||||
|
await this.cipherAction(info);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async generatePasswordToClipboard() {
|
||||||
|
const options = await this.passwordGenerationService.getOptions();
|
||||||
|
const password = PasswordGenerationService.generatePassword(options);
|
||||||
|
UtilsService.copyToClipboard(password);
|
||||||
|
this.passwordGenerationService.addHistory(password);
|
||||||
|
|
||||||
|
(window as any).ga('send', {
|
||||||
|
hitType: 'event',
|
||||||
|
eventAction: 'Generated Password From Context Menu',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async cipherAction(info: any) {
|
||||||
|
const id = info.menuItemId.split('_')[1];
|
||||||
|
if (id === 'noop') {
|
||||||
|
if (chrome.browserAction && chrome.browserAction.openPopup) {
|
||||||
|
chrome.browserAction.openPopup();
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ciphers = await this.cipherService.getAllDecrypted();
|
||||||
|
for (let i = 0; i < ciphers.length; i++) {
|
||||||
|
const cipher = ciphers[i];
|
||||||
|
if (cipher.id !== id) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (info.parentMenuItemId === 'autofill') {
|
||||||
|
(window as any).ga('send', {
|
||||||
|
hitType: 'event',
|
||||||
|
eventAction: 'Autofilled From Context Menu',
|
||||||
|
});
|
||||||
|
await this.startAutofillPage(cipher);
|
||||||
|
} else if (info.parentMenuItemId === 'copy-username') {
|
||||||
|
(window as any).ga('send', {
|
||||||
|
hitType: 'event',
|
||||||
|
eventAction: 'Copied Username From Context Menu',
|
||||||
|
});
|
||||||
|
UtilsService.copyToClipboard(cipher.login.username);
|
||||||
|
} else if (info.parentMenuItemId === 'copy-password') {
|
||||||
|
(window as any).ga('send', {
|
||||||
|
hitType: 'event',
|
||||||
|
eventAction: 'Copied Password From Context Menu',
|
||||||
|
});
|
||||||
|
UtilsService.copyToClipboard(cipher.login.password);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async startAutofillPage(cipher: any) {
|
||||||
|
this.main.loginToAutoFill = cipher;
|
||||||
|
const tab = await BrowserApi.getTabFromCurrentWindow();
|
||||||
|
if (tab == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
chrome.tabs.sendMessage(tab.id, {
|
||||||
|
command: 'collectPageDetails',
|
||||||
|
tab: tab,
|
||||||
|
sender: 'contextMenu',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,6 +5,7 @@ import { Cipher } from '../models/domain/cipher';
|
|||||||
import BrowserApi from '../browser/browserApi';
|
import BrowserApi from '../browser/browserApi';
|
||||||
|
|
||||||
import CommandsBackground from './commands.background';
|
import CommandsBackground from './commands.background';
|
||||||
|
import ContextMenusBackground from './contextMenus.background';
|
||||||
import RuntimeBackground from './runtime.background';
|
import RuntimeBackground from './runtime.background';
|
||||||
import TabsBackground from './tabs.background';
|
import TabsBackground from './tabs.background';
|
||||||
import WebRequestBackground from './webRequest.background';
|
import WebRequestBackground from './webRequest.background';
|
||||||
@@ -55,6 +56,7 @@ export default class MainBackground {
|
|||||||
loginsToAdd: any[] = [];
|
loginsToAdd: any[] = [];
|
||||||
|
|
||||||
private commandsBackground: CommandsBackground;
|
private commandsBackground: CommandsBackground;
|
||||||
|
private contextMenusBackground: ContextMenusBackground;
|
||||||
private runtimeBackground: RuntimeBackground;
|
private runtimeBackground: RuntimeBackground;
|
||||||
private tabsBackground: TabsBackground;
|
private tabsBackground: TabsBackground;
|
||||||
private webRequestBackground: WebRequestBackground;
|
private webRequestBackground: WebRequestBackground;
|
||||||
@@ -98,6 +100,8 @@ export default class MainBackground {
|
|||||||
|
|
||||||
// Background
|
// Background
|
||||||
this.commandsBackground = new CommandsBackground(this, this.passwordGenerationService);
|
this.commandsBackground = new CommandsBackground(this, this.passwordGenerationService);
|
||||||
|
this.contextMenusBackground = new ContextMenusBackground(this, this.cipherService,
|
||||||
|
this.passwordGenerationService);
|
||||||
this.runtimeBackground = new RuntimeBackground(this, this.autofillService, this.cipherService);
|
this.runtimeBackground = new RuntimeBackground(this, this.autofillService, this.cipherService);
|
||||||
this.tabsBackground = new TabsBackground(this);
|
this.tabsBackground = new TabsBackground(this);
|
||||||
this.webRequestBackground = new WebRequestBackground(this.utilsService, this.cipherService);
|
this.webRequestBackground = new WebRequestBackground(this.utilsService, this.cipherService);
|
||||||
@@ -105,64 +109,8 @@ export default class MainBackground {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async bootstrap() {
|
async bootstrap() {
|
||||||
// Chrome APIs
|
|
||||||
|
|
||||||
if (chrome.contextMenus) {
|
|
||||||
chrome.contextMenus.onClicked.addListener(async (info: any, tab: any) => {
|
|
||||||
if (info.menuItemId === 'generate-password') {
|
|
||||||
(window as any).ga('send', {
|
|
||||||
hitType: 'event',
|
|
||||||
eventAction: 'Generated Password From Context Menu',
|
|
||||||
});
|
|
||||||
const options = await this.passwordGenerationService.getOptions();
|
|
||||||
const password = PasswordGenerationService.generatePassword(options);
|
|
||||||
UtilsService.copyToClipboard(password);
|
|
||||||
await this.passwordGenerationService.addHistory(password);
|
|
||||||
} else if (info.parentMenuItemId === 'autofill' || info.parentMenuItemId === 'copy-username' ||
|
|
||||||
info.parentMenuItemId === 'copy-password') {
|
|
||||||
const id = info.menuItemId.split('_')[1];
|
|
||||||
if (id === 'noop') {
|
|
||||||
if ((window as any).chrome.browserAction.openPopup) {
|
|
||||||
(window as any).chrome.browserAction.openPopup();
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const ciphers = await this.cipherService.getAllDecrypted();
|
|
||||||
for (let i = 0; i < ciphers.length; i++) {
|
|
||||||
const cipher = ciphers[i];
|
|
||||||
if (cipher.id !== id) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (info.parentMenuItemId === 'autofill') {
|
|
||||||
(window as any).ga('send', {
|
|
||||||
hitType: 'event',
|
|
||||||
eventAction: 'Autofilled From Context Menu',
|
|
||||||
});
|
|
||||||
await this.startAutofillPage(cipher);
|
|
||||||
} else if (info.parentMenuItemId === 'copy-username') {
|
|
||||||
(window as any).ga('send', {
|
|
||||||
hitType: 'event',
|
|
||||||
eventAction: 'Copied Username From Context Menu',
|
|
||||||
});
|
|
||||||
UtilsService.copyToClipboard(cipher.login.username);
|
|
||||||
} else if (info.parentMenuItemId === 'copy-password') {
|
|
||||||
(window as any).ga('send', {
|
|
||||||
hitType: 'event',
|
|
||||||
eventAction: 'Copied Password From Context Menu',
|
|
||||||
});
|
|
||||||
UtilsService.copyToClipboard(cipher.login.password);
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Bootstrap
|
|
||||||
await this.commandsBackground.init();
|
await this.commandsBackground.init();
|
||||||
|
await this.contextMenusBackground.init();
|
||||||
await this.runtimeBackground.init();
|
await this.runtimeBackground.init();
|
||||||
await this.tabsBackground.init();
|
await this.tabsBackground.init();
|
||||||
await this.webRequestBackground.init();
|
await this.webRequestBackground.init();
|
||||||
@@ -174,6 +122,104 @@ export default class MainBackground {
|
|||||||
await this.fullSync(true);
|
await this.fullSync(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async setIcon() {
|
||||||
|
if (!chrome.browserAction && !this.sidebarAction) {
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
async refreshBadgeAndMenu() {
|
||||||
|
if (!chrome.windows || !chrome.contextMenus) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const tab = await BrowserApi.getTabFromCurrentWindowId();
|
||||||
|
if (!tab) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const disabled = await this.utilsService.getObjFromStorage<boolean>(ConstantsService.disableContextMenuItemKey);
|
||||||
|
if (!disabled) {
|
||||||
|
await this.buildContextMenu();
|
||||||
|
await this.contextMenuReady(tab, true);
|
||||||
|
} else {
|
||||||
|
await this.contextMenusRemoveAll();
|
||||||
|
await this.contextMenuReady(tab, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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(),
|
||||||
|
]);
|
||||||
|
|
||||||
|
chrome.runtime.sendMessage({
|
||||||
|
command: 'doneLoggingOut', expired: expired,
|
||||||
|
});
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
chrome.tabs.sendMessage(tab.id, {
|
||||||
|
command: 'collectPageDetails',
|
||||||
|
tab: tab,
|
||||||
|
sender: sender,
|
||||||
|
}, options, () => {
|
||||||
|
if (chrome.runtime.lastError) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private async buildContextMenu() {
|
private async buildContextMenu() {
|
||||||
if (!chrome.contextMenus || this.buildingContextMenu) {
|
if (!chrome.contextMenus || this.buildingContextMenu) {
|
||||||
return;
|
return;
|
||||||
@@ -232,45 +278,6 @@ export default class MainBackground {
|
|||||||
this.buildingContextMenu = false;
|
this.buildingContextMenu = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
async setIcon() {
|
|
||||||
if (!chrome.browserAction && !this.sidebarAction) {
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
|
|
||||||
async refreshBadgeAndMenu() {
|
|
||||||
if (!chrome.windows || !chrome.contextMenus) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const tab = await BrowserApi.getTabFromCurrentWindowId();
|
|
||||||
if (!tab) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const disabled = await this.utilsService.getObjFromStorage<boolean>(ConstantsService.disableContextMenuItemKey);
|
|
||||||
if (!disabled) {
|
|
||||||
await this.buildContextMenu();
|
|
||||||
await this.contextMenuReady(tab, true);
|
|
||||||
} else {
|
|
||||||
await this.contextMenusRemoveAll();
|
|
||||||
await this.contextMenuReady(tab, false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async contextMenuReady(tab: any, contextMenuEnabled: boolean) {
|
private async contextMenuReady(tab: any, contextMenuEnabled: boolean) {
|
||||||
await this.loadMenuAndUpdateBadge(tab.url, tab.id, contextMenuEnabled);
|
await this.loadMenuAndUpdateBadge(tab.url, tab.id, contextMenuEnabled);
|
||||||
this.onUpdatedRan = this.onReplacedRan = false;
|
this.onUpdatedRan = this.onReplacedRan = false;
|
||||||
@@ -382,63 +389,6 @@ export default class MainBackground {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async startAutofillPage(cipher: any) {
|
|
||||||
this.loginToAutoFill = cipher;
|
|
||||||
const tab = await BrowserApi.getTabFromCurrentWindow();
|
|
||||||
if (tab == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
chrome.tabs.sendMessage(tab.id, {
|
|
||||||
command: 'collectPageDetails',
|
|
||||||
tab: tab,
|
|
||||||
sender: 'contextMenu',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
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(),
|
|
||||||
]);
|
|
||||||
|
|
||||||
chrome.runtime.sendMessage({
|
|
||||||
command: 'doneLoggingOut', expired: expired,
|
|
||||||
});
|
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
chrome.tabs.sendMessage(tab.id, {
|
|
||||||
command: 'collectPageDetails',
|
|
||||||
tab: tab,
|
|
||||||
sender: sender,
|
|
||||||
}, options, () => {
|
|
||||||
if (chrome.runtime.lastError) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private cleanupLoginsToAdd() {
|
private cleanupLoginsToAdd() {
|
||||||
for (let i = this.loginsToAdd.length - 1; i >= 0; i--) {
|
for (let i = this.loginsToAdd.length - 1; i >= 0; i--) {
|
||||||
if (this.loginsToAdd[i].expires < new Date()) {
|
if (this.loginsToAdd[i].expires < new Date()) {
|
||||||
@@ -449,22 +399,6 @@ export default class MainBackground {
|
|||||||
setTimeout(() => this.cleanupLoginsToAdd(), 2 * 60 * 1000); // check every 2 minutes
|
setTimeout(() => this.cleanupLoginsToAdd(), 2 * 60 * 1000); // check every 2 minutes
|
||||||
}
|
}
|
||||||
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private doCheck(tab: any) {
|
private doCheck(tab: any) {
|
||||||
if (tab == null) {
|
if (tab == null) {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -104,7 +104,11 @@ export default class RuntimeBackground {
|
|||||||
break;
|
break;
|
||||||
case 'contextMenu':
|
case 'contextMenu':
|
||||||
clearTimeout(this.autofillTimeout);
|
clearTimeout(this.autofillTimeout);
|
||||||
this.pageDetailsToAutoFill.push({ frameId: sender.frameId, tab: msg.tab, details: msg.details });
|
this.pageDetailsToAutoFill.push({
|
||||||
|
frameId: sender.frameId,
|
||||||
|
tab: msg.tab,
|
||||||
|
details: msg.details,
|
||||||
|
});
|
||||||
this.autofillTimeout = setTimeout(async () => await this.autofillPage(), 300);
|
this.autofillTimeout = setTimeout(async () => await this.autofillPage(), 300);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
|||||||
Reference in New Issue
Block a user