# notion-enhancer # (c) 2020 dragonwocky # (c) 2020 TarasokUA # (https://dragonwocky.me/) under the MIT license import re import os import sys import platform import subprocess from shutil import copyfile, rmtree from time import sleep # to smooth the update process enhancer_version = '0.7.0' # for toggling notion visibility hotkey = 'CmdOrCtrl+Shift+A' # '=== title ===' = headers # '*' = information # '...' = actions # '##' = warnings # '>' = exit print('=== NOTION ENHANCER CUSTOMISATION LOG ===\n') try: filepath = '' __dirname__ = os.path.dirname(__file__) enhancer_folder = os.path.realpath(__dirname__) if 'microsoft' in platform.uname()[3].lower() and sys.platform == 'linux': filepath = '/mnt/c/' + \ subprocess.run( ['cmd.exe', '/c', 'echo', '%localappdata%'], stdout=subprocess.PIPE).stdout \ .rstrip().decode('utf-8')[3:].replace('\\', '/') + '/Programs/Notion/resources' drive = enhancer_folder[5].capitalize() if enhancer_folder.startswith( '/mnt/') else 'C' enhancer_folder = drive + ':\\' + enhancer_folder[6:] elif sys.platform == 'win32': filepath = subprocess.run(['echo', '%localappdata%'], shell=True, capture_output=True).stdout \ .rstrip().decode('utf-8') + '\\Programs\\Notion\\resources' elif sys.platform == 'linux': if os.path.exists('/opt/notion-app'): filepath = '/opt/notion-app' elif os.path.exists('/opt/notion'): filepath = '/opt/notion' elif os.path.exists('/usr/lib/notion-desktop/resources'): filepath = '/usr/lib/notion-desktop/resources' elif sys.platform == 'darwin': filepath = '/Applications/Notion.app/Contents/Resources' else: print(' > script not compatible with your os!\n (report this to dragonwocky#8449 on discord)') exit() if not filepath or not os.path.exists(filepath): print( ' > nothing found: exiting. notion install is either corrupted or non-existent.') exit() unpacking_asar = True if not os.path.isfile(os.path.join(filepath, 'app.asar')): print(f' ## file {os.path.join(filepath, "app.asar")} not found!') print(' * attempting to locate') if os.path.exists(os.path.join(filepath, 'app')): unpacking_asar = False print(' * app.asar was already unpacked: checking version.') cleaning_asar = True if os.path.isfile(os.path.join(filepath, 'app', 'ENHANCER_VERSION.txt')): with open(os.path.join(filepath, 'app', 'ENHANCER_VERSION.txt'), 'r', encoding='UTF-8') as content: if content.read() == enhancer_version: cleaning_asar = False if cleaning_asar: unpacking_asar = True print(' * version does not match: cleaning.') if os.path.exists(os.path.join(filepath, 'app')): print( f' ...removing folder {os.path.join(filepath, "app")}') rmtree(os.path.join(filepath, 'app')) else: print( f' * {os.path.join(filepath, "app")} was not found: step skipped.') if os.path.isfile(os.path.join(filepath, 'app.asar.bak')): print(' ...renaming asar.app.bak to asar.app') os.rename(os.path.join(filepath, 'app.asar.bak'), os.path.join(filepath, 'app.asar')) else: print( f' * {os.path.join(filepath, "app.asar.bak")} was not found: exiting. notion install is corrupted.') exit() else: print(' * version matches: continuing.') else: print( ' > nothing found: exiting. notion install is either corrupted or non-existent.') exit() if unpacking_asar: print(' ...unpacking app.asar') subprocess.run(['asar', 'extract', os.path.join(filepath, 'app.asar'), os.path.join( filepath, 'app')], shell=(True if sys.platform == 'win32' else False)) print(' ...renaming asar.app to asar.app.bak') os.rename(os.path.join(filepath, 'app.asar'), os.path.join(filepath, 'app.asar.bak')) with open(os.path.join(filepath, 'app', 'ENHANCER_VERSION.txt'), 'w', encoding='UTF-8') as write: write.write(enhancer_version) if os.path.isfile(os.path.join(filepath, 'app', 'renderer', 'preload.js')): print( f' ...adding preload.js to {os.path.join(filepath, "app","renderer","preload.js")}') with open(os.path.join(filepath, 'app', 'renderer', 'preload.js'), 'r', encoding='UTF-8') as content: if '/* === INJECTION MARKER === */' in content.read(): print(' * preload.js already added. replacing it.') content.seek(0) original = [] for num, line in enumerate(content): if '/* === INJECTION MARKER === */' in line: break original += line with open(os.path.join(filepath, 'app', 'renderer', 'preload.js'), 'w', encoding='UTF-8') as write: write.writelines(original) else: with open(os.path.join(filepath, 'app', 'renderer', 'preload.js'), 'a', encoding='UTF-8') as append: append.write('\n\n') with open(os.path.join(filepath, 'app', 'renderer', 'preload.js'), 'a', encoding='UTF-8') as append: print( f' ...linking to {os.path.join(".", "resources", "user.css")}') with open(os.path.join(__dirname__, 'resources', 'preload.js'), 'r', encoding='UTF-8') as insert: append.write(insert.read().replace( '☃☃☃resources☃☃☃', os.path.join(enhancer_folder, 'resources').replace('\\', '/')) .replace('☃☃☃version☃☃☃', enhancer_version)) else: print( f' * {os.path.join(filepath, "app","renderer","preload.js")} was not found: step skipped.') if os.path.isfile(os.path.join(filepath, "app", "main", "createWindow.js")): with open(os.path.join(filepath, "app", "main", "createWindow.js"), 'r', encoding='UTF-8') as content: content = content.read() print( f' ...making window frameless @ {os.path.join(filepath, "app", "main", "createWindow.js")}') if '{ frame: false, show: false' not in content: content = content.replace( '{ show: false', '{ frame: false, show: false') print( f' ...adding "open hidden" capabilities to {os.path.join(filepath, "app", "main", "createWindow.js")}') content = re.sub('\\s*\\/\\* === INJECTION START === \\*\\/.*?\\/\\* === INJECTION END === \\*\\/\\s*', 'window.show()', content, flags=re.DOTALL).replace('window.show()', """ /* === INJECTION START === */ const path = require('path'), store = require(path.join(__dirname, '..', 'store.js'))({ config: 'user-preferences', defaults: { openhidden: false, maximized: false, tray: false, } }); if (!store.openhidden || electron_1.BrowserWindow.getAllWindows().some(win => win.isVisible())) { window.show(); if (store.maximized) window.maximize(); } let appQuit = false; window.on('close', (e) => { if (appQuit || !store.tray) window = null; else { e.preventDefault(); window.hide(); } }); electron_1.app.on('activate', () => window.show()); electron_1.app.on('before-quit', () => (appQuit = true)); /* === INJECTION END === */ """) with open(os.path.join(filepath, "app", "main", "createWindow.js"), 'w', encoding='UTF-8') as write: write.write(content) else: print( f' * {os.path.join(filepath, "app", "main", "createWindow.js")} was not found: step skipped.') if os.path.isfile(os.path.join(filepath, "app", "renderer", "index.js")): with open(os.path.join(filepath, "app", "renderer", "index.js"), 'r', encoding='UTF-8') as content: print( f' ...adjusting drag area for frameless window in {os.path.join(filepath, "app", "renderer", "index.js")}') content = content.read() loc = content.rfind('dragRegionStyle') content = content[:loc] + content[loc:] \ .replace('top: 0', 'top: 1', 1) \ .replace('height: 34', 'height: 10', 1) with open(os.path.join(filepath, "app", "renderer", "index.js"), 'w', encoding='UTF-8') as write: write.write(content) else: print( f' * {os.path.join(filepath, "app", "renderer", "index.js")} was not found: step skipped.') if os.path.isfile(os.path.join(filepath, "app", "main", "main.js")): with open(os.path.join(filepath, "app", "main", "main.js"), 'r', encoding='UTF-8') as content: print( f' ...adding tray support (inc. context menu with settings) to {os.path.join(filepath, "app", "main", "main.js")}') print( f' ...adding window toggle hotkey to {os.path.join(filepath, "app", "main", "main.js")}') content = content.read() with open(os.path.join(filepath, "app", "main", "main.js"), 'w', encoding='UTF-8') as write: if '/* === INJECTION MARKER === */' in content: print(' * tray.js already added. replacing it.') original = [] for line in content.splitlines(): if '/* === INJECTION MARKER === */' in line: break original.append(line) write.write('\n'.join(original)) else: write.write(content.replace( 'electron_1.app.on("ready", handleReady);', 'electron_1.app.on("ready", () => handleReady() && enhancements());') + '\n') with open(os.path.join(filepath, "app", "main", "main.js"), 'a', encoding='UTF-8') as append: with open(os.path.join(__dirname__, 'resources', 'tray.js'), 'r', encoding='UTF-8') as insert: append.write('\n' + insert.read().replace( '☃☃☃hotkey☃☃☃', hotkey).replace( '☃☃☃resources☃☃☃', os.path.join(enhancer_folder, 'resources').replace('\\', '/')) ) print( f' ...copying datastore wrapper {os.path.join(".", "resources", "store.js")} to {os.path.join(filepath, "app")}') copyfile(os.path.join(__dirname__, 'resources', 'store.js'), os.path.join(filepath, "app", "store.js")) else: print( f' * {os.path.join(filepath, "app", "main", "main.js")} was not found: step skipped.') if sys.platform == 'linux' and 'microsoft' not in platform.uname()[3].lower(): print( f' ...patching app launcher') subprocess.call( ['sed', '-i', r's/electron\ app\.asar/electron\ app/', '/usr/bin/notion-app' if os.path.exists('/usr/bin/notion-app') else '/usr/bin/notion']) print('\n>>> SUCCESSFULLY CUSTOMISED <<<') except Exception as e: print( f'\n### ERROR (report this to dragonwocky#8449 on discord) ###\n{str(e)}') print('\n=== END OF LOG ===')