1.前言

WPF并没有文件选择对话框,要用也就只有使用Winform版的控件。至今我也没有寻找到一个WPF版本的文件选择对话框。

可能是我眼浊,如果各位知道有功能比较健全的WPF版文件选择对话框、文件打开对话框,还请留言告知。

这次做的是一个精简版的文件选择对话框。包含一个UserControl和一个承载UserControl的Window。

另外TreeView的样式引用自Mahspps中的样式。也就是如果需要使用这个文件选择对话框,就必须要引用Mahapps的相关dll。

当然,我会提供整个项目的源代码。如果大家不嫌弃,可以自己移除项目中的Mahspps相关引用。当然,这样做可能会使得控件界面显得比较难看,不过你还可以自己给TreeView提供你自己喜欢的样式。

这个文件选择对话框功能比较精简,仅仅提供文件夹选择的功能。其他诸如右键菜单、新建文件夹、文件夹拖放等功能目前并未实现。要完整使用WPF实现微软完整版的FolderBrowserDialog,我相信这工作量还是相当大的。

2.开始分析

先打开Winform版本的文件选择对话框看看是什么样子:

可以看到,从上到下依次是:桌面(不可展开)、库、计算机(展开就是各个磁盘驱动器)、网络、控制面板、回收站、以及桌面上的各个文件夹。

不管微软的这个习惯设计得怎么样,反正用户现在也习惯了这个设计。所以,本次实现的WPF版文件选择对话框也大致采用这个设计。其中有几点:

库用得太少了,去掉。

网络这个似乎有些是要输入密码的,我这没有远程的共享主机,所以不清楚微软这个是怎么让用户输入密码的。暂不实现。

至于控制版面和回收站,我不知道微软把这两个放在文件选择对话框里面算是什么意思?活跃气氛么~~~

其实说到这里,我有个想法。微软这个文件选择对话框里面的操作文件的方式(包括右键菜单、文件拖放等)和系统的资源管理器实在太像了。所以我怀疑这个文件选择对话框

里面的那棵树实际上是嵌入的一个另类的资源管理器,而并非微软单独开发的一个树。。。这样说来,也可以解释为什么微软不提供WPF版的文件选择对话框了,因为Explorer内部实现可能不是采用的WPF方式实现,所以改起来工作量很大...然后就不提供了。

当然,这些都是猜测,猜测。。

好了,话说回来,这个控件最关键的一点就是怎么把整个磁盘的文件通过一棵树加载起来,总不能初始化的时候先把整个磁盘的文件组织成一棵树?效率的原因决不允许你那样做。

有没有一种方法可以在用户展开某个节点的时候才初始化它的子节点?

微软为TreeView提供了一个模版:HierarchicalDataTemplate

使用这个模版,当用户展开某个节点时,TreeView会展开这个节点并且初始化这个节点的子节点的子节点,也就是HierarchicalDataTemplate第一次初始化的时候就被初始化第一层和第二层,当用户展开第一层的时候,它就开始初始化第三层...依次类推。因为使用这个模版初始化TreeView效率还是相当可观的。

到此,我们为TreeView准备一个模型(Model):

这个模型应该至少有这几个属性:

1.Name,节点的名称

2.FullName,节点所在位置的完整磁盘路径

3.Children,节点的所有子节点

另外你如果还需要其他类似图标等等也可以继续加。

在本例中,Model如下:

可以看到,多出了一个Type,这个Type主要指节点的类型,比如上面提到的网络、库、控制面板等等,这个说它们是文件夹但又不太是,所以没办法和文件夹一样统一处理。

所以给每个节点赋予一个类型,既方便处理,也方便以后的扩展。

Type的实现如下:

/// <summary>
/// 文件项类型
/// </summary>
internal enum MetroFolderBrowserControlModelType
{
/// <summary>
/// 表明这是一个文件夹
/// </summary>
Directory,
/// <summary>
/// 桌面
/// </summary>
Desktop,
/// <summary>
/// 计算机
/// </summary>
Computer,
/// <summary>
/// 磁盘驱动器
/// </summary>
Disk,
}

考虑到不同类型的节点给它们不同的节点图标显得更好看,所以有这样做:

        /// <summary>
