taken control of createWindow

This commit is contained in:
dragonwocky 2020-07-19 16:24:34 +10:00
parent d423bdbfd0
commit 21facae40c
Signed by: dragonwocky
GPG Key ID: C7A48B7846AA706D
20 changed files with 170 additions and 57 deletions

View File

@ -6,20 +6,19 @@ these guidelines are designed for smooth communication, management and developme
following them shows respect to the developer/s spending their free time on it, and makes it easiest for them to improve the tool. following them shows respect to the developer/s spending their free time on it, and makes it easiest for them to improve the tool.
**found a bug / something isn't working as expected?** create a **found a bug / something isn't working as expected?** create a
[bug report](https://github.com/dragonwocky/notion-enhancer/issues/new?assignees=&labels=bug&template=bug-report.md&title=). [bug report](https://github.com/dragonwocky/notion-enhancer/issues/new?labels=bug&template=bug-report.md).
> SECURITY ISSUE? (e.g. PERSONAL/NOTION DATA BEING INTERFERED WITH) > SECURITY ISSUE? (e.g. PERSONAL/NOTION DATA BEING INTERFERED WITH)
> EMAIL ME INSTEAD: [thedragonring.bod@gmail.com](mailto:thedragonring.bod@gmail.com) > EMAIL ME INSTEAD: [thedragonring.bod@gmail.com](mailto:thedragonring.bod@gmail.com)
**have a cool new feature idea / there's something you just wish you could do?** submit a **have a cool new feature idea / there's something you just wish you could do?** submit a
[feature request](https://github.com/dragonwocky/notion-enhancer/issues/new?assignees=&labels=enhancement&template=feature-request.md&title=). [feature request](https://github.com/dragonwocky/notion-enhancer/issues/new?labels=enhancement&template=feature-request.md).
> enhancements are applied only locally. > enhancements are applied only locally -
> features should be designed only to improve the user experience - > features should be designed only to improve the user experience.
> affecting the way notion internals work is against their ToS.
**using a not-yet-supported operating system or notion installation?** ask for **using a not-yet-supported operating system or notion installation?** ask for
[platform support](https://github.com/dragonwocky/notion-enhancer/issues/new?assignees=&labels=enhancement&template=platform-support.md&title=). [platform support](https://github.com/dragonwocky/notion-enhancer/issues/new?labels=enhancement&template=platform-support.md).
> mobile clients cannot currently be modded. > mobile clients cannot currently be modded.

View File

@ -84,10 +84,14 @@ module.exports = {
#### hacks #### hacks
each "hack" is a function taking a single **`store`** argument, which allows each "hack" is a function taking 2 arguments.
access to the module settings/options defined in `mod.js` (those set in the menu,
or used internally by the module). each module store is saved to + automatically syncs with `~/.notion-enhancer/id.json`. 1. the **`store`** argument, which allows access to the module
it can be initialised with `const data = store({ defaults })`, then used as if it were a normal object. settings/options defined in `mod.js` (those set in the menu, or used internally by the module).
each module store is saved to + automatically syncs with `~/.notion-enhancer/id.json`.
it can be initialised with `const settings = store({ defaults })`, then used as if it were a normal object.
2. the **`__exports`** argument, which is the `module.exports` of the file being modded.
this can be used to call or replace functions from notion.
this hack is applied to whichever file (`.js`-only) is set as the function key. these can be found within the `app` folder. this hack is applied to whichever file (`.js`-only) is set as the function key. these can be found within the `app` folder.
@ -104,11 +108,11 @@ e.g.
```js ```js
// sayhi.js // sayhi.js
module.exports = function (store) { module.exports = function (store, __exports) {
document.addEventListener('readystatechange', (event) => { document.addEventListener('readystatechange', (event) => {
if (document.readyState !== 'complete') return false; if (document.readyState !== 'complete') return false;
const data = store({ name: 'dragonwocky' }); const settings = store({ name: 'dragonwocky' });
console.log(data.name); console.log(settings.name);
}); });
}; };
// mod.js // mod.js

View File

@ -48,7 +48,7 @@ want to contribute? check the the [contribution guidelines](CONTRIBUTING.md).
(it can also be run from the wsl to apply enhancements to the windows app.) (it can also be run from the wsl to apply enhancements to the windows app.)
**using a not-yet-supported operating system or notion installation?** ask for **using a not-yet-supported operating system or notion installation?** ask for
[platform support](https://github.com/dragonwocky/notion-enhancer/issues/new?assignees=&labels=enhancement&template=platform-support.md&title=). [platform support](https://github.com/dragonwocky/notion-enhancer/issues/new?labels=enhancement&template=platform-support.md).
mobile clients are not supported and due to system limitations/restrictions cannot be. mobile clients are not supported and due to system limitations/restrictions cannot be.

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="250" height="250" fill="none" viewBox="0 0 250 250"><path fill="#000" d="M124.859 234.52L67.5474 135.736H102.683V12.184H147.323V135.736H182.459L124.859 234.52Z"/></svg>

After

Width:  |  Height:  |  Size: 215 B

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="250" height="250" fill="none" viewBox="0 0 250 250"><path fill="#000" d="M102.683 234.52V110.968H67.5474L124.859 12.184L182.459 110.968H147.323V234.52H102.683Z"/></svg>

After

Width:  |  Height:  |  Size: 215 B

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="250" height="250" fill="none" viewBox="0 0 250 250"><line x1="21.393" x2="233.525" y1="229.525" y2="17.393" stroke="#000" stroke-miterlimit="4.139" stroke-width="30"/><line x1="17.607" x2="229.739" y1="17.393" y2="229.525" stroke="#000" stroke-linejoin="round" stroke-width="30"/></svg>

After

Width:  |  Height:  |  Size: 333 B

View File

Before

Width:  |  Height:  |  Size: 7.0 KiB

After

Width:  |  Height:  |  Size: 7.0 KiB

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="250" height="250" fill="none" viewBox="0 0 250 250"><path fill="#000" d="M14.7346 227.26V7.03998H235.215V227.26H14.7346ZM46.4546 195.8H203.495V70.48H46.4546V195.8Z"/></svg>

After

Width:  |  Height:  |  Size: 219 B

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="250" height="250" fill="none" viewBox="0 0 250 250"><path fill="#000" d="M16.1311 225.96V76.72H84.5111V8.07999H233.751V157.32H165.371V225.96H16.1311ZM110.771 53.58V76.72H165.371V131.32H207.491V53.58H110.771ZM42.3911 199.96H139.111V122.22H42.3911V199.96Z"/></svg>

After

Width:  |  Height:  |  Size: 309 B

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="250" height="250" fill="none" viewBox="0 0 250 250"><path fill="#000" d="M17.8021 138.04V106.072H232.074V138.04H17.8021Z"/></svg>

After

Width:  |  Height:  |  Size: 176 B

View File

Before

Width:  |  Height:  |  Size: 110 KiB

After

Width:  |  Height:  |  Size: 110 KiB

View File

@ -7,7 +7,8 @@
const defaults = { const defaults = {
openhidden: false, openhidden: false,
maximized: false, maximized: false,
tray: false, close_to_tray: false,
frameless: true,
hotkey: 'CmdOrCtrl+Shift+A', hotkey: 'CmdOrCtrl+Shift+A',
}; };
@ -24,10 +25,13 @@ module.exports = {
options: [], options: [],
hacks: { hacks: {
'main/main.js': require('./tray.js')(defaults), 'main/main.js': require('./tray.js')(defaults),
'renderer/preload.js': function (store) { 'main/createWindow.js': require('./window.js')(defaults),
const data = store({ name: 'dragonwocky' }); 'renderer/preload.js': function (store, __exports) {
console.log(data.name); const window = require('electron').remote.getCurrentWindow();
data.name = 'tom'; document.defaultView.addEventListener('keyup', (event) => {
if (event.code === 'F5') window.reload();
// if (event.code === 'F4' && event.altKey) window.close();
});
}, },
}, },
}; };

View File

@ -8,7 +8,7 @@
let tray; let tray;
module.exports = (defaults) => module.exports = (defaults) =>
function (store) { function (store, __exports) {
const electron = require('electron'), const electron = require('electron'),
path = require('path'), path = require('path'),
is_mac = process.platform === 'darwin', is_mac = process.platform === 'darwin',
@ -18,9 +18,9 @@ module.exports = (defaults) =>
electron.app.on('ready', () => { electron.app.on('ready', () => {
tray = new electron.Tray( tray = new electron.Tray(
is_win is_win
? path.normalize(`${__dirname}/windows.ico`) ? path.resolve(`${__dirname}/icons/windows.ico`)
: new electron.nativeImage.createFromPath( : new electron.nativeImage.createFromPath(
path.normalize(`${__dirname}/mac+linux.png`) path.resolve(`${__dirname}/icons/mac+linux.png`)
).resize({ ).resize({
width: 16, width: 16,
height: 16, height: 16,
@ -31,10 +31,20 @@ module.exports = (defaults) =>
{ {
type: 'normal', type: 'normal',
label: 'Bug Report', label: 'Bug Report',
click: () => {
electron.shell.openExternal(
'https://github.com/dragonwocky/notion-enhancer/issues/new?labels=bug&template=bug-report.md'
);
},
}, },
{ {
type: 'normal', type: 'normal',
label: 'Feature Request', label: 'Feature Request',
click: () => {
electron.shell.openExternal(
'https://github.com/dragonwocky/notion-enhancer/issues/new?labels=enhancement&template=feature-request.md'
);
},
}, },
{ {
type: 'separator', type: 'separator',
@ -42,6 +52,11 @@ module.exports = (defaults) =>
{ {
type: 'normal', type: 'normal',
label: 'Docs', label: 'Docs',
click: () => {
electron.shell.openExternal(
'https://github.com/dragonwocky/notion-enhancer/tree/js'
);
},
}, },
{ {
type: 'normal', type: 'normal',
@ -70,6 +85,7 @@ module.exports = (defaults) =>
windows.forEach((win) => [win.isFocused() && win.blur(), win.hide()]); windows.forEach((win) => [win.isFocused() && win.blur(), win.hide()]);
if (is_mac) electron.app.hide(); if (is_mac) electron.app.hide();
} }
tray.on('click', () => { tray.on('click', () => {
const windows = electron.BrowserWindow.getAllWindows(); const windows = electron.BrowserWindow.getAllWindows();
if (windows.some((win) => win.isVisible())) hideWindows(); if (windows.some((win) => win.isVisible())) hideWindows();

85
mods/core/window.js Normal file
View File

@ -0,0 +1,85 @@
/*
* notion-enhancer
* (c) 2020 dragonwocky <thedragonring.bod@gmail.com>
* (c) 2020 TarasokUA
* (https://dragonwocky.me/) under the MIT license
*/
module.exports = (defaults) =>
function (store, __exports) {
const electron = require('electron'),
allWindows = electron.BrowserWindow.getAllWindows,
createWindow = __exports.createWindow,
path = require('path'),
is_mac = process.platform === 'darwin',
settings = store(defaults),
helpers = require('../../pkg/helpers.js'),
__notion = helpers.getNotion();
__exports.createWindow = function (relativeUrl) {
if (!relativeUrl) relativeUrl = '';
const window_state = require(`${__notion.replace(
/\\/g,
'/'
)}/app/node_modules/electron-window-state/index.js`)({
defaultWidth: 1320,
defaultHeight: 860,
}),
rect = {
x: window_state.x,
y: window_state.y,
width: window_state.width,
height: window_state.height,
},
focused_window = electron.BrowserWindow.getFocusedWindow();
if (focused_window && !focused_window.isMaximized()) {
rect.x = focused_window.getPosition()[0] + 20;
rect.y = focused_window.getPosition()[1] + 20;
rect.width = focused_window.getSize()[0];
rect.height = focused_window.getSize()[1];
}
const window = new electron.BrowserWindow({
show: false,
backgroundColor: '#ffffff',
titleBarStyle: 'hiddenInset',
frame: !settings.frameless,
webPreferences: {
preload: path.resolve(`${__notion}/app/renderer/index.js`),
webviewTag: true,
session: electron.session.fromPartition('persist:notion'),
},
...rect,
});
window.once('ready-to-show', function () {
if (
!settings.openhidden ||
allWindows().some((win) => win.isVisible() && win.id != window.id)
) {
window.show();
if (settings.maximized) window.maximize();
if (
(focused_window && focused_window.isFullScreen()) ||
window_state.isFullScreen
)
window.setFullScreen(true);
}
});
let intended_quit = false;
window.on('close', (e) => {
if (
intended_quit ||
!settings.close_to_tray ||
allWindows().length > 1
) {
window_state.saveState(window);
window = null;
} else {
e.preventDefault();
window.hide();
}
});
electron.app.on('before-quit', () => (intended_quit = true));
window.loadURL(__exports.getIndexUrl(relativeUrl));
return window;
};
};

View File

@ -56,16 +56,14 @@ module.exports = async function ({ overwrite_version } = {}) {
}); });
} }
console.info(' ...unpacking app.asar'); console.info(' ...unpacking app.asar');
const asar_app = path.normalize(`${__notion}/app.asar`), const asar_app = path.resolve(`${__notion}/app.asar`),
asar_exec = path.normalize( asar_exec = path.resolve(`${__dirname}/../node_modules/asar/bin/asar.js`);
`${__dirname}/../node_modules/asar/bin/asar.js`
);
await promisify(exec)( await promisify(exec)(
`"${asar_exec}" extract "${asar_app}" "${path.normalize( `"${asar_exec}" extract "${asar_app}" "${path.resolve(
`${__notion}/app` `${__notion}/app`
)}"` )}"`
); );
fs.move(asar_app, path.normalize(`${__notion}/app.asar.bak`)); fs.move(asar_app, path.resolve(`${__notion}/app.asar.bak`));
// patching launch script target of custom wrappers // patching launch script target of custom wrappers
if ( if (
@ -94,31 +92,28 @@ module.exports = async function ({ overwrite_version } = {}) {
} }
for await (let insertion_target of readdirIterator( for await (let insertion_target of readdirIterator(
path.normalize(`${__notion}/app`), path.resolve(`${__notion}/app`),
{ {
deep: (stats) => stats.path.indexOf('node_modules') === -1, deep: (stats) => stats.path.indexOf('node_modules') === -1,
filter: (stats) => stats.isFile() && stats.path.endsWith('.js'), filter: (stats) => stats.isFile() && stats.path.endsWith('.js'),
} }
)) { )) {
insertion_target = path.normalize(`${__notion}/app/${insertion_target}`); insertion_target = path.resolve(`${__notion}/app/${insertion_target}`);
fs.appendFile( fs.appendFile(
insertion_target, insertion_target,
`\n\n//notion-enhancer\nrequire('${helpers.realpath( `\n\n//notion-enhancer\nrequire('${helpers.realpath(
__dirname __dirname
)}/loader.js')(__filename);` )}/loader.js')(__filename, exports);`
); );
} }
// not resolved, nothing depends on it so it's just a "let it do its thing" // not resolved, nothing depends on it so it's just a "let it do its thing"
console.info(' ...recording enhancement version.'); console.info(' ...recording enhancement version.');
fs.outputFile( fs.outputFile(
path.normalize(`${__notion}/app/ENHANCER_VERSION.txt`), path.resolve(`${__notion}/app/ENHANCER_VERSION.txt`),
version
);
fs.outputFile(
path.normalize(`${helpers.data_folder}/version.txt`),
version version
); );
fs.outputFile(path.resolve(`${helpers.data_folder}/version.txt`), version);
console.info(' ~~ success.'); console.info(' ~~ success.');
return true; return true;

View File

@ -14,11 +14,11 @@ const fs = require('fs-extra'),
let __notion = helpers.getNotion(); let __notion = helpers.getNotion();
module.exports = async function () { module.exports = async function () {
const version_path = path.normalize(`${__notion}/app/ENHANCER_VERSION.txt`), const version_path = path.resolve(`${__notion}/app/ENHANCER_VERSION.txt`),
installed_version = (await fs.pathExists(version_path)) installed_version = (await fs.pathExists(version_path))
? await fs.readFile(version_path, 'utf8') ? await fs.readFile(version_path, 'utf8')
: '?.?.?'; : '?.?.?';
if (await fs.pathExists(path.normalize(`${__notion}/app.asar`))) { if (await fs.pathExists(path.resolve(`${__notion}/app.asar`))) {
return { return {
msg: `notion-enhancer has not been applied.`, msg: `notion-enhancer has not been applied.`,
code: 0, code: 0,

View File

@ -25,7 +25,7 @@ const is_wsl =
process.platform === 'linux' && process.platform === 'linux' &&
os.release().toLowerCase().includes('microsoft'), os.release().toLowerCase().includes('microsoft'),
// ~/.notion-enhancer absolute path. // ~/.notion-enhancer absolute path.
data_folder = path.normalize( data_folder = path.resolve(
`${ `${
is_wsl is_wsl
? (() => { ? (() => {
@ -86,19 +86,19 @@ function getNotion() {
if (!folder) if (!folder)
throw new EnhancerError( throw new EnhancerError(
'platform not supported: open a request in the github repo:\n' + 'platform not supported: open a request in the github repo:\n' +
'https://github.com/dragonwocky/notion-enhancer/issues/new?assignees=&labels=enhancement&template=platform-support.md' 'https://github.com/dragonwocky/notion-enhancer/issues/new?labels=enhancement&template=platform-support.md'
); );
// check if actual app files are present. // check if actual app files are present.
// if app/app.asar are missing but app.asar.bak present it will be moved to app.asar // if app/app.asar are missing but app.asar.bak present it will be moved to app.asar
const app_asar = path.normalize(`${folder}/app.asar`); const app_asar = path.resolve(`${folder}/app.asar`);
if ( if (
!( !(
fs.pathExistsSync(folder) && fs.pathExistsSync(folder) &&
(fs.pathExistsSync(app_asar) || (fs.pathExistsSync(app_asar) ||
fs.pathExistsSync(path.normalize(`${folder}/app`))) fs.pathExistsSync(path.resolve(`${folder}/app`)))
) )
) { ) {
const asar_bak = path.normalize(`${__notion}/app.asar.bak`); const asar_bak = path.resolve(`${__notion}/app.asar.bak`);
if (fs.pathExistsSync(asar_bak)) { if (fs.pathExistsSync(asar_bak)) {
fs.moveSync(asar_bak, app_asar); fs.moveSync(asar_bak, app_asar);
} else } else

View File

@ -11,13 +11,13 @@ const fs = require('fs-extra'),
store = require('./store.js'); store = require('./store.js');
let __notion = helpers.getNotion(); let __notion = helpers.getNotion();
module.exports = function (__file) { module.exports = function (__file, __exports) {
__file = __file __file = __file
.slice(path.normalize(`${__notion}/app`).length + 1) .slice(path.resolve(`${__notion}/app`).length + 1)
.replace(/\\/g, '/'); .replace(/\\/g, '/');
const modules = { const modules = {
source: fs.readdirSync(path.normalize(`${__dirname}/../mods`)), source: fs.readdirSync(path.resolve(`${__dirname}/../mods`)),
invalid: [], invalid: [],
loaded: [], loaded: [],
}; };
@ -34,13 +34,17 @@ module.exports = function (__file) {
throw Error; throw Error;
if (mod.type === 'core' || store('mods', { [mod.id]: false })[mod.id]) { if (mod.type === 'core' || store('mods', { [mod.id]: false })[mod.id]) {
if (mod.hacks && mod.hacks[__file]) if (mod.hacks && mod.hacks[__file])
mod.hacks[__file]((...args) => mod.hacks[__file](
args.length === 1 ? store(mod.id, args[0]) : store(args[0], args[1]) (...args) =>
args.length === 1
? store(mod.id, args[0])
: store(args[0], args[1]),
__exports
); );
if ( if (
__file === 'renderer/preload.js' && __file === 'renderer/preload.js' &&
fs.pathExistsSync( fs.pathExistsSync(
path.normalize(`${__dirname}/../mods/${dir}/styles.css`) path.resolve(`${__dirname}/../mods/${dir}/styles.css`)
) )
) { ) {
document.addEventListener('readystatechange', (event) => { document.addEventListener('readystatechange', (event) => {
@ -63,7 +67,7 @@ module.exports = function (__file) {
.session.fromPartition('persist:notion') .session.fromPartition('persist:notion')
.protocol.registerFileProtocol('enhancement', (req, callback) => { .protocol.registerFileProtocol('enhancement', (req, callback) => {
callback({ callback({
path: path.normalize( path: path.resolve(
`${__dirname}/../mods/${req.url.slice('enhancement://'.length)}` `${__dirname}/../mods/${req.url.slice('enhancement://'.length)}`
), ),
}); });

View File

@ -23,18 +23,18 @@ module.exports = async function ({ overwrite_asar, delete_data } = {}) {
const file_operations = []; const file_operations = [];
// extracted asar: modded // extracted asar: modded
const app_folder = path.normalize(`${__notion}/app`); const app_folder = path.resolve(`${__notion}/app`);
if (await fs.pathExists(app_folder)) { if (await fs.pathExists(app_folder)) {
console.info(` ...removing folder ${app_folder}`); console.info(` ...removing folder ${app_folder}`);
file_operations.push(fs.remove(app_folder)); file_operations.push(fs.remove(app_folder));
} else console.warn(` * ${app_folder} not found: step skipped.`); } else console.warn(` * ${app_folder} not found: step skipped.`);
// restoring original asar // restoring original asar
const asar_bak = path.normalize(`${__notion}/app.asar.bak`); const asar_bak = path.resolve(`${__notion}/app.asar.bak`);
if (await fs.pathExists(asar_bak)) { if (await fs.pathExists(asar_bak)) {
console.info(' ...moving asar.app.bak to app.asar'); console.info(' ...moving asar.app.bak to app.asar');
if (await fs.pathExists(path.normalize(`${__notion}/app.asar`))) { if (await fs.pathExists(path.resolve(`${__notion}/app.asar`))) {
console.warn(' * app.asar already exists!'); console.warn(' * app.asar already exists!');
if (overwrite_asar === undefined) { if (overwrite_asar === undefined) {
do { do {
@ -56,7 +56,7 @@ module.exports = async function ({ overwrite_asar, delete_data } = {}) {
file_operations.push( file_operations.push(
overwrite_asar || overwrite_asar === undefined overwrite_asar || overwrite_asar === undefined
? fs.move(asar_bak, path.normalize(`${__notion}/app.asar`), { ? fs.move(asar_bak, path.resolve(`${__notion}/app.asar`), {
overwrite: true, overwrite: true,
}) })
: fs.remove(asar_bak) : fs.remove(asar_bak)
@ -83,7 +83,7 @@ module.exports = async function ({ overwrite_asar, delete_data } = {}) {
); );
if (delete_data) { if (delete_data) {
file_operations.push(fs.remove(helpers.data_folder)); file_operations.push(fs.remove(helpers.data_folder));
} else fs.remove(path.normalize(`${helpers.data_folder}/version.txt`)); } else fs.remove(path.resolve(`${helpers.data_folder}/version.txt`));
} else console.warn(` * ${helpers.data_folder} not found: step skipped.`); } else console.warn(` * ${helpers.data_folder} not found: step skipped.`);
await Promise.all(file_operations); await Promise.all(file_operations);

View File

@ -11,11 +11,11 @@ const path = require('path'),
// a wrapper for accessing data stored in a JSON file. // a wrapper for accessing data stored in a JSON file.
module.exports = (namespace, defaults = {}) => { module.exports = (namespace, defaults = {}) => {
namespace = path.normalize(`${data_folder}/${namespace}.json`); namespace = path.resolve(`${data_folder}/${namespace}.json`);
fs.ensureDirSync(data_folder); fs.ensureDirSync(data_folder);
const getData = () => ({ ...defaults, ...getJSON(namespace) }); const getData = () => ({ ...defaults, ...getJSON(namespace) });
fs.writeJsonSync(namespace, getData()); // fs.writeJsonSync(namespace, getData());
return new Proxy(defaults, { return new Proxy(defaults, {
get(obj, prop) { get(obj, prop) {