У меня есть простой ListView
, связанный со списком объектов. В ListView.ItemTemplate
я хотел бы отформатировать каждый объект и связать его с TextBlock
, используя привязку функции x: Bind следующим образом: <TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(???)}" />
. Однако я не уверен, как передать ссылку на текущий элемент в функцию.
Это прекрасно работает, если мне нужно передать свойство элемента в функцию или даже несколько свойств, например как <TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(Name, Model)}" />
, но я не могу понять, как передать весь объект.
Я пробовал все это безрезультатно:
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(this)}" />
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(self)}" />
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget({x:Bind})}" />
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget({Binding})}" />
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(Item)}" />
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(DataContext)}" />
<TextBlock Name="Root" Text="{x:Bind local:MainViewModel.FormatWidget(Root.DataContext)}" />
В качестве обходного пути я может создать и привязать свойство к элементу данных, которое возвращает себя, например, так:
public Widget MeMyselfAndI { get => this; }
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(MeMyselfAndI)}" />
Однако это хак, и в моем реальном проекте я не могу изменить источник данных Класс элемента для добавления свойства.
Как передать ссылку на текущий элемент в функцию?
Вот ответ:
MainPage.xaml
<Page
x:Class="ListItemFunctionBinding.MainPage"
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:local="using:ListItemFunctionBinding"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Background="{ThemeResource ApplicationPageBackgroundThemeBrush}"
mc:Ignorable="d">
<Grid>
<ListView ItemsSource="{x:Bind ViewModel.Widgets}" SelectedItem="{x:Bind ViewModel.SelectedItem}">
<ListView.ItemTemplate>
<DataTemplate x:DataType="local:Widget">
<TextBlock Text="{x:Bind local:MainViewModel.FormatWidget(MeMyselfAndI)}" />
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</Grid>
</Page>
MainPage.xaml.cs
using Windows.UI.Xaml.Controls;
namespace ListItemFunctionBinding
{
public sealed partial class MainPage : Page
{
public MainViewModel ViewModel { get; } = new MainViewModel();
public MainPage()
{
InitializeComponent();
}
}
}
MainViewModel.cs
using System;
using System.Collections.ObjectModel;
using System.ComponentModel;
namespace ListItemFunctionBinding
{
public class MainViewModel : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
private Widget _selectedItem;
public Widget SelectedItem
{
get => _selectedItem;
set
{
if (_selectedItem != value)
{
_selectedItem = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(SelectedItem)));
}
}
}
public ObservableCollection<Widget> Widgets { get; } = new ObservableCollection<Widget>()
{
new Widget
{
Id = Guid.NewGuid(),
Name = "Regular Widget",
Model = "WX2020-01",
Description = "Your typical everyday widget."
},
new Widget
{
Id = Guid.NewGuid(),
Name = "Super Widget",
Model = "WX2020-02",
Description = "An extra special upgraded widget."
},
new Widget
{
Id = Guid.NewGuid(),
Name = "Broken Widget",
Model = "WX2020-03",
Description = "A widget that has been used and abused."
},
new Widget
{
Id = Guid.NewGuid(),
Name = "Fake Widget",
Model = "WX2020-04",
Description = "It's not really a widget at all!"
},
new Widget
{
Id = Guid.NewGuid(),
Name = "Surprise Widget",
Model = "WX2020-05",
Description = "What kind of widget will it be?"
},
new Widget
{
Id = Guid.NewGuid(),
Name = "Invisible Widget",
Model = "WX2020-06",
Description = "Our most inexpensive widget."
},
new Widget
{
Id = Guid.NewGuid(),
Name = "Backwards Widget",
Model = "WX2020-07",
Description = "Really more of a tegdiw, come to think of it."
}
};
public static string FormatWidget(Widget widget)
{
if (widget == null)
return "No widget selected";
else
return $"{widget.Name} [{widget.Model}] {widget.Description}";
}
}
}
Widget.cs
using System;
using System.ComponentModel;
namespace ListItemFunctionBinding
{
public class Widget : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
private Guid _id;
private string _name;
private string _model;
private string _description;
public Widget MeMyselfAndI { get => this; }
public Guid Id
{
get => _id;
set
{
if (_id != value)
{
_id = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Id)));
}
}
}
public string Name
{
get => _name;
set
{
if (_name != value)
{
_name = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Name)));
}
}
}
public string Model
{
get => _model;
set
{
if (_model != value)
{
_model = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Model)));
}
}
}
public string Description
{
get => _description;
set
{
if (_description != value)
{
_description = value;
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Description)));
}
}
}
}
}