bitblt函数 透明底图_手绘楼盘摆位图怎么画

bitblt函数 透明底图_手绘楼盘摆位图怎么画CDC::BitBlt函数,应用在GDI的绘图中,有时候我们需要显示的位图有一部分是透明的,例如在棋类游戏中,棋盘图片和棋子图片融合的时候,棋子边缘应该是透明。    在图1中,棋盘和棋子图片融合在一起,需要设置棋子图片边缘的“品红色”为透明。     1、BitBlt函数说明:        BOOLBitBlt(intx,int

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

       CDC::BitBlt函数,应用在GDI的绘图中,有时候我们需要显示的位图有一部分是透明的,例如在棋类游戏中,棋盘图片和棋子图片融合的时候,棋子边缘应该是透明。

       在图1中,棋盘和棋子图片融合在一起,需要设置棋子图片边缘的“品红色”为透明。bitblt函数 透明底图_手绘楼盘摆位图怎么画

BitBlt函数说明:              

                    

BOOL BitBlt( int x, int y, int nWidth, int nHeight, CDC* pSrcDC, int xSrc, int ySrc, DWORD dwRop );

          
CSDN中关于bitblt函数的说明 :Copies a bitmap from the source device context to this current device context.   将源设备上下文中一个位图,拷贝到当前设备上下文中。其中x,y是当前设备上下文的绘图位置,nWidth和nHeight是位图大小,xSrc和ySrc是源设备上下文的位图位置。dwRop 是源位图和当前位图的运算方式,下面是其取值

BLACKNESS 用黑色填充目标矩形区域.

DSTINVERT 将目标矩形图象进行反相.

MERGECOPY 将源矩形图象与指定的图案刷(Pattern)进行布尔”与”运算.

MERGEPAINT 将源矩形图形经过反相后,与目标矩形图象进行布尔”或”运算.

NOTSRCCOPY 将源矩形图象经过反相后,复制到目标矩形上.

NOTSRCERASE 先将源矩形图象与目标矩形图象进行布尔”或”运算,然后再将得图象进行反相.

PATCOPY 将指定的图案刷复制到目标矩形上.

PATINVERT 将指定的图案刷与目标矩形图象进行布尔”异或”运算.

PATPAINT 先将源矩形图象进行反相,与指定的图案刷进行布尔”或”运算,

再与目标矩形图象进行布尔”或”运算

SRCAND 将源矩形图象与目标矩形图象进行布尔”与”运算.

SRCCOPY 将源矩形图象直接复制到目标矩形上.

SRCERASE 将目标矩形图象进行反相,再与源矩形图象进行布尔”与”运算.

SRCINVERT 将源矩形图象与目标矩形图象进行布尔”异或”运算.

SRCPAINT 将源矩形图象与目标矩形图象进行布尔”或”运算.

WHITENESS 用白色填充目标矩形区域.

  下面举例说明 位图的运算方式,两个位图按照对应像素 按位运算          例如像素(R1,G1,B1)和(R2,G2,B2)采用SRCAND “与”运算,则结果为(R1&R2,G1&G2,B1&B2)。

测试:建立MFC单文档,在资源中添加BITMAP资源IDB_BITMAP1,IDB_BITMAP2,在view类的OnDraw中,添加以下代码:

<span style="font-size:18px;">void CXXXXXXXXView::OnDraw(CDC* pDC)
{
	CceshiTransparentDoc* pDoc = GetDocument();
	ASSERT_VALID(pDoc);
	if (!pDoc)
		return;

	// TODO: 在此处为本机数据添加绘制代码
	//图片1
	CBitmap bitmap1;
	bitmap1.LoadBitmap(IDB_BITMAP1);
	CDC  dcMem1;
	dcMem1.CreateCompatibleDC(pDC);
	CBitmap *pOldBitmap1=dcMem1.SelectObject(&bitmap1);

	//获取图像尺寸
	BITMAP  bitmapInfo;
	bitmap1.GetBitmap(&bitmapInfo);
	
	//图片2
	CBitmap bitmap2;
	bitmap2.LoadBitmap(IDB_BITMAP2);
	CDC  dcMem2;
	dcMem2.CreateCompatibleDC(pDC);
	CBitmap *pOldBitmap2=dcMem2.SelectObject(&bitmap2);
	
	//两个位图取“与”
	pDC->BitBlt(100,100,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMem1,0,0,SRCCOPY);
	pDC->BitBlt(100,100,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMem2,0,0,SRCAND);
	 
	dcMem1.SelectObject(pOldBitmap1);
	dcMem2.SelectObject(pOldBitmap2);

	dcMem1.DeleteDC();
	dcMem2.DeleteDC();
}</span>

测试位图

bitblt函数 透明底图_手绘楼盘摆位图怎么画                                          bitblt函数 透明底图_手绘楼盘摆位图怎么画

结果:

bitblt函数 透明底图_手绘楼盘摆位图怎么画

从结果图可以看出,位图操作是逐pixel逐bit进行运算的。

 单色位图与彩色位图的相互转化


CSDN中关于bitblt函数还有以下说明,

If destination, source, and pattern bitmaps do not have the same color format, theBitBlt function converts the source and pattern bitmaps to match the destination. The foreground and background colors of the destination bitmap are used in the conversion.

When the BitBlt function converts a monochrome bitmap to color, it sets white bits (1) to the background color and black bits (0) to the foreground color. The foreground and background colors of the destination device context are used. To convert color to monochrome, BitBlt sets pixels that match the background color to white and sets all other pixels to black.BitBlt uses the foreground and background colors of the color device context to convert from color to monochrome.

翻译:如果目的,源和模式位图颜色模式不同,BitBlt 函数会转换源和模式位图,以适应目的位图。目的位图的前景色和背景色,应用在转换过程中。单色位图只有“1”和“0”两种像素值。

BitBlt函数转换“单色位图”(源)“彩色位图”(目的)时,会将“单色位图”中的“1”转化为背景色目的的背景色),“0”转化为前景色目的的前景色)。这里需要十分注意背景设置函数pdcDest->SetBkColor(…)和前景设置函数pdcDest->SetTextColor(…)的使用。

