import browser from 'webextension-polyfill'; import { getBlockExplorerLink } from '@metamask/etherscan-link'; import { startCase, toLower } from 'lodash'; import { getEnvironmentType } from '../lib/util'; import { ENVIRONMENT_TYPE_BACKGROUND } from '../../../shared/constants/app'; import { TransactionStatus } from '../../../shared/constants/transaction'; import { getURLHostName } from '../../../ui/helpers/utils/util'; export default class ExtensionPlatform { // // Public // reload() { browser.runtime.reload(); } async openTab(options) { const newTab = await browser.tabs.create(options); return newTab; } async openWindow(options) { const newWindow = await browser.windows.create(options); return newWindow; } async focusWindow(windowId) { await browser.windows.update(windowId, { focused: true }); } async updateWindowPosition(windowId, left, top) { await browser.windows.update(windowId, { left, top }); } async getLastFocusedWindow() { const windowObject = await browser.windows.getLastFocused(); return windowObject; } async closeCurrentWindow() { const windowDetails = await browser.windows.getCurrent(); browser.windows.remove(windowDetails.id); } getVersion() { const { version, version_name: versionName } = browser.runtime.getManifest(); const versionParts = version.split('.'); if (versionName) { if (versionParts.length < 4) { throw new Error(`Version missing build number: '${version}'`); } // On Chrome, a more descriptive representation of the version is stored in the // `version_name` field for display purposes. We use this field instead of the `version` // field on Chrome for non-main builds (i.e. Flask, Beta) because we want to show the // version in the SemVer-compliant format "v[major].[minor].[patch]-[build-type].[build-number]", // yet Chrome does not allow letters in the `version` field. return versionName; // A fourth version part is sometimes present for "rollback" Chrome builds } else if (![3, 4].includes(versionParts.length)) { throw new Error(`Invalid version: ${version}`); } else if (versionParts[2].match(/[^\d]/u)) { // On Firefox, the build type and build version are in the third part of the version. const [major, minor, patchAndPrerelease] = versionParts; const matches = patchAndPrerelease.match(/^(\d+)([A-Za-z]+)(\d)+$/u); if (matches === null) { throw new Error(`Version contains invalid prerelease: ${version}`); } const [, patch, buildType, buildVersion] = matches; return `${major}.${minor}.${patch}-${buildType}.${buildVersion}`; } // If there is no `version_name` and there are only 3 or 4 version parts, then this is not a // prerelease and the version requires no modification. return version; } getExtensionURL(route = null, queryString = null) { let extensionURL = browser.runtime.getURL('home.html'); if (route) { extensionURL += `#${route}`; } if (queryString) { extensionURL += `?${queryString}`; } return extensionURL; } openExtensionInBrowser( route = null, queryString = null, keepWindowOpen = false, ) { const extensionURL = this.getExtensionURL( route, queryString, keepWindowOpen, ); this.openTab({ url: extensionURL }); if ( getEnvironmentType() !== ENVIRONMENT_TYPE_BACKGROUND && !keepWindowOpen ) { window.close(); } } getPlatformInfo(cb) { try { const platformInfo = browser.runtime.getPlatformInfo(); cb(platformInfo); return; } catch (e) { cb(e); // eslint-disable-next-line no-useless-return return; } } async showTransactionNotification(txMeta, rpcPrefs) { const { status, txReceipt: { status: receiptStatus } = {} } = txMeta; if (status === TransactionStatus.confirmed) { // There was an on-chain failure receiptStatus === '0x0' ? await this._showFailedTransaction( txMeta, 'Transaction encountered an error.', ) : await this._showConfirmedTransaction(txMeta, rpcPrefs); } else if (status === TransactionStatus.failed) { await this._showFailedTransaction(txMeta); } } addOnRemovedListener(listener) { browser.windows.onRemoved.addListener(listener); } async getAllWindows() { const windows = await browser.windows.getAll(); return windows; } async getActiveTabs() { const tabs = await browser.tabs.query({ active: true }); return tabs; } async currentTab() { const tab = await browser.tabs.getCurrent(); return tab; } async switchToTab(tabId) { const tab = await browser.tabs.update(tabId, { highlighted: true }); return tab; } async switchToAnotherURL(tabId, url) { await browser.tabs.update(tabId, { url }); } async closeTab(tabId) { await browser.tabs.remove(tabId); } async _showConfirmedTransaction(txMeta, rpcPrefs) { this._subscribeToNotificationClicked(); const url = getBlockExplorerLink(txMeta, rpcPrefs); const nonce = parseInt(txMeta.txParams.nonce, 16); const view = startCase( toLower(getURLHostName(url).replace(/([.]\w+)$/u, '')), ); const title = 'Confirmed transaction'; const message = `Transaction ${nonce} confirmed! ${ url.length ? `View on ${view}` : '' }`; await this._showNotification(title, message, url); } async _showFailedTransaction(txMeta, errorMessage) { const nonce = parseInt(txMeta.txParams.nonce, 16); const title = 'Failed transaction'; let message = `Transaction ${nonce} failed! ${ errorMessage || txMeta.err.message }`; ///: BEGIN:ONLY_INCLUDE_IN(mmi) if (isNaN(nonce)) { message = `Transaction failed! ${errorMessage || txMeta.err.message}`; } ///: END:ONLY_INCLUDE_IN await this._showNotification(title, message); } async _showNotification(title, message, url) { const iconUrl = await browser.runtime.getURL('../../images/icon-64.png'); await browser.notifications.create(url, { type: 'basic', title, iconUrl, message, }); } _subscribeToNotificationClicked() { if (!browser.notifications.onClicked.hasListener(this._viewOnEtherscan)) { browser.notifications.onClicked.addListener(this._viewOnEtherscan); } } _viewOnEtherscan(url) { if (url.startsWith('https://')) { browser.tabs.create({ url }); } } }