-
Notifications
You must be signed in to change notification settings - Fork 10
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
How do we add an item after/before another item or at a specified index of the collection? #3
Comments
I think that allowing users to set the item's id manually would really help here. What you can do now are some hacks like reimplementing the Collection.Pluck method to sort the items before rendering: function pluckWithSorting (collection$, sinkProperty, sortProperty) {
const sinks = {};
function sink$ (item) {
const key = `${item.name}.${item.id}.${sinkProperty}`;
if (sinks[key] === undefined) {
if (sinkProperty === 'DOM') {
sinks[key] = item[sinkProperty].map(vtree => ({...vtree, key})).remember();
} else {
sinks[key] = item[sinkProperty].remember();
}
}
return sinks[key];
}
return collection$
.map(collection => collection.asArray()
.sort(fst, snd => fst[sortProperty] - snd[sortProperty])
.map(item => sink$(item)))
.map(sinkStreams => xs.combine((...items) => items, ...sinkStreams))
.flatten()
.startWith([]); But this shouldn't be really performant as your collection size grows. Alternatively, you can use flexbox with CSS |
@Hypnosphi Unfortunately flexbox ordering would be a hack not a solution - it would work only if the output is the DOM. If we'd like to use collections as sinks for other drivers, like Canvas, SVG, Audio or anything else for that matter, there usually won't be any equivalent hack available. |
Btw, unrelated to this repository, API like addAfter(newItem, compareFunction) {
let index = this.findIndex(item => !compareFunction(item, newItem));
return this.slice().splice(index, 0, newItem);
} |
I have a weird thought. It's a common pattern that you want to apply a sort order to a collection. It's one thing to manually maintain a sort order by inserting at indexes, but what if we provided a Now the trick part is that often you want to sort on an observable, right? I think the default case should be a reactive sort. So the signature would actually look like What do y'all think? |
This would be post-processing again. What niieani wants is to have some kind of |
@Hypnosphi this indeed makes a lot of sense! Passing a sorting function to the collection definition would ensure the collection is always sorted - there's something so lovably "functional" about that notion. I suppose this will be the most common case. @Widdershin's proposition seems good too, for other use-cases - it could be that we want to sort a list "onClick" - or even to sort it differently on different observables (e.g. sortable DataGrid / Excel). So both features would be welcome additions. |
It's a common use case that a user wants to be able to change the sorting of a collection after creation. If we use My problem with the non-reactive approach is what are you actually sorting on? If you're sorting on the properties of your items, it's convention for those to be Observable. And then the return value from that As for the whole optimization thing, I wonder if we can cross that bridge when we come to it. Who knows how fast xstream works out to be in practice? This might be a good test. |
@Widdershin Are you saying that static data which comes from the server should be, as a convention, converted to Observables of Observables? Why? Is there any benefit in doing so if any updates to the data come in the form of new objects altogether? |
Items in a collection must have some static key anyway. Currently it's auto-generated but it could be provided by user as well. It's sorting by key what can be done instantly. All the sortings by dynamic properties should be done on the view stage, off course. |
You just turn the sequence of each item's states into a separate stream by mapping over a responses stream. You should do that if you want reactive updates. |
Totally with you now, all "reactive" sorting should be done on top of I'm still not clear what exactly we would be sorting on for the sort on addition style. Could someone provide a real world example? I'm fine with the notion of providing a |
Yes, it is. In the original case of messages timestamp would be the key. It's safe, as it doesn't seem to change in the future. You may be able to remove or even edit your messages, but not to change the sending time =) |
@niieani: have you had the chance to try out |
@Whiddershin How do you control the position of the added items with gather? |
Hmmm, @Hypnosphi, is this possible? Or are we missing a |
I'm open to adding that, but not really sure that it can be done solely inside of |
Sorting shouldn't be done on the idAttribute, but on an arbitrary attribute. A chat can receive 2 messages at exactly the same time and thus time cannot always be used as the ID. |
Sounds reasonable, but that attribute should anyway be static (plain value, not stream). Otherwise post-processing is cheaper. And currently |
I'm not really well versed in the internals of this library, but for whatever its worth, I'd think using idAttribute should be fine. Theoretically events could happen at exactly the same time, but in reality even 1 millisecond difference is going to have have 2 events processed one after the other. Javascript especially doesn't really do any 2 things at the exact same time. Feel free to tell me I'm wrong, I probably am :) |
Hi guys, any updates on that matter? I feel like sorting is an important subject, am I wrong? I am using a collection created via I am actually not sure why this behavior is not the default behavior (given that the first rendering is following the array's order). I might be doing things wrong here, but right now I don't really know how to sort the elements I add to my collection :( |
@atomrc I just faced this exact same issue. I use gather and when I change the order of the array in the observable ( 3rd param ) it doesn't sort ! Hmm |
Just coming back to this now: Seems like there are really two issues here. One is that we want to be able to sort all items inserted into the collection (including via gather), and it should be efficient. Let's call this insertion sort. Secondly, we want to be able to control the sort order of our items in response to their data changing. For My proposal for insertion sorting is to allow collection/gather to take a comparator function to using binary sort to insert new items. This could only operate on static data in the sources, not anything wrapped in an Observable. To that end, we can use my friend Roger's newly published library, sorted-immutable-list. I think we should make a new issue to discuss potential solutions for "reactive sort". I like the idea of respecting the order of the incoming stream of objects, but it seems trickier to implement so I would like to think it through. What say you @niieani @Hypnosphi @atomrc @FeliciousX? Would a binary insertion sort on non-observable data (ids and timestamps etc) solve your usecases? |
As long as the insertion sort method is configurable, the proposed solution should be fine. |
Imagine a chat app which implements offline support:
This is just one use case, but there are many others (like reordering lists, where you'd need to remove an item and re-add it in a different place).
I'd imagine a lot of the times you don't know the index and you'd actually need to find after/before which specific item you want your new item to appear with some sort of a search function.
Not sure how a pure/functional API would look like for this case, but if it were a classic mutable state, it'd be something like:
This would iterate on all the elements until the return value of the search is
false
, then insert the new item at the index of the first item that returnedfalse
.How can we deal with this case using cycle-collection?
The text was updated successfully, but these errors were encountered: