2019-11-10 13:11:30 +00:00
import ' package:flutter/foundation.dart ' ;
2019-11-25 00:42:27 +00:00
import ' types.dart ' ;
2019-11-10 13:11:30 +00:00
2020-05-11 00:48:41 +00:00
///Class that represents a set of rules to use block content in the browser window.
2019-11-10 13:11:30 +00:00
///
///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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
ContentBlocker ( { @ required this . trigger , @ required this . action } ) ;
2019-10-26 20:11:23 +00:00
Map < String , Map < String , dynamic > > toMap ( ) {
2019-12-01 11:55:06 +00:00
return { " trigger " : trigger . toMap ( ) , " action " : action . toMap ( ) } ;
2019-10-26 20:11:23 +00:00
}
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-12-01 11:55:06 +00:00
Map < String , dynamic > . from ( map [ " trigger " ] ) ) ,
2019-11-10 13:11:30 +00:00
action: ContentBlockerAction . fromMap (
2019-12-01 11:55:06 +00:00
Map < String , dynamic > . from ( map [ " action " ] ) ) ) ;
2019-11-04 00:39:23 +00:00
}
2019-10-26 20:11:23 +00:00
}
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +00:00
ContentBlockerTrigger (
{ @ required String urlFilter ,
bool urlFilterIsCaseSensitive = false ,
List < ContentBlockerTriggerResourceType > resourceType = const [ ] ,
List < String > ifDomain = const [ ] ,
List < String > unlessDomain = const [ ] ,
List < ContentBlockerTriggerLoadType > loadType = const [ ] ,
List < String > ifTopUrl = const [ ] ,
List < String > unlessTopUrl = const [ ] } ) {
2019-10-27 03:35:05 +00:00
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
2019-12-01 11:55:06 +00:00
. where ( ( key ) = >
map [ key ] = = null | |
( map [ key ] is List & & ( map [ key ] as List ) . length = = 0 ) ) // filter keys
2019-10-27 03:35:05 +00:00
. 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 = [ ] ;
2019-12-01 11:55:06 +00:00
List < String > resourceTypeStringList =
List < String > . from ( map [ " resource-type " ] ? ? [ ] ) ;
2019-11-04 00:39:23 +00:00
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 " ] ? ? [ ] ) ,
2019-12-01 11:55:06 +00:00
unlessTopUrl: List < String > . from ( map [ " unless-top-url " ] ? ? [ ] ) ) ;
2019-11-04 00:39:23 +00:00
}
2019-10-26 20:11:23 +00:00
}
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-12-01 11:55:06 +00:00
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-12-01 11:55:06 +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-12-01 11:55:06 +00:00
Map < String , dynamic > map = { " type " : type . toValue ( ) , " selector " : selector } ;
2019-10-27 03:35:05 +00:00
map . keys
2019-12-01 11:55:06 +00:00
. where ( ( key ) = >
map [ key ] = = null | |
( map [ key ] is List & & ( map [ key ] as List ) . length = = 0 ) ) // filter keys
2019-10-27 03:35:05 +00:00
. 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-12-01 11:55:06 +00:00
selector: map [ " selector " ] ) ;
2019-11-04 00:39:23 +00:00
}
2019-12-01 11:55:06 +00:00
}