Подтвердить что ты не робот

WPF Событие привязки к ViewModel (для классов без команд)

Я работаю над второй версией приложения, и, как часть переписывания, мне нужно перейти к архитектуре MVVM. Я получаю давление, чтобы поместить абсолютно каждый бит кода в класс модели представления - наличие С# в коде за файлом недовольно. (Я знаю, я знаю... Я понимаю, что код позади - это не плохо, но на этот раз это не мой звонок).

Для объектов, реализующих интерфейс командной строки, это легко. Я смог найти массу информации о том, как связать команду этих объектов с ICommand в модели представления. Проблема заключается в объектах, которые не имеют этого интерфейса, например

<ListBox
   x:Name="myListBox"
   MouseDoubleClick="myCallbackFunction">

<!-- ... -->

</ListBox>

Я хочу знать, как связать событие MouseDoubleClick для Listbox с myCallbackFunction, которое реализовано в модели представления. Возможно ли это?

Спасибо!

4b9b3361

Ответ 1

Это невозможно. Это можно сделать с помощью Attached Property или Behavior, хотя было бы немного сложно найти и вызвать соответствующий метод (это можно сделать с помощью Reflection довольно легко).

Это, как правило, обрабатывается через ICommand. Например, MVVM Light имеет отличное поведение EventToCommand для отображения любого события в ICommand on ViewModel. Преимущество использования ICommand заключается в том, что вы все равно можете использовать DataBinding, так как ICommand отображается как свойство.

Ответ 2

Чтобы ответить на ваш вопрос, обратитесь к Почему избежать кода в шаблоне MVVM WPF? Он предлагает две возможные вещи, которые вы хотите.

Однако почему вы хотите привязать MouseDoubleClick к ListBox к вашей ICommand в режиме просмотра?

Альтернативный способ заключается в том, что вы пишете метод в коде для регистрации MouseDoubleClick. Это не плохо из-за следующих фактов.

  • Сознательная привязка данных - это взаимодействие между представлением и моделью просмотра. Например, когда пользователь вводит некоторый текст в TextBox, обновляется также модель просмотра. Напротив, если viewmodel получает данные из базы данных, это будет показано на виде. Однако не в этом случае ICommand в вашей модели viewmodel привязывается к представлению.

  • Конечно, CanExcute ICommand был бы важен для вашей модели просмотра, но во многих случаях он не связан с моделью просмотра или не имеет отношения. В этом случае разница между привязкой ICommand и написанием кода - это то, где событие MouseDoubleClick привязано к ICommand или зарегистрировано в обработчике событий.

Ответ 3

WPF поддерживает расширения разметки в событиях с .NET 4.5. Используя эту возможность, я внедрил невероятно универсальный способ связывания расширений и написал об этом здесь:

http://www.singulink.com/CodeIndex/post/building-the-ultimate-wpf-event-method-binding-extension

Может использоваться для привязки к методу с использованием полного синтаксиса пути свойств, поддерживает привязки и другие расширения разметки в качестве аргументов и автоматически маршрутизирует метод, соответствующий сигнатуре предоставленных аргументов. Вот несколько примеров использования:

<!--  Basic usage  -->
<Button Click="{data:MethodBinding OpenFromFile}" Content="Open" />

<!--  Pass in a binding as a method argument  -->
<Button Click="{data:MethodBinding Save, {Binding CurrentItem}}" Content="Save" />

<!--  Another example of a binding, but this time to a property on another element  -->
<ComboBox x:Name="ExistingItems" ItemsSource="{Binding ExistingItems}" />
<Button Click="{data:MethodBinding Edit, {Binding SelectedItem, ElementName=ExistingItems}}" />

<!--  Pass in a hard-coded method argument, XAML string automatically converted to the proper type  -->
<ToggleButton Checked="{data:MethodBinding SetWebServiceState, True}"
                Content="Web Service"
                Unchecked="{data:MethodBinding SetWebServiceState, False}" />

<!--  Pass in sender, and match method signature automatically -->
<Canvas PreviewMouseDown="{data:MethodBinding SetCurrentElement, {data:EventSender}, ThrowOnMethodMissing=False}">
    <controls:DesignerElementTypeA />
    <controls:DesignerElementTypeB />
    <controls:DesignerElementTypeC />
</Canvas>

    <!--  Pass in EventArgs  -->
<Canvas MouseDown="{data:MethodBinding StartDrawing, {data:EventArgs}}"
        MouseMove="{data:MethodBinding AddDrawingPoint, {data:EventArgs}}"
        MouseUp="{data:MethodBinding EndDrawing, {data:EventArgs}}" />

<!-- Support binding to methods further in a property path -->
<Button Content="SaveDocument" Click="{data:MethodBinding CurrentDocument.DocumentService.Save, {Binding CurrentDocument}}" />

Просмотр сигнатур метода модели:

public void OpenFromFile();
public void Save(DocumentModel model);
public void Edit(DocumentModel model);

public void SetWebServiceState(bool state);

public void SetCurrentElement(DesignerElementTypeA element);
public void SetCurrentElement(DesignerElementTypeB element);
public void SetCurrentElement(DesignerElementTypeC element);

public void StartDrawing(MouseEventArgs e);
public void AddDrawingPoint(MouseEventArgs e);
public void EndDrawing(MouseEventArgs e);

public class Document
{
    // Fetches the document service for handling this document
    public DocumentService DocumentService { get; }
}

public class DocumentService
{
    public void Save(Document document);
}

Ответ 4

Один из способов может состоять в том, чтобы обработать событие в коде позади и вызвать соответствующий метод модели представления из кода, стоящего за

Вы также можете найти готовую командную библиотеку, например this, в которой используется ACB