SCMLife.com

 找回密码
 立即注册

QQ登录

只需一步,快速开始

扫一扫,访问微社区

查看: 3615|回复: 1

[推荐] Dojo 手势功能介绍

[复制链接]
发表于 2013-2-17 16:48:01 | 显示全部楼层 |阅读模式
本帖最后由 技术狂人 于 2013-2-17 16:50 编辑
& }" i2 ]1 C: A" y( s- I* b2 P3 k" S7 l5 t/ ~
Dojo 目前提供支持的手势有:
1 n2 B6 @7 X+ \# Y) \2 O, w
  • Tap:在屏幕上单击。
  • Double tap:在屏幕上双击。
  • Hold:手指轻点屏幕并持续一段时间。
  • Swipe:手指在屏幕上滑动。Dojo Api 对该手势的支持能够提供滑动持续的时间与位移信息。
    ; S2 W7 q  E5 S1 B& Q( c
HTML 中的 Touch 事件; u% N. d5 w5 ^- p, z& F+ m3 T
HTML Touch 系列事件是实现手势的基础。通过对 Touch 事件的监听,我们可以得到手指在屏幕上的滑动轨迹,从而判断出用户的手势动作。因此,要想研究 Dojo 中手势的实现,我们需要先了解 HTML 中 Touch 系列事件的触发顺序和事件对象所包含的数据。
8 H, Z' g) w/ j6 ]- S触控设备(例如智能手机)上支持触摸事件的浏览器一般都实现了以下几个事件:
- E  P9 E8 Q7 S! J8 y9 T6 f; Y# N
  • touchstart:该事件在用户手指落在设备屏幕上时发生。
  • touchmove:该事件当用户手指在屏幕上滑动时发生。
  • touchend:该事件当用户手指离开设备屏幕时发生。
  • touchcancel:该事件当用户手指移出文档范围或接触点数目超过浏览器支持时发生。5 a: E8 I1 Y+ I! ]: b9 j/ Z  X2 E
在 W3C 关于 Touch 事件的规范中规定了 Touch 事件的数据结构。; b+ `5 E! z; |# |
7 H" \. c2 Q" O( [6 J
清单 1 Touch 事件接口- p& w9 [9 N( H8 U1 j+ R
interface TouchEvent : UIEvent {
  q, I3 c8 z9 `. O! B; q    readonly attribute TouchList   touches; ! Y8 E, S7 B; {% ^
    readonly attribute TouchList   targetTouches; ! n# }- V2 Y( @& Z/ @% G
    readonly attribute TouchList   changedTouches; 9 _! D! O5 _( p6 X; T8 ]1 C
    readonly attribute boolean     altKey; # n: z' V; o* m
    readonly attribute boolean     metaKey; 4 O8 m! l( N$ w! J2 i/ E
    readonly attribute boolean     ctrlKey; - \+ `* j" ~* [
    readonly attribute boolean     shiftKey;
$ h8 k, j) \* S5 q+ W    readonly attribute EventTarget relatedTarget;
; U& I1 q0 D+ [ };

4 A8 p2 z1 J" L! _在该接口中,我们注意到有三个属性的类型都是 TouchList。 其中 touches 中包含所有发生在当前文档节点上的 Touch 信息,targetTouches 包含所有以当前文档节点为 target 的 Touch 信息,因为 Touch 事件是以 Bubble 方式在文档树中传递,所以这两个列表不一定相同。9 D' o% W- w5 J4 h7 X, `
changedTouches 属性比较特殊,在不同的事件中其有不同的含义。对于 touchstart 事件,其中存储的是激活的触摸点。对于 touchmove 事件,其中存储的是从上个事件发生后移动的点(即从上个事件以来接触点的轨迹)。
! b2 J, C7 Q9 J7 D+ H: ?8 M% RTocuhList 和其元素 Touch 的数据结构见清单 2。4 R/ J/ M1 n8 ?% @  g  x7 E
# {9 U$ O- |* l* [) P+ {' `& N
清单 2 TouchList 接口和 Touch 接口- @2 v; Y! c( j2 Z
interface TouchList {
) p! r4 h/ I5 f' D" @0 ^6 K: V    readonly attribute unsigned long length;
2 Z3 w  X( e5 Q& l    getter Touch item (unsigned long index); " G8 R6 c/ R/ R9 y- a" T2 w) C+ |5 N9 ]
    Touch        identifiedTouch (long identifier); 9 D2 Z$ Z$ w) N7 _0 _0 _! f5 m9 \1 N/ g
}; 3 y* I) T! F. N7 `# |
1 S" `5 j7 ]! L# \4 @- T, ]4 b* Q
interface Touch {
7 s! D# @2 g/ `    readonly attribute long        identifier; # M0 _5 Y5 E- l6 h: |2 n; K
    readonly attribute EventTarget target;
9 ?" X3 ~7 Q6 D  ]& p$ |5 s* {3 W    readonly attribute long        screenX;
/ h: g# D* T5 I. w' `    readonly attribute long        screenY; 7 h. {6 i. I! S; r
    readonly attribute long        clientX;
6 M! }1 p+ v" N2 o    readonly attribute long        clientY;
+ t! [) v1 A; c( A) S    readonly attribute long        pageX;
3 U# P! d; ^7 j5 `9 w! [0 ~    readonly attribute long        pageY; 2 o% L! S5 w# h
    readonly attribute long        radiusX;
