辛辛苦苦写了一个网站,可是用手机浏览器打开以后,在bottom的位置常常有广告显示,这是因为在wifi被dns劫持的状况下导入的广告脚本种子加载出的。
javascript
知道病情才能对症下药,广告脚本种子基本上是动态建立script来加载脚本的:css
var ele;
if (/\.css[^\.]*$/.test(asset.url)) {
ele = doc.createElement("link");
ele.type = "text/" + (asset.type || "css");
ele.rel = "stylesheet";
ele.href = asset.url
}
else {
ele = doc.createElement("script");
ele.type = "text/" + (asset.type || "javascript");
ele.src = asset.url
}
没错,document.createElement
java
/** 1. 禁用动态添加脚本,防止广告加载 */
(function () {
var createElement = document.createElement;
document.createElement = function (tag) {
switch (tag) {
case 'script':
console.log('禁用动态添加脚本,防止广告加载');
break;
default:
return createElement.apply(this, arguments);
}
}
})();
可能会误伤友军,好比要加载百度的脚本等,当本身须要用到这个时,能够考虑监控script.src的值的变化来过滤脚本,可是经过定义各类setter规则来监听script都会报错,因此只能换一种思路:在script被添加前作处理。web
/** * 禁用动态添加脚本,防止广告加载 * * @param valid bool? true(valid)|false(invalid)|other(off) * @param rule array 配置容许(valid)|不容许(invalid)的脚本规则:支持regex、string、function */
(function(valid, rule) {
if(typeof Element === 'undefined') console.log('IE8如下浏览器无效');
var origin = new RegExp('^' + location.origin),Ele = Element;
each(['appendChild', 'insertBefore', 'insertAfter'], proxy);
function proxy(prop) {
var proxy_obj = Ele.prototype[prop];
Ele.prototype[prop] = function(elem) {
if (!elem.children.length) {
var tag = elem.tagName.toLowerCase();
if (tag == 'script' && isBanScript(elem)) {
console.log('禁用脚本:' + elem.src);
var substitute = document.createElement('script');
substitute.innerHTML = '// 禁用脚本:' + elem.src;
elem = substitute;
}
}
return proxy_obj.apply(this, arguments);
};
}
function isBanScript(script) {
if (origin.test(script.src)) return false;
return valid === each(rule, match);
function match(val) {
var type = typeof val;
if (type === 'string') {
if (script.src == val) return true;
} else if (type === 'function') {
if (val(script)) return true;
} else {
if (val.test(script.src)) return true;
}
return false;
}
}
function each(arr, fn) {
if (arr) {
for (var i = 0, n = arr.length; i < n; i++) {
if (fn.call(arr[i], arr[i], i) === true) return false;
}
}
return true;
}
})(true, []);
//表示有效的脚本规则列表