在jQuery对话框中敲除'with'binding和select2

时间:2015-05-16 10:05:42

标签: jquery knockout.js binding jquery-select2 jquery-dialog

问题:

当在嵌套在使用knockout with数据绑定的元素下的jQuery对话框上使用时,select2 jQuery插件不起作用。删除with绑定,select2工作正常。如果with绑定到嵌套属性,则它将停止工作。

背景

所以我必须在3小时的最佳时间里努力争取让select2在jQuery对话框表单上工作....谈论对于谚语错误的树,我认为它纯粹是jQuery对话框和选择2。它可能从一开始就使用_allowInteraction修复。直到我把问题直接解决为简单的步骤,并且因为开始显露自己。问题在于with绑定。

声明

道歉,因为我在阻止jsFiddle的asinine公司工作。此外,由于实际模型非常大,我为了说明目的而细分了我的实现。

// models

function Department() {
  this.name         = ko.observable('dept1');
  this.selectedTeam = ko.observable( new Team() );
}
    
function Team() {
  this.name = ko.observable('team1');
}
    
function MainModel() {
  this.department = new Department();
  this.showTeam   = function() { 
    $('#addTeamDialog').dialog('open');
  };
}

// setup

ko.applyBindings( new MainModel() );
    	
$('#addTeamDialog').dialog({
  // fix allow select2 to work on the jq dialog
  _allowInteraction: function (event) {
    return !!$(event.target).is(".select2-input") || this._super(event);
  }		
});
    	
$('#someList').select2({
  data: [
    { id: 0, text: 'enhancement' },
    { id: 1, text: 'bug' },
    { id: 2, text: 'duplicate' },
    { id: 3, text: 'invalid' },
    { id: 4, text: 'wontfix' }
  ]
});
<link href="http://code.jquery.com/ui/1.11.4/themes/ui-lightness/jquery-ui.css" rel="stylesheet"/>
<link href="//cdnjs.cloudflare.com/ajax/libs/select2/4.0.0/css/select2.min.css" rel="stylesheet" />

<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<script src="http://code.jquery.com/ui/1.11.4/jquery-ui.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.2.0/knockout-min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/select2/4.0.0/js/select2.full.min.js"></script>

<button data-bind="click: showTeam">Add Team</button>

<div id="addTeamDialog">
  <fieldset data-bind="with: department">
    
    <div class="lite-dialog-field">
      <div class="label">
        <span data-bind="text: name"></span>
      </div>
      <div class="field">
        <input type="hidden" id="someList" />
      </div>    
    </div>
        
  </fieldset>
</div>

删除data-bind上的fieldset和select2可以正常工作。

data-bind上的fieldset设置为department时,select2正常工作。

data-bind上的fieldset设置为department.selectedTeam时,select2无效。

1 个答案:

答案 0 :(得分:4)

当你使用Knockout时,强烈建议在绑定中包装select2等外部库。虽然您只对其进行初始化一次,但withtemplateforeach等绑定可以在此之后的任何时间修改DOM。

你面临着

的危险
  1. 在Knockout尚未渲染任何内容时过早地初始化select2,或
  2. 淘汰并在稍后重新渲染标记,以便你的select2突然不再被绑定
  3. 例如,当Department.selectedTeam发生变化时会发生这种情况。

    我从Knockouts找到了一个快速而又脏的select2绑定&#39; rniemeyer himself here。除此之外,为了保持一致性和安全性,我只将select2标记更改为标准<select>并将MainModel.department变为适当的可观察值。

    &#13;
    &#13;
    ko.bindingHandlers.select2 = {
        init: function(element, valueAccessor) {
          var options = ko.toJS(valueAccessor()) || {};
          setTimeout(function() { 
              $(element).select2(options);
          }, 0);
        }
    };
    
    // models
    
    function Department() {
      this.name         = ko.observable('dept1');
      this.selectedTeam = ko.observable( new Team() );
    };
        
    function Team() {
      this.name     = ko.observable('team1');
      this.values   = ["red", "grey", "blue"];
      this.selected = ko.observableArray(["blue"]);
    };
        
    function MainModel() {
      this.department = ko.observable( new Department() );
      this.showTeam   = function() { 
        $('#addTeamDialog').dialog('open');
      };
    };
    
    // setup
    
    ko.applyBindings( new MainModel() );
        	
    $('#addTeamDialog').dialog({
      // fix allow select2 to work on the jq dialog
      _allowInteraction: function (event) {
        return !!$(event.target).is(".select2-input") || this._super(event);
      }		
    });
    &#13;
    select {
      width: 200px;
    }
    &#13;
    <link href="http://code.jquery.com/ui/1.11.4/themes/ui-lightness/jquery-ui.css" rel="stylesheet"/>
    <link href="//cdnjs.cloudflare.com/ajax/libs/select2/4.0.0/css/select2.min.css" rel="stylesheet" />
    <script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
    <script src="http://code.jquery.com/ui/1.11.4/jquery-ui.min.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.2.0/knockout-min.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/select2/4.0.0/js/select2.full.min.js"></script>
    
    <button data-bind="click: showTeam">Add Team</button>
    
    <div id="addTeamDialog">
      <fieldset data-bind="with: department().selectedTeam">
        
        <select data-bind="options: values,
                           selectedOptions: selected, 
                           select2: { placeholder: 'pick some colors' }">
        </select>
            
      </fieldset>
    </div>
    &#13;
    &#13;
    &#13;