phanpy/src/components/shortcuts.jsx

253 lines
7.4 KiB
React
Raw Normal View History

2023-02-16 17:51:54 +08:00
import './shortcuts.css';
2024-08-13 15:26:23 +08:00
import { t, Trans } from '@lingui/macro';
import { useLingui } from '@lingui/react';
2024-04-04 17:03:30 +08:00
import { MenuDivider } from '@szhsin/react-menu';
2023-10-22 19:24:59 +08:00
import { memo } from 'preact/compat';
import { useRef, useState } from 'preact/hooks';
import { useHotkeys } from 'react-hotkeys-hook';
2023-02-16 17:51:54 +08:00
import { useNavigate } from 'react-router-dom';
import { useSnapshot } from 'valtio';
import { SHORTCUTS_META } from '../components/shortcuts-settings';
2023-04-14 11:13:14 +08:00
import { api } from '../utils/api';
import { getLists } from '../utils/lists';
2023-02-16 17:51:54 +08:00
import states from '../utils/states';
import AsyncText from './AsyncText';
import Icon from './icon';
import Link from './link';
2023-03-28 15:59:20 +08:00
import MenuLink from './menu-link';
import Menu2 from './menu2';
2024-04-04 17:03:30 +08:00
import SubMenu2 from './submenu2';
2023-02-16 17:51:54 +08:00
function Shortcuts() {
2024-08-13 15:26:23 +08:00
const { _ } = useLingui();
2023-04-14 11:13:14 +08:00
const { instance } = api();
2023-02-16 17:51:54 +08:00
const snapStates = useSnapshot(states);
2023-10-21 17:40:03 +08:00
const { shortcuts, settings } = snapStates;
2023-02-16 17:51:54 +08:00
if (!shortcuts.length) {
return null;
}
2023-10-21 17:40:03 +08:00
if (
settings.shortcutsViewMode === 'multi-column' ||
(!settings.shortcutsViewMode && settings.shortcutsColumnsMode)
2023-10-21 17:40:03 +08:00
) {
return null;
}
2023-02-16 17:51:54 +08:00
const menuRef = useRef();
const hasLists = useRef(false);
const formattedShortcuts = shortcuts
.map((pin, i) => {
const { type, ...data } = pin;
if (!SHORTCUTS_META[type]) return null;
let { id, path, title, subtitle, icon } = SHORTCUTS_META[type];
2023-02-16 17:51:54 +08:00
if (typeof id === 'function') {
id = id(data, i);
}
if (typeof path === 'function') {
path = path(
{
...data,
instance: data.instance || instance,
},
i,
);
}
if (typeof title === 'function') {
title = title(data, i);
2024-08-13 15:26:23 +08:00
} else {
title = _(title);
}
if (typeof subtitle === 'function') {
subtitle = subtitle(data, i);
2024-08-13 15:26:23 +08:00
} else {
subtitle = _(subtitle);
}
if (typeof icon === 'function') {
icon = icon(data, i);
}
2023-02-16 17:51:54 +08:00
if (id === 'lists') {
hasLists.current = true;
}
return {
id,
path,
title,
subtitle,
icon,
};
})
.filter(Boolean);
2023-02-16 17:51:54 +08:00
const navigate = useNavigate();
useHotkeys(['1', '2', '3', '4', '5', '6', '7', '8', '9'], (e, handler) => {
const index = parseInt(handler.keys[0], 10) - 1;
if (index < formattedShortcuts.length) {
const { path } = formattedShortcuts[index];
if (path) {
navigate(path);
2023-02-19 22:07:15 +08:00
menuRef.current?.closeMenu?.();
2023-02-16 17:51:54 +08:00
}
}
});
const [lists, setLists] = useState([]);
2023-02-16 17:51:54 +08:00
return (
<div id="shortcuts">
2023-02-27 23:59:41 +08:00
{snapStates.settings.shortcutsViewMode === 'tab-menu-bar' ? (
<nav
class="tab-bar"
onContextMenu={(e) => {
e.preventDefault();
states.showShortcutsSettings = true;
}}
>
2023-02-27 23:59:41 +08:00
<ul>
{formattedShortcuts.map(
({ id, path, title, subtitle, icon }, i) => {
return (
2023-11-06 23:50:49 +08:00
<li key={`${i}-${id}-${title}-${subtitle}-${path}`}>
<Link
class={subtitle ? 'has-subtitle' : ''}
to={path}
onClick={(e) => {
if (e.target.classList.contains('is-active')) {
e.preventDefault();
const page = document.getElementById(`${id}-page`);
console.log(id, page);
if (page) {
page.scrollTop = 0;
const updatesButton =
page.querySelector('.updates-button');
if (updatesButton) {
updatesButton.click();
}
2023-02-27 23:59:41 +08:00
}
}
}}
>
<Icon icon={icon} size="xl" alt={title} />
<span>
<AsyncText>{title}</AsyncText>
{subtitle && (
<>
<br />
<small>{subtitle}</small>
</>
)}
</span>
</Link>
</li>
);
},
)}
2023-02-27 23:59:41 +08:00
</ul>
</nav>
) : (
<Menu2
2023-02-27 23:59:41 +08:00
instanceRef={menuRef}
overflow="auto"
viewScroll="close"
menuClassName="glass-menu shortcuts-menu"
2023-06-13 17:46:37 +08:00
gap={8}
2023-02-27 23:59:41 +08:00
position="anchor"
onMenuChange={(e) => {
if (e.open && hasLists.current) {
getLists().then(setLists);
}
}}
2023-02-27 23:59:41 +08:00
menuButton={
<button
type="button"
id="shortcuts-button"
class="plain"
onContextMenu={(e) => {
e.preventDefault();
states.showShortcutsSettings = true;
}}
2023-02-27 23:59:41 +08:00
onTransitionStart={(e) => {
// Close menu if the button disappears
try {
const { target } = e;
if (getComputedStyle(target).pointerEvents === 'none') {
menuRef.current?.closeMenu?.();
}
} catch (e) {}
}}
>
2024-08-13 15:26:23 +08:00
<Icon icon="shortcut" size="xl" alt={t`Shortcuts`} />
2023-02-27 23:59:41 +08:00
</button>
}
>
2023-11-06 23:50:49 +08:00
{formattedShortcuts.map(({ id, path, title, subtitle, icon }, i) => {
if (id === 'lists') {
return (
2024-04-04 17:03:30 +08:00
<SubMenu2
menuClassName="glass-menu"
overflow="auto"
gap={-8}
label={
<>
<Icon icon={icon} size="l" />
<span class="menu-grow">
<AsyncText>{title}</AsyncText>
</span>
<Icon icon="chevron-right" />
</>
}
>
<MenuLink to="/l">
2024-08-13 15:26:23 +08:00
<span>
<Trans>All Lists</Trans>
</span>
</MenuLink>
<MenuDivider />
{lists?.map((list) => (
<MenuLink key={list.id} to={`/l/${list.id}`}>
<span>{list.title}</span>
</MenuLink>
))}
2024-04-04 17:03:30 +08:00
</SubMenu2>
);
}
2023-02-27 23:59:41 +08:00
return (
2023-11-06 23:50:49 +08:00
<MenuLink
to={path}
key={`${i}-${id}-${title}-${subtitle}-${path}`}
class="glass-menu-item"
>
2023-02-27 23:59:41 +08:00
<Icon icon={icon} size="l" />{' '}
<span class="menu-grow">
<span>
<AsyncText>{title}</AsyncText>
</span>
{subtitle && (
<>
{' '}
<small class="more-insignificant">{subtitle}</small>
</>
)}
2023-02-27 23:59:41 +08:00
</span>
<span class="menu-shortcut hide-until-focus-visible">
{i + 1}
</span>
</MenuLink>
);
})}
</Menu2>
2023-02-27 23:59:41 +08:00
)}
2023-02-16 17:51:54 +08:00
</div>
);
}
2023-10-22 19:24:59 +08:00
export default memo(Shortcuts);