
2013-07-16
void CDibView::OnFDImage() //产生"浮雕"效果图函数
{
HANDLE data1handle;//用来存放图像数据的句柄;
LPBITMAPINFOHEADER lpBi;//图像的信息头结构;
CDibDoc *pDoc=GetDocument();//得到文挡指针;
HDIB hdib;//用来存放图像数据的句柄;
unsigned char *pData;//指向原始图像数据的指针;
unsigned char *data;//指向处理后图像数据的指针;
hdib=pDoc->m_hDIB;//拷贝存放已经读取的图像文件数据句柄;
lpBi=(LPBITMAPINFOHEADER)GlobalLock((HGLOBAL)hdib);//获取图像信息头
pData=(unsigned char*)FindDIBBits((LPSTR)lpBi);
//FindDIBBits是我定义的一个函数、根据图像的结构得到位图的灰度值数据、
pDoc->SetModifiedFlag(TRUE);
//设置文档修改标志为"真"、为后续的修改存盘作准备;
data1handle=GlobalAlloc(GMEM_SHARE,WIDTHBYTES(lpBi->biWidth*8)*lpBi->biHeight); //声明一个缓冲区用来暂存处理后的图像数据;
data=(unsigned char*)GlobalLock((HGLOBAL)data1handle);//得到该缓冲区的指针;
AfxGetApp()->BeginWaitCursor();
int i,j,buf;
for( i=lpBi->biHeight; i>=2; i--)//从图像右下角开始对图像的各个像素进行"浮雕"处理;
for( j=lpBi->biWidth; j>=2; j--)
{
//浮雕处理
buf=*(pData+(lpBi->biHeight-i)*WIDTHBYTES(lpBi->biWidth*8)+j)-*(pData+(lpBi->biHeight-i+1)*WIDTHBYTES(lpBi->biWidth*8)+j-1)+128;
if(buf>255) buf=255;
if(buf<0)buf=0; *(data+(lpBi->biHeight-i)*WIDTHBYTES(lpBi->biWidth*8)+j)=(BYTE)buf;
}
for( j=0; jbiHeight; j++)
for( i=0; ibiWidth; i++)
//重新写回原始图像的数据缓冲区;
*(pData+i*WIDTHBYTES(lpBi->biWidth*8)+j)=*(data+i*WIDTHBYTES(lpBi->biWidth*8)+j); AfxGetApp()->EndWaitCursor();
pDoc->m_hDIB =hdib//将处理过的图像数据写回pDoc中的图像缓冲区;
GlobalUnlock((HGLOBAL)hdib);//解锁、释放缓冲区;
GlobalUnlock((HGLOBAL)data1handle);
GlobalFree((HGLOBAL)hdib);
GlobalFree((HGLOBAL)data1handle);
Invalidate(TRUE);//显示图像
}
是下面这种HDIB吗?
HBITMAP转成CBitmap:
HBITMAP hBmp;
//hBmp的初始化代码
CBitmap* pBmp = CBitmap::FromHandle(hBmp);
HBITMAP hbmp;
CBitmap bmp;
bmp.Attach(hbmp);
CBitmap bmp;
bmp.LoadBitmap(IDB_BITMAP1);
HBITMAP hbm = (HBITMAP)bmp;
CBitmap bmp1;
bmp1.Attach(hbm);
2.CBitmap转成HBITMAP:
CBitmap* pBmp;
HBITMAP hBmp = (HBITMAP)(*pBmp);//(HBITMAP)是CBitmap类重载了的操作符
CBitmap::FromHandle() 转换出来的临时 CBitmap 对象似乎并不是可靠的!MSDN 上说这个临时只在某一时间段内有效。等到某个空闲的时候,这个临时 CBitmap 对象会被清除掉。
所以觉得还是 new Bitmap().Attach(HBITMAP) 好一点,至少我用的时候没出问题。。。
用fopen打开一个bmp文件,请问怎样取得它的HDIB句柄?
MSDN中对HDIB的运用是:BMP文件除去文件头(BITMAPFILEHEADER)以后的所有内容。当然你也可以定义其批向BMP文件的点阵信息!
HDIB是要你自己获得的,它不过是个内存句柄而已,用GlobalAlloc分配一块用于保存位图数据
的内存,返回值就是一个HDIB,不过,内存里面一定要有图像数据
BOOL ReadDIBFile(LPCTSTR strFileName)
{
BITMAPFILEHEADER bmfHeader;
DWORD dwBitsSize;
HDIB hDIB;
LPSTR pDIB;
CFile file;
if(!file.Open(strFileName,CFile::modeRead))
{
TRACE("failed to open img file\n");
return FALSE;
}
// 获取DIB(文件)长度(字节)
dwBitsSize = file.GetLength();
// 尝试读取DIB文件头
if (file.Read((LPSTR)&bmfHeader, sizeof(bmfHeader)) != sizeof(bmfHeader))
{
TRACE("failed to read image file\n");
file.Close();
return FALSE;
}
// 判断是否是DIB对象,检查头两个字节是否是"BM"
if (bmfHeader.bfType != DIB_HEADER_MARKER)
{
TRACE("not a valid bmp file\n");
file.Close();
return FALSE;
}
// 为DIB分配内存
hDIB = (HDIB) ::GlobalAlloc(GMEM_MOVEABLE | GMEM_ZEROINIT, dwBitsSize);
if (hDIB == 0)
{
TRACE("failed to alloc memory when read img file to buffer\n");
return FALSE;
}
// 锁定
pDIB = (LPSTR) ::GlobalLock((HGLOBAL) hDIB);
// 读象素
if (file.ReadHuge(pDIB, dwBitsSize - sizeof(BITMAPFILEHEADER)) !=
dwBitsSize - sizeof(BITMAPFILEHEADER) )
{
TRACE("not a valid file length\n");
::GlobalUnlock((HGLOBAL) hDIB);
::GlobalFree((HGLOBAL) hDIB);
return FALSE;
}
file.Close();
return TRUE;
}