chore: simplify api imports

This commit is contained in:
dragonwocky 2024-02-17 00:27:35 +11:00
parent f0e2570448
commit 1f717b98ca
Signed by: dragonwocky
GPG Key ID: 7998D08F7D7BD7A8
9 changed files with 224 additions and 220 deletions

View File

@ -1,6 +1,6 @@
/**
* notion-enhancer
* (c) 2023 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (c) 2024 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
@ -13,6 +13,7 @@ import {
expandVariantGroup,
} from "../vendor/@unocss-core.mjs";
import { presetUno } from "../vendor/@unocss-preset-uno.mjs";
import "../assets/icons.svg.js";
// prettier-ignore
// https://developer.mozilla.org/en-US/docs/Web/SVG/Element
@ -21,71 +22,87 @@ import { presetUno } from "../vendor/@unocss-preset-uno.mjs";
const svgElements = ["animate","animateMotion","animateTransform","circle","clipPath","defs","desc","discard","ellipse","feBlend","feColorMatrix","feComponentTransfer","feComposite","feConvolveMatrix","feDiffuseLighting","feDisplacementMap","feDistantLight","feDropShadow","feFlood","feFuncA","feFuncB","feFuncG","feFuncR","feGaussianBlur","feImage","feMerge","feMergeNode","feMorphology","feOffset","fePointLight","feSpecularLighting","feSpotLight","feTile","feTurbulence","filter","foreignObject","g","hatch","hatchpath","image","line","linearGradient","marker","mask","metadata","mpath","path","pattern","polygon","polyline","radialGradient","rect","script","set","stop","style","svg","switch","symbol","text","textPath","title","tspan","use","view"],
htmlAttributes = ["accept","accept-charset","accesskey","action","align","allow","alt","async","autocapitalize","autocomplete","autofocus","autoplay","background","bgcolor","border","buffered","capture","challenge","charset","checked","cite","class","code","codebase","color","cols","colspan","content","contenteditable","contextmenu","controls","coords","crossorigin","csp","data","data-*","datetime","decoding","default","defer","dir","dirname","disabled","download","draggable","enctype","enterkeyhint","for","form","formaction","formenctype","formmethod","formnovalidate","formtarget","headers","height","hidden","high","href","hreflang","http-equiv","icon","id","importance","integrity","inputmode","ismap","itemprop","keytype","kind","label","lang","loading","list","loop","low","max","maxlength","minlength","media","method","min","multiple","muted","name","novalidate","open","optimum","pattern","ping","placeholder","playsinline","poster","preload","radiogroup","readonly","referrerpolicy","rel","required","reversed","role","rows","rowspan","sandbox","scope","selected","shape","size","sizes","slot","span","spellcheck","src","srcdoc","srclang","srcset","start","step","style","tabindex","target","title","translate","type","usemap","value","width","wrap","accent-height","accumulate","additive","alignment-baseline","alphabetic","amplitude","arabic-form","ascent","attributeName","attributeType","azimuth","baseFrequency","baseline-shift","baseProfile","bbox","begin","bias","by","calcMode","cap-height","clip","clipPathUnits","clip-path","clip-rule","color-interpolation","color-interpolation-filters","color-profile","color-rendering","contentScriptType","contentStyleType","cursor","cx","cy","d","decelerate","descent","diffuseConstant","direction","display","divisor","dominant-baseline","dur","dx","dy","edgeMode","elevation","enable-background","end","exponent","fill","fill-opacity","fill-rule","filter","filterRes","filterUnits","flood-color","flood-opacity","font-family","font-size","font-size-adjust","font-stretch","font-style","font-variant","font-weight","format","from","fr","fx","fy","g1","g2","glyph-name","glyph-orientation-horizontal","glyph-orientation-vertical","glyphRef","gradientTransform","gradientUnits","hanging","horiz-adv-x","horiz-origin-x","ideographic","image-rendering","in","in2","intercept","k","k1","k2","k3","k4","kernelMatrix","kernelUnitLength","kerning","keyPoints","keySplines","keyTimes","lengthAdjust","letter-spacing","lighting-color","limitingConeAngle","local","marker-end","marker-mid","marker-start","markerHeight","markerUnits","markerWidth","mask","maskContentUnits","maskUnits","mathematical","mode","numOctaves","offset","opacity","operator","order","orient","orientation","origin","overflow","overline-position","overline-thickness","panose-1","paint-order","path","pathLength","patternContentUnits","patternTransform","patternUnits","pointer-events","points","pointsAtX","pointsAtY","pointsAtZ","preserveAlpha","preserveAspectRatio","primitiveUnits","r","radius","referrerPolicy","refX","refY","rendering-intent","repeatCount","repeatDur","requiredExtensions","requiredFeatures","restart","result","rotate","rx","ry","scale","seed","shape-rendering","slope","spacing","specularConstant","specularExponent","speed","spreadMethod","startOffset","stdDeviation","stemh","stemv","stitchTiles","stop-color","stop-opacity","strikethrough-position","strikethrough-thickness","string","stroke","stroke-dasharray","stroke-dashoffset","stroke-linecap","stroke-linejoin","stroke-miterlimit","stroke-opacity","stroke-width","surfaceScale","systemLanguage","tableValues","targetX","targetY","text-anchor","text-decoration","text-rendering","textLength","to","transform","transform-origin","u1","u2","underline-position","underline-thickness","unicode","unicode-bidi","unicode-range","units-per-em","v-alphabetic","v-hanging","v-ideographic","v-mathematical","values","vector-effect","version","vert-adv-y","vert-origin-x","vert-origin-y","viewBox","viewTarget","visibility","widths","word-spacing","writing-mode","x","x-height","x1","x2","xChannelSelector","xlink:actuate","xlink:arcrole","xlink:href","xlink:role","xlink:show","xlink:title","xlink:type","xml:base","xml:lang","xml:space","y","y1","y2","yChannelSelector","z","zoomAndPan"];
const kebabToPascalCase = (string) =>
string[0].toUpperCase() +
string.replace(/-[a-z]/g, (match) => match.slice(1).toUpperCase()).slice(1),
hToString = (type, props, ...children) =>
`<${type}${Object.entries(props)
.map(([attr, value]) => ` ${attr}="${value}"`)
.join("")}>${children
.map((child) => (Array.isArray(child) ? hToString(...child) : child))
.join("")}</${type}>`,
svgToUri = (svg) => {
// https://gist.github.com/jennyknuth/222825e315d45a738ed9d6e04c7a88d0
const xlmns = ~svg.indexOf("xmlns")
? "<svg"
: '<svg xmlns="http://www.w3.org/2000/svg"';
return `url("data:image/svg+xml;utf8,${svg
.replace("<svg", xlmns)
.replace(/"/g, "'")
.replace(/%/g, "%25")
.replace(/#/g, "%23")
.replace(/{/g, "%7B")
.replace(/}/g, "%7D")
.replace(/</g, "%3C")
.replace(/>/g, "%3E")
.replace(/\s+/g, " ")
.trim()}")`;
};
const iconPattern = /^i-((?:\w|-)+)(?:\?(mask|bg|auto))?$/,
presetIcons = ([, icon, mode]) => {
let svg;
if (!["bg", "mask"].includes(mode)) mode = undefined;
if (icon === "notion-enhancer") {
const { iconColour, iconMonochrome } = globalThis.__enhancerApi;
svg = mode === "mask" ? iconMonochrome : iconColour;
} else {
icon = kebabToPascalCase(icon);
if (!lucide[icon]) return;
const [type, props, children] = lucide[icon];
svg = hToString(type, props, ...children);
// accelerators approximately match electron accelerators.
// logic used when recording hotkeys in menu matches logic used
// when triggering hotkeys => detection should be reliable.
// default hotkeys using "alt" may trigger an altcode or
// accented character on some keyboard layouts (not recommended).
let keyListeners = [];
const modifierAliases = [
["metaKey", ["meta", "os", "win", "cmd", "command"]],
["ctrlKey", ["ctrl", "control"]],
["shiftKey", ["shift"]],
["altKey", ["alt"]],
],
addKeyListener = (accelerator, callback, waitForKeyup = false) => {
if (typeof accelerator === "string") accelerator = accelerator.split("+");
accelerator = accelerator.map((key) => key.toLowerCase());
keyListeners.push([accelerator, callback, waitForKeyup]);
},
removeKeyListener = (callback) => {
keyListeners = keyListeners.filter(([, c]) => c !== callback);
},
handleKeypress = (event, keyListeners) => {
for (const [accelerator, callback] of keyListeners) {
const acceleratorModifiers = [],
combinationTriggered =
accelerator.every((key) => {
for (const [modifier, aliases] of modifierAliases) {
if (aliases.includes(key)) {
acceleratorModifiers.push(modifier);
return true;
}
}
if (key === "space") key = " ";
if (key === "plus") key = "equal";
if (key === "minus") key = "-";
if (key === "\\") key = "backslash";
if (key === ",") key = "comma";
if (key === ".") key = "period";
const keyPressed = [
event.key.toLowerCase(),
event.code.toLowerCase(),
].includes(key);
return keyPressed;
}) &&
modifierAliases.every(([modifier]) => {
// required && used -> matches accelerator
// !required && !used -> matches accelerator
// (required && !used) || (!required && used) -> no match
// differentiates e.g.ctrl + x from ctrl + shift + x
return acceleratorModifiers.includes(modifier) === event[modifier];
});
if (combinationTriggered) callback(event);
}
mode ??= svg.includes("currentColor") ? "mask" : "bg";
return {
// https://antfu.me/posts/icons-in-pure-css
display: "inline-block",
height: "1em",
width: "1em",
...(mode === "mask"
? {
mask: `${svgToUri(svg)} no-repeat`,
"mask-size": "100% 100%",
"background-color": "currentColor",
}
: {
background: `${svgToUri(svg)} no-repeat`,
"background-size": "100% 100%",
"background-color": "transparent",
}),
};
},
onKeyup = (event) => {
const keyupListeners = keyListeners //
.filter(([, , waitForKeyup]) => waitForKeyup);
handleKeypress(event, keyupListeners);
},
onKeydown = (event) => {
const keydownListeners = keyListeners //
.filter(([, , waitForKeyup]) => !waitForKeyup);
handleKeypress(event, keydownListeners);
};
document.removeEventListener("keyup", onKeyup);
document.removeEventListener("keydown", onKeydown);
document.addEventListener("keyup", onKeyup);
document.addEventListener("keydown", onKeydown);
// mutation listeners observe updates to the dom.
// by default, the criteria for matching a selector
// is very broad. custom opts can be passed when
// adding a listener to reduce handler calls
let documentObserver,
observerDefaults = {
// whether to observe attribute updates
attributes: true,
// whether to observe innerText updates
characterData: true,
// whether to observe added/removed nodes
childList: true,
// whether to observe parent/descendant nodes
subtree: true,
},
mutationListeners = [];
@ -109,7 +126,6 @@ const _mutations = [],
matchesParent = opts.subtree && target.matches(`${selector} *`),
matchesChild = opts.subtree && target.querySelector(selector),
matchesAdded =
// was matching element added?
opts.childList &&
[...(mutation.addedNodes || [])].some((node) => {
if (!(node instanceof HTMLElement)) node = node.parentElement;
@ -136,12 +152,21 @@ const _mutations = [],
document.addEventListener("readystatechange", attachObserver);
attachObserver();
// combines instance-provided element props
// with a template of element props such that
// island/component/template props handlers
// and styles can be preserved and extended
// rather than overwritten
const extendProps = (props, extend) => {
const kebabToPascalCase = (string) =>
string[0].toUpperCase() +
string.replace(/-[a-z]/g, (match) => match.slice(1).toUpperCase()).slice(1),
hToString = (type, props, ...children) =>
`<${type}${Object.entries(props)
.map(([attr, value]) => ` ${attr}="${value}"`)
.join("")}>${children
.map((child) => (Array.isArray(child) ? hToString(...child) : child))
.join("")}</${type}>`,
// combines instance-provided element props
// with a template of element props such that
// island/component/template props handlers
// and styles can be preserved and extended
// rather than overwritten
extendProps = (props, extend) => {
for (const key in extend) {
const { [key]: value } = props;
if (typeof extend[key] === "function") {
@ -194,6 +219,53 @@ const extendProps = (props, extend) => {
},
html = htm.bind(h);
const iconPattern = /^i-((?:\w|-)+)(?:\?(mask|bg|auto))?$/,
svgToUri = (svg) => {
// https://gist.github.com/jennyknuth/222825e315d45a738ed9d6e04c7a88d0
const xlmns = ~svg.indexOf("xmlns")
? "<svg"
: '<svg xmlns="http://www.w3.org/2000/svg"';
return `url("data:image/svg+xml;utf8,${svg
.replace("<svg", xlmns)
.replace(/"/g, "'")
.replace(/%/g, "%25")
.replace(/#/g, "%23")
.replace(/{/g, "%7B")
.replace(/}/g, "%7D")
.replace(/</g, "%3C")
.replace(/>/g, "%3E")
.replace(/\s+/g, " ")
.trim()}")`;
},
// prefer custom preset over @unocss/preset-icons:
// limits icons to single set, avoids loading over
// cdn (otherwise could cause issues when submitting
// to the chrome webstore). also makes custom icon
// handling straightforward
presetIcons = ([, icon, mode]) => {
let svg,
mask = mode === "mask";
if (icon === "notion-enhancer") {
const { iconColour, iconMonochrome } = globalThis.__enhancerApi;
svg = mask ? iconMonochrome : iconColour;
} else {
icon = kebabToPascalCase(icon);
if (!lucide[icon]) return;
const [type, props, children] = lucide[icon];
svg = hToString(type, props, ...children);
}
mask ||= mode !== "bg" && svg.includes("currentColor");
return {
// https://antfu.me/posts/icons-in-pure-css
display: "inline-block",
height: "1em",
width: "1em",
[mask ? "mask" : "background"]: `${svgToUri(svg)} no-repeat`,
[mask ? "mask-size" : "background-size"]: "100% 100%",
"background-color": mask ? "currentColor" : "transparent",
};
};
let _renderedTokens = -1;
const _tokens = new Set(),
_stylesheet = html`<style id="__unocss"></style>`,
@ -229,6 +301,8 @@ renderStylesheet();
Object.assign((globalThis.__enhancerApi ??= {}), {
html,
extendProps,
addKeyListener,
removeKeyListener,
addMutationListener,
removeMutationListener,
});

View File

@ -1,6 +1,6 @@
/**
* notion-enhancer
* (c) 2023 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (c) 2024 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/

70
src/api/state.js Normal file
View File

@ -0,0 +1,70 @@
/**
* notion-enhancer
* (c) 2023 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
"use strict";
// batch event callbacks to avoid over-handling
// and any conflicts / perf.issues that may
// otherwise result. initial call is immediate,
// following calls are delayed. a wait time of
// ~200ms is recommended (the avg. human visual
// reaction time is ~180-200ms)
const sleep = async (ms) => {
return new Promise((res, rej) => setTimeout(res, ms));
},
debounce = (callback, ms = 200) => {
let delay, update;
const next = () =>
sleep(ms).then(() => {
if (!update) return (delay = undefined);
update(), (update = undefined);
delay = next();
});
return (...args) => {
if (delay) update = callback.bind(this, ...args);
return delay || ((delay = next()), callback(...args));
};
};
// provides basic key/value reactivity:
// this is shared between all active mods,
// i.e. mods can read and update other mods'
// reactive states. this enables interop
// between a mod's component islands and
// supports inter-mod communication if so
// required. caution should be used in
// naming keys to avoid conflicts
const _state = {},
_subscribers = [],
setState = (state) => {
Object.assign(_state, state);
const updates = Object.keys(state);
_subscribers
.filter(([keys]) => updates.some((key) => keys.includes(key)))
.forEach(([keys, callback]) => callback(keys.map((key) => _state[key])));
},
// useState(["keyA", "keyB"]) => returns [valueA, valueB]
// useState(["keyA", "keyB"], callback) => registers callback
// to be triggered after each update to either keyA or keyB,
// with [valueA, valueB] passed to the callback's first arg
useState = (keys, callback) => {
const state = keys.map((key) => _state[key]);
if (callback) {
callback = debounce(callback);
_subscribers.push([keys, callback]);
callback(state);
}
return state;
},
dumpState = () => _state;
Object.assign((globalThis.__enhancerApi ??= {}), {
sleep,
debounce,
setState,
useState,
dumpState,
});

View File

@ -1,6 +1,6 @@
/**
* notion-enhancer
* (c) 2023 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (c) 2024 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
@ -8,9 +8,10 @@
const IS_ELECTRON = typeof module !== "undefined",
IS_RENDERER = IS_ELECTRON && process.type === "renderer",
whenReady = new Promise((res, rej) => {
API_LOADED = new Promise((res, rej) => {
(globalThis.__enhancerApi ??= {}).__isReady = res;
});
}),
whenReady = (callback) => API_LOADED.then(callback);
// expected values: 'linux', 'win32', 'darwin' (== macos), 'firefox'
// and 'chromium' (inc. chromium-based browsers like edge and brave)
@ -159,5 +160,5 @@ Object.assign((globalThis.__enhancerApi ??= {}), {
readJson,
initDatabase,
reloadApp,
whenReady: (callback) => whenReady.then(callback),
whenReady,
});

View File

@ -1,140 +0,0 @@
/**
* notion-enhancer
* (c) 2023 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
"use strict";
// batch event callbacks to avoid over-handling
// and any conflicts / perf.issues that may
// otherwise result. initial call is immediate,
// following calls are delayed. a wait time of
// ~200ms is recommended (the avg. human visual
// reaction time is ~180-200ms)
const sleep = async (ms) => {
return new Promise((res, rej) => setTimeout(res, ms));
},
debounce = (callback, ms = 200) => {
let delay, update;
const next = () =>
sleep(ms).then(() => {
if (!update) return (delay = undefined);
update(), (update = undefined);
delay = next();
});
return (...args) => {
if (delay) update = callback.bind(this, ...args);
return delay || ((delay = next()), callback(...args));
};
};
// provides basic key/value reactivity:
// this is shared between all active mods,
// i.e. mods can read and update other mods'
// reactive states. this enables interop
// between a mod's component islands and
// supports inter-mod communication if so
// required. caution should be used in
// naming keys to avoid conflicts
const _state = {},
_subscribers = [],
setState = (state) => {
Object.assign(_state, state);
const updates = Object.keys(state);
_subscribers
.filter(([keys]) => updates.some((key) => keys.includes(key)))
.forEach(([keys, callback]) => callback(keys.map((key) => _state[key])));
},
// useState(["keyA", "keyB"]) => returns [valueA, valueB]
// useState(["keyA", "keyB"], callback) => registers callback
// to be triggered after each update to either keyA or keyB,
// with [valueA, valueB] passed to the callback's first arg
useState = (keys, callback) => {
const state = keys.map((key) => _state[key]);
if (callback) {
callback = debounce(callback);
_subscribers.push([keys, callback]);
callback(state);
}
return state;
},
dumpState = () => _state;
let keyListeners = [];
// accelerators approximately match electron accelerators.
// logic used when recording hotkeys in menu matches logic used
// when triggering hotkeys => detection should be reliable.
// default hotkeys using "alt" may trigger an altcode or
// accented character on some keyboard layouts (not recommended).
const modifierAliases = [
["metaKey", ["meta", "os", "win", "cmd", "command"]],
["ctrlKey", ["ctrl", "control"]],
["shiftKey", ["shift"]],
["altKey", ["alt"]],
],
addKeyListener = (accelerator, callback, waitForKeyup = false) => {
if (typeof accelerator === "string") accelerator = accelerator.split("+");
accelerator = accelerator.map((key) => key.toLowerCase());
keyListeners.push([accelerator, callback, waitForKeyup]);
},
removeKeyListener = (callback) => {
keyListeners = keyListeners.filter(([, c]) => c !== callback);
},
handleKeypress = (event, keyListeners) => {
for (const [accelerator, callback] of keyListeners) {
const acceleratorModifiers = [],
combinationTriggered =
accelerator.every((key) => {
for (const [modifier, aliases] of modifierAliases) {
if (aliases.includes(key)) {
acceleratorModifiers.push(modifier);
return true;
}
}
if (key === "space") key = " ";
if (key === "plus") key = "equal";
if (key === "minus") key = "-";
if (key === "\\") key = "backslash";
if (key === ",") key = "comma";
if (key === ".") key = "period";
const keyPressed = [
event.key.toLowerCase(),
event.code.toLowerCase(),
].includes(key);
return keyPressed;
}) &&
modifierAliases.every(([modifier]) => {
// required && used -> matches accelerator
// !required && !used -> matches accelerator
// (required && !used) || (!required && used) -> no match
// differentiates e.g.ctrl + x from ctrl + shift + x
return acceleratorModifiers.includes(modifier) === event[modifier];
});
if (combinationTriggered) callback(event);
}
},
onKeyup = (event) => {
const keyupListeners = keyListeners //
.filter(([, , waitForKeyup]) => waitForKeyup);
handleKeypress(event, keyupListeners);
},
onKeydown = (event) => {
const keydownListeners = keyListeners //
.filter(([, , waitForKeyup]) => !waitForKeyup);
handleKeypress(event, keydownListeners);
};
document.removeEventListener("keyup", onKeyup);
document.removeEventListener("keydown", onKeydown);
document.addEventListener("keyup", onKeyup);
document.addEventListener("keydown", onKeydown);
Object.assign((globalThis.__enhancerApi ??= {}), {
sleep,
debounce,
setState,
useState,
dumpState,
addKeyListener,
removeKeyListener,
});

View File

@ -11,8 +11,8 @@ function View({ id }, ...children) {
// set padding on last child to maintain pad on overflow
$view = html`<article
id=${id}
class="notion-enhancer--menu-view h-full w-full min-w-[580px]
absolute px-[60px] pt-[36px] important:[&>*]:last:pb-[36px]"
class="notion-enhancer--menu-view min-h-full w-full
absolute px-[60px] py-[36px] min-w-[580px]"
>
${children}
</article>`;

View File

@ -14,8 +14,8 @@ const isElectron = () => {
};
if (isElectron()) {
require("./common/system.js");
require("./common/registry.js");
require("./api/system.js");
require("./api/registry.js");
const { enhancerUrl } = globalThis.__enhancerApi,
{ getMods, isEnabled, modDatabase } = globalThis.__enhancerApi;
@ -49,6 +49,6 @@ if (isElectron()) {
}
};
} else {
import(chrome.runtime.getURL("/common/system.js")) //
import(chrome.runtime.getURL("/api/system.js")) //
.then(() => import(chrome.runtime.getURL("/load.mjs")));
}

View File

@ -39,12 +39,11 @@ export default (async () => {
// in both situations, modules that attach to
// the dom must be re-imported
await Promise.all([
// i.e. if (not_menu) or (is_menu && not_electron), then import
!(!IS_MENU || !IS_ELECTRON) || import(enhancerUrl("assets/icons.svg.js")),
!(!IS_MENU || !IS_ELECTRON) || import(enhancerUrl("common/registry.js")),
import(enhancerUrl("common/scaffold.mjs")),
import(enhancerUrl("common/events.js")),
IS_ELECTRON || import(enhancerUrl("common/registry.js")),
(IS_ELECTRON && IS_MENU) || import(enhancerUrl("api/state.js")),
import(enhancerUrl("api/interface.mjs")),
]);
globalThis.__enhancerApi.__isReady(globalThis.__enhancerApi);