BitBlt函数转换“彩色位图”(源)“单色位图”(目的)时,将“彩色位图”中像素值等于“背景色”(通过pdcSource->SetBkColor(…)设置)的像素转化为“单色位图”中为“1”,其他的像素值变成了“0”。

测试图片:

bitblt函数 透明底图_手绘楼盘摆位图怎么画

测试代码:

	//第1步:加载源位图bitmapDest,选入内存设备dcDest
	CDC dcSource;
	dcSource.CreateCompatibleDC(pDC);
	CBitmap bitmapDest;
	bitmapDest.LoadBitmap(IDB_BITMAP3);
	CBitmap *pOldBitmapDest=dcSource.SelectObject(&bitmapDest);          //源位图 选到了设备上下文

	BITMAP bitmapInfo;                                                 //获取位图信息
	bitmapDest.GetBitmap(&bitmapInfo);


	//第2步:建立单值位图bitmapMonochrome,选入内存设备dcMonoChrome

	CDC dcMonoChrome;                       //单色设备上下文
	dcMonoChrome.CreateCompatibleDC(pDC);
	CBitmap bitmapMonochrome;
	bitmapMonochrome.CreateBitmap(bitmapInfo.bmWidth,bitmapInfo.bmHeight,1,1,NULL);
	CBitmap *pOldBitmapMonoChrome=dcMonoChrome.SelectObject(&bitmapMonochrome);        



	//第3步:彩色位图 -----------> 单色位图 

	///注意这里设置  源位图的背景色
	dcSource.SetBkColor(RGB(164,51,39));  

	//源位图中   像素值为(164,51,39)的 对应到单色位图中的1,其他值变为0
	dcMonoChrome.BitBlt(0,0,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcSource,0,0,SRCCOPY);  


	第4步:画出转化后的单色图

	// pDC->BitBlt(50,50,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMonoChrome,0,0,SRCCOPY);

	//第5步:单色图   ---------->   彩色位图
	pDC->SetBkColor(RGB(220,216,67));                 //单色中的1--------------->(220,216,67)
	pDC->SetTextColor(RGB(173,98,50));                //单色中的0--------------->(173,98,50)
	pDC->BitBlt(50,50,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMonoChrome,0,0,SRCCOPY);



	//释放
	dcSource.SelectObject(pOldBitmapDest);
	bitmapDest.DeleteObject();
	dcSource.DeleteDC();

	 
	dcMonoChrome.SelectObject(&bitmapMonochrome); 
	bitmapMonochrome.DeleteObject();
	dcMonoChrome.DeleteDC();

