繁体   English   中英

我的 WPF 自定义控件的数据上下文正在取代父级的

[英]My WPF custom control's Data Context is superseding parent's

在我的主窗口中,我尝试绑定到bool ,但它正在查看我的自定义控件的DataContext 如果我不在用户控件中分配DataContext ,则主窗口的绑定有效,但是(显然)这会阻止用户控件中的绑定。

这是错误:

System.Windows.Data 错误:40:BindingExpression 路径错误:在“对象”“MyUserControlModel”(HashCode=1453241)上找不到“MyControlVisible”属性。 BindingExpression:Path=MyControlVisible; DataItem='MyUserControlModel' (HashCode=1453241); 目标元素是 'MyUserControl' (Name='_myUserControl'); 目标属性是“可见性”(类型“可见性”)

我需要绑定才能在两个控件上工作,但我不希望用户控件的DataContext取代窗口的。

这是代码:

<Window x:Class="Sandbox.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:Controls="clr-namespace:Sandbox.Controls" Title="Sandbox">
    <DockPanel LastChildFill="True">
        <DockPanel.Resources>
            <BooleanToVisibilityConverter x:Key="boolToVis" />
        </DockPanel.Resources>
        <Grid>
            <Controls:MyUserControl x:Name="_myUserControl" Visibility="{Binding MyControlVisible, Converter={StaticResource boolToVis}}"/>
        </Grid>
    </DockPanel>
</Window>

namespace Sandbox
{
    public partial class MainWindow
    {
        private MainWindowModel model;
        public MainWindow()
        {
            InitializeComponent();
            DataContext = model = new MainWindowModel();
            _myUserControl.Initialize(model.MyUControlModel);
        }
    }
}

using System.ComponentModel;
using Sandbox.Controls;

namespace Sandbox
{
    public class MainWindowModel : BaseModel
    {
        public MyUserControlModel MyUControlModel { get; set; }
        public bool MyControlVisible { get; set; }
        public MainWindowModel()
        {
            MyUControlModel = new MyUserControlModel();
            MyControlVisible = false;
            OnChange("");
        }
    }

    public class BaseModel : INotifyPropertyChanged
    {
        public event PropertyChangedEventHandler PropertyChanged;
        protected void OnChange(string s)
        {
            var handler = PropertyChanged;
            if (handler != null)
            {
                handler(this, new PropertyChangedEventArgs(s));
            }
        }
    }
}

<UserControl x:Class="Sandbox.Controls.MyUserControl"
             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">
    <Grid>
        <TextBlock Text="{Binding MyBoundText}"/>
    </Grid>
</UserControl>

namespace Sandbox.Controls
{
    public partial class MyUserControl
    {
        public MyUserControl()
        {
            InitializeComponent();
        }

        public void Initialize(MyUserControlModel context)
        {
            DataContext = context;
        }
    }

}

namespace Sandbox.Controls
{
    public class MyUserControlModel : BaseModel
    {
        public string MyBoundText { get; set; }
        public MyUserControlModel()
        {
            MyBoundText = "Hello World!";
            OnChange("");
        }
    }
}

这是您永远不应该直接从UserControl本身设置DataContext的众多原因之一。

当您这样做时,您不能再使用任何其他DataContext ,因为 UserControl 的DataContext是硬编码的。

对于您的绑定,通常DataContext将被继承,因此Visibility绑定可以在当前DataContext上找到属性MyControlVisible ,但是因为您在 UserControl 的构造函数中对DataContext进行了硬编码,所以找不到该属性。

您可以在绑定中指定不同的绑定源,例如

<Controls:MyUserControl Visibility="{Binding 
    RelativeSource={RelativeSource AncestorType={x:Type Window}}, 
    Path=DataContext.MyControlVisible, 
    Converter={StaticResource boolToVis}}" ... />

但是,这只是针对此特定情况的问题的解决方法,在我看来,这不是永久解决方案。 更好的解决方案是不要在UserControlDataContext进行硬编码

根据用户控件的用途和应用程序的设计方式,您可以采用几种不同的方法。

  • 您可以在 UserControl 上创建一个DependencyProperty以传入值,并绑定到该值。

     <Controls:MyUserControl UcModel="{Binding MyUControlModelProperty}" ... />

    <UserControl x:Class="Sandbox.Controls.MyUserControl" ElementName=MyUserControl...> <Grid DataContext="{Binding UCModel, ElementName=MyUserControl}"> <TextBlock Text="{Binding MyBoundText}"/> </Grid> </UserControl>
  • 或者您可以构建您的UserControl期望特定属性将在DataContext传递给它。 这通常是我做的,结合DataTemplates

     <Controls:MyUserControl DataContext="{Binding MyUControlModelProperty}" ... />

    <UserControl x:Class="Sandbox.Controls.MyUserControl"...> <Grid> <TextBlock Text="{Binding MyBoundText}"/> </Grid> </UserControl>
  • 正如我上面所说的,我喜欢使用DataTemplates来显示我的UserControls ,它们期望为它们的DataContext提供特定类型的Model ,所以通常我的主窗口的 XAML 看起来像这样:

     <DataTemplate DataType="{x:Type local:MyUControlModel}"> <Controls:MyUserControl /> </DataTemplate> <ContentPresenter Content="{Binding MyUControlModelProperty}" ... />

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM