uBlock/js/ublock.js

257 lines
7.3 KiB
JavaScript
Raw Normal View History

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-08-02 09:40:27 -06:00
/* global µBlock */
2014-06-23 16:42:43 -06:00
/******************************************************************************/
2014-08-02 09:40:27 -06:00
µBlock.getNetFilteringSwitch = function(url, domain) {
var keyHostname = this.URI.hostnameFromURI(url);
var pos = url.indexOf('#');
var keyURL = pos !== -1 ? url.slice(0, pos) : url;
// The caller may provide an already known domain -- convenient to reduce
// overhead of extracting a domain from the url
if ( domain === undefined ) {
domain = this.URI.domainFromHostname(keyHostname);
}
if ( !domain ) {
2014-06-23 16:42:43 -06:00
return false;
}
2014-08-02 09:40:27 -06:00
var exceptions = this.netWhitelist[domain];
if ( !exceptions ) {
return true;
}
var i = exceptions.length;
var exception;
while ( i-- ) {
exception = exceptions[i];
if ( exception.indexOf('/') !== -1 ) {
if ( exception.slice(-1) === '*' ) {
exception = exception.slice(0, -1);
if ( keyURL.slice(0, exception.length) === exception ) {
return false;
}
} else if ( keyURL === exception ) {
return false;
}
} else if ( keyHostname.slice(-exception.length) === exception ) {
2014-06-23 16:42:43 -06:00
return false;
}
}
return true;
};
/******************************************************************************/
2014-08-02 09:40:27 -06:00
µBlock.toggleNetFilteringSwitch = function(url, scope, newState) {
var keyHostname = this.URI.hostnameFromURI(url);
var pos = url.indexOf('#');
var keyURL = pos !== -1 ? url.slice(0, pos) : url;
var key = scope === 'page' ? keyURL : keyHostname;
// The caller may provide an already known domain -- convenient to reduce
// overhead of extracting a domain from `key`
var domain = this.URI.domainFromHostname(keyHostname);
if ( !domain ) {
return false;
}
var currentState = this.getNetFilteringSwitch(url, domain);
2014-06-23 16:42:43 -06:00
if ( newState === undefined ) {
newState = !currentState;
}
if ( newState === currentState ) {
return currentState;
}
2014-08-02 09:40:27 -06:00
var netWhitelist = this.netWhitelist;
var exceptions = netWhitelist[domain];
if ( !exceptions ) {
exceptions = netWhitelist[domain] = [];
}
2014-06-23 16:42:43 -06:00
// Add to exception list
if ( !newState ) {
2014-08-02 09:40:27 -06:00
exceptions.push(key);
this.saveWhitelist();
2014-06-23 16:42:43 -06:00
return true;
}
// Remove from exception list
2014-08-02 09:40:27 -06:00
var i = exceptions.length;
var exception;
while ( i-- ) {
exception = exceptions[i];
if ( exception.indexOf('/') !== -1 ) {
if ( exception.slice(-1) === '*' ) {
exception = exception.slice(0, -1);
if ( keyURL.slice(0, exception.length) === exception ) {
exceptions.splice(i, 1);
}
} else if ( keyURL === exception ) {
exceptions.splice(i, 1);
}
} else if ( keyHostname.slice(-exception.length) === exception ) {
exceptions.splice(i, 1);
2014-06-23 16:42:43 -06:00
}
}
2014-08-02 09:40:27 -06:00
if ( exceptions.length === 0 ) {
delete netWhitelist[domain];
}
this.saveWhitelist();
return true;
2014-06-23 16:42:43 -06:00
};
/******************************************************************************/
2014-06-23 17:23:36 -06:00
// For now we will use the net exception list
2014-08-02 09:40:27 -06:00
µBlock.getCosmeticFilteringSwitch = function(url, domain) {
return this.getNetFilteringSwitch(url, domain);
};
/******************************************************************************/
µBlock.stringFromWhitelist = function(exceptions) {
var r = {};
var i, bucket;
for ( var domain in exceptions ) {
if ( exceptions.hasOwnProperty(domain) === false ) {
continue;
}
bucket = exceptions[domain];
for ( i = 0; i < bucket.length; i++ ) {
r[bucket[i]] = true;
2014-06-23 17:23:36 -06:00
}
}
2014-08-02 09:40:27 -06:00
return Object.keys(r).sort(function(a,b){return a.localeCompare(b);}).join('\n');
2014-06-23 17:23:36 -06:00
};
/******************************************************************************/
2014-08-02 09:40:27 -06:00
µBlock.whitelistFromString = function(s) {
var exceptions = {};
var lines = s.split(/[\n\r]+/);
var line, domain, bucket;
for ( var i = 0; i < lines.length; i++ ) {
line = lines[i].trim();
domain = line.indexOf('/') !== -1 ?
this.URI.domainFromURI(line) :
this.URI.domainFromHostname(line);
if ( !domain ) {
continue;
}
bucket = exceptions[domain];
if ( bucket === undefined ) {
exceptions[domain] = [line];
} else {
bucket.push(line);
}
}
return exceptions;
2014-06-23 16:42:43 -06:00
};
/******************************************************************************/
2014-06-24 16:29:55 -06:00
// Return all settings if none specified.
2014-06-23 16:42:43 -06:00
µBlock.changeUserSettings = function(name, value) {
2014-06-24 16:29:55 -06:00
var µb = µBlock;
if ( name === undefined ) {
return µb.userSettings;
}
2014-06-23 16:42:43 -06:00
if ( typeof name !== 'string' || name === '' ) {
return;
}
// Do not allow an unknown user setting to be created
if ( µb.userSettings[name] === undefined ) {
return;
}
if ( value === undefined ) {
return µb.userSettings[name];
}
// Pre-change
switch ( name ) {
default:
break;
2014-06-23 16:42:43 -06:00
}
// Change
µb.userSettings[name] = value;
// Post-change
switch ( name ) {
default:
break;
2014-06-23 16:42:43 -06:00
}
µb.saveUserSettings();
};
/******************************************************************************/
µBlock.transposeType = function(type, path) {
if ( type === 'other' ) {
var pos = path.lastIndexOf('.');
if ( pos > 0 ) {
var ext = path.slice(pos);
if ( '.eot.ttf.otf.svg.woff'.indexOf(ext) >= 0 ) {
return 'stylesheet';
}
if ( '.ico.png'.indexOf(ext) >= 0 ) {
return 'image';
}
}
}
return type;
};
/******************************************************************************/
µBlock.formatCount = function(count) {
if ( typeof count !== 'number' ) {
return '';
}
var s = count.toFixed(0);
if ( count >= 1000 ) {
if ( count < 10000 ) {
s = '>' + s.slice(0,1) + 'K';
} else if ( count < 100000 ) {
s = s.slice(0,2) + 'K';
} else if ( count < 1000000 ) {
s = s.slice(0,3) + 'K';
} else if ( count < 10000000 ) {
s = s.slice(0,1) + 'M';
} else {
s = s.slice(0,-6) + 'M';
}
}
return s;
};