From be3e7b69bc3fec1b780e8d709567f1acab239dfd Mon Sep 17 00:00:00 2001 From: seoyoung-min Date: Sat, 3 Feb 2024 12:17:31 +0900 Subject: [PATCH] =?UTF-8?q?Feat:=20=EB=A6=AC=EC=8A=A4=ED=8A=B8=20=EC=95=84?= =?UTF-8?q?=EC=9D=B4=ED=85=9C=20=EA=B5=AC=ED=98=84=20=EB=B0=8F=20=EB=A6=AC?= =?UTF-8?q?=EC=8A=A4=ED=8A=B8=20=EC=83=9D=EC=84=B1=20=ED=8E=98=EC=9D=B4?= =?UTF-8?q?=EC=A7=80=20=ED=95=A9=EC=B2=B4=20(#13)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Chore: 라이브러리 설치 및 이미지 컴포넌트 환경 설정 * Design: 글로벌css 설정 * Feat: 공용 모달 컴포넌트 구현 * Feat: 아이템 링크 추가 모달 구현 * Feat: 라벨 공용 컴포넌트 추가 * Feat: 링크 미리보기 컴포넌트 구현 * Feat: 아이템 추가 버튼 UI 구현 * Refactor: 레이아웃에 portal용 div 추가 * Feat: 아이템 레이아웃 컴포넌트 구현 * Feat: 아이템 리스트 컴포넌트 및 상수 파일 추가 * Feat: FormProvider 활용 리스트 생성 페이지 추가 및 CreateItem 컴포넌트 구현 * Feat: LinkModal 에러메시지 및 비활성화 기능 추가 * Design: LinkPreview 박스 커서 포인터 추가 * Fix: LinkPreview 삭제 버튼 클릭시 링크 열리는 오류 수정 * Feat: 코멘트 글자수 초과시 입력 제한 기능 추가 및 타이틀 에러시 플레이스홀더 색상 변경 기능 추가 * Refactor: 리스트 생성 페이지 두 컴포넌트 합체 --- .vscode/settings.json | 2 + next.config.js | 8 + package.json | 5 +- public/icons/add.svg | 3 + public/icons/attach_image.svg | 4 + public/icons/back.svg | 3 + public/icons/clear_x_black.svg | 3 + public/icons/clear_x_gray.svg | 3 + public/icons/dnd.svg | 5 + public/icons/link.svg | 3 + .../create/_components/AddItemButton.css.ts | 23 +++ src/app/create/_components/AddItemButton.tsx | 14 ++ src/app/create/_components/CreateItem.css.ts | 71 +++++++ src/app/create/_components/CreateItem.tsx | 46 +++++ src/app/create/_components/CreateList.tsx | 6 +- src/app/create/_components/ItemLayout.css.ts | 76 +++++++ src/app/create/_components/ItemLayout.tsx | 77 ++++++++ src/app/create/_components/Items.css.ts | 102 ++++++++++ src/app/create/_components/Items.tsx | 187 ++++++++++++++++++ src/app/create/_components/LinkModal.tsx | 54 +++++ src/app/create/_components/LinkPreview.css.ts | 26 +++ src/app/create/_components/LinkPreview.tsx | 39 ++++ src/app/create/page.tsx | 25 ++- src/app/layout.tsx | 2 + src/components/Label/Label.css.ts | 40 ++++ src/components/Label/Label.tsx | 28 +++ src/components/Modal/Modal.css.ts | 32 +++ src/components/Modal/Modal.tsx | 34 ++++ src/components/Modal/ModalButton.css.ts | 51 +++++ src/components/Modal/ModalButton.tsx | 27 +++ src/components/Modal/ModalTitle.css.ts | 8 + src/components/Modal/ModalTitle.tsx | 6 + src/components/ModalPortal.ts | 14 ++ src/components/StrictModeDroppable.tsx | 21 ++ src/components/init.ts | 2 - src/hooks/useBooleanOutput.ts | 22 +++ src/hooks/useOnClickOutside.ts | 23 +++ src/lib/constants/formInputValidationRules.ts | 15 ++ src/lib/constants/placeholder.ts | 6 + src/lib/constants/regExpressions.ts | 2 + src/styles/globalStyles.css.ts | 2 + yarn.lock | 111 ++++++++++- 42 files changed, 1212 insertions(+), 19 deletions(-) create mode 100644 .vscode/settings.json create mode 100644 public/icons/add.svg create mode 100644 public/icons/attach_image.svg create mode 100644 public/icons/back.svg create mode 100644 public/icons/clear_x_black.svg create mode 100644 public/icons/clear_x_gray.svg create mode 100644 public/icons/dnd.svg create mode 100644 public/icons/link.svg create mode 100644 src/app/create/_components/AddItemButton.css.ts create mode 100644 src/app/create/_components/AddItemButton.tsx create mode 100644 src/app/create/_components/CreateItem.css.ts create mode 100644 src/app/create/_components/CreateItem.tsx create mode 100644 src/app/create/_components/ItemLayout.css.ts create mode 100644 src/app/create/_components/ItemLayout.tsx create mode 100644 src/app/create/_components/Items.css.ts create mode 100644 src/app/create/_components/Items.tsx create mode 100644 src/app/create/_components/LinkModal.tsx create mode 100644 src/app/create/_components/LinkPreview.css.ts create mode 100644 src/app/create/_components/LinkPreview.tsx create mode 100644 src/components/Label/Label.css.ts create mode 100644 src/components/Label/Label.tsx create mode 100644 src/components/Modal/Modal.css.ts create mode 100644 src/components/Modal/Modal.tsx create mode 100644 src/components/Modal/ModalButton.css.ts create mode 100644 src/components/Modal/ModalButton.tsx create mode 100644 src/components/Modal/ModalTitle.css.ts create mode 100644 src/components/Modal/ModalTitle.tsx create mode 100644 src/components/ModalPortal.ts create mode 100644 src/components/StrictModeDroppable.tsx delete mode 100644 src/components/init.ts create mode 100644 src/hooks/useBooleanOutput.ts create mode 100644 src/hooks/useOnClickOutside.ts create mode 100644 src/lib/constants/formInputValidationRules.ts create mode 100644 src/lib/constants/placeholder.ts create mode 100644 src/lib/constants/regExpressions.ts diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 00000000..7a73a41b --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,2 @@ +{ +} \ No newline at end of file diff --git a/next.config.js b/next.config.js index 95dff287..f06257b6 100644 --- a/next.config.js +++ b/next.config.js @@ -10,6 +10,14 @@ const nextConfig = { }); return config; }, + images: { + remotePatterns: [ + { + protocol: 'https', + hostname: '*', + }, + ], + }, }; module.exports = withVanillaExtract(nextConfig); diff --git a/package.json b/package.json index 33569613..b90a5765 100644 --- a/package.json +++ b/package.json @@ -31,8 +31,9 @@ "axios": "^1.6.5", "next": "14.0.4", "react": "^18", + "react-beautiful-dnd": "^13.1.1", "react-dom": "^18", - "react-hook-form": "^7.49.3", + "react-hook-form": "^7.50.0", "react-scripts": "^5.0.1", "zustand": "^4.4.7" }, @@ -40,6 +41,7 @@ "@svgr/webpack": "^8.1.0", "@commitlint/cli": "^18.6.0", "@commitlint/config-conventional": "^18.6.0", + "@svgr/webpack": "^8.1.0", "@testing-library/jest-dom": "^6.2.0", "@testing-library/react": "^14.1.2", "@testing-library/react-hooks": "^8.0.1", @@ -48,6 +50,7 @@ "@types/jest": "^29.5.11", "@types/node": "^20", "@types/react": "^18", + "@types/react-beautiful-dnd": "^13.1.8", "@types/react-dom": "^18", "eslint": "^8", "eslint-config-next": "14.0.4", diff --git a/public/icons/add.svg b/public/icons/add.svg new file mode 100644 index 00000000..35b07a43 --- /dev/null +++ b/public/icons/add.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/icons/attach_image.svg b/public/icons/attach_image.svg new file mode 100644 index 00000000..38cdc43e --- /dev/null +++ b/public/icons/attach_image.svg @@ -0,0 +1,4 @@ + + + + diff --git a/public/icons/back.svg b/public/icons/back.svg new file mode 100644 index 00000000..8d74e91b --- /dev/null +++ b/public/icons/back.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/icons/clear_x_black.svg b/public/icons/clear_x_black.svg new file mode 100644 index 00000000..ae632feb --- /dev/null +++ b/public/icons/clear_x_black.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/icons/clear_x_gray.svg b/public/icons/clear_x_gray.svg new file mode 100644 index 00000000..1a896c83 --- /dev/null +++ b/public/icons/clear_x_gray.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/icons/dnd.svg b/public/icons/dnd.svg new file mode 100644 index 00000000..6f03a70f --- /dev/null +++ b/public/icons/dnd.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/public/icons/link.svg b/public/icons/link.svg new file mode 100644 index 00000000..a10401e5 --- /dev/null +++ b/public/icons/link.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/app/create/_components/AddItemButton.css.ts b/src/app/create/_components/AddItemButton.css.ts new file mode 100644 index 00000000..e28b7a6f --- /dev/null +++ b/src/app/create/_components/AddItemButton.css.ts @@ -0,0 +1,23 @@ +import { style } from '@vanilla-extract/css'; + +export const addButton = style({ + width: '100%', + height: '60px', + + display: 'flex', + justifyContent: 'center', + alignItems: 'center', + gap: '12px', + + //body1 + fontSize: '1.6rem', + fontWeight: '400', + lineHeight: '1.6rem', + letterSpacing: '-0.48px', + color: '#61646B', + + backgroundColor: '#FFF', + + border: 'solid 1px #AFB1B6 ', + borderRadius: '15px', +}); diff --git a/src/app/create/_components/AddItemButton.tsx b/src/app/create/_components/AddItemButton.tsx new file mode 100644 index 00000000..1f48d3f2 --- /dev/null +++ b/src/app/create/_components/AddItemButton.tsx @@ -0,0 +1,14 @@ +import AddIcon from '/public/icons/add.svg'; +import * as styles from './AddItemButton.css'; + +interface AddItemButton { + handleAddButtonClick: () => void; +} + +export default function AddItemButton({ handleAddButtonClick }: AddItemButton) { + return ( + + ); +} diff --git a/src/app/create/_components/CreateItem.css.ts b/src/app/create/_components/CreateItem.css.ts new file mode 100644 index 00000000..0ce41170 --- /dev/null +++ b/src/app/create/_components/CreateItem.css.ts @@ -0,0 +1,71 @@ +import { style } from '@vanilla-extract/css'; + +export const header = style({ + width: '100%', + height: '90px', + paddingLeft: '20px', + paddingRight: '20px', + + position: 'sticky', + top: '0', + left: '0', + zIndex: '10', + + display: 'flex', + flexDirection: 'row', + alignItems: 'center', + justifyContent: 'space-between', + + backgroundColor: '#fff', + + borderBottom: '1px solid rgba(0, 0, 0, 0.10)', +}); + +export const headerTitle = style({ + fontSize: '2rem', +}); + +export const headerNextButton = style({ + fontSize: '1.6rem', + backgroundColor: 'transparent', +}); + +export const headerNextButtonDisabled = style([ + headerNextButton, + { + color: '#AFB1B6', //활성화 검정, 아닐때는 회색 + }, +]); + +export const article = style({ + padding: '16px 20px 30px', +}); + +//body1 +export const label = style({ + marginBottom: '1.6rem', + + fontSize: '1.6rem', + fontWeight: '600', + letterSpacing: '-0.048rem', +}); + +export const required = style({ + marginLeft: '6px', + + fontSize: '1.6rem', + fontWeight: '500', + letterSpacing: '-0.048rem', + color: '#FF5454', +}); + +//body3 +export const description = style({ + marginBottom: '1.6rem', + + fontSize: '1.4rem', + color: '#8A8A8E', + fontWeight: '400', + lineHeight: '2.5rem', + letterSpacing: '-0.042rem', +}); diff --git a/src/app/create/_components/CreateItem.tsx b/src/app/create/_components/CreateItem.tsx new file mode 100644 index 00000000..7cd378e6 --- /dev/null +++ b/src/app/create/_components/CreateItem.tsx @@ -0,0 +1,46 @@ +import { useFormContext } from 'react-hook-form'; + +import BackIcon from '/public/icons/back.svg'; +import Items from './Items'; +import * as styles from './CreateItem.css'; + +interface CreateItemProps { + onBackClick: () => void; +} + +export default function CreateItem({ onBackClick }: CreateItemProps) { + const { + formState: { isValid }, + } = useFormContext(); + + return ( +
+
+ +

리스트 생성

+ +
+
+

+ 아이템 추가 * +

+ +

+ 최소 3개, 최대 10개까지 아이템을 추가할 수 있어요.
+ 아이템의 순서대로 순위가 정해져요. +

+ +
+
+ ); +} diff --git a/src/app/create/_components/CreateList.tsx b/src/app/create/_components/CreateList.tsx index adf67112..8e80d36c 100644 --- a/src/app/create/_components/CreateList.tsx +++ b/src/app/create/_components/CreateList.tsx @@ -21,7 +21,7 @@ interface UserProfileType { nickname: string; } -function CreateList() { +function CreateList({ onNextClick }: { onNextClick: () => void }) { const { register, getValues, setValue, setError, control, formState } = useFormContext(); const { errors, isValid } = formState; @@ -63,9 +63,9 @@ function CreateList() {

리스트 생성

- +
diff --git a/src/app/create/_components/ItemLayout.css.ts b/src/app/create/_components/ItemLayout.css.ts new file mode 100644 index 00000000..09d3172a --- /dev/null +++ b/src/app/create/_components/ItemLayout.css.ts @@ -0,0 +1,76 @@ +import { style } from '@vanilla-extract/css'; + +export const itemHeader = style({ + width: '100%', + + display: 'flex', + alignItems: 'center', + gap: '12px', + + overflow: 'hidden', +}); + +export const rankAndTitle = style({ + width: '100%', + + display: 'flex', + gap: '8px', +}); + +export const line = style({ + width: '100%', + margin: '0px', + + border: 'solid 1px #AFB1B6', +}); + +export const moreInfo = style({ + display: 'flex', + flexDirection: 'column', + gap: '8px', +}); + +export const toolbar = style({ + width: '100%', + + display: 'flex', + justifyContent: 'space-between', + alignItems: 'center', +}); + +export const fileButtons = style({ + height: '18px', + + display: 'flex', + alignItems: 'center', + justifyContent: 'center', + gap: '10px', +}); + +export const previewContainer = style({ + display: 'flex', + gap: '10px', +}); + +export const previewBox = style({ + width: '90px', + height: '90px', + + display: 'flex', + justifyContent: 'center', + alignItems: 'center', + + position: 'relative', + + background: '#EBF4FF', + + borderRadius: '10px', + whiteSpace: 'pre-wrap', + overflow: 'hidden', +}); + +export const clearButton = style({ + position: 'absolute', + top: '5px', + right: '5px', +}); diff --git a/src/app/create/_components/ItemLayout.tsx b/src/app/create/_components/ItemLayout.tsx new file mode 100644 index 00000000..17581947 --- /dev/null +++ b/src/app/create/_components/ItemLayout.tsx @@ -0,0 +1,77 @@ +import { ReactNode } from 'react'; + +import DndIcon from '/public/icons/dnd.svg'; +import ClearGrayIcon from '/public/icons/clear_x_gray.svg'; +import ClearBlackIcon from '/public/icons/clear_x_black.svg'; +import ImageIcon from '/public/icons/attach_image.svg'; +import Label from '@/components/Label/Label'; +import * as styles from './ItemLayout.css'; + +interface ItemLayoutProps { + index: number; + handleDeleteItem: () => void; + itemLength: number; + titleInput: ReactNode; + commentTextArea: ReactNode; + commentLength: ReactNode; + linkModal: ReactNode; + linkPreview: ReactNode; +} + +export default function ItemLayout({ + index, + handleDeleteItem, + itemLength, + titleInput, + commentTextArea, + commentLength, + linkModal, + linkPreview, +}: ItemLayoutProps) { + return ( + <> +
+ +
+ + {titleInput} +
+ {itemLength > 3 && ( + + )} +
+ +
+ +
+ {commentTextArea} +
+
+ {linkModal} + +
+ {commentLength} +
+ +
+ {linkPreview} +
+ 사진칸 + +
+
+
+ + ); +} diff --git a/src/app/create/_components/Items.css.ts b/src/app/create/_components/Items.css.ts new file mode 100644 index 00000000..ef5a48a8 --- /dev/null +++ b/src/app/create/_components/Items.css.ts @@ -0,0 +1,102 @@ +import { style } from '@vanilla-extract/css'; + +export const itemsContainer = style({ + display: 'flex', + flexDirection: 'column', + gap: '16px', +}); + +export const item = style({ + padding: '12px 18px', + + display: 'flex', + flexDirection: 'column', + gap: '12px', + + backgroundColor: '#fff', + + fontSize: '1.6rem', + border: 'solid 1px #AFB1B6', + borderRadius: '6px', + + transition: 'box-shadow 0.3s ease', + boxShadow: 'rgba(0, 0, 0, 0.1) 0px 2px 2px;', +}); + +export const draggingItem = style([ + item, + { + boxShadow: '0px 20px 50px -5px #AFB1B6', + }, +]); + +export const title = style({ + //body1 + fontSize: '1.6rem', + fontWeight: '400', + lineHeight: '1.6rem', + letterSpacing: '-0.48px', + + '::placeholder': { + color: '#AFB1B6', + }, +}); + +export const errorTitle = style([ + title, + { + '::placeholder': { + color: '#FF5454', + }, + }, +]); + +export const comment = style({ + width: '100%', + resize: 'none', + + flexGrow: '1', + + //body2 + fontSize: '1.5rem', + lineHeight: '2.5rem', + letterSpacing: '-0.45px', + + border: 'none', + outline: 'none', + + '::placeholder': { + color: '#AFB1B6', + }, +}); + +export const linkModalChildren = style({ + width: '100%', +}); + +export const linkInput = style([ + title, + { + width: '100%', + padding: '8px', + + border: 'solid 1px #AFB1B6', + borderRadius: '4px', + }, +]); + +export const countLength = style({ + //body2 + fontSize: '1.5rem', + letterSpacing: '-0.45px', + color: '#61646B', +}); + +export const error = style({ + marginTop: '8px', + marginLeft: '4px', + + flexShrink: '0', + color: '#FF5454', + fontSize: '1.5rem', +}); diff --git a/src/app/create/_components/Items.tsx b/src/app/create/_components/Items.tsx new file mode 100644 index 00000000..4792155a --- /dev/null +++ b/src/app/create/_components/Items.tsx @@ -0,0 +1,187 @@ +import { useState } from 'react'; +import { useFieldArray, useFormContext, useWatch } from 'react-hook-form'; +import { DragDropContext, Draggable, DropResult } from 'react-beautiful-dnd'; + +import { itemPlaceholder } from '@/lib/constants/placeholder'; +import { itemTitleRules, itemCommentRules, itemLinkRules } from '@/lib/constants/formInputValidationRules'; +import { StrictModeDroppable } from '@/components/StrictModeDroppable'; +import { FormErrors } from '../page'; +import ItemLayout from './ItemLayout'; +import LinkModal from './LinkModal'; +import LinkPreview from './LinkPreview'; +import * as styles from './Items.css'; +import AddItemButton from './AddItemButton'; + +// http:// 없을경우 추가 +const ensureHttp = (link: string) => { + if (!link.startsWith('http://' || 'https://')) { + return 'http://' + link; + } + return link; +}; + +// 링크 도메인만 추출 (e.g. naver.com) +const urlToDomain = (link: string) => { + const domain = new URL(link).hostname.replace('www.', ''); + return domain; +}; + +export default function Items() { + const { + register, + control, + getValues, + setValue, + formState: { errors }, + } = useFormContext(); + + const { + fields: items, + append, + remove, + } = useFieldArray({ + name: 'items', + control, + rules: { minLength: 3, maxLength: 10 }, + }); + + const [current, setCurrent] = useState(null); + + const watchItems = useWatch({ control, name: 'items' }); + + //--- LinkModal 핸들러 + const handleLinkModalOpen = (index: number) => { + setCurrent(getValues().items[index]?.link); + }; + + const handleLinkModalCancel = (index: number) => { + setValue(`items.${index}.link`, current); + }; + + const handleLinkModalConfirm = (index: number) => { + if (watchItems[index]?.link) { + setValue(`items.${index}.link`, ensureHttp(watchItems[index]?.link)); + } + }; + + //--- 드래그 되었을 때 실행되는 이벤트 + const onDragEnd = ({ source, destination }: DropResult) => { + if (destination && source.index !== destination.index) { + const currentArray = [...getValues().items]; + const sourceItem = currentArray[source.index]; + currentArray.splice(source.index, 1); + currentArray.splice(destination.index, 0, sourceItem); + setValue('items', currentArray); + } + }; + + return ( + + + {(provided) => ( +
+ {items.map((item, index) => { + const errorMessage = (field: 'title' | 'comment' | 'link') => + (errors as FormErrors)?.items?.[index]?.[field]?.message; + const titleError = errorMessage('title'); + const commentError = errorMessage('comment'); + const linkError = errorMessage('link'); + return ( + + {(provided, snapshot) => ( +
+ remove(index)} + itemLength={watchItems.length} + titleInput={ + + } + commentTextArea={ +