好的,我真的想知道MVVM专业开发人员如何处理WPF中的openfile对话框。
我真的不想在我的ViewModel中执行此操作(其中'Browse'通过DelegateCommand引用)
void Browse(object param)
{
//Add code here
OpenFileDialog d = new OpenFileDialog();
if (d.ShowDialog() == true)
{
//Do stuff
}
}
因为我认为这违背了MVVM方法论。
我该怎么办?
答案 0 :(得分:35)
这里最好的办法是使用服务。
服务只是您从中央服务存储库(通常是IOC容器)访问的类。然后,该服务实现了您所需的功能,如OpenFileDialog。
因此,假设Unity容器中有IFileDialogService
,您可以执行...
void Browse(object param)
{
var fileDialogService = container.Resolve<IFileDialogService>();
string path = fileDialogService.OpenFileDialog();
if (!string.IsNullOrEmpty(path))
{
//Do stuff
}
}
答案 1 :(得分:9)
我本来想对其中一个答案发表评论,但唉,我的声誉还不够高。
调用OpenFileDialog()会违反MVVM模式,因为它意味着视图模型中的视图(对话框)。视图模型可以调用类似GetFileName()的东西(也就是说,如果简单绑定不够),但它不应该关心如何获取文件名。
答案 2 :(得分:7)
我使用的服务例如可以传递给viewModel的构造函数或通过依赖注入解析。 e.g。
public interface IOpenFileService
{
string FileName { get; }
bool OpenFileDialog()
}
和实现它的类,使用OpenFileDialog。在viewModel中,我只使用接口,因此可以根据需要进行模拟/替换。
答案 3 :(得分:6)
ViewModel不应该打开对话框,甚至不知道它们的存在。如果VM位于单独的DLL中,则项目不应引用PresentationFramework。
我喜欢在视图中使用辅助类来进行常见对话。
辅助类公开窗口在XAML中绑定的命令(不是事件)。这意味着在视图中使用RelayCommand。辅助类是DepencyObject,因此它可以绑定到视图模型。
class DialogHelper : DependencyObject
{
public ViewModel ViewModel
{
get { return (ViewModel)GetValue(ViewModelProperty); }
set { SetValue(ViewModelProperty, value); }
}
public static readonly DependencyProperty ViewModelProperty =
DependencyProperty.Register("ViewModel", typeof(ViewModel), typeof(DialogHelper),
new UIPropertyMetadata(new PropertyChangedCallback(ViewModelProperty_Changed)));
private static void ViewModelProperty_Changed(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
if (ViewModelProperty != null)
{
Binding myBinding = new Binding("FileName");
myBinding.Source = e.NewValue;
myBinding.Mode = BindingMode.OneWayToSource;
BindingOperations.SetBinding(d, FileNameProperty, myBinding);
}
}
private string FileName
{
get { return (string)GetValue(FileNameProperty); }
set { SetValue(FileNameProperty, value); }
}
private static readonly DependencyProperty FileNameProperty =
DependencyProperty.Register("FileName", typeof(string), typeof(DialogHelper),
new UIPropertyMetadata(new PropertyChangedCallback(FileNameProperty_Changed)));
private static void FileNameProperty_Changed(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
Debug.WriteLine("DialogHelper.FileName = {0}", e.NewValue);
}
public ICommand OpenFile { get; private set; }
public DialogHelper()
{
OpenFile = new RelayCommand(OpenFileAction);
}
private void OpenFileAction(object obj)
{
OpenFileDialog dlg = new OpenFileDialog();
if (dlg.ShowDialog() == true)
{
FileName = dlg.FileName;
}
}
}
帮助程序类需要对ViewModel实例的引用。请参阅资源字典。构建完成后,将设置ViewModel属性(在XAML的同一行中)。这是当辅助类上的FileName属性绑定到视图模型上的FileName属性时。
<Window x:Class="DialogExperiment.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:DialogExperiment"
xmlns:vm="clr-namespace:DialogExperimentVM;assembly=DialogExperimentVM"
Title="MainWindow" Height="350" Width="525">
<Window.Resources>
<vm:ViewModel x:Key="viewModel" />
<local:DialogHelper x:Key="helper" ViewModel="{StaticResource viewModel}"/>
</Window.Resources>
<DockPanel DataContext="{StaticResource viewModel}">
<Menu DockPanel.Dock="Top">
<MenuItem Header="File">
<MenuItem Header="Open" Command="{Binding Source={StaticResource helper}, Path=OpenFile}" />
</MenuItem>
</Menu>
</DockPanel>
</Window>
答案 4 :(得分:2)
拥有服务就像从viewmodel打开一个视图。 我在视图中有一个Dependency属性,在属性的chnage上,我打开FileDialog并读取路径,更新属性,从而更新VM的绑定属性
答案 5 :(得分:2)
我已经这样解决了这个问题:
CommandImpl未在下面的代码中实现。
namespace ViewModels.Interfaces
{
using System.Collections.Generic;
public interface IDialogWindow
{
List<string> ExecuteFileDialog(object owner, string extFilter);
}
}
namespace ViewModels
{
using ViewModels.Interfaces;
public class MyViewModel
{
public ICommand DoSomeThingCmd { get; } = new CommandImpl((dialogType) =>
{
var dlgObj = Activator.CreateInstance(dialogType) as IDialogWindow;
var fileNames = dlgObj?.ExecuteFileDialog(null, "*.txt");
//Do something with fileNames..
});
}
}
namespace Views
{
using ViewModels.Interfaces;
using Microsoft.Win32;
using System.Collections.Generic;
using System.Linq;
using System.Windows;
public class OpenFilesDialog : IDialogWindow
{
public List<string> ExecuteFileDialog(object owner, string extFilter)
{
var fd = new OpenFileDialog();
fd.Multiselect = true;
if (!string.IsNullOrWhiteSpace(extFilter))
{
fd.Filter = extFilter;
}
fd.ShowDialog(owner as Window);
return fd.FileNames.ToList();
}
}
}
XAML:
<Window
xmlns:views="clr-namespace:Views"
xmlns:viewModels="clr-namespace:ViewModels"
>
<Window.DataContext>
<viewModels:MyViewModel/>
</Window.DataContext>
<Grid>
<Button Content = "Open files.." Command="{Binding DoSomeThingCmd}" CommandParameter="{x:Type views:OpenFilesDialog}"/>
</Grid>
</Window>