使用SHFileInfo时为什么会出现错误的SpecialFolder图标?

Rac*_*hel 2 c# winapi

SHFileInfo用来检索文件和文件夹的系统图标,但是我发现特殊文件夹没有返回正确的文件夹图标.

例如,桌面文件夹将返回与常规文件夹相同的文件夹图标而不是桌面图标,并且MyComputer图标看起来像旧的Windows 98图标,而不是我期望的Windows 7 MyComputer图标.

为什么我得到特殊文件夹的错误图标,如何使用特定文件夹检索正确的系统图标SHFileInfo

我的原始代码来自这个代码项目文章,但它已被修改了一下.执行的实际代码仍然非常相似,如下所示:

public static System.Drawing.Icon GetFolderIcon(string folderPath, IconSize size, FolderType folderType)
{
    try
    {
        // Need to add size check, although errors generated at present!
        Int64 flags = WinApi.SHGFI_ICON | WinApi.SHGFI_USEFILEATTRIBUTES;

        if (FolderType.Open == folderType)
            flags |= WinApi.SHGFI_OPENICON;

        if (IconSize.Small == size)
            flags |= WinApi.SHGFI_SMALLICON;
        else
            flags |= WinApi.SHGFI_LARGEICON;

        // Get the folder icon
        WinApi.SHFILEINFO shfi = new WinApi.SHFILEINFO();
        WinApi.SHGetFileInfo(folderPath,
            WinApi.FILE_ATTRIBUTE_DIRECTORY,
            ref shfi,
            (Int32)System.Runtime.InteropServices.Marshal.SizeOf(shfi),
            flags);

        if (shfi.hIcon == IntPtr.Zero)
            return null;

        // Now clone the icon, so that it can be successfully stored in an ImageList
        System.Drawing.Icon icon = (System.Drawing.Icon)System.Drawing.Icon.FromHandle(shfi.hIcon).Clone();

        WinApi.DestroyIcon(shfi.hIcon);     // Cleanup
        return icon;
    }
    catch (Exception ex)
    {
        // Log Error
    }

    return null;
}
Run Code Online (Sandbox Code Playgroud)

随着它的调用看起来像这样:

var icon = IconUtil.GetFolderIcon(
    Environment.GetFolderPath(Environment.SpecialFolder.Desktop), 
    IconUtil.IconSize.Large, IconUtil.FolderType.Closed);
Run Code Online (Sandbox Code Playgroud)

我得到的图标看起来像这样

在此输入图像描述

而不是这个

在此输入图像描述

在此输入图像描述

在此输入图像描述

Rac*_*hel 6

根据MSDN,SHGFI_USEFILEATTRIBUTES标志:

表示该函数不应尝试访问pszPath指定的文件.相反,它应该表现为pszPath指定的文件与dwFileAttributes中传递的文件属性一样.

我认为Raymond Chen的评论提供了一个更容易理解的解释:

你通过的SHGFI_USEFILEATTRIBUTES意思是"忽略文件实际上是什么,只是假装它是我告诉你的." 而你的假装文件属性FILE_ATTRIBUTE_DIRECTORY意味着"只是一个简单无聊的目录".

所以为了解决我的问题,我只需要SHGFI_USEFILEATTRIBUTES在我想获取特定于文件夹的图标时删除该标志.

Eric Brown的评论也为使用PIDL提供了一种有用的替代方法.可以在此处找到代码的示例.