(C#)Windows Shell 外壳编程系列5 - 获取图标
(本系列文章由柠檬的(lc_mtt)原创,转载请注明出处,谢谢~)
接上一节:(C#)Windows Shell 外壳编程系列4 - 上下文菜单(iContextMenu)(二)嵌入菜单和执行命令
有关 PIDL
PIDL亦有“绝对路径”与“相对路径”的概念。表示“相对路径”的PIDL(本文简称为“相对PIDL”)只有一个ITEMIDLIST结构的元素,用于标识相对于父文件夹的“路径”;表示“绝对路径”的PIDL(简称为“绝对PIDL”)有若干个ITEMIDLIST结构的元素,第一个元素表示外壳名字空间根文件夹(“桌面”)下的某一子文件夹A,第二个元素则表示文件夹A下的某一子文件夹B,其余依此类推。这样绝对PIDL就通过保存一条从“桌面”下的直接子文件夹或文件的绝对PIDL与相对PIDL是相同的,而其他的文件夹或文件的相对PIDL就只是其绝对PIDL的最后一部分了。
为什么要说这些呢?因为有些函数,必须使用绝对PIDL,例如图标,如果不使用绝对PIDL,某些图标是无法正常获得的(驱动器、控制面板等)。
但使用 EnumObjects 获得的,仅仅是相对PIDL,如果通过相对PIDL获取绝对PIDL呢?我参考了开源项目 C# FileBrowser 中的 PIDL 类
 PIDL.cs
PIDL.cs
该类实现了 PIDL 的复制和结合功能。现在我们修改 ShellItem 类,使它带有父节点的 IShellFolder 以及提供获取绝对 PIDL 的属性:
 private ShellItem m_ParentItem;
private ShellItem m_ParentItem;

 public ShellItem ParentItem
public ShellItem ParentItem
 {
{
 get { return m_ParentItem; }
    get { return m_ParentItem; }
 set { m_ParentItem = value; }
    set { m_ParentItem = value; }
 }
}

 /// <summary>
/// <summary>
 /// 绝对 PIDL
/// 绝对 PIDL
 /// </summary>
/// </summary>
 public PIDL PIDLFull
public PIDL PIDLFull
 {
{
 get
    get
 {
    {
 PIDL pidlFull = new PIDL(PIDL, true);
        PIDL pidlFull = new PIDL(PIDL, true);
 ShellItem current = ParentItem;
        ShellItem current = ParentItem;
 while (current != null)
        while (current != null)
 {
        {
 pidlFull.Insert(current.PIDL);
            pidlFull.Insert(current.PIDL);
 current = current.ParentItem;
            current = current.ParentItem;
 }
        }
 return pidlFull;
        return pidlFull;
 }
    }
 }
}
获取图标
言归正传,既然已经获得绝对 PIDL,那么获取图标就是很简单的事情了,我们使用的是 SHGetFileInfo 这个API:
 [DllImport("shell32", EntryPoint = "SHGetFileInfo", ExactSpelling = false,
[DllImport("shell32", EntryPoint = "SHGetFileInfo", ExactSpelling = false, 
 CharSet = CharSet.Auto, SetLastError = true)]
    CharSet = CharSet.Auto, SetLastError = true)]
 public static extern IntPtr SHGetFileInfo(
public static extern IntPtr SHGetFileInfo(
 IntPtr ppidl,
    IntPtr ppidl, 
 FILE_ATTRIBUTE dwFileAttributes,
    FILE_ATTRIBUTE dwFileAttributes, 
 ref SHFILEINFO sfi,
    ref SHFILEINFO sfi, 
 int cbFileInfo,
    int cbFileInfo, 
 SHGFI uFlags);
    SHGFI uFlags);

 [DllImport("Shell32.dll", CharSet = CharSet.Auto)]
[DllImport("Shell32.dll", CharSet = CharSet.Auto)]
 public static extern IntPtr SHGetFileInfo(
public static extern IntPtr SHGetFileInfo(
 string Path,
    string Path, 
 FILE_ATTRIBUTE fileAttributes,
    FILE_ATTRIBUTE fileAttributes, 
 out SHFILEINFO sfi,
    out SHFILEINFO sfi, 
 int cbFileInfo, SHGFI flags);
    int cbFileInfo, SHGFI flags);
 
这里提供了一个重载,你可以选择是通过 PIDL 还是 路径 获取图标(如果是路径,那么仅仅能获取 文件夹/文件 的图标)。
 /// <summary>
/// <summary>
 /// 获取小图标索引
/// 获取小图标索引
 /// </summary>
/// </summary>
 public static int GetSmallIconIndex(string strFilename)
public static int GetSmallIconIndex(string strFilename)
 {
{
 SHFILEINFO psfi = new SHFILEINFO();
    SHFILEINFO psfi = new SHFILEINFO();
 IntPtr ipIcon = SHGetFileInfo(strFilename, 0, out psfi, Marshal.SizeOf(psfi),
    IntPtr ipIcon = SHGetFileInfo(strFilename, 0, out psfi, Marshal.SizeOf(psfi),
 SHGFI.ICON | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);
        SHGFI.ICON | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);

 return psfi.iIcon;
    return psfi.iIcon;
 }
}

 public static int GetSmallIconIndex(IntPtr ipIDList)
public static int GetSmallIconIndex(IntPtr ipIDList)
 {
{
 SHFILEINFO psfi = new SHFILEINFO();
    SHFILEINFO psfi = new SHFILEINFO();
 IntPtr ipIcon = SHGetFileInfo(ipIDList, 0, ref psfi, Marshal.SizeOf(psfi),
    IntPtr ipIcon = SHGetFileInfo(ipIDList, 0, ref psfi, Marshal.SizeOf(psfi),
 SHGFI.ICON | SHGFI.PIDL | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);
        SHGFI.ICON | SHGFI.PIDL | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);

 return psfi.iIcon;
    return psfi.iIcon;
 }
}
大家也许会觉得奇怪,GetSmallIconIndex 返回的是 int ,到底要怎么使用?
其实没错,GetSmallIconIndex 仅仅是返回该图标在系统图像列表(System ImageList)的索引(Index)而已。我们只要获取系统图像列表的指针,再把它关联到你的 TreeView 或 ListView ,即可通过 Icon Index 来显示图标了。
 IntPtr m_ipSmallSystemImageList;
IntPtr m_ipSmallSystemImageList;
 IntPtr m_ipLargeSystemImageList;
IntPtr m_ipLargeSystemImageList;

 //获取系统 ImageList
//获取系统 ImageList
 SHFILEINFO shfi = new SHFILEINFO();
SHFILEINFO shfi = new SHFILEINFO();

 m_ipSmallSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)),
