您的位置:首页 > 其它

删除任务栏锁定,任务栏图标的位置,查找可行性文件的方式

2015-07-09 10:16 399 查看
这几天改bug,遇到一个删除任务栏快捷图标的问题。本来想着应该挺简单的,估计是在注册表或者C盘的某个位置放置着一个链接,一删除就可以搞定。没想到没那么简单。

C:\Users\LeoLi\AppData\Roaming\Microsoft\Internet Explorer\Quick Launch\User Pinned\TaskBar这个目录下放的就是任务栏中的快捷方式。我之前以为只要把这个地方的lnk删除了,就相当于去掉了任务栏快捷图标,但是结果还是存在,只是你点击的时候系统提示该链接已失效。看来这个方法是行不通的。

网上去搜资料,发现有篇介绍任务栏列表的文章:http://www.codeproject.com/Articles/36561/Windows-Goodies-in-C-Jump-Lists。这篇文章不是介绍如何删除任务栏锁定,而且介绍了为什么我们在任务栏的图标上右键的时候,会看到很多额外的信息,都是程序自己加上去的。所以可以适当的了解下。然后其实了解了这块内容,删除任务栏锁定就不是什么难题了。代码如下:

bool UnLockWinTaskBarLink( const String runPath )
{
HINSTANCE hModule = ::LoadLibrary(_T("Shell32.dll")) ;
if (hModule == NULL)
{
return false ;
}
typedef HRESULT (__stdcall* SHCreateItemFromParsingName)(PCWSTR pszPath, IBindCtx *pbc, REFIID riid, void **ppv) ;
SHCreateItemFromParsingName mySHCreateItemFromParsingName = (SHCreateItemFromParsingName)GetProcAddress(hModule, "SHCreateItemFr        omParsingName") ;
if (mySHCreateItemFromParsingName == NULL)
{
return false ;
}
bool result = true ;
do
{
CComPtr<IShellItem> spItem ;
HRESULT hr = mySHCreateItemFromParsingName(
runPath.c_str(), NULL, IID_PPV_ARGS(&spItem)) ;
if (FAILED(hr))
{
result = false ;
break ;
}
CComPtr<IStartMenuPinnedList> spStartMenuPinnedList ;
hr = CoCreateInstance(CLSID_StartMenuPin, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&spStartMenuPinnedList)) ;
if (FAILED(hr))
{
result = false ;
break ;
}
hr = spStartMenuPinnedList->RemoveFromList(spItem) ;
if (FAILED(hr))
{
result = false ;
}
} while (false) ;
FreeLibrary(hModule) ;
return result;
}
代码的逻辑应该是不用多说,就是利用Shell32.dll的导出函数SHCreateItemFromParsingName,系统对每个任务栏锁定都相应的记录,通过这个程序的可执行文件的位置,我们可以得到一个操作这个程序的“标识”,便于去删除。

其实我多次看到Shell.dll的调用,这是一个很有用的dll,提供了很多操作系统的方法,我网上找了下这方面的总结,比较少,等以后用的多的再继续更新。

现在的问题似乎解决了,但是不要忘记,这个函数需要一个参数,也就是被删除任务栏锁定的可执行文件的位置。这儿可能很多人都说,直接获取当前路径不就完了吗?有现成的API。但问题是,如果要你删除的不是你自己的程序呢,随便给一个程序,让你解除任务栏绑定,你怎么去找到这个程序的exe的位置呢?

我之前的思路是系统肯定知道所有exe的位置,那注册表里有记录,可以直接找了半天没什么收获(其实确实有的,稍后说)。那最直接的办法就是全盘搜索,耗时巨大。一般来说程序的安装目录不会有太多层,我们遍历个7,8层应该都是可以检索出来的。而且还可以优先遍历某几个很有可能的文件夹,这样的设计显然不好。于是就想了个歪点子,我们能找到快捷链接的位置,右击快捷方式不是可以看到目标位置吗?那有没有办法获取到快捷方式的信息呢?答案是:有的。代码如下:

BOOL GetIEQuickLaunchPath(TCHAR *pszIEQueickLaunchPath)
{
LPITEMIDLIST ppidl;
if (SHGetSpecialFolderLocation(NULL, CSIDL_APPDATA, &ppidl) == S_OK)
{
BOOL flag = SHGetPathFromIDList(ppidl, pszIEQueickLaunchPath);
::wcscat_s(pszIEQueickLaunchPath, MAX_PATH, _T("\\Microsoft\\Internet Explorer\\Quick Launch"));
CoTaskMemFree(ppidl);
return flag;
}
return FALSE;
}

bool SearchExePathByLnk(String StrTargetFile, WCHAR* szDesEXEPath)
{
::CoInitialize(NULL); //初始化COM接口
IShellLink *psl = NULL;

HRESULT hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, (LPVOID *)&psl);
if (SUCCEEDED(hr))
{
IPersistFile *ppf;
hr = psl->QueryInterface(IID_IPersistFile, (LPVOID*)&ppf);
if (SUCCEEDED(hr))
{
hr = ppf->Load(StrTargetFile.c_str(), STGM_READ);    //加载文件
if (SUCCEEDED(hr))
{
WIN32_FIND_DATA wfd;
psl->GetPath(szDesEXEPath, MAX_PATH, (WIN32_FIND_DATA*)&wfd, SLGP_SHORTPATH);  //获取目标路径
return true;
}
ppf->Release();
}
psl->Release();  //释放对象
}
::CoUninitialize();   //释放COM接口

return false;
}

String StrTargetFile;
PIDLIST_ABSOLUTE pid;
TCHAR szDocument[MAX_PATH + 1] = { 0 };
WCHAR szDesEXEPath[MAX_PATH] = L"";

GetIEQuickLaunchPath(szDocument);
StrTargetFile.append(szDocument);
StrTargetFile.append(_T("\\User Pinned\\TaskBar\\"));
StrTargetFile.append(_T("酷我音乐.lnk"));

SearchExePathByLnk(StrTargetFile, szDesEXEPath);
UnLockWinTaskBarLink(RCString(szDesEXEPath));
DeleteFile(StrTargetFile.c_str());


上面的代码中,通过lnk查找可执行文件的位置是参考的是另一位博友的博客:点击打开链接 /article/8733587.html 。只要任务栏有锁定,那么%AppData%Roaming\Microsoft\Internet Explorer\Quick Launch\User
Pinned\TaskBar中就一定有这个快捷方式,那这样的话,问题就解决了。

当然这不是什么好方法,后来我找到了注册表里关于信息的记录。在一个项名为Uninstall的下面,这个项有两个地方,不固定某一个,所以需要都检索一下,具体位置是:

HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall ;

HKEY_LOCAL_MACHINE, _T("Software\\Wow6432Node\\Microsoft\\Windows\\CurrentVersion\\Uninstall 。

找到这个地方,剩下的就是读取注册表了,这个应该就非常简单了吧?
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: