fix(menu): keyboard triggering of inputs and switches via space/enter, don't trap tab in hotkey input

This commit is contained in:
dragonwocky 2023-01-24 00:15:10 +11:00
parent c3317bd9ec
commit 50a23f17c5
Signed by: dragonwocky
GPG Key ID: 7998D08F7D7BD7A8
6 changed files with 59 additions and 27 deletions

View File

@ -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}

View File

@ -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]">

View File

@ -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}

View File

@ -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();
}
},
});
}

View File

@ -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]

View File

@ -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>