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

这是错误:



我需要绑定(bind)才能在两个控件上工作,但是我不希望用户控件的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("");
        }
    }
}

最佳答案

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

这样做时,您将无法再使用其他DataContext,因为UserControl的DataContext是硬编码的。

在绑定(bind)的情况下,通常会继承DataContext,因此Visibility绑定(bind)可以在当前MyControlVisible上找到属性DataContext,但是由于您在UserControl的构造函数中对DataContext进行了硬编码,因此找不到该属性。

您可以在绑定(bind)中指定其他绑定(bind)源,例如

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

但是,这只是针对这种特定情况的问题的解决方法,我认为这不是永久性的解决方案。更好的解决方案是不简单地将DataContext硬编码到您的UserControl

根据UserControl的用途和应用程序的设计方式,您可以采用几种不同的方法。
  • 您可以在UserControl上创建一个DependencyProperty来传递值并绑定(bind)到该值。
    <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,以为其Model期望使用特定类型的DataContext,因此通常,主窗口的XAML如下所示:
    <DataTemplate DataType="{x:Type local:MyUControlModel}">
        <Controls:MyUserControl />
    </DataTemplate>
    
    <ContentPresenter Content="{Binding MyUControlModelProperty}" ... />
    
  • 关于c# - 我的WPF自定义控件的数据上下文取代了父级的,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/16488284/

    10-13 07:05