Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[#54] Drag & Drop UI 구현 #55

Merged
merged 30 commits into from
Jan 4, 2024
Merged

[#54] Drag & Drop UI 구현 #55

merged 30 commits into from
Jan 4, 2024

Conversation

han-kimm
Copy link
Contributor

@han-kimm han-kimm commented Jan 4, 2024

Feature

Drag & Drop UI

: /dashboard/[boardId] 페이지에서 사용하는 드래그앤 드롭 기능입니다.

View

dragandend

youdame and others added 29 commits January 2, 2024 13:46
Co-authored-by: youdame <youdame@users.noreply.github.com>
Co-authored-by: Jisoo Ahn <An-jisu@users.noreply.github.com>
@han-kimm han-kimm requested a review from An-jisu January 4, 2024 02:24
@han-kimm han-kimm requested review from youdame and ayden94 January 4, 2024 02:24
@han-kimm han-kimm self-assigned this Jan 4, 2024
@han-kimm han-kimm linked an issue Jan 4, 2024 that may be closed by this pull request
Comment on lines +16 to +17
if (typeof result === "function") {
return result(obj);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sender 내부에서 makeUrl 함수를 호출 할 때 각각의 파라미터 순서를 지키는게 불편해서
객체로 받아서 내부에서 처리하도록 했습니다~

@@ -47,7 +41,7 @@ class Api {
},
});
const data = await res.json();
return { status: res.status, data };
return { status: res.status, data, message: data.message };
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

서버에서 오류 메시지가 있을 경우, 메시지를 담는 속성을 추가해두었습니다.

@@ -59,11 +53,11 @@ class Api {
});

