FlutterWebviewPlugin.m 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. #import "FlutterWebviewPlugin.h"
  2. static NSString *const CHANNEL_NAME = @"flutter_webview_plugin";
  3. // UIWebViewDelegate
  4. @interface FlutterWebviewPlugin() <UIWebViewDelegate> {
  5. BOOL _enableAppScheme;
  6. }
  7. @end
  8. @implementation FlutterWebviewPlugin
  9. + (void)registerWithRegistrar:(NSObject<FlutterPluginRegistrar>*)registrar {
  10. channel = [FlutterMethodChannel
  11. methodChannelWithName:CHANNEL_NAME
  12. binaryMessenger:[registrar messenger]];
  13. UIViewController *viewController = (UIViewController *)registrar.messenger;
  14. FlutterWebviewPlugin* instance = [[FlutterWebviewPlugin alloc] initWithViewController:viewController];
  15. [registrar addMethodCallDelegate:instance channel:channel];
  16. }
  17. - (instancetype)initWithViewController:(UIViewController *)viewController {
  18. self = [super init];
  19. if (self) {
  20. self.viewController = viewController;
  21. }
  22. return self;
  23. }
  24. - (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
  25. if ([@"launch" isEqualToString:call.method]) {
  26. if (!self.webview)
  27. [self initWebView:call];
  28. else
  29. [self launch:call];
  30. result(nil);
  31. } else if ([@"close" isEqualToString:call.method]) {
  32. [self closeWebView];
  33. result(nil);
  34. } else if ([@"eval" isEqualToString:call.method]) {
  35. result([self evalJavascript:call]);
  36. } else {
  37. result(FlutterMethodNotImplemented);
  38. }
  39. }
  40. - (void)initWebView:(FlutterMethodCall*)call {
  41. // NSNumber *withJavascript = call.arguments[@"withJavascript"];
  42. NSNumber *clearCache = call.arguments[@"clearCache"];
  43. NSNumber *clearCookies = call.arguments[@"clearCookies"];
  44. NSNumber *hidden = call.arguments[@"hidden"];
  45. NSDictionary *rect = call.arguments[@"rect"];
  46. _enableAppScheme = call.arguments[@"enableAppScheme"];
  47. NSString *userAgent = call.arguments[@"userAgent"];
  48. //
  49. if (clearCache != (id)[NSNull null] && [clearCache boolValue]) {
  50. [[NSURLCache sharedURLCache] removeAllCachedResponses];
  51. }
  52. if (clearCookies != (id)[NSNull null] && [clearCookies boolValue]) {
  53. [[NSURLSession sharedSession] resetWithCompletionHandler:^{
  54. }];
  55. }
  56. if (userAgent != (id)[NSNull null]) {
  57. [[NSUserDefaults standardUserDefaults] registerDefaults:@{@"UserAgent": userAgent}];
  58. }
  59. CGRect rc;
  60. if (rect != nil) {
  61. rc = CGRectMake([[rect valueForKey:@"left"] doubleValue],
  62. [[rect valueForKey:@"top"] doubleValue],
  63. [[rect valueForKey:@"width"] doubleValue],
  64. [[rect valueForKey:@"height"] doubleValue]);
  65. } else {
  66. // TODO: create top NavigatorController and push
  67. rc = self.viewController.view.bounds;
  68. }
  69. self.webview = [[UIWebView alloc] initWithFrame:rc];
  70. self.webview.delegate = self;
  71. if (hidden != (id)[NSNull null] && [hidden boolValue])
  72. self.webview.hidden = YES;
  73. [self.viewController.view addSubview:self.webview];
  74. [self launch:call];
  75. }
  76. - (void)launch:(FlutterMethodCall*)call {
  77. NSString *url = call.arguments[@"url"];
  78. NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:url]];
  79. [self.webview loadRequest:request];
  80. }
  81. - (NSString *)evalJavascript:(FlutterMethodCall*)call {
  82. NSString *code = call.arguments[@"code"];
  83. NSString *result = [self.webview stringByEvaluatingJavaScriptFromString:code];
  84. return result;
  85. }
  86. - (void)closeWebView {
  87. [self.webview stopLoading];
  88. [self.webview removeFromSuperview];
  89. self.webview.delegate = nil;
  90. self.webview = nil;
  91. // manually trigger onDestroy
  92. [channel invokeMethod:@"onDestroy" arguments:nil];
  93. }
  94. #pragma mark -- WebView Delegate
  95. - (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType {
  96. id data = @{@"url": request.URL.absoluteString,
  97. @"type": @"shouldStart",
  98. @"navigationType": [NSNumber numberWithInt:navigationType]};
  99. [channel invokeMethod:@"onState" arguments:data];
  100. if (navigationType == UIWebViewNavigationTypeBackForward)
  101. [channel invokeMethod:@"onBackPressed" arguments:nil];
  102. else {
  103. id data = @{@"url": request.URL.absoluteString};
  104. [channel invokeMethod:@"onUrlChanged" arguments:data];
  105. }
  106. if (_enableAppScheme)
  107. return YES;
  108. // disable some scheme
  109. return [request.URL.scheme isEqualToString:@"http"] ||
  110. [request.URL.scheme isEqualToString:@"https"] ||
  111. [request.URL.scheme isEqualToString:@"about"];
  112. }
  113. -(void)webViewDidStartLoad:(UIWebView *)webView {
  114. [channel invokeMethod:@"onState" arguments:@{@"type": @"startLoad"}];
  115. }
  116. - (void)webViewDidFinishLoad:(UIWebView *)webView {
  117. [channel invokeMethod:@"onState" arguments:@{@"type": @"finishLoad"}];
  118. }
  119. - (void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error {
  120. id data = [FlutterError errorWithCode:[NSString stringWithFormat:@"%ld", error.code]
  121. message:error.localizedDescription
  122. details:error.localizedFailureReason];
  123. [channel invokeMethod:@"onError" arguments:data];
  124. }
  125. #pragma mark -- WkWebView Delegate
  126. @end