利用GDI+处理图像的色彩
首先先介绍一下ColorMatrix结构体:表示颜色的变换关系,定义如下:
typedef struct { REAL m[][]; } ColorMatrix;
ColorMatrix结构体一般和ImageAttribute类配合使用,使用的方式是先调用ImageAttibute::SetColorMatrix,运用该颜色变化矩阵,然后在绘制函数中将ImageAttribute对象作为DrawImage函数参数。以下的图像色彩变换都会用到这个结构体。
获取对应编码器的CLSID
int GetEncoderClsid(const WCHAR* format, CLSID* pClisd); // 获取对应编码器的CLSID int CMyDlg::GetEncoderClsid(const WCHAR* format, CLSID* pClisd) // 获取对应编码器的CLSID
{
UINT num = ; // 图像编码器的数量
UINT size = ; // 图像编码器数组的字节数 Gdiplus::ImageCodecInfo* pImageCodecInfo = NULL;
GetImageEncodersSize(&num, &size);
if (size == )
return -;
pImageCodecInfo = (Gdiplus::ImageCodecInfo*)(malloc(size)); GetImageEncoders(num, size, pImageCodecInfo);
for(UINT j = ;j<num;++j)
{
if (wcscmp(pImageCodecInfo[j].MimeType, format) == )
{
*pClisd = pImageCodecInfo[j].Clsid;
free(pImageCodecInfo);
return j;
}
}
free(pImageCodecInfo);
return -;
}
一、改变图像的透明度:只需要缩放Alpha分量就能到达效果。
BOOL ChangeImageAlpha(const CString& imagePath, REAL alpha, const CString& savePath);
/**********************************************************************************
* 作用:改变指定图像的透明度,并将新图像按照指定的图像格式和路径保存
* 参数:imagePath 为原图路径
* alpha 为分量缩放系数
* savePath 为用于保存处理后图像的路径
* 返回值: 转换是否成功
**********************************************************************************/
BOOL CMyGDIDlg::ChangeImageAlpha(const CString& imagePath, REAL alpha, const CString& savePath)
{
Bitmap bitmap(imagePath);
if (bitmap.GetLastStatus() != Ok)
return false; int nWidth = bitmap.GetWidth();
int nHeight = bitmap.GetHeight(); // 构建新图像对象
Bitmap image(nWidth, nHeight);
Rect rect(, , nWidth, nHeight);
// 利用新图像对象绘制
Graphics graph(&image); // 构建颜色变化矩阵
ColorMatrix colorMatrix = {
, , , , ,
, , , , ,
, , , , ,
, , , alpha, ,
, , , ,
}; ImageAttributes imageAttr;
imageAttr.SetColorMatrix(&colorMatrix); // 运用颜色变换矩阵绘制新图像
graph.DrawImage(&bitmap, rect, , , nWidth, nHeight, UnitPixel, &imageAttr); CLSID encoderClsid; // 文件编码器的CLSID
CString strExt = savePath.Right();
strExt.MakeLower();
// 根据扩展名获得不同的CLSID
if (strExt == _T("png"))
GetEncoderClsid(_T("image/png"), &encoderClsid);
else if (strExt == _T("jpg"))
GetEncoderClsid(_T("image/jpg"), &encoderClsid);
else
GetEncoderClsid(_T("image/bmp"), &encoderClsid); if (image.Save(savePath, &encoderClsid, NULL) == Ok)
return true;
else
return false;
}
调用: ChangeImageAlpha(_T("E:\\素材\\jpg\\1.jpg"), 0.5, _T("D:\\1.png"));
二、将图像转换为灰度图:原理就是使图中红、绿、蓝3个分量值相等。一般有3中方式:
(1)平均值法:使每个像素的三原色值等于红、绿、蓝3分量的平均值
R = G = B = (R + G +B) / 3
(2)最大值法:每个像素的三原色等于红、绿、蓝3分量的最大值
R = G = B = max(R, G, B)
(3)加权平均值法:给予红、绿、蓝3分量不同的权值然后相加
R = G = B = WrR + WgG + WbB
人眼对于三原色的敏感度从高到底分别是绿、红、蓝,所以三原色权值取值关系应该是 Wg > Wr > Wb。
依据YUV颜色空间可知当 R = G = B = 0.299R + 0.587 + 0.114B时能够的到最合理的灰度图。
算法与前文类似,在这里只需修改一下颜色变化矩阵即可:
// 构建颜色变化矩阵 ColorMatrix colorMatrix = {
0.299f, 0.299f, 0.299f, , ,
0.587f, 0.587f, 0.587f, , ,
0.114f, 0.114f, 0.114f, , ,
, , , , ,
, , , ,
};
三、改变图像的亮度:是通过改变红、绿、蓝颜色分量的增量来实现的。公式如下:
// brightness 为亮度变化量 REAL f = brightness / 255.0f;
// 构建颜色变化矩阵
ColorMatrix colorMatrix = {
, , , , ,
, , , , ,
, , , , ,
, , , , ,
f, f, f, ,
};
四、改变图像的对比度:一般来说对比度越大,图像越清晰醒目,色彩也越鲜明艳丽;而对比度越小,则会让图画显得比较灰暗。
图像的对比度变化公式如下:其中f为对比度,默认为1.
Rt = 128 + (R - 128)f
Gt = 128 + (G -128)f
Bt = 128+ (B - 128)f
REAL f = 0.0f;
if (contrast >= )
f = (contrast + 10.0f) / 10.0f;
else
f = ( + contrast) / 255.0f;
// 构建颜色变化矩阵
ColorMatrix colorMatrix = {
f, , , , ,
, f, , , ,
, , f, , ,
, , , , ,
0.5f*(-f), 0.5f*( - f), 0.5f*( - f), ,
};
利用GDI+处理图像的色彩的更多相关文章
- 利用GDI+处理图像,包括图像的的裁剪显示、转置、镜像、简单旋转、变形等。
一.图像的裁剪显示:有时程序需要显示图像的一部分而不是全部.实例代码如下: CDC* pDC = GetDC(); Graphics graph(pDC->GetSafeHdc()); Imag ...
- 超全面的.NET GDI+图形图像编程教程
本篇主题内容是.NET GDI+图形图像编程系列的教程,不要被这个滚动条吓到,为了查找方便,我没有分开写,上面加了目录了,而且很多都是源码和图片~ (*^_^*) 本人也为了学习深刻,另一方面也是为了 ...
- 利用OpenCV给图像添加中文标注
利用OpenCV给图像添加中文标注 : 参考:http://blog.sina.com.cn/s/blog_6bbd2dd101012dbh.html 和https://blog.csdn.net/ ...
- [转]超全面的.NET GDI+图形图像编程教程
本篇主题内容是.NET GDI+图形图像编程系列的教程,不要被这个滚动条吓到,为了查找方便,我没有分开写,上面加了目录了,而且很多都是源码和图片~ GDI+绘图基础 编写图形程序时需要使用GDI(Gr ...
- 学习笔记:利用GDI+生成简单的验证码图片
学习笔记:利用GDI+生成简单的验证码图片 /// <summary> /// 单击图片时切换图片 /// </summary> /// <param name=&quo ...
- C#利用GDI+绘制旋转文字等效果
C#中利用GDI+绘制旋转文本的文字,网上有很多资料,基本都使用矩阵旋转的方式实现.但基本都只提及按点旋转,若要实现在矩形范围内旋转文本,资料较少.经过琢磨,可以将矩形内旋转转化为按点旋转,不过需要经 ...
- 8 个优秀的 Linux 图形图像及色彩工具
8 个优秀的 Linux 图形图像及色彩工具 1. 硬件色彩分析器LPROF LPROF 是一个用于创建设备兼容,如相机.扫描仪.显示器的ICC兼容型材的颜色分析器.这些配置提供跨设备的色彩一致性.他 ...
- 利用matlab求图像均值和方差的几种方法
一.求均值 % 求一副灰度图像的均值 close all; clear; clc; i=imread('d:/lena.jpg'); %载入真彩色图像 i=rgb2gray(i); %转换为灰度图 i ...
- OpenCV与Python之图像的读入与显示以及利用Numpy的图像转换
1:读入图像,显示图像与保存图像 代码: import cv2 img=cv2.imread('lena.jpg',cv2.IMREAD_COLOR) cv2.namedWindow('lena',c ...
随机推荐
- walsh矩阵
矩阵A(1) 1 矩阵A(2) 1 1 1 -1 矩阵A(2n) 由上一级矩阵组合,即用上一级矩阵的一行生成本级矩阵的一行. 生成规则是 A(2n)[k] = { A(n)[k], A(n)[k] ...
- Java中TreeMap的基本操作
TreeSet有四种种构造函数可以初始化 在代码中主要列出了常用的三种: 构造方法摘要 TreeSet() 构造一个新的空 set,该 set 根据其元素的自然顺序进行排序. Tr ...
- Ubuntu 中用 delphi 开发 apache
经过近15年的沉默.delphi 10.2 终于重新开始支持linux 开发了. 今天说一下在ubuntu中开发apache的方法. 首先安装ubuntu 的delphi 开发环境,请参考以前的文章 ...
- 对 Service中sqlsession对象的优化
在本线程中添加object数据,必须在本线程中才能获取出来..其他线程获取不到. public class Test { public static void main(String[] args) ...
- MyBatis 实现新增
MyBatis实现新增 1.概念学习:(角度不同) 1.1 功能:从应用程序角度出发,软件具有哪些功能 1.2 业务:完成功能时的逻辑,对应Service中一个方法 1.3 事务:从数据库角度出发,完 ...
- centos7 lvm实例
1.lvm安装 rpm -qa|grep lvm lvm version yum install -y lvm2* 2.磁盘分区 fdisk -l fdisk /dev/xvdb1 #分区结束标记 t ...
- presentation skills
下面是从一个网站摘录下来的关于presentation skill需要回答的14个问题:网站的地址为:http://www.mindtools.com/pages/article/newCS_96.h ...
- 671. Second Minimum Node In a Binary Tree
/** * Definition for a binary tree node. * struct TreeNode { * int val; * TreeNode *left; * TreeNode ...
- 高性能Java RPC框架Dubbo与zookeeper的使用
https://blog.csdn.net/qq_38982845/article/details/83795295
- ubuntu16.04无法关机解决方法
11 down vote In my case these simple steps worked for me fine: Press Ctrl+Alt+T to go to a termina ...