使用默认的CMenu菜单类或者继承CMenu实现的菜单扩展类,在显示的时候最外层都会有边框出现,或者说是具有3D外观(菜单阴影不算),当改变菜单背景色或者需要加个边框线时就会看上去很不美观。看过很多菜单的自定义实现类,一般可以有两种方式来实现外框的移除。
       第一种方法就是:自定义窗口,完全模拟菜单的实现,自给自足,倒是能够完全满足开发需要,不过实现的复杂让人头痛,此处略过不提。

下面介绍第二种比较简单直接的方法:安装钩子,在菜单创建时就改变其窗口属性。其实菜单应该也算是一个窗口类,不过实在是无从得知到底在哪创建的窗口,所以下下钩子,过程倒是明了许多。
 实现如下:

先在cpp前面申明一下:

static HHOOK g_hook=NULL; // 全局钩子
        static LRESULT WINAPI CallWndProc(int, WPARAM, LPARAM); // 安装的钩子的窗口过程
        static LRESULT WINAPI MenuWndProc(HWND, UINT, WPARAM, LPARAM); // 用来处理菜单的窗口过程

然后是钩子的实现:

/////////////////////////////////////////////////////////////////////////////
// 如果需要去除菜单的外部边框,需要通过安装钩子,设置外框属性并改变菜单大小
WNDPROC oldWndProc = NULL; // 用来保存被替换的窗口过程
LRESULT WINAPI CallWndProc(int code, WPARAM wParam, LPARAM lParam)
{
 CWPSTRUCT* pStruct = (CWPSTRUCT*)lParam;
 while (code == HC_ACTION)
 {
  HWND hWnd = pStruct->hwnd;
  // 捕捉创建消息WM_CREATE,后面筛选为是否是菜单的创建
  if ( pStruct->message != WM_CREATE)
   break;
  TCHAR sClassName[10];
  int Count = ::GetClassName(hWnd, sClassName, sizeof(sClassName)/sizeof(sClassName[0]));
  // 检查是否菜单窗口,#32768为菜单类名
  if ( Count != 6 ||  _tcscmp(sClassName, _T("#32768")) != 0 )   
   break;
  
  WNDPROC lastWndProc = (WNDPROC)GetWindowLong(hWnd, GWL_WNDPROC);   
  if (lastWndProc != MenuWndProc)   
  {   
   // 替换菜单窗口过程  
   SetWindowLong(hWnd, GWL_WNDPROC, (long)MenuWndProc);   
   // 保留原有的窗口过程   
   oldWndProc = lastWndProc;   
  }
  break;   
 }

return CallNextHookEx((HHOOK)WH_CALLWNDPROC, code, wParam, lParam); 
}

// 处理菜单的窗口过程
LRESULT WINAPI MenuWndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{   
 LRESULT lResult;
 switch (message)   
 {      
 case WM_CREATE:
  {   
   // 首先要去掉菜单窗口的一些扩展风格
   // 包括:WS_BORDER、WS_EX_DLGMODALFRAME、WS_EX_WINDOWEDGE
   lResult = CallWindowProc(oldWndProc, hWnd, message, wParam, lParam);   
   DWORD dwStyle = ::GetWindowLong(hWnd,   GWL_STYLE);   
   DWORD dwNewStyle = (dwStyle & ~WS_BORDER);
   ::SetWindowLong(hWnd, GWL_STYLE, dwNewStyle);   
   DWORD dwExStyle = ::GetWindowLong(hWnd, GWL_EXSTYLE);
   DWORD dwNewExStyle = (dwExStyle & ~(WS_EX_DLGMODALFRAME | WS_EX_WINDOWEDGE));   
   ::SetWindowLong(hWnd, GWL_EXSTYLE, dwNewExStyle);   
   return lResult; 
  }   
 case   WM_PRINT: // 此处阻止非客户区地绘制
   return CallWindowProc( oldWndProc, hWnd, WM_PRINTCLIENT, wParam, lParam);

case   WM_WINDOWPOSCHANGING:   
  {   
   // 最后,由于我们在MeasureItem里指定了菜单大小,而系统会自动替菜单加边框,
   // 因此必须去掉此部分额外地尺寸,将菜单大小改小
   LPWINDOWPOS lpPos = (LPWINDOWPOS)lParam;   
   lpPos->cx -= 2*GetSystemMetrics(SM_CXBORDER)+4;   
   lpPos->cy -= 2*GetSystemMetrics(SM_CYBORDER)+4;
   lResult = CallWindowProc(oldWndProc, hWnd, message, wParam, lParam);  
   return 0;
  }   
 case   WM_GETICON:
  return 0;    
 default:   
  return  CallWindowProc( oldWndProc, hWnd, message, wParam, lParam);   
 }   
}

/////////////////////////////////////////////////////////////

