Skip to content

Commit

Permalink
[#411][독서 모임] Detail 독서 모임 컴포넌트 (#428)
Browse files Browse the repository at this point in the history
* feat: 모임 목록 페이지 모임 상세 컴포넌트 구현

* fix: 컴포넌트명 변경

* fix: bookGroup 컴포넌트명 변경
  • Loading branch information
WooDaeHyun authored and gxxrxn committed Jun 17, 2024
1 parent 2277b0d commit cbb032a
Show file tree
Hide file tree
Showing 7 changed files with 1,945 additions and 1,818 deletions.
4 changes: 2 additions & 2 deletions public/icons/calendar.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
4 changes: 2 additions & 2 deletions public/icons/comments.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
4 changes: 2 additions & 2 deletions public/icons/members.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
39 changes: 39 additions & 0 deletions src/stories/bookGroup/DetailBookGroupCard.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
import { Meta, StoryObj } from '@storybook/react';

import DetailBookGroupCard from '@/v1/bookgroup/DetailBookGroupCard';

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

export default meta;

type Story = StoryObj<typeof DetailBookGroupCard>;

export const Default: Story = {
args: {
title: '프롱이 리팩터링 스터디',
description:
'제1차 프롱이 기수연합 독서 스터디 입니다. 마틴 파울러의 저서 ‘리팩터링 2판’과 함께 진행합니다.',
book: {
title: '리팩터링 2판',
bookImageSrc: 'https://image.yes24.com/goods/89649360/XL',
},
date: {
start: '2023-10-31',
end: '2023-11-27',
},
memberCount: 3,
owner: {
name: '소피아',
profileImageSrc: '',
},
isPublic: false,
commentCount: 12,
handleClick: () => {
alert('모임 상세 페이지로 이동');
},
},
};
249 changes: 249 additions & 0 deletions src/v1/bookgroup/DetailBookGroupCard.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,249 @@
import Image from 'next/image';

import Badge from '@/ui/Base/Badge';
import { IconCalendar, IconMembers, IconComments } from '@public/icons';

interface DetailBookGroupCardProps {
title: string;
description: string;
book: { title: string; bookImageSrc: string };
date: { start: string; end: string };
owner: { name: string; profileImageSrc: string };
memberCount: number;
commentCount: number;
isPublic: boolean;
handleClick: () => void;
}

type BookGroupStatus = 'before' | 'dday' | 'ongoing' | 'end';

const getBookGroupStatus = (ddayByStart: number, ddayByEnd: number) => {
if (ddayByStart > 0) {
return {
status: 'before' as const,
ddayCount: ddayByStart,
};
} else if (ddayByStart === 0 && ddayByEnd > 0) {
return {
status: 'dday' as const,
};
} else if (ddayByStart < 0 && ddayByEnd >= 0) {
return {
status: 'ongoing' as const,
};
} else {
return {
status: 'end' as const,
};
}
};

const toDayFromMillseconds = (value: number) => {
return Math.ceil(value / (1000 * 60 * 60 * 24));
};

const DetailBookGroupCard = ({
title,
description,
book,
date,
owner,
memberCount,
commentCount,
isPublic,
handleClick,
}: DetailBookGroupCardProps) => {
const ddayByStart = toDayFromMillseconds(
new Date(date.start).getTime() - new Date().getTime()
);
const ddayByEnd = toDayFromMillseconds(
new Date(date.end).getTime() - new Date().getTime()
);

const { ...ddayProps } = getBookGroupStatus(ddayByStart, ddayByEnd);

return (
<div
onClick={handleClick}
className="h-[16.142rem] w-[35.5rem] rounded-[0.4rem] px-[1.6rem] py-[0.9rem] shadow-[0_0_0.6rem_rgba(180,180,180,0.25)]"
>
<div className="mb-[1rem] flex gap-[0.5rem]">
<Dday {...ddayProps}></Dday>
<Public isPublic={isPublic}></Public>
</div>
<div className="flex gap-[1.4rem]">
<div className="flex flex-col gap-[0.63rem]">
<Title title={title}></Title>
<Description description={description}></Description>
<Duration start={date.start} end={date.end}></Duration>
<div className="flex w-[22.5rem] justify-between">
<Owner
name={owner.name}
profileImageSrc={owner.profileImageSrc}
></Owner>
<div className="flex gap-[0.5rem]">
<MemberCount memberCount={memberCount}></MemberCount>
<CommentCount commentCount={commentCount}></CommentCount>
</div>
</div>
</div>
<div>
<Book title={book.title} bookImageSrc={book.bookImageSrc}></Book>
</div>
</div>
</div>
);
};

export default DetailBookGroupCard;

const getDdayBadgeInfo = (status: BookGroupStatus, ddayCount?: number) => {
switch (status) {
case 'before':
return {
colorScheme: 'main' as const,
isFilled: true,
text: `D-${ddayCount}`,
};
case 'dday':
return {
colorScheme: 'main' as const,
isFilled: false,
text: 'D-day',
};
case 'ongoing':
return {
colorScheme: 'main' as const,
isFilled: true,
text: '진행중',
};
case 'end':
return {
colorScheme: 'grey' as const,
isFilled: true,
text: '모임종료',
};
}
};

const Dday = ({
status,
ddayCount,
}: {
status: BookGroupStatus;
ddayCount?: number;
}) => {
const { text, ...badgeProps } = getDdayBadgeInfo(status, ddayCount);

return (
<Badge size="large" {...badgeProps}>
{text}
</Badge>
);
};

const Public = ({ isPublic }: { isPublic: boolean }) => (
<Badge size="large" colorScheme="grey">
{isPublic ? '공개' : '비공개'}
</Badge>
);

const Title = ({ title }: { title: string }) => {
return (
<div>
<span className="text-md font-bold">{title}</span>
</div>
);
};

const Description = ({ description }: { description: string }) => {
return (
<div className="w-[22.5rem] truncate text-sm">
<span>{description}</span>
</div>
);
};

const Duration = ({ start, end }: { start: string; end: string }) => {
return (
<div className="flex items-center gap-[0.5rem]">
<div>
<IconCalendar className="h-[1.161rem] w-[1.1rem] fill-placeholder" />
</div>
<div className="text-xs text-placeholder">
<span className="pt-[0.1rem]">
{start} - {end}
</span>
</div>
</div>
);
};

const Owner = ({
name,
profileImageSrc,
}: {
name: string;
profileImageSrc: string;
}) => {
return (
<div className="flex h-[2rem] gap-[0.5rem]">
{/* 아바타 컴포넌트로 변경 예정 */}
<div className={'relative h-[2rem] w-[2rem] rounded-full bg-black-400'}>
{profileImageSrc && <Image alt={name} src={profileImageSrc} fill />}
</div>
<div className="flex items-center text-xs">
<span>{name}</span>
</div>
</div>
);
};

const MemberCount = ({ memberCount }: { memberCount: number }) => {
return (
<div className="flex items-center gap-[0.3rem]">
<span className="h-auto w-[1.3rem] fill-placeholder">
<IconMembers className="h-[0.9rem] w-[1.3rem] fill-placeholder" />
</span>
<span className="text-xs text-placeholder">
<span className="text-placeholder">{memberCount}</span>
</span>
</div>
);
};

const CommentCount = ({ commentCount }: { commentCount: number }) => {
return (
<div className="flex items-center gap-[0.3rem]">
<div className="h-auto w-[1.3rem] fill-placeholder">
<IconComments className="h-[1.2rem] w-[1.2rem] fill-placeholder" />
</div>
<span className="text-xs text-placeholder">
<span className="text-placeholder">{commentCount}</span>
</span>
</div>
);
};

const Book = ({
bookImageSrc,
title,
}: {
bookImageSrc: string;
title: string;
}) => {
return (
<div>
<div className="relative h-[10.442rem] w-[8rem]">
{bookImageSrc && (
<Image
src={bookImageSrc}
alt={title}
fill
className="object-fit rounded-[0.4rem] shadow-[0.1rem_0.2rem_0.4rem_0_rgba(0,0,0,0.25)]"
/>
)}
</div>
</div>
);
};
1 change: 1 addition & 0 deletions tailwind.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ module.exports = {
'./src/stories/**/*.{ts,tsx}',
'./src/v1/**/*.{ts,tsx}',
'./.storybook/**/*.{ts,tsx}',
'./src/v1/**/*.{ts,tsx}',
],
theme: {
extend: {
Expand Down
Loading

0 comments on commit cbb032a

Please sign in to comment.