[在我的主窗口中,我尝试绑定到bool
,但它在我的自定义控件的DataContext
中查找。如果我没有在用户控件中分配DataContext
,则主窗口的绑定有效,但是(显然)这会制动用户控件中的绑定。
这里是错误:
System.Windows.Data错误:40:BindingExpression路径错误:在'对象''MyUserControlModel'(HashCode = 1453241)'上找不到'MyControlVisible'属性。 BindingExpression:Path = MyControlVisible; DataItem ='MyUserControlModel'(HashCode = 1453241);目标元素是“ MyUserControl”(名称=“ _ 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("");
}
}
}
这是永远不要直接从DataContext
本身设置UserControl
的许多原因之一。
这样做时,您将无法再使用其他任何DataContext
,因为UserControl的DataContext
是硬编码的。
[在进行绑定的情况下,通常会继承DataContext
,因此Visibility
绑定可以在当前MyControlVisible
上找到属性DataContext
,但是,因为您在UserControl的构造函数中对DataContext
进行了硬编码,找不到该属性。
您可以在绑定中指定其他绑定源,例如
<Controls:MyUserControl Visibility="{Binding
RelativeSource={RelativeSource AncestorType={x:Type Window}},
Path=DataContext.MyControlVisible,
Converter={StaticResource boolToVis}}" ... />
然而,这只是针对此特定情况的问题的解决方法,我认为这不是永久性的解决方案。更好的解决方案是不简单地在DataContext
UserControl
进行硬编码根据UserControl的用途和应用程序的设计,您可以采用几种不同的方法。
您可以在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
,他们期望Model
使用特定类型的DataContext
,因此通常我在主窗口中使用的XAML看起来像这样:
<DataTemplate DataType="{x:Type local:MyUControlModel}">
<Controls:MyUserControl />
</DataTemplate>
<ContentPresenter Content="{Binding MyUControlModelProperty}" ... />