Skip to content
This repository has been archived by the owner on Apr 12, 2024. It is now read-only.

Commit

Permalink
fix(jqLite): correctly monkey-patch core jQuery methods
Browse files Browse the repository at this point in the history
When real jQuery is present, Angular monkey patch it to fire `$destroy` event.

This commit fixes two issues in the jQuery patch:
- passing a selector to the $.fn.remove method (only fire `$destroy` on the matched elements)
- using `$.fn.html` without parameters as a getter (do not fire `$destroy`)
  • Loading branch information
mgol authored and vojtajina committed May 23, 2013
1 parent 041f118 commit da5f537
Show file tree
Hide file tree
Showing 3 changed files with 72 additions and 24 deletions.
7 changes: 4 additions & 3 deletions src/Angular.js
Original file line number Diff line number Diff line change
Expand Up @@ -1040,9 +1040,10 @@ function bindJQuery() {
injector: JQLitePrototype.injector,
inheritedData: JQLitePrototype.inheritedData
});
JQLitePatchJQueryRemove('remove', true);
JQLitePatchJQueryRemove('empty');
JQLitePatchJQueryRemove('html');
// Method signature: JQLitePatchJQueryRemove(name, dispatchThis, filterElems, getterIfNoArguments)
JQLitePatchJQueryRemove('remove', true, true, false);
JQLitePatchJQueryRemove('empty', false, false, false);
JQLitePatchJQueryRemove('html', false, false, true);
} else {
jqLite = JQLite;
}
Expand Down
41 changes: 21 additions & 20 deletions src/jqLite.js
Original file line number Diff line number Diff line change
Expand Up @@ -107,37 +107,38 @@ function camelCase(name) {
/////////////////////////////////////////////
// jQuery mutation patch
//
// In conjunction with bindJQuery intercepts all jQuery's DOM destruction apis and fires a
// In conjunction with bindJQuery intercepts all jQuery's DOM destruction apis and fires a
// $destroy event on all DOM nodes being removed.
//
/////////////////////////////////////////////

function JQLitePatchJQueryRemove(name, dispatchThis) {
function JQLitePatchJQueryRemove(name, dispatchThis, filterElems, getterIfNoArguments) {
var originalJqFn = jQuery.fn[name];
originalJqFn = originalJqFn.$original || originalJqFn;
removePatch.$original = originalJqFn;
jQuery.fn[name] = removePatch;

function removePatch() {
var list = [this],
function removePatch(param) {
var list = filterElems && param ? [this.filter(param)] : [this],
fireEvent = dispatchThis,
set, setIndex, setLength,
element, childIndex, childLength, children,
fns, events;

while(list.length) {
set = list.shift();
for(setIndex = 0, setLength = set.length; setIndex < setLength; setIndex++) {
element = jqLite(set[setIndex]);
if (fireEvent) {
element.triggerHandler('$destroy');
} else {
fireEvent = !fireEvent;
}
for(childIndex = 0, childLength = (children = element.children()).length;
childIndex < childLength;
childIndex++) {
list.push(jQuery(children[childIndex]));
element, childIndex, childLength, children;

if (!getterIfNoArguments || param != null) {
while(list.length) {
set = list.shift();
for(setIndex = 0, setLength = set.length; setIndex < setLength; setIndex++) {
element = jqLite(set[setIndex]);
if (fireEvent) {
element.triggerHandler('$destroy');
} else {
fireEvent = !fireEvent;
}
for(childIndex = 0, childLength = (children = element.children()).length;

This comment has been minimized.

Copy link
@stsvilik

stsvilik Apr 18, 2014

This loop seem to cause performance issues and seems like it can easily be replaced by a more efficient code.
Is there any reason why it can't be written like so?

list = list.concat(element.children().toArray());

Why do you need to wrap every child in jQuery just to re-wrap the original element in jqLite (line 131)?
Every time I use .html() to replace a big chunk of code it take roughly 300ms just to go through this loop.

This comment has been minimized.

Copy link
@btesser

btesser Apr 19, 2014

Contributor

jqlite doesn't support toArray. But I'm sure they could do list.push children.eq(childIndex). I don't really understand why the list needs jQuery wrapped elements but the element is only jqlite wrapped

This comment has been minimized.

Copy link
@stsvilik

stsvilik Apr 22, 2014

I think this while loop could easily be replaced by broadcast-style recursive function which will fire $destroy for every child there is.

childIndex < childLength;
childIndex++) {
list.push(jQuery(children[childIndex]));
}
}
}
}
Expand Down
48 changes: 47 additions & 1 deletion test/jQueryPatchSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -49,9 +49,55 @@ if (window.jQuery) {
doc.empty();
});

it('should fire on html()', function() {
it('should fire on html(param)', function() {
doc.html('abc');
});

it('should fire on html(\'\')', function() {
doc.html('');
});
});
});

describe('jQuery patch eagerness', function() {

var doc = null;
var divSpy = null;
var spy1 = null;
var spy2 = null;

beforeEach(function() {
divSpy = jasmine.createSpy('div.$destroy');
spy1 = jasmine.createSpy('span1.$destroy');
spy2 = jasmine.createSpy('span2.$destroy');
doc = $('<div><span class=first>abc</span><span class=second>xyz</span></div>');
doc.find('span.first').bind('$destroy', spy1);
doc.find('span.second').bind('$destroy', spy2);
});

afterEach(function() {
expect(divSpy).not.toHaveBeenCalled();
expect(spy1).not.toHaveBeenCalled();
});

describe('$detach event is not invoked in too many cases', function() {

it('should fire only on matched elements on detach(selector)', function() {
doc.find('span').detach('.second');
expect(spy2).toHaveBeenCalled();
expect(spy2.callCount).toEqual(1);
});

it('should fire only on matched elements on remove(selector)', function() {
doc.find('span').remove('.second');
expect(spy2).toHaveBeenCalled();
expect(spy2.callCount).toEqual(1);
});

it('should not fire on html()', function() {
doc.html();
expect(spy2).not.toHaveBeenCalled();
});
});
});
}

0 comments on commit da5f537

Please sign in to comment.