Javascript函数执行两次,Meteor js

时间:2016-01-07 18:22:02

标签: javascript meteor iron-router

这是一个水平财富轮盘的简单脚本,使用光滑的滑块创建。问题:函数rollout()执行了两次(以及所有帮助程序),因为模板呈现了两次。我使用铁:路由器进行路由,代码:

Router.configure({
  loadingTemplate: "loading",
  layoutTemplate: "layout"
});
Router.map(function() {
  this.route("hello", {
    path: "/",
    waitOn: function() {
      return Meteor.subscribe("games");
    }
  });
});

hello.js:

Session.setDefault('currentSlide', 0);
    Template.hello.events({
      'click .create': function(e) {
        e.preventDefault();
        var ins = {
          gameStartTime: new Date(),
          finished: false
        }
        Games.insert({
          gameStartTime: new Date(),
          finished: "false"
        });
        Session.set("gameStatus", "waiting");
      }
    })
    Template.hello.helpers({
      currentSlide: function() {
        var value = Session.get('currentSlide');
        if (value === 25) {
          return 0;
        }
        return value;
      },
      games: function() {
        var game = Games.find();
        return game;
      },
      gameHandler: function() {
        var game = Games.findOne();
        var gameStatus = Session.get("gameStatus");
        if (game && game.finished === "false" && gameStatus === "waiting") {
          Games.update({_id: game._id}, {
            $set: {running: true}
          });

        } else if (gameStatus === "finished") {

          Meteor.call('updateGame', game._id); //remove item from collection after game over; Error here: TypeError: Cannot read property '_id' of undefined; 
          console.log('game finished');
        }
        if (game && game.running === true) {
          //Set Session variable and launch roulette
          Session.set("gameStatus", "running");
          rollout(game._id);
        }
      }
    });
    var rollout = function(gameid) {
      Session.set("gameStatus", "running");
      //make div's visible. They are have display:none by default.
      document.getElementById("roulette").style.display = "block";
      document.getElementById("result").style.display = "block";
      document.getElementById("wrapper").style.display = "none";
      var speed = 500; //default slider speed
      Session.setDefault('speed', speed);
      var r = $('#roulette').slick({
        centerMode: true,
        slidesToShow: 7,
        slidesToScroll: 1,
        autoplay: true,
        autoplaySpeed: 0,
        speed: speed,
        draggable: false,
        pauseOnHover: false,
        cssEase:'linear'
      });
      var maximum = 14;
      var minimum = 0;
      var randomnumber = Math.floor(Math.random() * (maximum - minimum + 1)) + minimum;
      var winResult = randomnumber;
      var speed = 80;
      var currentSlide = Session.get('currentSlide');
      function roll(callback) {
        Session.set('currentSlide', r.slickCurrentSlide());
        if (speed < 800) {
          speed += 20;
          r.slickSetOption("speed", speed, false);
          setTimeout(function() {
            roll(callback);
          }, 500);
        } else if (speed >= 600 && speed < 1300) {
          speed += 40;
          r.slickSetOption("speed", speed, false);
          setTimeout(function() {
            roll(callback);
          }, 300);
        } else if (speed >= 1300 && speed < 20000) {
          setTimeout(function() {
            speed += 3;
            r.slickSetOption('speed', speed, false);
            if (Session.get('currentSlide') === winResult) {
              r.slickPause();
              setTimeout(function() {
                //hide roulette div's
                document.getElementById("roulette").style.display = "none";
                document.getElementById("result").style.display = "none";
                document.getElementById("wrapper").style.display = "block";
                r.unslick();

                Session.set('currentSlide', 0);
                Session.set("gameStatus", "finished");
                callback(true);
                return
              }, 10000);
            } else {
              roll(callback);
            }
          }, 30)
        }  else {
          callback(true);
        }
      }
      roll(function(callback) {
        Meteor.call('updateGame', gameid); //remove item from collection
        Session.set("gameStatus", "finished");
      });
    }

在模板中我使用spinner by sacha:spin package并且工作正常。

1 个答案:

答案 0 :(得分:2)

gameHandler函数不应该在模板助手中,这只是为了向模板提供信息。相反,该逻辑需要移动到autorun function。只要其中的响应数据源发生更改,此autorun语句就会重新运行。

例如:

Template.hello.onCreated(function() {
    this.autorun(function() {
        var game = Games.findOne();
        var gameStatus = Session.get("gameStatus");
        // ...etc
    });
}