import React, { forwardRef, ReactElement, useCallback, useEffect, useImperativeHandle, useRef, } from 'react'; import { Image, View, ImageSourcePropType, HostComponent } from 'react-native'; import BatchedBridge from 'react-native/Libraries/BatchedBridge/BatchedBridge'; import EventEmitter from 'react-native/Libraries/vendor/emitter/EventEmitter'; import invariant from 'invariant'; import RNCWebView, { Commands, NativeProps } from './RNCWebViewNativeComponent'; import RNCTPWebViewModule from './NativeRNCTPWebViewModule'; import { defaultOriginWhitelist, defaultRenderError, defaultRenderLoading, useWebViewLogic, } from './WebViewShared'; import { AndroidWebViewProps, WebViewSourceUri, type WebViewMessageEvent, type ShouldStartLoadRequestEvent, } from './WebViewTypes'; import styles from './WebView.styles'; const { resolveAssetSource } = Image; const directEventEmitter = new EventEmitter(); const registerCallableModule: (name: string, module: Object) => void = // `registerCallableModule()` is available in React Native 0.74 and above. // Fallback to use `BatchedBridge.registerCallableModule()` for older versions. require('react-native').registerCallableModule ?? BatchedBridge.registerCallableModule.bind(BatchedBridge); registerCallableModule('RNCWebViewMessagingModule', { onShouldStartLoadWithRequest: ( event: ShouldStartLoadRequestEvent & { messagingModuleName?: string } ) => { directEventEmitter.emit('onShouldStartLoadWithRequest', event); }, onMessage: ( event: WebViewMessageEvent & { messagingModuleName?: string } ) => { directEventEmitter.emit('onMessage', event); }, }); /** * A simple counter to uniquely identify WebView instances. Do not use this for anything else. */ let uniqueRef = 0; const WebViewComponent = forwardRef<{}, AndroidWebViewProps>( ( { overScrollMode = 'always', javaScriptEnabled = true, thirdPartyCookiesEnabled = true, scalesPageToFit = true, allowsFullscreenVideo = false, allowFileAccess = false, saveFormDataDisabled = false, cacheEnabled = true, androidLayerType = 'none', originWhitelist = defaultOriginWhitelist, setSupportMultipleWindows = true, setBuiltInZoomControls = true, setDisplayZoomControls = false, nestedScrollEnabled = false, startInLoadingState, onNavigationStateChange, onLoadStart, onError, onLoad, onLoadEnd, onLoadProgress, onHttpError: onHttpErrorProp, onRenderProcessGone: onRenderProcessGoneProp, onMessage: onMessageProp, onOpenWindow: onOpenWindowProp, renderLoading, renderError, style, containerStyle, source, nativeConfig, onShouldStartLoadWithRequest: onShouldStartLoadWithRequestProp, injectedJavaScriptObject, ...otherProps }, ref ) => { const messagingModuleName = useRef( `WebViewMessageHandler${(uniqueRef += 1)}` ).current; const webViewRef = useRef > | null>(null); const onShouldStartLoadWithRequestCallback = useCallback( (shouldStart: boolean, url: string, lockIdentifier?: number) => { if (lockIdentifier) { RNCTPWebViewModule.shouldStartLoadWithLockIdentifier( shouldStart, lockIdentifier ); } else if (shouldStart && webViewRef.current) { Commands.loadUrl(webViewRef.current, url); } }, [] ); const { onLoadingStart, onShouldStartLoadWithRequest, onMessage, viewState, setViewState, lastErrorEvent, onHttpError, onLoadingError, onLoadingFinish, onLoadingProgress, onOpenWindow, onRenderProcessGone, } = useWebViewLogic({ onNavigationStateChange, onLoad, onError, onHttpErrorProp, onLoadEnd, onLoadProgress, onLoadStart, onRenderProcessGoneProp, onMessageProp, onOpenWindowProp, startInLoadingState, originWhitelist, onShouldStartLoadWithRequestProp, onShouldStartLoadWithRequestCallback, }); useImperativeHandle( ref, () => ({ goForward: () => webViewRef.current && Commands.goForward(webViewRef.current), goBack: () => webViewRef.current && Commands.goBack(webViewRef.current), reload: () => { setViewState('LOADING'); if (webViewRef.current) { Commands.reload(webViewRef.current); } }, stopLoading: () => webViewRef.current && Commands.stopLoading(webViewRef.current), postMessage: (data: string) => webViewRef.current && Commands.postMessage(webViewRef.current, data), injectJavaScript: (data: string) => webViewRef.current && Commands.injectJavaScript(webViewRef.current, data), requestFocus: () => webViewRef.current && Commands.requestFocus(webViewRef.current), clearFormData: () => webViewRef.current && Commands.clearFormData(webViewRef.current), clearCache: (includeDiskFiles: boolean) => webViewRef.current && Commands.clearCache(webViewRef.current, includeDiskFiles), clearHistory: () => webViewRef.current && Commands.clearHistory(webViewRef.current), }), [setViewState, webViewRef] ); useEffect(() => { const onShouldStartLoadWithRequestSubscription = directEventEmitter.addListener( 'onShouldStartLoadWithRequest', ( event: ShouldStartLoadRequestEvent & { messagingModuleName?: string; } ) => { if (event.messagingModuleName === messagingModuleName) { // eslint-disable-next-line @typescript-eslint/no-unused-vars const { messagingModuleName: _, ...rest } = event; onShouldStartLoadWithRequest(rest); } } ); const onMessageSubscription = directEventEmitter.addListener( 'onMessage', (event: WebViewMessageEvent & { messagingModuleName?: string }) => { if (event.messagingModuleName === messagingModuleName) { // eslint-disable-next-line @typescript-eslint/no-unused-vars const { messagingModuleName: _, ...rest } = event; onMessage(rest); } } ); return () => { onShouldStartLoadWithRequestSubscription.remove(); onMessageSubscription.remove(); }; }, [messagingModuleName, onMessage, onShouldStartLoadWithRequest]); let otherView: ReactElement | undefined; if (viewState === 'LOADING') { otherView = (renderLoading || defaultRenderLoading)(); } else if (viewState === 'ERROR') { invariant( lastErrorEvent != null, 'lastErrorEvent expected to be non-null' ); if (lastErrorEvent) { otherView = (renderError || defaultRenderError)( lastErrorEvent.domain, lastErrorEvent.code, lastErrorEvent.description ); } } else if (viewState !== 'IDLE') { console.error(`RNCWebView invalid state encountered: ${viewState}`); } const webViewStyles = [styles.container, styles.webView, style]; const webViewContainerStyle = [styles.container, containerStyle]; if (typeof source !== 'number' && source && 'method' in source) { if (source.method === 'POST' && source.headers) { console.warn( 'WebView: `source.headers` is not supported when using POST.' ); } else if (source.method === 'GET' && source.body) { console.warn('WebView: `source.body` is not supported when using GET.'); } } const NativeWebView = (nativeConfig?.component as typeof RNCWebView | undefined) || RNCWebView; const sourceResolved = resolveAssetSource(source as ImageSourcePropType); const newSource = typeof sourceResolved === 'object' ? Object.entries(sourceResolved as WebViewSourceUri).reduce( (prev, [currKey, currValue]) => { return { ...prev, [currKey]: currKey === 'headers' && currValue && typeof currValue === 'object' ? Object.entries(currValue).map(([key, value]) => { return { name: key, value, }; }) : currValue, }; }, {} ) : sourceResolved; const webView = ( ); return ( {webView} {otherView} ); } ); // native implementation should return "true" only for Android 5+ const { isFileUploadSupported } = RNCTPWebViewModule; const WebView = Object.assign(WebViewComponent, { isFileUploadSupported }); export default WebView;