WPF自定义控件和直接内容支持

时间:2011-01-08 11:08:55

标签: wpf wpf-controls

我对WPF很新,而且有点卡住,所以任何帮助都会受到赞赏。

我正在尝试编写WPF自定义控件,它封装了我已经工作过的几个功能元素(即排序,过滤,标准菜单等),但是在一个很好的整齐包中以避免重复。

无论如何我已经创建了自定义控件(基于控件),然后在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:Controls.ListViewExtended">
    <Style TargetType="{x:Type local:ListViewExtended}">
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="{x:Type local:ListViewExtended}">
                    <Border Background="{TemplateBinding Background}"
                            BorderBrush="{TemplateBinding BorderBrush}"
                            BorderThickness="{TemplateBinding BorderThickness}">
                        <ListView>
                            <ListView.View>
                                <GridView>
                                <!-- Content goes here -->                             
                                </GridView> 
                            </ListView.View>
                        </ListView>
                    </Border>
                </ControlTemplate>
            </Setter.Value>
        </Setter>
    </Style>
</ResourceDictionary>

当我尝试添加GridViewColumns(或任何控件时),如下所示......

<elv:ListViewExtended>
   <GridView>
        <GridViewColumn Width="140" Header="Column 1" />
        <GridViewColumn Width="140" Header="Column 2" />
        <GridViewColumn Width="140" Header="Column 3" />
   </GridView>
</elv:ListViewExtended>

我得到“...不支持直接内容”错误。

我已经创建了一个依赖属性(再次在下面),允许添加GridView,但它仍然不起作用。

public static DependencyProperty GridViewProperty;

public static string GridViewHeader(DependencyObject target) 
{
    return (string)target.GetValue(GridViewProperty); 
}
public static void GridViewHeader(DependencyObject target, string value) 
{
    target.SetValue(GridViewProperty, value); 
} 

提前致谢

4 个答案:

答案 0 :(得分:18)

您只需指定ContentPropertyAttribute。

[ContentProperty("MainContent")]
public class GroupPanel : Control
{
    public GroupPanel()
    {
        DefaultStyleKey = typeof(GroupPanel);
    }

    public object MainContent
    {
        get { return GetValue(MainContentProperty); }
        set { SetValue(MainContentProperty, value); }
    }

    public static readonly DependencyProperty MainContentProperty =
        DependencyProperty.Register("MainContent", typeof(object), typeof(GroupPanel), null);
}

参考:http://msdn.microsoft.com/en-us/library/system.windows.markup.contentpropertyattribute(v=vs.90).aspx

答案 1 :(得分:3)

我在项目中使用这个简单的解决方案来支持自定义控件中的直接内容:

在项目中添加“CustomControl”并从“UserControl”类而不是“Control”派生此控件:

public class MyCustomControl: UserControl
{
    static MyCustomControl()
    {
        DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), 
            new FrameworkPropertyMetadata(typeof(MyCustomControl)));
    }
}

当您向项目添加CustomControl时,Visual Studio(我使用的是2012)会自动添加一个文件夹“Themes”,其中包含一个名为“Generic.xaml”的文件。 该文件包含一个ResourceDictionary,用于定义CustomControl的样式(模板)。

您将找到CustomControl的基本模板,已用作DefaultStyle。对于直接内容支持,将ContentPresenter放置在此模板内的父内容绑定位置:

<Style TargetType="{x:Type local:MyCustomControl}">
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="{x:Type local:MyCustomControl}">
                <Border Background="{TemplateBinding Background}"
                        BorderBrush="{TemplateBinding BorderBrush}"
                        BorderThickness="{TemplateBinding BorderThickness}">
                    <ContentPresenter Content="{Binding RelativeSource={RelativeSource TemplatedParent}, Path=Content}" />
                </Border>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
</Style>

现在可以向CustomControl添加内容:

<Window x:Class="MyApplication.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:controls="clr-namespace:MyApplication"
        Title="MainWindow" Height="350" Width="525">
    <Grid>
        <controls:MyCustomControl>
                <TextBlock>Hello</TextBlock>
        </controls:MyCustomControl>
    </Grid>
</Window>

希望这有帮助!

答案 2 :(得分:2)

ListView继承您的自定义控件,而不是Control。这肯定会让您更改模板,但我建议您阅读更多关于如何操作的文档(例如Sacha Barber的文章:Creating and consuming a custom WPF control)。

祝你学习顺利!

答案 3 :(得分:0)

不使用

{Binding MainContent ElementName=Self}

使用

Content="{Binding RelativeSource={RelativeSource TemplatedParent}, Path=MainContent}" />

本可以为我节省很多时间。