OpenFileDialogService
- 6 minutes to read
The OpenFileDialogService is an IOpenFileDialogService implementation that allows you to browse and open files in the File System by using the standard dialog box.
To implement file browsing functionality in accordance with the MVVM pattern, use the OpenFileDialogService class provided by the MVVM Framework.
Using the OpenFileDialogService functionality requires an OpenFileDialogService instance to be attached to a View. You can do this by using the Quick Actions as described in Quick Actions or manually as shown in the following code snippet.
<UserControl x:Class="FileDialogServicesSample.Views.FileDialogsView"
...
xmlns:dxmvvm="http://schemas.devexpress.com/winfx/2008/xaml/mvvm">
<dxmvvm:Interaction.Behaviors>
<dxmvvm:OpenFileDialogService />
</dxmvvm:Interaction.Behaviors>
</UserControl>
To show the dialog box, access the attached OpenFileDialogService (a description on how to obtain a service from a View Model is available here: Services in POCO objects and Services in ViewModelBase descendants) and invoke its ShowDialog(Action<CancelEventArgs>, String) method.
[POCOViewModel]
public class FileDialogsViewModel {
protected IOpenFileDialogService OpenFileDialogService { get { return this.GetService<IOpenFileDialogService>(); } }
...
public void Open() {
if (OpenFileDialogService.ShowDialog()) {
...
}
}
}
When the shown dialog is closed and the ShowDialog method returns True, you can obtain the selected file in the IOpenFileDialogService.File property. If the OpenFileDialogService.Multiselect feature is enabled and multiple files are selected, use the IOpenFileDialogService.Files property instead.
Note
The OpenFileDialogService uses its own FileInfoWrapper class implementing the IFileInfo interface to represent the selected file(s). This class provides the capabilities of the standard FileInfo to work with objects of the FileStream type.
You can use the InitialDirectory
property to specify the initially opened folder. Once a user selects a file, its directory becomes initial. Set the RestorePreviouslySelectedDirectory
property to false
to change this behavior and always open the folder specified by the InitialDirectory
property.
To allow users to select a folder instead of a file, use the FolderBrowserDialogService.
Example
<UserControl
x:Class="FileDialogServicesSample.Views.FileDialogsView"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:dx="http://schemas.devexpress.com/winfx/2008/xaml/core"
xmlns:dxb="http://schemas.devexpress.com/winfx/2008/xaml/bars"
xmlns:dxe="http://schemas.devexpress.com/winfx/2008/xaml/editors"
xmlns:dxr="http://schemas.devexpress.com/winfx/2008/xaml/ribbon"
xmlns:dxmvvm="http://schemas.devexpress.com/winfx/2008/xaml/mvvm"
xmlns:ViewModels="clr-namespace:FileDialogServicesSample.ViewModels"
mc:Ignorable="d" d:DesignHeight="300" d:DesignWidth="300">
<UserControl.DataContext>
<ViewModels:FileDialogsViewModel/>
</UserControl.DataContext>
<dxmvvm:Interaction.Behaviors>
<dxmvvm:OpenFileDialogService Title="{Binding Title}"/>
<dxmvvm:SaveFileDialogService Title="{Binding Title}"/>
</dxmvvm:Interaction.Behaviors>
<DockPanel>
<dxr:RibbonControl
DockPanel.Dock="Top"
ShowApplicationButton="False"
ToolbarShowMode="Hide"
RibbonTitleBarVisibility="Collapsed">
<dxr:RibbonDefaultPageCategory Caption="Default Category">
<dxr:RibbonPage Caption="File">
<dxr:RibbonPageGroup Caption="Actions">
<dxb:BarButtonItem
Content="Open"
Command="{Binding OpenCommand}"
Glyph="{dx:DXImage Image=Open_16x16.png}"
LargeGlyph="{dx:DXImage Image=Open2_32x32.png}"
RibbonStyle="Large"/>
<dxb:BarButtonItem
Content="Save"
Command="{Binding SaveCommand}"
Glyph="{dx:DXImage Image=Save_16x16.png}"
LargeGlyph="{dx:DXImage Image=Save_32x32.png}"
RibbonStyle="Large"/>
</dxr:RibbonPageGroup>
<dxr:RibbonPageGroup Caption="Settings">
<dxb:BarEditItem Content="Title" EditValue="{Binding Title}">
<dxb:BarEditItem.EditSettings>
<dxe:TextEditSettings/>
</dxb:BarEditItem.EditSettings>
</dxb:BarEditItem>
<dxb:BarEditItem Content="Filter" EditValue="{Binding Filter}">
<dxb:BarEditItem.EditSettings>
<dxe:TextEditSettings/>
</dxb:BarEditItem.EditSettings>
</dxb:BarEditItem>
<dxb:BarEditItem Content="Filter Index" EditValue="{Binding FilterIndex}">
<dxb:BarEditItem.EditSettings>
<dxe:TextEditSettings/>
</dxb:BarEditItem.EditSettings>
</dxb:BarEditItem>
</dxr:RibbonPageGroup>
</dxr:RibbonPage>
</dxr:RibbonDefaultPageCategory>
</dxr:RibbonControl>
<dxr:RibbonStatusBarControl DockPanel.Dock="Bottom">
<dxr:RibbonStatusBarControl.RightItems>
<dxb:BarStaticItem Content="{Binding DialogResult}">
<dxb:BarStaticItem.ContentTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock Text="Dialog Result: "/>
<TextBlock Text="{Binding}"/>
</StackPanel>
</DataTemplate>
</dxb:BarStaticItem.ContentTemplate>
</dxb:BarStaticItem>
</dxr:RibbonStatusBarControl.RightItems>
</dxr:RibbonStatusBarControl>
<TextBox
Text="{Binding FileBody, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}"
AcceptsReturn="True"
TextWrapping="Wrap"/>
</DockPanel>
</UserControl>
using DevExpress.Mvvm;
using DevExpress.Mvvm.DataAnnotations;
using System.IO;
using System.Linq;
namespace FileDialogServicesSample.ViewModels {
public class FileDialogsViewModel : ViewModelBase {
#region Common properties
public string Filter { get { return GetValue<string>(); } set { SetValue(value); } }
public int FilterIndex { get { return GetValue<int>(); } set { SetValue(value); } }
public string Title { get { return GetValue<string>(); } set { SetValue(value); } }
public bool DialogResult { get { return GetValue<bool>(); } protected set { SetValue(value); } }
public string ResultFileName { get { return GetValue<string>(); } protected set { SetValue(value); } }
public string FileBody { get { return GetValue<string>(); } set { SetValue(value); } }
#endregion
#region SaveFileDialogService specific properties
public string DefaultExt { get { return GetValue<string>(); } set { SetValue(value); } }
public string DefaultFileName { get { return GetValue<string>(); } set { SetValue(value); } }
public bool OverwritePrompt { get { return GetValue<bool>(); } set { SetValue(value); } }
#endregion
protected ISaveFileDialogService SaveFileDialogService { get { return GetService<ISaveFileDialogService>(); } }
protected IOpenFileDialogService OpenFileDialogService { get { return GetService<IOpenFileDialogService>(); } }
public FileDialogsViewModel() {
Filter = "Text Files (.txt)|*.txt|All Files (*.*)|*.*";
FilterIndex = 1;
Title = "Custom Dialog Title";
DefaultExt = "txt";
DefaultFileName = "Document1";
OverwritePrompt = true;
}
[Command]
public void Open() {
OpenFileDialogService.Filter = Filter;
OpenFileDialogService.FilterIndex = FilterIndex;
DialogResult = OpenFileDialogService.ShowDialog();
if (!DialogResult) {
ResultFileName = string.Empty;
} else {
IFileInfo file = OpenFileDialogService.Files.First();
ResultFileName = file.Name;
using (var stream = file.OpenText()) {
FileBody = stream.ReadToEnd();
}
}
}
[Command]
public void Save() {
SaveFileDialogService.DefaultExt = DefaultExt;
SaveFileDialogService.DefaultFileName = DefaultFileName;
SaveFileDialogService.Filter = Filter;
SaveFileDialogService.FilterIndex = FilterIndex;
DialogResult = SaveFileDialogService.ShowDialog();
if (!DialogResult) {
ResultFileName = string.Empty;
} else {
using (var stream = new StreamWriter(SaveFileDialogService.OpenFile())) {
stream.Write(FileBody);
}
}
}
}
}