Whalebird-desktop-client-ma.../src/main/index.js

487 lines
12 KiB
JavaScript
Raw Normal View History

2018-03-07 14:28:48 +01:00
'use strict'
import { app, ipcMain, BrowserWindow, shell, Menu } from 'electron'
2018-03-08 15:08:33 +01:00
import Datastore from 'nedb'
import empty from 'is-empty'
import log from 'electron-log'
2018-03-24 15:23:25 +01:00
import windowStateKeeper from 'electron-window-state'
import simplayer from 'simplayer'
import path from 'path'
2018-03-08 15:08:33 +01:00
2018-03-08 09:41:39 +01:00
import Authentication from './auth'
import Account from './account'
import Streaming from './streaming'
2018-03-07 14:28:48 +01:00
/**
* Set log level
*/
log.transports.console.level = 'debug'
log.transports.file.level = 'info'
2018-03-07 14:28:48 +01:00
/**
* Set `__static` path to static files in production
* https://simulatedgreg.gitbooks.io/electron-vue/content/en/using-static-assets.html
*/
if (process.env.NODE_ENV !== 'development') {
global.__static = path.join(__dirname, '/static').replace(/\\/g, '\\\\')
2018-03-07 14:28:48 +01:00
}
let mainWindow
const winURL = process.env.NODE_ENV === 'development'
? `http://localhost:9080`
: `file://${__dirname}/index.html`
2018-03-22 08:55:58 +01:00
// https://github.com/louischatriot/nedb/issues/459
2018-03-22 08:49:39 +01:00
const userData = app.getPath('userData')
const databasePath = process.env.NODE_ENV === 'production'
? userData + '/db/account.db'
: 'account.db'
2018-03-08 15:08:33 +01:00
let db = new Datastore({
2018-03-22 08:49:39 +01:00
filename: databasePath,
2018-03-08 15:08:33 +01:00
autoload: true
})
2018-03-21 04:22:45 +01:00
async function listAccounts () {
try {
const account = new Account(db)
await account.cleanup()
2018-03-21 04:22:45 +01:00
const accounts = await account.listAccounts()
return accounts
} catch (err) {
return []
}
}
2018-03-07 14:28:48 +01:00
function createWindow () {
/**
2018-03-21 04:22:45 +01:00
* List accounts
*/
2018-03-21 04:22:45 +01:00
listAccounts()
.then((accounts) => {
const accountsChange = accounts.map((a, index) => {
return {
label: a.domain,
accelerator: `CmdOrCtrl+${index + 1}`,
click: () => {
2018-03-21 04:22:45 +01:00
mainWindow.webContents.send('change-account', Object.assign(a, { index: index }))
}
}
2018-03-21 04:22:45 +01:00
})
/**
* Set menu
*/
const template = [
2018-03-13 08:08:17 +01:00
{
2018-03-21 04:22:45 +01:00
label: 'Whalebird',
submenu: [
{
label: 'About Whalebird',
role: 'about'
},
{
type: 'separator'
},
{
label: 'Preferences...',
accelerator: 'CmdOrCtrl+,',
click: () => {
mainWindow.webContents.send('open-preferences')
}
},
{
type: 'separator'
},
2018-03-21 04:22:45 +01:00
{
label: 'Quit',
accelerator: 'CmdOrCtrl+Q',
role: 'quit'
}
]
2018-03-13 08:08:17 +01:00
},
{
2018-03-21 04:22:45 +01:00
label: 'Toot',
submenu: [
{
label: 'New Toot',
accelerator: 'CmdOrCtrl+N',
click: () => {
mainWindow.webContents.send('CmdOrCtrl+N')
}
}
]
2018-03-13 08:08:17 +01:00
},
{
2018-03-21 04:22:45 +01:00
label: 'Edit',
submenu: [
{
label: 'Undo',
accelerator: 'CmdOrCtrl+Z',
role: 'undo'
},
{
label: 'Redo',
accelerator: 'Shift+CmdOrCtrl+Z',
role: 'redo'
},
{
type: 'separator'
},
{
label: 'Cut',
accelerator: 'CmdOrCtrl+X',
role: 'cut'
},
{
label: 'Copy',
accelerator: 'CmdOrCtrl+C',
role: 'copy'
},
{
label: 'Paste',
accelerator: 'CmdOrCtrl+V',
role: 'paste'
},
{
label: 'Select All',
accelerator: 'CmdOrCtrl+A',
role: 'selectall'
}
]
2018-03-13 08:08:17 +01:00
},
{
2018-03-21 04:22:45 +01:00
label: 'Window',
submenu: [
{
label: 'Close Window',
role: 'close'
},
{
label: 'Minimize',
role: 'minimize'
},
{
type: 'separator'
}
].concat(accountsChange)
.concat([
{
type: 'separator'
},
{
label: 'Jump to',
accelerator: 'CmdOrCtrl+K',
click: () => {
mainWindow.webContents.send('CmdOrCtrl+K')
}
}
])
2018-03-13 08:08:17 +01:00
}
]
2018-03-21 04:22:45 +01:00
const menu = Menu.buildFromTemplate(template)
Menu.setApplicationMenu(menu)
2018-03-21 04:22:45 +01:00
/**
* Initial window options
*/
2018-03-24 15:23:25 +01:00
let mainWindowState = windowStateKeeper({
defaultWidth: 1000,
height: 563
})
2018-03-21 04:22:45 +01:00
mainWindow = new BrowserWindow({
2018-03-22 09:35:54 +01:00
titleBarStyle: 'hidden',
2018-03-24 15:23:25 +01:00
x: mainWindowState.x,
y: mainWindowState.y,
width: mainWindowState.width,
height: mainWindowState.height,
2018-03-21 04:22:45 +01:00
useContentSize: true,
icon: path.join(__dirname, '../../build/icons/256x256.png')
2018-03-21 04:22:45 +01:00
})
2018-03-24 15:23:25 +01:00
mainWindowState.manage(mainWindow)
2018-03-07 14:28:48 +01:00
2018-03-21 04:22:45 +01:00
mainWindow.loadURL(winURL)
2018-03-07 14:28:48 +01:00
2018-03-21 04:22:45 +01:00
mainWindow.on('closed', () => {
mainWindow = null
})
})
2018-03-07 14:28:48 +01:00
}
app.on('ready', createWindow)
app.on('window-all-closed', () => {
if (process.platform !== 'darwin') {
app.quit()
}
})
app.on('activate', () => {
if (mainWindow === null) {
createWindow()
}
})
let auth = new Authentication(new Account(db))
2018-03-08 09:41:39 +01:00
ipcMain.on('get-auth-url', (event, domain) => {
2018-03-10 16:48:40 +01:00
auth.getAuthorizationUrl(domain)
2018-03-08 09:41:39 +01:00
.then((url) => {
log.debug(url)
event.sender.send('response-get-auth-url', url)
2018-03-10 15:28:55 +01:00
// Open authorize url in default browser.
2018-03-08 09:41:39 +01:00
shell.openExternal(url)
})
.catch((err) => {
log.error(err)
event.sender.send('error-get-auth-url', err)
})
2018-03-08 09:41:39 +01:00
})
ipcMain.on('get-access-token', (event, code) => {
auth.getAccessToken(code)
.catch((err) => {
log.error(err)
event.sender.send('error-get-access-token', err)
})
.then((token) => {
db.findOne({
accessToken: token
}, (err, doc) => {
if (err) return event.sender.send('error-get-access-token', err)
if (empty(doc)) return event.sender.send('error-get-access-token', 'error document is empty')
event.sender.send('response-get-access-token', doc._id)
})
})
2018-03-08 09:41:39 +01:00
})
// environments
ipcMain.on('get-social-token', (event, _) => {
const token = process.env.SOCIAL_TOKEN
if (empty(token)) {
return event.sender.send('error-get-social-token', new EmptyTokenError())
}
event.sender.send('response-get-social-token', token)
})
// nedb
ipcMain.on('list-accounts', (event, _) => {
const account = new Account(db)
account.listAccounts()
2018-03-08 15:08:33 +01:00
.catch((err) => {
log.error(err)
event.sender.send('error-list-accounts', err)
2018-03-08 15:08:33 +01:00
})
.then((accounts) => {
event.sender.send('response-list-accounts', accounts)
2018-03-08 15:08:33 +01:00
})
})
2018-03-09 09:36:57 +01:00
ipcMain.on('get-local-account', (event, id) => {
const account = new Account(db)
account.getAccount(id)
.catch((err) => {
log.error(err)
2018-03-09 09:36:57 +01:00
event.sender.send('error-get-local-account', err)
})
.then((account) => {
event.sender.send('response-get-local-account', account)
2018-03-09 09:36:57 +01:00
})
})
ipcMain.on('update-account', (event, acct) => {
const account = new Account(db)
const id = acct._id
delete acct._id
account.updateAccount(id, acct)
.then((ac) => {
event.sender.send('response-update-account', ac)
})
.catch((err) => {
event.sender.send('error-update-account', err)
})
})
ipcMain.on('remove-account', (event, id) => {
const account = new Account(db)
account.removeAccount(id)
.then(() => {
event.sender.send('response-remove-account')
})
.catch((err) => {
event.sender.send('error-remove-account', err)
})
})
2018-04-02 02:07:09 +02:00
ipcMain.on('forward-account', (event, acct) => {
const account = new Account(db)
account.forwardAccount(acct)
.then(() => {
event.sender.send('response-forward-account')
})
.catch((err) => {
event.sender.send('error-forward-account', err)
})
})
ipcMain.on('backward-account', (event, acct) => {
const account = new Account(db)
account.backwardAccount(acct)
.then(() => {
event.sender.send('response-backward-account')
})
.catch((err) => {
event.sender.send('error-backward-account', err)
})
})
// streaming
let userStreaming = null
ipcMain.on('start-user-streaming', (event, ac) => {
const account = new Account(db)
account.getAccount(ac._id)
.catch((err) => {
log.error(err)
event.sender.send('error-start-user-streaming', err)
})
.then((account) => {
// Stop old user streaming
if (userStreaming !== null) {
userStreaming.stop()
userStreaming = null
}
userStreaming = new Streaming(account)
userStreaming.startUserStreaming(
(update) => {
event.sender.send('update-start-user-streaming', update)
},
(notification) => {
event.sender.send('notification-start-user-streaming', notification)
},
(err) => {
log.error(err)
event.sender.send('error-start-user-streaming', err)
}
)
})
})
ipcMain.on('stop-user-streaming', (event, _) => {
if (userStreaming !== null) {
userStreaming.stop()
userStreaming = null
}
})
let localStreaming = null
ipcMain.on('start-local-streaming', (event, ac) => {
const account = new Account(db)
account.getAccount(ac._id)
.catch((err) => {
log.error(err)
event.sender.send('error-start-local-streaming', err)
})
.then((account) => {
// Stop old local streaming
if (localStreaming !== null) {
localStreaming.stop()
localStreaming = null
}
localStreaming = new Streaming(account)
localStreaming.start(
'/streaming/public/local',
(update) => {
event.sender.send('update-start-local-streaming', update)
},
(err) => {
log.error(err)
event.sender.send('error-start-local-streaming', err)
}
)
})
})
ipcMain.on('stop-local-streaming', (event, _) => {
localStreaming.stop()
localStreaming = null
})
let publicStreaming = null
ipcMain.on('start-public-streaming', (event, ac) => {
const account = new Account(db)
account.getAccount(ac._id)
.catch((err) => {
log.error(err)
event.sender.send('error-start-public-streaming', err)
})
.then((account) => {
// Stop old public streaming
if (publicStreaming !== null) {
publicStreaming.stop()
publicStreaming = null
}
publicStreaming = new Streaming(account)
publicStreaming.start(
'/streaming/public',
(update) => {
event.sender.send('update-start-public-streaming', update)
},
(err) => {
log.error(err)
event.sender.send('error-start-public-streaming', err)
}
)
})
})
ipcMain.on('stop-public-streaming', (event, _) => {
publicStreaming.stop()
publicStreaming = null
})
// sounds
ipcMain.on('operation-sound01', (event, _) => {
const sound = process.env.NODE_ENV === 'development'
? path.join(__dirname, '../../build/sounds/operation_sound01.wav')
: path.join(process.resourcesPath, 'build/sounds/operation_sound01.wav')
simplayer(sound, (err) => {
if (err) log.error(err)
})
})
ipcMain.on('operation-sound02', (event, _) => {
const sound = process.env.NODE_ENV === 'development'
? path.join(__dirname, '../../build/sounds/operation_sound02.wav')
: path.join(process.resourcesPath, 'build/sounds/operation_sound02.wav')
simplayer(sound, (err) => {
if (err) log.error(err)
})
})
2018-03-07 14:28:48 +01:00
/**
* Auto Updater
*
* Uncomment the following code below and install `electron-updater` to
* support auto updating. Code Signing with a valid certificate is required.
* https://simulatedgreg.gitbooks.io/electron-vue/content/en/using-electron-builder.html#auto-updating
*/
/*
import { autoUpdater } from 'electron-updater'
autoUpdater.on('update-downloaded', () => {
autoUpdater.quitAndInstall()
})
app.on('ready', () => {
if (process.env.NODE_ENV === 'production') autoUpdater.checkForUpdates()
})
*/
class EmptyTokenError {}