新聞中心
技術文章寫得少,所以有時候想寫點什么卻下不了手,不知道該寫什么;往往到了準備要寫的時候才發(fā)現自己想寫的東西其實很無聊,甚至覺得很幼稚,于是又關掉了編緝器,呵呵。

創(chuàng)新互聯專注于企業(yè)成都全網營銷推廣、網站重做改版、黃山區(qū)網站定制設計、自適應品牌網站建設、成都h5網站建設、商城網站定制開發(fā)、集團公司官網建設、成都外貿網站制作、高端網站制作、響應式網頁設計等建站業(yè)務,價格優(yōu)惠性價比高,為黃山區(qū)等各大城市提供網站開發(fā)制作服務。
今天周五,很閑,坐在電腦前沒什么事可做,產品線的人也沒提什么新的需求,可能下周會有新的需求和工作安排,但那是下周的事了。今天就想寫點技術的東西,也就當作是記記筆記,本人水平有限,希望大家多多指教,嘴下留情,哈哈。
有時候我們會想擴展DOM元素的功能,可以添加一些自定義的方法,以讓它用起來更加靈活、方便;先來舉個例子:
DOM功能擴展 - 你好
毫無疑問,從以上代碼可以看出,當點擊A標簽的時候會彈出“你好”,tagA是一個DOM元素,除了有onclick事件以外,還有onmouseover,onmouseout,onmousemove等等,而這些事件都是DOM元素本身就具有的;但現在我們希望對它進行擴展,例如可以讓它支持tagA.bind,我可以用tagA.bind("click",function(){}),來代替tagA.onclick=function(){}。OK,現在的目的很明確,先看下面的代碼:
DOM功能擴展 - 你好
以上這段代碼就是功能擴展后的最終效果,它與上一段代碼實現的功能是一樣的,但現在它還不能執(zhí)行,要進行擴展后才可以,在此之前先來看一些基礎知識,這很重要,因為等下會用到:
1、HTMLElement,在DOM標準中,每個元素都繼承自HTMLElement,而HTMLElement又繼承自Element,Element又繼承自Node;于是我們可以使用HTMLElement的Prototype來擴展HTML元素的方法和屬性,如何實現?我們來看一段代碼:
DOM功能擴展 - 你好
以上代碼在頁面加載的時候就彈出“這是一個擴展方法”,不過相信你已經注意到了,在IE6,7,8里面會出錯,但在IE9以上或者Chrome,Firefox,Opera這些瀏覽器里面都能正常運行,這是兼容性問題,不用擔心,后面會解決。
以上的代碼靈活性不夠好,我們優(yōu)化一下,讓它更加靈活:
DOM功能擴展 - 你好
從以上代碼可以看出,有了DOMExtend這個方法以后,我們就可以通過傳入不用的name 和 fn 實現不同的擴展。
2、以上講完了HTMLElement,接下來講講事件的綁定,很多人都知道,IE和其他瀏覽器的事件綁定方式不一樣,實現兼容所有瀏覽器的事件綁定的代碼如下:
- function BindEvent(elem,event,fun){
- if(elem.addEventListener){
- elem.addEventListener(event,fun,false);
- }
- else{
- elem.attachEvent("on"+event,fun);
- }
- }
以下是事件綁定的使用例子:
DOM功能擴展 - 你好
以上代碼運行后,點擊“你好”就會彈出“這是事件綁定”,這里值得一提的就是addEvenListener的第三個參數,這里的值是false,意思是取消Capture方式而采用冒泡方式。標準的事件有兩種觸發(fā)方式,一種是捕獲型(caputre),另一種是冒泡型;而IE只支持冒泡型。捕獲型的特點是觸發(fā)方式是從外到內的方式觸發(fā)事件,而冒泡型就是從內到外的方式觸發(fā)事件,假設以上代碼的A元素外層包了一個DIV元素,如果A元素與它的父元素DIV都有一個onclick事件,那么冒泡型就是點擊A的時候會先觸發(fā)A的事件,然后再觸發(fā)DIV的事件,反之就是捕獲型。
OK,相信通過以上的分析,對HTMLElement擴展和事件綁定都有了相當的了解,結合這兩個知識點,我們可以寫出如下的代碼:
DOM功能擴展 - 你好
執(zhí)行以上這個頁面,在IE9,Chrome,Opera,Firefox等標準瀏覽器里都能正常觸發(fā)tagA的點擊事件,于是現在只剩下一個問題,就是要兼容其他瀏覽器;IE瀏覽器之所以出錯,是因為它們隱藏了對HTMLElement的訪問,于是針對IE瀏覽器,我們就不能用HTMLElement.prototype來進行擴展了,但我們可以通過重寫以下幾個函數來達到目的:
document.getElementById
document.getElementsByTagName
document.createElement
document.documentElement
document.body
(PS:記憶中獲取DOM元素好像就是以上這些方法了~不知道還有沒有其他)
重寫后,再進行一些處理變換就可以得到以下完整的頁面代碼:
DOM功能擴展 - 你好
- function DOMExtend(name, fn){
- if(typeof(HTMLElement)!="undefined"){
- eval("HTMLElement.prototype."+name+"="+fn);
- }
- else{
- var _getElementById=document.getElementById;
- document.getElementById=function(id){
- var _elem=_getElementById(id);
- eval("_elem."+name+"="+fn);
- return _elem;
- }
- var _getElementByTagName=document.getElementsByTagName;
- document.getElementsByTagName=function(tag){
- var _elem=_getElementByTagName(tag);
- var len=_elem.length;
- for(var i=0;i
- eval("_elem["+i+"]."+name+"="+fn);
- }
- return _elem;
- }
- var _createElement=document.createElement;
- document.createElement=function(tag){
- var _elem=_createElement(tag);
- eval("_elem."+name+"="+fn);
- return _elem;
- }
- var _documentElement=document.documentElement;
- eval("_documentElement."+name+"="+fn);
- var _documentBody=document.body;
- eval("_documentBody."+name+"="+fn);
- }
- }
- function BindEvent(event,fun){
- if(this.addEventListener){
- this.addEventListener(event,fun,false);
- }
- else{
- this.attachEvent("on"+event,fun);
- }
- }
- DOMExtend("bind",BindEvent);var wrap=document.getElementById("tagA");
- wrap.bind("click",function(){
- alert(this.innerHTML);
- })
OK,目前為止已經解決了兼容性問題,這是所有瀏覽器都能順利通過的DOM元素擴展的代碼,但是這樣還有一個小問題,細心的人會發(fā)現在IE瀏覽器里面彈出的結果是"undefined",而不是"你好";問題的原因在于IE的事件綁定上,看以上代碼,當調用alert(this.innerHTML)的時候,由于IE綁定事件用的是attachEvent,這時候this指向的是windows,于是現在的目標的要改變this指向的對像,將this指向tagA。于是經過修改,完整代碼如下:
DOM功能擴展 - 你好
- function DOMExtend(name, fn){
- if(typeof(HTMLElement)!="undefined"){
- eval("HTMLElement.prototype."+name+"="+fn);
- }
- else{
- var _getElementById=document.getElementById;
- document.getElementById=function(id){
- var _elem=_getElementById(id);
- eval("_elem."+name+"="+fn);
- return _elem;
- }
- var _getElementByTagName=document.getElementsByTagName;
- document.getElementsByTagName=function(tag){
- var _elem=_getElementByTagName(tag);
- var len=_elem.length;
- for(var i=0;i
- eval("_elem["+i+"]."+name+"="+fn);
- }
- return _elem;
- }
- var _createElement=document.createElement;
- document.createElement=function(tag){
- var _elem=_createElement(tag);
- eval("_elem."+name+"="+fn);
- return _elem;
- }
- var _documentElement=document.documentElement;
- eval("_documentElement."+name+"="+fn);
- var _documentBody=document.body;
- eval("_documentBody."+name+"="+fn);
- }
- }
- function BindEvent(event,fun){
- if(this.addEventListener){
- this.addEventListener(event,fun,false);
- }
- else{
- var tag=this;
- tag.attachEvent("on"+event,function(){
- fun.apply(tag,arguments);//這里是關鍵
- });
- }
- }
- DOMExtend("bind",BindEvent);var wrap=document.getElementById("tagA");
- wrap.bind("click",function(){
- alert(this.innerHTML);
- })
分享標題:兼容所有瀏覽器的DOM擴展功能
分享網址:http://fisionsoft.com.cn/article/dhgdeee.html


咨詢
建站咨詢
