当前位置:首页 > c++ > 正文内容

c++创建win32无边框窗口

xuwenyan2年前 (2022-01-07)c++2932

无边框窗口

没有系统默认标题栏和默认边框的窗口,可以根据我们的需要实现完全自绘,大多数界面库都是使用无边框窗口进行自绘控件的。

默认窗口效果和无边框窗口效果对比

默认win32窗口



无边框窗口



创建无边框窗口

创建一个默认的win32窗口,然后处理一下 WM_NCCALCSIZE 消息,就可以成为无边框窗口了,代码如下:

case WM_NCCALCSIZE:
  return 0;
break;

但是如果仅仅处理 WM_NCCALCSIZE 消息的话会有一些问题,1是无法拖动无法调整大小,2是窗口默认边框和默认标题栏会自动跑出来,下面就来解决这个两个问题。

解决系统默认边框和默认标题栏自动出现的问题

当我们截图窗口时,就会出现这样的问题,如下:



需要处理的消息是 WM_NCACTIVATE ,代码如下:

case WM_NCACTIVATE:
  return TRUE;
break;

解决无法拖动和调整大小的问题

需要处理的消息是 WM_NCHITTEST ,代码如下:

case WM_NCHITTEST:{
    POINT pt = { (int)(short)LOWORD(lParam) ,(int)(short)HIWORD(lParam) };

    const POINT border{
        ::GetSystemMetrics(SM_CXFRAME) + ::GetSystemMetrics(SM_CXPADDEDBORDER),
        ::GetSystemMetrics(SM_CYFRAME) + ::GetSystemMetrics(SM_CXPADDEDBORDER)
    };

    RECT window_rc = { 0,0,0,0 };
    ::GetWindowRect(hWnd, &window_rc);

    enum region_mask {
      client = 0,
      left = 1 << 0,
      right = 1 << 1,
      top = 1 << 2,
      bottom = 1 << 3
    };

    const auto result =
      left * (pt.x < (window_rc.left + border.x)) |
      right * (pt.x >= (window_rc.right - border.x)) |
      top * (pt.y < (window_rc.top + border.y)) |
      bottom * (pt.y >= (window_rc.bottom - border.y));

    switch (result) {
    case left: return HTLEFT;
    case right: return HTRIGHT;
    case top: return HTTOP;
    case bottom: return HTBOTTOM;
    case top | left: return HTTOPLEFT;
    case top | right: return HTTOPRIGHT;
    case bottom | left: return HTBOTTOMLEFT;
    case bottom | right: return HTBOTTOMRIGHT;
    }

    const int caption = ::GetSystemMetrics(SM_CYCAPTION);
    if (pt.y <= window_rc.top + caption)
      return HTCAPTION;
}
break;

解决拖动调整大小时抖动的问题

解决完上面问题,还有一个小优化需要处理一下,当我们拖动窗口边缘调整大小时,我们会发现窗口右边缘和下边缘随着拖动一直在抖动,体验非常不好。处理方法就是在 WM_NCCALCSIZE 消息中将客户区减小1像素,留出1像素的边框,这样就可以很好的解决抖动问题,需要注意的是这个方法对于透明窗口是无效的,目前透明窗口没有找到很好的解决办法,欢迎大家提供思路。代码如下:

case WM_NCCALCSIZE:{
    // 解决调整大小时抖动问题(透明窗口无效)
    if (wParam == TRUE) {
      auto& params = *reinterpret_cast<NCCALCSIZE_PARAMS*>(lParam);
      RECT& rect = params.rgrc[0];
      rect.top += 1;
    }

    return 0;
}
break;

完整demo代码

// test_win32_window.cpp : 定义应用程序的入口点。
//

#include "stdafx.h"
#include "test_win32_window.h"
#include <ShObjIdl.h>

#define MAX_LOADSTRING 100

// 全局变量:
HINSTANCE hInst;								// 当前实例
TCHAR szTitle[MAX_LOADSTRING];					// 标题栏文本
TCHAR szWindowClass[MAX_LOADSTRING];			// 主窗口类名


// 此代码模块中包含的函数的前向声明:
ATOM				MyRegisterClass(HINSTANCE hInstance);
BOOL				InitInstance(HINSTANCE, int);
LRESULT CALLBACK	WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK	About(HWND, UINT, WPARAM, LPARAM);

ULONG_PTR m_gdiplusToken;

