mirror of
https://github.com/heyman/heynote.git
synced 2025-01-08 23:19:48 +01:00
7be0a304d0
Add support for migrating old buffer file to new library. Add support for changing location for the notes library. Replace theme toggle in status bar with a dropdown in Appearance settings. Improve New Note and Update Note dialogs. Implement UI for confirming note delete (the actualal deltion is still to be implemented).
245 lines
6.7 KiB
TypeScript
245 lines
6.7 KiB
TypeScript
const { contextBridge } = require('electron')
|
|
import themeMode from "./theme-mode"
|
|
import { isMac, isWindows, isLinux, isDev } from "../detect-platform"
|
|
import { ipcRenderer } from "electron"
|
|
import {
|
|
WINDOW_CLOSE_EVENT,
|
|
OPEN_SETTINGS_EVENT,
|
|
SETTINGS_CHANGE_EVENT,
|
|
UPDATE_AVAILABLE_EVENT,
|
|
UPDATE_ERROR,
|
|
UPDATE_DOWNLOAD_PROGRESS,
|
|
UPDATE_NOT_AVAILABLE_EVENT,
|
|
UPDATE_START_DOWNLOAD,
|
|
UPDATE_INSTALL_AND_RESTART,
|
|
UPDATE_DOWNLOADED,
|
|
UPDATE_CHECK_FOR_UPDATES,
|
|
} from "../constants"
|
|
import CONFIG from "../config"
|
|
import getCurrencyData from "./currency"
|
|
|
|
|
|
contextBridge.exposeInMainWorld("heynote", {
|
|
defaultFontFamily: "Hack",
|
|
defaultFontSize: 12,
|
|
|
|
platform: {
|
|
isMac,
|
|
isWindows,
|
|
isLinux,
|
|
isWebApp: false,
|
|
},
|
|
|
|
isDev: isDev,
|
|
themeMode: themeMode,
|
|
|
|
init() {
|
|
ipcRenderer.on("buffer:change", (event, path, content) => {
|
|
// called on all changes to open buffer files
|
|
// go through all registered callbacks for this path and call them
|
|
if (this.buffer._onChangeCallbacks[path]) {
|
|
this.buffer._onChangeCallbacks[path].forEach(callback => callback(content))
|
|
}
|
|
})
|
|
},
|
|
|
|
quit() {
|
|
console.log("quitting")
|
|
//ipcRenderer.invoke("app_quit")
|
|
},
|
|
|
|
onWindowClose(callback) {
|
|
ipcRenderer.on(WINDOW_CLOSE_EVENT, callback)
|
|
},
|
|
|
|
onOpenSettings(callback) {
|
|
ipcRenderer.on(OPEN_SETTINGS_EVENT, callback)
|
|
},
|
|
|
|
buffer: {
|
|
async exists(path) {
|
|
return await ipcRenderer.invoke("buffer:exists", path)
|
|
},
|
|
|
|
async getList() {
|
|
return await ipcRenderer.invoke("buffer:getList")
|
|
},
|
|
|
|
async getDirectoryList() {
|
|
return await ipcRenderer.invoke("buffer:getDirectoryList")
|
|
},
|
|
|
|
async load(path) {
|
|
return await ipcRenderer.invoke("buffer:load", path)
|
|
},
|
|
|
|
async save(path, content) {
|
|
return await ipcRenderer.invoke("buffer:save", path, content)
|
|
},
|
|
|
|
async move(path, newPath) {
|
|
return await ipcRenderer.invoke("buffer:move", path, newPath)
|
|
},
|
|
|
|
async create(path, content) {
|
|
return await ipcRenderer.invoke("buffer:create", path, content)
|
|
},
|
|
|
|
async saveAndQuit(contents) {
|
|
return await ipcRenderer.invoke("buffer:saveAndQuit", contents)
|
|
},
|
|
|
|
async close(path) {
|
|
return await ipcRenderer.invoke("buffer:close", path)
|
|
},
|
|
|
|
_onChangeCallbacks: {},
|
|
addOnChangeCallback(path, callback) {
|
|
// register a callback to be called when the buffer content changes for a specific file
|
|
if (!this._onChangeCallbacks[path]) {
|
|
this._onChangeCallbacks[path] = []
|
|
}
|
|
this._onChangeCallbacks[path].push(callback)
|
|
},
|
|
removeOnChangeCallback(path, callback) {
|
|
if (this._onChangeCallbacks[path]) {
|
|
this._onChangeCallbacks[path] = this._onChangeCallbacks[path].filter(cb => cb !== callback)
|
|
}
|
|
},
|
|
|
|
async selectLocation() {
|
|
return await ipcRenderer.invoke("library:selectLocation")
|
|
},
|
|
|
|
setLibraryPathChangeCallback(callback) {
|
|
ipcRenderer.on("library:pathChanged", callback)
|
|
},
|
|
},
|
|
|
|
settings: CONFIG.get("settings"),
|
|
|
|
setSettings(settings) {
|
|
ipcRenderer.invoke("settings:set", settings)
|
|
},
|
|
|
|
async getCurrencyData() {
|
|
return await getCurrencyData()
|
|
},
|
|
|
|
onSettingsChange(callback) {
|
|
ipcRenderer.on(SETTINGS_CHANGE_EVENT, (event, settings) => callback(settings))
|
|
},
|
|
|
|
autoUpdate: {
|
|
callbacks(callbacks) {
|
|
ipcRenderer.on(UPDATE_AVAILABLE_EVENT, (event, info) => callbacks?.updateAvailable(info))
|
|
ipcRenderer.on(UPDATE_NOT_AVAILABLE_EVENT, (event) => callbacks?.updateNotAvailable())
|
|
ipcRenderer.on(UPDATE_DOWNLOADED, (event) => callbacks?.updateDownloaded())
|
|
ipcRenderer.on(UPDATE_ERROR, (event, error) => callbacks?.updateError(error))
|
|
ipcRenderer.on(UPDATE_DOWNLOAD_PROGRESS, (event, progress) => callbacks?.updateDownloadProgress(progress))
|
|
},
|
|
|
|
startDownload() {
|
|
ipcRenderer.invoke(UPDATE_START_DOWNLOAD)
|
|
},
|
|
installAndRestart() {
|
|
ipcRenderer.invoke(UPDATE_INSTALL_AND_RESTART)
|
|
},
|
|
checkForUpdates() {
|
|
ipcRenderer.invoke(UPDATE_CHECK_FOR_UPDATES)
|
|
},
|
|
},
|
|
|
|
async getVersion() {
|
|
return await ipcRenderer.invoke("getVersion")
|
|
},
|
|
|
|
async getInitErrors() {
|
|
return await ipcRenderer.invoke("getInitErrors")
|
|
},
|
|
})
|
|
|
|
|
|
function domReady(condition: DocumentReadyState[] = ['complete', 'interactive']) {
|
|
return new Promise((resolve) => {
|
|
if (condition.includes(document.readyState)) {
|
|
resolve(true)
|
|
} else {
|
|
document.addEventListener('readystatechange', () => {
|
|
if (condition.includes(document.readyState)) {
|
|
resolve(true)
|
|
}
|
|
})
|
|
}
|
|
})
|
|
}
|
|
|
|
const safeDOM = {
|
|
append(parent: HTMLElement, child: HTMLElement) {
|
|
if (!Array.from(parent.children).find(e => e === child)) {
|
|
return parent.appendChild(child)
|
|
}
|
|
},
|
|
remove(parent: HTMLElement, child: HTMLElement) {
|
|
if (Array.from(parent.children).find(e => e === child)) {
|
|
return parent.removeChild(child)
|
|
}
|
|
},
|
|
}
|
|
|
|
|
|
function useLoading() {
|
|
const className = `loaders-css__square-spin`
|
|
const styleContent = `
|
|
.app-loading-wrap {
|
|
position: fixed;
|
|
top: 0;
|
|
left: 0;
|
|
width: 100vw;
|
|
height: 100vh;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
background: #fff;
|
|
z-index: 9;
|
|
}
|
|
@media (prefers-color-scheme: dark) {
|
|
.${className} > div {
|
|
background: #fff;
|
|
}
|
|
.app-loading-wrap {
|
|
background: #262B37
|
|
}
|
|
}
|
|
`
|
|
const oStyle = document.createElement('style')
|
|
const oDiv = document.createElement('div')
|
|
|
|
oStyle.id = 'app-loading-style'
|
|
oStyle.innerHTML = styleContent
|
|
oDiv.className = 'app-loading-wrap'
|
|
oDiv.innerHTML = `<div class="${className}"></div>`
|
|
|
|
return {
|
|
appendLoading() {
|
|
safeDOM.append(document.head, oStyle)
|
|
safeDOM.append(document.body, oDiv)
|
|
},
|
|
removeLoading() {
|
|
safeDOM.remove(document.head, oStyle)
|
|
safeDOM.remove(document.body, oDiv)
|
|
},
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------
|
|
|
|
const { appendLoading, removeLoading } = useLoading()
|
|
domReady().then(appendLoading)
|
|
|
|
window.onmessage = (ev) => {
|
|
ev.data.payload === 'removeLoading' && removeLoading()
|
|
}
|
|
|
|
setTimeout(removeLoading, 4999)
|