c# - 我可以强制更新绑定(bind)到 ComboBox.SelectedItem 的 Textbox DataContext 吗?

标签 c# wpf data-binding undo

我正在为我的应用程序实现撤消/重做功能,但遇到了问题。基本上我有一个绑定(bind)到项目列表的组合框。然后,我有几个文本框,其中包含组合框中所选项目的 DataContext 以及绑定(bind)到该项目属性的文本。当用户发出撤消命令并且要撤消的项目是文本框的文本更改时,我希望首先选择组合框中的关联项目。然后,我想立即用原始文本更新文本框。理论上,这应该更新所选项目的数据绑定(bind)属性。然而,发生的情况是,选择发生了变化,但在我更改文本框的文本属性之前,数据绑定(bind)文本框并未更新,因此它更改了组合框中先前选定的项目的属性值。

这是我的组合框:

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

这是我的文本框(DataContext 在父网格上设置。如果在文本框本身上设置 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 上的“目的”属性。这确实有效,但如果我这样做的话,它不会集中和选择。另外,由于所有绑定(bind)都已经就位,并且其中一些绑定(bind)到整数,所以我将只更新文本并允许“魔法”来处理翻译。这样我就可以为任何文本框提供通用的解决方案。为了完整起见,这是我的 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 在需要时自行更新。

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

你应该做的是:

    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 绑定(bind)到该属性。您的文本框应将其文本属性绑定(bind)到网格的数据上下文 SelectedItem.SomeText 属性。

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

您只需更改对象 - UI 就会使用react。

这是更好的做法。这也是为什么我们有 MVVM 和 MVC 等模型来强制区分 UI 和逻辑。

关于c# - 我可以强制更新绑定(bind)到 ComboBox.SelectedItem 的 Textbox DataContext 吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36182039/

相关文章:

c# - 如何通过 HTTP 请求发送和发送电子邮件

c# - 取决于 UI 焦点状态的 WPF 命令路由行为不一致

c# - 有没有办法在 MVVM 中自动创建 DependencyProperty

c# - UIElement.TranslatePoint 时不时返回 (0,0)

java - 自动数据绑定(bind)

javascript - polymer 下拉数据与 AJAX 响应绑定(bind)

c# - 将 bool 类型传递给 C# 和 MS SQL Server 中的位参数类型

c# - 如何将信息传递到 Umbraco 中的所有页面

c# - observablecollection 到 dataView 或 dataset

WPF:如何触发 GUI 行为以响应 View 模型事件?