2014-06-23 16:42:43 -06:00
|
|
|
/*******************************************************************************
|
|
|
|
|
|
|
|
µBlock - a Chromium browser extension to block requests.
|
|
|
|
Copyright (C) 2014 Raymond Hill
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see {http://www.gnu.org/licenses/}.
|
|
|
|
|
|
|
|
Home: https://github.com/gorhill/uBlock
|
|
|
|
*/
|
|
|
|
|
2014-10-19 05:11:27 -06:00
|
|
|
/* global vAPI */
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2014-06-23 16:42:43 -06:00
|
|
|
|
|
|
|
// Injected into content pages
|
|
|
|
|
2014-10-19 05:11:27 -06:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2014-10-17 13:44:19 -06:00
|
|
|
// because Safari
|
|
|
|
if (location.protocol !== "http:" && location.protocol !== "https:") {
|
|
|
|
throw "uBlock> contentscript-end.js > Skipping page... " + location.protocol + location.host;
|
|
|
|
}
|
2014-06-23 16:42:43 -06:00
|
|
|
|
2014-10-19 05:11:27 -06:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2014-10-17 13:44:19 -06:00
|
|
|
var messager = vAPI.messaging.channel('contentscript-end.js');
|
2014-06-23 16:42:43 -06:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
// ABP cosmetic filters
|
|
|
|
|
2014-07-30 06:05:00 -06:00
|
|
|
(function() {
|
2014-07-02 10:02:29 -06:00
|
|
|
var queriedSelectors = {};
|
|
|
|
var injectedSelectors = {};
|
|
|
|
var classSelectors = null;
|
|
|
|
var idSelectors = null;
|
2014-08-12 10:19:54 -06:00
|
|
|
var highGenerics = null;
|
|
|
|
var contextNodes = [document];
|
2014-08-14 11:59:37 -06:00
|
|
|
var nullArray = { push: function(){} };
|
2014-07-02 10:02:29 -06:00
|
|
|
|
|
|
|
var domLoaded = function() {
|
2014-08-14 11:59:37 -06:00
|
|
|
var style = document.getElementById('ublock-preload-1ae7a5f130fc79b4fdb8a4272d9426b5');
|
2014-08-13 18:03:55 -06:00
|
|
|
if ( style ) {
|
|
|
|
// https://github.com/gorhill/uBlock/issues/14
|
|
|
|
// Treat any existing domain-specific exception selectors as if
|
|
|
|
// they had been injected already.
|
|
|
|
var selectors, i;
|
2014-08-14 11:59:37 -06:00
|
|
|
var exceptions = style.getAttribute('data-ublock-exceptions');
|
2014-08-13 18:03:55 -06:00
|
|
|
if ( exceptions ) {
|
|
|
|
selectors = JSON.parse(exceptions);
|
|
|
|
i = selectors.length;
|
|
|
|
while ( i-- ) {
|
|
|
|
injectedSelectors[selectors[i]] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Avoid re-injecting already injected CSS rules.
|
|
|
|
selectors = selectorsFromStyles(style);
|
|
|
|
i = selectors.length;
|
2014-07-02 10:02:29 -06:00
|
|
|
while ( i-- ) {
|
2014-08-13 18:03:55 -06:00
|
|
|
injectedSelectors[selectors[i]] = true;
|
2014-07-02 10:02:29 -06:00
|
|
|
}
|
2014-08-17 14:07:42 -06:00
|
|
|
// https://github.com/gorhill/uBlock/issues/158
|
|
|
|
// Ensure injected styles are enforced
|
|
|
|
hideElements(selectors.join(','));
|
2014-06-25 16:44:35 -06:00
|
|
|
}
|
2014-07-20 13:00:26 -06:00
|
|
|
idsFromNodeList(document.querySelectorAll('[id]'));
|
|
|
|
classesFromNodeList(document.querySelectorAll('[class]'));
|
2014-07-02 10:02:29 -06:00
|
|
|
retrieveGenericSelectors();
|
|
|
|
};
|
2014-06-23 16:42:43 -06:00
|
|
|
|
2014-08-13 18:03:55 -06:00
|
|
|
var selectorsFromStyles = function(styleRef) {
|
|
|
|
var selectors = [];
|
|
|
|
var styles = typeof styleRef === 'string' ?
|
|
|
|
document.querySelectorAll(styleRef):
|
|
|
|
[styleRef];
|
|
|
|
var i = styles.length;
|
|
|
|
var style, subset, lastSelector, pos;
|
|
|
|
while ( i-- ) {
|
|
|
|
style = styles[i];
|
|
|
|
subset = style.textContent.split(',\n');
|
|
|
|
lastSelector = subset.pop();
|
|
|
|
if ( lastSelector ) {
|
|
|
|
pos = lastSelector.indexOf('\n');
|
|
|
|
if ( pos !== -1 ) {
|
|
|
|
subset.push(lastSelector.slice(0, pos));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
selectors = selectors.concat(subset);
|
|
|
|
}
|
|
|
|
return selectors;
|
|
|
|
};
|
|
|
|
|
2014-07-02 10:02:29 -06:00
|
|
|
var retrieveGenericSelectors = function() {
|
|
|
|
var selectors = classSelectors !== null ? Object.keys(classSelectors) : [];
|
|
|
|
if ( idSelectors !== null ) {
|
|
|
|
selectors = selectors.concat(idSelectors);
|
2014-06-23 16:42:43 -06:00
|
|
|
}
|
2014-08-12 10:19:54 -06:00
|
|
|
if ( selectors.length > 0 || highGenerics === null ) {
|
2014-07-02 10:02:29 -06:00
|
|
|
//console.log('µBlock> ABP cosmetic filters: retrieving CSS rules using %d selectors', selectors.length);
|
2014-10-17 13:44:19 -06:00
|
|
|
messager.send({
|
2014-07-02 10:02:29 -06:00
|
|
|
what: 'retrieveGenericCosmeticSelectors',
|
|
|
|
pageURL: window.location.href,
|
2014-08-12 10:19:54 -06:00
|
|
|
selectors: selectors,
|
|
|
|
highGenerics: highGenerics === null
|
2014-07-02 10:02:29 -06:00
|
|
|
},
|
|
|
|
retrieveHandler
|
|
|
|
);
|
2014-08-12 10:19:54 -06:00
|
|
|
} else {
|
|
|
|
retrieveHandler(null);
|
2014-07-02 10:02:29 -06:00
|
|
|
}
|
|
|
|
idSelectors = null;
|
|
|
|
classSelectors = null;
|
|
|
|
};
|
2014-06-23 16:42:43 -06:00
|
|
|
|
2014-07-02 10:02:29 -06:00
|
|
|
var retrieveHandler = function(selectors) {
|
2014-08-12 10:19:54 -06:00
|
|
|
//console.debug('µBlock> contextNodes = %o', contextNodes);
|
|
|
|
if ( selectors && selectors.highGenerics ) {
|
|
|
|
highGenerics = selectors.highGenerics;
|
2014-06-23 16:42:43 -06:00
|
|
|
}
|
2014-08-12 10:19:54 -06:00
|
|
|
if ( selectors && selectors.donthide.length ) {
|
2014-08-14 11:59:37 -06:00
|
|
|
processLowGenerics(selectors.donthide, nullArray);
|
2014-08-12 10:19:54 -06:00
|
|
|
}
|
|
|
|
if ( highGenerics ) {
|
|
|
|
if ( highGenerics.donthideLowCount ) {
|
2014-08-14 11:59:37 -06:00
|
|
|
processHighLowGenerics(highGenerics.donthideLow, nullArray);
|
2014-08-12 10:19:54 -06:00
|
|
|
}
|
|
|
|
if ( highGenerics.donthideMediumCount ) {
|
2014-08-14 11:59:37 -06:00
|
|
|
processHighMediumGenerics(highGenerics.donthideMedium, nullArray);
|
2014-08-07 14:12:15 -06:00
|
|
|
}
|
|
|
|
}
|
2014-08-14 11:59:37 -06:00
|
|
|
// No such thing as high-high generic exceptions.
|
2014-08-12 10:19:54 -06:00
|
|
|
//if ( highGenerics.donthideHighCount ) {
|
2014-08-14 11:59:37 -06:00
|
|
|
// processHighHighGenerics(document, highGenerics.donthideHigh, nullArray);
|
2014-08-12 10:19:54 -06:00
|
|
|
//}
|
|
|
|
var hideSelectors = [];
|
|
|
|
if ( selectors && selectors.hide.length ) {
|
|
|
|
processLowGenerics(selectors.hide, hideSelectors);
|
|
|
|
}
|
|
|
|
if ( highGenerics ) {
|
|
|
|
if ( highGenerics.hideLowCount ) {
|
|
|
|
processHighLowGenerics(highGenerics.hideLow, hideSelectors);
|
|
|
|
}
|
|
|
|
if ( highGenerics.hideMediumCount ) {
|
|
|
|
processHighMediumGenerics(highGenerics.hideMedium, hideSelectors);
|
|
|
|
}
|
|
|
|
if ( highGenerics.hideHighCount ) {
|
2014-09-16 17:16:18 -06:00
|
|
|
processHighHighGenericsAsync();
|
2014-08-12 10:19:54 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( hideSelectors.length ) {
|
2014-09-16 17:16:18 -06:00
|
|
|
addStyleTag(hideSelectors);
|
2014-06-23 16:42:43 -06:00
|
|
|
}
|
2014-08-12 10:19:54 -06:00
|
|
|
contextNodes.length = 0;
|
2014-07-02 10:02:29 -06:00
|
|
|
};
|
2014-06-23 16:42:43 -06:00
|
|
|
|
2014-10-17 13:44:19 -06:00
|
|
|
// Ensure elements matching a set of selectors are visually removed
|
2014-09-16 17:16:18 -06:00
|
|
|
// from the page, by:
|
|
|
|
// - Modifying the style property on the elements themselves
|
|
|
|
// - Injecting a style tag
|
|
|
|
|
|
|
|
var addStyleTag = function(selectors) {
|
|
|
|
hideElements(selectors);
|
|
|
|
var style = document.createElement('style');
|
|
|
|
style.setAttribute('class', 'ublock-postload-1ae7a5f130fc79b4fdb8a4272d9426b5');
|
|
|
|
// The linefeed before the style block is very important: do no remove!
|
|
|
|
style.appendChild(document.createTextNode(selectors.join(',\n') + '\n{display:none !important;}'));
|
|
|
|
var parent = document.body || document.documentElement;
|
|
|
|
if ( parent ) {
|
|
|
|
parent.appendChild(style);
|
|
|
|
}
|
2014-10-17 13:44:19 -06:00
|
|
|
messager.send({
|
2014-09-16 17:16:18 -06:00
|
|
|
what: 'injectedSelectors',
|
|
|
|
type: 'cosmetic',
|
|
|
|
hostname: window.location.hostname,
|
|
|
|
selectors: selectors
|
|
|
|
});
|
|
|
|
//console.debug('µBlock> generic cosmetic filters: injecting %d CSS rules:', selectors.length, text);
|
|
|
|
};
|
|
|
|
|
2014-08-17 14:07:42 -06:00
|
|
|
var hideElements = function(selectors) {
|
2014-09-03 18:14:55 -06:00
|
|
|
// https://github.com/gorhill/uBlock/issues/207
|
|
|
|
// Do not call querySelectorAll() using invalid CSS selectors
|
|
|
|
if ( selectors.length === 0 ) {
|
|
|
|
return;
|
|
|
|
}
|
2014-07-02 10:02:29 -06:00
|
|
|
if ( document.body === null ) {
|
|
|
|
return;
|
2014-06-23 16:42:43 -06:00
|
|
|
}
|
2014-08-17 14:07:42 -06:00
|
|
|
// https://github.com/gorhill/uBlock/issues/158
|
|
|
|
// Using CSSStyleDeclaration.setProperty is more reliable
|
2014-07-02 10:02:29 -06:00
|
|
|
var elems = document.querySelectorAll(selectors);
|
|
|
|
var i = elems.length;
|
|
|
|
while ( i-- ) {
|
2014-08-17 14:07:42 -06:00
|
|
|
elems[i].style.setProperty('display', 'none', 'important');
|
2014-06-23 16:42:43 -06:00
|
|
|
}
|
2014-07-02 10:02:29 -06:00
|
|
|
};
|
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// Extract and return the staged nodes which (may) match the selectors.
|
|
|
|
|
2014-08-12 10:19:54 -06:00
|
|
|
var selectNodes = function(selector) {
|
|
|
|
var targetNodes = [];
|
|
|
|
var i = contextNodes.length;
|
|
|
|
var node, nodeList, j;
|
|
|
|
var doc = document;
|
|
|
|
while ( i-- ) {
|
|
|
|
node = contextNodes[i];
|
|
|
|
if ( node === doc ) {
|
|
|
|
return doc.querySelectorAll(selector);
|
|
|
|
}
|
|
|
|
targetNodes.push(node);
|
|
|
|
nodeList = node.querySelectorAll(selector);
|
|
|
|
j = nodeList.length;
|
|
|
|
while ( j-- ) {
|
|
|
|
targetNodes.push(nodeList[j]);
|
|
|
|
}
|
2014-07-04 14:47:34 -06:00
|
|
|
}
|
2014-08-12 10:19:54 -06:00
|
|
|
return targetNodes;
|
2014-07-04 14:47:34 -06:00
|
|
|
};
|
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// Low generics:
|
|
|
|
// - [id]
|
|
|
|
// - [class]
|
|
|
|
|
2014-08-12 10:19:54 -06:00
|
|
|
var processLowGenerics = function(generics, out) {
|
|
|
|
var i = generics.length;
|
|
|
|
var selector;
|
|
|
|
while ( i-- ) {
|
|
|
|
selector = generics[i];
|
|
|
|
if ( injectedSelectors[selector] !== undefined ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
injectedSelectors[selector] = true;
|
2014-08-14 11:59:37 -06:00
|
|
|
out.push(selector);
|
2014-07-04 14:47:34 -06:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// High-low generics:
|
|
|
|
// - [alt="..."]
|
|
|
|
// - [title="..."]
|
|
|
|
|
2014-08-12 10:19:54 -06:00
|
|
|
var processHighLowGenerics = function(generics, out) {
|
|
|
|
var attrs = ['title', 'alt'];
|
2014-08-13 18:03:55 -06:00
|
|
|
var attr, attrValue, nodeList, iNode, node;
|
|
|
|
var selector;
|
2014-08-12 10:19:54 -06:00
|
|
|
while ( attr = attrs.pop() ) {
|
|
|
|
nodeList = selectNodes('[' + attr + ']');
|
|
|
|
iNode = nodeList.length;
|
|
|
|
while ( iNode-- ) {
|
|
|
|
node = nodeList[iNode];
|
|
|
|
attrValue = node.getAttribute(attr);
|
|
|
|
if ( !attrValue ) { continue; }
|
|
|
|
selector = '[' + attr + '="' + attrValue + '"]';
|
2014-08-13 18:03:55 -06:00
|
|
|
if ( generics[selector] ) {
|
|
|
|
if ( injectedSelectors[selector] === undefined ) {
|
|
|
|
injectedSelectors[selector] = true;
|
2014-08-14 11:59:37 -06:00
|
|
|
out.push(selector);
|
2014-08-12 10:19:54 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
selector = node.tagName.toLowerCase() + selector;
|
2014-08-13 18:03:55 -06:00
|
|
|
if ( generics[selector] ) {
|
|
|
|
if ( injectedSelectors[selector] === undefined ) {
|
|
|
|
injectedSelectors[selector] = true;
|
2014-08-14 11:59:37 -06:00
|
|
|
out.push(selector);
|
2014-08-12 10:19:54 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-04 14:47:34 -06:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// High-medium generics:
|
|
|
|
// - [href^="http"]
|
|
|
|
|
2014-08-12 10:19:54 -06:00
|
|
|
var processHighMediumGenerics = function(generics, out) {
|
|
|
|
var nodeList = selectNodes('a[href^="http"]');
|
|
|
|
var iNode = nodeList.length;
|
2014-08-13 18:03:55 -06:00
|
|
|
var node, href, pos, hash, selectors, selector, iSelector;
|
2014-08-12 10:19:54 -06:00
|
|
|
while ( iNode-- ) {
|
|
|
|
node = nodeList[iNode];
|
|
|
|
href = node.getAttribute('href');
|
|
|
|
if ( !href ) { continue; }
|
|
|
|
pos = href.indexOf('://');
|
|
|
|
if ( pos === -1 ) { continue; }
|
|
|
|
hash = href.slice(pos + 3, pos + 11);
|
2014-08-13 18:03:55 -06:00
|
|
|
selectors = generics[hash];
|
|
|
|
if ( selectors === undefined ) { continue; }
|
|
|
|
selectors = selectors.split(',\n');
|
|
|
|
iSelector = selectors.length;
|
|
|
|
while ( iSelector-- ) {
|
|
|
|
selector = selectors[iSelector];
|
|
|
|
if ( injectedSelectors[selector] === undefined ) {
|
|
|
|
injectedSelectors[selector] = true;
|
2014-08-14 11:59:37 -06:00
|
|
|
out.push(selector);
|
2014-08-13 18:03:55 -06:00
|
|
|
}
|
2014-07-02 10:02:29 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-10-17 13:44:19 -06:00
|
|
|
// High-high generics are *very costly* to process, so we will coalesce
|
2014-09-16 17:16:18 -06:00
|
|
|
// requests to process high-high generics into as few requests as possible.
|
|
|
|
// The gain is *significant* on bloated pages.
|
|
|
|
|
|
|
|
var processHighHighGenericsTimer = null;
|
|
|
|
|
|
|
|
var processHighHighGenerics = function() {
|
|
|
|
processHighHighGenericsTimer = null;
|
2014-08-13 18:03:55 -06:00
|
|
|
if ( injectedSelectors['{{highHighGenerics}}'] !== undefined ) { return; }
|
2014-09-16 17:16:18 -06:00
|
|
|
if ( document.querySelector(highGenerics.hideHigh) === null ) { return; }
|
2014-08-13 18:03:55 -06:00
|
|
|
injectedSelectors['{{highHighGenerics}}'] = true;
|
2014-10-17 13:44:19 -06:00
|
|
|
// We need to filter out possible exception cosmetic filters from
|
2014-09-16 17:16:18 -06:00
|
|
|
// high-high generics selectors.
|
|
|
|
var selectors = highGenerics.hideHigh.split(',\n');
|
|
|
|
var i = selectors.length;
|
2014-08-13 18:03:55 -06:00
|
|
|
var selector;
|
2014-09-16 17:16:18 -06:00
|
|
|
while ( i-- ) {
|
|
|
|
selector = selectors[i];
|
|
|
|
if ( injectedSelectors.hasOwnProperty(selector) ) {
|
|
|
|
selectors.splice(i, 1);
|
|
|
|
} else {
|
2014-08-13 18:03:55 -06:00
|
|
|
injectedSelectors[selector] = true;
|
2014-07-02 10:02:29 -06:00
|
|
|
}
|
|
|
|
}
|
2014-09-16 17:16:18 -06:00
|
|
|
if ( selectors.length !== 0 ) {
|
|
|
|
addStyleTag(selectors);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var processHighHighGenericsAsync = function() {
|
|
|
|
if ( processHighHighGenericsTimer !== null ) {
|
|
|
|
clearTimeout(processHighHighGenericsTimer);
|
|
|
|
}
|
|
|
|
processHighHighGenericsTimer = setTimeout(processHighHighGenerics, 300);
|
2014-07-02 10:02:29 -06:00
|
|
|
};
|
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// Extract all ids: these will be passed to the cosmetic filtering
|
|
|
|
// engine, and in return we will obtain only the relevant CSS selectors.
|
|
|
|
|
2014-07-20 13:00:26 -06:00
|
|
|
var idsFromNodeList = function(nodes) {
|
2014-07-02 10:02:29 -06:00
|
|
|
if ( !nodes || !nodes.length ) {
|
|
|
|
return;
|
2014-06-23 16:42:43 -06:00
|
|
|
}
|
2014-07-02 10:02:29 -06:00
|
|
|
if ( idSelectors === null ) {
|
|
|
|
idSelectors = [];
|
|
|
|
}
|
|
|
|
var qq = queriedSelectors;
|
|
|
|
var ii = idSelectors;
|
2014-07-20 13:00:26 -06:00
|
|
|
var node, v;
|
2014-07-02 10:02:29 -06:00
|
|
|
var i = nodes.length;
|
|
|
|
while ( i-- ) {
|
|
|
|
node = nodes[i];
|
2014-07-20 13:00:26 -06:00
|
|
|
if ( node.nodeType !== 1 ) { continue; }
|
2014-07-02 10:02:29 -06:00
|
|
|
// id
|
2014-07-08 23:53:49 -06:00
|
|
|
v = nodes[i].id;
|
2014-07-20 13:00:26 -06:00
|
|
|
if ( typeof v !== 'string' ) { continue; }
|
2014-07-08 23:53:49 -06:00
|
|
|
v = v.trim();
|
2014-07-20 13:00:26 -06:00
|
|
|
if ( v === '' ) { continue; }
|
|
|
|
v = '#' + v;
|
|
|
|
if ( qq[v] ) { continue; }
|
|
|
|
ii.push(v);
|
|
|
|
qq[v] = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// Extract all classes: these will be passed to the cosmetic filtering
|
|
|
|
// engine, and in return we will obtain only the relevant CSS selectors.
|
|
|
|
|
2014-07-20 13:00:26 -06:00
|
|
|
var classesFromNodeList = function(nodes) {
|
|
|
|
if ( !nodes || !nodes.length ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if ( classSelectors === null ) {
|
|
|
|
classSelectors = {};
|
|
|
|
}
|
|
|
|
var qq = queriedSelectors;
|
|
|
|
var cc = classSelectors;
|
|
|
|
var node, v, vv, j;
|
|
|
|
var i = nodes.length;
|
|
|
|
while ( i-- ) {
|
|
|
|
node = nodes[i];
|
2014-11-05 08:29:19 -07:00
|
|
|
vv = node.classList;
|
|
|
|
if ( typeof vv !== 'object' ) { continue; }
|
|
|
|
j = vv.length || 0;
|
2014-07-02 10:02:29 -06:00
|
|
|
while ( j-- ) {
|
2014-11-05 08:29:19 -07:00
|
|
|
v = vv[j];
|
|
|
|
if ( typeof v !== 'string' ) { continue; }
|
2014-07-02 10:02:29 -06:00
|
|
|
v = '.' + v;
|
|
|
|
if ( qq[v] ) { continue; }
|
|
|
|
cc[v] = true;
|
|
|
|
qq[v] = true;
|
|
|
|
}
|
2014-06-23 16:42:43 -06:00
|
|
|
}
|
2014-07-02 10:02:29 -06:00
|
|
|
};
|
2014-06-23 16:42:43 -06:00
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// Start cosmetic filtering.
|
|
|
|
|
2014-07-02 10:02:29 -06:00
|
|
|
domLoaded();
|
2014-06-23 16:42:43 -06:00
|
|
|
|
2014-09-16 17:16:18 -06:00
|
|
|
// Below this point is the code which takes care to observe changes in
|
|
|
|
// the page and to add if needed relevant CSS rules as a result of the
|
|
|
|
// changes.
|
|
|
|
|
2014-07-30 06:05:00 -06:00
|
|
|
// Observe changes in the DOM only if...
|
|
|
|
// - there is a document.body
|
|
|
|
// - there is at least one `script` tag
|
|
|
|
if ( !document.body || !document.querySelector('script') ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-10-17 13:44:19 -06:00
|
|
|
if (!window.MutationObserver) {
|
2014-10-29 13:36:23 -06:00
|
|
|
window.MutationObserver = window.WebKitMutationObserver || window.MozMutationObserver;
|
|
|
|
|
|
|
|
// dummy shim for older browsers
|
|
|
|
if (!window.MutationObserver) {
|
|
|
|
window.MutationObserver = function(handler) {
|
|
|
|
this.observe = function(target) {
|
|
|
|
target.addEventListener('DOMNodeInserted', function(e) {
|
|
|
|
handler([{addedNodes: [e.target]}]);
|
|
|
|
}, true);
|
|
|
|
};
|
|
|
|
}
|
2014-10-17 13:44:19 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-12 10:19:54 -06:00
|
|
|
var ignoreTags = {
|
2014-08-14 11:59:37 -06:00
|
|
|
'link': true,
|
|
|
|
'LINK': true,
|
2014-08-12 10:19:54 -06:00
|
|
|
'script': true,
|
2014-08-14 11:59:37 -06:00
|
|
|
'SCRIPT': true,
|
|
|
|
'style': true,
|
|
|
|
'STYLE': true
|
2014-08-12 10:19:54 -06:00
|
|
|
};
|
|
|
|
|
2014-09-16 13:39:21 -06:00
|
|
|
// Added node lists will be cumulated here before being processed
|
|
|
|
var addedNodeLists = [];
|
|
|
|
var addedNodeListsTimer = null;
|
|
|
|
|
|
|
|
var mutationObservedHandler = function() {
|
2014-08-12 10:19:54 -06:00
|
|
|
var nodeList, iNode, node;
|
2014-09-16 13:39:21 -06:00
|
|
|
while ( nodeList = addedNodeLists.pop() ) {
|
2014-08-12 10:19:54 -06:00
|
|
|
iNode = nodeList.length;
|
|
|
|
while ( iNode-- ) {
|
|
|
|
node = nodeList[iNode];
|
2014-09-24 15:41:05 -06:00
|
|
|
if ( node.nodeType !== 1 ) {
|
2014-08-12 10:19:54 -06:00
|
|
|
continue;
|
|
|
|
}
|
2014-09-05 14:13:48 -06:00
|
|
|
if ( ignoreTags.hasOwnProperty(node.tagName) ) {
|
2014-08-12 10:19:54 -06:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
contextNodes.push(node);
|
2014-07-30 06:05:00 -06:00
|
|
|
}
|
|
|
|
}
|
2014-09-16 13:39:21 -06:00
|
|
|
addedNodeListsTimer = null;
|
2014-08-12 10:19:54 -06:00
|
|
|
if ( contextNodes.length !== 0 ) {
|
|
|
|
idsFromNodeList(selectNodes('[id]'));
|
|
|
|
classesFromNodeList(selectNodes('[class]'));
|
|
|
|
retrieveGenericSelectors();
|
2014-07-30 06:05:00 -06:00
|
|
|
}
|
2014-07-02 10:02:29 -06:00
|
|
|
};
|
2014-08-12 10:19:54 -06:00
|
|
|
|
2014-09-16 13:39:21 -06:00
|
|
|
// https://github.com/gorhill/uBlock/issues/205
|
|
|
|
// Do not handle added node directly from within mutation observer.
|
2014-09-24 15:41:05 -06:00
|
|
|
var treeMutationObservedHandlerAsync = function(mutations) {
|
2014-09-16 13:39:21 -06:00
|
|
|
var iMutation = mutations.length;
|
|
|
|
var nodeList;
|
|
|
|
while ( iMutation-- ) {
|
2014-09-24 15:41:05 -06:00
|
|
|
nodeList = mutations[iMutation].addedNodes;
|
|
|
|
if ( nodeList.length !== 0 ) {
|
2014-09-16 13:39:21 -06:00
|
|
|
addedNodeLists.push(nodeList);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( addedNodeListsTimer === null ) {
|
|
|
|
// I arbitrarily chose 50 ms for now:
|
2014-10-17 13:44:19 -06:00
|
|
|
// I have to compromise between the overhead of processing too few
|
2014-09-16 13:39:21 -06:00
|
|
|
// nodes too often and the delay of many nodes less often.
|
2014-09-24 15:41:05 -06:00
|
|
|
addedNodeListsTimer = setTimeout(mutationObservedHandler, 75);
|
2014-09-16 13:39:21 -06:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-07-30 06:05:00 -06:00
|
|
|
// https://github.com/gorhill/httpswitchboard/issues/176
|
2014-09-24 15:41:05 -06:00
|
|
|
var treeObserver = new MutationObserver(treeMutationObservedHandlerAsync);
|
|
|
|
treeObserver.observe(document.body, {
|
2014-07-30 06:05:00 -06:00
|
|
|
childList: true,
|
|
|
|
subtree: true
|
|
|
|
});
|
2014-07-02 10:02:29 -06:00
|
|
|
})();
|
2014-06-23 16:42:43 -06:00
|
|
|
|
2014-08-02 09:40:27 -06:00
|
|
|
/******************************************************************************/
|
2014-06-23 16:42:43 -06:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2014-09-14 14:20:40 -06:00
|
|
|
// Permanent
|
2014-06-27 15:06:42 -06:00
|
|
|
|
2014-07-20 13:00:26 -06:00
|
|
|
(function() {
|
|
|
|
// Listeners to mop up whatever is otherwise missed:
|
|
|
|
// - Future requests not blocked yet
|
|
|
|
// - Elements dynamically added to the page
|
|
|
|
// - Elements which resource URL changes
|
2014-08-14 11:59:37 -06:00
|
|
|
|
|
|
|
var loadedElements = {
|
|
|
|
'iframe': 'src'
|
|
|
|
};
|
|
|
|
|
|
|
|
var failedElements = {
|
|
|
|
'img': 'src',
|
2014-09-05 14:13:48 -06:00
|
|
|
'input': 'src',
|
2014-08-14 11:59:37 -06:00
|
|
|
'object': 'data'
|
|
|
|
};
|
|
|
|
|
|
|
|
var onResource = function(target, dict) {
|
|
|
|
if ( !target ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var tagName = target.tagName.toLowerCase();
|
|
|
|
var prop = dict[tagName];
|
|
|
|
if ( prop === undefined ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var src = target[prop];
|
2014-08-25 07:24:01 -06:00
|
|
|
if ( typeof src !== 'string' || src === '' ) {
|
2014-08-14 11:59:37 -06:00
|
|
|
return;
|
|
|
|
}
|
2014-09-14 14:20:40 -06:00
|
|
|
if ( src.slice(0, 4) !== 'http' ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-08-25 06:49:08 -06:00
|
|
|
// https://github.com/gorhill/uBlock/issues/174
|
|
|
|
// Do not remove fragment from src URL
|
|
|
|
|
2014-08-02 09:40:27 -06:00
|
|
|
var onAnswerReceived = function(details) {
|
2014-09-14 14:20:40 -06:00
|
|
|
if ( typeof details !== 'object' || details === null ) {
|
2014-08-14 11:59:37 -06:00
|
|
|
return;
|
2014-08-02 09:40:27 -06:00
|
|
|
}
|
2014-08-14 11:59:37 -06:00
|
|
|
// If `!important` is not there, going back using history will
|
|
|
|
// likely cause the hidden element to re-appear.
|
2014-08-15 08:40:25 -06:00
|
|
|
if ( details.collapse ) {
|
|
|
|
if ( target.parentNode ) {
|
|
|
|
target.parentNode.removeChild(target);
|
|
|
|
} else {
|
2014-09-05 14:13:48 -06:00
|
|
|
target.style.setProperty('display', 'none', 'important');
|
2014-08-15 08:40:25 -06:00
|
|
|
}
|
2014-09-05 14:13:48 -06:00
|
|
|
} else {
|
|
|
|
target.style.setProperty('visibility', 'hidden', 'important');
|
2014-08-14 11:59:37 -06:00
|
|
|
}
|
2014-10-17 13:44:19 -06:00
|
|
|
messager.send({
|
2014-08-14 11:59:37 -06:00
|
|
|
what: 'injectedSelectors',
|
|
|
|
type: 'net',
|
|
|
|
hostname: window.location.hostname,
|
|
|
|
selectors: tagName + '[' + prop + '="' + src + '"]'
|
|
|
|
});
|
2014-08-02 09:40:27 -06:00
|
|
|
};
|
2014-09-14 14:20:40 -06:00
|
|
|
|
|
|
|
var details = {
|
|
|
|
what: 'filterRequest',
|
|
|
|
tagName: tagName,
|
|
|
|
requestURL: src,
|
|
|
|
pageHostname: window.location.hostname,
|
|
|
|
pageURL: window.location.href
|
|
|
|
};
|
2014-10-17 13:44:19 -06:00
|
|
|
messager.send(details, onAnswerReceived);
|
2014-08-14 11:59:37 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
var onResourceLoaded = function(ev) {
|
2014-08-15 08:34:13 -06:00
|
|
|
//console.debug('Loaded %s[src="%s"]', ev.target.tagName, ev.target.src);
|
2014-08-14 11:59:37 -06:00
|
|
|
onResource(ev.target, loadedElements);
|
2014-08-02 09:40:27 -06:00
|
|
|
};
|
2014-08-14 11:59:37 -06:00
|
|
|
|
2014-08-02 09:40:27 -06:00
|
|
|
var onResourceFailed = function(ev) {
|
2014-09-05 14:13:48 -06:00
|
|
|
//console.debug('Failed to load %s[src="%s"]', ev.target.tagName, ev.target.src);
|
2014-08-14 11:59:37 -06:00
|
|
|
onResource(ev.target, failedElements);
|
2014-06-27 15:06:42 -06:00
|
|
|
};
|
2014-08-14 11:59:37 -06:00
|
|
|
|
2014-07-20 13:00:26 -06:00
|
|
|
document.addEventListener('load', onResourceLoaded, true);
|
2014-08-02 09:40:27 -06:00
|
|
|
document.addEventListener('error', onResourceFailed, true);
|
2014-07-02 10:02:29 -06:00
|
|
|
})();
|
2014-06-23 16:42:43 -06:00
|
|
|
|
|
|
|
/******************************************************************************/
|
2014-09-14 14:20:40 -06:00
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
// https://github.com/gorhill/uBlock/issues/7
|
|
|
|
|
|
|
|
// Executed only once
|
|
|
|
|
|
|
|
(function() {
|
|
|
|
var srcProps = {
|
|
|
|
'embed': 'src',
|
|
|
|
'iframe': 'src',
|
|
|
|
'img': 'src',
|
|
|
|
'object': 'data'
|
|
|
|
};
|
|
|
|
var elements = [];
|
|
|
|
|
|
|
|
var onAnswerReceived = function(details) {
|
|
|
|
if ( typeof details !== 'object' || details === null ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var requests = details.requests;
|
|
|
|
var collapse = details.collapse;
|
|
|
|
var selectors = [];
|
|
|
|
var i = requests.length;
|
|
|
|
var request, elem;
|
|
|
|
while ( i-- ) {
|
|
|
|
request = requests[i];
|
|
|
|
elem = elements[request.index];
|
|
|
|
if ( collapse ) {
|
|
|
|
if ( elem.parentNode ) {
|
|
|
|
elem.parentNode.removeChild(elem);
|
|
|
|
} else {
|
|
|
|
elem.style.setProperty('display', 'none', 'important');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
elem.style.setProperty('visibility', 'hidden', 'important');
|
|
|
|
}
|
|
|
|
selectors.push(request.tagName + '[' + srcProps[request.tagName] + '="' + request.url + '"]');
|
|
|
|
}
|
|
|
|
if ( selectors.length !== 0 ) {
|
2014-10-17 13:44:19 -06:00
|
|
|
messager.send({
|
2014-09-14 14:20:40 -06:00
|
|
|
what: 'injectedSelectors',
|
|
|
|
type: 'net',
|
|
|
|
hostname: window.location.hostname,
|
|
|
|
selectors: selectors
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var requests = [];
|
|
|
|
var tagNames = ['embed','iframe','img','object'];
|
|
|
|
var elementIndex = 0;
|
|
|
|
var tagName, elems, i, elem, prop, src;
|
|
|
|
while ( tagName = tagNames.pop() ) {
|
|
|
|
elems = document.getElementsByTagName(tagName);
|
|
|
|
i = elems.length;
|
|
|
|
while ( i-- ) {
|
|
|
|
elem = elems[i];
|
|
|
|
prop = srcProps[tagName];
|
|
|
|
if ( prop === undefined ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
src = elem[prop];
|
|
|
|
if ( typeof src !== 'string' || src === '' ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ( src.slice(0, 4) !== 'http' ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
requests.push({
|
|
|
|
index: elementIndex,
|
|
|
|
tagName: tagName,
|
|
|
|
url: src
|
|
|
|
});
|
|
|
|
elements[elementIndex] = elem;
|
|
|
|
elementIndex += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var details = {
|
|
|
|
what: 'filterRequests',
|
|
|
|
pageURL: window.location.href,
|
|
|
|
pageHostname: window.location.hostname,
|
|
|
|
requests: requests
|
|
|
|
};
|
2014-10-17 13:44:19 -06:00
|
|
|
messager.send(details, onAnswerReceived);
|
2014-09-14 14:20:40 -06:00
|
|
|
})();
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2014-09-28 12:38:17 -06:00
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
// To send mouse coordinates to context menu handler, as the chrome API fails
|
|
|
|
// to provide the mouse position to context menu listeners.
|
|
|
|
// This could be inserted in its own content script, but it's so simple that
|
|
|
|
// I feel it's not worth the overhead.
|
|
|
|
|
|
|
|
// Ref.: https://developer.mozilla.org/en-US/docs/Web/Events/contextmenu
|
|
|
|
|
|
|
|
(function() {
|
|
|
|
var onContextMenu = function(ev) {
|
2014-10-17 13:44:19 -06:00
|
|
|
messager.send({
|
2014-09-28 12:38:17 -06:00
|
|
|
what: 'contextMenuEvent',
|
|
|
|
clientX: ev.clientX,
|
|
|
|
clientY: ev.clientY
|
|
|
|
});
|
|
|
|
};
|
2014-10-17 13:44:19 -06:00
|
|
|
|
|
|
|
window.addEventListener('contextmenu', onContextMenu, true);
|
2014-09-28 12:38:17 -06:00
|
|
|
})();
|
|
|
|
|
|
|
|
/******************************************************************************/
|