notion enhancer
@@ -12,6 +12,7 @@ installation
currently, only win10 is supported. it is possible to run this script via the wsl to modify the win10 notion app.
+python scripts must be run from the bash terminal or windows command prompt - directly opening/running them may not work.
(the styles should also work for the web version. these can be installed via an extension like stylus or a built-in feature like userChrome.css.)
@@ -24,7 +25,7 @@ or a built-in feature like userChrome.css< even if you are running the script from the wsl).cleaner.py
resources/user.css
files to your liking.resources/user.css
file to your liking (see styling).customiser.py
to build changes.done: run notion and enjoy.
@@ -91,14 +92,16 @@ to your preference. you will need to run or re-runcustomiser.py
af
tray
--
-
- single-click to toggle app visibility. right click to open menu. -
- settings will be saved in
%localappdata%/Programs/Notion/resources/app/user-preferences.json
+ - run on startup: run notion on boot/startup. (default: true)
- hide on open: hide the launch of notion to the tray. (default: false)
- open maximised: maximise the app on open. (default: false)
- close to tray: close window to tray rather than closing outright
on click of
⨉
. does not apply if multiple notion windows are open. (default: false)
+ - load theme.css: loads the custom colour theme file. +see colour theming for more information. (default: false)
single-click to toggle app visibility. right click to open menu.
+settings will be saved in %localappdata%/Programs/Notion/resources/app/user-preferences.json
customiser.py
every time you want to change some styles).
these should also work for the web version, if copied into your css customiser.
css below will work for every instance of the element, but if you wish to hide only a specific element
-(e.g. the '+ new' table row) it is recommended that you prepend each selector with [data-block-id='ID']
(video tutorial on fetching IDs).
[data-block-id='ID']
(video tutorial on fetching IDs).
- - dark+ theme + colour theming
-/*** dark+ ***/
-.notion-dark-theme .window-buttons {
- background: rgb(10, 10, 10);
-}
-.notion-dark-theme .window-buttons:hover {
- background: rgb(23, 23, 23);
-}
-/* sidebar */
-[style*='background: rgb(55, 60, 63)'],
-[style*='background: rgb(80, 85, 88)'] {
- background: rgb(0, 0, 0) !important;
-}
-/* main content */
-.notion-body.dark,
-[style*='background: rgb(47, 52, 55)'] {
- background: rgb(10, 10, 10) !important;
-}
-/* hovered buttons + table column titles */
-.notion-table-view > :first-child > :first-child,
-[style*='background: rgb(71, 76, 80)'],
-[style*='background: rgb(98, 102, 104)'] {
- background: rgb(15, 15, 15) !important;
- box-shadow: 0 0 0 0.5px rgba(78, 78, 78, 0.7);
-}
-/* database elements: e.g. view select, calendar event */
-[style*='background: rgb(63, 68, 71)'] {
- background: rgb(4, 4, 4) !important;
- border: 0.5px solid rgba(59, 59, 59, 0.7);
-}
+ this replaces the default notion dark theme. the provided theme file is my custom dark+ theme: +if you have another you wish to share, please contact me. if a few themes are provided i will +set up a distribution method (either including as optional themes or sharing on the website).
+to enable, see the tray options.
+to modify, enter the theme.css
file and change the colour values within the :root {}
- value names
+should describe what each colour will affect.
custom fonts
-styles must be added to the top of the user.css
file, so that nothing (other than comments) is above them @import statement
the @import statement must be added to the top of the user.css
file (with nothing above it
+except comments or other @import statements)
@import url('https://fonts.googleapis.com/css2?family=Fira+Code&family=Oxygen&family=Roboto+Slab:wght@300&display=swap');
.notion-app-inner {
font-family: 'Oxygen', sans-serif !important;
diff --git a/resources/preload.js b/resources/preload.js
index 455e8bc..84a0bb4 100644
--- a/resources/preload.js
+++ b/resources/preload.js
@@ -12,19 +12,35 @@
// DO NOT REMOVE THE INJECTION MARKER ABOVE
require('electron').remote.getGlobal('setTimeout')(() => {
- /* style injection */
const fs = require('fs'),
- css = fs.readFileSync('___user.css___'), // will be set by python script
- style = document.createElement('style'),
- head = document.getElementsByTagName('head')[0];
- if (!head) return;
- style.type = 'text/css';
- style.innerHTML = css;
- head.appendChild(style);
+ path = require('path'),
+ store = require(path.join(__dirname, '..', 'store.js'))({
+ config: 'user-preferences',
+ defaults: {
+ openhidden: false,
+ maximised: false,
+ tray: false,
+ theme: false,
+ },
+ });
const intervalID = setInterval(injection, 100);
function injection() {
if (document.querySelector('div.notion-topbar > div') == undefined) return;
+ clearInterval(intervalID);
+
+ /* style injection */
+ const head = document.getElementsByTagName('head')[0],
+ css = ['user'];
+ if (store.theme) css.push('theme');
+ console.table(store);
+ css.forEach((file) => {
+ file = fs.readFileSync(`☃☃☃assets☃☃☃/${file}.css`); // will be set by python script
+ let style = document.createElement('style');
+ style.type = 'text/css';
+ style.innerHTML = file;
+ head.appendChild(style);
+ });
const appwindow = require('electron').remote.getCurrentWindow();
@@ -56,9 +72,16 @@ require('electron').remote.getGlobal('setTimeout')(() => {
// maximise
element = document.createElement('button');
element.classList.add('window-buttons');
- element.innerHTML = '▢';
- element.onclick = () =>
- appwindow.isMaximized() ? appwindow.unmaximize() : appwindow.maximize();
+ element.innerHTML = appwindow.isMaximized() ? '🗗' : '🗖';
+ element.onclick = function () {
+ if (appwindow.isMaximized()) {
+ appwindow.unmaximize();
+ this.innerHTML = '🗖';
+ } else {
+ appwindow.maximize();
+ this.innerHTML = '🗗';
+ }
+ };
node.appendChild(element);
// close
@@ -67,14 +90,8 @@ require('electron').remote.getGlobal('setTimeout')(() => {
element.classList.add('window-buttons');
element.innerHTML = '⨉';
element.onclick = () => {
- const store = new (require(path.join(__dirname, '..', 'store.js')))({
- config: 'user-preferences',
- defaults: {
- tray: false,
- },
- });
if (
- store.get('tray') &&
+ store.tray &&
require('electron').remote.BrowserWindow.getAllWindows().length === 1
) {
appwindow.hide();
@@ -82,8 +99,6 @@ require('electron').remote.getGlobal('setTimeout')(() => {
};
node.appendChild(element);
- clearInterval(intervalID);
-
/* reload window */
document.defaultView.addEventListener(
'keyup',
diff --git a/resources/store.js b/resources/store.js
index f2bb378..4300c7a 100644
--- a/resources/store.js
+++ b/resources/store.js
@@ -10,26 +10,34 @@
const path = require('path'),
fs = require('fs');
-class Store {
- constructor(opts) {
- this.path = path.join(__dirname, opts.config + '.json');
- this.data = parseDataFile(this.path, opts.defaults);
- }
- get(key) {
- return this.data[key];
- }
- set(key, val) {
- this.data[key] = val;
- fs.writeFileSync(this.path, JSON.stringify(this.data));
- }
-}
-
-function parseDataFile(path, defaults) {
+function getJSON(from) {
try {
- return JSON.parse(fs.readFileSync(path));
- } catch (error) {
- return defaults;
+ return JSON.parse(fs.readFileSync(from));
+ } catch {
+ return {};
}
}
-module.exports = Store;
+module.exports = (opts) => {
+ opts = {
+ config: 'user-preferences',
+ defaults: {},
+ ...opts,
+ };
+ const config = path.join(__dirname, opts.config + '.json');
+ return new Proxy(
+ {},
+ {
+ get(obj, prop) {
+ obj = { ...opts.defaults, ...getJSON(config) };
+ return obj[prop];
+ },
+ set(obj, prop, val) {
+ obj = { ...opts.defaults, ...getJSON(config) };
+ obj[prop] = val;
+ fs.writeFileSync(config, JSON.stringify(obj));
+ return true;
+ },
+ }
+ );
+};
diff --git a/resources/theme.css b/resources/theme.css
new file mode 100644
index 0000000..b1ca2d5
--- /dev/null
+++ b/resources/theme.css
@@ -0,0 +1,159 @@
+/*
+ * notion-enhancer
+ * (c) 2020 dragonwocky
+ * (c) 2020 TarasokUA
+ * (https://dragonwocky.me/) under the MIT license
+ */
+
+:root {
+ /** backgrounds **/
+ --theme-main: rgb(10, 10, 10);
+ --theme-sidebar: rgb(0, 0, 0);
+ --theme-primary: rgb(177, 24, 24);
+ --theme-primary_hover: rgb(202, 26, 26);
+ --theme-primary_click: rgb(219, 41, 41);
+ --theme-primary_indicator: rgb(202, 26, 26);
+
+ /** databases **/
+ --theme-card: rgb(4, 4, 4);
+ --theme-card_border: rgba(59, 59, 59, 0.7);
+
+ /** colours **/
+ --theme-text_gray: rgba(151, 154, 155, 0.95);
+ --theme-text_brown: rgb(147, 114, 100);
+ --theme-text_orange: rgb(255, 163, 68);
+ --theme-text_yellow: rgb(255, 220, 73);
+ --theme-text_teal: rgb(77, 171, 154);
+ --theme-text_blue: rgb(82, 156, 202);
+ --theme-text_purple: rgb(154, 109, 215);
+ --theme-text_pink: rgb(226, 85, 161);
+ --theme-text_red: rgb(218, 47, 35);
+ --theme-bg_default: rgba(37, 37, 37, 0.815);
+ --theme-bg_gray: rgba(126, 128, 129, 0.5);
+ --theme-bg_brown: #50331f;
+ --theme-bg_orange: rgba(255, 155, 0, 0.58);
+ --theme-bg_yellow: rgba(183, 155, 0, 1);
+ --theme-bg_green: rgb(77, 171, 154);
+ --theme-bg_blue: rgba(0, 90, 146, 0.71);
+ --theme-bg_purple: rgba(91, 49, 148, 0.74);
+ --theme-bg_pink: rgba(243, 61, 159, 0.5);
+ --theme-bg_red: rgb(122, 20, 20);
+}
+
+.notion-dark-theme .window-buttons:hover {
+ background: rgb(23, 23, 23);
+}
+
+/** backgrounds **/
+
+[style*='background: rgb(55, 60, 63)'] {
+ background: var(--theme-sidebar) !important;
+}
+.notion-dark-theme .window-buttons,
+.notion-body.dark,
+[style*='background: rgb(47, 52, 55)'] {
+ background: var(--theme-main) !important;
+}
+
+/** databases **/
+[style*='box-shadow: rgb(47, 52, 55) -3px 0px 0px'] {
+ box-shadow: none !important;
+}
+[style*='background: rgba(202, 204, 206, 0.3)'] {
+ background: rgba(119, 119, 119, 0.3) !important;
+}
+.notion-table-view > :first-child > :first-child,
+[style*='background: rgb(71, 76, 80)'],
+[style*='background: rgb(98, 102, 104)'] {
+ background: rgb(15, 15, 15) !important;
+ box-shadow: 0 0 0 0.5px rgba(78, 78, 78, 0.7);
+}
+[style*='background: rgb(63, 68, 71)'] {
+ background: var(--theme-card) !important;
+}
+.notion-frame [style*='background: rgb(63, 68, 71)'] {
+ border: 0.5px solid var(--theme-card_border);
+}
+
+/** colours **/
+
+[style*='background: rgb(46, 170, 220)'] {
+ background: var(--theme-primary) !important;
+}
+[style*='background: rgb(6, 156, 205)'] {
+ background: var(--theme-primary-hover) !important;
+}
+[style*='background: rgb(0, 141, 190)'] {
+ background: var(--theme-primary_click) !important;
+}
+[style*='background: rgb(235, 87, 87)'] {
+ background: var(--theme-primary-indicator) !important;
+}
+
+[style*='background: rgb(80, 85, 88)'] {
+ background: var(--theme-bg_default) !important;
+}
+
+[style*='background: rgba(151, 154, 155, 0.5)'],
+[style*='background:rgb(69,75,78)'] {
+ background: var(--theme-bg_gray) !important;
+}
+[style*='color:rgba(151,154,155,0.95)'] {
+ color: var(--theme-text_gray) !important;
+}
+[style*='background: rgba(147, 114, 100, 0.5)'],
+[style*='background:rgb(67,64,64)'] {
+ background: var(--theme-bg_brown) !important;
+}
+[style*='color:rgb(147,114,100)'] {
+ color: var(--theme-text_brown) !important;
+}
+[style*='background: rgba(255, 163, 68, 0.5)'],
+[style*='background:rgb(89,74,58)'] {
+ background: var(--theme-bg_orange) !important;
+}
+[style*='color:rgb(255,163,68)'] {
+ color: var(--theme-text_orange) !important;
+}
+[style*='background: rgba(255, 220, 73, 0.5)'],
+[style*='background:rgb(89,86,59)'] {
+ background: var(--theme-bg_yellow) !important;
+}
+[style*='color:rgb(255,220,73)'] {
+ color: var(--theme-text_yellow) !important;
+}
+[style*='background: rgba(77, 171, 154, 0.5)'],
+[style*='background:rgb(53,76,75)'] {
+ background: var(--theme-bg_green) !important;
+}
+[style*='color:rgb(77,171,154)'] {
+ color: var(--theme-text_green) !important;
+}
+[style*='background: rgba(82, 156, 202, 0.5)'],
+[style*='background:rgb(54,73,84)'] {
+ background: var(--theme-bg_blue) !important;
+}
+[style*='color:rgb(82,156,202)'] {
+ color: var(--theme-text_blue) !important;
+}
+[style*='background: rgba(154, 109, 215, 0.5)'],
+[style*='background:rgb(68,63,87)'] {
+ background: var(--theme-bg_purple) !important;
+}
+[style*='color:rgb(154,109,215)'] {
+ color: var(--theme-text_purple) !important;
+}
+[style*='background: rgba(226, 85, 161, 0.5)'],
+[style*='background:rgb(83,59,76)'] {
+ background: var(--theme-bg_pink) !important;
+}
+[style*='color:rgb(226,85,161)'] {
+ color: var(--theme-text_pink) !important;
+}
+[style*='background: rgba(255, 115, 105, 0.5);'],
+[style*='background:rgb(89,65,65)'] {
+ background: var(--theme-bg_red) !important;
+}
+[style*='color:rgb(255,115,105)'] {
+ color: var(--theme-text_red) !important;
+}
diff --git a/resources/hotkey.js b/resources/tray.js
similarity index 63%
rename from resources/hotkey.js
rename to resources/tray.js
index 519da91..3dd046e 100644
--- a/resources/hotkey.js
+++ b/resources/tray.js
@@ -17,28 +17,22 @@ let tray;
function enhancements() {
const { Tray, Menu } = require('electron'),
path = require('path'),
- store = new (require(path.join(__dirname, '..', 'store.js')))({
+ store = require(path.join(__dirname, '..', 'store.js'))({
config: 'user-preferences',
defaults: {
openhidden: false,
maximised: false,
tray: false,
+ theme: false,
},
- }),
- states = {
- startup: electron_1.app.getLoginItemSettings().openAtLogin,
- openhidden: store.get('openhidden'),
- maximised: store.get('maximised'),
- tray: store.get('tray'),
- };
-
- tray = new Tray(path.join(__dirname, './notion.ico'));
+ });
+ tray = new Tray(path.join(__dirname, 'notion.ico'));
const contextMenu = Menu.buildFromTemplate([
{
id: 'startup',
label: 'run on startup',
type: 'checkbox',
- checked: states.startup,
+ checked: electron_1.app.getLoginItemSettings().openAtLogin,
click: () =>
contextMenu.getMenuItemById('startup').checked
? electron_1.app.setLoginItemSettings({ openAtLogin: true })
@@ -48,31 +42,43 @@ function enhancements() {
id: 'openhidden',
label: 'hide on open',
type: 'checkbox',
- checked: states.openhidden,
+ checked: store.openhidden,
click: () =>
contextMenu.getMenuItemById('openhidden').checked
- ? store.set('openhidden', true)
- : store.set('openhidden', false),
+ ? (store.openhidden = true)
+ : (store.openhidden = false),
},
{
id: 'maximised',
label: 'open maximised',
type: 'checkbox',
- checked: states.maximised,
+ checked: store.maximised,
click: () =>
contextMenu.getMenuItemById('maximised').checked
- ? store.set('maximised', true)
- : store.set('maximised', false),
+ ? (store.maximised = true)
+ : (store.maximised = false),
},
{
id: 'tray',
label: 'close to tray',
type: 'checkbox',
- checked: states.tray,
+ checked: store.tray,
click: () =>
contextMenu.getMenuItemById('tray').checked
- ? store.set('tray', true)
- : store.set('tray', false),
+ ? (store.tray = true)
+ : (store.tray = false),
+ },
+ {
+ id: 'theme',
+ label: 'load theme.css',
+ type: 'checkbox',
+ checked: store.theme,
+ click: () => {
+ contextMenu.getMenuItemById('theme').checked
+ ? (store.theme = true)
+ : (store.theme = false);
+ electron_1.BrowserWindow.getAllWindows().forEach((win) => win.reload());
+ },
},
{
type: 'separator',
@@ -86,22 +92,18 @@ function enhancements() {
tray.on('click', function () {
const win = electron_1.BrowserWindow.getAllWindows()[0];
- if (win.isVisible()) {
- if (win.isMinimized()) {
- win.show();
- } else win.hide();
- } else {
- if (contextMenu.getMenuItemById('maximised').checked) {
- win.maximize();
- } else win.show();
- }
+ if ((win.isVisible() && win.isMinimized()) || !store.maximised) {
+ win.show();
+ } else if (win.isVisible()) {
+ win.hide();
+ } else win.maximize();
});
- const hotkey = '___hotkey___'; // will be set by python script
+ const hotkey = '☃☃☃hotkey☃☃☃'; // will be set by python script
electron_1.globalShortcut.register(hotkey, () => {
const windows = electron_1.BrowserWindow.getAllWindows();
if (windows.some((win) => !win.isVisible())) {
- if (contextMenu.getMenuItemById('maximised').checked) {
+ if (store.maximised) {
windows.forEach((win) => win.maximize());
} else windows.forEach((win) => win.show());
} else windows.forEach((win) => win.hide());