Win32 C++ 实现对话框居中显示

2024-08-27 21:12

本文主要是介绍Win32 C++ 实现对话框居中显示,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        使用 MessageBox 对话框显示信息时, 对话框位置总是在屏幕中间, 而不是主窗口的中间, 如何以最简单的方式将对话框移到父窗口中间呢?  那就是使用 CBT 钩子 , 在窗口创建完成前(窗口句柄已经创建完成), 修改窗口的位置, 即可实现对话框在父窗口上居中显示.

首先简单写一个 CBT 钩子类, 这个类暂且叫做 CMessageBoxCenter

MessageBoxCenter.h

#pragma once
#include <windows.h>class CMessageBoxCenter
{public:CMessageBoxCenter();~CMessageBoxCenter();// 获取子窗口位于父窗口的居中位置static POINT GetChildWindowCenterPos(int nWidth,         //子窗口宽度int nHeight,        //子窗口高度HWND hParent        //父窗口句柄);private:// WH_CBT // 线程或全局// https://learn.microsoft.com/zh-cn/windows/win32/winmsg/cbtproc// 安装用于接收对 CBT 应用程序有用的通知的挂钩过程static LRESULT CALLBACK CBTProc(int nCode, WPARAM wParam, LPARAM lParam);private:static HHOOK m_hHook;
};

MessageBoxCenter.cpp

#include "MessageBoxCenter.h"HHOOK CMessageBoxCenter::m_hHook;CMessageBoxCenter::CMessageBoxCenter()
{m_hHook = ::SetWindowsHookEx(WH_CBT, CBTProc, NULL, ::GetCurrentThreadId());
}CMessageBoxCenter::~CMessageBoxCenter()
{::UnhookWindowsHookEx(m_hHook);
}POINT CMessageBoxCenter::GetChildWindowCenterPos(int nWidth,         //子窗口宽度int nHeight,        //子窗口高度HWND hParent        //父窗口句柄
)
{RECT rectParent = { 0 };LONG nParentW = 0;LONG nParentH = 0;// 如果父窗口句柄为空, 则以桌面窗口为父窗口if (NULL == hParent){hParent = ::GetDesktopWindow();}// 获取并统计父窗口宽度和高度::GetWindowRect(hParent, &rectParent);nParentW = rectParent.right - rectParent.left;nParentH = rectParent.bottom - rectParent.top;return {rectParent.left + (nParentW - nWidth) / 2, rectParent.top + (nParentH - nHeight) / 2};
}LRESULT CALLBACK CMessageBoxCenter::CBTProc(int nCode, WPARAM wParam, LPARAM lParam)
{// 即将创建一个窗口if (HCBT_CREATEWND == nCode){HWND hWndNew = (HWND)wParam;LPCBT_CREATEWND pStruct = (LPCBT_CREATEWND)lParam;HWND hParent = pStruct->lpcs->hwndParent;// 检查是对话框, 则修改显示位置为相对父窗口居中位置if (32770 == ::GetClassLongPtr(hWndNew, GCW_ATOM)){POINT nPos = GetChildWindowCenterPos(pStruct->lpcs->cx, pStruct->lpcs->cy, hParent);pStruct->lpcs->x = nPos.x;pStruct->lpcs->y = nPos.y;}}return ::CallNextHookEx(m_hHook, nCode, wParam, lParam);
}

以下测试例子简单写了个对话框过程, 仅简单测试对话框

main.cpp

