diff --git a/ElectronMainApp/config/default.json b/ElectronMainApp/config/default.json
index fc7f1b17..0a89a0d2 100644
--- a/ElectronMainApp/config/default.json
+++ b/ElectronMainApp/config/default.json
@@ -24,6 +24,16 @@
"LANGUAGE_SPECIFIC_ID": 7,
"CUSTOM_FILTERS_GROUP_ID": 0
},
+ "SafariExtensionBundles": {
+ "GENERAL": "com.adguard.safari.AdGuard.BlockerExtension",
+ "PRIVACY": "com.adguard.safari.AdGuard.BlockerPrivacy",
+ "SOCIAL_WIDGETS_AND_ANNOYANCES": "com.adguard.safari.AdGuard.BlockerSocial",
+ "SECURITY": "com.adguard.safari.AdGuard.BlockerSecurity",
+ "OTHER": "com.adguard.safari.AdGuard.BlockerOther",
+ "CUSTOM": "com.adguard.safari.AdGuard.BlockerCustom",
+ "ADVANCED_BLOCKING": "com.adguard.safari.AdGuard.AdvancedBlocking",
+ "ICON": "com.adguard.safari.AdGuard.Extension"
+ },
"backendUrl": "https://chrome.adtidy.org",
"backendApiKey": "4DDBE80A3DA94D819A00523252FB6380",
"filtersUrl": "https://filters.adtidy.org/extension/safari",
diff --git a/ElectronMainApp/locales/en.json b/ElectronMainApp/locales/en.json
index 9a24062c..868156b2 100644
--- a/ElectronMainApp/locales/en.json
+++ b/ElectronMainApp/locales/en.json
@@ -284,6 +284,9 @@
"options_about": {
"message": "About"
},
+ "options_content_blockers": {
+ "message": "Content blockers"
+ },
"options_compare_btn": {
"message": "How they compare"
},
@@ -303,7 +306,7 @@
"message": "Github"
},
"onboarding_safari_desc": {
- "message": "Check the box next to \"AdGuard Safari Icon\", \"AdGuard Advanced Blocking\" and \"AdGuard\" in Safari preferences"
+ "message": "Check the boxes next to AdGuard extensions in Safari preferences"
},
"onboarding_safari_btn": {
"message": "Open preferences"
@@ -344,16 +347,52 @@
"options_verbose_logging_desc": {
"message": "If Verbose logging is enabled, AdGuard Safari extensions will print additional information to the browser console"
},
- "settings_notification_title": {
- "message": "Attention! AdGuard extensions are disabled"
- },
- "settings_notification_text": {
- "message": "You can use these extensions to manage AdGuard for Safari. For instance, pause blocking on a website, or block an ad manually. You can enable AdGuard Icon and Advanced Blocking here<\/a>."
- },
"settings_notification_protection_title": {
"message": "Attention! Protection was disabled"
},
"settings_notification_protection_text": {
- "message": "AdGuard protection is paused. Enable AdGuard<\/a>."
+ "message": "AdGuard protection is paused."
+ },
+ "settings_notification_protection_button_text": {
+ "message": "Fix it"
+ },
+ "settings_notification_extensions_button_text": {
+ "message": "Fix it"
+ },
+ "settings_notification_extensions_title": {
+ "message": "Some AdGuard extensions are disabled!"
+ },
+ "settings_notification_extensions_text": {
+ "message": "You should enable all AdGuard extensions for correctly ad blocking in Safari"
+ },
+ "settings_notification_cb_extensions_title": {
+ "message": "Some AdGuard content blocker extensions are disabled!"
+ },
+ "settings_notification_cb_extensions_text": {
+ "message": "You should enable all AdGuard content blocker extensions for correctly ad blocking in Safari"
+ },
+ "options_cb_general_title": {
+ "message": "AdGuard General"
+ },
+ "options_cb_privacy_title": {
+ "message": "AdGuard Privacy"
+ },
+ "options_cb_social_title": {
+ "message": "AdGuard Social"
+ },
+ "options_cb_security_title": {
+ "message": "AdGuard Security"
+ },
+ "options_cb_other_title": {
+ "message": "AdGuard Other"
+ },
+ "options_cb_custom_title": {
+ "message": "AdGuard Custom"
+ },
+ "options_cb_rules_info": {
+ "message": "%s rules used"
+ },
+ "options_cb_disabled_warning": {
+ "message": "Disabled in Safari"
}
}
\ No newline at end of file
diff --git a/ElectronMainApp/safari-ext/index.js b/ElectronMainApp/safari-ext/index.js
index 96f5358a..a55f2d4d 100644
--- a/ElectronMainApp/safari-ext/index.js
+++ b/ElectronMainApp/safari-ext/index.js
@@ -9,6 +9,9 @@ const addon = require('bindings')('safari_ext_addon');
*/
module.exports = (() => {
+ const ADVANCED_BLOCKING_BUNDLE_ID = "com.adguard.safari.AdGuard.AdvancedBlocking";
+ const ICON_EXTENSION_BUNDLE_ID = "com.adguard.safari.AdGuard.Extension";
+
/**
* Initializes toolbar
*
@@ -134,30 +137,18 @@ module.exports = (() => {
};
/**
- * Getting state of the content blocker extension.
- * Returns true in callback if extension enabled, else returns false.
- * @param callback = (enabled as bool) => {}
- */
- const extensionContentBlockerState = (callback) => {
- addon.extensionContentBlockerState(callback);
- };
-
- /**
- * Getting state of the icon of Safari app extension.
- * Returns true in callback if extension enabled, else returns false.
- * @param callback = (enabled as bool) => {}
- */
- const extensionSafariIconState = (callback) => {
- addon.extensionSafariIconState(callback);
- };
-
- /**
- * Getting state of Advanced Blocking Safari extension.
- * Returns true in callback if extension enabled, else returns false.
- * @param callback = (enabled as bool) => {}
+ * Get safari extensions states info
+ * @param bundleId extension bundle identifier
+ * @param callback {*}
*/
- const extensionAdvancedBlockingState = (callback) => {
- addon.extensionAdvancedBlockingState(callback);
+ const getExtensionState = (bundleId, callback) => {
+ if (bundleId === ADVANCED_BLOCKING_BUNDLE_ID) {
+ addon.extensionAdvancedBlockingState(callback);
+ } else if (bundleId === ICON_EXTENSION_BUNDLE_ID) {
+ addon.extensionSafariIconState(callback);
+ } else {
+ addon.getExtensionContentBlockerState(bundleId, callback);
+ }
};
/**
@@ -192,9 +183,7 @@ module.exports = (() => {
whitelistDomains: whitelistDomains,
setUserFilter: setUserFilter,
userFilter: userFilter,
- extensionContentBlockerState: extensionContentBlockerState,
- extensionSafariIconState: extensionSafariIconState,
- extensionAdvancedBlockingState: extensionAdvancedBlockingState,
+ getExtensionState: getExtensionState,
openExtensionsPreferenses: openExtensionsPreferenses,
debugLog: debugLog,
setVerboseLogging: setVerboseLogging
diff --git a/ElectronMainApp/safari-ext/src.mm b/ElectronMainApp/safari-ext/src.mm
index 0ba9348e..e1d73e45 100644
--- a/ElectronMainApp/safari-ext/src.mm
+++ b/ElectronMainApp/safari-ext/src.mm
@@ -399,19 +399,22 @@ static void AsyncSendHandler(uv_async_t *handle) {
}];
}
-NAN_METHOD(extensionContentBlockerState){
+NAN_METHOD(getExtensionContentBlockerState){
- if (info.Length() < 1) {
- ThrowTypeError("Wrong number of arguments");
- return;
- }
+ if (info.Length() < 2) {
+ ThrowTypeError("Wrong number of arguments");
+ return;
+ }
- if (!info[0]->IsFunction()) {
- ThrowTypeError("Wrong arguments");
- return;
- }
+ if (!info[0]->IsString() || !info[1]->IsFunction()) {
+ ThrowTypeError("Wrong arguments");
+ return;
+ }
- Nan::Callback *cb = new Nan::Callback(info[0].As());
+ Nan::Utf8String message (info[0]);
+ NSString *bundleId = [NSString stringWithCString:*message encoding:NSUTF8StringEncoding];
+
+ Nan::Callback *cb = new Nan::Callback(info[1].As());
void (^resultBlock)(BOOL result) = ^void(BOOL result) {
@@ -425,7 +428,7 @@ static void AsyncSendHandler(uv_async_t *handle) {
});
};
- [SFContentBlockerManager getStateOfContentBlockerWithIdentifier:AESharedResources.blockerBundleId
+ [SFContentBlockerManager getStateOfContentBlockerWithIdentifier:bundleId
completionHandler:^(SFContentBlockerState * _Nullable state, NSError * _Nullable error) {
resultBlock(error == nil && state.enabled);
}];
@@ -732,8 +735,8 @@ static void AsyncSendHandler(uv_async_t *handle) {
Nan::Set(target, New("whitelistDomains").ToLocalChecked(),
GetFunction(New(whitelistDomains)).ToLocalChecked());
- Nan::Set(target, New("extensionContentBlockerState").ToLocalChecked(),
- GetFunction(New(extensionContentBlockerState)).ToLocalChecked());
+ Nan::Set(target, New("getExtensionContentBlockerState").ToLocalChecked(),
+ GetFunction(New(getExtensionContentBlockerState)).ToLocalChecked());
Nan::Set(target, New("extensionSafariIconState").ToLocalChecked(),
GetFunction(New(extensionSafariIconState)).ToLocalChecked());
diff --git a/ElectronMainApp/src/main/app/content-blocker/content-blocker-adapter.js b/ElectronMainApp/src/main/app/content-blocker/content-blocker-adapter.js
index e5b9e945..f5910bd7 100644
--- a/ElectronMainApp/src/main/app/content-blocker/content-blocker-adapter.js
+++ b/ElectronMainApp/src/main/app/content-blocker/content-blocker-adapter.js
@@ -1,3 +1,4 @@
+const config = require('config');
const listeners = require('../../notifier');
const events = require('../../events');
const settings = require('../settings-manager');
@@ -6,7 +7,7 @@ const {jsonFromFilters} = require('../libs/JSConverter');
const whitelist = require('../whitelist');
const log = require('../utils/log');
const concurrent = require('../utils/concurrent');
-const {groupRules} = require('./rule-groups');
+const {groupRules, rulesGroupsBundles} = require('./rule-groups');
/**
* Safari Content Blocker Adapter
@@ -29,19 +30,6 @@ module.exports = (function () {
}
];
- /**
- * Rules groups to extension bundle identifiers dictionary
- */
- const rulesGroupsBundles = {
- general: "com.adguard.safari.AdGuard.BlockerExtension",
- privacy: "com.adguard.safari.AdGuard.BlockerPrivacy",
- socialWidgetsAndAnnoyances: "com.adguard.safari.AdGuard.BlockerSocial",
- security: "com.adguard.safari.AdGuard.BlockerSecurity",
- other: "com.adguard.safari.AdGuard.BlockerOther",
- custom: "com.adguard.safari.AdGuard.BlockerCustom",
- advancedBlocking: "com.adguard.safari.AdGuard.AdvancedBlocking"
- };
-
/**
* Load content blocker
*/
@@ -64,6 +52,13 @@ module.exports = (function () {
}
setSafariContentBlocker(rulesGroupsBundles[group.key], json);
+
+ listeners.notifyListeners(events.CONTENT_BLOCKER_EXTENSION_UPDATED, {
+ rulesCount: group.rules.length,
+ bundleId: rulesGroupsBundles[group.key],
+ overlimit: result && result.overLimit,
+ filterGroups: group.filterGroups
+ });
}
const advancedBlocking = setAdvancedBlocking(rules.map(x => x.ruleText));
diff --git a/ElectronMainApp/src/main/app/content-blocker/rule-groups.js b/ElectronMainApp/src/main/app/content-blocker/rule-groups.js
index 444a87af..7caad8d0 100644
--- a/ElectronMainApp/src/main/app/content-blocker/rule-groups.js
+++ b/ElectronMainApp/src/main/app/content-blocker/rule-groups.js
@@ -15,6 +15,8 @@ module.exports = (function () {
const AntiBannerFilterGroupsId = config.get('AntiBannerFilterGroupsId');
const AntiBannerFiltersId = config.get('AntiBannerFiltersId');
+ const SafariExtensionBundles = config.get('SafariExtensionBundles');
+
/**
* Rules groups
*
@@ -62,6 +64,19 @@ module.exports = (function () {
all: [groups.general, groups.privacy, groups.security, groups.socialWidgetsAndAnnoyances, groups.other, groups.custom]
};
+ /**
+ * Rules groups to extension bundle identifiers dictionary
+ */
+ const rulesGroupsBundles = {
+ general: SafariExtensionBundles.GENERAL,
+ privacy: SafariExtensionBundles.PRIVACY,
+ socialWidgetsAndAnnoyances: SafariExtensionBundles.SOCIAL_WIDGETS_AND_ANNOYANCES,
+ security: SafariExtensionBundles.SECURITY,
+ other: SafariExtensionBundles.OTHER,
+ custom: SafariExtensionBundles.CUSTOM,
+ advancedBlocking: SafariExtensionBundles.ADVANCED_BLOCKING
+ };
+
/**
* Groups provided rules
*
@@ -103,6 +118,7 @@ module.exports = (function () {
const result = [];
for (let groupName in groups) {
const key = groups[groupName].key;
+ const filterGroups = groups[groupName].filterGroups;
if (rulesByAffinityBlocks[key]) {
rulesByGroup[key] = rulesByGroup[key].concat(rulesByAffinityBlocks[key]);
@@ -110,7 +126,8 @@ module.exports = (function () {
result.push({
key: key,
- rules: rulesByGroup[key]
+ rules: rulesByGroup[key],
+ filterGroups: filterGroups
});
log.info(`Affinity block ${key}: rules length: ${rulesByGroup[key].length}`);
@@ -180,9 +197,32 @@ module.exports = (function () {
return result;
};
+ let bundleGroups;
+ const filterGroupsBundles = () => {
+ if (bundleGroups) {
+ return bundleGroups;
+ }
+
+ const result = [];
+
+ for (let key in rulesGroupsBundles) {
+ if (groups[key]) {
+ result.push({
+ bundleId: rulesGroupsBundles[key],
+ groupIds: groups[key].filterGroups
+ });
+ }
+ }
+
+ bundleGroups = result;
+ return result;
+ };
+
return {
groupRules,
- groups
+ groups,
+ rulesGroupsBundles,
+ filterGroupsBundles
};
})();
diff --git a/ElectronMainApp/src/main/events.js b/ElectronMainApp/src/main/events.js
index 66b34158..8215c66c 100644
--- a/ElectronMainApp/src/main/events.js
+++ b/ElectronMainApp/src/main/events.js
@@ -27,6 +27,7 @@ const EventNotifierTypes = module.exports = {
NOTIFY_UPDATE_USER_FILTER_RULES: "event.notify.update.user.filter.rules",
UPDATE_WHITELIST_FILTER_RULES: "event.update.whitelist.filter.rules",
CONTENT_BLOCKER_UPDATED: "event.content.blocker.updated",
+ CONTENT_BLOCKER_EXTENSION_UPDATED: "event.content.blocker.extension.updated",
CONTENT_BLOCKER_UPDATE_REQUIRED: "event.content.blocker.update.required",
SHOW_OPTIONS_FILTERS_TAB: "event.show.options.filters",
SHOW_OPTIONS_USER_FILTER_TAB: "event.show.options.user.filter",
diff --git a/ElectronMainApp/src/main/startup.js b/ElectronMainApp/src/main/startup.js
index f5c0616a..74850957 100644
--- a/ElectronMainApp/src/main/startup.js
+++ b/ElectronMainApp/src/main/startup.js
@@ -6,6 +6,7 @@ const log = require('./app/utils/log');
const contentBlockerListener = require('./app/content-blocker/content-blocker-listener');
const notificationController = require('./notification-controller');
const safariToolbar = require('safari-ext');
+const toolbarController = require('./toolbar-controller');
/**
* Application startup
@@ -45,9 +46,9 @@ module.exports = (() => {
safariToolbar.sendReady();
// Check safari extensions
- safariToolbar.extensionContentBlockerState((result) => {
- if (!result) {
- log.warn('Safari extension content blocker is turned off!');
+ toolbarController.getExtensionsState((result) => {
+ if (!result || !result.contentBlockersEnabled) {
+ log.warn('Safari content blockers are turned off!');
showWindow();
}
diff --git a/ElectronMainApp/src/main/toolbar-controller.js b/ElectronMainApp/src/main/toolbar-controller.js
index 9f3b0319..e6be320f 100644
--- a/ElectronMainApp/src/main/toolbar-controller.js
+++ b/ElectronMainApp/src/main/toolbar-controller.js
@@ -1,3 +1,4 @@
+const config = require('config');
const safariToolbar = require('safari-ext');
const applicationApi = require('./api');
const listeners = require('./notifier');
@@ -13,6 +14,16 @@ const { shell } = require('electron');
*/
module.exports = (() => {
+ const SafariExtensionBundles = config.get('SafariExtensionBundles');
+ const ContentBlockerExtensions = [
+ SafariExtensionBundles.GENERAL,
+ SafariExtensionBundles.PRIVACY,
+ SafariExtensionBundles.SOCIAL_WIDGETS_AND_ANNOYANCES,
+ SafariExtensionBundles.SECURITY,
+ SafariExtensionBundles.OTHER,
+ SafariExtensionBundles.CUSTOM
+ ];
+
/**
* Protection status has been changed from toolbar
*
@@ -162,6 +173,55 @@ module.exports = (() => {
});
};
+ /**
+ * Get safari extensions states info
+ * @param callback {*}
+ */
+ const getExtensionsState = (callback) => {
+ const dfds = [];
+ const extensions = {};
+
+ for (let bundle in SafariExtensionBundles) {
+ const bundleId = SafariExtensionBundles[bundle];
+ dfds.push(new Promise((resolve) => {
+ safariToolbar.getExtensionState(bundleId, (info) => {
+ extensions[bundleId] = info;
+ resolve();
+ });
+ }));
+ }
+
+ Promise.all(dfds).then(function () {
+ let allContentBlockersDisabled = true;
+ let contentBlockersEnabled = true;
+ let minorExtensionsEnabled = true;
+
+ for (let extension in extensions) {
+ const extensionEnabled = extensions[extension];
+ if (!extensionEnabled) {
+ if (ContentBlockerExtensions.indexOf(extension) >= 0) {
+ contentBlockersEnabled = false;
+ } else {
+ minorExtensionsEnabled = false;
+ }
+ } else {
+ if (ContentBlockerExtensions.indexOf(extension) >= 0) {
+ allContentBlockersDisabled = false;
+ }
+ }
+ }
+
+ const result = {
+ extensions,
+ contentBlockersEnabled,
+ minorExtensionsEnabled,
+ allContentBlockersDisabled
+ };
+
+ callback(result);
+ });
+ };
+
/**
* Sets verbose logging
*
@@ -172,7 +232,8 @@ module.exports = (() => {
};
return {
- initToolbarController
+ initToolbarController,
+ getExtensionsState
};
})();
\ No newline at end of file
diff --git a/ElectronMainApp/src/main/ui-event-handler.js b/ElectronMainApp/src/main/ui-event-handler.js
index ff84de99..55be2d07 100644
--- a/ElectronMainApp/src/main/ui-event-handler.js
+++ b/ElectronMainApp/src/main/ui-event-handler.js
@@ -12,6 +12,8 @@ const safariToolbar = require('safari-ext');
const applicationApi = require('./api');
const updater = require('./updater');
const log = require('./app/utils/log');
+const toolbarController = require('./toolbar-controller');
+const {filterGroupsBundles} = require('./app/content-blocker/rule-groups');
/**
@@ -84,22 +86,13 @@ module.exports.init = function () {
filters.addAndEnableFilters([filter.filterId]);
});
break;
- case 'checkContentBlockerExtension':
- safariToolbar.extensionContentBlockerState((result) => {
- sendResponse(event, 'checkContentBlockerExtensionResponse', result);
+ case 'getSafariExtensionsState':
+ toolbarController.getExtensionsState((result) => {
+ sendResponse(event, 'getSafariExtensionsStateResponse', result);
});
break;
- case 'checkSafariExtensions':
- safariToolbar.extensionSafariIconState((result) => {
- if (!result) {
- sendResponse(event, 'checkSafariExtensionsResponse', result);
- return;
- }
-
- safariToolbar.extensionAdvancedBlockingState((result) => {
- sendResponse(event, 'checkSafariExtensionsResponse', result);
- });
- });
+ case 'getContentBlockersMetadata':
+ sendResponse(event, 'getContentBlockersMetadataResponse', filterGroupsBundles());
break;
case 'openSafariExtensionsPrefs':
safariToolbar.openExtensionsPreferenses(()=> {
diff --git a/ElectronMainApp/src/main/ui/css/style.css b/ElectronMainApp/src/main/ui/css/style.css
index c6393555..6b715946 100644
--- a/ElectronMainApp/src/main/ui/css/style.css
+++ b/ElectronMainApp/src/main/ui/css/style.css
@@ -6,6 +6,57 @@
margin-top: 140px;
height: 100%;
}
+.button {
+ cursor: pointer;
+ border: 0;
+ height: 40px;
+ border-radius: 4px;
+ font-size: 13px;
+ line-height: 42px;
+ transition: 0.3s ease background-color;
+ font-weight: 600;
+ outline: none;
+}
+.button--green {
+ padding: 0 30px;
+ background-color: #68bc86;
+ color: #ffffff;
+}
+.button--orange {
+ background-color: #eb9300;
+ color: #fff;
+}
+.button--green:hover {
+ background-color: #5ba575;
+}
+.button--transparent {
+ padding: 0 20px;
+ box-shadow: 0 0 0 1px rgba(216, 216, 216, .7);
+ background-color: transparent;
+}
+.button--transparent:hover {
+ background-color: rgba(0, 0, 0, .05);
+}
+.button--link {
+ padding: 0 40px;
+ color: #fff;
+ display: inline-block;
+ text-decoration: none;
+ min-width: 175px;
+ text-align: center;
+}
+.button--green-bd {
+ padding: 0 40px;
+ color: #66b574;
+ box-shadow: inset 0 0 0 1px #66b574;
+ display: inline-block;
+ text-decoration: none;
+ min-width: 175px;
+ transition: 0.3s ease background-color;
+}
+.button--green-bd:hover {
+ background-color: rgba(239, 239, 239, 0.5);
+}
/* st */
.tab-pane {
display: none;
@@ -345,28 +396,44 @@
padding-right: 70px;
opacity: 0.5;
}
+.opts-list--content-blocker .block-type {
+ cursor: default;
+}
.opts-list li.active .block-type {
opacity: 1;
}
.settings-notification {
margin-top: auto;
padding: 24px;
- background: rgba(245, 166, 35, 0.2);
+ background: #fff3df;
border-radius: 8px 8px 8px 8px;
margin-bottom: 15px;
}
+.settings-notification--cols {
+ display: none;
+ justify-content: space-between;
+ align-items: center;
+}
.settings-notification__title {
font-size: 16px;
- color: #4d4d4d;
width: 470px;
- line-height: 17px;
- margin-bottom: 8px;
+ line-height: 20px;
+ margin-bottom: 4px;
+ color: #4d4d4d;
}
.settings-notification__text {
color: #888888;
- font-size: 14px;
+ line-height: 14px;
+ font-size: 12px;
max-width: 553px;
- line-height: 17px;
+}
+.settings-notification__btn {
+ font-size: 16px;
+ min-width: 200px;
+ text-transform: uppercase;
+ height: 48px;
+ border-radius: 8px;
+ outline: none;
}
.block-type__desc {
display: flex;
@@ -416,6 +483,27 @@
.block-type__ico--0 {
background-image: url();
}
+.block-type__ico-info {
+ width: 32px;
+ height: 32px;
+ margin-right: 25px;
+ background-repeat: no-repeat;
+ display: inline-block;
+ flex-shrink: 0;
+ background-position: 50%;
+}
+.block-type__ico-info--warning {
+ background-image: url();
+}
+
+.block-type__ico-info--check {
+ background-image: url();
+}
+
+.block-type__ico-info--load {
+ background-image: url();
+ animation: clockwise 2s linear infinite;
+}
.block-type__desc-title {
color: #66b574;
@@ -424,6 +512,24 @@
.block-type:hover .block-type__desc-title {
text-decoration: underline;
}
+.block-type__info-title {
+ font-size: 16px;
+ line-height: 20px;
+ color: #4d4d4d;
+ margin-bottom: 4px;
+}
+.block-type__info-desc {
+ font-size: 12px;
+ line-height: 14px;
+ color: #888;
+ margin-bottom: 10px;
+}
+.cb_disabled_warning {
+ color: #9C1D00;
+}
+.block-type__info-desc:last-child {
+ margin-bottom: 0;
+}
.desc--filters {
color: rgba(77, 77, 77, 0.5);
}
@@ -688,58 +794,13 @@ li.active .toggler:before {
background-size: contain;
background-image: url(../images/arrow-down.svg);
}
-.button {
- cursor: pointer;
- border: 0;
- height: 40px;
- border-radius: 4px;
- font-size: 13px;
- line-height: 42px;
- transition: 0.3s ease background-color;
- font-weight: 600;
-}
-.button--green {
- padding: 0 30px;
- background-color: #68bc86;
- color: #ffffff;
-}
-.button--green:hover {
- background-color: #5ba575;
-}
-.button--transparent {
- padding: 0 20px;
- box-shadow: 0 0 0 1px rgba(216, 216, 216, .7);
- background-color: transparent;
-}
-.button--transparent:hover {
- background-color: rgba(0, 0, 0, .05);
-}
-.button--link {
- padding: 0 40px;
- color: #fff;
- display: inline-block;
- text-decoration: none;
- min-width: 175px;
- text-align: center;
-}
+
.wlimit .settings-actions .button--link {
padding: 0 35px;
}
.settings-actions .button--link {
color: #fff;
}
-.button--green-bd {
- padding: 0 40px;
- color: #66b574;
- box-shadow: inset 0 0 0 1px #66b574;
- display: inline-block;
- text-decoration: none;
- min-width: 175px;
- transition: 0.3s ease background-color;
-}
-.button--green-bd:hover {
- background-color: rgba(239, 239, 239, 0.5);
-}
.settings-actions .button--green-bd {
color: #66b574;
margin-left: 20px;
@@ -1467,6 +1528,10 @@ li.active .toggler:before {
display: none;
}
+body {
+ overflow: auto;
+}
+
.onboarding {
position: absolute;
display: none;
diff --git a/ElectronMainApp/src/main/ui/js/options.js b/ElectronMainApp/src/main/ui/js/options.js
index b0f4bcd8..6fe3f25a 100644
--- a/ElectronMainApp/src/main/ui/js/options.js
+++ b/ElectronMainApp/src/main/ui/js/options.js
@@ -956,6 +956,10 @@ const AntiBannerFilters = function (options) {
if (hash && hash.indexOf('#antibanner') === 0) {
TopMenu.toggleTab();
}
+
+ ipcRenderer.send('renderer-to-main', JSON.stringify({
+ 'type': 'getContentBlockersMetadata'
+ }));
});
ipcRenderer.send('renderer-to-main', JSON.stringify({
@@ -1305,13 +1309,39 @@ const AntiBannerFilters = function (options) {
}
}
+ /**
+ * Creates filters info string
+ *
+ * @param groupIds [] array of enabled groups
+ */
+ const getFiltersInfo = (groupIds) => {
+ if (!groupIds || groupIds.length === 0) {
+ return null;
+ }
+
+ if (loadedFiltersInfo.filters.length === 0) {
+ return null;
+ }
+
+ let filters = [];
+ for (let groupId of groupIds) {
+ if (loadedFiltersInfo.isCategoryEnabled(groupId)) {
+ const groupFilters = getFiltersByGroupId(groupId, loadedFiltersInfo.filters);
+ filters = filters.concat(groupFilters);
+ }
+ }
+
+ return generateFiltersNamesDescription(filters);
+ };
+
return {
render: renderCategoriesAndFilters,
updateRulesCountInfo: updateRulesCountInfo,
onFilterStateChanged: onFilterStateChanged,
onCategoryStateChanged: onCategoryStateChanged,
onFilterDownloadStarted: onFilterDownloadStarted,
- onFilterDownloadFinished: onFilterDownloadFinished
+ onFilterDownloadFinished: onFilterDownloadFinished,
+ getFiltersInfo: getFiltersInfo
};
};
@@ -1475,7 +1505,7 @@ const Settings = function () {
if (protectionEnabled) {
enableProtectionNotification.style.display = 'none';
} else {
- enableProtectionNotification.style.display = 'block';
+ enableProtectionNotification.style.display = 'flex';
}
};
@@ -1511,6 +1541,127 @@ const Settings = function () {
};
};
+/**
+ * Content blockers tab
+ *
+ * @returns {*}
+ * @constructor
+ */
+const ContentBlockersScreen = function (antiBannerFilters) {
+ 'use strict';
+
+ /**
+ * Elements to extension bundles dictionary
+ *
+ * @type {{*}}
+ */
+ const extensionElements = {
+ "com.adguard.safari.AdGuard.BlockerExtension": "cb_general",
+ "com.adguard.safari.AdGuard.BlockerPrivacy": "cb_privacy",
+ "com.adguard.safari.AdGuard.BlockerSocial": "cb_social",
+ "com.adguard.safari.AdGuard.BlockerSecurity": "cb_security",
+ "com.adguard.safari.AdGuard.BlockerOther": "cb_other",
+ "com.adguard.safari.AdGuard.BlockerCustom": "cb_custom"
+ };
+
+ /**
+ * Extension element for bundle identifier
+ *
+ * @param bundleId
+ * @return {*}
+ */
+ const getExtensionElement = (bundleId) => {
+ const elementId = extensionElements[bundleId];
+ if (elementId) {
+ return document.getElementById(elementId);
+ }
+
+ return null;
+ };
+
+ /**
+ * Updates content blockers info
+ *
+ * @param info
+ */
+ const updateContentBlockers = (info) => {
+ for (let extensionId in info.extensions) {
+ const state = info.extensions[extensionId];
+
+ const element = getExtensionElement(extensionId);
+ if (element) {
+ const icon = element.querySelector('.extension-block-ico');
+ const warning = element.querySelector('.cb_disabled_warning');
+ const rulesCount = element.querySelector('.cb_rules_count');
+
+ icon.classList.remove("block-type__ico-info--load");
+
+ icon.classList.add(state ? "block-type__ico-info--check" : "block-type__ico-info--warning");
+ warning.style.display = state ? 'none' : 'flex';
+
+ rulesCount.style.display = state ? 'flex' : 'none';
+ }
+ }
+ };
+
+ /**
+ * Updates extension rules count
+ *
+ * @param bundleId
+ * @param info
+ * @param filtersInfo
+ */
+ const updateExtensionState = (bundleId, info, filtersInfo) => {
+ const element = getExtensionElement(bundleId);
+ if (element) {
+ if (info) {
+ const rulesInfoElement = element.querySelector('.cb_rules_count');
+ rulesInfoElement.textContent = i18n.__("options_cb_rules_info.message", info.rulesCount);
+
+ //TODO: We might show rules overlimit here
+ }
+
+ if (filtersInfo) {
+ const filtersInfoElement = element.querySelector('.cb_filters_info');
+ filtersInfoElement.textContent = filtersInfo;
+ }
+ }
+ };
+
+ /**
+ * Sets loading state for extensions
+ */
+ const setLoading = () => {
+ const extensionsIcons = document.querySelectorAll('.extension-block-ico');
+ extensionsIcons.forEach((ext) => {
+ ext.classList.remove("block-type__ico-info--warning");
+ ext.classList.remove("block-type__ico-info--check");
+
+ ext.classList.add("block-type__ico-info--load");
+ });
+ };
+
+ /**
+ * Initialize
+ *
+ */
+ const init = () => {
+ ipcRenderer.on('getContentBlockersMetadataResponse', (e, response) => {
+ for (let extension of response) {
+ const filtersInfo = antiBannerFilters.getFiltersInfo(extension.groupIds);
+ updateExtensionState(extension.bundleId, null, filtersInfo);
+ }
+ });
+ };
+
+ return {
+ updateContentBlockers,
+ setLoading,
+ updateExtensionState,
+ init
+ };
+};
+
/**
* Page controller
*
@@ -1542,7 +1693,6 @@ PageController.prototype = {
this.aboutUpdatesRelaunch = document.getElementById('about-updates-relaunch');
this._initBoardingScreen();
- this._initSafariExtensionsMessage();
this._initUpdatesBlock();
},
@@ -1571,46 +1721,41 @@ PageController.prototype = {
_initBoardingScreen: function () {
let body = document.querySelector('body');
let onBoardingScreenEl = body.querySelector('#boarding-screen-placeholder');
- ipcRenderer.on('checkContentBlockerExtensionResponse', (e, arg) => {
- body.style.overflow = arg ? 'auto' : 'hidden';
- onBoardingScreenEl.style.display = arg ? 'none' : 'flex';
- });
+ const enableExtensionsNotification = document.getElementById('enableExtensionsNotification');
+ const enableCbExtensionsNotification = document.getElementById('enableCbExtensionsNotification');
- let openSafariSettingsButton = document.querySelector('#open-safari-extensions-settings-btn');
- openSafariSettingsButton && openSafariSettingsButton.addEventListener('click', (e) => {
- e.preventDefault();
- this._openSafariExtensionsPrefs();
- });
+ let self = this;
+ ipcRenderer.on('getSafariExtensionsStateResponse', (e, arg) => {
+ const allContentBlockersDisabled = arg.allContentBlockersDisabled;
+ const contentBlockersEnabled = arg.contentBlockersEnabled;
+ const minorExtensionsEnabled = arg.minorExtensionsEnabled;
- this._checkContentBlockerExtension();
- window.addEventListener("focus", () => this._checkContentBlockerExtension());
- },
+ body.style.overflow = !allContentBlockersDisabled ? 'auto' : 'hidden';
+ onBoardingScreenEl.style.display = !allContentBlockersDisabled ? 'none' : 'flex';
- _initSafariExtensionsMessage: function() {
- const enableExtensionsNotification = document.getElementById('enableExtensionsNotification');
- ipcRenderer.on('checkSafariExtensionsResponse', (e, arg) => {
- enableExtensionsNotification.style.display = arg ? 'none' : 'block';
+ enableExtensionsNotification.style.display = (contentBlockersEnabled && minorExtensionsEnabled) ? 'none' : 'flex';
+ enableCbExtensionsNotification.style.display = contentBlockersEnabled ? 'none' : 'flex';
+
+ self.contentBlockers.updateContentBlockers(arg);
});
- this._checkSafariExtensions();
- window.addEventListener("focus", () => this._checkSafariExtensions());
+ const openSafariSettingsButtons = document.querySelectorAll('.open-safari-extensions-settings-btn');
+ openSafariSettingsButtons.forEach((but) => {
+ but.addEventListener('click', (e) => {
+ e.preventDefault();
+ this._openSafariExtensionsPrefs();
+ });
+ });
- const notificationEnableExtensionsLink = document.getElementById('notificationEnableIconLink');
- notificationEnableExtensionsLink && notificationEnableExtensionsLink.addEventListener('click', (e) => {
- e.preventDefault();
- this._openSafariExtensionsPrefs();
- })
+ this.checkSafariExtensions();
+ window.addEventListener("focus", () => this.checkSafariExtensions());
},
- _checkContentBlockerExtension: function () {
- ipcRenderer.send('renderer-to-main', JSON.stringify({
- 'type': 'checkContentBlockerExtension'
- }));
- },
+ checkSafariExtensions: function() {
+ this.contentBlockers.setLoading();
- _checkSafariExtensions: function() {
ipcRenderer.send('renderer-to-main', JSON.stringify({
- 'type': 'checkSafariExtensions'
+ 'type': 'getSafariExtensionsState'
}));
},
@@ -1659,6 +1804,10 @@ PageController.prototype = {
this.antiBannerFilters = new AntiBannerFilters({ rulesInfo: contentBlockerInfo });
this.antiBannerFilters.render();
+ // Initialize Content blockers
+ this.contentBlockers = new ContentBlockersScreen(this.antiBannerFilters);
+ this.contentBlockers.init();
+
document.querySelector('#about-version-placeholder').textContent = i18n.__("options_about_version.message", environmentOptions.appVersion);
},
@@ -1726,12 +1875,14 @@ const initPage = function (response) {
case EventNotifierTypes.FILTER_ENABLE_DISABLE:
controller.antiBannerFilters.onFilterStateChanged(options);
controller.settings.updateAcceptableAdsCheckbox(options);
+ controller.contentBlockers.setLoading();
break;
case EventNotifierTypes.FILTER_ADD_REMOVE:
controller.antiBannerFilters.render();
break;
case EventNotifierTypes.FILTER_GROUP_ENABLE_DISABLE:
controller.antiBannerFilters.onCategoryStateChanged(options);
+ controller.contentBlockers.setLoading();
break;
case EventNotifierTypes.START_DOWNLOAD_FILTER:
controller.antiBannerFilters.onFilterDownloadStarted(options);
@@ -1742,12 +1893,19 @@ const initPage = function (response) {
break;
case EventNotifierTypes.UPDATE_USER_FILTER_RULES:
controller.userFilter.updateUserFilterRules();
+ controller.contentBlockers.setLoading();
break;
case EventNotifierTypes.UPDATE_WHITELIST_FILTER_RULES:
controller.whiteListFilter.updateWhiteListDomains();
+ controller.contentBlockers.setLoading();
break;
case EventNotifierTypes.CONTENT_BLOCKER_UPDATED:
controller.antiBannerFilters.updateRulesCountInfo(options);
+ controller.checkSafariExtensions();
+ break;
+ case EventNotifierTypes.CONTENT_BLOCKER_EXTENSION_UPDATED:
+ const filtersInfo = controller.antiBannerFilters.getFiltersInfo(options.filterGroups);
+ controller.contentBlockers.updateExtensionState(options.bundleId, options, filtersInfo);
break;
case EventNotifierTypes.SHOW_OPTIONS_FILTERS_TAB:
window.location.hash = 'antibanner';
diff --git a/ElectronMainApp/src/main/ui/options.html b/ElectronMainApp/src/main/ui/options.html
index 15f15a7c..5f66ade7 100644
--- a/ElectronMainApp/src/main/ui/options.html
+++ b/ElectronMainApp/src/main/ui/options.html
@@ -44,6 +44,11 @@
+
+
+
+
+
@@ -57,11 +62,35 @@
+
+
-
@@ -328,6 +349,115 @@
+
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+
+
+