int APIENTRY _tWinMain(HINSTANCE hInstance,
  HINSTANCE hPrevInstance,
  LPTSTR    lpCmdLine,
  int       nCmdShow) {
  UNREFERENCED_PARAMETER(hPrevInstance);
  UNREFERENCED_PARAMETER(lpCmdLine);
  ::CoInitialize(NULL);

  // TODO: 在此放置代码。
  MSG msg;
  HACCEL hAccelTable;

  // 初始化全局字符串
  LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
  LoadString(hInstance, IDC_TEST_WIN32_WINDOW, szWindowClass, MAX_LOADSTRING);
  MyRegisterClass(hInstance);

  // 执行应用程序初始化:
  if (!InitInstance(hInstance, nCmdShow)) {
    return FALSE;
  }

  hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_TEST_WIN32_WINDOW));

  // 主消息循环:
  while (GetMessage(&msg, NULL, 0, 0)) {
    if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) {
      TranslateMessage(&msg);
      DispatchMessage(&msg);
    }
  }

  ::CoUninitialize();

  return (int)msg.wParam;
}

//
//  函数: MyRegisterClass()
//
//  目的: 注册窗口类。
//
//  注释:
//
//    仅当希望
//    此代码与添加到 Windows 95 中的“RegisterClassEx”
//    函数之前的 Win32 系统兼容时,才需要此函数及其用法。调用此函数十分重要,
//    这样应用程序就可以获得关联的
//    “格式正确的”小图标。
//
ATOM MyRegisterClass(HINSTANCE hInstance) {
  WNDCLASSEX wcex;

  wcex.cbSize = sizeof(WNDCLASSEX);

  wcex.style = CS_HREDRAW | CS_VREDRAW;
  wcex.lpfnWndProc = WndProc;
  wcex.cbClsExtra = 0;
  wcex.cbWndExtra = 0;
  wcex.hInstance = hInstance;
  wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_TEST_WIN32_WINDOW));
  wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
  wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
  wcex.lpszMenuName = MAKEINTRESOURCE(IDC_TEST_WIN32_WINDOW);
  wcex.lpszClassName = szWindowClass;
  wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));

  return RegisterClassEx(&wcex);
}

//
//   函数: InitInstance(HINSTANCE, int)
//
//   目的: 保存实例句柄并创建主窗口
//
//   注释:
//
//        在此函数中,我们在全局变量中保存实例句柄并
//        创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow) {
  HWND hWnd = NULL;

  hInst = hInstance; // 将实例句柄存储在全局变量中

  hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
    CW_USEDEFAULT, 0, 640, 480, NULL, NULL, hInstance, NULL);

  if (!hWnd) {
    return FALSE;
  }

  ShowWindow(hWnd, nCmdShow);
  UpdateWindow(hWnd);

  return TRUE;
}

bool Maximized(HWND hwnd) {
  WINDOWPLACEMENT placement = { 0 };
  if (!::GetWindowPlacement(hwnd, &placement)) {
    return false;
  }

  return placement.showCmd == SW_MAXIMIZE;
}

//
//  函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
//  目的: 处理主窗口的消息。
//
//  WM_COMMAND	- 处理应用程序菜单
//  WM_PAINT	- 绘制主窗口
//  WM_DESTROY	- 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) {
  int wmId, wmEvent;
  PAINTSTRUCT ps;
  HDC hdc;

  switch (message) {
  case WM_COMMAND:
    wmId = LOWORD(wParam);
    wmEvent = HIWORD(wParam);
    // 分析菜单选择:
    switch (wmId) {
    case IDM_ABOUT:
      DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
      break;
    case IDM_EXIT:
      DestroyWindow(hWnd);
      break;
    default:
      return DefWindowProc(hWnd, message, wParam, lParam);
    }
    break;
  case WM_PAINT:
  {
    hdc = BeginPaint(hWnd, &ps);
    // TODO: 在此添加任意绘图代码...
    //RECT rc = { 100,200,300,220 };
    //::DrawText(hdc, L"hello word", -1, &rc, 0);

    EndPaint(hWnd, &ps);
  }
  break;
  case WM_DESTROY:
    PostQuitMessage(0);
    break;
  case WM_NCHITTEST:
  {
    POINT pt = { (int)(short)LOWORD(lParam) ,(int)(short)HIWORD(lParam) };

    const POINT border{
        ::GetSystemMetrics(SM_CXFRAME) + ::GetSystemMetrics(SM_CXPADDEDBORDER),
        ::GetSystemMetrics(SM_CYFRAME) + ::GetSystemMetrics(SM_CXPADDEDBORDER)
    };

    RECT window_rc = { 0,0,0,0 };
    ::GetWindowRect(hWnd, &window_rc);

    enum region_mask {
      client = 0,
      left = 1 << 0,
      right = 1 << 1,
      top = 1 << 2,
      bottom = 1 << 3
    };

    const auto result =
      left * (pt.x < (window_rc.left + border.x)) |
      right * (pt.x >= (window_rc.right - border.x)) |
      top * (pt.y < (window_rc.top + border.y)) |
      bottom * (pt.y >= (window_rc.bottom - border.y));

    switch (result) {
    case left: return HTLEFT;
    case right: return HTRIGHT;
    case top: return HTTOP;
    case bottom: return HTBOTTOM;
    case top | left: return HTTOPLEFT;
    case top | right: return HTTOPRIGHT;
    case bottom | left: return HTBOTTOMLEFT;
    case bottom | right: return HTBOTTOMRIGHT;
    }

    const int caption = ::GetSystemMetrics(SM_CYCAPTION);
    if (pt.y <= window_rc.top + caption)
      return HTCAPTION;
  }
  break;
  case WM_NCCALCSIZE:
  {
    if (wParam == TRUE) {
      auto& params = *reinterpret_cast<NCCALCSIZE_PARAMS*>(lParam);
      RECT& rect = params.rgrc[0];
      rect.top += 1;
    }

    return 0;
  }
  break;

  case WM_NCACTIVATE:
    return TRUE;
    break;

  default:
    return DefWindowProc(hWnd, message, wParam, lParam);
  }
  return 0;
}

