chore: switch to unocss for styling

faster + more powerful + better documented + more active than twind. sticking with custom preset-icons to avoid fetching over cdn + to load custom icons
This commit is contained in:
dragonwocky 2024-02-16 18:16:58 +11:00
parent 7fe993d356
commit f0e2570448
Signed by: dragonwocky
GPG Key ID: 7998D08F7D7BD7A8
42 changed files with 804 additions and 776 deletions

View File

@ -9,10 +9,41 @@ import { existsSync } from "node:fs";
import { resolve } from "node:path";
import { fileURLToPath } from "node:url";
const esmVersion = "135",
esTarget = "es2022",
esmBundle = ({ name, version, path = "", exports = [] }) => {
const scopedName = name;
if (name.startsWith("@")) name = name.split("/")[1];
path ||= `${name}.bundle.mjs`;
let bundleSrc = `https://esm.sh/v${esmVersion}/${scopedName}@${version}/${esTarget}/${path}`;
if (exports.length) bundleSrc += `?bundle&exports=${exports.join()}`;
return { [`${scopedName.replace(/\//g, "-")}.mjs`]: bundleSrc };
};
const dependencies = {
"htm.min.js": "https://unpkg.com/htm@3.1.1/mini/index.js",
"twind.min.js": "https://unpkg.com/@twind/cdn@1.0.8/cdn.global.js",
"lucide.min.js": "https://unpkg.com/lucide@0.319.0/dist/umd/lucide.min.js",
...esmBundle({ name: "htm", version: "3.1.1" }),
...esmBundle({
name: "lucide",
version: "0.319.0",
path: "dist/umd/lucide.mjs",
}),
...esmBundle({
name: "@unocss/core",
version: "0.58.5",
exports: ["createGenerator", "expandVariantGroup"],
}),
...esmBundle({
name: "@unocss/preset-uno",
version: "0.58.5",
exports: ["presetUno"],
}),
...esmBundle({
name: "@unocss/preset-icons",
version: "0.58.5",
exports: ["presetIcons"],
}),
"@unocss-preflight-tailwind.css":
"https://esm.sh/@unocss/reset@0.58.5/tailwind.css",
"coloris.min.js":
"https://cdn.jsdelivr.net/gh/mdbassit/Coloris@v0.22.0/dist/coloris.min.js",
"coloris.min.css":
@ -29,22 +60,3 @@ for (const file in dependencies) {
res = await (await fetch(source)).text();
await write(file, res);
}
// expose vendored twind cdn
await append("twind.min.js", `\n;globalThis.twind = twind;`);
// build content type lookup script from mime-db to avoid
// re-processing entire the database every time a file is
// requested via notion://www.notion.so/__notion-enhancer/
let contentTypes = [];
for (const [type, { extensions, charset }] of Object.entries(
await (await fetch("https://unpkg.com/mime-db@1.52.0/db.json")).json()
)) {
if (!extensions) continue;
const contentType = charset
? `${type}; charset=${charset.toLowerCase()}`
: type;
for (const ext of extensions) contentTypes.push([ext, contentType]);
}
contentTypes = `module.exports=new Map(${JSON.stringify(contentTypes)});`;
await write("content-types.min.js", contentTypes);

View File

@ -61,56 +61,6 @@ const _state = {},
},
dumpState = () => _state;
let documentObserver,
mutationListeners = [];
const mutationQueue = [],
addMutationListener = (selector, callback, subtree = true) => {
mutationListeners.push([selector, callback, subtree]);
},
removeMutationListener = (callback) => {
mutationListeners = mutationListeners.filter(([, c]) => c !== callback);
},
selectorMutated = (mutation, selector, subtree) => {
const target =
mutation.type === "characterData"
? mutation.target.parentElement
: mutation.target,
matchesTarget = target?.matches(selector);
if (!subtree) return matchesTarget;
const descendsFromTarget = target?.matches(`${selector} *`),
addedToTarget = [...(mutation.addedNodes || [])].some(
(node) =>
node instanceof HTMLElement &&
(node?.matches(`${selector}, ${selector} *`) ||
node?.querySelector(selector))
);
return matchesTarget || descendsFromTarget || addedToTarget;
},
handleMutations = () => {
while (mutationQueue.length) {
const mutation = mutationQueue.shift();
for (const [selector, callback, subtree] of mutationListeners) {
const matches = selectorMutated(mutation, selector, subtree);
if (matches) callback(mutation);
}
}
},
attachObserver = () => {
if (document.readyState !== "complete") return;
documentObserver ??= new MutationObserver((mutations, _observer) => {
if (!mutationQueue.length) requestIdleCallback(handleMutations);
mutationQueue.push(...mutations);
});
documentObserver.observe(document.body, {
attributes: true,
characterData: true,
childList: true,
subtree: true,
});
};
document.addEventListener("readystatechange", attachObserver);
attachObserver();
let keyListeners = [];
// accelerators approximately match electron accelerators.
// logic used when recording hotkeys in menu matches logic used
@ -163,17 +113,21 @@ const modifierAliases = [
});
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.addEventListener("keyup", (event) => {
const keyupListeners = keyListeners //
.filter(([, , waitForKeyup]) => waitForKeyup);
handleKeypress(event, keyupListeners);
});
document.addEventListener("keydown", (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,
@ -181,8 +135,6 @@ Object.assign((globalThis.__enhancerApi ??= {}), {
setState,
useState,
dumpState,
addMutationListener,
removeMutationListener,
addKeyListener,
removeKeyListener,
});

View File

@ -1,598 +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";
const { twind, htm, lucide } = globalThis,
{ iconColour, iconMonochrome } = globalThis.__enhancerApi;
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}>`;
const encodeSvg = (svg) =>
// https://gist.github.com/jennyknuth/222825e315d45a738ed9d6e04c7a88d0
svg
.replace(
"<svg",
~svg.indexOf("xmlns")
? "<svg"
: '<svg xmlns="http://www.w3.org/2000/svg"'
)
.replace(/"/g, "'")
.replace(/%/g, "%25")
.replace(/#/g, "%23")
.replace(/{/g, "%7B")
.replace(/}/g, "%7D")
.replace(/</g, "%3C")
.replace(/>/g, "%3E")
.replace(/\s+/g, " "),
presetIcons = ([, icon, mode]) => {
let svg;
// manually register i-notion-enhancer: renders the colour
// version by default, renders the monochrome version when
// mask mode is requested via i-notion-enhancer?mask
if (icon === "notion-enhancer") {
svg = mode === "mask" ? iconMonochrome : iconColour;
} else {
icon = kebabToPascalCase(icon);
if (!lucide[icon]) return;
const [type, props, children] = lucide[icon];
svg = hToString(type, props, ...children);
}
// https://antfu.me/posts/icons-in-pure-css
const dataUri = `url("data:image/svg+xml;utf8,${encodeSvg(svg)}")`;
if (mode === "auto") mode = undefined;
mode ??= svg.includes("currentColor") ? "mask" : "bg";
return {
display: "inline-block",
height: "1em",
width: "1em",
...(mode === "mask"
? {
mask: `${dataUri} no-repeat`,
"mask-size": "100% 100%",
"background-color": "currentColor",
color: "inherit",
}
: {
background: `${dataUri} no-repeat`,
"background-size": "100% 100%",
"background-color": "transparent",
}),
};
};
// at-runtime utility class evaluation w/ twind:
// - feature parity w/ tailwind v3
// - useful for building self-contained components
// (mods can extend interfaces w/out needing to
// import additional stylesheets)
// - integrated with lucide to render icons w/out
// complex markup, e.g. `<i class="i-bookmark" />`
twind.install({
darkMode: "class",
rules: [
["text-(wrap|nowrap|balance|pretty)", "textWrap"],
[/^i-((?:\w|-)+)(?:\?(mask|bg|auto))?$/, presetIcons],
[/^size-\[([^\]]+)\]$/, ({ 1: $1 }) => ({ height: $1, width: $1 })],
],
variants: [
["children", "&>*"],
["siblings", "&~*"],
["sibling", "&+*"],
[/^&/, (match) => match.input],
[/^has-\[([^\]]+)\]/, (match) => `&:has(${match[1]})`],
[
/^not-([a-z-]+|\[.+\])/,
({ 1: $1 }) => `&:not(${($1[0] == "[" ? "" : ":") + $1})`,
],
],
});
// https://developer.mozilla.org/en-US/docs/Web/SVG/Element
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 = [
// https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes
"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",
// https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute[
"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",
];
// enables use of the jsx-like htm syntax
// for building components and interfaces
// with tagged templates. instantiates dom
// elements directly, does not use a vdom.
// e.g. html`<div class=${className}></div>`
const h = (type, props, ...children) => {
children = children.flat(Infinity);
// html`<${Component} attr="value">Click Me<//>`
if (typeof type === "function") {
return type(props ?? {}, ...children);
}
const elem = svgElements.includes(type)
? document.createElementNS("http://www.w3.org/2000/svg", type)
: document.createElement(type);
for (const prop in props ?? {}) {
if (typeof props[prop] === "undefined") continue;
const isAttr =
htmlAttributes.includes(prop) ||
prop.startsWith("data-") ||
prop.startsWith("aria-");
if (isAttr) {
if (typeof props[prop] === "boolean") {
if (!props[prop]) continue;
elem.setAttribute(prop, "");
} else elem.setAttribute(prop, props[prop]);
} else elem[prop] = props[prop];
}
if (type === "style") {
elem.append(children.join("").replace(/\s+/g, " "));
} else elem.append(...children);
return elem;
},
// 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]: userProvided } = props;
if (typeof extend[key] === "function") {
props[key] = (...args) => {
extend[key](...args);
userProvided?.(...args);
};
} else if (key === "class") {
if (userProvided) props[key] += " ";
if (!userProvided) props[key] = "";
props[key] += extend[key];
} else props[key] = extend[key] ?? userProvided;
}
return props;
},
html = htm.bind(h);
Object.assign((globalThis.__enhancerApi ??= {}), {
html,
extendProps,
});

