mirror of https://github.com/skeeto/pixelcity.git
746 lines
22 KiB
C++
746 lines
22 KiB
C++
/*-----------------------------------------------------------------------------
|
|
|
|
World.cpp
|
|
|
|
2009 Shamus Young
|
|
|
|
-------------------------------------------------------------------------------
|
|
|
|
This holds a bunch of variables used by the other modules. It has the
|
|
claim system, which tracks all of the "property" is being used: As roads,
|
|
buildings, etc.
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
#define HUE_COUNT (sizeof(hue_list)/sizeof(float))
|
|
#define LIGHT_COLOR_COUNT (sizeof(light_colors)/sizeof(HSL))
|
|
#define CARS 600
|
|
#define WORLD_EDGE 200
|
|
#if SCREENSAVER
|
|
#define RESET_INTERVAL 120 //seconds
|
|
#define FADE_TIME 500 //milliseconds
|
|
#else
|
|
#define RESET_INTERVAL 999 //seconds
|
|
#define FADE_TIME 1500 //milliseconds
|
|
#endif
|
|
|
|
#include <windows.h>
|
|
#include <gl\gl.h>
|
|
#include <gl\glu.h>
|
|
#include <gl\glaux.h>
|
|
#include <math.h>
|
|
#include <time.h>
|
|
|
|
#include "glTypes.h"
|
|
#include "building.h"
|
|
#include "car.h"
|
|
#include "deco.h"
|
|
#include "camera.h"
|
|
#include "light.h"
|
|
#include "macro.h"
|
|
#include "math.h"
|
|
#include "mesh.h"
|
|
#include "random.h"
|
|
#include "render.h"
|
|
#include "sky.h"
|
|
#include "texture.h"
|
|
#include "visible.h"
|
|
#include "world.h"
|
|
|
|
struct plot
|
|
{
|
|
int x;
|
|
int z;
|
|
int width;
|
|
int depth;
|
|
|
|
};
|
|
|
|
enum {
|
|
FADE_IDLE,
|
|
FADE_OUT,
|
|
FADE_WAIT,
|
|
FADE_IN,
|
|
};
|
|
|
|
struct HSL
|
|
{
|
|
float hue;
|
|
float sat;
|
|
float lum;
|
|
};
|
|
|
|
class CStreet
|
|
{
|
|
public:
|
|
int _x;
|
|
int _y;
|
|
int _width;
|
|
int _depth;
|
|
CMesh* _mesh;
|
|
|
|
CStreet (int x, int y, int width, int depth);
|
|
~CStreet();
|
|
void Render ();
|
|
|
|
};
|
|
|
|
static HSL light_colors[] =
|
|
{
|
|
0.04f, 0.9f, 0.93f, //Amber / pink
|
|
0.055f, 0.95f, 0.93f, //Slightly brighter amber
|
|
0.08f, 0.7f, 0.93f, //Very pale amber
|
|
0.07f, 0.9f, 0.93f, //Very pale orange
|
|
0.1f, 0.9f, 0.85f, //Peach
|
|
0.13f, 0.9f, 0.93f, //Pale Yellow
|
|
0.15f, 0.9f, 0.93f, //Yellow
|
|
0.17f, 1.0f, 0.85f, //Saturated Yellow
|
|
0.55f, 0.9f, 0.93f, //Cyan
|
|
0.55f, 0.9f, 0.93f, //Cyan - pale, almost white
|
|
0.6f, 0.9f, 0.93f, //Pale blue
|
|
0.65f, 0.9f, 0.93f, //Pale Blue II, The Palening
|
|
0.65f, 0.4f, 0.99f, //Pure white. Bo-ring.
|
|
0.65f, 0.0f, 0.8f, //Dimmer white.
|
|
0.65f, 0.0f, 0.6f, //Dimmest white.
|
|
};
|
|
|
|
static float hue_list[] = { 0.04f, 0.07f, 0.1f, 0.5f, 0.6f }; //Yellows and blues - good for lights
|
|
static GLrgba bloom_color;
|
|
static long last_update;
|
|
static char world[WORLD_SIZE][WORLD_SIZE];
|
|
static CSky* sky;
|
|
static int fade_state;
|
|
static unsigned fade_start;
|
|
static float fade_current;
|
|
static int modern_count;
|
|
static int tower_count;
|
|
static int blocky_count;
|
|
static bool reset_needed;
|
|
static int skyscrapers;
|
|
static GLbbox hot_zone;
|
|
static unsigned start_time;
|
|
static int logo_index;
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static GLrgba get_light_color (float sat, float lum)
|
|
{
|
|
|
|
int index;
|
|
|
|
index = RandomVal (LIGHT_COLOR_COUNT);
|
|
return glRgbaFromHsl (light_colors[index].hue, sat, lum);
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static void claim (int x, int y, int width, int depth, int val)
|
|
{
|
|
|
|
int xx, yy;
|
|
|
|
for (xx = x; xx < (x + width); xx++) {
|
|
for (yy = y; yy < (y + depth); yy++) {
|
|
world[CLAMP (xx,0,WORLD_SIZE - 1)][CLAMP (yy,0,WORLD_SIZE - 1)] |= val;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static bool claimed (int x, int y, int width, int depth)
|
|
{
|
|
|
|
int xx, yy;
|
|
|
|
for (xx = x; xx < x + width; xx++) {
|
|
for (yy = y; yy < y + depth; yy++) {
|
|
if (world[CLAMP (xx,0,WORLD_SIZE - 1)][CLAMP (yy,0,WORLD_SIZE - 1)])
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static void build_road (int x1, int y1, int width, int depth)
|
|
{
|
|
|
|
int lanes;
|
|
int divider;
|
|
int sidewalk;
|
|
|
|
//the given rectangle defines a street and its sidewalk. See which way it goes.
|
|
if (width > depth)
|
|
lanes = depth;
|
|
else
|
|
lanes = width;
|
|
//if we dont have room for both lanes and sidewalk, abort
|
|
if (lanes < 4)
|
|
return;
|
|
//if we have an odd number of lanes, give the extra to a divider.
|
|
if (lanes % 2) {
|
|
lanes--;
|
|
divider = 1;
|
|
} else
|
|
divider = 0;
|
|
//no more than 10 traffic lanes, give the rest to sidewalks
|
|
sidewalk = MAX (2, (lanes - 10));
|
|
lanes -= sidewalk;
|
|
sidewalk /= 2;
|
|
//take the remaining space and give half to each direction
|
|
lanes /= 2;
|
|
//Mark the entire rectangle as used
|
|
claim (x1, y1, width, depth, CLAIM_WALK);
|
|
//now place the directional roads
|
|
if (width > depth) {
|
|
claim (x1, y1 + sidewalk, width, lanes, CLAIM_ROAD | MAP_ROAD_WEST);
|
|
claim (x1, y1 + sidewalk + lanes + divider, width, lanes, CLAIM_ROAD | MAP_ROAD_EAST);
|
|
} else {
|
|
claim (x1 + sidewalk, y1, lanes, depth, CLAIM_ROAD | MAP_ROAD_SOUTH);
|
|
claim (x1 + sidewalk + lanes + divider, y1, lanes, depth, CLAIM_ROAD | MAP_ROAD_NORTH);
|
|
}
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static plot find_plot (int x, int z)
|
|
{
|
|
|
|
plot p;
|
|
int x1, x2, z1, z2;
|
|
|
|
//We've been given the location of an open bit of land, but we have no
|
|
//idea how big it is. Find the boundary.
|
|
x1 = x2 = x;
|
|
while (!claimed (x1 - 1, z, 1, 1) && x1 > 0)
|
|
x1--;
|
|
while (!claimed (x2 + 1, z, 1, 1) && x2 < WORLD_SIZE)
|
|
x2++;
|
|
z1 = z2 = z;
|
|
while (!claimed (x, z1 - 1, 1, 1) && z1 > 0)
|
|
z1--;
|
|
while (!claimed (x, z2 + 1, 1, 1) && z2 < WORLD_SIZE)
|
|
z2++;
|
|
p.width = (x2 - x1);
|
|
p.depth = (z2 - z1);
|
|
p.x = x1;
|
|
p.z = z1;
|
|
return p;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static plot make_plot (int x, int z, int width, int depth)
|
|
{
|
|
|
|
plot p = {x, z, width, depth};
|
|
return p;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
void do_building (plot p)
|
|
{
|
|
|
|
int height;
|
|
int seed;
|
|
int area;
|
|
int type;
|
|
GLrgba color;
|
|
bool square;
|
|
|
|
//now we know how big the rectangle plot is.
|
|
area = p.width * p.depth;
|
|
color = WorldLightColor (RandomVal ());
|
|
seed = RandomVal ();
|
|
//Make sure the plot is big enough for a building
|
|
if (p.width < 10 || p.depth < 10)
|
|
return;
|
|
//If the area is too big for one building, sub-divide it.
|
|
if (area > 600) {
|
|
if (p.width > p.depth) {
|
|
p.width /= 2;
|
|
if (COIN_FLIP)
|
|
do_building (make_plot (p.x, p.z, p.width, p.depth));
|
|
else
|
|
do_building (make_plot (p.x + p.width, p.z, p.width, p.depth));
|
|
return;
|
|
} else {
|
|
p.depth /= 2;
|
|
if (COIN_FLIP)
|
|
do_building (make_plot (p.x, p.z, p.width, p.depth));
|
|
else
|
|
do_building (make_plot (p.x, p.z + p.depth, p.width, p.depth));
|
|
return;
|
|
}
|
|
}
|
|
if (area < 100)
|
|
return;
|
|
//The plot is "square" if width & depth are close
|
|
square = abs (p.width - p.depth) < 10;
|
|
//mark the land as used so other buildings don't appear here, even if we don't use it all.
|
|
claim (p.x, p.z, p.width, p.depth, CLAIM_BUILDING);
|
|
//The roundy mod buildings look best on square plots.
|
|
if (square && p.width > 20) {
|
|
height = 45 + RandomVal (10);
|
|
modern_count++;
|
|
skyscrapers++;
|
|
new CBuilding (BUILDING_MODERN, p.x, p.z, height, p.width, p.depth, seed, color);
|
|
return;
|
|
}
|
|
//This spot isn't ideal for any particular building, but try to keep a good mix
|
|
if (tower_count < modern_count && tower_count < blocky_count) {
|
|
type = BUILDING_TOWER;
|
|
tower_count++;
|
|
} else if (blocky_count < modern_count) {
|
|
type = BUILDING_BLOCKY;
|
|
blocky_count++;
|
|
} else {
|
|
type = BUILDING_MODERN;
|
|
modern_count++;
|
|
}
|
|
height = 45 + RandomVal (10);
|
|
new CBuilding (type, p.x, p.z, height, p.width, p.depth, seed, color);
|
|
skyscrapers++;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static int build_light_strip (int x1, int z1, int direction)
|
|
{
|
|
|
|
CDeco* d;
|
|
GLrgba color;
|
|
int x2, z2;
|
|
int length;
|
|
int width, depth;
|
|
int dir_x, dir_z;
|
|
float size_adjust;
|
|
|
|
//We adjust the size of the lights with this.
|
|
size_adjust = 2.5f;
|
|
color = glRgbaFromHsl (0.09f, 0.99f, 0.85f);
|
|
switch (direction) {
|
|
case NORTH:
|
|
dir_z = 1; dir_x = 0;break;
|
|
case SOUTH:
|
|
dir_z = 1; dir_x = 0;break;
|
|
case EAST:
|
|
dir_z = 0; dir_x = 1;break;
|
|
case WEST:
|
|
dir_z = 0; dir_x = 1;break;
|
|
}
|
|
//So we know we're on the corner of an intersection
|
|
//look in the given until we reach the end of the sidewalk
|
|
x2 = x1;
|
|
z2 = z1;
|
|
length = 0;
|
|
while (x2 > 0 && x2 < WORLD_SIZE && z2 > 0 && z2 < WORLD_SIZE) {
|
|
if ((world[x2][z2] & CLAIM_ROAD))
|
|
break;
|
|
length++;
|
|
x2 += dir_x;
|
|
z2 += dir_z;
|
|
}
|
|
if (length < 10)
|
|
return length;
|
|
width = MAX (abs(x2 - x1), 1);
|
|
depth = MAX (abs(z2 - z1), 1);
|
|
d = new CDeco;
|
|
if (direction == EAST)
|
|
d->CreateLightStrip ((float)x1, (float)z1 - size_adjust, (float)width, (float)depth + size_adjust, 2, color);
|
|
else if (direction == WEST)
|
|
d->CreateLightStrip ((float)x1, (float)z1, (float)width, (float)depth + size_adjust, 2, color);
|
|
else if (direction == NORTH)
|
|
d->CreateLightStrip ((float)x1, (float)z1, (float)width + size_adjust, (float)depth, 2, color);
|
|
else
|
|
d->CreateLightStrip ((float)x1 - size_adjust, (float)z1, (float)width + size_adjust, (float)depth, 2, color);
|
|
return length;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
static void do_reset (void)
|
|
{
|
|
|
|
int x, y;
|
|
int width, depth, height;
|
|
int attempts;
|
|
bool broadway_done;
|
|
bool road_left, road_right;
|
|
GLrgba light_color;
|
|
GLrgba building_color;
|
|
float west_street, north_street, east_street, south_street;
|
|
|
|
//Re-init Random to make the same city each time. Good for debugging.
|
|
//RandomInit (6);
|
|
reset_needed = false;
|
|
broadway_done = false;
|
|
skyscrapers = 0;
|
|
logo_index = 0;
|
|
tower_count = blocky_count = modern_count = 0;
|
|
hot_zone = glBboxClear ();
|
|
EntityClear ();
|
|
LightClear ();
|
|
CarClear ();
|
|
TextureReset ();
|
|
//Pick a tint for the bloom
|
|
bloom_color = get_light_color(0.5f + (float)RandomVal (10) / 20.0f, 0.75f);
|
|
light_color = glRgbaFromHsl (0.11f, 1.0f, 0.65f);
|
|
ZeroMemory (world, WORLD_SIZE * WORLD_SIZE);
|
|
for (y = WORLD_EDGE; y < WORLD_SIZE - WORLD_EDGE; y += RandomVal (30) + 20) {
|
|
if (!broadway_done && y > WORLD_HALF - 20) {
|
|
build_road (0, y, WORLD_SIZE, 19);
|
|
y += 20;
|
|
broadway_done = true;
|
|
} else {
|
|
depth = 6 + RandomVal (6);
|
|
if (y < WORLD_HALF / 2)
|
|
north_street = (float)(y + depth / 2);
|
|
if (y < (WORLD_SIZE - WORLD_HALF / 2))
|
|
south_street = (float)(y + depth / 2);
|
|
build_road (0, y, WORLD_SIZE, depth);
|
|
}
|
|
}
|
|
|
|
broadway_done = false;
|
|
for (x = WORLD_EDGE; x < WORLD_SIZE - WORLD_EDGE; x += RandomVal (30) + 20) {
|
|
if (!broadway_done && x > WORLD_HALF - 20) {
|
|
build_road (x, 0, 19, WORLD_SIZE);
|
|
x += 20;
|
|
broadway_done = true;
|
|
} else {
|
|
width = 6 + RandomVal (6);
|
|
if (x <= WORLD_HALF / 2)
|
|
west_street = (float)(x + width / 2);
|
|
if (x <= WORLD_HALF + WORLD_HALF / 2)
|
|
east_street = (float)(x + width / 2);
|
|
build_road (x, 0, width, WORLD_SIZE);
|
|
}
|
|
}
|
|
//We kept track of the positions of streets that will outline the high-detail hot zone
|
|
//in the middle of the world. Save this in a bounding box so that later we can
|
|
//have the camera fly around without clipping through buildings.
|
|
hot_zone = glBboxContainPoint (hot_zone, glVector (west_street, 0.0f, north_street));
|
|
hot_zone = glBboxContainPoint (hot_zone, glVector (east_street, 0.0f, south_street));
|
|
|
|
//Scan for places to put runs of streetlights on the east & west side of the road
|
|
for (x = 1; x < WORLD_SIZE - 1; x++) {
|
|
for (y = 0; y < WORLD_SIZE; y++) {
|
|
//if this isn't a bit of sidewalk, then keep looking
|
|
if (!(world[x][y] & CLAIM_WALK))
|
|
continue;
|
|
//If it's used as a road, skip it.
|
|
if ((world[x][y] & CLAIM_ROAD))
|
|
continue;
|
|
road_left = (world[x + 1][y] & CLAIM_ROAD) != 0;
|
|
road_right = (world[x - 1][y] & CLAIM_ROAD) != 0;
|
|
//if the cells to our east and west are not road, then we're not on a corner.
|
|
if (!road_left && !road_right)
|
|
continue;
|
|
//if the cell to our east AND west is road, then we're on a median. skip it
|
|
if (road_left && road_right)
|
|
continue;
|
|
y += build_light_strip (x, y, road_right ? SOUTH : NORTH);
|
|
}
|
|
}
|
|
//Scan for places to put runs of streetlights on the north & south side of the road
|
|
for (y = 1; y < WORLD_SIZE - 1; y++) {
|
|
for (x = 1; x < WORLD_SIZE - 1; x++) {
|
|
//if this isn't a bit of sidewalk, then keep looking
|
|
if (!(world[x][y] & CLAIM_WALK))
|
|
continue;
|
|
//If it's used as a road, skip it.
|
|
if ((world[x][y] & CLAIM_ROAD))
|
|
continue;
|
|
road_left = (world[x][y + 1] & CLAIM_ROAD) != 0;
|
|
road_right = (world[x][y - 1] & CLAIM_ROAD) != 0;
|
|
//if the cell to our east AND west is road, then we're on a median. skip it
|
|
if (road_left && road_right)
|
|
continue;
|
|
//if the cells to our north and south are not road, then we're not on a corner.
|
|
if (!road_left && !road_right)
|
|
continue;
|
|
x += build_light_strip (x, y, road_right ? EAST : WEST);
|
|
}
|
|
}
|
|
|
|
//Scan over the center area of the map and place the big buildings
|
|
attempts = 0;
|
|
while (skyscrapers < 50 && attempts < 350) {
|
|
x = (WORLD_HALF / 2) + (RandomVal () % WORLD_HALF);
|
|
y = (WORLD_HALF / 2) + (RandomVal () % WORLD_HALF);
|
|
if (!claimed (x, y, 1,1)) {
|
|
do_building (find_plot (x, y));
|
|
skyscrapers++;
|
|
}
|
|
attempts++;
|
|
}
|
|
//now blanket the rest of the world with lesser buildings
|
|
for (x = 0; x < WORLD_SIZE; x ++) {
|
|
for (y = 0; y < WORLD_SIZE; y ++) {
|
|
if (world[CLAMP (x,0,WORLD_SIZE)][CLAMP (y,0,WORLD_SIZE)])
|
|
continue;
|
|
width = 12 + RandomVal (20);
|
|
depth = 12 + RandomVal (20);
|
|
height = MIN (width, depth);
|
|
if (x < 30 || y < 30 || x > WORLD_SIZE - 30 || y > WORLD_SIZE - 30)
|
|
height = RandomVal (15) + 20;
|
|
else if (x < WORLD_HALF / 2)
|
|
height /= 2;
|
|
while (width > 8 && depth > 8) {
|
|
if (!claimed (x, y, width, depth)) {
|
|
claim (x, y, width, depth,CLAIM_BUILDING);
|
|
building_color = WorldLightColor (RandomVal ());
|
|
//if we're out of the hot zone, use simple buildings
|
|
if (x < hot_zone.min.x || x > hot_zone.max.x || y < hot_zone.min.z || y > hot_zone.max.z) {
|
|
height = 5 + RandomVal (height) + RandomVal (height);
|
|
new CBuilding (BUILDING_SIMPLE, x + 1, y + 1, height, width - 2, depth - 2, RandomVal (), building_color);
|
|
} else { //use fancy buildings.
|
|
height = 15 + RandomVal (15);
|
|
width -=2;
|
|
depth -=2;
|
|
if (COIN_FLIP)
|
|
new CBuilding (BUILDING_TOWER, x + 1, y + 1, height, width, depth, RandomVal (), building_color);
|
|
else
|
|
new CBuilding (BUILDING_BLOCKY, x + 1, y + 1, height, width, depth, RandomVal (), building_color);
|
|
}
|
|
break;
|
|
}
|
|
width--;
|
|
depth--;
|
|
}
|
|
//leave big gaps near the edge of the map, no need to pack detail there.
|
|
if (y < WORLD_EDGE || y > WORLD_SIZE - WORLD_EDGE)
|
|
y += 32;
|
|
}
|
|
//leave big gaps near the edge of the map
|
|
if (x < WORLD_EDGE || x > WORLD_SIZE - WORLD_EDGE)
|
|
x += 28;
|
|
}
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
This will return a random color which is suitible for light sources, taken
|
|
from a narrow group of hues. (Yellows, oranges, blues.)
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
GLrgba WorldLightColor (unsigned index)
|
|
{
|
|
|
|
index %= LIGHT_COLOR_COUNT;
|
|
return glRgbaFromHsl (light_colors[index].hue, light_colors[index].sat, light_colors[index].lum);
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
char WorldCell (int x, int y)
|
|
{
|
|
|
|
return world[CLAMP (x, 0,WORLD_SIZE - 1)][CLAMP (y, 0, WORLD_SIZE - 1)];
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
GLrgba WorldBloomColor ()
|
|
{
|
|
|
|
return bloom_color;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
int WorldLogoIndex ()
|
|
{
|
|
|
|
return logo_index++;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
GLbbox WorldHotZone ()
|
|
{
|
|
|
|
return hot_zone;
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
void WorldTerm (void)
|
|
{
|
|
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
void WorldReset (void)
|
|
{
|
|
|
|
//If we're already fading out, then this is the developer hammering on the
|
|
//"rebuild" button. Let's hurry things up for the nice man...
|
|
if (fade_state == FADE_OUT)
|
|
do_reset ();
|
|
//If reset is called but the world isn't ready, then don't bother fading out.
|
|
//The program probably just started.
|
|
fade_state = FADE_OUT;
|
|
fade_start = GetTickCount ();
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
void WorldRender ()
|
|
{
|
|
|
|
//Cheap - render the crappy one-texture streets
|
|
if (!SHOW_DEBUG_GROUND)
|
|
return;
|
|
glDepthMask (false);
|
|
glDisable (GL_CULL_FACE);
|
|
glDisable (GL_BLEND);
|
|
glEnable (GL_TEXTURE_2D);
|
|
glColor3f (1,1,1);
|
|
glBindTexture (GL_TEXTURE_2D, TextureId (TEXTURE_GROUND));
|
|
glBegin (GL_QUADS);
|
|
glTexCoord2f (0, 0); glVertex3f ( 0., 0, 0);
|
|
glTexCoord2f (0, 1); glVertex3f ( 0, 0, 1024);
|
|
glTexCoord2f (1, 1); glVertex3f ( 1024, 0, 1024);
|
|
glTexCoord2f (1, 0); glVertex3f ( 1024, 0, 0);
|
|
glEnd ();
|
|
glDepthMask (true);
|
|
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
float WorldFade (void)
|
|
{
|
|
|
|
return fade_current;
|
|
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
void WorldUpdate (void)
|
|
{
|
|
|
|
unsigned fade_delta;
|
|
unsigned now, elapsed;
|
|
|
|
now = GetTickCount ();
|
|
if (reset_needed) {
|
|
do_reset (); //Now we've faded out the scene, rebuild it
|
|
}
|
|
if (fade_state != FADE_IDLE) {
|
|
if (fade_state == FADE_WAIT && TextureReady () && EntityReady ()) {
|
|
fade_state = FADE_IN;
|
|
fade_start = now;
|
|
fade_current = 1.0f;
|
|
}
|
|
fade_delta = now - fade_start;
|
|
//See if we're done fading in or out
|
|
if (fade_delta > FADE_TIME && fade_state != FADE_WAIT) {
|
|
if (fade_state == FADE_OUT) {
|
|
reset_needed = true;
|
|
fade_state = FADE_WAIT;
|
|
fade_current = 1.0f;
|
|
} else {
|
|
fade_state = FADE_IDLE;
|
|
fade_current = 0.0f;
|
|
start_time = time (NULL);
|
|
}
|
|
} else {
|
|
fade_current = (float)fade_delta / FADE_TIME;
|
|
if (fade_state == FADE_IN)
|
|
fade_current = 1.0f - fade_current;
|
|
if (fade_state == FADE_WAIT)
|
|
fade_current = 1.0f;
|
|
}
|
|
if (!TextureReady ())
|
|
fade_current = 1.0f;
|
|
}
|
|
if (fade_state == FADE_IDLE && !TextureReady ()) {
|
|
fade_state = FADE_IN;
|
|
fade_start = now;
|
|
}
|
|
elapsed = time (NULL) - start_time;
|
|
if (fade_state == FADE_IDLE && elapsed > RESET_INTERVAL)
|
|
WorldReset ();
|
|
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
void WorldInit (void)
|
|
{
|
|
|
|
last_update = GetTickCount ();
|
|
for (int i = 0; i < CARS; i++)
|
|
new CCar ();
|
|
sky = new CSky ();
|
|
WorldReset ();
|
|
fade_state = FADE_OUT;
|
|
fade_start = 0;
|
|
|
|
}
|