4,498
社区成员




void SavePicture(CString strFileFrom, CString strFileTo)
{
CFileStatus status;
if(!CFile::GetStatus(strFileFrom,status))
{
AfxMessageBox("文件 " + strFileFrom + " 不存在!");
return;
}
CBitmap frombmp;
HBITMAP bmpHandle = (HBITMAP)LoadImage(NULL,strFileFrom,IMAGE_BITMAP,0,0,LR_LOADFROMFILE);
frombmp.Attach(bmpHandle);
//frombmp.Attach(ShrinkBitmap(bmpHandle,32,32));
//SaveBitmapToFile(frombmp,"logo1.bmp");
//CPaintDC dc(this); //绘制图象用
CBitmap BmpBack;
BmpBack.Attach(bmpHandle);//加载位图,IDB_XXXX
CDC dcMem;//载入大位图
dcMem.CreateCompatibleDC(NULL) ;
CBitmap* pOldBitmap = dcMem.SelectObject(&BmpBack);
CBitmap BmpTemp;//存储切割后的位图
BmpTemp.CreateCompatibleBitmap(&dcMem,137,83);
CDC DcTemp;
DcTemp.CreateCompatibleDC(NULL);
CBitmap* pOldBmpTemp = DcTemp.SelectObject(&BmpTemp);
BITMAP bm;
BmpBack.GetBitmap(&bm);
//下面代码可以实现缩放,
int iOldMode = SetStretchBltMode(DcTemp, COLORONCOLOR); //该语句可以解决失真问题
DcTemp.StretchBlt(0,0,137,83,&dcMem,0,0,bm.bmWidth,bm.bmHeight,SRCCOPY);
SetStretchBltMode(DcTemp, iOldMode);
//////////////////////////////////////
SaveBitmapToFile((HBITMAP)BmpTemp,strFileTo);
DcTemp.DeleteDC();//删除DC,否则Bitmap不可同时使用
dcMem.DeleteDC();//
}
bool SaveBitmapToFile(HBITMAP hBitmap, LPCTSTR lpFileName)
{
HDC hDC; //设备描述表
int iBits; //当前显示分辨率下每个像素所占字节数
WORD wBitCount; //位图中每个像素所占字节数
DWORD dwPaletteSize=0, //定义调色板大小, 位图中像素字节大小 ,位图文件大小 , 写入文件字节数
dwBmBitsSize,
dwDIBSize, dwWritten;
BITMAP Bitmap; //位图属性结构
BITMAPFILEHEADER bmfHdr; //位图文件头结构
BITMAPINFOHEADER bi; //位图信息头结构
LPBITMAPINFOHEADER lpbi; //指向位图信息头结构
HANDLE fh, hDib, hPal,hOldPal=NULL; //定义文件,分配内存句柄,调色板句柄
//计算位图文件每个像素所占字节数
HDC hWndDC = CreateDC("DISPLAY",NULL,NULL,NULL);
hDC = ::CreateCompatibleDC( hWndDC ) ;
int bitspixel=GetDeviceCaps(hDC, BITSPIXEL);
int planes= GetDeviceCaps(hDC, PLANES);
iBits = bitspixel* planes;
//iBits=1;
DeleteDC(hDC);
if (iBits <= 1)
wBitCount = 1;
else if (iBits <= 4)
wBitCount = 4;
else if (iBits <= 8)
wBitCount = 8;
else if (iBits <= 24)
wBitCount = 24;
else
wBitCount = 1 ;
//计算调色板大小
if (wBitCount <= 8)
dwPaletteSize = (1 << wBitCount) * sizeof(RGBQUAD);
//设置位图信息头结构
GetObject(hBitmap, sizeof(BITMAP), (LPSTR)&Bitmap);
bi.biSize = sizeof(BITMAPINFOHEADER);
bi.biWidth = Bitmap.bmWidth;
bi.biHeight = Bitmap.bmHeight;
bi.biPlanes = 1;
bi.biBitCount = wBitCount;
bi.biCompression = BI_RGB;
bi.biSizeImage = 0;
bi.biXPelsPerMeter = 0;
bi.biYPelsPerMeter = 0;
bi.biClrUsed = 0;
bi.biClrImportant = 0;
dwBmBitsSize = ((Bitmap.bmWidth * wBitCount+31)/32) * 4 * Bitmap.bmHeight ;
//为位图内容分配内存
hDib = GlobalAlloc(GHND,dwBmBitsSize+dwPaletteSize+sizeof(BITMAPINFOHEADER));
lpbi = (LPBITMAPINFOHEADER)GlobalLock(hDib);
*lpbi = bi;
// 处理调色板
hPal = GetStockObject(DEFAULT_PALETTE);
if (hPal)
{
hDC = ::GetDC(NULL);
hOldPal = ::SelectPalette(hDC, (HPALETTE)hPal, FALSE);
RealizePalette(hDC);
}
// 获取该调色板下新的像素值
GetDIBits(hDC, hBitmap, 0, (UINT) Bitmap.bmHeight,
(LPSTR)lpbi + sizeof(BITMAPINFOHEADER)
+dwPaletteSize,
(LPBITMAPINFO )
lpbi, DIB_RGB_COLORS);
//恢复调色板
if (hOldPal)
{
SelectPalette(hDC, (HPALETTE)hOldPal, TRUE);
RealizePalette(hDC);
::ReleaseDC(NULL, hDC);
}
//创建位图文件
fh = CreateFile(lpFileName, GENERIC_WRITE,
0, NULL, CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN, NULL);
if (fh == INVALID_HANDLE_VALUE)
return FALSE;
// 设置位图文件头
bmfHdr.bfType = 0x4D42; // "BM"
dwDIBSize = sizeof(BITMAPFILEHEADER)
+ sizeof(BITMAPINFOHEADER)
+ dwPaletteSize + dwBmBitsSize;
bmfHdr.bfSize = dwDIBSize;
bmfHdr.bfReserved1 = 0;
bmfHdr.bfReserved2 = 0;
bmfHdr.bfOffBits = (DWORD)sizeof(BITMAPFILEHEADER)
+ (DWORD)sizeof(BITMAPINFOHEADER)
+ dwPaletteSize;
// 写入位图文件头
WriteFile(fh, (LPSTR)&bmfHdr, sizeof(BITMAPFILEHEADER), &dwWritten, NULL);
// 写入位图文件其余内容
WriteFile(fh, (LPSTR)lpbi, dwDIBSize,
&dwWritten, NULL);
//清除
GlobalUnlock(hDib);
GlobalFree(hDib);
CloseHandle(fh);
return TRUE;
}
BOOL ImageScale(CImage &imgSrc, CImage &imgDst, double alpha)
{
if (alpha < 1)
{
return FALSE;
}
int nHeight = imgSrc.GetHeight();
int nWidth = imgSrc.GetWidth();
int nLineBytes = imgSrc.GetLineSizeInBytes();
LPBYTE lpDIBBits = imgSrc.LockRawImg();
int nNewWidth = nWidth / alpha;
int nNewHeight = nHeight / alpha;
int nNewLineBytes = (nNewWidth + 31) / 32 * 4;
imgDst.Initialize(nNewWidth, nNewHeight, 1, 1);
LPBYTE lpNewDIBBits = imgDst.LockRawImg();
memset(lpNewDIBBits, 0, nNewLineBytes * nNewHeight);
int i, j;
LPBYTE lpSrc, lpDst;
for (i = 0; i < nNewHeight; i++)
{
for (j =0; j < nNewWidth; j++)
{
double q = i * alpha;
double p = j * alpha;
int y1, y2;
y1 = (int)q;
y2 = y1 + 1;
q = q - y1;
int x1, x2;
x1 = (int)p;
x2 = x1+1;
p = p - x1;
if ((x2 >nWidth) || (y2 >nHeight))
continue;
if (x2 == nWidth)
x2 = x2 -1 ;
if (y2 == nHeight)
y2 = y2 -1;
double a = 0.0, b = 0.0, c = 0.0, d = 0.0;
a = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y1, x1);
b = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y1, x2);
c = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y2, x1);
d = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y2, x2);
double t1,t2,t3;
t1 = (1-p)*a + p*b;
t2 = (1-p)*c + p*d;
t3 = (1-q)*t1 + q*t2;
if (t3>0.5) {
SetPixValue(lpNewDIBBits, nNewLineBytes, nNewHeight, i, j, 1);
}
else{
SetPixValue(lpNewDIBBits, nNewLineBytes, nNewHeight, i, j, 0);
}
}
}
imgSrc.UnlockRawImg();
imgDst.UnlockRawImg();
return TRUE;
}
BYTE GetPixValue(LPBYTE lpDIBBits, int nLineBytes, int nHeight, int i, int j)
{
LPBYTE lpSrc = lpDIBBits + nLineBytes * (nHeight - 1 - i) + j/8;
BYTE pix = (BYTE)*lpSrc;
BYTE pos = (BYTE)(7 - j%8);
pix &= (0x01<<pos);
return (BYTE)(pix >> pos);
}
void SetPixValue(LPBYTE lpDIBBits, int nLineBytes, int nHeight, int i, int j, BYTE pixel)
{
LPBYTE lpDst = lpDIBBits + nLineBytes * (nHeight - 1 - i) + j/8;
BYTE pix = (BYTE)*lpDst;
BYTE pos = (BYTE)(7 - j%8);
*lpDst &= ~(0x01<<pos);
*lpDst |= ((pixel & 0x01)<<pos);
return;
}