如何在Iron Router的onBeforeAction中等待http调用?

dan*_*bes 11 javascript meteor iron-router

我想创建一个预加载脚本,执行许多异步函数来下载外部内容.我离这里很近,但我还没弄明白如何在我的onBeforeAction函数中推迟调用this.next().在下面的代码中,您可以看到我使用循环和setTimeout但是我以某种方式丢失了路由器的上下文,并且this.next()未定义.我相信这是因为我的preloadProject函数到了最后,路由器警告我,我忘了调用this.next(); 在我的waitToRender功能完成之前.

if (Meteor.isClient) {
    IR_BeforeHooks = {
        preloadProject: function() {
            var itemsProcessed = 0;
            _.each(items.items, function(e) {
                    HTTP.get(e.S3URL, {
                            headers: {
                                'Accept': '*/*'
                            },
                            responseType: 'arraybuffer' //requires aldeed:http
                        }, function(error, result) {
                            if (error) {
                                Session.set('error', {
                                    'title': 'Could not download',
                                    'message': error
                                });
                            }
                            if (result) {
                                itemsProcessed = itemsProcessed + 1;
                            }
                        }) //http get
                }) //each
            function waitToRender(router) {
                console.log('waiting...')
                var progress = (itemsProcessed / items.items.length) * 100;
                if (progress < 100) {
                    $('.progress-bar').css('width', Math.floor(progress) + '%');
                    setTimeout((function() {
                        waitToRender(router);
                    }), 50);
                    //console.log('timeout for a few ms here')
                }
                else {
                    console.log('all done, render');
                    router.next(); // I get this is not a function error here
                }
            }
            waitToRender(this);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

和我的路由器

Router.before(
    IR_BeforeHooks.preloadProject, 
    {
        only:['editor', 'viewer', 'embedder']
    }
);
Run Code Online (Sandbox Code Playgroud)

由于我项目中的复杂需求,我需要在渲染之前执行这些任务.如何在不阻止浏览器的情况下让Iron Router等待我的http调用?

dan*_*bes 1

我找到了解决方案。正如我怀疑的那样,问题在于当我的主要钩子函数完成其同步工作时,Iron Router 正在将自身归零。因此,为了解决这个问题,我在开始工作时设置了自己的 this.next 副本,并在异步工作完成时调用该函数的副本。

IR_BeforeHooks = {
  preloadProject: function() {
    var myNext = this.next;
    ...//http calls and etc
    function waitToRender(router) {
      console.log('waiting...')
      var progress = (itemsProcessed / items.items.length) * 100;
      if (progress < 100) {
        ...//use set timeout to recall waitToRender
      }
      else{
        myNext();
      }
    }
  }
}
Run Code Online (Sandbox Code Playgroud)