在非纯色背景上,叠加背景透明的BUTTON和STATIC_TEXT控件

例子:我们要把这个十字准星叠到一个BUTTON控件上,BUTTON位于一个非纯色的背景的窗口上。显示时,紫红色(RGB 0xFF00FF)区域要被挖去,而显示底下的背景。资源图片如下图

实现:
1、按照以前的经验,如果这个Button由我们自己CreateWindowEx建立起来的话,只要在RegisterClass的WNDCLASS结构体里,把HBRUSH设置成GetStockObject(NULL_BRUSH),并且在WM_PAINT消息里首先做SetBkMode(TRANSPARENT)就可以了。但由于现在走了GWES和eVC里RESOURCE文件结合的方法提高开发速度,所以我们不能回到自己CreateWindow的原始时代了。应该考虑在MS制定的GWES游戏规则里跳舞。

2、首先实现图片的去色。实际上很容易,从网上的资料看,以前还得做个单色“掩色”hdc, 然后怎么怎么处理。而WINCE已经把这个功能做好了,提供函数TransparentBlt。
现在我画上去的按钮成这样了:

问题很明显,图像本身是达到去色的目的了,但是按钮的背景仍然是灰色刷子刷出来的,必须把按钮背景变成透明的。

2、首先我想到在WM_DRAWITEM消息里做SekBkMode,没有效果。然后非常想把已注册的WNDCLASS改掉,于是找到函数SetClassLong和GetClassLong,并且INDEX参数给出了GCL_HBRBACKGROUND类型。一切看起来都很美好。于是我写了这样的代码
HBRUSH hbr = GetStockObject(NULL_BRUSH);
int nReturn = SetWindowLong(hWnd, GCL_HBRBACKGROUND, (int)hbr);
结果编译不让过,说没有GCL_HBRBACKGROUND这个东东。我查了下STANDARDSDK_500给出的头文件,里面的确没有GCL_HBRBACKGROUND的定义。于是干脆翻出VS2005在XP下的SDK,硬是找到了GCL_HBRBACKGROUND = (-10). 好,我就强制
SetWindowLong(hWnd, -10, (int)hbr)
结果编译过去了,运行起来nReturn = 0. 伤心啊。再试一把,
HBRUSH hbr = GetWindowLong(hwnd, -10)
运行结果hbr = NULL. 伤心啊,拔凉拔凉的啊。仔细看WINCE产品文档,虽然对用表格列出了一大堆nIndex,但是下面小字Remarks写了“The only values supported for the nIndex parameter are GCL_HICON and GCL_STYLE”,最多再加个GCL_HCURSOR,其他的参数类型WINCE都不支持。我日,浪费感情。

3、在父窗口的PROC里抓住WM_CTLCOLORBTN消息,在里面搞点鬼
case WM_CTLCOLORBTN:
{
 HBRUSH hbr = (HBRUSH)GetStockObject(NULL_BRUSH);
 SetBkMode( HDC(wParam), TRANSPARENT );
 return (int)hbr;
}
郁闷了,父窗口不画这个区域,BUTTON自己又不擦背景,桌面的字都透出来了。像下图这样:


然后我画了个把小时搞InvalidateRect, ValidateRect, 对着那点小地方折腾半天,父窗口还是不愿意画上子窗口的区域,真像开工作会议时的扯皮,不是本部门负责的区域,坚决不做,打死也不做。这就是为什么文章标题强调“非纯色背景”了。如果是纯色背景,我在这里设置好HBRUSH的颜色就搞定了,麻烦就麻烦在为了透明而用NULL_BRUSH才出了问题。

4、吃过饭,灵光了,想到个很容易的方法。昨天经理刚批评我不踏实做事,喜欢找捷径。嘿嘿这次又被我抄个小捷径了:我先前在代码里已经保留了父窗口背景的hdcOffScreen,现在把子窗口区域的部分拿过来,先画到子窗口上,然后再把子窗口的十字准星叠加上去。实际上,前面一直想的是如何去除背景色,这篇文章的标题也这样写,而实际上背景色根本没被去掉,只是上面又覆盖了一层父窗口的背景图而已,逆向思维。