3 F$ T9 P0 P+ I, y    readonly attribute long        radiusY; : ]6 l# Y8 [* B; c. k& a
    readonly attribute float       rotationAngle; * N! j* f' v  ?+ L" S; y% L' s6 j
    readonly attribute float       force; 0 J' h" M+ ^1 e0 E5 L
};

7 G* x+ i3 o+ g1 H0 \从清单 2 可以看出,我们可以从 TouchList 结构得到接触点的个数和每个点的坐标以及大小信息。对于不支持多点触摸的浏览器来说,touches 和 targetTouches 中永远只有一个元素。
' |/ y/ d  b! Q7 ?' z$ d2 v对于支持鼠标事件(例如 click 事件)的 DOM 元素,触摸动作不单会触发 Touch 事件还可能触发鼠标事件。具体的触发顺序是先触发 Touch 事件,当手指离开触摸屏幕并发生 touchend 事件之后触发鼠标事件。如果 touchstart 事件和 touchend 事件中间间隔的时间较长或者在 Touch 事件的处理函数中调用了 preventDefault 函数,则鼠标事件不会被触发。
$ f7 U3 |, \6 j' X3 h为了屏蔽浏览器对 Touch 事件实现上的差异,Dojo 在包 dojo.touch 中对 Touch 事件做了封装。Dojo 在 dojo.touch 包下定义了 press、move、release 和 cancel 四个事件,并将 touches 中的第一个元素移除,该元素的所有属性都被复制到事件对象上。
  M' J" M5 {3 X4 ^/ A* f1 @8 N* S
