我正在尝试使用自定义数据模板创建一个ListBox
,但绑定(bind)似乎是错误的。当我尝试使用 PacketItem 模板将项目添加到列表中时,它们会被添加,但三个 HexItem 文本框中的文本仍为空白。当我将列表框的项目模板直接设置为 HexItem 数据模板时,效果很好。仅当嵌套在 PacketItem 数据模板中时才会出现问题。
我确信这是一个简单的修复,似乎当模板嵌套时它不知道在哪里寻找文本绑定(bind)。我认为绑定(bind)到相对源以从父级获取数据上下文是可行的,但这似乎是错误的方法。
修改了 XAML
<UserControl x:Class="wpfPacketBox.ctlPacketBox"
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:wpfPacketBox"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="600">
<UserControl.Resources>
<DataTemplate x:Key="PacketItem">
<Expander IsExpanded="True"
d:DesignWidth="600">
<Expander.HeaderTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock Text="PacketNum" />
<TextBlock Text="Source" />
<TextBlock Text="Dest" />
<TextBlock Text="PacketID (FF0A)" />
</StackPanel>
</DataTemplate>
</Expander.HeaderTemplate>
<Expander.ContentTemplate>
<DataTemplate>
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="10*" />
<ColumnDefinition Width="65*" />
<ColumnDefinition Width="25*" />
</Grid.ColumnDefinitions>
<Grid Grid.Column="0">
<TextBox Text="{Binding Offset}"
FontFamily="Courier New"
Foreground="Blue"
Background="#FFCCCCCC"
TextWrapping="Wrap"
BorderThickness="0" />
</Grid>
<Grid Grid.Column="1">
<TextBox Text="{Binding Hex}"
FontFamily="Courier New"
TextWrapping="Wrap"
BorderThickness="0" />
</Grid>
<Grid Grid.Column="2">
<TextBox Text="{Binding Ascii}"
FontFamily="Courier New"
Background="#FFCCCCCC"
TextWrapping="Wrap"
BorderThickness="0" />
</Grid>
</Grid>
</DataTemplate>
</Expander.ContentTemplate>
</Expander>
</DataTemplate>
</UserControl.Resources>
<Grid>
<ListBox ItemsSource="{Binding Path=Packets}"
ItemTemplate="{StaticResource PacketItem}"
Grid.Row="1"
ScrollViewer.HorizontalScrollBarVisibility="Disabled" />
</Grid>
</UserControl>
代码
using System;
using System.Windows.Controls;
using System.Collections.ObjectModel;
namespace wpfTester.UserControls
{
public partial class test : UserControl
{
public ObservableCollection<clsPacket> Packets;
public test()
{
InitializeComponent();
Packets = new ObservableCollection<clsPacket>();
PacketListBox.ItemsSource = Packets;
for (int i = 0; i <= 100; i++)
{
Packets.Add(new clsPacket() { Offset = i, Hex = "00 11 22 33 44 55 66 77", Ascii = "ABCDEFGH" });
}
}
}
public class clsPacket
{
public long Offset { get; set; }
public String Hex { get; set; }
public String Ascii { get; set; }
}
}
解决方案
经过一番挖掘,我找到了解决方案 here 。问题出在扩展器上。我正在设置扩展器的标题/内容模板。但从未指定标题/内容本身。这是修改后的扩展器标签,解决了我的问题。
<Expander IsExpanded="True" Header="{Binding}" Content="{Binding}">
最佳答案
您需要为其设置DataTemplates
DataType
(clsPacket) 来解析绑定(bind)。
我不确定另一个 DataTemplate
的用途,但这应该为您指明正确的方向,使您的绑定(bind)正常工作。
<UserControl x:Class="WpfApplication11.test"
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:WpfApplication11"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="300">
<UserControl.Resources>
<DataTemplate DataType="{x:Type local:clsPacket}">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="10*" />
<ColumnDefinition Width="65*" />
<ColumnDefinition Width="25*" />
</Grid.ColumnDefinitions>
<Grid Grid.Column="0">
<TextBox Text="{Binding Offset}"
FontFamily="Courier New"
Foreground="Blue"
Background="#FFCCCCCC"
TextWrapping="Wrap"
BorderThickness="0" />
</Grid>
<Grid Grid.Column="1">
<TextBox Text="{Binding Hex}"
FontFamily="Courier New"
TextWrapping="Wrap"
BorderThickness="0" />
</Grid>
<Grid Grid.Column="2">
<TextBox Text="{Binding Ascii}"
FontFamily="Courier New"
Background="#FFCCCCCC"
TextWrapping="Wrap"
BorderThickness="0" />
</Grid>
</Grid>
</DataTemplate>
</UserControl.Resources>
<Grid>
<ListBox ItemsSource="{Binding Path=Packets}"
Grid.Row="1"
ScrollViewer.HorizontalScrollBarVisibility="Disabled" />
</Grid>
</UserControl>
代码:
namespace WpfApplication11
{
public partial class test : UserControl
{
private ObservableCollection<clsPacket> _packets = new ObservableCollection<clsPacket>();
public test()
{
InitializeComponent();
DataContext = this;
for (int i = 0; i <= 100; i++)
{
Packets.Add(new clsPacket() { Offset = i, Hex = "00 11 22 33 44 55 66 77", Ascii = "ABCDEFGH" });
}
}
public ObservableCollection<clsPacket> Packets
{
get { return _packets; }
set { _packets = value; }
}
}
public class clsPacket
{
public long Offset { get; set; }
public String Hex { get; set; }
public String Ascii { get; set; }
}
}
关于WPF 列表框与 DataTemplate 绑定(bind)问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14117727/