WPF: как связать объект с ComboBox
Попытка узнать, как привязывать объекты к различным типам элементов управления. В этом случае я хочу, чтобы образцы данных в моем объекте появлялись в ComboBox. Код работает, но то, что появляется вместо значений (David, Helen, Joe), это текст "TheProtect.UserControls.Client" )
XAML: (ucDataBindingObject.xaml)
<UserControl x:Class="TheProject.UserControls.ucDataBindingObject"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Width="Auto"
Height="Auto"
mc:Ignorable="d">
<Grid Width="130"
Height="240"
Margin="0">
<ComboBox Width="310"
HorizontalAlignment="Left"
VerticalAlignment="Top"
ItemsSource="{Binding Path=Clients}" />
</Grid>
</UserControl>
С#: ucDataBindingObject.xaml.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Windows.Controls;
namespace TheProject.UserControls
{
public partial class ucDataBindingObject : UserControl
{
public List<Client> Clients { get; set; }
public ucDataBindingObject()
{
Clients = new List<Client>();
Clients.Add(new Client(1, "David")); // sample data
Clients.Add(new Client(2, "Helen"));
Clients.Add(new Client(3, "Joe"));
InitializeComponent();
this.DataContext = this;
}
}
С# Client.cs
using System;
using System.Linq;
namespace TheProject.UserControls
{
public class Client
{
public int ID { get; set; }
public string Name { get; set; }
public Client(int id, string name)
{
this.ID = id;
this.Name = name;
}
}
}
Ответы
Ответ 1
Существует несколько способов рассказать структуре, что отображать
1) Используйте DisplayMemberPath
в ComboBox (это отобразит именованное свойство):
<ComboBox ItemsSource="{Binding Path=Clients}"
DisplayMemberPath="Name"
/>
2) Установите ItemTemplate
в ComboBox. Это похоже на # 1, за исключением того, что вы можете определить шаблон для отображения, а не просто свойство:
<ComboBox ItemsSource="{Binding Path=Clients}">
<ComboBox.ItemTemplate>
<DataTemplate>
<Border BorderBrush="Green" BorderThickness="1" Padding="5">
<TextBlock Text="{Binding Path=Name,StringFormat='Name: {0}'}" />
</Border>
</DataTemplate>
</ComboBox.ItemTemplate>
</ComboBox>
3) Добавьте переопределение ToString()
в исходный класс. Полезно, если вы всегда хотите отображать одну и ту же строку для данного класса. (Обратите внимание, что по умолчанию ToString()
- это просто имя типа класса, поэтому вы видите "TheProtect.UserControls.Client".)
public class Client
{
// ...
public override string ToString()
{
return string.Format("{0} ({1})", Name, ID);
}
}
4) Добавьте DataTemplate
к ресурсам XAML. Это полезно для сопоставления заданного типа класса с более сложным или стилизованным шаблоном.
<UserControl xmlns:local="clr-namespace:TheProject.UserControls">
<UserControl.Resources>
<DataTemplate DataType="local:Client">
<TextBlock Text="{Binding Name}" />
</DataTemplate>
</UserControl.Resources>
// ...
</UserControl>
Ответ 2
В DisplayMemberPath
укажите имя свойства, которое вы хотите показать в comboBox. В SelectedValuePath
укажите имя свойства, которое вы хотите выбрать. Когда вы выполните ComboBox.SelectedValue
, вы получите значение этого свойства.