// “关于”框的消息处理程序。
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam) {
  UNREFERENCED_PARAMETER(lParam);
  switch (message) {
  case WM_INITDIALOG:
    return (INT_PTR)TRUE;

  case WM_COMMAND:
    if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) {
      EndDialog(hDlg, LOWORD(wParam));
      return (INT_PTR)TRUE;
    }
    break;
  }
  return (INT_PTR)FALSE;
}
    文章作者:xuwenyan
    版权声明:本文为本站原创文章,转载请注明出处,非常感谢,如版权漏申明或您觉得任何有异议的地方欢迎与本站取得联系。

    扫描二维码推送至手机访问。

    版权声明:本文由艺文笔记发布,如需转载请注明出处。

    本文链接:https://www.xuwenyan.com/archives/2362

    分享给朋友:

    “c++创建win32无边框窗口” 的相关文章

    C++从dll导出lib

    C++从dll导出lib

    一、使用VC++的工具DUMPBIN将DLL中的导出函数表导出到一定义(.DEF)文件EXAMPLE: DUMPBIN VideoDeCoder.dll /EXPROTS /OUT:VideoDeCoder.def 二、将导出的.DEF文件整理为一符合.DEF个数的函数导出文件EX...

    C++如何获取控制台程序的输出内容?

    C++如何获取控制台程序的输出内容?

    很多工具程序(如ffmpeg)的进度显示往往都是以控制台字符显示的方法,我们可能需要调用这种控制台工具去完成工作,但同时又希望以友好的ui界面去显示当前的工作状态(如进度)。此时我们能想到的就是运行控制台程序,然后以某种方式去获取到控制台程序的输出,然后转换到我们的ui界面上去显示。 有多种...

    VC的ATL工程向导同时生成一个PS工程是做什么的?可以不要吗?

    VC的ATL工程向导同时生成一个PS工程是做什么的?可以不要吗?

    例如,我用VC2015的工程向导新建一个ATL的工程名字叫myAtl,那么VC会同时给我生成一个叫做myAtlPS的工程。这个myAtlPS工程是做什么的?什么情况下可以不需要它?什么情况下它又是必须存在的? PS工程是什么?可以不要吗? 这个PS工程叫做代理与存根(proxy&nbs...

    解决程序在xp系统总是莫名奇妙的崩溃问题(/Zc:threadSafeInit- )

    解决程序在xp系统总是莫名奇妙的崩溃问题(/Zc:threadSafeInit- )

    现象:程序在xp系统上面总是莫名其妙的崩溃,检查代码看不出任何问题,感觉代码都很好。即使你远程调试,找到了崩溃的点,当你注释了崩溃点之后,还是会崩溃到别的地方。当你遇到了这种情况的时候,不妨参照一下下面的方法看看,说不定可以解决问题。如何解决?将崩溃程序相关的所有工程代码全部关闭全局变量的线程安全检...

    uafxcwd.lib(afxmem.obj) : error LNK2005:

    uafxcwd.lib(afxmem.obj) : error LNK2005: "void * __cdecl operator new(unsigned int)"解决办法

    如果在编译MFC程序的时候出现下列及类似的错误: 1˃uafxcwd.lib(afxmem.obj) : error LNK2005: "void * __cdecl operator new(unsigned int)" (??2@YAPAXI@Z) 已经在 LIBCMTD.lib(new...

    c++函数模板参数类型限定

    c++函数模板参数类型限定

    函数模板函数模板可以实现对不同数据类型做统一操作,比如比较两个数据的大小:template<typename T> bool compare(T& a,T& b) {   return a...