if (res.status === 204) {
return { status: res.status, data: null };
return { status: res.status, data: null as any };
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

delete 일때는 데이터가 안 담겨져서 오는데 data 속성을 null 로 해도 타입 통과가 안돼서, 일단은 any 선언했습니다.

Comment on lines +101 to 104
.imageWrapper {
width: 100%;
height: auto;
}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

카드에 담긴 imge를 반응형으로 창 크기에 따라서 너비가 커지게 만들었습니다.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

화긴요

Comment on lines +35 to +45
const entireData: EntireData = {
cards: {},
columns: {},
columnOrder: [],
};

for (const value of columnData) {
entireData.columnOrder.push(value.id);
entireData.columns[value.id] = value;
}

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

columnOrder의 순서대로 column 컴포넌트를 mapping하고, columnId로 entireData.columns[columnId]로 해당 칼럼의 데이터를 찾아옵니다. card 데이터는 entireData.cards[columnId] 값으로 찾아옵니다.

@@ -82,7 +82,7 @@ const Card = ({ data, index, columnTitle, setCardList }: CardProps) => {
data={data}
columnTitle={columnTitle}
handleModalClose={handleCardModalToggle}
setCardList={setCardList}
setEntireList={setEntireList}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이건 결국 그럼 모든 컬럼에 있는 모든 카드를 하나의 스테이트에서 가져오기로 한 거군요

<Button buttonType="plus_icon" color="white" onClick={handleCreateModalToggle}>
<ChipPlus size="lg" />
</Button>
{entireList.cards[columnId]?.map((card, index) => (
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[ cards: {
  columnA : [{컬럼 A 카드 A}, {컬럼 A 카드 B}],
  columnB : [{컬럼 B 카드 A}, {컬럼 B 카드 B}],
  columnC : [{컬럼 C 카드 A}, {컬럼 C 카드 B}],
 }]

대충 이런 형태인가요?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@ayden94
맞아요~

const newCardList = prevValue.filter((card) => card.id !== data.id);
return newCardList;
});
setEntireList(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

모든 카드 부분을 이거로 바꾸다니.. 갱장한 노가다였겠군요 ㄷㄷ


setEntireList((prev) => ({
...prev,
cards: { ...prev.cards, [dragStartColumnId]: startCardList, [dragEndColumnId]: endCardList },
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이게 그 문제의 코드군요 ㅋㅋㅋㅋㅋ

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

미친럼처럼 spread만 남발..


useEffect(() => {
setMounted(true);
setMount(true);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이거는 혹시 다른 코드랑 중복되는, 가령 라이브러리의 예약어라던가, 그런 게 있었던 건가요??

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

아뇨?? 아마 코드가 지워졌는데 제가 mounted 랑 mount 헷갈린거같아요

(a, b) => Number(new Date(a.createdAt)) - Number(new Date(b.createdAt))
),
[dragEndColumnId]: endCardList.sort((a, b) => Number(new Date(a.createdAt)) - Number(new Date(b.createdAt))),
},
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

키아… 타이거즈

}
};
}, [dragRef]);
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

근데 이 훅은 뭘 리턴하나요??

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리턴은 안하고 Ref로 요리조리 씹뜯즐 하고 이벤트리스너만 부착합니다

"바라는 일이 되기를 바라.",
"오늘도 좋은 하루 되세요!",
"할 수 있는 만큼만 하기.",
];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이건 헤더의 빈 부분에서 랜덤으로 알아서 튀어나오는 건가요?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

네 랜덤메시지 그냥 이스터에그로 넣어봣어요

const mouseTouchDown = (e: MouseEvent | TouchEvent) => {
// 마우스를 눌렀을 때 이벤트 함수
const target = e.target as HTMLElement;
if (target.className.includes("Card")) return;
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

카드를 눌렀을 때는 스와이핑이 일어나지 않고,
드래그앤 드롭이 일어나도록 했습니다.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

코드는 추가해놨는데 지금 ModalWrapper 구조에서 스크롤이 막혀있어서~ 사용할 일이 없네요!
진호님 설계 굿

Comment on lines +146 to +155
if (prevColumnId === newColumnId) {
setEntireList((prev) => ({
...prev,
cards: {
...prev.cards,
[newColumnId]: prev.cards[newColumnId].map((v) => (v.id === editedCard.id ? editedCard : v)),
},
}));
return;
}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

상태관리 툴이 없으니까 setState 구조가 괴랄하네요

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

수정하는 경우에는 해당 카드의 아이디를 찾아서, map 메서드로 response로 응답받은 카드 정보로 바꿔치기 합니다.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이 코드는 칼럼 외의 내용을 수정할 경우에 적용됩니다.

Comment on lines +157 to +166
setEntireList((prev) => ({
...prev,
cards: {
...prev.cards,
[prevColumnId]: prev.cards[prevColumnId].filter((v) => v.id !== editedCard.id),
[newColumnId]:
(prev.cards[newColumnId].push(editedCard),
prev.cards[newColumnId].sort((a, b) => Number(new Date(a.createdAt)) - Number(new Date(b.createdAt)))),
},
}));
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

칼럼까지 수정할 경우 이 코드를 실행합니다.

항상 createdAt 기준으로 정렬을 하기 때문에, 서버상태와 일치합니다.

<Button buttonType="plus_icon" color="white" onClick={handleCreateModalToggle}>
<ChipPlus size="lg" />
</Button>
{entireList.cards[columnId]?.map((card, index) => (
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@ayden94
맞아요~


setEntireList((prev) => ({
...prev,
cards: { ...prev.cards, [dragStartColumnId]: startCardList, [dragEndColumnId]: endCardList },
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

미친럼처럼 spread만 남발..


useEffect(() => {
setMounted(true);
setMount(true);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

아뇨?? 아마 코드가 지워졌는데 제가 mounted 랑 mount 헷갈린거같아요

}
};
}, [dragRef]);
}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

리턴은 안하고 Ref로 요리조리 씹뜯즐 하고 이벤트리스너만 부착합니다

"바라는 일이 되기를 바라.",
"오늘도 좋은 하루 되세요!",
"할 수 있는 만큼만 하기.",
];
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

네 랜덤메시지 그냥 이스터에그로 넣어봣어요

@han-kimm han-kimm merged commit ce73577 into develop Jan 4, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[#54] Drag & Drop UI 구현
3 participants