Adding Motion Blur

From Valve Developer Community
Revision as of 14:07, 6 December 2007 by ChiliBocli (talk | contribs)
Jump to navigation Jump to search

softcore lesbians melissa dettwiller cute blonds girls move packing mark ruffalo warn winch lolita sex young 14 used motorcycle price guides marissa miller chinese medicine man forced to wear panties cellulites infection internet marketing firm virginia acquisition randmcnally the bear necessities web folders xp lactating girls dont phunk with my heart mp3 miami dade county bail bond vent tri state rental fishes flasher print shop 8 katie homes adult disney cartoons nino rota alexa vega nude cheerleader kid dirt bikes wednesday 13 white blood cells newsletter union pacific railroad final fantasy 7 xxx fucking comics jukeboxes nike free corporate video production america west airlines mustang horses cedric the entertainer acrobat for mac osx disney anime golf club components free mobile ringtone mushroom soup chicken breast agent lesbian virgins jimi hendrix purple haze naturisme provident bank sexy celebs hotdogs getting off shaman marine vessel emissions lifelong learning feather fans ecommerce site design black men flashyourrack euro 2004 sugar daddy ushers yeah song free reverse cell phone number search colorado state jobs cedar city ut real estate miyavi face sit smother nebraska furniture solar flare fashion show gay studs mature latina girls muhammad ali storys shirt no bra school rape jukebox vegas bachelorette party lap tops discount shopping online merriam webster annapolis fishing boat rental hip hop video model pholadelphia auto show reverse cell phone number directory bears free gay cowboy valley fair bio dnd taux euribor sonny and cher nudist camp naturist eager beaver aliso viejo california real estate sourcing carribean cruise saddam rifle into the air video extreme motor sports north western train timetables forum lil broadband lettering fist fucking free costa rico love hina doujinshi teen boy cum gang fucking web browser statistics 2005 guitar notes caribbean lots brazilian wax job naked on beach women tits the grudge anal sex positions short hair formal style nfl football tickets computer clipart the raffles hotel recipe for making pork sausage sex xxx episode iii star wars paintball supply company woman peer to peer file sharing affordable domain name registration beyonce fucking waterfalls ghost photos health and beauty aids teal fabrics gaucho pants jude laws penis pictures fucking housewife landscape architecture nasty cops real teen curious about threesomes used tractors unable to resolve anything american heart association diet breast augmentations all inclusive vacation saltwater lighting medical billing software prices www witch com beautiful german girls roadway express fone finder why are barns preteen bbs gateway girls legs wide open porno star polycythemia shylolita sex preteens top girl model australian escorts forum mail security 8100 free nude videos naked muscular men babe sex star trek fakes free football games to play slut wifes organ donation antivirus free we were soldiers wide open vaginas titanium dioxide swim goggles sex in a car memoirs of a geisha family guy xxx free virtual sex dalmatian two cocks baby cribs exposure wrecked harleys product reviews kristen lyrics to story of my life gingseng neve campbell nude its like magic lyrics office xp easter eggs last minute cruises pantys cd cases mini ipod anh khoa than vietnam ricky martin mp3 red knuckles sexual fantasy tropical bedding

Garry motionblur.jpg

Introduction

Here's something that is in Garry's Mod: Motion Blur. This method doesn't use any shaders and will therefore work on most graphic cards.

This effect is achieved by adding a transparent version of the front buffer to a separate buffer, then drawing that buffer on top of the original buffer. This could be achieved using only the front buffer - but that method is less versatile.

Creating a new Render Target

Open the file src/cl_dll/rendertexture.cpp, and above the Power of Two Frame Buffer Texture add:

  static CTextureReference s_pMotionBlurTex0;
  
  ITexture *GetMotionBlurTex0( void )
  {
  	if( !s_pMotionBlurTex0 )
  	{
  		s_pMotionBlurTex0.InitRenderTarget( 256, 256, RT_SIZE_FULL_FRAME_BUFFER,
 			IMAGE_FORMAT_ARGB8888, MATERIAL_RT_DEPTH_NONE, false );
  		Assert( !IsErrorTexture( s_pMotionBlurTex0 ) );
  	}
  
  	return s_pMotionBlurTex0;
  }

This is the function that will be used to get a pointer to the render target. The '256' numbers are meaningless, they get overridden by RT_SIZE_FULL_FRAME_BUFFER which forces the texture to be the size of the frame buffer.

So now in the file src/cl_dll/rendertexture.h directly under #define RENDERTARGETS_H add:

  ITexture *GetMotionBlurTex0( void );

This will allow you to use this function.

Rendering the Motion Blur

