我可以强制更新绑定到 ComboBox.SelectedItem 的文本框 DataContext 吗?

Can I force an update to a Textbox DataContext bound to a ComboBox.SelectedItem?

我正在为我的应用程序实现 undo/redo 功能,但遇到了 运行 问题。基本上我有一个绑定到项目列表的组合框。然后我有几个文本框,它们具有组合框中 selected 项目的 DataContext 和绑定到该项目属性的文本。当用户发出撤消命令并且要撤消的项目是文本框的文本更改时,我希望组合框中的关联项目首先被 selected。然后,我想立即用原始文本更新文本框。从理论上讲,这应该会更新 selected 项目的数据绑定 属性。然而,发生的事情是 selection 发生了变化,但是在我更改文本框的文本 属性 之前数据绑定文本框没有更新,因此它正在更改先前 属性 的值 select组合框中的项目。

这是我的组合框:

<ComboBox x:Name="myComboBox" 
          ItemsSource="{Binding MyItems, UpdateSourceTrigger=PropertyChanged}"
          DisplayMemberPath="Name"
          SelectionChanged="myComboBox_SelectionChanged" />

这是我的文本框(DataContext 是在父 Grid 上设置的。如果 DataContext 是在文本框本身上设置的,也会发生相同的行为)

<TextBox x:Name="purposeTxtBox" 
         Text="{Binding Purpose, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />

撤消启动后我使用的代码(片段)是:

case UndoAction.PURPOSE_CHANGE:
    SelectComboBoxItem(myComboBox, itemToSelect);
    purposeTxtBox.Text = itemValue;
    FocusAndSelect(purposeTxtBox);

以及 SelectComboBoxItem 方法:

private void SelectComboBoxItem(ComboBox box, object item)
{
    Dispatcher.BeginInvoke(DispatcherPriority.Input,
                           new Action(delegate()
                           {
                               box.SelectedItem = item;
                           }));
}

我知道我可以更新我的 DataContext 上的 'Purpose' 属性。这确实有效,但它不会集中注意力,如果我这样做的话 select。另外,由于所有绑定都已经到位并且其中一些绑定到整数,我打算只更新文本并允许 'magic' 处理 t运行 slations。这样我就可以为任何文本框提供通用解决方案。为了完整起见,这是我的 FocusAndSelect 函数:

private void FocusAndSelect(TextBox box)
{
    box.SelectAll();
    Dispatcher.BeginInvoke(DispatcherPriority.Input,
                           new Action(delegate()
                           {
                               box.Focus();         // Set Logical Focus
                               Keyboard.Focus(box); // Set Keyboard Focus
                           }));
}

我认为问题在于此:

Dispatcher.BeginInvoke(DispatcherPriority.Input,
                       new Action(delegate()
                       {
                           box.SelectedItem = item;
                       }));

您告诉系统更新组合框值,但您使用的是异步方法(我认为)。这意味着它会在 UI 线程上执行它(虽然我认为你是从 UI 线程调用它)但无论如何 - 它不会发生 现在.

然后在下一行中,您直接设置 purposeTxtBox.Text = itemValue,但复选框尚未更新其选中的值,因此尚未执行值更改。

最终,问题是您将逻辑与 UI 混为一谈。您不应该通过进入 UI 并更改某些内容来更改所选值,以便您的数据得到更新。您应该直接更改数据,并让 UI 在需要时自行更新。

举个简单的例子,一个显示布尔值的复选框,你不想在UI中创建一个复选框并在你的业务逻辑或数据逻辑中选中和取消选中它,复选框应该' 持有该值 - 你的程序应该,并且复选框应该绑定到它。您需要将该值更改为 false,将布尔值设置为 false,而不是转到复选框,将其设置为未选中并让绑定更新布尔值。这有点倒退!

你应该做的是:

    private Item selectedItem;
    public Item SelectedItem
    {
        get { return selectedItem; }
        set
        {
            if (selectedItem != value)
            {
                selectedItem = value;
                //perform your ItemChange events here, if you have any
                OnPropertyChanged("SelectedItem");
            }
        }
    }

您的组合框应该将它的 SelectedItem 绑定到 属性。您的文本框应将其文本属性绑定到网格的数据上下文 SelectedItem.SomeText 属性。

当您更改选择时,您不会转到 UI,更改组件,等待 UI 线程处理它,返回,触发事件等等等等

您只需更改对象 - UI 会做出反应。

这是更好的做法。这也是为什么我们有像 MVVM 和 MVC 这样的模型,来强制 UI 和逻辑之间的这种划分。

与你的问题没有直接关系(我希望它会有用),但你可以用委托来组织撤消:

readonly Stack<Action> _undo = new Stack<Action>();

string _someProperty;
// property to bind
public string SomeProperty
{
    get { return _someProperty; }
    set
    {
        var old = _someProperty; // capture old value
        _undo.Push(() =>
        {
            _someProperty = old;
            OnPropertyChanged();
        });
        _someProperty = value;
        OnPropertyChanged();
    }
}

void Undo()
{
    if (_undo.Count > 0)
        _undo.Pop()();
}

更改 属性 值(通过绑定)将 记录 委托设置先前的值。调用 Undo() 将简单地 play 从最后到第一个委托(后进先出)。

您也可以将选择和焦点操作添加到委托中(不确定这是否是个好主意)。