2015-03-14 08:22:25 +00:00
|
|
|
/**
|
2015-03-23 20:35:08 +00:00
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
2015-03-14 08:22:25 +00:00
|
|
|
*
|
|
|
|
* @providesModule WebView
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var EdgeInsetsPropType = require('EdgeInsetsPropType');
|
|
|
|
var React = require('React');
|
2015-05-08 16:45:43 +00:00
|
|
|
var ReactNativeViewAttributes = require('ReactNativeViewAttributes');
|
2015-03-14 08:22:25 +00:00
|
|
|
var StyleSheet = require('StyleSheet');
|
2015-11-27 13:39:00 +00:00
|
|
|
var UIManager = require('UIManager');
|
2015-03-14 08:22:25 +00:00
|
|
|
var View = require('View');
|
|
|
|
|
|
|
|
var keyMirror = require('keyMirror');
|
|
|
|
var merge = require('merge');
|
2015-11-18 16:24:26 +00:00
|
|
|
var requireNativeComponent = require('requireNativeComponent');
|
2015-03-14 08:22:25 +00:00
|
|
|
|
|
|
|
var PropTypes = React.PropTypes;
|
|
|
|
|
2015-03-17 10:08:46 +00:00
|
|
|
var RCT_WEBVIEW_REF = 'webview';
|
2015-03-14 08:22:25 +00:00
|
|
|
|
|
|
|
var WebViewState = keyMirror({
|
|
|
|
IDLE: null,
|
|
|
|
LOADING: null,
|
|
|
|
ERROR: null,
|
|
|
|
});
|
|
|
|
|
2015-10-29 16:00:33 +00:00
|
|
|
/**
|
2015-12-17 19:47:37 +00:00
|
|
|
* Renders a native WebView.
|
2015-10-29 16:00:33 +00:00
|
|
|
*/
|
2015-03-14 08:22:25 +00:00
|
|
|
var WebView = React.createClass({
|
|
|
|
|
|
|
|
propTypes: {
|
2015-11-18 16:24:26 +00:00
|
|
|
...View.propTypes,
|
2016-01-01 02:03:37 +00:00
|
|
|
renderError: PropTypes.func,
|
2015-12-17 19:47:37 +00:00
|
|
|
renderLoading: PropTypes.func,
|
2016-01-15 17:30:47 +00:00
|
|
|
onLoad: PropTypes.func,
|
|
|
|
onLoadEnd: PropTypes.func,
|
|
|
|
onLoadStart: PropTypes.func,
|
|
|
|
onError: PropTypes.func,
|
2015-07-10 19:21:19 +00:00
|
|
|
url: PropTypes.string,
|
|
|
|
html: PropTypes.string,
|
2015-03-14 08:22:25 +00:00
|
|
|
automaticallyAdjustContentInsets: PropTypes.bool,
|
|
|
|
contentInset: EdgeInsetsPropType,
|
|
|
|
onNavigationStateChange: PropTypes.func,
|
|
|
|
startInLoadingState: PropTypes.bool, // force WebView to show loadingView on first load
|
|
|
|
style: View.propTypes.style,
|
2015-12-17 19:47:37 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Used on Android only, JS is enabled by default for WebView on iOS
|
|
|
|
* @platform android
|
|
|
|
*/
|
2016-01-05 18:31:42 +00:00
|
|
|
javaScriptEnabled: PropTypes.bool,
|
2015-07-08 00:07:52 +00:00
|
|
|
|
2016-01-01 02:03:37 +00:00
|
|
|
/**
|
|
|
|
* Used on Android only, controls whether DOM Storage is enabled or not
|
|
|
|
* @platform android
|
|
|
|
*/
|
2016-01-05 18:31:42 +00:00
|
|
|
domStorageEnabled: PropTypes.bool,
|
2016-01-01 02:03:37 +00:00
|
|
|
|
2015-07-08 00:07:52 +00:00
|
|
|
/**
|
|
|
|
* Sets the JS to be injected when the webpage loads.
|
|
|
|
*/
|
|
|
|
injectedJavaScript: PropTypes.string,
|
|
|
|
|
2015-06-12 09:40:49 +00:00
|
|
|
/**
|
2015-12-17 19:47:37 +00:00
|
|
|
* Sets the user-agent for this WebView. The user-agent can also be set in native using
|
|
|
|
* WebViewConfig. This prop will overwrite that config.
|
2015-06-12 09:40:49 +00:00
|
|
|
*/
|
|
|
|
userAgent: PropTypes.string,
|
2015-12-17 19:47:37 +00:00
|
|
|
|
2015-03-14 08:22:25 +00:00
|
|
|
/**
|
|
|
|
* Used to locate this view in end-to-end tests.
|
|
|
|
*/
|
|
|
|
testID: PropTypes.string,
|
|
|
|
},
|
|
|
|
|
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
|
|
|
viewState: WebViewState.IDLE,
|
|
|
|
lastErrorEvent: null,
|
|
|
|
startInLoadingState: true,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
componentWillMount: function() {
|
|
|
|
if (this.props.startInLoadingState) {
|
|
|
|
this.setState({viewState: WebViewState.LOADING});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
render: function() {
|
|
|
|
var otherView = null;
|
|
|
|
|
|
|
|
if (this.state.viewState === WebViewState.LOADING) {
|
2015-04-09 23:20:59 +00:00
|
|
|
otherView = this.props.renderLoading && this.props.renderLoading();
|
2015-03-14 08:22:25 +00:00
|
|
|
} else if (this.state.viewState === WebViewState.ERROR) {
|
|
|
|
var errorEvent = this.state.lastErrorEvent;
|
2015-04-09 23:20:59 +00:00
|
|
|
otherView = this.props.renderError && this.props.renderError(
|
2015-03-14 08:22:25 +00:00
|
|
|
errorEvent.domain,
|
|
|
|
errorEvent.code,
|
|
|
|
errorEvent.description);
|
|
|
|
} else if (this.state.viewState !== WebViewState.IDLE) {
|
2015-05-18 22:30:17 +00:00
|
|
|
console.error('RCTWebView invalid state encountered: ' + this.state.loading);
|
2015-03-14 08:22:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var webViewStyles = [styles.container, this.props.style];
|
|
|
|
if (this.state.viewState === WebViewState.LOADING ||
|
|
|
|
this.state.viewState === WebViewState.ERROR) {
|
|
|
|
// if we're in either LOADING or ERROR states, don't show the webView
|
|
|
|
webViewStyles.push(styles.hidden);
|
|
|
|
}
|
|
|
|
|
2016-01-05 18:31:42 +00:00
|
|
|
var {javaScriptEnabled, domStorageEnabled} = this.props;
|
|
|
|
if (this.props.javaScriptEnabledAndroid) {
|
|
|
|
console.warn('javaScriptEnabledAndroid is deprecated. Use javaScriptEnabled instead');
|
|
|
|
javaScriptEnabled = this.props.javaScriptEnabledAndroid;
|
|
|
|
}
|
|
|
|
if (this.props.domStorageEnabledAndroid) {
|
|
|
|
console.warn('domStorageEnabledAndroid is deprecated. Use domStorageEnabled instead');
|
|
|
|
domStorageEnabled = this.props.domStorageEnabledAndroid;
|
|
|
|
}
|
|
|
|
|
2015-03-14 08:22:25 +00:00
|
|
|
var webView =
|
|
|
|
<RCTWebView
|
2015-03-17 10:08:46 +00:00
|
|
|
ref={RCT_WEBVIEW_REF}
|
2015-03-14 08:22:25 +00:00
|
|
|
key="webViewKey"
|
|
|
|
style={webViewStyles}
|
|
|
|
url={this.props.url}
|
2015-07-10 19:21:19 +00:00
|
|
|
html={this.props.html}
|
2015-07-08 00:07:52 +00:00
|
|
|
injectedJavaScript={this.props.injectedJavaScript}
|
2015-06-12 09:40:49 +00:00
|
|
|
userAgent={this.props.userAgent}
|
2016-01-05 18:31:42 +00:00
|
|
|
javaScriptEnabled={javaScriptEnabled}
|
|
|
|
domStorageEnabled={domStorageEnabled}
|
2015-03-14 08:22:25 +00:00
|
|
|
contentInset={this.props.contentInset}
|
|
|
|
automaticallyAdjustContentInsets={this.props.automaticallyAdjustContentInsets}
|
|
|
|
onLoadingStart={this.onLoadingStart}
|
|
|
|
onLoadingFinish={this.onLoadingFinish}
|
|
|
|
onLoadingError={this.onLoadingError}
|
|
|
|
testID={this.props.testID}
|
|
|
|
/>;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<View style={styles.container}>
|
|
|
|
{webView}
|
|
|
|
{otherView}
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
goForward: function() {
|
2015-11-27 13:39:00 +00:00
|
|
|
UIManager.dispatchViewManagerCommand(
|
2015-05-05 09:51:17 +00:00
|
|
|
this.getWebWiewHandle(),
|
2015-11-27 13:39:00 +00:00
|
|
|
UIManager.RCTWebView.Commands.goForward,
|
2015-05-05 09:51:17 +00:00
|
|
|
null
|
|
|
|
);
|
2015-03-14 08:22:25 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
goBack: function() {
|
2015-11-27 13:39:00 +00:00
|
|
|
UIManager.dispatchViewManagerCommand(
|
2015-05-05 09:51:17 +00:00
|
|
|
this.getWebWiewHandle(),
|
2015-11-27 13:39:00 +00:00
|
|
|
UIManager.RCTWebView.Commands.goBack,
|
2015-05-05 09:51:17 +00:00
|
|
|
null
|
|
|
|
);
|
2015-03-14 08:22:25 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
reload: function() {
|
2015-11-27 13:39:00 +00:00
|
|
|
UIManager.dispatchViewManagerCommand(
|
2015-05-05 09:51:17 +00:00
|
|
|
this.getWebWiewHandle(),
|
2015-11-27 13:39:00 +00:00
|
|
|
UIManager.RCTWebView.Commands.reload,
|
2015-05-05 09:51:17 +00:00
|
|
|
null
|
|
|
|
);
|
2015-03-14 08:22:25 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* We return an event with a bunch of fields including:
|
|
|
|
* url, title, loading, canGoBack, canGoForward
|
|
|
|
*/
|
|
|
|
updateNavigationState: function(event) {
|
|
|
|
if (this.props.onNavigationStateChange) {
|
|
|
|
this.props.onNavigationStateChange(event.nativeEvent);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
getWebWiewHandle: function() {
|
2015-05-13 01:55:13 +00:00
|
|
|
return React.findNodeHandle(this.refs[RCT_WEBVIEW_REF]);
|
2015-03-14 08:22:25 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
onLoadingStart: function(event) {
|
2016-01-15 17:30:47 +00:00
|
|
|
var onLoadStart = this.props.onLoadStart;
|
|
|
|
onLoadStart && onLoadStart(event);
|
2015-03-14 08:22:25 +00:00
|
|
|
this.updateNavigationState(event);
|
|
|
|
},
|
|
|
|
|
|
|
|
onLoadingError: function(event) {
|
|
|
|
event.persist(); // persist this event because we need to store it
|
2016-01-15 17:30:47 +00:00
|
|
|
var {onError, onLoadEnd} = this.props;
|
|
|
|
onError && onError(event);
|
|
|
|
onLoadEnd && onLoadEnd(event);
|
2015-05-18 22:30:17 +00:00
|
|
|
console.error('Encountered an error loading page', event.nativeEvent);
|
2015-03-14 08:22:25 +00:00
|
|
|
|
|
|
|
this.setState({
|
|
|
|
lastErrorEvent: event.nativeEvent,
|
|
|
|
viewState: WebViewState.ERROR
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
onLoadingFinish: function(event) {
|
2016-01-15 17:30:47 +00:00
|
|
|
var {onLoad, onLoadEnd} = this.props;
|
|
|
|
onLoad && onLoad(event);
|
|
|
|
onLoadEnd && onLoadEnd(event);
|
2015-03-14 08:22:25 +00:00
|
|
|
this.setState({
|
|
|
|
viewState: WebViewState.IDLE,
|
|
|
|
});
|
|
|
|
this.updateNavigationState(event);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2015-11-18 16:24:26 +00:00
|
|
|
var RCTWebView = requireNativeComponent('RCTWebView', WebView);
|
2015-03-14 08:22:25 +00:00
|
|
|
|
|
|
|
var styles = StyleSheet.create({
|
|
|
|
container: {
|
|
|
|
flex: 1,
|
|
|
|
},
|
|
|
|
hidden: {
|
|
|
|
height: 0,
|
|
|
|
flex: 0, // disable 'flex:1' when hiding a View
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
module.exports = WebView;
|