mirror of
https://github.com/notion-enhancer/notion-enhancer.git
synced 2025-04-04 04:39:03 +00:00
feat(menu): add searchbar + descriptions to mod lists, fix(menu): natural keyboard interactions w/ <Select> component
This commit is contained in:
parent
a8eb03ee67
commit
530be53e70
@ -87,9 +87,44 @@ function View({ id }, ...children) {
|
||||
return $el;
|
||||
}
|
||||
|
||||
function List({}, ...children) {
|
||||
function List({ description }, ...children) {
|
||||
const { html } = globalThis.__enhancerApi;
|
||||
return html`<div class="flex flex-col gap-y-[14px]">${children}</div>`;
|
||||
return html`<div class="flex flex-col gap-y-[14px]">
|
||||
<${Search} items=${children} />
|
||||
<p
|
||||
class="text-([12px] [color:var(--theme--fg-secondary)])"
|
||||
innerHTML=${description}
|
||||
></p>
|
||||
${children}
|
||||
</div>`;
|
||||
}
|
||||
|
||||
function Search({ items, oninput, ...props }) {
|
||||
const { html, addKeyListener } = globalThis.__enhancerApi,
|
||||
$search = html`<${Input}
|
||||
size="lg"
|
||||
type="text"
|
||||
icon="search"
|
||||
placeholder="Search ('/' to focus)"
|
||||
oninput=${(event) => {
|
||||
oninput?.(event);
|
||||
const query = event.target.value.toLowerCase();
|
||||
for (const $item of items) {
|
||||
const matches = $item.innerText.toLowerCase().includes(query);
|
||||
$item.style.display = matches ? "" : "none";
|
||||
}
|
||||
}}
|
||||
...${props}
|
||||
/>`;
|
||||
addKeyListener("/", (event) => {
|
||||
if (document.activeElement?.nodeName === "INPUT") return;
|
||||
// offsetParent == null if parent has "display: none;"
|
||||
if ($search.offsetParent) {
|
||||
event.preventDefault();
|
||||
$search.focus();
|
||||
}
|
||||
});
|
||||
return $search;
|
||||
}
|
||||
|
||||
function Mod({
|
||||
@ -211,7 +246,8 @@ function Option({ type, value, description, _get, _set, ...props }) {
|
||||
return html`<${type === "toggle" ? "label" : "div"}
|
||||
class="notion-enhancer--menu-option flex items-center justify-between
|
||||
mb-[18px] ${type === "toggle" ? "cursor-pointer" : ""}"
|
||||
><div class="flex flex-col ${type === "text" ? "w-full" : "mr-[10%]"}">
|
||||
>
|
||||
<div class="flex flex-col ${type === "text" ? "w-full" : "mr-[10%]"}">
|
||||
<h5 class="text-[14px] mb-[2px] mt-0">${label}</h5>
|
||||
${type === "text" ? $input : ""}
|
||||
<p
|
||||
@ -224,30 +260,34 @@ function Option({ type, value, description, _get, _set, ...props }) {
|
||||
<//>`;
|
||||
}
|
||||
|
||||
function Input({ wide, transparent, icon, onrerender, ...props }) {
|
||||
function Input({ size, icon, transparent, onrerender, ...props }) {
|
||||
const { html } = globalThis.__enhancerApi,
|
||||
$input = html`<input
|
||||
class="appearance-none h-[28px] w-full bg-transparent
|
||||
pl-[8px] pr-[32px] pb-px text-[14px] leading-[1.2]"
|
||||
class="${size === "lg"
|
||||
? "h-[36px] pl-[12px] pr-[40px]"
|
||||
: "h-[28px] pl-[8px] pr-[32px]"}
|
||||
w-full pb-px text-[14px] leading-[1.2]
|
||||
appearance-none bg-transparent"
|
||||
...${props}
|
||||
/>`,
|
||||
$icon = html`<i
|
||||
class="i-${icon} pointer-events-none
|
||||
absolute right-[8px] top-[6px] w-[16px] h-[16px]
|
||||
class="i-${icon} absolute w-[16px] h-[16px] pointer-events-none
|
||||
${size === "lg" ? "right-[12px] top-[10px]" : "right-[8px] top-[6px]"}
|
||||
text-[color:var(--theme--fg-secondary)]"
|
||||
></i>`;
|
||||
useState(["rerender"], () => onrerender?.($input, $icon));
|
||||
return html`<label
|
||||
focus=${() => $input.focus()}
|
||||
class="notion-enhancer--menu-input
|
||||
relative overflow-hidden rounded-[4px]
|
||||
${wide ? "block w-full mt-[4px] mb-[8px]" : "shrink-0 w-[192px]"}
|
||||
${transparent
|
||||
? `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)]"}"
|
||||
focus-within:ring-(& [color:var(--theme--accent-primary)])
|
||||
${size === "lg" ? "block w-full" : ""}
|
||||
${size === "md" ? "block w-full mt-[4px] mb-[8px]" : ""}
|
||||
${size === "sm" ? "shrink-0 w-[192px]" : ""}
|
||||
bg-${transparent
|
||||
? `([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])`
|
||||
: "[color:var(--theme--bg-hover)]"}"
|
||||
>${$input}${$icon}
|
||||
</label>`;
|
||||
}
|
||||
@ -255,7 +295,7 @@ function Input({ wide, transparent, icon, onrerender, ...props }) {
|
||||
function TextInput({ _get, _set, onchange, ...props }) {
|
||||
const { html } = globalThis.__enhancerApi;
|
||||
return html`<${Input}
|
||||
wide
|
||||
size="md"
|
||||
type="text"
|
||||
icon="text-cursor"
|
||||
onchange=${(event) => {
|
||||
@ -272,6 +312,7 @@ function TextInput({ _get, _set, onchange, ...props }) {
|
||||
function NumberInput({ _get, _set, onchange, ...props }) {
|
||||
const { html } = globalThis.__enhancerApi;
|
||||
return html`<${Input}
|
||||
size="sm"
|
||||
type="number"
|
||||
icon="hash"
|
||||
onchange=${(event) => {
|
||||
@ -314,6 +355,7 @@ function HotkeyInput({ _get, _set, onkeydown, ...props }) {
|
||||
event.target.dispatchEvent(new Event("change"));
|
||||
};
|
||||
return html`<${Input}
|
||||
size="sm"
|
||||
type="text"
|
||||
icon="command"
|
||||
onkeydown=${(event) => {
|
||||
@ -349,6 +391,7 @@ function ColorInput({ _get, _set, oninput, ...props }) {
|
||||
};
|
||||
return html`<${Input}
|
||||
transparent
|
||||
size="sm"
|
||||
type="text"
|
||||
icon="pipette"
|
||||
data-coloris
|
||||
@ -439,6 +482,9 @@ function Select({ values, _get, _set, ...props }) {
|
||||
transition duration-[20ms] hover:bg-[color:var(--theme--bg-hover)]"
|
||||
...${props}
|
||||
></div>`,
|
||||
$options = values.map((value) => {
|
||||
return html`<${SelectOption} ...${{ value, _get, _set }} />`;
|
||||
}),
|
||||
$popup = html`<div
|
||||
class="group absolute top-0 left-0
|
||||
flex flex-col justify-center items-end
|
||||
@ -452,38 +498,48 @@ function Select({ values, _get, _set, ...props }) {
|
||||
transition duration-[200ms] opacity-0 scale-95 rounded-[4px]
|
||||
group-open:(pointer-events-auto opacity-100 scale-100)"
|
||||
>
|
||||
${values.map((value) => {
|
||||
return html`<${SelectOption} ...${{ value, _get, _set }} />`;
|
||||
})}
|
||||
${$options}
|
||||
</div>
|
||||
</div>
|
||||
</div>`;
|
||||
|
||||
const { onclick } = $select;
|
||||
const { onclick, onkeydown } = $select,
|
||||
openPopup = () => {
|
||||
$popup.setAttribute("open", true);
|
||||
$options.forEach(($opt) => ($opt.tabIndex = 0));
|
||||
setState({ popupOpen: true });
|
||||
},
|
||||
closePopup = (value) => {
|
||||
$popup.removeAttribute("open");
|
||||
$options.forEach(($opt) => ($opt.tabIndex = -1));
|
||||
$select.style.width = `${$select.offsetWidth}px`;
|
||||
$select.style.background = "transparent";
|
||||
if (value) $select.innerText = value;
|
||||
setTimeout(() => {
|
||||
$select.style.width = "";
|
||||
$select.style.background = "";
|
||||
setState({ popupOpen: false });
|
||||
}, 200);
|
||||
};
|
||||
$select.onclick = (event) => {
|
||||
onclick?.(event);
|
||||
$popup.setAttribute("open", true);
|
||||
setState({ popupOpen: true });
|
||||
openPopup();
|
||||
};
|
||||
$select.onkeydown = (event) => {
|
||||
onkeydown?.(event);
|
||||
if (event.key === "Enter") openPopup();
|
||||
};
|
||||
useState(["rerender"], () => {
|
||||
_get?.().then((value) => {
|
||||
if ($popup.hasAttribute("open")) {
|
||||
$popup.removeAttribute("open");
|
||||
$select.style.width = `${$select.offsetWidth}px`;
|
||||
$select.style.background = "transparent";
|
||||
$select.innerText = value;
|
||||
setTimeout(() => {
|
||||
$select.style.width = "";
|
||||
$select.style.background = "";
|
||||
setState({ popupOpen: false });
|
||||
}, 200);
|
||||
closePopup(value);
|
||||
} else $select.innerText = value;
|
||||
});
|
||||
});
|
||||
document.addEventListener("click", (event) => {
|
||||
if (!$popup.hasAttribute("open")) return;
|
||||
if ($popup.contains(event.target) || event.target === $select) return;
|
||||
_set?.($select.innerText);
|
||||
closePopup();
|
||||
});
|
||||
|
||||
return html`<div class="notion-enhancer--menu-select relative">
|
||||
@ -501,7 +557,6 @@ function SelectOption({ value, _get, _set, ...props }) {
|
||||
$selected = html`<i class="ml-auto i-check w-[16px] h-[16px]"></i>`,
|
||||
$option = html`<div
|
||||
role="button"
|
||||
tabindex="0"
|
||||
class="select-none cursor-pointer rounded-[3px]
|
||||
flex items-center w-full h-[28px] px-[12px] leading-[1.2]
|
||||
transition duration-[20ms] hover:bg-[color:var(--theme--bg-hover)]"
|
||||
@ -512,11 +567,15 @@ function SelectOption({ value, _get, _set, ...props }) {
|
||||
</div>
|
||||
</div>`;
|
||||
|
||||
const { onclick } = $option;
|
||||
const { onclick, onkeydown } = $option;
|
||||
$option.onclick = (event) => {
|
||||
onclick?.(event);
|
||||
_set?.(value);
|
||||
};
|
||||
$option.onkeydown = (event) => {
|
||||
onkeydown?.(event);
|
||||
if (event.key === "Enter") _set?.(value);
|
||||
};
|
||||
useState(["rerender"], () => {
|
||||
_get?.().then((actualValue) => {
|
||||
if (actualValue === value) {
|
||||
|
@ -58,7 +58,7 @@ const compatibleMods = (mods) => {
|
||||
return required && compatible;
|
||||
});
|
||||
},
|
||||
renderList = async (mods) => {
|
||||
renderList = async (mods, description) => {
|
||||
const { html, getProfile, initDatabase } = globalThis.__enhancerApi,
|
||||
enabledMods = initDatabase([await getProfile(), "enabledMods"]);
|
||||
mods = compatibleMods(mods).map(async (mod) => {
|
||||
@ -69,30 +69,31 @@ const compatibleMods = (mods) => {
|
||||
};
|
||||
return html`<${Mod} ...${{ ...mod, _get, _set }} />`;
|
||||
});
|
||||
return html`<${List}>${await Promise.all(mods)}<//>`;
|
||||
return html`<${List} description=${description}>
|
||||
${await Promise.all(mods)}
|
||||
<//>`;
|
||||
},
|
||||
renderOptionViews = async (parentView, mods) => {
|
||||
const { html, getProfile, initDatabase } = globalThis.__enhancerApi,
|
||||
enabledMods = initDatabase([await getProfile(), "enabledMods"]);
|
||||
mods = compatibleMods(mods)
|
||||
.filter((mod) => {
|
||||
return mod.options?.filter((opt) => opt.type !== "heading").length;
|
||||
})
|
||||
.map(async (mod) => {
|
||||
const _get = () => enabledMods.get(mod.id),
|
||||
_set = async (enabled) => {
|
||||
await enabledMods.set(mod.id, enabled);
|
||||
setState({ rerender: true });
|
||||
};
|
||||
return html`<${View} id=${mod.id}>
|
||||
<${Mod} ...${{ ...mod, options: [], _get, _set }} />
|
||||
${await renderOptions(mod)}<//
|
||||
>`;
|
||||
});
|
||||
return Promise.all(mods);
|
||||
};
|
||||
|
||||
const renderOptionViews = async (parentView, mods) => {
|
||||
const { html, getProfile, initDatabase } = globalThis.__enhancerApi,
|
||||
enabledMods = initDatabase([await getProfile(), "enabledMods"]);
|
||||
mods = compatibleMods(mods)
|
||||
.filter((mod) => {
|
||||
return mod.options?.filter((opt) => opt.type !== "heading").length;
|
||||
})
|
||||
.map(async (mod) => {
|
||||
const _get = () => enabledMods.get(mod.id),
|
||||
_set = async (enabled) => {
|
||||
await enabledMods.set(mod.id, enabled);
|
||||
setState({ rerender: true });
|
||||
};
|
||||
return html`<${View} id=${mod.id}>
|
||||
<${Mod} ...${{ ...mod, options: [], _get, _set }} />
|
||||
${await renderOptions(mod)}<//
|
||||
>`;
|
||||
});
|
||||
return Promise.all(mods);
|
||||
};
|
||||
|
||||
const render = async () => {
|
||||
const { html, getCore, getThemes } = globalThis.__enhancerApi,
|
||||
{ getExtensions, getIntegrations } = globalThis.__enhancerApi,
|
||||
@ -150,13 +151,32 @@ const render = async () => {
|
||||
html`<${View} id="welcome">welcome<//>`,
|
||||
html`<${View} id="core">${await renderOptions(await getCore())}<//>`
|
||||
);
|
||||
for (const { id, mods } of [
|
||||
{ id: "themes", mods: await getThemes() },
|
||||
{ id: "extensions", mods: await getExtensions() },
|
||||
{ id: "integrations", mods: await getIntegrations() },
|
||||
for (const { id, mods, description } of [
|
||||
{
|
||||
id: "themes",
|
||||
mods: await getThemes(),
|
||||
description: `Themes override Notion's colour schemes. To switch between
|
||||
dark mode and light mode, go to <span class="py-[2px] px-[4px] rounded-[3px]
|
||||
bg-[color:var(--theme--bg-hover)]">Settings & members → My notifications &
|
||||
settings → My settings → Appearance</span>.`,
|
||||
},
|
||||
{
|
||||
id: "extensions",
|
||||
mods: await getExtensions(),
|
||||
description: `Extensions add to the functionality and layout of the Notion
|
||||
client, interacting with and modifying existing interfaces.`,
|
||||
},
|
||||
{
|
||||
id: "integrations",
|
||||
mods: await getIntegrations(),
|
||||
description: `<span class="text-[color:var(--theme--fg-red)]">
|
||||
Integrations access and modify Notion content. They interact directly with
|
||||
<span class="py-[2px] px-[4px] rounded-[3px] bg-[color:var(--theme--bg-hover)]">
|
||||
https://www.notion.so/api/v3</span>. Use at your own risk.</span>`,
|
||||
},
|
||||
]) {
|
||||
document.body.append(
|
||||
html`<${View} id=${id}>${await renderList(mods)}<//>`,
|
||||
html`<${View} id=${id}>${await renderList(mods, description)}<//>`,
|
||||
...(await renderOptionViews(id, mods))
|
||||
);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user