2017-09-02 01:50:31 +00:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* MIT License
|
|
|
|
*
|
2018-04-08 00:22:32 +00:00
|
|
|
* Copyright (c) 2018 F-List
|
2017-09-02 01:50:31 +00:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
|
|
* copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
|
|
|
*
|
|
|
|
* This license header applies to this file and all of the non-third-party assets it includes.
|
|
|
|
* @file The entry point for the Electron main thread of F-Chat 3.0.
|
2018-04-08 00:22:32 +00:00
|
|
|
* @copyright 2018 F-List
|
2017-09-02 01:50:31 +00:00
|
|
|
* @author Maya Wolf <maya@f-list.net>
|
|
|
|
* @version 3.0
|
|
|
|
* @see {@link https://github.com/f-list/exported|GitHub repo}
|
|
|
|
*/
|
2019-07-08 19:08:16 +00:00
|
|
|
|
|
|
|
// import { DebugLogger } from './debug-logger';
|
|
|
|
// // @ts-ignore
|
|
|
|
// const dl = new DebugLogger('main');
|
|
|
|
|
2017-09-02 01:50:31 +00:00
|
|
|
import * as electron from 'electron';
|
2017-12-05 01:47:27 +00:00
|
|
|
import log from 'electron-log'; //tslint:disable-line:match-default-export-name
|
2018-01-06 16:14:21 +00:00
|
|
|
import * as fs from 'fs';
|
2017-09-02 01:50:31 +00:00
|
|
|
import * as path from 'path';
|
2020-03-15 16:23:39 +00:00
|
|
|
// import * as url from 'url';
|
2018-01-06 16:14:21 +00:00
|
|
|
import l from '../chat/localize';
|
2019-09-17 17:14:14 +00:00
|
|
|
import {defaultHost, GeneralSettings} from './common';
|
2020-04-03 15:00:38 +00:00
|
|
|
import { getSafeLanguages, knownLanguageNames, updateSupportedLanguages } from './language';
|
2017-09-02 01:50:31 +00:00
|
|
|
import * as windowState from './window_state';
|
2018-01-06 16:14:21 +00:00
|
|
|
import BrowserWindow = Electron.BrowserWindow;
|
2018-03-28 13:51:05 +00:00
|
|
|
import MenuItem = Electron.MenuItem;
|
2020-04-01 22:19:55 +00:00
|
|
|
import { ElectronBlocker } from '@cliqz/adblocker-electron';
|
|
|
|
import fetch from 'node-fetch';
|
2020-04-02 00:27:21 +00:00
|
|
|
import MenuItemConstructorOptions = Electron.MenuItemConstructorOptions;
|
2020-04-03 00:46:36 +00:00
|
|
|
import * as _ from 'lodash';
|
2020-04-04 20:22:42 +00:00
|
|
|
import DownloadItem = Electron.DownloadItem;
|
2017-09-02 01:50:31 +00:00
|
|
|
|
|
|
|
// Module to control application life.
|
|
|
|
const app = electron.app;
|
|
|
|
|
|
|
|
// Keep a global reference of the window object, if you don't, the window will
|
|
|
|
// be closed automatically when the JavaScript object is garbage collected.
|
2017-12-05 01:47:27 +00:00
|
|
|
const windows: Electron.BrowserWindow[] = [];
|
2018-01-06 16:14:21 +00:00
|
|
|
const characters: string[] = [];
|
|
|
|
let tabCount = 0;
|
2017-09-02 01:50:31 +00:00
|
|
|
|
|
|
|
const baseDir = app.getPath('userData');
|
2019-09-17 17:14:14 +00:00
|
|
|
fs.mkdirSync(baseDir, {recursive: true});
|
2018-08-10 16:59:37 +00:00
|
|
|
let shouldImportSettings = false;
|
|
|
|
|
|
|
|
const settingsDir = path.join(baseDir, 'data');
|
2019-09-17 17:14:14 +00:00
|
|
|
fs.mkdirSync(settingsDir, {recursive: true});
|
2018-08-10 16:59:37 +00:00
|
|
|
const settingsFile = path.join(settingsDir, 'settings');
|
|
|
|
const settings = new GeneralSettings();
|
2017-09-02 01:50:31 +00:00
|
|
|
|
2018-09-28 00:08:10 +00:00
|
|
|
if(!fs.existsSync(settingsFile)) shouldImportSettings = true;
|
|
|
|
else
|
|
|
|
try {
|
|
|
|
Object.assign(settings, <GeneralSettings>JSON.parse(fs.readFileSync(settingsFile, 'utf8')));
|
|
|
|
} catch(e) {
|
|
|
|
log.error(`Error loading settings: ${e}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!settings.hwAcceleration) {
|
|
|
|
log.info('Disabling hardware acceleration.');
|
|
|
|
app.disableHardwareAcceleration();
|
|
|
|
}
|
|
|
|
|
2020-04-03 00:46:36 +00:00
|
|
|
// async function setDictionary(lang: string | undefined): Promise<void> {
|
|
|
|
// if(lang !== undefined) await ensureDictionary(lang);
|
|
|
|
// settings.spellcheckLang = lang;
|
|
|
|
// setGeneralSettings(settings);
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
|
|
export function updateSpellCheckerLanguages(langs: string[]): void {
|
|
|
|
// console.log('Language support:', langs);
|
2020-04-11 14:48:09 +00:00
|
|
|
electron.session.defaultSession.setSpellCheckerLanguages(langs);
|
2020-04-03 00:46:36 +00:00
|
|
|
|
|
|
|
for (const w of windows) {
|
|
|
|
// console.log('LANG SEND');
|
2020-04-11 14:48:09 +00:00
|
|
|
w.webContents.session.setSpellCheckerLanguages(langs);
|
2020-04-03 00:46:36 +00:00
|
|
|
w.webContents.send('update-dictionaries', langs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function toggleDictionary(lang: string): Promise<void> {
|
|
|
|
const activeLangs = getSafeLanguages(settings.spellcheckLang);
|
|
|
|
|
|
|
|
// console.log('INITIAL LANG', activeLangs, lang);
|
|
|
|
|
|
|
|
let newLangs: string[] = [];
|
|
|
|
|
|
|
|
if (_.indexOf(activeLangs, lang) >= 0) {
|
|
|
|
newLangs = _.reject(activeLangs, (al) => (al === lang));
|
|
|
|
} else {
|
|
|
|
activeLangs.push(lang);
|
|
|
|
newLangs = activeLangs;
|
|
|
|
}
|
|
|
|
|
|
|
|
settings.spellcheckLang = newLangs;
|
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
setGeneralSettings(settings);
|
2020-04-03 00:46:36 +00:00
|
|
|
|
|
|
|
// console.log('NEW LANG', newLangs);
|
|
|
|
|
|
|
|
updateSpellCheckerLanguages(newLangs);
|
2018-01-06 16:14:21 +00:00
|
|
|
}
|
2017-09-02 01:50:31 +00:00
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
function setGeneralSettings(value: GeneralSettings): void {
|
|
|
|
fs.writeFileSync(path.join(settingsDir, 'settings'), JSON.stringify(value));
|
|
|
|
for(const w of electron.webContents.getAllWebContents()) w.send('settings', settings);
|
|
|
|
shouldImportSettings = false;
|
2017-09-02 01:50:31 +00:00
|
|
|
}
|
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
async function addSpellcheckerItems(menu: Electron.Menu): Promise<void> {
|
2020-04-03 00:46:36 +00:00
|
|
|
const selected = getSafeLanguages(settings.spellcheckLang);
|
|
|
|
const langs = electron.session.defaultSession.availableSpellCheckerLanguages;
|
|
|
|
|
|
|
|
const sortedLangs = _.sortBy(
|
|
|
|
_.map(
|
|
|
|
langs,
|
|
|
|
(lang) => ({lang, name: (lang in knownLanguageNames) ? `${(knownLanguageNames as any)[lang]} (${lang})` : lang})
|
|
|
|
),
|
|
|
|
'name'
|
|
|
|
);
|
|
|
|
|
|
|
|
for (const lang of sortedLangs)
|
2018-01-06 16:14:21 +00:00
|
|
|
menu.append(new electron.MenuItem({
|
2020-04-03 00:46:36 +00:00
|
|
|
type: 'checkbox',
|
|
|
|
label: lang.name,
|
|
|
|
checked: (_.indexOf(selected, lang.lang) >= 0),
|
|
|
|
click: async() => toggleDictionary(lang.lang)
|
2018-01-06 16:14:21 +00:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
function setUpWebContents(webContents: Electron.WebContents): void {
|
2017-09-02 01:50:31 +00:00
|
|
|
const openLinkExternally = (e: Event, linkUrl: string) => {
|
|
|
|
e.preventDefault();
|
2018-03-04 02:32:26 +00:00
|
|
|
const profileMatch = linkUrl.match(/^https?:\/\/(www\.)?f-list.net\/c\/([^/#]+)\/?#?/);
|
2018-01-06 16:14:21 +00:00
|
|
|
if(profileMatch !== null && settings.profileViewer) webContents.send('open-profile', decodeURIComponent(profileMatch[2]));
|
2019-09-17 17:14:14 +00:00
|
|
|
else return electron.shell.openExternal(linkUrl);
|
2017-09-02 01:50:31 +00:00
|
|
|
};
|
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
webContents.on('will-navigate', openLinkExternally);
|
|
|
|
webContents.on('new-window', openLinkExternally);
|
2017-09-02 01:50:31 +00:00
|
|
|
}
|
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
function createWindow(): Electron.BrowserWindow | undefined {
|
|
|
|
if(tabCount >= 3) return;
|
2017-09-02 01:50:31 +00:00
|
|
|
const lastState = windowState.getSavedWindowState();
|
2018-01-06 16:14:21 +00:00
|
|
|
const windowProperties: Electron.BrowserWindowConstructorOptions & {maximized: boolean} = {
|
2019-06-08 22:51:04 +00:00
|
|
|
...lastState, center: lastState.x === undefined, show: false,
|
2020-04-01 22:19:55 +00:00
|
|
|
webPreferences: { webviewTag: true, nodeIntegration: true, spellcheck: true }
|
2018-01-06 16:14:21 +00:00
|
|
|
};
|
2020-03-15 15:22:55 +00:00
|
|
|
|
|
|
|
if(process.platform === 'darwin') {
|
2020-03-15 16:23:39 +00:00
|
|
|
windowProperties.titleBarStyle = 'hiddenInset';
|
|
|
|
// windowProperties.frame = true;
|
2020-03-15 15:22:55 +00:00
|
|
|
} else {
|
|
|
|
windowProperties.frame = false;
|
|
|
|
}
|
|
|
|
|
2017-12-05 01:47:27 +00:00
|
|
|
const window = new electron.BrowserWindow(windowProperties);
|
|
|
|
windows.push(window);
|
2017-09-02 01:50:31 +00:00
|
|
|
|
2020-04-03 00:46:36 +00:00
|
|
|
const safeLanguages = settings.spellcheckLang ? _.castArray(settings.spellcheckLang) : [];
|
|
|
|
electron.session.defaultSession.setSpellCheckerLanguages(safeLanguages);
|
2020-04-11 14:48:09 +00:00
|
|
|
window.webContents.session.setSpellCheckerLanguages(safeLanguages);
|
2020-04-03 00:46:36 +00:00
|
|
|
|
2020-04-02 00:27:21 +00:00
|
|
|
// tslint:disable-next-line:no-floating-promises
|
|
|
|
ElectronBlocker.fromLists(
|
|
|
|
fetch,
|
|
|
|
[
|
|
|
|
'https://easylist.to/easylist/easylist.txt',
|
|
|
|
'https://easylist.to/easylist/easyprivacy.txt', // EasyPrivacy
|
2020-04-05 19:10:59 +00:00
|
|
|
'https://easylist-downloads.adblockplus.org/easylist-cookie.txt', // Easy Cookies
|
2020-04-02 00:27:21 +00:00
|
|
|
'https://easylist.to/easylist/fanboy-social.txt', // Fanboy Social
|
|
|
|
'https://easylist.to/easylist/fanboy-annoyance.txt', // Fanboy Annoyances
|
|
|
|
'https://filters.adtidy.org/extension/chromium/filters/2.txt', // AdGuard Base
|
|
|
|
'https://filters.adtidy.org/extension/chromium/filters/11.txt', // AdGuard Mobile Ads
|
|
|
|
'https://filters.adtidy.org/extension/chromium/filters/4.txt', // AdGuard Social Media
|
|
|
|
'https://filters.adtidy.org/extension/chromium/filters/14.txt', // AdGuard Annoyances
|
2020-04-05 19:10:59 +00:00
|
|
|
'https://raw.githubusercontent.com/uBlockOrigin/uAssets/master/filters/annoyances.txt', // uBlock Origin Annoyances
|
2020-04-02 00:27:21 +00:00
|
|
|
'https://raw.githubusercontent.com/uBlockOrigin/uAssets/master/filters/filters.txt', // uBlock Origin Filters
|
|
|
|
'https://raw.githubusercontent.com/uBlockOrigin/uAssets/master/filters/privacy.txt', // uBlock Origin Privacy
|
2020-04-05 19:10:59 +00:00
|
|
|
'https://raw.githubusercontent.com/uBlockOrigin/uAssets/master/filters/badware.txt', // uBlock Origin Badware
|
|
|
|
'https://raw.githubusercontent.com/uBlockOrigin/uAssets/master/filters/resource-abuse.txt', // uBlock Origin Resource Abuse
|
|
|
|
'https://raw.githubusercontent.com/uBlockOrigin/uAssets/master/filters/unbreak.txt' // uBlock Origin Unbreak
|
2020-04-02 00:27:21 +00:00
|
|
|
]
|
|
|
|
).then(
|
2020-04-01 22:19:55 +00:00
|
|
|
(blocker) => {
|
|
|
|
blocker.enableBlockingInSession(electron.session.defaultSession);
|
|
|
|
|
2020-04-05 19:10:59 +00:00
|
|
|
// blocker.on('request-blocked', (request: Request) => {
|
|
|
|
// console.log('blocked', request.url);
|
|
|
|
// });
|
2020-04-03 00:46:36 +00:00
|
|
|
//
|
2020-04-05 19:10:59 +00:00
|
|
|
// blocker.on('request-redirected', (request: Request) => {
|
|
|
|
// console.log('redirected', request.url);
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// blocker.on('request-whitelisted', (request: Request) => {
|
|
|
|
// console.log('whitelisted', request.url);
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// blocker.on('csp-injected', (request: Request) => {
|
|
|
|
// console.log('csp', request.url);
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// blocker.on('script-injected', (script: string, url: string) => {
|
|
|
|
// console.log('script', script.length, url);
|
|
|
|
// });
|
|
|
|
//
|
|
|
|
// blocker.on('style-injected', (style: string, url: string) => {
|
|
|
|
// console.log('style', style.length, url);
|
|
|
|
// });
|
2020-04-01 22:19:55 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2020-04-04 20:22:42 +00:00
|
|
|
|
|
|
|
// This prevents automatic download prompts on certain webview URLs without
|
|
|
|
// stopping conversation logs from being downloaded
|
|
|
|
electron.session.defaultSession.on(
|
|
|
|
'will-download',
|
|
|
|
(e: Event, item: DownloadItem) => {
|
|
|
|
if (!item.getURL().match(/^blob:file:/)) {
|
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
|
2020-04-09 23:06:48 +00:00
|
|
|
console.log('GOT HERE');
|
|
|
|
|
2020-04-02 00:27:21 +00:00
|
|
|
// tslint:disable-next-line:no-floating-promises
|
2020-03-15 15:22:55 +00:00
|
|
|
window.loadFile(
|
|
|
|
path.join(__dirname, 'window.html'),
|
|
|
|
{
|
2020-04-02 00:27:21 +00:00
|
|
|
query: {settings: JSON.stringify(settings), import: shouldImportSettings ? 'true' : ''}
|
2020-03-15 15:22:55 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2020-04-01 22:19:55 +00:00
|
|
|
|
2020-03-15 15:22:55 +00:00
|
|
|
// window.loadURL(url.format({ //tslint:disable-line:no-floating-promises
|
|
|
|
// pathname: path.join(__dirname, 'window.html'),
|
|
|
|
// protocol: 'file:',
|
|
|
|
// slashes: true,
|
|
|
|
// query: {settings: JSON.stringify(settings), import: shouldImportSettings ? 'true' : []}
|
|
|
|
// }));
|
2017-09-02 01:50:31 +00:00
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
setUpWebContents(window.webContents);
|
2017-09-02 01:50:31 +00:00
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
// Save window state when it is being closed.
|
|
|
|
window.on('close', () => windowState.setSavedWindowState(window));
|
2017-12-05 01:47:27 +00:00
|
|
|
window.on('closed', () => windows.splice(windows.indexOf(window), 1));
|
2018-03-28 13:51:05 +00:00
|
|
|
window.once('ready-to-show', () => {
|
|
|
|
window.show();
|
|
|
|
if(lastState.maximized) window.maximize();
|
|
|
|
});
|
2019-06-09 23:33:52 +00:00
|
|
|
|
2018-01-06 16:14:21 +00:00
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
|
|
|
function showPatchNotes(): void {
|
2019-09-17 17:14:14 +00:00
|
|
|
electron.shell.openExternal('https://wiki.f-list.net/F-Chat_3.0#Changelog'); //tslint:disable-line:no-floating-promises
|
2018-01-06 16:14:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function onReady(): void {
|
2018-08-10 16:59:37 +00:00
|
|
|
log.transports.file.level = 'debug';
|
|
|
|
log.transports.console.level = 'debug';
|
|
|
|
log.transports.file.maxSize = 5 * 1024 * 1024;
|
|
|
|
log.transports.file.file = path.join(baseDir, 'log.txt');
|
|
|
|
log.info('Starting application.');
|
|
|
|
|
2020-04-03 15:00:38 +00:00
|
|
|
updateSupportedLanguages(electron.session.defaultSession.availableSpellCheckerLanguages);
|
|
|
|
|
2018-08-10 16:59:37 +00:00
|
|
|
app.setAppUserModelId('com.squirrel.fchat.F-Chat');
|
2018-01-06 16:14:21 +00:00
|
|
|
app.on('open-file', createWindow);
|
|
|
|
|
|
|
|
if(settings.version !== app.getVersion()) {
|
|
|
|
showPatchNotes();
|
2019-09-17 17:14:14 +00:00
|
|
|
if(settings.host === 'wss://chat.f-list.net:9799')
|
|
|
|
settings.host = defaultHost;
|
2018-01-06 16:14:21 +00:00
|
|
|
settings.version = app.getVersion();
|
|
|
|
setGeneralSettings(settings);
|
|
|
|
}
|
|
|
|
|
2018-08-10 16:59:37 +00:00
|
|
|
const updaterUrl = `https://client.f-list.net/${process.platform}`;
|
2018-01-06 16:14:21 +00:00
|
|
|
if(process.env.NODE_ENV === 'production') {
|
2018-08-10 16:59:37 +00:00
|
|
|
electron.autoUpdater.setFeedURL({url: updaterUrl + (settings.beta ? '?channel=beta' : ''), serverType: 'json'});
|
|
|
|
setTimeout(() => electron.autoUpdater.checkForUpdates(), 10000);
|
|
|
|
const updateTimer = setInterval(() => electron.autoUpdater.checkForUpdates(), 3600000);
|
|
|
|
electron.autoUpdater.on('update-downloaded', () => {
|
2018-01-06 16:14:21 +00:00
|
|
|
clearInterval(updateTimer);
|
2018-03-04 02:32:26 +00:00
|
|
|
const menu = electron.Menu.getApplicationMenu()!;
|
2018-03-28 13:51:05 +00:00
|
|
|
const item = menu.getMenuItemById('update') as MenuItem | null;
|
|
|
|
if(item !== null) item.visible = true;
|
|
|
|
else
|
|
|
|
menu.append(new electron.MenuItem({
|
|
|
|
label: l('action.updateAvailable'),
|
|
|
|
submenu: electron.Menu.buildFromTemplate([{
|
|
|
|
label: l('action.update'),
|
|
|
|
click: () => {
|
|
|
|
for(const w of windows) w.webContents.send('quit');
|
2018-08-10 16:59:37 +00:00
|
|
|
electron.autoUpdater.quitAndInstall();
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
}, {
|
|
|
|
label: l('help.changelog'),
|
|
|
|
click: showPatchNotes
|
|
|
|
}]),
|
|
|
|
id: 'update'
|
|
|
|
}));
|
2018-01-06 16:14:21 +00:00
|
|
|
electron.Menu.setApplicationMenu(menu);
|
2018-03-28 13:51:05 +00:00
|
|
|
for(const w of windows) w.webContents.send('update-available', true);
|
|
|
|
});
|
2018-08-10 16:59:37 +00:00
|
|
|
electron.autoUpdater.on('update-not-available', () => {
|
2018-03-28 13:51:05 +00:00
|
|
|
for(const w of windows) w.webContents.send('update-available', false);
|
|
|
|
const item = electron.Menu.getApplicationMenu()!.getMenuItemById('update') as MenuItem | null;
|
|
|
|
if(item !== null) item.visible = false;
|
2018-01-06 16:14:21 +00:00
|
|
|
});
|
2018-08-10 16:59:37 +00:00
|
|
|
electron.autoUpdater.on('error', (e) => log.error(e));
|
2018-01-06 16:14:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const viewItem = {
|
|
|
|
label: `&${l('action.view')}`,
|
|
|
|
submenu: <Electron.MenuItemConstructorOptions[]>[
|
|
|
|
{role: 'resetzoom'},
|
|
|
|
{role: 'zoomin'},
|
|
|
|
{role: 'zoomout'},
|
|
|
|
{type: 'separator'},
|
|
|
|
{role: 'togglefullscreen'}
|
|
|
|
]
|
|
|
|
};
|
|
|
|
if(process.env.NODE_ENV !== 'production')
|
2020-04-02 00:27:21 +00:00
|
|
|
viewItem.submenu.unshift({role: 'reload'}, {role: 'forceReload'}, {role: 'toggleDevTools'}, {type: 'separator'});
|
2018-01-06 16:14:21 +00:00
|
|
|
const spellcheckerMenu = new electron.Menu();
|
|
|
|
//tslint:disable-next-line:no-floating-promises
|
|
|
|
addSpellcheckerItems(spellcheckerMenu);
|
|
|
|
const themes = fs.readdirSync(path.join(__dirname, 'themes')).filter((x) => x.substr(-4) === '.css').map((x) => x.slice(0, -4));
|
|
|
|
const setTheme = (theme: string) => {
|
|
|
|
settings.theme = theme;
|
|
|
|
setGeneralSettings(settings);
|
|
|
|
};
|
|
|
|
electron.Menu.setApplicationMenu(electron.Menu.buildFromTemplate([
|
|
|
|
{
|
|
|
|
label: `&${l('title')}`,
|
|
|
|
submenu: [
|
|
|
|
{label: l('action.newWindow'), click: createWindow, accelerator: 'CmdOrCtrl+n'},
|
|
|
|
{
|
|
|
|
label: l('action.newTab'),
|
2020-04-03 00:46:36 +00:00
|
|
|
click: (_m: Electron.MenuItem, w: Electron.BrowserWindow) => {
|
2018-01-06 16:14:21 +00:00
|
|
|
if(tabCount < 3) w.webContents.send('open-tab');
|
|
|
|
},
|
|
|
|
accelerator: 'CmdOrCtrl+t'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: l('settings.logDir'),
|
2020-04-03 00:46:36 +00:00
|
|
|
click: (_m, window: BrowserWindow) => {
|
2020-04-02 00:27:21 +00:00
|
|
|
const dir = electron.dialog.showOpenDialogSync(
|
2019-09-17 17:14:14 +00:00
|
|
|
{defaultPath: settings.logDirectory, properties: ['openDirectory']});
|
2018-01-06 16:14:21 +00:00
|
|
|
if(dir !== undefined) {
|
2018-04-08 00:22:32 +00:00
|
|
|
if(dir[0].startsWith(path.dirname(app.getPath('exe'))))
|
|
|
|
return electron.dialog.showErrorBox(l('settings.logDir'), l('settings.logDir.inAppDir'));
|
2020-04-02 00:27:21 +00:00
|
|
|
const button = electron.dialog.showMessageBoxSync(window, {
|
2018-01-06 16:14:21 +00:00
|
|
|
message: l('settings.logDir.confirm', dir[0], settings.logDirectory),
|
|
|
|
buttons: [l('confirmYes'), l('confirmNo')],
|
|
|
|
cancelId: 1
|
|
|
|
});
|
|
|
|
if(button === 0) {
|
2018-03-28 13:51:05 +00:00
|
|
|
for(const w of windows) w.webContents.send('quit');
|
2018-01-06 16:14:21 +00:00
|
|
|
settings.logDirectory = dir[0];
|
|
|
|
setGeneralSettings(settings);
|
|
|
|
app.quit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: l('settings.closeToTray'), type: 'checkbox', checked: settings.closeToTray,
|
|
|
|
click: (item: Electron.MenuItem) => {
|
|
|
|
settings.closeToTray = item.checked;
|
|
|
|
setGeneralSettings(settings);
|
|
|
|
}
|
|
|
|
}, {
|
|
|
|
label: l('settings.profileViewer'), type: 'checkbox', checked: settings.profileViewer,
|
|
|
|
click: (item: Electron.MenuItem) => {
|
|
|
|
settings.profileViewer = item.checked;
|
|
|
|
setGeneralSettings(settings);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{label: l('settings.spellcheck'), submenu: spellcheckerMenu},
|
|
|
|
{
|
|
|
|
label: l('settings.theme'),
|
|
|
|
submenu: themes.map((x) => ({
|
|
|
|
checked: settings.theme === x,
|
|
|
|
click: () => setTheme(x),
|
|
|
|
label: x,
|
|
|
|
type: <'radio'>'radio'
|
|
|
|
}))
|
2018-09-28 00:08:10 +00:00
|
|
|
}, {
|
|
|
|
label: l('settings.hwAcceleration'), type: 'checkbox', checked: settings.hwAcceleration,
|
|
|
|
click: (item: Electron.MenuItem) => {
|
|
|
|
settings.hwAcceleration = item.checked;
|
|
|
|
setGeneralSettings(settings);
|
|
|
|
}
|
2018-03-04 02:32:26 +00:00
|
|
|
}, {
|
|
|
|
label: l('settings.beta'), type: 'checkbox', checked: settings.beta,
|
2018-03-28 13:51:05 +00:00
|
|
|
click: async(item: Electron.MenuItem) => {
|
2018-03-04 02:32:26 +00:00
|
|
|
settings.beta = item.checked;
|
|
|
|
setGeneralSettings(settings);
|
2018-08-10 16:59:37 +00:00
|
|
|
electron.autoUpdater.setFeedURL({url: updaterUrl + (item.checked ? '?channel=beta' : ''), serverType: 'json'});
|
|
|
|
return electron.autoUpdater.checkForUpdates();
|
2018-03-04 02:32:26 +00:00
|
|
|
}
|
2018-03-28 13:51:05 +00:00
|
|
|
}, {
|
|
|
|
label: l('fixLogs.action'),
|
2020-04-03 00:46:36 +00:00
|
|
|
click: (_m, window: BrowserWindow) => window.webContents.send('fix-logs')
|
2018-01-06 16:14:21 +00:00
|
|
|
},
|
|
|
|
{type: 'separator'},
|
|
|
|
{role: 'minimize'},
|
|
|
|
{
|
|
|
|
accelerator: process.platform === 'darwin' ? 'Cmd+Q' : undefined,
|
|
|
|
label: l('action.quit'),
|
2020-04-03 00:46:36 +00:00
|
|
|
click(_m: Electron.MenuItem, window: Electron.BrowserWindow): void {
|
2018-01-06 16:14:21 +00:00
|
|
|
if(characters.length === 0) return app.quit();
|
2020-04-02 00:27:21 +00:00
|
|
|
const button = electron.dialog.showMessageBoxSync(window, {
|
2018-01-06 16:14:21 +00:00
|
|
|
message: l('chat.confirmLeave'),
|
|
|
|
buttons: [l('confirmYes'), l('confirmNo')],
|
|
|
|
cancelId: 1
|
|
|
|
});
|
2018-04-08 00:22:32 +00:00
|
|
|
if(button === 0) {
|
|
|
|
for(const w of windows) w.webContents.send('quit');
|
|
|
|
app.quit();
|
|
|
|
}
|
2018-01-06 16:14:21 +00:00
|
|
|
}
|
|
|
|
}
|
2020-04-02 00:27:21 +00:00
|
|
|
] as MenuItemConstructorOptions[]
|
2018-01-06 16:14:21 +00:00
|
|
|
}, {
|
|
|
|
label: `&${l('action.edit')}`,
|
|
|
|
submenu: [
|
|
|
|
{role: 'undo'},
|
|
|
|
{role: 'redo'},
|
|
|
|
{type: 'separator'},
|
|
|
|
{role: 'cut'},
|
|
|
|
{role: 'copy'},
|
|
|
|
{role: 'paste'},
|
|
|
|
{role: 'selectall'}
|
2020-04-02 00:27:21 +00:00
|
|
|
] as MenuItemConstructorOptions[]
|
2018-01-06 16:14:21 +00:00
|
|
|
}, viewItem, {
|
|
|
|
label: `&${l('help')}`,
|
|
|
|
submenu: [
|
|
|
|
{
|
|
|
|
label: l('help.fchat'),
|
|
|
|
click: () => electron.shell.openExternal('https://wiki.f-list.net/F-Chat_3.0')
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: l('help.feedback'),
|
|
|
|
click: () => electron.shell.openExternal('https://goo.gl/forms/WnLt3Qm3TPt64jQt2')
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: l('help.rules'),
|
|
|
|
click: () => electron.shell.openExternal('https://wiki.f-list.net/Rules')
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: l('help.faq'),
|
|
|
|
click: () => electron.shell.openExternal('https://wiki.f-list.net/Frequently_Asked_Questions')
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: l('help.report'),
|
|
|
|
click: () => electron.shell.openExternal('https://wiki.f-list.net/How_to_Report_a_User#In_chat')
|
|
|
|
},
|
|
|
|
{label: l('version', app.getVersion()), click: showPatchNotes}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]));
|
2020-04-03 00:46:36 +00:00
|
|
|
electron.ipcMain.on('tab-added', (_event: Event, id: number) => {
|
2018-01-06 16:14:21 +00:00
|
|
|
const webContents = electron.webContents.fromId(id);
|
|
|
|
setUpWebContents(webContents);
|
|
|
|
++tabCount;
|
|
|
|
if(tabCount === 3)
|
|
|
|
for(const w of windows) w.webContents.send('allow-new-tabs', false);
|
|
|
|
});
|
|
|
|
electron.ipcMain.on('tab-closed', () => {
|
|
|
|
--tabCount;
|
|
|
|
for(const w of windows) w.webContents.send('allow-new-tabs', true);
|
|
|
|
});
|
2020-04-03 00:46:36 +00:00
|
|
|
electron.ipcMain.on('save-login', (_event: Event, account: string, host: string) => {
|
2018-01-06 16:14:21 +00:00
|
|
|
settings.account = account;
|
|
|
|
settings.host = host;
|
|
|
|
setGeneralSettings(settings);
|
|
|
|
});
|
|
|
|
electron.ipcMain.on('connect', (e: Event & {sender: Electron.WebContents}, character: string) => {
|
|
|
|
if(characters.indexOf(character) !== -1) return e.returnValue = false;
|
2019-09-17 17:14:14 +00:00
|
|
|
characters.push(character);
|
2018-01-06 16:14:21 +00:00
|
|
|
e.returnValue = true;
|
|
|
|
});
|
2020-04-03 00:46:36 +00:00
|
|
|
electron.ipcMain.on('dictionary-add', (_event: Event, word: string) => {
|
|
|
|
// if(settings.customDictionary.indexOf(word) !== -1) return;
|
|
|
|
// settings.customDictionary.push(word);
|
|
|
|
// setGeneralSettings(settings);
|
|
|
|
for(const w of windows) w.webContents.session.addWordToSpellCheckerDictionary(word);
|
2018-08-10 16:59:37 +00:00
|
|
|
});
|
2020-04-03 00:46:36 +00:00
|
|
|
electron.ipcMain.on('dictionary-remove', (_event: Event /*, word: string*/) => {
|
|
|
|
// settings.customDictionary.splice(settings.customDictionary.indexOf(word), 1);
|
|
|
|
// setGeneralSettings(settings);
|
2018-08-10 16:59:37 +00:00
|
|
|
});
|
2020-04-03 00:46:36 +00:00
|
|
|
electron.ipcMain.on('disconnect', (_event: Event, character: string) => {
|
2019-01-03 17:38:17 +00:00
|
|
|
const index = characters.indexOf(character);
|
|
|
|
if(index !== -1) characters.splice(index, 1);
|
|
|
|
});
|
2018-01-06 16:14:21 +00:00
|
|
|
const emptyBadge = electron.nativeImage.createEmpty();
|
|
|
|
//tslint:disable-next-line:no-require-imports
|
|
|
|
const badge = electron.nativeImage.createFromPath(path.join(__dirname, <string>require('./build/badge.png')));
|
|
|
|
electron.ipcMain.on('has-new', (e: Event & {sender: Electron.WebContents}, hasNew: boolean) => {
|
|
|
|
if(process.platform === 'darwin') app.dock.setBadge(hasNew ? '!' : '');
|
2018-03-28 13:51:05 +00:00
|
|
|
const window = electron.BrowserWindow.fromWebContents(e.sender) as BrowserWindow | undefined;
|
|
|
|
if(window !== undefined) window.setOverlayIcon(hasNew ? badge : emptyBadge, hasNew ? 'New messages' : '');
|
2018-01-06 16:14:21 +00:00
|
|
|
});
|
|
|
|
createWindow();
|
2017-09-02 01:50:31 +00:00
|
|
|
}
|
|
|
|
|
2018-08-10 16:59:37 +00:00
|
|
|
const isSquirrelStart = require('electron-squirrel-startup'); //tslint:disable-line:no-require-imports
|
2018-09-28 00:08:10 +00:00
|
|
|
if(isSquirrelStart || process.env.NODE_ENV === 'production' && !app.requestSingleInstanceLock()) app.quit();
|
2018-01-06 16:14:21 +00:00
|
|
|
else app.on('ready', onReady);
|
2018-09-28 00:08:10 +00:00
|
|
|
app.on('second-instance', createWindow);
|
2020-03-14 19:28:14 +00:00
|
|
|
app.on('window-all-closed', () => app.quit());
|