我迷失了这个,我希望我的Viewmodel使用事件委托,以便我可以订阅它,打开一些对话框并等待对话框结果。稍后,ViewModel应该对对话框结果做任何想做的事情。
这是我实现它的方式(恢复的代码):
public class MyViewModel()
{
public delegate TributaryDocument SearchDocumentEventHandler();
public event SearchDocumentEventHandler SearchDocument;
//Command for the search button
public CommandRelay SearchDocumentCommand { get; set; }
//Document that i found in the dialog.
public TributaryDocument Document { get; set; }
public MyViewModel()
{
SearchDocumentCommand = new CommandRelay(DoSearchDocument);
}
//The command execution
public void DoSearchDocument()
{
//Event used here !
Document = SearchDocument?.Invoke();
}
}
public class MyUIControl : UserControl
{
public MainWindow MainWindow { get; }
public MyUIControl()
{
MainWindow = Application.Current.Windows[0] as MainWindow;
DataContextChanged += MyUIControl_DataContextChanged;
}
private void MyUIControl_DataContextChanged(object sender, DependencyPropertyChangedEventArgs e)
{
var modelView = (MyViewModel)DataContext;
modelView.SearchDocument += MyUIControl_SearchDocument;
}
private TributaryDocument MyUIControl_SearchDocument()
{
//Dont know what to do here... i am lost on this part.
return await MainWindow.ShowDialog(new MyDocumentSearcherDialog());
}
}
//The signature for MainWindow.ShowDialog
public async Task<object> ShowDialog(object dialog)
{
return await DialogHost.Show(dialog, "MainDialog");
}
MyDocumentSearcherDialog
只是一个对话框,我在其中搜索并返回一个TributaryDocument
对象。
据我了解,问题出在这部分(因为我无法编译):
private TributaryDocument MyUIControl_SearchDocument()
{
return await MainWindow.ShowDialog(new MyDocumentSearcherDialog());
}
我不能在不将方法签名更改为异步的情况下使用await。如果我将其更改为异步,则必须返回Task<TributaryDocument>
并更改事件委托:
public delegate Task<TributaryDocument> SearchDocumentEventHandler();
//On MyUIControl
private Task<TributaryDocument> MyUIControl_SearchDocument()
{
return await MainWindow.ShowDialog(new MyDocumentSearcherDialog());
}
//On MyViewModel
public async void DoSearchDocument()
{
//Event used here !
Document = await Task.Run(async () => await SearchDocument?.Invoke());
}
如果执行此操作,则会出现以下异常:
其他信息:调用线程必须是STA,因为许多 UI组件需要这样做。
答案 0 :(得分:2)
似乎您需要做的就是删除Task.Run
(在这种情况下,无需卸载到另一个线程)。如果您从内部执行 UI ,则Task.Run
肯定会给您 STA线程异常。
但是,简而言之,异步和等待模式将使用当前的 SynchronisationContext 创建一个延续,因此无需担心
public async void DoSearchDocument()
{
await SearchDocument?.Invoke();
}
注意 :由于这是一个事件,因此大约可以使用async void
< / sub>