
๐ค ์๋ก
ํ์ฌ ๋๋ง๋ฌด ํ๋ก์ ํธ๋ฅผ ์งํ ์ค์ธ๋ฐ, ์ต๊ทผ ๊ฒ์๋ฌผ์ ์ด๋ป๊ฒ ๋ณด์ฌ์ค ์ ์์๊น ๊ณ ๋ฏผ์ ํ๋ค๊ฐ ์ธ์คํ๊ทธ๋จ์ ์คํ ๋ฆฌ ๊ธฐ๋ฅ์ด ์๊ฐ๋ฌ๊ณ ํด๋น ๋ถ๋ถ์ ์ธํฐ๋์ ์ ์ฐจ์ฉํ๋ ค ํ๋ค.
๐ฅ ๋ชฉํ

์ธ์คํ๊ทธ๋จ ์๋จ์๋ ์ต๊ทผ ์คํ ๋ฆฌ๋ฅผ ์ฌ๋ฆฐ ์ฌ๋์ ํ๋กํ์ด ๋ฌ๋ค. ํฐ์นํด์ ์ข์ฐ๋ฅผ ์์ ๋กญ๊ฒ ์์ง์ผ ์ ์๋ค.(๋ชจ๋ฐ์ผ ๊ธฐ์ค) ํด๋น ์ธํฐ๋์
์ฒ๋ผ ๋๋๊ทธ๋ก ํ๋กํ์ ์์ง์ด๋ ๊ฒ์ ๊ตฌํํ๋ ค ํ๋ค.
๐ฉ๐ป๐ป ๊ตฌํ
์ ์ฒด ์ฝ๋
const getFriends = async () => {
...
const [isMouseDown, setIsMouseDown] = useState<boolean>(false);
const [startX, setStartX] = useState<number>(0);
const [scrollLeft, setScrollLeft] = useState<number>(0);
...
const handleMouseDown = (
e: React.MouseEvent<HTMLUListElement, MouseEvent>
) => {
setIsMouseDown(true);
setStartX(e.clientX - (listRef.current?.offsetLeft || 0));
setScrollLeft(listRef.current?.scrollLeft || 0);
};
const handleMouseLeave = () => {
setIsMouseDown(false);
};
const handleMouseMove = (
e: React.MouseEvent<HTMLUListElement, MouseEvent>
) => {
if (!isMouseDown) return;
e.preventDefault();
const x = e.clientX - (listRef.current?.offsetLeft || 0);
const walk = x - startX; // ์ฌ๊ธฐ์ walk ๊ฐ์ผ๋ก ์คํฌ๋กค ์์ ์กฐ์ ํฉ๋๋ค
if (listRef.current) {
listRef.current.scrollLeft = scrollLeft - walk;
}
};
...
return (
<div>
<p className="text-3xl font-semibold">์น๊ตฌ</p>
<ul
ref={listRef}
className="flex justify-start py-5 overflow-x-hidden overflow-y-hidden scrolling-touch select-none w-full cursor-grab"
onMouseDown={handleMouseDown}
onMouseLeave={handleMouseLeave}
onMouseUp={handleMouseLeave}
onMouseMove={handleMouseMove}
>
{friends &&
friends.map((f) => {
return (
<li
key={f.id}
className="m-3 p-1 flex-shrink-0 bg-gradient-to-tr from-violet-600 to-yellow-300 rounded-full"
>
<img
src={f.image}
className="w-20 h-20 rounded-full transform transition hover:-rotate-6 cursor-pointer"
alt="profile"
/>
</li>
);
})}
</ul>
...
</div>
);
}
๋ถ๊ฐ ์ค๋ช
const handleMouseDown = (
e: React.MouseEvent<HTMLUListElement, MouseEvent>
) => {
setIsMouseDown(true);
setStartX(e.clientX - (listRef.current?.offsetLeft || 0));
setScrollLeft(listRef.current?.scrollLeft || 0);
};
const handleMouseLeave = () => {
setIsMouseDown(false);
};
const handleMouseMove = (
e: React.MouseEvent<HTMLUListElement, MouseEvent>
) => {
if (!isMouseDown) return;
e.preventDefault();
const x = e.clientX - (listRef.current?.offsetLeft || 0);
const walk = x - startX;
if (listRef.current) {
listRef.current.scrollLeft = scrollLeft - walk;
}
};

