zoukankan      html  css  js  c++  java
  • jquery event 封装的源源分析

    1. 对于javascript事件扩展,所有的lib都差不多。和jquery和prototype,yui和Ext,其要解决的首要问题是兼容性,所有lib都会对event进行包裹,统一其属性解决其兼容性。对于事件的操作无非是addEvent,fireEvent,removeEvent这三个事件方法。一般lib都会对浏览器的提供的函数做一些扩展,解决兼容性内存泄漏等问题。第三个问题就是如何得到domReady的状态。   
    2. 6.1 event的包裹   
    3. 浏览器的事件兼容性是一个令人头疼的问题。IE的event在是在全局的window下,而mozilla的event是事件源参数传入到回调函数中。还有很多的事件处理方式也一样。   
    4. Jquery提供了一个 event的包裹,这个相对于其它的lib提供的有点简单,但是足够使用。   
    5. //对事件进行包裹。   
    6.     fix : function(event) {   
    7.         if (event[expando] == truereturn event;//表明事件已经包裹过   
    8.         //保存原始event,同时clone一个。   
    9.         var originalEvent = event;                               ①   
    10.         event = {   originalEvent : originalEvent};   
    11.         for (var i = this.props.length, prop;i;) {   
    12.             prop = this.props[--i];   
    13.             event[prop] = originalEvent[prop];   
    14.         }          
    15.         event[expando] = true;         
    16.         //加上preventDefault and stopPropagation,在clone不会运行   
    17.         event.preventDefault = function() {                   ②   
    18.             // 在原始事件上运行   
    19.             if (originalEvent.preventDefault)   
    20.                 originalEvent.preventDefault();   
    21.             originalEvent.returnValue = false;   
    22.         };   
    23.         event.stopPropagation = function() {   
    24.             // 在原始事件上运行   
    25.             if (originalEvent.stopPropagation)   
    26.                 originalEvent.stopPropagation();   
    27.             originalEvent.cancelBubble = true;   
    28.         };   
    29.         // 修正 timeStamp   
    30.         event.timeStamp = event.timeStamp || now();   
    31.         // 修正target   
    32.         if (!event.target)                                       ③   
    33.             event.target = event.srcElement || document;               
    34.         if (event.target.nodeType == 3)//文本节点是父节点。   
    35.             event.target = event.target.parentNode;   
    36.         // relatedTarget   
    37.         if (!event.relatedTarget && event.fromElement)      ④   
    38.             event.relatedTarget = event.fromElement == event.target   
    39.                     ? event.toElement   : event.fromElement;   
    40.         // Calculate pageX/Y if missing and clientX/Y available   
    41.         if (event.pageX == null && event.clientX != null) {  ⑥   
    42.             var doc = document.documentElement, body = document.body;   
    43.           event.pageX = event.clientX   
    44.                 + (doc && doc.scrollLeft || body && body.scrollLeft || 0)   
    45.                     - (doc.clientLeft || 0);   
    46.             event.pageY = event.clientY   
    47.                 + (doc && doc.scrollTop || body && body.scrollTop || 0)   
    48.                     - (doc.clientTop || 0);   
    49.         }   
    50.   
    51.         // Add which for key events   
    52.     if (!event.which    && ((event.charCode || event.charCode === 0) ⑦   
    53.                         ? event.charCode    : event.keyCode))   
    54.             event.which = event.charCode || event.keyCode;   
    55.   
    56.     // Add metaKey to non-Mac browsers    
    57.         if (!event.metaKey && event.ctrlKey)                        ⑧   
    58.             event.metaKey = event.ctrlKey;   
    59.     // Add which for click: 1 == left; 2 == middle; 3 == right   
    60.     // Note: button is not normalized, so don't use it   
    61.         if (!event.which && event.button)                          ⑨   
    62.             event.which = (event.button & 1 ? 1 : (event.button & 2  
    63.                     ? 3 : (event.button & 4 ? 2 : 0)));   
    64.         return event;   
    65.     },   
    66. 上面的代码①处保留原始事件的引用,同时clone原始事件。在这个clone的事件上进行包裹。②处在原始事件上运行preventDefault 和 stopPropagation两个方法达到是否阻止默认的事件动作发生和是否停止冒泡事件事件向上传递。   
    67. ③处是修正target个,IE中采用srcElement,同时对于文本节点事件,应该把target传到其父节点。   
    68. ④处relatedTarget只是对于mouseout、mouseover有用。在IE中分成了to和from两个Target变量,在mozilla中没有分开。为了保证兼容,采用relatedTarget统一起来。   
    69. ⑥处是进行event的坐标位置。这个是相对于page。如果页面可以scroll,则要在其client上加上scroll。在IE中还应该减去默认的2px的body的边框。   
    70. ⑦处是把键盘事件的按键统一到event.which的属性上。Ext中的实现ev.charCode || ev.keyCode || 0; ⑨则是把鼠标事件的按键统一把event.which上。charCode、ev.keyCode一个是字符的按键,一个不是字符的按键。⑨处采用&的方式来进行兼容性的处理。 Ext 通过下面三行解决兼容问题。   
    71.   var btnMap = Ext.isIE ? {1:0,4:1,2:2} : (Ext.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2}); this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);   
    72. ①②③④⑤⑥⑦⑧⑨⑩  
  • 相关阅读:
    vue组件基本结构及各个生命周期
    vue项目目录结构详解
    http请求报文格式和响应报文格式
    H5中对history栈的操作
    原生js实现元素类名的判存、添加和移除
    Content-type解析
    IntelliJ Idea 常用快捷键列表
    深入ES6中的class类
    v-model和 .sync
    作用域插槽
  • 原文地址:https://www.cnblogs.com/rooney/p/1346467.html
Copyright © 2011-2022 走看看