似乎有些人在Silverlight中更新文本AutoCompleteBox时遇到了问题,遗憾的是我加入了这个行列。
我有一个名为EditableCombo的派生类,就像这样;
public class EditableCombo : AutoCompleteBox
{
ToggleButton _toggle;
Path _btnPath;
TextBox _textBox;
...animation and toggle button stuff...
public override void OnApplyTemplate()
{
...animation and toggle button stuff...
//required to overcome issue in AutoCompleteBox that prevents the text being updated in some instances
//http://stackoverflow.com/questions/8488968/silverlight-5-autocompletebox-bug?rq=1
_textBox = GetTemplateChild("Text") as TextBox;
if (_textBox == null)
throw new NullReferenceException();
if (_textBox != null)
_textBox.TextChanged += TextBoxChanged;
base.OnApplyTemplate();
}
void TextBoxChanged(object sender, TextChangedEventArgs e)
{
Debug.WriteLine("text box changed fired new value: " + _textBox.Text);
Text = _textBox.Text;
OnTextChanged(new RoutedEventArgs());
}
...animation and toggle button stuff...
}
使用户能够单击一个togglebutton并从下拉列表中选择以选择一个选项或键入一个新值,如标准组合框控件。
我的视图有一个EditableCombo控件绑定到包含Gender属性的viewmodel;
public string Gender
{
get
{
Debug.WriteLine("Gender get executed - Model.Gender = " + Model.Gender);
return Model.Gender;
}
set
{
if (Model.Gender == value) return;
MonitoredNotificationObject.RaisePropertyChanged(() => Model.Gender, value, this, true);
}
}
我的viewmodel使用MonitoredNotificationObject来维护撤消/重做历史记录并通知任何属性更改;
public void RaisePropertyChanged(Expression<Func<string>> propertyExpression,
string newValue,
object sender,
bool canChain)
{
PropertyExpressionHelper propertyExpressionHelper = new PropertyExpressionHelper(propertyExpression)
{
NewValue = newValue
};
#if DEBUG
VerifyPropertyExists(propertyExpressionHelper.Name, sender);
#endif
var monitoredAction = new MonitoredProperty<TViewModel, TModel>(this)
{
ObjectPropertyName = propertyExpressionHelper.MakeObjectPropertyName(),
PropertyExpressionHelper = propertyExpressionHelper,
Sender = (TViewModel) sender,
CanChain = canChain
};
propertyExpressionHelper.SetToNewValue();
RaisePropertyChanged(propertyExpressionHelper.Name, sender);
MaintainMonitorState(monitoredAction);
}
撤消和重做的实现如下(撤消显示);
public override bool UndoExecute(MonitoredObject<TViewModel, TModel> undoAction,
Stack<MonitoredObject<TViewModel, TModel>> redoActions,
Stack<MonitoredObject<TViewModel, TModel>> undoActions)
{
PropertyExpressionHelper.SetToNewValue();
redoActions.Push(undoAction);
var action = (MonitoredProperty<TViewModel, TModel>) undoAction;
HandleAutoInvokedProperties(action);
if (action.CanChain)
{
if (undoActions.Any())
{
if (CanDoChain(undoActions, action))
return true;
}
}
action.RaiseChange();
Sender.RaiseCanExecuteChanges();
return false;
}
属性更改通知会像这样提出;
protected virtual void RaiseChange()
{
MonitoredNotificationObject.RaisePropertyChanged(PropertyExpressionHelper.Name, Sender);
if (RaiseChangeAction != null)
RaiseChangeAction.Invoke();
}
使用上述工作可以正常用于普通文本框,并且成功地允许用户根据需要撤消和重做他们的更改。当用户在字段中键入条目时,这也适用于EditableCombo - 同样,撤消和重做按预期执行。
问题是当用户从下拉列表中选择EditableCombo中的新值时。字段更新,性别设置,一切都很好。单击撤消成功将字段更改回其原始值 - 一切看起来都很花哨。
但是,当用户尝试重做更改时,屏幕上的值不会更新。更改基础值,调用Gender属性的get并正确设置Model.Gender值。但没什么。屏幕不会更新。 editablecombo控件TextBoxChangedEvent不会触发,因此屏幕上的值不正确。
基本上没有通知控制变更。
有什么想法吗?
更新
包含EditableCombo的视图有一个包含Gender属性的viewmodel。该属性是这样绑定的;
<EditCombo:EditableCombo ItemsSource="{Binding Genders}"
ItemTemplate="{StaticResource editableComboDataTemplate}"
Style="{StaticResource EditableComboStyle}"
Text="{Binding Path=Gender,
UpdateSourceTrigger=PropertyChanged,
Mode=TwoWay,
ValidatesOnDataErrors=True}"
TextBoxStyle="{StaticResource editableComboDataEntryField}"
ValueMemberPath="Value" />
我的undo / redo实现适用于非editablecombo控件和editablecombo,当通过键盘输入新值时。重做问题仅在通过下拉切换按钮更改属性时才会显现。我知道基础值已正确更新,如前所述(例如,当ValidatesOnDataErrors打开时,当我重做并将Gender属性设置回有效值时,表示错误的红色边框消失 - 但是,文本保持不变)。
无论出于何种原因,TextBoxChanged事件永远不会在上述场景中触发。可能是事件正在其他地方处理吗?
答案 0 :(得分:0)
如果添加此行,是否有效:
Model.Gender = value;
到物业设定者?