import escapeStringRegexp from 'escape-string-regexp'; import React from 'react'; import { Linking, View, ActivityIndicator, Text } from 'react-native'; import { OnShouldStartLoadWithRequest, ShouldStartLoadRequestEvent, } from './WebViewTypes'; import styles from './WebView.styles'; const defaultOriginWhitelist = ['http://*', 'https://*']; const extractOrigin = (url: string): string => { const result = /^[A-Za-z][A-Za-z0-9+\-.]+:(\/\/)?[^/]*/.exec(url); return result === null ? '' : result[0]; }; const originWhitelistToRegex = (originWhitelist: string): string => `^${escapeStringRegexp(originWhitelist).replace(/\\\*/g, '.*')}`; const passesWhitelist = ( compiledWhitelist: readonly string[], url: string, ) => { const origin = extractOrigin(url); return compiledWhitelist.some(x => new RegExp(x).test(origin)); }; const compileWhitelist = ( originWhitelist: readonly string[], ): readonly string[] => ['about:blank', ...(originWhitelist || [])].map(originWhitelistToRegex); const createOnShouldStartLoadWithRequest = ( loadRequest: ( shouldStart: boolean, url: string, lockIdentifier: number, ) => void, originWhitelist: readonly string[], onShouldStartLoadWithRequest?: OnShouldStartLoadWithRequest, ) => { return ({ nativeEvent }: ShouldStartLoadRequestEvent) => { let shouldStart = true; const { url, lockIdentifier } = nativeEvent; if (!passesWhitelist(compileWhitelist(originWhitelist), url)) { Linking.canOpenURL(url).then((supported) => { if (supported) { return Linking.openURL(url); } console.warn(`Can't open url: ${url}`); return undefined; }).catch(e => { console.warn('Error opening URL: ', e); }); shouldStart = false; } else if (onShouldStartLoadWithRequest) { shouldStart = onShouldStartLoadWithRequest(nativeEvent); } loadRequest(shouldStart, url, lockIdentifier); }; }; const defaultRenderLoading = () => ( ); const defaultRenderError = ( errorDomain: string | undefined, errorCode: number, errorDesc: string, ) => ( Error loading page {`Domain: ${errorDomain}`} {`Error Code: ${errorCode}`} {`Description: ${errorDesc}`} ); export { defaultOriginWhitelist, createOnShouldStartLoadWithRequest, defaultRenderLoading, defaultRenderError, };