.. _js: *************** js *************** 技巧 ============================= checkbox半选状态:: indeterminate = true; element.attributes会返回所有属性的数组,每个属性都是一个key为name,value的object location.replace设置当前文档,并在history中移除,防止后退:: location.replace('about:blank') 空白页 location.replace('javascript:""') url不变,页面空白 checkbox使用鼠标选中时,在标签中不会添加checked属性,因此[checked]选择是无效的,可以直接使用node.checked来判断是否选中 is-disabled时禁止点击事件:: $(function() { $('#toggle').click(function() { if ($('.btn').hasClass('is-disabled')) { $('.btn').removeClass('is-disabled'); } else { $('.btn').addClass('is-disabled'); } }); $('.btn').click(function(e) { if ($(this).hasClass('is-disabled')) { e.stopImmediatePropagation(); } }); $('.btn').click(function() { console.log('some'); }); }); window.undefined可能会被覆盖,所以void 0才是真正的undefined,而且字符更少 使用concat来实现shallow flatten:: Array.prototype.concat.apply(Array.prototype,[[2,5],3]) //[2,5,3] var ctor = function(){}; ctor为constructor构造函数的缩写 IE中无法解析 Date.parse('2013-3-28 15:05:38') 必须为 Date.parse('2013/3/28 15:05:38') 对象为引用赋值,原对象改变新对象也跟着改变,但是如果原对象重新赋值,新对象不变:: var a = { name: 'tom' }; var b = a; console.log(b.name); //tom a.name = 'jerry'; console.log(b.name); //jerry a = null; console.log(b.name); //jerry document.activeElement用来管理DOM焦点,获得DOM中当前获得焦点的元素。 元素获得焦点的方式有页面加载、用户输入(通常是通过按Tab键)和在代码中调用focus()方法 继承 -------- :: function Animal() {} function Dog() {} Chrome下:: // 要让 Dog 继承 Animal, 只需: Dog.prototype.__proto__ = Animal.prototype; // 实例化后 var dog = new Dog(); // dog.__proto__ 指向 Dog.prototype // dog.__proto__.__proto__ 指向 Animal.prototype // 原型链已成功建立起来,而且很清晰 IE下只能:: Dog.prototype=new Animate; Arale做法:: function Ctor() { } // See: http://jsperf.com/object-create-vs-new-ctor var createProto = Object.__proto__ ? function(proto) { return { __proto__: proto } } : function(proto) { Ctor.prototype = proto return new Ctor() } Dog.prototype=createProto(Animate.prototype); 下面的a赋值会将a声明提前,但是赋值没有提前,因此undefined++会输出NaN. function a(){}与var a=function(){}不同,没有重新给a赋值:: function fn() { return a++; } function fn2() { var a = 1; return fn(); } //NaN console.log(fn2()); var a = 3; //number console.log(typeof a); function a() { console.log('d'); } //number console.log(typeof a); onclick事件中,如果return false会阻止默认事件发生:: eleBtn.onclick = function() { return false; }; 变量声明会自动提前到作用域的顶部,但是赋值不会提前:: if (!("a" in window)) { var a = 1; } alert(a); 相当于:: var a; if (!("a" in window)) { a = 1; } alert(a); 因此答案为undefined 函数声明:: function functionName(arg1, arg2){ //函数体 } 函数表达式:: b = function a(x) {} 函数声明会提前,函数表达式相当于变量赋值,没有新的声明。 函数声明会覆盖变量声明,但不会覆盖变量赋值:: function value(){ return 1; } var value; alert(typeof value); //"function" 赋值后:: function value(){ return 1; } var value = 1; alert(typeof value); //"number" arguments会更改传入参数:: function b(x, y, a) { arguments[2] = 10; alert(a); //10 } b(1, 2, 3); 如果第一个参数传入的对象调用者是null或者undefined,call方法将把全局对象(也就是window)作为this的值:: function a() { alert(this); //[object window] } a.call(null); case语句如果不使用break,会继续执行下面的case语句,而不再判断条件:: switch ('y') { case 'y': console.log('y'); case 'other': console.log('other'); } 结果为: y other 获取某位置处的元素:: document.elementFromPoint document.createDocumentFragment();可以避免每次进行Dom操作时的页面刷新 设置css:: .style.cssText [2,3].toString() == '2,3' console.dir()可以显示类.如使用log显示function会返回函数定义内容,而使用dir可以查看整个类结构. * 2进制转化为10进制:parseInt(a,2) * 10进制转化为16进制:a.toString(16)); ascii码为10进制 * 字符转ascii码:'a'.charCodeAt() == 97 * ascii码砖字符:String.fromCharCode(65) == 'A' :: $('pre').innerHTML .replace(/_/g,1) .split(/\s/) .map(function(i){ return String.fromCharCode(parseInt(i,2)); }).join(''); 判断为list,而不是object:: if (obj.length === +obj.length) {} * object.length undefined * +obj.length NaN undefined == null []会被认为是true:: > lst = []; > if (lst) console.log(true); true 在判断中赋值:: //else false if (a=false){ console.log(a); } else { console.log('else',a); } for (var key in dict) :: lst = ['a','b','c'] 2 in lst //true lst.length=8 会扩展array长度,不足的为undefined l.length=0 清空array 有length和序号,可以当作Array来使用:: var my_object = {     '0': 'zero',     '1': 'one',     '2': 'two',     '3': 'three',     '4': 'four',     length: 5 }; var sliced = Array.prototype.slice.call(my_object, 3); console.log(sliced); arg.length说明可以$.each sort默认按照字符编码对数组进行排序:: [2,10].sort() //[10,2] 按数字比较:: [2,10].sort(function(a,b){a-b}) //[2,10] 获取元素宽度,高度:: $element[0].offsetWidth $element[0].offsetHeight IE中array没有indexOf方法,数组中查找元素:: String.prototype.indexOf.call(['a','b'],'a') 如果html不规范(比如标签未关闭),那么在IE中赋值给$.html时会静静的失败 chrome中的滑动条控件:: css选择:: var $ = function(selector) { return document.querySelector(selector); }; forEach:: ["-webkit-", "-moz-", "-ms-", "-o-", ""].forEach(function(prefix) { 修改内容:: eleOpacityImage.innerHTML = '<' 修改css:: eleImages[1].style.opacity = value / 100; * 浏览器内部宽度 window.innerWidth * 整个浏览器的宽度 window.outerWidth * 显示器宽度 screen.width 解析url --------------------------- .. image:: location.png document.documentMode --------------------------- * 5 Internet Explorer 5 mode (also known as "quirks mode"). * 7 Internet Explorer 7 Standards mode. * 8 Internet Explorer 8 Standards mode. * 9 Internet Explorer 9 Standards mode. * 10 Internet Explorer 10 Standards mode. json标准格式 --------------- key加双引号 value加双引号,可以为数字(不加引号),可以为list([]) 位运算 --------------- http://www.w3school.com.cn/js/pro_js_operators_bitwise.asp :: << 左移 >> 右移 ~ 非 & 与 | 或 ^ 异或 所有整数字面量都默认存储为有符号整数。只有 ECMAScript 的位运算符才能创建无符号整数。 开发者不能直接访问第 32 个数位,即有符号整数的符号位(在最前面) 无符号整数的数值范围为 0 到 4294967295 32位最多存储整数42亿 :: var iNum = 18; alert(iNum.toString(2)); //输出 "10010" 位运算 NOT 实质上是对数字求负,然后减 1,因此 25 变 -26 escape ----------- * escape() 已经被废弃,不要使用 * encodeURI() url编码,编码后仍然可以使用 * encodeURIComponent() 作为url参数编码使用 原来:: http://www.google.com/a file with spaces.html encodeURI:: http://www.google.com/a%20file%20with%20spaces.html encodeURIComponent:: http%3A%2F%2Fwww.google.com%2Fa%20file%20with%20spaces.html 参数编码应用:: param1 = encodeURIComponent("http://xyz.com/?a=12&b=55") url = "http://domain.com/?param1=" + param1 + "¶m2=99"; 结果为:: http://www.domain.com/?param1=http%3A%2F%2Fxyz.com%2F%Ffa%3D12%26b%3D55¶m2=99 通过iframe实现跨域通信 ---------------------- http://blog.leezhong.com/tech/2011/01/25/iframe-crossdomain.html 主页面中获取iframe中的元素:: $(frames['bar'].document).find('#someid') iframe中获取其他iframe中的元素:: $(parent.frames['foo'].document).find('#someid') 通过改变隐藏iframe的size来通知发生了某事件,信息通过url hashtag或者页面元素赋值来传递 隐藏iframe:: 改变size:: $proxy.css('width', $proxy.width()+1+'px'); 监听resize事件:: $(window).resize(function(){}); constructor ------------------- 对象的constructor属性始终指向创建当前对象的构造函数 每个函数都有一个默认的属性prototype,而这个prototype的constructor默认指向这个函数 :: var Foo=function(){} var f = new Foo(); console.log(f.constructor === Foo); // true console.log(Foo.prototype.constructor === Foo);// true //合并起来 console.log(f.constructor.prototype.constructor===Foo);// true 但如果覆盖了prototype:: Foo.prototype = { getName: function() { return "name"; } }; 此时Person.prototype.constructor === Object 应采用重新覆盖的方式更改:: Person.prototype.constructor = Person; 播放声音 -------------- chrome中优先使用audio,使用ogg格式文件:: voice.self = Ext.DomHelper.append(document.body, {tag: 'audio', src: 'horse.ogg'}); voice.self.play(); 如果要使用wav格式:: IE下{tag: 'bgsound', src: file, loop: 1, autostart: true} 非IE{tag: 'embed', src: file, hidden: true, autostart: true, loop: false} 图形 ============= 开源库 * raphael svg/vml 绘图 * d3 数据绑定 * datav 数据可视化 方案 * svg在Android 3.0以上版本才支持 * paper.js canvas绘图 桌面平台使用raphael,移动端使用paper 思考 ============================= javascript三大主神 * Brendan Eich:javascript之父 * Dean Edwards:IE7.js、Base2框架、packer压缩器、cssQuery、moz-behaviors.xml的作者 * Douglas Crockford:JSON、JSLint、JSMin、ADSafe与蝴蝶书的作者 搞清组件的核心功能,果断调用。如jQuery为DOM/Ajax/Anim 操作类库 对于非核心功能,可以考虑在自己组件里实现。如$.extend 或 $.each IE6/7不支持JSON,需要借助json2.js,其他版本原生支持 js库 ============================= DD_belatedPNG解决IE6下PNG不透明问题 多选控件,包括自动补全,ajax等功能 http://textextjs.com/ 类似iphone的手指滑动,滚动屏幕效果 http://natrixnatrix89.github.com/promptu-menu/ moment进行日期解析:: var moment = require('moment'); moment().format('YYYY-MM-DD HH:mm:ss') http://modernizr.com/ 检测浏览器对html5和css3的支持情况的库 phantomjs提供headless的浏览器调用,可以模拟浏览器进行操作。 QUnit ---------- A JavaScript Unit Testing framework. stop(),start()提供了Asynchronous Test,如果不使用的话,异步操作中的断言会无法执行 测试接口是否提供,返回结果是否正确,行为是否符合预期 async ---------- 将多层嵌套变为一层,还是需要callback Jscex --------- series:: $await(op1()); $await(op2()); $await(op3()); parallel:: var resultArray = $await(whenAll(op1(), op2(), op3())); $await等待的是一个异步对象, 待该Task对象结束(返回结果或抛出错误);如果它尚未启动,则启动该任务;如果已经完成,则立即返回结果(或抛出错误) 在一般编程场景中,如果盲目使用await取代传统的callback,会带来不必要的封装,导致语句理解难度加大。如sample/weibo.html spm --------- 建立spm目录结构:: mkdir svg-personnel cd svg-personnel/ spm init 获取模块:: mkdir libs cd libs/ spm install all js文件名中不能有“.”,应该使用“-” js变量名中不能有“-”,改为驼峰式 require('./jquery.mousewheel')会导致spm build时认为要寻找jquery.js,因此改为require('./jquery-mousewheel') 如果不build到modules中,那么类中的this指向window 去掉coffee打包:: spm build --enableCoffee=false bootstrap -----------