结果分析:

bitblt函数 透明底图_手绘楼盘摆位图怎么画



彩色位图融合到彩色背景

返回文章开头提到的,棋子周围的图片透明处理。这里要借助单色位图,作为 “掩图”,去除棋子周围的透明区域。

性质1:二进制逻辑运算中,0[xor]0=0;0[xor]1=1,可见,“异或”操作中0具有保持原bit不变的性质。

性质2:1[xor]0[xor]0=1;1[xor]1[xor]1=1;0[xor]1[xor]1=0;0 [xor] 0[xor]0=0; 可见,一个bit1[xor]bit2[xor]bit2=bit1  连续“异或”,和“没有操作”一样。

借助一些小技巧,可以去掉图片周围的底色,现在有下面的“背景图”(A)和“源位图”(B),需要去掉“鸣人”周围的“棕红色”区域。

问题:根据性质2,A[异或]B[异或]B,可以去掉“棕红色”,但是糟糕的是“鸣人”也去掉了。

解决办法:在根据性质1,在第一次   [异或]B  后,将“鸣人”区域置为0,“棕红色”区域保持不变,办法是中间加上一个[与]操作。根据鸣人图片制作一个“掩码”图片,即图(C)。


图片叠加的步骤:A[异或]B[与]C[异或]B

bitblt函数 透明底图_手绘楼盘摆位图怎么画                

                             (A)            


 bitblt函数 透明底图_手绘楼盘摆位图怎么画


                        (B)


bitblt函数 透明底图_手绘楼盘摆位图怎么画


                           (C)

测试代码:

	//第1步:创建背景  
	CBitmap bitmapBack;
	bitmapBack.LoadBitmap(IDB_BEIJING);
	CDC dcMemBack;
	dcMemBack.CreateCompatibleDC(pDC);
	CBitmap *pOldBitmapBack=dcMemBack.SelectObject(&bitmapBack);

	BITMAP bitmapInfo;
	bitmapBack.GetBitmap(&bitmapInfo);

	//第2步:创建源图
	CBitmap bitmapSource;
	bitmapSource.LoadBitmap(IDB_BITMAP3);
	CDC  dcMemSource;
	dcMemSource.CreateCompatibleDC(pDC);
	CBitmap *pOldBitmapSource=dcMemSource.SelectObject(&bitmapSource);



	//第3步:创建单色图
	CBitmap  bitmapMonochrome;
	bitmapMonochrome.CreateBitmap(bitmapInfo.bmWidth,bitmapInfo.bmHeight,1,1,NULL);
	CDC  dcMemMonochrome;
	dcMemMonochrome.CreateCompatibleDC(pDC);
	CBitmap *pOldBitmapMonochrome=dcMemMonochrome.SelectObject(&bitmapMonochrome);

	//第4步:源位图----->单色图
	dcMemSource.SetBkColor(RGB(164,51,39));
	dcMemMonochrome.BitBlt(0,0,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMemSource,0,0,SRCCOPY);


	//第5步:“鸣人”透明叠加“背景”图中
	pDC->BitBlt(50,50,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMemBack,0,0,SRCCOPY);
	pDC->BitBlt(50,50,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMemSource,0,0,SRCINVERT);


	//注意这里涉及到将“单色位图”-------->“彩色位图”,然后在进行运算,
	//默认下背景色是(255,255,255),前景色是(0,0,0),下面两句可以不加,
	pDC->SetBkColor(RGB(255,255,255));   //将dcMemMonochrome中的“1”转化为(255,255,255)
	pDC->SetTextColor(RGB(0,0,0));       //将dcMemMonochrome中的“0”转化为(0,0,0)

	pDC->BitBlt(50,50,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMemMonochrome,0,0,SRCAND);


	pDC->BitBlt(50,50,bitmapInfo.bmWidth,bitmapInfo.bmHeight,&dcMemSource,0,0,SRCINVERT);


	//释放
	dcMemBack.SelectObject(pOldBitmapBack);
	bitmapBack.DeleteObject();
	dcMemBack.DeleteDC();
	
	dcMemSource.SelectObject(pOldBitmapSource);
	bitmapSource.DeleteObject();
	dcMemSource.DeleteDC();

	dcMemMonochrome.SelectObject(pOldBitmapMonochrome);
	bitmapMonochrome.DeleteObject();
	dcMemMonochrome.DeleteDC();

