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.
200 lines
6.2 KiB
200 lines
6.2 KiB
/**
|
|
* 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 })
|
|
if (process.env.HOT) {
|
|
const port = process.env.PORT || 1212
|
|
win.loadURL(`http://localhost:${port}`)
|
|
} else {
|
|
win.loadURL(`file://${__dirname}`)
|
|
}
|
|
|
|
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)
|
|
return res
|
|
})
|
|
}
|
|
|
|
const getSetting = (store, key) => {
|
|
const setting = store.find(s => s.key === key)
|
|
return setting && setting.hasOwnProperty('value') ? setting.value : null
|
|
}
|
|
|
|
/**
|
|
* Initialize Zap as soon as electron is ready.
|
|
*/
|
|
app.on('ready', async () => {
|
|
mainLog.timeEnd('Time until app is ready')
|
|
|
|
// 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
|
|
|
|
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({
|
|
show: false,
|
|
useContentSize: true,
|
|
titleBarStyle: 'hidden',
|
|
width: 950,
|
|
height: 600,
|
|
minWidth: 950,
|
|
minHeight: 425,
|
|
backgroundColor: get(theme, 'colors.darkestBackground', '#242633')
|
|
})
|
|
|
|
// 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()
|
|
}
|
|
|
|
/**
|
|
* 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()
|
|
})
|
|
})
|
|
|