/// 文件项类型
/// </summary>
public MetroFolderBrowserControlModelType Type
{
get
{
return type;
}
set
{
switch(value)
{
case MetroFolderBrowserControlModelType.Directory:
{
ItemImagePath = ImagePathHelper.FolderIconPath;
break;
}
case MetroFolderBrowserControlModelType.Computer:
{
ItemImagePath = ImagePathHelper.ComputerIconPath;
break;
}
case MetroFolderBrowserControlModelType.Desktop:
{
ItemImagePath = ImagePathHelper.DesktopIconPath;
break;
}
case MetroFolderBrowserControlModelType.Disk:
{
ItemImagePath = ImagePathHelper.DiskIconPath;
break;
}
default:
{
ItemImagePath = ImagePathHelper.FolderIconPath;
break;
}
}
type = value;
}
}

现在最主要的就是Children的实现还没做好。Children的如何实现也关系到整个控件的效率。

针对不同的类型,采用不同的办法获取其Children:

        /// <summary>
/// 子目录(文件 + 文件夹)
/// </summary>
public ObservableCollection<MetroFolderBrowserControlModel> Children
{
get
{
try
{
if (children != null)
{
return children;
}
children = new ObservableCollection<MetroFolderBrowserControlModel>();
switch(Type)
{
case MetroFolderBrowserControlModelType.Desktop:
{
break;
}
case MetroFolderBrowserControlModelType.Computer:
{
foreach (var device in Environment.GetLogicalDrives())
{
if (Directory.Exists(device))
{
MetroFolderBrowserControlModel model = new MetroFolderBrowserControlModel();
model.FileName = device;
model.FullName = device;
model.Type = MetroFolderBrowserControlModelType.Disk;
children.InvokeAdd<MetroFolderBrowserControlModel>(model);
}
}
break;
}
case MetroFolderBrowserControlModelType.Disk:
case MetroFolderBrowserControlModelType.Directory:
{
foreach (var item in ExplorerHelper.GetDirectoryChildrenItems(FullName, true, false, false))
{
MetroFolderBrowserControlModel model = new MetroFolderBrowserControlModel();
model.FileName = item.Name;
model.FullName = item.FullName;
children.Add(model);
}
break;
}
default:
{
break;
}
}
return children;
}
catch (Exception)
{
//出现异常,返回空的集合
return null;
}
}
}

接下来使用  绑好,这个树差不多就展示出来了:

        <TreeView ItemsSource="{Binding MetroFolderBrowserControlModels}" x:Name="treeview">
<TreeView.ItemTemplate>
<HierarchicalDataTemplate ItemsSource="{Binding Children}">
<StackPanel Orientation="Horizontal">
<Image Source="{Binding ItemImagePath, Mode=TwoWay}" Width="16" Height="16"/>
<TextBlock Text="{Binding FileName}" Margin="5,0,0,0"/>
</StackPanel>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>

至于其他获取用户选择的路径、对话框的返回值等等就不继续说了,有需要的可以下载源代码查看:

下载源代码:

http://download.csdn.net/detail/lyclovezmy/7655655

大致效果如下图:

