Sidebar.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. 'use client';
  2. import { useState, useEffect, useCallback } from 'react';
  3. import Link from 'next/link';
  4. import Image from 'next/image';
  5. import { usePathname } from 'next/navigation';
  6. import {
  7. ArrowLeft,
  8. Bell,
  9. Calculator,
  10. ChartLine,
  11. ChevronDown,
  12. ChevronsUpDown,
  13. CircleAlert,
  14. Coins,
  15. Gauge,
  16. Heart,
  17. Landmark,
  18. LogOut,
  19. FileText,
  20. Target,
  21. Trophy,
  22. User,
  23. Users,
  24. Wallet,
  25. } from 'lucide-react';
  26. import {
  27. Sidebar as SidebarRoot,
  28. SidebarContent,
  29. SidebarHeader,
  30. SidebarMenu,
  31. SidebarMenuButton,
  32. SidebarMenuItem,
  33. SidebarMenuSub,
  34. SidebarMenuSubButton,
  35. SidebarMenuSubItem,
  36. SidebarGroup,
  37. SidebarGroupContent,
  38. useSidebar,
  39. } from '@/components/ui/sidebar';
  40. import { Collapsible, CollapsibleContent, CollapsibleTrigger } from '@/components/ui/collapsible';
  41. import {
  42. DropdownMenu,
  43. DropdownMenuContent,
  44. DropdownMenuItem,
  45. DropdownMenuLabel,
  46. DropdownMenuSeparator,
  47. DropdownMenuTrigger,
  48. } from '@/components/ui/dropdown-menu';
  49. import { fetchApi } from '@/lib/utils/client';
  50. import type { StudioSettingsResponse } from '@/types/response/studio/settings';
  51. export default function Sidebar()
  52. {
  53. const pathname = usePathname();
  54. const { isMobile } = useSidebar();
  55. const [channel, setChannel] = useState<Pick<StudioSettingsResponse, 'isYouTubeConnected'|'channelName'|'handle'|'thumbnailUrl'>|null>(null);
  56. const isDonationPath = pathname.startsWith('/studio/donation');
  57. const isWalletPath = pathname.startsWith('/studio/wallet');
  58. const isSettlementPath = pathname.startsWith('/studio/settlement');
  59. const readCookie = (key: string, fallback: boolean) => {
  60. if (typeof document === 'undefined') return fallback;
  61. const match = document.cookie.match(new RegExp(`(?:^|; )${key}=([^;]*)`));
  62. if (!match) return fallback;
  63. return match[1] !== 'false';
  64. };
  65. const [donationOpen, setDonationOpen] = useState(() => readCookie('studio_menu_donation', true) || isDonationPath);
  66. const [walletOpen, setWalletOpen] = useState(() => readCookie('studio_menu_wallet', true) || isWalletPath);
  67. const [settlementOpen, setSettlementOpen] = useState(() => readCookie('studio_menu_settlement', true) || isSettlementPath);
  68. const saveCookie = useCallback((key: string, value: boolean) => {
  69. document.cookie = `${key}=${value}; path=/studio; max-age=${60 * 60 * 24 * 365}; SameSite=Lax`;
  70. }, []);
  71. const handleDonationToggle = useCallback((open: boolean) => {
  72. setDonationOpen(open);
  73. saveCookie('studio_menu_donation', open);
  74. }, [saveCookie]);
  75. const handleWalletToggle = useCallback((open: boolean) => {
  76. setWalletOpen(open);
  77. saveCookie('studio_menu_wallet', open);
  78. }, [saveCookie]);
  79. const handleSettlementToggle = useCallback((open: boolean) => {
  80. setSettlementOpen(open);
  81. saveCookie('studio_menu_settlement', open);
  82. }, [saveCookie]);
  83. const isActive = (href: string) => pathname === href || pathname.startsWith(href + '/');
  84. useEffect(() => {
  85. fetchApi<StudioSettingsResponse>('/api/studio/settings')
  86. .then(res => {
  87. if (res.data) {
  88. setChannel(res.data);
  89. }
  90. })
  91. .catch(() => {});
  92. }, []);
  93. const isConnected = channel?.isYouTubeConnected === true;
  94. return (
  95. <SidebarRoot collapsible="icon">
  96. <SidebarHeader>
  97. {/* 상단 Studio 라벨 + 돌아가기 버튼 */}
  98. <div className="flex items-center gap-1.5 px-2 py-1 group-data-[collapsible=icon]:hidden">
  99. <Link
  100. href="/"
  101. className="flex size-6 items-center justify-center rounded text-sidebar-foreground/50 hover:bg-sidebar-accent hover:text-sidebar-foreground transition-colors"
  102. title="뒤로가기"
  103. >
  104. <ArrowLeft className="size-3.5" />
  105. </Link>
  106. <span className="text-xs font-semibold text-sidebar-foreground/50">Studio</span>
  107. </div>
  108. {/* YouTube 채널 카드 */}
  109. <SidebarMenu>
  110. <SidebarMenuItem>
  111. <DropdownMenu>
  112. <DropdownMenuTrigger asChild>
  113. <SidebarMenuButton
  114. size="lg"
  115. className="data-[state=open]:bg-sidebar-accent data-[state=open]:text-sidebar-accent-foreground"
  116. tooltip={channel?.channelName ?? 'YouTube 채널'}
  117. >
  118. {/* 아이콘 (collapse 시 이것만 표시) */}
  119. <div className={`relative flex size-8 shrink-0 items-center justify-center rounded-lg ${!channel?.thumbnailUrl ? 'bg-sidebar-primary' : 'border'} text-sidebar-primary-foreground overflow-hidden`}>
  120. {channel?.thumbnailUrl ? (
  121. <Image
  122. src={channel.thumbnailUrl}
  123. alt={channel.channelName ?? 'YouTube'}
  124. width={32}
  125. height={32}
  126. className="size-full object-cover"
  127. />
  128. ) : (
  129. <span className="text-xs font-bold">
  130. <User />
  131. </span>
  132. )}
  133. {/* 미연동 경고 뱃지 */}
  134. {channel !== null && !isConnected && (
  135. <span className="absolute -right-1 -bottom-1 flex size-3.5 items-center justify-center rounded-full bg-background">
  136. <CircleAlert className="size-2.5 text-amber-500" />
  137. </span>
  138. )}
  139. </div>
  140. {/* 텍스트 (collapse 시 hidden) */}
  141. <div className="grid flex-1 text-left text-sm leading-tight">
  142. <span className="truncate font-semibold">
  143. {isConnected
  144. ? (channel?.channelName ?? '채널 이름')
  145. : '채널 미연동'}
  146. </span>
  147. <span className="truncate text-xs text-sidebar-foreground/60">
  148. {isConnected
  149. ? (channel?.handle ?? '')
  150. : '채널을 연결해 주세요'}
  151. </span>
  152. </div>
  153. <ChevronsUpDown className="ml-auto size-4 shrink-0" />
  154. </SidebarMenuButton>
  155. </DropdownMenuTrigger>
  156. <DropdownMenuContent
  157. className="w-[--anchor-width] min-w-56 rounded-lg"
  158. align="start"
  159. side={isMobile ? 'bottom' : 'right'}
  160. sideOffset={4}
  161. >
  162. {isConnected ? (
  163. <>
  164. <DropdownMenuLabel className="text-xs text-muted-foreground">
  165. {channel?.channelName}
  166. </DropdownMenuLabel>
  167. <DropdownMenuSeparator />
  168. <DropdownMenuItem asChild>
  169. <Link href="/studio/settings">채널 정보 보기</Link>
  170. </DropdownMenuItem>
  171. </>
  172. ) : (
  173. <>
  174. <DropdownMenuLabel className="text-xs text-muted-foreground">
  175. YouTube 채널이 연동되지 않았습니다
  176. </DropdownMenuLabel>
  177. <DropdownMenuSeparator />
  178. <DropdownMenuItem asChild>
  179. <Link href="/studio/settings">YouTube 연동하기</Link>
  180. </DropdownMenuItem>
  181. </>
  182. )}
  183. </DropdownMenuContent>
  184. </DropdownMenu>
  185. </SidebarMenuItem>
  186. </SidebarMenu>
  187. </SidebarHeader>
  188. <SidebarContent>
  189. <SidebarGroup>
  190. <SidebarGroupContent>
  191. <SidebarMenu>
  192. {/* 대시보드 */}
  193. <SidebarMenuItem>
  194. <SidebarMenuButton asChild isActive={isActive('/studio/dashboard')} tooltip="대시보드">
  195. <Link href="/studio/dashboard">
  196. <Gauge />
  197. <span>대시보드</span>
  198. </Link>
  199. </SidebarMenuButton>
  200. </SidebarMenuItem>
  201. {/* 후원 설정 */}
  202. <Collapsible open={donationOpen} onOpenChange={handleDonationToggle}>
  203. <SidebarMenuItem>
  204. <CollapsibleTrigger asChild>
  205. <SidebarMenuButton isActive={isDonationPath} tooltip="후원 설정">
  206. <Heart />
  207. <span>후원 설정</span>
  208. <ChevronDown
  209. className={`ml-auto size-4 transition-transform duration-200${donationOpen ? ' rotate-180' : ''}`}
  210. />
  211. </SidebarMenuButton>
  212. </CollapsibleTrigger>
  213. <CollapsibleContent>
  214. <SidebarMenuSub>
  215. <SidebarMenuSubItem>
  216. <SidebarMenuSubButton asChild isActive={isActive('/studio/donation/alert')}>
  217. <Link href="/studio/donation/alert">
  218. <Bell />
  219. <span>알림</span>
  220. </Link>
  221. </SidebarMenuSubButton>
  222. </SidebarMenuSubItem>
  223. <SidebarMenuSubItem>
  224. <SidebarMenuSubButton asChild isActive={isActive('/studio/donation/goal')}>
  225. <Link href="/studio/donation/goal">
  226. <Target />
  227. <span>목표</span>
  228. </Link>
  229. </SidebarMenuSubButton>
  230. </SidebarMenuSubItem>
  231. <SidebarMenuSubItem>
  232. <SidebarMenuSubButton asChild isActive={isActive('/studio/donation/rank')}>
  233. <Link href="/studio/donation/rank">
  234. <Trophy />
  235. <span>순위</span>
  236. </Link>
  237. </SidebarMenuSubButton>
  238. </SidebarMenuSubItem>
  239. <SidebarMenuSubItem>
  240. <SidebarMenuSubButton asChild isActive={isActive('/studio/donation/crew')}>
  241. <Link href="/studio/donation/crew">
  242. <Users />
  243. <span>크루</span>
  244. </Link>
  245. </SidebarMenuSubButton>
  246. </SidebarMenuSubItem>
  247. </SidebarMenuSub>
  248. </CollapsibleContent>
  249. </SidebarMenuItem>
  250. </Collapsible>
  251. {/* 지갑 */}
  252. <Collapsible open={walletOpen} onOpenChange={handleWalletToggle}>
  253. <SidebarMenuItem>
  254. <CollapsibleTrigger asChild>
  255. <SidebarMenuButton isActive={isWalletPath} tooltip="지갑">
  256. <Wallet />
  257. <span>지갑</span>
  258. <ChevronDown
  259. className={`ml-auto size-4 transition-transform duration-200${walletOpen ? ' rotate-180' : ''}`}
  260. />
  261. </SidebarMenuButton>
  262. </CollapsibleTrigger>
  263. <CollapsibleContent>
  264. <SidebarMenuSub>
  265. <SidebarMenuSubItem>
  266. <SidebarMenuSubButton asChild isActive={isActive('/studio/wallet/balance')}>
  267. <Link href="/studio/wallet/balance">
  268. <Coins />
  269. <span>잔액</span>
  270. </Link>
  271. </SidebarMenuSubButton>
  272. </SidebarMenuSubItem>
  273. <SidebarMenuSubItem>
  274. <SidebarMenuSubButton asChild isActive={isActive('/studio/wallet/revenue')}>
  275. <Link href="/studio/wallet/revenue">
  276. <ChartLine />
  277. <span>수익</span>
  278. </Link>
  279. </SidebarMenuSubButton>
  280. </SidebarMenuSubItem>
  281. <SidebarMenuSubItem>
  282. <SidebarMenuSubButton asChild isActive={isActive('/studio/wallet/withdraw')}>
  283. <Link href="/studio/wallet/withdraw">
  284. <LogOut />
  285. <span>출금</span>
  286. </Link>
  287. </SidebarMenuSubButton>
  288. </SidebarMenuSubItem>
  289. </SidebarMenuSub>
  290. </CollapsibleContent>
  291. </SidebarMenuItem>
  292. </Collapsible>
  293. {/* 정산 */}
  294. <Collapsible open={settlementOpen} onOpenChange={handleSettlementToggle}>
  295. <SidebarMenuItem>
  296. <CollapsibleTrigger asChild>
  297. <SidebarMenuButton isActive={isSettlementPath} tooltip="정산">
  298. <Calculator />
  299. <span>정산</span>
  300. <ChevronDown
  301. className={`ml-auto size-4 transition-transform duration-200${settlementOpen ? ' rotate-180' : ''}`}
  302. />
  303. </SidebarMenuButton>
  304. </CollapsibleTrigger>
  305. <CollapsibleContent>
  306. <SidebarMenuSub>
  307. <SidebarMenuSubItem>
  308. <SidebarMenuSubButton asChild isActive={isActive('/studio/settlement/account')}>
  309. <Link href="/studio/settlement/account">
  310. <Landmark />
  311. <span>계좌 관리</span>
  312. </Link>
  313. </SidebarMenuSubButton>
  314. </SidebarMenuSubItem>
  315. <SidebarMenuSubItem>
  316. <SidebarMenuSubButton asChild isActive={isActive('/studio/settlement/tax')}>
  317. <Link href="/studio/settlement/tax">
  318. <FileText />
  319. <span>원천징수 내역</span>
  320. </Link>
  321. </SidebarMenuSubButton>
  322. </SidebarMenuSubItem>
  323. </SidebarMenuSub>
  324. </CollapsibleContent>
  325. </SidebarMenuItem>
  326. </Collapsible>
  327. </SidebarMenu>
  328. </SidebarGroupContent>
  329. </SidebarGroup>
  330. </SidebarContent>
  331. </SidebarRoot>
  332. );
  333. }