为什么DataTemplate return class 名称而不是控件?
Why DataTemplate return class name instead of control?
我想创建自定义控件,例如添加 属性 Info 和 InfoTemplate。
我在 generic.xaml 中使用 ContetnPresenter 为 Info 属性.
定义了 ControlTemplate
当我不使用 InfoTemplate 时,它工作正常,但当我应用 ItemTemplate 时,内容表示为 class 名称字符串。应用于 GroupBox 的相同模板按预期工作。我做错了什么?我需要 OnApplyTemplate 中的一些额外代码吗?
贝娄是打印屏幕我的应用程序和来源。红色边框是一个 GroupBox,蓝色是我的控件。绿色边框是 DataTemplate 的一部分。
编辑:
为了测试,我创建 class MyGroupBox 继承自 GroupBox 并覆盖方法 OnHeaderChanged
public class MyGroupBox : GroupBox
{
protected override void OnHeaderChanged(object oldHeader, object newHeader)
{
//base.OnHeaderChanged(oldHeader, newHeader);
}
}
在那种情况下 GroupBox.Heder 表现得像我的 MyCustomControl 并显示文本而不是控件。所以问题是:我应该在我的控制事件中实现什么才能像我想要的那样工作?
MyCustomControl.cs
using System.Windows;
using System.Windows.Controls;
namespace WpfApplication7
{
public class MyCustomControl : ContentControl
{
static MyCustomControl()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), new FrameworkPropertyMetadata(typeof(MyCustomControl)));
}
public override void OnApplyTemplate()
{
base.OnApplyTemplate();
}
public object Info
{
get { return (object)GetValue(InfoProperty); }
set { SetValue(InfoProperty, value); }
}
public DataTemplate InfoTemplate
{
get { return (DataTemplate)GetValue(InfoTemplateProperty); }
set { SetValue(InfoTemplateProperty, value); }
}
public static readonly DependencyProperty InfoProperty =
DependencyProperty.Register(nameof(Info), typeof(object), typeof(MyCustomControl), new PropertyMetadata(null));
public static readonly DependencyProperty InfoTemplateProperty =
DependencyProperty.Register(nameof(InfoTemplate), typeof(DataTemplate), typeof(MyCustomControl), new PropertyMetadata(null));
}
}
Generic.xaml
<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:WpfApplication7">
<Style TargetType="{x:Type local:MyCustomControl}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type local:MyCustomControl}">
<StackPanel>
<TextBlock FontWeight="Bold">Info</TextBlock>
<ContentPresenter ContentSource="Info"/>
<TextBlock FontWeight="Bold">Content</TextBlock>
<ContentPresenter/>
</StackPanel>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</ResourceDictionary>
MainWindow.xml
<Window x:Class="WpfApplication7.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local="clr-namespace:WpfApplication7"
mc:Ignorable="d"
Title="MainWindow" Height="350" Width="525">
<Window.DataContext>
DATA_CONTEXT
</Window.DataContext>
<Window.Resources>
<DataTemplate x:Key="dataTemplate">
<Border BorderBrush="Green" BorderThickness="5">
<ContentPresenter Content="{Binding}"/>
</Border>
</DataTemplate>
</Window.Resources>
<StackPanel>
<Border BorderBrush="Red" BorderThickness="4">
<GroupBox HeaderTemplate="{StaticResource dataTemplate}">
<GroupBox.Header>
<TextBlock Text="{Binding}"/>
</GroupBox.Header>
</GroupBox>
</Border>
<Border BorderBrush="Blue" BorderThickness="4">
<local:MyCustomControl InfoTemplate="{StaticResource dataTemplate}">
<local:MyCustomControl.Info>
<TextBlock Text="{Binding}"/>
</local:MyCustomControl.Info>
My content
</local:MyCustomControl>
</Border>
</StackPanel>
</Window>
MainWindow.xaml.cs
using System.Collections.Generic;
using System.Windows;
namespace WpfApplication7
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
}
}
所以放弃我的回答并重新开始,因为我对所要求的内容有了更好的理解。这里的想法基本上是重新创建一个自定义 GroupBox
之类的控件。问题是自定义控件中 Info
属性 的 DataContext
(基本上是 GroupBox
的 Header
属性)不是t 作为自定义控件本身的 DataContext
出现,就像使用 GroupBox
时一样。
所以问题在于,您设置为 Info
属性 的 UI 块永远不会添加为控件的逻辑 child,因此它不会' 以继承 DataContext
的方式添加,就像在 GroupBox
中使用相同代码时发生的那样。为此,只需将您的自定义控件 class 更新为以下内容:
public class MyCustomControl : ContentControl
{
static MyCustomControl()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), new FrameworkPropertyMetadata(typeof(MyCustomControl)));
}
public override void OnApplyTemplate()
{
base.OnApplyTemplate();
}
public object Info
{
get { return (object)GetValue(InfoProperty); }
set { SetValue(InfoProperty, value); }
}
public DataTemplate InfoTemplate
{
get { return (DataTemplate)GetValue(InfoTemplateProperty); }
set { SetValue(InfoTemplateProperty, value); }
}
public static readonly DependencyProperty InfoProperty =
DependencyProperty.Register(nameof(Info), typeof(object), typeof(MyCustomControl), new FrameworkPropertyMetadata(null, new PropertyChangedCallback(MyCustomControl.OnHeaderChanged)));
private static void OnHeaderChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var obj = (MyCustomControl)d;
obj.RemoveLogicalChild(e.OldValue);
obj.AddLogicalChild(e.NewValue);
}
public static readonly DependencyProperty InfoTemplateProperty =
DependencyProperty.Register(nameof(InfoTemplate), typeof(DataTemplate), typeof(MyCustomControl), new PropertyMetadata(null));
}
这将解决这个问题,但是,应该指出的是,从 HeaderedContentControl
派生而不是从 ContentControl
派生可能会更清晰,因为 HeaderedContentControl
已经拥有所有这个设置为您开箱即用,并且已经有一个 Header
和 HeaderTemplate
可以用来代替您的 Info
和 InfoTemplate
属性,这将节省你一些代码。
如果你想让这个工作而不用担心逻辑 children 等等你可以只更新你设置 Info
的 UI 块中的绑定和让它使用 RelativeSource
来搜索自定义控件祖先,然后使用 "DataContext" 的路径,这将手动覆盖整个问题,尽管您必须记住每次都这样做。
我敢打赌,您最好只从 HeaderedContentControl
派生,它看起来应该包含您正在寻找的所有功能。
我想创建自定义控件,例如添加 属性 Info 和 InfoTemplate。
我在 generic.xaml 中使用 ContetnPresenter 为 Info 属性.
定义了 ControlTemplate
当我不使用 InfoTemplate 时,它工作正常,但当我应用 ItemTemplate 时,内容表示为 class 名称字符串。应用于 GroupBox 的相同模板按预期工作。我做错了什么?我需要 OnApplyTemplate 中的一些额外代码吗?
贝娄是打印屏幕我的应用程序和来源。红色边框是一个 GroupBox,蓝色是我的控件。绿色边框是 DataTemplate 的一部分。
编辑: 为了测试,我创建 class MyGroupBox 继承自 GroupBox 并覆盖方法 OnHeaderChanged
public class MyGroupBox : GroupBox
{
protected override void OnHeaderChanged(object oldHeader, object newHeader)
{
//base.OnHeaderChanged(oldHeader, newHeader);
}
}
在那种情况下 GroupBox.Heder 表现得像我的 MyCustomControl 并显示文本而不是控件。所以问题是:我应该在我的控制事件中实现什么才能像我想要的那样工作?
MyCustomControl.cs
using System.Windows;
using System.Windows.Controls;
namespace WpfApplication7
{
public class MyCustomControl : ContentControl
{
static MyCustomControl()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), new FrameworkPropertyMetadata(typeof(MyCustomControl)));
}
public override void OnApplyTemplate()
{
base.OnApplyTemplate();
}
public object Info
{
get { return (object)GetValue(InfoProperty); }
set { SetValue(InfoProperty, value); }
}
public DataTemplate InfoTemplate
{
get { return (DataTemplate)GetValue(InfoTemplateProperty); }
set { SetValue(InfoTemplateProperty, value); }
}
public static readonly DependencyProperty InfoProperty =
DependencyProperty.Register(nameof(Info), typeof(object), typeof(MyCustomControl), new PropertyMetadata(null));
public static readonly DependencyProperty InfoTemplateProperty =
DependencyProperty.Register(nameof(InfoTemplate), typeof(DataTemplate), typeof(MyCustomControl), new PropertyMetadata(null));
}
}
Generic.xaml
<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:WpfApplication7">
<Style TargetType="{x:Type local:MyCustomControl}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type local:MyCustomControl}">
<StackPanel>
<TextBlock FontWeight="Bold">Info</TextBlock>
<ContentPresenter ContentSource="Info"/>
<TextBlock FontWeight="Bold">Content</TextBlock>
<ContentPresenter/>
</StackPanel>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</ResourceDictionary>
MainWindow.xml
<Window x:Class="WpfApplication7.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local="clr-namespace:WpfApplication7"
mc:Ignorable="d"
Title="MainWindow" Height="350" Width="525">
<Window.DataContext>
DATA_CONTEXT
</Window.DataContext>
<Window.Resources>
<DataTemplate x:Key="dataTemplate">
<Border BorderBrush="Green" BorderThickness="5">
<ContentPresenter Content="{Binding}"/>
</Border>
</DataTemplate>
</Window.Resources>
<StackPanel>
<Border BorderBrush="Red" BorderThickness="4">
<GroupBox HeaderTemplate="{StaticResource dataTemplate}">
<GroupBox.Header>
<TextBlock Text="{Binding}"/>
</GroupBox.Header>
</GroupBox>
</Border>
<Border BorderBrush="Blue" BorderThickness="4">
<local:MyCustomControl InfoTemplate="{StaticResource dataTemplate}">
<local:MyCustomControl.Info>
<TextBlock Text="{Binding}"/>
</local:MyCustomControl.Info>
My content
</local:MyCustomControl>
</Border>
</StackPanel>
</Window>
MainWindow.xaml.cs
using System.Collections.Generic;
using System.Windows;
namespace WpfApplication7
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
}
}
所以放弃我的回答并重新开始,因为我对所要求的内容有了更好的理解。这里的想法基本上是重新创建一个自定义 GroupBox
之类的控件。问题是自定义控件中 Info
属性 的 DataContext
(基本上是 GroupBox
的 Header
属性)不是t 作为自定义控件本身的 DataContext
出现,就像使用 GroupBox
时一样。
所以问题在于,您设置为 Info
属性 的 UI 块永远不会添加为控件的逻辑 child,因此它不会' 以继承 DataContext
的方式添加,就像在 GroupBox
中使用相同代码时发生的那样。为此,只需将您的自定义控件 class 更新为以下内容:
public class MyCustomControl : ContentControl
{
static MyCustomControl()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), new FrameworkPropertyMetadata(typeof(MyCustomControl)));
}
public override void OnApplyTemplate()
{
base.OnApplyTemplate();
}
public object Info
{
get { return (object)GetValue(InfoProperty); }
set { SetValue(InfoProperty, value); }
}
public DataTemplate InfoTemplate
{
get { return (DataTemplate)GetValue(InfoTemplateProperty); }
set { SetValue(InfoTemplateProperty, value); }
}
public static readonly DependencyProperty InfoProperty =
DependencyProperty.Register(nameof(Info), typeof(object), typeof(MyCustomControl), new FrameworkPropertyMetadata(null, new PropertyChangedCallback(MyCustomControl.OnHeaderChanged)));
private static void OnHeaderChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var obj = (MyCustomControl)d;
obj.RemoveLogicalChild(e.OldValue);
obj.AddLogicalChild(e.NewValue);
}
public static readonly DependencyProperty InfoTemplateProperty =
DependencyProperty.Register(nameof(InfoTemplate), typeof(DataTemplate), typeof(MyCustomControl), new PropertyMetadata(null));
}
这将解决这个问题,但是,应该指出的是,从 HeaderedContentControl
派生而不是从 ContentControl
派生可能会更清晰,因为 HeaderedContentControl
已经拥有所有这个设置为您开箱即用,并且已经有一个 Header
和 HeaderTemplate
可以用来代替您的 Info
和 InfoTemplate
属性,这将节省你一些代码。
如果你想让这个工作而不用担心逻辑 children 等等你可以只更新你设置 Info
的 UI 块中的绑定和让它使用 RelativeSource
来搜索自定义控件祖先,然后使用 "DataContext" 的路径,这将手动覆盖整个问题,尽管您必须记住每次都这样做。
我敢打赌,您最好只从 HeaderedContentControl
派生,它看起来应该包含您正在寻找的所有功能。