?? hddrawsurf.cpp
字號:
// HDDrawSurf.cpp: implementation of the CDDrawSurf class.
//
//////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "HDDrawSurf.h"
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
CDDrawSurf::CDDrawSurf()
{
memset(&m_Dat,0,sizeof(_SURFACE_DATA));
m_Dib=0;
lpSourceSurf=0;
Buffer=0;
}
CDDrawSurf::~CDDrawSurf()
{
if(m_Dib)
{
delete m_Dib;
}
}
//在該函數中創建一個表面:
LPDIRECTDRAWSURFACE7 CDDrawSurf::Create(_SURFACE_DATA * dat)
{
//1. 賦值:
memcpy(&m_Dat,dat,sizeof(m_Dat));
DWORD bytes=m_Dat.Width*m_Dat.Height*(m_Dat.BPP/8);
if(m_Dat.BPP<8||m_Dat.BPP>32||m_Dat.DisplayBPP<8||m_Dat.DisplayBPP>32)
{
return 0;
}
if(m_Dat.lpDD==0)
{
return 0;
}
//2. 創建一個LPDIRECTDRAWSURFACE7表面:
DDSURFACEDESC2 desc;
ZeroMemory( &desc,sizeof(desc));
desc.dwSize = sizeof(desc);
desc.dwFlags = DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH;
desc.dwWidth = m_Dat.Width;
desc.dwHeight = m_Dat.Height;
desc.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN;
HRESULT r=m_Dat.lpDD->CreateSurface(&desc,&lpSourceSurf,NULL);
if (r!=DD_OK)
{
return 0;
}
//3. 獲取位圖數據緩沖區首地址:
if(m_Dat.m_bFromFile)
{
m_Dib=new CDib();
LoadBmpFromFile();
}
else
{
AssignBuffer();
}
if(Buffer==0)
{
lpSourceSurf=0;
return 0;
}
//4.將位圖數據傳送到表面對應的內存:
if(m_Dat.BPP==24&&m_Dat.DisplayBPP==32)
{
if(!CopyPixels24to32())
{
SaveInfo("拷貝24未位圖信息到32位表面錯誤.txt","CopyPixels()==0");
lpSourceSurf=0;
return 0;
}
}
else
{
if(!CopyPixels())
{
SaveInfo("拷貝位圖信息錯誤.txt","CopyPixels()==0");
lpSourceSurf=0;
return 0;
}
}
//5.設置色鍵:
if(!SetColorKey(m_Dat.Colorkey))
{
SaveInfo("設置色鍵錯誤.txt","CDDrawSurf::SetColorKey()=FALSE");
lpSourceSurf=0;
return 0;
}
return lpSourceSurf;
}
//問題:應該創建一個函數,即可以用單獨的位圖文件
// 創建表面,又可以從一幅大位圖中提取子畫面;
void CDDrawSurf::LoadBmpFromFile()
{
if(!m_Dat.m_bFromFile)
{
Buffer=0;
return;
}
if(m_Dib==0)
{
Buffer=0;
return;
}
m_Dib->LoadBmp(m_Dat.fn);//根據位圖文件名載入位圖;
m_Dat.BPP=m_Dib->BPP; //讀取位圖色彩深度;
m_Dat.Bpl=m_Dib->Bpl;
//找到欲提取的數據:
Buffer=FindPos();//找到(xt,yt)像素在內存中的位置;
}
void CDDrawSurf::DestroySurface()
{
}
void CDDrawSurf::AssignBuffer()
{
//如果沒有指定位圖色彩深度或使用了不支持的位圖色彩深度:
if(m_Dat.BPP!=8||m_Dat.BPP!=24)
{
Buffer=0;
return;
}
if(m_Dat.Bpl==0)
{
Buffer=0;
return;
}
Buffer=m_Dat.buf;
}
BYTE * CDDrawSurf::FindPos()
{
//獲取位置:
BYTE * pos;
pos=m_Dib->GetBuffer();
pos=pos+m_Dib->Bpl*(m_Dib->Height-1-m_Dat.YS)+m_Dat.XS*m_Dat.BPP/8;
return pos;
}
BOOL CDDrawSurf::CopyPixels()
{
if (lpSourceSurf==0 || Buffer==0)
{
return FALSE;
}
//首先檢查給出的坐標和需要截取的位圖是否有越界的情況:
DWORD t1=m_Dat.XS+m_Dat.Width;
DWORD t2=m_Dat.YS+m_Dat.Height;
if(t1>m_Dib->Width)
{
m_Dat.Width=m_Dib->Width-m_Dat.XS;
}
if(t2>m_Dib->Height)
{
m_Dat.Height=m_Dib->Height-m_Dat.YS;
}
DWORD len,offset;
len=((m_Dat.Width*m_Dat.BPP)>>3);//截取的子畫面的每一行字節寬度;
offset=((m_Dib->Width*m_Dat.BPP)>>3);//pos指針的偏移量,為位圖每一行的字節寬度;
DDSURFACEDESC2 desc;
ZeroMemory( &desc, sizeof(desc) );
desc.dwSize = sizeof(desc);
//開始寫數據:
HRESULT r=lpSourceSurf->Lock( 0, &desc, DDLOCK_WAIT | DDLOCK_WRITEONLY, 0 );
if (r!=DD_OK)
{
return FALSE;
}
//找到目的地址:
BYTE* surfbuf = (BYTE*)desc.lpSurface;
BYTE * pos = Buffer;
//逐行寫入位圖數據:
for( unsigned i=0;i<m_Dat.Height;i++)
{
memset(surfbuf,0,desc.lPitch);
memcpy(surfbuf,pos,len);
pos -= offset;
surfbuf += desc.lPitch;
}
//結束寫數據:
lpSourceSurf->Unlock( 0 );
return TRUE;
}
//將24位位圖拷貝到32位表面:
BOOL CDDrawSurf::CopyPixels24to32()
{
if (lpSourceSurf==0 || Buffer==0)
{
return FALSE;
}
if(m_Dat.BPP!=24)
{
return FALSE;
}
//首先檢查給出的坐標和需要截取的位圖是否有越界的情況:
DWORD t1=m_Dat.XS+m_Dat.Width;
DWORD t2=m_Dat.YS+m_Dat.Height;
if(t1>m_Dib->Width)
{
m_Dat.Width=m_Dib->Width-m_Dat.XS;
}
if(t2>m_Dib->Height)
{
m_Dat.Height=m_Dib->Height-m_Dat.YS;
}
DDSURFACEDESC2 desc;
ZeroMemory( &desc, sizeof(desc) );
desc.dwSize = sizeof(desc);
DWORD offset;
offset=((m_Dib->Width*m_Dat.BPP)>>3);//pos指針的偏移量,為位圖每一行的字節寬度;
//開始寫數據:
HRESULT r=lpSourceSurf->Lock( 0, &desc, DDLOCK_WAIT | DDLOCK_WRITEONLY, 0 );
if (r!=DD_OK)
{
return FALSE;
}
//找到目的地址:
BYTE * surfbuf = (BYTE*)desc.lpSurface;
BYTE * pos = Buffer;
BYTE * posline;
//逐行逐列寫入位圖數據:
for( unsigned i=0;i<m_Dat.Height;i++)
{
memset(surfbuf,1,desc.lPitch);
posline=pos;
for(unsigned j=0;j<m_Dat.Width;j++)
{
memcpy(surfbuf,posline,3);
posline+=3;
surfbuf+=4;
}
pos-=offset;
}
//結束寫數據:
lpSourceSurf->Unlock( 0 );
return TRUE;
}
BOOL CDDrawSurf::Blit(int x,int y,unsigned uck)
{
if(lpSourceSurf==0||m_Dat.lpDestSurf==0)
{
return 0;
}
BOOL bfastblt=TRUE;
DDSURFACEDESC2 destdesc;
ZeroMemory( &destdesc, sizeof(destdesc) );
destdesc.dwSize = sizeof(destdesc);
m_Dat.lpDestSurf->GetSurfaceDesc( &destdesc );
RECT destrect;
destrect.left=0;
destrect.top=0;
destrect.right=destdesc.dwWidth;
destrect.bottom=destdesc.dwHeight;
DDSURFACEDESC2 srcdesc;
ZeroMemory( &srcdesc, sizeof(srcdesc) );
srcdesc.dwSize = sizeof(srcdesc);
lpSourceSurf->GetSurfaceDesc( &srcdesc );
RECT srcrect;
srcrect.left=0;
srcrect.top=0;
srcrect.right=srcdesc.dwWidth;
srcrect.bottom=srcdesc.dwHeight;
//保證貼圖在目標表面之內:
if (x+srcrect.left>=destrect.right)
{
return FALSE;
}
if (y+srcrect.top>=destrect.bottom)
{
return FALSE;
}
if (x+srcrect.right<=destrect.left)
{
return FALSE;
}
if (y+srcrect.bottom<=destrect.top)
{
return FALSE;
}
//如果源表面有超出目標表面的情況,則進行裁剪:
if (x+srcrect.right>destrect.right)
{
srcrect.right-=x+srcrect.right-destrect.right;
}
if (y+srcrect.bottom>destrect.bottom)
{
srcrect.bottom-=y+srcrect.bottom-destrect.bottom;
}
RECT dr;
if (x<0)
{
srcrect.left=-x;
x=0;
dr.left=x;
dr.top=y;
dr.right=x+(srcrect.right-srcrect.left);
dr.bottom=y+(srcrect.bottom-srcrect.top);
bfastblt=FALSE;
}
if (y<0)
{
srcrect.top=-y;
y=0;
dr.left=x;
dr.top=y;
dr.right=x+(srcrect.left-srcrect.left);
dr.bottom=y+(srcrect.bottom-srcrect.top);
bfastblt=FALSE;
}
DWORD flags;
HRESULT re;
if (bfastblt)
{
flags=DDBLTFAST_WAIT;
if (uck)
{
flags |= DDBLTFAST_SRCCOLORKEY;
}
re=m_Dat.lpDestSurf->BltFast( x, y, lpSourceSurf, &srcrect, flags );
if(re!=DD_OK)
{
SaveBltError(re);
return FALSE;
}
}
else
{
flags=DDBLT_WAIT;
if (uck)
{
flags |= DDBLT_KEYSRC;
}
re=m_Dat.lpDestSurf->Blt( &dr, lpSourceSurf, &srcrect, flags, 0 );
if(re!=DD_OK)
{
SaveBltError(re);
return FALSE;
}
}
return TRUE;
}
BOOL CDDrawSurf::SetColorKey(COLORREF clr)
{
COLORREF rgbT;
HDC hdc;
DWORD dw = CLR_INVALID;
DDSURFACEDESC2 ddsd;
HRESULT hres;
if (clr != CLR_INVALID && lpSourceSurf->GetDC(&hdc) == DD_OK)
{
rgbT = GetPixel(hdc, 0, 0); //保存舊的像素點顏色
SetPixel(hdc, 0, 0, clr); //設置COLORREF指定的顏色
lpSourceSurf->ReleaseDC(hdc);
}
ddsd.dwSize = sizeof(ddsd);
while ((hres = lpSourceSurf->Lock(NULL, &ddsd, 0, NULL)) == DDERR_WASSTILLDRAWING)
;
if (hres == DD_OK)
{
dw = *(DWORD *) ddsd.lpSurface;//讀取顏色
if (ddsd.ddpfPixelFormat.dwRGBBitCount < 32)
dw &= (1 << ddsd.ddpfPixelFormat.dwRGBBitCount) - 1;
lpSourceSurf->Unlock(NULL);
}
if (clr != CLR_INVALID && lpSourceSurf->GetDC(&hdc) == DD_OK)
{
SetPixel(hdc, 0, 0, rgbT);
lpSourceSurf->ReleaseDC(hdc);
}
//設置色鍵:
DDCOLORKEY ddck;
ddck.dwColorSpaceLowValue = dw;
ddck.dwColorSpaceHighValue = ddck.dwColorSpaceLowValue;
if(lpSourceSurf->SetColorKey(DDCKEY_SRCBLT, &ddck)!=DD_OK)
{
return FALSE;
}
return TRUE;
}
BOOL CDDrawSurf::Restore()
{
//先恢復表面:
if(lpSourceSurf->Restore()!=DD_OK)
{
SaveInfo("恢復表面失敗.txt","Restore()失敗!");
return FALSE;
}
//再將位圖數據從Buffer中傳送到表面對應的內存:
if(m_Dat.BPP==24&&m_Dat.DisplayBPP==32)
{
if(!CopyPixels24to32())
{
SaveInfo("拷貝24位位圖信息到32位表面錯誤.txt","CopyPixels()==0");
return FALSE;
}
}
else
{
if(!CopyPixels())
{
SaveInfo("拷貝位圖信息錯誤.txt","CopyPixels()==0");
return FALSE;
}
}
return TRUE;
}
void CDDrawSurf::SaveBltError(HRESULT re)
{
switch(re)
{
case DDERR_GENERIC:
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_GENERIC");
break;
}
case DDERR_INVALIDCLIPLIST:
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_INVALIDCLIPLIST");
break;
}
case DDERR_INVALIDOBJECT:
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_INVALIDOBJECT ");
break;
}
case DDERR_INVALIDPARAMS:
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_INVALIDPARAMS ");
break;
}
case DDERR_INVALIDRECT :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_INVALIDRECT ");
break;
}
case DDERR_NOALPHAHW :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NOALPHAHW ");
break;
}
case DDERR_NOBLTHW :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NOBLTHW ");
break;
}
case DDERR_NOCLIPLIST :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NOCLIPLIST ");
break;
}
case DDERR_NODDROPSHW :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NODDROPSHW ");
break;
}
case DDERR_NOMIRRORHW :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NOMIRRORHW ");
break;
}
case DDERR_NORASTEROPHW :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NORASTEROPHW ");
break;
}
case DDERR_NOROTATIONHW :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NOROTATIONHW ");
break;
}
case DDERR_NOSTRETCHHW :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NOSTRETCHHW ");
break;
}
case DDERR_NOZBUFFERHW:
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_NOZBUFFERHW");
break;
}
case DDERR_SURFACEBUSY :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_SURFACEBUSY ");
break;
}
case DDERR_SURFACELOST :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_SURFACELOST ");
break;
}
case DDERR_UNSUPPORTED :
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_UNSUPPORTED ");
break;
}
case DDERR_WASSTILLDRAWING:
{
SaveInfo("帶色鍵帖表面失敗.txt","DDERR_WASSTILLDRAWING");
break;
}
};
}
void CDDrawSurf::Delete()
{
if(lpSourceSurf!=0)
{
lpSourceSurf->Release();
lpSourceSurf=0;
}
}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -