我对Windows桌面任务栏自动隐藏功能的一点小小改进--不再自动弹出(2013-02-26更新)
tfref 由于是笔记本电脑, 由于屏幕太小, 也可能是不想让别人看见自己打开了哪些窗口, 所以选择了把桌面的任务栏的自动隐藏选项给钩上了, 现在只要任务栏失去焦点, 它就会自己隐藏了. 但很不爽的是, 任务栏在屏幕的最下面(其它位置也一样), 只要鼠标稍微碰到就会显示出任务栏, 太不爽了, 所以打算进行一点小的修改, 于是就有了下面的一小段代码...
当然, 采用了最简单的办法, DLL注入, Explorer.exe 在启动的时候会默认加载system32目录下的msimg32.dll, 由于这个dll导出的函数非常少, 所以我就把它"替换"成我自己的了.
如果想试试效果的, 可以按如下方式使用:
1.把下面的程序生成的msimg32.dll放到Explorer.exe的目录下(C:\Windows), 可能需要先在任务管理器中结束Explorer.exe桌面进程.
2.然后从命令提示符,或任务管理器的文件->新建任务(运行...)中执行explorer命令即可重新启动桌面进程.
最终的效果:
桌面的任务栏会自己隐藏(自己设置的,右键菜单->属性->自动隐藏任务栏), 然后鼠标放到任务栏上它也不会再出来了
要想再显示出任务栏的方式:
1.按住左边的Ctrl键(可能会不方便, 还是用第3种方法好), 再把鼠标移到任务栏上面去就可以了.
2.当前台窗口是桌面的时候, 任务栏会显示(2013-01-10)
3.用鼠标左键单击一下任务栏(在屏幕的最下边).(2013-01-11)
主要实现代码:
//当鼠标在NC区移动时会产生这个消息 if(uMsg == WM_NCHITTEST){ HWND hForeground = GetForegroundWindow(); if(hForeground!=hDesktopWindow && //前台窗口不是桌面 hForeground!=NULL && hForeground!=hTaskBar &&//前台窗口不是任务栏 !(GetAsyncKeyState(VK_LCONTROL)&0x8000)) //左Ctrl未按下 { return 0; } }
好了, 几句代码而已, 不过功能倒是基本实现了, 没什么多说的.
下面是代码, 最后有项目下载:
#include <windows.h> #include <process.h> //导出msing32.dll原来的所有函数 #pragma comment(linker, "/EXPORT:vSetDdrawflag=_AheadLib_vSetDdrawflag,@1") #pragma comment(linker, "/EXPORT:AlphaBlend=_AheadLib_AlphaBlend,@2") #pragma comment(linker, "/EXPORT:DllInitialize=_AheadLib_DllInitialize,@3") #pragma comment(linker, "/EXPORT:GradientFill=_AheadLib_GradientFill,@4") #pragma comment(linker, "/EXPORT:TransparentBlt=_AheadLib_TransparentBlt,@5") //裸代码 #define NAKED __declspec(naked) //原来的msimg32模块 HMODULE hModMsimg32 = NULL; //加载原模块 void Load(void) { char msimg32_path[MAX_PATH]; GetSystemDirectory(msimg32_path, sizeof(msimg32_path)); strcat(msimg32_path, "\\msimg32.dll"); hModMsimg32 = LoadLibrary(msimg32_path); if(hModMsimg32 == NULL){ MessageBoxW(NULL, L"无法加载原始的 msimg32.dll, 请移除该修改程序 !", NULL, MB_OK); ExitProcess(1); } } //取得原模块中的函数地址 FARPROC WINAPI GetAddress(char* ProcName) { FARPROC pAddress = NULL; pAddress = GetProcAddress(hModMsimg32, ProcName); if(pAddress == NULL){ MessageBox(NULL, "获取函数地址失败!", ProcName, MB_ICONERROR); ExitProcess(1); } return pAddress; } //卸载原模块 void Unload(void) { if(hModMsimg32){ FreeLibrary(hModMsimg32); hModMsimg32 = NULL; } } //任务条原来的窗口函数地址 WNDPROC OldWindowProc = NULL; //在桌面的时候可以响应 HWND hDesktopWindow = NULL; //子类后的窗口函数地址 LRESULT CALLBACK NewWinProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) {
//2013-02-26更新一个hForeground!=hTaskBar的判断 //当鼠标在NC区移动时会产生这个消息
if(uMsg == WM_NCHITTEST){
HWND hForeground = GetForegroundWindow();
if(hForeground!=hDesktopWindow && //前台窗口不是桌面
hForeground!=NULL && hForeground!=hTaskBar &&//前台窗口不是任务栏
!(GetAsyncKeyState(VK_LCONTROL)&0x8000)) //左Ctrl未按下
{
return 0;
}
} //其它的消息全部忽略 return CallWindowProc(OldWindowProc, hwnd, uMsg, wParam, lParam); } //临时用的新线程, 主要是保证DLL能正确加载 unsigned int WINAPI ThreadProc(PVOID pv) { HWND hShellTrayWnd = NULL; //DLL加载时窗口还没被创建,所以要等等 while(!(hShellTrayWnd=FindWindow("Shell_TrayWnd", NULL))) Sleep(100); //取得桌面窗口 while(!(hDesktopWindow = FindWindow("Progman", "Program Manager"))) Sleep(100); //更改窗口函数地址为我的窗口函数地址,处理自己想处理的消息 OldWindowProc = (WNDPROC)SetWindowLong(hShellTrayWnd, GWL_WNDPROC, (LONG)NewWinProc); return 0; } BOOL WINAPI DllMain(HANDLE hDllHandle, DWORD dwReason, LPVOID lpReserved) { if(dwReason == DLL_PROCESS_ATTACH){ DisableThreadLibraryCalls(hDllHandle); Load(); _beginthreadex(NULL, 0, ThreadProc, NULL, 0, NULL); }else if(dwReason == DLL_PROCESS_DETACH){ Unload(); } return TRUE; } //直接转到原来msimg32.dll中的函数调用 NAKED AheadLib_vSetDdrawflag(void) { GetAddress("vSetDdrawflag"); __asm jmp eax; } NAKED AheadLib_AlphaBlend(void) { GetAddress("AlphaBlend"); __asm jmp eax; } NAKED AheadLib_DllInitialize(void) { GetAddress("DllInitialize"); __asm jmp eax; } NAKED AheadLib_GradientFill(void) { GetAddress("GradientFill"); __asm jmp eax; } NAKED AheadLib_TransparentBlt(void) { GetAddress("TransparentBlt"); __asm jmp eax; }
最后需要声明一点:
由于msimg32本身是系统文件(位于C:\Windows\System32目录下), 所以在使用, 被安全软件报病毒, 系统文件被替换之类的是很正常的, 自己知道没问题就行了.
我的系统是XPSP3, 没试过Win7,Win8,不知道能否工作~
2013-03-14 更新:
已在Win7做测试, 完美运行.
项目下载:https://files.cnblogs.com/nbsofer/HookTaskBar.7z
女孩不哭(QQ:191035066)@2013-01-09 20:27:41 http://www.cnblogs.com/nbsofer