问题由来

linkbutton 是 jQuery EasyUI 中常用的一个控件,可以使用它创建按钮。用法很简单,使用 a 标签给一个easyui-linkbutton 的class就可以了。

<a id="btn" class="easyui-linkbutton">这是一个按钮</a>

或者使用代码方式。

$("#btn").linkbutton();

实际使用时,经常会使用 jQuery 添加事件处理函数。例如

   $("#btn").click(function () {
alert("点击事件");
});

在使用过程中可能会需要暂时禁用这个按钮,easyui 已经提供了禁用按钮的方法 disable,使用方法也很简单

$(function(){
$("#btn").linkbutton(“disable”);
})

到这儿问题就来了,再点击一下,提示框照依然弹了出来,disable方法好像并没有什么效果

原因分析:不同的事件处理方式

在 DOM 中,存在着两类不同的事件处理方式,DOM0式和 DOM2 方式。

DOM0通过在HTML中设置属性,或者在JavaScript中设置一个对象的属性(property)的方法注册事件. 例如

var btn = document.getElementById("btn");
btn.onclick = function () {
alert("clicked");
};

取消事件注册,将 onclick 赋予为null 即可

btn.onclick = null;

DOM 2通过addEventListener()方法注册事件

var btn = document.getElementById("btn");
btn.addEventListener("click", function () {
alert("clicked");
}, false);

通过addEventListener添加的事件只能处理程序只能使用removeEventListener来移除,移除时传入的参数与添加处理程序时使用的参数相同,这也意味着通过addEventListener添加的匿名函数将无法移除

问题在于,这两种方式是各自独立处理的,互相并不影响。

由于 easyui 中使用了 DOM0 方式处理按钮的启用和禁用,而 jQuery 则使用 DOM2 方式进行事件的注册,所以之前的禁用失效了。

使用  DOM Level 0 方式,我们可以通过事件名称,方便地取得原来注册的事件处理函数,保存起来,以便以后回复。这正是 linkbutton 现在已经完成的。

解决方法:修复 linkbutton 的启用和禁用

修改的方式就是在原来的基础上,增加对于通过 DOM Level 2 方式注册的事件进行处理。检查是否注册了 jQuery 的 click 事件处理函数,如果有,在数据缓存对象上增加一个名为 savedHandlers 的数组来保存原来的点击事件处理函数,然后,从原来的对象上取消已经注册的事件处理函数。恢复的时候,检查数据对象上是否有保存的事件处理函数数组,如果存在,遍历这个数组,将这些事件处理函数重新注册到元素上。

function setButtonState(domElem, disabled) {                    // 设置按钮状态

    var data = $.data(domElem, "linkbutton");                   // 获取对象的数据
if (disabled) { // 禁用按钮
data.options.disabled = true;
var href = $(domElem).attr("href"); // 获取超级连接
if (href) {
data.href = href; // 保存原来的超级链接
$(domElem).attr("href", "javascript:void(0)"); // 重新设置
}
if (domElem.onclick) { // 是否有点击事件处理
data.onclick = domElem.onclick;
domElem.onclick = null; // 取消掉
}
var eventData = $(domElem).data("events") || $._data(domElem, 'events');
if (eventData && eventData["click"]) {
var clickHandlerObjects = eventData["click"];
data.savedHandlers = [];
for (var i = ; i < clickHandlerObjects.length; i++) {
if (clickHandlerObjects[i].namespace != "menu") {
var handler = clickHandlerObjects[i]["handler"];
$(domElem).unbind('click', handler);
data.savedHandlers.push(handler);
}
}
} $(domElem).addClass("l-btn-disabled"); // 使用样式
} else {
data.options.disabled = false; // 启用按钮
if (data.href) { // 恢复原来的超级链接
$(domElem).attr("href", data.href);
}
if (data.onclick) { // 恢复原来的点击事件处理
domElem.onclick = data.onclick;
}
if (data.savedHandlers) {
for (var i = ; i < data.savedHandlers.length; i++) {
$(domElem).click(data.savedHandlers[i]);
}
} $(domElem).removeClass("l-btn-disabled");
}
}

如果不想修改easyui的源代码,可以使用扩展方法来解决这个问题

/**
* linkbutton方法扩展
* @param {Object} jq
*/
$.extend($.fn.linkbutton.methods, {
/**
* 激活选项(覆盖重写)
* @param {Object} jq
*/
enable: function(jq){
return jq.each(function(){
var state = $.data(this, 'linkbutton');
if ($(this).hasClass('l-btn-disabled')) {
var itemData = state._eventsStore;
//恢复超链接
if (itemData.href) {
$(this).attr("href", itemData.href);
}
//回复点击事件
if (itemData.onclicks) {
for (var j = ; j < itemData.onclicks.length; j++) {
$(this).bind('click', itemData.onclicks[j]);
}
}
//设置target为null,清空存储的事件处理程序
itemData.target = null;
itemData.onclicks = [];
$(this).removeClass('l-btn-disabled');
}
});
},
/**
* 禁用选项(覆盖重写)
* @param {Object} jq
*/
disable: function(jq){
return jq.each(function(){
var state = $.data(this, 'linkbutton');
if (!state._eventsStore)
state._eventsStore = {};
if (!$(this).hasClass('l-btn-disabled')) {
var eventsStore = {};
eventsStore.target = this;
eventsStore.onclicks = [];
//处理超链接
var strHref = $(this).attr("href");
if (strHref) {
eventsStore.href = strHref;
$(this).attr("href", "javascript:void(0)");
}
//处理直接耦合绑定到onclick属性上的事件
var onclickStr = $(this).attr("onclick");
if (onclickStr && onclickStr != "") {
eventsStore.onclicks[eventsStore.onclicks.length] = new Function(onclickStr);
$(this).attr("onclick", "");
}
//处理使用jquery绑定的事件
var eventDatas = $(this).data("events") || $._data(this, 'events');
if (eventDatas["click"]) {
var eventData = eventDatas["click"];
for (var i = ; i < eventData.length; i++) {
if (eventData[i].namespace != "menu") {
eventsStore.onclicks[eventsStore.onclicks.length] = eventData[i]["handler"];
$(this).unbind('click', eventData[i]["handler"]);
i--;
}
}
}
state._eventsStore = eventsStore;
$(this).addClass('l-btn-disabled');
}
});
}
});

[js]jQuery EasyUI的linkbutton组件disable方法无法禁用jQuery绑定事件的问题分析的更多相关文章

  1. 第一百九十七节,jQuery EasyUI,LinkButton(按钮)组件

    jQuery EasyUI,LinkButton(按钮)组件 学习要点: 1.加载方式 2.属性列表 3.方法列表 本节课重点了解 EasyUI 中 LinkButton(按钮)组件的使用方法,这个组 ...

  2. jQuery EasyUI,LinkButton(按钮)组件

    转自:https://www.cnblogs.com/adc8868/p/6639570.html jQuery EasyUI,LinkButton(按钮)组件 学习要点: 1.加载方式 2.属性列表 ...

  3. JQuery EasyUI的常用组件

    jQuery EasyUI 是一个基于 jQuery 的框架,集成了各种用户界面插件,该框架提供了创建网页所需的一切,帮助您轻松建立站点. 注:本次介绍的JQuery EasyUI版本为1.5版. 一 ...

  4. jQuery EasyUI 的editor组件使用

    问题:最近在优化一个项目时,前端用到了 easyui这个插件来实现表格,搞了很久,才实现出一部分功能,但是还是有很多地方不熟悉,故记录一下,以后再研究 第一个实例------------------- ...

  5. jQuery delegate方法实现Ajax请求绑定事件不丢失

    给元素绑定click事件后 ,遇到一个问题:当执行一些ajax请求,再次调用此页面,里面的这个click事件就失效了 比如说:我的分页是一个ajax请求 但我点下一页时 后生成的元素a就没有了clic ...

  6. 从零开始学 Web 之 jQuery(五)操作元素其他属性,为元素绑定事件

    大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...

  7. 用live()方法给新增节点绑定事件

    jQuery 给所有匹配的元素附加一个事件处理函数,即使这个元素是以后再添加进来的也有效. 这个方法是基本是的 .bind() 方法的一个变体.使用 .bind() 时,选择器匹配的元素会附加一个事件 ...

  8. (Jquery)关于给动态加载的页面元素,绑定事件

    如果使用Jquery给元素绑定事件,一般会用bind,或者类似click函数来直接绑定. 但是对于动态生成的元素,会发现常规绑定无法生效,比如: <div class'div'></ ...

  9. Jquery EasyUI选项卡-Tabs的使用方法

    以下是easyUI的tabs的简单实用介绍. var e =$('#main').tabs('exists','accordion'); if(e==true){  $('#main').tabs(' ...

随机推荐

  1. socket学习

    对应的代码: 服务端: import socket phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM) #买手机 phone.bind(( ...

  2. React Native踩坑之旅

    原文连接:http://www.studyshare.cn/blog-front/blog/details/1137 最近做一个app,使用React Native实现,如果严格按照RN官方文档去配置 ...

  3. css 效果之转换

    在css3 中,有一个转换效果,也可以替代js,并且比 js 做得好,那就是转换,即transform 属性,只需要个这个属性加上值,就可以实现转化效果了;有三种值,平移(translate),旋转( ...

  4. html页面转jsp后 乱码问题。

    在jsp文件中的html显示乱码是因为服务端和客户端的编码不一致导致的.如果Java和JSP编译成class文件过程中,使用的编码方式与源文件的编码不一致,就会出现乱码.解决办法:1.未指定使用字符集 ...

  5. Prometheus

    官方网址:https://prometheus.io/ GitHub网址:https://github.com/prometheus/prometheus 软件下载地址:https://prometh ...

  6. thymeleaf标签必须由匹配的结束标记终止

    问题描述 springboot使用Thymeleaf标签时会报元素类型必须由匹配的结果标记终止. 如下所示 如果我们一个个的给这些元素后面加上终止标记也是件很麻烦的事~~~~ 解决办法 方法一: 在p ...

  7. splice() 方法通过删除现有元素和/或添加新元素来更改一个数组的内容。

    var myFish = ["angel", "clown", "mandarin", "surgeon"]; //从第 ...

  8. 贝叶斯公式与最大后验估计(MAP)

    1, 频率派思想 频率派思想认为概率乃事情发生的频率,概率是一固定常量,是固定不变的 2, 最大似然估计 假设有100个水果由苹果和梨混在一起,具体分配比例未知,于是你去随机抽取10次,抽到苹果标记为 ...

  9. Day07 - Ruby比一比:Symbol符号与String字串

    前情提要: 第六天我们透过Ruby代码练习public,protected和privatemethod时,发现冒号在前面的参数,:mydraft,:myspace,这些就是符号Symbol.在今天,我 ...

  10. SpringCloud-day04-Eureka高可用集群配置

    5.4Eureka高可用集群配置 在高并发的情况下一个注册中心难以满足,因此一般需要集群配置多台. 我们再新建两个module  microservice-eureka-server-2002,  m ...