234
src/common/markup.mjs Normal file
View File

@ -0,0 +1,234 @@
/**
* notion-enhancer
* (c) 2023 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
"use strict";
import htm from "../vendor/htm.mjs";
import lucide from "../vendor/lucide.mjs";
import {
createGenerator,
expandVariantGroup,
} from "../vendor/@unocss-core.mjs";
import { presetUno } from "../vendor/@unocss-preset-uno.mjs";
// prettier-ignore
// https://developer.mozilla.org/en-US/docs/Web/SVG/Element
// https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute
// https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes
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);
}
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",
}),
};
};
let documentObserver,
observerDefaults = {
attributes: true,
characterData: true,
childList: true,
subtree: true,
},
mutationListeners = [];
const _mutations = [],
addMutationListener = (selector, callback, opts) => {
opts = { ...observerDefaults, ...opts };
mutationListeners.push([selector, callback, opts]);
},
removeMutationListener = (callback) => {
mutationListeners = mutationListeners.filter(([, c]) => c !== callback);
},
selectorMutated = (mutation, selector, opts) => {
if (!opts.attributes && mutation.type === "attributes") return false;
if (!opts.characterData && mutation.type === "characterData") return false;
const target =
mutation.type === "characterData"
? mutation.target.parentElement
: mutation.target;
if (!target) return false;
const matchesTarget = target.matches(selector),
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;
return node?.querySelector(selector);
});
return matchesTarget || matchesParent || matchesChild || matchesAdded;
},
handleMutations = () => {
let mutation;
while ((mutation = _mutations.shift())) {
for (const [selector, callback, subtree] of mutationListeners)
if (selectorMutated(mutation, selector, subtree)) callback(mutation);
}
},
attachObserver = () => {
if (document.readyState !== "complete") return;
document.removeEventListener("readystatechange", attachObserver);
(documentObserver ??= new MutationObserver((mutations, _observer) => {
if (!_mutations.length) requestIdleCallback(handleMutations);
_mutations.push(...mutations);
})).disconnect();
documentObserver.observe(document.body, observerDefaults);
};
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) => {
for (const key in extend) {
const { [key]: value } = props;
if (typeof extend[key] === "function") {
props[key] = (...args) => {
extend[key](...args);
if (typeof value === "function") value(...args);
};
} else if (key === "class") {
props[key] = value ? `${value} ${extend[key]}` : extend[key];
} else props[key] = extend[key] ?? value;
}
return props;
},
// enables use of the jsx-like htm syntax
// for building components and interfaces
// with tagged templates. instantiates dom
// elements directly, does not use a vdom.
// e.g. html`<div class=${className}></div>`
h = function (type, props, ...children) {
// disables element caching
this[0] = 3;
children = children.flat(Infinity);
if (typeof type === "function") {
// html`<${Component} attr="value">Click Me<//>`
return type(props ?? {}, ...children);
}
const elem = svgElements.includes(type)
? document.createElementNS("http://www.w3.org/2000/svg", type)
: document.createElement(type);
for (const prop in props ?? {}) {
if (typeof props[prop] === "undefined") continue;
if (["class", "className"].includes(prop)) {
// collapse multiline classes &
// expand utility variant class groups
props[prop] = props[prop].replace(/\s+/g, " ");
props[prop] = expandVariantGroup(props[prop]).trim();
elem.setAttribute("un-cloak", "");
}
if (htmlAttributes.includes(prop) || prop.includes("-")) {
if (typeof props[prop] === "boolean") {
if (!props[prop]) continue;
elem.setAttribute(prop, "");
} else elem.setAttribute(prop, props[prop]);
} else elem[prop] = props[prop];
}
if (type === "style") {
elem.append(children.join("").replace(/\s+/g, " "));
} else elem.append(...children);
return elem;
},
html = htm.bind(h);
let _renderedTokens = -1;
const _tokens = new Set(),
_stylesheet = html`<style id="__unocss"></style>`,
uno = createGenerator({
presets: [presetUno()],
preflights: [{ getCSS: () => `[un-cloak]{display:none!important}` }],
rules: [[iconPattern, presetIcons, { layer: "icons" }]],
layers: { preflights: -2, icons: -1, default: 1 },
}),
extractTokens = ($root) => {
if (!$root?.classList) return;
for (const t of $root.classList) _tokens.add(t);
for (const $ of $root.children) extractTokens($);
$root.removeAttribute("un-cloak");
},
renderStylesheet = async () => {
if (_renderedTokens === _tokens.size) return;
_renderedTokens = _tokens.size;
const res = await uno.generate(_tokens);
if (!document.contains(_stylesheet)) document.head.append(_stylesheet);
if (_stylesheet.innerHTML !== res.css) _stylesheet.innerHTML = res.css;
};
addMutationListener("*", (mutation) => {
const targets = [];
if (mutation.type === "childList") {
for (const node of mutation.addedNodes) extractTokens(node);
} else if (mutation.type === "attributes") extractTokens(mutation.target);
else return;
renderStylesheet();
});
renderStylesheet();
Object.assign((globalThis.__enhancerApi ??= {}), {
html,
extendProps,
addMutationListener,
removeMutationListener,
});

View File

@ -72,7 +72,7 @@ const insertMenu = async (api, db) => {
// pass notion-enhancer api to electron menu process
if (["linux", "win32", "darwin"].includes(platform)) {
const apiKey = "__enhancerApi";
this.contentWindow[apiKey] = globalThis[apiKey];
this.contentWindow[apiKey] = { ...globalThis[apiKey] };
}
_contentWindow = this.contentWindow;
updateMenuTheme();
@ -99,7 +99,7 @@ const insertMenu = async (api, db) => {
<b>Configure the notion-enhancer and its mods</b>
<//>`.attach($button, "right");
addMutationListener(notionSidebar, appendToDom);
addMutationListener(".notion-app-inner", updateMenuTheme, false);
addMutationListener(".notion-app-inner", updateMenuTheme, { subtree: false });
appendToDom();
addKeyListener(openMenuHotkey, (event) => {

View File

@ -18,7 +18,7 @@ const setupWrapper = () => {
const $wrapper = html`<div
class="notion-enhancer--floating-buttons z-50
absolute bottom-[calc(26px+env(safe-area-inset-bottom))]
flex gap-[12px] !(&>.notion-help-button:static)"
flex gap-[12px] important:[&>.notion-help-button]:static"
style="right:${$help.style.right}"
></div>`;
removeMutationListener(addToDom);

View File

@ -99,12 +99,12 @@ function Panel({
<i
class="i-chevrons-left size-[20px]
text-[color:var(--theme--fg-secondary)] transition-transform
group-&[data-pinned]/panel:rotate-180 duration-[${transitionDuration}ms]"
group-[&[data-pinned]]/panel:rotate-180 duration-[${transitionDuration}ms]"
/>
</button>`,
$panel = html`<div
class="notion-enhancer--panel group/panel order-2
shrink-0 &[data-pinned]:w-[var(--panel--width,0)]"
shrink-0 [&[data-pinned]]:w-[var(--panel--width,0)]"
>
<style>
.notion-frame {
@ -120,8 +120,8 @@ function Panel({
</style>
<aside
class="border-(l-1 [color:var(--theme--fg-border)]) w-0
group-&[data-pinned]/panel:(w-[var(--panel--width,0)]) h-[calc(100vh-45px)] bottom-0)
absolute right-0 z-20 bg-[color:var(--theme--bg-primary)] group-&[data-peeked]/panel:(
group-[&[data-pinned]]/panel:(w-[var(--panel--width,0)]) h-[calc(100vh-45px)] bottom-0)
absolute right-0 z-20 bg-[color:var(--theme--bg-primary)] group-[&[data-peeked]]/panel:(
w-[var(--panel--width,0)] h-[calc(100vh-120px)] bottom-[60px] rounded-l-[8px] border-(t-1 b-1))"
>
<div
@ -138,7 +138,7 @@ function Panel({
</div>`;
const topbarId = "e0700ce3-a9ae-45f5-92e5-610ded0e348d",
topbarFavorite = ".notion-topbar-favorite-button",
topbarFavorite = ".notion-topbar .notion-topbar-favorite-button",
$topbarToggle = html`<${TopbarButton}
aria-label="Toggle side panel"
icon="panel-right"
@ -149,7 +149,7 @@ function Panel({
};
$panelToggle.onclick = $topbarToggle.onclick = () => $panel.toggle();
addMutationListener(topbarFavorite, addToTopbar);
addToTopbar(topbarFavorite);
addToTopbar();
isEnabled(topbarId).then(async (topbarEnabled) => {
if (!topbarEnabled) return;
@ -225,11 +225,11 @@ function Panel({
class="absolute opacity-0 h-full w-[3px] left-[-2px]
active:cursor-text bg-[color:var(--theme--fg-border)] z-20
transition duration-300 hover:(cursor-col-resize opacity-100)
group-&[data-peeked]/panel:(w-[8px] left-[-1px] rounded-l-[7px])"
group-[&[data-peeked]]/panel:(w-[8px] left-[-1px] rounded-l-[7px])"
>
<div
class="ml-[2px] bg-[color:var(--theme--bg-primary)]
group-&[data-peeked]/panel:(my-px h-[calc(100%-2px)] rounded-l-[6px])"
group-[&[data-peeked]]/panel:(my-px h-[calc(100%-2px)] rounded-l-[6px])"
></div>
</div>`,
startDrag = async (event) => {
@ -284,8 +284,8 @@ function Panel({
const coreId = "0f0bf8b6-eae6-4273-b307-8fc43f2ee082",
$peekTrigger = html`<div
class="absolute z-10 right-0 h-[calc(100vh-120px)] bottom-[60px] w-[96px]
group-&[data-peeked]/panel:(w-[calc(var(--panel--width,0)+8px)])
group-&[data-pinned]/panel:(w-[calc(var(--panel--width,0)+8px)])"
group-[&[data-peeked]]/panel:(w-[calc(var(--panel--width,0)+8px)])
group-[&[data-pinned]]/panel:(w-[calc(var(--panel--width,0)+8px)])"
></div>`;
modDatabase(coreId).then(async (db) => {
_peekPanelOnHover = await db.get("peekPanelOnHover");

View File

@ -21,7 +21,7 @@ function Tooltip(props, ...children) {
leading-[1.4] font-medium py-[4px] px-[8px] rounded-[4px]
drop-shadow-md transition duration-100 opacity-0
group-open/tooltip:(pointer-events-auto opacity-100)
&>b:text-[color:var(--theme--fg-primary)]"
[&>b]:text-[color:var(--theme--fg-primary)]"
>
${children}
</div>

View File

@ -15,9 +15,9 @@ function TopbarButton({ icon, ...props }, ...children) {
select-none h-[28px] w-[33px] duration-[20ms]
transition inline-flex items-center justify-center
rounded-[3px] hover:bg-[color:var(--theme--bg-hover)]
has-[span]:w-auto &>span:(text-[14px] leading-[1.2] px-[8px])
&[data-active]:bg-[color:var(--theme--bg-hover)]
&>i:size-[20px]`,
has-[span]:w-auto [&>span]:(text-[14px] leading-[1.2] px-[8px])
[&[data-active]]:bg-[color:var(--theme--bg-hover)]
[&>i]:size-[20px]`,
});
return html`<button ...${props}>

View File

@ -64,7 +64,7 @@ function Circle(rect) {
const { html } = globalThis.__enhancerApi;
return html`<div
class="absolute rounded-full
border-(& purple-500) bg-purple-400"
border-(~ purple-500) bg-purple-400"
style=${rectToStyle(rect)}
></div>`;
}
@ -115,7 +115,7 @@ function Banner({ updateAvailable, isDevelopmentBuild }) {
const $welcome = html`<div
class="relative flex overflow-hidden h-[192px] rounded-t-[4px]
border-(& purple-400) bg-purple-500 from-white/20 to-transparent
border-(~ purple-400) bg-purple-500 from-white/20 to-transparent
text-white bg-[linear-gradient(225deg,var(--tw-gradient-stops))]"
>
<${Circle} width="128px" height="128px" bottom="-64px" left="-64px" />
@ -138,7 +138,7 @@ function Banner({ updateAvailable, isDevelopmentBuild }) {
class="absolute bottom-0 right-0 py-[24px]
px-[32px] md:px-[48px] lg:px-[64px]"
>
<div class="relative flex-(& col)">
<div class="relative flex-(~ col)">
<i class="i-notion-enhancer text-[42px] mx-auto mb-[8px]"></i>
${$version}
</div>
@ -146,7 +146,7 @@ function Banner({ updateAvailable, isDevelopmentBuild }) {
</div>`,
$sponsorship = html`<div
class="py-[18px] px-[16px] rounded-b-[4px]
border-(& [color:var(--theme--fg-border)]) bg-[color:var(--theme--bg-secondary)]"
border-(~ [color:var(--theme--fg-border)]) bg-[color:var(--theme--bg-secondary)]"
>
<div class="flex items-center gap-[16px]">
<p class="text-[14px] font-semibold">

View File

@ -19,13 +19,13 @@ function Button({ icon, variant, tagName, ...props }, ...children) {
hover:bg-[color:var(--theme--accent-primary\\_hover)]`
: variant === "secondary"
? `text-[color:var(--theme--accent-secondary)]
border-(& [color:var(--theme--accent-secondary)])
border-(~ [color:var(--theme--accent-secondary)])
hover:bg-[color:var(--theme--accent-secondary\\_hover)]`
: variant === "brand"
? `text-white border-(& purple-400)
? `text-white border-(~ purple-400)
bg-purple-500 hover:(from-white/20 to-transparent
bg-[linear-gradient(225deg,var(--tw-gradient-stops))])`
: `border-(& [color:var(--theme--fg-border)])
: `border-(~ [color:var(--theme--fg-border)])
not-disabled:hover:bg-[color:var(--theme--bg-hover)]
disabled:text-[color:var(--theme--fg-secondary)]`
}`,

View File

@ -11,11 +11,9 @@ function Checkbox({ _get, _set, _requireReload = true, ...props }) {
const { html, extendProps, setState, useState } = globalThis.__enhancerApi,
$input = html`<input
type="checkbox"
class="hidden checked:&+div:(px-px
bg-[color:var(--theme--accent-primary)])
not-checked:&+div:(&>i:text-transparent
border-(& [color:var(--theme--fg-primary)])
hover:bg-[color:var(--theme--bg-hover)])"
class="hidden [&:checked+div]:(px-px bg-[color:var(--theme--accent-primary)])
[&:not(:checked)+div>i]:text-transparent [&:not(:checked)+div]:(border-(~
[color:var(--theme--fg-primary)]) hover:bg-[color:var(--theme--bg-hover)])"
...${props}
/>`;
extendProps($input, { onchange: () => _set?.($input.checked) });

View File

@ -52,7 +52,7 @@ function Footer({ categories, transitionDuration = 150 }) {
return html`<footer
class="notion-enhancer--menu-footer px-[60px] py-[16px]
flex w-full bg-[color:var(--theme--bg-primary)] h-[64px]
border-t-(& [color:var(--theme--fg-border)])"
border-t-(~ [color:var(--theme--fg-border)])"
>
${$categories.map(([, $btn]) => $btn)}${$reload}
</footer>`;

View File

@ -11,7 +11,7 @@ function Heading(props, ...children) {
extendProps(props, {
class: `notion-enhancer--menu-heading text-[16px]
font-semibold mb-[16px] mt-[48px] first:mt-0 pb-[12px]
border-b-(& [color:var(--theme--fg-border)])`,
border-b-(~ [color:var(--theme--fg-border)])`,
});
return html`<h4 ...${props}>${children}</h4>`;
}

View File

@ -111,7 +111,7 @@ function Input({
${type === "hotkey" ? "text-[color:var(--theme--fg-secondary)]" : ""}
${type === "color"
? "font-medium"
: "border-(& [color:var(--theme--fg-border)])"}"
: "border-(~ [color:var(--theme--fg-border)])"}"
data-coloris=${type === "color"}
...${props}
/>`,
@ -175,7 +175,7 @@ function Input({
px-[8px] bg-[color:var(--theme--bg-secondary)]
h-[28px] rounded-[4px] transition duration-[20ms]
text-([14px] [color:var(--theme--fg-secondary)])
border-(& [color:var(--theme--fg-border)])
border-(~ [color:var(--theme--fg-border)])
hover:bg-[color:var(--theme--bg-hover)]"
onkeydown=${(event) => {
if ([" ", "Enter"].includes(event.key)) {
@ -191,7 +191,7 @@ function Input({
class="notion-enhancer--menu-input
${variant === "lg" ? "h-[32px]" : "h-[28px]"}
relative overflow-hidden rounded-[4px] w-full inline-block
focus-within:ring-(& [color:var(--theme--accent-primary)])
focus-within:ring-(~ [color:var(--theme--accent-primary)])
${className ?? ""} ${type === "color"
? "bg-([image:repeating-linear-gradient(45deg,#aaa_25%,transparent_25%,transparent_75%,#aaa_75%,#aaa),repeating-linear-gradient(45deg,#aaa_25%,#fff_25%,#fff_75%,#aaa_75%,#aaa)] [position:0_0,4px_4px] [size:8px_8px])"
: "bg-[color:var(--theme--bg-hover)]"}"

View File

@ -64,7 +64,7 @@ function List({ id, mods, description }) {
};
return html`<${Mod} ...${{ ...mod, _get, _set }} />`;
});
return html`<div class="flex-(& col) gap-y-[14px]">
return html`<div class="flex-(~ col) gap-y-[14px]">
<${Search} items=${$mods} itemType=${id} />
<${Description} innerHTML=${description} />
${$mods}

View File

@ -31,7 +31,7 @@ function Mod({
bg-[color:var(--theme--bg-secondary)] w-full py-[18px] px-[16px]
border border-[color:var(--theme--fg-border)] cursor-pointer
duration-[20ms] hover:bg-[color:var(--theme--bg-hover)]
transition &+.notion-enhancer--menu-option:mt-[24px]"
transition [&+.notion-enhancer--menu-option]:mt-[24px]"
>
${thumbnail
? html`<img
@ -39,7 +39,7 @@ function Mod({
class="rounded-[4px] mr-[12px] h-[74px] my-auto"
/>`
: ""}
<div class="flex-(& col) w-full">
<div class="flex-(~ col) w-full">
<div class="flex flex-wrap items-center gap-[8px] text-[14px] mb-[5px]">
<h3 class="my-0">${name}</h3>
${[`v${version}`, ...tags].map((tag) => {

View File

@ -65,7 +65,7 @@ function Onboarding() {
};
const $regularGreeting = html`<div
class="mt-[16px] grid-(& cols-3) gap-[16px]"
class="mt-[16px] grid-(~ cols-3) gap-[16px]"
>
<${Tile}
href="https://notion-enhancer.github.io/getting-started/basic-usage/"
@ -93,7 +93,7 @@ function Onboarding() {
A few awesome companies out there have teamed up with me to provide
you with the notion-enhancer, free forever. Check them out!
<//>
<div class="mt-[16px] grid-(& cols-1) gap-[16px]"></div>
<div class="mt-[16px] grid-(~ cols-1) gap-[16px]"></div>
<${Description} class="mt-[12px]">
<a href="mailto:thedragonring.bod@gmail.com">Join this list.</a>
<//>

View File

@ -45,7 +45,7 @@ function Option({ _get, _set, ...opt }) {
class="notion-enhancer--menu-option flex items-center justify-between
mb-[18px] ${opt.type === "toggle" ? "cursor-pointer" : ""}"
>
<div class="flex-(& col) ${opt.type === "text" ? "w-full" : "mr-[10%]"}">
<div class="flex-(~ col) ${opt.type === "text" ? "w-full" : "mr-[10%]"}">
<h5 class="text-[14px] mb-[2px] mt-0">${opt.label}</h5>
${opt.type === "text"
? html`<${Input}

View File

@ -19,7 +19,7 @@ function Popup(
extendProps(props, {
class: `notion-enhancer--menu-popup group/popup
absolute top-0 left-0 z-20 text-left font-normal
flex-(& col) justify-center pointer-events-none
flex-(~ col) justify-center pointer-events-none
items-end w-full ${isDropdown ? "" : "h-full"}`,
});

View File

@ -133,7 +133,7 @@ function Profile({ id }) {
<p class="text-[14px] py-[2px] px-[8px]">
Are you sure you want to delete the profile ${$confirmName} permanently?
</p>
<div class="flex-(& col) gap-[8px] py-[6px] px-[8px]">
<div class="flex-(~ col) gap-[8px] py-[6px] px-[8px]">
<${Button}
tabindex="0"
icon="trash"

View File

@ -62,7 +62,7 @@ function Sidebar({ items, categories }) {
</span>`,
$sidebar = html`<aside
class="notion-enhancer--menu-sidebar h-full
px-[4px] overflow-y-auto flex-(& col) row-span-1
px-[4px] overflow-y-auto flex-(~ col) row-span-1
bg-[color:var(--theme--bg-secondary)]"
>
${items.map((item) => {

View File

@ -10,7 +10,7 @@ function Tile({ icon, title, tagName, ...props }, ...children) {
const { html, extendProps } = globalThis.__enhancerApi;
extendProps(props, {
class: `flex items-center gap-[12px] rounded-[4px]
border-(& [color:var(--theme--fg-border)]) px-[16px]
border-(~ [color:var(--theme--fg-border)]) px-[16px]
bg-[color:var(--theme--bg-secondary)] py-[12px]
hover:bg-[color:var(--theme--bg-hover)]`,
});

View File

@ -11,7 +11,7 @@ function Toggle({ _get, _set, _requireReload = true, ...props }) {
const { html, extendProps, setState, useState } = globalThis.__enhancerApi,
$input = html`<input
type="checkbox"
class="hidden checked:&+div>div:(
class="hidden [&:checked+div>div]:(
bg-[color:var(--theme--accent-primary)]
after:translate-x-[12px])"
...${props}

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 absolute h-full w-full
min-w-[580px] px-[60px] pt-[36px] !&>*:last:pb-[36px]"
class="notion-enhancer--menu-view h-full w-full min-w-[580px]
absolute px-[60px] pt-[36px] important:[&>*]:last:pb-[36px]"
>
${children}
</article>`;

View File

@ -127,7 +127,7 @@ const renderMenu = async () => {
/>`,
$main = html`
<main
class="flex-(& col) overflow-hidden transition-[height]"
class="flex-(~ col) overflow-hidden transition-[height]"
style="height: calc(100% + 65px)"
>
<!-- wrappers necessary for transitions and breakpoints -->

View File

@ -77,7 +77,7 @@
"value": true
}
],
"clientStyles": ["variables.css"],
"clientStyles": ["variables.css", "../vendor/@unocss-preflight-tailwind.css"],
"clientScripts": ["client.mjs"],
"electronScripts": [[".webpack/main/index", "electron.cjs"]]
}

View File

@ -16,7 +16,7 @@ function Heading({ indent, ...props }, ...children) {
decoration-(2 [color:var(--theme--fg-border)])
hover:bg-[color:var(--theme--bg-hover)]
py-[6px] pr-[2px] pl-[${indent * 18}px]
underline-(& offset-4) last:mb-[24px]"
underline-(~ offset-4) last:mb-[24px]"
...${props}
>
${children}
@ -131,6 +131,6 @@ export default async (api, db) => {
const semanticHeadings = '[class$="header-block"] :is(h2, h3, h4)';
addMutationListener(`${page} ${semanticHeadings}`, updateHeadings);
addMutationListener(`${page}, ${scroller}`, setup, false);
addMutationListener(`${page}, ${scroller}`, setup, { subtree: false });
setup();
};

View File

@ -55,6 +55,6 @@ export default async (api, db) => {
$scroller?.addEventListener("scroll", onScroll);
onScroll();
};
addMutationListener(scroller, setup, false);
addMutationListener(scroller, setup, { subtree: false });
setup();
};

View File

@ -14,7 +14,7 @@ export default async (api, db) => {
const { onMessage, addMutationListener, removeMutationListener } = api,
$buttons = await createWindowButtons(),
topbarMore = ".notion-topbar-more-button",
topbarMore = ".notion-topbar .notion-topbar-more-button",
addToTopbar = () => {
if (document.contains($buttons)) removeMutationListener(addToTopbar);
document.querySelector(topbarMore)?.after($buttons);

View File

@ -78,7 +78,7 @@ export default async function (api, db) {
}
});
const favoriteSelector = " n",
const favoriteSelector = ".notion-topbar-favorite-button",
favoriteButton = await db.get("favoriteButton"),
favoriteIcon = await db.get("favoriteIcon"),
$favoriteIcon = favoriteIcon ? html(favoriteIcon.content) : undefined;
@ -108,8 +108,7 @@ export default async function (api, db) {
const alwaysOnTopButton = await db.get("alwaysOnTopButton");
if (alwaysOnTopButton === "Disabled") return;
const topbarFavorite = ".notion-topbar-favorite-button",
const topbarFavorite = `.notion-topbar ${favoriteSelector}`,
pinIcon = await db.get("pinIcon"),
unpinIcon = await db.get("unpinIcon"),
$pin = html`<${TopbarButton}
@ -138,10 +137,8 @@ export default async function (api, db) {
icon="pin-off"
/>`,
addToTopbar = () => {
const $topbarFavorite = document.querySelector(topbarFavorite);
if (!$topbarFavorite) return;
$topbarFavorite.after($pin, $unpin);
removeMutationListener(addToTopbar);
if (document.contains($pin)) removeMutationListener(addToTopbar);
document.querySelector(topbarFavorite)?.after($pin, $unpin);
};
html`<${Tooltip}><b>${pinTooltip}</b><//>`.attach($pin, "bottom");
html`<${Tooltip}><b>${unpinTooltip}</b><//>`.attach($unpin, "bottom");

View File

@ -38,29 +38,16 @@ export default (async () => {
// extension:// pages can access chrome apis
// in both situations, modules that attach to
// the dom must be re-imported, and should not
// be used until import is complete, otherwise
// their local states will be cleared (e.g.,
// references to registered hotkeys)
const _state = globalThis.__enhancerApi?.dumpState?.();
// 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")),
import(enhancerUrl("vendor/twind.min.js")),
import(enhancerUrl("vendor/lucide.min.js")),
import(enhancerUrl("vendor/htm.min.js")),
]);
await Promise.all([
!(!IS_MENU || !IS_ELECTRON) || import(enhancerUrl("common/registry.js")),
import(enhancerUrl("common/scaffold.mjs")),
import(enhancerUrl("common/events.js")),
import(enhancerUrl("common/markup.js")),
]);
// copy across state from prev. module imports if
// useState / setState are called early, otherwise
// e.g. menu will not persist theme from initial msg
globalThis.__enhancerApi.setState(_state ?? {});
globalThis.__enhancerApi.__isReady(globalThis.__enhancerApi);
const { getMods, isEnabled, modDatabase } = globalThis.__enhancerApi;
for (const mod of await getMods()) {
if (!(await isEnabled(mod.id))) continue;
@ -85,5 +72,4 @@ export default (async () => {
}
if (IS_MENU) console.log("notion-enhancer: ready");
globalThis.__enhancerApi.__isReady(globalThis.__enhancerApi);
})();

4
src/vendor/@unocss-core.mjs vendored Normal file

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,387 @@
/*
1. Prevent padding and border from affecting element width. (https://github.com/mozdevs/cssremedy/issues/4)
2. Allow adding a border to an element by just adding a border-width. (https://github.com/tailwindcss/tailwindcss/pull/116)
2. [UnoCSS]: allow to override the default border color with css var `--un-default-border-color`
*/
*,
::before,
::after {
box-sizing: border-box; /* 1 */
border-width: 0; /* 2 */
border-style: solid; /* 2 */
border-color: var(--un-default-border-color, #e5e7eb); /* 2 */
}
::before,
::after {
--un-content: '';
}
/*
1. Use a consistent sensible line-height in all browsers.
2. Prevent adjustments of font size after orientation changes in iOS.
3. Use a more readable tab size.
4. Use the user's configured `sans` font-family by default.
5. Use the user's configured `sans` font-feature-settings by default.
6. Use the user's configured `sans` font-variation-settings by default.
7. Disable tap highlights on iOS.
*/
html,
:host {
line-height: 1.5; /* 1 */
-webkit-text-size-adjust: 100%; /* 2 */
-moz-tab-size: 4; /* 3 */
tab-size: 4; /* 3 */
font-family: ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"; /* 4 */
font-feature-settings: normal; /* 5 */
font-variation-settings: normal; /* 6 */
-webkit-tap-highlight-color: transparent; /* 7 */
}
/*
1. Remove the margin in all browsers.
2. Inherit line-height from `html` so users can set them as a class directly on the `html` element.
*/
body {
margin: 0; /* 1 */
line-height: inherit; /* 2 */
}
/*
1. Add the correct height in Firefox.
2. Correct the inheritance of border color in Firefox. (https://bugzilla.mozilla.org/show_bug.cgi?id=190655)
3. Ensure horizontal rules are visible by default.
*/
hr {
height: 0; /* 1 */
color: inherit; /* 2 */
border-top-width: 1px; /* 3 */
}
/*
Add the correct text decoration in Chrome, Edge, and Safari.
*/
abbr:where([title]) {
text-decoration: underline dotted;
}
/*
Remove the default font size and weight for headings.
*/
h1,
h2,
h3,
h4,
h5,
h6 {
font-size: inherit;
font-weight: inherit;
}
/*
Reset links to optimize for opt-in styling instead of opt-out.
*/
a {
color: inherit;
text-decoration: inherit;
}
/*
Add the correct font weight in Edge and Safari.
*/
b,
strong {
font-weight: bolder;
}
/*
1. Use the user's configured `mono` font-family by default.
2. Use the user's configured `mono` font-feature-settings by default.
3. Use the user's configured `mono` font-variation-settings by default.
4. Correct the odd `em` font sizing in all browsers.
*/
code,
kbd,
samp,
pre {
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; /* 1 */
font-feature-settings: normal; /* 2 */
font-variation-settings: normal; /* 3 */
font-size: 1em; /* 4 */
}
/*
Add the correct font size in all browsers.
*/
small {
font-size: 80%;
}
/*
Prevent `sub` and `sup` elements from affecting the line height in all browsers.
*/
sub,
sup {
font-size: 75%;
line-height: 0;
position: relative;
vertical-align: baseline;
}
sub {
bottom: -0.25em;
}
sup {
top: -0.5em;
}
/*
1. Remove text indentation from table contents in Chrome and Safari. (https://bugs.chromium.org/p/chromium/issues/detail?id=999088, https://bugs.webkit.org/show_bug.cgi?id=201297)
2. Correct table border color inheritance in all Chrome and Safari. (https://bugs.chromium.org/p/chromium/issues/detail?id=935729, https://bugs.webkit.org/show_bug.cgi?id=195016)
3. Remove gaps between table borders by default.
*/
table {
text-indent: 0; /* 1 */
border-color: inherit; /* 2 */
border-collapse: collapse; /* 3 */
}
/*
1. Change the font styles in all browsers.
2. Remove the margin in Firefox and Safari.
3. Remove default padding in all browsers.
*/
button,
input,
optgroup,
select,
textarea {
font-family: inherit; /* 1 */
font-feature-settings: inherit; /* 1 */
font-variation-settings: inherit; /* 1 */
font-size: 100%; /* 1 */
font-weight: inherit; /* 1 */
line-height: inherit; /* 1 */
color: inherit; /* 1 */
margin: 0; /* 2 */
padding: 0; /* 3 */
}
/*
Remove the inheritance of text transform in Edge and Firefox.
*/
button,
select {
text-transform: none;
}
/*
1. Correct the inability to style clickable types in iOS and Safari.
2. Remove default button styles.
*/
button,
[type='button'],
[type='reset'],
[type='submit'] {
-webkit-appearance: button; /* 1 */
background-color: transparent; /* 2 */
background-image: none; /* 2 */
}
/*
Use the modern Firefox focus style for all focusable elements.
*/
:-moz-focusring {
outline: auto;
}
/*
Remove the additional `:invalid` styles in Firefox. (https://github.com/mozilla/gecko-dev/blob/2f9eacd9d3d995c937b4251a5557d95d494c9be1/layout/style/res/forms.css#L728-L737)
*/
:-moz-ui-invalid {
box-shadow: none;
}
/*
Add the correct vertical alignment in Chrome and Firefox.
*/
progress {
vertical-align: baseline;
}
/*
Correct the cursor style of increment and decrement buttons in Safari.
*/
::-webkit-inner-spin-button,
::-webkit-outer-spin-button {
height: auto;
}
/*
1. Correct the odd appearance in Chrome and Safari.
2. Correct the outline style in Safari.
*/
[type='search'] {
-webkit-appearance: textfield; /* 1 */
outline-offset: -2px; /* 2 */
}
/*
Remove the inner padding in Chrome and Safari on macOS.
*/
::-webkit-search-decoration {
-webkit-appearance: none;
}
/*
1. Correct the inability to style clickable types in iOS and Safari.
2. Change font properties to `inherit` in Safari.
*/
::-webkit-file-upload-button {
-webkit-appearance: button; /* 1 */
font: inherit; /* 2 */
}
/*
Add the correct display in Chrome and Safari.
*/
summary {
display: list-item;
}
/*
Removes the default spacing for appropriate elements.
*/
blockquote,
dl,
dd,
h1,
h2,
h3,
h4,
h5,
h6,
hr,
figure,
p,
pre {
margin: 0;
}
fieldset {
margin: 0;
padding: 0;
}
legend {
padding: 0;
}
ol,
ul,
menu {
list-style: none;
margin: 0;
padding: 0;
}
dialog {
padding: 0;
}
/*
Prevent resizing textareas horizontally by default.
*/
textarea {
resize: vertical;
}
/*
1. Reset the default placeholder opacity in Firefox. (https://github.com/tailwindlabs/tailwindcss/issues/3300)
2. Set the default placeholder color to the user's configured gray 400 color.
*/
input::placeholder,
textarea::placeholder {
opacity: 1; /* 1 */
color: #9ca3af; /* 2 */
}
/*
Set the default cursor for buttons.
*/
button,
[role="button"] {
cursor: pointer;
}
/*
Make sure disabled buttons don't get the pointer cursor.
*/
:disabled {
cursor: default;
}
/*
1. Make replaced elements `display: block` by default. (https://github.com/mozdevs/cssremedy/issues/14)
2. Add `vertical-align: middle` to align replaced elements more sensibly by default. (https://github.com/jensimmons/cssremedy/issues/14#issuecomment-634934210)
This can trigger a poorly considered lint error in some tools but is included by design.
*/
img,
svg,
video,
canvas,
audio,
iframe,
embed,
object {
display: block; /* 1 */
vertical-align: middle; /* 2 */
}
/*
Constrain images and videos to the parent width and preserve their intrinsic aspect ratio. (https://github.com/mozdevs/cssremedy/issues/14)
*/
img,
video {
max-width: 100%;
height: auto;
}
/*
Make elements with the HTML hidden attribute stay hidden by default.
*/
[hidden] {
display: none;
}

26
src/vendor/@unocss-preset-icons.mjs vendored Normal file

File diff suppressed because one or more lines are too long

27
src/vendor/@unocss-preset-uno.mjs vendored Normal file

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1 +0,0 @@
!function(){var n=(new Map,function(n){for(var e,l,s=arguments,t=1,u="",r="",o=[0],f=function(n){1===t&&(n||(u=u.replace(/^\s*\n\s*|\s*\n\s*$/g,"")))?o.push(n?s[n]:u):3===t&&(n||u)?(o[1]=n?s[n]:u,t=2):2===t&&"..."===u&&n?o[2]=Object.assign(o[2]||{},s[n]):2===t&&u&&!n?(o[2]=o[2]||{})[u]=!0:t>=5&&(5===t?((o[2]=o[2]||{})[l]=n?u?u+s[n]:s[n]:u,t=6):(n||u)&&(o[2][l]+=n?u+s[n]:u)),u=""},i=0;i<n.length;i++){i&&(1===t&&f(),f(i));for(var p=0;p<n[i].length;p++)e=n[i][p],1===t?"<"===e?(f(),o=[o,"",null],t=3):u+=e:4===t?"--"===u&&">"===e?(t=1,u=""):u=e+u[0]:r?e===r?r="":u+=e:'"'===e||"'"===e?r=e:">"===e?(f(),t=1):t&&("="===e?(t=5,l=u,u=""):"/"===e&&(t<5||">"===n[i][p+1])?(f(),3===t&&(o=o[0]),t=o,(o=o[0]).push(this.apply(null,t.slice(1))),t=0):" "===e||"\t"===e||"\n"===e||"\r"===e?(f(),t=2):u+=e),3===t&&"!--"===u&&(t=4,o=o[0])}return f(),o.length>2?o.slice(1):o[1]});"undefined"!=typeof module?module.exports=n:self.htm=n}();

4
src/vendor/htm.mjs vendored Normal file
View File

@ -0,0 +1,4 @@
/* esm.sh - esbuild bundle(htm@3.1.1) es2022 production */
var a=function(p,f,c,n){var l;f[0]=0;for(var u=1;u<f.length;u++){var g=f[u++],o=f[u]?(f[0]|=g?1:2,c[f[u++]]):f[++u];g===3?n[0]=o:g===4?n[1]=Object.assign(n[1]||{},o):g===5?(n[1]=n[1]||{})[f[++u]]=o:g===6?n[1][f[++u]]+=o+"":g?(l=p.apply(o,a(p,o,c,["",null])),n.push(l),o[0]?f[0]|=2:(f[u-2]=0,f[u]=l)):n.push(o)}return n},M=new Map;function b(p){var f=M.get(this);return f||(f=new Map,M.set(this,f)),(f=a(this,f.get(p)||(f.set(p,f=function(c){for(var n,l,u=1,g="",o="",i=[0],s=function(v){u===1&&(v||(g=g.replace(/^\s*\n\s*|\s*\n\s*$/g,"")))?i.push(0,v,g):u===3&&(v||g)?(i.push(3,v,g),u=2):u===2&&g==="..."&&v?i.push(4,v,0):u===2&&g&&!v?i.push(5,0,!0,g):u>=5&&((g||!v&&u===5)&&(i.push(u,0,g,l),u=6),v&&(i.push(u,v,0,l),u=6)),g=""},t=0;t<c.length;t++){t&&(u===1&&s(),s(t));for(var w=0;w<c[t].length;w++)n=c[t][w],u===1?n==="<"?(s(),i=[i],u=3):g+=n:u===4?g==="--"&&n===">"?(u=1,g=""):g=n+g[0]:o?n===o?o="":g+=n:n==='"'||n==="'"?o=n:n===">"?(s(),u=1):u&&(n==="="?(u=5,l=g,g=""):n==="/"&&(u<5||c[t][w+1]===">")?(s(),u===3&&(i=i[0]),u=i,(i=i[0]).push(2,0,u),u=0):n===" "||n===" "||n===`
`||n==="\r"?(s(),u=2):g+=n),u===3&&g==="!--"&&(u=4,i=i[0])}return s(),i}(p)),f),arguments,[])).length>1?f:f[0]}export{b as default};
//# sourceMappingURL=htm.bundle.mjs.map

File diff suppressed because one or more lines are too long

16
src/vendor/lucide.mjs vendored Normal file

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long