diff --git a/README.md b/README.md
index 672e56b..cd7bdf2 100644
--- a/README.md
+++ b/README.md
@@ -1,11 +1,13 @@
**Tray** is an [Obsidian](https://obsidian.md/) plugin that can be used to launch the app
-on system startup and run it in the background, adding an icon to the system tray that it
-can be minimised to and a global hotkey to toggle visibility of the app's windows.
+on system startup and run it in the background, adding global hotkeys and a tray menu that
+toggle app window visibility and can create quick notes from anywhere in your operating system.
## Configuration
+### Window management
+
| Option | Description | Default |
| -------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------- |
| Launch on startup | Open Obsidian automatically whenever you log into your computer. | Disabled |
@@ -13,7 +15,15 @@ can be minimised to and a global hotkey to toggle visibility of the app's window
| Run in background | Hide the app and continue to run it in the background instead of quitting it when pressing the window close button or toggle focus hotkey. | Disabled |
| Hide taskbar icon | Hides the window's icon from from the dock/taskbar. This may not work on all Linux-based OSes. | Disabled |
| Create tray icon | Add an icon to your system tray/menubar to bring hidden Obsidian windows back into focus on click or force a full quit/relaunch of the app through the right-click menu. _Changing this option requires a restart to take effect._ | Enabled |
-| Toggle window focus hotkey | Format: [Electron accelerator](https://www.electronjs.org/docs/latest/api/accelerator) | CmdOrCtrl+Shift+Tab |
+| Toggle window focus hotkey | This hotkey is registered globally and will be detected even if Obsidian does not have keyboard focus. Format: [Electron accelerator](https://www.electronjs.org/docs/latest/api/accelerator) | CmdOrCtrl+Shift+Tab |
+
+### Quick notes
+
+| Option | Description | Default |
+| ---------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------- |
+| Quick note location | New quick notes will be placed in this folder. | |
+| Quick note date format | New quick notes will use a filename of this pattern. Format: [Moment.js format string](https://momentjs.com/docs/#/displaying/format/) | YYYY-MM-DD |
+| Quick note hotkey | This hotkey is registered globally and will be detected even if Obsidian does not have keyboard focus. Format: [Electron accelerator](https://www.electronjs.org/docs/latest/api/accelerator) | Disabled |
## Installation
diff --git a/main.js b/main.js
index 8354517..4e61a45 100644
--- a/main.js
+++ b/main.js
@@ -3,6 +3,8 @@
"use strict";
+const DEFAULT_DATE_FORMAT = "YYYY-MM-DD";
+
let tray;
const obsidian = require("obsidian"),
{
@@ -61,7 +63,17 @@ const onWindowClose = (event) => {
closeBtn.removeEventListener("click", onWindowClose, true);
};
-const setHideTaskbarIcon = (plugin) => {
+const addQuickNote = (plugin) => {
+ const { quickNoteLocation, quickNoteDateFormat } = plugin.settings,
+ format = quickNoteDateFormat || DEFAULT_DATE_FORMAT,
+ date = obsidian.moment().format(format),
+ name = obsidian
+ .normalizePath(`${quickNoteLocation ?? ""}/${date}`)
+ .replace(/\*|"|\\|<|>|:|\||\?/g, "-");
+ plugin.app.fileManager.createAndOpenMarkdownFile(name);
+ showWindows();
+ },
+ setHideTaskbarIcon = (plugin) => {
const win = getCurrentWindow();
win.setSkipTaskbar(plugin.settings.hideTaskbarIcon);
},
@@ -84,6 +96,12 @@ const createTrayIcon = (plugin) => {
``
),
contextMenu = Menu.buildFromTemplate([
+ {
+ type: "normal",
+ label: "Add Quick Note",
+ accelerator: plugin.settings.quickNoteHotkey,
+ click: () => addQuickNote(plugin),
+ },
{
type: "normal",
label: "Open Obsidian",
@@ -112,25 +130,36 @@ const createTrayIcon = (plugin) => {
tray.on("click", () => toggleWindows(plugin.settings.runInBackground, false));
};
-const registerHotkey = (plugin) => {
+const registerHotkeys = (plugin) => {
console.log("obsidian-tray: registering hotkey");
try {
- const accelerator = plugin.settings.toggleWindowFocusHotkey;
- globalShortcut.register(accelerator, () => {
- const runInBackground = plugin.settings.runInBackground;
- toggleWindows(runInBackground);
- });
+ const toggleAccelerator = plugin.settings.toggleWindowFocusHotkey,
+ quicknoteAccelerator = plugin.settings.quickNoteHotkey;
+ if (toggleAccelerator) {
+ globalShortcut.register(toggleAccelerator, () => {
+ const runInBackground = plugin.settings.runInBackground;
+ toggleWindows(runInBackground);
+ });
+ }
+ if (quicknoteAccelerator) {
+ globalShortcut.register(quicknoteAccelerator, () => {
+ addQuickNote(plugin);
+ });
+ }
} catch {}
},
- unregisterHotkey = (plugin) => {
+ unregisterHotkeys = (plugin) => {
console.log("obsidian-tray: unregistering hotkey");
try {
- const accelerator = plugin.settings.toggleWindowFocusHotkey;
- globalShortcut.unregister(accelerator);
+ const toggle = plugin.settings.toggleWindowFocusHotkey,
+ quicknote = plugin.settings.quickNoteHotkey;
+ globalShortcut.unregister(toggle);
+ globalShortcut.unregister(quicknote);
} catch {}
};
const OPTIONS = [
+ "Window management",
{
key: "launchOnStartup",
desc: "Open Obsidian automatically whenever you log into your computer.",
@@ -185,16 +214,26 @@ const OPTIONS = [
},
{
key: "toggleWindowFocusHotkey",
- desc: `
- Format:
-
- Electron accelerator
-
- `,
- type: "text",
+ type: "hotkey",
default: "CmdOrCtrl+Shift+Tab",
- onBeforeChange: unregisterHotkey,
- onChange: registerHotkey,
+ },
+ "Quick notes",
+ {
+ key: "quickNoteLocation",
+ desc: "New quick notes will be placed in this folder.",
+ type: "text",
+ placeholder: "Example: notes/quick",
+ },
+ {
+ key: "quickNoteDateFormat",
+ desc: "New quick notes will use a filename of this pattern.",
+ type: "moment",
+ default: DEFAULT_DATE_FORMAT,
+ },
+ {
+ key: "quickNoteHotkey",
+ type: "hotkey",
+ default: "CmdOrCtrl+Shift+Q",
},
];
@@ -210,6 +249,18 @@ const keyToLabel = (key) =>
.createRange()
.createContextualFragment((html ?? "").replace(/\s+/g, " "));
+const acceleratorFormat = `
+ This hotkey is registered globally and will be detected even if Obsidian does
+ not have keyboard focus. Format:
+
+ Electron accelerator
+ `,
+ momentFormat = `
+ Format:
+
+ Moment.js format string
+
Preview:
+ `;
class SettingsTab extends obsidian.PluginSettingTab {
constructor(app, plugin) {
super(app, plugin);
@@ -218,32 +269,58 @@ class SettingsTab extends obsidian.PluginSettingTab {
display() {
this.containerEl.empty();
for (const opt of OPTIONS) {
- const name = keyToLabel(opt.key),
- desc = htmlToFragment(opt.desc),
- onChange = async (value) => {
+ const setting = new obsidian.Setting(this.containerEl);
+ if (typeof opt === "string") {
+ setting.setName(opt);
+ setting.setHeading();
+ } else {
+ if (opt.default) {
+ opt.placeholder ??= `Example: ${opt.default}`;
+ }
+ if (opt.type === "hotkey") {
+ opt.desc ??= "";
+ opt.desc += acceleratorFormat;
+ opt.onBeforeChange = unregisterHotkeys;
+ opt.onChange = registerHotkeys;
+ }
+ if (opt.type === "moment") {
+ opt.desc = `${opt.desc ? `${opt.desc}
` : ""}${momentFormat}`;
+ }
+
+ setting.setName(keyToLabel(opt.key));
+ setting.setDesc(htmlToFragment(opt.desc));
+ const onChange = async (value) => {
await opt.onBeforeChange?.(this.plugin);
this.plugin.settings[opt.key] = value;
await this.plugin.saveSettings();
await opt.onChange?.(this.plugin);
};
- const setting = new obsidian.Setting(this.containerEl)
- .setName(name)
- .setDesc(desc);
- switch (opt.type) {
- case "toggle":
- setting.addToggle((toggle) =>
- toggle.setValue(this.plugin.settings[opt.key]).onChange(onChange)
- );
- break;
- case "text":
- default:
- setting.addText((text) =>
+ if (opt.type === "toggle") {
+ setting.addToggle((toggle) => {
+ toggle
+ .setValue(this.plugin.settings[opt.key] ?? opt.default)
+ .onChange(onChange);
+ });
+ } else if (opt.type === "moment") {
+ setting.addMomentFormat((moment) => {
+ const sampleEl = setting.descEl.createEl("b");
+ sampleEl.className = "u-pop";
+ moment
+ .setPlaceholder(opt.placeholder)
+ .setDefaultFormat(opt.default ?? "")
+ .setValue(this.plugin.settings[opt.key] ?? opt.default ?? "")
+ .setSampleEl(sampleEl)
+ .onChange(onChange);
+ });
+ } else {
+ setting.addText((text) => {
text
- .setPlaceholder(opt.default)
- .setValue(this.plugin.settings[opt.key])
- .onChange(onChange)
- );
+ .setPlaceholder(opt.placeholder)
+ .setValue(this.plugin.settings[opt.key] ?? opt.default ?? "")
+ .onChange(onChange);
+ });
+ }
}
}
}
@@ -256,7 +333,7 @@ class TrayPlugin extends obsidian.Plugin {
this.addSettingTab(new SettingsTab(this.app, this));
const { settings } = this;
- registerHotkey(this);
+ registerHotkeys(this);
setHideTaskbarIcon(this);
setLaunchOnStartup(this);
if (settings.createTrayIcon) createTrayIcon(this);
@@ -273,7 +350,7 @@ class TrayPlugin extends obsidian.Plugin {
}
}
onunload() {
- unregisterHotkey(this);
+ unregisterHotkeys(this);
cleanupWindowClose();
}