2019-11-10 13:11:30 +00:00
import ' package:flutter/foundation.dart ' ;
///ContentBlocker class represents a set of rules to use block content in the browser window.
///
///On iOS, it uses [WKContentRuleListStore](https://developer.apple.com/documentation/webkit/wkcontentruleliststore).
///On Android, it uses a custom implementation because such functionality doesn't exist.
2019-11-10 10:50:01 +00:00
///
2019-11-10 13:11:30 +00:00
///In general, this [article](https://developer.apple.com/documentation/safariservices/creating_a_content_blocker) can be used to get an overview about this functionality
///but on Android there are two types of [action] that are unavailable: `block-cookies` and `ignore-previous-rules`.
2019-10-26 20:11:23 +00:00
class ContentBlocker {
2019-11-10 13:11:30 +00:00
///Trigger of the content blocker. The trigger tells to the WebView when to perform the corresponding action.
2019-10-26 20:11:23 +00:00
ContentBlockerTrigger trigger ;
2019-11-10 13:11:30 +00:00
///Action associated to the trigger. The action tells to the WebView what to do when the trigger is matched.
2019-10-26 20:11:23 +00:00
ContentBlockerAction action ;
2019-11-10 13:11:30 +00:00
ContentBlocker ( { @ required this . trigger , @ required this . action } ) ;
2019-10-26 20:11:23 +00:00
Map < String , Map < String , dynamic > > toMap ( ) {
return {
" trigger " : trigger . toMap ( ) ,
" action " : action . toMap ( )
} ;
}
2019-11-04 00:39:23 +00:00
static ContentBlocker fromMap ( Map < dynamic , Map < dynamic , dynamic > > map ) {
return ContentBlocker (
2019-11-10 13:11:30 +00:00
trigger: ContentBlockerTrigger . fromMap (
2019-11-04 00:39:23 +00:00
Map < String , dynamic > . from ( map [ " trigger " ] )
) ,
2019-11-10 13:11:30 +00:00
action: ContentBlockerAction . fromMap (
2019-11-04 00:39:23 +00:00
Map < String , dynamic > . from ( map [ " action " ] )
)
) ;
}
2019-10-26 20:11:23 +00:00
}
2019-11-10 13:11:30 +00:00
///ContentBlockerTriggerResourceType class represents the possible resource type defined for a [ContentBlockerTrigger].
2019-10-26 20:11:23 +00:00
class ContentBlockerTriggerResourceType {
final String _value ;
const ContentBlockerTriggerResourceType . _internal ( this . _value ) ;
2019-11-04 00:39:23 +00:00
static ContentBlockerTriggerResourceType fromValue ( String value ) {
2019-11-10 13:11:30 +00:00
return ( [ " document " , " image " , " style-sheet " , " script " , " font " ,
2019-11-04 00:39:23 +00:00
" media " , " svg-document " , " raw " ] . contains ( value ) ) ? ContentBlockerTriggerResourceType . _internal ( value ) : null ;
}
2019-10-28 03:58:25 +00:00
toValue ( ) = > _value ;
2019-10-26 20:11:23 +00:00
static const DOCUMENT = const ContentBlockerTriggerResourceType . _internal ( ' document ' ) ;
static const IMAGE = const ContentBlockerTriggerResourceType . _internal ( ' image ' ) ;
static const STYLE_SHEET = const ContentBlockerTriggerResourceType . _internal ( ' style-sheet ' ) ;
static const SCRIPT = const ContentBlockerTriggerResourceType . _internal ( ' script ' ) ;
static const FONT = const ContentBlockerTriggerResourceType . _internal ( ' font ' ) ;
static const MEDIA = const ContentBlockerTriggerResourceType . _internal ( ' media ' ) ;
static const SVG_DOCUMENT = const ContentBlockerTriggerResourceType . _internal ( ' svg-document ' ) ;
2019-11-10 13:11:30 +00:00
///Any untyped load
2019-10-26 20:11:23 +00:00
static const RAW = const ContentBlockerTriggerResourceType . _internal ( ' raw ' ) ;
}
2019-11-10 13:11:30 +00:00
///ContentBlockerTriggerLoadType class represents the possible load type for a [ContentBlockerTrigger].
2019-10-27 03:35:05 +00:00
class ContentBlockerTriggerLoadType {
final String _value ;
const ContentBlockerTriggerLoadType . _internal ( this . _value ) ;
2019-11-04 00:39:23 +00:00
static ContentBlockerTriggerLoadType fromValue ( String value ) {
return ( [ " first-party " , " third-party " ] . contains ( value ) ) ? ContentBlockerTriggerLoadType . _internal ( value ) : null ;
}
2019-10-28 03:58:25 +00:00
toValue ( ) = > _value ;
2019-10-27 03:35:05 +00:00
2019-11-10 13:11:30 +00:00
///FIRST_PARTY is triggered only if the resource has the same scheme, domain, and port as the main page resource.
2019-10-27 03:35:05 +00:00
static const FIRST_PARTY = const ContentBlockerTriggerLoadType . _internal ( ' first-party ' ) ;
2019-11-10 13:11:30 +00:00
///THIRD_PARTY is triggered if the resource is not from the same domain as the main page resource.
2019-10-27 03:35:05 +00:00
static const THIRD_PARTY = const ContentBlockerTriggerLoadType . _internal ( ' third-party ' ) ;
}
2019-11-10 13:11:30 +00:00
///Trigger of the content blocker. The trigger tells to the WebView when to perform the corresponding action.
///A trigger dictionary must include an [ContentBlockerTrigger.urlFilter], which specifies a pattern to match the URL against.
///The remaining properties are optional and modify the behavior of the trigger.
///For example, you can limit the trigger to specific domains or have it not apply when a match is found on a specific domain.
2019-10-26 20:11:23 +00:00
class ContentBlockerTrigger {
2019-11-10 13:11:30 +00:00
///A regular expression pattern to match the URL against.
2019-10-26 20:11:23 +00:00
String urlFilter ;
2019-11-10 13:11:30 +00:00
///Used only by iOS. A Boolean value. The default value is false.
2019-10-27 03:35:05 +00:00
bool urlFilterIsCaseSensitive ;
2019-11-10 13:11:30 +00:00
///A list of [ContentBlockerTriggerResourceType] representing the resource types (how the browser intends to use the resource) that the rule should match.
///If not specified, the rule matches all resource types.
2019-10-26 20:11:23 +00:00
List < ContentBlockerTriggerResourceType > resourceType ;
2019-11-10 13:11:30 +00:00
///A list of strings matched to a URL's domain; limits action to a list of specific domains.
///Values must be lowercase ASCII, or punycode for non-ASCII. Add * in front to match domain and subdomains. Can't be used with [ContentBlockerTrigger.unlessDomain].
2019-10-27 03:35:05 +00:00
List < String > ifDomain ;
2019-11-10 13:11:30 +00:00
///A list of strings matched to a URL's domain; acts on any site except domains in a provided list.
///Values must be lowercase ASCII, or punycode for non-ASCII. Add * in front to match domain and subdomains. Can't be used with [ContentBlockerTrigger.ifDomain].
2019-10-27 03:35:05 +00:00
List < String > unlessDomain ;
2019-11-10 13:11:30 +00:00
///A list of [ContentBlockerTriggerLoadType] that can include one of two mutually exclusive values. If not specified, the rule matches all load types.
2019-10-27 03:35:05 +00:00
List < ContentBlockerTriggerLoadType > loadType ;
2019-11-10 13:11:30 +00:00
///A list of strings matched to the entire main document URL; limits the action to a specific list of URL patterns.
///Values must be lowercase ASCII, or punycode for non-ASCII. Can't be used with [ContentBlockerTrigger.unlessTopUrl].
2019-10-27 03:35:05 +00:00
List < String > ifTopUrl ;
2019-11-10 13:11:30 +00:00
///An array of strings matched to the entire main document URL; acts on any site except URL patterns in provided list.
///Values must be lowercase ASCII, or punycode for non-ASCII. Can't be used with [ContentBlockerTrigger.ifTopUrl].
2019-10-27 03:35:05 +00:00
List < String > unlessTopUrl ;
2019-10-26 20:11:23 +00:00
2019-11-10 13:11:30 +00:00
ContentBlockerTrigger ( { @ required String urlFilter , bool urlFilterIsCaseSensitive = false , List < ContentBlockerTriggerResourceType > resourceType = const [ ] ,
2019-10-27 03:35:05 +00:00
List < String > ifDomain = const [ ] , List < String > unlessDomain = const [ ] , List < ContentBlockerTriggerLoadType > loadType = const [ ] ,
List < String > ifTopUrl = const [ ] , List < String > unlessTopUrl = const [ ] } ) {
this . urlFilter = urlFilter ;
2019-11-10 13:11:30 +00:00
assert ( this . urlFilter ! = null ) ;
2019-10-27 03:35:05 +00:00
this . resourceType = resourceType ;
this . urlFilterIsCaseSensitive = urlFilterIsCaseSensitive ;
this . ifDomain = ifDomain ;
this . unlessDomain = unlessDomain ;
assert ( ! ( this . ifDomain . isEmpty | | this . unlessDomain . isEmpty ) = = false ) ;
this . loadType = loadType ;
assert ( this . loadType . length < = 2 ) ;
this . ifTopUrl = ifTopUrl ;
this . unlessTopUrl = unlessTopUrl ;
assert ( ! ( this . ifTopUrl . isEmpty | | this . unlessTopUrl . isEmpty ) = = false ) ;
}
2019-10-26 20:11:23 +00:00
Map < String , dynamic > toMap ( ) {
List < String > resourceTypeStringList = [ ] ;
resourceType . forEach ( ( type ) {
2019-10-28 03:58:25 +00:00
resourceTypeStringList . add ( type . toValue ( ) ) ;
2019-10-26 20:11:23 +00:00
} ) ;
2019-10-27 03:35:05 +00:00
List < String > loadTypeStringList = [ ] ;
loadType . forEach ( ( type ) {
2019-10-28 03:58:25 +00:00
loadTypeStringList . add ( type . toValue ( ) ) ;
2019-10-27 03:35:05 +00:00
} ) ;
2019-10-26 20:11:23 +00:00
2019-10-27 03:35:05 +00:00
Map < String , dynamic > map = {
2019-10-26 20:11:23 +00:00
" url-filter " : urlFilter ,
2019-10-27 03:35:05 +00:00
" url-filter-is-case-sensitive " : urlFilterIsCaseSensitive ,
" if-domain " : ifDomain ,
" unless-domain " : unlessDomain ,
" resource-type " : resourceTypeStringList ,
" load-type " : loadTypeStringList ,
" if-top-url " : ifTopUrl ,
" unless-top-url " : unlessTopUrl
2019-10-26 20:11:23 +00:00
} ;
2019-10-27 03:35:05 +00:00
map . keys
. where ( ( key ) = > map [ key ] = = null | | ( map [ key ] is List & & ( map [ key ] as List ) . length = = 0 ) ) // filter keys
. toList ( ) // create a copy to avoid concurrent modifications
. forEach ( map . remove ) ;
return map ;
2019-10-26 20:11:23 +00:00
}
2019-11-04 00:39:23 +00:00
static ContentBlockerTrigger fromMap ( Map < String , dynamic > map ) {
List < ContentBlockerTriggerResourceType > resourceType = [ ] ;
List < ContentBlockerTriggerLoadType > loadType = [ ] ;
List < String > resourceTypeStringList = List < String > . from ( map [ " resource-type " ] ? ? [ ] ) ;
resourceTypeStringList . forEach ( ( type ) {
resourceType . add ( ContentBlockerTriggerResourceType . fromValue ( type ) ) ;
} ) ;
List < String > loadTypeStringList = List < String > . from ( map [ " load-type " ] ? ? [ ] ) ;
loadTypeStringList . forEach ( ( type ) {
loadType . add ( ContentBlockerTriggerLoadType . fromValue ( type ) ) ;
} ) ;
return ContentBlockerTrigger (
2019-11-10 13:11:30 +00:00
urlFilter: map [ " url-filter " ] ,
2019-11-04 00:39:23 +00:00
urlFilterIsCaseSensitive: map [ " url-filter-is-case-sensitive " ] ,
ifDomain: List < String > . from ( map [ " if-domain " ] ? ? [ ] ) ,
unlessDomain: List < String > . from ( map [ " unless-domain " ] ? ? [ ] ) ,
resourceType: resourceType ,
loadType: loadType ,
ifTopUrl: List < String > . from ( map [ " if-top-url " ] ? ? [ ] ) ,
unlessTopUrl: List < String > . from ( map [ " unless-top-url " ] ? ? [ ] )
) ;
}
2019-10-26 20:11:23 +00:00
}
2019-11-10 13:11:30 +00:00
///ContentBlockerActionType class represents the kind of action that can be used with a [ContentBlockerTrigger].
2019-10-26 20:11:23 +00:00
class ContentBlockerActionType {
final String _value ;
const ContentBlockerActionType . _internal ( this . _value ) ;
2019-11-04 00:39:23 +00:00
static ContentBlockerActionType fromValue ( String value ) {
return ( [ " block " , " css-display-none " , " make-https " ] . contains ( value ) ) ? ContentBlockerActionType . _internal ( value ) : null ;
}
2019-10-28 03:58:25 +00:00
toValue ( ) = > _value ;
2019-10-26 20:11:23 +00:00
2019-11-10 13:11:30 +00:00
///Stops loading of the resource. If the resource was cached, the cache is ignored.
2019-10-26 20:11:23 +00:00
static const BLOCK = const ContentBlockerActionType . _internal ( ' block ' ) ;
2019-11-10 13:11:30 +00:00
///Hides elements of the page based on a CSS selector. A selector field contains the selector list. Any matching element has its display property set to none, which hides it.
///
///**NOTE**: on Android, JavaScript must be enabled.
2019-10-26 20:11:23 +00:00
static const CSS_DISPLAY_NONE = const ContentBlockerActionType . _internal ( ' css-display-none ' ) ;
2019-11-10 13:11:30 +00:00
///Changes a URL from http to https. URLs with a specified (nondefault) port and links using other protocols are unaffected.
2019-10-26 20:11:23 +00:00
static const MAKE_HTTPS = const ContentBlockerActionType . _internal ( ' make-https ' ) ;
}
2019-11-10 13:11:30 +00:00
///Action associated to the trigger. The action tells to the WebView what to do when the trigger is matched.
///When a trigger matches a resource, the browser queues the associated action for execution.
///The WebView evaluates all the triggers, it executes the actions in order.
///When a domain matches a trigger, all rules after the triggered rule that specify the same action are skipped.
///Group the rules with similar actions together to improve performance.
2019-10-26 20:11:23 +00:00
class ContentBlockerAction {
2019-11-10 13:11:30 +00:00
///Type of the action.
2019-10-26 20:11:23 +00:00
ContentBlockerActionType type ;
2019-11-10 13:11:30 +00:00
///If the action type is [ContentBlockerActionType.CSS_DISPLAY_NONE], then also the [selector] property is required, otherwise it is ignored.
///It specify a string that defines a selector list. Use CSS identifiers as the individual selector values, separated by commas.
2019-10-26 20:11:23 +00:00
String selector ;
2019-11-10 13:11:30 +00:00
ContentBlockerAction ( { @ required ContentBlockerActionType type , String selector } ) {
2019-10-26 20:11:23 +00:00
this . type = type ;
2019-11-10 13:11:30 +00:00
assert ( this . type ! = null ) ;
2019-10-26 20:11:23 +00:00
if ( this . type = = ContentBlockerActionType . CSS_DISPLAY_NONE ) {
assert ( selector ! = null ) ;
}
this . selector = selector ;
}
Map < String , dynamic > toMap ( ) {
2019-10-27 03:35:05 +00:00
Map < String , dynamic > map = {
2019-10-28 03:58:25 +00:00
" type " : type . toValue ( ) ,
2019-10-26 20:11:23 +00:00
" selector " : selector
} ;
2019-10-27 03:35:05 +00:00
map . keys
. where ( ( key ) = > map [ key ] = = null | | ( map [ key ] is List & & ( map [ key ] as List ) . length = = 0 ) ) // filter keys
. toList ( ) // create a copy to avoid concurrent modifications
. forEach ( map . remove ) ;
return map ;
2019-10-26 20:11:23 +00:00
}
2019-11-04 00:39:23 +00:00
static ContentBlockerAction fromMap ( Map < String , dynamic > map ) {
return ContentBlockerAction (
2019-11-10 13:11:30 +00:00
type: ContentBlockerActionType . fromValue ( map [ " type " ] ) ,
2019-11-04 00:39:23 +00:00
selector: map [ " selector " ]
) ;
}
2019-10-26 20:11:23 +00:00
}