我正在尝试在我的WPF应用程序中实现MVVM架构,我希望能够在执行命令后修改模型。请注意,我没有使用任何类型的MVVM框架。
我有一个基本的Command类,如下所示:
namespace MyApplication.Commands {
public abstract class CommandBase : ICommand {
protected static BackgroundWorker Worker = new BackgroundWorker();
protected static string _result;
public string Result {
get {
return _result;
}
set {
_result = value;
}
}
public abstract void DoWork(DoWorkEventArgs args);
public virtual void Execute(object parameter) {
Worker = new BackgroundWorker();
Worker.DoWork += (o, args) => DoWork(args);
Worker.RunWorkerCompleted += (sender, args) => {
RaiseCanExecuteChanged(EventArgs.Empty);
};
Worker.ProgressChanged += (sender, args) => {
RaiseCanExecuteChanged(EventArgs.Empty);
};
Worker.WorkerReportsProgress = true;
Worker.RunWorkerAsync(parameter);
}
public virtual bool CanExecute(object parameter) {
return !Worker.IsBusy;
}
public event EventHandler CanExecuteChanged;
protected virtual void RaiseCanExecuteChanged(EventArgs e) {
CanExecuteChanged(this, e);
}
}
}
我已经获得了实现此功能的实际命令,如下所示:
namespace MyApplication.Commands {
internal class DoSomethingCommand : CommandBase {
public override void DoWork(DoWorkEventArgs args) {
Worker.ReportProgress(0);
var success = false;
try {
var parameter = args.Argument as int?;
success = DoSomething(parameter);
} finally {
args.Result = success;
}
}
private static bool DoSomething(int parameter) {
// Do something expensive here...
System.Func func = (arg) => {
Thread.Sleep(arg);
return true;
};
// etc etc...
var success = func.Invoke(parameter);
return success;
}
}
}
我使用它作为执行昂贵操作的方法,同时仍然维护MVVM模型。我必须在进度和完成事件中执行RaiseCanExecuteChanged
,否则UI将不会更新相应的按钮状态。
但是,现在我想要修改当前正在访问的Model
的属性(在ListView
控件中选择的AKA),该属性作为ViewModel
的属性公开。如何在保持MVVM架构的同时做到这一点?
还有更好的方法将ViewModel
的当前状态传递给我的命令吗?目前,我使用MultiBinding
加IMultiValueConverter
只允许传递object[]
。
答案 0 :(得分:1)
为什么不在命令的构造函数中传递对viewModel的引用,并在字段中将其保存在本地。这样,您应该能够使用其公开的方法和属性来操纵整个viewModel。
关于你的第二个问题:
还有更好的方法将ViewModel的当前状态传递给我的命令吗?
您的命令应该在viewModel中实现,因此您不必提供viewModel的状态。如果情况并非如此,那么为什么不通过在控件的DataContext
属性中传递CommandParameter
来传递整个viewModel。