1
0
mirror of https://github.com/rhysd/Mstdn.git synced 2025-01-22 21:52:09 +01:00
Mstdn/main/app.ts

199 lines
6.6 KiB
TypeScript
Raw Normal View History

2017-04-15 16:32:44 +09:00
import * as path from 'path';
import {app, BrowserWindow, globalShortcut, Tray, shell, dialog} from 'electron';
2017-04-15 16:32:44 +09:00
import windowState = require('electron-window-state');
import * as menubar from 'menubar';
import log from './log';
2017-04-16 12:44:53 +09:00
import {Config, Account} from './config';
2017-04-15 16:32:44 +09:00
const IS_DEBUG = process.env.NODE_ENV === 'development';
const IS_DARWIN = process.platform === 'darwin';
2017-04-16 22:07:13 +09:00
const APP_ICON = path.join(__dirname, '..', 'resources', 'icon', 'icon.png');
2017-04-16 02:27:06 +09:00
const PRELOAD_JS = path.join(__dirname, '..', 'renderer', 'preload.js');
2017-04-15 16:32:44 +09:00
export class App {
2017-04-15 17:05:32 +09:00
private account: Account;
2017-04-15 16:32:44 +09:00
constructor(private win: Electron.BrowserWindow, private config: Config) {
if (config.accounts.length === 0) {
throw new Error('No account found. Please check the config.');
}
2017-04-15 17:05:32 +09:00
this.account = this.config.accounts[0];
2017-04-15 16:32:44 +09:00
}
open() {
this.win.loadURL(`https://${this.account.host}${this.account.default_page}`);
2017-04-16 02:57:54 +09:00
this.win.webContents.on('will-navigate', (e, url) => {
if (!url.startsWith(`https://${this.account.host}`)) {
e.preventDefault();
shell.openExternal(url);
}
});
this.win.webContents.on('new-window', (e, url) => {
e.preventDefault();
shell.openExternal(url);
});
this.win.webContents.session.setPermissionRequestHandler((contents, permission, callback) => {
if (permission !== 'geolocation' && permission !== 'media') {
// Granted
callback(true);
return;
}
dialog.showMessageBox({
type: 'question',
buttons: ['Accept', 'Reject'],
message: `Permission '${permission}' is requested by ${contents.getURL()}`,
detail: "Please choose one of 'Accept' or 'Reject'",
}, (buttonIndex: number) => {
const granted = buttonIndex === 0;
callback(granted);
});
});
2017-04-15 16:32:44 +09:00
this.win.show();
}
}
function trayIcon(color: string) {
2017-04-15 17:05:32 +09:00
return path.join(__dirname, '..', 'resources', 'icon', `tray-icon-${
2017-04-15 16:32:44 +09:00
color === 'white' ? 'white' : 'black'
}@2x.png`);
}
2017-04-16 12:44:53 +09:00
export default function startApp(config: Config) {
2017-04-15 16:32:44 +09:00
return (config.normal_window ? startNormalWindow : startMenuBar)(config)
.then(win => new App(win, config));
}
function startNormalWindow(config: Config): Promise<Electron.BrowserWindow> {
log.debug('Setup a normal window');
return new Promise<Electron.BrowserWindow>(resolve => {
const state = windowState({
defaultWidth: 600,
defaultHeight: 800,
});
const win = new BrowserWindow({
width: state.width,
height: state.height,
x: state.x,
y: state.y,
icon: APP_ICON,
show: false,
useContentSize: true,
autoHideMenuBar: true,
webPreferences: {
nodeIntegration: false,
sandbox: true,
2017-04-16 02:27:06 +09:00
preload: PRELOAD_JS,
2017-04-15 16:32:44 +09:00
},
});
win.once('ready-to-show', () => {
win.show();
});
win.once('closed', () => {
app.quit();
});
if (state.isFullScreen) {
win.setFullScreen(true);
} else if (state.isMaximized) {
win.maximize();
}
state.manage(win);
const toggleWindow = () => {
if (win.isFocused()) {
log.debug('Toggle window: shown -> hidden');
if (IS_DARWIN) {
app.hide();
} else {
win.hide();
}
} else {
log.debug('Toggle window: hidden -> shown');
win.show();
}
};
win.webContents.on('dom-ready', () => {
log.debug('Send config to renderer procress');
win.webContents.send('mstdn:config', config);
});
win.webContents.once('dom-ready', () => {
log.debug('Normal window application was launched');
if (config.hot_key) {
globalShortcut.register(config.hot_key, toggleWindow);
log.debug('Hot key was set to:', config.hot_key);
}
if (IS_DEBUG) {
win.webContents.openDevTools({mode: 'detach'});
}
});
const normalIcon = trayIcon(config.icon_color);
const tray = new Tray(normalIcon);
tray.on('click', toggleWindow);
tray.on('double-click', toggleWindow);
if (IS_DARWIN) {
tray.setHighlightMode('never');
2017-04-15 17:05:32 +09:00
app.dock.setIcon(APP_ICON);
2017-04-15 16:32:44 +09:00
}
resolve(win);
2017-04-15 16:32:44 +09:00
});
}
function startMenuBar(config: Config): Promise<Electron.BrowserWindow> {
log.debug('Setup a menubar window');
return new Promise<Electron.BrowserWindow>(resolve => {
const state = windowState({
2017-04-17 12:15:59 +09:00
defaultWidth: 350,
defaultHeight: 420,
2017-04-15 16:32:44 +09:00
});
const icon = trayIcon(config.icon_color);
const mb = menubar({
icon,
width: state.width,
height: state.height,
alwaysOnTop: IS_DEBUG || !!config.always_on_top,
tooltip: 'Mstdn',
autoHideMenuBar: true,
useContentSize: true,
show: false,
webPreferences: {
nodeIntegration: false,
sandbox: true,
2017-04-16 02:27:06 +09:00
preload: PRELOAD_JS,
2017-04-15 16:32:44 +09:00
},
});
mb.once('ready', () => mb.showWindow());
mb.once('after-create-window', () => {
log.debug('Menubar application was launched');
if (config.hot_key) {
globalShortcut.register(config.hot_key, () => {
if (mb.window.isFocused()) {
log.debug('Toggle window: shown -> hidden');
mb.hideWindow();
} else {
log.debug('Toggle window: hidden -> shown');
mb.showWindow();
}
});
log.debug('Hot key was set to:', config.hot_key);
}
if (IS_DEBUG) {
mb.window.webContents.openDevTools({mode: 'detach'});
}
mb.window.webContents.on('dom-ready', () => {
log.debug('Send config to renderer procress');
mb.window.webContents.send('mstdn:config', config);
});
state.manage(mb.window);
resolve(mb.window);
});
mb.once('after-close', () => {
app.quit();
});
});
}