javascript十個最常用的自定義函數


如果不使用類庫或者沒有自己的類庫,儲備一些常用函數總是有好處的。

(10)addEvent

網上最流行的版本是Scott Andrew的,據說javascript界曾舉行一場比賽(此事件我們可以在Pro Javascript Techniques第100頁看到)或瀏覽PPK的網站,征求添加事件與移除事件的函數,他就是其獲獎者。下面就是他的實現:

  function addEvent(elm, evType, fn, useCapture) {
    if (elm.addEventListener) {
      elm.addEventListener(evType, fn, useCapture);//DOM2.0
      return true;
    }
    else if (elm.attachEvent) {
      var r = elm.attachEvent('on' + evType, fn);//IE5+
      return r;
    }
    else {
      elm['on' + evType] = fn;//DOM 0
    }
  }

下面是Dean Edwards 的版本

 // addEvent/removeEvent written by Dean Edwards, 2005
// with input from Tino Zijdel
// http://dean.edwards.name/weblog/2005/10/add-event/
function addEvent(element, type, handler) {
    //為每一個事件處理函數分派一個唯一的ID
    if (!handler.$$guid) handler.$$guid = addEvent.guid++;
    //為元素的事件類型創建一個哈希表
    if (!element.events) element.events = {};
    //為每一個"元素/事件"對創建一個事件處理程序的哈希表
    var handlers = element.events[type];
    if (!handlers) {
        handlers = element.events[type] = {};
        //存儲存在的事件處理函數(如果有)
        if (element["on" + type]) {
            handlers[0] = element["on" + type];
        }
    }
    //將事件處理函數存入哈希表
    handlers[handler.$$guid] = handler;
    //指派一個全局的事件處理函數來做所有的工作
    element["on" + type] = handleEvent;
};
//用來創建唯一的ID的計數器
addEvent.guid = 1;
function removeEvent(element, type, handler) {
    //從哈希表中刪除事件處理函數
    if (element.events && element.events[type]) {
        delete element.events[type][handler.$$guid];
    }
};
function handleEvent(event) {
    var returnValue = true;
    //抓獲事件對象(IE使用全局事件對象)
    event = event || fixEvent(window.event);
    //取得事件處理函數的哈希表的引用
    var handlers = this.events[event.type];
    //執行每一個處理函數
    for (var i in handlers) {
        this.$$handleEvent = handlers[i];
        if (this.$$handleEvent(event) === false) {
            returnValue = false;
        }
    }
    return returnValue;
};
//為IE的事件對象添加一些“缺失的”函數
function fixEvent(event) {
    //添加標准的W3C方法
    event.preventDefault = fixEvent.preventDefault;
    event.stopPropagation = fixEvent.stopPropagation;
    return event;
};
fixEvent.preventDefault = function() {
    this.returnValue = false;
};
fixEvent.stopPropagation = function() {
    this.cancelBubble = true;
};

功能非常強悍,解決IE的this指向問題,event總是作為第一個參數傳入,跨瀏覽器就更不在話下。

另,我還珍藏了一個HTML5工作組的版本:

var addEvent=(function(){
    if(document.addEventListener){
        return function(el,type,fn){
            if(el.length){
                for(var i=0;i<el.length;i++){
                    addEvent(el[i],type,fn);
                }
            }else{
                el.addEventListener(type,fn,false);
            }
        };
    }else{
        return function(el,type,fn){
            if(el.length){
                for(var i=0;i<el.length;i++){
                    addEvent(el[i],type,fn);
                }
            }else{
                el.attachEvent('on'+type,function(){
                    return fn.call(el,window.event);
                });
            }
        };
    }
})();

(9)addLoadEvent()

我以前討論過這函數,不細說,就是慢了一點,各大類庫基本無視它,自行實現domReady版本。下面是Simon Willison 的實現:

  var addLoadEvent = function(fn) {
    var oldonload = window.onload;
    if (typeof window.onload != 'function') {
      window.onload = fn;
    }else {
      window.onload = function() {
        oldonload();
        fn();
      }
    }
  }

(8) getElementsByClass()

我有收集癖,手頭上擁有許多版本,最后集思廣益自己實現了一個。下面是我的實現:

var getElementsByClassName = function (searchClass, node,tag) {
    if(document.getElementsByClassName){
        return  document.getElementsByClassName(searchClass)
    }else{
        node = node || document;
        tag = tag || "*";
        var classes = searchClass.split(" "),
        elements = (tag === "*" && node.all)? node.all : node.getElementsByTagName(tag),
        patterns = [],
        returnElements = [],
        current,
        match;
        var i = classes.length;
        while(--i >= 0){
            patterns.push(new RegExp("(^|\\s)" + classes[i] + "(\\s|$)"));
        }
        var j = elements.length;
        while(--j >= 0){
            current = elements[j];
            match = false;
            for(var k=0, kl=patterns.length; k<kl; k++){
                match = patterns[k].test(current.className);
                if (!match)  break;
            }
            if (match)  returnElements.push(current);
        }
        return returnElements;
    }
}

(7)cssQuery()

