// // Copyright (c) PlaceholderCompany. All rights reserved. // namespace SystemTrayMenu.Utilities { using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Drawing; using System.Drawing.Imaging; using System.IO; using System.Runtime.InteropServices; using System.Threading.Tasks; // from https://www.codeproject.com/Articles/2532/Obtaining-and-managing-file-and-folder-icons-using // added ImageList_GetIcon, IconCache, AddIconOverlay /// /// Provides static methods to read system icons for both folders and files. /// /// /// IconReader.GetFileIcon("c:\\general.xls"); /// public static class IconReader { private static readonly ConcurrentDictionary DictIconCache = new ConcurrentDictionary(); // private static readonly object ReadIcon = new object(); public enum IconSize { Large = 0, // 32x32 pixels Small = 1, // 16x16 pixels } public enum FolderType { Open = 0, Closed = 1, } public static void Dispose() { foreach (Icon icon in DictIconCache.Values) { icon?.Dispose(); } } public static Icon GetFileIconWithCache(string filePath, bool linkOverlay, IconSize size = IconSize.Small) { Icon icon = null; string extension = Path.GetExtension(filePath); if (IsExtensionWitSameIcon(extension)) { icon = DictIconCache.GetOrAdd(extension, GetIcon); Icon GetIcon(string keyExtension) { return GetFileIconSTA(filePath, linkOverlay, size); } } else { icon = GetFileIconSTA(filePath, linkOverlay, size); } return icon; } [System.Diagnostics.CodeAnalysis.SuppressMessage("Reliability", "CA2008:Do not create tasks without passing a TaskScheduler", Justification = "todo")] public static Icon GetFolderIconSTA( string directoryPath, FolderType folderType, bool linkOverlay, IconSize size = IconSize.Small) { Icon icon = null; Task task = Task.Factory.StartNew(() => GetFolderIcon( directoryPath, folderType, linkOverlay, size)); icon = task.Result; return icon; } public static Icon GetFolderIcon( string directoryPath, FolderType folderType, bool linkOverlay, IconSize size = IconSize.Small) { Icon icon = null; // Need to add size check, although errors generated at present! // uint flags = Shell32.SHGFI_ICON | Shell32.SHGFI_USEFILEATTRIBUTES; // MH: Removed SHGFI_USEFILEATTRIBUTES, otherwise was wrong folder icon uint flags = DllImports.NativeMethods.ShgfiIcon; // | Shell32.SHGFI_USEFILEATTRIBUTES; if (linkOverlay) { flags += DllImports.NativeMethods.ShgfiLINKOVERLAY; } if (folderType == FolderType.Open) { flags += DllImports.NativeMethods.ShgfiOPENICON; } if (size == IconSize.Small) { flags += DllImports.NativeMethods.ShgfiSMALLICON; } else { flags += DllImports.NativeMethods.ShgfiLARGEICON; } // Get the folder icon DllImports.NativeMethods.SHFILEINFO shfi = default; IntPtr success = DllImports.NativeMethods.Shell32SHGetFileInfo( directoryPath, DllImports.NativeMethods.FileAttributeDirectory, ref shfi, (uint)Marshal.SizeOf(shfi), flags); if (success != IntPtr.Zero && shfi.hIcon != IntPtr.Zero) { try { icon = (Icon)Icon.FromHandle(shfi.hIcon).Clone(); DllImports.NativeMethods.User32DestroyIcon(shfi.hIcon); } #pragma warning disable CA1031 // Do not catch general exception types catch (Exception ex) #pragma warning restore CA1031 // Do not catch general exception types { Log.Error($"directoryPath:'{directoryPath}'", ex); } } return icon; } public static Icon AddIconOverlay(Icon originalIcon, Icon overlay) { Icon icon = null; if (originalIcon != null) { using Bitmap target = new Bitmap( originalIcon.Width, originalIcon.Height, PixelFormat.Format32bppArgb); Graphics graphics = Graphics.FromImage(target); graphics.DrawIcon(originalIcon, 0, 0); graphics.DrawIcon(overlay, 0, 0); target.MakeTransparent(target.GetPixel(1, 1)); icon = Icon.FromHandle(target.GetHicon()); } return icon; } private static bool IsExtensionWitSameIcon(string fileExtension) { bool isExtensionWitSameIcon = true; List extensionsWithDiffIcons = new List { string.Empty, ".EXE", ".LNK", ".ICO", ".URL" }; if (extensionsWithDiffIcons.Contains(fileExtension.ToUpperInvariant())) { isExtensionWitSameIcon = false; } return isExtensionWitSameIcon; } [System.Diagnostics.CodeAnalysis.SuppressMessage("Reliability", "CA2008:Do not create tasks without passing a TaskScheduler", Justification = "todo")] private static Icon GetFileIconSTA(string filePath, bool linkOverlay, IconSize size = IconSize.Small) { Icon icon = null; Task task = Task.Factory.StartNew(() => GetFileIcon(filePath, linkOverlay, size)); icon = task.Result; return icon; } private static Icon GetFileIcon(string filePath, bool linkOverlay, IconSize size = IconSize.Small) { Icon icon = null; DllImports.NativeMethods.SHFILEINFO shfi = default; uint flags = DllImports.NativeMethods.ShgfiIcon | DllImports.NativeMethods.ShgfiSYSICONINDEX; if (linkOverlay) { flags += DllImports.NativeMethods.ShgfiLINKOVERLAY; } /* Check the size specified for return. */ if (size == IconSize.Small) { flags += DllImports.NativeMethods.ShgfiSMALLICON; } else { flags += DllImports.NativeMethods.ShgfiLARGEICON; } IntPtr hImageList = DllImports.NativeMethods.Shell32SHGetFileInfo( filePath, DllImports.NativeMethods.FileAttributeNormal, ref shfi, (uint)Marshal.SizeOf(shfi), flags); if (hImageList != IntPtr.Zero) { IntPtr hIcon; if (linkOverlay) { hIcon = shfi.hIcon; // Get icon directly } else { // Get icon from .ink without overlay hIcon = DllImports.NativeMethods.ImageList_GetIcon(hImageList, shfi.iIcon, DllImports.NativeMethods.IldTransparent); } try { // Copy (clone) the returned icon to a new object, thus allowing us to clean-up properly icon = (Icon)Icon.FromHandle(hIcon).Clone(); } #pragma warning disable CA1031 // Do not catch general exception types catch (Exception ex) #pragma warning restore CA1031 // Do not catch general exception types { Log.Error($"filePath:'{filePath}'", ex); } // Cleanup if (!linkOverlay) { DllImports.NativeMethods.User32DestroyIcon(hIcon); } DllImports.NativeMethods.User32DestroyIcon(shfi.hIcon); } return icon; } } }