Skip to content

Commit

Permalink
[#366] BottomSheet 컴포넌트 (#394)
Browse files Browse the repository at this point in the history
* feat: BottomSheet 구현

* refactor: close 아이콘 크기 설정

* feat: logo svgr로 import하도록 수정
  • Loading branch information
gxxrxn authored Sep 16, 2023
1 parent e1d402a commit b8c9d77
Show file tree
Hide file tree
Showing 5 changed files with 135 additions and 2 deletions.
3 changes: 3 additions & 0 deletions public/icons/index.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
// 로고
export { default as LogoWithText } from './logo-with-text.svg';

// 상단 Nav & Header
export { default as IconArrowLeft } from './arrow-left.svg';
export { default as IconClose } from './close.svg';
Expand Down
2 changes: 1 addition & 1 deletion public/icons/kakao.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
2 changes: 1 addition & 1 deletion public/icons/logo-with-text.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
77 changes: 77 additions & 0 deletions src/stories/Base/BottomSheet.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
import { Meta, StoryObj } from '@storybook/react';

import { IconClose, IconKakao, LogoWithText } from '@public/icons';
import Button from '@/ui/Base/Button';
import BottomSheet from '@/ui/Base/BottomSheet';
import useDisclosure from '@/hooks/useDisclosure';

const meta: Meta<typeof BottomSheet> = {
title: 'Base/BottomSheet',
component: BottomSheet,
tags: ['autodocs'],
};

const DefaultBottomSheet = () => {
const { isOpen, onOpen, onClose } = useDisclosure();

return (
<>
<Button onClick={onOpen}>기본 바텀시트 열기</Button>
<BottomSheet isShow={isOpen} onClose={onClose}>
<p className="text-center text-lg">바텀시트 예시</p>
</BottomSheet>
</>
);
};

const LoginBottomSheet = () => {
const { isOpen, onOpen, onClose } = useDisclosure();

return (
<>
<Button onClick={onOpen}>로그인 바텀시트 열기</Button>
<BottomSheet isShow={isOpen} onClose={onClose}>
<div
className="absolute right-0 top-0 mr-[2rem] mt-[2rem] h-[2rem] w-[2rem] cursor-pointer"
onClick={onClose}
>
<IconClose />
</div>
<div className="m-auto flex w-full max-w-[38rem] flex-col items-center gap-[2.5rem] px-[2rem] pt-[5rem]">
<div className="h-auto w-[6rem]">
<LogoWithText />
</div>
<p className="text-lg font-bold text-black-700">
로그인이 필요한 서비스에요!
</p>
<p className="text-center text-sm font-normal text-placeholder">
간편하게 카카오로 로그인을 하고,
<br />
<span className="text-main-900">다독다독</span>의 다양한 기능을
이용해보세요.
</p>
<Button colorScheme="kakao" size="full">
<div className="flex w-full items-center justify-center gap-[0.7rem]">
<div className="h-auto w-[1.6rem]">
<IconKakao />
</div>
<span className="text-md font-normal">카카오 로그인</span>
</div>
</Button>
</div>
</BottomSheet>
</>
);
};

export default meta;

type Story = StoryObj<typeof BottomSheet>;

export const Default: Story = {
render: DefaultBottomSheet,
};

export const Login: Story = {
render: LoginBottomSheet,
};
53 changes: 53 additions & 0 deletions src/ui/Base/BottomSheet.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
import { Dialog, Transition } from '@headlessui/react';
import { Fragment, PropsWithChildren } from 'react';

interface BottomSheetProps {
isShow?: boolean;
onClose: () => void;
}

const BottomSheet = ({
isShow = false,
onClose,
children,
}: PropsWithChildren<BottomSheetProps>) => {
return (
<Transition.Root show={isShow} as={Fragment}>
<Dialog as="div" className="relative z-10" onClose={onClose}>
<Transition.Child
as={Fragment}
enter="ease-in-out duration-300"
enterFrom="opacity-0"
enterTo="opacity-100"
leave="ease-in-out duration-300"
leaveFrom="opacity-100"
leaveTo="opacity-0"
>
<div className="fixed inset-0 bg-overlay/60 transition-opacity" />
</Transition.Child>

<div className="fixed inset-0 overflow-hidden">
<div className="absolute inset-0 overflow-hidden">
<div className="pointer-events-none fixed bottom-0 left-1/2 flex max-w-full -translate-x-1/2">
<Transition.Child
as={Fragment}
enter="transform transition ease-in-out duration-300"
enterFrom="translate-y-full"
enterTo="translate-y-0"
leave="transform transition ease-in-out duration-300"
leaveFrom="translate-y-0"
leaveTo="translate-y-full"
>
<Dialog.Panel className="pointer-events-auto relative max-h-[100dvh] w-[43rem] rounded-t-[1.5rem] bg-white p-[2rem]">
{children}
</Dialog.Panel>
</Transition.Child>
</div>
</div>
</div>
</Dialog>
</Transition.Root>
);
};

export default BottomSheet;

0 comments on commit b8c9d77

Please sign in to comment.