2024-07-26 11:30:25 +02:00
|
|
|
import { Exception } from "sass";
|
2024-01-04 14:27:04 +01:00
|
|
|
import { SETTINGS_CHANGE_EVENT, OPEN_SETTINGS_EVENT } from "../electron/constants";
|
2024-10-29 17:03:36 +01:00
|
|
|
import { NoteFormat } from "../src/common/note-format";
|
2024-01-04 14:27:04 +01:00
|
|
|
|
2023-12-25 14:18:44 +01:00
|
|
|
const mediaMatch = window.matchMedia('(prefers-color-scheme: dark)')
|
|
|
|
let themeCallback = null
|
2023-12-28 16:48:56 +01:00
|
|
|
mediaMatch.addEventListener("change", async (event) => {
|
|
|
|
if (themeCallback) {
|
|
|
|
themeCallback((await Heynote.themeMode.get()).computed)
|
|
|
|
}
|
|
|
|
})
|
2023-12-25 14:18:44 +01:00
|
|
|
|
2024-01-11 20:21:44 +01:00
|
|
|
const isMobileDevice = window.matchMedia("(max-width: 600px)").matches
|
|
|
|
|
2023-12-28 16:48:56 +01:00
|
|
|
let autoUpdateCallbacks = null
|
2023-12-25 14:18:44 +01:00
|
|
|
let currencyData = null
|
|
|
|
|
2023-12-28 18:01:10 +01:00
|
|
|
let platform
|
|
|
|
const uaPlatform = window.navigator?.userAgentData?.platform || window.navigator.platform
|
|
|
|
if (uaPlatform.indexOf("Win") !== -1) {
|
|
|
|
platform = {
|
|
|
|
isMac: false,
|
|
|
|
isWindows: true,
|
|
|
|
isLinux: false,
|
|
|
|
}
|
|
|
|
} else if (uaPlatform.indexOf("Linux") !== -1) {
|
|
|
|
platform = {
|
|
|
|
isMac: false,
|
|
|
|
isWindows: false,
|
|
|
|
isLinux: true,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
platform = {
|
2023-12-25 14:18:44 +01:00
|
|
|
isMac: true,
|
|
|
|
isWindows: false,
|
|
|
|
isLinux: false,
|
2023-12-28 18:01:10 +01:00
|
|
|
}
|
|
|
|
}
|
2024-01-12 14:40:44 +01:00
|
|
|
platform.isWebApp = true
|
2023-12-28 18:01:10 +01:00
|
|
|
|
2024-01-04 14:27:04 +01:00
|
|
|
|
|
|
|
class IpcRenderer {
|
|
|
|
constructor() {
|
|
|
|
this.callbacks = {}
|
|
|
|
}
|
|
|
|
|
|
|
|
on(event, callback) {
|
|
|
|
if (!this.callbacks[event]) {
|
|
|
|
this.callbacks[event] = []
|
|
|
|
}
|
|
|
|
this.callbacks[event].push(callback)
|
|
|
|
}
|
|
|
|
|
|
|
|
send(event, ...args) {
|
|
|
|
if (this.callbacks[event]) {
|
|
|
|
for (const callback of this.callbacks[event]) {
|
|
|
|
callback(null, ...args)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const ipcRenderer = new IpcRenderer()
|
|
|
|
|
|
|
|
// get initial settings
|
|
|
|
let settingsData = localStorage.getItem("settings")
|
|
|
|
let initialSettings = {
|
|
|
|
keymap: "default",
|
2024-01-05 16:53:00 +01:00
|
|
|
emacsMetaKey: "alt",
|
2024-01-04 14:27:04 +01:00
|
|
|
showLineNumberGutter: true,
|
|
|
|
showFoldGutter: true,
|
2024-01-04 14:52:41 +01:00
|
|
|
bracketClosing: false,
|
2024-01-04 14:27:04 +01:00
|
|
|
}
|
|
|
|
if (settingsData !== null) {
|
|
|
|
initialSettings = Object.assign(initialSettings, JSON.parse(settingsData))
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-10-29 16:47:55 +01:00
|
|
|
const NOTE_KEY_PREFIX = "heynote-library__"
|
|
|
|
|
|
|
|
function noteKey(path) {
|
|
|
|
return NOTE_KEY_PREFIX + path
|
|
|
|
}
|
|
|
|
|
|
|
|
function getNoteMetadata(content) {
|
|
|
|
const firstSeparator = content.indexOf("\n∞∞∞")
|
|
|
|
if (firstSeparator === -1) {
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
const metadata = JSON.parse(content.slice(0, firstSeparator).trim())
|
|
|
|
return {"name": metadata.name}
|
|
|
|
} catch (e) {
|
|
|
|
return {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-29 17:03:36 +01:00
|
|
|
// Migrate single buffer (Heynote pre 2.0) in localStorage to notes library
|
|
|
|
// At some point we can remove this migration code
|
|
|
|
function migrateBufferFileToLibrary() {
|
|
|
|
if (!("buffer" in localStorage)) {
|
|
|
|
// nothing to migrate
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (Object.keys(localStorage).filter(key => key.startsWith(NOTE_KEY_PREFIX)).length > 0) {
|
|
|
|
// already migrated
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log("Migrating single buffer to notes library")
|
|
|
|
|
|
|
|
let content = localStorage.getItem("buffer")
|
|
|
|
const metadata = getNoteMetadata(content)
|
|
|
|
if (!metadata || !metadata.name) {
|
|
|
|
console.log("Adding metadata to Scratch note")
|
|
|
|
const note = NoteFormat.load(content)
|
|
|
|
note.metadata.name = "Scratch"
|
|
|
|
content = note.serialize()
|
|
|
|
}
|
|
|
|
localStorage.setItem("heynote-library__scratch.txt", content)
|
|
|
|
localStorage.removeItem("buffer")
|
|
|
|
}
|
|
|
|
migrateBufferFileToLibrary()
|
|
|
|
|
2023-12-28 18:01:10 +01:00
|
|
|
const Heynote = {
|
|
|
|
platform: platform,
|
2024-01-11 20:21:44 +01:00
|
|
|
defaultFontFamily: "Hack",
|
|
|
|
defaultFontSize: isMobileDevice ? 16 : 12,
|
2023-12-25 14:18:44 +01:00
|
|
|
|
|
|
|
buffer: {
|
2024-07-24 13:52:44 +02:00
|
|
|
async load(path) {
|
2024-10-29 16:47:55 +01:00
|
|
|
//console.log("loading", path)
|
|
|
|
const content = localStorage.getItem(noteKey(path))
|
|
|
|
return content === null ? '{"formatVersion":"1.0.0","name":"Scratch"}\n∞∞∞text-a\n' : content
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
|
|
|
|
2024-07-24 13:52:44 +02:00
|
|
|
async save(path, content) {
|
2024-10-29 16:47:55 +01:00
|
|
|
//console.log("saving", path, content)
|
|
|
|
localStorage.setItem(noteKey(path), content)
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
|
|
|
|
2024-07-26 11:30:25 +02:00
|
|
|
async create(path, content) {
|
2024-10-29 16:47:55 +01:00
|
|
|
localStorage.setItem(noteKey(path), content)
|
|
|
|
},
|
|
|
|
|
|
|
|
async delete(path) {
|
|
|
|
localStorage.removeItem(noteKey(path))
|
|
|
|
},
|
|
|
|
|
|
|
|
async move(path, newPath) {
|
|
|
|
const content = localStorage.getItem(noteKey(path))
|
|
|
|
localStorage.setItem(noteKey(newPath), content)
|
|
|
|
localStorage.removeItem(noteKey(path))
|
2024-07-26 11:30:25 +02:00
|
|
|
},
|
|
|
|
|
2024-07-24 13:52:44 +02:00
|
|
|
async saveAndQuit(contents) {
|
2023-12-25 14:18:44 +01:00
|
|
|
|
|
|
|
},
|
2024-01-01 19:04:40 +01:00
|
|
|
|
2024-07-24 13:52:44 +02:00
|
|
|
async exists(path) {
|
2024-10-29 16:47:55 +01:00
|
|
|
return localStorage.getItem(noteKey(path)) !== null
|
2024-07-24 13:52:44 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
async getList(path) {
|
2024-10-29 16:47:55 +01:00
|
|
|
//return {"scratch.txt": {name:"Scratch"}}
|
|
|
|
const notes = {}
|
|
|
|
for (let [key, content] of Object.entries(localStorage)) {
|
|
|
|
if (key.startsWith(NOTE_KEY_PREFIX)) {
|
|
|
|
const path = key.slice(NOTE_KEY_PREFIX.length)
|
|
|
|
notes[path] = getNoteMetadata(content)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return notes
|
2024-07-24 13:52:44 +02:00
|
|
|
},
|
|
|
|
|
2024-07-25 13:25:19 +02:00
|
|
|
async getDirectoryList() {
|
2024-10-29 16:47:55 +01:00
|
|
|
const directories = new Set()
|
|
|
|
for (let key in localStorage) {
|
|
|
|
if (key.startsWith(NOTE_KEY_PREFIX)) {
|
|
|
|
const path = key.slice(NOTE_KEY_PREFIX.length)
|
|
|
|
const parts = path.split("/")
|
|
|
|
if (parts.length > 1) {
|
|
|
|
for (let i = 1; i < parts.length; i++) {
|
|
|
|
directories.add(parts.slice(0, i).join("/"))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//console.log("directories", directories)
|
|
|
|
return [...directories]
|
2024-07-25 13:25:19 +02:00
|
|
|
},
|
|
|
|
|
2024-07-24 13:52:44 +02:00
|
|
|
async close(path) {
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
_onChangeCallbacks: {},
|
|
|
|
addOnChangeCallback(path, callback) {
|
|
|
|
|
|
|
|
},
|
|
|
|
removeOnChangeCallback(path, callback) {
|
2024-01-01 19:04:40 +01:00
|
|
|
|
|
|
|
},
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
onWindowClose(callback) {
|
|
|
|
//ipcRenderer.on(WINDOW_CLOSE_EVENT, callback)
|
|
|
|
},
|
|
|
|
|
2024-01-04 14:27:04 +01:00
|
|
|
settings: initialSettings,
|
|
|
|
|
2023-12-25 14:18:44 +01:00
|
|
|
onOpenSettings(callback) {
|
2024-01-04 14:27:04 +01:00
|
|
|
ipcRenderer.on(OPEN_SETTINGS_EVENT, callback)
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
onSettingsChange(callback) {
|
2024-01-04 14:27:04 +01:00
|
|
|
ipcRenderer.on(SETTINGS_CHANGE_EVENT, (event, settings) => callback(settings))
|
|
|
|
},
|
|
|
|
|
|
|
|
setSettings(settings) {
|
|
|
|
localStorage.setItem("settings", JSON.stringify(settings))
|
|
|
|
ipcRenderer.send(SETTINGS_CHANGE_EVENT, settings)
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
themeMode: {
|
|
|
|
set: (mode) => {
|
2023-12-28 16:48:56 +01:00
|
|
|
localStorage.setItem("theme", mode)
|
2023-12-25 14:18:44 +01:00
|
|
|
themeCallback(mode)
|
2024-10-29 16:47:55 +01:00
|
|
|
//console.log("set theme to", mode)
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
|
|
|
get: async () => {
|
2023-12-28 16:48:56 +01:00
|
|
|
const theme = localStorage.getItem("theme") || "system"
|
|
|
|
const systemTheme = mediaMatch.matches ? "dark" : "light"
|
2023-12-25 14:18:44 +01:00
|
|
|
return {
|
2023-12-28 16:48:56 +01:00
|
|
|
theme: theme,
|
|
|
|
computed: theme === "system" ? systemTheme : theme,
|
2023-12-25 14:18:44 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
onChange: (callback) => {
|
|
|
|
themeCallback = callback
|
|
|
|
},
|
|
|
|
removeListener() {
|
2023-12-28 16:48:56 +01:00
|
|
|
themeCallback = null
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
2023-12-28 16:48:56 +01:00
|
|
|
initial: localStorage.getItem("theme") || "system",
|
2023-12-25 14:18:44 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
getCurrencyData: async () => {
|
|
|
|
if (currencyData !== null) {
|
|
|
|
return currencyData
|
|
|
|
}
|
|
|
|
const response = await fetch("https://currencies.heynote.com/rates.json", {cache: "no-cache"})
|
|
|
|
currencyData = JSON.parse(await response.text())
|
|
|
|
return currencyData
|
|
|
|
},
|
2024-01-12 15:08:53 +01:00
|
|
|
|
|
|
|
async getVersion() {
|
|
|
|
return __APP_VERSION__ + " (" + __GIT_HASH__ + ")"
|
|
|
|
},
|
2024-07-24 12:31:19 +02:00
|
|
|
|
|
|
|
async getInitErrors() {
|
|
|
|
|
|
|
|
},
|
2023-12-25 14:18:44 +01:00
|
|
|
}
|
2023-12-28 16:48:56 +01:00
|
|
|
|
2024-01-04 14:27:04 +01:00
|
|
|
export { Heynote, ipcRenderer}
|