2

我有一个来自第三方的 WPF 控件 ParentWPFControl,我想从中继承(我们称其为子类 ChildWPFControl)。在此过程中,我计划重写一些后端逻辑和部分前端样式。我可以做前者就好,但我做后者有问题。

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

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

现在,我想我可以编写一个 ResourceDictionary XAML 并在那里定义前端,但是如果我想向 XAML 添加事件处理程序,这就会成为一个问题(至少我找不到这样做的方法)

我的另一种选择是直接在使用 ChildWPFControl 的对象中定义覆盖模板,但这会使设计的模块化程度降低。

我能想到的最后一个替代方法是创建一个 xaml <-> xaml.cs 对,它是一个 XAML 样式容器,然后强制 ChildWPFControl 使用通过后端事件处理程序定义的 ControlTemplate。

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

谢谢

4

3 回答 3

7

完全覆盖 WPF 控件需要几个步骤。有些是必需的,有些是可选的,具体取决于您的需要。我将为您解释两个重要的:

创建新的默认样式

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

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

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

覆盖类时的事件处理程序

在 a 中是正确的, ControlTemplate否则Style您不能使用使用 event like 的语法糖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 blend 对它进行了一些处理,我个人用它来告诉其他人什么样的内部控件对于使这个控件起作用是绝对必要的。

个人建议

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

最后,一篇不错的 MSDN文章涵盖了整个主题。

希望有帮助

于 2013-06-19T17:46:43.543 回答
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)
    {
       ...
    }

    ...

}
于 2013-06-19T15:02:50.493 回答
-1

看起来你的继承权比不允许的要多。xaml 的根元素必须与 xaml.cs 的基类匹配。

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

于 2013-06-19T15:20:05.830 回答