#include <windows.h>
#include <tchar.h>
#include <locale.h>
#include"resource.h"
#include "MessageBoxCenter.h"#pragma comment(linker,"\"/manifestdependency:type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"")INT_PTR WINAPI DialogProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam);int WINAPI WinMain(_In_ HINSTANCE hInstance,_In_opt_ HINSTANCE hPrevInstance,_In_ LPSTR lpCmdLine,_In_ int nShowCmd)
{setlocale(LC_ALL, "");UNREFERENCED_PARAMETER(hInstance);UNREFERENCED_PARAMETER(hPrevInstance);UNREFERENCED_PARAMETER(lpCmdLine);UNREFERENCED_PARAMETER(nShowCmd);WORD wID = ::DialogBoxParam(GetModuleHandle(NULL),  MAKEINTRESOURCE(IDD_DIALOG_FRAME), nullptr, DialogProc, (LPARAM)0);return wID;
}INT_PTR WINAPI DialogProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{if (WM_INITDIALOG == uMsg){return (INT_PTR)TRUE;}if (WM_CLOSE == uMsg){::EndDialog(hWnd, 0);return (INT_PTR)TRUE;}if (WM_COMMAND == uMsg){//消息源   HIWORD(wParam)      LOWORD(wParam)     lParam //菜单     0                   菜单ID             0//快捷键   1                   快捷键ID           0//控件     控件定义的通知码    控件ID             控件窗口句柄WORD wNotify = HIWORD(wParam);WORD wID = LOWORD(wParam);HWND hWndCtrl = (HWND)lParam;if (wID >= IDOK && wID <= IDNO){::EndDialog(hWnd, wID);}if (IDC_BUTTON_CBT == wID){CMessageBoxCenter boxCenter;MessageBox(hWnd, _T("WH_CBT 钩子测试, 居中显示对话框"), _T("WH_CBT 钩子"), MB_OK);}return (INT_PTR)TRUE;}return (INT_PTR)FALSE;
}

resource.h

//{{NO_DEPENDENCIES}}
// Microsoft Visual C++ 生成的包含文件。
// 供 Win32Hook.rc 使用
//
#define IDD_DIALOG_FRAME                101
#define IDC_BUTTON_CBT                  1001// Next default values for new objects
// 
#ifdef APSTUDIO_INVOKED
#ifndef APSTUDIO_READONLY_SYMBOLS
#define _APS_NEXT_RESOURCE_VALUE        103
#define _APS_NEXT_COMMAND_VALUE         40001
#define _APS_NEXT_CONTROL_VALUE         1002
#define _APS_NEXT_SYMED_VALUE           101
#endif
#endif

Win32Hook.rc

// Microsoft Visual C++ generated resource script.
//
#include "resource.h"#define APSTUDIO_READONLY_SYMBOLS
/
//
// Generated from the TEXTINCLUDE 2 resource.
//
#include "winres.h"/
#undef APSTUDIO_READONLY_SYMBOLS/
// 中文(简体,中国) resources#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_CHS)
LANGUAGE LANG_CHINESE, SUBLANG_CHINESE_SIMPLIFIED#ifdef APSTUDIO_INVOKED
/
//
// TEXTINCLUDE
//1 TEXTINCLUDE 
BEGIN"resource.h\0"
END2 TEXTINCLUDE 
BEGIN"#include ""winres.h""\r\n""\0"
END3 TEXTINCLUDE 
BEGIN"\r\n""\0"
END#endif    // APSTUDIO_INVOKED/
//
// Dialog
//IDD_DIALOG_FRAME DIALOGEX 0, 0, 309, 176
STYLE DS_SETFONT | DS_MODALFRAME | DS_FIXEDSYS | DS_CENTER | WS_POPUP | WS_CAPTION | WS_SYSMENU
CAPTION "WinHook"
FONT 8, "MS Shell Dlg", 400, 0, 0x1
BEGINDEFPUSHBUTTON   "确定",IDOK,73,142,50,14PUSHBUTTON      "取消",IDCANCEL,181,141,50,14PUSHBUTTON      "WH_CBT 居中显示对话框",IDC_BUTTON_CBT,19,18,91,14
END/
//
// DESIGNINFO
//#ifdef APSTUDIO_INVOKED
GUIDELINES DESIGNINFO
BEGINIDD_DIALOG_FRAME, DIALOGBEGINLEFTMARGIN, 7RIGHTMARGIN, 302TOPMARGIN, 7BOTTOMMARGIN, 169END
END
#endif    // APSTUDIO_INVOKED/
//
// AFX_DIALOG_LAYOUT
//IDD_DIALOG_FRAME AFX_DIALOG_LAYOUT
BEGIN0
END#endif    // 中文(简体,中国) resources
/#ifndef APSTUDIO_INVOKED
/
//
// Generated from the TEXTINCLUDE 3 resource.
///
#endif    // not APSTUDIO_INVOKED

对话框效果

未使用钩子效果

使用钩子后效果:

这篇关于Win32 C++ 实现对话框居中显示的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1112797

相关文章

MyBatis-Plus逻辑删除实现过程

《MyBatis-Plus逻辑删除实现过程》本文介绍了MyBatis-Plus如何实现逻辑删除功能,包括自动填充字段、配置与实现步骤、常见应用场景,并展示了如何使用remove方法进行逻辑删除,逻辑删... 目录1. 逻辑删除的必要性编程1.1 逻辑删除的定义1.2 逻辑删php除的优点1.3 适用场景2.

C#借助Spire.XLS for .NET实现在Excel中添加文档属性

《C#借助Spire.XLSfor.NET实现在Excel中添加文档属性》在日常的数据处理和项目管理中,Excel文档扮演着举足轻重的角色,本文将深入探讨如何在C#中借助强大的第三方库Spire.... 目录为什么需要程序化添加Excel文档属性使用Spire.XLS for .NET库实现文档属性管理Sp

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

Python+FFmpeg实现视频自动化处理的完整指南

《Python+FFmpeg实现视频自动化处理的完整指南》本文总结了一套在Python中使用subprocess.run调用FFmpeg进行视频自动化处理的解决方案,涵盖了跨平台硬件加速、中间素材处理... 目录一、 跨平台硬件加速:统一接口设计1. 核心映射逻辑2. python 实现代码二、 中间素材处

Java数组动态扩容的实现示例

《Java数组动态扩容的实现示例》本文主要介绍了Java数组动态扩容的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1 问题2 方法3 结语1 问题实现动态的给数组添加元素效果,实现对数组扩容,原始数组使用静态分配

Python实现快速扫描目标主机的开放端口和服务

《Python实现快速扫描目标主机的开放端口和服务》这篇文章主要为大家详细介绍了如何使用Python编写一个功能强大的端口扫描器脚本,实现快速扫描目标主机的开放端口和服务,感兴趣的小伙伴可以了解下... 目录功能介绍场景应用1. 网络安全审计2. 系统管理维护3. 网络故障排查4. 合规性检查报错处理1.

Python轻松实现Word到Markdown的转换

《Python轻松实现Word到Markdown的转换》在文档管理、内容发布等场景中,将Word转换为Markdown格式是常见需求,本文将介绍如何使用FreeSpire.DocforPython实现... 目录一、工具简介二、核心转换实现1. 基础单文件转换2. 批量转换Word文件三、工具特性分析优点局

Springboot3统一返回类设计全过程(从问题到实现)

《Springboot3统一返回类设计全过程(从问题到实现)》文章介绍了如何在SpringBoot3中设计一个统一返回类,以实现前后端接口返回格式的一致性,该类包含状态码、描述信息、业务数据和时间戳,... 目录Spring Boot 3 统一返回类设计:从问题到实现一、核心需求:统一返回类要解决什么问题?

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造