覆盖WPF控件

时间:2013-06-19 14:46:31

标签: c# wpf inheritance

我有一个来自第三方的WPF控件ParentWPFControl我想继承(让我们调用子类ChildWPFControl)。在此过程中,我计划覆盖一些后端逻辑和前端样式的一部分。我可以做前者就好了但我做后者时遇到了问题。

我尝试使用xaml< - >子国家/地区的xaml.cs结构,但似乎不允许来自VS的以下警告:

Partial declarations of 'ChildWPFControl' must not specify different base classes

现在,我想我可以编写一个ResourceDictionary XAML并在那里定义前端,但如果我想向XAML添加事件处理程序(至少我找不到办法做到这一点),这就成了一个问题< / p>

我的另一个选择是直接在使用ChildWPFControl的对象中定义覆盖模板,但这样可以减少模块化设计。

我能想到的最后一个选择是制作一个xaml&lt; - &gt; xaml.cs对是一个XAML样式的容器,然后强制ChildWPFControl使用通过后端事件处理程序定义的ControlTemplate。

无论如何,我正在寻找的是一个优雅的模块化解决方案来解决我的问题。任何建议都会受到欢迎。

由于

3 个答案:

答案 0 :(得分:6)

完全覆盖WPF控件需要几个步骤。有些是必要的,根据您的需要,有些是可选的。我将为您解释两个重要的问题:

创建新的默认样式

每个WPF控件都有一个默认样式,其中包含可视表示和覆盖属性。现在,如果你从控件派生,WPF仍然认为你想使用这个默认样式,要改变你在这样的静态构造函数中更改DefaultStyle

class MyButton : Button
{
    static MyButton()
    {
        DefaultStyleKeyProperty.OverrideMetadata(typeof(MyButton), new FrameworkPropertyMetadata(typeof(MyButton)));
    }
}

现在,如果您使用MyButton,WPF尝试为MyButton查找样式,而不是Button。 OverridesDefaultStyle是一种风格的属性,在某些方面也可能很方便。通常这些默认样式应放在与主题相关的xaml中。

覆盖课程时的事件处理程序

ControlTemplateStyle这是正确的,你不能使用像Click="OnClick"这样的事件的语法糖。关键是,视觉表示尽可能地与逻辑部分分离。使用OnApplyTemplate方法还有其他方法可以解决这个问题。通过覆盖它,您可以询问模板“给我这个控件”,然后在那里添加您的事件。

override OnApplyTemplate()
{
    var innerChild = Template.FindName("PART_InnerChild", this) as MyInnerControl;
    if(innerChild != null)
        innerChild.SomeEvent += OnSomeEvent;
}

注意:这些控件的名称通常以PART_方式开头,这可以在WPF基本控件中看到。告诉设计师这是一个很好的方式“没有这个控制,逻辑部分可能会破坏”。还有属性TemplatePart,但它并不重要,WPF并不关心它。 AFAIK Expression混合使用它,我个人用它来告诉其他人什么样的内部控制是绝对必要的,以使这个控制工作。

个人建议

从类中派生通常是我们在尝试自定义控件时所做的最后一步。因为需要做大量工作才能使其完全正常工作并且可能会限制可重用性,所以我们会尽量避免使用它,例如除了模板覆盖和样式之外还有一个很好的替代方案; attached behaviors

最后, 整篇主题包含在nice MSDN文章中。

希望有所帮助

答案 1 :(得分:2)

您可以将用户控件创建为包含基本控件的包装器。通过这种方式,您可以在xaml中更改样式,在C#中为包装控件添加一些逻辑。但这是一个繁琐的过程。

编辑:添加样本(telerik的包装器:RadComboBox)

XAML:

<UserControl x:Class="Controls.SingleDictionaryValueSelector"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:CardControls="clr-namespace:Controls"
             xmlns:telerik="http://schemas.telerik.com/2008/xaml/presentation" MinWidth="150" MinHeight="25" >


    <Grid >
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="*"></ColumnDefinition>
            <ColumnDefinition Width="25"></ColumnDefinition>
        </Grid.ColumnDefinitions>
        <!-- customize visual for wrapped control -->
        <telerik:RadComboBox x:Name="cb" 
                            Grid.Column="0"
                            VerticalAlignment="Center"
                            SelectedValuePath="Key"        
                            ClearSelectionButtonContent="Clear"
                            ClearSelectionButtonVisibility="Visible"
                            CanAutocompleteSelectItems="True"
                            CanKeyboardNavigationSelectItems="True"
                            SelectAllTextEvent="None"
                            OpenDropDownOnFocus="false"
                            IsFilteringEnabled="True"
                            TextSearchMode="Contains"
                            EmptyText="Select item" 
                            telerik:StyleManager.Theme="Metro"
                            FontFamily="Calibri"
                            FontSize="14"
                            IsEditable="True"
                            Foreground="#666" 
                            KeyDown="cb_KeyDown"
                            SelectionChanged="cb_SelectionChanged"
                            GotMouseCapture="cb_GotMouseCapture" 
                            DropDownOpened="cb_DropDownOpened" 
                            KeyUp="cb_KeyUp">

            <telerik:RadComboBox.ItemTemplate>
                <DataTemplate>
                    <TextBlock TextWrapping="Wrap" Width="{Binding RelativeSource={RelativeSource AncestorType=telerik:RadComboBox},Path=ActualWidth}" Text="{Binding Path=Value}" />
                </DataTemplate>
            </telerik:RadComboBox.ItemTemplate>
        </telerik:RadComboBox>

        <CardControls:ErrorInfo x:Name="errorInfoControl"  Grid.Column="1"  Visibility="Hidden"></CardControls:ErrorInfo>
    </Grid>
</UserControl>

CS:

public partial class SingleDictionaryValueSelector : IMyCustomInterface
{
     ....

    private void cb_KeyDown(object sender, KeyEventArgs e)
    {
        RadComboBox senderCombo = sender as RadComboBox;
        ...

    }

    private void cb_KeyUp(object sender, KeyEventArgs e)
    {
        SearchExecute();
    }





    private void cb_SelectionChanged(object sender, System.Windows.Controls.SelectionChangedEventArgs e)
    {
        RadComboBox senderCombo = sender as RadComboBox;

        ...  
    }

    private void cb_DropDownOpened(object sender, EventArgs e)
    {
       ...
    }

    ...

}

答案 2 :(得分:-1)

看起来你的继承混淆得比不允许的更多。 xaml的root元素必须与xaml.cs的基类匹配。

如果要在同一个项目中定义基类,则无法将其用作xaml中的基类,因为它本身仍然是xaml而不是已编译的控件。解决这个问题的一些方法:你可以在一个单独的项目中编译它并引用它,你可以完全用.cs编译基类而不是部分类,或者你可以使用一些样式魔法。以下是最后两个示例的链接:http://svetoslavsavov.blogspot.ca/2009/09/user-control-inheritance-in-wpf.html