-
Notifications
You must be signed in to change notification settings - Fork 27.4k
fix(orderBy): ensure correct ordering with arrays of objects and no predicate #12072
Conversation
I guess this should be backported to 1.3.x - but not to 1.2.x since it uses |
Here is a plunker with this build of AngularJS demonstrating the original bug is fixed: http://plnkr.co/edit/WxAW671zu7BePcN4wgBn?p=preview |
expect(orderBy([2, 1, 3], [''])).toEqual([1, 2, 3]); | ||
// expect(orderBy([2, 1, 3], '')).toEqual([1, 2, 3]); | ||
// expect(orderBy([2, 1, 3], [])).toEqual([1, 2, 3]); | ||
// expect(orderBy([2, 1, 3], [''])).toEqual([1, 2, 3]); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this intentionally commented out?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oops. No
Seems to be working fine when setting reverse true, although it doesn't seem to be working when using the '-' notation instead of reverse true for array with objects. Pulnker with the issue: |
Agh! I knew it wasn't going to be so easy. |
235849f
to
7020ecc
Compare
OK, so I fixed it... Actually, the reason for the refactoring was that it made this scenario much easier to handle. But I am not 100% happy that we have to guess at whether the object is a POJO or not, when we decide to use the index rather than the identity for its ordering value. Also, we now treat |
Agh! So safari wasn't happy comparing nulls, so I have now fixed that too |
For example in this plunkr is this behaviour desired for the third ng-repeat?: http://plnkr.co/edit/gyw8B10H9tpGdu77g1e7?p=preview According to documentation if reverse is true it should reverse the order of the array, no matter what type the contents are. It's not that clear what it should do with '-'. EDIT: Unless we assume that orderBy will always order the array in ascending order unless it's a POJO where it would order by index. But personally I would rather make it order by index if it's object like and ascending if it's a string/number. |
if (!(isArrayLike(array))) return array; | ||
sortPredicate = isArray(sortPredicate) ? sortPredicate : [sortPredicate]; | ||
|
||
if (!isArray(sortPredicate)) { sortPredicate = [sortPredicate]; } | ||
if (sortPredicate.length === 0) { sortPredicate = ['+']; } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should this be
sortPredicate.push('+');
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Mmm... thinking a little more about this, I think my proposal and the current code are wrong as we are modifying the original sortPredicate
and we should not do that
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree. We should not modify the passed in predicate
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How is the current implementation modifying the passed in predicate ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's not. But I agree that we should start doing that either :-)
Do you think it would make the code any harder to keep the original behavior? I do not mind this going one way or another, but do think this is a small breaking change |
I'm guessing that this "small breaking change" might be accidentally fixing (or at least changing behavior of) #11312? |
I think we can probably move |
cafc829
to
5fb4e14
Compare
No breaking change no more: the algorithm sorts nulls in the same way as
This means that dates are sorted as expected, but also that arrays are sorted by their I have noted that the idiom used is a Schwartzian Transform and that it closes #4282. |
|
||
function doComparison(v1, v2) { | ||
var result = 0; | ||
for (var index=0, length = predicates.length; index < length; ++index) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
for (...; !result && index < length; ...) {
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I like
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But it is a bit obscure. I would prefer to make it more explicit:
function doComparison(v1, v2) {
var result = 0;
for (var index=0, length = predicates.length; index < length; ++index) {
result = compare(v1.predicateValues[index], v2.predicateValues[index]) * predicates[index].descending;
if (result) break;
}
return result;
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Works for me
@lgalfaso I updated the code. |
LGTM |
…redicate By refactoring to use a Schwartzian transform, we can ensure that objects with no custom `toString` or `toValue` methods are just ordered using their position in the original collection. Closes angular#11866 Closes angular#11312 Closes angular#4282
d317e1e
to
afdd1df
Compare
This fix also considerably refactors how the
orderBy
filter works in internally.The new algorithm precomputes the predicate values for the array being ordered. This makes the algorith easier to follow but also ensures that this computation is done a maximum of n times, whereas in the previous algorithm it could be greater. The downside is that we must temporarily store this intermediate array.
There could be performance implications, but they are as likely to be beneficial as detrimental. It could probably do with benchmarking.
Closes #11866