/* * notion-enhancer * (c) 2020 dragonwocky * (https://dragonwocky.me/) under the MIT license */ 'use strict'; const store = require('../../pkg/store.js'), helpers = require('../../pkg/helpers.js'), fs = require('fs-extra'), path = require('path'), electron = require('electron'), browser = electron.remote.getCurrentWindow(); window['__start'] = async () => { const buttons = require('./buttons.js')(() => ({ frameless: true })); document.querySelector('#menu-titlebar').appendChild(buttons.element); document.defaultView.addEventListener('keyup', (event) => { if (event.code === 'F5') window.reload(); if ((event.ctrlKey || event.metaKey) && event.key === 'e') browser.close(); }); electron.ipcRenderer.on('enhancer:set-theme', (event, theme) => { document.body.className = `notion-${theme.mode}-theme`; for (const style of theme.rules) document.body.style.setProperty(style[0], style[1]); }); function createElement(html) { const template = document.createElement('template'); template.innerHTML = html.trim(); return template.content.firstElementChild; } function createAlert(type, message) { if (!type) throw Error(': no alert type specified'); const el = createElement(` `); return { el, resolve() { el.outerHTML = ''; }, prepend() { document.querySelector('#alerts').prepend(el); return this; }, append() { document.querySelector('#alerts').appendChild(el); return this; }, }; } // update checker fetch( `https://api.github.com/repos/dragonwocky/notion-enhancer/releases/latest` ) .then((res) => res.json()) .then((res) => { const raw_v = require('./mod.js').version, version = { local: raw_v.split(/[~-]/g)[0], repo: res.tag_name.slice(1), }; if (version.local == version.repo) return; // compare func from https://github.com/substack/semver-compare version.sorted = [version.local, version.repo].sort((a, b) => { const pa = a.split('.'), pb = b.split('.'); for (let i = 0; i < 3; i++) { let na = Number(pa[i]), nb = Number(pb[i]); if (na > nb) return 1; if (nb > na) return -1; if (!isNaN(na) && isNaN(nb)) return 1; if (isNaN(na) && !isNaN(nb)) return -1; } return 0; }); createAlert( 'warning', version.sorted[0] == version.local ? `update v${version.repo} available!
run npm i -g notion-enhancer
(or yarn global add notion-enhancer),
and notion-enhancer apply.` : `local build v${raw_v} is unstable.` ).prepend(); }); // mod loader const modules = helpers.getEnhancements(); if (modules.loaded.length) console.info( ` enhancements loaded: ${modules.loaded .map((mod) => mod.name) .join(', ')}.` ); if (modules.invalid.length) { createAlert( 'error', `invalid mods found: ${modules.invalid .map((mod) => `${mod}`) .join(', ')}.` ).append(); } // mod info + options function markdown(string) { const parsed = string .split('\n') .map((line) => line .trim() .replace(/\s+/g, ' ') // > quote .replace(/^>\s+(.+)$/g, '
$1
') // ~~strikethrough~~ .replace(/([^\\])?~~((?:(?!~~).)*[^\\])~~/g, '$1$2') // __underline__ .replace(/([^\\])?__((?:(?!__).)*[^\\])__/g, '$1$2') // **bold** .replace(/([^\\])?\*\*((?:(?!\*\*).)*[^\\])\*\*/g, '$1$2') // *italic* .replace(/([^\\])?\*([^*]*[^\\*])\*/g, '$1$2') // _italic_ .replace(/([^\\])?_([^_]*[^\\_])_/g, '$1$2') // `code` .replace(/([^\\])?`([^`]*[^\\`])`/g, '$1$2') // ![image_title](source) .replace( /([^\\])?\!\[([^\]]*[^\\\]]?)\]\(([^)]*[^\\)])\)/g, '$1$2' ) // [link](destination) .replace( /([^\\])?\[([^\]]*[^\\\]]?)\]\(([^)]*[^\\)])\)/g, '$1$2' ) ) .map((line) => line.startsWith('
') ? line : `

${line}

` ) .join(''); return parsed; } let modified_notice; function modified() { if (modified_notice) return; modified_notice = createAlert( 'info', `changes may not fully apply until app restart.` ); modified_notice.append(); } const $modules = document.querySelector('#modules'); for (let mod of modules.loaded.sort((a, b) => a.tags.includes('core') || store('mods', { [a.id]: { pinned: false } }).pinned ? -1 : b.tags.includes('core') || store('mods', { [b.id]: { pinned: false } }).pinned ? 1 : a.name.localeCompare(b.name) )) { const menuStore = store('mods', { [mod.id]: { enabled: false } }); mod.elem = createElement(` `); const $enable = mod.elem.querySelector(`#enable_${mod.id}`); if ($enable) $enable.addEventListener('click', (event) => { menuStore[mod.id].enabled = $enable.checked; mod.elem.className = menuStore[mod.id].enabled ? 'enabled' : 'disabled'; }); const $options = mod.elem.querySelector('.options'); let file_icon; if ($options) for (const opt of mod.options) { let $opt; switch (opt.type) { case 'toggle': $opt = createElement(`

`); break; case 'select': $opt = createElement(`

`); break; case 'input': $opt = createElement(`

`); break; case 'file': if (!file_icon) file_icon = await fs.readFile( path.resolve(`${__dirname}/icons/file.svg`) ); $opt = createElement(`

`); $opt.querySelector('.clear').addEventListener('click', (event) => { store(mod.id)[opt.key] = ''; $opt.querySelector('.path').innerText = store(mod.id)[opt.key] ? store(mod.id)[opt.key].split(path.sep).reverse()[0] : 'choose a file...'; }); break; } if (opt.type !== 'file') { $opt.querySelector( `#${opt.type}_${mod.id}--${opt.key}` ).value = store(mod.id, { [opt.key]: opt.type === 'select' ? opt.value[0] : opt.value, })[opt.key]; } $opt .querySelector(`#${opt.type}_${mod.id}--${opt.key}`) .addEventListener('change', (event) => { if (opt.type === 'toggle') { store(mod.id)[opt.key] = event.target.checked; } else if (opt.type === 'file') { if (event.target.files.length) store(mod.id)[opt.key] = event.target.files[0].path; $opt.querySelector('.path').innerText = store(mod.id)[opt.key] ? store(mod.id)[opt.key].split(path.sep).reverse()[0] : 'choose a file...'; } else store(mod.id)[opt.key] = typeof opt.value === 'number' ? Number(event.target.value) : event.target.value; modified(); }); $options.appendChild($opt); } $modules.append(mod.elem); } };