- handleMouseDown
์ฌ์ฉ์๊ฐ ๋ง์ฐ์ค ๋ฒํผ์ ๋๋ฅด๊ณ ์์ ๋ ํธ์ถ๋๋ ํจ์๋ก, setIsMouseDown(true)๋ฅผ ์ฌ์ฉํ์ฌ ์ฌ์ฉ์๊ฐ ๋ง์ฐ์ค๋ฅผ ๋๋ฅด๊ณ ์์์ ๋ํ๋ธ๋ค. e.clientX๋ ๋ธ๋ผ์ฐ์ ๋ทฐํฌํธ๋ฅผ ๊ธฐ์ค์ผ๋ก ๋ง์ฐ์ค ํฌ์ธํฐ์ ์ํ ์์น(x์ขํ)๋ฅผ ๊ฐ์ ธ์ค๊ณ , listRef.current?.offsetLeft๋ ul ์์์ ์ผ์ชฝ ๊ฒฝ๊ณ๊ฐ ๋ทฐํฌํธ์ ์ผ์ชฝ ๊ฒฝ๊ณ๋ก๋ถํฐ ์ผ๋ง๋ ๋จ์ด์ ธ์๋์ง(ํฝ์ ์)๋ฅผ ๋ํ๋ธ๋ค. - || 0์ ๊ฐ์ด ์กด์ฌํ์ง ์์ ๋ ๋ฐ์ํ ์ ์๋ ์ค๋ฅ๋ฅผ ๋ฐฉ์งํ๋ค.
๐ค ๋ธ๋ผ์ฐ์ ๋ทฐํฌํธ(viewport)๋?
์น ๋ธ๋ผ์ฐ์ ์ฐฝ ๋ด์์ ์ค์ ๋ก ์น ํ์ด์ง์ ๋ด์ฉ์ด ํ์๋๋ ์์ญ
- handleMouseMove
๋ง์ฐ์ค๊ฐ ul ์์ ์์์ ์์ง์ผ ๋ ํธ์ถ๋๋ ํจ์์ด๋ค. e.preventDefault()๋ ๊ธฐ๋ณธ ๋ง์ฐ์ค ์ด๋ฒคํธ๋ฅผ ๋ง์ ์ฌ์ฉ์ ์ ์ ๋๋๊ทธ ๋์๋ง ๋ฐ์ํ๋ค. ์ฒ์ ๋ง์ฐ์ค๋ฅผ ํด๋ฆญํ์ ๋ ์ ์ฅํ ๋ง์ฐ์ค ์์น์ธ startX๋ก๋ถํฐ ๋จ์ด์ง ๊ฑฐ๋ฆฌ๋ฅผ ๊ณ์ฐํด walk์ ์ ์ฅํ๋ค. ๋ง์ง๋ง์ผ๋ก listRef.current.scrollLeft๋ฅผ ์ ๋ฐ์ดํธํ์ฌ ul ์์๋ฅผ ์คํฌ๋กคํ๋ค. scrollleft๋ ์ฒ์ ๋ง์ฐ์ค๋ฅผ ๋๋ ์ ๋ ์คํฌ๋กค์ด ์ด๋์ ์์๋์ง๋ฅผ ๋ํ๋ด๊ณ , walk๋ ๋ง์ฐ์ค๋ฅผ ์์ง์ธ ๊ฑฐ๋ฆฌ๋ฅผ ๋ํ๋ด์ด, scrollLeft-walk๋ ์๋ก์ด ์คํฌ๋กค ์์น๋ฅผ ๊ณ์ฐํ๊ณ listRef.current.scrollLeft๋ฅผ ์ค์ ํ์ฌ ์คํฌ๋กค์ด ์์ง์ด๊ฒ ๋๋ค.
๐ค Element.scrollLeft
์์์ ์ฝํ ์ธ ๊ฐ ์ผ์ชฝ ๊ฐ์ฅ์๋ฆฌ์์ ์คํฌ๋กค๋๋ ํฝ์ ์๋ฅผ ๊ฐ์ ธ์ค๊ฑฐ๋ ์ค์ ํ๋ ์์ฑ
๐ค ์ ๋ง์ง๋ง ์ค์ด scrollLeft + walk์ด ์๋๋ผ scrollLeft - walk์ธ๊ฐ?
scrollLeft๋ ์์์ ์คํฌ๋กค ์์น๋ฅผ ๋ํ๋ด๋ฉฐ ์ผ์ชฝ ๋์์ ์์ํ์ฌ ์ค๋ฅธ์ชฝ์ผ๋ก ๊ฐ์๋ก ์ฆ๊ฐํ๋ค. walk์ ๋ง์ฐ์ค๋ฅผ ๋๋ฅธ ์์น์ ํ์ฌ ๋ง์ฐ์ค์ ์์น ์ฌ์ด์ ์ฐจ์ด๋ฅผ ๋ํ๋ธ๋ค. ๋ง์ฝ ์ฌ์ฉ์๊ฐ ๋ง์ฐ์ค๋ฅผ ์ค๋ฅธ์ชฝ์ผ๋ก ๋๋๊ทธํ๋ฉด walk๋ ์์๊ฐ ๋๋ค. ํ์ง๋ง ๋ง์ฐ์ค๋ฅผ ์ค๋ฅธ์ชฝ์ผ๋ก ๋๋๊ทธํ๋ฉด ์คํฌ๋กค์ด ์ผ์ชฝ์ผ๋ก ์ด๋ํด์ผ ํ๊ธฐ ๋๋ฌธ์ scrollLeft๊ฐ์ ๊ฐ์์์ผ์ผ ํ๋ค. ์ฆ, ์คํฌ๋กค ๋ฐฉํฅ๊ณผ ๋ง์ฐ์ค ๋๋๊ทธ ๋ฐฉํฅ์ด ๋ฐ๋์ด๊ธฐ ๋๋ฌธ์ด๋ค.
<ul
ref={listRef}
className="flex justify-start py-5 overflow-x-hidden overflow-y-hidden scrolling-touch w-full cursor-grab"
onMouseDown={handleMouseDown}
onMouseLeave={handleMouseLeave}
onMouseUp={handleMouseLeave}
onMouseMove={handleMouseMove}
>
- w-full๋ก ๊ฐ๋ก ๊ธธ์ด๋ฅผ ์ ํ๊ณ overflow-x-hidden์ผ๋ก ์ ํด์ง ๊ฐ๋ก ๊ธธ์ด๋ณด๋ค ๋ด์ฉ์ด ๋์น๋ฉด ์จ๊น๋๋ค. ๊ทธ๋ฆฌ๊ณ ์์์ ๊ฒฝ๊ณ ์์ ๋ค์ด์ฌ ๊ฒฝ์ฐ์ ์ปค์ ๋ชจ์์ grab์ผ๋ก ์ค์ ํ๋ค.
- onMouseDown, onMouseLeave, onMouseUp, onMouseMove๋ ์น ๋ธ๋ผ์ฐ์ ์์ ๊ธฐ๋ณธ์ ์ผ๋ก ์ ๊ณตํ๋ ๋ง์ฐ์ค ์ด๋ฒคํธ์ด๋ค.
<img
src={f.image}
className="w-20 h-20 rounded-full transform transition hover:-rotate-6 cursor-pointer"
alt="profile"
/>
- hover:-rotate-6์ผ๋ก ์ด๋ฏธ์ง์ hover(์ปค์๋ฅผ ๊ฐ์ ธ๋ค ๋์ ๋)ํ์ ๋ ์ด๋ฏธ์ง๊ฐ ํ์ ํ๋ค.
- cursor-pointer๋ฅผ ์ค์ ํ์ฌ img์ ๊ฒฝ๊ณ ์์ ๋ค์ด์ฌ ๊ฒฝ์ฐ ์ปค์ ๋ชจ์์ด pointer๊ฐ ๋๋ค.
โจ ๊ฒฐ๊ณผ

์ฐธ๊ณ ์๋ฃ
'W > CSS' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
TailwindCSS ์์ํ๊ธฐ (0) | 2024.07.17 |
---|---|
[React/CSS] ๋ชจ๋ฌ ์ฐฝ ๋ฐ๊นฅ์ ํด๋ฆญํ์ ๋ ๋ชจ๋ฌ์ฐฝ ๊บผ์ง๊ฒ ํ๊ธฐ (2) | 2024.07.17 |