31

I would like to pass a parameter defined in the XAML (View) of my application to the ViewModel class by using the RelayCommand. I followed Josh Smith's excellent article on MVVM and have implemented the following.

XAML Code

 <Button Command="{Binding Path=ACommandWithAParameter}" CommandParameter="Orange" HorizontalAlignment="Left" Style="{DynamicResource SimpleButton}" VerticalAlignment="Top" Content="Button"/> 

ViewModel Code

 public RelayCommand _aCommandWithAParameter; /// <summary> /// Returns a command with a parameter /// </summary> public RelayCommand ACommandWithAParameter { get { if (_aCommandWithAParameter == null) { _aCommandWithAParameter = new RelayCommand( param => this.CommandWithAParameter("Apple") ); } return _aCommandWithAParameter; } } public void CommandWithAParameter(String aParameter) { String theParameter = aParameter; } #endregion 

I set a breakpoint in the CommandWithAParameter method and observed that aParameter was set to "Apple", and not "Orange". This seems obvious as the method CommandWithAParameter is being called with the literal String "Apple".

However, looking up the execution stack, I can see that "Orange", the CommandParameter I set in the XAML is the parameter value for RelayCommand implemenation of the ICommand Execute interface method.

That is the value of parameter in the method below of the execution stack is "Orange",

 public void Execute(object parameter) { _execute(parameter); } 

What I am trying to figure out is how to create the RelayCommand ACommandWithAParameter property such that it can call the CommandWithAParameter method with the CommandParameter "Orange" defined in the XAML.

Is there a way to do this?

Why do I want to do this? Part of "On The Fly Localization" In my particular implementation I want to create a SetLanguage RelayCommand that can be bound to multiple buttons. I would like to pass the two character language identifier ("en", "es", "ja", etc) as the CommandParameter and have that be defined for each "set language" button defined in the XAML. I want to avoid having to create a SetLanguageToXXX command for each language supporting and hard coding the two character language identifier into each RelayCommand in the ViewModel.

1
  • 5
    None of the provided answers to this question helped me to create a RelayCommand object to communicate with a method requiring parameter input. If anyone is having a similar issue, this thread helped me: stackoverflow.com/questions/5298910/… Commented May 2, 2012 at 18:28

6 Answers 6

41

I don't understand why you have the extra complexity of specifying the lambda in the first place. Why not just do this:

if (_aCommandWithAParameter == null) { _aCommandWithAParameter = new RelayCommand<object>(CommandWithAParameter); } private void CommandWithAParameter(object state) { var str = state as string; } 
Sign up to request clarification or add additional context in comments.

8 Comments

readonly Action<object> _execute; readonly Predicate<object> _canExecute; public RelayCommand(Action<object> execute) : this(execute, null) { } public RelayCommand(Action<object> execute, Predicate<object> canExecute) { if (execute == null) throw new ArgumentNullException("execute"); _execute = execute; _canExecute = canExecute; } Action<object> is a delegate and requires a lambda expression when calling the constructor. Not sure why, but I get a compiler error otherwise.
Because it's Action<object>, not Action<string>. Updated my post to clarify.
Sorry about spacing in first comment. In any case the point is that the lambda expression is required because of the parameter argument in the RelayCommand constructor. Namely public RelayCommand(Action<object> execute) : this(execute, null). Since the constructor of RelayCommand takes a delegate, then when creating a new RelayCommand the lambda expression as shown in my post must be used. The parameter type in the method that is used on the right hand side of the lambda expression does alter the fact that a lambda expression must be used when creating a RelayCommand.
Don't you have to use a lambda expression if there is no parameter? It won't compile for me otherwise.
The RelayCommand constructor takes Action<object>, so the method you pass to it must match that delegate. The one in my above code sample does, because it takes an object parameter. If your method does not take an object parameter, the easiest way to transform is indeed to use a lambda: new RelayCommand(_ => MyMethod()) (an underscore is the conventional name for an unused lambda parameter)
|
15

You'll pass the param in the lambda to the command like so:

if (_aCommandWithAParameter == null) { _aCommandWithAParameter = new RelayCommand( param => this.CommandWithAParameter(param) ); } 

4 Comments

Paul, I get the Compiler Error "Invalid expression ')'" when I use the syntax you suggest above.
the ** characters are to show emphasis - they should not be in the code you actually cut and paste.
Paul thanks you are right. I should have realized the ** on either side were for emphasis. Your solution works. I marked Kent's solution as the answer because it is simpler (or should I say easier to understand without the lambda expressions).
Would you mind editing your answer to show = new RelayCommand(param => this.CommandWithAParameter(param)). I think that will be easier for others to understand and they won't make the same cut and paste mistake I made. I am guessing I don't have enough reputation points to edit other's answers yet otherwise I would do it myself.
8

Nothing posted here before worked for me.

Turns out, all answers are missing the <object> after RelayCommand!

This works for me:

public RelayCommand<object> OKCommand { get { if (_okCommand == null) _okCommand = new RelayCommand<object>(OkCommand_Execute); return _okCommand; } } private RelayCommand<object> _okCommand = null; private void OkCommand_Execute(object obj) { Result = true; } 

If you want to use aCanExecute method, use the following code:

_okCommand = new RelayCommand<object>(OkCommand_Execute, OkCommand_CanExecute); private bool OkCommand_CanExecute(object obj) { } 

Comments

5

Here is a simple solution for the commandparameter as I was looking for help on the subject. I could not find anything online that was simple enough. The following solution works well when you are using a relaycommand. I had a few hyperlinks for which I needed to get the url value that was clicked using the command parameter.

Step 1: In your relay command, create a simple property that will hold the parameter object value. You could call it parametervalue or any name that you prefer.

public object ParameterValue { get; set; } 

Step 2: In the Execute Method of the RelayCommand class, set the value or the property created above to the parameter from the Execute method.

readonly Action<object> m_execute; // Action to execute public void Execute(object parameter) { this.ParameterValue = parameter; m_execute(parameter); } 

Step 3: Now when you can bind the CommandParameter in xaml to any value you want to retrieve when the command is executed. example:

<TextBlock> <Hyperlink Command="{Binding Path=NavigateUrlCmd}" CommandParameter="{Binding ElementName=tbwebsite, Path=Text}"> <TextBlock Name="tbwebsite" Text="{Binding Path=website}"/> </Hyperlink> </TextBlock> 

If you have a command called chickenCommand, when executed you could access the parameter: chickenCommand.ParameterValue

I hope this helps somebody. Thank you for all your previous help.

Comments

2

I am just trying to sell my point, check this out whether this works...

http://mywpf-visu.blogspot.com/2009/12/relay-command-how-to-pass-parameter.html

Comments

1

I cannot substitute a reference to the method name for the lamda expression withing a compile error. Apparently, and by no means surprisingly, a non-static method name reference cannot be used in place of a lambda. I hardly see it as "added complexity". Consistently passing lamdas makes sense to me.

Comments

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.