mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-06-05 03:49:35 +00:00
355 lines
12 KiB
TypeScript
355 lines
12 KiB
TypeScript
import { useCallback, useEffect, useMemo, useState } from 'react';
|
|
import type { ComponentType } from 'react';
|
|
import { useLocation, useNavigate } from 'react-router-dom';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { Drawer, Layout, Menu } from 'antd';
|
|
import type { MenuProps } from 'antd';
|
|
import {
|
|
ApiOutlined,
|
|
CloseOutlined,
|
|
CloudServerOutlined,
|
|
ClusterOutlined,
|
|
CodeOutlined,
|
|
DashboardOutlined,
|
|
DatabaseOutlined,
|
|
GithubOutlined,
|
|
HeartOutlined,
|
|
ImportOutlined,
|
|
LogoutOutlined,
|
|
MenuOutlined,
|
|
MessageOutlined,
|
|
MoonFilled,
|
|
MoonOutlined,
|
|
SafetyOutlined,
|
|
SettingOutlined,
|
|
SunOutlined,
|
|
SwapOutlined,
|
|
TagsOutlined,
|
|
TeamOutlined,
|
|
ToolOutlined,
|
|
UploadOutlined,
|
|
} from '@ant-design/icons';
|
|
|
|
import { HttpUtil } from '@/utils';
|
|
import { pauseAnimationsUntilLeave, useTheme } from '@/hooks/useTheme';
|
|
import { useAllSettings } from '@/api/queries/useAllSettings';
|
|
import './AppSidebar.css';
|
|
|
|
const SIDEBAR_COLLAPSED_KEY = 'isSidebarCollapsed';
|
|
const DONATE_URL = 'https://donate.sanaei.dev/';
|
|
const REPO_URL = 'https://github.com/MHSanaei/3x-ui';
|
|
const LOGOUT_KEY = '__logout__';
|
|
|
|
type IconName = 'dashboard' | 'inbound' | 'team' | 'groups' | 'setting' | 'tool' | 'cluster' | 'logout' | 'apidocs';
|
|
|
|
const iconByName: Record<IconName, ComponentType> = {
|
|
dashboard: DashboardOutlined,
|
|
inbound: ImportOutlined,
|
|
team: TeamOutlined,
|
|
groups: TagsOutlined,
|
|
setting: SettingOutlined,
|
|
tool: ToolOutlined,
|
|
cluster: ClusterOutlined,
|
|
logout: LogoutOutlined,
|
|
apidocs: ApiOutlined,
|
|
};
|
|
|
|
function readCollapsed(): boolean {
|
|
try {
|
|
return JSON.parse(localStorage.getItem(SIDEBAR_COLLAPSED_KEY) || 'false');
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function DonateButton({ ariaLabel }: { ariaLabel: string }) {
|
|
return (
|
|
<a
|
|
href={DONATE_URL}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className="sidebar-donate"
|
|
aria-label={ariaLabel}
|
|
title={ariaLabel}
|
|
>
|
|
<HeartOutlined />
|
|
</a>
|
|
);
|
|
}
|
|
|
|
function VersionBadge({ version, collapsed }: { version: string; collapsed?: boolean }) {
|
|
if (!version) return null;
|
|
const label = `v${version}`;
|
|
return (
|
|
<a
|
|
href={REPO_URL}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className={`sider-version${collapsed ? ' is-collapsed' : ''}`}
|
|
aria-label={`GitHub ${label}`}
|
|
title={label}
|
|
>
|
|
<GithubOutlined />
|
|
{!collapsed && <span className="sider-version-text">{label}</span>}
|
|
</a>
|
|
);
|
|
}
|
|
|
|
function ThemeCycleButton({ id, isDark, isUltra, onCycle, ariaLabel }: {
|
|
id: string;
|
|
isDark: boolean;
|
|
isUltra: boolean;
|
|
onCycle: () => void;
|
|
ariaLabel: string;
|
|
}) {
|
|
const icon = !isDark ? <SunOutlined /> : !isUltra ? <MoonOutlined /> : <MoonFilled />;
|
|
return (
|
|
<button
|
|
id={id}
|
|
type="button"
|
|
className="sidebar-theme-cycle"
|
|
aria-label={ariaLabel}
|
|
title={ariaLabel}
|
|
onClick={onCycle}
|
|
>
|
|
{icon}
|
|
</button>
|
|
);
|
|
}
|
|
|
|
export default function AppSidebar() {
|
|
const { t } = useTranslation();
|
|
const { isDark, isUltra, toggleTheme, toggleUltra } = useTheme();
|
|
const navigate = useNavigate();
|
|
const { pathname, hash } = useLocation();
|
|
const { allSetting } = useAllSettings();
|
|
const showSubFormats = !!(allSetting.subJsonEnable || allSetting.subClashEnable);
|
|
|
|
const [collapsed, setCollapsed] = useState<boolean>(() => readCollapsed());
|
|
const [drawerOpen, setDrawerOpen] = useState(false);
|
|
|
|
const currentTheme: 'light' | 'dark' = isDark ? 'dark' : 'light';
|
|
const panelVersion = window.X_UI_CUR_VER || '';
|
|
|
|
const tabs = useMemo<{ key: string; icon: IconName; title: string }[]>(() => [
|
|
{ key: '/', icon: 'dashboard', title: t('menu.dashboard') },
|
|
{ key: '/inbounds', icon: 'inbound', title: t('menu.inbounds') },
|
|
{ key: '/clients', icon: 'team', title: t('menu.clients') },
|
|
{ key: '/groups', icon: 'groups', title: t('menu.groups') },
|
|
{ key: '/nodes', icon: 'cluster', title: t('menu.nodes') },
|
|
{ key: '/settings', icon: 'setting', title: t('menu.settings') },
|
|
{ key: '/xray', icon: 'tool', title: t('menu.xray') },
|
|
{ key: '/api-docs', icon: 'apidocs', title: t('menu.apiDocs') },
|
|
{ key: LOGOUT_KEY, icon: 'logout', title: t('logout') },
|
|
], [t]);
|
|
|
|
const navItems = useMemo(() => tabs.filter((tab) => tab.icon !== 'logout'), [tabs]);
|
|
const utilItems = useMemo(() => tabs.filter((tab) => tab.icon === 'logout'), [tabs]);
|
|
|
|
const settingsChildren = useMemo<NonNullable<MenuProps['items']>>(() => {
|
|
const children: NonNullable<MenuProps['items']> = [
|
|
{ key: '/settings#general', icon: <SettingOutlined />, label: t('pages.settings.panelSettings') },
|
|
{ key: '/settings#security', icon: <SafetyOutlined />, label: t('pages.settings.securitySettings') },
|
|
{ key: '/settings#telegram', icon: <MessageOutlined />, label: t('pages.settings.TGBotSettings') },
|
|
{ key: '/settings#subscription', icon: <CloudServerOutlined />, label: t('pages.settings.subSettings') },
|
|
];
|
|
if (showSubFormats) {
|
|
children.push({ key: '/settings#subscription-formats', icon: <CodeOutlined />, label: 'Sub Formats' });
|
|
}
|
|
return children;
|
|
}, [t, showSubFormats]);
|
|
|
|
const xrayChildren = useMemo<NonNullable<MenuProps['items']>>(() => [
|
|
{ key: '/xray#basic', icon: <SettingOutlined />, label: t('pages.xray.basicTemplate') },
|
|
{ key: '/xray#routing', icon: <SwapOutlined />, label: t('pages.xray.Routings') },
|
|
{ key: '/xray#outbound', icon: <UploadOutlined />, label: t('pages.xray.Outbounds') },
|
|
{ key: '/xray#balancer', icon: <ClusterOutlined />, label: t('pages.xray.Balancers') },
|
|
{ key: '/xray#dns', icon: <DatabaseOutlined />, label: 'DNS' },
|
|
{ key: '/xray#advanced', icon: <CodeOutlined />, label: t('pages.xray.advancedTemplate') },
|
|
], [t]);
|
|
|
|
const settingsActive = pathname === '/settings';
|
|
const xrayActive = pathname === '/xray';
|
|
const selectedKey = settingsActive
|
|
? `/settings${hash || '#general'}`
|
|
: xrayActive
|
|
? `/xray${hash || '#basic'}`
|
|
: (pathname === '' ? '/' : pathname);
|
|
|
|
const openSubmenu = settingsActive ? '/settings' : xrayActive ? '/xray' : null;
|
|
const [openKeys, setOpenKeys] = useState<string[]>(() => (openSubmenu ? [openSubmenu] : []));
|
|
useEffect(() => {
|
|
if (openSubmenu) {
|
|
setOpenKeys((keys) => (keys.includes(openSubmenu) ? keys : [...keys, openSubmenu]));
|
|
}
|
|
}, [openSubmenu]);
|
|
|
|
const toMenuItems = useCallback((items: typeof tabs): MenuProps['items'] =>
|
|
items.map((tab) => {
|
|
const Icon = iconByName[tab.icon];
|
|
if (tab.key === '/settings') {
|
|
return { key: tab.key, icon: <Icon />, label: tab.title, children: settingsChildren };
|
|
}
|
|
if (tab.key === '/xray') {
|
|
return { key: tab.key, icon: <Icon />, label: tab.title, children: xrayChildren };
|
|
}
|
|
return { key: tab.key, icon: <Icon />, label: tab.title };
|
|
}),
|
|
[settingsChildren, xrayChildren]);
|
|
|
|
const openLink = useCallback(async (key: string) => {
|
|
if (key === LOGOUT_KEY) {
|
|
await HttpUtil.post('/logout');
|
|
window.location.href = window.X_UI_BASE_PATH || '/';
|
|
return;
|
|
}
|
|
navigate(key);
|
|
}, [navigate]);
|
|
|
|
const onMenuClick = useCallback<NonNullable<MenuProps['onClick']>>(({ key }) => {
|
|
openLink(String(key));
|
|
}, [openLink]);
|
|
|
|
const onSiderCollapse = useCallback((isCollapsed: boolean, type: 'clickTrigger' | 'responsive') => {
|
|
if (type === 'clickTrigger') {
|
|
localStorage.setItem(SIDEBAR_COLLAPSED_KEY, String(isCollapsed));
|
|
setCollapsed(isCollapsed);
|
|
}
|
|
}, []);
|
|
|
|
const cycleTheme = useCallback((id: string) => {
|
|
pauseAnimationsUntilLeave(id);
|
|
if (!isDark) {
|
|
toggleTheme();
|
|
if (isUltra) toggleUltra();
|
|
} else if (!isUltra) {
|
|
toggleUltra();
|
|
} else {
|
|
toggleUltra();
|
|
toggleTheme();
|
|
}
|
|
}, [isDark, isUltra, toggleTheme, toggleUltra]);
|
|
|
|
return (
|
|
<div className="ant-sidebar">
|
|
<Layout.Sider
|
|
theme={currentTheme}
|
|
width={220}
|
|
collapsible
|
|
collapsed={collapsed}
|
|
breakpoint="md"
|
|
onCollapse={onSiderCollapse}
|
|
>
|
|
<div className={`sider-brand${collapsed ? ' sider-brand-collapsed' : ''}`}>
|
|
<div className="brand-block">
|
|
<span className="brand-text">{collapsed ? '3X' : '3X-UI'}</span>
|
|
</div>
|
|
{!collapsed && (
|
|
<div className="brand-actions">
|
|
<DonateButton ariaLabel={t('menu.donate') || 'Donate'} />
|
|
<ThemeCycleButton
|
|
id="theme-cycle"
|
|
isDark={isDark}
|
|
isUltra={isUltra}
|
|
onCycle={() => cycleTheme('theme-cycle')}
|
|
ariaLabel={t('menu.theme')}
|
|
/>
|
|
</div>
|
|
)}
|
|
</div>
|
|
<Menu
|
|
theme={currentTheme}
|
|
mode="inline"
|
|
selectedKeys={[selectedKey]}
|
|
openKeys={collapsed ? undefined : openKeys}
|
|
onOpenChange={(keys) => setOpenKeys(keys as string[])}
|
|
className="sider-nav"
|
|
items={toMenuItems(navItems)}
|
|
onClick={onMenuClick}
|
|
/>
|
|
<Menu
|
|
theme={currentTheme}
|
|
mode="inline"
|
|
selectedKeys={[selectedKey]}
|
|
className="sider-utility"
|
|
items={toMenuItems(utilItems)}
|
|
onClick={onMenuClick}
|
|
/>
|
|
<div className="sider-footer">
|
|
<VersionBadge version={panelVersion} collapsed={collapsed} />
|
|
</div>
|
|
</Layout.Sider>
|
|
|
|
<Drawer
|
|
placement="left"
|
|
closable={false}
|
|
open={drawerOpen}
|
|
rootClassName={currentTheme}
|
|
size="min(82vw, 320px)"
|
|
styles={{
|
|
wrapper: { padding: 0 },
|
|
body: { padding: 0, display: 'flex', flexDirection: 'column', height: '100%' },
|
|
header: { display: 'none' },
|
|
}}
|
|
onClose={() => setDrawerOpen(false)}
|
|
>
|
|
<div className="drawer-header">
|
|
<div className="brand-block">
|
|
<span className="drawer-brand">3X-UI</span>
|
|
</div>
|
|
<div className="drawer-header-actions">
|
|
<DonateButton ariaLabel={t('menu.donate') || 'Donate'} />
|
|
<ThemeCycleButton
|
|
id="theme-cycle-drawer"
|
|
isDark={isDark}
|
|
isUltra={isUltra}
|
|
onCycle={() => cycleTheme('theme-cycle-drawer')}
|
|
ariaLabel={t('menu.theme')}
|
|
/>
|
|
<button
|
|
className="drawer-close"
|
|
type="button"
|
|
aria-label={t('close')}
|
|
onClick={() => setDrawerOpen(false)}
|
|
>
|
|
<CloseOutlined />
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<Menu
|
|
theme={currentTheme}
|
|
mode="inline"
|
|
selectedKeys={[selectedKey]}
|
|
openKeys={openKeys}
|
|
onOpenChange={(keys) => setOpenKeys(keys as string[])}
|
|
className="drawer-menu drawer-nav"
|
|
items={toMenuItems(navItems)}
|
|
onClick={(info) => { onMenuClick(info); setDrawerOpen(false); }}
|
|
/>
|
|
<Menu
|
|
theme={currentTheme}
|
|
mode="inline"
|
|
selectedKeys={[selectedKey]}
|
|
className="drawer-menu drawer-utility"
|
|
items={toMenuItems(utilItems)}
|
|
onClick={(info) => { onMenuClick(info); setDrawerOpen(false); }}
|
|
/>
|
|
<div className="drawer-footer">
|
|
<VersionBadge version={panelVersion} />
|
|
</div>
|
|
</Drawer>
|
|
|
|
{!drawerOpen && (
|
|
<button
|
|
className="drawer-handle"
|
|
type="button"
|
|
aria-label={t('menu.dashboard')}
|
|
onClick={() => setDrawerOpen(true)}
|
|
>
|
|
<MenuOutlined />
|
|
</button>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|