Kairyou's Blog

专注于前端开发, 追求更好的用户体验, 更好的开发体验 [长沙前端QQ群:234746733]
  • javascript 实现 css 的 nth-child选择符

    / 分类: 开发,实践 / No Comments

    最近精简了JQuery,精简到33KB,一些功能肯定被K掉了。保留了常用的:
    ready, attr, data, css, html, text, val, empty, append, siblings, wrap, trigger, bind, unbind, delegate, mouseenter, mouseleave, anim, stop, ajax, jsonp等等。
    除了attr,data,css用了JQ的,并做了些改动,其他的基本都没在JQ的源码上精简,算个山寨货?。。。
    功能肯定没JQ强大,但是文件大小从JQ1.6的80K到30K,用在一些不是特别复杂的应用上还是不错的。
    语法也是用了链式操作,因为一些方法没参考JQ源码,所以有些语法和JQ不一样了。JQ的一些插件也不能直接用了,不过改改插件,基本上还是可以用的。

    因为选择器没采用用sizzle,所以没法用一些高级的选择符,比如odd,even,nth-child,这些就需要另外写方法实现了。
    odd,even的实现比较省事,取奇偶就可以了,i % 2的余数是0还是1就OK。

    nth-child的实现费了一些力,不仅要匹配允许的选择符,还要返回符合规范的结果。
    如果不知道CSS的nth-child选择符可以自己Search一下,偶就不介绍了。
    一些例子:nth-child(2n+1), nth-child(2), nth-child(3n), nth-child(n+2),nth-child(-n+2)

    代码如下:

    function nthChild(nodeList, selector ) {
    	var ret = [],
    		reg = /(-?\d*)[n]*([+-]\d+)*/,
    		m = selector.match(reg);
    	if (selector === m[1]) { // nth-child:(2) - 纯数字,直接返回
    		return [nodeList[(parseInt(m[1])-1)]];
    	}
    	var filter = function(i){
    		++i; //nth从1开始, elem从0开始,elem的index要+1
    		if (m[2]) { // 类似:nth-child:(-2n-1) / nth-child(n+2)
    			if ('' === m[1]){ // nth-child(n+2)
    				m[1] = 1;
    			}else if ('-' === m[1]) { // nth-child(-n+2)
    				m[1] = -1;
    			}// else nth-child(-2n-1) / nth-child(2n+10)
    		}else { // nth-child:(2n)
    			m[2] = 0;
    		}
    		var n = (i-parseInt(m[2])) / parseInt(m[1]);
    		// 正整数返回true
    		return ( n === parseInt(n) && n >= 0) ? true : false;
    	}
    	each(nodeList, function(i){ //each 方法需要单独写
    		if (filter(i)) {
    			ret.push(this)
    		}
    	});
    	return ret;
    }
    

    实现nth-child的思路:

    nth-child的公式为:nth-child(x*n+y);
    n从0开始++,(x*n+y)要>0
    x *n + y = i; // n,为自然数,从0开始; i为要取的第几个元素(正整数)
    上面得到:n = (i-y) / x ; 整除,余0
    那么,当i满足:(i-y) % x === 0 && (i-y) / x >= 0,就是符合条件的了

    使用:

    function each( obj, fn ) {
    	var i = 0, len = obj.length, val;
    	for (; i < len; i++) {
    		val = fn.call(obj[i], i, obj[i]);
    		if ( val === false ) break;
    	}
    }
    var list = nthChild(document.getElementsByTagName("li"), '2n+1'); //奇数行
    //使用FF/CM/OP/IE9等支持nth-child选择符的浏览器,得到NodeList的结果,可以用来与nthChild的结果比较
    	//比如:document.querySelectorAll('li:nth-child(2n+1)')
    each(list, function(){ this.className = 'selected'; }); //给奇数行设置class=selected
    
  • 区分IE8 、IE9 的专属css hack

    / 分类: 开发 / 8 Comments

    一般来说,我们写的结构比较好的时候,IE8/9下是没区别的.所以可能很少人关注只有IE8或只有IE9才识别的css hack.

    因为IE8及以下版本是不支持CSS3的,但是我们如果使用css3,在IE下IE9正常渲染,但我们又想让IE8及以下的浏览器实现同样的效果,且不希望使用css3pie或htc或条件注释等方法时,可能就会需要用到IE8和IE9的专属css hack了.

    .test{ /* 1. */
    	/* color:#09F\0; 以前是IE8/9, 现在10也支持 */
    	color:#09F\0/; /* 以前是IE8 only, 现在IE9/10也支持. 如要排除IE9需要使用下面的rule重设IE9样式 */
    }
    @media all and (min-width:0) { /* 2. */
        .test{color:red\9; }/* IE9 only, 现在IE10也支持 */
        /* Ps:老外的方法都是\0,根本没考虑Opera */
    }
    @media screen and (-ms-high-contrast: active), (-ms-high-contrast: none) { /* 3. */
       .test { color: green; } /* IE10+ */
    }
    :root .test { color:#963\9; } /* 以前IE9 only, 现在10也支持, 优先级高于@media, 优先级太高, 尽量少用 */
    

    注: IE老不按规矩, 以前IE8 only/IE9 only的hack, IE10 一出现都乱套了~
    为了避免看此文的同学失望而归, 从新又整理了一下内容(2013-01-28), 如有错误望指出.
    现在, 要想写出专门针对ie8或ie9的hack, 把1/2/3种组合在一起使用. 如果可以, 使用CSS条件注释更靠谱.

    此hack已经加入到我08年写的那个用css 写的一个浏览器检测中。