2 V* S+ b* {! a8 D! H/ B

% [$ V3 r7 S1 nDojo Gesture 分析
5 R! I* e% P: ^) iDojo 在包 dojox.gesture 中提供了对手势功能的支持。在本节中,我从对 dojox.gesture 的使用着手为大家分析 Dojo 如何实现的手势功能。+ u' S! ?* @# v$ F: `1 T* `# U
在 Dojo 源代码 dojox.gesture.test 包中有一个测试页面 test_gesture.html。清单 3 是其部分代码。( ^$ A: m' q: t# C& L, _
+ z% w% p4 |* M6 Z  h6 q
清单 3.test_gesture.html dojo.ready 部分代码
$ S4 U+ \1 {2 w0 W! t) k
var inner = html.byId("inner"); 0 T% p  W+ k: ^
on(inner, tap, action); 6 J0 a/ h1 {* F0 N1 v8 R
on(inner, tap.hold, action); 6 {/ }+ C2 {0 @% Y! e
on(inner, tap.doubletap, action); & N; j: @6 s  J6 z; H
on(inner, swipe, swipeAction);
* C- o' x. K8 l8 h! T
在这段代码中:inner 是 HTML 中的一个 DIV 元素。tap 和 swip 是 Dojo 提供的两种手势对象。我们都知道,on 函数是 Dojo 1.7 中新添加的一种事件机制,其作用是把发生在 DOM 节点上的事件和对应的事件处理程序关联起来。# h6 \2 K( O! B/ L# C% M! U/ @  C
从这段代码可以看出,Dojo 对手势的实现思路是先监测用户的手势动作,如果是符合定义的手势则触发自定义的手势事件。
/ c* t  M) C+ o7 s3 Q接下来,我们以 dojox.gesture.tap 对象为例来研究下如何实现一个手势。
$ M5 f8 \6 R# ?; ^0 t5 tdojox.gesture.tap 对象的类是从 dojox.gesture.Base 继承而来。我们比较一下两个类的代码可以发现,dojox.gesture.tap 对父类的修改有以下几个方面。
3 N: c0 E% R7 D9 Z& a9 \
  • 设置属性 defaultEvent 的值为 tap
  • 设置属性 subEvents 的值为数组 ["hold","doubletap"]. 可以猜测,清单 3 代码中的 tap.hold 和 tap.doubletap 应该和该属性有关。
  • 重写了父类的 press 和 release 方法。  r# }6 Y+ W, d. t/ o% e4 R0 z
让我们先来研究下 press 和 release 这两个方法。这两个方法在 dojox.gesture.Base 类中只有声明,和它们一起的还有 move 和 cancel 两个方法。根据 Base 类的注释,这四个方法代表了手势的四个阶段。Tap 的字面意思是拍打、轻击。根据字面意思,tap 手势只有 press 和 release 两个阶段,因此 dojox.gesture.tap 中只实现了 press 和 release 方法。
1 c5 l1 m. g% [! W1 L4 @0 Y6 |% n% O
清单 4.dojox.gesture.tap 部分代码
7 r- ?! r' {2 x/ ~
press: function(/*Object*/data, /*Event*/e){+ }( m) I/ ]+ a: v6 w5 ^1 y
// summary:
+ ^2 ^  `  Z+ S3 \' L% c//记录tap手势的初始信息并初始化一个Timeout句柄用来检测hold手势8 T, Z9 }; g5 q9 x
                if(e.touches && e.touches.length >= 2){) d( Q# t1 u3 p* h( f( m' ]
                        //tap 手势仅支持单点触摸  f( N. `" _. W4 k7 l2 e, `. Q0 v
                        delete data.context;+ D9 h& g( l' V2 s- V, f
                        return;
# _  v) A% J- i* O                }$ D; H5 S9 |* E# T2 |
                var target = e.target;
! J4 R4 ?0 X6 @& j. `                this._initTap(data, e);$ F: I7 O% v# K" ~
                data.tapTimeOut = setTimeout(lang.hitch(this, function(){
; M/ y! _* e1 E. [9 D% K/ D1 O$ e" c                        if(this._isTap(data, e)){
. D* S5 h( v. A; E, `% b                                this.fire(target, {type: "tap.hold"});# M, W) @) N) w5 L5 K
                        }% G, N; _0 O" @$ c
                        delete data.context;
) {* V. p1 Z( l                }), this.holdThreshold);
" f7 j1 I1 m0 j1 F7 L        },
6 A7 \; ?- `9 f. r, {' {! A) N        release: function(/*Object*/data, /*Event*/e){
/ w4 a3 n$ b+ z2 T; d8 a" ~                // summary:: @/ b! k9 I9 p
                //在touchend事件发生时检测并触发tap或double tap手势
+ u- l! N& x" g                if(!data.context){
1 n3 n! `  b' A$ b/ F                        clearTimeout(data.tapTimeOut);
# U+ `" b# O; y/ p0 E                        return;$ Z% Q3 C: O! Q! N
                }
  I. G: S# j% w3 l4 b. ^  N                if(this._isTap(data, e)){0 T6 ~1 d# u3 @; f
                        switch(data.context.c){8 q' @; n0 L5 H- Y) J
                        case 1:
0 H6 g, j- M+ F) C; [                                this.fire(e.target, {type: "tap"});+ X" c8 z9 w- O" T
                                break;
/ N- {8 }) q/ n: i                        case 2:
2 ~+ |% g! J2 L3 N2 O, e2 b                                this.fire(e.target, {type: "tap.doubletap"});
' C' p7 N) A- U0 b$ ~2 Z                                break;. _4 j- w& F8 s. m: s5 C
                        }5 n+ ~8 k4 o8 Y( Y. F" M% S" Y
                }
4 {; m0 m, v' G5 ]                clearTimeout(data.tapTimeOut);6 n: l+ j5 u. D6 @- x
        },
/ c6 l4 c* t$ S# E        _initTap: function(/*Object*/data, /*Event*/e){& ~) v1 ?+ C# d  I$ A
                // summary:
