Xamarin формы передают выбранный элемент в команду в качестве параметра команды
Я только начал с форм Xamarin, и теперь у меня есть список элементов, которые я отображаю в пользовательском шаблоне.
Поведение, которое я хочу, заключается в том, что событие запускается в контексте страницы (с использованием Corcav.Behaviors), но я хочу передать выбранный элемент в команду. Я не могу заставить последнюю часть работать. В приведенной ниже реализации событие срабатывает корректно, но переданный параметр - MyEventsListModel, но я хочу, чтобы элемент был нажат.
Заметьте, что я предпочтительно хочу решение в xaml/viewmodel, а не в коде позади. И альтернативное решение, при котором оба события срабатывают на событии, тоже подойдет.
Xaml:
<?xml version="1.0" encoding="utf-8" ?>
<ContentPage xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:behaviors="clr-namespace:Corcav.Behaviors;assembly=Corcav.Behaviors"
x:Class="TheEventsApp.Mobile.MyEventsList"
Title="My Events"
x:Name="MainPage">
<ListView ItemsSource="{Binding Events}">
<behaviors:Interaction.Behaviors>
<behaviors:BehaviorCollection>
<behaviors:EventToCommand EventName="ItemTapped" Command="{Binding NavigateToEventDetails}" CommandParameter="{Binding .}" />
</behaviors:BehaviorCollection>
</behaviors:Interaction.Behaviors>
<ListView.ItemTemplate>
<DataTemplate>
<ViewCell>
<StackLayout Orientation="Vertical">
<Label Text="{Binding Name}" />
</StackLayout>
</ViewCell>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</ContentPage>
ViewModel:
[ImplementPropertyChanged]
public class MyEventsListModel : FreshBasePageModel
{
private readonly EventsDatastore eventsStore;
public MyEventsListModel(EventsDatastore eventsStore)
{
this.eventsStore = eventsStore;
}
protected async override void ViewIsAppearing(object sender, EventArgs e)
{
this.Events = await eventsStore.GetMyEventsAsync();
}
public ObservableCollection<Event> Events { get; set; } = new ObservableCollection<Event>();
public Command NavigateToEventDetails
{
get
{
return new Command(async (clickedEvent) =>
{
await CoreMethods.PushPageModel<EventDetailsPageModel>(clickedEvent);
});
}
}
}
1 ответ
Решение было довольно простым. Окунувшись в исходный код Corcav, было легко разобраться. Следующий код в EventToCommand.cs
private void OnFired(EventArgs e)
{
object param = this.PassEventArgument ? e : this.CommandParameter;
if (!string.IsNullOrEmpty(this.CommandName))
{
if (this.Command == null) this.CreateRelativeBinding();
}
if (this.Command == null) throw new InvalidOperationException("No command available, Is Command properly set up?");
if (e == null && this.CommandParameter == null) throw new InvalidOperationException("You need a CommandParameter");
if (this.Command != null && this.Command.CanExecute(param))
{
this.Command.Execute(param);
}
}
Обращается с этим. Просто установив "PassEventArgument" в true, я решил эту проблему.
<behaviors:EventToCommand EventName="ItemTapped" Command="{Binding NavigateToEventDetails}" PassEventArgument="True" />