react-native-webview/ios/RNCWKWebViewManager.m

224 lines
8.5 KiB
Mathematica
Raw Normal View History

/**
* Copyright (c) 2015-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
2018-07-31 03:21:48 +00:00
2018-09-18 16:32:16 +00:00
#import "RNCWKWebViewManager.h"
2018-07-31 03:21:48 +00:00
#import <React/RCTUIManager.h>
#import <React/RCTDefines.h>
2018-09-18 16:32:16 +00:00
#import "RNCWKWebView.h"
2018-07-31 03:21:48 +00:00
2018-09-18 16:32:16 +00:00
@interface RNCWKWebViewManager () <RNCWKWebViewDelegate>
2018-07-31 03:21:48 +00:00
@end
@implementation RCTConvert (UIScrollView)
#if defined(__IPHONE_OS_VERSION_MAX_ALLOWED) && __IPHONE_OS_VERSION_MAX_ALLOWED >= 110000 /* __IPHONE_11_0 */
RCT_ENUM_CONVERTER(UIScrollViewContentInsetAdjustmentBehavior, (@{
@"automatic": @(UIScrollViewContentInsetAdjustmentAutomatic),
@"scrollableAxes": @(UIScrollViewContentInsetAdjustmentScrollableAxes),
@"never": @(UIScrollViewContentInsetAdjustmentNever),
@"always": @(UIScrollViewContentInsetAdjustmentAlways),
}), UIScrollViewContentInsetAdjustmentNever, integerValue)
#endif
@end
2018-09-18 16:32:16 +00:00
@implementation RNCWKWebViewManager
2018-07-31 03:21:48 +00:00
{
NSConditionLock *_shouldStartLoadLock;
BOOL _shouldStartLoad;
}
RCT_EXPORT_MODULE()
- (UIView *)view
{
2018-09-18 16:32:16 +00:00
RNCWKWebView *webView = [RNCWKWebView new];
2018-07-31 03:21:48 +00:00
webView.delegate = self;
return webView;
}
RCT_EXPORT_VIEW_PROPERTY(source, NSDictionary)
RCT_EXPORT_VIEW_PROPERTY(onLoadingStart, RCTDirectEventBlock)
RCT_EXPORT_VIEW_PROPERTY(onLoadingFinish, RCTDirectEventBlock)
RCT_EXPORT_VIEW_PROPERTY(onLoadingError, RCTDirectEventBlock)
RCT_EXPORT_VIEW_PROPERTY(onLoadingProgress, RCTDirectEventBlock)
2018-07-31 03:21:48 +00:00
RCT_EXPORT_VIEW_PROPERTY(onShouldStartLoadWithRequest, RCTDirectEventBlock)
RCT_EXPORT_VIEW_PROPERTY(injectedJavaScript, NSString)
RCT_EXPORT_VIEW_PROPERTY(javaScriptEnabled, BOOL)
RCT_EXPORT_VIEW_PROPERTY(allowsInlineMediaPlayback, BOOL)
RCT_EXPORT_VIEW_PROPERTY(mediaPlaybackRequiresUserAction, BOOL)
#if WEBKIT_IOS_10_APIS_AVAILABLE
RCT_EXPORT_VIEW_PROPERTY(dataDetectorTypes, WKDataDetectorTypes)
#endif
RCT_EXPORT_VIEW_PROPERTY(contentInset, UIEdgeInsets)
RCT_EXPORT_VIEW_PROPERTY(automaticallyAdjustContentInsets, BOOL)
RCT_EXPORT_VIEW_PROPERTY(hideKeyboardAccessoryView, BOOL)
RCT_EXPORT_VIEW_PROPERTY(allowsBackForwardNavigationGestures, BOOL)
RCT_EXPORT_VIEW_PROPERTY(incognito, BOOL)
RCT_EXPORT_VIEW_PROPERTY(pagingEnabled, BOOL)
RCT_EXPORT_VIEW_PROPERTY(userAgent, NSString)
RCT_EXPORT_VIEW_PROPERTY(applicationNameForUserAgent, NSString)
RCT_EXPORT_VIEW_PROPERTY(cacheEnabled, BOOL)
RCT_EXPORT_VIEW_PROPERTY(allowsLinkPreview, BOOL)
RCT_EXPORT_VIEW_PROPERTY(allowingReadAccessToURL, NSString)
2018-07-31 03:21:48 +00:00
#if defined(__IPHONE_OS_VERSION_MAX_ALLOWED) && __IPHONE_OS_VERSION_MAX_ALLOWED >= 110000 /* __IPHONE_11_0 */
RCT_EXPORT_VIEW_PROPERTY(contentInsetAdjustmentBehavior, UIScrollViewContentInsetAdjustmentBehavior)
#endif
/**
* Expose methods to enable messaging the webview.
*/
RCT_EXPORT_VIEW_PROPERTY(messagingEnabled, BOOL)
RCT_EXPORT_VIEW_PROPERTY(onMessage, RCTDirectEventBlock)
RCT_EXPORT_VIEW_PROPERTY(onScroll, RCTDirectEventBlock)
RCT_EXPORT_METHOD(postMessage:(nonnull NSNumber *)reactTag message:(NSString *)message)
2018-07-31 03:21:48 +00:00
{
2018-09-18 16:32:16 +00:00
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RNCWKWebView *> *viewRegistry) {
RNCWKWebView *view = viewRegistry[reactTag];
if (![view isKindOfClass:[RNCWKWebView class]]) {
RCTLogError(@"Invalid view returned from registry, expecting RNCWKWebView, got: %@", view);
2018-07-31 03:21:48 +00:00
} else {
[view postMessage:message];
2018-07-31 03:21:48 +00:00
}
}];
}
2018-09-18 16:32:16 +00:00
RCT_CUSTOM_VIEW_PROPERTY(bounces, BOOL, RNCWKWebView) {
view.bounces = json == nil ? true : [RCTConvert BOOL: json];
}
RCT_CUSTOM_VIEW_PROPERTY(useSharedProcessPool, BOOL, RNCWKWebView) {
view.useSharedProcessPool = json == nil ? true : [RCTConvert BOOL: json];
}
2018-09-18 16:32:16 +00:00
RCT_CUSTOM_VIEW_PROPERTY(scrollEnabled, BOOL, RNCWKWebView) {
view.scrollEnabled = json == nil ? true : [RCTConvert BOOL: json];
}
RCT_CUSTOM_VIEW_PROPERTY(sharedCookiesEnabled, BOOL, RNCWKWebView) {
view.sharedCookiesEnabled = json == nil ? false : [RCTConvert BOOL: json];
}
2018-09-18 16:32:16 +00:00
RCT_CUSTOM_VIEW_PROPERTY(decelerationRate, CGFloat, RNCWKWebView) {
view.decelerationRate = json == nil ? UIScrollViewDecelerationRateNormal : [RCTConvert CGFloat: json];
}
RCT_CUSTOM_VIEW_PROPERTY(directionalLockEnabled, BOOL, RNCWKWebView) {
view.directionalLockEnabled = json == nil ? true : [RCTConvert BOOL: json];
}
RCT_CUSTOM_VIEW_PROPERTY(showsHorizontalScrollIndicator, BOOL, RNCWKWebView) {
view.showsHorizontalScrollIndicator = json == nil ? true : [RCTConvert BOOL: json];
}
RCT_CUSTOM_VIEW_PROPERTY(showsVerticalScrollIndicator, BOOL, RNCWKWebView) {
view.showsVerticalScrollIndicator = json == nil ? true : [RCTConvert BOOL: json];
}
RCT_CUSTOM_VIEW_PROPERTY(keyboardDisplayRequiresUserAction, BOOL, RNCWKWebView) {
view.keyboardDisplayRequiresUserAction = json == nil ? true : [RCTConvert BOOL: json];
}
RCT_EXPORT_METHOD(injectJavaScript:(nonnull NSNumber *)reactTag script:(NSString *)script)
2018-07-31 03:21:48 +00:00
{
2018-09-18 16:32:16 +00:00
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RNCWKWebView *> *viewRegistry) {
RNCWKWebView *view = viewRegistry[reactTag];
if (![view isKindOfClass:[RNCWKWebView class]]) {
RCTLogError(@"Invalid view returned from registry, expecting RNCWKWebView, got: %@", view);
2018-07-31 03:21:48 +00:00
} else {
[view injectJavaScript:script];
2018-07-31 03:21:48 +00:00
}
}];
}
RCT_EXPORT_METHOD(goBack:(nonnull NSNumber *)reactTag)
2018-07-31 03:21:48 +00:00
{
2018-09-18 16:32:16 +00:00
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RNCWKWebView *> *viewRegistry) {
RNCWKWebView *view = viewRegistry[reactTag];
if (![view isKindOfClass:[RNCWKWebView class]]) {
RCTLogError(@"Invalid view returned from registry, expecting RNCWKWebView, got: %@", view);
2018-07-31 03:21:48 +00:00
} else {
[view goBack];
2018-07-31 03:21:48 +00:00
}
}];
}
RCT_EXPORT_METHOD(goForward:(nonnull NSNumber *)reactTag)
2018-07-31 03:21:48 +00:00
{
2018-09-18 16:32:16 +00:00
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RNCWKWebView *> *viewRegistry) {
RNCWKWebView *view = viewRegistry[reactTag];
if (![view isKindOfClass:[RNCWKWebView class]]) {
RCTLogError(@"Invalid view returned from registry, expecting RNCWKWebView, got: %@", view);
2018-07-31 03:21:48 +00:00
} else {
[view goForward];
2018-07-31 03:21:48 +00:00
}
}];
}
RCT_EXPORT_METHOD(reload:(nonnull NSNumber *)reactTag)
2018-07-31 03:21:48 +00:00
{
2018-09-18 16:32:16 +00:00
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RNCWKWebView *> *viewRegistry) {
RNCWKWebView *view = viewRegistry[reactTag];
if (![view isKindOfClass:[RNCWKWebView class]]) {
RCTLogError(@"Invalid view returned from registry, expecting RNCWKWebView, got: %@", view);
2018-07-31 03:21:48 +00:00
} else {
[view reload];
2018-07-31 03:21:48 +00:00
}
}];
}
RCT_EXPORT_METHOD(stopLoading:(nonnull NSNumber *)reactTag)
2018-07-31 03:21:48 +00:00
{
2018-09-18 16:32:16 +00:00
[self.bridge.uiManager addUIBlock:^(__unused RCTUIManager *uiManager, NSDictionary<NSNumber *, RNCWKWebView *> *viewRegistry) {
RNCWKWebView *view = viewRegistry[reactTag];
if (![view isKindOfClass:[RNCWKWebView class]]) {
RCTLogError(@"Invalid view returned from registry, expecting RNCWKWebView, got: %@", view);
2018-07-31 03:21:48 +00:00
} else {
[view stopLoading];
2018-07-31 03:21:48 +00:00
}
}];
}
#pragma mark - Exported synchronous methods
2018-09-18 16:32:16 +00:00
- (BOOL) webView:(RNCWKWebView *)webView
2018-07-31 03:21:48 +00:00
shouldStartLoadForRequest:(NSMutableDictionary<NSString *, id> *)request
withCallback:(RCTDirectEventBlock)callback
2018-07-31 03:21:48 +00:00
{
_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: "
"got %lld, expected %lld", (long long)lockIdentifier, (long long)_shouldStartLoadLock.condition);
}
}
@end