, ^, h# h) `" d                //        更新tap手势的数据
# g' i6 k% b3 h; q  N6 G) i                if(!data.context){
2 I" v1 j- Y8 U6 ^" ]. K: B                        data.context = {x: 0, y: 0, t: 0, c: 0};
$ c4 N$ h+ F. i4 \* `8 P4 V4 }0 g                }
% n7 `: N5 M  {. u% E& h* A$ Y                var ct = new Date().getTime();
9 N7 X5 @2 M6 S9 }) G# h1 J                if(ct - data.context.t <= this.doubleTapTimeout){
$ T9 D" v# x! N                        data.context.c++;
1 f7 o6 q5 a, n- @+ t& v                }else{
5 Q5 p$ B' a9 x" _/ G/ ?" Z8 T0 q                        data.context.c = 1;
: l: @3 ~$ M$ r8 G# B( O- k( y# t! V( w                        data.context.x = e.screenX;
" m# `) D/ u# r- w; b                        data.context.y = e.screenY;
) U; ^8 l& y! y0 t: b                }
+ y0 F5 m1 P$ j$ a) `+ p                data.context.t = ct;* ?# v+ l1 N* m) p& i8 J
        },
5 z: a0 [+ C( _4 e_isTap: function(/*Object*/data, /*Event*/e){
% `9 a4 }1 Q0 j$ G+ M6 U* _! c                // summary:
1 |# s( K6 A: C. L' t$ B  |                //                Check whether it's an valid tap6 T' G) F; _' W' G+ I' D0 d, X
                var dx = Math.abs(data.context.x - e.screenX);$ Y- ]% R4 L& O" j- s: h
                var dy = Math.abs(data.context.y - e.screenY);
/ ~# c0 ?! o6 @6 T$ s2 O5 M                return dx <= this.tapRadius && dy <= this.tapRadius;3 _" v. K8 j6 v8 {
        }
( |) l0 x5 S% g
我们发现,在 press 和 release 函数中,fire 函数被反复调用。函数 fire 是在 Base 类中定义的,其作用是利用 Dojo 的事件 API dojo.on 在指定 DOM 节点上触发特定事件。
. n5 b0 O* k$ G8 `( F我们再来看看是怎样检测手势的。在清单 4 中有一个名为 _isTap 的函数。从字面意思来看,我们可以断定其是用来判断当前发生的手势是否是 Tap 手势。通过阅读代码,我们可以看到该函数在 release 函数中被调用,函数内部通过检测 release 事件发生时的坐标和 press 事件的坐标(在 press 函数中通过调用 _initTap 来保存)间的距离来判断是否为 Tap 手势。请注意这几个函数的参数列表中的 data 参数,它是用来保存判断手势需要的数据的。5 F9 E- n4 Q: r. S0 A# o. |
到此,我们大致了解了怎么样利用 Dojo Gesture 库实现自定义的手势。但是,还有几个关键的问题我们没有搞清楚。
* d5 w% w! j/ H$ L! q/ _2 a/ _
  • 我们发出的事件有什么特性?
  • 手势的四个阶段是怎样被触发的?
  • 我们实现的手势能够在非触摸设备上使用吗?9 G4 |0 F4 q2 s3 m, u/ t
让我们继续来看 Dojo 自带的例子 test_gesture.html. 如果在 inner div 元素上轻敲作出 tap 手势的话,你会发现其父节点 outer div 元素也能接收到 tap 事件。这说明我们发出的 tap 事件是具有 bubble 性质的。! x' P7 w3 K# Q  E- m% ^
要得到剩下的两个问题的答案,我们还需要继续研究 dojox.gesture._Base 的代码。请看下面的代码:
; v$ L; o+ A+ u5 B  I
+ V$ v1 ]1 P2 }- }% t- @# W清单 5.dojox.gesture._Base 的 _add 方法* p& T4 m  L& L. m5 B* i
_add: function(/*Dom*/node, /*String*/type, /*function*/listener){
- @1 _9 D. S8 U/ k$ v- c2 e                         // summary: ) H. T0 ~1 z( s# o; g% v0 f
                         //         将 touch 几个阶段的事件绑定到手势类的相应处理函数上
) u, k. }) E  i! s! p             var element = this._getGestureElement(node); , p, q' c* Q3 X
             if(!element){ 4 k( v( x, |5 v
                // the first time listening to the node % t8 [1 j4 W4 x9 A+ g/ J: T" n
                 element = {
2 k3 P& r) a# u6 b2 G                     target: node,
$ D+ k# f& l3 M2 s  F                     data: {}, 9 M! Y3 S: W+ K9 D, c1 x
                     handles: {}
3 ]5 O0 j" F+ O                                 };
7 H9 R: G& P3 T0 `4 V6 i" Y6 d) p
% X7 T* ?$ X1 \- B4 f                 var _press = lang.hitch(this, "_process", element, "press"); $ X6 b& `' w7 Z' V) R
                 var _move = lang.hitch(this, "_process", element, "move");
+ z, S  G3 y) v6 N4 C+ s' X                 var _release = lang.hitch(this, "_process", element, "release");
5 r8 G: p7 L& g& P                 var _cancel = lang.hitch(this, "_process", element, "cancel");
4 G- j. |0 C3 M! W0 k
7 b' V& L8 }" e, i                 var handles = element.handles; * W' W, L2 v/ B1 [, g" j2 i5 @
                 if(this.touchOnly){
7 i+ Q# |. C/ c) r# `                     handles.press = on(node, 'touchstart', _press);
* U- i- ?! v$ ]( M+ S# |+ A/ L                     handles.move = on(node, 'touchmove', _move);
& X7 P8 E- @5 s3 \/ X* \9 z/ R                     handles.release = on(node, 'touchend', _release); ! L6 ^% ~2 G: g( v. {2 r% ?0 N& E
                     handles.cancel = on(node, 'touchcancel', _cancel); 8 z4 K' s6 U1 R" ]' q
                 }else{ 7 b2 h- S- y1 ?3 }: m/ S; j
                     handles.press = touch.press(node, _press); " J* y/ i1 O5 f
                     handles.move = touch.move(node, _move);
" f- O. o2 _( H                     handles.release = touch.release(node, _release);
# j* o# w8 ~/ m8 I                     handles.cancel = touch.cancel(node, _cancel);
% L) \: n- ]- L3 m8 a                                 }
( y) J! q$ x: O4 D/ s                 this._elements.push(element);
3 F" B% B8 X  o+ M9 G. F             }
7 A- y: u, a+ d# O8 H* O/ r* ^1 N% F             // track num of listeners for the gesture event - type ! c# M+ u/ a# F" l* c
            // so that we can release element if no more gestures being monitored
