topical media & game development 
  
 
 
 
 
  
    
    
  
 game-xna-intro-XnaShooterUIProject-Shaders-ShaderEffect.cs / cs
  // Project: XnaGraphicEngine, File: ShaderEffect.cs
  // Namespace: XnaGraphicEngine.Shaders, Class: ShaderEffect
  // Path: C:\code\XnaGraphicEngine\Shaders, Author: Abi
  // Code lines: 904, Size of file: 24,94 KB
  // Creation date: 07.09.2006 05:56
  // Last modified: 05.11.2006 00:32
  // Generated with Commenter by abi.exDream.com
  
  #region Using directives
  if DEBUG
  //using NUnit.Framework;
  endif
  using Microsoft.Xna.Framework;
  using Microsoft.Xna.Framework.Graphics;
  using System;
  using System.Collections.Generic;
  using System.IO;
  using System.Text;
  using XnaGraphicEngine.Graphics;
  using XnaGraphicEngine.Helpers;
  using Texture = XnaGraphicEngine.Graphics.Texture;
  using XnaTexture = Microsoft.Xna.Framework.Graphics.Texture;
  using Model = XnaGraphicEngine.Graphics.Model;
  using XnaModel = Microsoft.Xna.Framework.Graphics.Model;
  using XnaGraphicEngine.Game;
  #endregion
  
  namespace XnaGraphicEngine.Shaders
  {
   <summary>
 Shader effect class. You can either directly use this class by
 providing a fx filename in the constructor or derive from this class
 for special shader functionality (see post screen shaders for a more
 complex example).
 
</summary>
        public class ShaderEffect : IGraphicContent
        {
                #region Some shaders
 
<summary>
 Line rendering shader
 
</summary>
                public static ShaderEffect lineRendering =
                        new ShaderEffect("LineRendering.fx");
   <summary>
 Normal mapping shader
 
</summary>
                public static ShaderEffect normalMapping =
                        new ShaderEffect("NormalMapping.fx");
                #endregion
                  #region Variables
   <summary>
 Content name for this shader
 
</summary>
                private string shaderContentName = "";
   <summary>
 Effect
 
</summary>
                protected Effect xnaEffect = null;
 
<summary>
 Effect handles for shaders.
 
</summary>
                protected EffectParameter worldViewProj,
                        viewProj,
                        world,
                        viewInverse,
                        lightDir,
                        ambientColor,
                        diffuseColor,
                        specularColor,
                        specularPower,
                        diffuseTexture,
                        normalTexture;
                #endregion
                  #region Properties
   <summary>
 Is this shader valid to render? If not we can't perform any rendering.
 
</summary>
 <returns>Bool
</returns>
                public bool Valid
                {
                        get
                        {
                                return xnaEffect != null;
                        } // get
                } // Valid
   <summary>
 Effect
 
</summary>
 <returns>Effect
</returns>
                public Effect Effect
                {
                        get
                        {
                                return xnaEffect;
                        } // get
                } // Effect
   <summary>
 Number of techniques
 
</summary>
 <returns>Int
</returns>
                public int NumberOfTechniques
                {
                        get
                        {
                                return xnaEffect.Techniques.Count;
                        } // get
                } // NumberOfTechniques
   <summary>
 Get technique
 
</summary>
 <param name="techniqueName">Technique name
</param>
 <returns>Effect technique
</returns>
                public EffectTechnique GetTechnique(string techniqueName)
                {
                        return xnaEffect.Techniques
[techniqueName];
                } // GetTechnique(techniqueName)
   <summary>
 Set value helper to set an effect parameter.
 
</summary>
 <param name="param">Param
</param>
 <param name="setMatrix">Set matrix
</param>
                private void SetValue(EffectParameter param,
                        ref Matrix lastUsedMatrix, Matrix newMatrix)
                {
                        
/*obs, always update, matrices change every frame anyway!
 matrix compare takes too long, it eats up almost 50% of this method.
                        if (param != null &&
                                lastUsedMatrix != newMatrix)
  
                          {
                                  lastUsedMatrix = newMatrix;
                                  param.SetValue(newMatrix);
                          } // if (param)
                  } // SetValue(param, setMatrix)
  
   <summary>
 Set value helper to set an effect parameter.
 </summary>
 <param name="param">Param</param>
 <param name="lastUsedVector">Last used vector</param>
 <param name="newVector">New vector</param>
                private void SetValue(EffectParameter param,
                        ref Vector3 lastUsedVector, Vector3 newVector)
                {
                        if (param != null &&
                                lastUsedVector != newVector)
                        {
                                lastUsedVector = newVector;
                                param.SetValue(newVector);
                        } // if (param)
                } // SetValue(param, lastUsedVector, newVector)
   <summary>
 Set value helper to set an effect parameter.
 </summary>
 <param name="param">Param</param>
 <param name="lastUsedColor">Last used color</param>
 <param name="newColor">New color</param>
                private void SetValue(EffectParameter param,
                        ref Color lastUsedColor, Color newColor)
                {
                        // Note: This check eats few % of the performance, but the color
                        // often stays the change (around 50%).
                        if (param != null &&
                                //slower: lastUsedColor != newColor)
                                lastUsedColor.PackedValue != newColor.PackedValue)
                        {
                                lastUsedColor = newColor;
                                //obs: param.SetValue(ColorHelper.ConvertColorToVector4(newColor));
                                param.SetValue(newColor.ToVector4());
                        } // if (param)
                } // SetValue(param, lastUsedColor, newColor)
   <summary>
 Set value helper to set an effect parameter.
 </summary>
 <param name="param">Param</param>
 <param name="lastUsedValue">Last used value</param>
 <param name="newValue">New value</param>
                private void SetValue(EffectParameter param,
                        ref float lastUsedValue, float newValue)
                {
                        if (param != null &&
                                lastUsedValue != newValue)
                        {
                                lastUsedValue = newValue;
                                param.SetValue(newValue);
                        } // if (param)
                } // SetValue(param, lastUsedValue, newValue)
   <summary>
 Set value helper to set an effect parameter.
 </summary>
 <param name="param">Param</param>
 <param name="lastUsedValue">Last used value</param>
 <param name="newValue">New value</param>
                private void SetValue(EffectParameter param,
                        ref XnaTexture lastUsedValue, XnaTexture newValue)
                {
                        if (param != null &&
                                lastUsedValue != newValue)
                        {
                                lastUsedValue = newValue;
                                param.SetValue(newValue);
                        } // if (param)
                } // SetValue(param, lastUsedValue, newValue)
                  protected Matrix lastUsedWorldViewProjMatrix = Matrix.Identity;
   <summary>
 Set world view proj matrix
 </summary>
                protected Matrix WorldViewProjMatrix
                {
                        set
                        {
                                SetValue(worldViewProj, ref lastUsedWorldViewProjMatrix, value);
                        } // set
                } // WorldViewProjMatrix
                  protected Matrix lastUsedViewProjMatrix = Matrix.Identity;
   <summary>
 Set view proj matrix
 </summary>
                protected Matrix ViewProjMatrix
                {
                        set
                        {
                                SetValue(viewProj, ref lastUsedViewProjMatrix, value);
                        } // set
                } // ViewProjMatrix
                  //obs: protected Matrix lastUsedWorldMatrix = Matrix.Identity;
   <summary>
 Set world matrix
 </summary>
                public Matrix WorldMatrix
                {
                        set
                        {
                                // This is the most used property here.
                                //obs: SetValue(world, ref lastUsedWorldMatrix, value);
                                /*obs, world matrix ALWAYS changes! and it is always used!
                                if (world != null &&
                                        lastUsedWorldMatrix != value)
                                {
                                        lastUsedWorldMatrix = value;
                                        world.SetValue(lastUsedWorldMatrix);
                                } // if (world)
  
  
                                  // Faster, we checked world matrix in constructor.
                                  world.SetValue(value);
                          } // set
                  } // WorldMatrix
  
                  protected Matrix lastUsedInverseViewMatrix = Matrix.Identity;
   <summary>
 Set view inverse matrix
 </summary>
                protected Matrix InverseViewMatrix
                {
                        set
                        {
                                SetValue(viewInverse, ref lastUsedInverseViewMatrix, value);
                        } // set
                } // InverseViewMatrix
                  protected Vector3 lastUsedLightDir = Vector3.Zero;
   <summary>
 Set light direction
 </summary>
                protected Vector3 LightDir
                {
                        set
                        {
                                // Make sure lightDir is normalized (fx files are optimized
                                // to work with a normalized lightDir vector)
                                value.Normalize();
                                // Set negative value, shader is optimized not to negate dir!
                                SetValue(lightDir, ref lastUsedLightDir, -value);
                        } // set
                } // LightDir
                  protected Color lastUsedAmbientColor = ColorHelper.Empty;
   <summary>
 Ambient color
 </summary>
                public Color AmbientColor
                {
                        set
                        {
                                SetValue(ambientColor, ref lastUsedAmbientColor, value);
                        } // set
                } // AmbientColor
                  protected Color lastUsedDiffuseColor = ColorHelper.Empty;
   <summary>
 Diffuse color
 </summary>
                public Color DiffuseColor
                {
                        set
                        {
                                SetValue(diffuseColor, ref lastUsedDiffuseColor, value);
                        } // set
                } // DiffuseColor
                  protected Color lastUsedSpecularColor = ColorHelper.Empty;
   <summary>
 Specular color
 </summary>
                public Color SpecularColor
                {
                        set
                        {
                                SetValue(specularColor, ref lastUsedSpecularColor, value);
                        } // set
                } // SpecularColor
                  private float lastUsedSpecularPower = 0;
   <summary>
 SpecularPower for specular color
 </summary>
                public float SpecularPower
                {
                        set
                        {
                                SetValue(specularPower, ref lastUsedSpecularPower, value);
                        } // set
                } // SpecularPower
                  protected XnaTexture lastUsedDiffuseTexture = null;
   <summary>
 Set diffuse texture
 </summary>
                public Texture DiffuseTexture
                {
                        set
                        {
                                SetValue(diffuseTexture, ref lastUsedDiffuseTexture,
                                        value != null ? value.XnaTexture : null);
                        } // set
                } // DiffuseTexture
                  protected XnaTexture lastUsedNormalTexture = null;
   <summary>
 Set normal texture for normal mapping
 </summary>
                public Texture NormalTexture
                {
                        set
                        {
                                SetValue(normalTexture, ref lastUsedNormalTexture,
                                        value != null ? value.XnaTexture : null);
                        } // set
                } // NormalTexture
                #endregion
                  #region Constructor
                  public ShaderEffect(string shaderName)
                  {
                          if (BaseGame.Device == null)
                                  throw new NullReferenceException(
                                          "XNA device is not initialized, can't create ShaderEffect.");
  
                          shaderContentName = StringHelper.ExtractFilename(shaderName, true);
  
                          Load();
  
                          BaseGame.RegisterGraphicContentObject(this);
                  } // SimpleShader()
                  #endregion
  
                  #region Dispose
   <summary>
 Dispose
 </summary>
                public virtual void Dispose()
                {
                        // Dispose shader effect
                        if (xnaEffect != null)
                                xnaEffect.Dispose();
                        xnaEffect = null;
                } // Dispose()
                #endregion
                  #region Reload effect
   <summary>
 Reload effect (can be useful if we change the fx file dynamically).
 </summary>
                public virtual void Load()
                {
                        if (xnaEffect != null)
                                return;
                          /*obs
                          // Dispose old shader
                          if (effect != null)
                                  Dispose();
  			 */
  
                          // Load shader
                          try
                          {
                                  // We have to try, there is no "Exists" method.
                                  // We could try to check the xnb filename, but why bother? ^^
                                  xnaEffect = BaseGame.Content.Load<Effect>(
                                          Path.Combine(Directories.ContentDirectory, shaderContentName));
                          } // try
  if XBOX360
                          catch (Exception ex)
                          {
                                  Log.Write("Failed to load shader "+shaderContentName+". " +
                                          "Error: " + ex.ToString());
                                  // Rethrow error, app can't continue!
                                  throw ex;
                          }
  else
                          catch
                          {
                                  // Try again by loading by filename (only allowed for windows!)
                                  // Content file was most likely removed for easier testing :)
                                  try
                                  {
                                          CompiledEffect compiledEffect = Effect.CompileEffectFromFile(
                                                  Path.Combine("Shaders", shaderContentName + ".fx"),
                                                  null, null, CompilerOptions.None,
                                                  TargetPlatform.Windows);
  
                                          xnaEffect = new Effect(BaseGame.Device,
                                                  compiledEffect.GetEffectCode(), CompilerOptions.None, null);
                                  } // try
                                  catch (Exception ex)
                                  {
                                          Log.Write("Failed to load shader "+shaderContentName+". " +
                                                  "Error: " + ex.ToString());
                                          // Rethrow error, app can't continue!
                                          throw ex;
                                  } // catch
                          } // catch
  endif
  
                          GetParameters();
                  } // Load()
                  #endregion
  
                  #region Get parameters
   <summary>
 Get parameters, override to support more
 </summary>
                protected virtual void GetParameters()
                {
                        worldViewProj = xnaEffect.Parameters["worldViewProj"];
                        viewProj = xnaEffect.Parameters["viewProj"];
                        world = xnaEffect.Parameters["world"];
                        viewInverse = xnaEffect.Parameters["viewInverse"];
                        lightDir = xnaEffect.Parameters["lightDir"];
                        ambientColor = xnaEffect.Parameters["ambientColor"];
                        diffuseColor = xnaEffect.Parameters["diffuseColor"];
                        specularColor = xnaEffect.Parameters["specularColor"];
                        specularPower = xnaEffect.Parameters["shininess"];
                        diffuseTexture = xnaEffect.Parameters["diffuseTexture"];
                        normalTexture = xnaEffect.Parameters["normalTexture"];
                } // GetParameters()
                #endregion
                  #region SetParameters
   <summary>
 Set parameters, this overload sets all material parameters too.
 </summary>
                public virtual void SetParameters(Material setMat)
                {
                        if (worldViewProj != null)
                                worldViewProj.SetValue(BaseGame.WorldViewProjectionMatrix);
                        if (viewProj != null)
                                viewProj.SetValue(BaseGame.ViewProjectionMatrix);
                        if (world != null)
                                world.SetValue(BaseGame.WorldMatrix);
                        if (viewInverse != null)
                                viewInverse.SetValue(BaseGame.InverseViewMatrix);
                        if (lightDir != null)
                                lightDir.SetValue(BaseGame.LightDirection);
                          // Set all material properties
                          if (setMat != null)
                          {
                                  AmbientColor = setMat.ambientColor;
                                  DiffuseColor = setMat.diffuseColor;
                                  SpecularColor = setMat.specularColor;
                                  SpecularPower = setMat.specularPower;
                                  DiffuseTexture = setMat.diffuseTexture;
                                  NormalTexture = setMat.normalTexture;
                          } // if (setMat)
                  } // SetParameters()
  
   <summary>
 Set parameters, this overload sets all material parameters too.
 </summary>
                public virtual void SetParameters(XnaTexture setTex)
                {
                        if (worldViewProj != null)
                                worldViewProj.SetValue(BaseGame.WorldViewProjectionMatrix);
                        if (viewProj != null)
                                viewProj.SetValue(BaseGame.ViewProjectionMatrix);
                        if (world != null)
                                world.SetValue(BaseGame.WorldMatrix);
                        if (viewInverse != null)
                                viewInverse.SetValue(BaseGame.InverseViewMatrix);
                        if (lightDir != null)
                                lightDir.SetValue(BaseGame.LightDirection);
                          // Set all material properties
                          if (setTex != null)
                          {
                                  diffuseTexture.SetValue(setTex);
                          } // if (setMat)
                  } // SetParameters()
  
   <summary>
 Set parameters, override to set more
 </summary>
                public virtual void SetParameters()
                {
                        SetParameters((Material)null);
                } // SetParameters()
   <summary>
 Set parameters, this overload sets all material parameters too.
 </summary>
                public virtual void SetParametersOptimizedGeneral()
                {
                        //if (worldViewProj != null)
                        //        worldViewProj.SetValue(BaseGame.WorldViewProjectionMatrix);
                        if (viewProj != null)
                                viewProj.SetValue(BaseGame.ViewProjectionMatrix);
                        if (world != null)
                                world.SetValue(BaseGame.WorldMatrix);
                        if (viewInverse != null)
                                viewInverse.SetValue(BaseGame.InverseViewMatrix);
                        if (lightDir != null)
                                lightDir.SetValue(BaseGame.LightDirection);
                          /*obs
                          // Set the reflection cube texture only once
                          if (lastUsedReflectionCubeTexture == null &&
                                  reflectionCubeTexture != null)
                          {
                                  ReflectionCubeTexture = BaseGame.skyCube.SkyCubeMapTexture;
                          } // if (lastUsedReflectionCubeTexture)
  			 */
  
                          // This shader is used for MeshRenderManager and we want all
                          // materials to be opacque, else hotels will look wrong.
                          //obs: AlphaFactor = 1.0f;
  
                          // lastUsed parameters for colors and textures are not used,
                          // but we overwrite the values in SetParametersOptimized.
                          // We fix this by clearing all lastUsed values we will use later.
                          lastUsedAmbientColor = ColorHelper.Empty;
                          lastUsedDiffuseColor = ColorHelper.Empty;
                          lastUsedSpecularColor = ColorHelper.Empty;
                          lastUsedDiffuseTexture = null;
                          lastUsedNormalTexture = null;
                  } // SetParametersOptimizedGeneral()
  
   <summary>
 Set parameters, this overload sets all material parameters too.
 </summary>
                public virtual void SetParametersOptimized(Material setMat)
                {
                        // No need to set world matrix, will be done later in mesh rendering
                        // in the MeshRenderManager. All the rest is set with help of the
                        // SetParametersOptimizedGeneral above.
                          // Only update ambient, diffuse, specular and the textures, the rest
                          // will not change for a material change in MeshRenderManager.
                          ambientColor.SetValue(setMat.ambientColor.ToVector4());
                          diffuseColor.SetValue(setMat.diffuseColor.ToVector4());
                          specularColor.SetValue(setMat.specularColor.ToVector4());
                          if (setMat.diffuseTexture != null)
                                  diffuseTexture.SetValue(setMat.diffuseTexture.XnaTexture);
                          if (setMat.normalTexture != null)
                                  normalTexture.SetValue(setMat.normalTexture.XnaTexture);
                  } // SetParametersOptimized(setMat)
                  #endregion
  
                  #region Update
   <summary>
 Update
 </summary>
                public void Update()
                {
                        xnaEffect.CommitChanges();
                } // Update()
                #endregion
                  #region Render
   <summary>
 Render
 </summary>
 <param name="setMat">Set matrix</param>
 <param name="techniqueName">Technique name</param>
 <param name="renderDelegate">Render delegate</param>
                public void Render(Material setMat,
                        string techniqueName,
                        BaseGame.RenderDelegate renderDelegate)
                {
                        SetParameters(setMat);
                          /*will become important later in the book.
                          // Can we do the requested technique?
                          // For graphic cards not supporting ps2.0, fall back to ps1.1
                          if (BaseGame.CanUsePS20 == false &&
                                  techniqueName.EndsWith("20"))
                                  // Use same technique without the 20 ending!
                                  techniqueName = techniqueName.Substring(0, techniqueName.Length - 2);
  			 */
  
                          // Start shader
                          xnaEffect.CurrentTechnique = xnaEffect.Techniques[techniqueName];
                          xnaEffect.Begin(SaveStateMode.None);
  
                          // Render all passes (usually just one)
                          //foreach (EffectPass pass in effect.CurrentTechnique.Passes)
                          for (int num = 0; num < xnaEffect.CurrentTechnique.Passes.Count; num++)
                          {
                                  EffectPass pass = xnaEffect.CurrentTechnique.Passes[num];
  
                                  pass.Begin();
                                  renderDelegate();
                                  pass.End();
                          } // foreach (pass)
  
                          // End shader
                          xnaEffect.End();
                  } // Render(passName, renderDelegate)
  
   <summary>
 Render
 </summary>
 <param name="techniqueName">Technique name</param>
 <param name="renderDelegate">Render delegate</param>
                public void Render(string techniqueName,
                        BaseGame.RenderDelegate renderDelegate)
                {
                        Render(null, techniqueName, renderDelegate);
                } // Render(techniqueName, renderDelegate)
                #endregion
                  #region Render single pass shader
   <summary>
 Render single pass shader, little faster and simpler than
 Render and it just uses the current technique and renderes only
 the first pass (most shaders have only 1 pass anyway).
 Used for MeshRenderManager!
 </summary>
 <param name="renderDelegate">Render delegate</param>
                public void RenderSinglePassShader(
                        BaseGame.RenderDelegate renderDelegate)
                {
                        // Start effect (current technique should be set)
                        xnaEffect.Begin(SaveStateMode.None);
                        // Start first pass
                        xnaEffect.CurrentTechnique.Passes[0].Begin();
                          // Render
                          renderDelegate();
  
                          // End pass and shader
                          xnaEffect.CurrentTechnique.Passes[0].End();
                          xnaEffect.End();
                  } // RenderSinglePassShader(renderDelegate)
                  #endregion
          } // class ShaderEffect
  } // namespace XnaGraphicEngine.Shaders
  
  
  
(C) Æliens 
20/2/2008
You may not copy or print any of this material without explicit permission of the author or the publisher. 
In case of other copyright issues, contact the author.