diff options
-rw-r--r-- | Changelog | 31 | ||||
-rw-r--r-- | assets/pages/gootaGoFast.png | bin | 20631 -> 20668 bytes | |||
-rw-r--r-- | frig.frag | 2 | ||||
-rw-r--r-- | include/common.hpp | 13 | ||||
-rw-r--r-- | include/entities.hpp | 47 | ||||
-rw-r--r-- | include/ui.hpp | 62 | ||||
-rw-r--r-- | include/ui_menu.hpp | 63 | ||||
-rw-r--r-- | include/world.hpp | 6 | ||||
-rw-r--r-- | main.cpp | 48 | ||||
-rw-r--r-- | setup.mk | 2 | ||||
-rw-r--r-- | shader.frag | 14 | ||||
-rw-r--r-- | src/entities.cpp | 105 | ||||
-rw-r--r-- | src/gameplay.cpp | 33 | ||||
-rw-r--r-- | src/inventory.cpp | 5 | ||||
-rw-r--r-- | src/ui.cpp | 570 | ||||
-rw-r--r-- | src/ui_menu.cpp | 275 | ||||
-rw-r--r-- | src/world.cpp | 165 | ||||
-rw-r--r-- | storyXML/0000_SpawnHill.xml.dat | 3 | ||||
-rw-r--r-- | test.frag | 39 | ||||
-rw-r--r-- | xml/playerSpawnHill1.xml | 10 |
20 files changed, 774 insertions, 719 deletions
@@ -879,3 +879,34 @@ - began work with threading pages/triggers - added debug flag for settings.xml - improved particle drawing + +4/6/2016: +========= + + - began considering vertex-based rendering + - discovered really weird thread handling issues + - broke world transitions + +4/7/2016: +========= + + - fixed world transitions + - implemented rain + - added custom health values in XML + - player can't leave arenas until mob death + - typeOut speed is consistent + - continued smart renderinging + +4/11/2016: +========== + + - heavily revised merchant code + - fixed bug that caused game to crash idly + - fixed bug with spawning NPCs at specific coordinates + - added functionality to ui to auto-complete the next typeOut + +4/12/2016: +========== + + - potentially fixed inventory drawing bug + - moved menu library to its own file diff --git a/assets/pages/gootaGoFast.png b/assets/pages/gootaGoFast.png Binary files differindex 639c0f0..9477463 100644 --- a/assets/pages/gootaGoFast.png +++ b/assets/pages/gootaGoFast.png @@ -1,4 +1,4 @@ -#version 130
+#version 120
uniform sampler2D sampler;
uniform int numLight;
diff --git a/include/common.hpp b/include/common.hpp index c9837dd..a62d75a 100644 --- a/include/common.hpp +++ b/include/common.hpp @@ -35,6 +35,19 @@ typedef unsigned int uint; #undef near #endif +/** + * Defines how many game ticks should occur in one second, affecting how often + * game logic is handled. + */ + +#define TICKS_PER_SEC 20 + +/** + * Defines how many milliseconds each game tick will take. + */ + +#define MSEC_PER_TICK ( 1000 / TICKS_PER_SEC ) + //#define SEGFAULT /** diff --git a/include/entities.hpp b/include/entities.hpp index 3bc2f98..5ab4066 100644 --- a/include/entities.hpp +++ b/include/entities.hpp @@ -56,15 +56,7 @@ class Trade{ public: std::string item[2]; int quantity[2]; - Trade(int qo, const char* o, int qt, const char* t){ - item[0] = o; - item[1] = t; - - quantity[0] = qo; - quantity[1] = qt; - - std::cout << "Trading: " << quantity[0] << " " << item[0] << " for " << quantity[1] << " " << item[1] << std::endl; - } + Trade(int qo, std::string o, int qt, std::string t); Trade(){} }; @@ -75,8 +67,7 @@ public: vec2 loc; float width; float height; - float velx; - float vely; + vec2 vel; Color color; vec2 index; float duration; @@ -84,25 +75,25 @@ public: bool fountain; bool gravity; bool behind; + bool bounce; Particles(float x, float y, float w, float h, float vx, float vy, Color c, float d){ loc.x = x; loc.y = y; + vel.x = vx; + vel.y = vy; width = w; height = h; - velx = vx; - vely = vy; color = c; duration = d; fountain = false; gravity = true; behind = false; + bounce = false; index = Texture::getIndex(c); } ~Particles(){ - } void draw(){ - //glEnable(GL_TEXTURE_2D); glColor3ub(255,255,255); glBegin(GL_QUADS); glTexCoord2f(.25*index.x, .125*index.y); glVertex2i(loc.x, loc.y); @@ -110,15 +101,23 @@ public: glTexCoord2f(.25*index.x, .125*index.y); glVertex2i(loc.x + width, loc.y + height); glTexCoord2f(.25*index.x, .125*index.y); glVertex2i(loc.x, loc.y + width); glEnd(); - //glDisable(GL_TEXTURE_2D); - //glUseProgram(0); + } + void update( float _gravity, float ground_y ) { + // handle ground collision + if ( loc.y < ground_y ) { + loc.y = ground_y; + if ( bounce ) { + vel.y *= -0.2f; + vel.x /= 4; + } else { + vel.x = vel.y = 0; + canMove = false; + } + } else if ( gravity && vel.y > -1 ) + vel.y -= _gravity * deltaTime; } bool kill(float delta){ - duration -= delta; - if(duration <= 0.0f){ - return true; - } - else return false; + return (duration -= delta) <= 0; } }; @@ -141,6 +140,8 @@ public: float speed; // A speed factor for X movement + unsigned int hitCooldown; + /* * Movement flags */ @@ -229,7 +230,7 @@ public: virtual void wander(int); }; -class Merchant : public NPC{ +class Merchant : public NPC { public: std::vector<Trade>trade; uint currTrade; diff --git a/include/ui.hpp b/include/ui.hpp index 0142f6f..6fa2451 100644 --- a/include/ui.hpp +++ b/include/ui.hpp @@ -11,54 +11,16 @@ #include <config.hpp> #include <world.hpp> +#include <ui_menu.hpp> + #include <ft2build.h> #include <SDL2/SDL_opengl.h> #include <thread> #include FT_FREETYPE_H -#define DEBUG +#define SDL_KEY e.key.keysym.sym -typedef void(*menuFunc)(); - -struct menuItem{ - int member; - union{ - struct{ - vec2 loc; - dim2 dim; - Color color; - const char* text; - menuFunc func; - }button; - struct{ - vec2 loc; - dim2 dim; - Color color; - float minValue; - float maxValue; - const char* text; - float* var; - - float sliderLoc; - }slider; - }; -}; - -class Menu{ -public: - std::vector<menuItem>items; - Menu *child; - Menu *parent; - ~Menu(){ - child = NULL; - parent = NULL; - delete child; - delete parent; - } - - void gotoChild(); - void gotoParent(); -}; +#define DEBUG typedef uint8_t BYTE; typedef uint16_t WORD; @@ -87,15 +49,13 @@ typedef struct{ } __attribute__ ((packed)) BITMAPINFOHEADER; namespace ui { - menuItem createButton(vec2 l, dim2 d, Color c, const char* t, menuFunc f); - menuItem createChildButton(vec2 l, dim2 d, Color c, const char* t); - menuItem createParentButton(vec2 l, dim2 d, Color c, const char* t); - menuItem createSlider(vec2 l, dim2 d, Color c, float min, float max, const char* t, float* v); + /** * Contains the coordinates of the mouse inside the window. */ extern vec2 mouse; + extern vec2 premouse; /* * These flags are used elsewhere. @@ -154,9 +114,11 @@ namespace ui { void merchantBox(); void closeBox(); void waitForDialog(void); - bool pageExists( void ); + bool pageExists( void ); void drawPage( std::string path ); + + void dontTypeOut( void ); /* * Draws a larger string in the center of the screen. Drawing is done inside this function. */ @@ -170,12 +132,8 @@ namespace ui { void draw(void); - - /* - * Draw various menu items - */ void quitGame(); - void drawMenu(Menu* menu); + /* diff --git a/include/ui_menu.hpp b/include/ui_menu.hpp new file mode 100644 index 0000000..9a51739 --- /dev/null +++ b/include/ui_menu.hpp @@ -0,0 +1,63 @@ +#ifndef UI_MENU_H_ +#define UI_MENU_H_ + +#include <common.hpp> +#include <config.hpp> + +typedef void (*menuFunc)(void); + +struct menuItem { + int member; + union { + struct { + vec2 loc; + dim2 dim; + Color color; + + const char *text; + menuFunc func; + } button; + struct { + vec2 loc; + dim2 dim; + Color color; + + float minValue; + float maxValue; + float sliderLoc; + + const char *text; + float *var; + } slider; + }; +}; + +class Menu { +public: + std::vector<menuItem> items; + Menu *child, *parent; + + ~Menu() { + // TODO you CANNOT delete null pointers! + /*child = NULL; + parent = NULL; + delete child; + delete parent;*/ + } + + void gotoChild(void); + void gotoParent(void); +}; + +namespace ui { + namespace menu { + menuItem createButton(vec2 l, dim2 d, Color c, const char* t, menuFunc f); + menuItem createChildButton(vec2 l, dim2 d, Color c, const char* t); + menuItem createParentButton(vec2 l, dim2 d, Color c, const char* t); + menuItem createSlider(vec2 l, dim2 d, Color c, float min, float max, const char* t, float* v); + + void draw( void ); + } +} + +#endif // UI_MENU_H_ diff --git a/include/world.hpp b/include/world.hpp index 40e4a38..cae3808 100644 --- a/include/world.hpp +++ b/include/world.hpp @@ -20,6 +20,12 @@ #define PLAYER_SPEED_CONSTANT 0.15f /** + * Gravity thing + */ + +#define GRAVITY_CONSTANT 0.001f + +/** * Defines how many game ticks it takes for a day to elapse. */ @@ -25,19 +25,6 @@ using namespace tinyxml2; #include <entities.hpp> /** - * Defines how many game ticks should occur in one second, affecting how often - * game logic is handled. - */ - -#define TICKS_PER_SEC 20 - -/** - * Defines how many milliseconds each game tick will take. - */ - -#define MSEC_PER_TICK ( 1000 / TICKS_PER_SEC ) - -/** * The window object returned by SDL when we create the main window. */ @@ -232,7 +219,7 @@ int main(int argc, char *argv[]){ * textures for the entities and stuff. */ - if(!(IMG_Init(IMG_INIT_PNG) & IMG_INIT_PNG) | !(IMG_Init(IMG_INIT_JPG) & IMG_INIT_JPG)){ + if(!(IMG_Init(IMG_INIT_PNG | IMG_INIT_JPG) & (IMG_INIT_PNG | IMG_INIT_JPG))){ std::cout << "Could not init image libraries! Error: " << IMG_GetError() << std::endl; return -1; } @@ -357,7 +344,6 @@ int main(int argc, char *argv[]){ GLint bufferln = GL_FALSE; int logLength; - fragShader = glCreateShader(GL_FRAGMENT_SHADER); glShaderSource(fragShader, 1, &shaderSource, NULL); glCompileShader(fragShader); @@ -370,9 +356,8 @@ int main(int argc, char *argv[]){ glGetShaderInfoLog(fragShader, logLength, NULL, &fragShaderError[0]); std::cout << &fragShaderError[0] << std::endl; - if(bufferln == GL_FALSE){ - std::cout << "Error compiling shader" << std::endl; - } + if ( bufferln == GL_FALSE ) + UserError("Error compiling shader"); shaderProgram = glCreateProgram(); glAttachShader(shaderProgram, fragShader); @@ -396,9 +381,7 @@ int main(int argc, char *argv[]){ fadeIntensity = 250; - std::cout << "emem" << std::endl; initEverything(); - std::cout << "meme" << std::endl; if(!currentWorld){ std::cout<<"currentWorld == NULL!"<<std::endl; @@ -725,14 +708,10 @@ void render() { } - if(currentMenu){ - ui::drawMenu(currentMenu); - } - - /* - * Draw a white triangle as a replacement for the mouse's cursor. - */ + if ( currentMenu ) + ui::menu::draw(); + // draw the mouse cursor glColor3ub(255,255,255); glEnable(GL_TEXTURE_2D); glBindTexture(GL_TEXTURE_2D, mouseTex); @@ -748,17 +727,6 @@ void render() { **** END RENDERING **** **************************/ - /* - * These next two function finish the rendering - * - * glPopMatrix This anchors all of the matrices and blends them to a single - * matrix so the renderer can draw this to the screen, since screens - * are only 2 dimensions, we have to combine the matrixes to be 2d. - * - * SDL_GL_SwapWindow Since SDL has control over our renderer, we need to now give our - * new matrix to SDL so it can pass it to the window. - */ - glPopMatrix(); SDL_GL_SwapWindow(window); } @@ -781,6 +749,7 @@ void logic(){ e->health -= 25; e->hit = true; e->forcedMove = true; + e->hitCooldown = 10; e->vel.x = 0.5f * (player->left ? -1 : 1); e->vel.y = 0.2f; break; @@ -879,7 +848,7 @@ void logic(){ switch(m->subtype){ case MS_RABBIT: case MS_BIRD: - m->wander((rand()%240 + 15)); // Make the mob wander :) + m->wander((rand()%240 + 15)); // Make the mob wander break; case MS_TRIGGER: case MS_PAGE: @@ -967,6 +936,7 @@ void logic(){ { 0, 0, 255 }, // RGB color 2500 // duration (ms) ); + currentWorld->particles.back().bounce = true; } } else if ( weather == WorldWeather::Snowy ) { for ( unsigned int r = (randGet() % 25) + 11; r--; ) { @@ -2,4 +2,4 @@ TARGET_OS = linux # Bits setting, 32 or 64 -TARGET_BITS = 32 +TARGET_BITS = 64 diff --git a/shader.frag b/shader.frag deleted file mode 100644 index 55527a1..0000000 --- a/shader.frag +++ /dev/null @@ -1,14 +0,0 @@ -#version 120
-
-uniform vec2 lightLocation;
-uniform vec3 lightColor;
-uniform float lightStrength;
-uniform float screenHeight;
-
-void main(){
- float distance = length(lightLocation - gl_FragCoord.xy);
- float attenuation = lightStrength / distance;
- vec4 color = vec4(attenuation, attenuation, attenuation, (pow(attenuation, 3)) * vec4(lightColor, 2))+.5;
-
- gl_FragColor = color;
-}
diff --git a/src/entities.cpp b/src/entities.cpp index 5470245..df32a5b 100644 --- a/src/entities.cpp +++ b/src/entities.cpp @@ -63,15 +63,29 @@ void getRandomName(Entity *e){ switch(bufs[0]){ default : - case 'm': e->gender = MALE; break; - case 'f': e->gender = FEMALE;break; + case 'm': + e->gender = MALE; + break; + case 'f': + e->gender = FEMALE; + break; } - strcpy(e->name,bufs+1); + strcpy( e->name, bufs + 1 ); delete[] bufs; } +Trade::Trade(int qo, std::string o, int qt, std::string t){ + item[0] = o; + item[1] = t; + + quantity[0] = qo; + quantity[1] = qt; + + std::cout << "Trading: " << quantity[0] << " " << item[0] << " for " << quantity[1] << " " << item[1] << std::endl; +} + void Entity::spawn(float x, float y){ //spawns the entity you pass to it based off of coords and global entity settings loc.x = x; loc.y = y; @@ -89,6 +103,7 @@ void Entity::spawn(float x, float y){ //spawns the entity you pass to it based o forcedMove = false; ticksToUse = 0; + hitCooldown = 0; if(!maxHealth)health = maxHealth = 1; @@ -376,7 +391,12 @@ void Entity::draw(void){ //draws the entities tex->bind(0); break; } - glColor3ub(255,255,255); + + if ( hitCooldown ) + glColor3ub(255,255,0); + else + glColor3ub(255,255,255); + glUseProgram(shaderProgram); glUniform1i(glGetUniformLocation(shaderProgram, "sampler"), 0); glBegin(GL_QUADS); @@ -410,6 +430,9 @@ wander( int timeRun ) if ( forcedMove ) return; + if ( hitCooldown ) + hitCooldown--; + if ( followee ) { if ( loc.x < followee->loc.x - 40 ) direction = 1; @@ -500,9 +523,13 @@ void Merchant::wander(int timeRun){ if( vel.x < 0) currentWorld->goWorldLeft( this ); - if(inside != nullptr){ - if(loc.x <= inside->loc.x)loc.x = inside->loc.x; - if(loc.x + width >= inside->loc.x + inside->width)loc.x = inside->loc.x + inside->width - width; + if ( inside != nullptr ) { + loc.y = inside->loc.y + HLINE * 2; + vel.y = GRAVITY_CONSTANT * 5; + if ( loc.x <= inside->loc.x + HLINE * 5 ) + loc.x = inside->loc.x + HLINE * 5; + else if ( loc.x + width >= inside->loc.x + inside->width - HLINE * 5 ) + loc.x = inside->loc.x + inside->width - width - HLINE * 5; } ticksToUse--; } @@ -511,20 +538,43 @@ void Merchant::interact(){ std::thread([this]{ ui::merchantBox(name, trade[currTrade], ":Accept:Good-Bye", false, "Welcome to Smithy\'s. Buy your sausages here you freaking meme lording screw-face"); ui::waitForDialog(); - if(ui::dialogOptChosen == 1){ - if(!(player->inv->takeItem(trade[currTrade].item[1],trade[currTrade].quantity[1]))) + + // handle normal dialog options + switch ( ui::dialogOptChosen ) { + // Accept + case 1: + if ( !(player->inv->takeItem( trade[currTrade].item[1], trade[currTrade].quantity[1])) ) player->inv->addItem(trade[currTrade].item[0],trade[currTrade].quantity[0]); - }else if(ui::dialogOptChosen == 2){ - }else if(ui::merchOptChosen == 1){ - if(currTrade != 0){ + break; + + // Good-bye + case 2: + break; + + default: + break; + } + + // handle merchant-specific dialog options + switch ( ui::merchOptChosen ) { + // left arrow + case 1: + if ( currTrade ) currTrade--; - interact(); - } - }else if(ui::merchOptChosen == 2){ - if(currTrade < trade.size()){ + ui::dontTypeOut(); + interact(); // TODO should we nest like this? + break; + + // right arrow + case 2: + if ( currTrade < trade.size() - 1 ) currTrade++; - interact(); - } + ui::dontTypeOut(); + interact(); + break; + + default: + break; } }).detach(); } @@ -676,20 +726,11 @@ void Mob::wander(int timeRun){ //hey(this); break; case MS_PAGE: - if(player->loc.x > loc.x - 100 && - player->loc.x < loc.x + 100 && - ui::mouse.x > loc.x && - ui::mouse.x < loc.x + width && - ui::mouse.y > loc.y - width / 2 && - ui::mouse.y < loc.y + width * 1.5 && - SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_RIGHT)){ - std::thread([this]{hey(this);}).detach(); - /*if(speed != 666){ - speed = 666; - hey(this); - speed = 0; - }*/ - } + if(player->loc.x > loc.x - 100 && player->loc.x < loc.x + 100 && // within player ranger + ui::mouse.x > loc.x && ui::mouse.x < loc.x + width && // mouse x + ui::mouse.y > loc.y - width / 2 && ui::mouse.y < loc.y + width * 1.5 && // mouse y + SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_RIGHT)) // right click + hey(this); break; default: break; diff --git a/src/gameplay.cpp b/src/gameplay.cpp index ce514a4..7b5b3ba 100644 --- a/src/gameplay.cpp +++ b/src/gameplay.cpp @@ -219,20 +219,9 @@ CONT: } void commonPageFunc( Mob *callee ){ - //static bool lock = false; - - /*if ( !lock ) { - lock = true;*/ - if ( !ui::dialogBoxExists ) { - std::cout<<"begin\n"; - ui::drawPage( callee->heyid ); - while( ui::pageExists() ); - std::cout<<"done\n"; - //ui::waitForDialog(); - - callee->health = 0; - //lock = false; - } + ui::drawPage( callee->heyid ); + ui::waitForDialog(); + callee->health = 0; } void commonTriggerFunc(Mob *callee){ @@ -310,20 +299,20 @@ void initEverything(void){ } } - pauseMenu.items.push_back(ui::createParentButton({-256/2,0},{256,75},{0.0f,0.0f,0.0f}, "Resume")); - pauseMenu.items.push_back(ui::createChildButton({-256/2,-100},{256,75},{0.0f,0.0f,0.0f}, "Options")); - pauseMenu.items.push_back(ui::createButton({-256/2,-200},{256,75},{0.0f,0.0f,0.0f}, "Save and Quit", ui::quitGame)); - pauseMenu.items.push_back(ui::createButton({-256/2,-300},{256,75},{0.0f,0.0f,0.0f}, "Segfault", segFault)); + pauseMenu.items.push_back(ui::menu::createParentButton({-256/2,0},{256,75},{0.0f,0.0f,0.0f}, "Resume")); + pauseMenu.items.push_back(ui::menu::createChildButton({-256/2,-100},{256,75},{0.0f,0.0f,0.0f}, "Options")); + pauseMenu.items.push_back(ui::menu::createButton({-256/2,-200},{256,75},{0.0f,0.0f,0.0f}, "Save and Quit", ui::quitGame)); + pauseMenu.items.push_back(ui::menu::createButton({-256/2,-300},{256,75},{0.0f,0.0f,0.0f}, "Segfault", segFault)); pauseMenu.child = &optionsMenu; pauseMenu.parent = NULL; - optionsMenu.items.push_back(ui::createSlider({0-(float)SCREEN_WIDTH/4,0-(512/2)}, {50,512}, {0.0f, 0.0f, 0.0f}, 0, 100, "Master", &VOLUME_MASTER)); - optionsMenu.items.push_back(ui::createSlider({-200,100}, {512,50}, {0.0f, 0.0f, 0.0f}, 0, 100, "Music", &VOLUME_MUSIC)); - optionsMenu.items.push_back(ui::createSlider({-200,000}, {512,50}, {0.0f, 0.0f, 0.0f}, 0, 100, "SFX", &VOLUME_SFX)); + optionsMenu.items.push_back(ui::menu::createSlider({0-(float)SCREEN_WIDTH/4,0-(512/2)}, {50,512}, {0.0f, 0.0f, 0.0f}, 0, 100, "Master", &VOLUME_MASTER)); + optionsMenu.items.push_back(ui::menu::createSlider({-200,100}, {512,50}, {0.0f, 0.0f, 0.0f}, 0, 100, "Music", &VOLUME_MUSIC)); + optionsMenu.items.push_back(ui::menu::createSlider({-200,000}, {512,50}, {0.0f, 0.0f, 0.0f}, 0, 100, "SFX", &VOLUME_SFX)); optionsMenu.child = NULL; optionsMenu.parent = &pauseMenu; - // optionsMenu.push_back(ui::createButton({-256/2,-200},{256,75},{0.0f,0.0f,0.0f}, (const char*)("Save and Quit"), ); + // optionsMenu.push_back(ui::menu::createButton({-256/2,-200},{256,75},{0.0f,0.0f,0.0f}, (const char*)("Save and Quit"), ); /* * Spawn the player and begin the game. diff --git a/src/inventory.cpp b/src/inventory.cpp index 521b695..64254e9 100644 --- a/src/inventory.cpp +++ b/src/inventory.cpp @@ -257,7 +257,7 @@ void Inventory::draw(void){ a++; } a = 0; - for ( unsigned int i = 0; i < massOrder.size() ; i++, a++ ) { + while ( ++a < massOrder.size() ) { if ( !a || massDfp[ massOrder[a - 1] ] > massRange * 0.75f ) massDfp[ massOrder[a] ] += 5.0f * deltaTime; if ( massDfp[ massOrder[a] ] > massRange ) @@ -276,7 +276,8 @@ void Inventory::draw(void){ cd -= 1.0f * deltaTime; } - for ( unsigned int i = 0; i < massRay.size(); i++, a++ ) { + a = 0; + while ( ++a < massRay.size() ) { if ( !a || massDfp[ massOrderClosing[a - 1] ] <= 0 ) massDfp[ massOrderClosing[a] ] -= 10.0f * deltaTime; if ( massDfp[ massOrderClosing[a - 1] ] < 0 ) @@ -1,12 +1,8 @@ #include <ui.hpp> -/* - * Create a macro to easily access SDL keypresses -*/ - -#define SDL_KEY e.key.keysym.sym - extern std::vector<menuItem> optionsMenu; +extern Menu *currentMenu; +extern Menu pauseMenu; extern SDL_Window *window; @@ -65,16 +61,9 @@ static unsigned char fontColor[4] = {255,255,255,255}; static std::vector<std::pair<std::string,vec3>> dialogOptText; static std::string dialogBoxText; -static vec3 merchArrowLoc[2]; +static std::vector<vec3> merchArrowLoc ( 2, vec3 { 0, 0, 0 } ); static bool typeOutDone = true; - -/* - * Menu-related objects - */ - -extern Menu* currentMenu; -extern Menu pauseMenu; - +static bool typeOutSustain = false; static Mix_Chunk *dialogClick; @@ -95,19 +84,7 @@ Mix_Chunk *battleStart; Mix_Chunk *sanic; static GLuint pageTex = 0; - -void Menu::gotoParent(){ - if(parent == NULL){ - currentMenu = NULL; - config::update(); - }else{ - currentMenu = parent; - } -} - -void Menu::gotoChild(){ - currentMenu = child; -} +static bool pageTexReady = false; void loadFontSize( unsigned int size, std::vector<GLuint> &tex, std::vector<FT_Info> &dat ) { @@ -171,7 +148,7 @@ namespace ui { */ vec2 mouse; - static vec2 premouse={0,0}; + vec2 premouse={0,0}; /* * Variety of keydown bools @@ -209,6 +186,8 @@ namespace ui { unsigned int fontSize; + void takeScreenshot(GLubyte* pixels); + /* * Initialises the Freetype library, and sets a font size. */ @@ -264,13 +243,15 @@ namespace ui { } ftex = &ftex16; ftdat = &ftdat16; - } else if ( size == 24 ){ + fontSize = 16; + } else if ( size == 24 ) { if ( !ft24loaded ) { loadFontSize( fontSize = size, ftex24, ftdat24 ); ft24loaded = true; } ftex = &ftex24; ftdat = &ftdat24; + fontSize = 24; } } @@ -410,6 +391,14 @@ namespace ui { return width; } + /** + * Prevents typeOut from typing the next string it's given. + */ + + void dontTypeOut( void ) { + typeOutSustain = true; + } + /* * Draw a string in a typewriter-esque fashion. Each letter is rendered as calls are made * to this function. Passing a different string to the function will reset the counters. @@ -417,34 +406,30 @@ namespace ui { std::string ret; std::string typeOut( std::string str ) { - static unsigned int sinc, // Acts as a delayer for the space between each character. + static unsigned int tadv = TICKS_PER_SEC / 12; + static unsigned int tickk, linc=0, // Contains the number of letters that should be drawn. size=0; // Contains the full size of the current string. - /* - * Reset values if a new string is being passed. - */ - - if(strncmp(ret.c_str(),str.c_str(),linc-1)){ - ret.clear(); // Zero the buffer - size=str.size(); // Set the new target string size - linc=0; // Reset the incrementers - sinc=1; - typeOutDone = false; + // reset values if a new string is being passed. + if ( !linc || ret.substr( 0, linc ) != str.substr( 0, linc ) ) { + tickk = tickCount + tadv; + ret = str.substr( 0, 1 ); + size = str.size(); // Set the new target string size + linc = 1; // Reset the incrementers + if ( (typeOutDone = typeOutSustain) ) + typeOutSustain = false; } - /* - * Draw the next letter if necessary. - */ - - if(typeOutDone) + if ( typeOutDone ) return str; - else if(++sinc==2){ - sinc=0; - ret.append(str, linc, 1); + // Draw the next letter if necessary. + else if ( tickk <= tickCount ) { + tickk = tickCount + tadv; + ret += str[linc]; - if(linc<size) + if ( linc < size ) linc++; else typeOutDone = true; @@ -519,15 +504,11 @@ namespace ui { ret.clear(); } - void merchantBox(const char *name,Trade trade,const char *opt,bool passive,const char *text,...){ va_list dialogArgs; std::unique_ptr<char[]> printfbuf (new char[512]); dialogPassive = passive; - - std::cout << "Market Trading: " << trade.quantity[0] << " " << trade.item[0] << " for " << trade.quantity[1] << " " << trade.item[1] << std::endl; - merchTrade = trade; // clear the buffer @@ -575,24 +556,21 @@ namespace ui { * Wait for a dialog box to be dismissed. */ - void waitForDialog(void){ - do{ - //std::thread(dialogAdvance); - //mainLoop(); - }while(dialogBoxExists); + void waitForDialog ( void ) { + while ( dialogBoxExists ); } - void waitForCover(void){ - do{ + + void waitForCover ( void ) { + while ( fadeIntensity < 255 ) mainLoop(); - }while(fadeIntensity < 255); fadeIntensity = 255; } - void waitForNothing(unsigned int ms){ + + void waitForNothing ( unsigned int ms ) { unsigned int target = millis() + ms; - do{ - mainLoop(); - }while(millis() < target); + while ( millis() < target ); } + void importantText(const char *text,...){ va_list textArgs; char *printfbuf; @@ -632,11 +610,7 @@ namespace ui { void drawPage( std::string path ) { pageTex = Texture::loadTexture( path ); - std::cout<<"page set\n"; - } - - bool pageExists( void ) { - return pageTex; + pageTexReady = true; } void draw(void){ @@ -644,11 +618,8 @@ namespace ui { float x,y,tmp; std::string rtext; - if ( pageTex ) { - - std::cout<<"page draw\n"; - - glEnable( GL_TEXTURE_2D); + if ( pageTexReady ) { + glEnable( GL_TEXTURE_2D ); glBindTexture( GL_TEXTURE_2D, pageTex ); glBegin( GL_QUADS ); glTexCoord2i( 0, 0 ); glVertex2i( offset.x - 300, SCREEN_HEIGHT - 100 ); @@ -656,9 +627,9 @@ namespace ui { glTexCoord2i( 1, 1 ); glVertex2i( offset.x + 300, SCREEN_HEIGHT - 600 ); glTexCoord2i( 0, 1 ); glVertex2i( offset.x - 300, SCREEN_HEIGHT - 600 ); glEnd(); - glDisable( GL_TEXTURE_2D); + glDisable( GL_TEXTURE_2D ); - } else if (dialogBoxExists){ + } else if (dialogBoxExists) { rtext=typeOut(dialogBoxText); @@ -677,10 +648,11 @@ namespace ui { putStringCentered(offset.x,offset.y,rtext.c_str()); setFontSize(16); } - }else if(dialogMerchant){ + }else if ( dialogMerchant ) { x=offset.x-SCREEN_WIDTH/6; y=(offset.y+SCREEN_HEIGHT/2)-HLINE*8; + // draw the box border glColor3ub(255,255,255); glBegin(GL_LINE_STRIP); glVertex2f(x-1 ,y+1); @@ -690,6 +662,7 @@ namespace ui { glVertex2f(x - 1,y+1); glEnd(); + // draw the box glColor3ub(0,0,0); glRectf(x,y,x+SCREEN_WIDTH/3,y-SCREEN_HEIGHT*.6); @@ -757,12 +730,12 @@ namespace ui { setFontColor(255, 255, 255); // draw option + dialogOptText[i].second.y = y - SCREEN_HEIGHT / 2 - (fontSize + HLINE) * (i + 1); tmp = putStringCentered(offset.x, dialogOptText[i].second.y, dialogOptText[i].first); // get coordinate information on option dialogOptText[i].second.z = offset.x + tmp; dialogOptText[i].second.x = offset.x - tmp; - dialogOptText[i].second.y = y - SCREEN_HEIGHT / 2 - (fontSize + HLINE) * (i + 1); // make text yellow if the mouse hovers over the text if(mouse.x > dialogOptText[i].second.x && mouse.x < dialogOptText[i].second.z && @@ -792,7 +765,7 @@ namespace ui { glColor3ub(0,0,0); glRectf(x,y,x+SCREEN_WIDTH-HLINE*16,y-SCREEN_HEIGHT/4); - rtext=typeOut(dialogBoxText); + rtext = typeOut(dialogBoxText); putString(x+HLINE,y-fontSize-HLINE,rtext); @@ -813,8 +786,11 @@ namespace ui { setFontColor(255,255,255); } - if ( rtext != dialogBoxText ) - Mix_PlayChannel(1,dialogClick,0); + static unsigned int rtext_oldsize = 0; + if ( rtext_oldsize != rtext.size() ) { + if ( !isspace( rtext[(rtext_oldsize = rtext.size()) - 1] ) ) + Mix_PlayChannel( 1, dialogClick, 0 ); + } }if(!fadeIntensity){ vec2 hub = { @@ -877,331 +853,6 @@ namespace ui { config::save(); } - menuItem createButton(vec2 l, dim2 d, Color c, const char* t, menuFunc f){ - menuItem temp; - temp.member = 0; - - temp.button.loc = l; - temp.button.dim = d; - temp.button.color = c; - - temp.button.text = t; - - temp.button.func = f; - - return temp; - } - - menuItem createChildButton(vec2 l, dim2 d, Color c, const char* t){ - menuItem temp; - temp.member = -1; - - temp.button.loc = l; - temp.button.dim = d; - temp.button.color = c; - - temp.button.text = t; - - temp.button.func = NULL; - - return temp; - } - - menuItem createParentButton(vec2 l, dim2 d, Color c, const char* t){ - menuItem temp; - temp.member = -2; - - temp.button.loc = l; - temp.button.dim = d; - temp.button.color = c; - - temp.button.text = t; - - temp.button.func = NULL; - - return temp; - } - - menuItem createSlider(vec2 l, dim2 d, Color c, float min, float max, const char* t, float* v){ - menuItem temp; - temp.member = 1; - - temp.slider.loc = l; - temp.slider.dim = d; - temp.slider.color = c; - temp.slider.minValue = min; - temp.slider.maxValue = max; - - temp.slider.text = t; - - temp.slider.var = v; - - temp.slider.sliderLoc = *v; - - return temp; - } - - /* - * Draws the menu - */ - - void drawMenu(Menu *menu){ - setFontSize(24); - config::update(); - SDL_Event e; - - mouse.x=premouse.x+offset.x-(SCREEN_WIDTH/2); - mouse.y=(offset.y+SCREEN_HEIGHT/2)-premouse.y; - - //custom event polling for menu's so all other events are disregarded - while(SDL_PollEvent(&e)){ - switch(e.type){ - case SDL_QUIT: - gameRunning = false; - return; - break; - case SDL_MOUSEMOTION: - premouse.x=e.motion.x; - premouse.y=e.motion.y; - break; - case SDL_KEYUP: - if(SDL_KEY == SDLK_ESCAPE){ - menu->gotoParent(); - return; - } - break; - default:break; - } - } - - //draw the dark transparent background - glColor4f(0.0f, 0.0f, 0.0f, .8f); - glRectf(offset.x-SCREEN_WIDTH/2,0,offset.x+SCREEN_WIDTH/2,SCREEN_HEIGHT); - - //loop through all elements of the menu - for(auto &m : menu->items){ - //if the menu is any type of button - if(m.member == 0 || m.member == -1 || m.member == -2){ - - //draw the button background - glColor3f(m.button.color.red,m.button.color.green,m.button.color.blue); - glRectf(offset.x+m.button.loc.x, - offset.y+m.button.loc.y, - offset.x+m.button.loc.x + m.button.dim.x, - offset.y+m.button.loc.y + m.button.dim.y); - //draw the button text - putStringCentered(offset.x + m.button.loc.x + (m.button.dim.x/2), - (offset.y + m.button.loc.y + (m.button.dim.y/2)) - ui::fontSize/2, - m.button.text); - - //tests if the mouse is over the button - if(mouse.x >= offset.x+m.button.loc.x && mouse.x <= offset.x+m.button.loc.x + m.button.dim.x){ - if(mouse.y >= offset.y+m.button.loc.y && mouse.y <= offset.y+m.button.loc.y + m.button.dim.y){ - - //if the mouse if over the button, it draws this white outline - glColor3f(1.0f,1.0f,1.0f); - glBegin(GL_LINE_STRIP); - glVertex2f(offset.x+m.button.loc.x, offset.y+m.button.loc.y); - glVertex2f(offset.x+m.button.loc.x+m.button.dim.x, offset.y+m.button.loc.y); - glVertex2f(offset.x+m.button.loc.x+m.button.dim.x, offset.y+m.button.loc.y+m.button.dim.y); - glVertex2f(offset.x+m.button.loc.x, offset.y+m.button.loc.y+m.button.dim.y); - glVertex2f(offset.x+m.button.loc.x, offset.y+m.button.loc.y); - glEnd(); - - //if the mouse is over the button and clicks - if(SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_LEFT)){ - switch(m.member){ - case 0: //normal button - m.button.func(); - break; - case -1: - menu->gotoChild(); //goto child menu - break; - case -2: - menu->gotoParent(); //goto parent menu - default:break; - } - } - } - } - - //if element is a slider - }else if(m.member == 1){ - //combining slider text with variable amount - char outSV[32]; - sprintf(outSV, "%s: %.1f",m.slider.text, *m.slider.var); - - float sliderW, sliderH; - - if(m.slider.dim.y > m.slider.dim.x){ - //width of the slider handle - sliderW = m.slider.dim.x; - sliderH = m.slider.dim.y * .05; - //location of the slider handle - m.slider.sliderLoc = m.slider.minValue + (*m.slider.var/m.slider.maxValue)*(m.slider.dim.y-sliderW); - }else{ - //width of the slider handle - sliderW = m.slider.dim.x * .05; - sliderH = m.slider.dim.y; - //location of the slider handle - m.slider.sliderLoc = m.slider.minValue + (*m.slider.var/m.slider.maxValue)*(m.slider.dim.x-sliderW); - } - //draw the background of the slider - glColor4f(m.slider.color.red,m.slider.color.green,m.slider.color.blue, .5f); - glRectf(offset.x+m.slider.loc.x, - offset.y+m.slider.loc.y, - offset.x+m.slider.loc.x + m.slider.dim.x, - offset.y+m.slider.loc.y + m.slider.dim.y); - - //draw the slider handle - glColor4f(m.slider.color.red,m.slider.color.green,m.slider.color.blue, 1.0f); - if(m.slider.dim.y > m.slider.dim.x){ - glRectf(offset.x+m.slider.loc.x, - offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05), - offset.x+m.slider.loc.x + sliderW, - offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); - - //draw the now combined slider text - putStringCentered(offset.x + m.slider.loc.x + (m.slider.dim.x/2), (offset.y + m.slider.loc.y + (m.slider.dim.y*1.05)) - ui::fontSize/2, outSV); - }else{ - glRectf(offset.x+m.slider.loc.x+m.slider.sliderLoc, - offset.y+m.slider.loc.y, - offset.x+m.slider.loc.x + m.slider.sliderLoc + sliderW, - offset.y+m.slider.loc.y + sliderH); - - //draw the now combined slider text - putStringCentered(offset.x + m.slider.loc.x + (m.slider.dim.x/2), (offset.y + m.slider.loc.y + (m.slider.dim.y/2)) - ui::fontSize/2, outSV); - } - //test if mouse is inside of the slider's borders - if(mouse.x >= offset.x+m.slider.loc.x && mouse.x <= offset.x+m.slider.loc.x + m.slider.dim.x){ - if(mouse.y >= offset.y+m.slider.loc.y && mouse.y <= offset.y+m.slider.loc.y + m.slider.dim.y){ - - //if it is we draw a white border around it - glColor3f(1.0f,1.0f,1.0f); - glBegin(GL_LINE_STRIP); - glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y); - glVertex2f(offset.x+m.slider.loc.x+m.slider.dim.x, offset.y+m.slider.loc.y); - glVertex2f(offset.x+m.slider.loc.x+m.slider.dim.x, offset.y+m.slider.loc.y+m.slider.dim.y); - glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y+m.slider.dim.y); - glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y); - - if(m.slider.dim.y > m.slider.dim.x){ - //and a border around the slider handle - glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)); - glVertex2f(offset.x+m.slider.loc.x + sliderW, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)); - glVertex2f(offset.x+m.slider.loc.x + sliderW, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); - glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); - glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)); - }else{ - //and a border around the slider handle - glVertex2f(offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y); - glVertex2f(offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), offset.y+m.slider.loc.y); - glVertex2f(offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), offset.y+m.slider.loc.y+m.slider.dim.y); - glVertex2f(offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y+m.slider.dim.y); - glVertex2f(offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y); - } - - glEnd(); - - //if we are inside the slider and click it will set the slider to that point - if(SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_LEFT)){ - //change handle location - if(m.slider.dim.y > m.slider.dim.x){ - *m.slider.var = (((mouse.y-offset.y) - m.slider.loc.y)/m.slider.dim.y)*100; - //draw a white box over the handle - glColor3f(1.0f,1.0f,1.0f); - glRectf(offset.x+m.slider.loc.x, - offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05), - offset.x+m.slider.loc.x + sliderW, - offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); - - }else{ - *m.slider.var = (((mouse.x-offset.x) - m.slider.loc.x)/m.slider.dim.x)*100; - //draw a white box over the handle - glColor3f(1.0f,1.0f,1.0f); - glRectf(offset.x+m.slider.loc.x + m.slider.sliderLoc, - offset.y+m.slider.loc.y, - offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), - offset.y+m.slider.loc.y + m.slider.dim.y); - } - } - - //makes sure handle can't go below or above min and max values - if(*m.slider.var >= m.slider.maxValue)*m.slider.var = m.slider.maxValue; - else if(*m.slider.var <= m.slider.minValue)*m.slider.var = m.slider.minValue; - } - } - } - } - setFontSize(16); - } - - void takeScreenshot(GLubyte* pixels){ - std::vector<GLubyte> bgr (SCREEN_WIDTH * SCREEN_HEIGHT * 3, 0); - - for(uint x = 0; x < SCREEN_WIDTH*SCREEN_HEIGHT*3; x+=3){ - bgr[x] = pixels[x+2]; - bgr[x+1] = pixels[x+1]; - bgr[x+2] = pixels[x]; - } - - time_t epoch = time(NULL); - struct tm* timen = localtime(&epoch); - - std::string name = "screenshots/"; - name += std::to_string(1900 + timen->tm_year) += "-"; - name += std::to_string(timen->tm_mon + 1) += "-"; - name += std::to_string(timen->tm_mday) += "_"; - name += std::to_string(timen->tm_hour) += "-"; - name += std::to_string(timen->tm_min) += "-"; - name += std::to_string(timen->tm_sec); - name += ".bmp"; - FILE* bmp = fopen(name.c_str(), "w+"); - - // unsigned long header_size = sizeof(BITMAPFILEHEADER) + - // sizeof(BITMAPINFOHEADER); - - BITMAPFILEHEADER bmfh; - BITMAPINFOHEADER bmih; - - memset(&bmfh, 0, sizeof(BITMAPFILEHEADER)); - memset(&bmih, 0, sizeof(BITMAPINFOHEADER)); - - bmfh.bfType = 0x4d42; - - bmfh.bfOffBits = 54; - bmfh.bfSize = sizeof(BITMAPFILEHEADER) + - sizeof(BITMAPINFOHEADER); - bmfh.bfReserved1 = 0; - bmfh.bfReserved2 = 0; - - - bmih.biSize = sizeof(BITMAPINFOHEADER); - bmih.biBitCount = 24; - - bmih.biClrImportant = 0; - bmih.biClrUsed = 0; - - bmih.biCompression = 0; - - bmih.biWidth = SCREEN_WIDTH; - bmih.biHeight = SCREEN_HEIGHT; - - bmih.biPlanes = 1; - bmih.biSizeImage = 0; - - bmih.biXPelsPerMeter = 0x0ec4; - bmih.biYPelsPerMeter = 0x0ec4; - - fwrite(&bmfh, 1,sizeof(BITMAPFILEHEADER),bmp); - fwrite(&bmih, 1,sizeof(BITMAPINFOHEADER),bmp); - fwrite(&bgr, 1,3*SCREEN_WIDTH*SCREEN_HEIGHT,bmp); - - delete[] pixels; - - fclose(bmp); - } - void closeBox(){ dialogBoxExists = false; dialogMerchant = false; @@ -1211,16 +862,16 @@ namespace ui { unsigned char i; if ( pageTex ) { - std::cout<<"rip page\n"; glDeleteTextures( 1, &pageTex ); pageTex = 0; + pageTexReady = false; return; } - if(!typeOutDone){ + /*if(!typeOutDone){ typeOutDone = true; return; - } + }*/ for(i=0;i<dialogOptText.size();i++){ if(mouse.x > dialogOptText[i].second.x && @@ -1231,10 +882,14 @@ namespace ui { goto EXIT; } } - if(dialogMerchant){ - for(i=0;i<2;i++){ + + if ( dialogMerchant ) { + for ( i = 0; i < merchArrowLoc.size(); i++ ) { + + // TODO neaten this if statement + if(((merchArrowLoc[i].x < merchArrowLoc[i].z) ? - (mouse.x > merchArrowLoc[i].x && mouse.x < merchArrowLoc[i].z) : + (mouse.x > merchArrowLoc[i].x && mouse.x < merchArrowLoc[i].z) : (mouse.x < merchArrowLoc[i].x && mouse.x > merchArrowLoc[i].z) && mouse.y > merchArrowLoc[i].y - 8 && mouse.y < merchArrowLoc[i].y + 8)){ merchOptChosen = i + 1; @@ -1288,14 +943,10 @@ EXIT: break; case SDL_MOUSEBUTTONUP: - - // right click advances dialog - if ( ( e.button.button & SDL_BUTTON_RIGHT ) && (dialogBoxExists | pageTex) ) - dialogAdvance(); - if ( ig ) { ig->vel.x = (fr.x - mouse.x) / 50.0f; ig->vel.y = (fr.y - mouse.y) / 50.0f; + //ig->forcedMove = true; // kills vel.x too quickly ig = NULL; } break; @@ -1303,6 +954,10 @@ EXIT: // mouse clicks case SDL_MOUSEBUTTONDOWN: + // right click advances dialog + if ( ( e.button.button & SDL_BUTTON_RIGHT ) && (dialogBoxExists | pageTexReady) ) + dialogAdvance(); + // left click uses item if ( ( e.button.button & SDL_BUTTON_LEFT ) && !dialogBoxExists ) player->inv->usingi = true; @@ -1405,8 +1060,8 @@ EXIT: currentWorld = ((Arena *)currentWorld)->exitArena( player ); if ( tmp != currentWorld ) toggleBlackFast(); - } else if( (tmp = currentWorld->goInsideStructure( player )) != currentWorld ) - currentWorld = tmp; + } else if ( (tmp = currentWorld->goInsideStructure( player )) != currentWorld ) + currentWorld = tmp; break; case SDLK_LSHIFT: if(debug){ @@ -1454,7 +1109,7 @@ EXIT: debug ^= true; break; case SDLK_z: - weather = WorldWeather::Snowy; + weather = WorldWeather::Rain; break; case SDLK_i: if ( isCurrentWorldIndoors() && Indoorp(currentWorld)->isFloorAbove( player ) ) { @@ -1517,9 +1172,6 @@ EXIT: static GLubyte* pixels; pixels = new GLubyte[ 3 * SCREEN_WIDTH * SCREEN_HEIGHT]; glReadPixels(0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, GL_RGB, GL_UNSIGNED_BYTE, pixels); - //static std::thread scr; - //scr = std::thread(takeScreenshot,pixels); - //scr.detach(); takeScreenshot(pixels); std::cout << "Took screenshot" << std::endl; @@ -1543,7 +1195,7 @@ EXIT: } } - // Flush preloaded AI functions if necessary + // Flush preloaded AI functions if necessary if ( !dialogBoxExists && AIpreaddr.size() ) { while ( !AIpreaddr.empty() ) { AIpreaddr.front()->addAIFunc( AIpreload.front(), false ); @@ -1575,4 +1227,70 @@ EXIT: Mix_PlayChannel( 1, battleStart, 0 ); } + + void takeScreenshot(GLubyte* pixels){ + std::vector<GLubyte> bgr (SCREEN_WIDTH * SCREEN_HEIGHT * 3, 0); + + for(uint x = 0; x < SCREEN_WIDTH*SCREEN_HEIGHT*3; x+=3){ + bgr[x] = pixels[x+2]; + bgr[x+1] = pixels[x+1]; + bgr[x+2] = pixels[x]; + } + + time_t epoch = time(NULL); + struct tm* timen = localtime(&epoch); + + std::string name = "screenshots/"; + name += std::to_string(1900 + timen->tm_year) += "-"; + name += std::to_string(timen->tm_mon + 1) += "-"; + name += std::to_string(timen->tm_mday) += "_"; + name += std::to_string(timen->tm_hour) += "-"; + name += std::to_string(timen->tm_min) += "-"; + name += std::to_string(timen->tm_sec); + name += ".bmp"; + FILE* bmp = fopen(name.c_str(), "w+"); + + // unsigned long header_size = sizeof(BITMAPFILEHEADER) + + // sizeof(BITMAPINFOHEADER); + + BITMAPFILEHEADER bmfh; + BITMAPINFOHEADER bmih; + + memset(&bmfh, 0, sizeof(BITMAPFILEHEADER)); + memset(&bmih, 0, sizeof(BITMAPINFOHEADER)); + + bmfh.bfType = 0x4d42; + + bmfh.bfOffBits = 54; + bmfh.bfSize = sizeof(BITMAPFILEHEADER) + + sizeof(BITMAPINFOHEADER); + bmfh.bfReserved1 = 0; + bmfh.bfReserved2 = 0; + + + bmih.biSize = sizeof(BITMAPINFOHEADER); + bmih.biBitCount = 24; + + bmih.biClrImportant = 0; + bmih.biClrUsed = 0; + + bmih.biCompression = 0; + + bmih.biWidth = SCREEN_WIDTH; + bmih.biHeight = SCREEN_HEIGHT; + + bmih.biPlanes = 1; + bmih.biSizeImage = 0; + + bmih.biXPelsPerMeter = 0x0ec4; + bmih.biYPelsPerMeter = 0x0ec4; + + fwrite(&bmfh, 1,sizeof(BITMAPFILEHEADER),bmp); + fwrite(&bmih, 1,sizeof(BITMAPINFOHEADER),bmp); + fwrite(&bgr, 1,3*SCREEN_WIDTH*SCREEN_HEIGHT,bmp); + + delete[] pixels; + + fclose(bmp); + } } diff --git a/src/ui_menu.cpp b/src/ui_menu.cpp new file mode 100644 index 0000000..775ea81 --- /dev/null +++ b/src/ui_menu.cpp @@ -0,0 +1,275 @@ +#include <ui_menu.hpp> + +extern bool gameRunning; + +extern Menu *currentMenu; +extern Menu pauseMenu; + +void Menu:: +gotoParent( void ) +{ + if ( !parent ) { + currentMenu = NULL; + config::update(); + } else + currentMenu = parent; +} + +void Menu:: +gotoChild( void ) +{ + currentMenu = child; +} + +namespace ui { + namespace menu { + menuItem createButton(vec2 l, dim2 d, Color c, const char* t, menuFunc f){ + menuItem temp; + + temp.member = 0; + temp.button.loc = l; + temp.button.dim = d; + temp.button.color = c; + temp.button.text = t; + temp.button.func = f; + + return temp; + } + + menuItem createChildButton(vec2 l, dim2 d, Color c, const char* t){ + menuItem temp; + + temp.member = -1; + temp.button.loc = l; + temp.button.dim = d; + temp.button.color = c; + temp.button.text = t; + temp.button.func = NULL; + + return temp; + } + + menuItem createParentButton(vec2 l, dim2 d, Color c, const char* t){ + menuItem temp; + + temp.member = -2; + temp.button.loc = l; + temp.button.dim = d; + temp.button.color = c; + temp.button.text = t; + temp.button.func = NULL; + + return temp; + } + + menuItem createSlider(vec2 l, dim2 d, Color c, float min, float max, const char* t, float* v){ + menuItem temp; + + temp.member = 1; + temp.slider.loc = l; + temp.slider.dim = d; + temp.slider.color = c; + temp.slider.minValue = min; + temp.slider.maxValue = max; + temp.slider.text = t; + temp.slider.var = v; + temp.slider.sliderLoc = *v; + + return temp; + } + + void draw( void ) { + SDL_Event e; + + setFontSize(24); + config::update(); + + mouse.x = ui::premouse.x+offset.x-(SCREEN_WIDTH/2); + mouse.y = (offset.y+SCREEN_HEIGHT/2)-ui::premouse.y; + + //custom event polling for menu's so all other events are ignored + while(SDL_PollEvent(&e)){ + switch(e.type){ + case SDL_QUIT: + gameRunning = false; + return; + break; + case SDL_MOUSEMOTION: + premouse.x=e.motion.x; + premouse.y=e.motion.y; + break; + case SDL_KEYUP: + if(SDL_KEY == SDLK_ESCAPE){ + currentMenu->gotoParent(); + return; + } + break; + default:break; + } + } + + //draw the dark transparent background + glColor4f(0.0f, 0.0f, 0.0f, .8f); + glRectf(offset.x-SCREEN_WIDTH/2,0,offset.x+SCREEN_WIDTH/2,SCREEN_HEIGHT); + + //loop through all elements of the menu + for(auto &m : currentMenu->items){ + //if the menu is any type of button + if(m.member == 0 || m.member == -1 || m.member == -2){ + + //draw the button background + glColor3f(m.button.color.red,m.button.color.green,m.button.color.blue); + glRectf(offset.x+m.button.loc.x, + offset.y+m.button.loc.y, + offset.x+m.button.loc.x + m.button.dim.x, + offset.y+m.button.loc.y + m.button.dim.y); + //draw the button text + putStringCentered(offset.x + m.button.loc.x + (m.button.dim.x/2), + (offset.y + m.button.loc.y + (m.button.dim.y/2)) - ui::fontSize/2, + m.button.text); + + //tests if the mouse is over the button + if(mouse.x >= offset.x+m.button.loc.x && mouse.x <= offset.x+m.button.loc.x + m.button.dim.x){ + if(mouse.y >= offset.y+m.button.loc.y && mouse.y <= offset.y+m.button.loc.y + m.button.dim.y){ + + //if the mouse if over the button, it draws this white outline + glColor3f(1.0f,1.0f,1.0f); + glBegin(GL_LINE_STRIP); + glVertex2f(offset.x+m.button.loc.x, offset.y+m.button.loc.y); + glVertex2f(offset.x+m.button.loc.x+m.button.dim.x, offset.y+m.button.loc.y); + glVertex2f(offset.x+m.button.loc.x+m.button.dim.x, offset.y+m.button.loc.y+m.button.dim.y); + glVertex2f(offset.x+m.button.loc.x, offset.y+m.button.loc.y+m.button.dim.y); + glVertex2f(offset.x+m.button.loc.x, offset.y+m.button.loc.y); + glEnd(); + + //if the mouse is over the button and clicks + if(SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_LEFT)){ + switch(m.member){ + case 0: //normal button + m.button.func(); + break; + case -1: + currentMenu->gotoChild(); + break; + case -2: + currentMenu->gotoParent(); + default:break; + } + } + } + } + + //if element is a slider + }else if(m.member == 1){ + //combining slider text with variable amount + char outSV[32]; + sprintf(outSV, "%s: %.1f",m.slider.text, *m.slider.var); + + float sliderW, sliderH; + + if(m.slider.dim.y > m.slider.dim.x){ + //width of the slider handle + sliderW = m.slider.dim.x; + sliderH = m.slider.dim.y * .05; + //location of the slider handle + m.slider.sliderLoc = m.slider.minValue + (*m.slider.var/m.slider.maxValue)*(m.slider.dim.y-sliderW); + }else{ + //width of the slider handle + sliderW = m.slider.dim.x * .05; + sliderH = m.slider.dim.y; + //location of the slider handle + m.slider.sliderLoc = m.slider.minValue + (*m.slider.var/m.slider.maxValue)*(m.slider.dim.x-sliderW); + } + //draw the background of the slider + glColor4f(m.slider.color.red,m.slider.color.green,m.slider.color.blue, .5f); + glRectf(offset.x+m.slider.loc.x, + offset.y+m.slider.loc.y, + offset.x+m.slider.loc.x + m.slider.dim.x, + offset.y+m.slider.loc.y + m.slider.dim.y); + + //draw the slider handle + glColor4f(m.slider.color.red,m.slider.color.green,m.slider.color.blue, 1.0f); + if(m.slider.dim.y > m.slider.dim.x){ + glRectf(offset.x+m.slider.loc.x, + offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05), + offset.x+m.slider.loc.x + sliderW, + offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); + + //draw the now combined slider text + putStringCentered(offset.x + m.slider.loc.x + (m.slider.dim.x/2), (offset.y + m.slider.loc.y + (m.slider.dim.y*1.05)) - ui::fontSize/2, outSV); + }else{ + glRectf(offset.x+m.slider.loc.x+m.slider.sliderLoc, + offset.y+m.slider.loc.y, + offset.x+m.slider.loc.x + m.slider.sliderLoc + sliderW, + offset.y+m.slider.loc.y + sliderH); + + //draw the now combined slider text + putStringCentered(offset.x + m.slider.loc.x + (m.slider.dim.x/2), (offset.y + m.slider.loc.y + (m.slider.dim.y/2)) - ui::fontSize/2, outSV); + } + //test if mouse is inside of the slider's borders + if(mouse.x >= offset.x+m.slider.loc.x && mouse.x <= offset.x+m.slider.loc.x + m.slider.dim.x){ + if(mouse.y >= offset.y+m.slider.loc.y && mouse.y <= offset.y+m.slider.loc.y + m.slider.dim.y){ + + //if it is we draw a white border around it + glColor3f(1.0f,1.0f,1.0f); + glBegin(GL_LINE_STRIP); + glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y); + glVertex2f(offset.x+m.slider.loc.x+m.slider.dim.x, offset.y+m.slider.loc.y); + glVertex2f(offset.x+m.slider.loc.x+m.slider.dim.x, offset.y+m.slider.loc.y+m.slider.dim.y); + glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y+m.slider.dim.y); + glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y); + + if(m.slider.dim.y > m.slider.dim.x){ + //and a border around the slider handle + glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)); + glVertex2f(offset.x+m.slider.loc.x + sliderW, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)); + glVertex2f(offset.x+m.slider.loc.x + sliderW, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); + glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); + glVertex2f(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)); + }else{ + //and a border around the slider handle + glVertex2f(offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y); + glVertex2f(offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), offset.y+m.slider.loc.y); + glVertex2f(offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), offset.y+m.slider.loc.y+m.slider.dim.y); + glVertex2f(offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y+m.slider.dim.y); + glVertex2f(offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y); + } + + glEnd(); + + //if we are inside the slider and click it will set the slider to that point + if(SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_LEFT)){ + //change handle location + if(m.slider.dim.y > m.slider.dim.x){ + *m.slider.var = (((mouse.y-offset.y) - m.slider.loc.y)/m.slider.dim.y)*100; + //draw a white box over the handle + glColor3f(1.0f,1.0f,1.0f); + glRectf(offset.x+m.slider.loc.x, + offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05), + offset.x+m.slider.loc.x + sliderW, + offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH); + + }else{ + *m.slider.var = (((mouse.x-offset.x) - m.slider.loc.x)/m.slider.dim.x)*100; + //draw a white box over the handle + glColor3f(1.0f,1.0f,1.0f); + glRectf(offset.x+m.slider.loc.x + m.slider.sliderLoc, + offset.y+m.slider.loc.y, + offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), + offset.y+m.slider.loc.y + m.slider.dim.y); + } + } + + //makes sure handle can't go below or above min and max values + if(*m.slider.var >= m.slider.maxValue)*m.slider.var = m.slider.maxValue; + else if(*m.slider.var <= m.slider.minValue)*m.slider.var = m.slider.minValue; + } + } + } + } + setFontSize(16); + } + + + } +} diff --git a/src/world.cpp b/src/world.cpp index ffddcab..be8e682 100644 --- a/src/world.cpp +++ b/src/world.cpp @@ -20,12 +20,6 @@ using namespace tinyxml2; #define INDOOR_FLOOR_THICKNESS 50 #define INDOOR_FLOOR_HEIGHTT 400 -/** - * Gravity thing - */ - -#define GRAVITY_CONSTANT 0.001f - extern Player *player; // main.cpp? extern World *currentWorld; // main.cpp extern World *currentWorldToLeft; // main.cpp @@ -265,8 +259,8 @@ generate( unsigned int width ) // create slopes from the points that were just defined, populate the rest of the WorldData structure - for(wditer = worldData.begin(); wditer != worldData.end(); wditer++){ - if ((*wditer).groundHeight) + for(wditer = worldData.begin() + 1; wditer != worldData.end(); wditer++){ + if ( (*wditer).groundHeight && wditer + GROUND_HILLINESS < worldData.end() ) // wditer + GROUND_HILLINESS can go out of bounds (invalid read) geninc = ( (*(wditer + GROUND_HILLINESS)).groundHeight - (*wditer).groundHeight ) / (float)GROUND_HILLINESS; else @@ -335,8 +329,8 @@ update( Player *p, unsigned int delta ) particles.erase( std::remove_if( particles.begin(), particles.end(), [&delta](Particles &part){return part.kill(delta);}), particles.end()); for ( auto part = particles.begin(); part != particles.end(); part++ ) { if ( (*part).canMove ) { - (*part).loc.y += (*part).vely * delta; - (*part).loc.x += (*part).velx * delta; + (*part).loc.y += (*part).vel.y * delta; + (*part).loc.x += (*part).vel.x * delta; for ( auto &b : build ) { if ( b->bsubtype == FOUNTAIN ) { @@ -429,7 +423,19 @@ void World::draw(Player *p){ // the sunny wallpaper is faded with the night depending on tickCount bgTex->bind( 0 ); - safeSetColorA( 255, 255, 255, weather == WorldWeather::Snowy ? 150 : 255 - worldShade * 4); + int alpha; + switch( weather ) { + case WorldWeather::Snowy: + alpha = 150; + break; + case WorldWeather::Rain: + alpha = 0; + break; + default: + alpha = 255 - worldShade * 4; + break; + } + safeSetColorA( 255, 255, 255, alpha ); glBegin( GL_QUADS ); glTexCoord2i( 0, 0 ); glVertex2i( offset.x - SCREEN_WIDTH/2-5, offset.y + SCREEN_HEIGHT/2 ); @@ -439,7 +445,7 @@ void World::draw(Player *p){ glEnd(); bgTex->bindNext(); - safeSetColorA( 255, 255, 255, worldShade * 4); + safeSetColorA( 255, 255, 255, !alpha ? 255 : worldShade * 4); glBegin( GL_QUADS ); glTexCoord2i( 0, 0 ); glVertex2i( offset.x - SCREEN_WIDTH/2-5, offset.y + SCREEN_HEIGHT/2 ); @@ -532,8 +538,17 @@ void World::draw(Player *p){ glUseProgram(0); - for ( auto &b : build ) - b->draw(); + for ( auto &b : build ) { + if ( b->bsubtype == STALL_MARKET ) { + for ( auto &n : npc ) { + if ( n->type == MERCHT && ((Merchant *)n)->inside == b ) { + n->draw(); + break; + } + } + } + b->draw(); + } // draw light elements? @@ -570,9 +585,8 @@ void World::draw(Player *p){ } } - for(uint i = 0; i < light.size(); i++){ + for ( uint i = 0; i < light.size(); i++ ) flameArray[i] = light[i].fireFlicker; - } glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT ); @@ -685,8 +699,10 @@ void World::draw(Player *p){ glUseProgram(0); - for ( auto &n : npc ) - n->draw(); + for ( auto &n : npc ) { + if ( n->type != MERCHT ) + n->draw(); + } for ( auto &m : mob ) m->draw(); @@ -818,7 +834,7 @@ singleDetect( Entity *e ) // if the entity is under the world/line, pop it back to the surface if ( e->loc.y < worldData[i].groundHeight ) { int dir = e->vel.x < 0 ? -1 : 1; - if ( worldData[i + (dir * 8)].groundHeight - 30 > worldData[i + dir].groundHeight ) { + if ( i + (dir * 2) < worldData.size() && worldData[i + (dir * 2)].groundHeight - 30 > worldData[i + dir].groundHeight ) { e->loc.x -= ( PLAYER_SPEED_CONSTANT + 2.7 ) * e->speed * 2 * dir; e->vel.x = 0; } else { @@ -869,11 +885,11 @@ detect( Player *p ) int l; // handle the player - std::thread( &World::singleDetect, this, p).detach(); + std::thread( &World::singleDetect, this, p ).detach(); // handle other entities for ( auto &e : entity ) - std::thread(&World::singleDetect,this,e).detach(); + std::thread( &World::singleDetect, this, e ).detach(); // handle particles for ( auto &part : particles ) { @@ -887,14 +903,7 @@ detect( Player *p ) if ( l > (int)(lineCount - 1) ) l = lineCount - 1; - // handle ground collision - if ( part.loc.y < worldData[l].groundHeight ) { - part.loc.y = worldData[l].groundHeight; - part.vely = 0; - part.velx = 0; - part.canMove = false; - } else if ( part.gravity && part.vely > -2 ) - part.vely -= GRAVITY_CONSTANT * deltaTime; + part.update( GRAVITY_CONSTANT, worldData[l].groundHeight ); } // handle particle creation @@ -1143,14 +1152,17 @@ goInsideStructure( Player *p ) void World:: addHole( unsigned int start, unsigned int end ) { - for ( unsigned int i = end; i-- > start; ) + if ( end > worldData.size() ) + end = worldData.size(); + + for ( unsigned int i = start; i < end; i++ ) worldData[i].groundHeight = 0; } void World:: addHill( const ivec2 peak, const unsigned int width ) { - int start = peak.x - width / 2, end = start + width, offset; + int start = peak.x - width / 2, end = start + width, offset = 0; const float thing = peak.y - worldData[start].groundHeight; const float period = PI / width; @@ -1500,8 +1512,9 @@ Arena::~Arena(void){ World *Arena::exitArena(Player *p){ World *tmp; - if(p->loc.x + p->width / 2 > mob[0]->loc.x && - p->loc.x + p->width / 2 < mob[0]->loc.x + HLINE * 12 ){ + if ( !mmob->alive && + p->loc.x + p->width / 2 > mob[0]->loc.x && + p->loc.x + p->width / 2 < mob[0]->loc.x + HLINE * 12 ) { tmp = battleNest.front(); battleNest.erase(battleNest.begin()); @@ -1640,6 +1653,10 @@ loadWorldFromXMLNoSave( std::string path ) { } } + // tells what world is outside, if in a structure + else if ( Indoor && (ptr = wxml->Attribute("outside")) ) + inside.push_back( ptr ); + // error, invalid link tag else UserError("XML Error: Invalid <link> tag in " + currentXML + "!"); @@ -1692,6 +1709,10 @@ loadWorldFromXMLNoSave( std::string path ) { // indoor spawning floor selection if ( Indoor && wxml->QueryUnsignedAttribute( "floor", &flooor ) == XML_NO_ERROR ) Indoorp(tmp)->moveToFloor( tmp->npc.back(), flooor ); + + // custom health value + if ( wxml->QueryFloatAttribute( "health", &spawnx) == XML_NO_ERROR ) + tmp->mob.back()->health = tmp->mob.back()->maxHealth = spawnx; } // npc creation @@ -1700,7 +1721,7 @@ loadWorldFromXMLNoSave( std::string path ) { // spawn at coordinates if desired if ( wxml->QueryFloatAttribute( "x", &spawnx ) == XML_NO_ERROR) - tmp->addNPC( spawnx, wxml->FloatAttribute("y") ); + tmp->addNPC( spawnx, 100 ); else tmp->addNPC( 0, 100 ); @@ -1720,6 +1741,10 @@ loadWorldFromXMLNoSave( std::string path ) { if ( Indoor && wxml->QueryUnsignedAttribute( "floor", &flooor ) == XML_NO_ERROR ) Indoorp(tmp)->moveToFloor( tmp->npc.back(), flooor ); + + // custom health value + if ( wxml->QueryFloatAttribute( "health", &spawnx) == XML_NO_ERROR ) + tmp->mob.back()->health = tmp->mob.back()->maxHealth = spawnx; } // structure creation @@ -1732,7 +1757,7 @@ loadWorldFromXMLNoSave( std::string path ) { wxml->StrAttribute("inside") ); } else if ( name == "trigger" ) { - tmp->addMob(MS_TRIGGER,wxml->FloatAttribute("x"),0,commonTriggerFunc); + tmp->addMob( MS_TRIGGER, wxml->FloatAttribute("x"), 0, commonTriggerFunc ); tmp->mob.back()->heyid = wxml->Attribute("id"); } else if ( name == "page" ) { tmp->addMob( MS_PAGE, wxml->FloatAttribute("x"), 0, commonPageFunc ); @@ -1748,6 +1773,7 @@ loadWorldFromXMLNoSave( std::string path ) { Indoorp(tmp)->addFloor( wxml->UnsignedAttribute("width") ); } + spawnx = 0; wxml = wxml->NextSiblingElement(); } @@ -1775,34 +1801,53 @@ loadWorldFromXMLNoSave( std::string path ) { 100, vil->StrAttribute("texture"), vil->StrAttribute("inside")); - }else if ( name == "stall" ) { - if(!strcmp(vil->Attribute("type"),"market")){ - tmp->addStructure((BUILD_SUB)70, - vil->QueryFloatAttribute("x", &spawnx) != XML_NO_ERROR ? - randx : spawnx, - 100, - vil->StrAttribute("texture"), - vil->StrAttribute("inside")); - tmp->addMerchant(0,100); + } else if ( name == "stall" ) { + sptr = vil->StrAttribute("type"); + + // handle markets + if ( sptr == "market" ) { + + // create a structure and a merchant, and pair them + tmp->addStructure( STALL_MARKET, + vil->QueryFloatAttribute("x", &spawnx) != XML_NO_ERROR ? randx : spawnx, + 100, + vil->StrAttribute("texture"), + vil->StrAttribute("inside") + ); + tmp->addMerchant( 0, 100 ); + tmp->merchant.back()->inside = tmp->build.back(); - if(vil->FirstChildElement("buy")){ - }else if(vil->FirstChildElement("sell")){ - }else if(vil->FirstChildElement("trade")){ - tmp->merchant.back()->trade.push_back(Trade(vil->FirstChildElement("trade")->IntAttribute("quantity"), - vil->FirstChildElement("trade")->Attribute("item"), - vil->FirstChildElement("trade")->IntAttribute("quantity1"), - vil->FirstChildElement("trade")->Attribute("item1"))); - tmp->merchant.back()->trade.push_back(Trade(1,"Wood Sword", 420, "Dank MayMay")); + } + + // handle traders + else if ( sptr == "trader") { + tmp->addStructure( STALL_TRADER, + vil->QueryFloatAttribute("x", &spawnx) != XML_NO_ERROR ? randx : spawnx, + 100, + vil->StrAttribute("texture"), + vil->StrAttribute("inside") + ); + } + + // loop through buy/sell/trade tags + XMLElement *sxml = vil->FirstChildElement(); + std::string tag; + while ( sxml ) { + tag = sxml->Name(); + + if ( tag == "buy" ) { + // TODO + } else if ( tag == "sell" ) { + // TODO + } else if ( tag == "trade" ) { + tmp->merchant.back()->trade.push_back( Trade( sxml->IntAttribute("quantity"), + sxml->StrAttribute("item"), + sxml->IntAttribute("quantity1"), + sxml->StrAttribute("item1") + )); } - strcpy(tmp->merchant.back()->name,"meme"); - }else if(!strcmp(vil->Attribute("type"),"trader")){ - tmp->addStructure((BUILD_SUB)71, - vil->QueryFloatAttribute("x", &spawnx) != XML_NO_ERROR ? - randx : spawnx, - 100, - vil->StrAttribute("texture"), - vil->StrAttribute("inside")); + sxml = sxml->NextSiblingElement(); } } diff --git a/storyXML/0000_SpawnHill.xml.dat b/storyXML/0000_SpawnHill.xml.dat deleted file mode 100644 index 6508abc..0000000 --- a/storyXML/0000_SpawnHill.xml.dat +++ /dev/null @@ -1,3 +0,0 @@ --300 -72 -dOnE
\ No newline at end of file diff --git a/test.frag b/test.frag deleted file mode 100644 index 07b4a8a..0000000 --- a/test.frag +++ /dev/null @@ -1,39 +0,0 @@ -#version 120
-uniform sampler2D sampler;
-
-uniform int numLight;
-uniform vec2 lightLocation[64];
-uniform vec3 lightColor;
-uniform float amb;
-
-float b = .0005;
-float minLight = .05;
-float radius = sqrt(1.0 / (b * minLight));
-
-//float radius = b*minlight;
-
-void main(){
- vec4 color = vec4(0.0,0.0,0.0,0.0);
- for(int i = 0; i < numLight; i++){
- vec2 loc = lightLocation[i];
- float dist = length(loc - gl_FragCoord.xy);
- //float attenuation=1.0/(1.0+0.01*dist+0.00000000001*dist*dist);
- float attenuation = clamp(1.0 - dist*dist/(radius*radius), 0.0, 1.0); attenuation *= attenuation;
-
- color += vec4(attenuation, attenuation, attenuation, 1.0) * vec4(lightColor, 1.0);
- }
- vec2 coords = gl_TexCoord[0].st;
- vec4 tex = texture2D(sampler, coords);
-
- color += vec4(amb,amb,amb,1.0+amb);
- gl_FragColor = tex * vec4(color)*tex.a;
-}
-
-/* b values
- .002 10
- .008 50
- .0005 200
- .00008 500
- .00002 1000
- .00005 2000
-*/
\ No newline at end of file diff --git a/xml/playerSpawnHill1.xml b/xml/playerSpawnHill1.xml index 936731e..53f5ad2 100644 --- a/xml/playerSpawnHill1.xml +++ b/xml/playerSpawnHill1.xml @@ -7,15 +7,15 @@ <hill peakx="0" peaky="1000" width="50" /> - <mob x="300" type="1" aggressive="false" /> + <mob x="300" type="1" aggressive="false" health="1000" /> <!--<trigger x="-300" id="Test" />--> - <npc name="Ralph" hasDialog="true" /> - <npc name="Johnny" hasDialog="false" /> - <npc name="Big Dave" hasDialog="true"/> + <npc name="Ralph" hasDialog="true" x="300" /> + <npc name="Johnny" hasDialog="false" x="300" /> + <npc name="Big Dave" hasDialog="true" x="300" /> - <page x="-200" id="assets/pages/gootaGoFast.png" /> + <page x="-200" id="assets/door.png" /> <village name="Swaggggggggggggg"> <structure type="0" x="-300" inside="playerSpawnHill1_Building1.xml"/> |