2023-01-20 15:33:26 +01:00
|
|
|
|
import { app, BrowserWindow, shell, ipcMain, Menu, nativeTheme, globalShortcut } from 'electron'
|
2023-01-12 18:55:55 +01:00
|
|
|
|
import { release } from 'node:os'
|
|
|
|
|
import { join } from 'node:path'
|
2023-01-19 00:20:50 +01:00
|
|
|
|
import * as jetpack from "fs-jetpack";
|
2023-03-06 01:30:37 +01:00
|
|
|
|
|
2023-01-15 12:45:06 +01:00
|
|
|
|
import menu from './menu'
|
2023-01-19 00:20:50 +01:00
|
|
|
|
import { initialContent, initialDevContent } from '../initial-content'
|
2023-01-27 17:18:31 +01:00
|
|
|
|
import { WINDOW_CLOSE_EVENT, SETTINGS_CHANGE_EVENT } from '../constants';
|
2023-01-20 15:33:26 +01:00
|
|
|
|
import CONFIG from "../config"
|
|
|
|
|
import { onBeforeInputEvent } from "../keymap"
|
2023-12-23 11:17:13 +01:00
|
|
|
|
import { isDev } from '../detect-platform';
|
2023-12-06 13:31:47 +01:00
|
|
|
|
import { initializeAutoUpdate, checkForUpdates } from './auto-update';
|
2023-03-20 11:09:56 +01:00
|
|
|
|
import { fixElectronCors } from './cors';
|
2023-12-23 11:17:13 +01:00
|
|
|
|
import { getBufferFilePath } from './buffer';
|
2023-03-06 01:30:37 +01:00
|
|
|
|
|
2023-01-12 18:55:55 +01:00
|
|
|
|
|
|
|
|
|
// The built directory structure
|
|
|
|
|
//
|
|
|
|
|
// ├─┬ dist-electron
|
|
|
|
|
// │ ├─┬ main
|
|
|
|
|
// │ │ └── index.js > Electron-Main
|
|
|
|
|
// │ └─┬ preload
|
|
|
|
|
// │ └── index.js > Preload-Scripts
|
|
|
|
|
// ├─┬ dist
|
|
|
|
|
// │ └── index.html > Electron-Renderer
|
|
|
|
|
//
|
|
|
|
|
process.env.DIST_ELECTRON = join(__dirname, '..')
|
|
|
|
|
process.env.DIST = join(process.env.DIST_ELECTRON, '../dist')
|
|
|
|
|
process.env.PUBLIC = process.env.VITE_DEV_SERVER_URL
|
2023-01-15 10:53:09 +01:00
|
|
|
|
? join(process.env.DIST_ELECTRON, '../public')
|
|
|
|
|
: process.env.DIST
|
2023-01-12 18:55:55 +01:00
|
|
|
|
|
|
|
|
|
// Disable GPU Acceleration for Windows 7
|
|
|
|
|
if (release().startsWith('6.1')) app.disableHardwareAcceleration()
|
|
|
|
|
|
|
|
|
|
// Set application name for Windows 10+ notifications
|
|
|
|
|
if (process.platform === 'win32') app.setAppUserModelId(app.getName())
|
|
|
|
|
|
2023-01-13 01:57:50 +01:00
|
|
|
|
if (!process.env.VITE_DEV_SERVER_URL && !app.requestSingleInstanceLock()) {
|
2023-01-15 10:53:09 +01:00
|
|
|
|
app.quit()
|
|
|
|
|
process.exit(0)
|
2023-01-12 18:55:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2023-01-15 12:45:06 +01:00
|
|
|
|
// Set custom application menu
|
|
|
|
|
Menu.setApplicationMenu(menu)
|
2023-01-13 01:57:50 +01:00
|
|
|
|
|
2023-01-19 02:38:19 +01:00
|
|
|
|
|
2023-01-12 18:55:55 +01:00
|
|
|
|
// Remove electron security warnings
|
|
|
|
|
// This warning only shows in development mode
|
|
|
|
|
// Read more on https://www.electronjs.org/docs/latest/tutorial/security
|
|
|
|
|
// process.env['ELECTRON_DISABLE_SECURITY_WARNINGS'] = 'true'
|
|
|
|
|
|
2023-07-08 16:47:25 +02:00
|
|
|
|
export let win: BrowserWindow | null = null
|
2023-01-12 18:55:55 +01:00
|
|
|
|
// Here, you can also use other preload
|
|
|
|
|
const preload = join(__dirname, '../preload/index.js')
|
|
|
|
|
const url = process.env.VITE_DEV_SERVER_URL
|
|
|
|
|
const indexHtml = join(process.env.DIST, 'index.html')
|
2023-01-20 15:33:26 +01:00
|
|
|
|
|
2023-01-29 12:49:54 +01:00
|
|
|
|
let currentKeymap = CONFIG.get("settings.keymap")
|
2023-01-19 00:20:50 +01:00
|
|
|
|
let contentSaved = false
|
|
|
|
|
|
2023-12-06 13:31:47 +01:00
|
|
|
|
// if this version is a beta version, set the release channel to beta
|
|
|
|
|
const isBetaVersion = app.getVersion().includes("beta")
|
|
|
|
|
if (isBetaVersion) {
|
2023-12-07 00:42:29 +01:00
|
|
|
|
CONFIG.set("settings.allowBetaVersions", true)
|
2023-12-06 13:31:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
2023-01-12 18:55:55 +01:00
|
|
|
|
|
|
|
|
|
async function createWindow() {
|
2023-01-19 11:45:40 +01:00
|
|
|
|
// read any stored window settings from config, or use defaults
|
2023-01-19 02:38:19 +01:00
|
|
|
|
let windowConfig = {
|
2023-01-19 11:45:40 +01:00
|
|
|
|
width: CONFIG.get("windowConfig.width", 900) as number,
|
|
|
|
|
height: CONFIG.get("windowConfig.height", 680) as number,
|
|
|
|
|
isMaximized: CONFIG.get("windowConfig.isMaximized", false) as boolean,
|
|
|
|
|
isFullScreen: CONFIG.get("windowConfig.isFullScreen", false) as boolean,
|
2023-01-19 02:38:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
win = new BrowserWindow(Object.assign({
|
|
|
|
|
title: 'heynote',
|
2023-01-16 16:07:39 +01:00
|
|
|
|
icon: join(process.env.PUBLIC, 'favicon.ico'),
|
2023-01-18 21:36:31 +01:00
|
|
|
|
backgroundColor: nativeTheme.shouldUseDarkColors ? '#262B37' : '#FFFFFF',
|
2023-01-15 10:53:09 +01:00
|
|
|
|
//titleBarStyle: 'customButtonsOnHover',
|
2023-01-15 20:00:10 +01:00
|
|
|
|
autoHideMenuBar: true,
|
2023-01-15 10:53:09 +01:00
|
|
|
|
webPreferences: {
|
|
|
|
|
preload,
|
|
|
|
|
// Warning: Enable nodeIntegration and disable contextIsolation is not secure in production
|
|
|
|
|
// Consider using contextBridge.exposeInMainWorld
|
|
|
|
|
// Read more on https://www.electronjs.org/docs/latest/tutorial/context-isolation
|
|
|
|
|
nodeIntegration: true,
|
2023-01-15 11:59:17 +01:00
|
|
|
|
contextIsolation: true,
|
2023-01-15 10:53:09 +01:00
|
|
|
|
},
|
2023-01-19 02:38:19 +01:00
|
|
|
|
|
|
|
|
|
}, windowConfig))
|
2023-01-19 11:45:40 +01:00
|
|
|
|
|
2023-01-19 02:47:57 +01:00
|
|
|
|
// maximize window if it was maximized last time
|
2023-01-19 02:38:19 +01:00
|
|
|
|
if (windowConfig.isMaximized) {
|
|
|
|
|
win.maximize()
|
|
|
|
|
}
|
2023-01-19 11:45:40 +01:00
|
|
|
|
if (windowConfig.isFullScreen) {
|
|
|
|
|
win.setFullScreen(true)
|
|
|
|
|
}
|
2023-01-15 10:53:09 +01:00
|
|
|
|
|
2023-01-19 00:20:50 +01:00
|
|
|
|
win.on("close", (event) => {
|
|
|
|
|
// Prevent the window from closing, and send a message to the renderer which will in turn
|
|
|
|
|
// send a message to the main process to save the current buffer and close the window.
|
|
|
|
|
if (!contentSaved) {
|
|
|
|
|
event.preventDefault()
|
2023-01-19 02:38:19 +01:00
|
|
|
|
win?.webContents.send(WINDOW_CLOSE_EVENT)
|
|
|
|
|
} else {
|
|
|
|
|
// save window config
|
|
|
|
|
Object.assign(windowConfig, {
|
2023-01-19 11:45:40 +01:00
|
|
|
|
isMaximized: win.isMaximized(),
|
|
|
|
|
isFullScreen: win.isFullScreen(),
|
2023-01-19 02:38:19 +01:00
|
|
|
|
}, win.getNormalBounds())
|
|
|
|
|
CONFIG.set("windowConfig", windowConfig)
|
2023-01-19 00:20:50 +01:00
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
2023-03-09 09:46:21 +01:00
|
|
|
|
nativeTheme.themeSource = CONFIG.get("theme")
|
2023-01-15 10:53:09 +01:00
|
|
|
|
|
|
|
|
|
if (process.env.VITE_DEV_SERVER_URL) { // electron-vite-vue#298
|
2023-01-15 20:24:31 +01:00
|
|
|
|
win.loadURL(url + '?dev=1')
|
2023-01-15 10:53:09 +01:00
|
|
|
|
// Open devTool if the app is not packaged
|
|
|
|
|
//win.webContents.openDevTools()
|
|
|
|
|
} else {
|
|
|
|
|
win.loadFile(indexHtml)
|
|
|
|
|
//win.webContents.openDevTools()
|
|
|
|
|
}
|
2023-01-20 15:33:26 +01:00
|
|
|
|
|
|
|
|
|
// custom keyboard shortcuts for Emacs keybindings
|
|
|
|
|
win.webContents.on("before-input-event", function (event, input) {
|
|
|
|
|
onBeforeInputEvent({event, input, win, currentKeymap})
|
|
|
|
|
})
|
2023-01-15 10:53:09 +01:00
|
|
|
|
|
|
|
|
|
// Test actively push message to the Electron-Renderer
|
|
|
|
|
win.webContents.on('did-finish-load', () => {
|
|
|
|
|
win?.webContents.send('main-process-message', new Date().toLocaleString())
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Make all links open with the browser, not with the application
|
|
|
|
|
win.webContents.setWindowOpenHandler(({ url }) => {
|
2023-03-08 13:07:58 +01:00
|
|
|
|
if (url.startsWith('https:') || url.startsWith('http:')) {
|
|
|
|
|
shell.openExternal(url)
|
|
|
|
|
}
|
2023-01-15 10:53:09 +01:00
|
|
|
|
return { action: 'deny' }
|
|
|
|
|
})
|
2023-03-20 11:09:56 +01:00
|
|
|
|
|
|
|
|
|
fixElectronCors(win)
|
2023-01-12 18:55:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-10 22:59:42 +01:00
|
|
|
|
function registerGlobalHotkey() {
|
|
|
|
|
globalShortcut.unregisterAll()
|
|
|
|
|
if (CONFIG.get("settings.enableGlobalHotkey")) {
|
|
|
|
|
try {
|
|
|
|
|
const ret = globalShortcut.register(CONFIG.get("settings.globalHotkey"), () => {
|
2023-12-10 23:44:58 +01:00
|
|
|
|
if (!win) {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if (win.isFocused()) {
|
2023-12-11 00:31:40 +01:00
|
|
|
|
if (!!app.hide) {
|
|
|
|
|
// app.hide() only available on macOS
|
|
|
|
|
app?.hide()
|
|
|
|
|
} else {
|
|
|
|
|
win.blur()
|
|
|
|
|
}
|
2023-12-10 23:44:58 +01:00
|
|
|
|
} else {
|
|
|
|
|
app.focus({steal: true})
|
|
|
|
|
if (win.isMinimized()) {
|
|
|
|
|
win.restore()
|
|
|
|
|
}
|
2023-12-10 23:32:17 +01:00
|
|
|
|
}
|
2023-12-10 22:59:42 +01:00
|
|
|
|
})
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.log("Could not register global hotkey:", error)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-06 01:30:37 +01:00
|
|
|
|
app.whenReady().then(createWindow).then(async () => {
|
2023-03-07 01:45:23 +01:00
|
|
|
|
initializeAutoUpdate(win)
|
2023-12-10 22:59:42 +01:00
|
|
|
|
registerGlobalHotkey()
|
2023-03-06 01:30:37 +01:00
|
|
|
|
})
|
2023-01-12 18:55:55 +01:00
|
|
|
|
|
|
|
|
|
app.on('window-all-closed', () => {
|
2023-01-15 10:53:09 +01:00
|
|
|
|
win = null
|
|
|
|
|
if (process.platform !== 'darwin') app.quit()
|
2023-01-12 18:55:55 +01:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
app.on('second-instance', () => {
|
2023-01-15 10:53:09 +01:00
|
|
|
|
if (win) {
|
|
|
|
|
// Focus on the main window if the user tried to open another
|
|
|
|
|
if (win.isMinimized()) win.restore()
|
|
|
|
|
win.focus()
|
|
|
|
|
}
|
2023-01-12 18:55:55 +01:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
app.on('activate', () => {
|
2023-01-15 10:53:09 +01:00
|
|
|
|
const allWindows = BrowserWindow.getAllWindows()
|
|
|
|
|
if (allWindows.length) {
|
|
|
|
|
allWindows[0].focus()
|
|
|
|
|
} else {
|
|
|
|
|
createWindow()
|
|
|
|
|
}
|
2023-01-12 18:55:55 +01:00
|
|
|
|
})
|
|
|
|
|
|
2023-01-15 11:59:17 +01:00
|
|
|
|
ipcMain.handle('dark-mode:set', (event, mode) => {
|
2023-03-09 09:46:21 +01:00
|
|
|
|
CONFIG.set("theme", mode)
|
2023-01-15 11:59:17 +01:00
|
|
|
|
nativeTheme.themeSource = mode
|
2023-01-12 18:55:55 +01:00
|
|
|
|
})
|
2023-01-15 11:59:17 +01:00
|
|
|
|
|
|
|
|
|
ipcMain.handle('dark-mode:get', () => nativeTheme.themeSource)
|
2023-01-19 00:20:50 +01:00
|
|
|
|
|
2023-12-23 11:17:13 +01:00
|
|
|
|
ipcMain.handle('buffer-content:load', async () => {
|
|
|
|
|
let bufferPath = getBufferFilePath()
|
2023-01-19 00:20:50 +01:00
|
|
|
|
if (jetpack.exists(bufferPath) === "file") {
|
|
|
|
|
return await jetpack.read(bufferPath, 'utf8')
|
|
|
|
|
} else {
|
|
|
|
|
return isDev? initialDevContent : initialContent
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
async function save(content) {
|
2023-12-23 11:17:13 +01:00
|
|
|
|
return await jetpack.write(getBufferFilePath(), content, {
|
2023-01-19 00:20:50 +01:00
|
|
|
|
atomic: true,
|
|
|
|
|
mode: '600',
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ipcMain.handle('buffer-content:save', async (event, content) => {
|
|
|
|
|
return await save(content)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
ipcMain.handle('buffer-content:saveAndQuit', async (event, content) => {
|
|
|
|
|
await save(content)
|
|
|
|
|
contentSaved = true
|
|
|
|
|
app.quit()
|
|
|
|
|
})
|
2023-01-20 15:33:26 +01:00
|
|
|
|
|
2023-01-27 17:18:31 +01:00
|
|
|
|
ipcMain.handle('settings:set', (event, settings) => {
|
2023-12-06 11:56:10 +01:00
|
|
|
|
if (settings.keymap !== CONFIG.get("settings.keymap")) {
|
2023-01-27 17:18:31 +01:00
|
|
|
|
currentKeymap = settings.keymap
|
|
|
|
|
}
|
2023-12-10 22:59:42 +01:00
|
|
|
|
let globalHotkeyChanged = settings.enableGlobalHotkey !== CONFIG.get("settings.enableGlobalHotkey") || settings.globalHotkey !== CONFIG.get("settings.globalHotkey")
|
|
|
|
|
|
2023-01-27 17:18:31 +01:00
|
|
|
|
CONFIG.set("settings", settings)
|
2023-12-10 22:59:42 +01:00
|
|
|
|
|
2023-01-27 17:18:31 +01:00
|
|
|
|
win?.webContents.send(SETTINGS_CHANGE_EVENT, settings)
|
2023-12-10 22:59:42 +01:00
|
|
|
|
|
|
|
|
|
if (globalHotkeyChanged) {
|
|
|
|
|
registerGlobalHotkey()
|
|
|
|
|
}
|
2023-01-20 15:33:26 +01:00
|
|
|
|
})
|