使用Backbone.js我已经设置了一个带有比较器功能的集合。这很好地排序模型,但我想颠倒顺序。
如何按降序排序模型而不是升序?
答案 0 :(得分:133)
好吧,你可以从比较器返回负值。例如,如果我们采用Backbone网站上的示例并希望颠倒顺序,它将如下所示:
var Chapter = Backbone.Model;
var chapters = new Backbone.Collection;
chapters.comparator = function(chapter) {
return -chapter.get("page"); // Note the minus!
};
chapters.add(new Chapter({page: 9, title: "The End"}));
chapters.add(new Chapter({page: 5, title: "The Middle"}));
chapters.add(new Chapter({page: 1, title: "The Beginning"}));
alert(chapters.pluck('title'));
答案 1 :(得分:56)
就个人而言,我对这里给出的任何解决方案都不满意:
当排序类型为非数字时,乘以-1的解决方案无效。虽然有很多方法(例如通过调用Date.getTime()
),但这些案例是具体的。我想要一种通用的方法来反转任何类型的方向,而不必担心被分类字段的具体类型。
对于字符串解决方案,一次构造一个字符串似乎是性能瓶颈,尤其是在使用大型集合(或者实际上是大型排序字段字符串)时
这是一个适用于字符串,日期和数字字段的解决方案:
首先,声明一个比较器,它将反转sortBy函数的结果,如下所示:
function reverseSortBy(sortByFunction) {
return function(left, right) {
var l = sortByFunction(left);
var r = sortByFunction(right);
if (l === void 0) return -1;
if (r === void 0) return 1;
return l < r ? 1 : l > r ? -1 : 0;
};
}
现在,如果你想要改变排序的方向,我们需要做的就是:
var Chapter = Backbone.Model;
var chapters = new Backbone.Collection;
// Your normal sortBy function
chapters.comparator = function(chapter) {
return chapter.get("title");
};
// If you want to reverse the direction of the sort, apply
// the reverseSortBy function.
// Assuming the reverse flag is kept in a boolean var called reverseDirection
if(reverseDirection) {
chapters.comparator = reverseSortBy(chapters.comparator);
}
chapters.add(new Chapter({page: 9, title: "The End"}));
chapters.add(new Chapter({page: 5, title: "The Middle"}));
chapters.add(new Chapter({page: 1, title: "The Beginning"}));
alert(chapters.pluck('title'));
这有效的原因是,如果sort函数有两个参数,Backbone.Collection.sort
的行为会有所不同。在这种情况下,它的行为方式与比较器传入Array.sort
的方式相同。此解决方案的工作原理是将单个参数sortBy函数调整为两个参数比较器并反转结果。
答案 2 :(得分:46)
Backbone.js的集合比较器依赖于Underscore.js方法_.sortBy。 sortBy实现的方式最终以一种使反向排序字符串变得困难的方式“包装”javascript .sort()。简单地否定字符串最终返回NaN并打破排序。
如果你需要使用字符串执行反向排序,例如反向字母排序,这是一种真正的hackish方式:
comparator: function (Model) {
var str = Model.get("name");
str = str.toLowerCase();
str = str.split("");
str = _.map(str, function(letter) {
return String.fromCharCode(-(letter.charCodeAt(0)));
});
return str;
}
它绝不是漂亮的,但它是一个“字符串否定者”。如果您对javascript中的本机对象类型没有任何疑虑,可以通过提取字符串否定符并将其作为String.Prototype上的方法添加来使代码更清晰。但是,如果你知道自己在做什么,你可能只想这样做,因为在javascript中修改本机对象类型会产生无法预料的后果。
答案 3 :(得分:28)
我的解决方案是在排序后反转结果。
为防止双重渲染首先使用静音排序,然后触发'重置'。
collections.sort({silent:true})
collections.models = collections.models.reverse();
collections.trigger('reset', collections, {});
答案 4 :(得分:13)
修改比较器函数以返回一些反向的proporitional值,而不是返回您当前的数据。 一些代码来自:http://documentcloud.github.com/backbone/#Collection-comparator
示例:
var Chapter = Backbone.Model;
var chapters = new Backbone.Collection;
/* Method 1: This sorts by page number */
chapters.comparator = function(chapter) {
return chapter.get("page");
};
/* Method 2: This sorts by page number in reverse */
chapters.comparator = function(chapter) {
return -chapter.get("page");
};
chapters.add(new Chapter({page: 9, title: "The End"}));
chapters.add(new Chapter({page: 5, title: "The Middle"}));
chapters.add(new Chapter({page: 1, title: "The Beginning"}));
答案 5 :(得分:9)
以下对我有用:
comparator: function(a, b) {
// Optional call if you want case insensitive
name1 = a.get('name').toLowerCase();
name2 = b.get('name').toLowerCase();
if name1 < name2
ret = -1;
else if name1 > name2
ret = 1;
else
ret = 0;
if this.sort_dir === "desc"
ret = -ret
return ret;
}
collection.sort_dir = "asc";
collection.sort(); // returns collection in ascending order
collection.sort_dir = "desc";
collection.sort(); // returns collection in descending order
答案 6 :(得分:8)
我读了一些Backbone比较器函数使用或模仿JavaScript Array.prototype.sort()函数的地方。这意味着它使用1,-1或0来决定集合中每个模型的排序顺序。这会不断更新,并且基于比较器,收集将保持正确的顺序。
有关Array.prototype.sort()的信息,请访问:https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort?redirectlocale=en-US&redirectslug=JavaScript%2FReference%2FGlobal_Objects%2FArray%2Fsort
关于此问题的许多答案只是在将其退回页面之前撤消订单。这不太理想。
使用上面关于Mozilla的文章作为指南,我能够使用以下代码以相反的顺序对我的集合进行排序,然后我们只是将集合呈现为以当前顺序排列,我们可以使用标志(reverseSortDirection)扭转秩序。
//Assuming this or similar will be in your Backbone.Collection.
sortKey: 'id', //The field name of the item in your collection
reverseSortDirection: false,
comparator: function(a, b) {
var sampleDataA = a.get(this.sortKey),
sampleDataB = b.get(this.sortKey);
if (this.reverseSortDirection) {
if (sampleDataA > sampleDataB) { return -1; }
if (sampleDataB > sampleDataA) { return 1; }
return 0;
} else {
if (sampleDataA < sampleDataB) { return -1; }
if (sampleDataB < sampleDataA) { return 1; }
return 0;
}
},
除了两个模型之外的比较器,对集合或模型的更改运行隔离函数并更改集合的顺序。
我已经使用Numbers和Strings测试了这种方法,它似乎对我很有效。
我真的希望这个答案可以提供帮助,因为我多次努力解决这个问题并且通常最终会使用解决方法。
答案 7 :(得分:5)
这可以通过重写sortBy方法来优雅地完成。这是一个例子
var SortedCollection = Backbone.Collection.extend({
initialize: function () {
// Default sort field and direction
this.sortField = "name";
this.sortDirection = "ASC";
},
setSortField: function (field, direction) {
this.sortField = field;
this.sortDirection = direction;
},
comparator: function (m) {
return m.get(this.sortField);
},
// Overriding sortBy (copied from underscore and just swapping left and right for reverse sort)
sortBy: function (iterator, context) {
var obj = this.models,
direction = this.sortDirection;
return _.pluck(_.map(obj, function (value, index, list) {
return {
value: value,
index: index,
criteria: iterator.call(context, value, index, list)
};
}).sort(function (left, right) {
// swap a and b for reverse sort
var a = direction === "ASC" ? left.criteria : right.criteria,
b = direction === "ASC" ? right.criteria : left.criteria;
if (a !== b) {
if (a > b || a === void 0) return 1;
if (a < b || b === void 0) return -1;
}
return left.index < right.index ? -1 : 1;
}), 'value');
}
});
所以你可以像这样使用它:
var collection = new SortedCollection([
{ name: "Ida", age: 26 },
{ name: "Tim", age: 5 },
{ name: "Rob", age: 55 }
]);
//sort by "age" asc
collection.setSortField("age", "ASC");
collection.sort();
//sort by "age" desc
collection.setSortField("age", "DESC");
collection.sort();
此解决方案不依赖于字段类型。
答案 8 :(得分:3)
// For numbers, dates, and other number-like things
var things = new Backbone.Collection;
things.comparator = function (a, b) { return b - a };
// For strings
things.comparator = function (a, b) {
if (a === b) return 0;
return a < b ? -1 : 1;
};
答案 9 :(得分:2)
对于那些只想翻转当前订单的人来说,这是一个非常简单的解决方案。如果您有一个表,其列可以在两个方向上排序,那么它很有用。
collection.models = collection.models.reverse()
如果您对表格案例(coffeescript)感兴趣,可以将它与此类结合起来:
collection.comparator = (model) ->
model.get(sortByType)
collection.sort()
if reverseOrder
collection.models = collection.models.reverse()
答案 10 :(得分:1)
对于id:
的逆序comparator: function(object) { return (this.length - object.id) + 1; }
答案 11 :(得分:0)
我的要求略有不同,因为我在表格中显示我的模型,我希望能够通过任何collumn(模型属性)以及升序或降序对它们进行排序。
为了让所有案例都有效,它已经花费了相当多的时间(值可以是字符串,空字符串,日期,数字。但是我认为这非常接近。
inverseString: function(str)
{
return str.split("").map(function (letter) { return String.fromCharCode(-(letter.charCodeAt(0))); }).join("");
}
getComparisonValue: function (val, desc)
{
var v = 0;
// Is a string
if (typeof val === "string")
{
// Is an empty string, upgrade to a space to avoid strange ordering
if(val.length === 0)
{
val = " ";
return desc ? this.inverseString(val) : val;
}
// Is a (string) representing a number
v = Number(val);
if (!isNaN(v))
{
return desc ? -1 * v : v;
}
// Is a (string) representing a date
v = Date.parse(val);
if (!isNaN(v))
{
return desc ? -1 * v : v;
}
// Is just a string
return desc ? this.inverseString(val) : val;
}
// Not a string
else
{
return desc ? -1 * val : val;
}
},
使用:
comparator: function (item)
{
// Store the collumn to 'sortby' in my global model
var property = app.collections.global.get("sortby");
// Store the direction of the sorting also in the global model
var desc = app.collections.global.get("direction") === "DESC";
// perform a comparison based on property & direction
return this.getComparisonValue(item.get(property), desc);
},
答案 12 :(得分:0)
我刚刚覆盖sort函数以在完成时反转模型数组。我也不再触发“排序”。事件发生之前直到反向完成。
sort : function(options){
options = options || {};
Backbone.Collection.prototype.sort.call(this, {silent:true});
this.models.reverse();
if (!options.silent){
this.trigger('sort', this, options);
}
return this;
},
答案 13 :(得分:0)
最近遇到了这个问题,我决定添加一个rsort方法。
Collection = Backbone.Collection.extend({
comparator:function(a, b){
return a>b;
},
rsort:function(){
var comparator = this.comparator;
this.comparator = function(){
return -comparator.apply(this, arguments);
}
this.sort();
this.comparator = comparator;
}
});
希望有人能找到这个有用的
答案 14 :(得分:0)
这是使用UnderscoreJS对集合模型进行反向排序的快捷方法
var reverseCollection = _.sortBy(this.models, function(model) {
return self.indexOf(model) * -1;
});
&#13;