2019-10-26 02:42:50 +00:00
import ' dart:async ' ;
2020-05-11 00:48:41 +00:00
import ' dart:collection ' ;
2019-10-26 02:42:50 +00:00
import ' package:flutter/services.dart ' ;
import ' types.dart ' ;
import ' in_app_browser.dart ' ;
///This class uses native [Chrome Custom Tabs](https://developer.android.com/reference/android/support/customtabs/package-summary) on Android
///and [SFSafariViewController](https://developer.apple.com/documentation/safariservices/sfsafariviewcontroller) on iOS.
///
2019-11-10 13:11:30 +00:00
///[browserFallback] represents the [InAppBrowser] instance fallback in case `Chrome Custom Tabs`/`SFSafariViewController` is not available.
2019-10-26 02:42:50 +00:00
class ChromeSafariBrowser {
2021-01-28 16:10:15 +00:00
late String uuid ;
InAppBrowser ? browserFallback ;
2020-05-11 00:48:41 +00:00
Map < int , ChromeSafariBrowserMenuItem > _menuItems = new HashMap ( ) ;
2019-10-26 02:42:50 +00:00
bool _isOpened = false ;
2021-01-28 16:10:15 +00:00
late MethodChannel _channel ;
2020-05-29 17:56:03 +00:00
static const MethodChannel _sharedChannel =
const MethodChannel ( ' com.pichillilorenzo/flutter_chromesafaribrowser ' ) ;
2019-10-26 02:42:50 +00:00
///Initialize the [ChromeSafariBrowser] instance with an [InAppBrowser] fallback instance or `null`.
2019-12-01 11:55:06 +00:00
ChromeSafariBrowser ( { bFallback } ) {
2019-10-26 02:42:50 +00:00
uuid = uuidGenerator . v4 ( ) ;
2019-11-10 13:11:30 +00:00
browserFallback = bFallback ;
2019-12-18 20:34:40 +00:00
this . _channel =
MethodChannel ( ' com.pichillilorenzo/flutter_chromesafaribrowser_ $ uuid ' ) ;
this . _channel . setMethodCallHandler ( handleMethod ) ;
2019-10-26 02:42:50 +00:00
_isOpened = false ;
}
Future < dynamic > handleMethod ( MethodCall call ) async {
2019-12-01 11:55:06 +00:00
switch ( call . method ) {
2019-10-26 02:42:50 +00:00
case " onChromeSafariBrowserOpened " :
onOpened ( ) ;
break ;
2019-12-18 00:56:21 +00:00
case " onChromeSafariBrowserCompletedInitialLoad " :
onCompletedInitialLoad ( ) ;
2019-10-26 02:42:50 +00:00
break ;
case " onChromeSafariBrowserClosed " :
onClosed ( ) ;
this . _isOpened = false ;
break ;
2020-05-11 00:48:41 +00:00
case " onChromeSafariBrowserMenuItemActionPerform " :
String url = call . arguments [ " url " ] ;
String title = call . arguments [ " title " ] ;
int id = call . arguments [ " id " ] . toInt ( ) ;
2021-01-28 16:10:15 +00:00
if ( this . _menuItems [ id ] ! = null ) {
this . _menuItems [ id ] ! . action ( url , title ) ;
}
2020-05-11 00:48:41 +00:00
break ;
2019-10-26 02:42:50 +00:00
default :
throw UnimplementedError ( " Unimplemented ${ call . method } method " ) ;
}
}
///Opens an [url] in a new [ChromeSafariBrowser] instance.
///
2019-11-10 13:11:30 +00:00
///[url]: The [url] to load. Call [encodeUriComponent()] on this if the [url] contains Unicode characters.
2019-10-26 02:42:50 +00:00
///
2019-11-10 13:11:30 +00:00
///[options]: Options for the [ChromeSafariBrowser].
2019-10-26 02:42:50 +00:00
///
2019-11-10 13:11:30 +00:00
///[headersFallback]: The additional header of the [InAppBrowser] instance fallback to be used in the HTTP request for this URL, specified as a map from name to value.
2019-10-26 02:42:50 +00:00
///
2019-11-10 13:11:30 +00:00
///[optionsFallback]: Options used by the [InAppBrowser] instance fallback.
2020-05-21 01:34:39 +00:00
///
///[contextMenuFallback]: Context Menu used by the [InAppBrowser] instance fallback.
2019-12-01 11:55:06 +00:00
Future < void > open (
2021-01-28 16:10:15 +00:00
{ required String url ,
ChromeSafariBrowserClassOptions ? options ,
Map < String , String > ? headersFallback = const { } ,
InAppBrowserClassOptions ? optionsFallback } ) async {
assert ( url . isNotEmpty ) ;
2019-10-26 02:42:50 +00:00
this . throwIsAlreadyOpened ( message: ' Cannot open $ url ! ' ) ;
2019-10-27 03:35:05 +00:00
2021-01-28 16:10:15 +00:00
List < Map < String , dynamic > > menuItemList = [ ] ;
2020-05-11 00:48:41 +00:00
_menuItems . forEach ( ( key , value ) {
2020-05-29 17:56:03 +00:00
menuItemList . add ( { " id " : value . id , " label " : value . label } ) ;
2020-05-11 00:48:41 +00:00
} ) ;
2019-10-27 03:35:05 +00:00
2019-10-26 02:42:50 +00:00
Map < String , dynamic > args = < String , dynamic > { } ;
args . putIfAbsent ( ' uuid ' , ( ) = > uuid ) ;
args . putIfAbsent ( ' url ' , ( ) = > url ) ;
2020-05-11 00:48:41 +00:00
args . putIfAbsent ( ' options ' , ( ) = > options ? . toMap ( ) ? ? { } ) ;
2020-05-21 01:34:39 +00:00
args . putIfAbsent ( ' menuItemList ' , ( ) = > menuItemList ) ;
2020-05-29 17:56:03 +00:00
args . putIfAbsent ( ' uuidFallback ' , ( ) = > browserFallback ? . uuid ) ;
args . putIfAbsent ( ' headersFallback ' , ( ) = > headersFallback ? ? { } ) ;
2020-05-11 00:48:41 +00:00
args . putIfAbsent ( ' optionsFallback ' , ( ) = > optionsFallback ? . toMap ( ) ? ? { } ) ;
2020-05-29 17:56:03 +00:00
args . putIfAbsent ( ' contextMenuFallback ' ,
( ) = > browserFallback ? . contextMenu ? . toMap ( ) ? ? { } ) ;
2019-12-18 20:34:40 +00:00
await _sharedChannel . invokeMethod ( ' open ' , args ) ;
2019-10-26 02:42:50 +00:00
this . _isOpened = true ;
}
2020-05-11 00:48:41 +00:00
///Closes the [ChromeSafariBrowser] instance.
Future < void > close ( ) async {
Map < String , dynamic > args = < String , dynamic > { } ;
await _channel . invokeMethod ( " close " , args ) ;
}
///Adds a [ChromeSafariBrowserMenuItem] to the menu.
void addMenuItem ( ChromeSafariBrowserMenuItem menuItem ) {
this . _menuItems [ menuItem . id ] = menuItem ;
}
///Adds a list of [ChromeSafariBrowserMenuItem] to the menu.
void addMenuItems ( List < ChromeSafariBrowserMenuItem > menuItems ) {
menuItems . forEach ( ( menuItem ) {
this . _menuItems [ menuItem . id ] = menuItem ;
} ) ;
}
2020-06-30 12:29:19 +00:00
///On Android, returns `true` if Chrome Custom Tabs is available.
///On iOS, returns `true` if SFSafariViewController is available.
///Otherwise returns `false`.
static Future < bool > isAvailable ( ) async {
Map < String , dynamic > args = < String , dynamic > { } ;
return await _sharedChannel . invokeMethod ( " isAvailable " , args ) ;
}
2019-10-26 02:42:50 +00:00
///Event fires when the [ChromeSafariBrowser] is opened.
2019-12-01 11:55:06 +00:00
void onOpened ( ) { }
2019-10-26 02:42:50 +00:00
2019-12-18 00:56:21 +00:00
///Event fires when the initial URL load is complete.
void onCompletedInitialLoad ( ) { }
2019-10-26 02:42:50 +00:00
///Event fires when the [ChromeSafariBrowser] is closed.
2019-12-01 11:55:06 +00:00
void onClosed ( ) { }
2019-10-26 02:42:50 +00:00
2019-11-25 22:04:17 +00:00
///Returns `true` if the [ChromeSafariBrowser] instance is opened, otherwise `false`.
2019-10-26 02:42:50 +00:00
bool isOpened ( ) {
return this . _isOpened ;
}
void throwIsAlreadyOpened ( { String message = ' ' } ) {
if ( this . isOpened ( ) ) {
2019-12-01 11:55:06 +00:00
throw Exception ( [
' Error: ${ ( message . isEmpty ) ? ' ' : message + ' ' } The browser is already opened. '
] ) ;
2019-10-26 02:42:50 +00:00
}
}
void throwIsNotOpened ( { String message = ' ' } ) {
if ( ! this . isOpened ( ) ) {
2019-12-01 11:55:06 +00:00
throw Exception ( [
' Error: ${ ( message . isEmpty ) ? ' ' : message + ' ' } The browser is not opened. '
] ) ;
2019-10-26 02:42:50 +00:00
}
}
2019-12-01 11:55:06 +00:00
}
2020-05-11 00:48:41 +00:00
2020-05-29 17:56:03 +00:00
///Class that represents a custom menu item for a [ChromeSafariBrowser] instance.
2020-05-11 00:48:41 +00:00
class ChromeSafariBrowserMenuItem {
2020-05-29 17:56:03 +00:00
///The menu item id
2020-05-11 00:48:41 +00:00
int id ;
2020-05-29 17:56:03 +00:00
///The label of the menu item
2020-05-11 00:48:41 +00:00
String label ;
2020-05-29 17:56:03 +00:00
///Callback function to be invoked when the menu item is clicked
2020-05-11 00:48:41 +00:00
final void Function ( String url , String title ) action ;
2020-05-29 17:56:03 +00:00
ChromeSafariBrowserMenuItem (
2021-01-28 16:10:15 +00:00
{ required this . id , required this . label , required this . action } ) ;
2020-05-29 17:56:03 +00:00
Map < String , dynamic > toMap ( ) {
return { " id " : id , " label " : label } ;
}
Map < String , dynamic > toJson ( ) {
return this . toMap ( ) ;
}
@ override
String toString ( ) {
return toMap ( ) . toString ( ) ;
}
}