(重新)渲染更改事件处理程序中的Backbone视图不起作用

时间:2016-04-03 00:47:16

标签: javascript backbone.js backbone-views backbone-events backbone-stickit

我有两个表单元素,都是通过backbone.stickit进行双向数据绑定。 第二个表单元素( #input )只是化妆品 - 显示它实际上正在工作。

每次下拉( #select )菜单中的选项发生变化时,我的想法就是(重新)呈现我的视图。

我试图通过捕获#select的'changed'事件并调用this.render()来(重新)渲染视图来实现这一点。

显然这不起作用。选定的选项不会保存回模型中,我无法理解原因。

我不是在寻找解决方案,而不是解释,为什么以下代码不起作用。解决方案(如:对我有用)是小提琴的一部分 - 注释掉。

HTML:

<script type="text/template" id="tpl">
  <h1>Hello <%= select %></h1>
  <select id="select">
  </select>
  <p>Select:
    <%= select %>
  </p>
  <hr>
  <input type="text" id="input">
  <p>Input:
    <%= input %>
  </p>
</script>

<div id="ctr"></div>

JavaScript的:

Foo = Backbone.Model.extend({
  defaults: {
    select: "",
    input: "",
  }
});
FooView = Backbone.View.extend({
  el: '#ctr',
  template: _.template($('#tpl').html()),
  initialize() {
    this.model.bind('change', function() {
      console.log("model change:");
      console.log(this.model.get('select'));
      console.log(this.model.get('input'));
    }, this);
    //this.model.bind('change:select', function() { this.render(); }, this); // <--------------------- WORKS
  },
  render: function() {
    this.$el.html(this.template(this.model.toJSON()));
    this.stickit();
    return this;
  },
  events: {
    'change #select': function(ev) {
      console.log('change event triggered:');
      console.log(this.model.get('select'));
      console.log(this.model.get('input'));
      this.render(); // <--------------------- DOES NOT WORK - WHY?
    },
    /* 'click #render': function(ev) {
      console.log('render event triggered:');
      console.log(this.model.get('select'));
      console.log(this.model.get('input'));
      this.render();
    } */
  },
  bindings: {
    '#input': 'input',
    '#select': {
      observe: 'select',
      selectOptions: {
        collection: function() {
          return [{
            value: '1',
            label: 'Foo'
          }, {
            value: '2',
            label: 'Bar'
          }, {
            value: '3',
            label: 'Blub'
          }]
        }
      }
    },
  },
});
new FooView({
  model: new Foo()
}).render();

https://jsfiddle.net/r7vL9u07/9/

1 个答案:

答案 0 :(得分:1)

this.render()事件处理程序中调用change #select无效的原因是因为您正在破坏Backbone.stickit为您提供的双向数据绑定。流程如下:

  • 用户更改&#39; #select&#39;。
  • 的值
  • 您的change #select处理程序触发并调用this.render()
  • render使用未选择#ctr的新选择菜单重新填充option
  • Backbone.stickit响应对#select的更改。
  • Backbone.stickit尝试获取#select的值,但由于它不包含选定的option,因此值为undefined
  • Backbone.sticket将model&#39; select属性设置为undefined

如果将this.render()调用移到model change:select处理程序中,它的工作原理是因为Backbone.stickit能够正确更新模型DOM在获得机会之前发生变化。