My example of spying on a method is failing with "Expected spy handle_click to have been called." when it should pass. However, I am getting the console log "Foo handle_click called!", so I know it's being called.
Foo.js
function Foo() {
this.$btn = $('<a href="#">Foo</a>');
this.$btn.on('click', this.handle_click);
};
Foo.prototype.handle_click = function(evt) {
evt.preventDefault();
console.log('Foo handle_click called!');
};
Foo_spec.js:
it('should be called when trigger is clicked', function() {
var foo = new Foo();
spyOn( foo, 'handle_click' ).andCallThrough();
foo.$btn.click();
expect( foo.handle_click ).toHaveBeenCalled();
});
I am using jasmine-1.2.0, jasmin-html and jasmine-jquery, but not jasmine-sinon; at-least I don't think it's bundled in there. Any help is much appreciated!
Update This was answered below. However, I wanted to document the solution in the case of a jQuery Plugin:
Foo.js:
function Foo() { ... }
Foo.prototype.handle_click = function(evt) { ... }
$.fn.foo = function(options) {
return new Foo(this, options || {});
};
$.fn.foo.prototype = Foo.prototype;
Foo_spec.js:
it('should be called when clicked', function() {
spyOn( $.fn.foo.prototype, 'handle_click');
var plugin = $('#selector-for-plugin').foo();
plugin.$btn.click();
expect( plugin.handle_click ).toHaveBeenCalled();
});
The problem is that you bind the handle_click
function in the constructor. So when you create a new instance a reference to the function is binded to the event. After that you replace the foo.handle_click
with a spy. But this will not effect the function that was binded to the event as this is still your original function. You have to spy on the Foo.prototype.handle_click
function before you create the instance, so the spied function can be bound to the event.
it('should be called when trigger is clicked', function() {
spyOn( Foo.prototype, 'handle_click' );
var foo = new Foo();
foo.$btn.click();
expect( foo.handle_click ).toHaveBeenCalled();
});