假设我有一个包含另一个控件的自定义控件(例如MyCustomButton)。我公开了一个属性Content,它包含了内部控件:
public object Content
{
get { return innerControl.Content; }
set { innerControl.Content = value; }
}
为了让使用者绑定到这个属性,我需要为它定义一个DependencyProperty:
public static DependencyProperty ContentProperty = DependencyProperty.Register("Content", typeof (object), typeof (MyCustomButton));
但现在我需要我的属性定义来使用GetValue / SetValue:
public object Content
{
get { return GetValue(ContentProperty); }
set { SetValue(ContentProperty, value); }
}
所以我不再包装内部控件的值了。
我可以定义PropertyMetadata来处理DependencyProperty的PropertyChanged事件,但是我需要一堆管道代码来保持值同步并防止无限制的环回更改。
更新:我不能只从Button派生,因为我的UserControl有各种其他问题。
有更好的方法吗?
答案 0 :(得分:2)
那么,根据您使用用户控件包装按钮的原因的详细信息,您可以定义一个从按钮继承的自定义控件。然后,您可以简单地覆盖要定义自定义控件的行为的方法和属性,而不是包装按钮并公开所需的包装方法和属性。这样,您将获得按钮的所有功能,而无需重新发明轮子。
这是一个谷歌链接,引导您完成它(我找到的第一个 - 有很多):http://knol.google.com/k/creating-custom-controls-with-c-net#
如果用户控件有其他问题,这可能不适合你,但我提供这个答案,因为你提到它的唯一目的是包装按钮。我个人赞成创建自定义控件并继承而不是用户控件和包装,如果有问题的控件只是一种更具体的包装/继承控件(即你的情况下的按钮)。
编辑:根据更新的问题......
你可以沿着这些方向做点什么。以下是用户控件客户端的XAML:
<Grid>
<local:MyControl ButtonContent="Click Me!"/>
</Grid>
</Window>
以下是用户控件本身的XAML:
<UserControl x:Class="GuiScratch.MyControl"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:GuiScratch"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="300">
<Grid>
<StackPanel>
<ContentControl Content="Asdf"/>
<Button Content="{Binding RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type local:MyControl}},Path=ButtonContent}"/>
</StackPanel>
</Grid>
</UserControl>
而且,这是背后的代码:
public partial class MyControl : UserControl
{
public static readonly DependencyProperty ButtonContentProperty =
DependencyProperty.Register("ButtonContent", typeof(object), typeof(MyControl),
new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.AffectsRender));
public object ButtonContent
{
get { return (object)GetValue(ButtonContentProperty); }
set { SetValue(ButtonContentProperty, value); }
}
public MyControl()
{
InitializeComponent();
}
}
因此,您无需通过代码处理绑定。您的客户端XAML绑定到您的依赖项属性,用户控件本身的XAML也是如此。以这种方式,他们共享依赖属性设置。我在我的小便笺簿中运行它,结果是(至少我的理解)你正在寻找什么。主窗口将用户控件显示为带有文本“Asdf”的堆栈面板,然后显示带有“Click Me!”文本的按钮