• Shell扩展:定制上下文菜单


    如若需要获取某个选定文件的完整路径,小弟经常机械性地先复制Shell窗口中的路径,紧接着复制文件名并拼在路径之后.有时候复制文件路径是为了程序使用,这就必须将路径中的所有"\"换成"\\",总之一切都是很无聊的操作.还好我是个程序员,完全可以定制一些程序来方便自己.在此分享给大家.

    程序实现的功能很明确:在Shell的上下文菜单中加入一菜单项目"获取文件路径并保存到剪贴板",点击此项可以将选中的一个或多个文件的完整路径保存到剪贴板中.多个文件路径之间以换行"\r\n"间隔.若需要获取的路径是程序格式("\"换成"\\"),则可在Ctrl键按下的状态下单击该菜单项.

    实现:定制Shell的菜单项,需实现IContextMenu接口,同时也需要实现IShellExtInit接口来完成初始化的工作.

          首先定义一些需要使用的宏:

    //菜单ID
    #define  ID_COPY_PATH  0

    //用于剪贴板格式
    #ifdef _UNICODE
    #define CF_TEXT_FORMAT        CF_UNICODETEXT
    #else
    #define CF_TEXT_FORMAT        CF_TEXT
    #endif

          定义一数组保存选中的文件(夹)列表,定义如下:

    CAtlArray<CString>  m_arrFilePath;

          IShellExtInit接口就一个Initialize方法,在这里用于显示上下文菜单之前的初始化工作.我在实现中将当前选中的文件(夹)列表保存到m_arrFilePath数组,代码如下:

    HRESULT STDMETHODCALLTYPE Initialize( LPCITEMIDLIST pidlFolder, IDataObject *pdtobj, HKEY hkeyProgID)
    {
        m_arrFilePath.RemoveAll();

        
    //文件列表
        if( pdtobj != NULL ) 
        {
            STGMEDIUM medium 
    = { 0 };
            FORMATETC fe 
    = { CF_HDROP, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
            
    if( SUCCEEDED( pdtobj->GetData(&fe, &medium) ) ) 
            {
                HDROP hDrop 
    = (HDROP) ::GlobalLock(medium.hGlobal);
                UINT uCount 
    = ::DragQueryFile( hDrop, (UINT) -1, NULL, 0 );
                
    for( UINT uIndex = 0; uIndex < uCount; uIndex++ ) 
                {
                    TCHAR szFileName[MAX_PATH] 
    = { 0 };
                    ::DragQueryFile(hDrop, uIndex, szFileName, (
    sizeof(szFileName) / sizeof(TCHAR)) - 1);
                    
                    
    //szFileName为文件(夹)名
                    m_arrFilePath.Add( szFileName );
                }
                ::GlobalUnlock(medium.hGlobal);
                ::ReleaseStgMedium(
    &medium);
            }
        }    
        
    return S_OK;
    }

    IContextMenu接口则是实现上下文菜单的主体,其有三个成员方法要实现:

    QueryContextMenu方法:可以用来添加自己的菜单项,实现如下: 

    HRESULT STDMETHODCALLTYPE QueryContextMenu( HMENU hmenu, UINT indexMenu,    UINT idCmdFirst, UINT idCmdLast,    UINT uFlags)
    {
        CString strMenuText;
        strMenuText.Format( _T(
    "获取文件路径并保存到剪贴板") );
        ::InsertMenu(hmenu, indexMenu
    ++, MF_STRING | MF_BYPOSITION,    idCmdFirst + ID_COPY_PATH, strMenuText);
        
    return ResultFromScode(MAKE_SCODE(SEVERITY_SUCCESS, FACILITY_NULL, (USHORT)(ID_COPY_PATH + 1)));
    }
    InvokeCommand方法:用来处理菜单项的命令:
    HRESULT STDMETHODCALLTYPE InvokeCommand( LPCMINVOKECOMMANDINFO lpici )
    {
        
    //Ctrl键按下时拷贝的路径的"\"用"\\"代替,这样可以直接应用在代码中
        BOOL  bCtrlPress = ( ( ::GetKeyState( VK_CONTROL ) & 0x8000 ) != 0 );

        
    if( LOWORD( lpici->lpVerb ) == ID_COPY_PATH )
        {    
    //如果是拷贝文件路径菜单项
            CString  strClipboard(_T(""));
            
    for ( int nIndex = 0; nIndex <  (int)m_arrFilePath.GetCount(); nIndex++ )
            {    
    //遍历选中的文件(夹)列表
                CString  strItem = m_arrFilePath.GetAt( nIndex );
                
    if( bCtrlPress )
                {
                    
    //获取路径的代码格式
                    strItem.Replace( _T("\\"), _T("\\\\") );
                }
                
    //添加到总串并以换行结束
                strClipboard.Append( strItem );
                strClipboard.Append( _T(
    "\r\n") );
            }

            
    //拷贝进剪贴板
            if( ::OpenClipboard( NULL ) )
            {
                ::EmptyClipboard();
                HGLOBAL hGlobal 
    = GlobalAlloc( GPTR, ( strClipboard.GetLength() + 1 ) * sizeof( TCHAR ) );
                LPTSTR lpszText 
    = (LPTSTR)::GlobalLock( hGlobal );
                
    if( lpszText != NULL )
                {
                    _tcscpy_s( lpszText, strClipboard.GetLength() 
    + 1, strClipboard );
                }
                ::SetClipboardData( CF_TEXT_FORMAT, hGlobal);
                ::GlobalUnlock( hGlobal );
                ::CloseClipboard();
            }
        }
        
    return S_OK;
    }

    GetCommandString方法:在此处用不到,简单地返回E_NOTIMPL即可.

    程序实现已完结,最后一步就是注册,需要在注册表的HKEY_CLASSES_ROOT\*\ShellEx\ContextMenuHandlers\键一自己的键,并把键值设为刚刚编写的COM的CLSID.

    由于是用ATL实现,我只需在RGS文件的HKCR键下加上如下脚本:

        NoRemove *
        {
            NoRemove ShellEx
            {
                NoRemove ContextMenuHandlers
                {
                    ForceRemove 
    'GetFilePath' =  s '{AAD8C1A8-017E-44B3-8271-DFBA4CD8E75C}'
                }
            }
        }

    AAD8C1A8-017E-44B3-8271-DFBA4CD8E75C是我的CLSID.

    一切都已完成。希望大家愉快.

    PS: 我不知道博客园能不能上传附件,如果有需要源码的朋友可以留下Email。

  • 相关阅读:
    android -- eclipse 环境搭建
    android AVD坑(1) -- no images system installed for this target
    android AVD坑(1) -- no images system installed for this target
    简单的轮播
    C# 使用NPOI 处理Excel(Datable与Excel相互转换)
    C盘突然爆满
    IIS中 flv、swf 文件无法播放
    webservice 尝试加载 Oracle 客户端库时引发 BadImageFormatException。如果在安装 32 位 Oracle 客户端组件的情况下运行,将出现此问题
    window server 2008 安装Oracle10g
    ora-01033 oracle initialization or
  • 原文地址:https://www.cnblogs.com/fangkm/p/1432932.html
Copyright © 2020-2023  润新知