feat(panel): add tooltips to panel toggles

This commit is contained in:
dragonwocky 2024-01-20 23:58:12 +11:00
parent 68b73246fb
commit 23f8e3b560
Signed by: dragonwocky
GPG Key ID: D408552AAEF956CB
7 changed files with 148 additions and 119 deletions

View File

@ -112,6 +112,7 @@ const modifierAliases = [
keyListeners = keyListeners.filter(([, c]) => c !== callback);
},
handleKeypress = (event, keyListeners) => {
console.log(event);
for (const [accelerator, callback] of keyListeners) {
const acceleratorModifiers = [],
combinationTriggered =
@ -125,6 +126,7 @@ const modifierAliases = [
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 = [

View File

@ -121,22 +121,31 @@ const insertMenu = async (api, db) => {
const insertPanel = async (api, db) => {
const notionFrame = ".notion-frame",
notionTopbarBtn = ".notion-topbar-more-button",
togglePanelHotkey = await db.get("togglePanelHotkey"),
{ addPanelView, addMutationListener, addKeyListener } = api,
{ html, setState, useState } = api;
{ addPanelView, addMutationListener } = api,
{ html, setState } = api;
const $panel = html`<${Panel}
...${Object.assign(
...["Width", "Open", "View"].map((key) => ({
[`_get${key}`]: () => db.get(`sidePanel${key}`),
[`_set${key}`]: async (value) => {
await db.set(`sidePanel${key}`, value);
setState({ rerender: true });
},
}))
)}
/>`;
hotkey="${togglePanelHotkey}"
...${Object.assign(
...["Width", "Open", "View"].map((key) => ({
[`_get${key}`]: () => db.get(`sidePanel${key}`),
[`_set${key}`]: async (value) => {
await db.set(`panel${key}`, value);
setState({ rerender: true });
},
}))
)}
/>`,
appendToDom = () => {
const $frame = document.querySelector(notionFrame);
if (!$frame) return;
if (!$frame.contains($panel)) $frame.append($panel);
if (!$frame.style.flexDirection !== "row")
$frame.style.flexDirection = "row";
};
addMutationListener(notionFrame, appendToDom);
appendToDom();
const $helloThere = html`<div class="p-[16px]">hello there</div>`,
$generalKenobi = html`<div class="p-[16px]">general kenobi</div>`;
@ -158,43 +167,6 @@ const insertPanel = async (api, db) => {
$icon: "type",
$view: $generalKenobi,
});
// setTimeout(() => {
// removePanelView($helloThere);
// removePanelView($generalKenobi);
// }, 5000);
const $panelTopbarBtn = html`<${TopbarButton}
aria-label="Open side panel"
icon="panel-right"
onclick=${$panel.toggle}
/>`,
appendToDom = () => {
const $frame = document.querySelector(notionFrame);
if (!$frame) return;
if (!$frame.contains($panel)) $frame.append($panel);
if (!$frame.style.flexDirection !== "row")
$frame.style.flexDirection = "row";
if (!document.contains($panelTopbarBtn)) {
const $notionTopbarBtn = document.querySelector(notionTopbarBtn);
$notionTopbarBtn?.before($panelTopbarBtn);
}
};
addMutationListener(`${notionFrame}, ${notionTopbarBtn}`, appendToDom);
appendToDom();
useState(["panelOpen"], ([panelOpen]) => {
if (panelOpen) $panelTopbarBtn.setAttribute("data-active", true);
else $panelTopbarBtn.removeAttribute("data-active");
});
useState(["panelViews"], ([panelViews = []]) => {
$panelTopbarBtn.style.display = panelViews.length ? "" : "none";
});
addKeyListener(togglePanelHotkey, (event) => {
event.preventDefault();
event.stopPropagation();
$panel.toggle();
});
};
export default async (api, db) =>

View File

@ -6,6 +6,7 @@
*/
import { Tooltip } from "./Tooltip.mjs";
import { TopbarButton } from "./TopbarButton.mjs";
import { Select } from "../menu/islands/Select.mjs";
// note: these islands are not reusable.
@ -48,15 +49,12 @@ function View({ _get }) {
}
function Switcher({ _get, _set, minWidth, maxWidth }) {
const { html, setState, useState } = globalThis.__enhancerApi,
const { html, useState } = globalThis.__enhancerApi,
$switcher = html`<div
class="relative flex items-center grow
font-medium p-[8.5px] ml-[4px] select-none"
></div>`,
setView = (view) => {
_set?.(view);
setState({ activePanelView: view });
};
setView = (view) => _set?.(view);
useState(["panelViews"], ([panelViews = []]) => {
const values = panelViews.map(([{ title, $icon }]) => {
// panel switcher internally uses the select island,
@ -76,6 +74,7 @@ function Switcher({ _get, _set, minWidth, maxWidth }) {
}
function Panel({
hotkey,
_getWidth,
_setWidth,
_getOpen,
@ -86,8 +85,24 @@ function Panel({
maxWidth = 640,
transitionDuration = 300,
}) {
const { html, setState, useState } = globalThis.__enhancerApi,
const { html, useState, addKeyListener } = globalThis.__enhancerApi,
{ addMutationListener, removeMutationListener } = globalThis.__enhancerApi,
$topbarToggle = html`<${TopbarButton}
aria-label="Toggle side panel"
icon="panel-right"
/>`,
$panelToggle = html`<button
aria-label="Toggle side panel"
class="user-select-none h-[24px] w-[24px] duration-[20ms]
transition inline-flex items-center justify-center mr-[10px]
rounded-[3px] hover:bg-[color:var(--theme--bg-hover)]"
>
<i
class="i-chevrons-left w-[20px] h-[20px]
text-[color:var(--theme--fg-secondary)] transition-transform
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)]"
@ -118,23 +133,13 @@ function Panel({
<${Switcher}
...${{ _get: _getView, _set: _setView, minWidth, maxWidth }}
/>
<button
aria-label="Toggle side panel"
class="user-select-none h-[24px] w-[24px] duration-[20ms]
transition inline-flex items-center justify-center mr-[10px]
rounded-[3px] hover:bg-[color:var(--theme--bg-hover)]"
onclick=${() => $panel.toggle()}
>
<i
class="i-chevrons-left w-[20px] h-[20px]
text-[color:var(--theme--fg-secondary)] transition-transform
group-&[data-pinned]/panel:rotate-180 duration-[${transitionDuration}ms]"
/>
</button>
${$panelToggle}
</div>
<${View} ...${{ _get: _getView }} />
</aside>
</div>`;
$panelToggle.onclick = $topbarToggle.onclick = () => $panel.toggle();
$topbarToggle.addToTopbar();
let preDragWidth, dragStartX, _animatedAt;
const getWidth = async (width) => {
@ -206,20 +211,6 @@ function Panel({
group-&[data-peeked]/panel:(my-px h-[calc(100%-2px)] rounded-l-[6px])"
></div>
</div>`,
$onResizeClick = html`<span>close</span>`,
$resizeTooltip = html`<${Tooltip}>
<b>Drag</b> to resize<br />
<b>Click</b> to ${$onResizeClick}
<//>`,
showTooltip = (event) => {
setTimeout(() => {
if (!$resizeHandle.matches(":hover")) return;
const open = $panel.hasAttribute("open"),
{ x } = $resizeHandle.getBoundingClientRect();
$onResizeClick.innerText = open ? "close" : "lock open";
$resizeTooltip.show(x, event.clientY);
}, 200);
},
startDrag = async (event) => {
dragStartX = event.clientX;
preDragWidth = await getWidth();
@ -240,11 +231,43 @@ function Panel({
if (dragStartX - event.clientX === 0) $panel.toggle();
preDragWidth = dragStartX = undefined;
};
$resizeHandle.addEventListener("mouseout", $resizeTooltip.hide);
$resizeHandle.addEventListener("mousedown", startDrag);
$resizeHandle.addEventListener("mouseover", showTooltip);
$panel.lastElementChild.prepend($resizeHandle);
// add tooltips to panel pin/unpin toggles
const $resizeTooltipClick = html`<span></span>`,
$resizeTooltip = html`<${Tooltip}
onbeforeshow=${() => {
$resizeTooltipClick.innerText = isPinned() ? "close" : "lock open";
}}
><b>Drag</b> to resize<br />
<b>Click</b> to ${$resizeTooltipClick}
<//>`,
$toggleTooltipClick = html`<b></b>`,
$toggleTooltip = html`<${Tooltip}
class="text-start"
onbeforeshow=${() => {
$toggleTooltipClick.innerText = isPinned()
? "Close sidebar"
: "Lock sidebar open";
}}
>${$toggleTooltipClick}<br />
${hotkey}
<//>`,
alignTooltipBelow = ($target) => {
const rect = $target.getBoundingClientRect();
return {
x: rect.right,
y: rect.bottom + $toggleTooltip.clientHeight / 2 + 6,
};
};
$resizeTooltip.attach($resizeHandle, () => {
const rect = $resizeHandle.getBoundingClientRect();
return { x: rect.x - 6 };
});
$toggleTooltip.attach($topbarToggle, () => alignTooltipBelow($topbarToggle));
$toggleTooltip.attach($panelToggle, () => alignTooltipBelow($panelToggle));
// hovering over the peek trigger will temporarily
// pop out an interactive preview of the panel
let _peekDebounce;
@ -274,10 +297,9 @@ function Panel({
if (!$notionHelp) return;
width ??= await getWidth();
if (isNaN(width)) width = minWidth;
if (!$panel.hasAttribute("open")) width = 0;
if (isClosed()) width = 0;
const to = `${26 + width}px`,
from = $notionHelp.style.getPropertyValue("right"),
opts = { duration: transitionDuration, easing };
from = $notionHelp.style.getPropertyValue("right");
if (from === to) return;
$notionHelp.style.setProperty("right", to);
animate($notionHelp, [({ right: from }, { right: to })]);
@ -292,9 +314,9 @@ function Panel({
animate($panel, [closedWidth, openWidth]);
$panel.removeAttribute("data-peeked");
$panel.dataset.pinned = true;
$topbarToggle.setAttribute("data-active", true);
setInteractive(true);
_setOpen(true);
setState({ panelOpen: true });
$panel.resize();
};
$panel.peek = () => {
@ -314,8 +336,8 @@ function Panel({
};
$panel.close = async () => {
if (isClosed()) return;
setState({ panelOpen: false });
if (panelViews.length) _setOpen(false);
$topbarToggle.removeAttribute("data-active");
const width = (animationState.width = `${await getWidth()}px`);
// only animate container close if it is actually taking up space,
// otherwise will unnaturally grow + retrigger peek on peek mouseout
@ -348,9 +370,18 @@ function Panel({
};
useState(["panelViews"], async ([panelViews = []]) => {
$topbarToggle.style.display = panelViews.length ? "" : "none";
if (panelViews.length && (await _getOpen())) $panel.pin();
else $panel.close();
});
if (!hotkey) return $panel;
addKeyListener(hotkey, (event) => {
event.preventDefault();
event.stopPropagation();
$panel.toggle();
});
return $panel;
}

View File

@ -8,16 +8,16 @@ function Tooltip(props, ...children) {
const { html, extendProps } = globalThis.__enhancerApi;
extendProps(props, {
role: "dialog",
class: `absolute group/tooltip z-[999] pointer-events-none`,
class: `absolute group/tooltip z-[999] text-center pointer-events-none`,
});
const notionApp = ".notion-app-inner",
$tooltip = html`<div ...${props}>
<div
class="bg-[color:var(--theme--bg-secondary)]
text-([color:var(--theme--fg-secondary)] [12px] center)
text-([color:var(--theme--fg-secondary)] [12px] nowrap)
leading-[1.4] font-medium py-[4px] px-[8px] rounded-[4px]
drop-shadow-md transition duration-200 opacity-0
drop-shadow-md transition duration-100 opacity-0
group-open/tooltip:(pointer-events-auto opacity-100)
&>b:text-[color:var(--theme--fg-primary)]"
>
@ -27,12 +27,14 @@ function Tooltip(props, ...children) {
$tooltip.show = (x, y) => {
const $notionApp = document.querySelector(notionApp);
if (!document.contains($tooltip)) $notionApp?.append($tooltip);
if ($tooltip.hasAttribute("open")) return;
requestAnimationFrame(() => {
$tooltip.onbeforeshow?.();
$tooltip.setAttribute("open", true);
x -= $tooltip.clientWidth + 6;
if (x < 0) x += $tooltip.clientWidth + 12;
x -= $tooltip.clientWidth;
if (x < 0) x = $tooltip.clientWidth + 12;
y -= $tooltip.clientHeight / 2;
if (y < 0) y += $tooltip.clientHeight / 2;
if (y < 0) y = $tooltip.clientHeight + 12;
$tooltip.style.left = `${x}px`;
$tooltip.style.top = `${y}px`;
$tooltip.onshow?.();
@ -45,6 +47,16 @@ function Tooltip(props, ...children) {
$tooltip.onhide?.();
}, 200);
};
$tooltip.attach = ($target, calcPos) => {
$target.addEventListener("mouseover", (event) => {
setTimeout(() => {
if (!$target.matches(":hover")) return;
const { x = event.clientX, y = event.clientY } = calcPos?.(event) ?? {};
$tooltip.show(x, y);
}, 200);
});
$target.addEventListener("mouseout", $tooltip.hide);
};
return $tooltip;
}

View File

@ -4,8 +4,8 @@
* (https://notion-enhancer.github.io/) under the MIT license
*/
function TopbarButton({ icon, ...props }, ...children) {
const { html, extendProps } = globalThis.__enhancerApi;
function TopbarButton({ icon, ...props }) {
const { html, extendProps, addMutationListener } = globalThis.__enhancerApi;
extendProps(props, {
tabindex: 0,
role: "button",
@ -15,12 +15,23 @@ function TopbarButton({ icon, ...props }, ...children) {
rounded-[3px] hover:bg-[color:var(--theme--bg-hover)]
&[data-active]:bg-[color:var(--theme--bg-hover)]`,
});
return html`<button ...${props}>
<i
class="i-${icon} w-[20px] h-[20px]
fill-[color:var(--theme--fg-secondary)]"
/>
</button>`;
const notionTopbar = ".notion-topbar-favorite-button",
$button = html`<button ...${props}>
<i
class="i-${icon} w-[20px] h-[20px]
fill-[color:var(--theme--fg-secondary)]"
/>
</button>`,
addToTopbar = () => {
if (document.contains($button)) return;
document.querySelector(notionTopbar)?.after($button);
};
$button.addToTopbar = () => {
addToTopbar();
addMutationListener(notionTopbar, addToTopbar);
};
return $button;
}
export { TopbarButton };

View File

@ -22,6 +22,7 @@ const updateHotkey = (event) => {
if (key === " ") key = "Space";
if (["+", "="].includes(key)) key = "Plus";
if (key === "-") key = "Minus";
if (key === "|") key = "\\";
if (event.code === "Comma") key = ",";
if (event.code === "Period") key = ".";
if (key === "Control") key = "Ctrl";

View File

@ -24,25 +24,25 @@
},
{
"type": "hotkey",
"key": "toggleWindowHotkey",
"description": "Toggles focus of the Notion window anywhere, even when your Notion app isn't active.",
"value": "Ctrl+Shift+A"
"key": "togglePanelHotkey",
"description": "Toggles the side panel used by some notion-enhancer extensions to display additional information and interfaces within the Notion app.",
"value": "Ctrl+Shift+\\"
},
{
"type": "hotkey",
"key": "togglePanelHotkey",
"description": "Toggles the side panel used by some notion-enhancer extensions to display additional information and interfaces within the Notion app.",
"value": "Ctrl+Shift+|"
"key": "toggleWindowHotkey",
"description": "Toggles focus of the Notion window anywhere, even when your Notion app isn't active.",
"value": "Ctrl+Shift+A"
},
{
"type": "heading",
"label": "Appearance"
},
{
"type": "select",
"key": "menuButtonIconStyle",
"description": "Sets whether the notion-enhancer icon added to Notion's sidebar should be coloured or monochrome. The latter style will match the theme's icon colour for users who would like the icon to be less noticeable.",
"values": ["Colour", "Monochrome"]
"type": "file",
"key": "customStyles",
"description": "Adds the styles from an uploaded .css file to Notion. Use this if you would like to customise the current theme or <a href=\"https://notion-enhancer.github.io/advanced/tweaks\">otherwise tweak Notion's appearance</a>.",
"extensions": ["css"]
},
{
"type": "select",
@ -51,10 +51,10 @@
"values": ["Auto", "Enabled", "Disabled"]
},
{
"type": "file",
"key": "customStyles",
"description": "Adds the styles from an uploaded .css file to Notion. Use this if you would like to customise the current theme or <a href=\"https://notion-enhancer.github.io/advanced/tweaks\">otherwise tweak Notion's appearance</a>.",
"extensions": ["css"]
"type": "select",
"key": "menuButtonIconStyle",
"description": "Sets whether the notion-enhancer icon added to Notion's sidebar should be coloured or monochrome. The latter style will match the theme's icon colour for users who would like the icon to be less noticeable.",
"values": ["Colour", "Monochrome"]
},
{
"type": "heading",
@ -66,7 +66,7 @@
"key": "developerMode",
"description": "Activates built-in debugging tools accessible through the application menu.",
"platforms": ["linux", "win32", "darwin"],
"value": false
"value": true
}
],
"clientStyles": ["variables.css"],