gpt4 book ai didi

C# WPF 无法从 UserControl 访问 ControlTemplate 中定义的按钮

转载 作者:太空宇宙 更新时间:2023-11-03 16:05:36 25 4
gpt4 key购买 nike

这是我在 stackoverlow 上的第一篇文章,我认为这是 C# 和 WPF 的最佳在线帮助网站。

在过去三天里,我研究了大约六个多小时,试图找出如何将按钮处理程序事件附加到我在样式中定义的按钮。我的情况似乎与我在网上看到的所有其他示例不同。因此,我非常感谢您的意见。

MainWindow.xaml 中的一个片段,我在网格中定义了以下内容

 <ScrollViewer Grid.Column="2" Grid.ColumnSpan="2" 
HorizontalScrollBarVisibility="Visible"
VerticalScrollBarVisibility="Hidden"
Background="AliceBlue">
<ItemsControl ItemsSource="{Binding LaserDataItems}">
<ItemsControl.ItemsPanel>
<ItemsPanelTemplate>
<StackPanel Orientation="Horizontal" Background="AliceBlue" />
</ItemsPanelTemplate>
</ItemsControl.ItemsPanel>
</ItemsControl>
</ScrollViewer>

LaserDataItems 上的此绑定(bind)将包含要显示的用户控件列表。这些是根据需要动态创建的。但我实际上试图在控件的代码后面完成我的所有工作(如下)

在我向您展示 UserControl 之前,这里是我要与 UserControl 一起使用的样式。我的第一个目标是在加载按钮后在按钮上创建点击事件

<Style x:Key="HeaderStyle" TargetType="{x:Type DataGridColumnHeader}">            
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="{x:Type DataGridColumnHeader}">
<Button x:Name="MyButton" Height="20"/>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>

这是用户控件

<UserControl x:Class="EClient.Controls.GetAllLaserData"
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"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="300">
<DataGrid Name="laserDataGrid"
ColumnHeaderStyle="{StaticResource HeaderStyle}"
AutoGenerateColumns="False"
ItemsSource="{Binding }"
IsReadOnly="True"
RowHeaderWidth="0"
Height="567"
VerticalAlignment="Top"
RowHeight="17">
<DataGrid.Columns>
<DataGridTextColumn
Width="53"
Header="{Binding HeaderName}"
Binding="{Binding Name}"></DataGridTextColumn>
</DataGrid.Columns>
<DataGrid.RowStyle>
<Style TargetType="DataGridRow">
<Setter Property="IsEnabled" Value="False"/>
</Style>
</DataGrid.RowStyle>
</DataGrid>
</UserControl>

在此 UserControl 的 DataGrid 中是一个 ColumnHeaderStyle,它被分配了模板:HeaderStyle。这种样式有一个按钮,我需要在它被实例化时附加一个点击处理程序。我该怎么做呢。我试图通过覆盖 OnApplyTemplate 方法来访问它,但我尝试的所有操作都返回 null。有没有可能这个方法运行的时候按钮还没有实例化?必须有办法到达此按钮。

当我运行它并成功创建这些用户控件(数据网格)的列表时,可以在第一个单元格(这是每个网格的标题)中看到一个按钮。

我还尝试使用遍历可视化树的通用例程,但无济于事。这其实是我应该追求的东西吗,我真的是一筹莫展。


以下文章于 2013 年 10 月 26 日添加

虽然我已经找到了解决这个问题的另一种方法,但我仍然有兴趣解决它。我真的认为这是一个很好的问题。

无论如何,我想我应该放一张我正在尝试做的事情的照片。也许这样可以更好地看待手头的问题。

Datagrid with Button Above Grid

最佳答案

不是直接的答案 - 但这是我处理问题的方法。我不使用数据网格 - 而是使用 itemscontrols,因此我必须使用 ItemsControl 作为示例。

当处理列表中的按钮时,我通常将包含按钮的控件拆分到它自己的用户控件中(根据需要应用样式)。然后我会在后台有一个事件处理程序,例如XAML

<UserControl x:Class="BindableTest.UserControls.CatItem"
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"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="400">
<StackPanel VerticalAlignment="Top" HorizontalAlignment="Left" Margin="0 0 10 0">
<Label Content="{Binding Path=Model.DisplayNumber}" HorizontalAlignment="Left" Height="25" Width="25" />
<TextBox BorderBrush="Black" Text="{Binding Path=Model.SomeText, UpdateSourceTrigger=PropertyChanged}" Width="150" Height="25" />
<StackPanel Orientation="Horizontal">
<Button x:Name="btnDelete" Margin="5 0 0 0" Content="Delete" Click="btnDelete_Click_1"></Button>
</StackPanel>
</StackPanel>
</UserControl>

和代码隐藏...

  /// <summary>
/// Interaction logic for CatItem.xaml
/// </summary>
public partial class CatItem : UserControl
{
public CatItem()
{
InitializeComponent();
}

private void btnDelete_Click_1(object sender, RoutedEventArgs e)
{
if (DeleteClick != null)
DeleteClick(sender, e);
}

public event RoutedEventHandler DeleteClick;


}

然后您可以从调用 xaml 中使用此事件,如下所示。

<ItemsControl x:Name="itmsList" ItemsSource="{Binding Path=ModelListVMs}">
<ItemsControl.ItemsPanel>
<ItemsPanelTemplate>
<StackPanel Orientation="Horizontal"></StackPanel>
</ItemsPanelTemplate>
</ItemsControl.ItemsPanel>
<ItemsControl.ItemTemplate>
<DataTemplate>
<uc:CatItem DeleteClick="ItemDeleteClick"></uc:CatItem> <!-- Using click here -->
</DataTemplate>
</ItemsControl.ItemTemplate>
</ItemsControl>

然后您可以在窗口的背景中选择任何按钮的事件。

public void ItemDeleteClick(object sender, RoutedEventArgs e)
{
var ctx = (sender as Button);

//can get datacontext and other properties easily from here
}

抱歉,这不是一个直接的答案,但希望它能帮助您朝着正确的方向前进。

关于C# WPF 无法从 UserControl 访问 ControlTemplate 中定义的按钮,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19576470/

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