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

c++创建win32无边框窗口

xuwenyan11个月前 (01-07)c++899

无边框窗口

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

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

默认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++创建快捷方式(lnk、url)并添加到开始菜单、桌面、任务栏

    C++创建快捷方式(lnk、url)并添加到开始菜单、桌面、任务栏

    windows快捷方式有lnk和url两种,其中lnk以打开应用程序为主,而url以打开网页链接为主。下面创建快捷方式的方法均是调用Com接口,所以一定要在调用此方法前初始化com接口(CoInitialize)。创建lnk快捷方式#include <ShlObj.h>...

    C++如何实现挂起进程、恢复进程

    C++如何实现挂起进程、恢复进程

    1:枚举进程的所有线程,使用SuspendThread函数挂起每一个线程,需要恢复时使用ResumeThread函数恢复。因为挂起和恢复的顺序是不可预知的,所以可能会导致一些多线程程序崩溃,单线程程序可能不受影响。void SuspendProcess(DWORD process...

    C++实现win32窗口文件拖拽

    C++实现win32窗口文件拖拽

    如题,C++如何实现win32窗口文件拖拽,直接上代码 方法1:使用win32消息实现 此方法的弊端在于,无法过滤可以拖拽的文件,拖拽时显示的图标也默认的,无法像资源管理器那样自定义。实现步骤大致分为以下三步: 第一步:首先创建窗口时必须在exStyle加上WS_EX_ACCEP...

    C++如何实现远程注入dll

    C++如何实现远程注入dll

    如何把我们的代码放到别人的进程里面运行?我们需要做一个dll动态库,然后使用远程注入技术,将我们的dll注入到别人的进程里面,然后加载起来。这样我们的代码就可以在别人的进程里面工作了。注入代码#include "stdafx.h" int EnableD...

    使用GDI、MFC_GDI、GDI+绘制数组RGBA序列

    使用GDI、MFC_GDI、GDI+绘制数组RGBA序列

    学习ffmpeg时遇到一个问题,ffmpeg解码出RGB颜色后怎么绘制到屏幕上,于是将GDI、MFC_GDI、GDI+等方式都记录一下 1:注意按windows的要求,R、G、B、A顺序要调整为B、G、R、A 。 2:GDI不支持透明通道A,透明通道A的值读进去以后没有作用。想要支持透...

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

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

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