mirror of
https://github.com/Sendouc/sendou.ink.git
synced 2026-05-06 05:07:36 -05:00
397 lines
9.3 KiB
TypeScript
397 lines
9.3 KiB
TypeScript
import clsx from "clsx";
|
|
import {
|
|
Calendar,
|
|
ChevronRight,
|
|
LogIn,
|
|
Menu,
|
|
MessageSquare,
|
|
Settings,
|
|
Tv,
|
|
User,
|
|
Users,
|
|
X,
|
|
} from "lucide-react";
|
|
import * as React from "react";
|
|
import { Dialog, Modal, ModalOverlay } from "react-aria-components";
|
|
import { useTranslation } from "react-i18next";
|
|
import { Link } from "react-router";
|
|
import { useUser } from "~/features/auth/core/user";
|
|
import { FriendMenu } from "~/features/friends/components/FriendMenu";
|
|
import { SENDOUQ_ACTIVITY_LABEL } from "~/features/friends/friends-constants";
|
|
import type { RootLoaderData } from "~/root";
|
|
import {
|
|
EVENTS_PAGE,
|
|
FRIENDS_PAGE,
|
|
navIconUrl,
|
|
SETTINGS_PAGE,
|
|
userPage,
|
|
} from "~/utils/urls";
|
|
import { Avatar } from "./Avatar";
|
|
import { EventsList } from "./EventsList";
|
|
import { SendouButton } from "./elements/Button";
|
|
import { Image } from "./Image";
|
|
import { MOCK_TOTAL_UNREAD } from "./layout/ChatSidebar";
|
|
import { LogInButtonContainer } from "./layout/LogInButtonContainer";
|
|
import {
|
|
NotificationContent,
|
|
useNotifications,
|
|
} from "./layout/NotificationPopover";
|
|
import { navItems } from "./layout/nav-items";
|
|
import styles from "./MobileNav.module.css";
|
|
import { NotificationDot } from "./NotificationDot";
|
|
import { StreamListItems } from "./StreamListItems";
|
|
|
|
type SidebarData = RootLoaderData["sidebar"] | undefined;
|
|
type PanelType = "closed" | "menu" | "friends" | "tourneys" | "you";
|
|
|
|
export function MobileNav({ sidebarData }: { sidebarData: SidebarData }) {
|
|
const [activePanel, setActivePanel] = React.useState<PanelType>("closed");
|
|
const user = useUser();
|
|
const { unseenIds } = useNotifications();
|
|
|
|
const hasUnseenNotifications = unseenIds.length > 0;
|
|
const hasFriendInSendouQ =
|
|
sidebarData?.friends.some((f) => f.subtitle === SENDOUQ_ACTIVITY_LABEL) ??
|
|
false;
|
|
|
|
const closePanel = () => setActivePanel("closed");
|
|
|
|
return (
|
|
<div className={styles.mobileNav}>
|
|
{activePanel === "menu" ? (
|
|
<MenuOverlay
|
|
streams={sidebarData?.streams ?? []}
|
|
onClose={closePanel}
|
|
/>
|
|
) : null}
|
|
|
|
{activePanel === "friends" ? (
|
|
<FriendsPanel
|
|
friends={sidebarData?.friends ?? []}
|
|
onClose={closePanel}
|
|
/>
|
|
) : null}
|
|
|
|
{activePanel === "tourneys" ? (
|
|
<TourneysPanel
|
|
events={sidebarData?.events ?? []}
|
|
onClose={closePanel}
|
|
/>
|
|
) : null}
|
|
|
|
{activePanel === "you" ? <YouPanel onClose={closePanel} /> : null}
|
|
|
|
<MobileTabBar
|
|
activePanel={activePanel}
|
|
onTabPress={setActivePanel}
|
|
isLoggedIn={Boolean(user)}
|
|
hasUnseenNotifications={hasUnseenNotifications}
|
|
hasFriendInSendouQ={hasFriendInSendouQ}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function MobileTabBar({
|
|
activePanel,
|
|
onTabPress,
|
|
isLoggedIn,
|
|
hasUnseenNotifications,
|
|
hasFriendInSendouQ,
|
|
}: {
|
|
activePanel: PanelType;
|
|
onTabPress: (panel: PanelType) => void;
|
|
isLoggedIn: boolean;
|
|
hasUnseenNotifications: boolean;
|
|
hasFriendInSendouQ: boolean;
|
|
}) {
|
|
const { t } = useTranslation(["front", "common"]);
|
|
|
|
return (
|
|
<nav className={styles.tabBar}>
|
|
<MobileTab
|
|
icon={<Menu />}
|
|
label={t("front:mobileNav.menu")}
|
|
isActive={activePanel === "menu"}
|
|
onPress={() => onTabPress("menu")}
|
|
/>
|
|
|
|
{isLoggedIn ? (
|
|
<>
|
|
<MobileTab
|
|
icon={<Users />}
|
|
label={t("front:mobileNav.friends")}
|
|
isActive={activePanel === "friends"}
|
|
onPress={() => onTabPress("friends")}
|
|
showNotificationDot={hasFriendInSendouQ}
|
|
/>
|
|
<MobileTab
|
|
icon={<Calendar />}
|
|
label={t("front:sideNav.myCalendar")}
|
|
isActive={activePanel === "tourneys"}
|
|
onPress={() => onTabPress("tourneys")}
|
|
/>
|
|
<MobileTab
|
|
icon={<MessageSquare />}
|
|
label={t("front:mobileNav.chat")}
|
|
isActive={false}
|
|
onPress={() => {}}
|
|
unreadCount={MOCK_TOTAL_UNREAD}
|
|
/>
|
|
<MobileTab
|
|
icon={<User />}
|
|
label={t("front:mobileNav.you")}
|
|
isActive={activePanel === "you"}
|
|
onPress={() => onTabPress("you")}
|
|
showNotificationDot={hasUnseenNotifications}
|
|
/>
|
|
</>
|
|
) : (
|
|
<LogInButtonContainer>
|
|
<button type="submit" className={styles.tab}>
|
|
<span className={styles.tabIcon}>
|
|
<LogIn />
|
|
</span>
|
|
<span>{t("front:mobileNav.login")}</span>
|
|
</button>
|
|
</LogInButtonContainer>
|
|
)}
|
|
</nav>
|
|
);
|
|
}
|
|
|
|
function MobileTab({
|
|
icon,
|
|
label,
|
|
isActive,
|
|
onPress,
|
|
showNotificationDot,
|
|
unreadCount,
|
|
}: {
|
|
icon: React.ReactNode;
|
|
label: string;
|
|
isActive: boolean;
|
|
onPress: () => void;
|
|
showNotificationDot?: boolean;
|
|
unreadCount?: number;
|
|
}) {
|
|
return (
|
|
<button
|
|
type="button"
|
|
className={styles.tab}
|
|
data-active={isActive}
|
|
onClick={onPress}
|
|
>
|
|
<span className={styles.tabIcon}>
|
|
{icon}
|
|
{showNotificationDot ? <NotificationDot /> : null}
|
|
{unreadCount ? (
|
|
<span className={styles.tabBadge}>{unreadCount}</span>
|
|
) : null}
|
|
</span>
|
|
<span>{label}</span>
|
|
</button>
|
|
);
|
|
}
|
|
|
|
function MobilePanel({
|
|
title,
|
|
onClose,
|
|
children,
|
|
}: {
|
|
title: string;
|
|
onClose: () => void;
|
|
children: React.ReactNode;
|
|
}) {
|
|
return (
|
|
<ModalOverlay className={styles.panelOverlay} isOpen isDismissable={false}>
|
|
<Modal className={clsx(styles.panel, "scrollbar")}>
|
|
<Dialog className={styles.panelDialog}>
|
|
<header className={styles.panelHeader}>
|
|
<h2 className={styles.panelTitle}>{title}</h2>
|
|
<SendouButton
|
|
icon={<X />}
|
|
variant="minimal"
|
|
shape="circle"
|
|
onPress={onClose}
|
|
/>
|
|
</header>
|
|
<div className={styles.panelContent}>{children}</div>
|
|
</Dialog>
|
|
</Modal>
|
|
</ModalOverlay>
|
|
);
|
|
}
|
|
|
|
function MenuOverlay({
|
|
streams,
|
|
onClose,
|
|
}: {
|
|
streams: NonNullable<SidebarData>["streams"];
|
|
onClose: () => void;
|
|
}) {
|
|
const { t } = useTranslation(["front", "common"]);
|
|
const user = useUser();
|
|
|
|
return (
|
|
<ModalOverlay className={styles.panelOverlay} isOpen isDismissable={false}>
|
|
<Modal className={clsx(styles.menuOverlay, "scrollbar")}>
|
|
<Dialog className={styles.panelDialog}>
|
|
<header className={styles.menuHeader}>
|
|
<h2 className={styles.menuTitle}>{t("front:mobileNav.menu")}</h2>
|
|
<SendouButton
|
|
icon={<X />}
|
|
variant="minimal"
|
|
shape="circle"
|
|
onPress={onClose}
|
|
/>
|
|
</header>
|
|
|
|
<nav aria-label={t("front:mobileNav.menu")}>
|
|
<ul className={styles.navGrid}>
|
|
{navItems.map((item) => (
|
|
<li key={item.name}>
|
|
<Link
|
|
to={`/${item.url}`}
|
|
className={styles.navItem}
|
|
onClick={onClose}
|
|
>
|
|
<div className={styles.navItemImage}>
|
|
<Image
|
|
path={navIconUrl(item.name)}
|
|
height={32}
|
|
width={32}
|
|
alt=""
|
|
/>
|
|
</div>
|
|
<span>{t(`common:pages.${item.name}` as any)}</span>
|
|
</Link>
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</nav>
|
|
|
|
<section className={styles.streamsSection}>
|
|
<header className={styles.streamsSectionHeader}>
|
|
<Tv />
|
|
<h3>{t("front:sideNav.streams")}</h3>
|
|
</header>
|
|
{streams.length === 0 ? (
|
|
<div className={styles.sideNavEmpty}>
|
|
{t("front:sideNav.noStreams")}
|
|
</div>
|
|
) : null}
|
|
<ul className={styles.streamsList}>
|
|
<StreamListItems
|
|
streams={streams}
|
|
onClick={onClose}
|
|
isLoggedIn={Boolean(user)}
|
|
/>
|
|
</ul>
|
|
</section>
|
|
</Dialog>
|
|
</Modal>
|
|
</ModalOverlay>
|
|
);
|
|
}
|
|
|
|
function FriendsPanel({
|
|
friends,
|
|
onClose,
|
|
}: {
|
|
friends: NonNullable<SidebarData>["friends"];
|
|
onClose: () => void;
|
|
}) {
|
|
const { t } = useTranslation(["front", "common"]);
|
|
const user = useUser();
|
|
|
|
return (
|
|
<MobilePanel title={t("front:sideNav.friends")} onClose={onClose}>
|
|
{friends.length > 0 ? (
|
|
friends.map((friend) => (
|
|
<FriendMenu key={friend.id} {...friend} onNavigate={onClose} />
|
|
))
|
|
) : (
|
|
<div className="text-lighter text-sm p-2 text-center">
|
|
{user
|
|
? t("front:sideNav.friends.noFriends")
|
|
: t("front:sideNav.friends.notLoggedIn")}
|
|
</div>
|
|
)}
|
|
<Link
|
|
to={FRIENDS_PAGE}
|
|
className={styles.panelSectionLink}
|
|
onClick={onClose}
|
|
>
|
|
{t("common:actions.viewAll")}
|
|
<ChevronRight size={14} />
|
|
</Link>
|
|
</MobilePanel>
|
|
);
|
|
}
|
|
|
|
function TourneysPanel({
|
|
events,
|
|
onClose,
|
|
}: {
|
|
events: NonNullable<SidebarData>["events"];
|
|
onClose: () => void;
|
|
}) {
|
|
const { t } = useTranslation(["front", "common"]);
|
|
|
|
return (
|
|
<MobilePanel title={t("front:sideNav.myCalendar")} onClose={onClose}>
|
|
<EventsList events={events} onClick={onClose} />
|
|
<Link
|
|
to={EVENTS_PAGE}
|
|
className={styles.panelSectionLink}
|
|
onClick={onClose}
|
|
>
|
|
{t("common:actions.viewAll")}
|
|
<ChevronRight size={14} />
|
|
</Link>
|
|
</MobilePanel>
|
|
);
|
|
}
|
|
|
|
function YouPanel({ onClose }: { onClose: () => void }) {
|
|
const { t } = useTranslation(["front", "common"]);
|
|
const user = useUser();
|
|
const { notifications, unseenIds } = useNotifications();
|
|
|
|
if (!user) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<MobilePanel title={t("front:mobileNav.you")} onClose={onClose}>
|
|
<div className={styles.youPanelUserRow}>
|
|
<Link
|
|
to={userPage(user)}
|
|
className={styles.youPanelUser}
|
|
onClick={onClose}
|
|
>
|
|
<Avatar user={user} size="sm" />
|
|
<span className={styles.youPanelUsername}>{user.username}</span>
|
|
</Link>
|
|
<Link
|
|
to={SETTINGS_PAGE}
|
|
className={styles.youPanelSettingsButton}
|
|
onClick={onClose}
|
|
aria-label={t("common:pages.settings")}
|
|
>
|
|
<Settings />
|
|
</Link>
|
|
</div>
|
|
|
|
{notifications ? (
|
|
<NotificationContent
|
|
notifications={notifications}
|
|
unseenIds={unseenIds}
|
|
onClose={onClose}
|
|
/>
|
|
) : null}
|
|
</MobilePanel>
|
|
);
|
|
}
|