WPF:自定义Metro样式文件夹选择对话框FolderBrowserDialog的更多相关文章

  1. c# 打开、保存文件对话框 和 文件夹选择对话框

    1. OpenFileDialog openImageDialog = new OpenFileDialog(); openImageDialog.Filter = "Image Files ...

  2. [VB.NET]调用系统的文件夹选择对话框

    以下示例代码展示如何调用系统的文件夹选择对话框: Private Function SelectFolder(ByVal Describe As String, Optional ByVal Show ...

  3. 在WPF中使用文件夹选择对话框

    开发中有时会想实现"选择某个文件夹"的效果: 在WPF中,使用Microsoft.Win32.OpenFileDialog只能选择文件,FolderBrowserDialog只能用 ...

  4. C++文件(夹)选择对话框

    由于各种应用,我们需要调用系统的打开文件对话框或者打开文件夹对话框,或两者兼有.今遇到这个情况已经解决,特写下这篇博文. 1.打开文件对话框常用的方法是使用系统的CFileDialog.这里介绍另外一 ...

  5. 【转】python qt(pyqt)的文件打开、文件保存、文件夹选择对话框

    import PyQt4.QtCore,PyQt4.QtGui # 获取文件路径对话框 file_name = QFileDialog.getOpenFileName(self,"open ...

  6. 文件夹选择之FolderBrowserDialog控件

    应用程序可能只允许用户选择文佳夹而非文件,例如在播放MP3时,用户可能把所有的MP3放在一个文佳夹内,在添加时,只要选择添加这个文佳夹,将会把在这个文件内的所有MP3添加的播放器里.在这里对播放器来说 ...

  7. (WPF) 文件和文件夹选择对话框。

    点击button,选择一个excel文件,并将文件名显示在textbox上. private void btnSelectErrorTableFile_Click(object sender, Rou ...

  8. MFC 文件夹选择对话框

    CString setSavePath() { CString strPath = _T(""); HRESULT hr; LPITEMIDLIST pItemList; BROW ...

  9. Asp.net点击按钮弹出文件夹选择框的实现(网页)

    本文地址:http://www.cnblogs.com/PiaoMiaoGongZi/p/4092112.html 在Asp.net网站实际的开发中,比如:需要实现点击一个类似于FileUpload的 ...

随机推荐

  1. Zookeeper-watcher机制源码分析(一)

    Watcher的基本流程 ZooKeeper 的 Watcher 机制,总的来说可以分为三个过程:客户端注册 Watcher.服务器处理 Watcher 和客户端回调 Watcher 客户端注册wat ...

  2. 我对java String的理解 及 源码浅析

    摘要: 摘要: 原创出处: http://www.cnblogs.com/Alandre/ 泥沙砖瓦浆木匠 希望转载,保留摘要,谢谢! 每天起床告诉自己,自己的目标是 ”技术 + 英语 还有生活“! ...

  3. 如何一步步在生产环境上部署django和vue

    本文由云+社区发表 本文主要讲述了如何一步步在生产环境上部署django和vue,操作系统默认为centos 说明:后文中出现的以下字符串均表示具体的路径或者名称,含义如下: DJANGO_DIR-- ...

  4. 浅谈Web服务器和应用服务器的区别

    1Web服务器和应用服务器简介 通俗的讲,Web服务器传送页面使浏览器可以浏览,然而应用程序服务器提供的是客户端应用程序可以调用(call)的方法(methods).确切一点,你可以说:Web服务器专 ...

  5. MySQL系列详解十:MySQL多源复制演示-技术流ken

    前言 多源复制即多主一从结构,多个主服务器端的数据都会同步到后端一个从服务器上面.至于为什么要做多源复制下面的总结很到位. 1.灾备作用:将各个库汇总在一起,就算是其他库都挂了(整个机房都无法连接了) ...

  6. 【转】JAVA解压.TAR.Z及.ZIP文件

     解压.ZIP文件 package app.qdupr.Method; import java.io.File; import java.io.FileOutputStream; import jav ...

  7. [翻译]一个新式的基于文本的浏览器 Browsh

    介绍 什么是Browsh? Browsh是一个纯文本浏览器,可以运行在大多数的TTY终端环境和任何浏览器.目前1 ,终端客户端比浏览器客户端更先进2. TTY 客户端 终端客户端即时更新和交付,以便于 ...

  8. [转]Angular引入第三方库

    本文转自: https://blog.csdn.net/yuzhiqiang_1993/article/details/71215232 版权声明:本文为博主原创文章,转载请注明地址.如果文中有什么纰 ...

  9. 《深入理解Java虚拟机》(三)垃圾收集器与内存分配策略

    垃圾收集器与内存分配策略 详解 3.1 概述 本文参考的是周志明的 <深入理解Java虚拟机>第三章 ,为了整理思路,简单记录一下,方便后期查阅. 3.2 对象已死吗 在垃圾收集器进行回收 ...

  10. C#调用存储过程执行缓慢,但在数据库中执行却很快的问题

    参考: http://www.debugease.com/mssqlbasic/976568.html https://www.cnblogs.com/Irving/p/3951220.html ht ...