m_ipSmallSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)),
 SHGFI.SYSICONINDEX | SHGFI.SMALLICON | SHGFI.USEFILEATTRIBUTES);
    SHGFI.SYSICONINDEX | SHGFI.SMALLICON | SHGFI.USEFILEATTRIBUTES);

 m_ipLargeSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)),
m_ipLargeSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)),
 SHGFI.SYSICONINDEX | SHGFI.LARGEICON | SHGFI.USEFILEATTRIBUTES);
    SHGFI.SYSICONINDEX | SHGFI.LARGEICON | SHGFI.USEFILEATTRIBUTES);

 //把系统 ImageList 关联到 TreeView 和 ListView
//把系统 ImageList 关联到 TreeView 和 ListView
 API.SendMessage(Tree1.Handle, API.TVM_SETIMAGELIST, API.TVSIL_NORMAL, m_ipSmallSystemImageList);
API.SendMessage(Tree1.Handle, API.TVM_SETIMAGELIST, API.TVSIL_NORMAL, m_ipSmallSystemImageList);
 API.SendMessage(lvFile.Handle, API.LVM_SETIMAGELIST, API.LVSIL_NORMAL, m_ipLargeSystemImageList);
API.SendMessage(lvFile.Handle, API.LVM_SETIMAGELIST, API.LVSIL_NORMAL, m_ipLargeSystemImageList);
OK,我们修改以往的例子,就可以在 Tree 节点上显示图标了:



 ShellItem shellItem=new ShellItem(pidlSub, iSub, sItem);
ShellItem shellItem=new ShellItem(pidlSub, iSub, sItem);
 int imgIndex = API.GetSmallIconIndex(shellItem.PIDLFull.Ptr);
int imgIndex = API.GetSmallIconIndex(shellItem.PIDLFull.Ptr);
 TreeNode nodeSub = new TreeNode(name, imgIndex, imgIndex);
TreeNode nodeSub = new TreeNode(name, imgIndex, imgIndex);


 
(注:关于文中出现的一些结构体或常量,读者可以自行查阅 MSDN,精力有限实在不能一一说明。)
我们来看一下效果:

事实上,这个代码改了很多,也涉及到下一节的部分内容,因此代码将在下一节中抛出...
接上一节:(C#)Windows Shell 外壳编程系列4 - 上下文菜单(iContextMenu)(二)嵌入菜单和执行命令
有关 PIDL
PIDL亦有“绝对路径”与“相对路径”的概念。表示“相对路径”的PIDL(本文简称为“相对PIDL”)只有一个ITEMIDLIST结构的元素,用于标识相对于父文件夹的“路径”;表示“绝对路径”的PIDL(简称为“绝对PIDL”)有若干个ITEMIDLIST结构的元素,第一个元素表示外壳名字空间根文件夹(“桌面”)下的某一子文件夹A,第二个元素则表示文件夹A下的某一子文件夹B,其余依此类推。这样绝对PIDL就通过保存一条从“桌面”下的直接子文件夹或文件的绝对PIDL与相对PIDL是相同的,而其他的文件夹或文件的相对PIDL就只是其绝对PIDL的最后一部分了。
为什么要说这些呢?因为有些函数,必须使用绝对PIDL,例如图标,如果不使用绝对PIDL,某些图标是无法正常获得的(驱动器、控制面板等)。
但使用 EnumObjects 获得的,仅仅是相对PIDL,如果通过相对PIDL获取绝对PIDL呢?我参考了开源项目 C# FileBrowser 中的 PIDL 类
 PIDL.cs
PIDL.cs该类实现了 PIDL 的复制和结合功能。现在我们修改 ShellItem 类,使它带有父节点的 IShellFolder 以及提供获取绝对 PIDL 的属性:
 private ShellItem m_ParentItem;
private ShellItem m_ParentItem;
 public ShellItem ParentItem
public ShellItem ParentItem {
{ get { return m_ParentItem; }
    get { return m_ParentItem; } set { m_ParentItem = value; }
    set { m_ParentItem = value; } }
}
 /// <summary>
/// <summary> /// 绝对 PIDL
/// 绝对 PIDL /// </summary>
/// </summary> public PIDL PIDLFull
public PIDL PIDLFull {
{ get
    get {
    { PIDL pidlFull = new PIDL(PIDL, true);
        PIDL pidlFull = new PIDL(PIDL, true); ShellItem current = ParentItem;
        ShellItem current = ParentItem; while (current != null)
        while (current != null) {
        { pidlFull.Insert(current.PIDL);
            pidlFull.Insert(current.PIDL); current = current.ParentItem;
            current = current.ParentItem; }
        } return pidlFull;
        return pidlFull; }
    } }
}获取图标
言归正传,既然已经获得绝对 PIDL,那么获取图标就是很简单的事情了,我们使用的是 SHGetFileInfo 这个API:
 [DllImport("shell32", EntryPoint = "SHGetFileInfo", ExactSpelling = false,
[DllImport("shell32", EntryPoint = "SHGetFileInfo", ExactSpelling = false,  CharSet = CharSet.Auto, SetLastError = true)]
    CharSet = CharSet.Auto, SetLastError = true)] public static extern IntPtr SHGetFileInfo(
public static extern IntPtr SHGetFileInfo( IntPtr ppidl,
    IntPtr ppidl,  FILE_ATTRIBUTE dwFileAttributes,
    FILE_ATTRIBUTE dwFileAttributes,  ref SHFILEINFO sfi,
    ref SHFILEINFO sfi,  int cbFileInfo,
    int cbFileInfo,  SHGFI uFlags);
    SHGFI uFlags);
 [DllImport("Shell32.dll", CharSet = CharSet.Auto)]
[DllImport("Shell32.dll", CharSet = CharSet.Auto)] public static extern IntPtr SHGetFileInfo(
public static extern IntPtr SHGetFileInfo( string Path,
    string Path,  FILE_ATTRIBUTE fileAttributes,
    FILE_ATTRIBUTE fileAttributes,  out SHFILEINFO sfi,
    out SHFILEINFO sfi,  int cbFileInfo, SHGFI flags);
    int cbFileInfo, SHGFI flags);
这里提供了一个重载,你可以选择是通过 PIDL 还是 路径 获取图标(如果是路径,那么仅仅能获取 文件夹/文件 的图标)。
 /// <summary>
/// <summary> /// 获取小图标索引
/// 获取小图标索引 /// </summary>
/// </summary> public static int GetSmallIconIndex(string strFilename)
public static int GetSmallIconIndex(string strFilename) {
{ SHFILEINFO psfi = new SHFILEINFO();
    SHFILEINFO psfi = new SHFILEINFO(); IntPtr ipIcon = SHGetFileInfo(strFilename, 0, out psfi, Marshal.SizeOf(psfi),
    IntPtr ipIcon = SHGetFileInfo(strFilename, 0, out psfi, Marshal.SizeOf(psfi), SHGFI.ICON | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);
        SHGFI.ICON | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);
 return psfi.iIcon;
    return psfi.iIcon; }
}
 public static int GetSmallIconIndex(IntPtr ipIDList)
public static int GetSmallIconIndex(IntPtr ipIDList) {
{ SHFILEINFO psfi = new SHFILEINFO();
    SHFILEINFO psfi = new SHFILEINFO(); IntPtr ipIcon = SHGetFileInfo(ipIDList, 0, ref psfi, Marshal.SizeOf(psfi),
    IntPtr ipIcon = SHGetFileInfo(ipIDList, 0, ref psfi, Marshal.SizeOf(psfi), SHGFI.ICON | SHGFI.PIDL | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);
        SHGFI.ICON | SHGFI.PIDL | SHGFI.SMALLICON | SHGFI.SYSICONINDEX);
 return psfi.iIcon;
    return psfi.iIcon; }
}大家也许会觉得奇怪,GetSmallIconIndex 返回的是 int ,到底要怎么使用?
其实没错,GetSmallIconIndex 仅仅是返回该图标在系统图像列表(System ImageList)的索引(Index)而已。我们只要获取系统图像列表的指针,再把它关联到你的 TreeView 或 ListView ,即可通过 Icon Index 来显示图标了。
 IntPtr m_ipSmallSystemImageList;
IntPtr m_ipSmallSystemImageList; IntPtr m_ipLargeSystemImageList;
IntPtr m_ipLargeSystemImageList;
 //获取系统 ImageList
//获取系统 ImageList SHFILEINFO shfi = new SHFILEINFO();
SHFILEINFO shfi = new SHFILEINFO();
 m_ipSmallSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)),
