feat: render headings in outliner

This commit is contained in:
dragonwocky 2024-01-31 00:12:14 +11:00
parent ff1e5f7550
commit 6661c5559b
Signed by: dragonwocky
GPG Key ID: 7998D08F7D7BD7A8
6 changed files with 88 additions and 122 deletions

View File

@ -63,19 +63,18 @@ const isEnabled = async (id) => {
};
const modDatabase = async (id) => {
const optionDefaults =
(await getMods())
.find((mod) => mod.id === id)
?.options?.map?.((opt) => {
let value = opt.value;
value ??= opt.values?.[0]?.value;
value ??= opt.values?.[0];
return [opt.key, value];
})
?.filter?.(([, value]) => typeof value !== "undefined") ?? {};
const optionDefaults = (await getMods())
.find((mod) => mod.id === id)
?.options?.map?.((opt) => {
let value = opt.value;
value ??= opt.values?.[0]?.value;
value ??= opt.values?.[0];
return [opt.key, value];
})
?.filter?.(([, value]) => typeof value !== "undefined");
return globalThis.__enhancerApi.initDatabase(
[await getProfile(), id],
Object.fromEntries(optionDefaults)
Object.fromEntries(optionDefaults ?? [])
);
};

View File

@ -58,11 +58,11 @@ const insertMenu = async (api, db) => {
let _contentWindow;
const updateMenuTheme = () => {
const darkMode = document.body.classList.contains("dark"),
notionTheme = darkMode ? "dark" : "light";
menuPing.theme = notionTheme;
_contentWindow?.postMessage?.(menuPing, "*");
};
const darkMode = document.body.classList.contains("dark"),
notionTheme = darkMode ? "dark" : "light";
menuPing.theme = notionTheme;
_contentWindow?.postMessage?.(menuPing, "*");
};
const $modal = html`<${Modal}>
<${Frame}
@ -122,14 +122,13 @@ const insertMenu = async (api, db) => {
const insertPanel = async (api, db) => {
const notionFrame = ".notion-frame",
togglePanelHotkey = await db.get("togglePanelHotkey"),
{ addMutationListener, removeMutationListener } = api,
{ html, setState, addPanelView } = api;
{ html, setState, addMutationListener, removeMutationListener } = api;
const $panel = html`<${Panel}
hotkey="${togglePanelHotkey}"
...${Object.assign(
...["Width", "Open", "View"].map((key) => ({
[`_get${key}`]: () => db.get(`sidePanel${key}`),
[`_get${key}`]: () => db.get(`panel${key}`),
[`_set${key}`]: async (value) => {
await db.set(`panel${key}`, value);
setState({ rerender: true });
@ -146,27 +145,6 @@ const insertPanel = async (api, db) => {
};
addMutationListener(notionFrame, appendToDom);
appendToDom();
const $helloThere = html`<div class="p-[16px]">hello there</div>`,
$generalKenobi = html`<div class="p-[16px]">general kenobi</div>`;
addPanelView({
title: "outliner",
// prettier-ignore
$icon: html`<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24">
<circle cx="5" cy="7" r="2.8"/>
<circle cx="5" cy="17" r="2.79"/>
<path d="M21,5.95H11c-0.55,0-1-0.45-1-1v0c0-0.55,0.45-1,1-1h10c0.55,0,1,0.45,1,1v0C22,5.5,21.55,5.95,21,5.95z"/>
<path d="M17,10.05h-6c-0.55,0-1-0.45-1-1v0c0-0.55,0.45-1,1-1h6c0.55,0,1,0.45,1,1v0C18,9.6,17.55,10.05,17,10.05z"/>
<path d="M21,15.95H11c-0.55,0-1-0.45-1-1v0c0-0.55,0.45-1,1-1h10c0.55,0,1,0.45,1,1v0C22,15.5,21.55,15.95,21,15.95z" />
<path d="M17,20.05h-6c-0.55,0-1-0.45-1-1v0c0-0.55,0.45-1,1-1h6c0.55,0,1,0.45,1,1v0C18,19.6,17.55,20.05,17,20.05z"/>
</svg>`,
$view: $helloThere,
});
addPanelView({
title: "word counter",
$icon: "type",
$view: $generalKenobi,
});
};
export default async (api, db) =>

View File

@ -1,22 +1,26 @@
/**
* notion-enhancer: outliner
* (c) 2020 CloudHill <rl.cloudhill@gmail.com> (https://github.com/CloudHill)
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (c) 2021 CloudHill <rl.cloudhill@gmail.com> (https://github.com/CloudHill)
* (c) 2024 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
export default async function ({ web, components }, db) {
const dbNoticeText = 'Open a page to see its table of contents.',
pageNoticeText = 'Click on a heading to jump to it.',
$notice = web.html`<p id="outliner--notice">${dbNoticeText}</p>`;
"use strict";
const $headingList = web.html`<div></div>`;
let viewFocused = false,
$page;
await components.addPanelView({
id: '87e077cc-5402-451c-ac70-27cc4ae65546',
icon: web.html`<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24">
export default async (api, db) => {
const { html, addMutationListener, addPanelView } = api,
frame = ".notion-sidebar-container + div",
page = ".notion-page-content",
headings = [
".notion-header-block",
".notion-sub_header-block",
".notion-sub_sub_header-block",
],
$view = html`<div></div>`;
addPanelView({
title: "Outliner",
// prettier-ignore
$icon: html`<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24">
<circle cx="5" cy="7" r="2.8"/>
<circle cx="5" cy="17" r="2.79"/>
<path d="M21,5.95H11c-0.55,0-1-0.45-1-1v0c0-0.55,0.45-1,1-1h10c0.55,0,1,0.45,1,1v0C22,5.5,21.55,5.95,21,5.95z"/>
@ -24,61 +28,47 @@ export default async function ({ web, components }, db) {
<path d="M21,15.95H11c-0.55,0-1-0.45-1-1v0c0-0.55,0.45-1,1-1h10c0.55,0,1,0.45,1,1v0C22,15.5,21.55,15.95,21,15.95z" />
<path d="M17,20.05h-6c-0.55,0-1-0.45-1-1v0c0-0.55,0.45-1,1-1h6c0.55,0,1,0.45,1,1v0C18,19.6,17.55,20.05,17,20.05z"/>
</svg>`,
title: 'Outliner',
$content: web.render(web.html`<div></div>`, $notice, $headingList),
onFocus: () => {
viewFocused = true;
$page = document.getElementsByClassName('notion-page-content')[0];
updateHeadings();
},
onBlur: () => {
viewFocused = false;
},
$view,
});
await web.whenReady();
function updateHeadings() {
if (!$page) return;
$notice.innerText = pageNoticeText;
$headingList.style.display = '';
const $headerBlocks = $page.querySelectorAll('[class^="notion-"][class*="header-block"]'),
$fragment = web.html`<div></div>`;
let indent = 0;
for (const $header of $headerBlocks) {
const id = $header.dataset.blockId.replace(/-/g, ''),
placeholder = $header.querySelector('[placeholder]').getAttribute('placeholder'),
headerDepth = +[...placeholder].reverse()[0];
indent = (headerDepth-1) * 18;
const $outlineHeader = web.render(
web.html`<a href="#${id}" class="outliner--header"
placeholder="${web.escape(placeholder)}"
style="--outliner--indent:${indent}px;"></a>`,
$header.innerText
);
$outlineHeader.addEventListener('click', (event) => {
location.hash = '';
});
web.render($fragment, $outlineHeader);
}
if ($fragment.innerHTML !== $headingList.innerHTML) {
web.render(web.empty($headingList), ...$fragment.children);
}
function Heading({ indent, ...props }, ...children) {
return html`<div
role="button"
class="notion-enhancer--outliner-heading
block cursor-pointer select-none text-[14px]
m-px py-[6px] pr-[2px] pl-[${indent * 18}px]
decoration-(2 [color:var(--theme--fg-border)])
hover:bg-[color:var(--theme--bg-hover)]
underline-(& offset-4)"
...${props}
>
${children}
</div>`;
}
const pageObserver = () => {
if (!viewFocused) return;
if (document.contains($page)) {
updateHeadings();
} else {
$page = document.querySelector('.notion-page-content');
if (!$page) {
$notice.innerText = dbNoticeText;
$headingList.style.display = 'none';
} else updateHeadings();
}
};
web.addDocumentObserver(pageObserver, [
'.notion-page-content',
'.notion-collection_view_page-block',
]);
pageObserver();
}
const getHeadings = () => {
return [...document.querySelectorAll(headings.join(", "))];
},
getHeadingLevel = ($heading) => {
for (let i = 0; i < headings.length; i++)
if ($heading.matches(headings[i])) return i + 1;
},
updateHeadings = () => {
$view.innerHTML = "";
for (const $heading of getHeadings()) {
const title = $heading.innerText,
indent = getHeadingLevel($heading);
if (!title) continue;
const $h = html`<${Heading} indent=${indent} onclick=${() => {
// todo: scroll into view
}}>${title}</p>`;
$view.append($h);
}
};
let $page;
addMutationListener(page, () => {
if (!document.contains($page)) $page = document.querySelector(page);
if ($page) updateHeadings();
});
};

View File

@ -1,23 +1,21 @@
{
"name": "outliner",
"name": "Outliner",
"version": "0.5.0",
"id": "87e077cc-5402-451c-ac70-27cc4ae65546",
"version": "0.4.0",
"description": "adds a table of contents to the side panel.",
"preview": "outliner.png",
"tags": ["extension", "panel"],
"description": "Adds a table of contents to the side panel to overview and navigate the current page's headings and subheadings.",
"thumbnail": "outliner.png",
"tags": ["panel"],
"authors": [
{
"name": "dragonwocky",
"homepage": "https://dragonwocky.me/",
"avatar": "https://dragonwocky.me/avatar.jpg"
},
{
"name": "CloudHill",
"email": "rh.cloudhill@gmail.com",
"homepage": "https://github.com/CloudHill",
"avatar": "https://avatars.githubusercontent.com/u/54142180"
}
],
"js": {
"client": ["client.mjs"]
},
"css": {
"client": ["client.css"]
},
"options": []
"clientScripts": ["client.mjs"]
}

View File

@ -27,7 +27,7 @@
"type": "number",
"key": "distanceScrolledUntilShown",
"description": "How far down a page you must be scrolled for the scroll to top button to appear.",
"value": 50
"value": 0
},
{
"type": "select",

View File

@ -2,6 +2,7 @@
"core",
"extensions/titlebar",
"extensions/topbar",
"extensions/outliner",
"extensions/scroll-to-top",
"themes/classic-dark"
]