mirror of
https://github.com/documenso/documenso.git
synced 2025-11-13 00:03:33 +10:00
feat: avatar images
This commit is contained in:
@ -3,6 +3,7 @@ import type { Metadata } from 'next';
|
|||||||
import { getRequiredServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
import { getRequiredServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
||||||
|
|
||||||
import { SettingsHeader } from '~/components/(dashboard)/settings/layout/header';
|
import { SettingsHeader } from '~/components/(dashboard)/settings/layout/header';
|
||||||
|
import { AvatarImageForm } from '~/components/forms/avatar-image';
|
||||||
import { ProfileForm } from '~/components/forms/profile';
|
import { ProfileForm } from '~/components/forms/profile';
|
||||||
|
|
||||||
import { DeleteAccountDialog } from './delete-account-dialog';
|
import { DeleteAccountDialog } from './delete-account-dialog';
|
||||||
@ -18,6 +19,7 @@ export default async function ProfileSettingsPage() {
|
|||||||
<div>
|
<div>
|
||||||
<SettingsHeader title="Profile" subtitle="Here you can edit your personal details." />
|
<SettingsHeader title="Profile" subtitle="Here you can edit your personal details." />
|
||||||
|
|
||||||
|
<AvatarImageForm className="mb-8 max-w-xl" user={user} />
|
||||||
<ProfileForm className="mb-8 max-w-xl" user={user} />
|
<ProfileForm className="mb-8 max-w-xl" user={user} />
|
||||||
|
|
||||||
<hr className="my-4 max-w-xl" />
|
<hr className="my-4 max-w-xl" />
|
||||||
|
|||||||
@ -1,7 +1,6 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
import { getServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
import { getServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
||||||
import type { GetTeamsResponse } from '@documenso/lib/server-only/team/get-teams';
|
|
||||||
import { getTeams } from '@documenso/lib/server-only/team/get-teams';
|
import { getTeams } from '@documenso/lib/server-only/team/get-teams';
|
||||||
|
|
||||||
import { RefreshOnFocus } from '~/components/(dashboard)/refresh-on-focus/refresh-on-focus';
|
import { RefreshOnFocus } from '~/components/(dashboard)/refresh-on-focus/refresh-on-focus';
|
||||||
@ -16,18 +15,15 @@ type PublicProfileLayoutProps = {
|
|||||||
export default async function PublicProfileLayout({ children }: PublicProfileLayoutProps) {
|
export default async function PublicProfileLayout({ children }: PublicProfileLayoutProps) {
|
||||||
const { user, session } = await getServerComponentSession();
|
const { user, session } = await getServerComponentSession();
|
||||||
|
|
||||||
let teams: GetTeamsResponse = [];
|
// I wouldn't typically do this but it's better than the `let` statement
|
||||||
|
const teams = user && session ? await getTeams({ userId: user.id }) : undefined;
|
||||||
if (user && session) {
|
|
||||||
teams = await getTeams({ userId: user.id });
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<NextAuthProvider session={session}>
|
<NextAuthProvider session={session}>
|
||||||
<div className="min-h-screen">
|
<div className="min-h-screen">
|
||||||
<ProfileHeader user={user} teams={teams} />
|
<ProfileHeader user={user} teams={teams} />
|
||||||
|
|
||||||
<main className="mb-8 mt-8 px-4 md:mb-12 md:mt-12 md:px-8">{children}</main>
|
<main className="my-8 px-4 md:my-12 md:px-8">{children}</main>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<RefreshOnFocus />
|
<RefreshOnFocus />
|
||||||
|
|||||||
@ -5,10 +5,12 @@ import { notFound, redirect } from 'next/navigation';
|
|||||||
import { FileIcon } from 'lucide-react';
|
import { FileIcon } from 'lucide-react';
|
||||||
import { DateTime } from 'luxon';
|
import { DateTime } from 'luxon';
|
||||||
|
|
||||||
|
import { NEXT_PUBLIC_WEBAPP_URL } from '@documenso/lib/constants/app';
|
||||||
|
import { getServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
||||||
import { getPublicProfileByUrl } from '@documenso/lib/server-only/profile/get-public-profile-by-url';
|
import { getPublicProfileByUrl } from '@documenso/lib/server-only/profile/get-public-profile-by-url';
|
||||||
import { extractInitials } from '@documenso/lib/utils/recipient-formatter';
|
import { extractInitials } from '@documenso/lib/utils/recipient-formatter';
|
||||||
import { formatDirectTemplatePath } from '@documenso/lib/utils/templates';
|
import { formatDirectTemplatePath } from '@documenso/lib/utils/templates';
|
||||||
import { Avatar, AvatarFallback } from '@documenso/ui/primitives/avatar';
|
import { Avatar, AvatarFallback, AvatarImage } from '@documenso/ui/primitives/avatar';
|
||||||
import { Button } from '@documenso/ui/primitives/button';
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
import {
|
import {
|
||||||
Table,
|
Table,
|
||||||
@ -52,19 +54,27 @@ export default async function PublicProfilePage({ params }: PublicProfilePagePro
|
|||||||
notFound();
|
notFound();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const { user } = await getServerComponentSession();
|
||||||
|
|
||||||
const { profile, templates } = publicProfile;
|
const { profile, templates } = publicProfile;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col items-center justify-center py-4 sm:py-32">
|
<div className="flex flex-col items-center justify-center py-4 sm:py-32">
|
||||||
<div className="flex flex-col items-center">
|
<div className="flex flex-col items-center">
|
||||||
<Avatar className="dark:border-border h-24 w-24 border-2 border-solid">
|
<Avatar className="dark:border-border h-24 w-24 border-2 border-solid">
|
||||||
|
{publicProfile.avatarImageId && (
|
||||||
|
<AvatarImage
|
||||||
|
src={`${NEXT_PUBLIC_WEBAPP_URL()}/api/avatar/${publicProfile.avatarImageId}`}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
<AvatarFallback className="text-sm text-gray-400">
|
<AvatarFallback className="text-sm text-gray-400">
|
||||||
{extractInitials(publicProfile.name)}
|
{extractInitials(publicProfile.name)}
|
||||||
</AvatarFallback>
|
</AvatarFallback>
|
||||||
</Avatar>
|
</Avatar>
|
||||||
|
|
||||||
<div className="mt-4 flex flex-row items-center justify-center">
|
<div className="mt-4 flex flex-row items-center justify-center">
|
||||||
<h2 className="font-bold">{publicProfile.name}</h2>
|
<h2 className="text-xl font-semibold md:text-2xl">{publicProfile.name}</h2>
|
||||||
|
|
||||||
{publicProfile.badge && (
|
{publicProfile.badge && (
|
||||||
<Tooltip>
|
<Tooltip>
|
||||||
@ -88,7 +98,7 @@ export default async function PublicProfilePage({ params }: PublicProfilePagePro
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="ml-2">
|
<div className="ml-2">
|
||||||
<p className="text-foreground text-base font-bold">
|
<p className="text-foreground text-base font-semibold">
|
||||||
{BADGE_DATA[publicProfile.badge.type].name}
|
{BADGE_DATA[publicProfile.badge.type].name}
|
||||||
</p>
|
</p>
|
||||||
<p className="text-muted-foreground mt-0.5 text-sm">
|
<p className="text-muted-foreground mt-0.5 text-sm">
|
||||||
@ -100,13 +110,43 @@ export default async function PublicProfilePage({ params }: PublicProfilePagePro
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="text-muted-foreground mt-4 max-w-lg whitespace-pre-wrap break-words text-center">
|
<div className="text-muted-foreground mt-4 space-y-1">
|
||||||
{profile.bio}
|
{(profile.bio ?? '').split('\n').map((line, index) => (
|
||||||
|
<p
|
||||||
|
key={index}
|
||||||
|
className="max-w-[60ch] whitespace-pre-wrap break-words text-center text-sm"
|
||||||
|
>
|
||||||
|
{line}
|
||||||
|
</p>
|
||||||
|
))}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{templates.length === 0 && (
|
||||||
|
<div className="mt-4 w-full max-w-xl border-t pt-4">
|
||||||
|
<p className="text-muted-foreground max-w-[60ch] whitespace-pre-wrap break-words text-center text-sm leading-relaxed">
|
||||||
|
It looks like {publicProfile.name} hasn't added any documents to their profile yet.{' '}
|
||||||
|
{!user?.id && (
|
||||||
|
<span className="mt-2 inline-block">
|
||||||
|
While waiting for them to do so you can create your own Documenso account and get
|
||||||
|
started with document signing right away.
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
{'userId' in profile && user?.id === profile.userId && (
|
||||||
|
<span className="mt-2 inline-block">
|
||||||
|
Go to your{' '}
|
||||||
|
<Link href="/settings/public-profile" className="underline">
|
||||||
|
public profile settings
|
||||||
|
</Link>{' '}
|
||||||
|
to add documents.
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{templates.length > 0 && (
|
{templates.length > 0 && (
|
||||||
<div className="mt-8 w-full max-w-3xl rounded-md border">
|
<div className="mt-8 w-full max-w-xl rounded-md border">
|
||||||
<Table className="w-full" overflowHidden>
|
<Table className="w-full" overflowHidden>
|
||||||
<TableHeader>
|
<TableHeader>
|
||||||
<TableRow>
|
<TableRow>
|
||||||
@ -119,16 +159,18 @@ export default async function PublicProfilePage({ params }: PublicProfilePagePro
|
|||||||
{templates.map((template) => (
|
{templates.map((template) => (
|
||||||
<TableRow key={template.id}>
|
<TableRow key={template.id}>
|
||||||
<TableCell className="text-muted-foreground flex flex-col justify-between overflow-hidden text-sm sm:flex-row">
|
<TableCell className="text-muted-foreground flex flex-col justify-between overflow-hidden text-sm sm:flex-row">
|
||||||
<div className="flex flex-1 items-start gap-2">
|
<div className="flex flex-1 items-start justify-start gap-2">
|
||||||
<FileIcon
|
<FileIcon
|
||||||
className="text-muted-foreground/40 h-8 w-8 flex-shrink-0"
|
className="text-muted-foreground/40 h-8 w-8 flex-shrink-0"
|
||||||
strokeWidth={1.5}
|
strokeWidth={1.5}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="flex flex-1 flex-col gap-4 overflow-hidden md:flex-row md:items-center md:justify-between">
|
<div className="flex flex-1 flex-col gap-4 overflow-hidden md:flex-row md:items-start md:justify-between">
|
||||||
<div>
|
<div>
|
||||||
<p className="text-sm font-bold">{template.publicTitle}</p>
|
<p className="text-foreground text-sm font-semibold leading-none">
|
||||||
<p className="line-clamp-3 max-w-[70ch] whitespace-normal text-xs text-neutral-400">
|
{template.publicTitle}
|
||||||
|
</p>
|
||||||
|
<p className="text-muted-foreground mt-1 line-clamp-3 max-w-[70ch] whitespace-normal text-xs">
|
||||||
{template.publicDescription}
|
{template.publicDescription}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -62,9 +62,9 @@ export const ProfileHeader = ({ user, teams = [] }: ProfileHeaderProps) => {
|
|||||||
</Link>
|
</Link>
|
||||||
|
|
||||||
<div className="flex flex-row items-center justify-center">
|
<div className="flex flex-row items-center justify-center">
|
||||||
<p className="mr-4 text-neutral-400">
|
<p className="text-muted-foreground mr-4">
|
||||||
<span className="text-sm sm:hidden">Want your own public profile?</span>
|
<span className="text-sm sm:hidden">Want your own public profile?</span>
|
||||||
<span className="hidden sm:block">
|
<span className="hidden text-sm sm:block">
|
||||||
Like to have your own public profile with agreements?
|
Like to have your own public profile with agreements?
|
||||||
</span>
|
</span>
|
||||||
</p>
|
</p>
|
||||||
|
|||||||
@ -13,6 +13,7 @@ import { AddTeamEmailDialog } from '~/components/(teams)/dialogs/add-team-email-
|
|||||||
import { DeleteTeamDialog } from '~/components/(teams)/dialogs/delete-team-dialog';
|
import { DeleteTeamDialog } from '~/components/(teams)/dialogs/delete-team-dialog';
|
||||||
import { TransferTeamDialog } from '~/components/(teams)/dialogs/transfer-team-dialog';
|
import { TransferTeamDialog } from '~/components/(teams)/dialogs/transfer-team-dialog';
|
||||||
import { UpdateTeamForm } from '~/components/(teams)/forms/update-team-form';
|
import { UpdateTeamForm } from '~/components/(teams)/forms/update-team-form';
|
||||||
|
import { AvatarImageForm } from '~/components/forms/avatar-image';
|
||||||
|
|
||||||
import { TeamEmailDropdown } from './team-email-dropdown';
|
import { TeamEmailDropdown } from './team-email-dropdown';
|
||||||
import { TeamTransferStatus } from './team-transfer-status';
|
import { TeamTransferStatus } from './team-transfer-status';
|
||||||
@ -44,6 +45,8 @@ export default async function TeamsSettingsPage({ params }: TeamsSettingsPagePro
|
|||||||
transferVerification={team.transferVerification}
|
transferVerification={team.transferVerification}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<AvatarImageForm className="mb-8" team={team} user={session.user} />
|
||||||
|
|
||||||
<UpdateTeamForm teamId={team.id} teamName={team.name} teamUrl={team.url} />
|
<UpdateTeamForm teamId={team.id} teamName={team.name} teamUrl={team.url} />
|
||||||
|
|
||||||
<section className="mt-6 space-y-6">
|
<section className="mt-6 space-y-6">
|
||||||
|
|||||||
@ -7,6 +7,7 @@ import { motion } from 'framer-motion';
|
|||||||
import { CheckCircle2, ChevronsUpDown, Plus, Settings2 } from 'lucide-react';
|
import { CheckCircle2, ChevronsUpDown, Plus, Settings2 } from 'lucide-react';
|
||||||
import { signOut } from 'next-auth/react';
|
import { signOut } from 'next-auth/react';
|
||||||
|
|
||||||
|
import { NEXT_PUBLIC_WEBAPP_URL } from '@documenso/lib/constants/app';
|
||||||
import { TEAM_MEMBER_ROLE_MAP, TEAM_URL_REGEX } from '@documenso/lib/constants/teams';
|
import { TEAM_MEMBER_ROLE_MAP, TEAM_URL_REGEX } from '@documenso/lib/constants/teams';
|
||||||
import { isAdmin } from '@documenso/lib/next-auth/guards/is-admin';
|
import { isAdmin } from '@documenso/lib/next-auth/guards/is-admin';
|
||||||
import type { GetTeamsResponse } from '@documenso/lib/server-only/team/get-teams';
|
import type { GetTeamsResponse } from '@documenso/lib/server-only/team/get-teams';
|
||||||
@ -99,6 +100,9 @@ export const MenuSwitcher = ({ user, teams: initialTeamsData }: MenuSwitcherProp
|
|||||||
className="relative flex h-12 flex-row items-center px-0 py-2 ring-0 focus:outline-none focus-visible:border-0 focus-visible:ring-0 focus-visible:ring-transparent md:px-2"
|
className="relative flex h-12 flex-row items-center px-0 py-2 ring-0 focus:outline-none focus-visible:border-0 focus-visible:ring-0 focus-visible:ring-transparent md:px-2"
|
||||||
>
|
>
|
||||||
<AvatarWithText
|
<AvatarWithText
|
||||||
|
avatarSrc={`${NEXT_PUBLIC_WEBAPP_URL()}/api/avatar/${
|
||||||
|
selectedTeam ? selectedTeam.avatarImageId : user.avatarImageId
|
||||||
|
}`}
|
||||||
avatarFallback={formatAvatarFallback(selectedTeam?.name)}
|
avatarFallback={formatAvatarFallback(selectedTeam?.name)}
|
||||||
primaryText={selectedTeam ? selectedTeam.name : user.name}
|
primaryText={selectedTeam ? selectedTeam.name : user.name}
|
||||||
secondaryText={formatSecondaryAvatarText(selectedTeam)}
|
secondaryText={formatSecondaryAvatarText(selectedTeam)}
|
||||||
@ -122,6 +126,11 @@ export const MenuSwitcher = ({ user, teams: initialTeamsData }: MenuSwitcherProp
|
|||||||
<DropdownMenuItem asChild>
|
<DropdownMenuItem asChild>
|
||||||
<Link href={formatRedirectUrlOnSwitch()}>
|
<Link href={formatRedirectUrlOnSwitch()}>
|
||||||
<AvatarWithText
|
<AvatarWithText
|
||||||
|
avatarSrc={
|
||||||
|
user.avatarImageId
|
||||||
|
? `${NEXT_PUBLIC_WEBAPP_URL()}/api/avatar/${user.avatarImageId}`
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
avatarFallback={formatAvatarFallback()}
|
avatarFallback={formatAvatarFallback()}
|
||||||
primaryText={user.name}
|
primaryText={user.name}
|
||||||
secondaryText={formatSecondaryAvatarText()}
|
secondaryText={formatSecondaryAvatarText()}
|
||||||
@ -180,10 +189,15 @@ export const MenuSwitcher = ({ user, teams: initialTeamsData }: MenuSwitcherProp
|
|||||||
href={formatRedirectUrlOnSwitch(team.url)}
|
href={formatRedirectUrlOnSwitch(team.url)}
|
||||||
>
|
>
|
||||||
<AvatarWithText
|
<AvatarWithText
|
||||||
|
avatarSrc={
|
||||||
|
team.avatarImageId
|
||||||
|
? `${NEXT_PUBLIC_WEBAPP_URL()}/api/avatar/${team.avatarImageId}`
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
avatarFallback={formatAvatarFallback(team.name)}
|
avatarFallback={formatAvatarFallback(team.name)}
|
||||||
primaryText={team.name}
|
primaryText={team.name}
|
||||||
secondaryText={
|
secondaryText={
|
||||||
<div className="relative">
|
<div className="relative w-full">
|
||||||
<motion.span
|
<motion.span
|
||||||
className="overflow-hidden"
|
className="overflow-hidden"
|
||||||
variants={{
|
variants={{
|
||||||
|
|||||||
188
apps/web/src/components/forms/avatar-image.tsx
Normal file
188
apps/web/src/components/forms/avatar-image.tsx
Normal file
@ -0,0 +1,188 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
|
import { useRouter } from 'next/navigation';
|
||||||
|
|
||||||
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
|
import { ErrorCode, useDropzone } from 'react-dropzone';
|
||||||
|
import { useForm } from 'react-hook-form';
|
||||||
|
import { match } from 'ts-pattern';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import { NEXT_PUBLIC_WEBAPP_URL } from '@documenso/lib/constants/app';
|
||||||
|
import { base64 } from '@documenso/lib/universal/base64';
|
||||||
|
import { extractInitials } from '@documenso/lib/utils/recipient-formatter';
|
||||||
|
import type { Team, User } from '@documenso/prisma/client';
|
||||||
|
import { TRPCClientError } from '@documenso/trpc/client';
|
||||||
|
import { trpc } from '@documenso/trpc/react';
|
||||||
|
import { cn } from '@documenso/ui/lib/utils';
|
||||||
|
import { Avatar, AvatarFallback, AvatarImage } from '@documenso/ui/primitives/avatar';
|
||||||
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
|
import {
|
||||||
|
Form,
|
||||||
|
FormControl,
|
||||||
|
FormField,
|
||||||
|
FormItem,
|
||||||
|
FormLabel,
|
||||||
|
FormMessage,
|
||||||
|
} from '@documenso/ui/primitives/form/form';
|
||||||
|
import { useToast } from '@documenso/ui/primitives/use-toast';
|
||||||
|
|
||||||
|
export const ZAvatarImageFormSchema = z.object({
|
||||||
|
bytes: z.string().nullish(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TAvatarImageFormSchema = z.infer<typeof ZAvatarImageFormSchema>;
|
||||||
|
|
||||||
|
export type AvatarImageFormProps = {
|
||||||
|
className?: string;
|
||||||
|
user: User;
|
||||||
|
team?: Team;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const AvatarImageForm = ({ className, user, team }: AvatarImageFormProps) => {
|
||||||
|
const { toast } = useToast();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const { mutateAsync: setProfileImage } = trpc.profile.setProfileImage.useMutation();
|
||||||
|
|
||||||
|
const initials = extractInitials(team?.name || user.name || '');
|
||||||
|
|
||||||
|
const hasAvatarImage = useMemo(() => {
|
||||||
|
if (team) {
|
||||||
|
return team.avatarImageId !== null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return user.avatarImageId !== null;
|
||||||
|
}, [team, user.avatarImageId]);
|
||||||
|
|
||||||
|
const avatarImageId = team ? team.avatarImageId : user.avatarImageId;
|
||||||
|
|
||||||
|
const form = useForm<TAvatarImageFormSchema>({
|
||||||
|
values: {
|
||||||
|
bytes: null,
|
||||||
|
},
|
||||||
|
resolver: zodResolver(ZAvatarImageFormSchema),
|
||||||
|
});
|
||||||
|
|
||||||
|
const { getRootProps, getInputProps } = useDropzone({
|
||||||
|
maxSize: 1024 * 1024,
|
||||||
|
accept: {
|
||||||
|
'image/*': ['.png', '.jpg', '.jpeg'],
|
||||||
|
},
|
||||||
|
multiple: false,
|
||||||
|
onDropAccepted: ([file]) => {
|
||||||
|
void file.arrayBuffer().then((buffer) => {
|
||||||
|
const contents = base64.encode(new Uint8Array(buffer));
|
||||||
|
|
||||||
|
form.setValue('bytes', contents);
|
||||||
|
void form.handleSubmit(onFormSubmit)();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onDropRejected: ([file]) => {
|
||||||
|
form.setError('bytes', {
|
||||||
|
type: 'onChange',
|
||||||
|
message: match(file.errors[0].code)
|
||||||
|
.with(ErrorCode.FileTooLarge, () => 'Uploaded file is too large')
|
||||||
|
.with(ErrorCode.FileTooSmall, () => 'Uploaded file is too small')
|
||||||
|
.with(ErrorCode.FileInvalidType, () => 'Uploaded file not an allowed file type')
|
||||||
|
.otherwise(() => 'An unknown error occurred'),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const onFormSubmit = async (data: TAvatarImageFormSchema) => {
|
||||||
|
try {
|
||||||
|
await setProfileImage({
|
||||||
|
bytes: data.bytes,
|
||||||
|
teamId: team?.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: 'Avatar Updated',
|
||||||
|
description: 'Your avatar has been updated successfully.',
|
||||||
|
duration: 5000,
|
||||||
|
});
|
||||||
|
|
||||||
|
router.refresh();
|
||||||
|
} catch (err) {
|
||||||
|
if (err instanceof TRPCClientError && err.data?.code === 'BAD_REQUEST') {
|
||||||
|
toast({
|
||||||
|
title: 'An error occurred',
|
||||||
|
description: err.message,
|
||||||
|
variant: 'destructive',
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
toast({
|
||||||
|
title: 'An unknown error occurred',
|
||||||
|
variant: 'destructive',
|
||||||
|
description:
|
||||||
|
'We encountered an unknown error while attempting to update the avatar. Please try again later.',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Form {...form}>
|
||||||
|
<form
|
||||||
|
className={cn('flex w-full flex-col gap-y-4', className)}
|
||||||
|
// onSubmit={form.handleSubmit(onFormSubmit)}
|
||||||
|
>
|
||||||
|
<fieldset className="flex w-full flex-col gap-y-4" disabled={form.formState.isSubmitting}>
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="bytes"
|
||||||
|
render={() => (
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel>Avatar</FormLabel>
|
||||||
|
|
||||||
|
<FormControl>
|
||||||
|
<div className="flex items-center gap-8">
|
||||||
|
<div className="relative">
|
||||||
|
<Avatar className="h-16 w-16 border-2 border-solid">
|
||||||
|
{avatarImageId && (
|
||||||
|
<AvatarImage
|
||||||
|
src={`${NEXT_PUBLIC_WEBAPP_URL()}/api/avatar/${avatarImageId}`}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<AvatarFallback className="text-sm text-gray-400">
|
||||||
|
{initials}
|
||||||
|
</AvatarFallback>
|
||||||
|
</Avatar>
|
||||||
|
|
||||||
|
{hasAvatarImage && (
|
||||||
|
<button
|
||||||
|
className="bg-background/70 text-destructive absolute inset-0 flex cursor-pointer items-center justify-center text-xs opacity-0 transition-opacity hover:opacity-100"
|
||||||
|
disabled={form.formState.isSubmitting}
|
||||||
|
onClick={() => void onFormSubmit({ bytes: null })}
|
||||||
|
>
|
||||||
|
Remove
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
type="button"
|
||||||
|
variant="secondary"
|
||||||
|
size="sm"
|
||||||
|
{...getRootProps()}
|
||||||
|
loading={form.formState.isSubmitting}
|
||||||
|
disabled={form.formState.isSubmitting}
|
||||||
|
>
|
||||||
|
Upload Avatar
|
||||||
|
<input {...getInputProps()} />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</FormControl>
|
||||||
|
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</fieldset>
|
||||||
|
</form>
|
||||||
|
</Form>
|
||||||
|
);
|
||||||
|
};
|
||||||
@ -169,6 +169,8 @@ export const ManagePublicTemplateDialog = ({
|
|||||||
description: 'Template has been updated.',
|
description: 'Template has been updated.',
|
||||||
duration: 5000,
|
duration: 5000,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
onOpenChange(false);
|
||||||
} catch {
|
} catch {
|
||||||
toast({
|
toast({
|
||||||
title: 'An unknown error occurred',
|
title: 'An unknown error occurred',
|
||||||
|
|||||||
34
apps/web/src/pages/api/avatar/[id].tsx
Normal file
34
apps/web/src/pages/api/avatar/[id].tsx
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
import type { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
|
||||||
|
import { getAvatarImage } from '@documenso/lib/server-only/profile/get-avatar-image';
|
||||||
|
|
||||||
|
export default async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
if (req.method !== 'GET') {
|
||||||
|
return res.status(405).json({
|
||||||
|
status: 'error',
|
||||||
|
message: 'Method not allowed',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = req.query;
|
||||||
|
|
||||||
|
if (typeof id !== 'string') {
|
||||||
|
return res.status(400).json({
|
||||||
|
status: 'error',
|
||||||
|
message: 'Missing id',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = await getAvatarImage({ id });
|
||||||
|
|
||||||
|
if (!result) {
|
||||||
|
return res.status(404).json({
|
||||||
|
status: 'error',
|
||||||
|
message: 'Not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
res.setHeader('Content-Type', result.contentType);
|
||||||
|
res.setHeader('Cache-Control', 'public, max-age=31536000, immutable');
|
||||||
|
res.send(result.content);
|
||||||
|
}
|
||||||
3
package-lock.json
generated
3
package-lock.json
generated
@ -32,7 +32,7 @@
|
|||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18.0.0",
|
"node": ">=18.0.0",
|
||||||
"npm": ">=8.6.0"
|
"npm": ">=10.7.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/marketing": {
|
"apps/marketing": {
|
||||||
@ -31843,6 +31843,7 @@
|
|||||||
"playwright": "1.43.0",
|
"playwright": "1.43.0",
|
||||||
"react": "18.2.0",
|
"react": "18.2.0",
|
||||||
"remeda": "^1.27.1",
|
"remeda": "^1.27.1",
|
||||||
|
"sharp": "^0.33.1",
|
||||||
"stripe": "^12.7.0",
|
"stripe": "^12.7.0",
|
||||||
"ts-pattern": "^5.0.5",
|
"ts-pattern": "^5.0.5",
|
||||||
"zod": "^3.22.4"
|
"zod": "^3.22.4"
|
||||||
|
|||||||
@ -49,6 +49,7 @@
|
|||||||
"playwright": "1.43.0",
|
"playwright": "1.43.0",
|
||||||
"react": "18.2.0",
|
"react": "18.2.0",
|
||||||
"remeda": "^1.27.1",
|
"remeda": "^1.27.1",
|
||||||
|
"sharp": "^0.33.1",
|
||||||
"stripe": "^12.7.0",
|
"stripe": "^12.7.0",
|
||||||
"ts-pattern": "^5.0.5",
|
"ts-pattern": "^5.0.5",
|
||||||
"zod": "^3.22.4"
|
"zod": "^3.22.4"
|
||||||
|
|||||||
26
packages/lib/server-only/profile/get-avatar-image.ts
Normal file
26
packages/lib/server-only/profile/get-avatar-image.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import sharp from 'sharp';
|
||||||
|
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
|
export type GetAvatarImageOptions = {
|
||||||
|
id: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getAvatarImage = async ({ id }: GetAvatarImageOptions) => {
|
||||||
|
const avatarImage = await prisma.avatarImage.findFirst({
|
||||||
|
where: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!avatarImage) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const bytes = Buffer.from(avatarImage.bytes, 'base64');
|
||||||
|
|
||||||
|
return {
|
||||||
|
contentType: 'image/jpeg',
|
||||||
|
content: await sharp(bytes).toFormat('jpeg').toBuffer(),
|
||||||
|
};
|
||||||
|
};
|
||||||
@ -25,6 +25,7 @@ type PublicDirectLinkTemplate = Template & {
|
|||||||
type BaseResponse = {
|
type BaseResponse = {
|
||||||
url: string;
|
url: string;
|
||||||
name: string;
|
name: string;
|
||||||
|
avatarImageId?: string | null;
|
||||||
badge?: {
|
badge?: {
|
||||||
type: 'Premium' | 'EarlySupporter';
|
type: 'Premium' | 'EarlySupporter';
|
||||||
since: Date;
|
since: Date;
|
||||||
@ -149,6 +150,7 @@ export const getPublicProfileByUrl = async ({
|
|||||||
badge,
|
badge,
|
||||||
profile: user.profile,
|
profile: user.profile,
|
||||||
url: profileUrl,
|
url: profileUrl,
|
||||||
|
avatarImageId: user.avatarImageId,
|
||||||
name: user.name || '',
|
name: user.name || '',
|
||||||
templates: user.Template.filter(
|
templates: user.Template.filter(
|
||||||
(template): template is PublicDirectLinkTemplate =>
|
(template): template is PublicDirectLinkTemplate =>
|
||||||
@ -166,6 +168,7 @@ export const getPublicProfileByUrl = async ({
|
|||||||
},
|
},
|
||||||
profile: team.profile,
|
profile: team.profile,
|
||||||
url: profileUrl,
|
url: profileUrl,
|
||||||
|
avatarImageId: team.avatarImageId,
|
||||||
name: team.name || '',
|
name: team.name || '',
|
||||||
templates: team.templates.filter(
|
templates: team.templates.filter(
|
||||||
(template): template is PublicDirectLinkTemplate =>
|
(template): template is PublicDirectLinkTemplate =>
|
||||||
|
|||||||
106
packages/lib/server-only/profile/set-avatar-image.ts
Normal file
106
packages/lib/server-only/profile/set-avatar-image.ts
Normal file
@ -0,0 +1,106 @@
|
|||||||
|
import sharp from 'sharp';
|
||||||
|
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
|
import type { RequestMetadata } from '../../universal/extract-request-metadata';
|
||||||
|
|
||||||
|
export type SetAvatarImageOptions = {
|
||||||
|
userId: number;
|
||||||
|
teamId?: number | null;
|
||||||
|
bytes?: string | null;
|
||||||
|
requestMetadata?: RequestMetadata;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setAvatarImage = async ({
|
||||||
|
userId,
|
||||||
|
teamId,
|
||||||
|
bytes,
|
||||||
|
requestMetadata,
|
||||||
|
}: SetAvatarImageOptions) => {
|
||||||
|
let oldAvatarImageId: string | null = null;
|
||||||
|
|
||||||
|
const user = await prisma.user.findUnique({
|
||||||
|
where: {
|
||||||
|
id: userId,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
avatarImage: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
throw new Error('User not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
oldAvatarImageId = user.avatarImageId;
|
||||||
|
|
||||||
|
if (teamId) {
|
||||||
|
const team = await prisma.team.findFirst({
|
||||||
|
where: {
|
||||||
|
id: teamId,
|
||||||
|
members: {
|
||||||
|
some: {
|
||||||
|
userId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!team) {
|
||||||
|
throw new Error('Team not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
oldAvatarImageId = team.avatarImageId;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (oldAvatarImageId) {
|
||||||
|
await prisma.avatarImage.delete({
|
||||||
|
where: {
|
||||||
|
id: oldAvatarImageId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let newAvatarImageId: string | null = null;
|
||||||
|
|
||||||
|
if (bytes) {
|
||||||
|
const optimisedBytes = await sharp(Buffer.from(bytes, 'base64'))
|
||||||
|
.resize(512, 512)
|
||||||
|
.toFormat('jpeg', { quality: 75 })
|
||||||
|
.toBuffer();
|
||||||
|
|
||||||
|
const avatarImage = await prisma.avatarImage.create({
|
||||||
|
data: {
|
||||||
|
bytes: optimisedBytes.toString('base64'),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
newAvatarImageId = avatarImage.id;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (teamId) {
|
||||||
|
await prisma.team.update({
|
||||||
|
where: {
|
||||||
|
id: teamId,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
avatarImageId: newAvatarImageId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// TODO: Audit Logs
|
||||||
|
} else {
|
||||||
|
await prisma.user.update({
|
||||||
|
where: {
|
||||||
|
id: userId,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
avatarImageId: newAvatarImageId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// TODO: Audit Logs
|
||||||
|
}
|
||||||
|
|
||||||
|
return newAvatarImageId;
|
||||||
|
};
|
||||||
@ -0,0 +1,19 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "Team" ADD COLUMN "avatarImageId" TEXT;
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "User" ADD COLUMN "avatarImageId" TEXT;
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "AvatarImage" (
|
||||||
|
"id" TEXT NOT NULL,
|
||||||
|
"bytes" TEXT NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "AvatarImage_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "User" ADD CONSTRAINT "User_avatarImageId_fkey" FOREIGN KEY ("avatarImageId") REFERENCES "AvatarImage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "Team" ADD CONSTRAINT "Team_avatarImageId_fkey" FOREIGN KEY ("avatarImageId") REFERENCES "AvatarImage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
|
||||||
@ -24,19 +24,21 @@ enum Role {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model User {
|
model User {
|
||||||
id Int @id @default(autoincrement())
|
id Int @id @default(autoincrement())
|
||||||
name String?
|
name String?
|
||||||
customerId String? @unique
|
customerId String? @unique
|
||||||
email String @unique
|
email String @unique
|
||||||
emailVerified DateTime?
|
emailVerified DateTime?
|
||||||
password String?
|
password String?
|
||||||
source String?
|
source String?
|
||||||
signature String?
|
signature String?
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @default(now()) @updatedAt
|
updatedAt DateTime @default(now()) @updatedAt
|
||||||
lastSignedIn DateTime @default(now())
|
lastSignedIn DateTime @default(now())
|
||||||
roles Role[] @default([USER])
|
roles Role[] @default([USER])
|
||||||
identityProvider IdentityProvider @default(DOCUMENSO)
|
identityProvider IdentityProvider @default(DOCUMENSO)
|
||||||
|
avatarImageId String?
|
||||||
|
|
||||||
accounts Account[]
|
accounts Account[]
|
||||||
sessions Session[]
|
sessions Session[]
|
||||||
Document Document[]
|
Document Document[]
|
||||||
@ -58,6 +60,7 @@ model User {
|
|||||||
Webhooks Webhook[]
|
Webhooks Webhook[]
|
||||||
siteSettings SiteSettings[]
|
siteSettings SiteSettings[]
|
||||||
passkeys Passkey[]
|
passkeys Passkey[]
|
||||||
|
avatarImage AvatarImage? @relation(fields: [avatarImageId], references: [id], onDelete: SetNull)
|
||||||
|
|
||||||
@@index([email])
|
@@index([email])
|
||||||
}
|
}
|
||||||
@ -474,17 +477,20 @@ enum TeamMemberInviteStatus {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model Team {
|
model Team {
|
||||||
id Int @id @default(autoincrement())
|
id Int @id @default(autoincrement())
|
||||||
name String
|
name String
|
||||||
url String @unique
|
url String @unique
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
customerId String? @unique
|
avatarImageId String?
|
||||||
ownerUserId Int
|
customerId String? @unique
|
||||||
|
ownerUserId Int
|
||||||
|
|
||||||
members TeamMember[]
|
members TeamMember[]
|
||||||
invites TeamMemberInvite[]
|
invites TeamMemberInvite[]
|
||||||
teamEmail TeamEmail?
|
teamEmail TeamEmail?
|
||||||
emailVerification TeamEmailVerification?
|
emailVerification TeamEmailVerification?
|
||||||
transferVerification TeamTransferVerification?
|
transferVerification TeamTransferVerification?
|
||||||
|
avatarImage AvatarImage? @relation(fields: [avatarImageId], references: [id], onDelete: SetNull)
|
||||||
|
|
||||||
profile TeamProfile?
|
profile TeamProfile?
|
||||||
owner User @relation(fields: [ownerUserId], references: [id], onDelete: Cascade)
|
owner User @relation(fields: [ownerUserId], references: [id], onDelete: Cascade)
|
||||||
@ -676,3 +682,11 @@ model BackgroundJobTask {
|
|||||||
jobId String
|
jobId String
|
||||||
backgroundJob BackgroundJob @relation(fields: [jobId], references: [id], onDelete: Cascade)
|
backgroundJob BackgroundJob @relation(fields: [jobId], references: [id], onDelete: Cascade)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
model AvatarImage {
|
||||||
|
id String @id @default(cuid())
|
||||||
|
bytes String
|
||||||
|
|
||||||
|
team Team[]
|
||||||
|
user User[]
|
||||||
|
}
|
||||||
|
|||||||
@ -3,6 +3,7 @@ import { TRPCError } from '@trpc/server';
|
|||||||
import { IS_BILLING_ENABLED } from '@documenso/lib/constants/app';
|
import { IS_BILLING_ENABLED } from '@documenso/lib/constants/app';
|
||||||
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
import { jobsClient } from '@documenso/lib/jobs/client';
|
import { jobsClient } from '@documenso/lib/jobs/client';
|
||||||
|
import { setAvatarImage } from '@documenso/lib/server-only/profile/set-avatar-image';
|
||||||
import { getSubscriptionsByUserId } from '@documenso/lib/server-only/subscription/get-subscriptions-by-user-id';
|
import { getSubscriptionsByUserId } from '@documenso/lib/server-only/subscription/get-subscriptions-by-user-id';
|
||||||
import { deleteUser } from '@documenso/lib/server-only/user/delete-user';
|
import { deleteUser } from '@documenso/lib/server-only/user/delete-user';
|
||||||
import { findUserSecurityAuditLogs } from '@documenso/lib/server-only/user/find-user-security-audit-logs';
|
import { findUserSecurityAuditLogs } from '@documenso/lib/server-only/user/find-user-security-audit-logs';
|
||||||
@ -22,6 +23,7 @@ import {
|
|||||||
ZForgotPasswordFormSchema,
|
ZForgotPasswordFormSchema,
|
||||||
ZResetPasswordFormSchema,
|
ZResetPasswordFormSchema,
|
||||||
ZRetrieveUserByIdQuerySchema,
|
ZRetrieveUserByIdQuerySchema,
|
||||||
|
ZSetProfileImageMutationSchema,
|
||||||
ZUpdatePasswordMutationSchema,
|
ZUpdatePasswordMutationSchema,
|
||||||
ZUpdateProfileMutationSchema,
|
ZUpdateProfileMutationSchema,
|
||||||
ZUpdatePublicProfileMutationSchema,
|
ZUpdatePublicProfileMutationSchema,
|
||||||
@ -246,4 +248,32 @@ export const profileRouter = router({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
setProfileImage: authenticatedProcedure
|
||||||
|
.input(ZSetProfileImageMutationSchema)
|
||||||
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
try {
|
||||||
|
const { bytes, teamId } = input;
|
||||||
|
|
||||||
|
return await setAvatarImage({
|
||||||
|
userId: ctx.user.id,
|
||||||
|
teamId,
|
||||||
|
bytes,
|
||||||
|
requestMetadata: extractNextApiRequestMetadata(ctx.req),
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
console.error(err);
|
||||||
|
|
||||||
|
let message = 'We were unable to update your profile image. Please try again.';
|
||||||
|
|
||||||
|
if (err instanceof Error) {
|
||||||
|
message = err.message;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new TRPCError({
|
||||||
|
code: 'BAD_REQUEST',
|
||||||
|
message,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
|
|||||||
@ -9,15 +9,21 @@ export const ZFindUserSecurityAuditLogsSchema = z.object({
|
|||||||
perPage: z.number().optional(),
|
perPage: z.number().optional(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type TFindUserSecurityAuditLogsSchema = z.infer<typeof ZFindUserSecurityAuditLogsSchema>;
|
||||||
|
|
||||||
export const ZRetrieveUserByIdQuerySchema = z.object({
|
export const ZRetrieveUserByIdQuerySchema = z.object({
|
||||||
id: z.number().min(1),
|
id: z.number().min(1),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type TRetrieveUserByIdQuerySchema = z.infer<typeof ZRetrieveUserByIdQuerySchema>;
|
||||||
|
|
||||||
export const ZUpdateProfileMutationSchema = z.object({
|
export const ZUpdateProfileMutationSchema = z.object({
|
||||||
name: z.string().min(1),
|
name: z.string().min(1),
|
||||||
signature: z.string(),
|
signature: z.string(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type TUpdateProfileMutationSchema = z.infer<typeof ZUpdateProfileMutationSchema>;
|
||||||
|
|
||||||
export const ZUpdatePublicProfileMutationSchema = z.object({
|
export const ZUpdatePublicProfileMutationSchema = z.object({
|
||||||
bio: z
|
bio: z
|
||||||
.string()
|
.string()
|
||||||
@ -37,28 +43,37 @@ export const ZUpdatePublicProfileMutationSchema = z.object({
|
|||||||
.optional(),
|
.optional(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type TUpdatePublicProfileMutationSchema = z.infer<typeof ZUpdatePublicProfileMutationSchema>;
|
||||||
|
|
||||||
export const ZUpdatePasswordMutationSchema = z.object({
|
export const ZUpdatePasswordMutationSchema = z.object({
|
||||||
currentPassword: ZCurrentPasswordSchema,
|
currentPassword: ZCurrentPasswordSchema,
|
||||||
password: ZPasswordSchema,
|
password: ZPasswordSchema,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type TUpdatePasswordMutationSchema = z.infer<typeof ZUpdatePasswordMutationSchema>;
|
||||||
|
|
||||||
export const ZForgotPasswordFormSchema = z.object({
|
export const ZForgotPasswordFormSchema = z.object({
|
||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type TForgotPasswordFormSchema = z.infer<typeof ZForgotPasswordFormSchema>;
|
||||||
|
|
||||||
export const ZResetPasswordFormSchema = z.object({
|
export const ZResetPasswordFormSchema = z.object({
|
||||||
password: ZPasswordSchema,
|
password: ZPasswordSchema,
|
||||||
token: z.string().min(1),
|
token: z.string().min(1),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type TResetPasswordFormSchema = z.infer<typeof ZResetPasswordFormSchema>;
|
||||||
|
|
||||||
export const ZConfirmEmailMutationSchema = z.object({
|
export const ZConfirmEmailMutationSchema = z.object({
|
||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
});
|
});
|
||||||
|
|
||||||
export type TFindUserSecurityAuditLogsSchema = z.infer<typeof ZFindUserSecurityAuditLogsSchema>;
|
|
||||||
export type TRetrieveUserByIdQuerySchema = z.infer<typeof ZRetrieveUserByIdQuerySchema>;
|
|
||||||
export type TUpdateProfileMutationSchema = z.infer<typeof ZUpdateProfileMutationSchema>;
|
|
||||||
export type TUpdatePasswordMutationSchema = z.infer<typeof ZUpdatePasswordMutationSchema>;
|
|
||||||
export type TForgotPasswordFormSchema = z.infer<typeof ZForgotPasswordFormSchema>;
|
|
||||||
export type TResetPasswordFormSchema = z.infer<typeof ZResetPasswordFormSchema>;
|
|
||||||
export type TConfirmEmailMutationSchema = z.infer<typeof ZConfirmEmailMutationSchema>;
|
export type TConfirmEmailMutationSchema = z.infer<typeof ZConfirmEmailMutationSchema>;
|
||||||
|
|
||||||
|
export const ZSetProfileImageMutationSchema = z.object({
|
||||||
|
bytes: z.string().nullish(),
|
||||||
|
teamId: z.number().min(1).nullish(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TSetProfileImageMutationSchema = z.infer<typeof ZSetProfileImageMutationSchema>;
|
||||||
|
|||||||
@ -50,6 +50,7 @@ AvatarFallback.displayName = AvatarPrimitive.Fallback.displayName;
|
|||||||
|
|
||||||
type AvatarWithTextProps = {
|
type AvatarWithTextProps = {
|
||||||
avatarClass?: string;
|
avatarClass?: string;
|
||||||
|
avatarSrc?: string | null;
|
||||||
avatarFallback: string;
|
avatarFallback: string;
|
||||||
className?: string;
|
className?: string;
|
||||||
primaryText: React.ReactNode;
|
primaryText: React.ReactNode;
|
||||||
@ -61,6 +62,7 @@ type AvatarWithTextProps = {
|
|||||||
|
|
||||||
const AvatarWithText = ({
|
const AvatarWithText = ({
|
||||||
avatarClass,
|
avatarClass,
|
||||||
|
avatarSrc,
|
||||||
avatarFallback,
|
avatarFallback,
|
||||||
className,
|
className,
|
||||||
primaryText,
|
primaryText,
|
||||||
@ -72,6 +74,7 @@ const AvatarWithText = ({
|
|||||||
<Avatar
|
<Avatar
|
||||||
className={cn('dark:border-border h-10 w-10 border-2 border-solid border-white', avatarClass)}
|
className={cn('dark:border-border h-10 w-10 border-2 border-solid border-white', avatarClass)}
|
||||||
>
|
>
|
||||||
|
{avatarSrc && <AvatarImage src={avatarSrc} />}
|
||||||
<AvatarFallback className="text-xs text-gray-400">{avatarFallback}</AvatarFallback>
|
<AvatarFallback className="text-xs text-gray-400">{avatarFallback}</AvatarFallback>
|
||||||
</Avatar>
|
</Avatar>
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user