Major changes: - Docker Compose: full Supabase stack (11 services) equivalent to supabase CLI - Fischerei module: 16 DB tables, waters/species/stocking/catch books/competitions - Sitzungsprotokolle module: meeting protocols, agenda items, task tracking - Verbandsverwaltung module: federation management, member clubs, contacts, fees - Per-account module activation via Modules page toggle - Site Builder: live CMS data in Puck blocks (courses, events, membership registration) - Public registration APIs: course signup, event registration, membership application - Document generation: PDF member cards, Excel reports, HTML labels - Landing page: real Com.BISS content (no filler text) - UX audit fixes: AccountNotFound component, shared status badges, confirm dialog, pagination, duplicate heading removal, emoji→badge replacement, a11y fixes - QA: healthcheck fix, API auth fix, enum mismatch fix, password required attribute
195 lines
4.5 KiB
TypeScript
195 lines
4.5 KiB
TypeScript
'use client';
|
|
|
|
import Link from 'next/link';
|
|
import { useRouter } from 'next/navigation';
|
|
|
|
import { Home, LogOut, Menu } from 'lucide-react';
|
|
import * as z from 'zod';
|
|
|
|
import { AccountSelector } from '@kit/accounts/account-selector';
|
|
import { useSignOut } from '@kit/supabase/hooks/use-sign-out';
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
DialogTrigger,
|
|
} from '@kit/ui/dialog';
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuSeparator,
|
|
DropdownMenuTrigger,
|
|
} from '@kit/ui/dropdown-menu';
|
|
import { NavigationConfigSchema } from '@kit/ui/navigation-schema';
|
|
import { Trans } from '@kit/ui/trans';
|
|
|
|
import featureFlagsConfig from '~/config/feature-flags.config';
|
|
import pathsConfig from '~/config/paths.config';
|
|
|
|
type Accounts = Array<{
|
|
label: string | null;
|
|
value: string | null;
|
|
image: string | null;
|
|
}>;
|
|
|
|
const features = {
|
|
enableTeamAccounts: featureFlagsConfig.enableTeamAccounts,
|
|
enableTeamCreation: featureFlagsConfig.enableTeamCreation,
|
|
};
|
|
|
|
export const TeamAccountLayoutMobileNavigation = (
|
|
props: React.PropsWithChildren<{
|
|
account: string;
|
|
userId: string;
|
|
accounts: Accounts;
|
|
config: z.output<typeof NavigationConfigSchema>;
|
|
}>,
|
|
) => {
|
|
const signOut = useSignOut();
|
|
|
|
const Links = props.config.routes.map(
|
|
(item, index) => {
|
|
if ('children' in item) {
|
|
return item.children.map((child) => {
|
|
return (
|
|
<DropdownLink
|
|
key={child.path}
|
|
Icon={child.Icon}
|
|
path={child.path}
|
|
label={child.label}
|
|
/>
|
|
);
|
|
});
|
|
}
|
|
|
|
if ('divider' in item) {
|
|
return <DropdownMenuSeparator key={index} />;
|
|
}
|
|
},
|
|
);
|
|
|
|
return (
|
|
<DropdownMenu>
|
|
<DropdownMenuTrigger>
|
|
<Menu className={'h-9'} />
|
|
</DropdownMenuTrigger>
|
|
|
|
<DropdownMenuContent sideOffset={10} className={'w-screen rounded-none'}>
|
|
<TeamAccountsModal
|
|
userId={props.userId}
|
|
accounts={props.accounts}
|
|
account={props.account}
|
|
/>
|
|
|
|
{Links}
|
|
|
|
<DropdownMenuSeparator />
|
|
|
|
<SignOutDropdownItem onSignOut={() => signOut.mutateAsync()} />
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
);
|
|
};
|
|
|
|
function DropdownLink(
|
|
props: React.PropsWithChildren<{
|
|
path: string;
|
|
label: string;
|
|
Icon: React.ReactNode;
|
|
}>,
|
|
) {
|
|
return (
|
|
<DropdownMenuItem
|
|
render={
|
|
<Link
|
|
href={props.path}
|
|
className={'flex h-12 w-full items-center gap-x-3 px-3'}
|
|
>
|
|
{props.Icon}
|
|
|
|
<span>
|
|
<Trans i18nKey={props.label} defaults={props.label} />
|
|
</span>
|
|
</Link>
|
|
}
|
|
/>
|
|
);
|
|
}
|
|
|
|
function SignOutDropdownItem(
|
|
props: React.PropsWithChildren<{
|
|
onSignOut: () => unknown;
|
|
}>,
|
|
) {
|
|
return (
|
|
<DropdownMenuItem
|
|
className={'flex h-12 w-full items-center space-x-2'}
|
|
onClick={props.onSignOut}
|
|
>
|
|
<LogOut className={'h-4'} />
|
|
|
|
<span>
|
|
<Trans i18nKey={'common.signOut'} />
|
|
</span>
|
|
</DropdownMenuItem>
|
|
);
|
|
}
|
|
|
|
function TeamAccountsModal(props: {
|
|
accounts: Accounts;
|
|
userId: string;
|
|
account: string;
|
|
}) {
|
|
const router = useRouter();
|
|
|
|
return (
|
|
<Dialog>
|
|
<DialogTrigger
|
|
render={
|
|
<DropdownMenuItem
|
|
className={'flex h-12 w-full items-center space-x-2'}
|
|
onSelect={(e) => e.preventDefault()}
|
|
>
|
|
<Home className={'h-4'} />
|
|
|
|
<span>
|
|
<Trans i18nKey={'common.yourAccounts'} />
|
|
</span>
|
|
</DropdownMenuItem>
|
|
}
|
|
/>
|
|
|
|
<DialogContent>
|
|
<DialogHeader>
|
|
<DialogTitle>
|
|
<Trans i18nKey={'common.yourAccounts'} />
|
|
</DialogTitle>
|
|
</DialogHeader>
|
|
|
|
<div className={'py-6'}>
|
|
<AccountSelector
|
|
className={'w-full max-w-full'}
|
|
userId={props.userId}
|
|
accounts={props.accounts}
|
|
features={features}
|
|
selectedAccount={props.account}
|
|
onAccountChange={(value) => {
|
|
if (value) {
|
|
document.cookie = `last-selected-team=${encodeURIComponent(value)}; path=/; max-age=${60 * 60 * 24 * 30}; SameSite=Lax`;
|
|
}
|
|
|
|
const path = value
|
|
? pathsConfig.app.accountHome.replace('[account]', value)
|
|
: pathsConfig.app.home;
|
|
|
|
router.replace(path);
|
|
}}
|
|
/>
|
|
</div>
|
|
</DialogContent>
|
|
</Dialog>
|
|
);
|
|
}
|