Smoother coastlines, and a specular map for future e shader use.

git-svn-id: http://svn.berlios.de/svnroot/repos/oolite-linux/trunk@2823 127b21dd-08f5-0310-b4b7-95ae10353056
This commit is contained in:
Jens Ayton 2009-12-01 22:59:53 +00:00
parent 363e33d209
commit 89424c71d4
2 changed files with 72 additions and 19 deletions

View File

@ -35,6 +35,12 @@ typedef struct
} FloatRGB; } FloatRGB;
typedef struct
{
float r, g, b, a;
} FloatRGBA;
@interface OOPlanetTextureGenerator: OOTextureGenerator @interface OOPlanetTextureGenerator: OOTextureGenerator
{ {
@private @private

View File

@ -25,12 +25,17 @@ MA 02110-1301, USA.
*/ */
#define DEBUG_DUMP ( 1 && !defined(NDEBUG))
#define DISABLE_SPECULAR ( 0 && DEBUG_DUMP) // No transparency in debug dump to make life easier.
#import "OOPlanetTextureGenerator.h" #import "OOPlanetTextureGenerator.h"
#import "OOCollectionExtractors.h" #import "OOCollectionExtractors.h"
#import "OOColor.h" #import "OOColor.h"
#import "OOTexture.h" #import "OOTexture.h"
#ifndef NDEBUG #if DEBUG_DUMP
#import "Universe.h" #import "Universe.h"
#import "MyOpenGLView.h" #import "MyOpenGLView.h"
#endif #endif
@ -49,7 +54,7 @@ static void AddNoise(float *buffer, unsigned width, unsigned height, unsigned oc
float QFactor(float *accbuffer, int x, int y, unsigned width, unsigned height, float polar_y_value, float impress, float bias); float QFactor(float *accbuffer, int x, int y, unsigned width, unsigned height, float polar_y_value, float impress, float bias);
static FloatRGB PlanetMix(float q, float impress, float seaBias, FloatRGB landColor, FloatRGB seaColor, FloatRGB paleLandColor, FloatRGB paleSeaColor); static FloatRGBA PlanetMix(float q, float impress, float seaBias, FloatRGB landColor, FloatRGB seaColor, FloatRGB paleLandColor, FloatRGB paleSeaColor);
@implementation OOPlanetTextureGenerator @implementation OOPlanetTextureGenerator
@ -137,9 +142,6 @@ static FloatRGB PlanetMix(float q, float impress, float seaBias, FloatRGB landCo
} }
static FloatRGB PlanetMix(float q, float impress, float seaBias, FloatRGB landColor, FloatRGB seaColor, FloatRGB polarLandColor, FloatRGB polarSeaColor);
- (void) loadTexture - (void) loadTexture
{ {
OOLog(@"planetTex.temp", @"Started generator %@", self); OOLog(@"planetTex.temp", @"Started generator %@", self);
@ -193,15 +195,28 @@ static FloatRGB PlanetMix(float q, float impress, float seaBias, FloatRGB landCo
Vector norm = vector_normal(make_vector(24.0f * (yW - yE), 24.0f * (yS - yN), 2.0f)); Vector norm = vector_normal(make_vector(24.0f * (yW - yE), 24.0f * (yS - yN), 2.0f));
// FIXME: powf() is very expensive, can we use an approximation or change exponent to 3.0? FloatRGBA color = PlanetMix(q, impress, seaBias, _landColor, _seaColor, _polarLandColor, _polarSeaColor);
GLfloat shade = powf(norm.z, 3.2);
FloatRGB color = PlanetMix(q, impress, seaBias, _landColor, _seaColor, _polarLandColor, _polarSeaColor); /* Terrain shading
was: _powf(norm.z, 3.2). Changing exponent to 3 makes very
little difference, other than being faster.
*/
GLfloat shade = norm.z * norm.z * norm.z;
/* We don't want terrain shading in the sea. The alpha channel
of color is a measure of "seaishness" for the specular map,
so we can recycle that to avoid branching.
*/
shade = color.a + (1.0f - color.a) * shade;
*px++ = 255 * color.r * shade; *px++ = 255 * color.r * shade;
*px++ = 255 * color.g * shade; *px++ = 255 * color.g * shade;
*px++ = 255 * color.b * shade; *px++ = 255 * color.b * shade;
#if DISABLE_SPECULAR
*px++ = 255; *px++ = 255;
#else
*px++ = 255 * color.a;
#endif
} }
} }
success = YES; success = YES;
@ -214,7 +229,8 @@ END:
else free(buffer); else free(buffer);
OOLog(@"planetTex.temp", @"Completed generator %@ %@successfully", self, success ? @"" : @"un"); OOLog(@"planetTex.temp", @"Completed generator %@ %@successfully", self, success ? @"" : @"un");
#ifndef NDEBUG
#if DEBUG_DUMP
if (success) if (success)
{ {
[[UNIVERSE gameView] dumpRGBAToFileNamed:[NSString stringWithFormat:@"planet-%u-%u-new", _seed.high, _seed.low] [[UNIVERSE gameView] dumpRGBAToFileNamed:[NSString stringWithFormat:@"planet-%u-%u-new", _seed.high, _seed.low]
@ -237,29 +253,62 @@ static FloatRGB Blend(float fraction, FloatRGB a, FloatRGB b)
{ {
fraction * a.r + prime * b.r, fraction * a.r + prime * b.r,
fraction * a.g + prime * b.g, fraction * a.g + prime * b.g,
fraction * a.b + prime * b.b fraction * a.b + prime * b.b,
}; };
} }
static FloatRGB PlanetMix(float q, float impress, float seaBias, FloatRGB landColor, FloatRGB seaColor, FloatRGB paleLandColor, FloatRGB paleSeaColor) static FloatRGBA PlanetMix(float q, float impress, float seaBias, FloatRGB landColor, FloatRGB seaColor, FloatRGB paleLandColor, FloatRGB paleSeaColor)
{ {
float maxq = impress + seaBias; float maxq = impress + seaBias;
float hi = 0.66667 * maxq; float hi = 0.66667 * maxq;
float oh = 1.0 / hi; float oh = 1.0 / hi;
float ih = 1.0 / (1.0 - hi); float ih = 1.0 / (1.0 - hi);
#define RECIP_COASTLINE_PORTION (150.0f)
#define COASTLINE_PORTION (1.0f / RECIP_COASTLINE_PORTION)
const FloatRGB white = { 1.0f, 1.0f, 1.0f }; const FloatRGB white = { 1.0f, 1.0f, 1.0f };
FloatRGB result;
float specular = 0.0f;
if (q <= 0.0f) return seaColor; // Offset to reduce coastline-lowering effect of r2823 coastline smoothing improvement.
if (q > 1.0) return white; q -= COASTLINE_PORTION;
if (q < 0.01) return Blend(q * 100.0f, landColor, paleSeaColor); // Coastline if (q <= 0.0f)
{
if (q > -COASTLINE_PORTION)
{
// Coastal waters
result = Blend(-q * RECIP_COASTLINE_PORTION, seaColor, paleSeaColor);
specular = -(q * RECIP_COASTLINE_PORTION);
}
else
{
// Open sea
result = seaColor;
specular = 1.0;
}
}
else if (q > 1.0)
{
result = white;
}
else if (q < COASTLINE_PORTION)
{
// Coastline
result = Blend(q * RECIP_COASTLINE_PORTION, landColor, paleSeaColor);
}
else if (q > hi)
{
result = Blend((q - hi) * ih, white, paleLandColor); // Snow-capped peaks
}
else
{
result = Blend((hi - q) * oh, landColor, paleLandColor);
}
if (q > hi) return Blend((q - hi) * ih, white, paleLandColor); // Snow-capped peaks return (FloatRGBA){ result.r, result.g, result.b, specular };
return Blend((hi - q) * oh, landColor, paleLandColor);
} }
@ -335,7 +384,5 @@ float QFactor(float *accbuffer, int x, int y, unsigned width, unsigned height, f
polar_y *= polar_y; polar_y *= polar_y;
q = q * (1.0 - polar_y) + polar_y * polar_y_value; q = q * (1.0 - polar_y) + polar_y * polar_y_value;
q = OOClamp_0_1_f(q);
return q; return q;
} }