Skip to content
This repository has been archived by the owner on Jul 29, 2024. It is now read-only.

[FE] feat: 카테고리 및 글 드래그앤드롭 기능 구현 #418

Merged
merged 25 commits into from
Sep 20, 2023
Merged
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
19497a7
refactor: 카테고리 데이터가 null 인 경우 early return
nangkyeonglim Sep 12, 2023
543cb3b
feat: 글 및 카테고리 순서 수정 API, 훅 작성
nangkyeonglim Sep 16, 2023
27b6e00
feat: 타입 가드 유틸 함수 구현
nangkyeonglim Sep 16, 2023
d0f32bc
feat: 글 및 카테고리 순서 수정 API mocking
nangkyeonglim Sep 16, 2023
3a8269b
feat: 카테고리에 스크롤이 생길 때 `Header`가 고정되도록 수정
nangkyeonglim Sep 16, 2023
925a694
feat: 스크롤이 있을 때 카테고리 추가 시 맨 아래로 스크롤 되는 기능 구현
nangkyeonglim Sep 16, 2023
bd56b74
design: 카테고리 및 글 hover시 색상 변경
nangkyeonglim Sep 16, 2023
8d6cfe0
feat: `throttle` 유틸 함수 구현
nangkyeonglim Sep 16, 2023
1d007d3
feat: 같은 배열인지 확인하는 유틸 함수 작성
nangkyeonglim Sep 16, 2023
82c9fc2
feat: useDragAndDrop 훅 작성
nangkyeonglim Sep 16, 2023
eace190
feat: 카테고리 드래그 앤 드롭 구현
nangkyeonglim Sep 16, 2023
1d30e31
feat: 글 드래그 앤 드롭 구현
nangkyeonglim Sep 16, 2023
3dc15b3
feat: 드래그 영역 확장 및 throttle 적용
nangkyeonglim Sep 16, 2023
efb4468
refactor: `useDragAndDrop` 훅 파일 이동
nangkyeonglim Sep 16, 2023
16e602c
fix: dragEnd 시 버블링으로 인해 2번 API 요청 되는 현상 수정
nangkyeonglim Sep 19, 2023
fea808c
refactor: 카테고리 순서 변경 시 테이블 페이지의 글을 받아오는 쿼리 무효화 제거
nangkyeonglim Sep 19, 2023
5f4d2a4
refactor: 드래그 종류 확인 함수를 값으로 변경하고 네이밍 수정
nangkyeonglim Sep 19, 2023
7df8f3c
refactor: 마지막 드래그 영역 아이디 상수화
nangkyeonglim Sep 19, 2023
354e868
refactor: 드래그 영역 색 상수화
nangkyeonglim Sep 19, 2023
cb5f0bc
docs: 동작 이해를 위한 주석 추가
nangkyeonglim Sep 19, 2023
c24e73b
refactor: 기본 카테고리 id 로컬스토리지에서 받아오도록 변경
nangkyeonglim Sep 19, 2023
6db984a
refactor: `decideDraggingTarget` 조건문 분리
nangkyeonglim Sep 19, 2023
0baaa73
refactor: `handleDragStart` 버블링 막는 조건문 변경
nangkyeonglim Sep 19, 2023
56c03f6
refactor: 드래그 위치 인덱스 상수화
nangkyeonglim Sep 19, 2023
2939b6f
refactor: 드래그 불가한 조건문 명확하게 변경
nangkyeonglim Sep 20, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
11 changes: 8 additions & 3 deletions frontend/src/apis/category.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,8 @@ import {
AddCategoriesRequest,
GetCategoriesResponse,
GetCategoryDetailResponse,
PatchCategoryArgs,
UpdateCategoryOrderArgs,
UpdateCategoryTitleArgs,
} from 'types/apis/category';

