Direct3D: Added hardware motion blur
This commit is contained in:
parent
1b0ee71fd3
commit
d7031bc615
|
@ -68,7 +68,8 @@ private:
|
|||
D3DDISPLAYMODE mode;
|
||||
D3DPRESENT_PARAMETERS dpp;
|
||||
D3DFORMAT screenFormat;
|
||||
LPDIRECT3DTEXTURE9 emulatedImage;
|
||||
LPDIRECT3DTEXTURE9 emulatedImage[2];
|
||||
unsigned char mbCurrentTexture; // current texture for motion blur
|
||||
int width;
|
||||
int height;
|
||||
RECT destRect;
|
||||
|
@ -83,6 +84,12 @@ private:
|
|||
// 1 3
|
||||
// 0 2
|
||||
|
||||
struct TRANSP_VERTEX {
|
||||
FLOAT x, y, z, rhw;
|
||||
D3DCOLOR color;
|
||||
FLOAT tx, ty;
|
||||
} transpVertices[4];
|
||||
|
||||
void createFont();
|
||||
void destroyFont();
|
||||
void createTexture();
|
||||
|
@ -118,7 +125,9 @@ Direct3DDisplay::Direct3DDisplay()
|
|||
height = 0;
|
||||
failed = false;
|
||||
pFont = NULL;
|
||||
emulatedImage = NULL;
|
||||
emulatedImage[0] = NULL;
|
||||
emulatedImage[1] = NULL;
|
||||
mbCurrentTexture = 0;
|
||||
}
|
||||
|
||||
|
||||
|
@ -297,6 +306,7 @@ bool Direct3DDisplay::initialize()
|
|||
createFont();
|
||||
createTexture();
|
||||
setOption( _T("d3dFilter"), theApp.d3dFilter );
|
||||
setOption( _T("motionBlur"), theApp.d3dMotionBlur );
|
||||
calculateDestRect();
|
||||
|
||||
if(failed) return false;
|
||||
|
@ -353,7 +363,7 @@ void Direct3DDisplay::render()
|
|||
// copy pix to emulatedImage and apply pixel filter if selected
|
||||
D3DLOCKED_RECT lr;
|
||||
|
||||
if( FAILED( hr = emulatedImage->LockRect( 0, &lr, NULL, D3DLOCK_DISCARD ) ) ) {
|
||||
if( FAILED( hr = emulatedImage[ mbCurrentTexture ]->LockRect( 0, &lr, NULL, D3DLOCK_DISCARD ) ) ) {
|
||||
DXTRACE_ERR_MSGBOX( _T("Can not lock texture"), hr );
|
||||
return;
|
||||
} else {
|
||||
|
@ -395,16 +405,29 @@ void Direct3DDisplay::render()
|
|||
break;
|
||||
}
|
||||
}
|
||||
emulatedImage->UnlockRect( 0 );
|
||||
emulatedImage[ mbCurrentTexture ]->UnlockRect( 0 );
|
||||
}
|
||||
|
||||
// set emulatedImage as active texture
|
||||
pDevice->SetTexture( 0, emulatedImage );
|
||||
|
||||
// set current emulatedImage as active texture
|
||||
pDevice->SetTexture( 0, emulatedImage[ mbCurrentTexture ] );
|
||||
|
||||
// render textured triangles
|
||||
pDevice->SetFVF( D3DFVF_XYZRHW | D3DFVF_TEX1 );
|
||||
pDevice->DrawPrimitiveUP( D3DPT_TRIANGLESTRIP, 2, Vertices, sizeof(VERTEX) );
|
||||
|
||||
|
||||
if( theApp.d3dMotionBlur ) {
|
||||
// set old emulatedImage as active texture
|
||||
mbCurrentTexture ^= 1; // XOR 1 = switch 0/1
|
||||
pDevice->SetTexture( 0, emulatedImage[ mbCurrentTexture ] );
|
||||
|
||||
// render textured triangles
|
||||
pDevice->SetFVF( D3DFVF_XYZRHW | D3DFVF_DIFFUSE | D3DFVF_TEX1 );
|
||||
pDevice->DrawPrimitiveUP( D3DPT_TRIANGLESTRIP, 2, transpVertices, sizeof(TRANSP_VERTEX) );
|
||||
}
|
||||
|
||||
|
||||
// render speed and status messages
|
||||
D3DCOLOR color;
|
||||
RECT r;
|
||||
|
@ -511,16 +534,32 @@ void Direct3DDisplay::destroyFont()
|
|||
|
||||
void Direct3DDisplay::createTexture()
|
||||
{
|
||||
if( !emulatedImage ) {
|
||||
if( !emulatedImage[0] ) {
|
||||
HRESULT hr = pDevice->CreateTexture(
|
||||
width, height, 1, // 1 level, no mipmaps
|
||||
D3DUSAGE_DYNAMIC, dpp.BackBufferFormat,
|
||||
D3DPOOL_DEFAULT, // anything else won't work
|
||||
&emulatedImage,
|
||||
&emulatedImage[0],
|
||||
NULL );
|
||||
|
||||
if( FAILED( hr ) ) {
|
||||
DXTRACE_ERR_MSGBOX( _T("createTexture failed"), hr );
|
||||
DXTRACE_ERR_MSGBOX( _T("createTexture(0) failed"), hr );
|
||||
return;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
if( !emulatedImage[1] && theApp.d3dMotionBlur ) {
|
||||
HRESULT hr = pDevice->CreateTexture(
|
||||
width, height, 1, // 1 level, no mipmaps
|
||||
D3DUSAGE_DYNAMIC, dpp.BackBufferFormat,
|
||||
D3DPOOL_DEFAULT, // anything else won't work
|
||||
&emulatedImage[1],
|
||||
NULL );
|
||||
|
||||
if( FAILED( hr ) ) {
|
||||
DXTRACE_ERR_MSGBOX( _T("createTexture(1) failed"), hr );
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -528,9 +567,14 @@ void Direct3DDisplay::createTexture()
|
|||
|
||||
void Direct3DDisplay::destroyTexture()
|
||||
{
|
||||
if( emulatedImage ) {
|
||||
emulatedImage->Release();
|
||||
emulatedImage = NULL;
|
||||
if( emulatedImage[0] ) {
|
||||
emulatedImage[0]->Release();
|
||||
emulatedImage[0] = NULL;
|
||||
}
|
||||
|
||||
if( emulatedImage[1] ) {
|
||||
emulatedImage[1]->Release();
|
||||
emulatedImage[1] = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -593,6 +637,39 @@ void Direct3DDisplay::calculateDestRect()
|
|||
Vertices[3].rhw = 1.0f;
|
||||
Vertices[3].tx = 1.0f;
|
||||
Vertices[3].ty = 0.0f;
|
||||
|
||||
if( theApp.d3dMotionBlur ) {
|
||||
// configure semi-transparent triangles
|
||||
D3DCOLOR semiTrans = D3DCOLOR_ARGB( 0x7F, 0xFF, 0xFF, 0xFF );
|
||||
transpVertices[0].x = Vertices[0].x;
|
||||
transpVertices[0].y = Vertices[0].y;
|
||||
transpVertices[0].z = Vertices[0].z;
|
||||
transpVertices[0].rhw = Vertices[0].rhw;
|
||||
transpVertices[0].color = semiTrans;
|
||||
transpVertices[0].tx = Vertices[0].tx;
|
||||
transpVertices[0].ty = Vertices[0].ty;
|
||||
transpVertices[1].x = Vertices[1].x;
|
||||
transpVertices[1].y = Vertices[1].y;
|
||||
transpVertices[1].z = Vertices[1].z;
|
||||
transpVertices[1].rhw = Vertices[1].rhw;
|
||||
transpVertices[1].color = semiTrans;
|
||||
transpVertices[1].tx = Vertices[1].tx;
|
||||
transpVertices[1].ty = Vertices[1].ty;
|
||||
transpVertices[2].x = Vertices[2].x;
|
||||
transpVertices[2].y = Vertices[2].y;
|
||||
transpVertices[2].z = Vertices[2].z;
|
||||
transpVertices[2].rhw = Vertices[2].rhw;
|
||||
transpVertices[2].color = semiTrans;
|
||||
transpVertices[2].tx = Vertices[2].tx;
|
||||
transpVertices[2].ty = Vertices[2].ty;
|
||||
transpVertices[3].x = Vertices[3].x;
|
||||
transpVertices[3].y = Vertices[3].y;
|
||||
transpVertices[3].z = Vertices[3].z;
|
||||
transpVertices[3].rhw = Vertices[3].rhw;
|
||||
transpVertices[3].color = semiTrans;
|
||||
transpVertices[3].tx = Vertices[3].tx;
|
||||
transpVertices[3].ty = Vertices[3].ty;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
@ -631,6 +708,27 @@ void Direct3DDisplay::setOption( const char *option, int value )
|
|||
if( !_tcscmp( option, _T("fullScreenStretch") ) ) {
|
||||
calculateDestRect();
|
||||
}
|
||||
|
||||
if( !_tcscmp( option, _T("motionBlur") ) ) {
|
||||
switch( value )
|
||||
{
|
||||
case 0:
|
||||
mbCurrentTexture = 0;
|
||||
pDevice->SetRenderState( D3DRS_ALPHABLENDENABLE, FALSE );
|
||||
break;
|
||||
case 1:
|
||||
// enable vertex alpha blending
|
||||
pDevice->SetRenderState( D3DRS_ALPHABLENDENABLE, TRUE );
|
||||
pDevice->SetRenderState( D3DRS_DIFFUSEMATERIALSOURCE, D3DMCS_COLOR1 );
|
||||
pDevice->SetRenderState( D3DRS_SRCBLEND, D3DBLEND_SRCALPHA );
|
||||
pDevice->SetRenderState( D3DRS_DESTBLEND, D3DBLEND_INVSRCALPHA );
|
||||
// apply vertex alpha values to texture
|
||||
pDevice->SetTextureStageState( 0, D3DTSS_ALPHAARG1, D3DTA_DIFFUSE );
|
||||
calculateDestRect();
|
||||
createTexture(); // create the second texture
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
@ -658,6 +756,7 @@ bool Direct3DDisplay::resetDevice()
|
|||
}
|
||||
createTexture();
|
||||
setOption( _T("d3dFilter"), theApp.d3dFilter );
|
||||
setOption( _T("motionBlur"), theApp.d3dMotionBlur );
|
||||
failed = false;
|
||||
return true;
|
||||
}
|
||||
|
|
|
@ -450,6 +450,8 @@ BEGIN_MESSAGE_MAP(MainWnd, CWnd)
|
|||
ON_UPDATE_COMMAND_UI(ID_OUTPUTAPI_OPENAL, &MainWnd::OnUpdateOutputapiOpenal)
|
||||
ON_COMMAND(ID_OUTPUTAPI_OALCONFIGURATION, &MainWnd::OnOutputapiOalconfiguration)
|
||||
ON_UPDATE_COMMAND_UI(ID_OUTPUTAPI_OALCONFIGURATION, &MainWnd::OnUpdateOutputapiOalconfiguration)
|
||||
ON_COMMAND(ID_RENDERAPI_D3DMOTIONBLUR, &MainWnd::OnRenderapiD3dmotionblur)
|
||||
ON_UPDATE_COMMAND_UI(ID_RENDERAPI_D3DMOTIONBLUR, &MainWnd::OnUpdateRenderapiD3dmotionblur)
|
||||
END_MESSAGE_MAP()
|
||||
|
||||
|
||||
|
|
|
@ -443,6 +443,8 @@ public:
|
|||
afx_msg void OnUpdateOutputapiOalconfiguration(CCmdUI *pCmdUI);
|
||||
afx_msg void OnMoving(UINT fwSide, LPRECT pRect);
|
||||
afx_msg void OnSizing(UINT fwSide, LPRECT pRect);
|
||||
afx_msg void OnRenderapiD3dmotionblur();
|
||||
afx_msg void OnUpdateRenderapiD3dmotionblur(CCmdUI *pCmdUI);
|
||||
};
|
||||
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
|
|
|
@ -2028,3 +2028,22 @@ void MainWnd::OnUpdateOutputapiOalconfiguration(CCmdUI *pCmdUI)
|
|||
pCmdUI->Enable( FALSE );
|
||||
#endif
|
||||
}
|
||||
|
||||
void MainWnd::OnRenderapiD3dmotionblur()
|
||||
{
|
||||
#ifndef NO_D3D
|
||||
theApp.d3dMotionBlur = !theApp.d3dMotionBlur;
|
||||
if( theApp.display ) {
|
||||
theApp.display->setOption( _T("motionBlur"), theApp.d3dMotionBlur ? 1 : 0 );
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
void MainWnd::OnUpdateRenderapiD3dmotionblur(CCmdUI *pCmdUI)
|
||||
{
|
||||
#ifndef NO_D3D
|
||||
pCmdUI->SetCheck( theApp.d3dMotionBlur ? 1 : 0 );
|
||||
#else
|
||||
pCmdUI->Enable( FALSE );
|
||||
#endif
|
||||
}
|
||||
|
|
|
@ -287,7 +287,10 @@ VBA::VBA()
|
|||
ddrawUsingEmulationOnly = false;
|
||||
ddrawDebug = false;
|
||||
ddrawUseVideoMemory = false;
|
||||
#ifndef NO_D3D
|
||||
d3dFilter = 0;
|
||||
d3dMotionBlur = false;
|
||||
#endif
|
||||
glFilter = 0;
|
||||
GLSLShaders = 0;
|
||||
skin = NULL;
|
||||
|
@ -1517,10 +1520,14 @@ void VBA::loadSettings()
|
|||
ddrawUseVideoMemory = regQueryDwordValue("ddrawUseVideoMemory", true) ? true : false;
|
||||
tripleBuffering = regQueryDwordValue("tripleBuffering", false) ? true : false;
|
||||
|
||||
#ifndef NO_D3D
|
||||
d3dFilter = regQueryDwordValue("d3dFilter", 1);
|
||||
if(d3dFilter < 0 || d3dFilter > 1)
|
||||
d3dFilter = 1;
|
||||
|
||||
d3dMotionBlur = ( regQueryDwordValue("d3dMotionBlur", 0) == 1 ) ? true : false;
|
||||
#endif
|
||||
|
||||
glFilter = regQueryDwordValue("glFilter", 1);
|
||||
if(glFilter < 0 || glFilter > 1)
|
||||
glFilter = 1;
|
||||
|
@ -2583,7 +2590,11 @@ void VBA::saveSettings()
|
|||
regSetDwordValue("ddrawUseVideoMemory", ddrawUseVideoMemory);
|
||||
regSetDwordValue("tripleBuffering", tripleBuffering);
|
||||
|
||||
#ifndef NO_D3D
|
||||
regSetDwordValue("d3dFilter", d3dFilter);
|
||||
regSetDwordValue("d3dMotionBlur", d3dMotionBlur ? 1 : 0);
|
||||
#endif
|
||||
|
||||
regSetDwordValue("glFilter", glFilter);
|
||||
regSetDwordValue("GLSLShaders", GLSLShaders);
|
||||
|
||||
|
|
|
@ -161,7 +161,10 @@ class VBA : public CWinApp
|
|||
bool ddrawUsingEmulationOnly;
|
||||
bool ddrawDebug;
|
||||
bool ddrawUseVideoMemory;
|
||||
#ifndef NO_D3D
|
||||
int d3dFilter;
|
||||
bool d3dMotionBlur;
|
||||
#endif
|
||||
int glFilter;
|
||||
int GLSLShaders;
|
||||
bool dinputKeyFocus;
|
||||
|
|
|
@ -1592,6 +1592,7 @@ BEGIN
|
|||
MENUITEM "Direct&3D", ID_OPTIONS_VIDEO_RENDERMETHOD_DIRECT3D
|
||||
MENUITEM " Filter: Nearest", ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DNOFILTER
|
||||
MENUITEM " Filter: Bilinear", ID_OPTIONS_VIDEO_RENDEROPTIONS_D3DBILINEAR
|
||||
MENUITEM " Motion Blur", ID_RENDERAPI_D3DMOTIONBLUR
|
||||
MENUITEM SEPARATOR
|
||||
MENUITEM "&OpenGL", ID_OPTIONS_VIDEO_RENDERMETHOD_OPENGL
|
||||
MENUITEM " Filter: Nearest", ID_OPTIONS_VIDEO_RENDEROPTIONS_GLNEAREST
|
||||
|
|
|
@ -822,13 +822,15 @@
|
|||
#define ID_RENDERAPI_FILTER 40351
|
||||
#define ID_OPTIONS_VIDEO_RENDEROPTIONS_GLSLSHADERS 40352
|
||||
#define ID_OPTIONS_SPEED 40353
|
||||
#define ID_RENDERAPI_MOTIONBLUR 40354
|
||||
#define ID_RENDERAPI_D3DMOTIONBLUR 40355
|
||||
|
||||
// Next default values for new objects
|
||||
//
|
||||
#ifdef APSTUDIO_INVOKED
|
||||
#ifndef APSTUDIO_READONLY_SYMBOLS
|
||||
#define _APS_NEXT_RESOURCE_VALUE 161
|
||||
#define _APS_NEXT_COMMAND_VALUE 40354
|
||||
#define _APS_NEXT_COMMAND_VALUE 40356
|
||||
#define _APS_NEXT_CONTROL_VALUE 1272
|
||||
#define _APS_NEXT_SYMED_VALUE 103
|
||||
#endif
|
||||
|
|
Loading…
Reference in New Issue