对Windows桌面任务栏自动隐藏功能的一点小小改进 --- 不再自动弹出

陪她去流浪 桃子 2013年01月09日 阅读次数:5064

由于是笔记本电脑, 由于屏幕太小, 也可能是不想让别人看见自己打开了哪些窗口, 所以选择了把桌面的任务栏的自动隐藏选项给钩上了, 现在只要任务栏失去焦点, 它就会自己隐藏了. 但很不爽的是, 任务栏在屏幕的最下面(其它位置也一样), 只要鼠标稍微碰到就会显示出任务栏, 太不爽了, 所以打算进行一点小的修改, 于是就有了下面的一小段代码...

当然, 采用了最简单的办法, 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做测试, 完美运行.

项目下载: HookTaskBar.7z

标签:代码片段 · WinAPI · hook · 个性化