/* global e */ /* global hide */ /* global show */ /* global sendMessage */ /* global getOption_ */ /* global setOption_ */ "use strict"; /** * Handles rule (de)activation in the popup */ function toggleRuleLine(event) { if (event.target.matches("input[type=checkbox]")) { getTab(activeTab => { const set_ruleset = { active: event.target.checked, name: event.target.nextSibling.innerText, tab_id: activeTab.id, }; sendMessage("set_ruleset_active_status", set_ruleset, () => { // purge the name from the cache so that this unchecking is persistent. sendMessage("delete_from_ruleset_cache", set_ruleset.name, () => { // Now reload the selected tab of the current window. chrome.tabs.reload(set_ruleset.tab_id); }); }); }); } } /** * Creates rule lines (including checkbox and icon) for the popup * @param rulesets * @param list_div * @returns {*} */ function appendRulesToListDiv(rulesets, list_div) { if (rulesets && rulesets.length) { // template parent block for each ruleset let templateLine = document.createElement("div"); templateLine.className = "rule checkbox"; // label "container" let templateLabel = document.createElement("label"); // checkbox let templateCheckbox = document.createElement("input"); templateCheckbox.type = "checkbox"; // label text let templateLabelText = document.createElement("span"); // img "remove" button let templateRemove = document.createElement("img"); templateRemove.src = chrome.extension.getURL("images/remove.png"); templateRemove.className = "remove"; templateLabel.appendChild(templateCheckbox); templateLabel.appendChild(templateLabelText); templateLine.appendChild(templateLabel); for (const ruleset of rulesets) { let line = templateLine.cloneNode(true); let checkbox = line.querySelector("input[type=checkbox]"); let text = line.querySelector("span"); checkbox.checked = ruleset.active; text.innerText = ruleset.name; if (ruleset.note && ruleset.note.length) { line.title = ruleset.note; if (ruleset.note === "user rule") { let remove = templateRemove.cloneNode(true); line.appendChild(remove); remove.addEventListener("click", () => { sendMessage("remove_rule", { ruleset, src: 'popup' }, () => { list_div.removeChild(line); }); }); } } list_div.appendChild(line); } show(list_div); } } function showHttpNowhereUI() { // Set up checkbox for HTTP nowhere mode getOption_('httpNowhere', false, function(item) { if (item.httpNowhere) { e('http-nowhere-checkbox').checked = true; } e('HttpNowhere').style.visibility = "visible"; }); }; // Change the UI to reflect extension enabled/disabled function updateEnabledDisabledUI() { getOption_('globalEnabled', true, function(item) { e('onoffswitch').checked = item.globalEnabled; e('disableButton').style.visibility = "visible"; // Hide or show the rules sections if (item.globalEnabled) { document.body.className = "" showHttpNowhereUI() } else { document.body.className = "disabled" } }); } // Toggle extension enabled/disabled status function toggleEnabledDisabled() { var extension_toggle_effect = function() { updateEnabledDisabledUI(); // The extension state changed, so reload this tab. chrome.tabs.reload(); window.close(); } getOption_('globalEnabled', true, function(item) { setOption_('globalEnabled', !item.globalEnabled, extension_toggle_effect); }); } /** * Create the list of rules for a specific tab * @param activeTab */ function listRules(activeTab) { sendMessage("get_active_rulesets", activeTab.id, function(rulesets) { if (rulesets) { const stableRules = rulesets.filter(ruleset => ruleset.default_state); const unstableRules = rulesets.filter(ruleset => !ruleset.default_state); appendRulesToListDiv(stableRules, e("StableRules")); appendRulesToListDiv(unstableRules, e("UnstableRules")); // Add listener to capture the toggle event e("RuleManagement").addEventListener("click", toggleRuleLine); } // Only show the "Add a rule" section if we're on an HTTPS page if (/^https:/.test(activeTab.url)) { show(e("addRuleSection")); } }); } /** * Fill in content into the popup on load */ document.addEventListener("DOMContentLoaded", function () { getTab(tab => { const url = new URL(tab.url); sendMessage("check_if_site_disabled", url.host, disabled => { if(!disabled){ listRules(tab); } showEnableOrDisable(url, disabled); }); }); // Set up the enabled/disabled switch & hide/show rules updateEnabledDisabledUI(); e('onoffswitch').addEventListener('click', toggleEnabledDisabled); e('http-nowhere-checkbox').addEventListener('click', toggleHttpNowhere, false); e('reset-to-defaults').addEventListener('click', () => { if (confirm(chrome.i18n.getMessage("prefs_reset_defaults_message"))) { sendMessage("reset_to_defaults", null, () => { window.close(); }); } }); // Print the extension's current version. var the_manifest = chrome.runtime.getManifest(); var version_info = e('current-version'); version_info.innerText = the_manifest.version; let rulesets_versions = e('rulesets-versions'); sendMessage("get_ruleset_timestamps", null, timestamps => { for(let [update_channel, timestamp] of timestamps){ if(timestamp > 0){ let ruleset_date = new Date(timestamp * 1000); let ruleset_version_string = ruleset_date.getUTCFullYear() + "." + (ruleset_date.getUTCMonth() + 1) + "." + ruleset_date.getUTCDate(); let timestamp_span = document.createElement("span"); timestamp_span.className = "rulesets-version"; timestamp_span.innerText = chrome.i18n.getMessage("about_rulesets_version") + " " + update_channel + ": " + ruleset_version_string; rulesets_versions.appendChild(timestamp_span); } } }); e("aboutTitle").title = chrome.i18n.getMessage("about_title"); e("add-rule-link").addEventListener("click", addManualRule); e("disable-on-this-site").addEventListener("click", disableOnSite); e("enable-on-this-site").addEventListener("click", enableOnSite); }); var escapeForRegex = function( value ) { return value.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&"); }; function showEnableOrDisable(url, disabled) { if (["http:", "https:", "ftp:"].indexOf(url.protocol) != -1) { const disableLink = e("disable-on-this-site"); const enableLink = e("enable-on-this-site"); const addRuleSection = e("addRuleSection"); const resetToDefaults = e('reset-to-defaults'); if (disabled) { show(enableLink); hide(disableLink); hide(addRuleSection); hide(resetToDefaults); } else { show(disableLink); hide(enableLink); } } else { const disableEnableSection = e("disableEnableSection"); hide(disableEnableSection); } } /** * Handles the manual addition of rules */ function addManualRule() { getTab(function(tab) { hide(e("add-rule-link")); show(e("add-new-rule-div")); const url = new URL(tab.url); e("new-rule-host").value = url.host; const escapedHost = escapeForRegex(url.host); e("new-rule-regex").value = `^http://${escapedHost}/`; e("new-rule-redirect").value = `https://${url.host}/`; e("new-rule-name").value = "Manual rule for " + url.host; e("add-new-rule-button").addEventListener("click", function() { const params = { /** * FIXME: the current implementation forbide users setting custom * ruleset names... */ name: e("new-rule-host").value, target : [e("new-rule-host").value], rule: [{ to: e("new-rule-redirect").value, from: e("new-rule-regex").value }], default_off: "user rule" }; sendMessage("add_new_rule", params, function() { location.reload(); }); }); e("cancel-new-rule").addEventListener("click", function() { show(e("add-rule-link")); hide(e("add-new-rule-div")); }); e("new-rule-show-advanced-link").addEventListener("click", function() { show(e("new-rule-advanced")); hide(e("new-rule-regular-text")); }); e("new-rule-hide-advanced-link").addEventListener("click", function() { hide(e("new-rule-advanced")); show(e("new-rule-regular-text")); }); }); } /** * Disable HTTPS Everywhere on a particular FQDN */ function disableOnSite() { getTab(function(tab) { const url = new URL(tab.url); sendMessage("disable_on_site", url.host); chrome.tabs.reload(tab.id); window.close(); }); } function enableOnSite() { getTab(function(tab) { const url = new URL(tab.url); sendMessage("enable_on_site", url.host); chrome.tabs.reload(tab.id); window.close(); }); } function toggleHttpNowhere() { getOption_('httpNowhere', false, function(item) { setOption_('httpNowhere', !item.httpNowhere); }); } function getTab(callback) { let url = new URL(window.location.href); if (url.searchParams.has('tabId')) { let parentId = Number(url.searchParams.get('tabId')); return chrome.tabs.get(parentId, callback); } chrome.tabs.query({active: true, lastFocusedWindow: true}, tabs => callback(tabs[0])); }