最后是调用:
下钩子需要调用SetWindowsHookEx,参数包括主窗口的实例句柄theApp.m_hInstance,可以在调用时作为参数传入,同时也设一个参数作为是否安装钩子的标识,以便在退出时判断是否需要卸载钩子(UnhookWindowsHookEx(g_hook))。示例如下:

void CSkinMenu::RemoveMenuBorder(HINSTANCE hInst, BOOL bRemove /* = TRUE */)
{
 m_bRemoveBorder = bRemove; // 标识

// 需要移除边框时,要安装钩子
 if (m_bRemoveBorder)
 {
  DWORD id = ::GetCurrentThreadId(); // 获取当前线程的ID
  g_hook = SetWindowsHookEx(WH_CALLWNDPROC,CallWndProc,hInst,id);
 }
}

这样子,就搞定菜单的边框了,最后要记得,如果安装了钩子,需要卸载掉。

【转载】CMenu自绘---钩子---去除边框的更多相关文章

  1. 【Android】Android EditText 去除边框

    [Android]Android EditText 去除边框 将EditText属性设置修改 android:background="@null" //////////////// ...

  2. Android Listview 去除边框

    最近在做一个时间轴的功能,因为原生效果的Listview有item分隔边框,所以就需要去除边框,调用listview的setDivider方法就可以了: listView.setDivider(nul ...

  3. WPF去除边框的方法

    原文:WPF去除边框的方法 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/yangsen600/article/details/81978125 W ...

  4. QT自绘标题和边框

    在QT中如果想要自绘标题和边框,一般步骤是: 1) 在创建窗口前设置Qt::FramelessWindowHint标志,设置该标志后会创建一个无标题.无边框的窗口. 2)在客户区域的顶部创建一个自绘标 ...

  5. golang-imagick图片去除边框(黑边,白边,),添加中文字

    图片去除边框(黑边,白边) package main import ( "fmt" "github.com/gographics/imagick/imagick" ...

  6. (webapp)微信和safri 对于html5 部分功能不兼容,多选或单选下拉框去除边框无效果。

    1 appearance:none; 2 -moz-appearance:none; /* Firefox */ 3 -webkit-appearance:none; /* Safari 和 Chro ...

  7. [css小技巧]input去除边框问题

    border:none;是不够的 (1)在谷歌浏览器添加 outline: none;去除点击后产生的边框; (2)IE7下border: none;还会有边框存在,改用border: 0;即可,同时 ...

  8. winform重绘控件边框

    首先添加一个用户控件 对于重绘边框有三个需要考虑的东西 1:是否显示边框 2:边框颜色 3:边框宽度 所以定义三个私有变量 /// <summary>/// 是否显示边框/// </ ...

  9. 去除input边框 input去除边框 去除input获取焦点时的蓝色外边框

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

随机推荐

  1. bzoj千题计划185:bzoj1260: [CQOI2007]涂色paint

    http://www.lydsy.com/JudgeOnline/problem.php?id=1260 区间DP模型 dp[l][r] 表示涂完区间[l,r]所需的最少次数 从小到大们枚举区间[l, ...

  2. Java SSM框架之MyBatis3(七)MyBatis之参数取值

    在mybatis中,参数取值方式有两种:#{ } 和 ${ } 一.#{ } select * from student where name=#{name} 编译后执行的sql语句: select ...

  3. 一个简单的二叉搜索树(C++实现)

    参考:http://www.cnblogs.com/skywang12345/p/3576373.html 这里主要就是自己实现的代码,删除动作有点不一样: #ifndef __BSTREE_H__ ...

  4. 从零开始编写自己的JavaScript框架(一)

    1. 模块的定义和加载 1.1 模块的定义 一个框架想要能支撑较大的应用,首先要考虑怎么做模块化.有了内核和模块加载系统,外围的模块就可以一个一个增加.不同的JavaScript框架,实现模块化方式各 ...

  5. js和jquery使按钮失效为不可用状态的方法

    设置disabled属性为true即为不可用状态. html代码: <input type="button" value="提交" id="bt ...

  6. c++刷题(43/100)矩阵旋转打印

    题目1:矩阵旋转打印 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则 ...

  7. 洛谷 P1563 玩具谜题

    如果你想不耗费脑力做出这个题目,往下看: 本萌新看到这个题目,想到了乘法法则,题目中左右方向要判断两次,很耗脑力,和乘法中的正负号判断非常像. 抽象一点:这个人向内向外就是乘法中括号外的正负号,他的左 ...

  8. spring的普通类中获取session和request对像

    在使用spring时,经常需要在普通类中获取session,request等对像. 1.第一钟方式,针对Spring和Struts2集成的项目: 在有使用struts2时,因为struts2有一个接口 ...

  9. mybatis延迟加载——(十二)

    1.     什么是延迟加载 resultMap可以实现高级映射(使用association.collection实现一对一及一对多映射),association.collection具备延迟加载功能 ...

  10. Servlet笔记10--Session

    Web编程中的Session: 代码示例: package com.bjpowernode.javaweb.servlet; import java.io.IOException; import ja ...