You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

211 lines
6.5 KiB

8 years ago
/**
* This module executes inside of electron's main process. You can start
* electron renderer process from here and communicate with the other processes
* through IPC.
*
* When running `npm run build` or `npm run build-main`, this file is compiled to
* `./app/main.prod.js` using webpack. This gives us some performance wins.
*/
import { app, BrowserWindow, session } from 'electron'
import installExtension, {
REACT_DEVELOPER_TOOLS,
REDUX_DEVTOOLS
} from 'electron-devtools-installer'
import get from 'lodash.get'
import { mainLog } from './lib/utils/log'
import ZapMenuBuilder from './lib/zap/menuBuilder'
import ZapController from './lib/zap/controller'
import ZapUpdater from './lib/zap/updater'
import themes from './themes'
import { getDbName } from './store/db'
// Set up a couple of timers to track the app startup progress.
mainLog.time('Time until app is ready')
/**
* Fetch user settings from indexedDb.
* We do this by starting up a new browser window and accessing indexedDb from within it.
*
* @return {[type]} 'settings' store from indexedDb.
*/
const fetchSettings = () => {
const win = new BrowserWindow({ show: false, focusable: false })
if (process.env.HOT) {
const port = process.env.PORT || 1212
win.loadURL(`http://localhost:${port}/dist/empty.html`)
} else {
win.loadURL(`file://${__dirname}/dist/empty.html`)
}
// Once we have fetched (or failed to fetch) the user settings, destroy the window.
win.on('load-settings-done', () => process.nextTick(() => win.destroy()))
const dbName = getDbName()
mainLog.debug(`Fetching user settings from indexedDb (using database "%s")`, dbName)
return win.webContents
.executeJavaScript(
`
new Promise((resolve, reject) => {
var DBOpenRequest = window.indexedDB.open('${dbName}')
DBOpenRequest.onupgradeneeded = function(event) {
event.target.transaction.abort()
return reject(new Error('Database does not exist'))
}
DBOpenRequest.onerror = function() {
return reject(new Error('Error loading database'))
}
DBOpenRequest.onsuccess = function() {
const db = DBOpenRequest.result
var transaction = db.transaction(['settings'], 'readwrite')
transaction.onerror = function() {
return reject(transaction.error)
}
var objectStore = transaction.objectStore('settings')
var objectStoreRequest = objectStore.getAll()
objectStoreRequest.onsuccess = function() {
return resolve(objectStoreRequest.result)
}
}
})
`
)
.then(res => {
mainLog.debug('Got user settings: %o', res)
win.emit('load-settings-done')
return res
})
.catch(err => {
win.emit('load-settings-done')
throw err
})
}
const getSetting = (store, key) => {
const setting = store.find(s => s.key === key)
return setting && setting.hasOwnProperty('value') ? setting.value : null
}
8 years ago
/**
* Initialize Zap as soon as electron is ready.
8 years ago
*/
app.on('ready', async () => {
mainLog.timeEnd('Time until app is ready')
8 years ago
// Get the users preference so that we can:
// - set the background colour of the window to avoid unwanted flicker.
// - Initialise the Language menu with the users locale selected by default.
let theme = {}
let locale
if (!process.env.DISABLE_INIT) {
try {
const settings = await fetchSettings()
locale = getSetting(settings, 'locale')
const themeKey = getSetting(settings, 'theme')
theme = themes[themeKey]
} catch (e) {
mainLog.warn('Unable to determine user locale and theme', e)
}
}
// Create a new browser window.
const mainWindow = new BrowserWindow({
8 years ago
show: false,
useContentSize: true,
titleBarStyle: 'hidden',
width: 950,
height: 600,
minWidth: 950,
minHeight: 425,
backgroundColor: get(theme, 'colors.primaryColor', '#242633')
})
8 years ago
// Initialise the updater.
const updater = new ZapUpdater(mainWindow)
updater.init()
// Initialise the application.
const zap = new ZapController(mainWindow)
zap.init()
// Initialise the application menus.
const menuBuilder = new ZapMenuBuilder(mainWindow)
menuBuilder.buildMenu(locale)
/**
* In production mode, enable source map support.
*/
if (process.env.NODE_ENV === 'production') {
const sourceMapSupport = require('source-map-support') // eslint-disable-line global-require
sourceMapSupport.install()
}
8 years ago
/**
* In development mode or when DEBUG_PROD is set, enable debugging tools.
*/
if (process.env.NODE_ENV === 'development' || process.env.DEBUG_PROD === 'true') {
installExtension(REACT_DEVELOPER_TOOLS)
.then(name => mainLog.debug(`Added Extension: ${name}`))
.catch(err => mainLog.warn(`An error occurred when installing REACT_DEVELOPER_TOOLS: ${err}`))
installExtension(REDUX_DEVTOOLS)
.then(name => mainLog.debug(`Added Extension: ${name}`))
.catch(err => mainLog.warn(`An error occurred when installing REDUX_DEVTOOLS: ${err}`))
zap.mainWindow.webContents.once('dom-ready', () => {
zap.mainWindow.openDevTools()
})
}
/**
* Add application event listener:
* - Open zap payment form when lightning url is opened
*/
app.setAsDefaultProtocolClient('lightning')
app.on('open-url', (event, url) => {
mainLog.debug('open-url')
event.preventDefault()
const payReq = url.split(':')[1]
zap.sendMessage('lightningPaymentUri', { payReq })
zap.mainWindow.show()
})
// HACK: patch webrequest to fix devtools incompatibility with electron 2.x.
// See https://github.com/electron/electron/issues/13008#issuecomment-400261941
session.defaultSession.webRequest.onBeforeRequest({}, (details, callback) => {
if (details.url.indexOf('7accc8730b0f99b5e7c0702ea89d1fa7c17bfe33') !== -1) {
callback({
redirectURL: details.url.replace(
'7accc8730b0f99b5e7c0702ea89d1fa7c17bfe33',
'57c9d07b416b5a2ea23d28247300e4af36329bdc'
)
})
} else {
callback({ cancel: false })
}
})
/**
* Add application event listener:
* - Stop gRPC and kill lnd process before the app windows are closed and the app quits.
*/
app.on('before-quit', async event => {
if (!zap.is('terminated')) {
event.preventDefault()
zap.terminate()
} else {
if (zap.mainWindow) {
zap.mainWindow.forceClose = true
}
}
})
/**
* On OS X it's common to re-open a window in the app when the dock icon is clicked.
*/
app.on('activate', () => {
zap.mainWindow.show()
})
})