mirror of
https://github.com/lumehq/lume.git
synced 2025-10-11 00:22:38 +02:00
added profile update page
This commit is contained in:
@@ -12,6 +12,13 @@ export const ProfileMenu = memo(function ProfileMenu({ pubkey }: { pubkey: strin
|
|||||||
router.push(`/profile/${pubkey}`);
|
router.push(`/profile/${pubkey}`);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const updateProfile = () => {
|
||||||
|
router.push({
|
||||||
|
pathname: '/profile/update',
|
||||||
|
query: { pubkey: pubkey },
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
const copyPubkey = async () => {
|
const copyPubkey = async () => {
|
||||||
const npub = nip19.npubEncode(pubkey);
|
const npub = nip19.npubEncode(pubkey);
|
||||||
await writeText(npub);
|
await writeText(npub);
|
||||||
@@ -33,6 +40,11 @@ export const ProfileMenu = memo(function ProfileMenu({ pubkey }: { pubkey: strin
|
|||||||
className="group relative flex h-[30px] select-none items-center rounded px-1 pl-6 text-sm leading-none text-zinc-100 outline-none data-[disabled]:pointer-events-none data-[highlighted]:bg-zinc-700 data-[highlighted]:text-fuchsia-400 data-[disabled]:text-zinc-400">
|
className="group relative flex h-[30px] select-none items-center rounded px-1 pl-6 text-sm leading-none text-zinc-100 outline-none data-[disabled]:pointer-events-none data-[highlighted]:bg-zinc-700 data-[highlighted]:text-fuchsia-400 data-[disabled]:text-zinc-400">
|
||||||
View profile
|
View profile
|
||||||
</DropdownMenu.Item>
|
</DropdownMenu.Item>
|
||||||
|
<DropdownMenu.Item
|
||||||
|
onClick={() => updateProfile()}
|
||||||
|
className="group relative flex h-[30px] select-none items-center rounded px-1 pl-6 text-sm leading-none text-zinc-100 outline-none data-[disabled]:pointer-events-none data-[highlighted]:bg-zinc-700 data-[highlighted]:text-fuchsia-400 data-[disabled]:text-zinc-400">
|
||||||
|
Update profile
|
||||||
|
</DropdownMenu.Item>
|
||||||
<DropdownMenu.Item
|
<DropdownMenu.Item
|
||||||
onClick={() => copyPubkey()}
|
onClick={() => copyPubkey()}
|
||||||
className="group relative flex h-[30px] select-none items-center rounded px-1 pl-6 text-sm leading-none text-zinc-100 outline-none data-[disabled]:pointer-events-none data-[highlighted]:bg-zinc-700 data-[highlighted]:text-fuchsia-400 data-[disabled]:text-zinc-400">
|
className="group relative flex h-[30px] select-none items-center rounded px-1 pl-6 text-sm leading-none text-zinc-100 outline-none data-[disabled]:pointer-events-none data-[highlighted]:bg-zinc-700 data-[highlighted]:text-fuchsia-400 data-[disabled]:text-zinc-400">
|
||||||
|
21
src/layouts/newsfeedLayout.tsx
Normal file
21
src/layouts/newsfeedLayout.tsx
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
import AccountBar from '@components/accountBar';
|
||||||
|
import NavigatorBar from '@components/navigatorBar';
|
||||||
|
|
||||||
|
export default function NewsFeedLayout({ children }: { children: React.ReactNode }) {
|
||||||
|
return (
|
||||||
|
<div className="flex h-full w-full flex-row">
|
||||||
|
<div className="relative h-full w-[70px] shrink-0 border-r border-zinc-900">
|
||||||
|
<div data-tauri-drag-region className="absolute top-0 left-0 h-12 w-full" />
|
||||||
|
<AccountBar />
|
||||||
|
</div>
|
||||||
|
<div className="grid grow grid-cols-4">
|
||||||
|
<div className="col-span-1">
|
||||||
|
<NavigatorBar />
|
||||||
|
</div>
|
||||||
|
<div className="col-span-3 m-3 ml-0 overflow-hidden rounded-lg border border-zinc-800 bg-zinc-900 shadow-input shadow-black/20">
|
||||||
|
<div className="h-full w-full rounded-lg">{children}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
@@ -1,7 +1,14 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
import AccountBar from '@components/accountBar';
|
import AccountBar from '@components/accountBar';
|
||||||
import NavigatorBar from '@components/navigatorBar';
|
import ActiveLink from '@components/activeLink';
|
||||||
|
|
||||||
|
import { currentUser } from '@stores/currentUser';
|
||||||
|
|
||||||
|
import { useStore } from '@nanostores/react';
|
||||||
|
|
||||||
export default function UserLayout({ children }: { children: React.ReactNode }) {
|
export default function UserLayout({ children }: { children: React.ReactNode }) {
|
||||||
|
const $currentUser: any = useStore(currentUser);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex h-full w-full flex-row">
|
<div className="flex h-full w-full flex-row">
|
||||||
<div className="relative h-full w-[70px] shrink-0 border-r border-zinc-900">
|
<div className="relative h-full w-[70px] shrink-0 border-r border-zinc-900">
|
||||||
@@ -10,7 +17,31 @@ export default function UserLayout({ children }: { children: React.ReactNode })
|
|||||||
</div>
|
</div>
|
||||||
<div className="grid grow grid-cols-4">
|
<div className="grid grow grid-cols-4">
|
||||||
<div className="col-span-1">
|
<div className="col-span-1">
|
||||||
<NavigatorBar />
|
<div className="flex h-full flex-col flex-wrap justify-between overflow-hidden px-2 pt-3 pb-4">
|
||||||
|
{/* main */}
|
||||||
|
<div className="flex flex-col gap-4">
|
||||||
|
{/* menu */}
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
<div className="flex items-center justify-between px-2">
|
||||||
|
<h3 className="text-sm font-bold text-zinc-400">Menu</h3>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-1 text-zinc-500">
|
||||||
|
<ActiveLink
|
||||||
|
href={`/profile/${$currentUser.pubkey}`}
|
||||||
|
activeClassName="ring-1 ring-white/10 dark:bg-zinc-900 dark:text-white"
|
||||||
|
className="flex h-10 items-center gap-1 rounded-lg px-2.5 text-sm font-medium hover:bg-zinc-900">
|
||||||
|
<span>Personal Page</span>
|
||||||
|
</ActiveLink>
|
||||||
|
<ActiveLink
|
||||||
|
href={`/profile/update?pubkey=${$currentUser.pubkey}`}
|
||||||
|
activeClassName="ring-1 ring-white/10 dark:bg-zinc-900 dark:text-white"
|
||||||
|
className="flex h-10 items-center gap-1 rounded-lg px-2.5 text-sm font-medium hover:bg-zinc-900">
|
||||||
|
<span>Update Profile</span>
|
||||||
|
</ActiveLink>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="col-span-3 m-3 ml-0 overflow-hidden rounded-lg border border-zinc-800 bg-zinc-900 shadow-input shadow-black/20">
|
<div className="col-span-3 m-3 ml-0 overflow-hidden rounded-lg border border-zinc-800 bg-zinc-900 shadow-input shadow-black/20">
|
||||||
<div className="h-full w-full rounded-lg">{children}</div>
|
<div className="h-full w-full rounded-lg">{children}</div>
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import BaseLayout from '@layouts/baseLayout';
|
import BaseLayout from '@layouts/baseLayout';
|
||||||
import UserLayout from '@layouts/userLayout';
|
import NewsFeedLayout from '@layouts/newsfeedLayout';
|
||||||
|
|
||||||
import { Placeholder } from '@components/note/placeholder';
|
import { Placeholder } from '@components/note/placeholder';
|
||||||
import { Thread } from '@components/thread';
|
import { Thread } from '@components/thread';
|
||||||
@@ -52,7 +52,7 @@ Page.getLayout = function getLayout(
|
|||||||
) {
|
) {
|
||||||
return (
|
return (
|
||||||
<BaseLayout>
|
<BaseLayout>
|
||||||
<UserLayout>{page}</UserLayout>
|
<NewsFeedLayout>{page}</NewsFeedLayout>
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import BaseLayout from '@layouts/baseLayout';
|
import BaseLayout from '@layouts/baseLayout';
|
||||||
import UserLayout from '@layouts/userLayout';
|
import NewsFeedLayout from '@layouts/newsfeedLayout';
|
||||||
|
|
||||||
import { Placeholder } from '@components/note/placeholder';
|
import { Placeholder } from '@components/note/placeholder';
|
||||||
import { Thread } from '@components/thread';
|
import { Thread } from '@components/thread';
|
||||||
@@ -48,7 +48,7 @@ Page.getLayout = function getLayout(
|
|||||||
) {
|
) {
|
||||||
return (
|
return (
|
||||||
<BaseLayout>
|
<BaseLayout>
|
||||||
<UserLayout>{page}</UserLayout>
|
<NewsFeedLayout>{page}</NewsFeedLayout>
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
244
src/pages/profile/update.tsx
Normal file
244
src/pages/profile/update.tsx
Normal file
@@ -0,0 +1,244 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
|
import BaseLayout from '@layouts/baseLayout';
|
||||||
|
import UserLayout from '@layouts/userLayout';
|
||||||
|
|
||||||
|
import { currentUser } from '@stores/currentUser';
|
||||||
|
|
||||||
|
import { useStore } from '@nanostores/react';
|
||||||
|
import { useRouter } from 'next/router';
|
||||||
|
import { dateToUnix, useNostr } from 'nostr-react';
|
||||||
|
import { getEventHash, signEvent } from 'nostr-tools';
|
||||||
|
import { JSXElementConstructor, ReactElement, ReactFragment, ReactPortal, useState } from 'react';
|
||||||
|
import { useForm } from 'react-hook-form';
|
||||||
|
import Database from 'tauri-plugin-sql-api';
|
||||||
|
|
||||||
|
type FormValues = {
|
||||||
|
display_name: string;
|
||||||
|
name: string;
|
||||||
|
username: string;
|
||||||
|
picture: string;
|
||||||
|
banner: string;
|
||||||
|
about: string;
|
||||||
|
website: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
// TODO: update the design
|
||||||
|
export default function Page() {
|
||||||
|
const router = useRouter();
|
||||||
|
const { publish } = useNostr();
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
|
const $currentUser: any = useStore(currentUser);
|
||||||
|
const profile = JSON.parse($currentUser.metadata);
|
||||||
|
|
||||||
|
const {
|
||||||
|
register,
|
||||||
|
handleSubmit,
|
||||||
|
formState: { errors, isDirty, isValid },
|
||||||
|
} = useForm<FormValues>();
|
||||||
|
|
||||||
|
const onSubmit = async (data: any) => {
|
||||||
|
setLoading(true);
|
||||||
|
|
||||||
|
// publish account to relays
|
||||||
|
const event: any = {
|
||||||
|
content: JSON.stringify(data),
|
||||||
|
created_at: dateToUnix(),
|
||||||
|
kind: 0,
|
||||||
|
pubkey: $currentUser.pubkey,
|
||||||
|
tags: [],
|
||||||
|
};
|
||||||
|
event.id = getEventHash(event);
|
||||||
|
event.sig = signEvent(event, $currentUser.privkey);
|
||||||
|
publish(event);
|
||||||
|
|
||||||
|
// save account to database
|
||||||
|
const db = await Database.load('sqlite:lume.db');
|
||||||
|
await db.execute(
|
||||||
|
`UPDATE accounts SET metadata = '${JSON.stringify(data)}' WHERE pubkey = "${
|
||||||
|
$currentUser.pubkey
|
||||||
|
}"`
|
||||||
|
);
|
||||||
|
await db.close();
|
||||||
|
|
||||||
|
// set currentUser in global state
|
||||||
|
currentUser.set({
|
||||||
|
metadata: JSON.stringify(data),
|
||||||
|
npub: $currentUser.npub,
|
||||||
|
privkey: $currentUser.privkey,
|
||||||
|
pubkey: $currentUser.pubkey,
|
||||||
|
});
|
||||||
|
|
||||||
|
// redirect to newsfeed
|
||||||
|
setTimeout(() => {
|
||||||
|
setLoading(false);
|
||||||
|
router.push('/feed/following');
|
||||||
|
}, 1500);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<form
|
||||||
|
onSubmit={handleSubmit(onSubmit)}
|
||||||
|
className="flex h-full w-full flex-col justify-between px-6">
|
||||||
|
<div className="mb-8 flex flex-col gap-3 pt-8">
|
||||||
|
<h1 className="bg-gradient-to-br from-zinc-200 to-zinc-400 bg-clip-text text-3xl font-medium text-transparent">
|
||||||
|
Update profile
|
||||||
|
</h1>
|
||||||
|
<h2 className="w-3/4 text-zinc-400">
|
||||||
|
Your profile will be published to all relays, as long as you have the private key, you
|
||||||
|
always can recover your profile in any client
|
||||||
|
</h2>
|
||||||
|
</div>
|
||||||
|
<fieldset className="flex flex-col gap-2">
|
||||||
|
<div className="grid grid-cols-4">
|
||||||
|
<div className="col-span-1">
|
||||||
|
<label className="text-zinc-300">Display Name</label>
|
||||||
|
</div>
|
||||||
|
<div className="col-span-3 flex flex-col gap-2">
|
||||||
|
<div className="relative shrink-0 before:pointer-events-none before:absolute before:-inset-1 before:rounded-[11px] before:border before:border-blue-500 before:opacity-0 before:ring-2 before:ring-blue-500/20 before:transition after:pointer-events-none after:absolute after:inset-px after:rounded-[7px] after:shadow-highlight after:shadow-white/5 after:transition focus-within:before:opacity-100 focus-within:after:shadow-blue-500/100 dark:focus-within:after:shadow-blue-500/20">
|
||||||
|
<input
|
||||||
|
{...register('display_name')}
|
||||||
|
defaultValue={profile.display_name || ''}
|
||||||
|
className="relative w-full rounded-lg border border-black/5 px-3.5 py-2 shadow-input shadow-black/5 !outline-none placeholder:text-zinc-400 dark:bg-zinc-800 dark:text-zinc-200 dark:shadow-black/10 dark:placeholder:text-zinc-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="text-sm text-red-400">
|
||||||
|
{errors.display_name && <p>{errors.display_name.message}</p>}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-4">
|
||||||
|
<div className="col-span-1">
|
||||||
|
<label className="text-zinc-300">Name</label>
|
||||||
|
</div>
|
||||||
|
<div className="col-span-3 flex flex-col gap-2">
|
||||||
|
<div className="relative shrink-0 before:pointer-events-none before:absolute before:-inset-1 before:rounded-[11px] before:border before:border-blue-500 before:opacity-0 before:ring-2 before:ring-blue-500/20 before:transition after:pointer-events-none after:absolute after:inset-px after:rounded-[7px] after:shadow-highlight after:shadow-white/5 after:transition focus-within:before:opacity-100 focus-within:after:shadow-blue-500/100 dark:focus-within:after:shadow-blue-500/20">
|
||||||
|
<input
|
||||||
|
{...register('name')}
|
||||||
|
defaultValue={profile.name || ''}
|
||||||
|
className="relative w-full rounded-lg border border-black/5 px-3.5 py-2 shadow-input shadow-black/5 !outline-none placeholder:text-zinc-400 dark:bg-zinc-800 dark:text-zinc-200 dark:shadow-black/10 dark:placeholder:text-zinc-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="text-sm text-red-400">
|
||||||
|
{errors.name && <p>{errors.name.message}</p>}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-4">
|
||||||
|
<div className="col-span-1">
|
||||||
|
<label className="text-zinc-300">Username</label>
|
||||||
|
</div>
|
||||||
|
<div className="col-span-3 flex flex-col gap-2">
|
||||||
|
<div className="relative shrink-0 before:pointer-events-none before:absolute before:-inset-1 before:rounded-[11px] before:border before:border-blue-500 before:opacity-0 before:ring-2 before:ring-blue-500/20 before:transition after:pointer-events-none after:absolute after:inset-px after:rounded-[7px] after:shadow-highlight after:shadow-white/5 after:transition focus-within:before:opacity-100 focus-within:after:shadow-blue-500/100 dark:focus-within:after:shadow-blue-500/20">
|
||||||
|
<input
|
||||||
|
{...register('username')}
|
||||||
|
defaultValue={profile.username || ''}
|
||||||
|
className="relative w-full rounded-lg border border-black/5 px-3.5 py-2 shadow-input shadow-black/5 !outline-none placeholder:text-zinc-400 dark:bg-zinc-800 dark:text-zinc-200 dark:shadow-black/10 dark:placeholder:text-zinc-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="text-sm text-red-400">
|
||||||
|
{errors.username && <p>{errors.username.message}</p>}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-4">
|
||||||
|
<div className="col-span-1">
|
||||||
|
<label className="text-zinc-300">Profile Picture</label>
|
||||||
|
</div>
|
||||||
|
<div className="col-span-3 flex flex-col gap-2">
|
||||||
|
<div className="relative shrink-0 before:pointer-events-none before:absolute before:-inset-1 before:rounded-[11px] before:border before:border-blue-500 before:opacity-0 before:ring-2 before:ring-blue-500/20 before:transition after:pointer-events-none after:absolute after:inset-px after:rounded-[7px] after:shadow-highlight after:shadow-white/5 after:transition focus-within:before:opacity-100 focus-within:after:shadow-blue-500/100 dark:focus-within:after:shadow-blue-500/20">
|
||||||
|
<input
|
||||||
|
{...register('picture')}
|
||||||
|
defaultValue={profile.picture || ''}
|
||||||
|
className="relative w-full rounded-lg border border-black/5 px-3.5 py-2 shadow-input shadow-black/5 !outline-none placeholder:text-zinc-400 dark:bg-zinc-800 dark:text-zinc-200 dark:shadow-black/10 dark:placeholder:text-zinc-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="text-sm text-red-400">
|
||||||
|
{errors.picture && <p>{errors.picture.message}</p>}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-4">
|
||||||
|
<div className="col-span-1">
|
||||||
|
<label className="text-zinc-300">Banner</label>
|
||||||
|
</div>
|
||||||
|
<div className="col-span-3 flex flex-col gap-2">
|
||||||
|
<div className="relative shrink-0 before:pointer-events-none before:absolute before:-inset-1 before:rounded-[11px] before:border before:border-blue-500 before:opacity-0 before:ring-2 before:ring-blue-500/20 before:transition after:pointer-events-none after:absolute after:inset-px after:rounded-[7px] after:shadow-highlight after:shadow-white/5 after:transition focus-within:before:opacity-100 focus-within:after:shadow-blue-500/100 dark:focus-within:after:shadow-blue-500/20">
|
||||||
|
<input
|
||||||
|
{...register('banner')}
|
||||||
|
defaultValue={profile.banner || ''}
|
||||||
|
className="relative w-full rounded-lg border border-black/5 px-3.5 py-2 shadow-input shadow-black/5 !outline-none placeholder:text-zinc-400 dark:bg-zinc-800 dark:text-zinc-200 dark:shadow-black/10 dark:placeholder:text-zinc-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="text-sm text-red-400">
|
||||||
|
{errors.banner && <p>{errors.banner.message}</p>}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-4">
|
||||||
|
<div className="col-span-1">
|
||||||
|
<label className="text-zinc-300">About</label>
|
||||||
|
</div>
|
||||||
|
<div className="col-span-3 flex flex-col gap-2">
|
||||||
|
<div className="relative h-24 shrink-0 before:pointer-events-none before:absolute before:-inset-1 before:rounded-[11px] before:border before:border-blue-500 before:opacity-0 before:ring-2 before:ring-blue-500/20 before:transition after:pointer-events-none after:absolute after:inset-px after:rounded-[7px] after:shadow-highlight after:shadow-white/5 after:transition focus-within:before:opacity-100 focus-within:after:shadow-blue-500/100 dark:focus-within:after:shadow-blue-500/20">
|
||||||
|
<textarea
|
||||||
|
{...register('about')}
|
||||||
|
defaultValue={profile.about || ''}
|
||||||
|
className="relative h-24 w-full resize-none rounded-lg border border-black/5 px-3.5 py-2 shadow-input shadow-black/5 !outline-none placeholder:text-zinc-400 dark:bg-zinc-800 dark:text-zinc-200 dark:shadow-black/10 dark:placeholder:text-zinc-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="text-sm text-red-400">
|
||||||
|
{errors.about && <p>{errors.about.message}</p>}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</fieldset>
|
||||||
|
<div className="pb-5">
|
||||||
|
<div className="flex h-10 items-center">
|
||||||
|
{loading === true ? (
|
||||||
|
<svg
|
||||||
|
className="h-5 w-5 animate-spin text-white"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
fill="none"
|
||||||
|
viewBox="0 0 24 24">
|
||||||
|
<circle
|
||||||
|
className="opacity-25"
|
||||||
|
cx="12"
|
||||||
|
cy="12"
|
||||||
|
r="10"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth="4"></circle>
|
||||||
|
<path
|
||||||
|
className="opacity-75"
|
||||||
|
fill="currentColor"
|
||||||
|
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||||||
|
</svg>
|
||||||
|
) : (
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
disabled={!isDirty || !isValid}
|
||||||
|
className="transform rounded-lg bg-[radial-gradient(ellipse_at_bottom_right,_var(--tw-gradient-stops))] from-gray-300 via-fuchsia-600 to-orange-600 px-3.5 py-2 font-medium active:translate-y-1 disabled:cursor-not-allowed disabled:opacity-30">
|
||||||
|
<span className="drop-shadow-lg">Update</span>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Page.getLayout = function getLayout(
|
||||||
|
page:
|
||||||
|
| string
|
||||||
|
| number
|
||||||
|
| boolean
|
||||||
|
| ReactElement<unknown, string | JSXElementConstructor<unknown>>
|
||||||
|
| ReactFragment
|
||||||
|
| ReactPortal
|
||||||
|
) {
|
||||||
|
return (
|
||||||
|
<BaseLayout>
|
||||||
|
<UserLayout>{page}</UserLayout>
|
||||||
|
</BaseLayout>
|
||||||
|
);
|
||||||
|
};
|
Reference in New Issue
Block a user