// POST: 카테고리 추가
Expand All @@ -17,8 +18,12 @@ export const getCategories = (): Promise<GetCategoriesResponse> => http.get(cate
export const getWritingsInCategory = (categoryId: number): Promise<GetCategoryDetailResponse> =>
http.get(`${categoryURL}/${categoryId}`);

// PATCH: 카테고리 수정
export const patchCategory = ({ categoryId, body }: PatchCategoryArgs) =>
// PATCH: 카테고리 이름 수정
export const updateCategoryTitle = ({ categoryId, body }: UpdateCategoryTitleArgs) =>
http.patch(`${categoryURL}/${categoryId}`, { json: body });

// PATCH: 카테고리 순서 수정
export const updateCategoryOrder = ({ categoryId, body }: UpdateCategoryOrderArgs) =>
http.patch(`${categoryURL}/${categoryId}`, { json: body });

// DELETE: 카테고리 삭제
Expand Down
5 changes: 5 additions & 0 deletions frontend/src/apis/writings.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import type {
GetWritingResponse,
PublishWritingArgs,
UpdateWritingTitleArgs,
UpdateWritingOrderArgs,
} from 'types/apis/writings';

// 글 생성(글 업로드): POST
Expand Down Expand Up @@ -47,3 +48,7 @@ export const getDetailWritings = (categoryId: number): Promise<GetDetailWritings
// 글 제목 변경: PATCH
export const updateWritingTitle = ({ writingId, body }: UpdateWritingTitleArgs) =>
http.patch(`${writingURL}/${writingId}`, { json: body });

// 글 제목 순서 변경: PATCH
export const updateWritingOrder = ({ writingId, body }: UpdateWritingOrderArgs) =>
http.patch(`${writingURL}/${writingId}`, { json: body });
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ const S = {
border-radius: 4px;

&:hover {
background-color: ${({ theme }) => theme.color.gray4};
background-color: ${({ theme }) => theme.color.gray3};
}
`,

Expand Down
4 changes: 2 additions & 2 deletions frontend/src/components/Category/Category/Category.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ const Category = ({ categoryId, categoryName, isDefaultCategory }: Props) => {
isError,
setIsError,
} = useUncontrolledInput();
const { patchCategory, deleteCategory } = useCategoryMutation();
const { updateCategoryTitle, deleteCategory } = useCategoryMutation();
const { goWritingTablePage } = usePageNavigate();
const toast = useToast();

Expand All @@ -38,7 +38,7 @@ const Category = ({ categoryId, categoryName, isDefaultCategory }: Props) => {

validateCategoryName(categoryName);

patchCategory({
updateCategoryTitle({
categoryId,
body: {
categoryName,
Expand Down
12 changes: 10 additions & 2 deletions frontend/src/components/Category/Header/Header.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,11 @@ import { useToast } from 'hooks/@common/useToast';
import { getErrorMessage } from 'utils/error';
import { validateCategoryName } from 'utils/validators';

const Header = () => {
type Props = {
onCategoryAdded: () => void;
};

const Header = ({ onCategoryAdded }: Props) => {
const {
inputRef,
escapeInput: escapeAddCategory,
Expand All @@ -18,7 +22,7 @@ const Header = () => {
isError,
setIsError,
} = useUncontrolledInput();
const { addCategory } = useCategoryMutation();
const { addCategory } = useCategoryMutation(onCategoryAdded);
const toast = useToast();

const requestAddCategory: KeyboardEventHandler<HTMLInputElement> = async (e) => {
Expand Down Expand Up @@ -66,11 +70,15 @@ export default Header;

const S = {
Header: styled.header`
position: sticky;
top: 0;
z-index: 1;
display: flex;
justify-content: space-between;
align-items: center;
height: 3.6rem;
padding: 0.8rem;
background-color: ${({ theme }) => theme.color.spaceBackground};
font-size: 1.2rem;
font-weight: 400;
`,
Expand Down
110 changes: 93 additions & 17 deletions frontend/src/components/Category/Item/Item.tsx
Original file line number Diff line number Diff line change
@@ -1,34 +1,110 @@
import Accordion from 'components/@common/Accordion/Accordion';
import { useState } from 'react';
import { DragEvent, useState } from 'react';
import Category from '../Category/Category';
import WritingList from '../WritingList/WritingList';
import styled, { css } from 'styled-components';
import { INDEX_POSITION } from 'constants/drag';

type Props = {
categoryId: number;
categoryName: string;
isDefaultCategory: boolean;
draggingIndexList: number[];
dragOverIndexList: number[];
onDragStart: (...ids: number[]) => (e: DragEvent) => void;
onDragEnter: (...ids: number[]) => (e: DragEvent) => void;
onDragEnd: (e: DragEvent) => void;
isWritingDragging: boolean;
};

const Item = ({ categoryId, categoryName, isDefaultCategory }: Props) => {
const Item = ({
categoryId,
categoryName,
isDefaultCategory,
draggingIndexList,
dragOverIndexList,
onDragStart,
onDragEnter,
onDragEnd,
isWritingDragging,
}: Props) => {
const [isOpen, setIsOpen] = useState(false);

const decideDraggingTarget = () => {
const isCategoryDragOverTarget =
dragOverIndexList.length === 1 &&
categoryId === dragOverIndexList[INDEX_POSITION.CATEGORY_ID];
const isWritingDragOverTarget =
isCategoryDragOverTarget && dragOverIndexList.length !== draggingIndexList.length;

if (isWritingDragOverTarget) return 'writing';
if (isCategoryDragOverTarget) return 'category';
return 'none';
};

return (
<Accordion.Item key={categoryId}>
<Accordion.Title
onIconClick={() => setIsOpen((prev) => !prev)}
aria-label={`${categoryName} 카테고리 왼쪽 사이드바에서 열기`}
>
<Category
categoryId={categoryId}
categoryName={categoryName}
isDefaultCategory={isDefaultCategory}
/>
</Accordion.Title>
<Accordion.Panel>
<WritingList categoryId={categoryId} isOpen={isOpen} />
</Accordion.Panel>
</Accordion.Item>
<S.DragContainer
draggable={!isDefaultCategory}
$draggingTarget={decideDraggingTarget()}
onDragStart={onDragStart(categoryId)}
onDragEnter={onDragEnter(categoryId)}
jeonjeunghoon marked this conversation as resolved.
Show resolved Hide resolved
onDragEnd={onDragEnd}
>
<Accordion.Item key={categoryId}>
<Accordion.Title
onIconClick={() => setIsOpen((prev) => !prev)}
aria-label={`${categoryName} 카테고리 왼쪽 사이드바에서 열기`}
>
<Category
categoryId={categoryId}
categoryName={categoryName}
isDefaultCategory={isDefaultCategory}
/>
</Accordion.Title>
<Accordion.Panel>
<WritingList
categoryId={categoryId}
isOpen={isOpen}
dragOverIndexList={dragOverIndexList}
onDragStart={onDragStart}
onDragEnter={onDragEnter}
onDragEnd={onDragEnd}
isWritingDragging={isWritingDragging}
/>
</Accordion.Panel>
</Accordion.Item>
</S.DragContainer>
);
};

export default Item;

const S = {
DragContainer: styled.div<{
$draggingTarget: 'category' | 'writing' | 'none';
}>`
position: relative;
border-top: 0.4rem solid transparent;

${({ $draggingTarget }) =>
$draggingTarget === 'category' &&
css`
border-radius: 0;
border-top: 0.4rem solid ${({ theme }) => theme.color.dragArea};
`}

${({ $draggingTarget }) =>
$draggingTarget === 'writing' &&
css`
&::before {
content: '';
pointer-events: none;
position: absolute;
width: 100%;
height: 100%;
border-radius: 4px;
background-color: ${({ theme }) => theme.color.dragArea};
}
`}
`,
};
67 changes: 51 additions & 16 deletions frontend/src/components/Category/List/List.tsx
Original file line number Diff line number Diff line change
@@ -1,28 +1,63 @@
import Accordion from 'components/@common/Accordion/Accordion';
import { useCategories } from './useCategories';
import Item from '../Item/Item';
import { useDragAndDrop } from 'components/Category/useDragAndDrop';
import styled, { css } from 'styled-components';
import { INDEX_POSITION, LAST_DRAG_SECTION_ID } from 'constants/drag';

const List = () => {
const { categories } = useCategories();
const {
draggingIndexList,
dragOverIndexList,
handleDragEnd,
handleDragEnter,
handleDragStart,
isCategoryDragging,
isWritingDragging,
} = useDragAndDrop();

if (!categories) return null;

return (
<>
{categories ? (
<Accordion>
{categories.map((category, index) => {
return (
<Item
key={category.id}
categoryId={category.id}
categoryName={category.categoryName}
isDefaultCategory={Boolean(index === 0)}
/>
);
})}
</Accordion>
) : null}
</>
<Accordion>
{categories.map((category, index) => {
return (
<Item
key={category.id}
categoryId={category.id}
categoryName={category.categoryName}
isDefaultCategory={Boolean(index === 0)}
draggingIndexList={draggingIndexList}
dragOverIndexList={dragOverIndexList}
onDragStart={handleDragStart}
onDragEnter={handleDragEnter}
onDragEnd={handleDragEnd}
isWritingDragging={isWritingDragging}
/>
);
})}
<S.DragLastSection
onDragEnter={handleDragEnter(LAST_DRAG_SECTION_ID)}
$isDragOverTarget={
isCategoryDragging &&
dragOverIndexList[INDEX_POSITION.CATEGORY_ID] === LAST_DRAG_SECTION_ID
}
/>
</Accordion>
);
};

export default List;

const S = {
DragLastSection: styled.div<{ $isDragOverTarget: boolean }>`
height: 0.4rem;
background-color: transparent;
${({ $isDragOverTarget }) =>
$isDragOverTarget &&
css`
background-color: ${({ theme }) => theme.color.dragArea};
`};
`,
};
8 changes: 6 additions & 2 deletions frontend/src/components/Category/Section/Section.tsx
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
import { styled } from 'styled-components';
import Header from '../Header/Header';
import List from '../List/List';
import { throttle } from 'utils/functionRegulator';
import { useScroll } from 'hooks/@common/useScroll';

const Section = () => {
const { scrollRef, scrollToBottom, scrollInArea } = useScroll();

return (
<S.Section>
<Header />
<S.Section ref={scrollRef} onDrag={throttle(scrollInArea(100, 100), 100)}>
<Header onCategoryAdded={scrollToBottom} />
<List />
</S.Section>
);
Expand Down
Loading