C# 我做错了什么?具有数据绑定的自定义Usercontrol中的Dependency属性

C# 我做错了什么?具有数据绑定的自定义Usercontrol中的Dependency属性,c#,xaml,windows-runtime,windows-store-apps,winrt-xaml,C#,Xaml,Windows Runtime,Windows Store Apps,Winrt Xaml,(已编辑以修复DP的语法错误-但问题仍然存在) 我在一个简单的自定义用户控件中对依赖属性进行数据绑定时遇到了一些问题,不知道你们中的一些专家能否告诉我哪里出了问题 基本上,在我的usercontrol中,我有一个标签和两个按钮。标签绑定到“我的依赖项”属性,当您单击按钮时,计数根据按钮的单击而递增或递减,然后由标签显示 如果我在一个没有usercontrol的页面中执行所有这些操作,那么它就可以正常工作。然而,一旦我将其移动到usercontrol,它就会开始表现出奇怪的行为。初始计数0按预期在

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

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

基本上,在我的usercontrol中,我有一个标签和两个按钮。标签绑定到“我的依赖项”属性,当您单击按钮时,计数根据按钮的单击而递增或递减,然后由标签显示

如果我在一个没有usercontrol的页面中执行所有这些操作,那么它就可以正常工作。然而,一旦我将其移动到usercontrol,它就会开始表现出奇怪的行为。初始计数0按预期在开始时显示,当我单击按钮时,修订后的计数将在第一次单击时显示。但是,如果再次单击同一按钮,则会更新dependency属性,但不会显示该值(标签变为空白)。如果我点击另一个按钮,则会显示修改后的计数,但只是第一次点击-这一切都很奇怪,我被卡住了

如果我在'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>
我假设我的数据绑定语法有点不正确,但应该是什么呢

(编辑)-绘图变厚!我想补充一点,我刚刚在一个标准WPF应用程序中尝试了完全相同的语法,一切都按预期进行。因此,我想问题是,我是否发现WinRT/XAML存在缺陷,或者使用该平台时语法是否不同

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

  • 如果启动测试应用程序,请单击btnMore按钮,然后计数将增加,文本块将按预期更新。如果我再次单击同一按钮,计数将递增,但lblCount的文本值将变为空字符串
  • <> Li >如果我启动测试应用程序并按下BTNMULL,然后按下未使用的按钮,并不断重复这个值,那么这个值就会不断增加,TeXBube将继续更新——只要我不连续地按BTNFER两次,而是按下空白按钮(或者实际上只把光标聚焦到一个空白的文本框或任何其他UI的东西),然后文本块将继续显示递增的数字。第二次我多次单击BTN时,lblCount的值将变为空字符串

这非常奇怪……

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

在旋转Win8并为自己尝试代码后,我看到了与您相同的问题

由此我可以确定两件事:

1) 您的依赖项属性基本上是正确的。单击按钮时,我可以观察到属性值的增减。这向我表明问题是某种刷新问题

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


在我看来,beta版(err.'consumer preview')中存在某种刷新错误,您可以通过稍微改变应用程序的结构来加以解决。

我从RichTextBox继承了类似的东西

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

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

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

}

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

XAML:


@Rhyous-我想是的,但是我对一些代码有点了解,不想启动Win8。谢谢你的回答Ritch-你是对的,第三个参数应该是UserControl1,而不是UserControl-这是我发布时的一个错误。但问题是,即使在更正了源代码之后,问题仍然存在,我不知道为什么…谢谢Rhyous-我将看一看这篇文章。“TheCountProperty_Changed”仅用于验证DP值是否确实在更改。通过在那里放置一个断点,我可以计算e.NewValue,并看到该值每次都在变化——因此DP似乎在工作。这让我觉得我在XAML中的绑定语法有问题……感谢xyzzer,它工作得完美无缺!为了以防万一,我将在connect上调试原始问题。我可以看出微软希望我们如何异步地做这些事情,但是如果你同步地做的话,它不应该像它那样显示,对我来说,这似乎是一个bug。
private static void TheCountProperty_Changed(DependencyObject source, DependencyPropertyChangedEventArgs e)
{

}
<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();
        }
    }

}