UNPKG

iamport-react-native

Version:

리액트 네이티브용 아임포트 결제/본인인증 연동 라이브러리

227 lines (224 loc) 9.04 kB
"use strict"; import { createRef, useEffect, useRef, useState } from 'react'; import { Alert, Linking, Platform, View } from 'react-native'; import WebView from 'react-native-webview'; import { IMPConst } from "../../constants/index.js"; import viewStyles from "../../styles.js"; import IamportUrl from "../../utils/IamportUrl.js"; import ValidationForPayment from "../../utils/ValidationForPayment.js"; import ErrorOnParams from "../ErrorOnParams/index.js"; import Loading from "../Loading/index.js"; import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime"; function Payment({ userCode, tierCode, data, loading, callback }) { const [webviewSource, setWebviewSource] = useState({ html: IMPConst.WEBVIEW_SOURCE_HTML }); const injectedJavaScript = Platform.OS === 'ios' ? IMPConst.WEBVIEW_IOS_HYUNDAICARD_INJECTED_JAVASCRIPT : ''; const [isWebViewLoaded, setIsWebViewLoaded] = useState(false); const [showLoading, setShowLoading] = useState(true); const webview = /*#__PURE__*/createRef(); const smilepayRef = useRef(false); let redirectUrl = IMPConst.M_REDIRECT_URL; if (data.m_redirect_url !== undefined && data.m_redirect_url.trim() !== '') { redirectUrl = data.m_redirect_url; } else { data.m_redirect_url = redirectUrl; } useEffect(() => { const { pg } = data; if (pg.startsWith('smilepay') && Platform.OS === 'ios' && !smilepayRef.current) { /** * [feature/smilepay] IOS - 스마일페이 대비 코드 작성 * 스마일페이 결제창을 iframe 방식으로 띄우기 때문에 WKWebView에서 서드 파티 쿠키가 허용되지 않아 * WKWebView의 baseUrl을 강제로 스마일페이 URL로 적용 */ setWebviewSource({ ...webviewSource, baseUrl: IMPConst.SMILEPAY_BASE_URL }); smilepayRef.current = true; } }, [data, webviewSource]); useEffect(() => { const handleOpenURL = event => { const { pg, pay_method } = data; if (pay_method === 'trans') { const iamportUrl = new IamportUrl(event.url); /* 나이스 - 실시간 계좌이체 대비 */ if (pg.startsWith('nice')) { const queryParameters = iamportUrl.getQuery(); const scheme = iamportUrl.scheme; let niceTransRedirectionUrl; if (scheme === data.app_scheme?.toLowerCase()) { if (queryParameters.callbackparam1 != null) { niceTransRedirectionUrl = queryParameters.callbackparam1; } } webview.current?.injectJavaScript(` window.location.href = "${niceTransRedirectionUrl}?${iamportUrl.getStringifiedQuery()}"; `); } } }; const subscription = Linking.addEventListener('url', handleOpenURL); return function cleanup() { subscription.remove(); }; }, [data, redirectUrl, webview]); const removeLoadingNeeded = () => { if (showLoading && Platform.OS === 'android') { // 로딩상태. 안드로이드 플랫폼 if (isWebViewLoaded) { // 웹뷰 로드 끝. 리디렉션 방식 return true; } return isIframeWayPayment(); } // IOS return false; }; const isIframeWayPayment = () => { const { pg, pay_method, customer_uid } = data; if (pg.startsWith('html5_inicis') && customer_uid) { // 이니시스 빌링결제 return true; } if (pg.startsWith('mobilians') && pay_method === 'phone') { // 모빌리언스 휴대폰 소액결제 return true; } return pg.startsWith('danal') || pg.startsWith('danal_tpay') || pg.startsWith('smilepay') || pg.startsWith('payco') || pg.startsWith('bluewalnut') || pg.startsWith('settle_acc'); }; const validation = new ValidationForPayment(userCode, loading, callback, data); if (validation.getIsValid()) { const { loadingContainer, webViewContainer } = viewStyles; return /*#__PURE__*/_jsxs(_Fragment, { children: [/*#__PURE__*/_jsx(WebView, { containerStyle: webViewContainer, ref: webview, source: webviewSource, mixedContentMode: 'always', injectedJavaScript: injectedJavaScript, onError: event => { console.warn('Encountered an error loading page', event.nativeEvent); // 탐색이 실패하면 react-native-webview는 바로 오류 화면을 표시하려 한다 // 한편 별도 설정 안 한 WKWebView는 오류를 무시하여 결제를 계속 진행할 수 있는 경우가 있다 // 오류 화면 표시를 막는다 event.preventDefault(); // Domain=NSURLErrorDomain Code=-1005 Description="The network connection was lost." // 지속적으로 폴링하는 카드사(현대카드)의 경우 카드사 앱에서 돌아올 때 HTTP 요청이 강제 중단되어 -1005 오류 발생 // 별도 설정 안 한 WKWebView에서 간헐적으로 자동 리다이렉트가 안 되고 버튼을 누르라고 하는 경우 이 경우이다 // 무시한다 if (Platform.OS === 'ios' || Platform.OS === 'macos' && event.nativeEvent.code === -1005) { return; } // 이외의 경우에는 어떻게 처리해야 할지 확실하지 않지만 우선 알림을 보여 주도록 구현 Alert.alert(`탐색 오류가 발생했습니다\n도메인: ${event.nativeEvent.domain}\n오류 코드: ${event.nativeEvent.code}\n설명: ${event.nativeEvent.description}`); }, onLoadEnd: () => { if (!isWebViewLoaded) { if (data.pg.startsWith('eximbay')) { data.popup = false; } if (tierCode) { webview.current?.injectJavaScript(` setTimeout(function() { IMP.agency("${userCode}", "${tierCode}"); }); `); } else { webview.current?.injectJavaScript(` setTimeout(function() { IMP.init("${userCode}"); }); `); } webview.current?.injectJavaScript(` setTimeout(function() { IMP.request_pay(${JSON.stringify(data)}, function(response) { window.ReactNativeWebView.postMessage(JSON.stringify(response)); }); }); `); setIsWebViewLoaded(true); } // only for Android if (removeLoadingNeeded()) { setShowLoading(false); } } /* PG사가 callback을 지원하는 경우, 결제결과를 받아 callback을 실행한다 */, onMessage: e => { /** * [v1.6.0] 다날의 경우 response에 주문명(name)이 포함되어 있는데 * 주문명에 %가 들어갈 경우, decodeURIComponent시 URI malformed 에러가 발생하는 것 대비해 * 우선 encodeURIComponent를 한 후, decodeURIComponent가 끝나면 * 최종적으로 decodeURIComponent를 한 번 더 한다 */ const encoded = encodeURIComponent(e.nativeEvent.data); const decoded = decodeURIComponent(encoded); const response = JSON.parse(decoded); if (typeof callback === 'function') { callback(response); } }, originWhitelist: ['*'] // https://github.com/facebook/react-native/issues/19986 , sharedCookiesEnabled: true, onShouldStartLoadWithRequest: request => { const { url } = request; // console.log(`url: ${url}`); const iamportUrl = new IamportUrl(url); if (iamportUrl.isPaymentOver(redirectUrl, data)) { if (typeof callback === 'function') { if (data.pg.startsWith('html5_inicis') && data.pay_method === 'trans' && Platform.OS === 'ios') { callback(iamportUrl.getInicisTransQuery(redirectUrl)); } else { callback(iamportUrl.getQuery()); } } return false; } if (iamportUrl.isAppUrl()) { /* 3rd-party 앱 오픈 */ iamportUrl.launchApp(); return false; } if (isWebViewLoaded && showLoading && iamportUrl.isIframeLoaded()) { /** * only for IOS * iframe이 load되면(url이 about:blank 또는 https://service.iamport.kr이 아니면) * webview의 loading 상태를 해제한다 */ setShowLoading(false); } return true; } }), showLoading && /*#__PURE__*/_jsx(View, { style: loadingContainer, children: loading || /*#__PURE__*/_jsx(Loading, {}) })] }); } return /*#__PURE__*/_jsx(ErrorOnParams, { message: validation.getMessage() }); } export default Payment; //# sourceMappingURL=index.js.map