// DYNAMIC: "MODE"				"0..9"

// STATIC: "CONVERT_TO_SRGB"	"0..1"	[ps20b][= g_pHardwareConfig->NeedsShaderSRGBConversion()] [PC]
// STATIC: "CONVERT_TO_SRGB"	"0..0"	[= 0] [XBOX]
// STATIC: "LINEAR_TO_SRGB"		"0..1"	[ps20b]

#define HDRTYPE HDR_TYPE_NONE
#include "common_ps_fxc.h"

const float g_Alpha : register( c0 );

sampler BaseTextureSampler	: register( s0 );
sampler BaseTextureSampler2	: register( s1 );

struct PS_INPUT
{
	float2 baseTexCoord				: TEXCOORD0;
};
	   
float3 RGBtoHSV( in float3 rgb )
{
	float3 hsv;
	float fmin, fmax, delta;
	fmin = min( min( rgb.r, rgb.g ), rgb.b );
	fmax = max( max( rgb.r, rgb.g) , rgb.b );
	hsv.b = fmax;				// v
	delta = fmax - fmin;
	if( delta != 0 )
	{
		hsv.g = delta / fmax;		// s
		if( rgb.r == fmax )
			hsv.r = ( rgb.g - rgb.b ) / delta;		// between yellow & magenta
		else if( rgb.g == fmax )
			hsv.r = 2 + ( rgb.b - rgb.r ) / delta;	// between cyan & yellow
		else
			hsv.r = 4 + ( rgb.r - rgb.g ) / delta;	// between magenta & cyan
		hsv.r *= 60;				// degrees
		if( hsv.r < 0 )
			hsv.r += 360;
	}
	else 
	{
		// r = g = b = 0		// s = 0, v is undefined
		hsv.g = 0;
		hsv.r = -1;
	}
	return hsv;
}

float3 HSVtoRGB( in float3 hsv )
{
	int i;
	float3 rgb;
	float h = hsv.r;
	float s = hsv.g;
	float v = hsv.b;
	float f, p, q, t;
	if( s == 0 ) 
	{
		// achromatic (grey)
		rgb.rgb = v;
	}
	else
	{
		h /= 60;			// sector 0 to 5
		i = floor( h );
		f = h - i;			// factorial part of h
		p = v * ( 1 - s );
		q = v * ( 1 - s * f );
		t = v * ( 1 - s * ( 1 - f ) );
		if( h < 1)
		{
			rgb.r = v;
			rgb.g = t;
			rgb.b = p;
		}
		else if( h >= 1 && h < 2 )
		{
			rgb.r = q;
			rgb.g = v;
			rgb.b = p;
		}
		else if( h >= 2 && h < 3 )
		{
			rgb.r = p;
			rgb.g = v;
			rgb.b = t;
		}
		else if( h >= 3 && h < 4 )
		{
			rgb.r = p;
			rgb.g = q;
			rgb.b = v;
		}
		else if( h >= 4 && h < 5 )
		{
			rgb.r = t;
			rgb.g = p;
			rgb.b = v;
		}
		else // if ( h >= 5 )
		{
			rgb.r = v;
			rgb.g = p;
			rgb.b = q;
		}
	}
	return rgb;
}

// We have to run through this input converter on OpenGL if the
// rest of the shader code is expecting sRGB values
float3 SampleTexture( sampler texSampler, float2 tc )
{
	float3 c = tex2D( texSampler, tc ).xyz;

	#if ( LINEAR_TO_SRGB )
	{
		c = LinearToGamma( c );
	}
	#endif

	return c;
}

// We have to run through this output converter on OpenGL if we
// expect to be writing out sRGB values (since sRGB will be forced on)
float3 OutputColor( float3 result )
{
	#if ( LINEAR_TO_SRGB )
	{
		return GammaToLinear( result );
	}
	#endif

	return result;
}


