|
|
@@ -1,15 +1,29 @@
|
|
|
#import "FlutterWebviewPlugin.h"
|
|
|
|
|
|
static NSString *const CHANNEL_NAME = @"flutter_webview_plugin";
|
|
|
+static NSString *const EVENT_CHANNEL_NAME = @"flutter_webview_plugin_event";
|
|
|
+
|
|
|
+// UIWebViewDelegate
|
|
|
+@interface FlutterWebviewPlugin() <UIWebViewDelegate, FlutterStreamHandler> {
|
|
|
+ FlutterEventSink _eventSink;
|
|
|
+}
|
|
|
+@end
|
|
|
|
|
|
@implementation FlutterWebviewPlugin
|
|
|
+ (void)registerWithRegistrar:(NSObject<FlutterPluginRegistrar>*)registrar {
|
|
|
channel = [FlutterMethodChannel
|
|
|
methodChannelWithName:CHANNEL_NAME
|
|
|
binaryMessenger:[registrar messenger]];
|
|
|
+
|
|
|
UIViewController *viewController = (UIViewController *)registrar.messenger;
|
|
|
FlutterWebviewPlugin* instance = [[FlutterWebviewPlugin alloc] initWithViewController:viewController];
|
|
|
+
|
|
|
[registrar addMethodCallDelegate:instance channel:channel];
|
|
|
+
|
|
|
+ FlutterEventChannel* event =
|
|
|
+ [FlutterEventChannel eventChannelWithName:EVENT_CHANNEL_NAME
|
|
|
+ binaryMessenger:[registrar messenger]];
|
|
|
+ [event setStreamHandler:instance];
|
|
|
}
|
|
|
|
|
|
- (instancetype)initWithViewController:(UIViewController *)viewController {
|
|
|
@@ -22,7 +36,10 @@ static NSString *const CHANNEL_NAME = @"flutter_webview_plugin";
|
|
|
|
|
|
- (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
|
|
|
if ([@"launch" isEqualToString:call.method]) {
|
|
|
- [self showWebView:call];
|
|
|
+ if (!self.webview)
|
|
|
+ [self initWebView:call];
|
|
|
+ else
|
|
|
+ [self launch:call];
|
|
|
result(nil);
|
|
|
} else if ([@"close" isEqualToString:call.method]) {
|
|
|
[self closeWebView];
|
|
|
@@ -32,27 +49,97 @@ static NSString *const CHANNEL_NAME = @"flutter_webview_plugin";
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-- (void)showWebView:(FlutterMethodCall*)call {
|
|
|
+- (void)launch:(FlutterMethodCall*)call {
|
|
|
NSString *url = call.arguments[@"url"];
|
|
|
- NSNumber *withJavascript = call.arguments[@"withJavascript"];
|
|
|
+
|
|
|
+ NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:url]];
|
|
|
+ [self.webview loadRequest:request];
|
|
|
+}
|
|
|
+
|
|
|
+- (void)initWebView:(FlutterMethodCall*)call {
|
|
|
+ // NSNumber *withJavascript = call.arguments[@"withJavascript"];
|
|
|
NSNumber *clearCache = call.arguments[@"clearCache"];
|
|
|
NSNumber *clearCookies = call.arguments[@"clearCookies"];
|
|
|
- NSNumber *fullScreen = call.arguments[@"fullScreen"];
|
|
|
+ NSNumber *hidden = call.arguments[@"hidden"];
|
|
|
+ NSDictionary *rect = call.arguments[@"rect"];
|
|
|
+
|
|
|
+ //
|
|
|
+ if ([clearCache boolValue]) {
|
|
|
+ [[NSURLCache sharedURLCache] removeAllCachedResponses];
|
|
|
+ }
|
|
|
|
|
|
- self.webviewController = [[WebviewController alloc] initWithUrl:url withJavascript:withJavascript clearCache:clearCache clearCookes:clearCookies];
|
|
|
+ if ([clearCookies boolValue]) {
|
|
|
+ [[NSURLSession sharedSession] resetWithCompletionHandler:^{
|
|
|
+ }];
|
|
|
+ }
|
|
|
|
|
|
- if ([fullScreen boolValue]) {
|
|
|
- [self.viewController presentViewController:self.webviewController animated:YES completion:nil];
|
|
|
+ CGRect rc;
|
|
|
+ if (rect) {
|
|
|
+ rc = CGRectMake([[rect valueForKey:@"left"] doubleValue],
|
|
|
+ [[rect valueForKey:@"top"] doubleValue],
|
|
|
+ [[rect valueForKey:@"width"] doubleValue],
|
|
|
+ [[rect valueForKey:@"height"] doubleValue]);
|
|
|
} else {
|
|
|
- UINavigationController *navigation = [[UINavigationController alloc] initWithRootViewController:self.webviewController];
|
|
|
- [self.viewController presentModalViewController:navigation animated:YES];
|
|
|
+ rc = self.viewController.view.bounds;
|
|
|
}
|
|
|
+
|
|
|
+ self.webview = [[UIWebView alloc] initWithFrame:rc];
|
|
|
+ self.webview.delegate = self;
|
|
|
+
|
|
|
+ if (!hidden || ![hidden boolValue])
|
|
|
+ [self.viewController.view addSubview:self.webview];
|
|
|
+
|
|
|
+ [self launch:call];
|
|
|
}
|
|
|
|
|
|
- (void)closeWebView {
|
|
|
- [self.webviewController dismissViewControllerAnimated:YES completion:^{
|
|
|
- [channel invokeMethod:@"onDestroy" arguments:nil];
|
|
|
- }];
|
|
|
+ [self.webview stopLoading];
|
|
|
+ [self.webview removeFromSuperview];
|
|
|
+ self.webview.delegate = nil;
|
|
|
+ self.webview = nil;
|
|
|
}
|
|
|
|
|
|
+
|
|
|
+#pragma mark -- WebView Delegate
|
|
|
+- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType {
|
|
|
+ [self sendStateEvent:[NSString stringWithFormat:@"shouldStart %@", request.URL]];
|
|
|
+ return YES;
|
|
|
+}
|
|
|
+-(void)webViewDidStartLoad:(UIWebView *)webView {
|
|
|
+ [self sendStateEvent:@"startLoad"];
|
|
|
+}
|
|
|
+
|
|
|
+- (void)webViewDidFinishLoad:(UIWebView *)webView {
|
|
|
+ [self sendStateEvent:@"finishLoad"];
|
|
|
+}
|
|
|
+
|
|
|
+- (void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error {
|
|
|
+ id data = [FlutterError errorWithCode:[NSString stringWithFormat:@"%ld", error.code]
|
|
|
+ message:error.localizedDescription
|
|
|
+ details:error.localizedFailureReason];
|
|
|
+ [self sendStateEvent:data];
|
|
|
+}
|
|
|
+
|
|
|
+#pragma mark -- WkWebView Delegate
|
|
|
+
|
|
|
+#pragma mark -- FlutterStreamHandler impl
|
|
|
+
|
|
|
+- (FlutterError*)onListenWithArguments:(id)arguments eventSink:(FlutterEventSink)eventSink {
|
|
|
+ _eventSink = eventSink;
|
|
|
+ return nil;
|
|
|
+}
|
|
|
+
|
|
|
+- (FlutterError*)onCancelWithArguments:(id)arguments {
|
|
|
+ [[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
|
+ _eventSink = nil;
|
|
|
+ return nil;
|
|
|
+}
|
|
|
+
|
|
|
+- (void)sendStateEvent:(id)data {
|
|
|
+ // data should be @"" or [FlutterError]
|
|
|
+ if (!_eventSink)
|
|
|
+ return;
|
|
|
+
|
|
|
+ _eventSink(data);
|
|
|
+}
|
|
|
@end
|