lbry-desktop/electron/createWindow.js

233 lines
8.1 KiB
JavaScript
Raw Permalink Normal View History

import { WEBPACK_ELECTRON_PORT } from 'config';
import { app, BrowserWindow, dialog, screen, nativeImage } from 'electron';
import isDev from 'electron-is-dev';
import windowStateKeeper from 'electron-window-state';
2019-11-05 13:54:58 -05:00
import SUPPORTED_LANGUAGES from 'constants/supported_languages';
import { SUPPORTED_SUB_LANGUAGE_CODES, SUB_LANG_CODE_LEN } from 'constants/supported_sub_languages';
import SUPPORTED_BROWSER_LANGUAGES from 'constants/supported_browser_languages';
2020-08-13 13:57:00 -03:00
import { TO_TRAY_WHEN_CLOSED } from 'constants/settings';
2018-01-17 23:13:08 -03:00
import setupBarMenu from './menu/setupBarMenu';
2019-11-12 17:03:11 -05:00
import * as PAGES from 'constants/pages';
const remote = require('@electron/remote/main');
2022-02-16 18:48:09 -03:00
const shell = require('electron').shell;
function GetAppLangCode() {
// https://www.electronjs.org/docs/api/locales
// 1. Gets the user locale.
// 2. Converts unsupported sub-languages to its primary (e.g. "en-GB" -> "en").
// Note that the primary itself may or may not be a supported language
// (up to clients to verify against SUPPORTED_LANGUAGES).
const langCode = app.getLocale();
if (langCode.length === SUB_LANG_CODE_LEN && !SUPPORTED_SUB_LANGUAGE_CODES.includes(langCode)) {
return SUPPORTED_BROWSER_LANGUAGES[langCode.slice(0, 2)];
}
return SUPPORTED_BROWSER_LANGUAGES[langCode];
}
2018-02-23 18:20:12 -05:00
export default appState => {
// Get primary display dimensions from Electron.
const { width, height } = screen.getPrimaryDisplay().workAreaSize;
// Load the previous state with fallback to defaults.
const windowState = windowStateKeeper({
defaultWidth: width,
defaultHeight: height,
});
2019-11-18 10:30:15 -08:00
const startMinimized = (process.argv || []).includes('--hidden');
const windowConfiguration = {
2019-09-26 12:07:11 -04:00
backgroundColor: '#270f34', // Located in src/scss/init/_vars.scss `--color-background--splash`
2018-03-26 14:32:43 -07:00
minWidth: 950,
2018-01-17 23:13:08 -03:00
minHeight: 600,
autoHideMenuBar: true,
2019-10-09 12:34:18 -04:00
titleBarStyle: 'hiddenInset',
2018-01-17 23:13:08 -03:00
show: false,
// Create the window using the state information.
x: windowState.x,
y: windowState.y,
// If state is undefined, create window as maximized.
width: windowState.width === undefined ? width : windowState.width,
height: windowState.height === undefined ? height : windowState.height,
icon: nativeImage.createFromPath('static/img/tray/default/tray.png'),
webPreferences: {
// Disable renderer process's webSecurity on development to enable CORS.
webSecurity: !isDev,
plugins: true,
nodeIntegration: true,
contextIsolation: false,
enableRemoteModule: true, // see about removing this
},
};
2019-05-01 10:18:26 -04:00
const lbryProto = 'lbry://';
const lbryProtoQ = 'lbry://?';
2019-09-02 23:01:04 -04:00
const rendererURL = isDev ? `http://localhost:${WEBPACK_ELECTRON_PORT}` : `file://${__dirname}/index.html`;
2018-01-17 23:13:08 -03:00
let window = new BrowserWindow(windowConfiguration);
remote.enable(window.webContents);
2018-01-17 23:13:08 -03:00
// Let us register listeners on the window, so we can update the state
// automatically (the listeners will be removed when the window is closed)
// and restore the maximized or full screen state.
windowState.manage(window);
2018-01-17 23:13:08 -03:00
let deepLinkingURI;
2019-05-01 10:18:26 -04:00
2019-05-07 17:38:29 -04:00
if ((process.platform === 'win32' || process.platform === 'linux') && String(process.argv[1]).startsWith('lbry')) {
2019-05-01 10:18:26 -04:00
[, deepLinkingURI] = process.argv || '';
2018-01-17 23:13:08 -03:00
// Keep only command line / deep linked arguments
// Windows normalizes URIs when they're passed in from other apps. On Windows, this tries to
// restore the original URI that was typed.
// - If the URI has no path, Windows adds a trailing slash. LBRY URIs can't have a slash with no
// path, so we just strip it off.
// - In a URI with a claim ID, like lbry://channel#claimid, Windows interprets the hash mark as
// an anchor and converts it to lbry://channel/#claimid. We remove the slash here as well.
2018-06-05 11:22:15 -04:00
// - ? also interpreted as an anchor, remove slash also.
if (process.platform === 'win32') {
2018-06-05 11:22:15 -04:00
deepLinkingURI = deepLinkingURI
.replace(/\/$/, '')
.replace('/#', '#')
.replace('/?', '?');
}
} else {
2019-05-01 10:18:26 -04:00
deepLinkingURI = appState.macDeepLinkingURI || '';
}
// is it a lbry://? pointing to an app page
if (deepLinkingURI.includes(lbryProtoQ)) {
let path = deepLinkingURI.slice(lbryProtoQ.length);
2019-08-28 19:53:48 -04:00
let page = path.indexOf('?') >= 0 ? path.substring(0, path.indexOf('?')) : path;
if (Object.values(PAGES).includes(page)) {
2019-05-01 10:18:26 -04:00
deepLinkingURI = deepLinkingURI.replace(lbryProtoQ, '#/$/');
} else {
deepLinkingURI = '';
}
// else is it a claim
} else if (deepLinkingURI.includes(lbryProto)) {
deepLinkingURI = deepLinkingURI.replace(lbryProto, '#');
} else {
deepLinkingURI = '';
2018-01-17 23:13:08 -03:00
}
setupBarMenu();
2019-11-05 13:54:58 -05:00
window.loadURL(rendererURL + deepLinkingURI);
2018-02-23 18:20:12 -05:00
window.on('close', event => {
2020-08-13 13:57:00 -03:00
if (appState.isQuitting) {
return;
}
if (!appState.autoUpdateAccepted) {
2018-02-23 18:20:12 -05:00
event.preventDefault();
if (window.isFullScreen()) {
window.once('leave-full-screen', () => {
window.hide();
});
window.setFullScreen(false);
} else {
window.hide();
}
2018-02-23 18:20:12 -05:00
}
2020-08-13 13:57:00 -03:00
const getToTrayWhenClosedSetting = window.webContents.executeJavaScript(`localStorage.getItem('${TO_TRAY_WHEN_CLOSED}')`);
getToTrayWhenClosedSetting.then(toTrayWhenClosedSetting => {
const closeApp = toTrayWhenClosedSetting === 'false';
if (closeApp) {
app.quit();
}
});
2018-01-17 23:13:08 -03:00
});
window.on('focus', () => {
window.webContents.send('window-is-focused', null);
});
window.on('unresponsive', () => {
dialog.showMessageBox(
window,
{
type: 'warning',
buttons: ['Wait', 'Quit'],
title: 'LBRY Unresponsive',
defaultId: 1,
message: 'LBRY is not responding. Would you like to quit?',
cancelId: 0,
},
buttonIndex => {
if (buttonIndex === 1) app.quit();
}
);
});
2019-11-05 13:54:58 -05:00
2018-01-17 23:13:08 -03:00
window.once('ready-to-show', () => {
2019-11-18 10:30:15 -08:00
startMinimized ? window.hide() : window.show();
2018-01-17 23:13:08 -03:00
});
2019-11-18 14:42:29 -05:00
// A backup incase https://github.com/electron/electron/issues/7779 happens
2019-11-18 10:30:15 -08:00
window.webContents.once('dom-ready', () => {
startMinimized && window.hide();
});
2019-11-18 10:30:15 -08:00
2018-01-17 23:13:08 -03:00
window.webContents.on('did-finish-load', () => {
window.webContents.session.setUserAgent(`LBRY/${app.getVersion()}`);
2019-11-05 13:54:58 -05:00
// restore the user's previous language - we have to do this from here because only electron process can access app.getLocale()
window.webContents.executeJavaScript("localStorage.getItem('language')").then(storedLanguage => {
const language =
storedLanguage && storedLanguage !== 'undefined' && storedLanguage !== 'null'
? storedLanguage
: GetAppLangCode();
2019-11-05 13:54:58 -05:00
if (language !== 'en' && SUPPORTED_LANGUAGES[language]) {
window.webContents.send('language-set', language);
}
});
2018-01-17 23:13:08 -03:00
});
window.webContents.on('crashed', () => {
window = null;
});
window.webContents.setWindowOpenHandler((details) => {
// Only open http and https links to prevent
// security issues.
if (['https:', 'http:'].includes(new URL(details.url).protocol)) {
shell.openExternal(details.url);
}
return { action: 'deny' };
});
2018-01-17 23:13:08 -03:00
window.webContents.on('update-target-url', (event, url) => {
// Change internal links to the lbry protocol. External (https) links should remain unchanged.
let hoverUrlBase = `http://localhost:${WEBPACK_ELECTRON_PORT}/`;
if (!isDev) {
// Return format of 'update-target-url':
// Linux: file:///@claim
// Windows: file:///C:/@claim
// Use '__dirname' in case installation is not in C:
const path = require('path');
const exeRoot = path.parse(__dirname).root;
if (process.platform === 'win32') {
// Add extra "/" prefix. Convert "C:\" to "C:/"
hoverUrlBase = `file:///` + exeRoot.replace(/\\/g, '/');
} else {
hoverUrlBase = `file://` + exeRoot;
}
}
let dispUrl = url.replace(hoverUrlBase, lbryProto);
// Non-claims don't need the lbry protocol:
if (dispUrl === lbryProto) {
dispUrl = 'Home';
} else if (dispUrl.startsWith(lbryProto + '$/')) {
dispUrl = dispUrl.replace(lbryProto, '/');
}
window.webContents.send('update-target-url', dispUrl);
});
2018-01-17 23:13:08 -03:00
return window;
};