float4 main( PS_INPUT i ) : COLOR
{
	float3 result;
#if MODE == 0
	// negative greyscale of scene * gman
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );

	float scale = 1.0f / 3.0f;
	scene.xyz = dot( float3( scale, scale, scale), scene.xyz );
	scene = float3( 1, 1, 1 ) - scene;

	return FinalOutput( float4( OutputColor( scene * gman ), g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif

#if MODE == 1
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );
	float scale = 1.0f / 3.0f;
	scene.xyz = dot( float3( scale, scale, scale ), scene.xyz );

	float gmanLum = dot( float3( scale, scale, scale ), gman );
	if( gmanLum < 0.3 )
	{
		result = OutputColor( float3( 1, 1, 1 ) - gman );
		return FinalOutput( float4( result, g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
	}
	else
	{
		result = OutputColor( ( float3( 1, 1, 1 ) - gman ) * scene );
		return FinalOutput( float4( result, g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
	}
#endif

#if MODE == 2
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );

	float startRamp = .2;
	float endRamp = .5;
	
	float scale = 1.0f / 3.0f;
	float gmanLum = dot( float3( scale, scale, scale ), gman );
	float sceneLum = dot( float3( scale, scale, scale ), scene );

	float blend = ( gmanLum - startRamp ) * ( 1.0f / ( endRamp - startRamp ) );
	blend = saturate( blend );
	
//	return gmanLum * ( 1.0f - blend ) + scene * blend;

	result = OutputColor( min( gmanLum.xxx, scene ) );
	return FinalOutput( float4( result, g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif

#if MODE == 3
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );
	float scale = 1.0f / 3.0f;
	float gmanLum = dot( float3( scale, scale, scale ), gman );
	float sceneLum = dot( float3( scale, scale, scale ), scene );

	float a = 0.0f;
	float b = 0.4f;
	float c = 0.7f;
	float d = 1.0f;

	float blend;
	if( gmanLum < b )
	{
		blend = ( gmanLum - a ) / ( b - a );
	}
	else if( gmanLum > c )
	{
		blend = 1.0f - ( ( gmanLum - c) / ( d - c ) );
	}
	else
	{
		blend = 1.0f;
	}

	blend = saturate( blend );
	
	result = OutputColor( gmanLum.xxx * ( float3( 1, 1, 1 ) - blend.xxx ) + scene * blend.xxx );
	return FinalOutput( float4( result, g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif

#if MODE == 4
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );
	float scale = 1.0f / 3.0f;
	float gmanLum = dot( float3( scale, scale, scale ), gman );
	float sceneLum = dot( float3( scale, scale, scale ), scene );

	float a = 0.0f;
	float b = 0.4f;
	float c = 0.7f;
	float d = 1.0f;

	float blend;
	if( gmanLum < b )
	{
		blend = ( gmanLum - a ) / ( b - a );
	}
	else if( gmanLum > c )
	{
		blend = 1.0f - ( ( gmanLum - c) / ( d - c ) );
	}
	else
	{
		blend = 1.0f;
	}

	blend = saturate( blend );

	result = OutputColor( gman * ( float3( 1, 1, 1 ) - blend.xxx ) + scene * blend.xxx );
	return FinalOutput( float4( result, g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif

#if MODE == 5
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );
	float scale = 1.0f / 3.0f;
//	float sceneLum = dot( float3( scale, scale, scale ), scene );
	float sceneLum = scene.r;

	if( sceneLum > 0.0f )
	{
		return FinalOutput( float4( OutputColor( scene ), g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
	}
	else
	{
		float3 hsv = RGBtoHSV( gman );

//		float blend = saturate( hsv.b - .5 );
		float blend = hsv.b - .5;
		
		hsv.b *= 1.0f + blend;
		hsv.g *= 1.0f - blend;
		return FinalOutput( float4( OutputColor( HSVtoRGB( hsv ) ), g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
	}
#endif

#if MODE == 6
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );
	return FinalOutput( float4( OutputColor( scene + gman ), g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif

#if MODE == 7
	float3 scene = SampleTexture( BaseTextureSampler, i.baseTexCoord );
	return FinalOutput( float4( OutputColor( scene ), g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif

#if MODE == 8
	float3 gman = SampleTexture( BaseTextureSampler2, i.baseTexCoord );
	return FinalOutput( float4( OutputColor( gman ), g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif

#if MODE == 9
	// Fetch textures
	float3 cLayer1 = SampleTexture( BaseTextureSampler, i.baseTexCoord.xy );
	float3 cLayer2 = SampleTexture( BaseTextureSampler2, i.baseTexCoord.xy );

	/*
	// Put colors roughly back into gamma space
	float3 cGammaLayer1 = pow( cLayer1, 0.454545f );
	float3 cGammaLayer2 = pow( cLayer2, 0.454545f );

	// Brightness
	//float flLayer1Brightness = saturate( dot( cGammaLayer1.rgb, float3( 0.3f, 0.59f, 0.11f ) ) );
	//float flLayer2Brightness = saturate( dot( cGammaLayer2.rgb, float3( 0.3f, 0.59f, 0.11f ) ) );
	float flLayer1Brightness = saturate( dot( cGammaLayer1.rgb, float3( 0.333f, 0.334f, 0.333f ) ) );
	float flLayer2Brightness = saturate( dot( cGammaLayer2.rgb, float3( 0.333f, 0.334f, 0.333f ) ) );

	// Blend layers in rough gamma space
	float3 cGammaOverlayResult;
	if ( flLayer1Brightness < 0.5f )
	{
		cGammaOverlayResult.rgb = ( 2.0f * cGammaLayer1.rgb * cGammaLayer2.rgb );
	}
	else
	{
		cGammaOverlayResult.rgb = ( 1.0f - ( 2.0f * ( 1.0f - cGammaLayer1.rgb ) * ( 1.0f - cGammaLayer2.rgb ) ) );
	}

	// Convert back to linear space
	float3 cLinearOverlayResult = pow( cGammaOverlayResult.rgb, 2.2f );
	//*/

	float flLayer1Brightness = saturate( dot( cLayer1.rgb, float3( 0.333f, 0.334f, 0.333f ) ) );
	float flLayer2Brightness = saturate( dot( cLayer2.rgb, float3( 0.333f, 0.334f, 0.333f ) ) );

	// Modify layer 1 to be more contrasty
	cLayer1.rgb = saturate( cLayer1.rgb * cLayer1.rgb * 2.0f );
	float3 cLinearOverlayResult = cLayer1.rgb + cLayer2.rgb * saturate( 1.0f - flLayer1Brightness * 2.0f );

	// Tonemap, fog, etc.
	return FinalOutput( float4( OutputColor( cLinearOverlayResult.rgb ), g_Alpha ), 0, PIXEL_FOG_TYPE_NONE, TONEMAP_SCALE_NONE );
#endif
}