mirror of
https://github.com/kevinbentley/Descent3.git
synced 2025-01-22 19:55:23 +00:00
288 lines
7.9 KiB
C++
288 lines
7.9 KiB
C++
/*
|
|
* $Logfile: /DescentIII/Main/ddgr_win32/ddgrWin32DX.cpp $
|
|
* $Revision: 2 $
|
|
* $Date: 9/12/97 4:13p $
|
|
* $Author: Samir $
|
|
*
|
|
* DirectX implementation of DDGR library.
|
|
*
|
|
* $Log: /DescentIII/Main/ddgr_win32/ddgrWin32DX.cpp $
|
|
*
|
|
* 2 9/12/97 4:13p Samir
|
|
* Added some private data access functions and more DirectX
|
|
* functionality.
|
|
*
|
|
* $NoKeywords: $
|
|
*/
|
|
|
|
#include "DDAccess.h" // Allow this module full DD access
|
|
#include "ddgrWin32.h"
|
|
#include "ddgrWin32DX.h"
|
|
#include "Application.h"
|
|
#include "mono.h"
|
|
#include "pserror.h"
|
|
|
|
tDDGRDXInternalData DDGR_DX_lib_data;
|
|
|
|
// Initializes DX subsystem
|
|
bool ddgr_dx_Init(oeApplication *app) {
|
|
DX_DATA(hPrimaryWnd) = (HWND)((oeWin32Application *)app)->m_hWnd;
|
|
|
|
DX_DATA(lpDD) = NULL;
|
|
|
|
// Initialize DirectDraw system if we're going fullscreen, and GDIX system
|
|
HRESULT hres;
|
|
|
|
hres = DirectDrawCreate(NULL, &DX_DATA(lpDD), NULL);
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Failure to initialize DirectDraw driver (%d)", LOWORD(hres));
|
|
return false;
|
|
}
|
|
|
|
hres =
|
|
DX_DATA(lpDD)->SetCooperativeLevel(DX_DATA(hPrimaryWnd), DDSCL_ALLOWREBOOT | DDSCL_EXCLUSIVE | DDSCL_FULLSCREEN);
|
|
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Failed to set DirectDraw fullscreen access level (%d)", LOWORD(hres));
|
|
return false;
|
|
}
|
|
|
|
DX_DATA(vidrefs) = 0; // reset video reference count.
|
|
DX_DATA(init) = true;
|
|
|
|
mprintf((0, "DX system initialized.\n"));
|
|
|
|
return true;
|
|
}
|
|
|
|
// Closes DX subsystem
|
|
void ddgr_dx_Close() {
|
|
ASSERT(DX_DATA(vidrefs) == 0);
|
|
|
|
if (DX_DATA(lpDD)) {
|
|
DX_DATA(lpDD)->RestoreDisplayMode();
|
|
DX_DATA(lpDD)->Release();
|
|
}
|
|
DX_DATA(lpDD) = NULL;
|
|
DX_DATA(hPrimaryWnd) = NULL;
|
|
DX_DATA(init) = false;
|
|
|
|
mprintf((0, "DX system closed.\n"));
|
|
}
|
|
|
|
// Initializes the display for use with the DX subsystem.
|
|
bool ddgr_dx_surf_InitVideo(ddgr_surface *sf) {
|
|
tDXSurface *dxsf;
|
|
DDSURFACEDESC ddsd;
|
|
|
|
// we need to create a primary surface from the display mode we set, and possibly a backbuffer if
|
|
// requested.
|
|
if (DX_DATA(vidrefs) == 0) {
|
|
HRESULT hres;
|
|
DDSCAPS ddscaps;
|
|
|
|
hres = DX_DATA(lpDD)->SetDisplayMode(sf->w, sf->h, sf->bpp);
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Unable to set DirectDraw display mode (%d)\n", LOWORD(hres));
|
|
return false;
|
|
}
|
|
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
ddsd.dwFlags = DDSD_CAPS | DDSD_BACKBUFFERCOUNT;
|
|
ddsd.dwBackBufferCount = 1;
|
|
ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE | DDSCAPS_FLIP | DDSCAPS_COMPLEX;
|
|
|
|
hres = DX_DATA(lpDD)->CreateSurface(&ddsd, &DX_DATA(vidsurf.lpdds), NULL);
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Unable to capture DirectDraw display surface (%d)\n", LOWORD(hres));
|
|
return false;
|
|
}
|
|
|
|
ddscaps.dwCaps = DDSCAPS_BACKBUFFER;
|
|
|
|
hres = DX_DATA(vidsurf.lpdds)->GetAttachedSurface(&ddscaps, &DX_DATA(vidsurf.lpddsback));
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Unable to capture DirectDraw display back surface (%d)\n", LOWORD(hres));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
DX_DATA(vidrefs)++; // increment vidref count
|
|
|
|
dxsf = new tDXSurface;
|
|
dxsf->lpdds = DX_DATA(vidsurf.lpdds);
|
|
|
|
// this means that we are capturing the screen, not creating a screen.
|
|
// get the screen's width and height and bitdepth
|
|
if (sf->flags == 0) {
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
dxsf->lpdds->GetSurfaceDesc(&ddsd);
|
|
sf->w = ddsd.dwWidth;
|
|
sf->h = ddsd.dwHeight;
|
|
sf->bpp = ddsd.ddpfPixelFormat.dwRGBBitCount;
|
|
}
|
|
|
|
// we create a dummy surface for the screen's back buffer, or our accessable buffer
|
|
// specifying SURFFLAG_BACKBUFFER ensures that we create a GDI bitmap of size requested
|
|
// in 'sf' and it will be our renderable bitmap, or OUR primary render buffer
|
|
if (sf->flags & SURFFLAG_BACKBUFFER) {
|
|
// set our front buffer to this newly created surface and delete the gdi bitmap object
|
|
dxsf->lpddsback = DX_DATA(vidsurf.lpddsback);
|
|
dxsf->backbuffer = true;
|
|
} else {
|
|
dxsf->lpddsback = NULL;
|
|
dxsf->backbuffer = false;
|
|
}
|
|
|
|
sf->obj = (void *)dxsf;
|
|
|
|
return true;
|
|
}
|
|
|
|
// Deinitializes the video surface
|
|
void ddgr_dx_surf_CloseVideo(ddgr_surface *sf) {
|
|
tDXSurface *bm = (tDXSurface *)sf->obj;
|
|
|
|
ASSERT(DX_DATA(vidrefs) > 0);
|
|
|
|
DX_DATA(vidrefs)--;
|
|
if (DX_DATA(vidrefs) == 0) {
|
|
DX_DATA(vidsurf.lpdds)->Release();
|
|
DX_DATA(vidsurf.lpdds) = NULL;
|
|
DX_DATA(vidsurf.lpddsback) = NULL;
|
|
}
|
|
|
|
delete bm;
|
|
}
|
|
|
|
// flips the buffers in a surface. really only useful for video screens
|
|
bool ddgr_dx_surf_FlipVideo(ddgr_surface *sf) {
|
|
tDXSurface *bm = (tDXSurface *)sf->obj;
|
|
HRESULT hres;
|
|
|
|
if (bm->backbuffer && (sf->flags & SURFFLAG_BACKBUFFER)) {
|
|
hres = bm->lpdds->Flip(NULL, 0);
|
|
if (hres == DDERR_WASSTILLDRAWING || hres == DD_OK)
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/* input:
|
|
sf->name is optional.
|
|
sf->w, sf->h, sf->bpp are mandatory
|
|
sf->type and sf->flags are mandatory.
|
|
output:
|
|
sf->obj = surface object.
|
|
sf->locks = 0
|
|
*/
|
|
bool ddgr_dx_surf_Create(ddgr_surface *sf) { return true; }
|
|
|
|
void ddgr_dx_surf_Destroy(ddgr_surface *sf) {}
|
|
|
|
/* retrieves a pointer to surface memory. allowed to lock one surface multiple times.
|
|
ptr is the returned pointer to surface memory. used to unlock surface also
|
|
rowsize is the size in bytes of one row of memory.
|
|
*/
|
|
bool ddgr_dx_surf_Lock(ddgr_surface *sf, void **ptr, int *rowsize) {
|
|
HRESULT hres;
|
|
DDSURFACEDESC ddsd;
|
|
tDXSurface *bm = (tDXSurface *)sf->obj;
|
|
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
|
|
if (bm->backbuffer)
|
|
hres = bm->lpddsback->Lock(NULL, &ddsd, DDLOCK_WAIT, NULL);
|
|
else
|
|
hres = bm->lpdds->Lock(NULL, &ddsd, DDLOCK_WAIT, NULL);
|
|
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Unable to set lock DirectDraw surface(%d)\n", LOWORD(hres));
|
|
return false;
|
|
}
|
|
|
|
*rowsize = (int)ddsd.lPitch;
|
|
*ptr = ddsd.lpSurface;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ddgr_dx_surf_Unlock(ddgr_surface *sf, void *ptr) {
|
|
HRESULT hres;
|
|
tDXSurface *bm = (tDXSurface *)sf->obj;
|
|
|
|
if (bm->backbuffer)
|
|
hres = bm->lpddsback->Unlock(ptr);
|
|
else
|
|
hres = bm->lpdds->Unlock(ptr);
|
|
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Unable to unlock DirectDraw surface(%d)\n", LOWORD(hres));
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// attaches an OS handle to a surface
|
|
void ddgr_dx_surf_AttachHandle(ddgr_surface *sf, unsigned handle) {}
|
|
|
|
// The only graphic primatives
|
|
// clear
|
|
// blt.
|
|
void ddgr_dx_surf_Clear(ddgr_surface *dsf, ddgr_color col, int l, int t, int w, int h) {
|
|
HRESULT hres;
|
|
DDBLTFX ddbltfx;
|
|
tDXSurface *dbm = (tDXSurface *)dsf->obj;
|
|
LPDIRECTDRAWSURFACE lpdds;
|
|
RECT rect;
|
|
|
|
lpdds = (dbm->backbuffer) ? dbm->lpddsback : dbm->lpdds;
|
|
|
|
memset(&ddbltfx, 0, sizeof(DDBLTFX));
|
|
ddbltfx.dwSize = sizeof(ddbltfx);
|
|
ddbltfx.dwFillColor = RGB(DDGR_COLOR_RED(col), DDGR_COLOR_GREEN(col), DDGR_COLOR_BLUE(col));
|
|
SetRect(&rect, l, t, l + w - 1, t + h - 1);
|
|
|
|
hres = lpdds->Blt(&rect, // dest rect
|
|
NULL, // src surface
|
|
NULL, // src rect
|
|
DDBLT_COLORFILL | DDBLT_WAIT, &ddbltfx);
|
|
}
|
|
|
|
bool ddgr_dx_surf_Blt(ddgr_surface *dsf, int dx, int dy, ddgr_surface *ssf, int sx, int sy, int sw, int sh) {
|
|
HRESULT hres;
|
|
tDXSurface *sbm = (tDXSurface *)ssf->obj;
|
|
tDXSurface *dbm = (tDXSurface *)dsf->obj;
|
|
LPDIRECTDRAWSURFACE lpddsd;
|
|
LPDIRECTDRAWSURFACE lpddss;
|
|
RECT srect, drect;
|
|
|
|
if (ssf->flags & SURFFLAG_BACKBUFFER)
|
|
lpddss = sbm->lpddsback;
|
|
else
|
|
lpddss = sbm->lpdds;
|
|
|
|
if (dsf->flags & SURFFLAG_BACKBUFFER)
|
|
lpddsd = dbm->lpddsback;
|
|
else
|
|
lpddsd = dbm->lpdds;
|
|
|
|
SetRect(&srect, sx, sy, sx + sw - 1, sy + sh - 1);
|
|
SetRect(&drect, dx, dy, dx + sw - 1, dy + sh + 1);
|
|
|
|
hres = lpddsd->Blt(&drect, lpddss, &srect, DDBLT_WAIT, NULL);
|
|
if (hres != DD_OK) {
|
|
ddgr_PushError("Unable to blt surface %s to DirectDraw surface %s(%d)\n", ssf->name, dsf->name, LOWORD(hres));
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|