1096

I have an input type="image". This acts like the cell notes in Microsoft Excel. If someone enters a number into the text box that this input-image is paired with, I setup an event handler for the input-image. Then when the user clicks the image, they get a little popup to add some notes to the data.

My problem is that when a user enters a zero into the text box, I need to disable the input-image's event handler. I have tried the following, but to no avail.

$('#myimage').click(function { return false; });
Konrad Borowski
  • 9,885
  • 2
  • 50
  • 68
Randy L
  • 13,549
  • 12
  • 41
  • 72

20 Answers20

1695

jQuery ≥ 1.7

With jQuery 1.7 onward the event API has been updated, .bind()/.unbind() are still available for backwards compatibility, but the preferred method is using the on()/off() functions. The below would now be,

$('#myimage').click(function() { return false; }); // Adds another click event
$('#myimage').off('click');
$('#myimage').on('click.mynamespace', function() { /* Do stuff */ });
$('#myimage').off('click.mynamespace');

jQuery < 1.7

In your example code you are simply adding another click event to the image, not overriding the previous one:

$('#myimage').click(function() { return false; }); // Adds another click event

Both click events will then get fired.

As people have said you can use unbind to remove all click events:

$('#myimage').unbind('click');

If you want to add a single event and then remove it (without removing any others that might have been added) then you can use event namespacing:

$('#myimage').bind('click.mynamespace', function() { /* Do stuff */ });

and to remove just your event:

$('#myimage').unbind('click.mynamespace');
Ben Leggiero
  • 25,904
  • 38
  • 161
  • 267
samjudson
  • 53,412
  • 7
  • 55
  • 67
  • 6
    Is there a way to test if `unbind()` is working? I've added it and both events are still firing. – David Yell Sep 23 '10 at 10:41
  • 19
    Well if both events are still firing then obviously its not working. You'd have to give more info to get a proper answer. Try asking a separate question. – samjudson Oct 07 '10 at 14:02
  • 15
    Old answer -also like the the included namespace information- but I forget; does on/off return the jQ object? If so, perhaps a more complete answer would be to daisy chain: `$('#myimage').off('click').on('click',function(){...})` – vol7ron Jan 10 '13 at 19:25
  • 3
    Yes they do, so yes, if you wanted to clear all other click events and then add your own you could do the above chained call. – samjudson Jan 11 '13 at 10:47
  • Using `.unbind()` helped me in a situation where I was toggling a bunch of checkboxes, but relized i was re-adding the `.click()` handler repeatedly, causing the app to hang. – TecBrat Mar 18 '14 at 14:35
  • @samjudson: could you explain **why** we would like to use `off` instead of unbind? @the0ther mentions closure compiler but I'm not sure what this means nor whether this is the correct/only reason. thanks – Adrien Be Apr 16 '14 at 11:07
  • I believe @theOther is stating a preference for bind() over click(), not bind() over on(). on/off have replaced bind/unbind as the preferred method in jQuery since 1.7 as stated above. I don't know what the closure compiler is so can't really comment on that. – samjudson Apr 16 '14 at 14:09
  • just struck my comment from the record, it was three years old. on/off is the way to go. – Randy L Apr 16 '14 at 14:16
  • this one is a hack::: cleaning up all the events from the element regardless if they have set using `.on(..)` or even native `.onclick = ` --- clone the element with `.cloneNode(true)` hide the original element, insert the copied one before it, optionally removing the original element afterwards. –  Mar 04 '15 at 14:17
  • @samjudson if a button has attrib `onclick="return false;"` its jQuery `click` listener is still being fired..is that correct? – techie_28 May 23 '16 at 09:39
  • See **jwhittenburg**'s comment below re DOM object having inline `onclick="doFoo"` - use `.prop("onclick", null).attr("onclick", null)` – gordon Aug 03 '17 at 18:02
  • Was previously unaware of jQuery event namespacing: https://api.jquery.com/event.namespace/ (pretty slim documentation tbh). But precisely what I needed, allows sets of events to be bound / unbound as a group. – Morvael Apr 08 '19 at 10:58
64

This wasn't available when this question was answered, but you can also use the live() method to enable/disable events.

$('#myimage:not(.disabled)').live('click', myclickevent);

$('#mydisablebutton').click( function () { $('#myimage').addClass('disabled'); });

What will happen with this code is that when you click #mydisablebutton, it will add the class disabled to the #myimage element. This will make it so that the selector no longer matches the element and the event will not be fired until the 'disabled' class is removed making the .live() selector valid again.

This has other benefits by adding styling based on that class as well.

Scott Stafford
  • 40,202
  • 22
  • 116
  • 163
MacAnthony
  • 4,313
  • 2
  • 20
  • 26
  • 3
    Nice way to use `live()`, never thought about using it this way before ... Aside from coding convenience, is it faster or does it offer any other advantages to using bind/unbind? – chakrit Mar 11 '10 at 16:53
  • 2
    If you are adding/removing elements, then there are definite performance advantages over bind as the binding with live is only done once. jQuery seems to be moving to more live and delegate events rather than binding to specific elements. – MacAnthony Mar 26 '10 at 16:23
  • 3
    To disable live you can use die() – Moons Dec 07 '11 at 10:37
  • 10
    "As of jQuery 1.7, the .live() method is deprecated. Use .on() to attach event handlers. Users of older versions of jQuery should use .delegate() in preference to .live()." http://api.jquery.com/live/ – Lucas Pottersky Dec 29 '11 at 18:49
  • 13
    Ah, fickle jQuery, one day you bring 300 rep to a guy, the next day it's obsolete. – MrBoJangles Feb 27 '13 at 23:14
39

If you want to respond to an event just one time, the following syntax should be really helpful:

 $('.myLink').bind('click', function() {
   //do some things

   $(this).unbind('click', arguments.callee); //unbind *just this handler*
 });

Using arguments.callee, we can ensure that the one specific anonymous-function handler is removed, and thus, have a single time handler for a given event. Hope this helps others.

