2015-03-23 20:28:42 +00:00
|
|
|
/**
|
2018-09-11 22:27:47 +00:00
|
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
2015-03-23 20:28:42 +00:00
|
|
|
*
|
2018-02-17 02:24:55 +00:00
|
|
|
* This source code is licensed under the MIT license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree.
|
2015-03-23 20:28:42 +00:00
|
|
|
*/
|
2015-03-14 08:22:25 +00:00
|
|
|
|
|
|
|
#import "RCTWebViewManager.h"
|
|
|
|
|
|
|
|
#import "RCTBridge.h"
|
|
|
|
#import "RCTUIManager.h"
|
|
|
|
#import "RCTWebView.h"
|
2015-11-14 18:25:00 +00:00
|
|
|
#import "UIView+React.h"
|
2015-03-14 08:22:25 +00:00
|
|
|
|
2015-11-04 16:42:28 +00:00
|
|
|
@interface RCTWebViewManager () <RCTWebViewDelegate>
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
2015-11-19 15:09:09 +00:00
|
|
|
@implementation RCTWebViewManager
|
|
|
|
{
|
2015-11-04 16:42:28 +00:00
|
|
|
NSConditionLock *_shouldStartLoadLock;
|
|
|
|
BOOL _shouldStartLoad;
|
|
|
|
}
|
2015-03-14 08:22:25 +00:00
|
|
|
|
2015-04-08 12:42:43 +00:00
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
2015-03-14 08:22:25 +00:00
|
|
|
- (UIView *)view
|
|
|
|
{
|
2015-11-04 16:42:28 +00:00
|
|
|
RCTWebView *webView = [RCTWebView new];
|
|
|
|
webView.delegate = self;
|
|
|
|
return webView;
|
2015-03-14 08:22:25 +00:00
|
|
|
}
|
|
|
|
|
2016-02-02 02:00:18 +00:00
|
|
|
RCT_EXPORT_VIEW_PROPERTY(source, NSDictionary)
|
2015-11-19 15:09:09 +00:00
|
|
|
RCT_REMAP_VIEW_PROPERTY(bounces, _webView.scrollView.bounces, BOOL)
|
|
|
|
RCT_REMAP_VIEW_PROPERTY(scrollEnabled, _webView.scrollView.scrollEnabled, BOOL)
|
2016-01-28 13:35:14 +00:00
|
|
|
RCT_REMAP_VIEW_PROPERTY(decelerationRate, _webView.scrollView.decelerationRate, CGFloat)
|
2016-02-26 16:19:40 +00:00
|
|
|
RCT_EXPORT_VIEW_PROPERTY(scalesPageToFit, BOOL)
|
2016-10-16 13:29:14 +00:00
|
|
|
RCT_EXPORT_VIEW_PROPERTY(messagingEnabled, BOOL)
|
2015-11-19 15:09:09 +00:00
|
|
|
RCT_EXPORT_VIEW_PROPERTY(injectedJavaScript, NSString)
|
|
|
|
RCT_EXPORT_VIEW_PROPERTY(contentInset, UIEdgeInsets)
|
|
|
|
RCT_EXPORT_VIEW_PROPERTY(automaticallyAdjustContentInsets, BOOL)
|
|
|
|
RCT_EXPORT_VIEW_PROPERTY(onLoadingStart, RCTDirectEventBlock)
|
|
|
|
RCT_EXPORT_VIEW_PROPERTY(onLoadingFinish, RCTDirectEventBlock)
|
|
|
|
RCT_EXPORT_VIEW_PROPERTY(onLoadingError, RCTDirectEventBlock)
|
2016-10-16 13:29:14 +00:00
|
|
|
RCT_EXPORT_VIEW_PROPERTY(onMessage, RCTDirectEventBlock)
|
2015-11-19 15:09:09 +00:00
|
|
|
RCT_EXPORT_VIEW_PROPERTY(onShouldStartLoadWithRequest, RCTDirectEventBlock)
|
|
|
|
RCT_REMAP_VIEW_PROPERTY(allowsInlineMediaPlayback, _webView.allowsInlineMediaPlayback, BOOL)
|
2016-03-16 17:02:09 +00:00
|
|
|
RCT_REMAP_VIEW_PROPERTY(mediaPlaybackRequiresUserAction, _webView.mediaPlaybackRequiresUserAction, BOOL)
|
2016-07-13 21:31:43 +00:00
|
|
|
RCT_REMAP_VIEW_PROPERTY(dataDetectorTypes, _webView.dataDetectorTypes, UIDataDetectorTypes)
|
2015-08-11 13:37:12 +00:00
|
|
|
|
2015-07-31 13:55:47 +00:00
|
|
|
RCT_EXPORT_METHOD(goBack:(nonnull NSNumber *)reactTag)
|
2015-03-14 08:22:25 +00:00
|
|
|
{
|
2015-11-14 18:25:00 +00:00
|
|
|
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTWebView *> *viewRegistry) {
|
2015-03-14 08:22:25 +00:00
|
|
|
RCTWebView *view = viewRegistry[reactTag];
|
|
|
|
if (![view isKindOfClass:[RCTWebView class]]) {
|
2015-07-31 20:50:44 +00:00
|
|
|
RCTLogError(@"Invalid view returned from registry, expecting RCTWebView, got: %@", view);
|
2015-08-11 20:18:08 +00:00
|
|
|
} else {
|
|
|
|
[view goBack];
|
2015-03-14 08:22:25 +00:00
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-07-31 13:55:47 +00:00
|
|
|
RCT_EXPORT_METHOD(goForward:(nonnull NSNumber *)reactTag)
|
2015-03-14 08:22:25 +00:00
|
|
|
{
|
2018-08-16 20:34:22 +00:00
|
|
|
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTWebView *> *viewRegistry) {
|
|
|
|
RCTWebView *view = viewRegistry[reactTag];
|
2015-03-14 08:22:25 +00:00
|
|
|
if (![view isKindOfClass:[RCTWebView class]]) {
|
2015-07-31 20:50:44 +00:00
|
|
|
RCTLogError(@"Invalid view returned from registry, expecting RCTWebView, got: %@", view);
|
2015-08-11 20:18:08 +00:00
|
|
|
} else {
|
|
|
|
[view goForward];
|
2015-03-14 08:22:25 +00:00
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-07-31 13:55:47 +00:00
|
|
|
RCT_EXPORT_METHOD(reload:(nonnull NSNumber *)reactTag)
|
2015-03-14 08:22:25 +00:00
|
|
|
{
|
2015-11-14 18:25:00 +00:00
|
|
|
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTWebView *> *viewRegistry) {
|
2015-03-14 08:22:25 +00:00
|
|
|
RCTWebView *view = viewRegistry[reactTag];
|
|
|
|
if (![view isKindOfClass:[RCTWebView class]]) {
|
2015-08-11 20:18:08 +00:00
|
|
|
RCTLogError(@"Invalid view returned from registry, expecting RCTWebView, got: %@", view);
|
|
|
|
} else {
|
|
|
|
[view reload];
|
2015-03-14 08:22:25 +00:00
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
Implemented stopLoading
Summary:**Motivation:** In my app, I'm using a WebView that loads content from my mobile site. What I want to do is when a user presses a link on the loaded page, I want to stop the WebView's request, hijack the URL and open the URL in a new WebView, pushed to the top of the navigator stack. To me, this gives the overall app a more native feel, instead of implementing a rudimentary navbar on the main WebView to go back.
**Attempted Workarounds:** I've attempted to get similar functionality by capturing the onNavigationStateChange event in the WebView, and then within calling goBack + pushing the new view to the navigator stack. From a functionality standpoint, this works. However, from a UI standpoint, the user can clearly see the webview change states to a new page + go back before having the new view pushed on top of their nav stack.
Closes https://github.com/facebook/react-native/pull/6886
Differential Revision: D3212447
Pulled By: mkonicek
fb-gh-sync-id: 05911e583d9ba54ddbd54a772153c80ed227731e
fbshipit-source-id: 05911e583d9ba54ddbd54a772153c80ed227731e
2016-04-22 15:14:53 +00:00
|
|
|
RCT_EXPORT_METHOD(stopLoading:(nonnull NSNumber *)reactTag)
|
|
|
|
{
|
|
|
|
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTWebView *> *viewRegistry) {
|
|
|
|
RCTWebView *view = viewRegistry[reactTag];
|
|
|
|
if (![view isKindOfClass:[RCTWebView class]]) {
|
|
|
|
RCTLogError(@"Invalid view returned from registry, expecting RCTWebView, got: %@", view);
|
|
|
|
} else {
|
|
|
|
[view stopLoading];
|
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2016-10-16 13:29:14 +00:00
|
|
|
RCT_EXPORT_METHOD(postMessage:(nonnull NSNumber *)reactTag message:(NSString *)message)
|
|
|
|
{
|
|
|
|
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTWebView *> *viewRegistry) {
|
|
|
|
RCTWebView *view = viewRegistry[reactTag];
|
|
|
|
if (![view isKindOfClass:[RCTWebView class]]) {
|
|
|
|
RCTLogError(@"Invalid view returned from registry, expecting RCTWebView, got: %@", view);
|
|
|
|
} else {
|
|
|
|
[view postMessage:message];
|
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2017-01-07 04:13:20 +00:00
|
|
|
RCT_EXPORT_METHOD(injectJavaScript:(nonnull NSNumber *)reactTag script:(NSString *)script)
|
|
|
|
{
|
|
|
|
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTWebView *> *viewRegistry) {
|
|
|
|
RCTWebView *view = viewRegistry[reactTag];
|
|
|
|
if (![view isKindOfClass:[RCTWebView class]]) {
|
|
|
|
RCTLogError(@"Invalid view returned from registry, expecting RCTWebView, got: %@", view);
|
|
|
|
} else {
|
|
|
|
[view injectJavaScript:script];
|
|
|
|
}
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-11-04 16:42:28 +00:00
|
|
|
#pragma mark - Exported synchronous methods
|
|
|
|
|
|
|
|
- (BOOL)webView:(__unused RCTWebView *)webView
|
2015-11-14 18:25:00 +00:00
|
|
|
shouldStartLoadForRequest:(NSMutableDictionary<NSString *, id> *)request
|
2015-11-04 16:42:28 +00:00
|
|
|
withCallback:(RCTDirectEventBlock)callback
|
|
|
|
{
|
|
|
|
_shouldStartLoadLock = [[NSConditionLock alloc] initWithCondition:arc4random()];
|
|
|
|
_shouldStartLoad = YES;
|
|
|
|
request[@"lockIdentifier"] = @(_shouldStartLoadLock.condition);
|
|
|
|
callback(request);
|
|
|
|
|
|
|
|
// Block the main thread for a maximum of 250ms until the JS thread returns
|
|
|
|
if ([_shouldStartLoadLock lockWhenCondition:0 beforeDate:[NSDate dateWithTimeIntervalSinceNow:.25]]) {
|
|
|
|
BOOL returnValue = _shouldStartLoad;
|
|
|
|
[_shouldStartLoadLock unlock];
|
|
|
|
_shouldStartLoadLock = nil;
|
|
|
|
return returnValue;
|
|
|
|
} else {
|
|
|
|
RCTLogWarn(@"Did not receive response to shouldStartLoad in time, defaulting to YES");
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RCT_EXPORT_METHOD(startLoadWithResult:(BOOL)result lockIdentifier:(NSInteger)lockIdentifier)
|
|
|
|
{
|
|
|
|
if ([_shouldStartLoadLock tryLockWhenCondition:lockIdentifier]) {
|
|
|
|
_shouldStartLoad = result;
|
|
|
|
[_shouldStartLoadLock unlockWithCondition:0];
|
|
|
|
} else {
|
|
|
|
RCTLogWarn(@"startLoadWithResult invoked with invalid lockIdentifier: "
|
2017-09-25 17:23:02 +00:00
|
|
|
"got %lld, expected %lld", (long long)lockIdentifier, (long long)_shouldStartLoadLock.condition);
|
2015-11-04 16:42:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-14 08:22:25 +00:00
|
|
|
@end
|