我做错了什么?具有DataBinding的自定义用户控件内的依赖项属性

本文关键字:控件 用户 依赖 属性 自定义 什么 错了 具有 DataBinding | 更新日期: 2023-09-27 18:28:22

(经过编辑以修复DP的语法错误,但问题仍然存在)

我在一个简单的自定义用户控件内的依赖属性的数据绑定方面遇到了一些问题,不知道你们中的一些专家是否能告诉我哪里出了问题。

基本上,在我的用户控件中,我有一个标签和两个按钮。标签绑定到我的依赖属性,当您单击按钮时,计数会根据按钮的单击而增加或减少,然后由标签显示。

如果我在一个没有用户控件的页面内完成所有这些操作,那么一切都正常工作。然而,一旦我把它移到用户控件上,它就开始表现得很奇怪。初始计数0在开始时按预期显示,当我单击按钮时,修改后的计数在第一次单击时显示。但是,如果我再次单击相同的按钮,依赖项属性将更新,但不会显示值(标签为空)。如果我点击了另一个按钮,那么修改后的计数就会显示出来,但只是第一次点击——这一切都很奇怪,我被卡住了!

如果我在"TheCountProperty_Changed"方法上设置断点,我可以验证属性值是否正确更新。。

这是我的自定义控件的XAML:

<UserControl
x:Class="DPTest.UserControl1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:DPTest"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"
d:DesignHeight="300"
d:DesignWidth="400"
x:Name="TestControl"
x:Uid="TestControl">
<Grid>
    <TextBlock x:Name="lblCount" HorizontalAlignment="Left" Margin="0,0,0,0" TextWrapping="Wrap" Text="{Binding Path=TheCount, ElementName=TestControl, Mode=OneWay}" VerticalAlignment="Top" FontSize="36"/>
    <Button x:Name="btnLess" Content="&lt;" HorizontalAlignment="Left" Margin="0,71,0,0" VerticalAlignment="Top" Width="75" Click="btnLess_Click" ClickMode="Press"/>
    <Button x:Name="btnMore" Content="&gt;" HorizontalAlignment="Left" Margin="91,71,0,0" VerticalAlignment="Top" Width="75" Click="btnMore_Click" ClickMode="Press"/>
</Grid>

这是它的CS:

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Windows.Foundation;
using Windows.Foundation.Collections;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Controls.Primitives;
using Windows.UI.Xaml.Data;
using Windows.UI.Xaml.Input;
using Windows.UI.Xaml.Media;
using Windows.UI.Xaml.Navigation;
namespace DPTest
{
public sealed partial class UserControl1 : UserControl
{
    public static readonly DependencyProperty TheCountProperty = DependencyProperty.Register("TheCount", typeof(int), typeof(UserControl1), new PropertyMetadata(0, new PropertyChangedCallback(TheCountProperty_Changed)));
    public int TheCount
    {
        get { return (int)GetValue(TheCountProperty); }
        set { SetValue(TheCountProperty, value); }
    }
    private static void TheCountProperty_Changed(DependencyObject source, DependencyPropertyChangedEventArgs e)
    {
    }
    public UserControl1()
    {
        this.InitializeComponent();
    }
    private void btnLess_Click(object sender, RoutedEventArgs e)
    {
        //lblCount.Text = (Convert.ToInt32(lblCount.Text) - 1).ToString();
        this.TheCount -= 1;
    }
    private void btnMore_Click(object sender, RoutedEventArgs e)
    {
        //lblCount.Text = (Convert.ToInt32(lblCount.Text) + 1).ToString();
        this.TheCount += 1;
    }
}
}

我假设我的Databinding语法有点不正确,但它应该是什么?

(编辑)-情节越来越浓!我想补充一点,我刚刚在标准的WPF应用程序中尝试了完全相同的语法,一切都如预期的那样工作。所以我想问题是,我是否发现了WinRT/XAML的错误,或者使用该平台的语法不同?

(编辑#2对所有的编辑感到抱歉,但我仍在努力解决这个问题!)我开始相信这是显示问题或按钮控件的错误。我刚刚添加了一个额外的按钮,但没有为它或任何东西连接任何点击事件代码——它基本上只是一个什么都不做的按钮。有趣的是:

  • 如果我启动测试应用程序,请单击btnMore按钮,然后TheCount将递增,Textblock将按预期更新。如果我再次单击相同的按钮,TheCount会增加,但lblCount的Text值会变为空字符串
  • 然而,如果我启动测试应用程序并按下btnMore,然后按下未使用的按钮,并不断重复,那么值就会不断增加,TextBlock就会不断更新——只要我没有连续按下btnMore两次,而是按下空白按钮(或者实际上只是将光标聚焦到空白文本框或任何其他UI中),则文本块将继续显示递增的数字。第二次我多次单击btnMore,那么lblCount的值就会变成一个空字符串

这太可怕了。。。。

我做错了什么?具有DataBinding的自定义用户控件内的依赖项属性

DependencyProperty的第三个参数看起来应该是"UserControl1",而不是"UserControl"。

在旋转Win 8并亲自尝试代码后,我看到了与您相同的问题。

从中我可以确定两件事:

1) 您的依赖项属性基本正确。我可以观察到属性的值随着按钮的点击而增加和减少。这向我表明,这个问题是某种刷新问题。

2) 将代码从UserControl移动到Main xaml页面(在我的测试应用程序中,它是默认的"Blank.xaml"页面)允许数字正确刷新。

在我看来,测试版中存在某种刷新错误(错误…"消费者预览"),您可以通过稍微不同地构建应用程序来解决这个问题。

这看起来像是WinRT中的一些同步问题-我使用了一个转换器来检查我得到了什么,看起来int值被设置为null。一旦我修改了代码以异步调用对属性的更新,它就开始正常工作了。

XAML:

<UserControl
    x:Class="DPTest.UserControl1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:local="using:DPTest"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    mc:Ignorable="d"
    d:DesignHeight="300"
    d:DesignWidth="400"
    x:Name="TestControl"
    x:Uid="TestControl">
    <Grid>
        <Grid.Resources>
            <local:IntToStringConverter
                x:Key="IntToStringConverter" />
        </Grid.Resources>
        <TextBlock
            x:Name="lblCount"
            HorizontalAlignment="Left"
            Margin="0,0,0,0"
            TextWrapping="Wrap"
            Text="{Binding Path=TheCount, Converter={StaticResource IntToStringConverter}, ElementName=TestControl, Mode=OneWay}"
            VerticalAlignment="Top"
            FontSize="36" />
        <Button
            x:Name="btnLess"
            Content="&lt;"
            HorizontalAlignment="Left"
            Margin="0,71,0,0"
            VerticalAlignment="Top"
            Width="75"
            Click="btnLess_Click"
            ClickMode="Release" />
        <Button
            x:Name="btnMore"
            Content="&gt;"
            HorizontalAlignment="Left"
            Margin="91,71,0,0"
            VerticalAlignment="Top"
            Width="75"
            Click="btnMore_Click"
            ClickMode="Release" />
    </Grid>
</UserControl>

代码背后:

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Windows.Foundation;
using Windows.Foundation.Collections;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using Windows.UI.Xaml.Controls.Primitives;
using Windows.UI.Xaml.Data;
using Windows.UI.Xaml.Input;
using Windows.UI.Xaml.Media;
using Windows.UI.Xaml.Navigation;
namespace DPTest
{
    public sealed partial class UserControl1 : UserControl
    {
        public static readonly DependencyProperty TheCountProperty =
            DependencyProperty.Register(
                "TheCount",
                typeof(int),
                typeof(UserControl1),
                new PropertyMetadata(
                    0,
                    new PropertyChangedCallback(TheCountProperty_Changed)));
        public int TheCount
        {
            get { return (int)GetValue(TheCountProperty); }
            set { SetValue(TheCountProperty, value); }
        }
        private static void TheCountProperty_Changed(DependencyObject source, DependencyPropertyChangedEventArgs e)
        {
        }
        public UserControl1()
        {
            this.InitializeComponent();
        }
        private void btnLess_Click(object sender, RoutedEventArgs e)
        {
            //lblCount.Text = (Convert.ToInt32(lblCount.Text) - 1).ToString();
            this.Dispatcher.InvokeAsync(
                Windows.UI.Core.CoreDispatcherPriority.Low,
                (s, a) => this.TheCount -= 1,
                this,
                null);
        }
        private void btnMore_Click(object sender, RoutedEventArgs e)
        {
            //lblCount.Text = (Convert.ToInt32(lblCount.Text) + 1).ToString();
            this.Dispatcher.InvokeAsync(
                Windows.UI.Core.CoreDispatcherPriority.Low,
                (s, a) => this.TheCount += 1,
                this,
                null);
        }
    }
    public class IntToStringConverter : IValueConverter
    {
        public object Convert(object value, Type targetType, object parameter, string language)
        {
            return value.ToString();
        }
        public object ConvertBack(object value, Type targetType, object parameter, string language)
        {
            throw new NotImplementedException();
        }
    }
}

我从RichTextBox继承了一些非常相似的东西。

http://www.wpfsharp.com/2011/08/01/loading-a-richtextbox-from-an-rtf-file-using-binding-or-a-richtextfile-control/

我发现我没有将依赖属性设为只读(不确定这是否重要)

此外,我假设您打算填充以下内容:

private static void TheCountProperty_Changed(DependencyObject source, DependencyPropertyChangedEventArgs e)
{
}