gpt4 book ai didi

c# - 派生的 UserControl 未在 parent 更改 DependencyProperty 时得到通知

转载 作者:行者123 更新时间:2023-11-30 12:59:08 24 4
gpt4 key购买 nike

我创建了一个基础 ClassA,其中包含一个 DependencyProperty MyText。我从中派生了一个 ClassB。此 ClassB 嵌入在 MyCustomButton 中。在 MouseHover/MousePressed 上,我更改了嵌入式 ClassB 的 MyText-Value。但是,当父 ClassA 触发 PropertyChanged-Event 时,派生的 ClassB 永远不会更改属性。

将更改的 DependencyProperty 通知派生类 B 的正确方法是什么?我已经尝试过“Base class DependencyProperty value change in WPF”中提到的解决方案,但也无法使其正常工作。

我的代码:

主窗口仅包含一个包含 ClassB 实例的 MyCustomButton:

<Window.Resources>
<cc:ClassB x:Key="MyClassB" MyText="MyClassBText"/>
</Window.Resources>

<Grid>
<StackPanel>
<cc:MyCustomButton MyClassA="{StaticResource MyClassB}" Width="100" Height="100"/>
</StackPanel>
</Grid>

MyCustomButton 持有 ClassA 的实例(实际上是派生的 ClassB 的实例):

class MyCustomButton : Button
{
public ClassA MyClassA
{
get { return (ClassA)GetValue(MyClassAProperty); }
set { SetValue(MyClassAProperty, value); }
}
public static readonly DependencyProperty MyClassAProperty =
DependencyProperty.Register("MyClassA", typeof(ClassA), typeof(MyCustomButton));
}

<Style TargetType="{x:Type cc:MyCustomButton}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type cc:MyCustomButton}">

<Border BorderBrush="{TemplateBinding BorderBrush}" BorderThickness="{TemplateBinding BorderThickness}">
<StackPanel Background="Transparent">
<cc:ClassA x:Name="ButtonClassA" Content="{Binding MyClassA, RelativeSource={RelativeSource TemplatedParent}}" />
</StackPanel>
</Border>

<ControlTemplate.Triggers>
<Trigger Property="IsMouseOver" Value="true">
<Setter TargetName="ButtonClassA" Property="MyText" Value="HOVER"/>
</Trigger>
<Trigger Property="IsPressed" Value="true">
<Setter TargetName="ButtonClassA" Property="MyText" Value="PRESSED"/>
</Trigger>
</ControlTemplate.Triggers>

</ControlTemplate>

</Setter.Value>
</Setter>
</Style>

ClassA 拥有 DependencyProperty:

public class ClassA : UserControl
{
public string MyText
{
get { return (string)GetValue(MyTextProperty); }
set { SetValue(MyTextProperty, value); }
}
public static readonly DependencyProperty MyTextProperty =
DependencyProperty.Register("MyText", typeof(string), typeof(ClassA), new PropertyMetadata("ClassA Default Text", MyTextPropertyChanged));

private static void MyTextPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
Console.WriteLine("ClassA.MyTextPropertyChanged: " + e.NewValue);
}

public ClassA()
{
}
}

B类派生自A类

static ClassB()
{
MyTextProperty.AddOwner(typeof(ClassB), new PropertyMetadata(MyTextPropertyChanged));
}

private static void MyTextPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
Console.WriteLine("ClassB.MyTextPropertyChanged: " + e.NewValue); // This is just reached once at startup!
}

public ClassB()
{
InitializeComponent();
}

<local:ClassA
xmlns:local="clr-namespace:CustomControlPlayground.CustomControls"
x:Class="CustomControlPlayground.CustomControls.ClassB"
x:Name="ClassBXAML">

<Grid>
<TextBlock Text="{Binding MyText, ElementName=ClassBXAML}"/>
</Grid>
</local:ClassA>

更新:

到目前为止我得到的所有答案似乎都是正确的,但对我的问题没有帮助。我认为问题可能出在我绑定(bind)派生类的方式上。难道我的 ClassB MyClassB 总是被转换为 ClassA 吗?

在我的 MainWindow.xaml 中:

<Window.Resources>
<cc:ClassB x:Key="MyClassB" MyText="MyClassBText"/>
</Window.Resources>
<Grid>
<cc:MyCustomButton MyClassA="{StaticResource MyClassB}" Width="100" Height="100"/>
</Grid>

在 MyButton 样式中:

<StackPanel Background="Transparent">
<cc:ClassA x:Name="ButtonClassA" Content="{Binding MyClassA, RelativeSource={RelativeSource TemplatedParent}}" />
</StackPanel>

最佳答案

我看到的第一件事是,如果未指定其他内容,依赖属性 将设置为 BindsTwoWayByDefault=false。因此,您只有从源到目标的绑定(bind)。要启用依赖属性允许从目标到源的绑定(bind),您必须在依赖属性创建时声明它:

public static readonly DependencyProperty MyTextPropertyChanged =
DependencyProperty.Register
(
"MyTextProperty",
typeof(ClassA),
typeof(string),
new FrameworkPropertyMetadata
(
default(string),
//the next line enables the binding back to source:
FrameworkPropertyMetadataOptions.BindsTwoWayByDefault,
MyTextPropertyChanged
)
);

public static void MyTextPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs)
{
//handling changed event
}

检查示例项目后,发现您缺少 ClassA 和/或 ClassBstyle。只需删除您的 ClassB 的 XAML,然后使用 C# 代码并在 App.xaml 中添加以下行,然后一切正常:

<Application.Resources>
<Style TargetType="{x:Type customControls:ClassA}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type customControls:ClassA}">
<TextBlock Text="{Binding MyText,
RelativeSource={RelativeSource TemplatedParent}}"/>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</Application.Resources>

从您的 ClassB 中删除 XAML 文件,只使用 C# 代码。

最后我建议你在 ClassA 对应的 ClassB 中写如下:

static ClassA()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(ClassA), new FrameworkPropertyMetadata(typeof(ClassA)));
}

//and in ClassB

static ClassA()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(ClassA), new FrameworkPropertyMetadata(typeof(ClassA)));
}

关于c# - 派生的 UserControl 未在 parent 更改 DependencyProperty 时得到通知,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26930004/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com