mirror of
https://github.com/notion-enhancer/notion-enhancer.git
synced 2025-04-04 12:49:03 +00:00
fix(menu): keyboard triggering of inputs and switches via space/enter, don't trap tab in hotkey input
This commit is contained in:
parent
c3317bd9ec
commit
50a23f17c5
@ -25,7 +25,7 @@ function Button({ icon, variant, tagName, ...props }, ...children) {
|
||||
hover:bg-[color:var(--theme--bg-hover)]`
|
||||
}`,
|
||||
});
|
||||
return html`<${tagName ?? "button"} ...${props}>
|
||||
return html`<${tagName ?? "button"} tabindex="0" ...${props}>
|
||||
${icon
|
||||
? html`<i
|
||||
class="i-${icon}
|
||||
|
@ -25,6 +25,12 @@ function Checkbox({ _get, _set, ...props }) {
|
||||
return html`<label
|
||||
tabindex="0"
|
||||
class="notion-enhancer--menu-checkbox cursor-pointer"
|
||||
onkeydown=${(event) => {
|
||||
if ([" ", "Enter"].includes(event.key)) {
|
||||
event.preventDefault();
|
||||
$input.click();
|
||||
}
|
||||
}}
|
||||
>
|
||||
${$input}
|
||||
<div class="flex items-center h-[16px] transition duration-[200ms]">
|
||||
|
@ -7,13 +7,16 @@
|
||||
import { extendProps, useState } from "../state.mjs";
|
||||
|
||||
const updateHotkey = (event) => {
|
||||
event.preventDefault();
|
||||
const keys = [];
|
||||
for (const modifier of ["metaKey", "ctrlKey", "altKey", "shiftKey"]) {
|
||||
if (!event[modifier]) continue;
|
||||
const alias = modifier[0].toUpperCase() + modifier.slice(1, -3);
|
||||
keys.push(alias);
|
||||
}
|
||||
// retain tab for keyboard navigation of menu
|
||||
if (event.key === "Tab" && !keys.length) {
|
||||
return;
|
||||
} else event.preventDefault();
|
||||
if (!keys.length && ["Backspace", "Delete"].includes(event.key)) {
|
||||
event.target.value = "";
|
||||
} else if (event.key) {
|
||||
@ -146,6 +149,12 @@ function Input({
|
||||
h-[28px] px-[8px] bg-[color:var(--theme--bg-secondary)]
|
||||
text-([14px] [color:var(--theme--fg-secondary)]) rounded-[4px]
|
||||
transition duration-[20ms] hover:bg-[color:var(--theme--bg-hover)]"
|
||||
onkeydown=${(event) => {
|
||||
if ([" ", "Enter"].includes(event.key)) {
|
||||
event.preventDefault();
|
||||
$input.click();
|
||||
}
|
||||
}}
|
||||
>${$input}${$icon.children[0]}${$filename}
|
||||
</label>
|
||||
${$clear}
|
||||
|
@ -50,7 +50,10 @@ function Popup({ trigger, onopen, onclose, onbeforeclose }, ...children) {
|
||||
extendProps(trigger, {
|
||||
onclick: $popup.show,
|
||||
onkeydown(event) {
|
||||
if (event.key === "Enter") $popup.show();
|
||||
if ([" ", "Enter"].includes(event.key)) {
|
||||
event.preventDefault();
|
||||
$popup.show();
|
||||
}
|
||||
},
|
||||
});
|
||||
}
|
||||
|
@ -11,7 +11,8 @@ function Toggle({ _get, _set, ...props }) {
|
||||
$input = html`<input
|
||||
type="checkbox"
|
||||
class="hidden checked:sibling:children:(
|
||||
bg-[color:var(--theme--accent-primary)] after:translate-x-[12px])"
|
||||
bg-[color:var(--theme--accent-primary)]
|
||||
after:translate-x-[12px])"
|
||||
...${props}
|
||||
/>`;
|
||||
extendProps($input, { onchange: () => _set?.($input.checked) });
|
||||
@ -25,6 +26,12 @@ function Toggle({ _get, _set, ...props }) {
|
||||
tabindex="0"
|
||||
class="w-[30px] h-[18px] rounded-[44px] cursor-pointer
|
||||
transition duration-200 bg-[color:var(--theme--bg-hover)]"
|
||||
onkeydown=${(event) => {
|
||||
if ([" ", "Enter"].includes(event.key)) {
|
||||
event.preventDefault();
|
||||
$input.click();
|
||||
}
|
||||
}}
|
||||
>
|
||||
<div
|
||||
class="w-full h-full rounded-[44px] text-[12px]
|
||||
|
@ -39,14 +39,14 @@ function Profile({ id }) {
|
||||
class="py-[2px] px-[4px] rounded-[3px]
|
||||
bg-[color:var(--theme--bg-hover)]"
|
||||
></span>`,
|
||||
$success = html`<${Popup}
|
||||
$uploadSuccess = html`<${Popup}
|
||||
onopen=${async () => ($successName.innerText = await getName())}
|
||||
>
|
||||
<p class="py-[2px] px-[8px] text-[14px]">
|
||||
The profile ${$successName} has been updated successfully.
|
||||
</p>
|
||||
<//>`,
|
||||
$error = html`<${Popup}>
|
||||
$uploadError = html`<${Popup}>
|
||||
<p
|
||||
class="py-[2px] px-[8px] text-[14px]
|
||||
text-[color:var(--theme--accent-secondary)]"
|
||||
@ -60,16 +60,13 @@ function Profile({ id }) {
|
||||
reader.onload = async (progress) => {
|
||||
const res = progress.currentTarget.result;
|
||||
try {
|
||||
await profile.import({
|
||||
...JSON.parse(res),
|
||||
profileName: await getName(),
|
||||
});
|
||||
await profile.import(JSON.parse(res));
|
||||
setState({ rerender: true, databaseUpdated: true });
|
||||
$success.show();
|
||||
setTimeout(() => $success.hide(), 2000);
|
||||
$uploadSuccess.show();
|
||||
setTimeout(() => $uploadSuccess.hide(), 2000);
|
||||
} catch (err) {
|
||||
$error.show();
|
||||
setTimeout(() => $error.hide(), 2000);
|
||||
$uploadError.show();
|
||||
setTimeout(() => $uploadError.hide(), 2000);
|
||||
}
|
||||
};
|
||||
reader.readAsText(file);
|
||||
@ -94,7 +91,14 @@ function Profile({ id }) {
|
||||
$a.click();
|
||||
$a.remove();
|
||||
},
|
||||
deleteProfile = async () => {
|
||||
$uploadInput = html`<input
|
||||
type="file"
|
||||
class="hidden"
|
||||
accept=".json"
|
||||
onchange=${uploadProfile}
|
||||
/>`;
|
||||
|
||||
const deleteProfile = async () => {
|
||||
let profileIds = await db.get("profileIds");
|
||||
if (!profileIds?.length) profileIds = ["default"];
|
||||
// clear profile data
|
||||
@ -108,9 +112,8 @@ function Profile({ id }) {
|
||||
await db.remove("activeProfile");
|
||||
setState({ rerender: true, databaseUpdated: true });
|
||||
} else setState({ rerender: true });
|
||||
};
|
||||
|
||||
const $delete = html`<button
|
||||
},
|
||||
$delete = html`<button
|
||||
class="h-[14px] transition duration-[20ms]
|
||||
text-[color:var(--theme--fg-secondary)]
|
||||
hover:text-[color:var(--theme--fg-primary)]"
|
||||
@ -138,7 +141,7 @@ function Profile({ id }) {
|
||||
<${Button}
|
||||
tabindex="0"
|
||||
class="justify-center"
|
||||
onclick=${() => $confirm.close()}
|
||||
onclick=${() => $confirm.hide()}
|
||||
>
|
||||
Cancel
|
||||
<//>
|
||||
@ -151,14 +154,18 @@ function Profile({ id }) {
|
||||
onchange=${(event) => (event.target.checked = true)}
|
||||
/>
|
||||
<${Input} icon="file-cog" ...${{ _get: getName, _set: setName }} />
|
||||
<${Button} variant="sm" icon="import" class="relative" tagName="label">
|
||||
<input
|
||||
type="file"
|
||||
class="hidden"
|
||||
accept=".json"
|
||||
onchange=${uploadProfile}
|
||||
/>
|
||||
Import ${$success} ${$error}
|
||||
<${Button}
|
||||
icon="import"
|
||||
variant="sm"
|
||||
tagName="label"
|
||||
class="relative"
|
||||
onkeydown=${(event) => {
|
||||
if ([" ", "Enter"].includes(event.key)) {
|
||||
event.preventDefault();
|
||||
$uploadInput.click();
|
||||
}
|
||||
}}
|
||||
>${$uploadInput} Import ${$uploadSuccess}${$uploadError}
|
||||
<//>
|
||||
<${Button} variant="sm" icon="upload" onclick=${downloadProfile}>Export<//>
|
||||
<div class="relative flex">${$delete}${$confirm}</div>
|
||||
|
Loading…
Reference in New Issue
Block a user