4

我有一个问题,当父级已将该对象设置为数据绑定时,我无法创建使用自定义对象属性的用户控件。

尝试解释我的意思是代码。自定义对象:

public class MyObj
{
    public string Text { get; set; }

    public MyObj(string text)
    {
        Text = text;
    }
}

背后的用户控制代码:

/// <summary>
/// Interaction logic for MyControl.xaml
/// </summary>
public partial class MyControl : UserControl
{
    public static readonly DependencyProperty ObjectProperty =
        DependencyProperty.Register("Object", typeof (MyObj), typeof (MyControl), new PropertyMetadata(default(MyObj)));

    public MyObj Object
    {
        get { return (MyObj) GetValue(ObjectProperty); }
        set { SetValue(ObjectProperty, value); }
    }

    public MyControl()
    {
        InitializeComponent();
    }
}

用户控制 XAML:

<UserControl x:Class="Test.MyControl"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
         xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
         mc:Ignorable="d" 
         d:DesignHeight="300" d:DesignWidth="300" DataContext="{Binding RelativeSource={RelativeSource Self}}">
<TextBlock Text="{Binding Object.Text}"/>

所以我所期望的只是让 MyControl 显示一个 TextBlock,其中的文本显示 MyObj.Text 中的任何字符串;

如果我在代码中添加控件,没有任何绑定,那么这可以正常工作,例如

MyControl myControl = new MyControl(){ Object = new MyObj("Hello World!") };
grid.Children.Add(myControl);

但是,如果我尝试使用数据绑定,这不会显示任何内容,这是 MainWindow 的代码。

代码隐藏:

public partial class MainWindow : Window, INotifyPropertyChanged
{
    private MyObj _Object;
    public MyObj Object
    {
        get { return _Object; }
        set
        {
            _Object = value;
            OnPropertyChanged("Object");
        }
    }

    public MainWindow()
    {
        InitializeComponent();

        Object = new MyObj("HELLO");
    }

    public event PropertyChangedEventHandler PropertyChanged;

    [NotifyPropertyChangedInvocator]
    protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
    {
        PropertyChangedEventHandler handler = PropertyChanged;
        if (handler != null) handler(this, new PropertyChangedEventArgs(propertyName));
    }
}

XAML:

谁能指出我正确的方向,我想这与在 UserControl 上使用相对源绑定有关,但我不确定。

谢谢

4

2 回答 2

16

我个人从未在 UserControl 上使用过相对自绑定,所以我不确定它是否有效。您可以尝试设置x:Name您的UserControl,并在绑定中使用它。

<UserControl x:Class="Test.MyControl"
             ...
             x:Name="window">
    <TextBlock Text="{Binding ElementName=window, Path=Object.Text}"/>
</UserControl>

请注意,如果数据绑定在运行时绑定失败,您还应该在“输出”窗口中看到相关的错误消息。

于 2013-07-30T19:53:33.283 回答
-1

已经很久了..但是由于有一种新技术,我想在这里发布。

Compiled Time Binding :这是 Windows 10 引入的一种新型绑定。这种绑定具有很多性能优势经典绑定。

您无需为页面或控件本身设置任何DataContext的额外好处是您可以绑定到页面或控件中的任何内容的DataContext

<UserControl x:Class="Test.MyControl"
             ...
             x:Name="window">
    <TextBlock Text="{x:Bind Object.Text}"/>
</UserControl>

但这是否像您想象的那样完美地工作..不!不像你猜的那样。并且有一个答案。

编译时间绑定默认设置为OneTime,而不是经典绑定设置为OneWay

因此您需要将模式显式设置为OneWay以确保值始终更新。

<UserControl x:Class="Test.MyControl"
             ...
             x:Name="window">
    <TextBlock Text="{x:Bind Object.Text,Mode=OneWay}"/>
</UserControl>
于 2015-12-24T09:40:36.530 回答