| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 | 
							- /**
 -  * 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.
 -  */
 - 
 - #import "RNCWKWebViewManager.h"
 - 
 - #import <React/RCTUIManager.h>
 - #import <React/RCTDefines.h>
 - #import "RNCWKWebView.h"
 - 
 - @interface RNCWKWebViewManager () <RNCWKWebViewDelegate>
 - @end
 - 
 - @implementation RNCWKWebViewManager
 - {
 -   NSConditionLock *_shouldStartLoadLock;
 -   BOOL _shouldStartLoad;
 - }
 - 
 - RCT_EXPORT_MODULE()
 - 
 - - (UIView *)view
 - {
 -   RNCWKWebView *webView = [RNCWKWebView new];
 -   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)
 - RCT_EXPORT_VIEW_PROPERTY(onShouldStartLoadWithRequest, RCTDirectEventBlock)
 - RCT_EXPORT_VIEW_PROPERTY(injectedJavaScript, NSString)
 - 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(pagingEnabled, BOOL)
 - RCT_EXPORT_VIEW_PROPERTY(userAgent, NSString)
 - 
 - /**
 -  * Expose methods to enable messaging the webview.
 -  */
 - RCT_EXPORT_VIEW_PROPERTY(messagingEnabled, BOOL)
 - RCT_EXPORT_VIEW_PROPERTY(onMessage, RCTDirectEventBlock)
 - 
 - RCT_EXPORT_METHOD(postMessage:(nonnull NSNumber *)reactTag message:(NSString *)message)
 - {
 -   [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);
 -     } else {
 -       [view postMessage:message];
 -     }
 -   }];
 - }
 - 
 - RCT_CUSTOM_VIEW_PROPERTY(bounces, BOOL, RNCWKWebView) {
 -   view.bounces = json == nil ? true : [RCTConvert BOOL: json];
 - }
 - 
 - RCT_CUSTOM_VIEW_PROPERTY(scrollEnabled, BOOL, RNCWKWebView) {
 -   view.scrollEnabled = json == nil ? true : [RCTConvert BOOL: json];
 - }
 - 
 - RCT_CUSTOM_VIEW_PROPERTY(decelerationRate, CGFloat, RNCWKWebView) {
 -   view.decelerationRate = json == nil ? UIScrollViewDecelerationRateNormal : [RCTConvert CGFloat: json];
 - }
 - 
 - RCT_EXPORT_METHOD(injectJavaScript:(nonnull NSNumber *)reactTag script:(NSString *)script)
 - {
 -   [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);
 -     } else {
 -       [view injectJavaScript:script];
 -     }
 -   }];
 - }
 - 
 - RCT_EXPORT_METHOD(goBack:(nonnull NSNumber *)reactTag)
 - {
 -   [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);
 -     } else {
 -       [view goBack];
 -     }
 -   }];
 - }
 - 
 - RCT_EXPORT_METHOD(goForward:(nonnull NSNumber *)reactTag)
 - {
 -   [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);
 -     } else {
 -       [view goForward];
 -     }
 -   }];
 - }
 - 
 - RCT_EXPORT_METHOD(reload:(nonnull NSNumber *)reactTag)
 - {
 -   [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);
 -     } else {
 -       [view reload];
 -     }
 -   }];
 - }
 - 
 - RCT_EXPORT_METHOD(stopLoading:(nonnull NSNumber *)reactTag)
 - {
 -   [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);
 -     } else {
 -       [view stopLoading];
 -     }
 -   }];
 - }
 - 
 - #pragma mark - Exported synchronous methods
 - 
 - - (BOOL)          webView:(RNCWKWebView *)webView
 - shouldStartLoadForRequest:(NSMutableDictionary<NSString *, id> *)request
 -              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: "
 -                "got %lld, expected %lld", (long long)lockIdentifier, (long long)_shouldStartLoadLock.condition);
 -   }
 - }
 - 
 - @end
 
 
  |