Files
ncd-fe/packages/pl-fe/src/components/dropdown-navigation.tsx
nicole mikołajczyk c17561a561 nicolium: more ports
Signed-off-by: nicole mikołajczyk <git@mkljczk.pl>
2026-02-15 16:45:41 +01:00

587 lines
21 KiB
TypeScript

/* eslint-disable jsx-a11y/interactive-supports-focus */
import { useInfiniteQuery } from '@tanstack/react-query';
import { Link, type LinkOptions } from '@tanstack/react-router';
import clsx from 'clsx';
import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react';
import { FormattedMessage } from 'react-intl';
import { fetchOwnAccounts, logOut, switchAccount } from '@/actions/auth';
import { useAccount } from '@/api/hooks/accounts/use-account';
import Account from '@/components/account';
import Divider from '@/components/ui/divider';
import Icon from '@/components/ui/icon';
import Stack from '@/components/ui/stack';
import Text from '@/components/ui/text';
import ProfileStats from '@/features/ui/components/profile-stats';
import { useAppDispatch } from '@/hooks/use-app-dispatch';
import { useAppSelector } from '@/hooks/use-app-selector';
import { useFeatures } from '@/hooks/use-features';
import { useInstance } from '@/hooks/use-instance';
import { useRegistrationStatus } from '@/hooks/use-registration-status';
import { useFollowRequestsCount } from '@/queries/accounts/use-follow-requests';
import { scheduledStatusesCountQueryOptions } from '@/queries/statuses/scheduled-statuses';
import { useDraftStatusesCountQuery } from '@/queries/statuses/use-draft-statuses';
import { useInteractionRequestsCount } from '@/queries/statuses/use-interaction-requests';
import { makeGetOtherAccounts } from '@/selectors';
import { useSettings } from '@/stores/settings';
import { useIsSidebarOpen, useUiStoreActions } from '@/stores/ui';
import sourceCode from '@/utils/code';
import type { Account as AccountEntity } from 'pl-api';
interface IDropdownNavigationLink extends Partial<LinkOptions> {
href?: string;
icon: string;
text: string | JSX.Element;
onClick: React.EventHandler<React.MouseEvent>;
}
const DropdownNavigationLink: React.FC<IDropdownNavigationLink> = React.memo(
({ href, to, icon, text, onClick, ...rest }) => {
const body = (
<>
<div className='⁂-dropdown-navigation__link__icon'>
<Icon src={icon} />
</div>
<Text tag='span' weight='medium' theme='inherit'>
{text}
</Text>
</>
);
if (to) {
return (
<Link className='⁂-dropdown-navigation__link' to={to} {...rest} onClick={onClick}>
{body}
</Link>
);
}
return (
<a className='⁂-dropdown-navigation__link' href={href} target='_blank' onClick={onClick}>
{body}
</a>
);
},
);
const DropdownNavigation: React.FC = React.memo((): JSX.Element | null => {
const dispatch = useAppDispatch();
const isSidebarOpen = useIsSidebarOpen();
const { closeSidebar } = useUiStoreActions();
const me = useAppSelector((state) => state.me);
const features = useFeatures();
const authenticatedScheduledStatusesCountQueryOptions = useMemo(
() => ({
...scheduledStatusesCountQueryOptions,
enabled: !!me && features.scheduledStatuses,
}),
[me, features],
);
const getOtherAccounts = useCallback(makeGetOtherAccounts(), []);
const { account } = useAccount(me || undefined);
const otherAccounts = useAppSelector((state) => getOtherAccounts(state));
const settings = useSettings();
const followRequestsCount = useFollowRequestsCount().data ?? 0;
const interactionRequestsCount = useInteractionRequestsCount().data ?? 0;
const scheduledStatusCount =
useInfiniteQuery(authenticatedScheduledStatusesCountQueryOptions).data ?? 0;
const { data: draftCount = 0 } = useDraftStatusesCountQuery();
// const dashboardCount = useAppSelector((state) => state.admin.openReports.count() + state.admin.awaitingApproval.count());
const [sidebarVisible, setSidebarVisible] = useState(isSidebarOpen);
const touchStart = useRef(0);
const touchEnd = useRef<number | null>(null);
const { isOpen } = useRegistrationStatus();
const instance = useInstance();
const restrictUnauth = instance.pleroma.metadata.restrict_unauthenticated;
const containerRef = React.useRef<HTMLDivElement>(null);
const [switcher, setSwitcher] = React.useState(false);
const handleClose = () => {
setSwitcher(false);
closeSidebar();
};
const handleSwitchAccount =
(account: AccountEntity): React.MouseEventHandler =>
(e) => {
e.preventDefault();
e.stopPropagation();
dispatch(switchAccount(account.id));
};
const onClickLogOut: React.MouseEventHandler = (e) => {
e.preventDefault();
e.stopPropagation();
dispatch(logOut());
};
const handleSwitcherClick: React.MouseEventHandler = (e) => {
e.preventDefault();
e.stopPropagation();
setSwitcher((prevState) => !prevState);
};
const renderAccount = (account: AccountEntity) => (
<a
className='⁂-dropdown-navigation__account-switcher__account'
href='#'
onClick={handleSwitchAccount(account)}
key={account.id}
>
<div>
<Account
account={account}
showAccountHoverCard={false}
withRelationship={false}
withLinkToProfile={false}
/>
</div>
</a>
);
const handleKeyDown: React.KeyboardEventHandler<HTMLDivElement> = (e) => {
if (e.key === 'Escape') handleClose();
};
const handleTouchStart: React.TouchEventHandler<HTMLDivElement> = (e) =>
(touchStart.current = e.targetTouches[0].clientX);
const handleTouchMove: React.TouchEventHandler<HTMLDivElement> = (e) =>
(touchEnd.current = e.targetTouches[0].clientX);
const handleTouchEnd: React.TouchEventHandler<HTMLDivElement> = (e) => {
if (touchEnd.current !== null && touchStart.current - touchEnd.current > 100) {
handleClose();
}
touchEnd.current = null;
};
useEffect(() => {
dispatch(fetchOwnAccounts());
}, []);
useEffect(() => {
if (isSidebarOpen) containerRef.current?.querySelector('a')?.focus();
setTimeout(
() => {
setSidebarVisible(isSidebarOpen);
},
isSidebarOpen ? 0 : 150,
);
}, [isSidebarOpen]);
return (
<div
aria-expanded={isSidebarOpen}
className={clsx({
'⁂-dropdown-navigation__container': true,
'⁂-dropdown-navigation__container--partially-visible': isSidebarOpen || sidebarVisible,
'⁂-dropdown-navigation__container--visible': isSidebarOpen && sidebarVisible,
})}
ref={containerRef}
onKeyDown={handleKeyDown}
onTouchStart={handleTouchStart}
onTouchMove={handleTouchMove}
onTouchEnd={handleTouchEnd}
>
<div className='⁂-dropdown-navigation__overlay' role='button' onClick={handleClose} />
<div className='⁂-dropdown-navigation'>
{account ? (
<div>
<Link to='/@{$username}' params={{ username: account.acct }} onClick={closeSidebar}>
<Account account={account} showAccountHoverCard={false} withLinkToProfile={false} />
</Link>
{!settings.demetricator && (
<ProfileStats account={account} onClickHandler={handleClose} />
)}
<Stack space={4}>
<Divider />
<DropdownNavigationLink
to='/@{$username}'
params={{ username: account.acct }}
icon={require('@phosphor-icons/core/regular/user.svg')}
text={<FormattedMessage id='account.profile' defaultMessage='Profile' />}
onClick={closeSidebar}
/>
{(account.locked || followRequestsCount > 0) && (
<DropdownNavigationLink
to='/follow_requests'
icon={require('@phosphor-icons/core/regular/user-plus.svg')}
text={
<FormattedMessage
id='column.follow_requests'
defaultMessage='Follow requests'
/>
}
onClick={closeSidebar}
/>
)}
{interactionRequestsCount > 0 && (
<DropdownNavigationLink
to='/interaction_requests'
icon={require('@phosphor-icons/core/regular/heart-half.svg')}
text={
<FormattedMessage
id='column.interaction_requests'
defaultMessage='Interaction requests'
/>
}
onClick={closeSidebar}
/>
)}
{features.conversations && (
<DropdownNavigationLink
to='/conversations'
icon={require('@phosphor-icons/core/regular/envelope-simple.svg')}
text={<FormattedMessage id='column.direct' defaultMessage='Direct messages' />}
onClick={closeSidebar}
/>
)}
{features.bookmarks && (
<DropdownNavigationLink
to='/bookmarks'
icon={require('@phosphor-icons/core/regular/bookmarks.svg')}
text={<FormattedMessage id='column.bookmarks' defaultMessage='Bookmarks' />}
onClick={closeSidebar}
/>
)}
{features.groups && (
<DropdownNavigationLink
to='/groups'
icon={require('@phosphor-icons/core/regular/users-three.svg')}
text={<FormattedMessage id='column.groups' defaultMessage='Groups' />}
onClick={closeSidebar}
/>
)}
{features.lists && (
<DropdownNavigationLink
to='/lists'
icon={require('@phosphor-icons/core/regular/list-dashes.svg')}
text={<FormattedMessage id='column.lists' defaultMessage='Lists' />}
onClick={closeSidebar}
/>
)}
{features.circles && (
<DropdownNavigationLink
to='/circles'
icon={require('@phosphor-icons/core/regular/circles-three.svg')}
text={<FormattedMessage id='column.circles' defaultMessage='Circles' />}
onClick={closeSidebar}
/>
)}
{features.antennas && (
<DropdownNavigationLink
to='/antennas'
icon={require('@phosphor-icons/core/regular/broadcast.svg')}
text={<FormattedMessage id='column.antennas' defaultMessage='Antennas' />}
onClick={closeSidebar}
/>
)}
{features.drive && (
<DropdownNavigationLink
to='/drive/{-$folderId}'
icon={require('@phosphor-icons/core/regular/cloud.svg')}
text={<FormattedMessage id='column.drive' defaultMessage='Drive' />}
onClick={closeSidebar}
/>
)}
{features.events && (
<DropdownNavigationLink
to='/events'
icon={require('@phosphor-icons/core/regular/calendar-dots.svg')}
text={<FormattedMessage id='column.events' defaultMessage='Events' />}
onClick={closeSidebar}
/>
)}
{features.profileDirectory && (
<DropdownNavigationLink
to='/directory'
icon={require('@phosphor-icons/core/regular/address-book.svg')}
text={
<FormattedMessage id='column.directory' defaultMessage='Profile directory' />
}
onClick={closeSidebar}
/>
)}
{scheduledStatusCount > 0 && (
<DropdownNavigationLink
to='/scheduled_statuses'
icon={require('@phosphor-icons/core/regular/hourglass.svg')}
text={
<FormattedMessage
id='column.scheduled_statuses'
defaultMessage='Scheduled posts'
/>
}
onClick={closeSidebar}
/>
)}
{draftCount > 0 && (
<DropdownNavigationLink
to='/draft_statuses'
icon={require('@phosphor-icons/core/regular/pencil-simple.svg')}
text={<FormattedMessage id='column.draft_statuses' defaultMessage='Drafts' />}
onClick={closeSidebar}
/>
)}
{features.publicTimeline && (
<>
<Divider />
<DropdownNavigationLink
to='/timeline/local'
icon={require('@phosphor-icons/core/regular/planet.svg')}
text={
features.federating ? (
<FormattedMessage id='tabs_bar.local' defaultMessage='Local' />
) : (
<FormattedMessage id='tabs_bar.all' defaultMessage='All' />
)
}
onClick={closeSidebar}
/>
{features.bubbleTimeline && (
<DropdownNavigationLink
to='/timeline/bubble'
icon={require('@phosphor-icons/core/regular/graph.svg')}
text={<FormattedMessage id='tabs_bar.bubble' defaultMessage='Bubble' />}
onClick={closeSidebar}
/>
)}
{features.federating && (
<DropdownNavigationLink
to='/timeline/fediverse'
icon={require('@phosphor-icons/core/regular/fediverse-logo.svg')}
text={<FormattedMessage id='tabs_bar.fediverse' defaultMessage='Fediverse' />}
onClick={closeSidebar}
/>
)}
{features.wrenchedTimeline && (
<DropdownNavigationLink
to='/timeline/wrenched'
icon={require('@phosphor-icons/core/regular/wrench.svg')}
text={<FormattedMessage id='tabs_bar.wrenched' defaultMessage='Wrenched' />}
onClick={closeSidebar}
/>
)}
</>
)}
<Divider />
<DropdownNavigationLink
to='/settings'
icon={require('@phosphor-icons/core/regular/sliders-horizontal.svg')}
text={<FormattedMessage id='column.preferences' defaultMessage='Preferences' />}
onClick={closeSidebar}
/>
{features.followedHashtagsList && (
<DropdownNavigationLink
to='/followed_tags'
icon={require('@phosphor-icons/core/regular/hash.svg')}
text={
<FormattedMessage
id='column.followed_tags'
defaultMessage='Followed hashtags'
/>
}
onClick={closeSidebar}
/>
)}
{features.rssFeedSubscriptions && (
<DropdownNavigationLink
to='/rss_feed_subscriptions'
icon={require('@phosphor-icons/core/regular/rss.svg')}
text={
<FormattedMessage
id='column.rss_feed_subscriptions'
defaultMessage='Subscribed RSS feeds'
/>
}
onClick={closeSidebar}
/>
)}
{(account.is_admin ?? account.is_moderator) && (
<DropdownNavigationLink
to='/pl-fe/admin'
icon={require('@phosphor-icons/core/regular/gauge.svg')}
text={<FormattedMessage id='column.admin.dashboard' defaultMessage='Dashboard' />}
onClick={closeSidebar}
// count={dashboardCount} WIP
/>
)}
<Divider />
<DropdownNavigationLink
to='/logout'
icon={require('@phosphor-icons/core/regular/sign-out.svg')}
text={<FormattedMessage id='navigation_bar.logout' defaultMessage='Logout' />}
onClick={onClickLogOut}
/>
<Divider />
<DropdownNavigationLink
href={sourceCode.url}
icon={require('@phosphor-icons/core/regular/code.svg')}
text={<FormattedMessage id='navigation.source_code' defaultMessage='Source code' />}
onClick={closeSidebar}
/>
<Divider />
<div
className={clsx('⁂-dropdown-navigation__account-switcher', {
'⁂-dropdown-navigation__account-switcher--expanded': switcher,
})}
>
<button type='button' onClick={handleSwitcherClick}>
<Text tag='span'>
<FormattedMessage
id='profile_dropdown.switch_account'
defaultMessage='Switch accounts'
/>
</Text>
<Icon src={require('@phosphor-icons/core/regular/caret-down.svg')} />
</button>
{switcher && (
<div className='⁂-dropdown-navigation__account-switcher__accounts'>
{otherAccounts.map((account) => renderAccount(account))}
<Link
className='⁂-dropdown-navigation__account-switcher__add'
to='/login/add'
onClick={handleClose}
>
<Icon src={require('@phosphor-icons/core/regular/plus.svg')} />
<Text size='sm' weight='medium'>
<FormattedMessage
id='profile_dropdown.add_account'
defaultMessage='Add an existing account'
/>
</Text>
</Link>
</div>
)}
</div>
</Stack>
</div>
) : (
<div>
{features.publicTimeline && !restrictUnauth.timelines.local && (
<>
<DropdownNavigationLink
to='/timeline/local'
icon={require('@phosphor-icons/core/regular/planet.svg')}
text={
features.federating ? (
<FormattedMessage id='tabs_bar.local' defaultMessage='Local' />
) : (
<FormattedMessage id='tabs_bar.all' defaultMessage='All' />
)
}
onClick={closeSidebar}
/>
{features.bubbleTimeline && !restrictUnauth.timelines.bubble && (
<DropdownNavigationLink
to='/timeline/bubble'
icon={require('@phosphor-icons/core/regular/graph.svg')}
text={<FormattedMessage id='tabs_bar.bubble' defaultMessage='Bubble' />}
onClick={closeSidebar}
/>
)}
{features.federating && !restrictUnauth.timelines.federated && (
<DropdownNavigationLink
to='/timeline/fediverse'
icon={require('@phosphor-icons/core/regular/fediverse-logo.svg')}
text={<FormattedMessage id='tabs_bar.fediverse' defaultMessage='Fediverse' />}
onClick={closeSidebar}
/>
)}
{features.wrenchedTimeline && !restrictUnauth.timelines.wrenched && (
<DropdownNavigationLink
to='/timeline/wrenched'
icon={require('@phosphor-icons/core/regular/wrench.svg')}
text={<FormattedMessage id='tabs_bar.wrenched' defaultMessage='Wrenched' />}
onClick={closeSidebar}
/>
)}
<Divider />
</>
)}
<DropdownNavigationLink
to='/login'
icon={require('@phosphor-icons/core/regular/sign-in.svg')}
text={<FormattedMessage id='account.login' defaultMessage='Log in' />}
onClick={closeSidebar}
/>
{isOpen && (
<DropdownNavigationLink
to='/signup'
icon={require('@phosphor-icons/core/regular/user-plus.svg')}
text={<FormattedMessage id='account.register' defaultMessage='Sign up' />}
onClick={closeSidebar}
/>
)}
<Divider />
<DropdownNavigationLink
href={sourceCode.url}
icon={require('@phosphor-icons/core/regular/code.svg')}
text={<FormattedMessage id='navigation.source_code' defaultMessage='Source code' />}
onClick={closeSidebar}
/>
</div>
)}
</div>
</div>
);
});
export { DropdownNavigation as default };