DateTimePicker OnValueChanged事件被引发两次
本文关键字:两次 OnValueChanged 事件 DateTimePicker | 更新日期: 2023-09-27 18:13:33
我正在使用扩展WPF工具包社区版库版本2.5中的DateTimePicker WPF控件。
我的问题是,当我选择一个日期,OnValueChanged事件被引发两次,而不是仅仅一次。
下面是我使用的代码: XAML:<StackPanel>
<xctk:DateTimePicker AutoCloseCalendar="True" Name="picker" Width="400" Height="40" ValueChanged="UpDownBase_OnValueChanged"/>
<ListBox Height="300" Name="listbox"></ListBox>
</StackPanel>
c#代码:
private void UpDownBase_OnValueChanged(object sender, RoutedPropertyChangedEventArgs<object> e)
{
var value = picker.Value;
if (value == null)
listbox.Items.Add("[NULL]");
else
listbox.Items.Add(value.Value.ToString(CultureInfo.InvariantCulture));
}
现在,每当我选择一个新的日期时,列表框将填充两个新项目。我还调试了程序,确认事件处理程序实际上被调用了两次。
如何解决这个问题?
更新:
我尝试了2.4版本,似乎这个问题已经消失了。现在看来,这可能是2.5版的一个bug。
这似乎是因为2.5中的事件是从:
触发的at Xceed.Wpf.Toolkit.DateTimePicker.OnValueChanged(Nullable`1 oldValue, Nullable`1 newValue) in C:'Users'Mark Vinten'Downloads'wpftoolkit-114314'Main'Source'ExtendedWPFToolkitSolution'Src'Xceed.Wpf.Toolkit'DateTimePicker'Implementation'DateTimePicker.cs:line 264
然后从基类中:
at Xceed.Wpf.Toolkit.TimePicker.OnValueChanged(Nullable`1 oldValue, Nullable`1 newValue) in C:'Users'Mark Vinten'Downloads'wpftoolkit-114314'Main'Source'ExtendedWPFToolkitSolution'Src'Xceed.Wpf.Toolkit'TimePicker'Implementation'TimePicker.cs:line 264
现在基类似乎也经历了CLR绑定过程,这表明这是绑定值。我还在研究为什么会这样,但一个解决方法是使用Binding:
MainWindow.cs
public DateTime? DateTimeValue
{
get { return (DateTime?)GetValue(DateTimeValueProperty); }
set { SetValue(DateTimeValueProperty, value); }
}
// Using a DependencyProperty as the backing store for DateTimeValue. This enables animation, styling, binding, etc...
public static readonly DependencyProperty DateTimeValueProperty =
DependencyProperty.Register("DateTimeValue", typeof(DateTime?), typeof(MainWindow), new PropertyMetadata(null, new PropertyChangedCallback(DateTimeValueProperty_Changed)));
private static void DateTimeValueProperty_Changed(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
MainWindow mw = d as MainWindow;
System.Diagnostics.Debug.WriteLine("d is " + d == null ? "null" : d.GetType().FullName);
if (mw != null && e.Property == DateTimeValueProperty)
{
var value = e.NewValue as DateTime?;
var listbox = FindChild<ListBox>(mw, "listbox");
if (value == null)
listbox.Items.Add("[NULL]");
else
listbox.Items.Add(value.Value.ToString(System.Globalization.CultureInfo.InvariantCulture));
}
}
/// <summary>
/// Finds a Child of a given item in the visual tree.
/// </summary>
/// <param name="parent">A direct parent of the queried item.</param>
/// <typeparam name="T">The type of the queried item.</typeparam>
/// <param name="childName">x:Name or Name of child. </param>
/// <returns>The first parent item that matches the submitted type parameter.
/// If not matching item can be found,
/// a null parent is being returned.</returns>
public static T FindChild<T>(DependencyObject parent, string childName)
where T : DependencyObject
{
// Confirm parent and childName are valid.
if (parent == null) return null;
T foundChild = null;
int childrenCount = VisualTreeHelper.GetChildrenCount(parent);
for (int i = 0; i < childrenCount; i++)
{
var child = VisualTreeHelper.GetChild(parent, i);
// If the child is not of the request child type child
T childType = child as T;
if (childType == null)
{
// recursively drill down the tree
foundChild = FindChild<T>(child, childName);
// If the child is found, break so we do not overwrite the found child.
if (foundChild != null) break;
}
else if (!string.IsNullOrEmpty(childName))
{
var frameworkElement = child as FrameworkElement;
// If the child's name is set for search
if (frameworkElement != null && frameworkElement.Name == childName)
{
// if the child's name is of the request name
foundChild = (T)child;
break;
}
}
else
{
// child element found.
foundChild = (T)child;
break;
}
}
return foundChild;
}
MainWindow.xaml
<StackPanel DataContext="{Binding RelativeSource={RelativeSource AncestorType={x:Type local:MainWindow}}}">
<xctk:DateTimePicker AutoCloseCalendar="True" Name="picker" Width="400" Height="40" Value="{Binding DateTimeValue}" />
<ListBox Height="300" Name="listbox"></ListBox>
</StackPanel>
使用绑定系统,该系统会自动检查值是否已更改,如果已更改则只引发事件。
注意:FindChild<>是我在此发现的一个函数,我如何通过名称或类型找到WPF控件?邮报》
使用最终摘要更新
这样做的原因似乎是因为在DateTimePicker中嵌入了一个TimePicker来提供该功能。不幸的是,DateTimePicker和TimePicker都来自相同的基,因此在UpDownBase中引发相同的路由事件,其中T是DateTime?
如果你检查事件参数,e.RoutedEVent总是UpDownBase。OnValueChanged,因为这是引发事件的类。e.Source或e.OriginalSource总是DateTimePicker本身,这意味着你没有有效的方法来过滤掉一个或另一个事件。
有代码在DateTimeUpDown.RaiseValueChangedEvent()检查TemplatedParent是否是一个时间选择器,以防止重新引发,但事件是否从DateTimePicker或TimePicker中引发TemplatedParent似乎总是DateTimePicker,因此失败,因此您获得两次事件。
我在WPFToolkit项目网站上提出了一个错误:https://wpftoolkit.codeplex.com/workitem/22014
我通过检查事件的原始来源解决了这个问题。
if(e.OriginalSource is Xceed.Wpf.Toolkit.DateTimePicker)
{
if(((Xceed.Wpf.Toolkit.DateTimePicker)e.OriginalSource).IsFocused == true)
{
ResetDataTable();
}
}
因为我没有在这个控件中显示时间选择器,我还确保它是datetimepicker有焦点,而不是时间选择器。可能是多余的
我通过比较原始源代码和源代码来解决这个问题。
e.OriginalSource == e.Source