WPF 列表框与 DataTemplate 绑定(bind)问题

标签 wpf binding listbox datatemplate

我正在尝试使用自定义数据模板创建一个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/

相关文章:

xml - 如何指定 XmlAccessorType 用于 JAXB xjc 生成的类

c# - 使用 DataTemplate 在 WPF 中通过 LINQ to SQL 绑定(bind) ListBox 的 ListItem 属性的两种方式

.net - 按钮的默认背景颜色

wpf - 单击按钮时对 WPF 列表框进行排序?

javascript - Angular 绑定(bind)到带空格的方括号表示法属性

gwt - 在 GWT ListBox 中处理大量数据

c# - 列表框 Selected.value 抛出 null 异常

c# - 将 ObervableCollection 绑定(bind)到 ListBox

wpf - 使用 SelectionMode = Multiple 对 ListBox 进行数据绑定(bind)

java - BeansBinding、JGoodies Binding、JFace 或 Glazed Lists?