老旧浏览器API兼容
获取滚动距离
js
/**
* 兼容: 获取offsetTop/offsetLeft, 惰性函数(只会判断一次)
* @returns {Object: {top, left}}
*/
var getScrollOffset = (function () {
if (window.pageXOffset) {
return function () {
return {
top: window.pageXOffset,
left: window.pageYOffset,
};
};
}
return function () {
return {
top: document.body.scrollTop + document.documentElement.scrollTop,
left: document.body.scrollLeft + document.documentElement.scrollLeft,
};
};
})();
获取已经已滚动的宽/高度 PageX/Y
js
/**
* 兼容: 获取滚动的宽度/高度
* @returns {object: {width, height}}
*/
var getScrollSize = (function () {
if (document.body.scrollHeight) {
return function () {
return {
width: document.body.scrollWidth,
height: document.body.scrollHeight,
};
};
}
return function () {
return {
width: document.documentElement.scrollWidth,
height: document.documentElement.scrollHeight,
};
};
})();
/**
* 兼容: pageX/pageY
* @param {Event} e
* @returns {Object:{x, y}}
*/
var getPagePosition = function (e) {
var scrollPos = getScrollOffset(),
clientLeft = document.documentElement.clientLeft || 0,
clientTop = document.documentElement.clientTop || 0;
return {
x: e.clientX + scrollPos.left - clientLeft,
y: e.clientY + scrollPos.top - clientTop,
};
};
获取可视区域大小
js
/**
* 兼容: 获取可视区域大小
* @returns {Object:{width, height}}
*/
var getViewportSize = (function () {
if (window.innerWidth) {
return function () {
return {
width: window.innerWidth,
height: window.innerHeight,
};
};
} else {
return function () {
// ie
var key = document.compatMode === 'BackCompat' ? 'body' : 'documentElement';
return {
width: document[key].clientWidth,
height: document[key].clientHeight,
};
};
}
})();
获取 style 对象属性的值
js
/**
* 兼容: 获取 getComputedStyle 的值
* @returns {string|number|undefined}
*/
var getStyles = (function () {
if (window.getComputedStyle) {
return function (el, prop) {
var styles = window.getComputedStyle(el, null);
return styles[prop] ? styles[prop] : undefined;
};
}
return function (el, prop) {
// ie
var styles = el.currentStyle;
return styles[prop] ? styles[prop] : undefined;
};
})();
添加/移除事件监听函数
js
/**
* 兼容: 添加事件处理函数(惰性函数另一种实现,只会判断一次)
* @param {HTMLElement} el 事件源
* @param {String} type 事件类型
* @param {Function} handler 事件处理函数
* @returns
*/
function addEvent(el, type, handler, capture) {
if (el.addEventListener) {
addEvent = function (el, type, handler, capture) {
el.addEventListener(type, handler, capture);
};
} else if (el.attachEvent) {
addEvent = function (el, type, handler) {
el.attachEvent('on' + type, function () {
handler.call(el);
});
};
} else {
addEvent = function (el, type, handler) {
el['on' + type] = handler;
};
}
return addEvent(el, type, handler, capture);
}
/**
* 兼容: 移除事件处理函数
* @param {HTMLElement} el 事件源
* @param {String} type 事件类型
* @returns
*/
function removeEvent(el, type, handler) {
if (el.removeEventListener) {
removeEvent = function (el, type, handler) {
el.removeEventListener(type, handler);
};
} else if (el.detachEvent) {
removeEvent = function (el, type, handler) {
el.detachEvent('on' + type, handler);
};
} else {
removeEvent = function (el, type, _handler) {
el['on' + type] = null;
};
}
return removeEvent(el, type, handler); // 只有这一次执行会判断
}
取消冒泡/浏览器默认行为
js
/**
* 兼容: 取消冒泡
* @param {Event} e
* @returns
*/
function cancelBubble(e) {
if (e.stopPropagation) {
cancelBubble = function (e) {
e = e || window.event;
return e.stopPropagation();
};
} else {
cancelBubble = function (e) {
e = e || window.event; // window.event 兼容ie8及其以下版本的浏览器
e.cancelBubble = true;
};
}
return cancelBubble(e);
}
/**
* 兼容: 取消浏览器默认事件
* @param {Event} e
* @returns
*/
function preventDefault(e) {
var e = e || window.event;
if (e.preventDefault) {
// w3c standard
preventDefault = function (e) {
e.preventDefault(e);
};
} else {
// ie: 9/8/7/6
preventDefault = function (e) {
e = e || widow.event;
e.returnValue = false;
};
}
return preventDefault(e);
}
圣杯模式继承
兼容老旧浏览器, 无法使用 class 语法, 一个继承写起来太费劲了
js
var inherit = (function () {
function Middleman() {}
return function (child, parent) {
Middleman.prototype = parent.prototype;
child.prototype = new Middleman();
child.prototype.constructor = child;
child.super_class = parent;
};
})();