case WM_DRAWITEM:
{
 LPDRAWITEMSTRUCT lpDIS = (LPDRAWITEMSTRUCT)lParam;
 if( lpDIS->CtlID = IDC_CROSS )
 {
  BitBlt( lpDIS->hDC,  //这个是BUTTON窗口的hdc device,而不是background的,所以我们从{0,0}画起
            0,
     0,
     button.nWidth, //CROSS图片的宽和高
     button.nHeight,
     BackGround.hdcOffScreen,  //父窗口上背景图的OffScreen hdc, 也就是画在内存里的那份
     button.x,  //CROSS按钮在背景图上的左上角坐标。我们把背景图上那个区域的内容先画到BUTTON的hdcDevice上面,作为BUTTON窗口的背景。
     button.y,
     SRCCOPY );

  TransparentBlt( lpDIS->hDC,
     0,
     0,
     button.nWidth,
     button.nHeight,
     button.hdcOffScreen, //这次就把CROSS图片在画在内存里的那份hdcOffScreen拿出来,画到hdcDevice上
     0,
     0,
     button.nWidth,
     button.nHeight,
     0x00FF00FF);

  break;
 }
}


最终效果如图



5、用这个方法同样可以实现静态文本控件的“透明”显示, 不过这就不是在WM_DRAWITEM消息里做响应了,而是在WM_CTLCOLORSTATIC消息里,参考代码如下:

case WM_CTLCOLORSTATIC:
{
 HDC hdc = (HDC)wParam;
 HWND hWnd = (HWND)lParam;

 SetTextColor(hdc, FontColor); //设置文本颜色
 BitBlt(hdc, ...) //把父窗口的背景图抓出对应坐标和长宽的一块来当我们的背景。
 SetBkMode(hdc, TRANSPARENT); //设置文本区域背景透明,这是必要的。除了按钮的背景外,文本自己还有个底
 return (int)GetStockObject(NULL_BRUSH); //返回一个透明刷子,不然后面会把我们BitBlt上去的背景图给刷没了。
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
WinCE5.0下实现透明背景按钮(.net C#) 需要在wince5.0下实现图形化界面,用于功能导航,用过GPS导航就知道了,类似的界面。众所周知gif,PNG等图片,都是可以实现透明背景的,在win下这应该不是个问题。但在ce5.0下,无论是透明背景的ICON、PNG、GIF都无法简单实现透明背景,alpha通道会丢失。网上搜索了些资料,当然也有办法解决,大家可以查一下,但性能较低。最终按微软SDK自带的一个计算器的源码思路解决了透明背景按钮问题。 要点: 1.如何画透明背景的BMP 2.如何画按钮并实现事件 思路及解决: 1.画按钮的思路:   ImageAttributes imageAttr=new ImageAttributes();   imageAttr.SetColorKey(Color.FromArgb(255, 0, 255),Color.FromArgb(255, 0, 255));   然后使用Graphics.DrawImage(,,,,,imageAttr)函数在指定的位置上画出透明的图片。   2. 事件驱动的思路: 事先定义好各按钮的指令   public enum Command { cmd1 = 0,//无操作 cmd2,//第一项操作 cmd3,//第二项操作 cmd4,//可自己扩展 max } 在创建按钮的同时明确以下几个参数 容器控件、资源存放的目录、X坐标、Y坐标、背景图片、按钮的标题、激活时的文字颜色、激活时的文字颜色、触发的指令。创建窗口,针对窗体事件做如下定义 •在Form_Load时生成按钮, •在Form_OnPaint时使用按钮自身的Render函数根据自己状态(有没有被点中)重画, •在MouseDown时判断点击位置是否在某个按钮的内部,如果是在它内部就改变它的状态,设置状态的同时调用窗口控件的Graphics局部重画这个按钮,     •在Form_MouseUp时判断现在的位置是不是在按钮内部,如果按下了,抬起时又移出了范围则不处理。如果按下与抬起都是在同一个按钮的边界内部则执行这个按钮所设置的指令。 以下是我使用的资源,将作为按钮的图片需要透明的部分设置成RGB(255,0,255),那种常刺眼的颜色。以下图片可以另存为BMP使用,设置的分辨是800*600的CE设备。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值