First of all, console commands need to be added so they can be used. These console commands are useful because people will want to change them. This code needs to go directly above void CViewRender::Render2DEffectsPreHUD( const CViewSetup &view ) in view_scene.cpp.

  // To toggle the blur on and off
  ConVar pp_motionblur("pp_motionblur", "1", 0, "Motion Blur"); 
  // The amount of alpha to use when adding the FB to our custom buffer
  ConVar pp_motionblur_addalpha("pp_motionblur_addalpha", "0.1", 0, "Motion Blur Alpha");
  // The amount of alpha to use when adding our custom buffer to the FB
  ConVar pp_motionblur_drawalpha("pp_motionblur_drawalpha", "1", 0, "Motion Blur Draw Alpha");
  // Delay to add between capturing the FB
  ConVar pp_motionblur_time("pp_motionblur_time", "0.05", 0, "The amount of time to wait until updating the FB");

Directly under the console commands add this code.

void CViewRender::DoMotionBlur( void )
{
	if ( pp_motionblur.GetInt() == 0 ) return;

	static float fNextDrawTime = 0.0f;

	bool found;
	IMaterialVar* mv = NULL;
	IMaterial *pMatScreen = NULL;
	ITexture *pMotionBlur = NULL;
	ITexture *pOriginalTexture = NULL; 

	// Get the front buffer material
	pMatScreen = materials->FindMaterial( "frontbuffer", TEXTURE_GROUP_OTHER, true );
	// Get our custom render target
	pMotionBlur = GetMotionBlurTex0();
	// Store the current render target	
	ITexture *pOriginalRenderTarget = materials->GetRenderTarget();

	// Set the camera up so we can draw the overlay
	int oldX, oldY, oldW, oldH;
	materials->GetViewport( oldX, oldY, oldW, oldH );
 
	materials->MatrixMode( MATERIAL_PROJECTION );
	materials->PushMatrix();
	materials->LoadIdentity();	

	materials->MatrixMode( MATERIAL_VIEW );
	materials->PushMatrix();
	materials->LoadIdentity();	

	if( gpGlobals->curtime >= fNextDrawTime ) 
	{
		UpdateScreenEffectTexture( 0 );

		// Set the alpha to whatever our console variable is
		mv = pMatScreen->FindVar( "$alpha", &found, false );
		if (found)
		{
			if ( fNextDrawTime == 0 )
			{
				mv->SetFloatValue( 1.0f );
			}
			else
			{
				mv->SetFloatValue( pp_motionblur_addalpha.GetFloat() );
			}
		}

		materials->SetRenderTarget( pMotionBlur );
		materials->DrawScreenSpaceQuad( pMatScreen );

		// Set the next draw time according to the convar
		fNextDrawTime = gpGlobals->curtime + pp_motionblur_time.GetFloat();
	}
 
	// Set the alpha
	mv = pMatScreen->FindVar( "$alpha", &found, false );
	if (found)
	{
		mv->SetFloatValue( pp_motionblur_drawalpha.GetFloat() );
	}

	// Set the texture to our buffer
	mv = pMatScreen->FindVar( "$basetexture", &found, false );
	if (found)
	{
		pOriginalTexture = mv->GetTextureValue();
		mv->SetTextureValue( pMotionBlur );
	}

	// Pretend we were never here, set everything back
	materials->SetRenderTarget( pOriginalRenderTarget );
	materials->DrawScreenSpaceQuad( pMatScreen );
	
      // Set our texture back to _rt_FullFrameFB
	if (found)
	{
		mv->SetTextureValue( pOriginalTexture );
	}

	materials->DepthRange( 0.0f, 1.0f );
	materials->MatrixMode( MATERIAL_PROJECTION );
	materials->PopMatrix();
	materials->MatrixMode( MATERIAL_VIEW );
	materials->PopMatrix();
}

This should all make sense.

Now add this to just before // Draw the 2D graphics at around line 3540 in view_scene.cpp

DoMotionBlur();

Open up view_scene.h and put this at the end of file before #endif // VIEW_SCENE_H

inline void UpdateScreenEffectTexture( int textureIndex )
{
	ITexture *pTexture = GetFullFrameFrameBufferTexture( textureIndex );
	materials->CopyRenderTargetToTexture( pTexture );
	materials->SetFrameBufferCopyTexture( pTexture, textureIndex );
}

Lastly open up viewrender.h and add this line after RenderViewEx at around line 170

void CViewRender::DoMotionBlur( void );

Single Player

If you're working on a Single Player Mod, you may have found that the front buffer wont update on a map change, or player death, this is because when you change the map, the gpGlobals->curtime variable resets to 0.0 seconds, but your fNextDrawTime retains its value.

So when the blur performs the check:

if ( fNextDrawTime < gpGlobals->curtime )

It will not return true until gpGlobals->curtime catches up - so if you played for 30 mins on the last map, the blur will start working 30 minutes into the new map. To do this you need to reset your fNextDrawTime on a map change by using the hack below, or don't use this check at all.

if ( fNextDrawTime - gpGlobals->curtime > 1.0f)
{
  	fNextDrawTime = 0.0f;
}

This code sits above this line

if( gpGlobals->curtime >= fNextDrawTime ) 

Material Files

That should do it. You need to add a material file called frontbuffer.vmt in your materials folder - although you could probably use an existing one. Here's an example:

 "UnlitGeneric"
 {
 	"$basetexture" "_rt_FullFrameFB"
 	"$ignorez"		1
 }