挖空:在不清除视图模型中的值的情况下更改选择列表中的选项
Knockout: changing options in select list without clearing value in view-model
我有一个基于 Knockout JS 的问题,一些级联选项列表并切换出与它们相关的底层"活动"对象。
我创建了一个 jsfiddle 来演示这个问题。
我有一个 UI,用户在其中编辑主"标题"记录并添加/删除/编辑子记录。有一个用于编辑子记录的中心区域。这个想法是单击表中的子记录,这将成为中间区域中正在编辑的记录。
我遇到的问题是由于第二个下拉列表中的事物列表会根据第一个下拉列表而变化。在活动记录更改之前,这很好。如果类别因活动记录更改而更改,则"事物"列表也会更改。此时,将清除新活动子记录上选定的"事物"(第二个下拉列表(。
我假设新活动记录上的值发生了变化,但被清除了,因为它没有出现在旧列表中(如果类别发生了变化(。然后更改项列表本身(包括相应的值(,但此时该值已从视图模型中消失。
(我意识到这是一个相当冗长的解释,希望jsfiddle能说清楚(
如何更改下拉列表中的项目列表和视图模型中的选定值,而不会在此过程中丢失所选值?
.HTML:
<label>Some header field</label>
<input type="text" id="txtSomeHeaderField" data-bind="value: HeaderField" />
<fieldset>
<legend>Active Child Record</legend>
<label>Category</label>
<select id="ddlCategory" data-bind="options: categories, value: ActiveChildRecord().Category, optionsCaption:'Select category...'" ></select>
<label>Things</label>
<select id="ddlThings" data-bind="options: ThingsList, value: ActiveChildRecord().Favorite, optionsCaption:'Select favorite thing...'" ></select>
</fieldset>
<button data-bind="click: AddChildRecord" >Add a child record</button>
<table id="tblChildRecords" border>
<thead>
<tr>
<th>Category</th>
<th>Favorite Thing</th>
</tr>
</thead>
<tbody data-bind="foreach: ChildRecords">
<tr data-bind="click: ChildRecordClicked, css: {activeRow: ActiveChildRecord() === $data}" >
<td data-bind="text: Category"></td>
<td data-bind="text: Favorite"></td>
</tr>
</tbody>
</table>
<p>Steps to reproduce problem:</p>
<ol>
<li>Click "Add child record"</li>
<li>Click on that row to make it the "active" record</li>
<li>Choose category "Pets" and thing "Dog"</li>
<li>Click "Add child record"</li>
<li>Click on the new row to make it the "active" record</li>
<li>Choose category "Colours" and thing "Blue"</li>
<li>Now click back on the first row... <strong>"Dog" disappears!</strong></li>
</ol>
Javascript:
var categories = ["Pets", "Colours", "Foods"];
var MyViewModel = function(){
var _this = this;
this.HeaderField = ko.observable("this value is unimportant");
this.ChildRecords = ko.observableArray([]);
this.ActiveChildRecord = ko.observable({ Category: ko.observable("-"), Favorite: ko.observable("-")});
this.ThingsList = ko.observableArray();
this.AddChildRecord = function(){
_this.ChildRecords.push({ Category: ko.observable("-"), Favorite: ko.observable("-")});
}
this.ChildRecordClicked = function(childRecord){
_this.ActiveChildRecord(childRecord);
}
this.RefreshThingsList = ko.computed(function(){
var strActiveCategory = _this.ActiveChildRecord().Category();
switch(strActiveCategory){
case "Pets": _this.ThingsList(["Dog", "Cat", "Fish"]); break;
case "Colours": _this.ThingsList(["Red", "Green", "Blue", "Orange"]); break;
case "Foods": _this.ThingsList(["Apple", "Orange", "Strawberry"]); break;
}
});
}
ko.applyBindings(MyViewModel);
Knockout 的值 AllowUnset 绑定可能是一种更干净的方法。
http://jsfiddle.net/5mpwx501/8/
<select id="ddlCategory" data-bind="options: categories, value: ActiveChildRecord().Category, valueAllowUnset: true, optionsCaption:'Select category...'" ></select>
<select id="ddlThings" data-bind="options: ThingsList, value: ActiveChildRecord().Favorite, valueAllowUnset: true, optionsCaption:'Select favorite thing...'" ></select>
@super酷是 100% 正确的,但它未定义的原因是当您单击宠物行时 ActiveChildRecord 会发生变化,但此计算函数尚未执行,因此您有一个很小的时间范围,其中 Dog 是收藏夹,但选项仍然是颜色。由于 Dog 不是一个选项,因此下拉列表会将 ActiveChildRecord 上的 Favorite 属性设置为 undefined。
我会使用 valueAllowUnset 绑定。基本上,它告诉下拉列表,如果没有匹配项,请不要将我的值设置为 undefined,而是等待,因为选项可能正在更新。
使用此绑定的一个很好的副作用是,当您添加新的子记录时,它不会复制前一行。它会自然为您重置选择。
我使用了一种完全不同的方法,使用订阅来更新列表和值,并使用特殊的可观察值来保存编辑的记录。
<fieldset>
<legend>Active Child Record</legend>
<label>Category</label>
<select id="ddlCategory"
data-bind="options: categories, value: category,
optionsCaption:'Select category...'" ></select>
<label>Things</label>
<select id="ddlThings"
data-bind="options: things, value: thing,
optionsCaption:'Select favorite thing...'" ></select>
</fieldset>
<button data-bind="click: AddChildRecord" >Add a child record</button>
<table id="tblChildRecords" border>
<thead>
<tr>
<th>Category</th>
<th>Favorite Thing</th>
</tr>
</thead>
<tbody data-bind="foreach: childRecords">
<tr data-bind="click: ChildRecordClicked,
css: {activeRow: editedRecord() === $data}" >
<td data-bind="text: category"></td>
<td data-bind="text: thing"></td>
</tr>
</tbody>
</table>
JavaScript:
var categories = ["Pets", "Colours", "Foods"];
var MyViewModel = function(){
var _this = this;
this.categories = ko.observableArray(["Pets","Colours","Foods"]);
this.category = ko.observable();
this.category.subscribe(function(newCategory){
_this.refreshThings(newCategory);
if(editedRecord()) {
editedRecord().category(newCategory);
}
});
this.things = ko.observableArray([]);
this.thing = ko.observable();
_this.refreshThings = function(newCategory){
switch(newCategory){
case "Pets": _this.things(["Dog", "Cat", "Fish"]); break;
case "Colours": _this.things(["Red", "Green", "Blue", "Orange"]); break;
case "Foods": _this.things(["Apple", "Orange", "Strawberry"]); break;
}
};
this.thing.subscribe(function(newThing){
if(editedRecord()) {
editedRecord().thing(newThing);
}
});
this.childRecords = ko.observableArray([]);
this.editedRecord = ko.observable();
this.AddChildRecord = function(){
var newRecord = {
category: ko.observable(),
thing: ko.observable()
};
_this.childRecords.push(newRecord);
_this.editedRecord(newRecord);
_this.category('');
_this.thing('');
}
this.ChildRecordClicked = function(childRecord){
_this.editedRecord(null);
_this.category(childRecord.category())
_this.thing(childRecord.thing())
_this.editedRecord(childRecord);
}
}
ko.applyBindings(MyViewModel);
几点说明:
- 使用一个名为"editedRecord"的新可观察量。这可以保存当前编辑的记录的值(新建,通过单击它选择(,或者如果不应编辑任何内容,则为 null 值(此值在
AddChildRecord
和ChildrecordClicked
中设置,以便在更新列表时进行无效更改( - 有一个数组
categories
、一个可观察category
和一个更新事物列表以及编辑记录的类别属性(如果存在(的订阅 - 有一个数组
things
、一个可观察thing
和一个更新已编辑记录的事物属性(如果存在(的订阅 addChildRecord
创建一个新的空记录,并将其设置为已编辑的记录。除了初始化类别和事物的列表childRecordClick
将单击的记录设置为已编辑的记录
如您所见,使用这种技术,绑定仍然非常简单,您可以完全控制每时每刻发生的事情。
您可以使用与此类似的技术来允许取消版本和类似的事情。事实上,我通常在其他位置编辑记录,并在用户接受更改后添加它或应用其更改,从而允许他取消。
这是你修改的小提琴。
最后,如果要保留未编辑记录上的斜杠,请进行以下更改:
this.AddChildRecord = function(){
_this.editedRecord(null);
var newRecord = {
category: ko.observable("-"),
thing: ko.observable("-")
};
_this.childRecords.push(newRecord);
_this.category('');
_this.thing('');
_this.editedRecord(newRecord);
}
包含在此版本的小提琴中,但最好应用样式以使表格单元格具有最小高度,并将其保留为空,就像在以前的版本中一样。
我在你的小提琴中做了一个小的修改,效果很好.
查看模型:
this.RefreshThingsList = ko.computed(function(){
var store= ActiveChildRecord().Favorite();
var strActiveCategory = _this.ActiveChildRecord().Category();
switch(strActiveCategory){
case "Pets": _this.ThingsList(["Dog", "Cat", "Fish"]); break;
case "Colours": _this.ThingsList(["Red", "Green", "Blue", "Orange"]); break;
case "Foods": _this.ThingsList(["Apple", "Orange", "Strawberry"]); break;
}
alert(ActiveChildRecord().Favorite()); // debug here you get undefined on your step 7 so store the value upfront and use it .
ActiveChildRecord().Favorite(store);
});
在这里工作小提琴
以防万一您正在寻找其他东西,请告诉我们。
- 在我的情况下,使用带有变量失败的 jquery 选择器
- 如何创建只允许用户在选中前一个复选框的情况下选择复选框的复选框验证
- 获取 HTML 5 音频控制文件位置并在不使用 id 选择器的情况下更新 src
- 如何在不重新加载页面的情况下选择文件提交表单
- Javascript:检索在没有 JQuery 的情况下选择了哪个下拉菜单
- Jquery 如何在不作为父级的情况下选择低于 id/class 的元素
- 如何更新p的标签:在primefaces中ajax调用后,在不关闭组件的情况下选择CheckboxMenu
- 是否有任何处理程序可以检测在没有jQuery的情况下选择字段是否有任何选项
- 如何使第一个单选按钮在默认情况下选择一个列表中建立的for in循环使用trick
- 如何在不使用id的情况下选择javascript中的第二个单选按钮
- JavasScript:如何防止用户在不使用禁用/指针事件的情况下选择选择选项列表中的任何内容
- 选项:第一个孩子不会在所有情况下选择相同的选项
- 在没有键盘的情况下选择和取消选择列表中的多个选项
- 在不知道索引的情况下选择jquery tools选项卡
- Angular.js在不使用ng-option指令的情况下选择选项
- 在不使用Select属性的情况下选择一个Select选项(做一个假的鼠标点击)
- 在安装了Opera Mini的功能手机上,可以在不使用javascript的情况下选择要上传的文件来触发UI更改
- 使用普通JavaScript,在不按索引或ID进行选择的情况下选择和删除html选择选项
- 如何知道是否在不知道名称或 ID 的情况下选择了第 7 个单选按钮
- 为什么在某些情况下选择铁页不起作用