WPFToolkit AutoCompleteBox가 ListView 내에서 올바르게 바인딩되지 않음
WPFToolkit의 AutoCompleteBox 컨트롤을 시작하기 위해 약간 고생 한 후 ListView 내에서 AutoCompleteBox 를 사용하려고 할 때 또 다른 문제에 직면 했습니다. 거의 완벽하게 바인딩되지만 이유 때문에 처음에는 표시되지 않고 ValueMemberPath
대신 시도합니다. 객체를 Namespace.object
적절한 ValueMemberPath
값 대신 제공하는 문자열로 변환하려면 AutoCompleteBox에서 다른 항목을 선택하면 완벽하게 작동하며 다른 Namespace.object
.


다음은 내 코드입니다. 복사하여 붙여 넣으면 동일한 결과를 얻을 수 있습니다 ( DotNetProjects.WpfToolkit.Input
NuGet 패키지 관리자 에 추가하는 것을 잊지 마세요 ) .
- Namespace.MainWindow.xaml
<Window x:Class="Namespace.MainWindow"
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"
xmlns:local="clr-namespace:Namespace"
mc:Ignorable="d"
xmlns:toolkit="clr-namespace:System.Windows.Controls;assembly=DotNetProjects.Input.Toolkit"
Title="AutoCompleteBox in ListView" Height="300" Width="350" WindowStartupLocation="CenterScreen">
<!-- Required Template to show the names of the Items in the ItemsList -->
<Window.Resources>
<DataTemplate x:Key="AutoCompleteBoxItemTemplate">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" Background="Transparent">
<Label Content="{Binding Name}"/>
</StackPanel>
</DataTemplate>
</Window.Resources>
<StackPanel Margin="5">
<StackPanel Orientation="Horizontal" Margin="0 5 0 0">
<StackPanel Width="{Binding ElementName=FirstColumnWidth, Path=ActualWidth}">
<TextBlock Text="ACB binded to Cart.Item"/>
<!-- ACB that binds correctly -->
<toolkit:AutoCompleteBox
ItemsSource="{Binding Path=ItemsList}"
ValueMemberPath="Name"
SelectedItem="{Binding Path=Cart.Item, Mode=TwoWay}"
ItemTemplate="{StaticResource AutoCompleteBoxItemTemplate}"/>
</StackPanel>
<StackPanel Margin="15 0 0 0">
<TextBlock Text="Value of Cart.Item.Name"/>
<TextBlock Text="{Binding Path=Cart.Item.Name}"/>
</StackPanel>
</StackPanel>
<TextBlock Margin="0 30 0 0" HorizontalAlignment="Center" Text="ListView with CartsList as ItemsListSource"/>
<ListView ItemsSource="{Binding CartsList}">
<ListView.View>
<GridView>
<GridViewColumn x:Name="FirstColumnWidth">
<GridViewColumn.Header>
<TextBlock Text="ACB binded to each Cart.Item"/>
</GridViewColumn.Header>
<GridViewColumn.CellTemplate>
<DataTemplate>
<!-- ACB that doesn't bind correctly -->
<toolkit:AutoCompleteBox
ItemsSource="{
Binding RelativeSource={RelativeSource AncestorType=Window},
Path=DataContext.ItemsList}"
ValueMemberPath="Name"
SelectedItem="{Binding Path=Item, Mode=TwoWay}"
ItemTemplate="{StaticResource AutoCompleteBoxItemTemplate}"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
<GridViewColumn >
<GridViewColumn.Header>
<TextBlock Text="Value of each Cart.Item.Name"/>
</GridViewColumn.Header>
<GridViewColumn.CellTemplate>
<DataTemplate>
<TextBlock Text="{Binding Path=Item.Name}"/>
</DataTemplate>
</GridViewColumn.CellTemplate>
</GridViewColumn>
</GridView>
</ListView.View>
</ListView>
</StackPanel>
</Window>
- 코드 숨김 (MainWindow.xaml.cs)
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Runtime.CompilerServices;
using System.Windows;
namespace Namespace
{
public partial class MainWindow : Window, INotifyPropertyChanged
{
// INPC Implementation
public event PropertyChangedEventHandler PropertyChanged;
private void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}
// The list that contains Items that will be chosen in a Cart
private ObservableCollection<Item> _ItemsList;
public ObservableCollection<Item> ItemsList
{
get => _ItemsList;
set
{
_ItemsList = value;
OnPropertyChanged();
}
}
// The list that contains Carts that will be shown in the ListView
private ObservableCollection<Cart> _CartsList;
public ObservableCollection<Cart> CartsList
{
get => _CartsList;
set
{
_CartsList = value;
OnPropertyChanged();
}
}
// A signle Cart
private Cart _Cart;
public Cart Cart
{
get => _Cart;
set
{
_Cart = value;
OnPropertyChanged();
}
}
public MainWindow()
{
DataContext = this;
InitializeComponent();
// Populating ItemsList
ItemsList = new ObservableCollection<Item>()
{
new Item("T-shirt"), new Item("Jeans"), new Item("Boots"),
};
// Populating CartsList
CartsList = new ObservableCollection<Cart>()
{
new Cart(ItemsList[0]),
new Cart(ItemsList[2]),
new Cart(ItemsList[1]),
new Cart(ItemsList[0]),
new Cart(ItemsList[1]),
};
// Setting an Item to Cart
Cart = new Cart(ItemsList[2]);
}
}
// Cart Object
public class Cart
{
public Item Item { get; set; }
public Cart(Item item) => Item = item;
}
// Item Object
public class Item
{
// Important to be private set so it cannot be changed
public string Name { get; private set; }
public Item(string name) => Name = name;
}
}
답변
어떤 이유에 따라 텍스트가 업데이트하는 선택 변경을 위해 ValueMemberPath
네 스팅 할 때 트리거되지 않습니다 AutoCompleteBox
의를 ListView
. SelectionChanged
이벤트도 발생하지 않습니다. 정확히 왜 그리고 이것이 버그인지 아닌지 알 수 없었습니다. 그러나 패키지를 사용하여 해결 방법 을 보여 드릴 수 있습니다 Microsoft.Xaml.Behaviors.Wpf.
을 재설정하고에서 SelectedItem
다시 할당 하는 트리거 작업을 만들 수 있습니다 AutoCompleteBox
.
public class ForceUpdateSelectedItemAction : TriggerAction<AutoCompleteBox>
{
protected override void Invoke(object parameter)
{
var selectedItem = AssociatedObject.SelectedItem;
AssociatedObject.SetCurrentValue(AutoCompleteBox.SelectedItemProperty, null);
AssociatedObject.SetCurrentValue(AutoCompleteBox.SelectedItemProperty, selectedItem);
}
}
이 트리거 작업은의 Loaded
이벤트에 사용할 수 있습니다 AutoCompleteBox
.
<toolkit:AutoCompleteBox ...>
<b:Interaction.Triggers>
<b:EventTrigger EventName="Loaded">
<local:ForceUpdateSelectedItemAction/>
</b:EventTrigger>
</b:Interaction.Triggers>
</toolkit:AutoCompleteBox>
이렇게하면 bound 속성을 변경하지 않고 선택 항목과 텍스트가 업데이트됩니다.