1

我有两个表单元素,都是通过backbone.stickit 进行双向数据绑定的。第二个表单元素 ( #input ) 只是装饰物 - 用于显示它确实在工作。

这个想法是,每次下拉(#select)菜单中的选项发生变化时,我的视图都会被(重新)渲染。

我试图通过捕捉 #select 的“更改”事件并调用 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/

4

1 回答 1

1

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

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

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

于 2016-04-03T05:58:01.757 回答