写.真
⚠️ 공부한 내용을 멋대로 기록한 것입니다. 작성된 정보는 순 엉터리정보일 수 있으니 참고하실 때 조심하세요.
현재 서비스중인 앱에서 네트워크 상태를 확인하여 처리를 하는 요구사항이 들어왔다.
요구사항은 다음과 같다.
https://www.npmjs.com/package/@react-native-community/netinfo/v/6.0.6
네트워크 상태를 감지하기 위해 해당 라이브러리를 사용하였다. 이 라이브러리를 사용하면 인터넷 연결 여부, 네트워크 유형(예: WiFi, 셀룰러, 유선, 없음 등), 추가적인 연결 세부 정보(예: 속도나 IP 주소)를 알 수 있어 앱의 네트워크 기반 기능을 효율적으로 관리할 수 있다.
사용방법은 간단하다.
React hook과 같이 사용할 수 있다.
1import NetInfo from "@react-native-community/netinfo"; 2 3const TextComponent = () => { 4 const netInfo = useNetInfo(); 5 6 return ( 7 <View> 8 <Text>Type: {netInfo.type}</Text> 9 <Text>Is Connected? {netInfo.isConnected?.toString()}</Text> 10 </View> 11 ); 12} 13// console.log로 출력한 결과는 다음과 같다. 14# iOS 15{ 16 "details":{ 17 "ipAddress":"00.00.00.00", 18 "isConnectionExpensive":false, 19 "subnet":"000.000.000.0" 20 }, 21 "isConnected":true, 22 "isInternetReachable":true, 23 "type":"wifi" 24} 25 26# Android 27{ 28 "details":{ 29 "bssid":"00:00:00:00:00:00", 30 "frequency":2447, 31 "ipAddress":"00.0.0.00", 32 "isConnectionExpensive":false, 33 "linkSpeed":1, 34 "rxLinkSpeed":2, 35 "strength":99, 36 "subnet":"000.000.000.0", 37 "txLinkSpeed":1 38 }, 39 "isConnected":true, 40 "isInternetReachable":true, 41 "isWifiEnabled":true, 42 "type":"wifi" 43}
가장 최상단에 위치한 App.tsx에 네트워크 상태 변화를 감지하여 toast알림을 주는 방식을 선택하였다.
초기 네트워크 상태를 설정하고 네트워크 상태 변화를 state로 관리하는 custom hook을 생성하였다.
react-native-netinfo를 사용하는 가장 기본적인 훅 사용 방법이다.
1const useNetworkStatus = () => { 2 const [isConnected, setIsConnected] = useState<boolean | null>(null); 3 4 const networkAlert = () => { 5 if (isConnected === false) { 6 Alert.alert('네트워크 오류', '네트워크 연결을 확인해주세요.'); 7 throw new Error('네트워크 연결 에러'); 8 } 9 return; 10 }; 11 12 useEffect(() => { 13 const updateNetworkStatus = (state: NetInfoState) => { 14 console.log('Network status changed:', state.isConnected); 15 setIsConnected(state.isConnected); 16 }; 17 18 // 초기 네트워크 상태 설정 19 NetInfo.fetch().then(updateNetworkStatus); 20 21 const unsubscribe = NetInfo.addEventListener(updateNetworkStatus); 22 23 // 컴포넌트 언마운트 시 구독 해제 24 return () => { 25 unsubscribe(); 26 }; 27 }, []); 28 29 return { isConnected, setIsConnected, networkAlert }; 30}; 31 32export default useNetworkStatus; 33
해당 요구사항도 간단하다. 위의 커스텀훅을 활용하여 이벤트가 일어나는 함수에서 isConnected가 false일 경우 Alert를 띄우도록 구현하였다.
해당 요구사항을 구현을 하기 위해서는 store page network modal, user page network modal 등 다양한 곳에 사용할 것을 확인하여 모듈화를 진행하였다.
모달내에는 네트워크 연결이 끊겼을 시 재시도 버튼이 있어야하고, 재시도 버튼을 눌렀을 때는 해당 페이지의 데이터를 refetch 시켜야 한다. 재사용 가능한 컴포넌트를 만들고 props로 refetch action을 받기로 했다.
1interface IUseNetworkRetryProps { 2 onRetryAction: () => void; 3 isFullContainer?: boolean; 4} 5 6const useNetworkRetry = ({ 7 onRetryAction, 8 isFullContainer, 9}: IUseNetworkRetryProps) => { 10 const netInfo = useNetInfo(); 11 const modalAction = useModalAction(); 12 13 const onRetryConnect = () => { 14 onRetryAction(); 15 }; 16 17 useFocusEffect( 18 React.useCallback(() => { 19 if (netInfo.isConnected === false) { 20 modalAction.showModal( 21 <RetryModal 22 onRetry={onRetryConnect} 23 isFullContainer={isFullContainer} 24 />, 25 ); 26 27 Toast.show('네트워크 연결을 확인해주세요.', Toast.SHORT); 28 return; 29 } 30 return () => console.log('unmount'); 31 }, [netInfo.isConnected]), 32 ); 33}; 34 35export default useNetworkRetry;
네트워크가 연결이 되었을 경우 해당 훅을 통해 감지를 하고, retry 버튼을 눌렀을 때 기존의 캐싱된 데이터를 remove 시키고 refetch 받도록 하였다.
캐싱된 데이터를 삭제시켜준 이유는 이전의 캐싱된 데이터가 있다면 캐싱된 데이터를 보여주게 되는데 네트워크가 불안정할 경우 기존의 데이터를 최신 데이터로 느낄 수 있기 때문에 캐싱 데이터를 remove 시켜주었다.
그리고 네트워크 연결이 끊겼을 경우 재시도 버튼을 누르면 modal이 꺼지지 않게 처리하고 toast 메시지를 통해 네트워크 연결 상태를 유저에게 인지시키는 방식을 사용하였다.
만약 refetch가 성공할 경우 modal을 하이드 시켜주었다.
1// store query 2export const useStoreViewModel = () => { 3 const queryClient = useQueryClient(); 4 const storeId = useStoreIdx(); 5 6 const { 7 data: store, 8 isLoading, 9 refetch: storeRefetch, 10 } = useFetchStoreQuery(storeId, { 11 enabled: true, 12 }); 13 14 const removeStoreQuery = () => { 15 queryClient.removeQueries({ queryKey: ['store', storeId] }); 16 }; 17 18 return { 19 store, 20 isLoading, 21 storeRefetch, 22 removeStoreQuery, 23 }; 24}; 25 26// component use 27 28 useNetworkRetry({ 29 onRetryAction: async () => { 30 try { 31 await removeStoreQuery(); 32 33 const result = await storeRefetch(); 34 35 if (result.isSuccess && result.data) { 36 modalAction.hideModal(); 37 } else { 38 Toast.show('네트워크 연결을 확인해주세요.', Toast.SHORT); 39 } 40 } catch (error) { 41 console.log(error); 42 Toast.show('네트워크 연결을 확인해주세요.', Toast.SHORT); 43 } 44 }, 45 }); 46 47
home → user → home으로 이동하여 네트워크를 끊어보니 home page에서 모달이 띄워지는 문제를 겪었다.
useEffect를 통해 isConnected일 경우 modal을 띄워주는 형식으로 구현을 하였는데, 웹 개발을 하면서 useEffect hook에 대한 이해도가 쌓이다 보니 해당 문제를 겪은 것이다.
react-native의 navigator스택 또는 탭 내의 컴포넌트는 사용자가 네비게이션을 통해 이동할 때마다 마운트 및 언마운트되지 않고, 대신 포커스 상태가 변경된다. useEffect는 컴포넌트의 마운트와 언마운트에 반응하기 때문에, 화면이 포커스를 받거나 잃을 때 반응하지 않는다.
stack에 user 화면이 쌓여 useEffect가 감지되어 modal이 출력되는 것이였다. 해당 페이지가 Focus되었을 때 사이드 이펙트를 처리하기 위해서는 useEffect 대신 react-native hook인 useFocusEffect를 사용해야 했다.
react와 react-native 비슷하면서 다르다. 사이드 이펙트에 대한 무지가 앱 전체에 영향을 많이 주는거 같아 항상 조심스럽다.