Compare commits
15 commits
bb2006ded7
...
d1e6f0339b
Author | SHA1 | Date | |
---|---|---|---|
d1e6f0339b | |||
72a5c25838 | |||
171f0ae099 | |||
a351a9017d | |||
15fbf27459 | |||
9183117a20 | |||
386d72d914 | |||
6c479e80d6 | |||
abae5c74d5 | |||
3569ccc18e | |||
d4de7876cc | |||
0c93778c5a | |||
4b80c89050 | |||
16cde64761 | |||
a2a5eee49e |
13 changed files with 270 additions and 37 deletions
17
Dockerfile.dev
Normal file
17
Dockerfile.dev
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
FROM node:22-alpine@sha256:41e4389f3d988d2ed55392df4db1420ad048ae53324a8e2b7c6d19508288107e
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
RUN corepack enable
|
||||||
|
COPY package.json yarn.lock .yarnrc.yml ./
|
||||||
|
RUN yarn install --frozen-lockfile
|
||||||
|
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
ENV NODE_ENV=development
|
||||||
|
ENV NEXT_TELEMETRY_DISABLED=1
|
||||||
|
|
||||||
|
EXPOSE 3000
|
||||||
|
ENV HOSTNAME="0.0.0.0"
|
||||||
|
|
||||||
|
CMD ["/bin/ash", "entrypoint.dev.sh"]
|
10
README.md
10
README.md
|
@ -104,7 +104,7 @@ This project is built with a modern tech stack:
|
||||||
yarn prisma:generate
|
yarn prisma:generate
|
||||||
```
|
```
|
||||||
```bash
|
```bash
|
||||||
yarn prisa:db:push
|
yarn prisma:db:push
|
||||||
```
|
```
|
||||||
- Run the following command to apply migrations and generate Prisma Client:
|
- Run the following command to apply migrations and generate Prisma Client:
|
||||||
```bash
|
```bash
|
||||||
|
@ -129,6 +129,14 @@ This project is built with a modern tech stack:
|
||||||
password: password
|
password: password
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**Docker Development Environment:**
|
||||||
|
|
||||||
|
- The docker development environment can be started with the following command:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn dev_container
|
||||||
|
```
|
||||||
|
|
||||||
**Self-Hosting with Docker (Planned):**
|
**Self-Hosting with Docker (Planned):**
|
||||||
|
|
||||||
- A Docker image and `docker-compose.yml` file will be provided in the future to allow for easy self-hosting of the MeetUP application. This setup will also include database services. Instructions will be updated here once available.
|
- A Docker image and `docker-compose.yml` file will be provided in the future to allow for easy self-hosting of the MeetUP application. This setup will also include database services. Instructions will be updated here once available.
|
||||||
|
|
27
docker-compose.dev.yml
Normal file
27
docker-compose.dev.yml
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
services:
|
||||||
|
app:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: Dockerfile.dev
|
||||||
|
ports:
|
||||||
|
- '3000:3000'
|
||||||
|
environment:
|
||||||
|
- AUTH_SECRET=secret
|
||||||
|
- AUTH_URL=http://localhost:3000
|
||||||
|
- DATABASE_URL=file:/data/db.sqlite
|
||||||
|
env_file:
|
||||||
|
- .env.local
|
||||||
|
volumes:
|
||||||
|
- ./data:/data
|
||||||
|
develop:
|
||||||
|
watch:
|
||||||
|
- action: sync
|
||||||
|
path: ./src
|
||||||
|
target: /app/src
|
||||||
|
ignore:
|
||||||
|
- node_modules/
|
||||||
|
- action: rebuild
|
||||||
|
path: package.json
|
||||||
|
- action: sync+restart
|
||||||
|
path: prisma
|
||||||
|
target: /app/prisma
|
10
entrypoint.dev.sh
Normal file
10
entrypoint.dev.sh
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
echo "Running start script with user $(whoami) and NODE_ENV $NODE_ENV"
|
||||||
|
if [ -d "prisma" ]; then
|
||||||
|
echo "Syncing Prisma database"
|
||||||
|
yarn prisma:generate
|
||||||
|
yarn prisma:db:push
|
||||||
|
fi
|
||||||
|
|
||||||
|
exec yarn dev
|
|
@ -12,7 +12,8 @@
|
||||||
"prisma:generate": "dotenv -e .env.local -- prisma generate",
|
"prisma:generate": "dotenv -e .env.local -- prisma generate",
|
||||||
"prisma:studio": "dotenv -e .env.local -- prisma studio",
|
"prisma:studio": "dotenv -e .env.local -- prisma studio",
|
||||||
"prisma:db:push": "dotenv -e .env.local -- prisma db push",
|
"prisma:db:push": "dotenv -e .env.local -- prisma db push",
|
||||||
"prisma:migrate:reset": "dotenv -e .env.local -- prisma migrate reset"
|
"prisma:migrate:reset": "dotenv -e .env.local -- prisma migrate reset",
|
||||||
|
"dev_container": "docker compose -f docker-compose.dev.yml up --watch --build"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@auth/prisma-adapter": "^2.9.1",
|
"@auth/prisma-adapter": "^2.9.1",
|
||||||
|
@ -48,8 +49,8 @@
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@eslint/eslintrc": "3.3.1",
|
"@eslint/eslintrc": "3.3.1",
|
||||||
"@tailwindcss/postcss": "4.1.8",
|
"@tailwindcss/postcss": "4.1.8",
|
||||||
"@types/node": "22.15.30",
|
"@types/node": "22.15.31",
|
||||||
"@types/react": "19.1.6",
|
"@types/react": "19.1.7",
|
||||||
"@types/react-dom": "19.1.6",
|
"@types/react-dom": "19.1.6",
|
||||||
"dotenv-cli": "8.0.0",
|
"dotenv-cli": "8.0.0",
|
||||||
"eslint": "9.28.0",
|
"eslint": "9.28.0",
|
||||||
|
|
|
@ -26,12 +26,12 @@ export default async function LoginPage() {
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='flex flex-col items-center justify-center h-screen'>
|
<div className='flex flex-col items-center min-h-screen'>
|
||||||
<div className='flex flex-col items-center justify-center h-screen'>
|
<div className='flex flex-col items-center min-h-screen'>
|
||||||
<div className='absolute top-4 right-4'>
|
<div className='fixed top-4 right-4'>
|
||||||
<ThemePicker />
|
<ThemePicker />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div className='mt-auto mb-auto'>
|
||||||
<Card className='w-[350px] max-w-screen;'>
|
<Card className='w-[350px] max-w-screen;'>
|
||||||
<CardHeader className='grid place-items-center'>
|
<CardHeader className='grid place-items-center'>
|
||||||
<Logo colorType='colored' logoType='secondary'></Logo>
|
<Logo colorType='colored' logoType='secondary'></Logo>
|
||||||
|
|
|
@ -25,11 +25,7 @@ export default function SignOutPage() {
|
||||||
</CardDescription>
|
</CardDescription>
|
||||||
</CardHeader>
|
</CardHeader>
|
||||||
<CardContent className='gap-6 flex flex-col'>
|
<CardContent className='gap-6 flex flex-col'>
|
||||||
<Button
|
<Button type='submit' variant='secondary'>
|
||||||
className='hover:bg-blue-600 hover:text-white'
|
|
||||||
type='submit'
|
|
||||||
variant='secondary'
|
|
||||||
>
|
|
||||||
Logout
|
Logout
|
||||||
</Button>
|
</Button>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { cva, type VariantProps } from 'class-variance-authority';
|
||||||
import { cn } from '@/lib/utils';
|
import { cn } from '@/lib/utils';
|
||||||
|
|
||||||
const buttonVariants = cva(
|
const buttonVariants = cva(
|
||||||
"radius-lg inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-button transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
|
"radius-lg inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-button transition-all cursor-pointer disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
|
||||||
{
|
{
|
||||||
variants: {
|
variants: {
|
||||||
variant: {
|
variant: {
|
||||||
|
|
|
@ -7,6 +7,7 @@ export default function LabeledInput({
|
||||||
placeholder,
|
placeholder,
|
||||||
value,
|
value,
|
||||||
name,
|
name,
|
||||||
|
autocomplete,
|
||||||
error,
|
error,
|
||||||
...rest
|
...rest
|
||||||
}: {
|
}: {
|
||||||
|
@ -15,6 +16,7 @@ export default function LabeledInput({
|
||||||
placeholder?: string;
|
placeholder?: string;
|
||||||
value?: string;
|
value?: string;
|
||||||
name?: string;
|
name?: string;
|
||||||
|
autocomplete?: string;
|
||||||
error?: string;
|
error?: string;
|
||||||
} & React.InputHTMLAttributes<HTMLInputElement>) {
|
} & React.InputHTMLAttributes<HTMLInputElement>) {
|
||||||
return (
|
return (
|
||||||
|
@ -27,6 +29,7 @@ export default function LabeledInput({
|
||||||
defaultValue={value}
|
defaultValue={value}
|
||||||
id={name}
|
id={name}
|
||||||
name={name}
|
name={name}
|
||||||
|
autoComplete={autocomplete}
|
||||||
{...rest}
|
{...rest}
|
||||||
/>
|
/>
|
||||||
{error && <p className='text-red-500 text-sm mt-1'>{error}</p>}
|
{error && <p className='text-red-500 text-sm mt-1'>{error}</p>}
|
||||||
|
|
|
@ -1,14 +1,22 @@
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
|
import React, { useState, useRef } from 'react';
|
||||||
|
import { useRouter } from 'next/navigation';
|
||||||
|
|
||||||
import LabeledInput from '@/components/labeled-input';
|
import LabeledInput from '@/components/labeled-input';
|
||||||
import { Button } from '@/components/custom-ui/button';
|
import { Button } from '@/components/custom-ui/button';
|
||||||
|
|
||||||
import useZodForm from '@/hooks/useZodForm';
|
import useZodForm from '@/hooks/useZodForm';
|
||||||
import { loginSchema } from '@/validation';
|
import { loginSchema, registerSchema } from '@/validation';
|
||||||
import { loginAction } from '@/login';
|
import { loginAction } from '@/login';
|
||||||
import { useRouter } from 'next/navigation';
|
import { registerAction } from '@/register';
|
||||||
|
|
||||||
export default function LoginForm() {
|
function LoginFormElement({
|
||||||
|
setIsSignUp,
|
||||||
|
formRef,
|
||||||
|
}: {
|
||||||
|
setIsSignUp: (value: boolean | ((prev: boolean) => boolean)) => void;
|
||||||
|
formRef?: React.RefObject<HTMLFormElement | null>;
|
||||||
|
}) {
|
||||||
const { handleSubmit, formState, register, setError } =
|
const { handleSubmit, formState, register, setError } =
|
||||||
useZodForm(loginSchema);
|
useZodForm(loginSchema);
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
@ -40,7 +48,7 @@ export default function LoginForm() {
|
||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<form onSubmit={onSubmit} className='flex flex-col gap-5 w-full'>
|
<form className='flex flex-col gap-5 w-full' onSubmit={onSubmit}>
|
||||||
<LabeledInput
|
<LabeledInput
|
||||||
type='email'
|
type='email'
|
||||||
label='E-Mail or Username'
|
label='E-Mail or Username'
|
||||||
|
@ -59,7 +67,14 @@ export default function LoginForm() {
|
||||||
<Button type='submit' variant='primary'>
|
<Button type='submit' variant='primary'>
|
||||||
Login
|
Login
|
||||||
</Button>
|
</Button>
|
||||||
<Button type='submit' variant='outline_primary'>
|
<Button
|
||||||
|
type='button'
|
||||||
|
variant='outline_primary'
|
||||||
|
onClick={() => {
|
||||||
|
formRef?.current?.reset();
|
||||||
|
setIsSignUp((v) => !v);
|
||||||
|
}}
|
||||||
|
>
|
||||||
Sign Up
|
Sign Up
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
@ -71,3 +86,139 @@ export default function LoginForm() {
|
||||||
</form>
|
</form>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function RegisterFormElement({
|
||||||
|
setIsSignUp,
|
||||||
|
formRef,
|
||||||
|
}: {
|
||||||
|
setIsSignUp: (value: boolean | ((prev: boolean) => boolean)) => void;
|
||||||
|
formRef?: React.RefObject<HTMLFormElement | null>;
|
||||||
|
}) {
|
||||||
|
const { handleSubmit, formState, register, setError } =
|
||||||
|
useZodForm(registerSchema);
|
||||||
|
|
||||||
|
const onSubmit = handleSubmit(async (data) => {
|
||||||
|
try {
|
||||||
|
const { error } = await registerAction(data);
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
setError('root', {
|
||||||
|
message: error,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
formRef?.current?.reset();
|
||||||
|
setIsSignUp(false);
|
||||||
|
// TODO: Show registration success message (reminder to verify email)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} catch (error: unknown) {
|
||||||
|
if (error instanceof Error)
|
||||||
|
setError('root', {
|
||||||
|
message: error?.message,
|
||||||
|
});
|
||||||
|
else
|
||||||
|
setError('root', {
|
||||||
|
message: 'An unknown error occurred.',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<form
|
||||||
|
ref={formRef}
|
||||||
|
className='flex flex-col gap-5 w-full'
|
||||||
|
onSubmit={onSubmit}
|
||||||
|
>
|
||||||
|
<LabeledInput
|
||||||
|
type='text'
|
||||||
|
label='First Name'
|
||||||
|
placeholder='Your first name'
|
||||||
|
autocomplete='given-name'
|
||||||
|
error={formState.errors.firstName?.message}
|
||||||
|
{...register('firstName')}
|
||||||
|
/>
|
||||||
|
<LabeledInput
|
||||||
|
type='text'
|
||||||
|
label='Last Name'
|
||||||
|
placeholder='Your last name'
|
||||||
|
autocomplete='family-name'
|
||||||
|
error={formState.errors.lastName?.message}
|
||||||
|
{...register('lastName')}
|
||||||
|
/>
|
||||||
|
<LabeledInput
|
||||||
|
type='email'
|
||||||
|
label='E-Mail'
|
||||||
|
placeholder='Your email address'
|
||||||
|
autocomplete='email'
|
||||||
|
error={formState.errors.email?.message}
|
||||||
|
{...register('email')}
|
||||||
|
/>
|
||||||
|
<LabeledInput
|
||||||
|
type='text'
|
||||||
|
label='Username'
|
||||||
|
placeholder='Your username'
|
||||||
|
autocomplete='username'
|
||||||
|
error={formState.errors.username?.message}
|
||||||
|
{...register('username')}
|
||||||
|
/>
|
||||||
|
<LabeledInput
|
||||||
|
type='password'
|
||||||
|
label='Password'
|
||||||
|
placeholder='Create a password'
|
||||||
|
autocomplete='new-password'
|
||||||
|
error={formState.errors.password?.message}
|
||||||
|
{...register('password')}
|
||||||
|
/>
|
||||||
|
<LabeledInput
|
||||||
|
type='password'
|
||||||
|
label='Confirm Password'
|
||||||
|
placeholder='Repeat your password'
|
||||||
|
autocomplete='new-password'
|
||||||
|
error={formState.errors.confirmPassword?.message}
|
||||||
|
{...register('confirmPassword')}
|
||||||
|
/>
|
||||||
|
<div className='grid grid-rows-2 gap-2'>
|
||||||
|
<Button type='submit' variant='primary'>
|
||||||
|
Sign Up
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
type='button'
|
||||||
|
variant='outline_primary'
|
||||||
|
onClick={() => {
|
||||||
|
formRef?.current?.reset();
|
||||||
|
setIsSignUp((v) => !v);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Back to Login
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
{formState.errors.root?.message && (
|
||||||
|
<p className='text-red-500'>{formState.errors.root?.message}</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function LoginForm() {
|
||||||
|
const [isSignUp, setIsSignUp] = useState(false);
|
||||||
|
|
||||||
|
const formRef = useRef<HTMLFormElement>(null);
|
||||||
|
|
||||||
|
if (isSignUp) {
|
||||||
|
return (
|
||||||
|
<RegisterFormElement
|
||||||
|
setIsSignUp={setIsSignUp}
|
||||||
|
formRef={formRef}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<LoginFormElement
|
||||||
|
setIsSignUp={setIsSignUp}
|
||||||
|
formRef={formRef}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
|
@ -15,7 +15,7 @@ export async function registerAction(data: z.infer<typeof registerSchema>) {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const { email, password, username } = result.data;
|
const { email, password, firstName, lastName, username } = result.data;
|
||||||
|
|
||||||
const user = await prisma.user.findUnique({
|
const user = await prisma.user.findUnique({
|
||||||
where: {
|
where: {
|
||||||
|
@ -37,6 +37,9 @@ export async function registerAction(data: z.infer<typeof registerSchema>) {
|
||||||
email,
|
email,
|
||||||
name: username,
|
name: username,
|
||||||
password_hash: passwordHash,
|
password_hash: passwordHash,
|
||||||
|
first_name: firstName,
|
||||||
|
last_name: lastName,
|
||||||
|
emailVerified: new Date(), // TODO: handle email verification
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,14 @@ export const loginSchema = zod.object({
|
||||||
|
|
||||||
export const registerSchema = zod
|
export const registerSchema = zod
|
||||||
.object({
|
.object({
|
||||||
|
firstName: zod
|
||||||
|
.string()
|
||||||
|
.min(1, 'First name is required')
|
||||||
|
.max(32, 'First name must be at most 32 characters long'),
|
||||||
|
lastName: zod
|
||||||
|
.string()
|
||||||
|
.min(1, 'Last name is required')
|
||||||
|
.max(32, 'Last name must be at most 32 characters long'),
|
||||||
email: zod
|
email: zod
|
||||||
.string()
|
.string()
|
||||||
.email('Invalid email address')
|
.email('Invalid email address')
|
||||||
|
@ -35,6 +43,15 @@ export const registerSchema = zod
|
||||||
message: 'Passwords do not match',
|
message: 'Passwords do not match',
|
||||||
path: ['confirmPassword'],
|
path: ['confirmPassword'],
|
||||||
})
|
})
|
||||||
.refine((data) => !data.password.includes(data.username), {
|
.refine(
|
||||||
message: 'Password cannot contain username',
|
(data) =>
|
||||||
});
|
!data.password.includes(data.firstName) &&
|
||||||
|
!data.password.includes(data.lastName) &&
|
||||||
|
!data.password.includes(data.email) &&
|
||||||
|
!data.password.includes(data.username),
|
||||||
|
{
|
||||||
|
message:
|
||||||
|
'Username cannot contain your first name, last name, email, or username itself',
|
||||||
|
path: ['password'],
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
26
yarn.lock
26
yarn.lock
|
@ -1659,12 +1659,12 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@types/node@npm:22.15.30":
|
"@types/node@npm:22.15.31":
|
||||||
version: 22.15.30
|
version: 22.15.31
|
||||||
resolution: "@types/node@npm:22.15.30"
|
resolution: "@types/node@npm:22.15.31"
|
||||||
dependencies:
|
dependencies:
|
||||||
undici-types: "npm:~6.21.0"
|
undici-types: "npm:~6.21.0"
|
||||||
checksum: 10c0/ca330ac0e7fd502686d6df115fcc606aba46fd334220f749bbba2f639accdadcb23f7900603ceccdc8240be736739cad5c0b87c0fa92c9255a4dff245f07d664
|
checksum: 10c0/ef7d5dc890da41cfd554d35ab8998bc18be9e3a0caa642e720599ac4410a94a4879766e52b3c9cafa06c66b7b8aebdc51f322cf67df23a6489927890196a316d
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
@ -1677,12 +1677,12 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@types/react@npm:19.1.6":
|
"@types/react@npm:19.1.7":
|
||||||
version: 19.1.6
|
version: 19.1.7
|
||||||
resolution: "@types/react@npm:19.1.6"
|
resolution: "@types/react@npm:19.1.7"
|
||||||
dependencies:
|
dependencies:
|
||||||
csstype: "npm:^3.0.2"
|
csstype: "npm:^3.0.2"
|
||||||
checksum: 10c0/8b10b198e28997b3c57559750f8bcf5ae7b33c554b16b6f4fe2ece1d4de6a2fc8cb53e7effe08ec9cb939d2f479eb97c5e08aac2cf83b10a90164fe451cc8ea2
|
checksum: 10c0/3bb8fb865debad4328b0d623e1c669f2ee90e9302638a64e65a0a1c61efca4f4ef91f58b55ff94075358c190d80bb8472a5823c6901d8cdc9009dd436a1dcab1
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
@ -3890,8 +3890,8 @@ __metadata:
|
||||||
"@radix-ui/react-switch": "npm:^1.2.4"
|
"@radix-ui/react-switch": "npm:^1.2.4"
|
||||||
"@radix-ui/react-tabs": "npm:^1.1.11"
|
"@radix-ui/react-tabs": "npm:^1.1.11"
|
||||||
"@tailwindcss/postcss": "npm:4.1.8"
|
"@tailwindcss/postcss": "npm:4.1.8"
|
||||||
"@types/node": "npm:22.15.30"
|
"@types/node": "npm:22.15.31"
|
||||||
"@types/react": "npm:19.1.6"
|
"@types/react": "npm:19.1.7"
|
||||||
"@types/react-dom": "npm:19.1.6"
|
"@types/react-dom": "npm:19.1.6"
|
||||||
bcryptjs: "npm:^3.0.2"
|
bcryptjs: "npm:^3.0.2"
|
||||||
class-variance-authority: "npm:^0.7.1"
|
class-variance-authority: "npm:^0.7.1"
|
||||||
|
@ -4995,9 +4995,9 @@ __metadata:
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"tailwind-merge@npm:^3.2.0":
|
"tailwind-merge@npm:^3.2.0":
|
||||||
version: 3.3.0
|
version: 3.3.1
|
||||||
resolution: "tailwind-merge@npm:3.3.0"
|
resolution: "tailwind-merge@npm:3.3.1"
|
||||||
checksum: 10c0/a50cd141100486f98541dfab3705712af5860556689b7496dc6b0284374f02d12d5471f0f40035f6bb8b1c749c422060a1f3e5f8900057d8a7786b111c8472e6
|
checksum: 10c0/b84c6a78d4669fa12bf5ab8f0cdc4400a3ce0a7c006511af4af4be70bb664a27466dbe13ee9e3b31f50ddf6c51d380e8192ce0ec9effce23ca729d71a9f63818
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue