如何设置依赖属性值?

时间:2014-06-23 20:27:23

标签: c# wpf xaml data-binding dependency-properties

请参考下面的代码了解上下文。

  • 启动时,2个文本框的文本将为"这是原始值"。
  • 点击TestBox按钮("测试按钮")时:
    • TestBox' TextBox的文字将更改为"按测试按钮设置"
    • 其他TextBox的值不会更改。
  • 点击Window按钮时,BOTH TextBoxes 的文本应更改为" Set By Window"。 但是,只有普通的TextBox会更新,TestBox不会更新。 < - 这就是BUG!

似乎我在Test内设置TestBox属性的方式会消除绑定。

在不破坏绑定的情况下,在用户控件本身内更改依赖属性的正确方法是什么?


示例代码:

我有一个UserControl,TestBox看起来像这样:

TestBox.xaml:

<UserControl x:Class="Company.UserControls.TestBox"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             x:Name="TextBoxControl">
    <StackPanel>
        <TextBox MinWidth="100" Name="TestTextBox" 
            Text="{Binding Path=Test, ElementName=TextBoxControl, Mode=TwoWay}" 
         />
        <Button MinWidth="100" Content="Test Button" 
            Click="ButtonBase_OnClick" />
    </StackPanel>
</UserControl>

TestBox.xaml.cs:

using System.Windows;

namespace Company.UserControls
{
    public partial class TestBox
    {
        public const string TestString = "Set By Test Button";

        public TestBox()
        {
            InitializeComponent();
        }
        public static readonly DependencyProperty TestProperty = 
                DependencyProperty.Register(
                    "Test", 
                    typeof(string), typeof(TestBox),
            new FrameworkPropertyMetadata(null, 
                        FrameworkPropertyMetadataOptions.AffectsRender));
        public string Test
        {
            get { return (string)GetValue(TestProperty); }
            set { SetValue(TestProperty, value); }
        }
        private void ButtonBase_OnClick(object sender, RoutedEventArgs e)
        {
            /****** THIS OBLITERATES THE BINDING ******/
        Test = TestString;
            /****** THIS OBLITERATES THE BINDING ******/
        }
    }
}

一个使用这个控件的窗口:

MainWindow.xaml:

<Window x:Class="Company.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:u="clr-namespace:Company.UserControls"
        Title="MainWindow">
    <StackPanel x:Name="MyStackPanel">
        <TextBox Text="{Binding Path=MyTestValue, Mode=OneWay}"/>
        <u:TestBox x:Name="MyTestBox" 
                   Test="{Binding Path=MyTestValue, Mode=OneWay}"/>
        <Button Content="Click" Click="ButtonBase_OnClick" />
    </StackPanel>
</Window>

MainWindow.xaml.cs:

using System.Windows;

namespace Company
{
    public partial class MainWindow
    {
        public MainWindow()
        {
            InitializeComponent();

            MyStackPanel.DataContext = new MyThing
            {
                MyTestValue = "This is the Original Value"
            };
        }
        private void ButtonBase_OnClick(object sender, RoutedEventArgs e)
        {
            MyStackPanel.DataContext = new MyThing
            {
                MyTestValue = "Set by Window"
            };
        }
    }

    public class MyThing
    {
        public string MyTestValue { get; set; }
    }
}

1 个答案:

答案 0 :(得分:0)

问题是您要求绑定系统不同步。整个系统旨在使所有绑定元素保持同步。只有在绑定模式设置为&#34; TwoWay&#34;或者&#34; OneWayToSource&#34;。在这些条件下,值将传输回源,因此系统保持同步。但是,在您的情况下,双向绑定将导致两个按钮都更改两个文本框。

您需要使用两个依赖项属性TestBox。第一个依赖项属性将绑定到内部文本框,第二个属性将绑定到父窗口。然后,您需要将属性更改处理程序添加到第二个依赖项属性(在FrameworkPropertyMetadata中完成)。在此处理程序中,只需在第一个依赖项属性上设置值。

由于你正在使用带有代码的UserControl,一个更简单的解决方案是只有上面提到的第二个依赖属性,并直接将值(从你的事件处理程序和属性更改处理程序)设置到文本框上X:名称

如果您需要进一步澄清,请与我们联系。