Usar os comandos em um viewmodel

Concluído 200 XP

Você viu como obter dados dos seus viewmodels para sua interface do usuário e como usar vinculação bidirecional para obter dados de volta nos seus viewmodels.

Usar vinculações bidirecionais como essa é a maneira preferencial para reagir às alterações na interface do usuário sempre que os dados forem alterados. Muitas coisas que trataríamos como eventos podem ser manipuladas usando o MVVM e as vinculações bidirecionais. Outros exemplos são coisas como Switch.IsToggled e Slider.Value, que podem ser refletidas no nosso viewmodel como um valor booliano ou inteiro, sem ter que usar eventos.

Mas há algumas coisas como uma ativação Button ou MenuItem que não estão vinculadas diretamente à alteração de dados. Essas interações ainda requerem o tratamento semelhante a eventos. Como esses componentes de interface do usuário geralmente invocam algum tipo de lógica com os dados, queremos que essa lógica esteja no viewmodel. Mas não queremos tratá-los como eventos Clicked e Selected no code-behind, se possível. Queremos o máximo possível no viewmodel, dessa forma ele é testável.

Usar o padrão de comando

Muitos dos controles do .NET MAUI que têm esse tipo de interação dão suporte para associação a uma propriedade que expõe uma interface ICommand. Essa propriedade provavelmente tem o nome Command. O botão é um exemplo:

XAML
<Button Text="Give Bonus" Command="{Binding GiveBonusCommand}" />

O controle sabe quando invocar o comando. Por exemplo, um botão invoca o comando quando é pressionado. Neste exemplo, o comando está associado à propriedade GiveBonusCommand do viewmodel. O tipo de propriedade deve implementar a interface ICommand. O código seria semelhante a isto:

C#
public class EmployeeViewModel : INotifyPropertyChanged
{
    public ICommand GiveBonusCommand {get; private set;}
    ...
}

A interface ICommand tem um método Execute que é chamado quando o botão é clicado. Dessa forma, o ICommand.Execute diretamente substitui o código de manipulação de eventos Button.Click.

A interface completa ICommand possui dois métodos adicionais: CanExecute e CanExecuteChanged, que são usados para determinar se um controle deve aparecer habilitado ou desabilitado.

Um botão, por exemplo, poderá aparecer esmaecido se CanExecute retornar false.

Aqui está como a interface ICommand se parece em C#:

C#
public interface ICommand
{
    bool CanExecute(object parameter);
    void Execute(object parameter);
    event EventHandler CanExecuteChanged;
}

Use a classe Command

Esse padrão de comando permite manter uma separação clara entre o comportamento e a implementação da interface do usuário. No entanto, isso poderá complicar o seu código se for necessário criar uma classe separada para implementar cada manipulador de eventos.

Em vez de criar várias classes personalizadas que implementam a interface, é comum usar Command ou Command<T>. Essas classes implementam ICommand, mas expõem seu comportamento como propriedades no seu viewmodel que você pode definir. Isso permite que você implemente a propriedade GiveBonusCommand descrita anteriormente inteiramente dentro da nossa classe viewmodel:

C#
public class EmployeeViewModel : INotifyPropertyChanged
{
    public ICommand GiveBonusCommand {get; private set;}
    public EmployeeViewModel(Employee model)
    {
        GiveBonusCommand = new Command(GiveBonusExecute, GiveBonusCanExecute)
    }

    void GiveBonusExecute()
    {
        //logic for giving bonus
    }

    bool GiveBonusCanExecute()
    {
        //logic for deciding if "give bonus" button should be enabled.
    }
}

Nesse código, o comportamento Execute é sendo fornecido pelo método GiveBonusExecute. E CanExecute é fornecido por GiveBonusCanExecute. Os itens delegados a esses métodos são passados para o construtor Command. Neste exemplo, não há nenhuma implementação para CanExecuteChanged.

Simplificar com o Kit de Ferramentas do MVVM

A biblioteca MVVM Toolkit contém implementações de ICommand conhecidos como RelayCommand e AsyncRelayCommand. Ele também fornece geradores de origem para simplificar ainda mais esse código. No exemplo a seguir, GiveBonusCommand será gerado, configurando tanto o método para chamar para executar quanto para chamar para ver se ele pode ser executado. O atributo [RelayCommand] é usado no método GiveBonus e gerará o GiveBonusCommand. Além disso, definindo a propriedade CanExecute no atributo como o nome do método que desejamos conectar ao método CanExecute do ICommand, ele gerará o código para configurar isso para nós.

C#
public partial class EmployeeViewModel : ObservableObject
{
    public EmployeeViewModel(Employee model)
    {
    }

    [RelayCommand(CanExecute = nameof(GiveBonusCanExecute))]
    void GiveBonus()
    {
        //logic for giving bonus
    }

    bool GiveBonusCanExecute()
    {
        //logic for deciding if "give bonus" button should be enabled.
        return true;
    }
}

O Kit de Ferramentas MVVM também manipula métodos async, que são comuns na programação do .NET.

Comandos com parâmetros

A interface ICommand aceita um parâmetro object para os métodos CanExecute e Execute. O .NET MAUI implementa essa interface sem verificação de tipo através da classe Command. Os delegados que você anexa ao comando devem fazer sua própria verificação de tipo para garantir que o parâmetro correto seja passado. O .NET MAUI também fornece a implementação Command<T> na qual você define o tipo de parâmetro esperado. Quando você cria um comando que aceita um único tipo de parâmetro, use Command<T>.

Os controles do .NET MAUI que implementam o padrão de comando fornecem a propriedade CommandParameter. Ao definir essa propriedade, você pode passar um parâmetro para o comando quando ele é invocado com Execute, ou quando ele verifica o método CanExecute para obter o status.

Neste exemplo, o valor da cadeia de caracteres 25 é enviado para o comando:

XAML
<Button Text="Give Bonus" Command="{Binding GiveBonusCommand}" CommandParameter="25" />

O comando precisaria interpretar e converter esse parâmetro de cadeia de caracteres. Existem muitas maneiras de fornecer um parâmetro fortemente tipado.

  • Em vez de usar a sintaxe de atributo para definir CommandParameter, use elementos XAML.

    XAML
    <Button Text="Give Bonus" Command="{Binding GiveBonusCommand}">
        <Button.CommandParameter>
            <x:Int32>25</x:Int32>
        </Button.CommandParameter>
    </Button>
    
  • Vincule o CommandParameter a uma instância do tipo correto.

  • Se o CommandParameter estiver associado ao tipo incorreto, aplique um conversor para converter o valor para o tipo correto.

Verifique seu conhecimento

1.

Qual é o propósito da interface ICommand?

2.

Como as classes Command ou Command<T> podem simplificar o uso da interface ICommand?

3.

Qual é a função da propriedade CommandParameter nos controles do .NET MAUI?


Unidade seguinte: Exercício – Converter um manipulador de eventos em um comando

Anterior Avançar