Merge branch 'fe/feat/74-workspace-create-modal' into 'fe/develop'
Feat: WorkSpaceCreateModal 컴포넌트 구현 - S11P21S002-74 See merge request s11-s-project/S11P21S002!8
This commit is contained in:
commit
9d985c671f
@ -1,33 +0,0 @@
|
|||||||
import * as React from 'react';
|
|
||||||
import { cn } from '@/lib/utils';
|
|
||||||
|
|
||||||
interface ButtonProps {
|
|
||||||
isActive: boolean;
|
|
||||||
text: string;
|
|
||||||
onClick: () => void;
|
|
||||||
progress?: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function Button({ isActive, text, onClick, progress = 0 }: ButtonProps): JSX.Element {
|
|
||||||
const buttonText = progress === 100 ? '업로드 완료' : text;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<button
|
|
||||||
className={cn(
|
|
||||||
'relative flex h-12 w-full items-center justify-center rounded-lg border-2 px-5 py-2.5 transition-all duration-300',
|
|
||||||
isActive
|
|
||||||
? 'cursor-pointer border-primary bg-primary text-white'
|
|
||||||
: 'cursor-not-allowed border-gray-200 bg-gray-100 text-gray-500'
|
|
||||||
)}
|
|
||||||
disabled={!isActive}
|
|
||||||
onClick={isActive ? onClick : undefined}
|
|
||||||
style={{ '--progress-width': `${progress}%` } as React.CSSProperties}
|
|
||||||
>
|
|
||||||
<span className="relative z-10 font-sans text-base font-bold leading-6">{buttonText}</span>
|
|
||||||
<div
|
|
||||||
className="transition-width absolute left-0 top-0 z-0 h-full bg-white bg-opacity-20 duration-300"
|
|
||||||
style={{ width: `var(--progress-width, 0%)` }}
|
|
||||||
></div>
|
|
||||||
</button>
|
|
||||||
);
|
|
||||||
}
|
|
39
frontend/src/components/ImageUploadModal/ProgressButton.tsx
Normal file
39
frontend/src/components/ImageUploadModal/ProgressButton.tsx
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
import { cn } from '@/lib/utils';
|
||||||
|
import { Button as BaseButton, ButtonProps as BaseButtonProps } from '@/components/ui/button';
|
||||||
|
|
||||||
|
interface ProgressButtonProps extends BaseButtonProps {
|
||||||
|
isActive: boolean;
|
||||||
|
progress?: number;
|
||||||
|
text: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ProgressButton({
|
||||||
|
isActive,
|
||||||
|
text,
|
||||||
|
progress = 0,
|
||||||
|
variant,
|
||||||
|
size,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: ProgressButtonProps): JSX.Element {
|
||||||
|
const buttonText = progress === 100 ? '업로드 완료' : text;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BaseButton
|
||||||
|
className={cn('relative w-full overflow-hidden', className)}
|
||||||
|
variant={variant}
|
||||||
|
size={size}
|
||||||
|
disabled={!isActive}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{progress > 0 && (
|
||||||
|
<div
|
||||||
|
className="absolute left-0 top-0 h-full bg-primary transition-all duration-300"
|
||||||
|
style={{ width: `${progress}%` }}
|
||||||
|
></div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<span className={cn('relative z-10', progress > 0 && 'text-white')}>{buttonText}</span>
|
||||||
|
</BaseButton>
|
||||||
|
);
|
||||||
|
}
|
@ -1,6 +1,6 @@
|
|||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
import CloseButton from './CloseButton';
|
import CloseButton from './CloseButton';
|
||||||
import Button from './Button';
|
import ProgressButton from './ProgressButton';
|
||||||
import FileList from './FileList';
|
import FileList from './FileList';
|
||||||
import { uploadFiles } from '@/api/upload';
|
import { uploadFiles } from '@/api/upload';
|
||||||
|
|
||||||
@ -103,11 +103,13 @@ export default function ImageUploadModal({ title, buttonText, onClose }: ImageUp
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div className="flex justify-center">
|
<div className="flex justify-center">
|
||||||
<Button
|
<ProgressButton
|
||||||
isActive={files.length > 0 && !isUploading}
|
isActive={files.length > 0 && !isUploading}
|
||||||
text={isUploading ? `업로드 중... (${uploadProgress}%)` : buttonText}
|
text={isUploading ? `업로드 중... (${uploadProgress}%)` : buttonText}
|
||||||
onClick={handleUpload}
|
onClick={handleUpload}
|
||||||
progress={uploadProgress}
|
progress={uploadProgress}
|
||||||
|
type="submit"
|
||||||
|
variant="outlinePrimary"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -0,0 +1,78 @@
|
|||||||
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
|
import { useForm } from 'react-hook-form';
|
||||||
|
import { z } from 'zod';
|
||||||
|
import { Form, FormControl, FormField, FormItem, FormLabel, FormMessage } from '../ui/form';
|
||||||
|
import { Input } from '../ui/input';
|
||||||
|
import { Button } from '../ui/button';
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
workspaceName: z.string().max(50).min(1, {
|
||||||
|
message: '이름을 입력해주세요.',
|
||||||
|
}),
|
||||||
|
workspaceDescription: z.string().max(200).optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type WorkSpaceCreateFormValues = z.infer<typeof formSchema>;
|
||||||
|
|
||||||
|
const defaultValues: Partial<WorkSpaceCreateFormValues> = {
|
||||||
|
workspaceName: '',
|
||||||
|
workspaceDescription: '',
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function WorkSpaceCreateForm({ onSubmit }: { onSubmit: (data: WorkSpaceCreateFormValues) => void }) {
|
||||||
|
const form = useForm<WorkSpaceCreateFormValues>({
|
||||||
|
resolver: zodResolver(formSchema),
|
||||||
|
defaultValues,
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Form {...form}>
|
||||||
|
<form
|
||||||
|
onSubmit={form.handleSubmit(onSubmit)}
|
||||||
|
className="flex flex-col gap-5"
|
||||||
|
>
|
||||||
|
<FormField
|
||||||
|
name="workspaceName"
|
||||||
|
control={form.control}
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel className="body-strong">워크스페이스 이름</FormLabel>
|
||||||
|
<FormControl>
|
||||||
|
<Input
|
||||||
|
placeholder="이름을 입력해주세요."
|
||||||
|
maxLength={50}
|
||||||
|
{...field}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<FormField
|
||||||
|
name="workspaceDescription"
|
||||||
|
control={form.control}
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel className="body-strong">워크스페이스 설명</FormLabel>
|
||||||
|
<FormControl>
|
||||||
|
<Input
|
||||||
|
placeholder="워크스페이스 설명을 입력해주세요."
|
||||||
|
maxLength={200}
|
||||||
|
{...field}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
variant="outlinePrimary"
|
||||||
|
disabled={!form.formState.isValid}
|
||||||
|
>
|
||||||
|
워크스페이스 만들기
|
||||||
|
</Button>
|
||||||
|
</form>
|
||||||
|
</Form>
|
||||||
|
);
|
||||||
|
}
|
@ -0,0 +1,18 @@
|
|||||||
|
import '@/index.css';
|
||||||
|
import WorkSpaceCreateModal from '.';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
title: 'Modal/WorkSpaceCreateModal',
|
||||||
|
component: WorkSpaceCreateModal,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Default = () => (
|
||||||
|
<WorkSpaceCreateModal
|
||||||
|
onClose={() => {
|
||||||
|
console.log('close');
|
||||||
|
}}
|
||||||
|
onSubmit={(data) => {
|
||||||
|
console.log(data);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
25
frontend/src/components/WorkSpaceCreateModal/index.tsx
Normal file
25
frontend/src/components/WorkSpaceCreateModal/index.tsx
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import WorkSpaceCreateForm, { WorkSpaceCreateFormValues } from './WorkSpaceCreateForm';
|
||||||
|
import XIcon from '@/assets/icons/x.svg?react';
|
||||||
|
|
||||||
|
export default function WorkSpaceCreateModal({
|
||||||
|
onClose,
|
||||||
|
onSubmit,
|
||||||
|
}: {
|
||||||
|
onClose: () => void;
|
||||||
|
onSubmit: (data: WorkSpaceCreateFormValues) => void;
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<div className="flex w-[610px] flex-col gap-10 rounded-3xl border px-10 py-5 shadow-lg">
|
||||||
|
<header className="flex gap-5">
|
||||||
|
<h1 className="small-title w-full">새 워크스페이스</h1>
|
||||||
|
<button
|
||||||
|
className="flex h-8 w-8 items-center justify-center"
|
||||||
|
onClick={onClose}
|
||||||
|
>
|
||||||
|
<XIcon className="stroke-gray-900" />
|
||||||
|
</button>
|
||||||
|
</header>
|
||||||
|
<WorkSpaceCreateForm onSubmit={onSubmit} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user