8 W1 ~0 N/ Y0 s8 n* c: [# d            element.handles[type] = !element.handles[type] ? 1 : ++element.handles[type];
' M. c( y- A3 I4 S+ g
! j4 t( z1 r6 ?5 i# B+ V2 K                         return on(node, type, listener); //handle ' y" j2 S- R, }3 h! j- X
                 }

, s/ @; z- Z6 K, L9 P: ^$ C2 b0 z从注释上来看,这个方法是为 dom node 添加手势事件的处理句柄。3 P+ o" r* S0 z5 i
当事件只支持触摸设备时,该函数将相应的 touch 事件映射到 _Base 类相应方法上。这些方法又会调用事件的 press, move, release 和 cancel 方法从而触发我们的手势事件。
4 {6 O# N+ w8 ?+ z当事件同时支持触摸和鼠标动作时,该方法通过 dojo.touch 来处理相应触摸 / 鼠标事件的监听。
8 c4 r; T/ J+ q# w
' y2 V& g. p1 P' b( C- F9 O8 N1 c
5 ~! f- N8 P1 J
总结
( u3 W  ~/ M& B& ?通过上面的论述,我们可以总结出实现自定义手势的步骤
& O0 z. c2 J, _+ V* B, w: z
  • 从 dojox.gesture._Base 继承
  • 实现 press, move, release 和 cancel 函数,监听并记录 touch/ 鼠标事件的数据。这些数据可以保存在这几个函数的 data 参数中。
  • 在合适的时机通过 fire 方法发出手势事件
    1 Z- N1 F8 t$ d# ^" E

+ Z, Q* J0 K9 [# e: z
 楼主| 发表于 2013-2-17 16:51:17 | 显示全部楼层
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关闭

SCMLife推荐上一条 /4 下一条

QQ|小黑屋|手机版|无图版|SCMLife.com ( 京ICP备06056490号-1 )

GMT+8, 2018-12-11 02:45 , Processed in 0.070417 second(s), 6 queries , Gzip On, MemCache On.

Powered by SCMLife X3.4 Licensed

© 2001-2017 JoyShare.

快速回复 返回顶部 返回列表