I have a Backbone.js collection that I would like to be able to sort using jQuery UI's Sortable. Nothing fancy, I just have a list that I would like to be able to sort.
The problem is that I'm not sure how to get the current order of items after being sorted and communicate that to the collection. Sortable can serialize itself, but that won't give me the model data I need to give to the collection.
Ideally, I'd like to be able to just get an array of the current order of the models in the collection and use the reset method for the collection, but I'm not sure how to get the current order. Please share any ideas or examples for getting an array with the current model order.
http://jsfiddle.net/aJjW6/2/
HTML:
JavaScript:
CSS:
Just use Backbone.CollectionView!
Voila!
I've done this by using jQuery UI Sortable to trigger an event on the item view when an item is dropped. I can then trigger another event on the item view that includes the model as data which the collection view is bound to. The collection view can then be responsible for updating the sort order.
Working example
http://jsfiddle.net/7X4PX/260/
jQuery UI Sortable
The stop event is bound to a function that triggers
drop
on the DOM node for the item with the item's index (provided by jQuery UI) as data.Item view
The drop event is bound to the
drop
function which triggers anupdate-sort
event on the item view's DOM node with the data[this.model, index]
. That means we are passing the current model and it's index (from jQuery UI sortable) to whomever is bound to theupdate-sort
event.Items (collection) view
The
Items
view is bound to theupdate-sort
event and the function uses the data passed by the event (model and index). The model is removed from the collection, theordinal
attribute is updated on each remaining item and the order of items by id is sent to the server to store state.Collection
The collection has a comparator function defined which orders the collection by
ordinal
. This keeps the rendered order of items in sync as the "default order" of the collection is now by the value of theordinal
attribute.Note there is some duplication of effort: the model doesn't need to be removed and added back to the collection if a collection has a comparator function as the jsfiddle does. Also the view may not need to re-render itself.
Note: compared to the other answer, my feeling was that it was more correct to notify the model instance of the item that it needed to be updated instead of the collection directly. Both approaches are valid. The other answer here goes directly to the collection instead of taking the model-first approach. Pick whichever makes more sense to you.