首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >重写WPF控件

重写WPF控件
EN

Stack Overflow用户
提问于 2013-06-19 22:46:31
回答 3查看 8.3K关注 0票数 2

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

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

代码语言:javascript
复制
Partial declarations of 'ChildWPFControl' must not specify different base classes

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

我的另一种选择是直接在使用ChildWPFControl的对象中定义覆盖模板,但这会使设计变得不那么模块化。

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

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

谢谢

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2013-06-20 01:46:44

要完全重写WPF控件,需要执行几个步骤。有些是必需的,有些则是可选的,具体取决于您的需求。我将为您解释两个重要的问题:

创建新的默认样式

每个WPF控件都有一个默认样式,其中包含它的可视化表示形式和重写属性。现在,如果从控件派生的WPF仍然认为您想要使用这种默认样式,则可以更改静态构造函数中的DefaultStyle,如下所示

代码语言:javascript
复制
class MyButton : Button
{
    static MyButton()
    {
        DefaultStyleKeyProperty.OverrideMetadata(typeof(MyButton), new FrameworkPropertyMetadata(typeof(MyButton)));
    }
}

现在,如果你使用MyButton,WPF会尝试为MyButton找到一个样式,而不是为按钮。OverridesDefaultStyle是一种风格的属性,在某些情况下可能也很方便。通常,这些默认样式应该放在与xaml相关的主题中。

重写类时的事件处理程序

ControlTemplateStyle中它是正确的,你不能使用像Click="OnClick"这样的事件的语法糖。重点是,视觉表示尽可能地与逻辑部分解耦。不过,还有其他方法可以克服这个问题,使用OnApplyTemplate方法。通过重写它,你请求模板“给我这个控件”,然后你只需要在那里添加你的事件。

代码语言:javascript
复制
override OnApplyTemplate()
{
    var innerChild = Template.FindName("PART_InnerChild", this) as MyInnerControl;
    if(innerChild != null)
        innerChild.SomeEvent += OnSomeEvent;
}

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

个人建议

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

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

希望这能有所帮助

票数 7
EN

Stack Overflow用户

发布于 2013-06-19 23:02:50

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

编辑:添加示例(telerik的包装器:RadComboBox)

XAML:

代码语言:javascript
复制
<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>

政务司司长:

代码语言:javascript
复制
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
EN

Stack Overflow用户

发布于 2013-06-19 23:20:06

看起来你把你的遗产搞混了,而不是不允许的。xaml的根元素必须与xaml.cs的基类匹配。

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

票数 -1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/17194333

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档