use*_*540 2 wpf binding mvvm viewmodel
我是MVVM的新手,现在就完成我的第一个POC。但是,我一直在努力解决一个问题,持续了2天。向你们解释的想法可能会帮助您如此迅速地解决问题。现在让我简要介绍一下我的问题。我有WPF MVVM应用程序,其主视图绑定到MainViewModel。我在这里有Textblock,可以在加载屏幕的同时绑定来自视图模型的某些内容,这真棒。我也有绑定到ChildViewModel的ChildUserControl。现在,我需要在用户控件级别执行的某些操作上,将不同的内容从用户控件绑定到主窗口中的Textblock。怎么可能?
这是我有MainWindow.Xaml的示例代码
<Window.Resources>
<viewModel:MainViewModel x:Key="mainWindowViewModel"/></Window.Resources>
<TextBlock Name="txtStatus" Text="{Binding StatusMessage, Mode=OneWay }"/>
Run Code Online (Sandbox Code Playgroud)
ChildUserControl.xaml
<UserControl.Resources>
<viewModel:ChildModelView x:Key="ChildModelView"/> </UserControl.Resources>
public class ChildModelView : BaseViewModel
{
// Some child level logic..
// then need to update the txtStatus text block from parent
}
Run Code Online (Sandbox Code Playgroud)
非常感谢您的帮助。
一种简单的方法是使用IoC。创建子视图模型时,请将主视图模型的引用传递给子视图模型,并将其作为私有只读变量保存。现在,您可以访问所有主要的VM公众。
一种替代解决方案可能是使用Mediator模式。
敲了一个简单的应用程序来演示IoC解决方案。
App.xaml.cs
public partial class App : Application
{
protected override void OnStartup(StartupEventArgs e)
{
base.OnStartup(e);
var window = new MainWindow() {DataContext = new MainWindowViewModel() };
window.Show();
}
}
Run Code Online (Sandbox Code Playgroud)
MainWindowViewModel.cs
public class MainWindowViewModel : ViewModelBase
{
private string _statusMessage;
public string StatusMessage
{
get { return _statusMessage; }
set { _statusMessage = value; this.OnPropertyChanged("StatusMessage"); }
}
public ICommand OpenChildCommand { get; private set; }
public MainWindowViewModel()
{
this.StatusMessage = "No status";
this.OpenChildCommand = new DelegateCommand((o) => this.OpenChild());
}
private void OpenChild()
{
var window = new ChildWindow {DataContext = new ChildWindowViewModel(this)};
window.Show();
}
}
Run Code Online (Sandbox Code Playgroud)
ChildWindowViewModel.cs
public class ChildWindowViewModel : ViewModelBase
{
private readonly MainWindowViewModel _mainvm;
public ChildWindowViewModel(MainWindowViewModel mainvm)
{
_mainvm = mainvm;
this.UpdateStatusCommand = new DelegateCommand((o) => this.UpdateStatus());
}
public ICommand UpdateStatusCommand { get; private set; }
private void UpdateStatus()
{
this._mainvm.StatusMessage = "New Status";
}
}
Run Code Online (Sandbox Code Playgroud)
ViewModelBase.cs
public abstract class ViewModelBase : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
protected void OnPropertyChanged(string propertyName)
{
this.OnPropertyChanged(new PropertyChangedEventArgs(propertyName));
}
protected virtual void OnPropertyChanged(PropertyChangedEventArgs e)
{
var handler = this.PropertyChanged;
if (handler != null)
{
handler(this, e);
}
}
}
Run Code Online (Sandbox Code Playgroud)
DelegateCommand.cs
public class DelegateCommand : ICommand
{
/// <summary>
/// Action to be performed when this command is executed
/// </summary>
private Action<object> executionAction;
/// <summary>
/// Predicate to determine if the command is valid for execution
/// </summary>
private Predicate<object> canExecutePredicate;
/// <summary>
/// Initializes a new instance of the DelegateCommand class.
/// The command will always be valid for execution.
/// </summary>
/// <param name="execute">The delegate to call on execution</param>
public DelegateCommand(Action<object> execute)
: this(execute, null)
{
}
/// <summary>
/// Initializes a new instance of the DelegateCommand class.
/// </summary>
/// <param name="execute">The delegate to call on execution</param>
/// <param name="canExecute">The predicate to determine if command is valid for execution</param>
public DelegateCommand(Action<object> execute, Predicate<object> canExecute)
{
if (execute == null)
{
throw new ArgumentNullException("execute");
}
this.executionAction = execute;
this.canExecutePredicate = canExecute;
}
/// <summary>
/// Raised when CanExecute is changed
/// </summary>
public event EventHandler CanExecuteChanged
{
add { CommandManager.RequerySuggested += value; }
remove { CommandManager.RequerySuggested -= value; }
}
/// <summary>
/// Executes the delegate backing this DelegateCommand
/// </summary>
/// <param name="parameter">parameter to pass to predicate</param>
/// <returns>True if command is valid for execution</returns>
public bool CanExecute(object parameter)
{
return this.canExecutePredicate == null ? true : this.canExecutePredicate(parameter);
}
/// <summary>
/// Executes the delegate backing this DelegateCommand
/// </summary>
/// <param name="parameter">parameter to pass to delegate</param>
/// <exception cref="InvalidOperationException">Thrown if CanExecute returns false</exception>
public void Execute(object parameter)
{
if (!this.CanExecute(parameter))
{
throw new InvalidOperationException("The command is not valid for execution, check the CanExecute method before attempting to execute.");
}
this.executionAction(parameter);
}
}
Run Code Online (Sandbox Code Playgroud)
MainWindow.xaml
<Window x:Class="WpfApplication2.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="MainWindow" Height="350" Width="525">
<StackPanel>
<TextBlock Text="{Binding StatusMessage, Mode=OneWay}" />
<Button HorizontalAlignment="Center" VerticalAlignment="Center" Content="Open Child Window"
Command="{Binding Path=OpenChildCommand}"/>
</StackPanel>
Run Code Online (Sandbox Code Playgroud)
ChildWindow.xaml
<Window x:Class="WpfApplication2.ChildWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="ChildWindow" Height="300" Width="300">
<Grid>
<Button HorizontalAlignment="Center" VerticalAlignment="Center" Content="UpdateStatus"
Command="{Binding Path=UpdateStatusCommand}" />
</Grid>
Run Code Online (Sandbox Code Playgroud)
单击更新状态之前的图像

单击updatestatus后的图像

| 归档时间: |
|
| 查看次数: |
5892 次 |
| 最近记录: |