转载:http://blog.csdn.net/rmxming/article/details/11661365

对于我们这些控件狂来说,窗口阴影也是一个必不可少的实现需求。虽说其没多大用,但对于增加窗口立体感来说,那是挺有帮助的。

我实现了一个类似于360界面的阴影效果,其可以支持正常窗口,也支持半透明窗口。

阴影窗口对于正常窗口和半透明窗口,有区别么?且让我慢慢写来:)

阴影窗口的实现原理,简单来讲:就是在主窗口创建时,创建一个子窗口,吸附于主窗口的底部。然后在子窗口上做一个带半透明阴影效果的描绘。

以下代码是阴影窗口在父窗口的创建代码,是不是很简单?

  1. LRESULT OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/)
  2. {
  3. m_Shadow.Create(m_hWnd);
  4. m_Shadow.SetShadowSize(8);
  5. return TRUE;
  6. }

下面是阴影窗口实现步骤:

1. 在阴影窗口创建时,只设定阴影窗口的样式为WS_VISIBLE,我们这里不能用WS_CHILD,否则阴影窗口就跑到主窗口里面去了。

  1. // Create shadow window.
  2. HWND Create(const HWND wndParent)
  3. {
  4. ATLASSERT( ::IsWindow(wndParent) );
  5. m_hParentWnd = wndParent;
  6. CRect rc(1, 1, 1, 1);
  7. return CWindowImpl<CThemedShadowWnd, CWindow, CControlWinTraits>::Create(0, rc, NULL, WS_VISIBLE, NULL);
  8. }

2. 在阴影窗口执行WM_CREATE消息时,修改其样式为WS_EX_LAYERED | WS_EX_TRANSPARENT,注意这两个样式都要要。WS_EX_TRANSPARENT是让窗口无法接收点击消息,你总不想你的窗口阴影可以被用户点击且激活吧:)

  1. SetWindowLong(GWL_EXSTYLE, GetWindowLong(GWL_EXSTYLE) | WS_EX_LAYERED | WS_EX_TRANSPARENT);
  2. ModifyStyleEx(WS_EX_TOPMOST, WS_EX_NOACTIVATE);

3. 与此同时,阴影窗口注册父窗口的消息处理回调函数,此举是为了获取父窗口的移动、重绘和隐藏等重要消息。因为阴影窗口要跟随着父窗口的状态改变而改变。

  1. // Set parent window original processing.
  2. m_OriParentProc = ::GetWindowLong(m_hParentWnd, GWL_WNDPROC);
  3. ::SetWindowLong(m_hParentWnd, GWL_WNDPROC, (LONG)ParentProc);

回调函数要做的事情很简单,吸附于父窗口之下,像个小尾巴一样:

  1. // Get parent message.
  2. static LRESULT CALLBACK ParentProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
  3. {
  4. // Find the shadow window pointer via parent window handle.
  5. ATLASSERT( m_szShadowWindows.find(hwnd) != m_szShadowWindows.end() );
  6. CThemedShadowWnd *pThis = m_szShadowWindows[hwnd];
  7. WNDPROC pDefProc        = (WNDPROC)pThis->m_OriParentProc;
  8. switch(uMsg)
  9. {
  10. case WM_ERASEBKGND:
  11. case WM_PAINT:
  12. case WM_MOVE:
  13. case WM_ACTIVATE:
  14. case WM_NCACTIVATE:
  15. {
  16. if (::IsWindowVisible(hwnd))
  17. {
  18. pThis->AdjustWindowPos();
  19. }
  20. break;
  21. }
  22. case WM_DESTROY:
  23. {
  24. // Destroy the shadow window.
  25. pThis->DestroyWindow();
  26. break;
  27. }
  28. case WM_NCDESTROY:
  29. {
  30. // Remove shadow window from map.
  31. m_szShadowWindows.erase(hwnd);
  32. break;
  33. }
  34. case WM_SHOWWINDOW:
  35. {
  36. // the window is being hidden
  37. if (!wParam)
  38. {
  39. pThis->ShowWindow(SW_HIDE);
  40. }
  41. else
  42. {
  43. pThis->ShowWindow(SW_SHOW);
  44. }
  45. break;
  46. }
  47. default:
  48. {
  49. break;
  50. }
  51. }
  52. return pDefProc(hwnd, uMsg, wParam, lParam);
  53. }

好了,窗口消息机制处理完了,就要处理阴影画法了,我这里用的是GDI+的画法,如果有童鞋觉得效果不够好,可以尝试多改改参数配置,以达到理想效果:

  1. // Create shadow brush.
  2. PathGradientBrush brShadow(m_ShadowPath.m_pPath);
  3. Color clrShadow[3] = {Color::Transparent, Color(255, 0, 0, 0), Color(255, 0, 0, 0)};
  4. int nCount = 3;
  5. REAL szPos[3] = {0.0F, 0.05F, 1.0F};
  6. brShadow.SetInterpolationColors(clrShadow, szPos, nCount);
  7. // Draw shadow.
  8. rcShadow.Width  = rcShadow.Width - m_nShadowSize - m_nBlankArea;
  9. rcShadow.Height = rcShadow.Height - m_nShadowSize - m_nBlankArea;
  10. graphics.ExcludeClip(rcShadow);
  11. graphics.FillPath(&brShadow, m_ShadowPath.m_pPath);

注意我这里排除了一部分的阴影部分,那是为透明窗口制作的,排除的效果图如下,阴影窗口只显示在矩形的右下角,而其他地方是透明的。


如果我不排除一部分阴影区域,那么透明的窗口效果将变得很难看,如下图,透明背景被阴影遮盖了,这显然不符合美学的要求。

如果你的窗口的角是椭圆的,你可能还需要增宽阴影的显示区域,那么可以用如下函数进行阴影的宽度增长:

  1. // Set blank area right position.
  2. void SetRightOffsetArea(const int nRightPos)
  3. {
  4. m_nBlankArea = nRightPos;
  5. if (nRightPos < 0)
  6. {
  7. m_nBlankArea = 1;
  8. }
  9. }

阴影窗口免费实例代码下载:http://download.csdn.net/detail/renstarone/6267677

VC++界面编程之--阴影窗口的实现详解的更多相关文章

  1. Android 之窗口小部件详解(三)  部分转载

    原文地址:http://blog.csdn.net/iefreer/article/details/4626274. (一) 应用程序窗口小部件App Widgets 应用程序窗口小部件(Widget ...

  2. Android 之窗口小部件详解--App Widget

    Android 之窗口小部件详解--App Widget  版本号 说明 作者 日期  1.0  添加App Widge介绍和示例  Sky Wang 2013/06/27        1 App ...

  3. 转载 CSS3 经典教程系列:CSS3 盒阴影(box-shadow)详解

    目标大纲 文章转载 CSS3 经典教程系列:CSS3 盒阴影(box-shadow)详解 IE中CSS-filter滤镜小知识大全 CSS实现跨浏览器兼容性的盒阴影效果

  4. [置顶] VC++界面编程之--使用分层窗口实现界面皮肤

    使用分层界面来实现界面皮肤的好处是:可以保证图片边缘处理不失真,且能用于异形窗口上,如一些不规则的窗口,你很难用SetWindowRgn来达到理想效果. 在很多情况下,界面的漂亮与否,取决于PS的制作 ...

  5. VC++界面编程之--使用分层窗口实现界面皮肤

    使用分层界面来实现界面皮肤的好处是:可以保证图片边缘处理不失真,且能用于异形窗口上,如一些不规则的窗口,你很难用SetWindowRgn来达到理想效果. 在很多情况下,界面的漂亮与否,取决于PS的制作 ...

  6. VC/MFC中的CComboBox控件使用详解

    CComboBox控件详解 CComboBox控件又称作组合框控件,其有三种形态可供选择,1.简单组合框(Simple)2.下拉组合框(Drop-down)3.下拉列表式组合框(Drop-down l ...

  7. VMware 虚拟化编程(14) — VDDK 的高级传输模式详解

    目录 目录 前文列表 虚拟磁盘数据的传输方式 Transport Methods Local File Access NBD and NBDSSL Transport SAN Transport Ho ...

  8. 网页打开新窗口——Window.open()详解

    转载自:http://blog.csdn.net/business122/article/details/8281142 Window.Open详解 一.window.open()支持环境:JavaS ...

  9. Java 异步编程 (5 种异步实现方式详解)

    ​ 同步操作如果遇到一个耗时的方法,需要阻塞等待,那么我们有没有办法解决呢?让它异步执行,下面我会详解异步及实现@mikechen 目录 什么是异步? 一.线程异步 二.Future异步 三.Comp ...

随机推荐

  1. Codeforces Round #288 (Div. 2)

    A. Pasha and Pixels     题意就是给一个n*m的矩阵,k次操作,一开始矩阵全白,一次操作可以染黑一个格子,问第几次操作可以使得矩阵中存在一个2*2的黑色矩阵.直接模拟即可 代码: ...

  2. 《Focus On 3D Terrain Programming》中一段代码的注释一

    取自<Focus On 3D Terrain Programming>中的一段: //--------------------------------------------------- ...

  3. opencv载入,显示及保存图像

    1.声明一个表示图像的变量,在OpenCV2中,这个变量是cv::Mat类型,该类是用于保存图像以及其他矩阵数据的数据结构.默认情况下它们的尺寸为0. cv::Mat  image;       // ...

  4. 【SPFA】 最短路计数

    最短路计数 [问题描述]   给出一个N个顶点M条边的无向无权图,顶点编号为1-N.问从顶点1开始,到其他每个点的最短路有几条. [输入格式]   输入第一行包含2个正整数N,M,为图的顶点数与边数. ...

  5. android 应用架构随笔一(架构搭建)

    1.拷贝积累utils以及PagerTab类 2.定义BaseApplication类 3.定义BaseActivity类 4.改写MainActivity 5.定义布局文件 6.定义BaseFrag ...

  6. Mongodb 笔记07 分片、配置分片、选择片键、分片管理

    分片 1. 分片(sharding)是指将数据拆分,将其分散存放在不同的机器上的过程.有时也用分区(partitioning)来表示这个概念.将数据分散到不同的机器上,不需要功能强大的大型计算机就可以 ...

  7. 鸟哥的linux私房菜学习记录之账号管理与权限设定

    每个登录者都会取到两个ID,一个使用者ID,一个群组ID

  8. USB HID描述符【转】

    本文转载自: USB是个通用的总线,端口都是统一的.但是USB设备却各种各样,例如USB鼠标,USB键盘,U盘等等,那么USB主机是如何识别出不同的设备的呢?这就要依赖于描述符了.USB的描述符主要有 ...

  9. GPS模块的AT指令集

    AT+CPIN? 查询sim卡状态. 接着: AT+CGCLASS="B"AT+CGDCONT=1,"IP","CMNET"AT+CGATT ...

  10. JS 动态加载脚本 执行回调_转

    关于在javascript里面加载其它的js文件的问题可能很多人都遇到过,但很多朋友可能并不知道怎么判断我们要加载的js文件是否加载完成,如果没有加载完成我们就调用文件里面的函数是不会成功的.本文讲解 ...