4
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

C#でエクスプローラのコンテキストメニューに項目を追加する

Last updated at Posted at 2024-02-17

はじめに

Windowsのエクスプローラでファイルを右クリックしたときに出てくるメニュー(コンテキストメニュー)に項目を追加する方法です。SharpShellを使うほうが楽ですが、ちょっとしたメニューを追加する分にはこの方法で十分だと思います。以下の記事を参考にさせていただきました。

.NET で Windows エクスプローラのコンテキストメニューを拡張 #C# - Qiita

contextsample.png

作成方法

ここの作成方法と同じ方法を実施し、ビルドまで行います。但し、以下の変更をします。

  • プロジェクト名は「MyAddin」ではなく「ContextMenuExtension」とします。
  • プロジェクトを作成したら、System.Windows.Formsを参照に追加します。「プロジェクト」から「参照の追加」を選択し、「System.Windows.Forms」にチェックを入れて「OK」を押します。
  • 「コードを記載します。」の箇所のコードはベースコードを貼り付けます。
  • //GUID(変更必要)のコメント部は先頭の1箇所しかないため、この1箇所のみGUIDを新規作成し、置き換えます。
  • キーファイル名は「MyAddin」ではなく「ContextMenuExtension」とします。
  • 今回はExcelではなくWindowsが対象なので64bit Windowsであれば構成をx64に、32bit Windowsであれば構成をx86にします。

登録と解除

登録して使用する

管理者権限でコマンドプロンプトを開いてRegAsmを利用して登録します。登録するとファイルをクリックしたときに「ファイル名を表示(E)」メニューが追加されます。

REM 以下は64bit版Windowsの場合。32bitの場合はFramework64をFrameworkに変更する。
C:\Windows\Microsoft.NET\Framework64\v4.0.30319\RegAsm.exe /codebase ContextMenuExtension.dll

解除して使用をやめる

管理者権限でコマンドプロンプトを開いてRegAsmを利用して登録を解除します。その後再起動をしてContextMenuExtension.dllを削除します。

REM 以下は64bit版Windowsの場合。32bitの場合はFramework64をFrameworkに変更する。
C:\Windows\Microsoft.NET\Framework64\v4.0.30319\RegAsm.exe /u ContextMenuExtension.dll

カスタマイズ

ベースコード内の設定ここからから設定ここまでまでのソースコードを変更することでコンテキストメニューをカスタマイズできます。

メニュー名の変更

以下の箇所を変更することで表示するメニュー名を変更できます。

//メニュー名
private string MENU_NAME = "ファイル名を表示(&E)";

メニュー選択時の動作の変更

以下の箇所を変更することでメニューを選択したときに行われる動作を変更できます。選択されたファイルはstring[] filelistsに格納されています。

//メニューが選択されたときに行われる処理
private void ActionMethod()
{
    MessageBox.Show("選択されたファイル: " + Environment.NewLine + string.Join(Environment.NewLine, filelists));
}

メニューを表示するファイルの変更

以下の箇所を変更することでメニューを表示するファイルを変更することができます。

すべてのファイルにメニューを表示(ベースコードの場合)

//メニューを表示するかどうか
private bool IsShowMenu()//戻り値がtrueの場合メニューを表示
{
    return true;
}

例)拡張子がbmpのファイルにのみメニューを表示

//メニューを表示するかどうか
private bool IsShowMenu()//戻り値がtrueの場合メニューを表示
{
    //メニュー表示対象の拡張子
    string targetExtension = ".bmp";

    bool menuShow = true;

    if (filelists != null && filelists.Length > 0)
    {
        //選択されたファイルの拡張子が全てtargetExtensionかどうか
        foreach (string file in filelists)
        {
            if (Path.GetExtension(file).ToLower() != targetExtension.ToLower())
            {
                menuShow = false;
                break;
            }
        }
    }
    else
    {
        menuShow = false;
    }

    return menuShow;
}

例)ファイルサイズが1MB以上の場合のみメニューを表示

