Meteor - Using collection on client startup
Asked Answered
T

2

22

Why this code shows "0"? Shouldn't it return "1"?

Messages = new Meteor.Collection("messages");

if (Meteor.is_client) {
    Meteor.startup(function () {    
        alert(Messages.find().count());
    });
}

if (Meteor.is_server) {
    Meteor.startup(function () {
        Messages.insert({text: "server says hello"});
    });
}

If I do the "Messages.find().count()" later, it returns 1.

Troup answered 11/4, 2012 at 4:50 Comment(3)
@Raynos It doesn't look like a concurrency problem. It happens every time the client startup handler is executed.Troup
It is a concurrency issue, the issue is that your running two things concurrently one on the server and one on the client. and your expecting the server one to magically finish firstCopulate
@Copulate The insert in the server runs only once, when the server is started. The find in the client runs multiple times any time later.Troup
T
37

By default, when a Meteor client starts up, it connects to the server and subscribes to documents in any Meteor.Collection you defined. That takes some time to complete, since there's always some amount of delay in establishing the server connection and receiving documents.

Meteor.startup() on the client is a lot like $() in jQuery -- it runs its argument once the client DOM is ready. It does not wait for your client's collections to receive all their documents from the server. So the way you wrote the code, the call to find() will always run too early and return 0.

If you want to wait to run code until after a collection is first downloaded from the server, you need to use Meteor.subscribe() to explicitly subscribe to a collection. subscribe() takes a callback that will run when the initial set of documents are on the client.

See:

meteor-publish and meteor-subscribe

Textual answered 11/4, 2012 at 6:29 Comment(2)
When using Meteor.subscribe() do i have to somehow turn off the default "subscribe to all documents of a collection" behaviour?Clip
@Clip (or for others who'd have the same question), yes you need to remove the autopublish package: $ meteor remove autopublish.Apropos
A
0

Just to follow up with a code example of how to know when a collection is ready to use on the client.

As @debergalis described, you should use the Meteor.subscribe approach - it accepts a couple of callbacks, notably onReady

For example:

if(Meteor.isClient){

    Meteor.subscribe("myCollection", {

        onReady: function(){

            // do stuff with my collection

        }

    });

}
Ana answered 26/5, 2016 at 1:12 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.