ghayes
  • 8,100
  • 1
  • 34
  • 39
  • 2
    Absolutely, though sometimes you may wish to have certain logic about when the handler should be unbound (e.g. don't unbind unless they filed out a text box first). – ghayes Mar 24 '13 at 22:47
  • 6
    arguments.callee is depraceted in strict mode! – zloctb Nov 26 '13 at 07:38
38

This can be done by using the unbind function.

$('#myimage').unbind('click');

You can add multiple event handlers to the same object and event in jquery. This means adding a new one doesn't replace the old ones.

There are several strategies for changing event handlers, such as event namespaces. There are some pages about this in the online docs.

Look at this question (that's how I learned of unbind). There is some useful description of these strategies in the answers.

How to read bound hover callback functions in jquery

Community
  • 1
  • 1
Mnebuerquo
  • 5,370
  • 4
  • 41
  • 48
32

maybe the unbind method will work for you

$("#myimage").unbind("click");
John Boker
  • 78,333
  • 17
  • 93
  • 129
31

I had to set the event to null using the prop and the attr. I couldn't do it with one or the other. I also could not get .unbind to work. I am working on a TD element.

.prop("onclick", null).attr("onclick", null)
dwhittenburg
  • 608
  • 7
  • 13
  • 1
    I voted this up because I've been trying to unbind an onblur for about 30 mintues. '.unbind' didn't work, '.prop' didn't work, '.attr' didn't work, but this trick with both .prop and .attr together did the trick. odd. I'm using Chrome with jquery 1.7.2 – Jeremy May 31 '12 at 06:31
  • same here. unbind and off didn't work for me on FF with jq 1.7.2 and .prop("onclick", null) was also sufficient in my case – bryanallott Mar 27 '13 at 08:58
  • 4
    I think the reason the above methods didn't work for you guys is because of how you set it up. If you wrote the event handler into the DOM then yes, clearing this attribute is necessary to clear the event handler, however, I think most people (including myself) wish to keep such things (javascrip, event handlers, etc.) out of the DOM, so when we want to set up an event handler we use something like `$("#elementId").click(function(){//Event execution code goes here});` so that it is not in the html page at all. To clear that we actually need to use `.unbind()`, or the preferred `.off()` – VoidKing May 17 '13 at 20:32
  • Here's a like from someone who has just spent an hour to discover that properties added via jQuery don't show on Chrome's element panel. Been trying `unbind` and `off` and it wasn't solving the problem. Thank you very much! – Geeky Guy Oct 13 '14 at 19:58
  • Im with the "unbind() didnt work for me" crowd, but this did, and I'm using latest jquery. Go figure – Darkloki Mar 02 '15 at 20:48
14

If event is attached this way, and the target is to be unattached:

$('#container').on('click','span',function(eo){
    alert(1);

    $(this).off(); //seams easy, but does not work

    $('#container').off('click','span'); //clears click event for every span

    $(this).on("click",function(){return false;}); //this works.

});​
11

You may be adding the onclick handler as inline markup:

<input id="addreport" type="button" value="Add New Report" onclick="openAdd()" />

If so, the jquery .off() or .unbind() won't work. You need to add the original event handler in jquery as well:

$("#addreport").on("click", "", function (e) {
   openAdd();
});

Then the jquery has a reference to the event handler and can remove it:

$("#addreport").off("click")

VoidKing mentions this a little more obliquely in a comment above.

VK Da NINJA
  • 512
  • 7
  • 19
davaus
  • 979
  • 12
  • 16
9

Updated for 2014

Using the latest version of jQuery, you're now able to unbind all events on a namespace by simply doing $( "#foo" ).off( ".myNamespace" );

alexpls
  • 1,755
  • 18
  • 27
8

Best way to remove inline onclick event is $(element).prop('onclick', null);

Shahrukh Azeem
  • 265
  • 2
  • 10
8

To remove ALL event-handlers, this is what worked for me:

To remove all event handlers mean to have the plain HTML structure without all the event handlers attached to the element and its child nodes. To do this, jQuery's clone() helped.

var original, clone;
// element with id my-div and its child nodes have some event-handlers
original = $('#my-div');
clone = original.clone();
//
original.replaceWith(clone);

With this, we'll have the clone in place of the original with no event-handlers on it.

Good Luck...

alo Malbarez
  • 354
  • 2
  • 6
  • 15
Aakash
  • 14,077
  • 4
  • 77
  • 63
7

Thanks for the information. very helpful i used it for locking page interaction while in edit mode by another user. I used it in conjunction with ajaxComplete. Not necesarily the same behavior but somewhat similar.

function userPageLock(){
    $("body").bind("ajaxComplete.lockpage", function(){
        $("body").unbind("ajaxComplete.lockpage");
        executePageLock();      
    });
};  

function executePageLock(){
    //do something
}
Nowaker
  • 11,112
  • 4
  • 47
  • 60
jquery_user
  • 71
  • 1
  • 1
6

If you use $(document).on() to add a listener to a dynamically created element then you may have to use the following to remove it:

// add the listener
$(document).on('click','.element',function(){
    // stuff
});

// remove the listener
$(document).off("click", ".element");

ow3n
  • 4,612
  • 4
  • 38
  • 47
5

In case .on() method was previously used with particular selector, like in the following example:

$('body').on('click', '.dynamicTarget', function () {
    // Code goes here
});

Both unbind() and .off() methods are not going to work.

However, .undelegate() method could be used to completely remove handler from the event for all elements which match the current selector:

$("body").undelegate(".dynamicTarget", "click")
Ilia Rostovtsev
  • 12,128
  • 10
  • 47
  • 80
3

This also works fine .Simple and easy.see http://jsfiddle.net/uZc8w/570/

$('#myimage').removeAttr("click");
Somnath Kharat
  • 3,442
  • 2
  • 24
  • 49
3

I know this comes in late, but why not use plain JS to remove the event?

var myElement = document.getElementById("your_ID");
myElement.onclick = null;

or, if you use a named function as an event handler:

function eh(event){...}
var myElement = document.getElementById("your_ID");
myElement.addEventListener("click",eh); // add event handler
myElement.removeEventListener("click",eh); //remove it
Elia Weiss
  • 5,417
  • 8
  • 50
  • 81
Silviu Preda
  • 558
  • 1
  • 6
  • 27
2

if you set the onclick via html you need to removeAttr ($(this).removeAttr('onclick'))

if you set it via jquery (as the after the first click in my examples above) then you need to unbind($(this).unbind('click'))

VK Da NINJA
  • 512
  • 7
  • 19
Ishan Liyanage
  • 1,561
  • 19
  • 24
1

All the approaches described did not work for me because I was adding the click event with on() to the document where the element was created at run-time:

$(document).on("click", ".button", function() {
    doSomething();
});


My workaround:

As I could not unbind the ".button" class I just assigned another class to the button that had the same CSS styles. By doing so the live/on-event-handler ignored the click finally:

// prevent another click on the button by assigning another class
$(".button").attr("class","buttonOff");

Hope that helps.

Avatar
  • 11,039
  • 8
  • 98
  • 167
1

Hope my below code explains all. HTML:

(function($){

     $("#btn_add").on("click",function(){
       $("#btn_click").on("click",added_handler);
       alert("Added new handler to button 1");
     });

  
 
     $("#btn_remove").on("click",function(){
       $("#btn_click").off("click",added_handler);
       alert("Removed new handler to button 1");
     });

  
   function fixed_handler(){
    alert("Fixed handler");
  }
  
   function added_handler(){
    alert("new handler");
  }
  
  $("#btn_click").on("click",fixed_handler);
  $("#btn_fixed").on("click",fixed_handler);
  
  
})(jQuery);
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<button id="btn_click">Button 1</button>
  <button id="btn_add">Add Handler</button>
  <button id="btn_remove">Remove Handler</button>
  <button id="btn_fixed">Fixed Handler</button>
mysticmo
  • 47
  • 3
  • 1
    why are you passing in `})(jQuery);` into the IIFE when it can already be accessed globally? – Bryan P Jan 19 '16 at 07:01
  • 2
    Good question Bryan. This method has been entered in my gene due to practice. - There can be some possibility for non jquery plugin to have the variable $, hence I cannot use $ directly. - Frequent use of the word "jQuery" in an js script file can increase the byte size of the file, where jQuery is a 6 byte string. Hence I prefer to use "$" or any variable of single byte. – mysticmo Jan 19 '16 at 09:44
0

Just Simply remove that from button:

$('.classname').click(function(){
$( ".classname" ).remove();
});
Mitra
  • 137
  • 1
  • 1
  • 11