diff options
author | Clyne Sullivan <tullivan99@gmail.com> | 2015-11-03 08:45:29 -0500 |
---|---|---|
committer | Clyne Sullivan <tullivan99@gmail.com> | 2015-11-03 08:45:29 -0500 |
commit | 505027ab6dec50f991e2c21abf270c491128487a (patch) | |
tree | f4ae696cfaeac81fc05a9405f65c204d50625f33 | |
parent | f8ebf8fd3d5691cd45566281a1f54c5ecbc43762 (diff) |
ui improvements
-rw-r--r-- | Changelog | 9 | ||||
-rw-r--r-- | include/common.h | 2 | ||||
-rw-r--r-- | include/ui.h | 80 | ||||
-rw-r--r-- | include/world.h | 1 | ||||
-rw-r--r-- | main.cpp | 372 | ||||
-rw-r--r-- | shader.frag | 4 | ||||
-rw-r--r-- | src/entities.cpp | 2 | ||||
-rw-r--r-- | src/inventory.cpp | 2 | ||||
-rw-r--r-- | src/ui.cpp | 49 | ||||
-rw-r--r-- | src/world.cpp | 32 | ||||
-rw-r--r-- | test.frag | 4 |
11 files changed, 365 insertions, 192 deletions
@@ -222,3 +222,12 @@ - improved shaders (can now see drawn objects) - re-organized Goals.txt - began working on game concept/design/plot/everything (in google doc) + +11/3/2015: +========== + + - added world gen based off of functions + - added cutting to black + - added typewriter-esque text drawing + - documented ui.h + - continued work on GLSL shaders diff --git a/include/common.h b/include/common.h index b016020..db60dd6 100644 --- a/include/common.h +++ b/include/common.h @@ -58,7 +58,7 @@ typedef struct { #define GAME_NAME "Independent Study v.0.3 alpha" -#define SCREEN_WIDTH 1280 +#define SCREEN_WIDTH 1200 #define SCREEN_HEIGHT 720 //#define FULLSCREEN diff --git a/include/ui.h b/include/ui.h index 79ca9a7..6ebbd1f 100644 --- a/include/ui.h +++ b/include/ui.h @@ -2,39 +2,87 @@ #define UI_H #include <common.h> -#include <cstdarg> // For putText() +#include <cstdarg> -#include <world.h> // World-switching stuff -#include <ft2build.h> // FreeType stuff +#include <world.h> +#include <ft2build.h> #include FT_FREETYPE_H #define DEBUG -namespace ui { // Functions are kept in a namespace simply - // for organization +namespace ui { + + /* + * Contains the coordinates of the mouse in the window. + */ extern vec2 mouse; + /* + * These flags are used elsewhere. + */ + extern bool debug; extern bool posFlag; - extern bool dialogBoxExists; extern unsigned int fontSize; - void initFonts(void); // Checks for and initializes the FreeType 2 library + /* + * Initializes the FreeType system. + */ + + void initFonts(void); + + /* + * Sets the current font/font size. + */ + + void setFontFace(const char *ttf); + void setFontSize(unsigned int size); + + /* + * Draws a string of text at the given coordinates. + */ + + float putString(const float x,const float y,const char *s); + + /* + * Draws a formatted string at the given coordinates. + */ + + float putText(const float x,const float y,const char *str,...); + + /* + * Creates a dialogBox text string (format: `name`: `text`). This function simply sets up + * variables that are drawn in ui::draw(). When the dialog box exists player control is + * limited until a right click is given, closing the box. + */ + + void dialogBox(const char *name,const char *text,...); + + /* + * Draws a larger string in the center of the screen. Drawing is done inside this function. + */ + + void importantText(const char *text,...); + + /* + * Draw various UI elements (dialogBox, player health) + */ + + void draw(void); - void setFontFace(const char *ttf); // Checks and unpacks the TTF file for use by putString() and putText() - void setFontSize(unsigned int size); // Sets the size of the currently loaded font to 'size' pixels + /* + * Handle keyboard/mouse events. + */ - float putString(const float x,const float y,const char *s); // Draws the string 's' to the coordinates ('x','y'). The height (and therefore the width) - // are determined by what's currently set by setFontSize() - float putText(const float x,const float y,const char *str,...); // Draws the formatted string 'str' using putString() + void handleEvents(void); - void dialogBox(const char *name,const char *text,...); // Prepares a dialog box to be drawn (its drawn as a black background at the top of the - // screen and then 'text' is putString()'d + /* + * Toggle the black overlay thing. + */ - void draw(void); // Draws things like dialogBox's if necessary + void toggleBlack(void); - void handleEvents(void); // Handles keyboard and mouse events } #endif // UI_H diff --git a/include/world.h b/include/world.h index 85b02c9..81ec335 100644 --- a/include/world.h +++ b/include/world.h @@ -82,6 +82,7 @@ public: */ virtual void generate(unsigned int width); + void generateFunc(unsigned int width,unsigned int (*func)(unsigned int)); /* * Looks for the furthest back layer in this world and adds a new layer of width `width` behind it. @@ -8,21 +8,21 @@ /* * TICKS_PER_SEC & MSEC_PER_TICK - * + * * The game's main loop mainly takes care of two things: drawing to the * screen and handling game logic, from user input to world gravity stuff. * The call for rendering is made every time the main loop loops, and then * uses interpolation for smooth drawing to the screen. However, the call * for logic would be preferred to be run every set amount of time. - * - + * + * * The logic loop is currently implemented to run at a certain interval * that we call a 'tick'. As one may now guess, TICKS_PER_SEC defines the * amount of ticks that should be made every second. MSEC_PER_TICK then * does a simple calculation of how many milliseconds elapse per each * 'tick'. Simple math is then done in the main loop using MSEC_PER_TICK * to call the logic handler when necessary. - * + * */ #define TICKS_PER_SEC 20 @@ -30,13 +30,13 @@ /* * window & mainGLContext - * + * * In order to draw using SDL and its openGL facilities SDL requires * an SDL_Window object, which spawns a window for the program to draw * to. Once the SDL_Window is initialized, an SDL_GLContext is made so * that openGL calls can be made to SDL. The game requires both of these * variables to initialize. - * + * */ SDL_Window *window = NULL; @@ -47,20 +47,20 @@ SDL_GLContext mainGLContext = NULL; * background image. Currently there is only one background image for the * main world; this will be changed and bgImage likely removed once better * backgrounds are implemented. - * + * */ static GLuint bgDay, bgNight, bgMtn, bgTreesFront, bgTreesMid, bgTreesFar; /* * gameRunning - * + * * This is one of the most important variables in the program. The main * loop of the game is set to break once this variable is set to false. * The only call to modify this variable is made in src/ui.cpp, where it * is set to false if either an SDL_QUIT message is received (the user * closes the window through their window manager) or if escape is pressed. - * + * */ bool gameRunning; @@ -71,20 +71,20 @@ bool gameRunning; * variable. This should only be changed when layer switch * buttons are pressed (see src/ui.cpp), or when the player * enters a Structure/Indoor World (see src/ui.cpp again). - * + * * player - This points to a Player object, containing everything for * the player. Most calls made with currentWorld require a * Player object as an argument, and glOrtho is set based * off of the player's coordinates. This is probably the one * Entity-derived object that is not pointed to in the entity * array. - * + * * entity - Contains pointers to 'all' entities that have been created in * the game, including NPCs, Structures, and Mobs. World draws * and entity handling done by the world cycle through entities * using this array. Entities made that aren't added to this * array probably won't be noticable by the game. - * + * */ World *currentWorld=NULL; @@ -94,9 +94,9 @@ extern std::vector<Entity * > entity; /* * tickCount contains the number of ticks generated since main loop entrance. * This variable might be used anywhere. - * + * * deltaTime is used for interpolation stuff. - * + * * Pretty sure these variables are considered static as they might be externally * referenced somewhere. */ @@ -107,13 +107,13 @@ unsigned int deltaTime = 0; /* * */ + GLuint fragShader; GLuint shaderProgram; /* * names is used to open a file containing all possible NPC names. It is externally * referenced in src/entities.cpp for getting random names. - * */ FILE *names; @@ -121,7 +121,6 @@ FILE *names; /* * These variables are used by SDL_mixer to create sound. * horn is not currently used, although it may be set. - * */ Mix_Music *music; @@ -130,35 +129,34 @@ Mix_Chunk *horn; /* * loops is used for texture animation. It is believed to be passed to entity * draw functions, although it may be externally referenced instead. - * */ unsigned int loops = 0; // Used for texture animation /* * initEverything - * + * * Before the main loop, things like the player, entities, and worlds should * be created. This game has not reached the point that these can be scripted * or programmed, so this function substitues for that. It is defined in * src/gameplay.cpp. - * + * */ extern void initEverything(void); /* * mainLoop is in fact the main loop, which runs 'infinitely' (as long as gameRunning - - - + * + * + * * is set). Each loop updates timing values (tickCount and deltaTime), runs logic() * if MSEC_PER_TICK milliseconds have passed, and then runs render(). - * + * * logic handles all user input and entity/world physics. - * + * * render handles all drawing to the window, calling draw functions for everything. - * + * */ void logic(void); @@ -184,7 +182,7 @@ std::string readFile(const char *filePath) { return content; } -/* +/* * This offset is used as the player offset in the world drawing so * everything can be moved according to the player */ @@ -192,12 +190,12 @@ vec2 offset; /* * millis - * + * * We've encountered many problems when attempting to create delays for triggering * the logic function. As a result, we decided on using the timing libraries given * by <chrono> in the standard C++ library. This function simply returns the amount * of milliseconds that have passed sine the epoch. - * + * */ #ifdef __WIN32__ @@ -224,62 +222,60 @@ static vec2 star[100]; /******************************************************************************* * MAIN ************************************************************************ *******************************************************************************/ - + int main(int argc, char *argv[]){ gameRunning=false; - + /* * (Attempt to) Initialize SDL libraries so that we can use SDL facilities and eventually * make openGL calls. Exit if there was an error. - * */ - + if(SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO) < 0){ std::cout << "SDL was not able to initialize! Error: " << SDL_GetError() << std::endl; return -1; } - + // Run SDL_Quit when main returns - atexit(SDL_Quit); - - /* - * (Attempt to) Initialize SDL_image libraries with IMG_INIT_PNG so that we can load PNG - * textures for the entities and stuff. - * - */ - - if(!(IMG_Init(IMG_INIT_PNG) & IMG_INIT_PNG)){ + atexit(SDL_Quit); + + /* + * (Attempt to) Initialize SDL_image libraries with IMG_INIT_PNG so that we can load PNG + * textures for the entities and stuff. + */ + + if(!(IMG_Init(IMG_INIT_PNG) & IMG_INIT_PNG)){ std::cout << "Could not init image libraries! Error: " << IMG_GetError() << std::endl; return -1; } - + // Run IMG_Quit when main returns atexit(IMG_Quit); - + /* * (Attempt to) Initialize SDL_mixer libraries for loading and playing music/sound files. - * + * */ - + if(Mix_OpenAudio( 44100, MIX_DEFAULT_FORMAT, 2, 2048) < 0){ std::cout << "SDL_mixer could not initialize! Error: " << Mix_GetError() << std::endl; } - + // Run Mix_Quit when main returns atexit(Mix_Quit); - - /* - * Create a window for SDL to draw to. Most parameters are the default, except for the - * following which are defined in include/common.h: - * - * GAME_NAME the name of the game that is displayed in the window title bar - * SCREEN_WIDTH the width of the created window - * SCREEN_HEIGHT the height of the created window - * FULLSCREEN makes the window fullscreen - * - */ - - window = SDL_CreateWindow(GAME_NAME, + + /* + * Create a window for SDL to draw to. Most parameters are the default, except for the + * following which are defined in include/common.h: + * + * GAME_NAME the name of the game that is displayed in the window title bar + * SCREEN_WIDTH the width of the created window + * SCREEN_HEIGHT the height of the created window + * FULLSCREEN makes the window fullscreen + * + */ + + window = SDL_CreateWindow(GAME_NAME, SDL_WINDOWPOS_UNDEFINED, // Spawn the window at random (undefined) x and y coordinates SDL_WINDOWPOS_UNDEFINED, // SCREEN_WIDTH, @@ -288,35 +284,35 @@ int main(int argc, char *argv[]){ #ifdef FULLSCREEN | SDL_WINDOW_FULLSCREEN #endif // FULLSCREEN - ); - + ); + /* * Exit if the window cannot be created */ - + if(window==NULL){ std::cout << "The window failed to generate! SDL_Error: " << SDL_GetError() << std::endl; return -1; } - + /* * Create the SDL OpenGL context. Once created, we are allowed to use OpenGL functions. * Saving this context to mainGLContext does not appear to be necessary as mainGLContext * is never referenced again. - * + * */ - + if((mainGLContext = SDL_GL_CreateContext(window)) == NULL){ std::cout << "The OpenGL context failed to initialize! SDL_Error: " << SDL_GetError() << std::endl; return -1; } - + /* * Initialize GLEW libraries, and exit if there was an error. * Not sure what they're for yet. - * + * */ - + GLenum err; #ifndef __WIN32__ glewExperimental = GL_TRUE; @@ -364,6 +360,7 @@ int main(int argc, char *argv[]){ /* * Initializes our shaders so that the game has shadows. */ + #ifdef SHADERS std::cout << "Initializing shaders!" << std::endl; @@ -449,6 +446,10 @@ int main(int argc, char *argv[]){ initInventorySprites(); + /* + * Generate coordinates for stars that are drawn at night. + */ + unsigned int i; for(i=0;i<100;i++){ star[i].x=getRand()%currentWorld->getTheWidth()-currentWorld->getTheWidth()/2; @@ -557,7 +558,29 @@ void mainLoop(void){ render(); // Call the render loop } +extern bool fadeEnable; +static unsigned char fadeIntensity = 0; + void render(){ + + /* + * This offset variable is what we use to move the camera and locked + * objects on the screen so they always appear to be in the same relative area + */ + + offset = player->loc; + offset.x += player->width/2; + + /* + * If the camera will go off of the left or right of the screen we want to lock it so we can't + * see past the world render + */ + + if(player->loc.x - SCREEN_WIDTH/2 < currentWorld->getTheWidth() * -0.5f) + offset.x = ((currentWorld->getTheWidth() * -0.5f) + SCREEN_WIDTH / 2) + player->width / 2; + if(player->loc.x + SCREEN_WIDTH/2 > currentWorld->getTheWidth() * 0.5f) + offset.x = ((currentWorld->getTheWidth() * 0.5f) - SCREEN_WIDTH / 2) + player->width / 2; + /* * These functions run everyloop to update the current stacks presets * @@ -594,29 +617,16 @@ void render(){ * glLoadIdentity This scales the current matrix back to the origin so the * translations are seen normally on a stack. */ - - /* - * This offset variable is what we use to move the camera and locked - * objects on the screen so they always appear to be in the same relative area - */ - offset = player->loc; - offset.x += player->width/2; - - /* - * If the camera will go off of the left or right of the screen we want to lock it so we can't - * see past the world render - */ - if(player->loc.x - SCREEN_WIDTH/2 < currentWorld->getTheWidth()*-0.5f)offset.x = ((currentWorld->getTheWidth()*-0.5f) + SCREEN_WIDTH/2) + player->width/2; - if(player->loc.x + SCREEN_WIDTH/2 > currentWorld->getTheWidth()*0.5f)offset.x = ((currentWorld->getTheWidth()*0.5f) - SCREEN_WIDTH/2) + player->width/2; - glMatrixMode(GL_PROJECTION); //set the matrix mode as projection so we can set the ortho size and the camera settings later on - glPushMatrix(); //push the matrix to the top of the matrix stack - glLoadIdentity(); //replace the entire matrix stack with the updated GL_PROJECTION mode + + glMatrixMode(GL_PROJECTION); + glPushMatrix(); + glLoadIdentity(); glOrtho((offset.x-SCREEN_WIDTH/2),(offset.x+SCREEN_WIDTH/2),0,SCREEN_HEIGHT,-1,1); - glMatrixMode(GL_MODELVIEW); //set the matrix to modelview so we can draw objects - glPushMatrix(); //push the matrix to the top of the matrix stack - glLoadIdentity(); //replace the entire matrix stack with the updated GL_MODELVIEW mode + glMatrixMode(GL_MODELVIEW); + glPushMatrix(); + glLoadIdentity(); glEnable(GL_STENCIL_TEST); - glPushMatrix(); //basically here we put a blank canvas (new matrix) on the screen to draw on + glPushMatrix(); /* * glPushAttrib This passes attributes to the renderer so it knows what it can @@ -660,26 +670,41 @@ void render(){ glDisable(GL_TEXTURE_2D); + /* + * Draws stars if it is an appropriate time of day for them. + */ + if(((weather==DARK )&(tickCount%DAY_CYCLE)<DAY_CYCLE/2) || ((weather==SUNNY)&(tickCount%DAY_CYCLE)>DAY_CYCLE*.75) ){ - if(tickCount%DAY_CYCLE){ + + if(tickCount%DAY_CYCLE){ // The above if statement doesn't check for exact midnight. + glColor4ub(255,255,255,255); for(unsigned int i=0;i<100;i++){ glRectf(star[i].x+offset.x*.9,star[i].y,star[i].x+offset.x*.9+HLINE,star[i].y+HLINE); } + } } + /* + * Calculate the Y to start drawing the background at, and a value for how shaded the + * background elements should be. + */ + int base = 40 - (int)worldGetYBase(currentWorld); int shade = worldShade*2; glEnable(GL_TEXTURE_2D); + /* + * Draw the mountains. + */ + glBindTexture(GL_TEXTURE_2D, bgMtn); glBegin(GL_QUADS); safeSetColorA(150-shade,150-shade,150-shade,220); - //glColor4ub(150,150,150,220); - for(int i = 0; i <= currentWorld->getTheWidth()/1920; i++){ + for(int i = 0; i <= currentWorld->getTheWidth()/1920; i++){ glTexCoord2i(0,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * i)+offset.x*.85,base); glTexCoord2i(1,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * (i+1))+offset.x*.85,base); glTexCoord2i(1,0);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * (i+1))+offset.x*.85,base+1080); @@ -687,11 +712,13 @@ void render(){ } glEnd(); + /* + * Draw three layers of trees. + */ glBindTexture(GL_TEXTURE_2D, bgTreesFar); glBegin(GL_QUADS); safeSetColorA(100-shade,100-shade,100-shade,240); - //glColor4ub(100,100,100,240); for(int i = 0; i <= currentWorld->getTheWidth()/1920; i++){ glTexCoord2i(0,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * i)+offset.x*.6,base); glTexCoord2i(1,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * (i+1))+offset.x*.6,base); @@ -703,7 +730,6 @@ void render(){ glBindTexture(GL_TEXTURE_2D, bgTreesMid); glBegin(GL_QUADS); safeSetColorA(150-shade,150-shade,150-shade,250); - //glColor4ub(150,150,150,250); for(int i = 0; i <= currentWorld->getTheWidth()/1920; i++){ glTexCoord2i(0,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * i)+offset.x*.4,base); glTexCoord2i(1,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * (i+1))+offset.x*.4,base); @@ -715,7 +741,6 @@ void render(){ glBindTexture(GL_TEXTURE_2D, bgTreesFront); glBegin(GL_QUADS); safeSetColorA(255-shade,255-shade,255-shade,255); - //glColor4ub(255,255,255,255); for(int i = 0; i <= currentWorld->getTheWidth()/1920; i++){ glTexCoord2i(0,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * i)+offset.x*.25,base); glTexCoord2i(1,1);glVertex2i((currentWorld->getTheWidth()*-0.5f)+(1920 * (i+1))+offset.x*.25,base); @@ -734,25 +759,25 @@ void render(){ currentWorld->draw(player); + /* + * Apply shaders if desired. + */ + #ifdef SHADERS + glUseProgramObjectARB(shaderProgram); glUniform2f(glGetUniformLocation(shaderProgram, "lightLocation"), 640,100); glUniform3f(glGetUniformLocation(shaderProgram, "lightColor"), 1,1,1); glUniform1f(glGetUniformLocation(shaderProgram, "lightStrength"), 100 + (1000-(shade*10))); std::cout << 100 + (1000-(shade*10)) << std::endl; - //glBlendFunc(GL_ONE, GL_ONE); - #endif //SHADERS - - glColor4ub(0,0,0,200); - glRectf(offset.x-SCREEN_WIDTH/2,0,offset.x+SCREEN_WIDTH/2,SCREEN_HEIGHT); - - #ifdef SHADERS + glColor4ub(0,0,0,200); + glRectf(offset.x-SCREEN_WIDTH/2,0,offset.x+SCREEN_WIDTH/2,SCREEN_HEIGHT); glUseProgramObjectARB(0); + #endif //SHADERS player->inv->draw(); - /* * Draw UI elements. This includes the player's health bar and the dialog box. */ @@ -794,21 +819,36 @@ void render(){ glVertex2i(SCREEN_WIDTH/2+offset.x, player->loc.y); glEnd(); } - + } - + /* * Draw a white triangle as a replacement for the mouse's cursor. */ - + glColor3ub(255,255,255); - + glBegin(GL_TRIANGLES); glVertex2i(ui::mouse.x ,ui::mouse.y ); glVertex2i(ui::mouse.x+HLINE*3.5,ui::mouse.y ); glVertex2i(ui::mouse.x ,ui::mouse.y-HLINE*3.5); glEnd(); + /* + * Here we draw a black overlay if it's been requested. + */ + + if(fadeIntensity){ + glColor4ub(0,0,0,fadeIntensity); + glRectf(player->loc.x-SCREEN_WIDTH, + SCREEN_HEIGHT, + player->loc.x+SCREEN_WIDTH, + 0); + if(fadeIntensity == 255){ + ui::importantText("The screen is black."); + } + } + /************************** **** END RENDERING **** **************************/ @@ -829,143 +869,147 @@ void render(){ } void logic(){ - + /* * NPCSelected is used to insure that only one NPC is made interactable with the mouse * if, for example, multiple entities are occupying one space. */ - + static bool NPCSelected = false; - + /* * Handle user input (keyboard & mouse). */ - + ui::handleEvents(); - + /* * Run the world's detect function. This handles the physics of the player and any entities * that exist in this world. */ - + currentWorld->detect(player); if(player->loc.y<.02)gameRunning=false; - + /* * Entity logic: This loop finds every entity that is alive and in the current world. It then * basically runs their AI functions depending on what type of entity they are. For NPCs, * click detection is done as well for NPC/player interaction. - * + * */ - + for(int i = 0 ; i < entity.size(); i++){ if(!entity[i]->alive)std::cout<<"Entity "<<i<<" is not alive!"<<std::endl; - + /* * Check if the entity is in this world and is alive. */ - + if(entity[i]->inWorld == currentWorld && entity[i]->alive){ - + /* * Switch on the entity's type and handle them accordingly */ - + switch(entity[i]->type){ - + case NPCT: // Handle NPCs - + /* * Make the NPC 'wander' about the world if they're allowed to do so. * Entity->canMove is modified when a player interacts with an NPC so * that the NPC doesn't move when it talks to the player. - * + * */ - + if(entity[i]->canMove) NPCp(entity[i])->wander((rand() % 120 + 30)); - + /* * Don't bother handling the NPC if another has already been handled. */ - + if(NPCSelected){ entity[i]->near=false; break; } - + /* * Check if the NPC is under the mouse. */ - + if(ui::mouse.x >= entity[i]->loc.x && ui::mouse.x <= entity[i]->loc.x + entity[i]->width && ui::mouse.y >= entity[i]->loc.y && ui::mouse.y <= entity[i]->loc.y + entity[i]->width ){ - + /* * Check of the NPC is close enough to the player for interaction to be * considered legal. In other words, require the player to be close to - * the NPC in order to interact with it. - * + * the NPC in order to interact with it. + * * This uses the Pythagorean theorem to check for NPCs within a certain * radius (40 HLINEs) of the player's coordinates. - * - */ - + * + */ + if(pow((entity[i]->loc.x - player->loc.x),2) + pow((entity[i]->loc.y - player->loc.y),2) <= pow(40*HLINE,2)){ - + /* * Set Entity->near so that this NPC's name is drawn under them, and toggle NPCSelected * so this NPC is the only one that's clickable. */ - + entity[i]->near=true; NPCSelected=true; - + /* * Check for a right click, and allow the NPC to interact with the * player if one was made. */ - + if(SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_RIGHT)){ - + NPCp(entity[i])->interact(); //Mix_PlayChannel( -1, horn, 0); // Audio feedback - + } } - + /* * Hide the NPC's name if the mouse isn't on the NPC. */ - + }else entity[i]->near=false; - + break; // End case NPCT - + case MOBT: // Handle Mobs - + /* * Run the Mob's AI function. */ - + switch(entity[i]->subtype){ case MS_RABBIT: case MS_BIRD: Mobp(entity[i])->wander((rand()%240 + 15)); // Make the mob wander break; } - + break; // End case MOBT - + default:break; - + } } } - + + /* + * Switch between day and night (SUNNY and DARK) if necessary. + */ + if(!(tickCount%DAY_CYCLE)||!tickCount){ if(weather==SUNNY){ weather=DARK; @@ -973,15 +1017,31 @@ void logic(){ weather=SUNNY; } } - + + /* + * Calculate an in-game shading value (as opposed to GLSL shading). + */ + #define PI 3.1415926535 worldShade=50*sin((tickCount+(DAY_CYCLE/2))/(DAY_CYCLE/PI)); - + + /* + * Transition to and from black if necessary. + */ + + if(fadeEnable){ + if(fadeIntensity < 160)fadeIntensity+=5; + else if(fadeIntensity < 255)fadeIntensity+=1; + }else{ + if(fadeIntensity > 150)fadeIntensity-=1; + else if(fadeIntensity) fadeIntensity-=5; + } + /* * Increment a loop counter used for animating sprites. */ - + loops++; - tickCount++; - NPCSelected=false; + tickCount++; // Used for day/night cycles + NPCSelected=false; // See above } diff --git a/shader.frag b/shader.frag index 6f90a5f..857c6b9 100644 --- a/shader.frag +++ b/shader.frag @@ -1,4 +1,4 @@ -#version 130
+#version 120
uniform vec2 lightLocation;
uniform vec3 lightColor;
@@ -17,4 +17,4 @@ void main(){ vec4 color = vec4(0, 0, 0, 0.8f - pow(attenuation, 3)) * vec4(lightColor, 1);
gl_FragColor = color;
-}
\ No newline at end of file +}
diff --git a/src/entities.cpp b/src/entities.cpp index 42ed5a9..acc45bd 100644 --- a/src/entities.cpp +++ b/src/entities.cpp @@ -176,7 +176,7 @@ void Entity::draw(void){ //draws the entities glMatrixMode(GL_MODELVIEW); glPopMatrix(); if(near){ - ui::setFontSize(10); + ui::setFontSize(14); ui::putText(loc.x,loc.y-ui::fontSize-HLINE/2,"%s",name); } } diff --git a/src/inventory.cpp b/src/inventory.cpp index 11f2ad7..cb33deb 100644 --- a/src/inventory.cpp +++ b/src/inventory.cpp @@ -92,7 +92,7 @@ int Inventory::takeItem(ITEM_ID id,unsigned char count){ void Inventory::draw(void){ unsigned int i=0; float y=SCREEN_HEIGHT/2,xoff; - ui::setFontSize(12); + ui::setFontSize(14); ui::putText(offset.x-SCREEN_WIDTH/2,y,"Inventory:"); while(item[i].count){ y-=HLINE*12; @@ -16,6 +16,8 @@ static GLuint ftex; static char *dialogBoxText; +bool fadeEnable = false; + namespace ui { vec2 mouse; bool debug=false; @@ -139,6 +141,23 @@ namespace ui { }while(s[++i]); return xo; } + char *typeOut(char *str){ + static unsigned int sinc,linc,size; + static char *ret = NULL; + unsigned int i; + if(!ret || size != strlen(str)){ + size=strlen(str); + ret=(char *)calloc(size,sizeof(char)); + linc=0; + sinc=2; + } + if(++sinc==3){ + sinc=0; + strncpy(ret+linc,str+linc,1); + if(linc<size)linc++; + } + return ret; + } float putText(const float x,const float y,const char *str,...){ // putText() simply runs 'str' and the extra arguments though va_list args; // vsnprintf(), which'll store the complete string to a buffer char *buf; // that's then passed to putString() @@ -167,6 +186,21 @@ namespace ui { vsnprintf(dialogBoxText+name_len+2,512-name_len-2,text,dialogArgs); va_end(dialogArgs); } + void importantText(const char *text,...){ + va_list textArgs; + char *ttext; + va_start(textArgs,text); + ttext=(char *)calloc(512,sizeof(char)); + vsnprintf(ttext,512,text,textArgs); + va_end(textArgs); + setFontSize(24); + char *rtext; + rtext=typeOut(ttext); + putString(player->loc.x+player->width/2-SCREEN_WIDTH/2, + SCREEN_HEIGHT/2+fontSize, + rtext); + free(ttext); + } void draw(void){ float x,y; if(dialogBoxExists){ @@ -174,10 +208,12 @@ namespace ui { x=player->loc.x-SCREEN_WIDTH/2+HLINE*8; y=SCREEN_HEIGHT-HLINE*8; glRectf(x,y,x+SCREEN_WIDTH-HLINE*16,y-SCREEN_HEIGHT/4); - setFontSize(12); - putString(x+HLINE,y-fontSize-HLINE,dialogBoxText); + char *rtext; + rtext=typeOut(dialogBoxText); + setFontSize(16); + putString(x+HLINE,y-fontSize-HLINE,rtext); } - setFontSize(12); + setFontSize(14); putText(((SCREEN_WIDTH/2)+offset.x)-125,SCREEN_HEIGHT-fontSize,"Health: %u/%u",player->health>0?(unsigned)player->health:0, (unsigned)player->maxHealth); if(player->alive){ @@ -269,8 +305,9 @@ namespace ui { player->inv->useItem(); } if(SDL_KEY==SDLK_c){ - dialogBox("","You pressed `c`, but nothing happened."); + dialogBox("","You pressed `c`, but nothing happened?"); } + if(SDL_KEY==SDLK_p)toggleBlack(); if(SDL_KEY==SDLK_LSHIFT)player->speed = debug?4:3; // Sprint if(SDL_KEY==SDLK_LCTRL)player->speed = .5; } @@ -304,4 +341,8 @@ namespace ui { } } } + + void toggleBlack(void){ + fadeEnable ^= true; + } } diff --git a/src/world.cpp b/src/world.cpp index 36993b0..0ec0f35 100644 --- a/src/world.cpp +++ b/src/world.cpp @@ -33,6 +33,14 @@ float worldGetYBase(World *w){ } World::World(void){ + /* + * Nullify pointers to other worlds. + */ + + behind = + infront = + toLeft = + toRight = NULL; } void World::generate(unsigned int width){ // Generates the world and sets all variables contained in the World class. @@ -129,15 +137,21 @@ void World::generate(unsigned int width){ // Generates the world and sets all va */ x_start=0 - getWidth(this) / 2; - - /* - * Nullify pointers to other worlds. - */ - - behind = - infront = - toLeft = - toRight = NULL; +} + +void World::generateFunc(unsigned int width,unsigned int (*func)(unsigned int)){ + unsigned int i; + if((lineCount = width) <= 0) + abort(); + line=(struct line_t *)calloc(lineCount,sizeof(struct line_t)); + for(i=0;i<lineCount;i++){ + line[i].y=func(i); + line[i].color=rand() % 20 + 100; + line[i].gh[0]=(getRand() % 16) / 3.5 + 2; + line[i].gh[1]=(getRand() % 16) / 3.5 + 2; + line[i].gs=true; + } + x_start=0 - getWidth(this) / 2; } World::~World(void){ @@ -1,4 +1,4 @@ -#version 130
+#version 120
void main(){
gl_FragColor = vec4(1.0, 1.0, 1.0, 1.0);
-}
\ No newline at end of file +}
|