MFC之绘图总结
来源:互联网 发布:imovie windows破解版 编辑:程序博客网 时间:2024/05/07 14:33
MFC提供绘图API只支持Windows操作系统中的标准图像文件格式,即BMP格式,可以分成两类:设备相关位图(DDB)和设备无关位图(DIB)。
我的理解用代码与注释的方式表现了,有什么不正确的地方请指正,谢谢!
1)以整个文件的信息作为数据源的显示方式(文件格式为bmp)
LPBITMAPINFO bmi;
LPBYTE pBits;
// data为fwrite读取整个文件的数据
int LoadMemToBitmap(unsigned char* data, int size)
{
BITMAPFILEHEADER bmfh;
int pos = 0;
//读文件信息头
memcpy((LPVOID)&bmfh, data + pos, sizeof(bmfh) );
pos += sizeof(bmfh);
if(bmfh.bfType != 0x4d42)
{
AfxMessageBox("This is not a bmp file!");
return -1;
}
//读位图信息头
int infoSize = bmfh.bfOffBits - sizeof(bmfh);
bmi = (LPBITMAPINFO)new BYTE[infoSize];
memcpy((LPVOID)bmi, data + pos, infoSize);
pos += infoSize;
if(bmi->bmiHeader.biBitCount!=1 && bmi->bmiHeader.biBitCount!=4 && bmi->bmiHeader.biBitCount!=8 && bmi->bmiHeader.biBitCount!=24)
{
AfxMessageBox("The number of colors is not valid!");
return -2;
}
//读图像数据
pBits = new BYTE[bmi->bmiHeader.biSizeImage];
//file.Read( (LPVOID)pBits, bmi->bmiHeader.biSizeImage);
memcpy((LPVOID)pBits, data + pos, bmi->bmiHeader.biSizeImage);
pos += bmi->bmiHeader.biSizeImage;
return 0;
}
void OnPaint()
{
CPaintDC dc(this); // device context for painting
// TODO: Add your message handler code here
// Do not call CDialog::OnPaint() for painting messages
if (bmi != NULL)
{
int width = bmi->bmiHeader.biWidth;
int height = bmi->bmiHeader.biHeight;
int xDest = 0;
int yDest = 0;
int nRate = 2;
CRect rect;
GetClientRect(rect);//获取客户区大小,如果需要画整个位图,可以根据需要修改
xDest = (rect.Width() - width * nRate)/2;
yDest = (rect.Height() - height * nRate)/2 + 76;
StretchDIBits(dc.GetSafeHdc(),xDest,yDest,bmi->bmiHeader.biWidth * nRate,bmi->bmiHeader.biHeight * nRate,0,0,bmi->bmiHeader.biWidth,bmi->bmiHeader.biHeight,pBits,bmi,DIB_RGB_COLORS,SRCCOPY);
}
}
2)以图像数据块作为数据源的显示方式(格式任意)
int ConvertToBitmap(CBitmap &bitmap,unsigned char *Img,int SizeX, int SizeY, int Channels)
{
unsigned char*pImage;
pImage = Img;
int byteSize = 0;
int _sizeX = SizeX,_sizeY = SizeY,bds = Channels,_bands = Channels;
if(NULL == pImage)
{
return 0;
}
//注释该if分支,可减少嵌套结构,有利于阅读和优化 liu_sh 2012-5-17
//单通道图
if (bds == 1 )
{
int widthBytes = (_sizeX * 3 * 8 + 31) / 32 * 4;
byteSize = widthBytes * _sizeY;
int newlen = _sizeX * _sizeY * 3;
unsigned char * dstData = new unsigned char[newlen];
unsigned char * srcData = pImage;
for (int i = 0 ; i < 3 ; i ++)
{
for (int j = 0 ; j < _sizeX * _sizeY ; j ++)
{
dstData[i * _sizeX * _sizeY + j] = srcData[j];
}
}
debugout("dst data set suc");
if (dstData != NULL && byteSize)
{
BITMAPINFO bmpinfo;
memset(&bmpinfo, 0, sizeof(bmpinfo));
bmpinfo.bmiHeader.biBitCount = 8 * 3;//每个像素的二进制位数
bmpinfo.bmiHeader.biHeight = _sizeY;//图像高度
bmpinfo.bmiHeader.biPlanes = 1;//面数,必须设为
bmpinfo.bmiHeader.biSize = sizeof(bmpinfo.bmiHeader);//bmpinfo.bmiHeader的大小
bmpinfo.bmiHeader.biSizeImage = (DWORD)byteSize;//图像数据区大小,注意一个扫描行所占的字节数必须是的倍数,不足用填充.也就是如果宽*3不是的倍数就要加到是的倍数.*3是相当于位来说的.
bmpinfo.bmiHeader.biWidth = _sizeX;//图像宽度
bitmap.DeleteObject();
HBITMAP hBitmap;
hBitmap = CreateDIBSection(NULL,&bmpinfo, DIB_RGB_COLORS ,NULL ,0, 0);
bitmap.Attach(hBitmap);
if (!bitmap.m_hObject)
{
delete[] dstData;//防止内存泄露 liu_sh 2012-5-22
dstData=NULL;
return -1;
}
BITMAP bmp;
bitmap.GetBitmap(&bmp);
unsigned char* buff = new unsigned char[byteSize];
ConvertFromChannelToDib(dstData, buff, _sizeX, _sizeY, 3);
memcpy(bmp.bmBits, buff, byteSize);
delete[] buff;
buff = NULL;
}
if(NULL != dstData)
{
delete []dstData;
dstData = NULL;
}
}
else //多通道图
{
int widthBytes = (_sizeX * _bands * 8 + 31) / 32 * 4;//计算每行需要的字节数
byteSize = widthBytes * _sizeY;//计算图像占用的总字节数
if(pImage == NULL || byteSize <=0) //此时无法转换图像
return 0;
//if (pImage != NULL && byteSize)
//{
BITMAPINFO bmpinfo;
memset(&bmpinfo, 0, sizeof(bmpinfo));
bmpinfo.bmiHeader.biBitCount = 8 * _bands;//每个像素的二进制位数
bmpinfo.bmiHeader.biHeight = _sizeY;//图像高度
bmpinfo.bmiHeader.biPlanes=1;//面数,必须设为
bmpinfo.bmiHeader.biSize=sizeof(bmpinfo.bmiHeader);//bmpinfo.bmiHeader的大小
bmpinfo.bmiHeader.biSizeImage = (DWORD)byteSize;//图像数据区大小,注意一个扫描行所占的字节数必须是的倍数,不足用填充.也就是如果宽*3不是的倍数就要加到是的倍数.*3是相当于位来说的.
bmpinfo.bmiHeader.biWidth = _sizeX;//图像宽度
bitmap.DeleteObject();
HBITMAP hBitmap;
hBitmap = CreateDIBSection(NULL,&bmpinfo, DIB_RGB_COLORS ,NULL ,0, 0);
bitmap.Attach(hBitmap);
if (!bitmap.m_hObject)
return -1;
BITMAP bmp;
bitmap.GetBitmap(&bmp);
unsigned char* buff = new unsigned char[byteSize];
memset(buff, 0, byteSize);
ConvertFromChannelToDib(pImage, buff, _sizeX, _sizeY, _bands, byteSize);
memcpy(bmp.bmBits, buff, byteSize);
delete[] buff;
//}// end of if (pImage != NULL && byteSize) liu_sh 2012-5-17
}
return 0;
}
void ConvertFromChannelToDib(unsigned char * srcData, unsigned char * dstData, int SizeX , int SizeY , int nChannels, int Len)
{
int i, j, k;
unsigned char *pcolor[3];
CString logStr;
if(NULL == srcData || NULL == dstData || SizeX <=0 || SizeY <=0 || nChannels <=0 )
{
return;
}
int widthBytes = (SizeX * nChannels * 8 + 31) / 32 * 4;
for (k=0; k<nChannels; k++)
pcolor[k] = srcData + SizeX * SizeY * k;
for (i=0; i<SizeY; i++)
{
for (j=0; j<SizeX; j++)
{
for (k=0; k<nChannels; k++)
{
dstData[widthBytes * (SizeY - i -1) + nChannels * j + k] = *pcolor[nChannels - 1 - k];
pcolor[nChannels - 1 - k]++;
}
}
for (k=0; k<widthBytes - SizeX * 3; k++)
dstData[widthBytes * (SizeY - i -1) + SizeX * 3 + k] = 0;
}
}
unsigned char* m_imgData = NULL; //读入的数据块信息
int m_nSizeX; // 传入的数据块的宽度
int m_nSizeY; // 传入的数据块的高度
int m_nChannels ; // 传入的数据块的通道数
void OnPaint()
{
CPaintDC dc(this); // device context for painting
{
if (m_imgData != NULL)
{
CWnd *pwnd = GetDlgItem(IDC_PIC1); // 定义的picture控件
pwnd->ShowWindow(SW_SHOW);
if(pwnd == NULL)
{
return;
}
CDC *pDC = pwnd->GetDC();
if(pDC == NULL)
{
return;
}
CRect rect;
pwnd->GetClientRect(&rect);
CDC memDC;
memDC.CreateCompatibleDC(pDC);
CBitmap* pOldBmp;
CBitmap bmp;
ConvertToBitmap(bmp,m_imgData ,m_nSizeX,m_nSizeY,m_nChannels);
if (!bmp.m_hObject)
{
return;
}
pOldBmp = memDC.SelectObject(&bmp);
try
{
pDC->BitBlt(0, 0, rect.Width(), rect.Height(), &memDC, 0, 0, SRCCOPY);
}
catch(...)
{
debugout("\n\n+++pDC->BitBlt error+++\n\n");
}
debugout("###################xiejianping 010097");
memDC.SelectObject(pOldBmp);
ReleaseDC(pDC);
memDC.DeleteDC();
if (m_imgData != NULL)
{
delete [] m_imgData;
m_imgData = NULL;
}
}
}
}
- MFC之绘图总结
- MFC绘图基础之总结
- MFC 绘图控制 -- 总结
- MFC绘图总结
- MFC之简单绘图
- MFC之缓冲绘图
- MFC绘图之线
- mfc的dc绘图总结
- MFC之画刷绘图
- MFC绘图总结-DIB图形绘制
- MFC绘图总结(4):画图
- 使用MFC编写绘图程序的总结
- MFC编程之三: 绘图-1(画图)
- MFC编程之三: 绘图-1(画图)
- MFC学习笔记之简单绘图一
- MFC学习笔记之简单绘图二
- MFC学习笔记之绘图控制
- MFC学习笔记之简单绘图二
- RCNN学习笔记(0):rcnn简介
- sg函数训练——开窍篇
- android so文件的加载过程以及常规错误处理
- 对一个8位(一字节)数的倒序处理
- HDU 1013
- MFC之绘图总结
- Android Studio 2.0+高效开发之路
- jq js css3 实现瀑布流
- vs2010 AfxMessageBox 或者MessageBox char*强转成LPCTSTR 乱码显示
- 随机过程--Metropolis-Hastings算法
- 八大排序算法
- 浏览文本文件并进行修改保存
- Ubuntu 好用命令总结
- ViewController 生命周期-iOS初学