Как передать текущий элемент списка в функцию x: Bind? - PullRequest
0 голосов
/ 20 марта 2020

У меня есть простой 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)));
                }
            }
        }
    }
}

Ответы [ 2 ]

0 голосов
/ 24 марта 2020

Я нашел решение здесь :

Приведение без пути

Собственный анализатор связывания не предоставляет ключевое слово для представления this в качестве параметра функции, но он поддерживает преобразование без пути (например, {x:Bind (x:String)}), которое можно использовать в качестве параметра функции. Следовательно, {x:Bind MethodName((namespace:TypeOfThis))} является допустимым способом выполнения того, что концептуально эквивалентно {x:Bind MethodName(this)}.

Так что в моем исходном примере я могу использовать <TextBlock Text="{x:Bind local:MainViewModel.FormatWidget((local:Widget))}" /> для привязки текущего элемента списка.

Это отлично работает! Ура!

0 голосов
/ 20 марта 2020

В этом случае вы можете использовать метод Converter для передачи текущего элемента (например, Widget), а затем вызвать метод MainViewModel.FormatWidget в конвертере.

.xaml:

<Page.Resources>
    <local:MyConverter x:Key="MyConverter"/>
</Page.Resources>

<Grid>
    <ListView ItemsSource="{x:Bind ViewModel.Widgets}" SelectedItem="{x:Bind ViewModel.SelectedItem}">
        <ListView.ItemTemplate>
            <DataTemplate x:DataType="local:Widget">
                <TextBlock Text="{x:Bind Converter={StaticResource MyConverter}}" />
            </DataTemplate>
        </ListView.ItemTemplate>
    </ListView>
</Grid>

.cs:

public class MyConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, string language)
    {
        var widget = value as Widget;
        return MainViewModel.FormatWidget(widget);
    }

    public object ConvertBack(object value, Type targetType, object parameter, string language)
    {
        throw new NotImplementedException();
    }
}
...