+ jQuery.event.special.foo = {
+ setup: function( data, namespaces, handler ) {
+ ok( true, "Setup run." );
+ },
+ teardown: function( namespaces ) {
+ ok( true, "Teardown run." );
+ },
+ add: function( handleObj ) {
+ ok( true, "Add run." );
+ },
+ remove: function( handleObj ) {
+ ok( true, "Remove run." );
+ },
+ _default: function( event ) {
+ ok( true, "Default run." );
+ }
+ };
+
+ // Run: setup, add
+ jQuery("#liveSpan1").live("foo.a", function(e){
+ ok( true, "Handler 1 run." );
+ });
+
+ // Run: add
+ jQuery("#liveSpan1").live("foo.b", function(e){
+ ok( true, "Handler 2 run." );
+ });
+
+ // Run: Handler 1, Handler 2, Default
+ jQuery("#liveSpan1").trigger("foo");
+
+ // Run: Handler 1, Default
+ // TODO: Namespace doesn't trigger default (?)
+ jQuery("#liveSpan1").trigger("foo.a");
+
+ // Run: remove
+ jQuery("#liveSpan1").die("foo.a");
+
+ // Run: Handler 2, Default
+ jQuery("#liveSpan1").trigger("foo");
+
+ // Run: remove, teardown
+ jQuery("#liveSpan1").die("foo");
+
+ delete jQuery.event.special.foo;
+});
+
+test(".delegate()/.undelegate()", function() {
+ expect(65);
+
+ var submit = 0, div = 0, livea = 0, liveb = 0;
+
+ jQuery("#body").delegate("div", "submit", function(){ submit++; return false; });
+ jQuery("#body").delegate("div", "click", function(){ div++; });
+ jQuery("#body").delegate("div#nothiddendiv", "click", function(){ livea++; });
+ jQuery("#body").delegate("div#nothiddendivchild", "click", function(){ liveb++; });
+
+ // Nothing should trigger on the body
+ jQuery("body").trigger("click");
+ equals( submit, 0, "Click on body" );
+ equals( div, 0, "Click on body" );
+ equals( livea, 0, "Click on body" );
+ equals( liveb, 0, "Click on body" );
+
+ // This should trigger two events
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ jQuery("div#nothiddendiv").trigger("click");
+ equals( submit, 0, "Click on div" );
+ equals( div, 1, "Click on div" );
+ equals( livea, 1, "Click on div" );
+ equals( liveb, 0, "Click on div" );
+
+ // This should trigger three events (w/ bubbling)
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ jQuery("div#nothiddendivchild").trigger("click");
+ equals( submit, 0, "Click on inner div" );
+ equals( div, 2, "Click on inner div" );
+ equals( livea, 1, "Click on inner div" );
+ equals( liveb, 1, "Click on inner div" );
+
+ // This should trigger one submit
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ jQuery("div#nothiddendivchild").trigger("submit");
+ equals( submit, 1, "Submit on div" );
+ equals( div, 0, "Submit on div" );
+ equals( livea, 0, "Submit on div" );
+ equals( liveb, 0, "Submit on div" );
+
+ // Make sure no other events were removed in the process
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ jQuery("div#nothiddendivchild").trigger("click");
+ equals( submit, 0, "undelegate Click on inner div" );
+ equals( div, 2, "undelegate Click on inner div" );
+ equals( livea, 1, "undelegate Click on inner div" );
+ equals( liveb, 1, "undelegate Click on inner div" );
+
+ // Now make sure that the removal works
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ jQuery("#body").undelegate("div#nothiddendivchild", "click");
+ jQuery("div#nothiddendivchild").trigger("click");
+ equals( submit, 0, "undelegate Click on inner div" );
+ equals( div, 2, "undelegate Click on inner div" );
+ equals( livea, 1, "undelegate Click on inner div" );
+ equals( liveb, 0, "undelegate Click on inner div" );
+
+ // Make sure that the click wasn't removed too early
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ jQuery("div#nothiddendiv").trigger("click");
+ equals( submit, 0, "undelegate Click on inner div" );
+ equals( div, 1, "undelegate Click on inner div" );
+ equals( livea, 1, "undelegate Click on inner div" );
+ equals( liveb, 0, "undelegate Click on inner div" );
+
+ // Make sure that stopPropgation doesn't stop live events
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ jQuery("#body").delegate("div#nothiddendivchild", "click", function(e){ liveb++; e.stopPropagation(); });
+ jQuery("div#nothiddendivchild").trigger("click");
+ equals( submit, 0, "stopPropagation Click on inner div" );
+ equals( div, 1, "stopPropagation Click on inner div" );
+ equals( livea, 0, "stopPropagation Click on inner div" );
+ equals( liveb, 1, "stopPropagation Click on inner div" );
+
+ // Make sure click events only fire with primary click
+ submit = 0, div = 0, livea = 0, liveb = 0;
+ var event = jQuery.Event("click");
+ event.button = 1;
+ jQuery("div#nothiddendiv").trigger(event);
+
+ equals( livea, 0, "delegate secondary click" );
+
+ jQuery("#body").undelegate("div#nothiddendivchild", "click");
+ jQuery("#body").undelegate("div#nothiddendiv", "click");
+ jQuery("#body").undelegate("div", "click");
+ jQuery("#body").undelegate("div", "submit");
+
+ // Test binding with a different context
+ var clicked = 0, container = jQuery('#main')[0];
+ jQuery("#main").delegate("#foo", "click", function(e){ clicked++; });
+ jQuery("div").trigger('click');
+ jQuery("#foo").trigger('click');
+ jQuery("#main").trigger('click');
+ jQuery("body").trigger('click');
+ equals( clicked, 2, "delegate with a context" );
+
+ // Make sure the event is actually stored on the context
+ ok( jQuery.data(container, "events").live, "delegate with a context" );
+
+ // Test unbinding with a different context
+ jQuery("#main").undelegate("#foo", "click");
+ jQuery("#foo").trigger('click');
+ equals( clicked, 2, "undelegate with a context");
+
+ // Test binding with event data
+ jQuery("#body").delegate("#foo", "click", true, function(e){ equals( e.data, true, "delegate with event data" ); });
+ jQuery("#foo").trigger("click");
+ jQuery("#body").undelegate("#foo", "click");
+
+ // Test binding with trigger data
+ jQuery("#body").delegate("#foo", "click", function(e, data){ equals( data, true, "delegate with trigger data" ); });
+ jQuery("#foo").trigger("click", true);
+ jQuery("#body").undelegate("#foo", "click");
+
+ // Test binding with different this object
+ jQuery("#body").delegate("#foo", "click", jQuery.proxy(function(e){ equals( this.foo, "bar", "delegate with event scope" ); }, { foo: "bar" }));
+ jQuery("#foo").trigger("click");
+ jQuery("#body").undelegate("#foo", "click");
+
+ // Test binding with different this object, event data, and trigger data
+ jQuery("#body").delegate("#foo", "click", true, jQuery.proxy(function(e, data){
+ equals( e.data, true, "delegate with with different this object, event data, and trigger data" );
+ equals( this.foo, "bar", "delegate with with different this object, event data, and trigger data" );
+ equals( data, true, "delegate with with different this object, event data, and trigger data")
+ }, { foo: "bar" }));
+ jQuery("#foo").trigger("click", true);
+ jQuery("#body").undelegate("#foo", "click");
+
+ // Verify that return false prevents default action
+ jQuery("#body").delegate("#anchor2", "click", function(){ return false; });
+ var hash = window.location.hash;
+ jQuery("#anchor2").trigger("click");
+ equals( window.location.hash, hash, "return false worked" );
+ jQuery("#body").undelegate("#anchor2", "click");
+
+ // Verify that .preventDefault() prevents default action
+ jQuery("#body").delegate("#anchor2", "click", function(e){ e.preventDefault(); });
+ var hash = window.location.hash;
+ jQuery("#anchor2").trigger("click");
+ equals( window.location.hash, hash, "e.preventDefault() worked" );
+ jQuery("#body").undelegate("#anchor2", "click");
+
+ // Test binding the same handler to multiple points
+ var called = 0;
+ function callback(){ called++; return false; }
+
+ jQuery("#body").delegate("#nothiddendiv", "click", callback);
+ jQuery("#body").delegate("#anchor2", "click", callback);
+
+ jQuery("#nothiddendiv").trigger("click");
+ equals( called, 1, "Verify that only one click occurred." );
+
+ called = 0;
+ jQuery("#anchor2").trigger("click");
+ equals( called, 1, "Verify that only one click occurred." );
+
+ // Make sure that only one callback is removed
+ jQuery("#body").undelegate("#anchor2", "click", callback);
+
+ called = 0;
+ jQuery("#nothiddendiv").trigger("click");
+ equals( called, 1, "Verify that only one click occurred." );
+
+ called = 0;
+ jQuery("#anchor2").trigger("click");
+ equals( called, 0, "Verify that no click occurred." );
+
+ // Make sure that it still works if the selector is the same,
+ // but the event type is different
+ jQuery("#body").delegate("#nothiddendiv", "foo", callback);
+
+ // Cleanup
+ jQuery("#body").undelegate("#nothiddendiv", "click", callback);
+
+ called = 0;
+ jQuery("#nothiddendiv").trigger("click");
+ equals( called, 0, "Verify that no click occurred." );
+
+ called = 0;
+ jQuery("#nothiddendiv").trigger("foo");
+ equals( called, 1, "Verify that one foo occurred." );
+
+ // Cleanup
+ jQuery("#body").undelegate("#nothiddendiv", "foo", callback);