diff --git a/frontend/reply-module/.storybook/main.js b/frontend/reply-module/.storybook/main.js index 26dfeaf66..82a742044 100644 --- a/frontend/reply-module/.storybook/main.js +++ b/frontend/reply-module/.storybook/main.js @@ -1,10 +1,4 @@ module.exports = { - "stories": [ - "../src/**/*.stories.mdx", - "../src/**/*.stories.@(js|jsx|ts|tsx)" - ], - "addons": [ - "@storybook/addon-links", - "@storybook/addon-essentials" - ] -} \ No newline at end of file + stories: ["../src/**/*.stories.mdx", "../src/**/*.stories.@(js|jsx|ts|tsx)"], + addons: ["@storybook/addon-links", "@storybook/addon-essentials"] +}; diff --git a/frontend/reply-module/.storybook/preview.css b/frontend/reply-module/.storybook/preview.css new file mode 100644 index 000000000..ec23d958f --- /dev/null +++ b/frontend/reply-module/.storybook/preview.css @@ -0,0 +1,4 @@ +#root { + width: 100%; + max-width: 1080px; +} diff --git a/frontend/reply-module/.storybook/preview.js b/frontend/reply-module/.storybook/preview.js index 48afd568a..02afa7473 100644 --- a/frontend/reply-module/.storybook/preview.js +++ b/frontend/reply-module/.storybook/preview.js @@ -1,9 +1,23 @@ +import { configure, addDecorator } from "@storybook/react"; +import GlobalStyles from "../src/styles/GlobalStyles"; +import "./preview.css"; + export const parameters = { actions: { argTypesRegex: "^on[A-Z].*" }, + layout: "centered", controls: { matchers: { color: /(background|color)$/i, - date: /Date$/, - }, - }, -} \ No newline at end of file + date: /Date$/ + } + } +}; + +addDecorator(style => ( + <> + + <>{style()} + +)); + +configure(require.context("../src", true, /\.stories\.js?$/), module); diff --git a/frontend/reply-module/src/App.tsx b/frontend/reply-module/src/App.tsx index a5e22d64a..49837e41c 100644 --- a/frontend/reply-module/src/App.tsx +++ b/frontend/reply-module/src/App.tsx @@ -1,5 +1,5 @@ const App = () => { - return <>Hello World; + return null; }; export default App; diff --git a/frontend/reply-module/src/assets/svg/three-dots.svg b/frontend/reply-module/src/assets/svg/three-dots.svg new file mode 100644 index 000000000..bd4058aaa --- /dev/null +++ b/frontend/reply-module/src/assets/svg/three-dots.svg @@ -0,0 +1,3 @@ + + + diff --git a/frontend/reply-module/src/components/atoms/Avatar/Avatar.stories.tsx b/frontend/reply-module/src/components/atoms/Avatar/Avatar.stories.tsx new file mode 100644 index 000000000..cc19a8f65 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/Avatar/Avatar.stories.tsx @@ -0,0 +1,16 @@ +import { Story } from "@storybook/react"; +import Avatar, { Props } from "."; + +export default { + title: "atoms/Avatar", + component: Avatar, + argTypes: { children: { control: "text" } } +}; + +const Template: Story = args => ; + +export const Default = Template.bind({}); + +Default.args = { + imageURL: "" +}; diff --git a/frontend/reply-module/src/components/atoms/Avatar/index.tsx b/frontend/reply-module/src/components/atoms/Avatar/index.tsx new file mode 100644 index 000000000..37402cf45 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/Avatar/index.tsx @@ -0,0 +1,14 @@ +import { Container } from "./styles"; + +export type Size = "SM" | "MD" | "LG"; + +export interface Props { + imageURL: string; + size?: Size; +} + +const Avatar = ({ imageURL, size = "MD" }: Props) => { + return ; +}; + +export default Avatar; diff --git a/frontend/reply-module/src/components/atoms/Avatar/styles.ts b/frontend/reply-module/src/components/atoms/Avatar/styles.ts new file mode 100644 index 000000000..286d2c87e --- /dev/null +++ b/frontend/reply-module/src/components/atoms/Avatar/styles.ts @@ -0,0 +1,17 @@ +import styled from "styled-components"; + +import { Size } from "."; + +const avatarSizeBySize = { + SM: 30, + MD: 40, + LG: 90 +}; + +const Container = styled.img<{ size: Size }>` + border-radius: 50%; + width: ${props => `${avatarSizeBySize[props.size]}px`}; + height: ${props => `${avatarSizeBySize[props.size]}px`}; +`; + +export { Container }; diff --git a/frontend/reply-module/src/components/atoms/CommentOption/CommentOption.stories.tsx b/frontend/reply-module/src/components/atoms/CommentOption/CommentOption.stories.tsx new file mode 100644 index 000000000..8e127a394 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/CommentOption/CommentOption.stories.tsx @@ -0,0 +1,14 @@ +import { Story } from "@storybook/react"; +import CommentOption, { Props } from "."; + +export default { + title: "atoms/CommentOption", + component: CommentOption, + argTypes: { children: { control: "text" } } +}; + +const Template: Story = args => ; + +export const Default = Template.bind({}); + +Default.args = {}; diff --git a/frontend/reply-module/src/components/atoms/CommentOption/index.tsx b/frontend/reply-module/src/components/atoms/CommentOption/index.tsx new file mode 100644 index 000000000..d352943e3 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/CommentOption/index.tsx @@ -0,0 +1,29 @@ +import { useState } from "react"; +import threeDots from "../../../assets/svg/three-dots.svg"; +import { Container, DeleteButton, EditButton, OptionContainer, OptionIcon } from "./styles"; + +export interface Props { + onEdit?: () => void; + onDelete?: () => void; +} + +const CommentOption = ({ onEdit, onDelete }: Props) => { + const [isShowOptionBox, setShowOptionBox] = useState(false); + const onShowOptionBox = () => { + setShowOptionBox(state => !state); + }; + + return ( + + + {isShowOptionBox && ( + + 수정 + 삭제 + + )} + + ); +}; + +export default CommentOption; diff --git a/frontend/reply-module/src/components/atoms/CommentOption/styles.ts b/frontend/reply-module/src/components/atoms/CommentOption/styles.ts new file mode 100644 index 000000000..5fe7d3414 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/CommentOption/styles.ts @@ -0,0 +1,63 @@ +import styled from "styled-components"; +import { PALETTE } from "../../../styles/palette"; + +const Container = styled.div` + position: absolute; +`; + +const OptionIcon = styled.img` + cursor: pointer; +`; + +const OptionContainer = styled.div` + position: absolute; + right: -5px; + width: 6rem; + box-shadow: 1.04082px 1.04082px 6.24491px rgba(0, 0, 0, 0.25); + border-radius: 10px; + display: flex; + flex-direction: column; + align-items: center; + background-color: ${PALETTE.WHITE}; + + ::before { + content: ""; + position: absolute; + top: -5px; + right: 8px; + + border-bottom: 10px solid ${PALETTE.WHITE}; + border-left: 5px solid transparent; + border-right: 5px solid transparent; + } + + & > button { + width: 100%; + border: none; + outline: none; + background-color: ${PALETTE.WHITE}; + cursor: pointer; + font-weight: 600; + margin-bottom: 0.3rem; + + :first-child { + padding-top: 0.5rem; + border-radius: 10px 10px 0 0; + } + + :last-child { + margin-bottom: 0; + padding-bottom: 0.5rem; + border-radius: 0 0 10px 10px; + } + } +`; + +const EditButton = styled.button` + color: ${PALETTE.BLACK_700}; +`; +const DeleteButton = styled.button` + color: ${PALETTE.RED_600}; +`; + +export { Container, OptionIcon, OptionContainer, EditButton, DeleteButton }; diff --git a/frontend/reply-module/src/components/atoms/CommentTextBox/CommentTextBox.stories.tsx b/frontend/reply-module/src/components/atoms/CommentTextBox/CommentTextBox.stories.tsx new file mode 100644 index 000000000..d653da56b --- /dev/null +++ b/frontend/reply-module/src/components/atoms/CommentTextBox/CommentTextBox.stories.tsx @@ -0,0 +1,17 @@ +import { Story } from "@storybook/react"; +import CommentTextBox, { Props } from "."; + +export default { + title: "atoms/CommentTextBox", + component: CommentTextBox, + argTypes: { children: { control: "text" } } +}; + +const Template: Story = args => ; + +export const Default = Template.bind({}); + +Default.args = { + name: "곤이", + contentEditable: true +}; diff --git a/frontend/reply-module/src/components/atoms/CommentTextBox/index.tsx b/frontend/reply-module/src/components/atoms/CommentTextBox/index.tsx new file mode 100644 index 000000000..185c730c3 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/CommentTextBox/index.tsx @@ -0,0 +1,18 @@ +import { Container, Name, Text } from "./styles"; + +export interface Props { + name: string; + children: string; + contentEditable?: boolean; +} + +const CommentTextBox = ({ name, children, contentEditable = false }: Props) => { + return ( + + {name} + {children} + + ); +}; + +export default CommentTextBox; diff --git a/frontend/reply-module/src/components/atoms/CommentTextBox/styles.ts b/frontend/reply-module/src/components/atoms/CommentTextBox/styles.ts new file mode 100644 index 000000000..920d837ee --- /dev/null +++ b/frontend/reply-module/src/components/atoms/CommentTextBox/styles.ts @@ -0,0 +1,27 @@ +import styled from "styled-components"; +import { PALETTE } from "../../../styles/palette"; + +const Container = styled.div` + width: 100%; + background-color: ${PALETTE.GRAY_200}; + border-radius: 10px; + padding: 0.8rem 1rem; + display: flex; + flex-direction: column; +`; + +const Name = styled.span` + font-weight: 700; + font-size: 1.4rem; + margin-bottom: 0.7rem; +`; + +const Text = styled.div` + outline-color: ${PALETTE.BLACK_700}; + background-color: ${props => (props.contentEditable ? PALETTE.WHITE : PALETTE.GRAY_200)}; + border-radius: 10px; + min-width: 10rem; + max-width: 20rem; +`; + +export { Container, Name, Text }; diff --git a/frontend/reply-module/src/components/atoms/SubmitButton/SubmitButton.stories.tsx b/frontend/reply-module/src/components/atoms/SubmitButton/SubmitButton.stories.tsx new file mode 100644 index 000000000..a6ba8f1db --- /dev/null +++ b/frontend/reply-module/src/components/atoms/SubmitButton/SubmitButton.stories.tsx @@ -0,0 +1,14 @@ +import { Story } from "@storybook/react"; +import SubmitButton, { Props } from "."; + +export default { + title: "atoms/SubmitButton", + component: SubmitButton, + argTypes: { children: { control: "text" } } +}; + +const Template: Story = args => ; + +export const Default = Template.bind({}); + +Default.args = {}; diff --git a/frontend/reply-module/src/components/atoms/SubmitButton/index.tsx b/frontend/reply-module/src/components/atoms/SubmitButton/index.tsx new file mode 100644 index 000000000..7e56e3894 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/SubmitButton/index.tsx @@ -0,0 +1,12 @@ +import { Button } from "./styles"; + +export interface Props { + children: string; + onClick: () => void; +} + +const SubmitButton = ({ children, onClick }: Props) => { + return ; +}; + +export default SubmitButton; diff --git a/frontend/reply-module/src/components/atoms/SubmitButton/styles.ts b/frontend/reply-module/src/components/atoms/SubmitButton/styles.ts new file mode 100644 index 000000000..41cf756b3 --- /dev/null +++ b/frontend/reply-module/src/components/atoms/SubmitButton/styles.ts @@ -0,0 +1,15 @@ +import { PALETTE } from "./../../../styles/palette"; +import styled from "styled-components"; + +const Button = styled.button` + width: 6rem; + height: 3.6rem; + background-color: ${PALETTE.PRIMARY}; + color: ${PALETTE.WHITE}; + font-size: 1.6rem; + font-weight: 500; + border: none; + border-radius: 10px; +`; + +export { Button }; diff --git a/frontend/reply-module/src/components/molecules/Comment/Comment.stories.tsx b/frontend/reply-module/src/components/molecules/Comment/Comment.stories.tsx new file mode 100644 index 000000000..355a594b4 --- /dev/null +++ b/frontend/reply-module/src/components/molecules/Comment/Comment.stories.tsx @@ -0,0 +1,28 @@ +import { Story } from "@storybook/react"; +import Comment, { Props } from "."; + +export default { + title: "molecules/Comment", + component: Comment, + argTypes: { children: { control: "text" } } +}; + +const Template: Story = args => ; + +export const Default = Template.bind({}); + +Default.args = { + comment: { + id: 1, + content: "Donec accumsan neque enim sodales. Neque eget vulputate viverra convallis pharetra.", + user: { + id: 1, + imageURL: + "https://static.independent.co.uk/s3fs-public/thumbnails/image/2015/06/06/15/Chris-Pratt.jpg?width=982&height=726&auto=webp&quality=75", + nickName: "Robert Hill", + type: "Authorized" + }, + createdAt: "1시간 전" + }, + shouldShowOption: true +}; diff --git a/frontend/reply-module/src/components/molecules/Comment/index.tsx b/frontend/reply-module/src/components/molecules/Comment/index.tsx new file mode 100644 index 000000000..539f80af6 --- /dev/null +++ b/frontend/reply-module/src/components/molecules/Comment/index.tsx @@ -0,0 +1,33 @@ +import { Comment as CommentType } from "../../../types"; +import Avatar from "../../atoms/Avatar"; +import CommentOption from "../../atoms/CommentOption"; +import CommentTextBox from "../../atoms/CommentTextBox"; +import { Container, CommentTextBoxWrapper, Time, CommentOptionWrapper } from "./styles"; + +export interface Props { + comment: CommentType; + align?: "left" | "right"; + + shouldShowOption?: boolean; + onEdit?: () => void; + onDelete?: () => void; +} + +const Comment = ({ comment, align = "left", shouldShowOption, onEdit, onDelete }: Props) => { + return ( + + + + {comment.content} + + {shouldShowOption && ( + + + + )} + + + ); +}; + +export default Comment; diff --git a/frontend/reply-module/src/components/molecules/Comment/styles.ts b/frontend/reply-module/src/components/molecules/Comment/styles.ts new file mode 100644 index 000000000..af2340a0d --- /dev/null +++ b/frontend/reply-module/src/components/molecules/Comment/styles.ts @@ -0,0 +1,27 @@ +import styled from "styled-components"; + +const Container = styled.div<{ align: "left" | "right" }>` + display: flex; + flex-direction: ${props => (props.align === "left" ? "row" : "row-reverse")}; +`; + +const CommentTextBoxWrapper = styled.div<{ align: "left" | "right" }>` + position: relative; + display: flex; + flex-direction: column; + align-items: ${props => (props.align === "left" ? "flex-start" : "flex-end")}; + margin: ${props => (props.align === "left" ? "0 0 0 0.6rem" : "0 0.6rem 0 0")}; +`; + +const Time = styled.span` + margin: 0 1rem; + margin-top: 0.3rem; +`; + +const CommentOptionWrapper = styled.div` + position: absolute; + right: 30px; + top: 8px; +`; + +export { Container, CommentTextBoxWrapper, Time, CommentOptionWrapper }; diff --git a/frontend/reply-module/src/components/organisms/CommentInput/CommentInput.stories.tsx b/frontend/reply-module/src/components/organisms/CommentInput/CommentInput.stories.tsx new file mode 100644 index 000000000..07836f452 --- /dev/null +++ b/frontend/reply-module/src/components/organisms/CommentInput/CommentInput.stories.tsx @@ -0,0 +1,14 @@ +import { Story } from "@storybook/react"; +import CommentInput, { Props } from "."; + +export default { + title: "organisms/CommentInput", + component: CommentInput, + argTypes: { children: { control: "text" } } +}; + +const Template: Story = args => ; + +export const Default = Template.bind({}); + +Default.args = {}; diff --git a/frontend/reply-module/src/components/organisms/CommentInput/index.tsx b/frontend/reply-module/src/components/organisms/CommentInput/index.tsx new file mode 100644 index 000000000..862d3fdf5 --- /dev/null +++ b/frontend/reply-module/src/components/organisms/CommentInput/index.tsx @@ -0,0 +1,27 @@ +import SubmitButton from "../../atoms/SubmitButton"; +import { Container, TextArea, Wrapper, GuestInfo } from "./styles"; + +export interface Props {} + +const CommentInput = () => { + const isGuest = true; + + return ( + +