测试结果:

bitblt函数 透明底图_手绘楼盘摆位图怎么画



文中有错误的地方请留言!!!!





版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/180426.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • JS 实现2+2=5的代码 实现原理解析

    JS 实现2+2=5的代码 实现原理解析

  • EXTJS 教程目录

    EXTJS 教程目录  本人开发extjs有两三个月了,做了三个左右的项目,其中后台都是用它来完成的。现在想借此机会整理一下用extjs开发的一些思维。  其实本人并没有完全地看过一本extjs的书籍,只是在开发过程中遇到什么问题就去百度什么。结果到现在开发时基本的东西都记不住,每次都是从旧项目中拷贝要用的东西出来,结果效率很慢。ps:以下教程都是采用extjs3.4都编写的  言归正传,以下的目录…

  • Flutter Mac 安装全教程(Android / iOS)

    Flutter Mac 安装全教程(Android / iOS)本文来自于:https://flutter.io/docs/get-started/install/macos目录下载Flutter设置环境变量Flutter命令安装编辑IDE下载Flutterhttps://flutter.io/docs/development/tools/sdk/archive在这里获取Flutter的安装包,推荐使用stablechanne…

  • 无法停止usb大容量存储设备_大容量存储重启后就被禁用

    无法停止usb大容量存储设备_大容量存储重启后就被禁用需求:最近由于操作设置本机电脑组策略禁用可移动存储设备后,恢复不了USB大容量存储设备禁用后恢复不了问题解决方案:1:网上一大群所谓的知识分支提供了几乎拷贝的一致的答案:注册策略恢复设置。说多了都是泪啊,毛用没有。2:后面想到了下载软件(大势至usb控制系统8.1官方版)试一下吧:设置一下就可以了。该软件下载一个试用版本就行。下载链接如下:http://www.onlinedow…

  • list遍历的几种方式_arraylist cannot be cast to

    list遍历的几种方式_arraylist cannot be cast to目录复原现场疑问原因解决方案注意点复原现场上代码:疑问为啥指定了类型还会出现类型转换出错?原因在pcp代码规范里面有提到里面的代码可以看到,一个指定类型,一个是泛型,他们居然可以互相转换,这是为了兼容以前版本在上面代码Set<String>set=newHashSet<>(list);……

  • php sqrt函数,sqrt函数怎么使用「建议收藏」

    php sqrt函数,sqrt函数怎么使用「建议收藏」JavaScript中的sqrt函数是用于返回一个数的平方根,也就是开平方,下面的文章我们就来具体看一下sqrt函数的使用方法。我们来看一下sqrt函数的基本语法Math.sqrt(value)平方根数需要计算。返回作为参数传递的数字的平方根。需要注意的是:1、作为参数传递的非数字字符串返回NaN2、作为参数传递的多于1个整数的数组返回NaN3、作为参数传递的负数返回NaN4、作为参数传递的空字符…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号