別名為getElementsBySeletor,由Dean Edwards最先實現,Prototype.js,JQuery等類庫都有相應實現,其中JQuery把它整合到$()選擇器中,名聲蓋過其前輩。不過IE8等新銳瀏覽器已經實現querySelector與querySelectorAll方法,待到IE6與IE7報廢之日,它就無用了。無憂里有它的實現原理講解。由於太長,就不粘出來了,具體可到原作者網站看看。

(6)toggle()

用來顯示或隱藏一個DOM元素。

function toggle(obj) {
	var el = document.getElementById(obj);
	if ( el.style.display != 'none' ) {
		el.style.display = 'none';
	}
	else {
		el.style.display = '';
	}
}

(5)insertAfter()

DOM只提供了insertBefore,我們很有必要自己實現insertAfter。不過我認為 insertAdjacentElement是更好的選擇,現在除了火狐其他瀏覽器都實現這個方法。下面是Jeremy Keith的版本:

function insertAfter(parent, node, referenceNode) {
	parent.insertBefore(node, referenceNode.nextSibling);
}

(4)inArray()

用於判斷檢查數組中是否存在某個值,下面方法取自Prototype類庫。

Array.prototype.inArray = function (value) {
	for (var i=0,l = this.length ; i <l ; i++) {
		if (this[i] === value) {
			return true;
		}
	}
	return false;
};

另一個版本:

var inArray = function (arr,value) {
	for (var i=0,l = arr.length ; i <l ; i++) {
		if (arr[i] === value) {
			return true;
		}
	}
	return false;
};

(3) getCookie(), setCookie(), deleteCookie()

做BBS與商業網站的應該經常用到,無理由每次都要讓用戶輸入密碼登錄吧。我們需要借助cookie實現自動登錄功能。

function getCookie( name ) {
	var start = document.cookie.indexOf( name + "=" );
	var len = start + name.length + 1;
	if ( ( !start ) && ( name != document.cookie.substring( 0, name.length ) ) ) {
		return null;
	}
	if ( start == -1 ) return null;
	var end = document.cookie.indexOf( ';', len );
	if ( end == -1 ) end = document.cookie.length;
	return unescape( document.cookie.substring( len, end ) );
}
function setCookie( name, value, expires, path, domain, secure ) {
	var today = new Date();
	today.setTime( today.getTime() );
	if ( expires ) {
		expires = expires * 1000 * 60 * 60 * 24;
	}
	var expires_date = new Date( today.getTime() + (expires) );
	document.cookie = name+'='+escape( value ) +
		( ( expires ) ? ';expires='+expires_date.toGMTString() : '' ) + //expires.toGMTString()
		( ( path ) ? ';path=' + path : '' ) +
		( ( domain ) ? ';domain=' + domain : '' ) +
		( ( secure ) ? ';secure' : '' );
}
function deleteCookie( name, path, domain ) {
	if ( getCookie( name ) ) document.cookie = name + '=' +
			( ( path ) ? ';path=' + path : '') +
			( ( domain ) ? ';domain=' + domain : '' ) +
			';expires=Thu, 01-Jan-1970 00:00:01 GMT';
}

(2)getStyle()與setStyle()

所有UI控件都應該存在的函數,動態設置樣式與獲取樣式。這個可以寫得很短,也可以寫得很長,但要精確取得樣式,一個字:難!但我發現許多問題都是發端於IE,微軟的開發人員好像從來不打算給出getComputedStyle這樣的函數,與之相近的currentStyle會返回auto,inhert, ' '等讓你哭笑不得的值,這還沒有算上IE怪癖模式帶來的難度呢!各類庫的實現是非常長與難分離出來的,下面是我實現的版本:

 function setStyle(el,prop,value){
    if(prop == "opacity" && !+"\v1"){
      //IE7 bug:filter 濾鏡要求 hasLayout=true 方可執行(否則沒有效果)
      if (!el.currentStyle || !el.currentStyle.hasLayout) el.style.zoom = 1;
      prop = "filter";
      if(!!window.XDomainRequest){
        value ="progid:DXImageTransform.Microsoft.Alpha(style=0,opacity="+value*100+")";
      }else{
        value ="alpha(opacity="+value*100+")"
      }
    }
    el.style.cssText += ';' + (prop+":"+value);
  }

  function getStyle(el, style){
    if(!+"\v1"){
      style = style.replace(/\-(\w)/g, function(all, letter){
        return letter.toUpperCase();
      });
      return el.currentStyle[style];
    }else{
      return document.defaultView.getComputedStyle(el, null).getPropertyValue(style)
    }
  }

有關setStyle還可以看我另一篇博文,畢竟現在設置的樣式都是內聯樣式,與html混雜在一起。

(1)$()

實至名歸,最值錢的函數,可以節省多少流量啊。最先由Prototype.js實現的,那是洪荒時代遺留下來的珍獸,現在有許多變種。

function $() {
	var elements = [];
	for (var i = 0; i < arguments.length; i++) {
		var element = arguments[i];
		if (typeof element == 'string')
			element = document.getElementById(element);
		if (arguments.length == 1)
			return element;
		elements.push(element);
	}
	return elements;
}

注意!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系我们删除。



 
粤ICP备14056181号  © 2014-2021 ITdaan.com