为什么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(基本上是 GroupBoxHeader 属性)不是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 已经拥有所有这个设置为您开箱即用,并且已经有一个 HeaderHeaderTemplate 可以用来代替您的 InfoInfoTemplate 属性,这将节省你一些代码。

如果你想让这个工作而不用担心逻辑 children 等等你可以只更新你设置 Info 的 UI 块中的绑定和让它使用 RelativeSource 来搜索自定义控件祖先,然后使用 "DataContext" 的路径,这将手动覆盖整个问题,尽管您必须记住每次都这样做。

我敢打赌,您最好只从 HeaderedContentControl 派生,它看起来应该包含您正在寻找的所有功能。