国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

詳解js的事件代理(委托)

2019-11-19 18:20:25
字體:
來源:轉載
供稿:網友

JavaScript事件代理(委托)一般用于以下情況:

  1. 事件注冊在祖先級元素上,代理其子級元素。可以減少事件注冊數量,節約內存開銷,提高性能。

  2. 對js動態添加的子元素可自動綁定事件。

之前一直用各種js庫的事件代理,如 jQuery,非常方便實用。今天嘗試用原生 js 實現該功能。

var addEvent = (function () { if (document.addEventListener) { return function (element, type, handler) { element.addEventListener(type, handler, false); }; } else if (document.attachEvent) { return function (element, type, handler) { element.attachEvent('on' + type, function () { handler.apply(element, arguments); }); }; } else { return function (element, type, handler) { element['on' + type] = function () { return handler.apply(element, arguments); }; }; }})(),getClassElements = function (parentElement, classname) { var all, element, classArr = [], classElements = []; if (parentElement.getElementsByClassName) { return parentElement.getElementsByClassName(classname); } else { all = parentElement.getElementsByTagName('*'); for (var i = 0, len = all.length; i < len; i++) { element = all[i]; classArr = element && element.className && element.className.split(' '); if (classArr) { for (var j = 0; j < classArr.length; j++) {  if (classArr[j] === classname) {  classElements.push(element);  } } } } return classElements; }},delegate = function () { // 參數:element, type, [selector,] handler var args = arguments, element = args[0], type = args[1], handler; if (args.length === 3) { handler = args[2]; return addEvent(element, type, handler); } if (args.length === 4) { selector = args[2]; handler = args[3]; return addEvent(element, type, function (event) { var event = event || window.event, target = event.target || event.srcElement, quickExpr = /^(?:[a-zA-Z]*#([/w-]+)|(/w+)|[a-zA-Z]*/.([/w-]+))$/, match, idElement, elements, tagName, count = 0, len; if (typeof selector === 'string') { match = quickExpr.exec(selector); if (match) {  // #ID selector  if (match[1]) {  idElement = document.getElementById(match[1]);  tagName = match[0].slice(0, match[0].indexOf('#'));  // tag selector  } else if (match[2]) {  elements = element.getElementsByTagName(selector);  // .class selector  } else if (match[3]) {  elements = getClassElements(element, match[3]);  tagName = match[0].slice(0, match[0].indexOf('.'));  } } if (idElement) {  if ( tagName ? tagName === idElement.nodeName.toLowerCase() && target === idElement : target === idElement ) {  return handler.apply(idElement, arguments);  } } else if (elements) {  for (len = elements.length; count < len; count++) {  if ( tagName ? tagName === elements[count].nodeName.toLowerCase() && target === elements[count] : target === elements[count] ) {  return handler.apply(elements[count], arguments);  }  } } } }); }};

主要是用 apply 改變 this 的指向

handler.apply(idElement, arguments);handler.apply(elements[count], arguments);

測試一下:

<style>#outer {padding: 50px; background-color: lightpink;}#inner {padding: 30px; background-color: aliceblue;}#paragraph1, #paragraph3 {background-color: cadetblue}</style>
<div id="outer">outer <div id="inner">inner <p id="paragraph1" class="parag1">paragraph1</p> <p id="paragraph2" class="parag">paragraph2</p> <span>span</span> <p id="paragraph3" class="parag">paragraph3</p> </div></div>
var outer = document.getElementById('outer');delegate(outer, 'click', function () { console.log(this.id); // outer});
delegate(outer, 'click', 'p', function () { console.log(this.id); //點擊 paragraph1 元素,輸出其id為 "paragraph1"});

模仿 jQuery 的風格,優化代碼:

(function () { var $ = function (element) { return new _$(element); }; var _$ = function (element) { this.element = element && element.nodeType === 1 ? element : document; }; _$.prototype = { constructor: _$, addEvent: function (type, handler, useCapture) { var element = this.element; if (document.addEventListener) { element.addEventListener(type, handler, (useCapture ? useCapture : false)); } else if (document.attachEvent) { element.attachEvent('on' + type, function () {  handler.apply(element, arguments); }); } else { element['on' + type] = function () {  return handler.apply(element, arguments); }; } return this; }, getClassElements: function (classname) { var element = this.element, all, ele, classArr = [], classElements = []; if (element.getElementsByClassName) { return element.getElementsByClassName(classname); } else { all = element.getElementsByTagName('*'); for (var i = 0, len = all.length; i < len; i++) {  ele = all[i];  classArr = ele && ele.className && ele.className.split(' ');  if (classArr) {  for (var j = 0; j < classArr.length; j++) {  if (classArr[j] === classname) {  classElements.push(ele);  }  }  } } return classElements; } }, delegate: function () { //參數:type, [selector,] handler var self = this, element = this.element, type = arguments[0], handler; if (arguments.length === 2) { handler = arguments[1]; return self.addEvent(type, handler); } else if (arguments.length === 3) { selector = arguments[1]; handler = arguments[2]; return self.addEvent(type, function (event) {  var event = event || window.event,  target = event.target || event.srcElement,  quickExpr = /^(?:[a-zA-Z]*#([/w-]+)|(/w+)|[a-zA-Z]*/.([/w-]+))$/,  match,  idElement,  elements,  tagName,  count = 0,  len;  if (typeof selector === 'string') {  match = quickExpr.exec(selector);  if (match) {  // #ID selector  if (match[1]) {  idElement = document.getElementById(match[1]);  tagName = match[0].slice(0, match[0].indexOf('#'));  // tag selector  } else if (match[2]) {  elements = element.getElementsByTagName(selector);  // .class selector  } else if (match[3]) {  elements = self.getClassElements(match[3]);  tagName = match[0].slice(0, match[0].indexOf('.'));  }  }  if (idElement) {  if ( tagName ? tagName === idElement.nodeName.toLowerCase() && target === idElement ? target === idElement ) {  return handler.apply(idElement, arguments);  }  } else if (elements) {  for (len = elements.length; count < len; count++) {  if ( tagName ? tagName === elements[count].nodeName.toLowerCase() && target === elements[count] : target === elements[count] ) {   return handler.apply(elements[count], arguments);  }  }  }  } }); } } }; window.$ = $; return $;}());

使用如下:

var outer = document.getElementById('outer');$(outer).delegate('click', '.parag', function (event) { console.log(this.id);});

以上就是本文的全部內容,希望本文的內容對大家的學習或者工作能帶來一定的幫助,同時也希望多多支持武林網!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 德州市| 康保县| 开平市| 合山市| 桂东县| 临湘市| 温州市| 永春县| 布尔津县| 三亚市| 如皋市| 合山市| 泌阳县| 五华县| 龙州县| 丘北县| 黔西县| 绿春县| 遂川县| 苏尼特左旗| 博爱县| 胶州市| 玛多县| 浙江省| 商河县| 天柱县| 奇台县| 尉氏县| 石渠县| 高台县| 扎鲁特旗| 中牟县| 洛宁县| 民和| 灵寿县| 萨迦县| 韩城市| 林芝县| 韩城市| 西畴县| 肇庆市|