1
0
mirror of https://github.com/gorhill/uBlock.git synced 2024-11-17 16:02:33 +01:00
uBlock/src/js/contentscript-end.js

1003 lines
34 KiB
JavaScript
Raw Normal View History

2014-06-24 00:42:43 +02:00
/*******************************************************************************
2015-03-07 19:20:18 +01:00
µBlock - a browser extension to block requests.
2014-06-24 00:42:43 +02:00
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-06-24 00:42:43 +02:00
// Injected into content pages
2015-01-02 03:14:53 +01:00
(function() {
'use strict';
/******************************************************************************/
2015-05-09 00:28:01 +02:00
// I've seen this happens on Firefox
if ( window.location === null ) {
return;
2015-01-13 21:52:15 +01:00
}
// This can happen
2015-05-16 19:40:56 +02:00
if ( typeof vAPI !== 'object' ) {
2015-01-23 17:32:49 +01:00
//console.debug('contentscript-end.js > vAPI not found');
2015-01-13 21:52:15 +01:00
return;
}
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/587
2015-01-23 17:32:49 +01:00
// Pointless to execute without the start script having done its job.
if ( !vAPI.contentscriptStartInjected ) {
2015-01-13 21:52:15 +01:00
return;
}
2014-06-24 00:42:43 +02:00
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/456
// Already injected?
2015-01-02 03:14:53 +01:00
if ( vAPI.contentscriptEndInjected ) {
2015-01-23 17:32:49 +01:00
//console.debug('contentscript-end.js > content script already injected');
return;
}
2015-01-02 03:14:53 +01:00
vAPI.contentscriptEndInjected = true;
vAPI.styles = vAPI.styles || [];
/******************************************************************************/
var messager = vAPI.messaging.channel('contentscript-end.js');
2014-06-24 00:42:43 +02:00
// https://github.com/gorhill/uMatrix/issues/144
2015-04-08 01:34:22 +02:00
vAPI.shutdown.add(function() {
messager.close();
});
/******************************************************************************/
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/7
2015-03-29 18:13:28 +02:00
var uBlockCollapser = (function() {
var timer = null;
var requestId = 1;
var newRequests = [];
var pendingRequests = {};
2015-03-29 18:13:28 +02:00
var pendingRequestCount = 0;
2015-06-04 17:17:02 +02:00
var src1stProps = {
2015-03-29 18:13:28 +02:00
'embed': 'src',
'iframe': 'src',
2015-03-29 18:13:28 +02:00
'img': 'src',
'object': 'data'
};
2015-06-04 17:17:02 +02:00
var src2ndProps = {
'img': 'srcset'
};
2015-03-29 18:13:28 +02:00
var PendingRequest = function(target, tagName, attr) {
this.id = requestId++;
this.target = target;
this.tagName = tagName;
this.attr = attr;
pendingRequests[this.id] = this;
pendingRequestCount += 1;
};
// Because a while ago I have observed constructors are faster than
// literal object instanciations.
var BouncingRequest = function(id, tagName, url) {
this.id = id;
this.tagName = tagName;
this.url = url;
this.collapse = false;
};
var onProcessed = function(response) {
// https://github.com/gorhill/uMatrix/issues/144
if ( response.shutdown ) {
2015-04-08 01:34:22 +02:00
vAPI.shutdown.exec();
return;
}
var requests = response.result;
2015-03-29 18:13:28 +02:00
if ( requests === null || Array.isArray(requests) === false ) {
return;
}
var selectors = [];
var i = requests.length;
var request, entry, target, value;
while ( i-- ) {
request = requests[i];
if ( pendingRequests.hasOwnProperty(request.id) === false ) {
2015-03-29 18:13:28 +02:00
continue;
}
entry = pendingRequests[request.id];
2015-03-29 18:13:28 +02:00
delete pendingRequests[request.id];
pendingRequestCount -= 1;
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/869
2015-03-29 18:13:28 +02:00
if ( !request.collapse ) {
continue;
}
target = entry.target;
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/399
2015-03-29 18:13:28 +02:00
// Never remove elements from the DOM, just hide them
target.style.setProperty('display', 'none', 'important');
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/1048
2015-03-29 18:13:28 +02:00
// Use attribute to construct CSS rule
if ( (value = target.getAttribute(entry.attr)) ) {
2015-03-29 18:13:28 +02:00
selectors.push(entry.tagName + '[' + entry.attr + '="' + value + '"]');
}
}
if ( selectors.length !== 0 ) {
messager.send({
what: 'cosmeticFiltersInjected',
2015-03-29 18:13:28 +02:00
type: 'net',
hostname: window.location.hostname,
selectors: selectors
});
}
// Renew map: I believe that even if all properties are deleted, an
// object will still use more memory than a brand new one.
if ( pendingRequestCount === 0 ) {
pendingRequests = {};
2015-03-29 18:13:28 +02:00
}
};
var send = function() {
timer = null;
messager.send({
what: 'filterRequests',
pageURL: window.location.href,
pageHostname: window.location.hostname,
requests: newRequests
}, onProcessed);
newRequests = [];
};
var process = function(delay) {
if ( newRequests.length === 0 ) {
return;
}
if ( delay === 0 ) {
clearTimeout(timer);
send();
} else if ( timer === null ) {
timer = vAPI.setTimeout(send, delay || 20);
2015-03-29 18:13:28 +02:00
}
};
// If needed eventually, we could listen to `src` attribute changes
// for iframes.
var add = function(target) {
var tagName = target.localName;
2015-06-04 17:17:02 +02:00
var prop = src1stProps[tagName];
2015-03-29 18:13:28 +02:00
if ( prop === undefined ) {
return;
}
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/174
2015-03-29 18:13:28 +02:00
// Do not remove fragment from src URL
var src = target[prop];
2015-06-04 17:17:02 +02:00
if ( typeof src !== 'string' || src.length === 0 ) {
prop = src2ndProps[tagName];
if ( prop === undefined ) {
return;
}
src = target[prop];
if ( typeof src !== 'string' || src.length === 0 ) {
return;
}
2015-03-29 18:13:28 +02:00
}
if ( src.lastIndexOf('http', 0) !== 0 ) {
return;
}
var req = new PendingRequest(target, tagName, prop);
newRequests.push(new BouncingRequest(req.id, tagName, src));
};
2015-05-02 01:06:52 +02:00
var iframeSourceModified = function(mutations) {
var i = mutations.length;
while ( i-- ) {
addIFrame(mutations[i].target, true);
}
process();
};
var iframeSourceObserver = new MutationObserver(iframeSourceModified);
var iframeSourceObserverOptions = {
attributes: true,
attributeFilter: [ 'src' ]
};
var primeLocalIFrame = function(iframe) {
// Should probably also copy injected styles.
if ( vAPI.injectedScripts ) {
var scriptTag = document.createElement('script');
scriptTag.appendChild(document.createTextNode(vAPI.injectedScripts));
var parent = iframe.contentDocument && iframe.contentDocument.head;
if ( parent ) {
parent.appendChild(scriptTag);
}
}
};
2015-05-02 01:06:52 +02:00
var addIFrame = function(iframe, dontObserve) {
// https://github.com/gorhill/uBlock/issues/162
// Be prepared to deal with possible change of src attribute.
if ( dontObserve !== true ) {
iframeSourceObserver.observe(iframe, iframeSourceObserverOptions);
}
2015-03-29 18:13:28 +02:00
var src = iframe.src;
if ( src === '' || typeof src !== 'string' ) {
primeLocalIFrame(iframe);
2015-03-29 18:13:28 +02:00
return;
}
if ( src.lastIndexOf('http', 0) !== 0 ) {
return;
}
var req = new PendingRequest(iframe, 'iframe', 'src');
newRequests.push(new BouncingRequest(req.id, 'iframe', src));
};
2016-02-04 00:37:56 +01:00
var iframesFromNode = function(node) {
if ( node.localName === 'iframe' ) {
addIFrame(node);
}
var iframes = node.getElementsByTagName('iframe');
var i = iframes.length;
while ( i-- ) {
addIFrame(iframes[i]);
}
2015-03-29 18:13:28 +02:00
process();
};
return {
add: add,
addIFrame: addIFrame,
iframesFromNode: iframesFromNode,
process: process
};
})();
/******************************************************************************/
2015-02-14 18:16:36 +01:00
/******************************************************************************/
// Cosmetic filters
2014-06-24 00:42:43 +02:00
(function() {
2014-12-19 20:00:46 +01:00
if ( vAPI.skipCosmeticFiltering ) {
//console.debug('Abort cosmetic filtering');
2014-12-19 20:00:46 +01:00
return;
}
2015-10-31 05:55:10 +01:00
//console.debug('Start cosmetic filtering');
2015-03-02 02:26:33 +01:00
//var timer = window.performance || Date;
//var tStart = timer.now();
2015-12-08 20:06:33 +01:00
var hideElements = (function() {
if ( document.body === null ) {
return function() {};
}
if ( document.body.shadowRoot === undefined ) {
return function(selectors) {
// https://github.com/chrisaljoudi/uBlock/issues/207
// Do not call querySelectorAll() using invalid CSS selectors
if ( selectors.length === 0 ) { return; }
var elems = document.querySelectorAll(selectors);
var i = elems.length;
if ( i === 0 ) { return; }
// https://github.com/chrisaljoudi/uBlock/issues/158
// Using CSSStyleDeclaration.setProperty is more reliable
while ( i-- ) {
elems[i].style.setProperty('display', 'none', 'important');
}
};
}
return function(selectors) {
if ( selectors.length === 0 ) { return; }
var elems = document.querySelectorAll(selectors);
var i = elems.length;
if ( i === 0 ) { return; }
// https://github.com/gorhill/uBlock/issues/435
// Using shadow content so that we do not have to modify style
// attribute.
var sessionId = vAPI.sessionId;
var elem, shadow;
while ( i-- ) {
elem = elems[i];
shadow = elem.shadowRoot;
// https://www.chromestatus.com/features/4668884095336448
// "Multiple shadow roots is being deprecated."
if ( shadow !== null ) {
if ( shadow.className !== sessionId ) {
elem.style.setProperty('display', 'none', 'important');
}
continue;
}
// https://github.com/gorhill/uBlock/pull/555
// Not all nodes can be shadowed:
// https://github.com/w3c/webcomponents/issues/102
// https://github.com/gorhill/uBlock/issues/762
// Remove display style that might get in the way of the shadow
// node doing its magic.
try {
shadow = elem.createShadowRoot();
shadow.className = sessionId;
elem.style.removeProperty('display');
} catch (ex) {
elem.style.setProperty('display', 'none', 'important');
}
}
};
})();
2015-10-31 05:55:10 +01:00
// https://github.com/chrisaljoudi/uBlock/issues/789
// https://github.com/gorhill/uBlock/issues/873
// Be sure that our style tags used for cosmetic filtering are still applied.
var checkStyleTags = function() {
var doc = document,
html = doc.documentElement,
head = doc.head,
2015-12-08 20:06:33 +01:00
newParent = head || html;
2015-10-31 05:55:10 +01:00
if ( newParent === null ) {
return;
}
var styles = vAPI.styles || [],
style, oldParent;
for ( var i = 0; i < styles.length; i++ ) {
style = styles[i];
oldParent = style.parentNode;
2015-12-08 20:06:33 +01:00
// https://github.com/gorhill/uBlock/issues/1031
// If our style tag was disabled, force a re-insert into the page.
if (
style.disabled &&
oldParent !== null &&
style[vAPI.sessionId] === undefined
) {
oldParent.removeChild(style);
oldParent = null;
}
if ( oldParent === head || oldParent === html ) {
continue;
2015-10-31 05:55:10 +01:00
}
2015-12-08 20:06:33 +01:00
style.disabled = false;
newParent.appendChild(style);
// The page tried to get rid of us: reapply inline styles to
// blocked elements.
hideElements(style.textContent.slice(0, style.textContent.lastIndexOf('\n')));
2015-10-31 05:55:10 +01:00
}
};
checkStyleTags();
2014-07-02 18:02:29 +02:00
var queriedSelectors = {};
2015-01-01 05:09:22 +01:00
var injectedSelectors = {};
var lowGenericSelectors = [];
var highGenerics = null;
var contextNodes = [document];
var nullArray = { push: function(){} };
2014-07-02 18:02:29 +02:00
var retrieveGenericSelectors = function() {
2015-03-02 01:43:42 +01:00
if ( lowGenericSelectors.length !== 0 || highGenerics === null ) {
//console.log('µBlock> ABP cosmetic filters: retrieving CSS rules using %d selectors', lowGenericSelectors.length);
messager.send({
what: 'retrieveGenericCosmeticSelectors',
pageURL: window.location.href,
selectors: lowGenericSelectors,
firstSurvey: highGenerics === null
},
retrieveHandler
);
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/452
retrieveHandler = nextRetrieveHandler;
} else {
nextRetrieveHandler(null);
}
lowGenericSelectors = [];
2015-01-01 05:09:22 +01:00
};
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/452
// This needs to be executed *after* the response from our query is
2015-01-01 05:09:22 +01:00
// received, not at `DOMContentLoaded` time, or else there is a good
// likeliness to outrun contentscript-start.js, which may still be waiting
// on a response from its own query.
var firstRetrieveHandler = function(response) {
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/158
2014-11-16 20:06:29 +01:00
// Ensure injected styles are enforced
// rhill 2014-11-16: not sure this is needed anymore. Test case in
// above issue was fine without the line below..
2014-11-27 12:41:28 +01:00
var selectors = vAPI.hideCosmeticFilters;
if ( typeof selectors === 'object' ) {
2015-01-01 05:09:22 +01:00
injectedSelectors = selectors;
2015-03-28 16:09:36 +01:00
hideElements(Object.keys(selectors));
2014-06-26 00:44:35 +02:00
}
// Add exception filters into injected filters collection, in order
// to force them to be seen as "already injected".
2014-11-27 12:41:28 +01:00
selectors = vAPI.donthideCosmeticFilters;
if ( typeof selectors === 'object' ) {
2015-01-01 13:46:33 +01:00
for ( var selector in selectors ) {
2014-11-27 12:41:28 +01:00
if ( selectors.hasOwnProperty(selector) ) {
injectedSelectors[selector] = true;
}
}
}
// Flush dead code from memory
firstRetrieveHandler = null;
// These are sent only once
var result = response && response.result;
if ( result ) {
if ( result.highGenerics ) {
highGenerics = result.highGenerics;
}
if ( result.donthide ) {
processLowGenerics(result.donthide, nullArray);
}
}
nextRetrieveHandler(response);
2014-07-02 18:02:29 +02:00
};
2014-06-24 00:42:43 +02:00
var nextRetrieveHandler = function(response) {
// https://github.com/gorhill/uMatrix/issues/144
if ( response && response.shutdown ) {
2015-04-08 01:34:22 +02:00
vAPI.shutdown.exec();
return;
}
2015-03-02 02:26:33 +01:00
//var tStart = timer.now();
//console.debug('µBlock> contextNodes = %o', contextNodes);
var result = response && response.result;
var hideSelectors = [];
if ( result && result.hide.length ) {
processLowGenerics(result.hide, hideSelectors);
}
if ( highGenerics ) {
if ( highGenerics.hideLowCount ) {
processHighLowGenerics(highGenerics.hideLow, hideSelectors);
}
if ( highGenerics.hideMediumCount ) {
processHighMediumGenerics(highGenerics.hideMedium, hideSelectors);
}
if ( highGenerics.hideHighCount ) {
processHighHighGenericsAsync();
}
}
if ( hideSelectors.length !== 0 ) {
addStyleTag(hideSelectors);
2014-06-24 00:42:43 +02:00
}
contextNodes.length = 0;
2015-03-02 02:26:33 +01:00
//console.debug('%f: uBlock: CSS injection time', timer.now() - tStart);
2014-07-02 18:02:29 +02:00
};
2014-06-24 00:42:43 +02:00
var retrieveHandler = firstRetrieveHandler;
// Ensure elements matching a set of selectors are visually removed
// from the page, by:
// - Modifying the style property on the elements themselves
// - Injecting a style tag
var addStyleTag = function(selectors) {
2015-12-06 00:02:04 +01:00
// https://github.com/gorhill/uBlock/issues/1015
// Boost specificity of our CSS rules.
var styleText = ':root ' + selectors.join(',\n:root ');
var style = document.createElement('style');
2015-12-08 20:06:33 +01:00
style.setAttribute('type', 'text/css');
// The linefeed before the style block is very important: do no remove!
2015-12-06 00:02:04 +01:00
style.appendChild(document.createTextNode(styleText + '\n{display:none !important;}'));
2015-10-31 05:55:10 +01:00
var parent = document.head || document.documentElement;
if ( parent ) {
parent.appendChild(style);
vAPI.styles.push(style);
}
2015-12-06 00:02:04 +01:00
hideElements(styleText);
messager.send({
what: 'cosmeticFiltersInjected',
type: 'cosmetic',
hostname: window.location.hostname,
selectors: selectors
});
//console.debug('µBlock> generic cosmetic filters: injecting %d CSS rules:', selectors.length, text);
};
// Extract and return the staged nodes which (may) match the selectors.
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 22:47:34 +02:00
}
return targetNodes;
2014-07-04 22:47:34 +02:00
};
// Low generics:
// - [id]
// - [class]
var processLowGenerics = function(generics, out) {
var i = generics.length;
var selector;
while ( i-- ) {
selector = generics[i];
2015-01-01 05:09:22 +01:00
if ( injectedSelectors.hasOwnProperty(selector) ) {
continue;
}
injectedSelectors[selector] = true;
out.push(selector);
2014-07-04 22:47:34 +02:00
}
};
// High-low generics:
// - [alt="..."]
// - [title="..."]
var processHighLowGenerics = function(generics, out) {
var attrs = ['title', 'alt'];
2014-08-14 02:03:55 +02:00
var attr, attrValue, nodeList, iNode, node;
var selector;
while ( (attr = attrs.pop()) ) {
nodeList = selectNodes('[' + attr + ']');
iNode = nodeList.length;
while ( iNode-- ) {
node = nodeList[iNode];
attrValue = node.getAttribute(attr);
if ( !attrValue ) { continue; }
2015-01-02 01:09:40 +01:00
// Candidate 1 = generic form
2015-01-02 01:16:02 +01:00
// If generic form is injected, no need to process the specific
2015-01-02 01:09:40 +01:00
// form, as the generic will affect all related specific forms
selector = '[' + attr + '="' + attrValue + '"]';
2015-01-02 01:09:40 +01:00
if ( generics.hasOwnProperty(selector) ) {
2015-01-01 05:09:22 +01:00
if ( injectedSelectors.hasOwnProperty(selector) === false ) {
2014-08-14 02:03:55 +02:00
injectedSelectors[selector] = true;
out.push(selector);
2015-01-02 01:09:40 +01:00
continue;
}
}
2015-01-02 01:09:40 +01:00
// Candidate 2 = specific form
2015-03-29 18:13:28 +02:00
selector = node.localName + selector;
2015-01-02 01:09:40 +01:00
if ( generics.hasOwnProperty(selector) ) {
2015-01-01 05:09:22 +01:00
if ( injectedSelectors.hasOwnProperty(selector) === false ) {
2014-08-14 02:03:55 +02:00
injectedSelectors[selector] = true;
out.push(selector);
}
}
}
2014-07-04 22:47:34 +02:00
}
};
// High-medium generics:
// - [href^="http"]
var processHighMediumGenerics = function(generics, out) {
2015-11-01 17:25:36 +01:00
var doc = document;
var i = contextNodes.length;
var aa = [ null ];
var node, nodes;
while ( i-- ) {
node = contextNodes[i];
if ( node.localName === 'a' ) {
aa[0] = node;
processHighMediumGenericsForNodes(aa, generics, out);
}
nodes = node.getElementsByTagName('a');
if ( nodes.length === 0 ) { continue; }
processHighMediumGenericsForNodes(nodes, generics, out);
if ( node === doc ) {
break;
}
}
};
2015-11-01 17:25:36 +01:00
var processHighMediumGenericsForNodes = function(nodes, generics, out) {
var i = nodes.length;
var node, href, pos, hash, selectors, j, selector;
var aa = [ '' ];
while ( i-- ) {
node = nodes[i];
href = node.getAttribute('href');
if ( !href ) { continue; }
pos = href.indexOf('://');
if ( pos === -1 ) { continue; }
hash = href.slice(pos + 3, pos + 11);
2014-08-14 02:03:55 +02:00
selectors = generics[hash];
if ( selectors === undefined ) { continue; }
// A string.
if ( typeof selectors === 'string' ) {
2015-11-01 17:25:36 +01:00
aa[0] = selectors;
selectors = aa;
}
// An array of strings.
2015-11-01 17:25:36 +01:00
j = selectors.length;
while ( j-- ) {
selector = selectors[j];
if (
href.lastIndexOf(selector.slice(8, -2), 0) === 0 &&
injectedSelectors.hasOwnProperty(selector) === false
) {
2014-08-14 02:03:55 +02:00
injectedSelectors[selector] = true;
out.push(selector);
2014-08-14 02:03:55 +02:00
}
2014-07-02 18:02:29 +02:00
}
}
};
// High-high generics are *very costly* to process, so we will coalesce
// requests to process high-high generics into as few requests as possible.
// The gain is *significant* on bloated pages.
2015-03-02 16:54:15 +01:00
var processHighHighGenericsMisses = 8;
var processHighHighGenericsTimer = null;
var processHighHighGenerics = function() {
processHighHighGenericsTimer = null;
2015-03-13 17:26:54 +01:00
if ( highGenerics.hideHigh === '' ) {
return;
}
if ( injectedSelectors.hasOwnProperty('{{highHighGenerics}}') ) {
return;
}
2015-10-31 05:55:10 +01:00
// When there are too many misses for these highly generic CSS rules,
// we will just give up on looking whether they need to be injected.
if ( document.querySelector(highGenerics.hideHigh) === null ) {
2015-03-02 16:54:15 +01:00
processHighHighGenericsMisses -= 1;
if ( processHighHighGenericsMisses === 0 ) {
2015-03-02 02:26:33 +01:00
injectedSelectors['{{highHighGenerics}}'] = true;
}
return;
}
2014-08-14 02:03:55 +02:00
injectedSelectors['{{highHighGenerics}}'] = true;
// We need to filter out possible exception cosmetic filters from
// high-high generics selectors.
var selectors = highGenerics.hideHigh.split(',\n');
var i = selectors.length;
2014-08-14 02:03:55 +02:00
var selector;
while ( i-- ) {
selector = selectors[i];
if ( injectedSelectors.hasOwnProperty(selector) ) {
selectors.splice(i, 1);
} else {
2014-08-14 02:03:55 +02:00
injectedSelectors[selector] = true;
2014-07-02 18:02:29 +02:00
}
}
if ( selectors.length !== 0 ) {
addStyleTag(selectors);
}
};
var processHighHighGenericsAsync = function() {
if ( processHighHighGenericsTimer !== null ) {
clearTimeout(processHighHighGenericsTimer);
}
processHighHighGenericsTimer = vAPI.setTimeout(processHighHighGenerics, 300);
2014-07-02 18:02:29 +02: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 21:00:26 +02:00
var idsFromNodeList = function(nodes) {
2014-07-02 18:02:29 +02:00
if ( !nodes || !nodes.length ) {
return;
2014-06-24 00:42:43 +02:00
}
2014-07-02 18:02:29 +02:00
var qq = queriedSelectors;
var ll = lowGenericSelectors;
2014-07-20 21:00:26 +02:00
var node, v;
2014-07-02 18:02:29 +02:00
var i = nodes.length;
while ( i-- ) {
node = nodes[i];
2014-07-20 21:00:26 +02:00
if ( node.nodeType !== 1 ) { continue; }
2015-11-01 17:25:36 +01:00
v = node.id;
2014-07-20 21:00:26 +02:00
if ( typeof v !== 'string' ) { continue; }
2014-07-09 07:53:49 +02:00
v = v.trim();
2014-07-20 21:00:26 +02:00
if ( v === '' ) { continue; }
v = '#' + v;
if ( qq.hasOwnProperty(v) ) { continue; }
ll.push(v);
2014-07-20 21:00:26 +02:00
qq[v] = true;
}
};
// Extract all classes: these will be passed to the cosmetic filtering
// engine, and in return we will obtain only the relevant CSS selectors.
2015-09-04 22:30:53 +02:00
// https://github.com/gorhill/uBlock/issues/672
// http://www.w3.org/TR/2014/REC-html5-20141028/infrastructure.html#space-separated-tokens
// http://jsperf.com/enumerate-classes/6
2014-07-20 21:00:26 +02:00
var classesFromNodeList = function(nodes) {
if ( !nodes || !nodes.length ) {
return;
}
2015-09-04 22:30:53 +02:00
2014-07-20 21:00:26 +02:00
var qq = queriedSelectors;
var ll = lowGenericSelectors;
2015-09-04 22:30:53 +02:00
var v, vv, len, c, beg, end;
2014-07-20 21:00:26 +02:00
var i = nodes.length;
2015-09-04 22:30:53 +02:00
2014-07-20 21:00:26 +02:00
while ( i-- ) {
2015-09-04 22:30:53 +02:00
vv = nodes[i].className;
if ( typeof vv !== 'string' ) { continue; }
len = vv.length;
beg = 0;
for (;;) {
// Skip whitespaces
while ( beg !== len ) {
c = vv.charCodeAt(beg);
if ( c !== 0x20 && (c > 0x0D || c < 0x09) ) { break; }
beg++;
}
if ( beg === len ) { break; }
end = beg + 1;
// Skip non-whitespaces
while ( end !== len ) {
c = vv.charCodeAt(end);
if ( c === 0x20 || (c <= 0x0D && c >= 0x09) ) { break; }
end++;
}
v = '.' + vv.slice(beg, end);
if ( qq.hasOwnProperty(v) === false ) {
ll.push(v);
qq[v] = true;
}
if ( end === len ) { break; }
beg = end + 1;
2014-07-02 18:02:29 +02:00
}
2014-06-24 00:42:43 +02:00
}
2014-07-02 18:02:29 +02:00
};
2014-06-24 00:42:43 +02:00
// Start cosmetic filtering.
2015-02-14 18:16:36 +01:00
idsFromNodeList(document.querySelectorAll('[id]'));
classesFromNodeList(document.querySelectorAll('[class]'));
retrieveGenericSelectors();
2014-06-24 00:42:43 +02:00
2015-03-02 02:26:33 +01:00
//console.debug('%f: uBlock: survey time', timer.now() - tStart);
// 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.
// 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;
}
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/618
2015-03-29 18:13:28 +02:00
// Following is to observe dynamically added iframes:
// - On Firefox, the iframes fails to fire a `load` event
var ignoreTags = {
'link': true,
'script': true,
2015-03-29 18:13:28 +02:00
'style': true
};
2014-09-16 21:39:21 +02:00
// Added node lists will be cumulated here before being processed
var addedNodeLists = [];
var addedNodeListsTimer = null;
2015-10-31 05:55:10 +01:00
var removedNodeListsTimer = null;
2015-03-29 18:13:28 +02:00
var collapser = uBlockCollapser;
2014-09-16 21:39:21 +02:00
2015-10-31 05:55:10 +01:00
// The `cosmeticFiltersActivated` message is required: a new element could
// be matching an already injected but otherwise inactive cosmetic filter.
// This means the already injected cosmetic filter become active (has an
// effect on the document), and thus must be logged if needed.
var addedNodesHandler = function() {
addedNodeListsTimer = null;
2015-12-19 23:58:26 +01:00
var iNodeList = addedNodeLists.length,
nodeList, iNode, node;
while ( iNodeList-- ) {
nodeList = addedNodeLists[iNodeList];
iNode = nodeList.length;
while ( iNode-- ) {
node = nodeList[iNode];
if ( node.nodeType !== 1 ) {
continue;
}
2015-03-29 18:13:28 +02:00
if ( ignoreTags.hasOwnProperty(node.localName) ) {
continue;
}
contextNodes.push(node);
2015-03-29 18:13:28 +02:00
collapser.iframesFromNode(node);
}
}
2015-12-19 23:58:26 +01:00
addedNodeLists.length = 0;
if ( contextNodes.length !== 0 ) {
idsFromNodeList(selectNodes('[id]'));
classesFromNodeList(selectNodes('[class]'));
retrieveGenericSelectors();
messager.send({ what: 'cosmeticFiltersActivated' });
}
2014-07-02 18:02:29 +02:00
};
2015-10-31 05:55:10 +01:00
// https://github.com/gorhill/uBlock/issues/873
// This will ensure our style elements will stay in the DOM.
var removedNodesHandler = function() {
removedNodeListsTimer = null;
checkStyleTags();
};
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/205
2014-09-16 21:39:21 +02:00
// Do not handle added node directly from within mutation observer.
2015-10-31 05:55:10 +01:00
// I arbitrarily chose 100 ms for now: I have to compromise between the
// overhead of processing too few nodes too often and the delay of many
// nodes less often.
var domLayoutChanged = function(mutations) {
var removedNodeLists = false;
2014-09-16 21:39:21 +02:00
var iMutation = mutations.length;
2015-10-31 05:55:10 +01:00
var nodeList, mutation;
2014-09-16 21:39:21 +02:00
while ( iMutation-- ) {
2015-10-31 05:55:10 +01:00
mutation = mutations[iMutation];
nodeList = mutation.addedNodes;
if ( nodeList.length !== 0 ) {
2014-09-16 21:39:21 +02:00
addedNodeLists.push(nodeList);
}
2015-10-31 05:55:10 +01:00
if ( mutation.removedNodes.length !== 0 ) {
removedNodeLists = true;
}
}
if ( addedNodeLists.length !== 0 && addedNodeListsTimer === null ) {
addedNodeListsTimer = vAPI.setTimeout(addedNodesHandler, 100);
2014-09-16 21:39:21 +02:00
}
2015-10-31 05:55:10 +01:00
if ( removedNodeLists && removedNodeListsTimer === null ) {
removedNodeListsTimer = vAPI.setTimeout(removedNodesHandler, 100);
2014-09-16 21:39:21 +02:00
}
};
//console.debug('Starts cosmetic filtering\'s mutations observer');
// https://github.com/gorhill/httpswitchboard/issues/176
2015-10-31 05:55:10 +01:00
var domLayoutObserver = new MutationObserver(domLayoutChanged);
domLayoutObserver.observe(document.body, {
childList: true,
subtree: true
});
// https://github.com/gorhill/uMatrix/issues/144
2015-04-08 01:34:22 +02:00
vAPI.shutdown.add(function() {
2015-10-31 05:55:10 +01:00
domLayoutObserver.disconnect();
if ( addedNodeListsTimer !== null ) {
clearTimeout(addedNodeListsTimer);
}
});
2014-07-02 18:02:29 +02:00
})();
2014-06-24 00:42:43 +02:00
2014-08-02 17:40:27 +02:00
/******************************************************************************/
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2014-09-14 22:20:40 +02:00
// Permanent
2014-06-27 23:06:42 +02:00
2015-03-29 18:13:28 +02:00
// Listener to collapse blocked resources.
// - Future requests not blocked yet
// - Elements dynamically added to the page
// - Elements which resource URL changes
2015-02-04 06:24:12 +01:00
(function() {
var onResourceFailed = function(ev) {
//console.debug('onResourceFailed(%o)', ev);
uBlockCollapser.add(ev.target);
uBlockCollapser.process();
};
document.addEventListener('error', onResourceFailed, true);
// https://github.com/gorhill/uMatrix/issues/144
2015-04-08 01:34:22 +02:00
vAPI.shutdown.add(function() {
document.removeEventListener('error', onResourceFailed, true);
});
})();
2014-06-24 00:42:43 +02:00
/******************************************************************************/
2014-09-14 22:20:40 +02:00
/******************************************************************************/
2015-04-07 03:26:05 +02:00
// https://github.com/chrisaljoudi/uBlock/issues/7
2015-11-01 17:25:36 +01:00
// Executed only once.
// Preferring getElementsByTagName over querySelectorAll:
// http://jsperf.com/queryselectorall-vs-getelementsbytagname/145
2014-09-14 22:20:40 +02:00
(function() {
2015-03-29 18:13:28 +02:00
var collapser = uBlockCollapser;
var elems, i, elem;
elems = document.getElementsByTagName('embed');
i = elems.length;
while ( i-- ) {
collapser.add(elems[i]);
}
elems = document.getElementsByTagName('object');
i = elems.length;
while ( i-- ) {
collapser.add(elems[i]);
}
elems = document.getElementsByTagName('img');
i = elems.length;
2015-03-29 18:13:28 +02:00
while ( i-- ) {
elem = elems[i];
if ( elem.complete ) {
collapser.add(elem);
2015-02-25 02:10:10 +01:00
}
2015-03-29 18:13:28 +02:00
}
elems = document.getElementsByTagName('iframe');
i = elems.length;
while ( i-- ) {
collapser.addIFrame(elems[i]);
}
2015-03-29 18:13:28 +02:00
collapser.process(0);
2014-09-14 22:20:40 +02:00
})();
/******************************************************************************/
2014-09-28 20:38:17 +02:00
/******************************************************************************/
2015-12-12 22:13:00 +01:00
// To send mouse coordinates to main process, as the chrome API fails
2014-09-28 20:38:17 +02:00
// to provide the mouse position to context menu listeners.
2015-12-12 22:13:00 +01:00
// https://github.com/chrisaljoudi/uBlock/issues/1143
// Also, find a link under the mouse, to try to avoid confusing new tabs
// as nuisance popups.
2014-09-28 20:38:17 +02:00
// Ref.: https://developer.mozilla.org/en-US/docs/Web/Events/contextmenu
(function() {
if ( window !== window.top ) {
return;
}
var onMouseClick = function(ev) {
var elem = ev.target;
while ( elem !== null && elem.localName !== 'a' ) {
elem = elem.parentElement;
}
messager.send({
what: 'mouseClick',
x: ev.clientX,
y: ev.clientY,
url: elem !== null ? elem.href : ''
2014-09-28 20:38:17 +02:00
});
};
2015-09-10 20:01:16 +02:00
window.addEventListener('mousedown', onMouseClick, true);
// https://github.com/gorhill/uMatrix/issues/144
2015-04-08 01:34:22 +02:00
vAPI.shutdown.add(function() {
2015-09-10 20:01:16 +02:00
document.removeEventListener('mousedown', onMouseClick, true);
});
2014-09-28 20:38:17 +02:00
})();
/******************************************************************************/
2015-01-02 03:14:53 +01:00
/******************************************************************************/
})();
/******************************************************************************/