From 493c543be20e63b0ee3f7f24f361def2d48ff2af Mon Sep 17 00:00:00 2001 From: yoouyeon Date: Sun, 22 Dec 2024 22:44:49 +0900 Subject: [PATCH] =?UTF-8?q?Feat:=20=EC=8A=A4=ED=81=AC=EB=A1=A4=20=EC=9D=B4?= =?UTF-8?q?=EB=B2=A4=ED=8A=B8=EB=A5=BC=20=EB=AC=B4=EC=8B=9C=ED=95=98?= =?UTF-8?q?=EB=8A=94=20MegaphoneScroller=20=EC=BB=B4=ED=8F=AC=EB=84=8C?= =?UTF-8?q?=ED=8A=B8=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - https://virtuoso.dev/customize-structure/ - https://virtuoso.dev/custom-scroll-container/ - passive 기본값 설정 (true)를 바꾸기 위해 직접 이벤트 리스너를 등록함. --- components/takgu/Layout/MegaPhone.tsx | 74 +++++++++++++++++++++++---- 1 file changed, 63 insertions(+), 11 deletions(-) diff --git a/components/takgu/Layout/MegaPhone.tsx b/components/takgu/Layout/MegaPhone.tsx index e8142cb6c..6f8df0f75 100644 --- a/components/takgu/Layout/MegaPhone.tsx +++ b/components/takgu/Layout/MegaPhone.tsx @@ -1,6 +1,6 @@ import { useRouter } from 'next/router'; -import { useState, useEffect, useRef } from 'react'; -import { Virtuoso, VirtuosoHandle } from 'react-virtuoso'; +import { useState, useEffect, useRef, forwardRef } from 'react'; +import { Virtuoso, VirtuosoHandle, Components } from 'react-virtuoso'; import { Item } from 'types/takgu/itemTypes'; import useAxiosGet from 'hooks/useAxiosGet'; import useInterval from 'hooks/useInterval'; @@ -20,15 +20,7 @@ const adminContent: IMegaphoneContent = { intraId: '관리자', }; -export const MegaphoneItem = ({ content, intraId }: IMegaphoneContent) => { - return ( -
-
{intraId}
-
{content}
-
- ); -}; - +// 메인 확성기 컴포넌트 const Megaphone = () => { const [contents, setContents] = useState([]); const [itemList, setItemList] = useState([]); @@ -99,6 +91,9 @@ const Megaphone = () => { ref={virtuoso} itemContent={(_, data) => } style={{ height: '100%' }} + components={{ + Scroller: MegaphoneScroller, + }} /> )} @@ -106,4 +101,61 @@ const Megaphone = () => { ); }; +// 확성기 아이템 컴포넌트 +export const MegaphoneItem = ({ content, intraId }: IMegaphoneContent) => { + return ( +
+
{intraId}
+
{content}
+
+ ); +}; + +// 유저의 스크롤 이벤트 동작을 제거한 확성기 스크롤러 컴포넌트 +const MegaphoneScroller: Components['Scroller'] = forwardRef( + ({ children, ...props }, ref) => { + const localRef = useRef(null); + + // 전달받은 ref와 내부 ref가 동일한 element를 가리키도록 함 + const combinedRef = (node: HTMLDivElement | null) => { + localRef.current = node; + + if (ref) { + (ref as React.MutableRefObject).current = node; + } + }; + + useEffect(() => { + const element = localRef.current; + + if (element) { + const preventScroll = (e: Event) => { + e.preventDefault(); + e.stopPropagation(); + }; + + // wheel 및 touchmove 이벤트 차단 (pc 및 모바일) + element.addEventListener('wheel', preventScroll, { passive: false }); + element.addEventListener('touchmove', preventScroll, { + passive: false, + }); + + // 컴포넌트 언마운트 시 이벤트 제거 + return () => { + element.removeEventListener('wheel', preventScroll); + element.removeEventListener('touchmove', preventScroll); + }; + } + }, []); + + return ( +
+ {children} +
+ ); + } +); + +MegaphoneScroller.displayName = 'MegaphoneScroller'; + export default Megaphone;