aboutsummaryrefslogtreecommitdiff
path: root/src/app/(main)/admin
diff options
context:
space:
mode:
authorFuwn <[email protected]>2026-01-24 13:09:50 +0000
committerFuwn <[email protected]>2026-01-24 13:09:50 +0000
commit396acf3bbbe00a192cb0ea0a9ccf91b1d8d2850b (patch)
treeb9df4ca6a70db45cfffbae6fdd7252e20fb8e93c /src/app/(main)/admin
downloadumami-396acf3bbbe00a192cb0ea0a9ccf91b1d8d2850b.tar.xz
umami-396acf3bbbe00a192cb0ea0a9ccf91b1d8d2850b.zip
Initial commitHEADmain
Created from https://vercel.com/new
Diffstat (limited to 'src/app/(main)/admin')
-rw-r--r--src/app/(main)/admin/AdminLayout.tsx33
-rw-r--r--src/app/(main)/admin/AdminNav.tsx48
-rw-r--r--src/app/(main)/admin/layout.tsx17
-rw-r--r--src/app/(main)/admin/teams/AdminTeamsDataTable.tsx19
-rw-r--r--src/app/(main)/admin/teams/AdminTeamsPage.tsx19
-rw-r--r--src/app/(main)/admin/teams/AdminTeamsTable.tsx86
-rw-r--r--src/app/(main)/admin/teams/[teamId]/AdminTeamPage.tsx11
-rw-r--r--src/app/(main)/admin/teams/[teamId]/page.tsx12
-rw-r--r--src/app/(main)/admin/teams/page.tsx9
-rw-r--r--src/app/(main)/admin/users/UserAddButton.tsx32
-rw-r--r--src/app/(main)/admin/users/UserAddForm.tsx71
-rw-r--r--src/app/(main)/admin/users/UserDeleteButton.tsx35
-rw-r--r--src/app/(main)/admin/users/UserDeleteForm.tsx41
-rw-r--r--src/app/(main)/admin/users/UsersDataTable.tsx14
-rw-r--r--src/app/(main)/admin/users/UsersPage.tsx24
-rw-r--r--src/app/(main)/admin/users/UsersTable.tsx84
-rw-r--r--src/app/(main)/admin/users/[userId]/UserEditForm.tsx73
-rw-r--r--src/app/(main)/admin/users/[userId]/UserHeader.tsx9
-rw-r--r--src/app/(main)/admin/users/[userId]/UserPage.tsx19
-rw-r--r--src/app/(main)/admin/users/[userId]/UserProvider.tsx20
-rw-r--r--src/app/(main)/admin/users/[userId]/UserSettings.tsx25
-rw-r--r--src/app/(main)/admin/users/[userId]/UserWebsites.tsx15
-rw-r--r--src/app/(main)/admin/users/[userId]/page.tsx12
-rw-r--r--src/app/(main)/admin/users/page.tsx9
-rw-r--r--src/app/(main)/admin/websites/AdminWebsitesDataTable.tsx13
-rw-r--r--src/app/(main)/admin/websites/AdminWebsitesPage.tsx19
-rw-r--r--src/app/(main)/admin/websites/AdminWebsitesTable.tsx89
-rw-r--r--src/app/(main)/admin/websites/[websiteId]/AdminWebsitePage.tsx14
-rw-r--r--src/app/(main)/admin/websites/[websiteId]/page.tsx12
-rw-r--r--src/app/(main)/admin/websites/page.tsx9
30 files changed, 893 insertions, 0 deletions
diff --git a/src/app/(main)/admin/AdminLayout.tsx b/src/app/(main)/admin/AdminLayout.tsx
new file mode 100644
index 0000000..3c8fa20
--- /dev/null
+++ b/src/app/(main)/admin/AdminLayout.tsx
@@ -0,0 +1,33 @@
+'use client';
+import { Column, Grid } from '@umami/react-zen';
+import type { ReactNode } from 'react';
+import { PageBody } from '@/components/common/PageBody';
+import { useLoginQuery } from '@/components/hooks';
+import { AdminNav } from './AdminNav';
+
+export function AdminLayout({ children }: { children: ReactNode }) {
+ const { user } = useLoginQuery();
+
+ if (!user.isAdmin || process.env.cloudMode) {
+ return null;
+ }
+
+ return (
+ <Grid columns={{ xs: '1fr', lg: 'auto 1fr' }} width="100%" height="100%">
+ <Column
+ display={{ xs: 'none', lg: 'flex' }}
+ width="240px"
+ height="100%"
+ border="right"
+ backgroundColor
+ marginRight="2"
+ padding="3"
+ >
+ <AdminNav />
+ </Column>
+ <Column gap="6" margin="2">
+ <PageBody>{children}</PageBody>
+ </Column>
+ </Grid>
+ );
+}
diff --git a/src/app/(main)/admin/AdminNav.tsx b/src/app/(main)/admin/AdminNav.tsx
new file mode 100644
index 0000000..20c0115
--- /dev/null
+++ b/src/app/(main)/admin/AdminNav.tsx
@@ -0,0 +1,48 @@
+import { SideMenu } from '@/components/common/SideMenu';
+import { useMessages, useNavigation } from '@/components/hooks';
+import { Globe, User, Users } from '@/components/icons';
+
+export function AdminNav({ onItemClick }: { onItemClick?: () => void }) {
+ const { formatMessage, labels } = useMessages();
+ const { pathname } = useNavigation();
+
+ const items = [
+ {
+ label: formatMessage(labels.manage),
+ items: [
+ {
+ id: 'users',
+ label: formatMessage(labels.users),
+ path: '/admin/users',
+ icon: <User />,
+ },
+ {
+ id: 'websites',
+ label: formatMessage(labels.websites),
+ path: '/admin/websites',
+ icon: <Globe />,
+ },
+ {
+ id: 'teams',
+ label: formatMessage(labels.teams),
+ path: '/admin/teams',
+ icon: <Users />,
+ },
+ ],
+ },
+ ];
+
+ const selectedKey = items
+ .flatMap(e => e.items)
+ ?.find(({ path }) => path && pathname.startsWith(path))?.id;
+
+ return (
+ <SideMenu
+ items={items}
+ title={formatMessage(labels.admin)}
+ selectedKey={selectedKey}
+ allowMinimize={false}
+ onItemClick={onItemClick}
+ />
+ );
+}
diff --git a/src/app/(main)/admin/layout.tsx b/src/app/(main)/admin/layout.tsx
new file mode 100644
index 0000000..34cdd0b
--- /dev/null
+++ b/src/app/(main)/admin/layout.tsx
@@ -0,0 +1,17 @@
+import type { Metadata } from 'next';
+import { AdminLayout } from './AdminLayout';
+
+export default function ({ children }) {
+ if (process.env.cloudMode) {
+ return null;
+ }
+
+ return <AdminLayout>{children}</AdminLayout>;
+}
+
+export const metadata: Metadata = {
+ title: {
+ template: '%s | Admin | Umami',
+ default: 'Admin | Umami',
+ },
+};
diff --git a/src/app/(main)/admin/teams/AdminTeamsDataTable.tsx b/src/app/(main)/admin/teams/AdminTeamsDataTable.tsx
new file mode 100644
index 0000000..7da8531
--- /dev/null
+++ b/src/app/(main)/admin/teams/AdminTeamsDataTable.tsx
@@ -0,0 +1,19 @@
+import type { ReactNode } from 'react';
+import { DataGrid } from '@/components/common/DataGrid';
+import { useTeamsQuery } from '@/components/hooks';
+import { AdminTeamsTable } from './AdminTeamsTable';
+
+export function AdminTeamsDataTable({
+ showActions,
+}: {
+ showActions?: boolean;
+ children?: ReactNode;
+}) {
+ const queryResult = useTeamsQuery();
+
+ return (
+ <DataGrid query={queryResult} allowSearch={true}>
+ {({ data }) => <AdminTeamsTable data={data} showActions={showActions} />}
+ </DataGrid>
+ );
+}
diff --git a/src/app/(main)/admin/teams/AdminTeamsPage.tsx b/src/app/(main)/admin/teams/AdminTeamsPage.tsx
new file mode 100644
index 0000000..41e6f4a
--- /dev/null
+++ b/src/app/(main)/admin/teams/AdminTeamsPage.tsx
@@ -0,0 +1,19 @@
+'use client';
+import { Column } from '@umami/react-zen';
+import { PageHeader } from '@/components/common/PageHeader';
+import { Panel } from '@/components/common/Panel';
+import { useMessages } from '@/components/hooks';
+import { AdminTeamsDataTable } from './AdminTeamsDataTable';
+
+export function AdminTeamsPage() {
+ const { formatMessage, labels } = useMessages();
+
+ return (
+ <Column gap="6" margin="2">
+ <PageHeader title={formatMessage(labels.teams)} />
+ <Panel>
+ <AdminTeamsDataTable />
+ </Panel>
+ </Column>
+ );
+}
diff --git a/src/app/(main)/admin/teams/AdminTeamsTable.tsx b/src/app/(main)/admin/teams/AdminTeamsTable.tsx
new file mode 100644
index 0000000..9f2abd5
--- /dev/null
+++ b/src/app/(main)/admin/teams/AdminTeamsTable.tsx
@@ -0,0 +1,86 @@
+import { DataColumn, DataTable, Dialog, Icon, MenuItem, Modal, Row, Text } from '@umami/react-zen';
+import Link from 'next/link';
+import { useState } from 'react';
+import { DateDistance } from '@/components/common/DateDistance';
+import { useMessages } from '@/components/hooks';
+import { Edit, Trash } from '@/components/icons';
+import { MenuButton } from '@/components/input/MenuButton';
+import { TeamDeleteForm } from '../../teams/[teamId]/TeamDeleteForm';
+
+export function AdminTeamsTable({
+ data = [],
+ showActions = true,
+}: {
+ data: any[];
+ showActions?: boolean;
+}) {
+ const { formatMessage, labels } = useMessages();
+ const [deleteTeam, setDeleteTeam] = useState(null);
+
+ return (
+ <>
+ <DataTable data={data}>
+ <DataColumn id="name" label={formatMessage(labels.name)} width="1fr">
+ {(row: any) => <Link href={`/admin/teams/${row.id}`}>{row.name}</Link>}
+ </DataColumn>
+ <DataColumn id="websites" label={formatMessage(labels.members)} width="140px">
+ {(row: any) => row?._count?.members}
+ </DataColumn>
+ <DataColumn id="members" label={formatMessage(labels.websites)} width="140px">
+ {(row: any) => row?._count?.websites}
+ </DataColumn>
+ <DataColumn id="owner" label={formatMessage(labels.owner)}>
+ {(row: any) => {
+ const name = row?.members?.[0]?.user?.username;
+
+ return (
+ <Text title={name} truncate>
+ <Link href={`/admin/users/${row?.members?.[0]?.user?.id}`}>{name}</Link>
+ </Text>
+ );
+ }}
+ </DataColumn>
+ <DataColumn id="created" label={formatMessage(labels.created)} width="160px">
+ {(row: any) => <DateDistance date={new Date(row.createdAt)} />}
+ </DataColumn>
+ {showActions && (
+ <DataColumn id="action" align="end" width="50px">
+ {(row: any) => {
+ const { id } = row;
+
+ return (
+ <MenuButton>
+ <MenuItem href={`/admin/teams/${id}`} data-test="link-button-edit">
+ <Row alignItems="center" gap>
+ <Icon>
+ <Edit />
+ </Icon>
+ <Text>{formatMessage(labels.edit)}</Text>
+ </Row>
+ </MenuItem>
+ <MenuItem
+ id="delete"
+ onAction={() => setDeleteTeam(id)}
+ data-test="link-button-delete"
+ >
+ <Row alignItems="center" gap>
+ <Icon>
+ <Trash />
+ </Icon>
+ <Text>{formatMessage(labels.delete)}</Text>
+ </Row>
+ </MenuItem>
+ </MenuButton>
+ );
+ }}
+ </DataColumn>
+ )}
+ </DataTable>
+ <Modal isOpen={!!deleteTeam}>
+ <Dialog style={{ width: 400 }}>
+ <TeamDeleteForm teamId={deleteTeam} onClose={() => setDeleteTeam(null)} />
+ </Dialog>
+ </Modal>
+ </>
+ );
+}
diff --git a/src/app/(main)/admin/teams/[teamId]/AdminTeamPage.tsx b/src/app/(main)/admin/teams/[teamId]/AdminTeamPage.tsx
new file mode 100644
index 0000000..2150197
--- /dev/null
+++ b/src/app/(main)/admin/teams/[teamId]/AdminTeamPage.tsx
@@ -0,0 +1,11 @@
+'use client';
+import { TeamSettings } from '@/app/(main)/teams/[teamId]/TeamSettings';
+import { TeamProvider } from '@/app/(main)/teams/TeamProvider';
+
+export function AdminTeamPage({ teamId }: { teamId: string }) {
+ return (
+ <TeamProvider teamId={teamId}>
+ <TeamSettings teamId={teamId} />
+ </TeamProvider>
+ );
+}
diff --git a/src/app/(main)/admin/teams/[teamId]/page.tsx b/src/app/(main)/admin/teams/[teamId]/page.tsx
new file mode 100644
index 0000000..104766a
--- /dev/null
+++ b/src/app/(main)/admin/teams/[teamId]/page.tsx
@@ -0,0 +1,12 @@
+import type { Metadata } from 'next';
+import { AdminTeamPage } from './AdminTeamPage';
+
+export default async function ({ params }: { params: Promise<{ teamId: string }> }) {
+ const { teamId } = await params;
+
+ return <AdminTeamPage teamId={teamId} />;
+}
+
+export const metadata: Metadata = {
+ title: 'Team',
+};
diff --git a/src/app/(main)/admin/teams/page.tsx b/src/app/(main)/admin/teams/page.tsx
new file mode 100644
index 0000000..987f02b
--- /dev/null
+++ b/src/app/(main)/admin/teams/page.tsx
@@ -0,0 +1,9 @@
+import type { Metadata } from 'next';
+import { AdminTeamsPage } from './AdminTeamsPage';
+
+export default function () {
+ return <AdminTeamsPage />;
+}
+export const metadata: Metadata = {
+ title: 'Teams',
+};
diff --git a/src/app/(main)/admin/users/UserAddButton.tsx b/src/app/(main)/admin/users/UserAddButton.tsx
new file mode 100644
index 0000000..0525082
--- /dev/null
+++ b/src/app/(main)/admin/users/UserAddButton.tsx
@@ -0,0 +1,32 @@
+import { Button, Dialog, DialogTrigger, Icon, Modal, Text, useToast } from '@umami/react-zen';
+import { useMessages, useModified } from '@/components/hooks';
+import { Plus } from '@/components/icons';
+import { UserAddForm } from './UserAddForm';
+
+export function UserAddButton({ onSave }: { onSave?: () => void }) {
+ const { formatMessage, labels, messages } = useMessages();
+ const { toast } = useToast();
+ const { touch } = useModified();
+
+ const handleSave = () => {
+ toast(formatMessage(messages.saved));
+ touch('users');
+ onSave?.();
+ };
+
+ return (
+ <DialogTrigger>
+ <Button variant="primary" data-test="button-create-user">
+ <Icon>
+ <Plus />
+ </Icon>
+ <Text>{formatMessage(labels.createUser)}</Text>
+ </Button>
+ <Modal>
+ <Dialog title={formatMessage(labels.createUser)} style={{ width: 400 }}>
+ {({ close }) => <UserAddForm onSave={handleSave} onClose={close} />}
+ </Dialog>
+ </Modal>
+ </DialogTrigger>
+ );
+}
diff --git a/src/app/(main)/admin/users/UserAddForm.tsx b/src/app/(main)/admin/users/UserAddForm.tsx
new file mode 100644
index 0000000..6c36551
--- /dev/null
+++ b/src/app/(main)/admin/users/UserAddForm.tsx
@@ -0,0 +1,71 @@
+import {
+ Button,
+ Form,
+ FormButtons,
+ FormField,
+ FormSubmitButton,
+ ListItem,
+ PasswordField,
+ Select,
+ TextField,
+} from '@umami/react-zen';
+import { useMessages, useUpdateQuery } from '@/components/hooks';
+import { ROLES } from '@/lib/constants';
+
+export function UserAddForm({ onSave, onClose }) {
+ const { mutateAsync, error, isPending } = useUpdateQuery(`/users`);
+ const { formatMessage, labels, getErrorMessage } = useMessages();
+
+ const handleSubmit = async (data: any) => {
+ await mutateAsync(data, {
+ onSuccess: async () => {
+ onSave(data);
+ onClose();
+ },
+ });
+ };
+
+ return (
+ <Form onSubmit={handleSubmit} error={getErrorMessage(error)}>
+ <FormField
+ label={formatMessage(labels.username)}
+ name="username"
+ rules={{ required: formatMessage(labels.required) }}
+ >
+ <TextField autoComplete="new-username" data-test="input-username" />
+ </FormField>
+ <FormField
+ label={formatMessage(labels.password)}
+ name="password"
+ rules={{ required: formatMessage(labels.required) }}
+ >
+ <PasswordField autoComplete="new-password" data-test="input-password" />
+ </FormField>
+ <FormField
+ label={formatMessage(labels.role)}
+ name="role"
+ rules={{ required: formatMessage(labels.required) }}
+ >
+ <Select>
+ <ListItem id={ROLES.viewOnly} data-test="dropdown-item-viewOnly">
+ {formatMessage(labels.viewOnly)}
+ </ListItem>
+ <ListItem id={ROLES.user} data-test="dropdown-item-user">
+ {formatMessage(labels.user)}
+ </ListItem>
+ <ListItem id={ROLES.admin} data-test="dropdown-item-admin">
+ {formatMessage(labels.admin)}
+ </ListItem>
+ </Select>
+ </FormField>
+ <FormButtons>
+ <Button isDisabled={isPending} onPress={onClose}>
+ {formatMessage(labels.cancel)}
+ </Button>
+ <FormSubmitButton variant="primary" data-test="button-submit" isDisabled={false}>
+ {formatMessage(labels.save)}
+ </FormSubmitButton>
+ </FormButtons>
+ </Form>
+ );
+}
diff --git a/src/app/(main)/admin/users/UserDeleteButton.tsx b/src/app/(main)/admin/users/UserDeleteButton.tsx
new file mode 100644
index 0000000..ee8f2c1
--- /dev/null
+++ b/src/app/(main)/admin/users/UserDeleteButton.tsx
@@ -0,0 +1,35 @@
+import { Button, Dialog, DialogTrigger, Icon, Modal, Text } from '@umami/react-zen';
+import { useLoginQuery, useMessages } from '@/components/hooks';
+import { Trash } from '@/components/icons';
+import { UserDeleteForm } from './UserDeleteForm';
+
+export function UserDeleteButton({
+ userId,
+ username,
+ onDelete,
+}: {
+ userId: string;
+ username: string;
+ onDelete?: () => void;
+}) {
+ const { formatMessage, labels } = useMessages();
+ const { user } = useLoginQuery();
+
+ return (
+ <DialogTrigger>
+ <Button isDisabled={userId === user?.id} data-test="button-delete">
+ <Icon size="sm">
+ <Trash />
+ </Icon>
+ <Text>{formatMessage(labels.delete)}</Text>
+ </Button>
+ <Modal>
+ <Dialog title={formatMessage(labels.deleteUser)} style={{ width: 400 }}>
+ {({ close }) => (
+ <UserDeleteForm userId={userId} username={username} onSave={onDelete} onClose={close} />
+ )}
+ </Dialog>
+ </Modal>
+ </DialogTrigger>
+ );
+}
diff --git a/src/app/(main)/admin/users/UserDeleteForm.tsx b/src/app/(main)/admin/users/UserDeleteForm.tsx
new file mode 100644
index 0000000..8f6fd50
--- /dev/null
+++ b/src/app/(main)/admin/users/UserDeleteForm.tsx
@@ -0,0 +1,41 @@
+import { AlertDialog, Row } from '@umami/react-zen';
+import { useDeleteQuery, useMessages, useModified } from '@/components/hooks';
+
+export function UserDeleteForm({
+ userId,
+ username,
+ onSave,
+ onClose,
+}: {
+ userId: string;
+ username: string;
+ onSave?: () => void;
+ onClose?: () => void;
+}) {
+ const { messages, labels, formatMessage } = useMessages();
+ const { mutateAsync } = useDeleteQuery(`/users/${userId}`);
+ const { touch } = useModified();
+
+ const handleConfirm = async () => {
+ await mutateAsync(null, {
+ onSuccess: async () => {
+ touch('users');
+ touch(`users:${userId}`);
+ onSave?.();
+ onClose?.();
+ },
+ });
+ };
+
+ return (
+ <AlertDialog
+ title={formatMessage(labels.delete)}
+ onConfirm={handleConfirm}
+ onCancel={onClose}
+ confirmLabel={formatMessage(labels.delete)}
+ isDanger
+ >
+ <Row gap="1">{formatMessage(messages.confirmDelete, { target: username })}</Row>
+ </AlertDialog>
+ );
+}
diff --git a/src/app/(main)/admin/users/UsersDataTable.tsx b/src/app/(main)/admin/users/UsersDataTable.tsx
new file mode 100644
index 0000000..8467bd2
--- /dev/null
+++ b/src/app/(main)/admin/users/UsersDataTable.tsx
@@ -0,0 +1,14 @@
+import type { ReactNode } from 'react';
+import { DataGrid } from '@/components/common/DataGrid';
+import { useUsersQuery } from '@/components/hooks';
+import { UsersTable } from './UsersTable';
+
+export function UsersDataTable({ showActions }: { showActions?: boolean; children?: ReactNode }) {
+ const queryResult = useUsersQuery();
+
+ return (
+ <DataGrid query={queryResult} allowSearch={true}>
+ {({ data }) => <UsersTable data={data} showActions={showActions} />}
+ </DataGrid>
+ );
+}
diff --git a/src/app/(main)/admin/users/UsersPage.tsx b/src/app/(main)/admin/users/UsersPage.tsx
new file mode 100644
index 0000000..7e1b0f4
--- /dev/null
+++ b/src/app/(main)/admin/users/UsersPage.tsx
@@ -0,0 +1,24 @@
+'use client';
+import { Column } from '@umami/react-zen';
+import { PageHeader } from '@/components/common/PageHeader';
+import { Panel } from '@/components/common/Panel';
+import { useMessages } from '@/components/hooks';
+import { UserAddButton } from './UserAddButton';
+import { UsersDataTable } from './UsersDataTable';
+
+export function UsersPage() {
+ const { formatMessage, labels } = useMessages();
+
+ const handleSave = () => {};
+
+ return (
+ <Column gap="6" margin="2">
+ <PageHeader title={formatMessage(labels.users)}>
+ <UserAddButton onSave={handleSave} />
+ </PageHeader>
+ <Panel>
+ <UsersDataTable />
+ </Panel>
+ </Column>
+ );
+}
diff --git a/src/app/(main)/admin/users/UsersTable.tsx b/src/app/(main)/admin/users/UsersTable.tsx
new file mode 100644
index 0000000..9c10f3e
--- /dev/null
+++ b/src/app/(main)/admin/users/UsersTable.tsx
@@ -0,0 +1,84 @@
+import { DataColumn, DataTable, Icon, MenuItem, Modal, Row, Text } from '@umami/react-zen';
+import Link from 'next/link';
+import { useState } from 'react';
+import { DateDistance } from '@/components/common/DateDistance';
+import { useMessages } from '@/components/hooks';
+import { Edit, Trash } from '@/components/icons';
+import { MenuButton } from '@/components/input/MenuButton';
+import { ROLES } from '@/lib/constants';
+import { UserDeleteForm } from './UserDeleteForm';
+
+export function UsersTable({
+ data = [],
+ showActions = true,
+}: {
+ data: any[];
+ showActions?: boolean;
+}) {
+ const { formatMessage, labels } = useMessages();
+ const [deleteUser, setDeleteUser] = useState(null);
+
+ return (
+ <>
+ <DataTable data={data}>
+ <DataColumn id="username" label={formatMessage(labels.username)} width="2fr">
+ {(row: any) => <Link href={`/admin/users/${row.id}`}>{row.username}</Link>}
+ </DataColumn>
+ <DataColumn id="role" label={formatMessage(labels.role)}>
+ {(row: any) =>
+ formatMessage(
+ labels[Object.keys(ROLES).find(key => ROLES[key] === row.role)] || labels.unknown,
+ )
+ }
+ </DataColumn>
+ <DataColumn id="websites" label={formatMessage(labels.websites)}>
+ {(row: any) => row._count.websites}
+ </DataColumn>
+ <DataColumn id="created" label={formatMessage(labels.created)}>
+ {(row: any) => <DateDistance date={new Date(row.createdAt)} />}
+ </DataColumn>
+ {showActions && (
+ <DataColumn id="action" align="end" width="100px">
+ {(row: any) => {
+ const { id } = row;
+
+ return (
+ <MenuButton>
+ <MenuItem href={`/admin/users/${id}`} data-test="link-button-edit">
+ <Row alignItems="center" gap>
+ <Icon>
+ <Edit />
+ </Icon>
+ <Text>{formatMessage(labels.edit)}</Text>
+ </Row>
+ </MenuItem>
+ <MenuItem
+ id="delete"
+ onAction={() => setDeleteUser(row)}
+ data-test="link-button-delete"
+ >
+ <Row alignItems="center" gap>
+ <Icon>
+ <Trash />
+ </Icon>
+ <Text>{formatMessage(labels.delete)}</Text>
+ </Row>
+ </MenuItem>
+ </MenuButton>
+ );
+ }}
+ </DataColumn>
+ )}
+ </DataTable>
+ <Modal isOpen={!!deleteUser}>
+ <UserDeleteForm
+ userId={deleteUser?.id}
+ username={deleteUser?.username}
+ onClose={() => {
+ setDeleteUser(null);
+ }}
+ />
+ </Modal>
+ </>
+ );
+}
diff --git a/src/app/(main)/admin/users/[userId]/UserEditForm.tsx b/src/app/(main)/admin/users/[userId]/UserEditForm.tsx
new file mode 100644
index 0000000..28bf030
--- /dev/null
+++ b/src/app/(main)/admin/users/[userId]/UserEditForm.tsx
@@ -0,0 +1,73 @@
+import {
+ Form,
+ FormButtons,
+ FormField,
+ FormSubmitButton,
+ ListItem,
+ PasswordField,
+ Select,
+ TextField,
+} from '@umami/react-zen';
+import { useLoginQuery, useMessages, useUpdateQuery, useUser } from '@/components/hooks';
+import { ROLES } from '@/lib/constants';
+
+export function UserEditForm({ userId, onSave }: { userId: string; onSave?: () => void }) {
+ const { formatMessage, labels, messages, getMessage } = useMessages();
+ const user = useUser();
+ const { user: login } = useLoginQuery();
+
+ const { mutateAsync, error, toast, touch } = useUpdateQuery(`/users/${userId}`);
+
+ const handleSubmit = async (data: any) => {
+ await mutateAsync(data, {
+ onSuccess: async () => {
+ toast(formatMessage(messages.saved));
+ touch('users');
+ touch(`user:${user.id}`);
+ onSave?.();
+ },
+ });
+ };
+
+ return (
+ <Form onSubmit={handleSubmit} error={getMessage(error?.code)} values={user}>
+ <FormField name="username" label={formatMessage(labels.username)}>
+ <TextField data-test="input-username" />
+ </FormField>
+ <FormField
+ name="password"
+ label={formatMessage(labels.password)}
+ rules={{
+ minLength: { value: 8, message: formatMessage(messages.minPasswordLength, { n: '8' }) },
+ }}
+ >
+ <PasswordField autoComplete="new-password" data-test="input-password" />
+ </FormField>
+
+ {user.id !== login.id && (
+ <FormField
+ name="role"
+ label={formatMessage(labels.role)}
+ rules={{ required: formatMessage(labels.required) }}
+ >
+ <Select defaultValue={user.role}>
+ <ListItem id={ROLES.viewOnly} data-test="dropdown-item-viewOnly">
+ {formatMessage(labels.viewOnly)}
+ </ListItem>
+ <ListItem id={ROLES.user} data-test="dropdown-item-user">
+ {formatMessage(labels.user)}
+ </ListItem>
+ <ListItem id={ROLES.admin} data-test="dropdown-item-admin">
+ {formatMessage(labels.admin)}
+ </ListItem>
+ </Select>
+ </FormField>
+ )}
+ <FormButtons>
+ <FormSubmitButton data-test="button-submit" variant="primary">
+ {formatMessage(labels.save)}
+ </FormSubmitButton>
+ </FormButtons>
+ </Form>
+ );
+}
diff --git a/src/app/(main)/admin/users/[userId]/UserHeader.tsx b/src/app/(main)/admin/users/[userId]/UserHeader.tsx
new file mode 100644
index 0000000..1f82897
--- /dev/null
+++ b/src/app/(main)/admin/users/[userId]/UserHeader.tsx
@@ -0,0 +1,9 @@
+import { PageHeader } from '@/components/common/PageHeader';
+import { useUser } from '@/components/hooks';
+import { User } from '@/components/icons';
+
+export function UserHeader() {
+ const user = useUser();
+
+ return <PageHeader title={user?.username} icon={<User />} />;
+}
diff --git a/src/app/(main)/admin/users/[userId]/UserPage.tsx b/src/app/(main)/admin/users/[userId]/UserPage.tsx
new file mode 100644
index 0000000..5e0f8d1
--- /dev/null
+++ b/src/app/(main)/admin/users/[userId]/UserPage.tsx
@@ -0,0 +1,19 @@
+'use client';
+import { Column } from '@umami/react-zen';
+import { UserHeader } from '@/app/(main)/admin/users/[userId]/UserHeader';
+import { Panel } from '@/components/common/Panel';
+import { UserProvider } from './UserProvider';
+import { UserSettings } from './UserSettings';
+
+export function UserPage({ userId }: { userId: string }) {
+ return (
+ <UserProvider userId={userId}>
+ <Column gap="6">
+ <UserHeader />
+ <Panel>
+ <UserSettings userId={userId} />
+ </Panel>
+ </Column>
+ </UserProvider>
+ );
+}
diff --git a/src/app/(main)/admin/users/[userId]/UserProvider.tsx b/src/app/(main)/admin/users/[userId]/UserProvider.tsx
new file mode 100644
index 0000000..ea01915
--- /dev/null
+++ b/src/app/(main)/admin/users/[userId]/UserProvider.tsx
@@ -0,0 +1,20 @@
+import { Loading } from '@umami/react-zen';
+import { createContext, type ReactNode } from 'react';
+import { useUserQuery } from '@/components/hooks/queries/useUserQuery';
+import type { User } from '@/generated/prisma/client';
+
+export const UserContext = createContext<User>(null);
+
+export function UserProvider({ userId, children }: { userId: string; children: ReactNode }) {
+ const { data: user, isFetching, isLoading } = useUserQuery(userId);
+
+ if (isFetching && isLoading) {
+ return <Loading placement="absolute" />;
+ }
+
+ if (!user) {
+ return null;
+ }
+
+ return <UserContext.Provider value={user}>{children}</UserContext.Provider>;
+}
diff --git a/src/app/(main)/admin/users/[userId]/UserSettings.tsx b/src/app/(main)/admin/users/[userId]/UserSettings.tsx
new file mode 100644
index 0000000..3f17f3e
--- /dev/null
+++ b/src/app/(main)/admin/users/[userId]/UserSettings.tsx
@@ -0,0 +1,25 @@
+import { Column, Tab, TabList, TabPanel, Tabs } from '@umami/react-zen';
+import { useMessages } from '@/components/hooks';
+import { UserEditForm } from './UserEditForm';
+import { UserWebsites } from './UserWebsites';
+
+export function UserSettings({ userId }: { userId: string }) {
+ const { formatMessage, labels } = useMessages();
+
+ return (
+ <Column gap="6">
+ <Tabs>
+ <TabList>
+ <Tab id="details">{formatMessage(labels.details)}</Tab>
+ <Tab id="websites">{formatMessage(labels.websites)}</Tab>
+ </TabList>
+ <TabPanel id="details" style={{ width: 500 }}>
+ <UserEditForm userId={userId} />
+ </TabPanel>
+ <TabPanel id="websites">
+ <UserWebsites userId={userId} />
+ </TabPanel>
+ </Tabs>
+ </Column>
+ );
+}
diff --git a/src/app/(main)/admin/users/[userId]/UserWebsites.tsx b/src/app/(main)/admin/users/[userId]/UserWebsites.tsx
new file mode 100644
index 0000000..eeb173e
--- /dev/null
+++ b/src/app/(main)/admin/users/[userId]/UserWebsites.tsx
@@ -0,0 +1,15 @@
+import { WebsitesTable } from '@/app/(main)/websites/WebsitesTable';
+import { DataGrid } from '@/components/common/DataGrid';
+import { useUserWebsitesQuery } from '@/components/hooks';
+
+export function UserWebsites({ userId }) {
+ const queryResult = useUserWebsitesQuery({ userId });
+
+ return (
+ <DataGrid query={queryResult}>
+ {({ data }) => (
+ <WebsitesTable data={data} showActions={true} allowEdit={true} allowView={true} />
+ )}
+ </DataGrid>
+ );
+}
diff --git a/src/app/(main)/admin/users/[userId]/page.tsx b/src/app/(main)/admin/users/[userId]/page.tsx
new file mode 100644
index 0000000..16c9f36
--- /dev/null
+++ b/src/app/(main)/admin/users/[userId]/page.tsx
@@ -0,0 +1,12 @@
+import type { Metadata } from 'next';
+import { UserPage } from './UserPage';
+
+export default async function ({ params }: { params: Promise<{ userId: string }> }) {
+ const { userId } = await params;
+
+ return <UserPage userId={userId} />;
+}
+
+export const metadata: Metadata = {
+ title: 'User',
+};
diff --git a/src/app/(main)/admin/users/page.tsx b/src/app/(main)/admin/users/page.tsx
new file mode 100644
index 0000000..96e69eb
--- /dev/null
+++ b/src/app/(main)/admin/users/page.tsx
@@ -0,0 +1,9 @@
+import type { Metadata } from 'next';
+import { UsersPage } from './UsersPage';
+
+export default function () {
+ return <UsersPage />;
+}
+export const metadata: Metadata = {
+ title: 'Users',
+};
diff --git a/src/app/(main)/admin/websites/AdminWebsitesDataTable.tsx b/src/app/(main)/admin/websites/AdminWebsitesDataTable.tsx
new file mode 100644
index 0000000..2105992
--- /dev/null
+++ b/src/app/(main)/admin/websites/AdminWebsitesDataTable.tsx
@@ -0,0 +1,13 @@
+import { DataGrid } from '@/components/common/DataGrid';
+import { useWebsitesQuery } from '@/components/hooks';
+import { AdminWebsitesTable } from './AdminWebsitesTable';
+
+export function AdminWebsitesDataTable() {
+ const query = useWebsitesQuery();
+
+ return (
+ <DataGrid query={query} allowSearch={true}>
+ {props => <AdminWebsitesTable {...props} />}
+ </DataGrid>
+ );
+}
diff --git a/src/app/(main)/admin/websites/AdminWebsitesPage.tsx b/src/app/(main)/admin/websites/AdminWebsitesPage.tsx
new file mode 100644
index 0000000..1c2ac92
--- /dev/null
+++ b/src/app/(main)/admin/websites/AdminWebsitesPage.tsx
@@ -0,0 +1,19 @@
+'use client';
+import { Column } from '@umami/react-zen';
+import { PageHeader } from '@/components/common/PageHeader';
+import { Panel } from '@/components/common/Panel';
+import { useMessages } from '@/components/hooks';
+import { AdminWebsitesDataTable } from './AdminWebsitesDataTable';
+
+export function AdminWebsitesPage() {
+ const { formatMessage, labels } = useMessages();
+
+ return (
+ <Column gap="6" margin="2">
+ <PageHeader title={formatMessage(labels.websites)} />
+ <Panel>
+ <AdminWebsitesDataTable />
+ </Panel>
+ </Column>
+ );
+}
diff --git a/src/app/(main)/admin/websites/AdminWebsitesTable.tsx b/src/app/(main)/admin/websites/AdminWebsitesTable.tsx
new file mode 100644
index 0000000..cfda595
--- /dev/null
+++ b/src/app/(main)/admin/websites/AdminWebsitesTable.tsx
@@ -0,0 +1,89 @@
+import { DataColumn, DataTable, Dialog, Icon, MenuItem, Modal, Row, Text } from '@umami/react-zen';
+import Link from 'next/link';
+import { useState } from 'react';
+import { WebsiteDeleteForm } from '@/app/(main)/websites/[websiteId]/settings/WebsiteDeleteForm';
+import { DateDistance } from '@/components/common/DateDistance';
+import { useMessages } from '@/components/hooks';
+import { Edit, Trash, Users } from '@/components/icons';
+import { MenuButton } from '@/components/input/MenuButton';
+
+export function AdminWebsitesTable({ data = [] }: { data: any[] }) {
+ const { formatMessage, labels } = useMessages();
+ const [deleteWebsite, setDeleteWebsite] = useState(null);
+
+ return (
+ <>
+ <DataTable data={data}>
+ <DataColumn id="name" label={formatMessage(labels.name)}>
+ {(row: any) => (
+ <Text truncate>
+ <Link href={`/admin/websites/${row.id}`}>{row.name}</Link>
+ </Text>
+ )}
+ </DataColumn>
+ <DataColumn id="domain" label={formatMessage(labels.domain)}>
+ {(row: any) => <Text truncate>{row.domain}</Text>}
+ </DataColumn>
+ <DataColumn id="owner" label={formatMessage(labels.owner)}>
+ {(row: any) => {
+ if (row?.team) {
+ return (
+ <Row alignItems="center" gap>
+ <Icon>
+ <Users />
+ </Icon>
+ <Text truncate>
+ <Link href={`/admin/teams/${row?.team?.id}`}>{row?.team?.name}</Link>
+ </Text>
+ </Row>
+ );
+ }
+ return (
+ <Text truncate>
+ <Link href={`/admin/users/${row?.user?.id}`}>{row?.user?.username}</Link>
+ </Text>
+ );
+ }}
+ </DataColumn>
+ <DataColumn id="created" label={formatMessage(labels.created)} width="180px">
+ {(row: any) => <DateDistance date={new Date(row.createdAt)} />}
+ </DataColumn>
+ <DataColumn id="action" align="end" width="50px">
+ {(row: any) => {
+ const { id } = row;
+
+ return (
+ <MenuButton>
+ <MenuItem href={`/admin/websites/${id}`} data-test="link-button-edit">
+ <Row alignItems="center" gap>
+ <Icon>
+ <Edit />
+ </Icon>
+ <Text>{formatMessage(labels.edit)}</Text>
+ </Row>
+ </MenuItem>
+ <MenuItem
+ id="delete"
+ onAction={() => setDeleteWebsite(id)}
+ data-test="link-button-delete"
+ >
+ <Row alignItems="center" gap>
+ <Icon>
+ <Trash />
+ </Icon>
+ <Text>{formatMessage(labels.delete)}</Text>
+ </Row>
+ </MenuItem>
+ </MenuButton>
+ );
+ }}
+ </DataColumn>
+ </DataTable>
+ <Modal isOpen={!!deleteWebsite}>
+ <Dialog style={{ width: 400 }}>
+ <WebsiteDeleteForm websiteId={deleteWebsite} onClose={() => setDeleteWebsite(null)} />
+ </Dialog>
+ </Modal>
+ </>
+ );
+}
diff --git a/src/app/(main)/admin/websites/[websiteId]/AdminWebsitePage.tsx b/src/app/(main)/admin/websites/[websiteId]/AdminWebsitePage.tsx
new file mode 100644
index 0000000..5da82af
--- /dev/null
+++ b/src/app/(main)/admin/websites/[websiteId]/AdminWebsitePage.tsx
@@ -0,0 +1,14 @@
+'use client';
+import { WebsiteSettings } from '@/app/(main)/websites/[websiteId]/settings/WebsiteSettings';
+import { WebsiteProvider } from '@/app/(main)/websites/WebsiteProvider';
+import { Panel } from '@/components/common/Panel';
+
+export function AdminWebsitePage({ websiteId }: { websiteId: string }) {
+ return (
+ <WebsiteProvider websiteId={websiteId}>
+ <Panel>
+ <WebsiteSettings websiteId={websiteId} />
+ </Panel>
+ </WebsiteProvider>
+ );
+}
diff --git a/src/app/(main)/admin/websites/[websiteId]/page.tsx b/src/app/(main)/admin/websites/[websiteId]/page.tsx
new file mode 100644
index 0000000..557adbd
--- /dev/null
+++ b/src/app/(main)/admin/websites/[websiteId]/page.tsx
@@ -0,0 +1,12 @@
+import type { Metadata } from 'next';
+import { WebsiteSettingsPage } from '@/app/(main)/settings/websites/[websiteId]/WebsiteSettingsPage';
+
+export default async function ({ params }: { params: Promise<{ websiteId: string }> }) {
+ const { websiteId } = await params;
+
+ return <WebsiteSettingsPage websiteId={websiteId} />;
+}
+
+export const metadata: Metadata = {
+ title: 'Website',
+};
diff --git a/src/app/(main)/admin/websites/page.tsx b/src/app/(main)/admin/websites/page.tsx
new file mode 100644
index 0000000..d6da9f6
--- /dev/null
+++ b/src/app/(main)/admin/websites/page.tsx
@@ -0,0 +1,9 @@
+import type { Metadata } from 'next';
+import { AdminWebsitesPage } from './AdminWebsitesPage';
+
+export default function () {
+ return <AdminWebsitesPage />;
+}
+export const metadata: Metadata = {
+ title: 'Websites',
+};