Sidebar: make it possible to set the sidebar as collapsed (#72)
* Sidebar: make it possible to set the sidebar as collapsed
This commit is contained in:
committed by
GitHub
parent
d137df2675
commit
b2c27eb25b
@@ -52,6 +52,7 @@ function SuspendedPersonalAccountDropdown(props: { user: User | null }) {
|
|||||||
if (userData) {
|
if (userData) {
|
||||||
return (
|
return (
|
||||||
<PersonalAccountDropdown
|
<PersonalAccountDropdown
|
||||||
|
showProfileName={false}
|
||||||
paths={paths}
|
paths={paths}
|
||||||
features={features}
|
features={features}
|
||||||
user={userData}
|
user={userData}
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ export function AdminSidebar(props: { user: User }) {
|
|||||||
</SidebarContent>
|
</SidebarContent>
|
||||||
|
|
||||||
<SidebarContent className={'mt-5'}>
|
<SidebarContent className={'mt-5'}>
|
||||||
<SidebarGroup label={'Admin'} collapsible={false}>
|
<SidebarGroup label={'Admin'}>
|
||||||
<SidebarItem end path={'/admin'} Icon={<Home className={'h-4'} />}>
|
<SidebarItem end path={'/admin'} Icon={<Home className={'h-4'} />}>
|
||||||
Home
|
Home
|
||||||
</SidebarItem>
|
</SidebarItem>
|
||||||
@@ -35,7 +35,7 @@ export function AdminSidebar(props: { user: User }) {
|
|||||||
</SidebarContent>
|
</SidebarContent>
|
||||||
|
|
||||||
<SidebarContent className={'absolute bottom-4'}>
|
<SidebarContent className={'absolute bottom-4'}>
|
||||||
<ProfileAccountDropdownContainer user={props.user} collapsed={false} />
|
<ProfileAccountDropdownContainer user={props.user} />
|
||||||
</SidebarContent>
|
</SidebarContent>
|
||||||
</Sidebar>
|
</Sidebar>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
|
import { useContext } from 'react';
|
||||||
|
|
||||||
import { useRouter } from 'next/navigation';
|
import { useRouter } from 'next/navigation';
|
||||||
|
|
||||||
import { AccountSelector } from '@kit/accounts/account-selector';
|
import { AccountSelector } from '@kit/accounts/account-selector';
|
||||||
|
import { SidebarContext } from '@kit/ui/sidebar';
|
||||||
|
|
||||||
import featureFlagsConfig from '~/config/feature-flags.config';
|
import featureFlagsConfig from '~/config/feature-flags.config';
|
||||||
import pathsConfig from '~/config/paths.config';
|
import pathsConfig from '~/config/paths.config';
|
||||||
@@ -19,13 +22,13 @@ export function HomeAccountSelector(props: {
|
|||||||
}>;
|
}>;
|
||||||
|
|
||||||
userId: string;
|
userId: string;
|
||||||
collapsed: boolean;
|
|
||||||
}) {
|
}) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
const { collapsed } = useContext(SidebarContext);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AccountSelector
|
<AccountSelector
|
||||||
collapsed={props.collapsed}
|
collapsed={collapsed}
|
||||||
accounts={props.accounts}
|
accounts={props.accounts}
|
||||||
features={features}
|
features={features}
|
||||||
userId={props.userId}
|
userId={props.userId}
|
||||||
|
|||||||
@@ -53,14 +53,12 @@ export function HomeMenuNavigation(props: { workspace: UserWorkspace }) {
|
|||||||
<HomeAccountSelector
|
<HomeAccountSelector
|
||||||
userId={user.id}
|
userId={user.id}
|
||||||
accounts={accounts}
|
accounts={accounts}
|
||||||
collapsed={false}
|
|
||||||
/>
|
/>
|
||||||
</If>
|
</If>
|
||||||
|
|
||||||
<UserNotifications userId={user.id} />
|
<UserNotifications userId={user.id} />
|
||||||
|
|
||||||
<ProfileAccountDropdownContainer
|
<ProfileAccountDropdownContainer
|
||||||
collapsed={true}
|
|
||||||
user={user}
|
user={user}
|
||||||
account={workspace}
|
account={workspace}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -71,7 +71,6 @@ export function HomeMobileNavigation(props: { workspace: UserWorkspace }) {
|
|||||||
<HomeAccountSelector
|
<HomeAccountSelector
|
||||||
userId={props.workspace.user.id}
|
userId={props.workspace.user.id}
|
||||||
accounts={props.workspace.accounts}
|
accounts={props.workspace.accounts}
|
||||||
collapsed={false}
|
|
||||||
/>
|
/>
|
||||||
</DropdownMenuGroup>
|
</DropdownMenuGroup>
|
||||||
|
|
||||||
|
|||||||
@@ -11,25 +11,28 @@ import { UserNotifications } from '~/home/(user)/_components/user-notifications'
|
|||||||
import type { UserWorkspace } from '../_lib/server/load-user-workspace';
|
import type { UserWorkspace } from '../_lib/server/load-user-workspace';
|
||||||
import { HomeAccountSelector } from './home-account-selector';
|
import { HomeAccountSelector } from './home-account-selector';
|
||||||
|
|
||||||
export function HomeSidebar(props: { workspace: UserWorkspace }) {
|
interface HomeSidebarProps {
|
||||||
|
workspace: UserWorkspace;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function HomeSidebar(props: HomeSidebarProps) {
|
||||||
const { workspace, user, accounts } = props.workspace;
|
const { workspace, user, accounts } = props.workspace;
|
||||||
|
const collapsed = personalAccountNavigationConfig.sidebarCollapsed;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Sidebar>
|
<Sidebar collapsed={collapsed}>
|
||||||
<SidebarContent className={'h-16 justify-center'}>
|
<SidebarContent className={'h-16 justify-center'}>
|
||||||
<div className={'flex items-center justify-between space-x-2'}>
|
<div className={'flex items-center justify-between space-x-2'}>
|
||||||
<If
|
<If
|
||||||
condition={featuresFlagConfig.enableTeamAccounts}
|
condition={featuresFlagConfig.enableTeamAccounts}
|
||||||
fallback={<AppLogo className={'py-2'} />}
|
fallback={<AppLogo className={'py-2'} />}
|
||||||
>
|
>
|
||||||
<HomeAccountSelector
|
<HomeAccountSelector userId={user.id} accounts={accounts} />
|
||||||
userId={user.id}
|
|
||||||
collapsed={false}
|
|
||||||
accounts={accounts}
|
|
||||||
/>
|
|
||||||
</If>
|
</If>
|
||||||
|
|
||||||
<UserNotifications userId={user.id} />
|
<div className={'hidden group-aria-[expanded=true]/sidebar:block'}>
|
||||||
|
<UserNotifications userId={user.id} />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</SidebarContent>
|
</SidebarContent>
|
||||||
|
|
||||||
@@ -40,7 +43,6 @@ export function HomeSidebar(props: { workspace: UserWorkspace }) {
|
|||||||
<div className={'absolute bottom-4 left-0 w-full'}>
|
<div className={'absolute bottom-4 left-0 w-full'}>
|
||||||
<SidebarContent>
|
<SidebarContent>
|
||||||
<ProfileAccountDropdownContainer
|
<ProfileAccountDropdownContainer
|
||||||
collapsed={false}
|
|
||||||
user={user}
|
user={user}
|
||||||
account={workspace}
|
account={workspace}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -39,6 +39,7 @@ function UserHomeLayout({ children }: React.PropsWithChildren) {
|
|||||||
|
|
||||||
<PageMobileNavigation className={'flex items-center justify-between'}>
|
<PageMobileNavigation className={'flex items-center justify-between'}>
|
||||||
<AppLogo />
|
<AppLogo />
|
||||||
|
|
||||||
<HomeMobileNavigation workspace={workspace} />
|
<HomeMobileNavigation workspace={workspace} />
|
||||||
</PageMobileNavigation>
|
</PageMobileNavigation>
|
||||||
|
|
||||||
@@ -56,4 +57,4 @@ function getLayoutStyle() {
|
|||||||
(cookies().get('layout-style')?.value as PageLayoutStyle) ??
|
(cookies().get('layout-style')?.value as PageLayoutStyle) ??
|
||||||
personalAccountNavigationConfig.style
|
personalAccountNavigationConfig.style
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -20,6 +20,8 @@ export function TeamAccountAccountsSelector(params: {
|
|||||||
value: string | null;
|
value: string | null;
|
||||||
image: string | null;
|
image: string | null;
|
||||||
}>;
|
}>;
|
||||||
|
|
||||||
|
collapsed?: boolean;
|
||||||
}) {
|
}) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
@@ -28,7 +30,7 @@ export function TeamAccountAccountsSelector(params: {
|
|||||||
selectedAccount={params.selectedAccount}
|
selectedAccount={params.selectedAccount}
|
||||||
accounts={params.accounts}
|
accounts={params.accounts}
|
||||||
userId={params.userId}
|
userId={params.userId}
|
||||||
collapsed={false}
|
collapsed={params.collapsed}
|
||||||
features={features}
|
features={features}
|
||||||
onAccountChange={(value) => {
|
onAccountChange={(value) => {
|
||||||
const path = value
|
const path = value
|
||||||
|
|||||||
@@ -1,8 +1,14 @@
|
|||||||
import { User } from '@supabase/supabase-js';
|
'use client';
|
||||||
|
|
||||||
import { Sidebar, SidebarContent } from '@kit/ui/sidebar';
|
import { useContext } from 'react';
|
||||||
|
|
||||||
|
import type { User } from '@supabase/supabase-js';
|
||||||
|
|
||||||
|
import { Sidebar, SidebarContent, SidebarContext } from '@kit/ui/sidebar';
|
||||||
|
import { cn } from '@kit/ui/utils';
|
||||||
|
|
||||||
import { ProfileAccountDropdownContainer } from '~/components//personal-account-dropdown-container';
|
import { ProfileAccountDropdownContainer } from '~/components//personal-account-dropdown-container';
|
||||||
|
import { getTeamAccountSidebarConfig } from '~/config/team-account-navigation.config';
|
||||||
import { TeamAccountNotifications } from '~/home/[account]/_components/team-account-notifications';
|
import { TeamAccountNotifications } from '~/home/[account]/_components/team-account-notifications';
|
||||||
|
|
||||||
import { TeamAccountAccountsSelector } from '../_components/team-account-accounts-selector';
|
import { TeamAccountAccountsSelector } from '../_components/team-account-accounts-selector';
|
||||||
@@ -18,11 +24,12 @@ export function TeamAccountLayoutSidebar(props: {
|
|||||||
account: string;
|
account: string;
|
||||||
accountId: string;
|
accountId: string;
|
||||||
accounts: AccountModel[];
|
accounts: AccountModel[];
|
||||||
collapsed: boolean;
|
|
||||||
user: User;
|
user: User;
|
||||||
}) {
|
}) {
|
||||||
|
const collapsed = getTeamAccountSidebarConfig(props.account).sidebarCollapsed;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Sidebar>
|
<Sidebar collapsed={collapsed}>
|
||||||
<SidebarContainer
|
<SidebarContainer
|
||||||
account={props.account}
|
account={props.account}
|
||||||
accountId={props.accountId}
|
accountId={props.accountId}
|
||||||
@@ -37,28 +44,40 @@ function SidebarContainer(props: {
|
|||||||
account: string;
|
account: string;
|
||||||
accountId: string;
|
accountId: string;
|
||||||
accounts: AccountModel[];
|
accounts: AccountModel[];
|
||||||
collapsible?: boolean;
|
|
||||||
user: User;
|
user: User;
|
||||||
}) {
|
}) {
|
||||||
const { account, accounts, user } = props;
|
const { account, accounts, user } = props;
|
||||||
const userId = user.id;
|
const userId = user.id;
|
||||||
|
const { collapsed } = useContext(SidebarContext);
|
||||||
|
|
||||||
|
const className = cn(
|
||||||
|
'flex max-w-full items-center justify-between space-x-4',
|
||||||
|
{
|
||||||
|
'w-full justify-start space-x-0': collapsed,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<SidebarContent className={'h-16 justify-center'}>
|
<SidebarContent className={'h-16 justify-center'}>
|
||||||
<div
|
<div className={className}>
|
||||||
className={'flex max-w-full items-center justify-between space-x-4'}
|
|
||||||
>
|
|
||||||
<TeamAccountAccountsSelector
|
<TeamAccountAccountsSelector
|
||||||
userId={userId}
|
userId={userId}
|
||||||
selectedAccount={account}
|
selectedAccount={account}
|
||||||
accounts={accounts}
|
accounts={accounts}
|
||||||
|
collapsed={collapsed}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<TeamAccountNotifications
|
<div
|
||||||
userId={userId}
|
className={cn({
|
||||||
accountId={props.accountId}
|
hidden: collapsed,
|
||||||
/>
|
})}
|
||||||
|
>
|
||||||
|
<TeamAccountNotifications
|
||||||
|
userId={userId}
|
||||||
|
accountId={props.accountId}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</SidebarContent>
|
</SidebarContent>
|
||||||
|
|
||||||
@@ -70,7 +89,6 @@ function SidebarContainer(props: {
|
|||||||
<SidebarContent>
|
<SidebarContent>
|
||||||
<ProfileAccountDropdownContainer
|
<ProfileAccountDropdownContainer
|
||||||
user={props.user}
|
user={props.user}
|
||||||
collapsed={false}
|
|
||||||
/>
|
/>
|
||||||
</SidebarContent>
|
</SidebarContent>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -62,7 +62,6 @@ export function TeamAccountNavigationMenu(props: {
|
|||||||
<TeamAccountNotifications accountId={account.id} userId={user.id} />
|
<TeamAccountNotifications accountId={account.id} userId={user.id} />
|
||||||
|
|
||||||
<ProfileAccountDropdownContainer
|
<ProfileAccountDropdownContainer
|
||||||
collapsed={true}
|
|
||||||
user={user}
|
user={user}
|
||||||
account={account}
|
account={account}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ import { TeamAccountLayoutSidebar } from './_components/team-account-layout-side
|
|||||||
import { TeamAccountNavigationMenu } from './_components/team-account-navigation-menu';
|
import { TeamAccountNavigationMenu } from './_components/team-account-navigation-menu';
|
||||||
import { loadTeamWorkspace } from './_lib/server/team-account-workspace.loader';
|
import { loadTeamWorkspace } from './_lib/server/team-account-workspace.loader';
|
||||||
|
|
||||||
interface Params {
|
interface TeamWorkspaceLayoutParams {
|
||||||
account: string;
|
account: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -29,7 +29,7 @@ function TeamWorkspaceLayout({
|
|||||||
children,
|
children,
|
||||||
params,
|
params,
|
||||||
}: React.PropsWithChildren<{
|
}: React.PropsWithChildren<{
|
||||||
params: Params;
|
params: TeamWorkspaceLayoutParams;
|
||||||
}>) {
|
}>) {
|
||||||
const data = use(loadTeamWorkspace(params.account));
|
const data = use(loadTeamWorkspace(params.account));
|
||||||
const style = getLayoutStyle(params.account);
|
const style = getLayoutStyle(params.account);
|
||||||
@@ -45,7 +45,6 @@ function TeamWorkspaceLayout({
|
|||||||
<PageNavigation>
|
<PageNavigation>
|
||||||
<If condition={style === 'sidebar'}>
|
<If condition={style === 'sidebar'}>
|
||||||
<TeamAccountLayoutSidebar
|
<TeamAccountLayoutSidebar
|
||||||
collapsed={false}
|
|
||||||
account={params.account}
|
account={params.account}
|
||||||
accountId={data.account.id}
|
accountId={data.account.id}
|
||||||
accounts={accounts}
|
accounts={accounts}
|
||||||
|
|||||||
@@ -18,7 +18,6 @@ const features = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export function ProfileAccountDropdownContainer(props: {
|
export function ProfileAccountDropdownContainer(props: {
|
||||||
collapsed: boolean;
|
|
||||||
user: User;
|
user: User;
|
||||||
|
|
||||||
account?: {
|
account?: {
|
||||||
@@ -32,16 +31,13 @@ export function ProfileAccountDropdownContainer(props: {
|
|||||||
const userData = user.data as User;
|
const userData = user.data as User;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={props.collapsed ? '' : 'w-full'}>
|
<PersonalAccountDropdown
|
||||||
<PersonalAccountDropdown
|
className={'w-full'}
|
||||||
className={'w-full'}
|
paths={paths}
|
||||||
paths={paths}
|
features={features}
|
||||||
features={features}
|
user={userData}
|
||||||
showProfileName={!props.collapsed}
|
account={props.account}
|
||||||
user={userData}
|
signOutRequested={() => signOut.mutateAsync()}
|
||||||
account={props.account}
|
/>
|
||||||
signOutRequested={() => signOut.mutateAsync()}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,4 +32,5 @@ if (featureFlagsConfig.enablePersonalAccountBilling) {
|
|||||||
export const personalAccountNavigationConfig = NavigationConfigSchema.parse({
|
export const personalAccountNavigationConfig = NavigationConfigSchema.parse({
|
||||||
routes,
|
routes,
|
||||||
style: process.env.NEXT_PUBLIC_USER_NAVIGATION_STYLE,
|
style: process.env.NEXT_PUBLIC_USER_NAVIGATION_STYLE,
|
||||||
|
sidebarCollapsed: process.env.NEXT_PUBLIC_HOME_SIDEBAR_COLLAPSED,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -43,6 +43,7 @@ export function getTeamAccountSidebarConfig(account: string) {
|
|||||||
return NavigationConfigSchema.parse({
|
return NavigationConfigSchema.parse({
|
||||||
routes: getRoutes(account),
|
routes: getRoutes(account),
|
||||||
style: process.env.NEXT_PUBLIC_TEAM_NAVIGATION_STYLE,
|
style: process.env.NEXT_PUBLIC_TEAM_NAVIGATION_STYLE,
|
||||||
|
sidebarCollapsed: process.env.NEXT_PUBLIC_TEAM_SIDEBAR_COLLAPSED,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -101,7 +101,7 @@ export function AccountSelector({
|
|||||||
'dark:shadow-primary/10 group w-full min-w-0 px-2 lg:w-auto lg:max-w-fit',
|
'dark:shadow-primary/10 group w-full min-w-0 px-2 lg:w-auto lg:max-w-fit',
|
||||||
{
|
{
|
||||||
'justify-start': !collapsed,
|
'justify-start': !collapsed,
|
||||||
'justify-center': collapsed,
|
'm-auto justify-center px-4 lg:w-full': collapsed,
|
||||||
},
|
},
|
||||||
className,
|
className,
|
||||||
)}
|
)}
|
||||||
@@ -143,7 +143,11 @@ export function AccountSelector({
|
|||||||
)}
|
)}
|
||||||
</If>
|
</If>
|
||||||
|
|
||||||
<CaretSortIcon className="ml-2 h-4 w-4 shrink-0 opacity-50" />
|
<CaretSortIcon
|
||||||
|
className={cn('ml-2 h-4 w-4 shrink-0 opacity-50', {
|
||||||
|
hidden: collapsed,
|
||||||
|
})}
|
||||||
|
/>
|
||||||
</Button>
|
</Button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
|
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ export function PersonalAccountDropdown({
|
|||||||
className,
|
className,
|
||||||
user,
|
user,
|
||||||
signOutRequested,
|
signOutRequested,
|
||||||
showProfileName,
|
showProfileName = true,
|
||||||
paths,
|
paths,
|
||||||
features,
|
features,
|
||||||
account,
|
account,
|
||||||
@@ -56,8 +56,9 @@ export function PersonalAccountDropdown({
|
|||||||
enableThemeToggle: boolean;
|
enableThemeToggle: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
className?: string;
|
|
||||||
showProfileName?: boolean;
|
showProfileName?: boolean;
|
||||||
|
|
||||||
|
className?: string;
|
||||||
}) {
|
}) {
|
||||||
const { data: personalAccountData } = usePersonalAccountData(
|
const { data: personalAccountData } = usePersonalAccountData(
|
||||||
user.id,
|
user.id,
|
||||||
|
|||||||
@@ -7,6 +7,11 @@ const RouteMatchingEnd = z
|
|||||||
|
|
||||||
export const NavigationConfigSchema = z.object({
|
export const NavigationConfigSchema = z.object({
|
||||||
style: z.enum(['custom', 'sidebar', 'header']).default('sidebar'),
|
style: z.enum(['custom', 'sidebar', 'header']).default('sidebar'),
|
||||||
|
sidebarCollapsed: z
|
||||||
|
.enum(['false', 'true'])
|
||||||
|
.default('false')
|
||||||
|
.optional()
|
||||||
|
.transform((value) => value === `true`),
|
||||||
routes: z.array(
|
routes: z.array(
|
||||||
z.union([
|
z.union([
|
||||||
z.object({
|
z.object({
|
||||||
|
|||||||
@@ -30,14 +30,14 @@ function PageWithSidebar(props: PageProps) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={cn('flex bg-gray-50/50 dark:bg-background', props.className)}
|
className={cn('flex bg-gray-50/95 dark:bg-background/85', props.className)}
|
||||||
>
|
>
|
||||||
{Navigation}
|
{Navigation}
|
||||||
|
|
||||||
<div
|
<div
|
||||||
className={
|
className={
|
||||||
props.contentContainerClassName ??
|
props.contentContainerClassName ??
|
||||||
'mx-auto flex h-screen w-full flex-col overflow-y-auto px-4 lg:px-0'
|
'mx-auto flex h-screen w-full flex-col overflow-y-auto px-4 lg:px-0 bg-inherit'
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{MobileNavigation}
|
{MobileNavigation}
|
||||||
@@ -115,7 +115,7 @@ export function PageBody(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function PageNavigation(props: React.PropsWithChildren) {
|
export function PageNavigation(props: React.PropsWithChildren) {
|
||||||
return <div className={'hidden flex-1 lg:flex'}>{props.children}</div>;
|
return <div className={'hidden flex-1 lg:flex bg-inherit'}>{props.children}</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function PageDescription(props: React.PropsWithChildren) {
|
export function PageDescription(props: React.PropsWithChildren) {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { useContext, useId, useState } from 'react';
|
import { useContext, useId, useRef, useState } from 'react';
|
||||||
|
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import { usePathname } from 'next/navigation';
|
import { usePathname } from 'next/navigation';
|
||||||
@@ -24,8 +24,11 @@ import { Trans } from './trans';
|
|||||||
|
|
||||||
export type SidebarConfig = z.infer<typeof NavigationConfigSchema>;
|
export type SidebarConfig = z.infer<typeof NavigationConfigSchema>;
|
||||||
|
|
||||||
|
export { SidebarContext };
|
||||||
|
|
||||||
export function Sidebar(props: {
|
export function Sidebar(props: {
|
||||||
collapsed?: boolean;
|
collapsed?: boolean;
|
||||||
|
expandOnHover?: boolean;
|
||||||
className?: string;
|
className?: string;
|
||||||
children:
|
children:
|
||||||
| React.ReactNode
|
| React.ReactNode
|
||||||
@@ -35,19 +38,62 @@ export function Sidebar(props: {
|
|||||||
}) => React.ReactNode);
|
}) => React.ReactNode);
|
||||||
}) {
|
}) {
|
||||||
const [collapsed, setCollapsed] = useState(props.collapsed ?? false);
|
const [collapsed, setCollapsed] = useState(props.collapsed ?? false);
|
||||||
|
const isExpandedRef = useRef<boolean>(false);
|
||||||
|
|
||||||
const className = getClassNameBuilder(props.className ?? '')({
|
const expandOnHover =
|
||||||
|
props.expandOnHover ??
|
||||||
|
process.env.NEXT_PUBLIC_EXPAND_SIDEBAR_ON_HOVER === 'true';
|
||||||
|
|
||||||
|
const sidebarSizeClassName = getSidebarSizeClassName(
|
||||||
collapsed,
|
collapsed,
|
||||||
|
isExpandedRef.current,
|
||||||
|
);
|
||||||
|
|
||||||
|
const className = getClassNameBuilder(
|
||||||
|
cn(props.className ?? '', sidebarSizeClassName, {}),
|
||||||
|
)();
|
||||||
|
|
||||||
|
const containerClassName = cn(sidebarSizeClassName, 'bg-inherit', {
|
||||||
|
'max-w-[4rem]': expandOnHover && isExpandedRef.current,
|
||||||
});
|
});
|
||||||
|
|
||||||
const ctx = { collapsed, setCollapsed };
|
const ctx = { collapsed, setCollapsed };
|
||||||
|
|
||||||
|
const onMouseEnter =
|
||||||
|
props.collapsed && expandOnHover
|
||||||
|
? () => {
|
||||||
|
setCollapsed(false);
|
||||||
|
isExpandedRef.current = true;
|
||||||
|
}
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
const onMouseLeave =
|
||||||
|
props.collapsed && expandOnHover
|
||||||
|
? () => {
|
||||||
|
if (!isRadixPopupOpen()) {
|
||||||
|
setCollapsed(true);
|
||||||
|
isExpandedRef.current = false;
|
||||||
|
} else {
|
||||||
|
onRadixPopupClose(() => {
|
||||||
|
setCollapsed(true);
|
||||||
|
isExpandedRef.current = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
: undefined;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SidebarContext.Provider value={ctx}>
|
<SidebarContext.Provider value={ctx}>
|
||||||
<div className={className}>
|
<div
|
||||||
{typeof props.children === 'function'
|
className={containerClassName}
|
||||||
? props.children(ctx)
|
onMouseEnter={onMouseEnter}
|
||||||
: props.children}
|
onMouseLeave={onMouseLeave}
|
||||||
|
>
|
||||||
|
<div aria-expanded={!collapsed} className={className}>
|
||||||
|
{typeof props.children === 'function'
|
||||||
|
? props.children(ctx)
|
||||||
|
: props.children}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</SidebarContext.Provider>
|
</SidebarContext.Provider>
|
||||||
);
|
);
|
||||||
@@ -55,17 +101,22 @@ export function Sidebar(props: {
|
|||||||
|
|
||||||
export function SidebarContent({
|
export function SidebarContent({
|
||||||
children,
|
children,
|
||||||
className,
|
className: customClassName,
|
||||||
}: React.PropsWithChildren<{
|
}: React.PropsWithChildren<{
|
||||||
className?: string;
|
className?: string;
|
||||||
}>) {
|
}>) {
|
||||||
return (
|
const { collapsed } = useContext(SidebarContext);
|
||||||
<div
|
|
||||||
className={cn('flex w-full flex-col space-y-1.5 px-4 py-1', className)}
|
const className = cn(
|
||||||
>
|
'flex w-full flex-col space-y-1.5 py-1',
|
||||||
{children}
|
customClassName,
|
||||||
</div>
|
{
|
||||||
|
'px-4': !collapsed,
|
||||||
|
'px-2': collapsed,
|
||||||
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
return <div className={className}>{children}</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function SidebarGroup({
|
export function SidebarGroup({
|
||||||
@@ -96,7 +147,7 @@ export function SidebarGroup({
|
|||||||
|
|
||||||
const Wrapper = () => {
|
const Wrapper = () => {
|
||||||
const className = cn(
|
const className = cn(
|
||||||
'group flex items-center justify-between px-container space-x-2.5',
|
'px-container group flex items-center justify-between space-x-2.5',
|
||||||
{
|
{
|
||||||
'py-2.5': !sidebarCollapsed,
|
'py-2.5': !sidebarCollapsed,
|
||||||
},
|
},
|
||||||
@@ -131,7 +182,11 @@ export function SidebarGroup({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={'flex flex-col space-y-1 py-1'}>
|
<div
|
||||||
|
className={cn('flex flex-col', {
|
||||||
|
'space-y-1 py-1': !collapsed,
|
||||||
|
})}
|
||||||
|
>
|
||||||
<Wrapper />
|
<Wrapper />
|
||||||
|
|
||||||
<If condition={collapsible ? !isGroupCollapsed : true}>
|
<If condition={collapsible ? !isGroupCollapsed : true}>
|
||||||
@@ -164,48 +219,84 @@ export function SidebarItem({
|
|||||||
|
|
||||||
const active = isRouteActive(path, currentPath, end ?? false);
|
const active = isRouteActive(path, currentPath, end ?? false);
|
||||||
const variant = active ? 'secondary' : 'ghost';
|
const variant = active ? 'secondary' : 'ghost';
|
||||||
const size = collapsed ? 'icon' : 'sm';
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Button
|
<TooltipProvider delayDuration={0}>
|
||||||
asChild
|
<Tooltip disableHoverableContent>
|
||||||
className={cn('flex w-full text-sm shadow-none active:bg-secondary/60', {
|
<TooltipTrigger asChild>
|
||||||
'justify-start space-x-2.5': !collapsed,
|
<Button
|
||||||
'hover:bg-initial': active,
|
asChild
|
||||||
})}
|
className={cn(
|
||||||
size={size}
|
'flex w-full text-sm shadow-none active:bg-secondary/60',
|
||||||
variant={variant}
|
{
|
||||||
>
|
'justify-start space-x-2.5': !collapsed,
|
||||||
<Link key={path} href={path}>
|
'hover:bg-initial': active,
|
||||||
<If condition={collapsed} fallback={Icon}>
|
},
|
||||||
<TooltipProvider>
|
)}
|
||||||
<Tooltip>
|
size={'sm'}
|
||||||
<TooltipTrigger asChild>{Icon}</TooltipTrigger>
|
variant={variant}
|
||||||
|
>
|
||||||
|
<Link key={path} href={path}>
|
||||||
|
{Icon}
|
||||||
|
<span className={cn({ hidden: collapsed })}>{children}</span>
|
||||||
|
</Link>
|
||||||
|
</Button>
|
||||||
|
</TooltipTrigger>
|
||||||
|
|
||||||
<TooltipContent side={'right'} sideOffset={20}>
|
<If condition={collapsed}>
|
||||||
{children}
|
<TooltipContent side={'right'} sideOffset={10}>
|
||||||
</TooltipContent>
|
{children}
|
||||||
</Tooltip>
|
</TooltipContent>
|
||||||
</TooltipProvider>
|
|
||||||
</If>
|
</If>
|
||||||
|
</Tooltip>
|
||||||
<span className={cn({ hidden: collapsed })}>{children}</span>
|
</TooltipProvider>
|
||||||
</Link>
|
|
||||||
</Button>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function getClassNameBuilder(className: string) {
|
function getClassNameBuilder(className: string) {
|
||||||
return cva([cn('flex box-content h-screen flex-col relative', className)], {
|
return cva([
|
||||||
variants: {
|
cn(
|
||||||
collapsed: {
|
'group/sidebar transition-width fixed box-content flex h-screen w-2/12 flex-col bg-inherit backdrop-blur-sm duration-100',
|
||||||
true: `w-[6rem]`,
|
className,
|
||||||
false: `w-2/12 lg:w-[17rem]`,
|
),
|
||||||
},
|
]);
|
||||||
},
|
}
|
||||||
|
|
||||||
|
function getSidebarSizeClassName(collapsed: boolean, isExpanded: boolean) {
|
||||||
|
return cn(['z-10 flex w-full flex-col'], {
|
||||||
|
'dark:shadow-primary/20 lg:w-[17rem]': !collapsed,
|
||||||
|
'lg:w-[4rem]': collapsed,
|
||||||
|
shadow: isExpanded,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getRadixPopup() {
|
||||||
|
return document.querySelector('[data-radix-popper-content-wrapper]');
|
||||||
|
}
|
||||||
|
|
||||||
|
function isRadixPopupOpen() {
|
||||||
|
return getRadixPopup() !== null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function onRadixPopupClose(callback: () => void) {
|
||||||
|
const element = getRadixPopup();
|
||||||
|
|
||||||
|
if (element) {
|
||||||
|
const observer = new MutationObserver(() => {
|
||||||
|
if (!getRadixPopup()) {
|
||||||
|
callback();
|
||||||
|
|
||||||
|
observer.disconnect();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
observer.observe(element.parentElement!, {
|
||||||
|
childList: true,
|
||||||
|
subtree: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export function SidebarNavigation({
|
export function SidebarNavigation({
|
||||||
config,
|
config,
|
||||||
}: React.PropsWithChildren<{
|
}: React.PropsWithChildren<{
|
||||||
|
|||||||
Reference in New Issue
Block a user