import 'dart:async'; import 'package:flutter/services.dart'; import 'webview_feature.dart'; import '../types.dart'; ///Class that represents an Android-specific class that manages Service Workers used by [WebView]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerControllerCompat class AndroidServiceWorkerController { static AndroidServiceWorkerController? _instance; static const MethodChannel _channel = const MethodChannel( 'com.pichillilorenzo/flutter_inappwebview_android_serviceworkercontroller'); AndroidServiceWorkerClient? serviceWorkerClient; ///Gets the [AndroidServiceWorkerController] shared instance. static AndroidServiceWorkerController instance() { return (_instance != null) ? _instance! : _init(); } static AndroidServiceWorkerController _init() { _channel.setMethodCallHandler(_handleMethod); _instance = AndroidServiceWorkerController(); return _instance!; } static Future _handleMethod(MethodCall call) async { AndroidServiceWorkerController controller = AndroidServiceWorkerController.instance(); AndroidServiceWorkerClient? serviceWorkerClient = controller.serviceWorkerClient; switch (call.method) { case "shouldInterceptRequest": String url = call.arguments["url"]; String method = call.arguments["method"]; Map? headers = call.arguments["headers"]?.cast(); bool isForMainFrame = call.arguments["isForMainFrame"]; bool hasGesture = call.arguments["hasGesture"]; bool isRedirect = call.arguments["isRedirect"]; var request = new WebResourceRequest( url: url, method: method, headers: headers, isForMainFrame: isForMainFrame, hasGesture: hasGesture, isRedirect: isRedirect); if (serviceWorkerClient != null && serviceWorkerClient.shouldInterceptRequest != null) { return (await serviceWorkerClient.shouldInterceptRequest!(request)) ?.toMap(); } break; default: throw UnimplementedError("Unimplemented ${call.method} method"); } return null; } ///Gets whether Service Workers support content URL access. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_CONTENT_ACCESS]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#getAllowContentAccess() static Future getAllowContentAccess() async { Map args = {}; return await _channel.invokeMethod('getAllowContentAccess', args); } ///Gets whether Service Workers support file access. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_FILE_ACCESS]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#getAllowFileAccess() static Future getAllowFileAccess() async { Map args = {}; return await _channel.invokeMethod('getAllowFileAccess', args); } ///Gets whether Service Workers are prohibited from loading any resources from the network. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_BLOCK_NETWORK_LOADS]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#getBlockNetworkLoads() static Future getBlockNetworkLoads() async { Map args = {}; return await _channel.invokeMethod('getBlockNetworkLoads', args); } ///Gets the current setting for overriding the cache mode. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_CACHE_MODE]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#getCacheMode() static Future getCacheMode() async { Map args = {}; return AndroidCacheMode.fromValue(await _channel.invokeMethod('getCacheMode', args)); } ///Enables or disables content URL access from Service Workers. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_CONTENT_ACCESS]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#setAllowContentAccess(boolean) static Future setAllowContentAccess(bool allow) async { Map args = {}; args.putIfAbsent("allow", () => allow); await _channel.invokeMethod('setAllowContentAccess', args); } ///Enables or disables file access within Service Workers. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_FILE_ACCESS]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#setAllowFileAccess(boolean) static Future setAllowFileAccess(bool allow) async { Map args = {}; args.putIfAbsent("allow", () => allow); await _channel.invokeMethod('setAllowFileAccess', args); } ///Sets whether Service Workers should not load resources from the network. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_BLOCK_NETWORK_LOADS]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#setBlockNetworkLoads(boolean) static Future setBlockNetworkLoads(bool flag) async { Map args = {}; args.putIfAbsent("flag", () => flag); await _channel.invokeMethod('setBlockNetworkLoads', args); } ///Overrides the way the cache is used. ///This method should only be called if [AndroidWebViewFeature.isFeatureSupported] returns `true` for [AndroidWebViewFeature.SERVICE_WORKER_CACHE_MODE]. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerWebSettingsCompat#setCacheMode(int) static Future setCacheMode(AndroidCacheMode mode) async { Map args = {}; args.putIfAbsent("mode", () => mode.toValue()); await _channel.invokeMethod('setCacheMode', args); } } ///Class that represents an Android-specific class for clients to capture Service Worker related callbacks. /// ///**NOTE**: available on Android 24+. /// ///**Official Android API**: https://developer.android.com/reference/androidx/webkit/ServiceWorkerClientCompat class AndroidServiceWorkerClient { ///Notify the host application of a resource request and allow the application to return the data. ///If the return value is `null`, the Service Worker will continue to load the resource as usual. ///Otherwise, the return response and data will be used. /// ///This method is called only if [AndroidWebViewFeature.SERVICE_WORKER_SHOULD_INTERCEPT_REQUEST] is supported. ///You can check whether that flag is supported using [AndroidWebViewFeature.isFeatureSupported]. /// ///[request] represents an object containing the details of the request. /// ///**NOTE**: available on Android 24+. final Future Function(WebResourceRequest request)? shouldInterceptRequest; AndroidServiceWorkerClient({ this.shouldInterceptRequest }); }