m_ipSmallSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)), SHGFI.SYSICONINDEX | SHGFI.SMALLICON | SHGFI.USEFILEATTRIBUTES);
    SHGFI.SYSICONINDEX | SHGFI.SMALLICON | SHGFI.USEFILEATTRIBUTES);
 m_ipLargeSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)),
m_ipLargeSystemImageList = API.SHGetFileInfo("", 0, out shfi, Marshal.SizeOf(typeof(SHFILEINFO)), SHGFI.SYSICONINDEX | SHGFI.LARGEICON | SHGFI.USEFILEATTRIBUTES);
    SHGFI.SYSICONINDEX | SHGFI.LARGEICON | SHGFI.USEFILEATTRIBUTES);
 //把系统 ImageList 关联到 TreeView 和 ListView
//把系统 ImageList 关联到 TreeView 和 ListView API.SendMessage(Tree1.Handle, API.TVM_SETIMAGELIST, API.TVSIL_NORMAL, m_ipSmallSystemImageList);
API.SendMessage(Tree1.Handle, API.TVM_SETIMAGELIST, API.TVSIL_NORMAL, m_ipSmallSystemImageList); API.SendMessage(lvFile.Handle, API.LVM_SETIMAGELIST, API.LVSIL_NORMAL, m_ipLargeSystemImageList);
API.SendMessage(lvFile.Handle, API.LVM_SETIMAGELIST, API.LVSIL_NORMAL, m_ipLargeSystemImageList);OK,我们修改以往的例子,就可以在 Tree 节点上显示图标了:



 ShellItem shellItem=new ShellItem(pidlSub, iSub, sItem);
ShellItem shellItem=new ShellItem(pidlSub, iSub, sItem); int imgIndex = API.GetSmallIconIndex(shellItem.PIDLFull.Ptr);
int imgIndex = API.GetSmallIconIndex(shellItem.PIDLFull.Ptr); TreeNode nodeSub = new TreeNode(name, imgIndex, imgIndex);
TreeNode nodeSub = new TreeNode(name, imgIndex, imgIndex);


(注:关于文中出现的一些结构体或常量,读者可以自行查阅 MSDN,精力有限实在不能一一说明。)
我们来看一下效果:

事实上,这个代码改了很多,也涉及到下一节的部分内容,因此代码将在下一节中抛出...
 
 
     
                
            
         浙公网安备 33010602011771号
浙公网安备 33010602011771号