8

如果我通过设置 DataSourceUpdateMode = Never 关闭绑定数据源的自动更新,然后使用按钮更新整个批次(使用 binding.WriteValue),则会出现问题 - 即,仅更新第一个绑定控件的数据源。所有其他控件都重置为原始值。

这是因为当当前对象发生变化时(如上述WriteValue之后发生的情况),如果ControlUpdateMode = OnPropertyChange,那么所有其他控件都会重新从数据源中读取该值。

避免这个问题的标准方法是什么?

一种方法是从 BindingSource 派生一个类并添加一个 WriteAllValues 方法。此方法执行以下操作:

(1)对于每个Binding,保存ControlUpdateMode

(2) 对于每个Binding,设置ControlUpdateMode = Never

(3) 对于每个Binding,调用WriteValue方法

(4) 对于每个 Binding,将 ControlUpdateMode 重置为保存的值

(5) 对于每个Binding,如果ControlUpdateMode = OnPropertyChange,调用ReadValue方法。

你能看出这样做有什么问题吗?

如果使用您自己的类,实现 IEditableObject 会解决问题吗?

在我正在处理的另一个控件中,我实现了自己的绑定。我解决这个问题的方法是使用以下代码。(我已经做到了最低限度,希望你能遵守!):

Private Shared ControlDoingExplicitUpdate As MyCustomControl = Nothing

Private Sub UpdateDataSourceFromControl(ByVal item As Object, ByVal propertyName As String, ByVal value As Object)
  Dim p As PropertyDescriptor = Me.props(propertyName)
  Try
    ControlDoingExplicitUpdate = Me
    p.SetValue(item, value)
  Catch ex As Exception
    Throw
  Finally
    ControlDoingExplicitUpdate = Nothing
  End Try
End Sub

Private Sub DataBindingSource_CurrentItemChanged(ByVal sender As Object, ByVal e As System.EventArgs)
  If (ControlDoingExplicitUpdate IsNot Nothing) AndAlso (ControlDoingExplicitUpdate IsNot Me) Then Exit Sub
  Me.UpdateControlFromDataSource() 'Uses ReadValue
End Sub

因此,当调用 UpdateDataSourceFromControl 时,将为同一 BindingSource 中的所有其他控件调用所有 CurrentItemChanged 事件。但是,由于设置了 ControlDoingExplicitUpdate,它们不会从数据源重新读取值,除非它们恰好是进行更新的控件。在所有这些事件完成后,ControlDoingExplicitUpdate 设置为 Nothing,以便恢复正常服务。

我希望你能关注这个,并且 - 再次 - 我问,你能看到这有什么问题吗?

4

3 回答 3

5

我对表格有类似的要求。在我的情况下,我只希望在单击表单的“保存”按钮时发生所有表单控件的数据绑定。

我找到的最佳解决方案是将每个绑定的 DataSourceUpdateMode 设置为 OnValidation,然后将包含表单的 AutoValidate 属性设置为 Disable。当您在窗体上的控件之间更改焦点时,这可以防止绑定。然后在我的 Save 按钮的 Click 事件中,我手动验证表单的输入,如果可以,则调用表单的 ValidateChildren 方法来触发绑定。

此方法还具有让您完全控制如何验证输入的优势。默认情况下,WinForms 不包含执行此操作的好方法。

于 2011-08-26T15:58:19.040 回答
2

我相信我最近在 stackoverflow 上读到了这个答案: 禁用双向数据绑定

public static class DataBindingUtils
{
    public static void SuspendTwoWayBinding( BindingManagerBase bindingManager )
    {
        if( bindingManager == null )
        {
           throw new ArgumentNullException ("bindingManager");
        }

        foreach( Binding b in bindingManager.Bindings )
        {
            b.DataSourceUpdateMode = DataSourceUpdateMode.Never;
        }
    }

    public static void UpdateDataBoundObject( BindingManagerBase bindingManager )
    {
        if( bindingManager == null )
        {
           throw new ArgumentNullException ("bindingManager");
        }

        foreach( Binding b in bindingManager.Bindings )
        {
            b.WriteValue ();
        }
    }
}
于 2015-03-03T16:05:13.757 回答
0

我建议不要与数据绑定作斗争。将 DataSourceUpdateMode 保持为 OnValidation。然后选择以下 2 个选项之一。

1、在控件中更改值之前,请备份当前值。如果用户单击取消按钮,则将备份的值复制回数据源。并运行 userBindingSource.ResetCurrentItem() 将控件中的值刷新为原始值。

2,如果用户单击取消按钮,从数据库中按 ID 拉出一个新对象。并使用新对象替换绑定源中的当前对象。代码可能是这样的:

MyObject myObject = myObjectBindingSource.Current as MyObject;
MyObject originalObject = myObjectRepository.GetOneObjectFromDatabase(myObject.ID);
int currentIndex = myObjectBindingSource.Position;
myObjectBindingSource.Insert(currentIndex, originalObject);
myObjectBindingSource.Position = currentIndex;
myObjectBindingSource.RemoveAt(currentIndex+1);

我个人喜欢第二个选项,因为它避免了复制对象的工作,它可能需要复制每个属性,并且如果将来属性更改,维护会更加困难。它还提取了最新的数据,使记录非常新鲜。此外,第一个选项仅保存 1 个记录副本。如果您导航到其他记录并进行其他更改,则副本将丢失。但是,第二个选项从数据库中提取数据,因此您始终可以导航回任何以前修改但尚未保存的记录,然后单击取消按钮以恢复原始值。

于 2019-10-28T20:05:14.367 回答