tray: hotkey, run in background

window management features patched to reduce unexpected behaviour e.g. duplicate windows
This commit is contained in:
dragonwocky 2021-12-03 00:00:16 +11:00
parent 7f53445ab0
commit e761b309bf
9 changed files with 219 additions and 10 deletions

View File

@ -43,7 +43,7 @@
"key": "hotkey",
"label": "toggle header collapse hotkey",
"tooltip": "**opens/closes the currently focused header section**",
"value": "ctrl+enter"
"value": "Ctrl+Enter"
}
]
}

View File

@ -11,7 +11,7 @@ export const createWindowButtons = async ({ web, components }, db) => {
maximizeIcon = (await db.get(['maximize_icon'])) || (await components.feather('maximize')),
unmaximizeIcon =
(await db.get(['unmaximize_icon'])) || (await components.feather('minimize')),
closeIcon = await db.get(['close_icon'], await components.feather('x'));
closeIcon = (await db.get(['close_icon'])) || (await components.feather('x'));
minimizeIcon = minimizeIcon.trim();
maximizeIcon = maximizeIcon.trim();
unmaximizeIcon = unmaximizeIcon.trim();

View File

@ -16,11 +16,3 @@ module.exports = async function (api, db, __exports, __eval) {
};
`);
};
// hide on open
// maybe only do once to hide first window?
// exports.createWindow = (...args) => {
// warmWindowState.warmedWindow = true;
// createWindow(...args);
// };

View File

@ -4,6 +4,8 @@
"components",
"integrated-titlebar",
"tray",
"tweaks",
"font-chooser",
"outliner",

View File

@ -595,6 +595,9 @@ body,
box-shadow: var(--theme--accent_blue-active, rgba(26, 170, 220, 0.7)) 0px 0px 0px 1px inset,
var(--theme--accent_blue-active, rgba(26, 170, 220, 0.4)) 0px 0px 0px 2px !important;
}
.notion-sidebar-switcher:focus-visible {
box-shadow: none !important;
}
.notion-onboarding-plan-type-team[style*='box-shadow: rgb(46, 170, 220)'],
.notion-onboarding-plan-type-personal[style*='box-shadow: rgb(46, 170, 220)'] {
box-shadow: var(--theme--accent_blue) 0px 0px 0px 2px !important;

17
repo/tray/client.mjs Normal file
View File

@ -0,0 +1,17 @@
/*
* notion-enhancer: tray
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
export default async function ({ env, web }, db) {
const runInBackground = await db.get(['run_in_background']);
if (!runInBackground) return;
// force new window creation on create new window hotkey
// hotkey is built into notion, so can't be changed,
// but is broken by this mod's window duplication prevention
web.addHotkeyListener([env.name === 'darwin' ? 'Meta' : 'Ctrl', 'Shift', 'N'], () => {
__enhancerElectronApi.sendMessage('create-new-window');
});
}

View File

@ -0,0 +1,44 @@
/*
* notion-enhancer: tray
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
'use strict';
module.exports = async function (api, db, __exports, __eval) {
const electron = require('electron'),
{ isMenuOpen } = require('notion-enhancer/worker.cjs'),
runInBackground = await db.get(['run_in_background']);
if (!runInBackground) return;
let appQuit = false;
electron.app.on('before-quit', () => {
appQuit = true;
});
const notionCreateWindow = __exports.createWindow;
__exports.createWindow = (relativeUrl = '', args, force = false) => {
const windows = electron.BrowserWindow.getAllWindows();
if (windows.length) windows.forEach((win) => win.show());
if (force || !windows.length) {
// hijack close event to hide instead
const window = notionCreateWindow(relativeUrl, args);
window.prependListener('close', (e) => {
const isLastNotionWindow =
electron.BrowserWindow.getAllWindows().length === (isMenuOpen() ? 2 : 1);
if (!appQuit && isLastNotionWindow) {
window.hide();
e.preventDefault();
throw new Error('<fake error>: prevent window close');
}
});
return window;
} else {
// prevents duplicate windows on dock/taskbar click
windows[0].focus();
return windows[0];
}
};
};

112
repo/tray/main.cjs Normal file
View File

@ -0,0 +1,112 @@
/*
* notion-enhancer: tray
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
'use strict';
let tray;
module.exports = async function ({ env, registry }, db, __exports, __eval) {
const electron = require('electron'),
path = require('path'),
enhancerIcon = path.resolve(`${__dirname}/../../media/colour-x16.png`),
hotkey = await db.get(['hotkey']),
runInBackground = await db.get(['run_in_background']),
menuHotkey = await (
await registry.db('a6621988-551d-495a-97d8-3c568bca2e9e')
).get(['hotkey']);
const toggleWindows = (checkFocus = true) => {
const windows = electron.BrowserWindow.getAllWindows();
if (runInBackground) {
// hide
if (windows.some((win) => (checkFocus ? win.isFocused() : true) && win.isVisible())) {
windows.forEach((win) => [win.isFocused() && win.blur(), win.hide()]);
} else windows.forEach((win) => win.show());
} else {
// minimize
if (windows.some((win) => (checkFocus ? win.isFocused() : true) && !win.isMinimized())) {
windows.forEach((win) => win.minimize());
} else windows.forEach((win) => win.restore());
}
};
await electron.app.whenReady();
tray = new electron.Tray(enhancerIcon);
tray.setToolTip('notion-enhancer');
tray.on('click', () => toggleWindows(false));
electron.globalShortcut.register(hotkey, toggleWindows);
// connects to client hotkey listener
// manually forces new window creation
// since notion's default is broken by
// duplicate window prevention
const createWindow = () => {
const { createWindow } = env.notionRequire('main/createWindow.js');
createWindow('', null, true);
};
electron.ipcMain.on(`notion-enhancer:create-new-window`, createWindow);
const contextMenu = electron.Menu.buildFromTemplate([
{
type: 'normal',
label: 'notion-enhancer',
icon: enhancerIcon,
enabled: false,
},
{ type: 'separator' },
{
type: 'normal',
label: 'docs',
click: () => electron.shell.openExternal('https://notion-enhancer.github.io/'),
},
{
type: 'normal',
label: 'source code',
click: () => electron.shell.openExternal('https://github.com/notion-enhancer/'),
},
{
type: 'normal',
label: 'community',
click: () => electron.shell.openExternal('https://discord.gg/sFWPXtA'),
},
{
type: 'normal',
label: 'enhancements menu',
accelerator: menuHotkey,
click: env.focusMenu,
},
{ type: 'separator' },
{
type: 'normal',
label: 'toggle visibility',
accelerator: hotkey,
click: toggleWindows,
},
{
type: 'normal',
label: 'new window',
click: createWindow,
accelerator: 'CmdOrCtrl+Shift+N',
},
{
label: 'relaunch',
click: env.reload,
},
{
label: 'quit',
role: 'quit',
},
]);
tray.setContextMenu(contextMenu);
};
// hide on open
// maybe only do once to hide first window?
// exports.createWindow = (...args) => {
// warmWindowState.warmedWindow = true;
// createWindow(...args);
// };

39
repo/tray/mod.json Normal file
View File

@ -0,0 +1,39 @@
{
"name": "tray",
"id": "f96f4a73-21af-4e3f-a68f-ab4976b020da",
"environments": ["linux", "win32", "darwin"],
"version": "0.11.0",
"description": "adds an icon to the system tray/menubar for extra app/window management features (e.g. open on startup, a global hotkey).",
"tags": ["extension", "app"],
"authors": [
{
"name": "dragonwocky",
"email": "thedragonring.bod@gmail.com",
"homepage": "https://dragonwocky.me/",
"avatar": "https://dragonwocky.me/avatar.jpg"
}
],
"css": {},
"js": {
"client": ["client.mjs"],
"electron": [
{ "source": "main.cjs", "target": "main/main.js" },
{ "source": "createWindow.cjs", "target": "main/createWindow.js" }
]
},
"options": [
{
"type": "toggle",
"key": "run_in_background",
"label": "run notion in the background",
"tooltip": "**pressing the close button or toggling window visibility will hide the app, running notion in the background** (instead of quitting or minimizing it)",
"value": true
},
{
"type": "hotkey",
"key": "hotkey",
"label": "toggle window visibility hotkey",
"value": "Ctrl+Shift+A"
}
]
}