//メニューを表示するかどうか
private bool IsShowMenu()//戻り値がtrueの場合メニューを表示
{
    bool menuShow = true;

    if (filelists != null && filelists.Length > 0)
    {
        try
        {
            foreach (string file in filelists)
            {
                if ((new FileInfo(file)).Length < 1024 * 1024)
                {
                    menuShow = false;
                    break;
                }
            }
        }
        catch (Exception ex)
        {
            menuShow = false;
        }
    }
    else
    {
        menuShow = false;
    }

    return menuShow;
}

例)C:\work\配下のファイルにのみメニューを表示

//メニューを表示するかどうか
private bool IsShowMenu()//戻り値がtrueの場合メニューを表示
{
    bool menuShow = true;

    if (filelists != null && filelists.Length > 0)
    {
        try
        {
            foreach (string file in filelists)
            {
                if (!file.StartsWith(@"C:\work\"))
                {
                    menuShow = false;
                    break;
                }
            }
        }
        catch (Exception ex)
        {
            menuShow = false;
        }
    }
    else
    {
        menuShow = false;
    }

    return menuShow;
}

フォルダの余白のコンテキストメニューに追加

以下の箇所を変更することでフォルダの余白のコンテキストメニューに対してメニューを追加することができます。余白を右クリックされたフォルダはstring folderpathに格納されています。

//レジストリの場所 HKEY_CLASSES_ROOT\
private const string TargetFileType = @"Directory\Background";

具体的なコード例です。設定ここからから設定ここまでまでを以下に置き換えるとC:\work\配下のフォルダの余白を右クリックしたときのみ「コマンドプロンプトを表示(E)」メニューを表示します。

//--------------------------設定ここから--------------------------

//余白を右クリックされたフォルダは string folderpath に格納されている

//メニュー名
private string MENU_NAME = "コマンドプロンプトを表示(&E)";

//メニューが選択されたときに行われる処理
private void ActionMethod()
{
    System.Diagnostics.ProcessStartInfo processStartInfo = new System.Diagnostics.ProcessStartInfo();
    processStartInfo.FileName = "cmd.exe";
    processStartInfo.Arguments = "/s /k pushd " + "\"" + folderpath + "\"";
    System.Diagnostics.Process.Start(processStartInfo);
}

//メニューを表示するかどうか
private bool IsShowMenu()//戻り値がtrueの場合メニューを表示
{
    //C:\work配下の場合のみメニュー表示
    return folderpath != null && folderpath.StartsWith(@"C:\work");
}

//レジストリの場所 HKEY_CLASSES_ROOT\
private const string TargetFileType = @"Directory\Background";

//--------------------------設定ここまで--------------------------

ベースコード

Class1.cs
using Microsoft.Win32;
using System;
using System.Collections.Specialized;
using System.Runtime.InteropServices;
using System.Text;
using System.Windows.Forms;

namespace ContextMenuExtension//プロジェクト名
{
    [ClassInterface(ClassInterfaceType.None)]
    [Guid("90703BC8-1487-4B27-AA96-043C54268C97")]//GUID(変更必要)
    [ComVisible(true)]
    public class ContextMenuExtension : IShellExtInit, IContextMenu
    {
        //--------------------------設定ここから--------------------------

        //選択されたファイル/フォルダは string[] filelists に格納されている

        //メニュー名
        private string MENU_NAME = "ファイル名を表示(&E)";

        //メニューが選択されたときに行われる処理
        private void ActionMethod()
        {
            MessageBox.Show("選択されたファイル: " + Environment.NewLine + string.Join(Environment.NewLine, filelists));
        }

        //メニューを表示するかどうか
        private bool IsShowMenu()//戻り値がtrueの場合メニューを表示
        {
            return true;
        }

        //レジストリの場所 HKEY_CLASSES_ROOT\
        private const string TargetFileType = "*";

        //--------------------------設定ここまで--------------------------

        //以下おまじない

        //格納用変数
        private string[] filelists;//選択されたファイル/フォルダ
        private string folderpath;//どのフォルダで右クリックされたか

        public void Initialize(IntPtr pidlFolder, IntPtr pDataObj, IntPtr hKeyProgID)
        {
            if (TargetFileType == "*")
            {
                // 選択されているファイル/ディレクトリを取得
                if (pDataObj == IntPtr.Zero && TargetFileType == "*")
                {
                    throw new ArgumentException();
                }
                DataObject dataObject = new DataObject(Marshal.GetObjectForIUnknown(pDataObj));
                if (dataObject.ContainsFileDropList())
                {
                    StringCollection files = dataObject.GetFileDropList();
                    filelists = new string[files.Count];
                    files.CopyTo(filelists, 0);
                }
            }
            else if (TargetFileType == @"Directory\Background")
            {
                //余白をクリックされたフォルダパスを取得
                if (pidlFolder == IntPtr.Zero)
                {
                    throw new ArgumentException();
                }
                else
                {
                    var buff = new StringBuilder(260);
                    NativeMethods.SHGetPathFromIDList(pidlFolder, buff);
                    folderpath = buff.ToString();
                }
            }
            else
            {
                throw new ArgumentException();
            }
        }

        private uint IDM_SHOW_FILENAME = 0;

        public int QueryContextMenu(IntPtr hMenu, uint iMenu, uint idCmdFirst, uint idCmdLast, uint uFlags)
        {
            // メニュー項目を追加
            if (((uint)CMF.CMF_DEFAULTONLY & uFlags) != 0)
            {
                return WinError.MAKE_HRESULT(WinError.SEVERITY_SUCCESS, 0, 0);
            }

            //メニュー表示対象の拡張子でなければ追加しない
            if (!IsShowMenu())
            {
                return WinError.MAKE_HRESULT(WinError.SEVERITY_SUCCESS, 0, 0);
            }

            MENUITEMINFO mii = new MENUITEMINFO();
            mii.cbSize = (uint)Marshal.SizeOf(mii);
            mii.fMask = MIIM.MIIM_ID | MIIM.MIIM_TYPE | MIIM.MIIM_STATE;
            mii.wID = idCmdFirst + IDM_SHOW_FILENAME;
            mii.fType = MFT.MFT_STRING;
            mii.dwTypeData = MENU_NAME;
            mii.fState = MFS.MFS_ENABLED;
            if (!NativeMethods.InsertMenuItem(hMenu, iMenu, true, ref mii))
            {
                return Marshal.GetHRForLastWin32Error();
            }

            // 追加したメニュー項目の数を返す
            return WinError.MAKE_HRESULT(WinError.SEVERITY_SUCCESS, 0, 1);
        }

        public void InvokeCommand(IntPtr pici)
        {
            CMINVOKECOMMANDINFO ici = (CMINVOKECOMMANDINFO)Marshal.PtrToStructure(pici, typeof(CMINVOKECOMMANDINFO));

            if (NativeMethods.HighWord(ici.lpVerb.ToInt32()) != 0)
            {
                return;
            }

            if (NativeMethods.LowWord(ici.lpVerb.ToInt32()) == IDM_SHOW_FILENAME)
            {
                if (TargetFileType == "*")
                {
                    if (filelists != null && filelists.Length > 0)
                    {
                        ActionMethod();
                    }
                }
                else if (TargetFileType == @"Directory\Background")
                {
                    if (folderpath != null && System.IO.Directory.Exists(folderpath))
                    {
                        ActionMethod();
                    }
                }
            }
            else
            {
                Marshal.ThrowExceptionForHR(WinError.E_FAIL);
            }
        }

        private string MSG_SHOW_FILENAME = "";

        public void GetCommandString(UIntPtr idCmd, uint uFlags, IntPtr pReserved, StringBuilder pszName, uint cchMax)
        {
            if (idCmd.ToUInt32() != IDM_SHOW_FILENAME)
            {
                return;
            }
            if ((GCS)uFlags == GCS.GCS_HELPTEXTW)
            {
                if (MSG_SHOW_FILENAME.Length > cchMax - 1)
                {
                    Marshal.ThrowExceptionForHR(WinError.STRSAFE_E_INSUFFICIENT_BUFFER);
                }
                else
                {
                    pszName.Clear();
                    pszName.Append(MSG_SHOW_FILENAME);
                }
            }
        }

        private const string Description = "ContextMenuExtention Class";

        [ComRegisterFunction()]
        public static void Register(Type t)
        {
            try
            {
                Guid clsid = t.GUID;
                if (clsid == Guid.Empty)
                {
                    throw new ArgumentException("clsid must not be empty");
                }
                string keyName = string.Format("{0}\\shellex\\ContextMenuHandlers\\{1}", TargetFileType, clsid.ToString("B"));
                using (RegistryKey key = Registry.ClassesRoot.CreateSubKey(keyName))
                {
                    if (key != null && !string.IsNullOrEmpty(Description))
                    {
                        key.SetValue(null, Description);
                    }
                }
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
                throw;
            }
        }

        [ComUnregisterFunction()]
        public static void Unregister(Type t)
        {
            try
            {
                Guid clsid = t.GUID;
                if (clsid == Guid.Empty)
                {
                    throw new ArgumentException("clsid must not be empty");
                }
                string keyName = string.Format("{0}\\shellex\\ContextMenuHandlers\\{1}", TargetFileType, clsid.ToString("B"));
                Registry.ClassesRoot.DeleteSubKeyTree(keyName, false);
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
                throw;
            }
        }
    }

    [ComImport(),
    InterfaceType(ComInterfaceType.InterfaceIsIUnknown),
    Guid("000214e8-0000-0000-c000-000000000046")]
    internal interface IShellExtInit
    {
        void Initialize(IntPtr /*LPCITEMIDLIST*/ pidlFolder, IntPtr /*LPDATAOBJECT*/ pDataObj, IntPtr /*HKEY*/ hKeyProgID);
    }

    [ComImport(),
    InterfaceType(ComInterfaceType.InterfaceIsIUnknown),
    Guid("000214e4-0000-0000-c000-000000000046")]
    internal interface IContextMenu
    {
        [PreserveSig]
        int QueryContextMenu(IntPtr /*HMENU*/ hMenu, uint iMenu, uint idCmdFirst, uint idCmdLast, uint uFlags);
        void InvokeCommand(IntPtr pici);
        void GetCommandString(UIntPtr idCmd, uint uFlags, IntPtr pReserved, StringBuilder pszName, uint cchMax);
    }

    internal static class WinError
    {
        public const int S_OK = 0x0000;
        public const int S_FALSE = 0x0001;
        public const int E_FAIL = -2147467259;
        public const int E_INVALIDARG = -2147024809;
        public const int E_OUTOFMEMORY = -2147024882;
        public const int STRSAFE_E_INSUFFICIENT_BUFFER = -2147024774;

        public const uint SEVERITY_SUCCESS = 0;
        public const uint SEVERITY_ERROR = 1;

        public static int MAKE_HRESULT(uint sev, uint fac, uint code)
        {
            return (int)((sev << 31) | (fac << 16) | code);
        }
    }

    [Flags]
    internal enum MIIM : uint
    {
        MIIM_STATE = 0x00000001,
        MIIM_ID = 0x00000002,
        MIIM_SUBMENU = 0x00000004,
        MIIM_CHECKMARKS = 0x00000008,
        MIIM_TYPE = 0x00000010,
        MIIM_DATA = 0x00000020,
        MIIM_STRING = 0x00000040,
        MIIM_BITMAP = 0x00000080,
        MIIM_FTYPE = 0x00000100
    }

    internal enum MFT : uint
    {
        MFT_STRING = 0x00000000,
        MFT_BITMAP = 0x00000004,
        MFT_MENUBARBREAK = 0x00000020,
        MFT_MENUBREAK = 0x00000040,
        MFT_OWNERDRAW = 0x00000100,
        MFT_RADIOCHECK = 0x00000200,
        MFT_SEPARATOR = 0x00000800,
        MFT_RIGHTORDER = 0x00002000,
        MFT_RIGHTJUSTIFY = 0x00004000
    }

    internal enum MFS : uint
    {
        MFS_ENABLED = 0x00000000,
        MFS_UNCHECKED = 0x00000000,
        MFS_UNHILITE = 0x00000000,
        MFS_GRAYED = 0x00000003,
        MFS_DISABLED = 0x00000003,
        MFS_CHECKED = 0x00000008,
        MFS_HILITE = 0x00000080,
        MFS_DEFAULT = 0x00001000
    }

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    internal struct MENUITEMINFO
    {
        public uint cbSize;
        public MIIM fMask;
        public MFT fType;
        public MFS fState;
        public uint wID;
        public IntPtr hSubMenu;
        public IntPtr hbmpChecked;
        public IntPtr hbmpUnchecked;
        public UIntPtr dwItemData;
        public string dwTypeData;
        public uint cch;
        public IntPtr hbmpItem;
    }

    [Flags]
    internal enum CMF : uint
    {
        CMF_NORMAL = 0x00000000,
        CMF_DEFAULTONLY = 0x00000001,
        CMF_VERBSONLY = 0x00000002,
        CMF_EXPLORE = 0x00000004,
        CMF_NOVERBS = 0x00000008,
        CMF_CANRENAME = 0x00000010,
        CMF_NODEFAULT = 0x00000020,
        CMF_INCLUDESTATIC = 0x00000040,
        CMF_ITEMMENU = 0x00000080,
        CMF_EXTENDEDVERBS = 0x00000100,
        CMF_DISABLEDVERBS = 0x00000200,
        CMF_ASYNCVERBSTATE = 0x00000400,
        CMF_OPTIMIZEFORINVOKE = 0x00000800,
        CMF_SYNCCASCADEMENU = 0x00001000,
        CMF_DONOTPICKDEFAULT = 0x00002000,
        CMF_RESERVED = 0xFFFF0000
    }

    [Flags]
    internal enum CMIC : uint
    {
        CMIC_MASK_ICON = 0x00000010,
        CMIC_MASK_HOTKEY = 0x00000020,
        CMIC_MASK_NOASYNC = 0x00000100,
        CMIC_MASK_FLAG_NO_UI = 0x00000400,
        CMIC_MASK_UNICODE = 0x00004000,
        CMIC_MASK_NO_CONSOLE = 0x00008000,
        CMIC_MASK_ASYNCOK = 0x00100000,
        CMIC_MASK_NOZONECHECKS = 0x00800000,
        CMIC_MASK_FLAG_LOG_USAGE = 0x04000000,
        CMIC_MASK_SHIFT_DOWN = 0x10000000,
        CMIC_MASK_PTINVOKE = 0x20000000,
        CMIC_MASK_CONTROL_DOWN = 0x40000000
    }

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    public struct POINT
    {
        public int X;
        public int Y;
    }

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    internal struct CMINVOKECOMMANDINFO
    {
        public uint cbSize;
        public CMIC fMask;
        public IntPtr hwnd;
        public IntPtr lpVerb;
        [MarshalAs(UnmanagedType.LPStr)]
        public string parameters;
        [MarshalAs(UnmanagedType.LPStr)]
        public string directory;
        public int nShow;
        public uint dwHotKey;
        public IntPtr hIcon;
    }
    internal enum GCS : uint
    {
        GCS_VERBA = 0x00000000,
        GCS_HELPTEXTA = 0x00000001,
        GCS_VALIDATEA = 0x00000002,
        GCS_VERBW = 0x00000004,
        GCS_HELPTEXTW = 0x00000005,
        GCS_VALIDATEW = 0x00000006,
        GCS_VERBICONW = 0x00000014,
        GCS_UNICODE = 0x00000004
    }

    internal class NativeMethods
    {
        [DllImport("user32", CharSet = CharSet.Unicode, SetLastError = true)]
        [return: MarshalAs(UnmanagedType.Bool)]
        public static extern bool InsertMenuItem(IntPtr hMenu, uint uItem, [MarshalAs(UnmanagedType.Bool)] bool fByPosition, ref MENUITEMINFO mii);

        [DllImport("shell32", CharSet = CharSet.Unicode)]
        [return: MarshalAs(UnmanagedType.Bool)]
        public static extern bool SHGetPathFromIDList(IntPtr pidlFolder, [MarshalAs(UnmanagedType.LPWStr), Out] StringBuilder lpPathName);

        public static int HighWord(int number)
        {
            return ((number & 0x80000000) == 0x80000000) ? (number >> 16) : ((number >> 16) & 0xffff);
        }

        public static int LowWord(int number)
        {
            return number & 0xffff;
        }
    }
}

参考URL

全般

フォルダ

その他

4
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?