标签: JavaScript

  • Nexus S很诡异的单击变双击现象

    被报告了一个很诡异的Bug,在且只在三星Nexus S上出现,系统版本4.0.4和4.1.1都有:

    一次点击,会触发两次点击事件。两次事件的 target 和 currentTarget 都相同。

    因为只在三星Nexus S上出现,调试相当困难,反复无果。后来想起来我用zepto类库作为底层库,而且编译时把touch部分也编译进去了,所以尝试着将 click 替换为 tap ,居然解决了……

    具体问题症结,以后再研究吧。

  • 新版node.js安装教程

    知道node.js这个东西后,一直想尝鲜。今天终于下手,先要安装环境,看了好多教程,没看太明白,似乎很难的样子。

    最后按照官方文档自己尝试了才知道,原来现在安装node.js已经完全自动化了。我用的是Ubuntu 10.4,先更新一下源:

    sudo apt-get update
    sudo apt-get upgrade

    然后安装GNU make和git,接着clone代码,最后make就ok了。

    sudo apt-get install gcc
    sudo apt-get install git-core
    git clone git://github.com/joyent/node.git
    cd node
    ./configure
    make
    make install

    make的过程比较久,让它慢慢跑就是了,完成后,就可以在命令行里测试了。现在node.js已经是0.9.3-pre版了,看到版本号很高,心里很高兴呀~

    node -v // v0.9.3-pre
    node
    > console.log('hello, world') // hello, world

    今天先到这里,将来哪天开新项目的时候用node.js做后端吧。

  • 书评:《基于MVC的JavaScript Web富应用开发》

    封面
    基于MVC的JavaScript Web富应用开发一书的封面

    知道这本书源于那篇博客:《旅行,写作,编程》。读完文章后,我忍不住顿足捶胸,尼玛这才是生活啊,这才是程序员应有的生活啊!!最尼玛让人崩溃的是,作者才21啊!!就把我30岁之后的梦想实现了。感叹之后,屌丝的生活还得继续。谁让咱们一出生就是Hard模式呢,老实说我对目前的生活还算基本满意,当然如果我也能环游世界写本书啥的,我可能也觉得洒家这辈子值了。

    (更多…)

  • 使用jQuery托管事件对阻止冒泡的影响

    使用jQuery托管事件对阻止冒泡的影响

    列表范例
    移动App中常见的列表

    jQuery托管事件是个好东西,减少侦听器的数量,还能降低内存泄露的风险,尤其在列表类的应用比较常见。

    想象一个如右图所示的下载列表,点击各列表项会展开详情,用户可以在里面查看详细信息;不展开的话也可以直接点击右边的下载按钮,就会直接下载应用。使用jQuery来写代码大概是这样的:

    $('#app-list li').on('click', function (event) {
      // 显示/隐藏详情
      $(this).toggleClass('active');
    });
    $('#app-list .download-button').on('click', function (event) {
      // 下载通过超链进行
      // 这时为了不让详情展开,需要阻止事件冒泡
      event.stopPropagation();
    });
    

    假如列表里有20个选项,那么这样就会添加40个侦听器。在PC浏览器里这样做问题不大,但是移动设备内存比较少的话可能会引发问题。另外,如果我们还引入了“上拉读取更多”和“下拉刷新”的功能,那么就要为内容的更新添加更多的代码,以避免内存泄露。所以这时,就应当将事件托管到ul去处理。( 关于jQuery事件托管可以参考其文档(http://api.jquery.com/delegate/)。)

    $('#app-list').on('click', 'li', function (event) {
      $(this).toggleClass('active');
    });
    $('#app-list').on('click', '.download-button', function (event) {
      event.stopPropagation();
    });
    

    可以看到,我没有修改事件处理函数。这样做,我们只添加了2个侦听器,并且不管#app-list里的内容如何变化,都不需要重新注册侦听器。这给代码减了不少负。

    不过接下来我发现,点击按钮后,详情仍然会被展开收起,似乎阻止冒泡的代码没有生效。仔细一想,对了,我已经把事件注册到#app-list上了,所有的事件其实都是冒泡上来的,所以点击按钮,冒泡必然经过上层节点。于是继续修改代码,改变事件处理函数的逻辑:

    $('#app-list').on('click', 'li', function (event) {
      // 如果事件始于下载按钮,则退出
      if ($(event.target).is('.download-button')) {
        return;
      }
      $(this).toggleClass('active');
    });
    

    这样便万事大吉了。

  • JavaScript奇葩的参数机制

    JavaScript奇葩的参数机制

    今天实习生遇到一个问题:有一个数组,想在一个函数里将它清空,结果办不到。代码大概是这样的:

    function empty(arr) {
      arr = [];
    }
    var array = [1, 2];
    empty(array);
    console.log(array); // 预期 [],实际 [1, 2]
    

    对于我这种半路出家根基不实的人来说面对这种问题总是很挠头。于是只有翻书,在《JavaScript高级程序设计(第三版)》上看到,原来JavaScript的函数参数设计这么奇葩。

    首先,JS里的变量分为值类型和引用类型,这点我是知道的。基础类型只有NaN、null、undefined、String、Number这5个,其他都是引用类型,也就是赋值时传引用不复制值的类型。区分如下:

    // 值类型
    var a = 'test',
        b = a;
    b = 'temp';
    console.log(a); // 'test'
    
    // 引用类型
    var a = {id: 1},
        b = a;
    b.id = 2;
    console.log(a.id); // 2
    

    但是在用在函数参数的时候,又会有所不同。参数并不是传递的引用,而是传递的引用的引用。所以即使用“===”判断,也会返回true,因为最终指向的对象是一样的。但是如果在参数中对参数重新赋值的话,就相当于改变了引用地址,重新创建了一个对象,也就无法操作外面的对象了。这可能也是“运行时环境对象”造成的结果吧。

  • Google Map 和 Bootstrap 冲突的地方

    今天遇到一个诡异的问题(其实以前就遇到了,被我想办法绕开了):

    在Google Map里添加一个Marker,如果Marker能够被拖动(draggable),就显示不正常。表现为一个1/3宽度的图标带2个阴影,拖动后变成1/4大小的图标。

    反复尝试都无法解决,去查看了Gmap的源码,发现大家是一样的,也没能解决。最后还是Google之!

    原来是Bootstrap的问题,它为了不让图片撑开容器,在样式里规定img { max-width: 100%};,就是这句话导致图片被压缩(应该是默认占据更宽的位置,好显示阴影之类的东西)。

    修改我的样式,增加.map-container img { max-width: none; }之后,就一切正常了。

  • Backbone.js笔记

    关于事件

    • 使用Backbone里,我们可以继承Backbone.View,并且侦听UI事件。这些操作是通过jQuery或者Zepto的事件委托实现的,所以很重要的一点就是:这些事件都是UI事件,loaderror这些事件是无法在events属性里注册并被侦听到的。
    • 因为是托管的事件,事件处理函数最好用event.currentTarget来寻到节点
    • model的事件都会被collection转发,所以可以直接侦听collection;同理,除非remove并等待垃圾回收的model,也不应简单的调用off(),因为这会使collection没法侦听到事件,漏掉一些处理。

    路由解析规则

    这点文档中说得不算太详尽,我摸索如下:

    1. 路径分析以#/为起始,所以链接应该如#/app/add
    2. /是很重要的分隔符,末尾的/会被认为有下一级参数,比如app/list/的规则就不适用于http://domain.com/#/app/list这样的路径
    3. 规则只匹配一次,不会多次执行
    4. 刷新页面的方法:
      Backbone.history.loadUrl(Backbone.history.fragment);

    其它关于Backbone.js的文章

    Backbone.js经验两则

    重写Backbone.js的加载动作

  • Mobile Web开发笔记

    Mobile Web开发笔记

    这里记录使用Phonegap开发移动应用期间发现。

    性能

    使用jQuery托管事件对阻止冒泡的影响

    表现

    解决 iOS webkit 使用CSS动画时闪烁的问题

    CSS 兼容性

    CSS 属性 iOS Android IE 其它
    border-radius 3.x -webkit- Firefox 3.6- -moz-
    Safari 4- -webkit-
    box-shadow 2.3 -webkit- Safari 5- -webkit-

    网摘

    Developing Better PhoneGap Apps: Float Mobile Learning

    重点包括

    1. 使用tap取代click(这点我还需要进一步测试,我发现zepto+phonegap没法捕获tap事件,不知道是我不是敲键盘的姿势不对)
    2. 使用css动画取代js动画

    随记

    1. “click”和“tap“不是一回事儿,“click”会延迟300~400ms才处理,时间上会接近taphold之类的事件,“tap”更快
    2. 给元素加上:active伪类,可以给用户更快的反馈
    3. 调试不太容易,debug.phonegap.com似乎已经关了,得想办法把自己的weinre服务搭起来。暂时可以在js里写console.log(),接受一个参数,会在log cat输出。
  • Backbone.js经验两则

    Backbone.js经验两则

    使用HTML5,实现从桌面拖拽到网页

    使用HTML5新增加的API,可以很方便的实现拖拽, 包括从桌面拖拽到网页上(部分浏览器比如Chrome还可以把东西从网页拖拽到桌面),这个操作不在今天的讨论之内,可以参考:NATIVE HTML5 DRAG AND DROP这篇文章,讲得足够详细了。

    当我试图在Backbone框架上使用这个功能的时候,问题出现了。开始我没多想,直接这么写的:

    var myView = Backbone.View.extend({
      events: {
        "drop img": "img_dropHandler"
      },
      img_dropHandler: function (event) {
        var reader = new FileReader();
        var img = event.target;
        reader.onload = function (event) {
          $(img).attr('src', event.target.result);
        }
        result.readAsDataURL(event.dataTransfer.files[0]);
      }
    });
    

    结果运行时提示我,event对象没有dataTransfer属性。我用的是最新版本的Chrome,理应是对HTML5支持最好的,而且文中也说代码在Firefox和Chrome下运行通过。后来检查了一下event,发现是f.Event,似乎原本应该是MouseEvent或者Event什么的。于是我把属性展开,看到了originalEvent这个属性,是MouseEvent;再展开originalEvent,就看到了dataTransfer属性,里面有期望中的所有属性。看来是Backbone并没有直接使用原始事件,而是封装了一层再广播。(更正)事件代理是通过jQuery来做的,jQuery在这里把原始事件封装了一层再进行转播,导致原始事件的属性没有完全复制。解决方法很简单,多写几个字母就行了:

    // 其它地方都一样
    result.readAsDataURL(event.originalEvent.dataTransfer.files[0]);
    

    Model中数组的处理

    先看一段代码:

    var ModelClass = Backbone.Model.extend({
      defaults: {
        contents: []
      }
    });
    var model1 = new ModelClass();
    var arr = model1.get('contents');
    arr[0] = 'haha';
    var model2 = new ModelClass();
    console.log(model2.get('contents'));
    

    大家猜猜结果是什么?竟然是“[‘haha’]”!这个我只能认为是Backbone.js的bug了。解决方法是先复制一个数组,对数据操作后再赋值回去,如下:

    var model1 = new ModelClass();
    var arr = model1.get('contents').concat();
    arr[0] = 'haha';
    model1.set('contents', arr);
    
  • 使用PhoneGap+jQuery Moblie开发蜂鸟镜头库手机App

    蜂鸟镜头库app效果图
    蜂鸟镜头库app

    自从春节期间在家尝试了PhoneGap之后,一直想做点实际的东西,好总结点经验,以备不时之需。可惜后台苦手,数据源是个问题。后来想起来以前做蜂鸟镜头库的时候有两个接口,可以取得完整的数据,于是便动起手来,开始做这个app。

    开发本身并不复杂,大概花了一天时间搞出雏形,加上翻页基本就能发布了。不过还是遇到不少问题,今天简单总结下踩过的坑,以免以后再踩。
    (更多…)