aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authordrumsetmonkey <abelleisle@roadrunner.com>2016-03-01 08:00:55 -0500
committerdrumsetmonkey <abelleisle@roadrunner.com>2016-03-01 08:00:55 -0500
commit97c701f329bf3da154c23b0529f194d4d8823287 (patch)
treee003610fa009b82eff3580a6b72407f282b19510 /src
parent4df411931dd63f22258be76911e0648c3cdc3936 (diff)
parent26d71799f37bc325b6db0214268f4e72eb970ee9 (diff)
Work on merchants and yer mum
Diffstat (limited to 'src')
-rw-r--r--src/common.cpp18
-rw-r--r--src/entities.cpp18
-rw-r--r--src/threadpool.cpp95
-rw-r--r--src/ui.cpp1014
-rw-r--r--src/world.cpp55
5 files changed, 720 insertions, 480 deletions
diff --git a/src/common.cpp b/src/common.cpp
index faa7012..01c8779 100644
--- a/src/common.cpp
+++ b/src/common.cpp
@@ -10,8 +10,6 @@
#include <vector>
#endif // __WIN32__
-void *NULLPTR = NULL;
-
#ifndef __WIN32__
unsigned int millis(void){
@@ -21,22 +19,6 @@ unsigned int millis(void){
#endif // __WIN32__
-Condition::Condition(const char *_id,void *val){
- id = new char[strlen(_id)+1];
- strcpy(id,_id);
- value = val;
-}
-Condition::~Condition(){
- delete[] id;
-}
-
-bool Condition::sameID(const char *s){
- return !strcmp(id,s);
-}
-void *Condition::getValue(void){
- return value;
-}
-
void DEBUG_prints(const char* file, int line, const char *s,...){
va_list args;
printf("%s:%d: ",file,line);
diff --git a/src/entities.cpp b/src/entities.cpp
index afcc9ad..5e8d51c 100644
--- a/src/entities.cpp
+++ b/src/entities.cpp
@@ -150,20 +150,20 @@ Merchant::Merchant(){ //sets all of the Merchant specific traits on object creat
//tex = new Texturec(1,"assets/NPC.png");
//inv = new Inventory(NPC_INV_SIZE);
- inv = NULL;
-
+ //inv = new Inventory(1);
+
//randDialog = rand() % RAND_DIALOG_COUNT - 1;
dialogIndex = 0;
}
Merchant::~Merchant(){
- while(!aiFunc.empty()){
+ /*while(!aiFunc.empty()){
aiFunc.pop_back();
- }
+ }*/
- delete inv;
- delete tex;
- delete[] name;
+ //delete inv;
+ //delete tex;
+ //delete[] name;
}
Structures::Structures(){ //sets the structure type
@@ -458,7 +458,9 @@ void NPC::interact(){ //have the npc's interact back to the player
}
void Merchant::interact(){
- ui::merchantBox(name, ":Accept:Good Bye", false, "Welcome to smithy\'s. Buy your sausages here you freaking mean lording screw-face");
+ ui::merchantBox(name, &bsinv, ":Accept:Good-Bye", false, "Welcome to Smithy\'s. Buy your sausages here you freaking meme lording screw-face");
+ //ui::merchantBox();
+ ui::waitForDialog();
}
void Object::interact(void){
diff --git a/src/threadpool.cpp b/src/threadpool.cpp
new file mode 100644
index 0000000..d49217d
--- /dev/null
+++ b/src/threadpool.cpp
@@ -0,0 +1,95 @@
+#include "threadpool.h"
+
+// Constructor.
+ThreadPool::ThreadPool(int threads) :
+ terminate(false),
+ stopped(false)
+{
+ // Create number of required threads and add them to the thread pool vector.
+ for(int i = 0; i < threads; i++)
+ {
+ threadPool.emplace_back(thread(&ThreadPool::Invoke, this));
+ }
+}
+
+void ThreadPool::Enqueue(function<void()> f)
+{
+ // Scope based locking.
+ {
+ // Put unique lock on task mutex.
+ unique_lock<mutex> lock(tasksMutex);
+
+ // Push task into queue.
+ tasks.push(f);
+ }
+
+ // Wake up one thread.
+ condition.notify_one();
+}
+
+void ThreadPool::Invoke() {
+
+ function<void()> task;
+ while(true)
+ {
+ // Scope based locking.
+ {
+ // Put unique lock on task mutex.
+ unique_lock<mutex> lock(tasksMutex);
+
+ // Wait until queue is not empty or termination signal is sent.
+ condition.wait(lock, [this]{ return !tasks.empty() || terminate; });
+
+ // If termination signal received and queue is empty then exit else continue clearing the queue.
+ if (terminate && tasks.empty())
+ {
+ return;
+ }
+
+ // Get next task in the queue.
+ task = tasks.front();
+
+ // Remove it from the queue.
+ tasks.pop();
+ }
+
+ // Execute the task.
+ task();
+ }
+}
+
+void ThreadPool::ShutDown()
+{
+ // Scope based locking.
+ {
+ // Put unique lock on task mutex.
+ unique_lock<mutex> lock(tasksMutex);
+
+ // Set termination flag to true.
+ terminate = true;
+ }
+
+ // Wake up all threads.
+ condition.notify_all();
+
+ // Join all threads.
+ for(thread &thread : threadPool)
+ {
+ thread.join();
+ }
+
+ // Empty workers vector.
+ threadPool.empty();
+
+ // Indicate that the pool has been shut down.
+ stopped = true;
+}
+
+// Destructor.
+ThreadPool::~ThreadPool()
+{
+ if (!stopped)
+ {
+ ShutDown();
+ }
+} \ No newline at end of file
diff --git a/src/ui.cpp b/src/ui.cpp
index 836a98e..00c0864 100644
--- a/src/ui.cpp
+++ b/src/ui.cpp
@@ -1,160 +1,214 @@
#include <ui.h>
-/*
- * Create a macro to easily access SDL keypresses
-*/
+/**
+ * A macro for easier SDL key reading
+ */
#define SDL_KEY e.key.keysym.sym
-extern std::vector<menuItem>optionsMenu;
-
-extern SDL_Window *window;
-
/*
- * External references for updating player coords / current world.
-*/
+ * Important to have..
+ */
-extern Player *player;
-extern World *currentWorld;
+extern SDL_Window *window;
+extern Player *player;
+extern World *currentWorld;
+extern bool gameRunning;
/*
- * In the case of dialog, some NPC quests can be preloaded so that they aren't assigned until
- * the dialog box closes. Reference variables for that here.
-*/
+ * NPC AI functions are given to the NPCs within UI loops.
+ */
extern std::vector<int (*)(NPC *)> AIpreload;
extern std::vector<NPC *> AIpreaddr;
/*
- * Pressing ESC or closing the window will set this to false.
-*/
+ * Variables and objects used with the FreeType library and font rendering.
+ */
-extern bool gameRunning;
+#define FT_CHAR_COUNT 93
-/*
- * Freetype variables, and a GLuint for referencing rendered letters.
-*/
+typedef struct {
+ GLuint tex; /**< OpenGL texture object */
+ ivec2 wh; /**< Width and height of the character */
+ ivec2 bl; /**< Offset for drawing the character? */
+ ivec2 ad; /**< Number of pixels to advance cursor. */
+} FT_Tex;
+
+static FT_Tex ftmap[FT_CHAR_COUNT];
static FT_Library ftl;
static FT_Face ftf;
-static GLuint ftex[93];
-static vec2 ftexwh[93];
-static vec2 ftexbl[93];
-static vec2 ftexad[93];
-
static unsigned char fontColor[3] = {255,255,255};
/*
* Variables for dialog boxes / options.
-*/
+ */
-static char dialogBoxText[512];
-static char *dialogOptText[4];
-static float dialogOptLoc[4][3];
+static char dialogBoxText[512];
+static char *dialogOptText[4];
+static float merchAOptLoc[2][3];
+static float dialogOptLoc[4][3];
static unsigned char dialogOptCount = 0;
-static bool typeOutDone = true;
+static bool typeOutDone = true;
+
+/*
+ * Menu-related objects
+ */
-extern Menu* currentMenu;
+extern Menu *currentMenu;
extern Menu pauseMenu;
+/**
+ * The sound made when displaying characters with dialogBox or importantText.
+ */
static Mix_Chunk *dialogClick;
+/*
+ * Other sounds
+ */
+
+Mix_Chunk *battleStart;
+Mix_Chunk *sanic;
+
+/**
+ * A reference to the main loop function for functions like waitForCover().
+ */
+
extern void mainLoop(void);
/*
- * Toggled by pressing 'q', disables some controls when true.
-*/
+ * Overlay variables
+ */
bool fadeEnable = false;
-bool fadeWhite = false;
-bool fadeFast = false;
+bool fadeWhite = false;
+bool fadeFast = false;
+
unsigned int fadeIntensity = 0;
+/**
+ * Set to true when the player is in a battle area (i.e. currentWorld points to
+ * an Arena).
+ */
+
bool inBattle = false;
-Mix_Chunk *battleStart;
-Mix_Chunk *sanic;
+
void Menu::gotoParent(){
- if(parent == NULL){
+ if(!parent){
currentMenu = NULL;
updateConfig();
- }else{
+ }else
currentMenu = parent;
- }
}
void Menu::gotoChild(){
- if(child == NULL){
+ if(!child)
currentMenu = NULL;
- }else{
+ else
currentMenu = child;
- }
}
+static vec2 premouse={0,0};
+
namespace ui {
- /*
- * Mouse coordinates.
- */
+ /**
+ * The current position of the mouse.
+ */
vec2 mouse;
- static vec2 premouse={0,0};
- /*
- * Variety of keydown bools
- */
- bool edown;
-
- /*
- * Debugging flags.
- */
+ /**
+ * If true, debug information will be drawn to the screen.
+ */
+
+ bool debug = false;
+
+ /**
+ * If true, lines should be drawn to the player when the debug menu is open.
+ */
- bool debug=false;
bool posFlag=false;
+
+ /**
+ * If true, the player will be able to move when the current dialog box is
+ * displayed.
+ */
+
bool dialogPassive = false;
bool dialogMerchant = false;
+ std::vector<BuySell> *minv;
int dialogPassiveTime = 0;
-
- /*
- * Dialog stuff that needs to be 'public'.
- */
+ /**
+ * When set to true the dialog box will attempt to display.
+ */
bool dialogBoxExists = false;
+
+ /**
+ * When set to true the text will display as 'important' text.
+ */
+
bool dialogImportant = false;
+
+ /**
+ * Contains the last chosen dialog option.
+ */
+
unsigned char dialogOptChosen = 0;
+ /**
+ * Determines how many characters can be displayed in a dialog box before
+ * a new line is required.
+ */
+
unsigned int textWrapLimit = 110;
- /*
- * Current font size. Changing this WILL NOT change the font size, see setFontSize() for
- * actual font size changing.
- */
+ /**
+ * The current font size.
+ *
+ * DO NOT change this directly, use setFontSize() instead.
+ */
unsigned int fontSize;
- /*
- * Initialises the Freetype library, and sets a font size.
- */
+ /**
+ * Initializes the Freetype library, and other UI related variables.
+ */
void initFonts(void){
+
+ // init the FreeType library
if(FT_Init_FreeType(&ftl)){
- std::cout<<"Error! Couldn't initialize freetype."<<std::endl;
+ std::cout<<"Error! Couldn't initialize the FreeType library."<<std::endl;
abort();
}
- fontSize=0;
- memset(&ftex,0,93*sizeof(GLuint));
+
+ fontSize = 0;
+ memset(&ftmap, 0, FT_CHAR_COUNT * sizeof(FT_Tex));
+
#ifdef DEBUG
DEBUG_printf("Initialized FreeType2.\n",NULL);
#endif // DEBUG
+
+ /*
+ * Load UI related sounds.
+ */
+
dialogClick = Mix_LoadWAV("assets/sounds/click.wav");
battleStart = Mix_LoadWAV("assets/sounds/frig.wav");
- sanic = Mix_LoadWAV("assets/sounds/sanic.wav");
- //Mix_Volume(1,50);
+ sanic = Mix_LoadWAV("assets/sounds/sanic.wav");
}
+ /**
+ * Frees resources taken by the UI facilities
+ */
+
void destroyFonts(void){
FT_Done_Face(ftf);
FT_Done_FreeType(ftl);
@@ -164,95 +218,96 @@ namespace ui {
Mix_FreeChunk(sanic);
}
- /*
- * Sets a new font family to use (*.ttf).
- */
+ /**
+ * Sets a new font face to use (*.ttf).
+ */
void setFontFace(const char *ttf){
if(FT_New_Face(ftl,ttf,0,&ftf)){
std::cout<<"Error! Couldn't open "<<ttf<<"."<<std::endl;
abort();
}
+
#ifdef DEBUG
DEBUG_printf("Using font %s\n",ttf);
#endif // DEBUG
+
}
- /*
- * Sets a new font size (default: 12).
- */
+ /**
+ * Sets a new font size and renders the necessary characters (default font
+ * size: 16; 24 for importantText).
+ */
void setFontSize(unsigned int size){
mtx.lock();
unsigned int i,j;
- char *buf;
- fontSize=size;
-
- FT_Set_Pixel_Sizes(ftf,0,fontSize);
+ std::unique_ptr<uint8_t[]> rgbaBuf;
+ size_t rgbaBufSize = 0;
- /*
- * Pre-render 'all' the characters.
- */
+ FT_Set_Pixel_Sizes(ftf,0,(fontSize = size));
- glDeleteTextures(93,ftex); // delete[] any already-rendered textures
- glGenTextures(93,ftex); // Generate new texture name/locations?
+ // delete old characters, make space for new ones
+ for(i=0; i < FT_CHAR_COUNT; i++){
+ glDeleteTextures(1, &ftmap[i].tex);
+ glGenTextures(1, &ftmap[i].tex);
+ }
+ // Load all characters we expect to use
for(i=33;i<126;i++){
- /*
- * Load the character from the font family file.
- */
- //uiLoop ? std::cout << "Loop while render\n" : std::cout << "No loop while render\n";
+ // Load the bitmap for the current character.
if(FT_Load_Char(ftf,i,FT_LOAD_RENDER)){
std::cout<<"Error! Unsupported character "<<(char)i<<" ("<<i<<")."<<std::endl;
abort();
}
/*
- * Transfer the character's bitmap (?) to a texture for rendering.
- */
+ * Set up the OpenGL texture thing.
+ */
- glBindTexture(GL_TEXTURE_2D,ftex[i-33]);
- glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_WRAP_S ,GL_CLAMP_TO_EDGE);
- glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_WRAP_T ,GL_CLAMP_TO_EDGE);
- glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER ,GL_LINEAR );
- glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER ,GL_LINEAR );
- glPixelStorei(GL_UNPACK_ALIGNMENT,1);
+ glBindTexture(GL_TEXTURE_2D, ftmap[i-33].tex);
+
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S , GL_CLAMP_TO_EDGE);
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T , GL_CLAMP_TO_EDGE);
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER , GL_LINEAR );
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER , GL_LINEAR );
+
+ glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
/*
- * The just-created texture will render red-on-black if we don't do anything to it, so
- * here we create a buffer 4 times the size and transform the texture into an RGBA array,
- * making it white-on-black.
- */
-
- buf = new char[ftf->glyph->bitmap.width * ftf->glyph->bitmap.rows * 4];
+ * Convert the bitmap font given to us from FreeType into an RGBA
+ * format, for ease of drawing.
+ */
+
+ rgbaBuf.reset(new uint8_t [(rgbaBufSize = ftf->glyph->bitmap.width * ftf->glyph->bitmap.rows * 4)]);
+ rgbaBufSize /= 4;
- for(j=0;j<ftf->glyph->bitmap.width*ftf->glyph->bitmap.rows;j++){
- buf[j*4 ]=255;//fontColor[0];
- buf[j*4+1]=255;//fontColor[1];
- buf[j*4+2]=255;//fontColor[2];
- buf[j*4+3]=ftf->glyph->bitmap.buffer[j] ? 255 : 0;
- //buf[j*4+3]=ftf->glyph->bitmap.buffer[j];
+ // populate the buffer
+ for(j=0; j < rgbaBufSize; j++){
+ rgbaBuf[j * 4 ] =
+ rgbaBuf[j * 4 + 1] =
+ rgbaBuf[j * 4 + 2] = 255;
+ rgbaBuf[j * 4 + 3] = ftf->glyph->bitmap.buffer[j] ? 255 : 0;
}
- ftexwh[i-33].x=ftf->glyph->bitmap.width;
- ftexwh[i-33].y=ftf->glyph->bitmap.rows;
- ftexbl[i-33].x=ftf->glyph->bitmap_left;
- ftexbl[i-33].y=ftf->glyph->bitmap_top;
- ftexad[i-33].x=ftf->glyph->advance.x>>6;
- ftexad[i-33].y=ftf->glyph->advance.y>>6;
+ // save important character information
+ ftmap[i-33].wh = { (int)ftf->glyph->bitmap.width, (int)ftf->glyph->bitmap.rows };
+ ftmap[i-33].bl = { ftf->glyph->bitmap_left, ftf->glyph->bitmap_top };
+ ftmap[i-33].ad = { ftf->glyph->advance.x >> 6, ftf->glyph->advance.y >> 6 };
- glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA,ftf->glyph->bitmap.width,ftf->glyph->bitmap.rows,0,GL_RGBA,GL_UNSIGNED_BYTE,buf);
+ // do the thing
+ glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, ftf->glyph->bitmap.width, ftf->glyph->bitmap.rows, 0, GL_RGBA, GL_UNSIGNED_BYTE, rgbaBuf.get());
- delete[] buf; //free(buf);
+ rgbaBuf.release();
}
mtx.unlock();
}
- /*
- * Set a color for font rendering (default: white).
- */
+ /**
+ * Set a color for font rendering (default: white).
+ */
void setFontColor(unsigned char r,unsigned char g,unsigned char b){
fontColor[0]=r;
@@ -260,413 +315,430 @@ namespace ui {
fontColor[2]=b;
}
- /*
- * Draws a character at the specified coordinates, aborting if the character is unknown.
- */
+ /**
+ * Draws a character at the specified coordinates, aborting if the character is undrawable.
+ */
- vec2 putChar(float xx,float yy,char c){
- vec2 c1,c2;
+ ivec2 putChar(float x,float y,char c){
+ ivec2 c1,c2;
- int x = xx, y = yy;
-
- /*
- * Get the width and height of the rendered character.
- */
-
- c1={(float)floor(x)+ftexbl[c-33].x,
- (float)floor(y)+ftexbl[c-33].y};
- c2=ftexwh[c-33];
+ // calculate coordinates
+ c1 = { (int)floor(x) + ftmap[c-33].bl.x,
+ (int)floor(y) + ftmap[c-33].bl.y };
+ c2 = ftmap[c-33].wh;
/*
- * Draw the character:
- */
+ * Draw the character
+ */
glEnable(GL_TEXTURE_2D);
- glBindTexture(GL_TEXTURE_2D,ftex[c-33]);
+ glBindTexture(GL_TEXTURE_2D, ftmap[c-33].tex);
+
glPushMatrix();
glTranslatef(0,-c2.y,0);
+
glBegin(GL_QUADS);
- glColor3ub(fontColor[0],fontColor[1],fontColor[2]);
- glTexCoord2f(0,1);glVertex2f(c1.x ,c1.y );
- glTexCoord2f(1,1);glVertex2f(c1.x+c2.x,c1.y );
- glTexCoord2f(1,0);glVertex2f(c1.x+c2.x,c1.y+c2.y);
- glTexCoord2f(0,0);glVertex2f(c1.x ,c1.y+c2.y);
+ glColor3ub(fontColor[0], fontColor[1], fontColor[2]);
+
+ glTexCoord2f(0, 1); glVertex2f(c1.x , c1.y );
+ glTexCoord2f(1, 1); glVertex2f(c1.x + c2.x, c1.y );
+ glTexCoord2f(1, 0); glVertex2f(c1.x + c2.x, c1.y + c2.y);
+ glTexCoord2f(0, 0); glVertex2f(c1.x , c1.y + c2.y);
+
glEnd();
+
glPopMatrix();
glDisable(GL_TEXTURE_2D);
-
- /*
- * return the width.
- */
-
- return ftexad[c-33];//(vec2){c2.x,ftexad[c-33].y};
+
+ // return the number of pixels the cursor should move
+ return ftmap[c-33].ad;
}
- /*
- * Draw a string at the specified coordinates.
- */
+ /**
+ * Draw a string at the specified coordinates.
+ */
float putString(const float x,const float y,const char *s){
- unsigned int i=0;
- float xo=x,yo=y;
- vec2 add;
+ unsigned int i = 0;
+ ivec2 add;
+ vec2 off = { (float)floor(x), (float)floor(y) };
/*
* Loop on each character:
- */
+ */
do{
- if(i && ((i / 110.0) == (i / 110))){
- yo-=fontSize*1.05;
- xo=x;
+ // wrap text if necessary
+ if(i && (i / (float)textWrapLimit == i / textWrapLimit)){
+ off.y -= fontSize * 1.05;
+ off.x = x;
+
+ // skip a space if it's there since we just newline'd
if(s[i] == ' ')
i++;
}
+
+ // handle newlines
if(s[i] == '\n'){
- yo-=fontSize*1.05;
- xo=x;
+ off.y -= fontSize * 1.05;
+ off.x = x;
+
+ // (TODO) handle carriage returns and tabs
}else if(s[i] == '\r' || s[i] == '\t'){
- /*if(s[i] == '\n'){
- yo-=fontSize*1.05;
- xo=x;
- */}else if(s[i]==' '){ // Handle spaces
- xo+=fontSize/2;
- }else if(s[i]=='\b'){ // Handle backspaces?
- xo-=add.x;
+
+ // handle spaces
+ }else if(s[i]==' '){
+ off.x += fontSize / 2;
+
+ // handle backspaces
+ }else if(s[i]=='\b'){
+ off.x -= add.x;
+
+ // handle everything else
}else{
- add=putChar(floor(xo),floor(yo),s[i]);
- xo+=add.x;
- yo+=add.y;
+ add = putChar(off.x, off.y, s[i]);
+ off.x += add.x;
+ off.y += add.y;
}
+
}while(s[++i]);
- return xo; // i.e. the string width
+ // return string width
+ return off.x;
}
+ /**
+ * Print a string center-aligned on the specified coordinate.
+ */
+
float putStringCentered(const float x,const float y,const char *s){
unsigned int i = 0;
float width = 0;
+ /*
+ * Calculate the string's width by cycling through each character.
+ */
+
do{
- if(s[i]=='\n'){ // Handle newlines
- // TODO
- }else if(s[i]==' '){ // Handle spaces
- width+=fontSize/2;
- }else if(s[i]=='\b'){ // Handle backspaces?
- // Why?
- // Cuz
- }else{
- width+=ftexwh[i].x+fontSize*.1;
- }
+ // handle newlines
+ if(s[i]=='\n'){
+
+ // handle spaces
+ }else if(s[i]==' '){
+ width += fontSize / 2;
+
+ // handle backspaces
+ }else if(s[i]=='\b'){
+
+ // handle everything else
+ }else
+ width += ftmap[i].wh.x + fontSize * .1;
+
}while(s[++i]);
-
- putString(floor(x-width/2),y,s);
- return width;
+
+ // print the string
+ return putString(floor(x - width / 2), y, s);
}
- /*
- * 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.
- */
-
- static char *ret = NULL;
+ /**
+ * Draw a string in a typewriter-esque fashion.
+ *
+ * This function is expected to be called as it is rendered, slowly allowing
+ * more characters to be drawn as it is looped on. Only one call to this
+ * function can be handled at a time.
+ */
+
+ static char *typeOutStr = NULL;
char *typeOut(char *str){
- static unsigned int sinc, // Acts as a delayer for the space between each character.
+ static unsigned int sinc=0, // Acts as a delayer for the space between each character.
linc=0, // Contains the number of letters that should be drawn.
size=0; // Contains the full size of the current string.
- //static char *ret = NULL;
-
- /*
- * Create a well-sized buffer if we haven't yet.
- */
-
- if(!ret){
- ret = new char[512]; //(char *)calloc(512,sizeof(char));
- memset(ret,0,512*sizeof(char));
+
+ // allocate memory for the string if necessary
+ if(!typeOutStr){
+ typeOutStr = new char[512];
+ memset(typeOutStr, 0, 512 * sizeof(char));
}
/*
* Reset values if a new string is being passed.
- */
+ */
- if(strncmp(ret,str,linc-1)){
- memset(ret,0,512); // Zero the buffer
- size=strlen(str); // Set the new target string size
- linc=0; // Reset the incrementers
- sinc=1;
+ if(strncmp(typeOutStr, str, linc - 1)){
+ memset(typeOutStr, 0, 512); // Zero the buffer
+ size = strlen(str); // Set the new target string size
+ linc = 0; // Reset the incrementers
+ sinc = 1; //
typeOutDone = false;
}
/*
* Draw the next letter if necessary.
- */
+ */
if(typeOutDone)
return str;
- else if(++sinc==2){
- sinc=0;
+ else if(++sinc == 2){
+ sinc = 0;
- strncpy(ret+linc,str+linc,1); // Get next character
+ // add next character to output string
+ strncpy(typeOutStr + linc,str + linc, 1);
- if(linc<size)
+ if(linc < size)
linc++;
else
typeOutDone = true;
}
-
- return ret; // The buffered string.
+
+ // return the string
+ return typeOutStr;
}
- /*
- * Draw a formatted string to the specified coordinates.
- */
+ /**
+ * Draw a formatted string to the specified coordinates.
+ */
float putText(const float x,const float y,const char *str,...){
va_list args;
- char *buf;
- float width;
-
- /*
- * Create a wimpy buffer.
- */
-
- buf = new char[512]; //(char *)calloc(128,sizeof(char));
- memset(buf,0,512*sizeof(char));
-
- /*
- * Handle the formatted string, printing it to the buffer.
- */
+ std::unique_ptr<char[]> buf (new char[512]);
- va_start(args,str);
- vsnprintf(buf,512,str,args);
+ // create the formatted string
+ va_start(args, str);
+ vsnprintf(buf.get(), 512, str, args);
va_end(args);
- /*
- * Draw the string, free resources, return the width of the string.
- */
-
- width=putString(x,y,buf);
- delete[] buf; //free(buf);
-
- return width;
+ return putString(x, y, buf.get());
}
+
+ /**
+ * Prints a character dialog box.
+ *
+ * This function sets up the variables necessary to draw a dialog box. If
+ * `opt` contains a valid string, options will be printed with the dialog
+ * box. If the box is passive, the player will be allowed to move while it
+ * is being displayed.
+ */
+
void dialogBox(const char *name,const char *opt,bool passive,const char *text,...){
+ textWrapLimit = 110;
va_list dialogArgs;
- unsigned int len;
- char *sopt,*soptbuf;
+ size_t len;
dialogPassive = passive;
- /*
- * Set up the text buffer.
- */
+ // clear the buffer
+ memset(dialogBoxText, '\0', 512);
- memset(dialogBoxText,0,512);
-
- /*
- * Get the text ready for rendering.
- */
-
- len=strlen(name);
- strcpy(dialogBoxText ,name);
- strcpy(dialogBoxText+len,": ");
- len+=2;
+ // create the string
+ strcpy(dialogBoxText, name);
+ strcat(dialogBoxText, ": ");
+ len=strlen(dialogBoxText);
va_start(dialogArgs,text);
- vsnprintf(dialogBoxText+len,512-len,text,dialogArgs);
+ vsnprintf(dialogBoxText + len, 512 - len, text, dialogArgs);
va_end(dialogArgs);
- /*
- * Set up option text.
- */
-
+ // free old option text
while(dialogOptCount){
if(dialogOptText[dialogOptCount]){
- delete[] dialogOptText[dialogOptCount]; //free(dialogOptText[dialogOptCount]);
+ delete[] dialogOptText[dialogOptCount];
dialogOptText[dialogOptCount] = NULL;
}
+
dialogOptCount--;
};
- dialogOptCount = 0;
dialogOptChosen = 0;
- memset(&dialogOptLoc,0,sizeof(float)*12);
+ memset(&dialogOptLoc, 0, sizeof(float) * 12);
+ //char *sopt = (char*)malloc(255);
+ //dialogOptCount = 4;
- if(opt != NULL){
-
- soptbuf = new char[strlen(opt)+1];
- strcpy(soptbuf,opt);
-
- sopt=strtok(soptbuf,":");
- while(sopt != NULL){
- dialogOptText[dialogOptCount] = new char[strlen(sopt)+1]; //(char *)malloc(strlen(sopt));
- strcpy(dialogOptText[dialogOptCount++],sopt);
- sopt=strtok(NULL,":");
+ // handle options if desired
+ if(opt){
+ //std::vector<char*> soptbuf (new char[strlen(opt) + 1]);
+ //std::unique_ptr<char[]> soptbuf (new char[strlen(opt) + 1]);
+ char soptbuf[255];
+ strcpy(soptbuf, opt);
+ char *sopt = strtok(soptbuf, ":");
+
+ // cycle through options
+ while(sopt){
+ printf("%s",sopt);
+ strcpy( (dialogOptText[dialogOptCount++] = new char[strlen(sopt) + 1]), sopt);
+ sopt = strtok(NULL,":");
}
-
- delete[] soptbuf;
-
}
- /*
- * Tell draw() that the box is ready.
- */
-
+ // allow box to be displayed
dialogBoxExists = true;
dialogImportant = false;
- if(ret)
- ret[0] = '\0';
+ // kill the string created by typeOut if it contains something
+ if(typeOutStr)
+ *typeOutStr = '\0';
}
- void merchantBox(const char *name,const char *opt,bool passive,const char *text,...){
+
+ void merchantBox(const char *name,std::vector<BuySell> *bsinv,const char *opt,bool passive,const char *text,...){
std::cout << "Buying and selling on the bi-weekly!" << std::endl;
va_list dialogArgs;
- unsigned int len;
- char *sopt,*soptbuf;
+ size_t len;
+ minv = bsinv;
dialogPassive = passive;
- /*
- * Set up the text buffer.
- */
+ // clear the buffer
+ memset(dialogBoxText, '\0', 512);
- memset(dialogBoxText,0,512);
-
- /*
- * Get the text ready for rendering.
- */
-
- len=strlen(name);
- strcpy(dialogBoxText ,name);
- strcpy(dialogBoxText+len,": ");
- len+=2;
+ // create the string
+ strcpy(dialogBoxText, name);
+ strcat(dialogBoxText, ": ");
+ len=strlen(dialogBoxText);
va_start(dialogArgs,text);
- vsnprintf(dialogBoxText+len,512-len,text,dialogArgs);
+ vsnprintf(dialogBoxText + len, 512 - len, text, dialogArgs);
va_end(dialogArgs);
- /*
- * Set up option text.
- */
-
+ // free old option text
while(dialogOptCount){
if(dialogOptText[dialogOptCount]){
- delete[] dialogOptText[dialogOptCount]; //free(dialogOptText[dialogOptCount]);
+ delete[] dialogOptText[dialogOptCount];
dialogOptText[dialogOptCount] = NULL;
}
+
dialogOptCount--;
};
- dialogOptCount = 0;
dialogOptChosen = 0;
- memset(&dialogOptLoc,0,sizeof(float)*12);
+ memset(&dialogOptLoc, 0, sizeof(float) * 12);
- if(opt != NULL){
-
- soptbuf = new char[strlen(opt)+1];
- strcpy(soptbuf,opt);
-
- sopt=strtok(soptbuf,":");
- while(sopt != NULL){
- dialogOptText[dialogOptCount] = new char[strlen(sopt)+1]; //(char *)malloc(strlen(sopt));
- strcpy(dialogOptText[dialogOptCount++],sopt);
- sopt=strtok(NULL,":");
+ // handle options if desired
+ if(opt){
+ //std::unique_ptr<char[]> soptbuf (new char[strlen(opt) + 1]);
+ char soptbuf[255];
+ strcpy(soptbuf, opt);
+ char *sopt = strtok(soptbuf, ":");
+
+ // cycle through options
+ while(sopt){
+ strcpy( (dialogOptText[dialogOptCount++] = new char[strlen(sopt) + 1]), sopt);
+ sopt = strtok(NULL,":");
}
-
- delete[] soptbuf;
-
}
- /*
- * Tell draw() that the box is ready.
- */
-
+ // allow box to be displayed
dialogBoxExists = true;
dialogImportant = false;
dialogMerchant = true;
+ textWrapLimit = 50;
- if(ret)
- ret[0] = '\0';
+ // kill the string created by typeOut if it contains something
+ if(typeOutStr)
+ *typeOutStr = '\0';
+ }
+
+ void merchantBox(){
+ textWrapLimit = 50;
+ dialogMerchant = true;
}
+
+ /**
+ * Wait for a dialog box to be dismissed.
+ */
+
void waitForDialog(void){
do{
mainLoop();
}while(ui::dialogBoxExists);
}
+
+ /**
+ * Wait for the screen to be fully covered through toggle___().
+ */
+
void waitForCover(void){
do{
mainLoop();
}while(fadeIntensity < 255);
fadeIntensity = 255;
}
- void waitForNothing(unsigned int ms){
- unsigned int target = millis() + ms;
- do{
- mainLoop();
- }while(millis() < target);
- }
+
+ /**
+ * Prepare formatted 'important' string for drawing.
+ *
+ * "Important" text will display in a bigger font size at the center of the
+ * screen. Usually accompanied by a cover from toggle___() and limits on
+ * player controls.
+ */
+
void importantText(const char *text,...){
va_list textArgs;
- //if(!player->ground)return;
-
- memset(dialogBoxText,0,512);
+ // clear dialog buffer (we share the same)
+ memset(dialogBoxText, '\0', 512);
+ // format the string
va_start(textArgs,text);
- vsnprintf(dialogBoxText,512,text,textArgs);
+ vsnprintf(dialogBoxText, 512, text, textArgs);
va_end(textArgs);
+ // set draw flags
dialogBoxExists = true;
dialogImportant = true;
- //toggleBlack();
}
+
+ /**
+ * Draw a passive 'important' text for a certain duration.
+ */
+
void passiveImportantText(int duration, const char *text,...){
va_list textArgs;
- //if(!player->ground)return;
-
- memset(dialogBoxText,0,512);
+ // clear buffer
+ memset(dialogBoxText, '\0', 512);
+ // format the string
va_start(textArgs,text);
- vsnprintf(dialogBoxText,512,text,textArgs);
+ vsnprintf(dialogBoxText, 512, text, textArgs);
va_end(textArgs);
+ // set draw flags
dialogBoxExists = true;
dialogImportant = true;
dialogPassive = true;
dialogPassiveTime = duration;
}
+ /**
+ * Draws all UI-related elements to the screen.
+ */
void draw(void){
- uiLoop = true;
unsigned char i;
float x,y,tmp;
char *rtext;
+ // handle dialog box / important text
if(dialogBoxExists){
- rtext=typeOut(dialogBoxText);
+ rtext = typeOut(dialogBoxText);
if(dialogImportant){
- setFontColor(255,255,255);
- if(dialogPassive){
- dialogPassiveTime -= deltaTime;
- if(dialogPassiveTime < 0){
- dialogPassive = false;
- dialogImportant = false;
- dialogBoxExists = false;
- }
+ setFontColor(255, 255, 255);
+
+ // handle timeout
+ if(dialogPassive && (dialogPassiveTime -= deltaTime) <= 0){
+ dialogPassive = false;
+ dialogImportant = false;
+ dialogBoxExists = false;
}
+
+ // draw text
if(fadeIntensity == 255 || dialogPassive){
setFontSize(24);
- putStringCentered(offset.x,offset.y,rtext);
+ putStringCentered(offset.x, offset.y, rtext);
setFontSize(16);
}
}else if(dialogMerchant){
- /*static std::string merchOpt[3] = {"Accept",
- "Cancel"};*/
-
x=offset.x-SCREEN_WIDTH/6;
y=(offset.y+SCREEN_HEIGHT/2)-HLINE*8;
@@ -683,114 +755,157 @@ namespace ui {
glColor3ub(0,0,0);
glRectf(x,y,x+SCREEN_WIDTH/3,y-SCREEN_HEIGHT*.6);
- for(i=0;i<dialogOptCount;i++){
- dialogOptLoc[i][1] = (y-SCREEN_HEIGHT*.55) - i*fontSize;
- setFontColor(255,255,255);
- tmp = putStringCentered(offset.x,dialogOptLoc[i][1],dialogOptText[i]);
+ // draw typeOut'd text
+ putString(x + HLINE, y - fontSize - HLINE, (rtext = typeOut(dialogBoxText)));
+ merchAOptLoc[0][0] = offset.x - (SCREEN_WIDTH / 5.5);
+ merchAOptLoc[0][1] = offset.x + (SCREEN_WIDTH / 5.5);
+ merchAOptLoc[1][0] = offset.y + (SCREEN_HEIGHT *.25);
+ merchAOptLoc[1][1] = offset.y + (SCREEN_HEIGHT *.25);
+ merchAOptLoc[2][0] = offset.x - (SCREEN_WIDTH / 5.5) - 16;
+ merchAOptLoc[2][1] = offset.x + (SCREEN_WIDTH / 5.5) + 16;
+
+ glColor3ub(255,255,255);
+ glBegin(GL_TRIANGLES);
+ glVertex2f(merchAOptLoc[2][0],merchAOptLoc[1][0]);
+ glVertex2f(merchAOptLoc[0][0],merchAOptLoc[1][0]-8);
+ glVertex2f(merchAOptLoc[0][0],merchAOptLoc[1][0]+8);
+ glEnd();
+
+ // draw / handle dialog options if they exist
+ for(i = 0; i < dialogOptCount; i++){
+ setFontColor(255, 255, 255);
+
+ // draw option
+ tmp = putStringCentered(offset.x, dialogOptLoc[i][1], dialogOptText[i]);
+
+ // get coordinate information on option
dialogOptLoc[i][2] = offset.x + tmp;
dialogOptLoc[i][0] = offset.x - tmp;
- //merchOptLoc[i][1] = y - SCREEN_HEIGHT / 4 + (fontSize + HLINE) * (i + 1);
- if(mouse.x > dialogOptLoc[i][0] &&
- mouse.x < dialogOptLoc[i][2] &&
- mouse.y > dialogOptLoc[i][1] &&
- mouse.y < dialogOptLoc[i][1] + fontSize){ // fontSize
- setFontColor(255,255,0);
- putStringCentered(offset.x,dialogOptLoc[i][1],dialogOptText[i]);
+ dialogOptLoc[i][1] = y - SCREEN_HEIGHT / 2 - (fontSize + HLINE) * (i + 1);
+
+ // make text yellow if the mouse hovers over the text
+ if(mouse.x > dialogOptLoc[i][0] && mouse.x < dialogOptLoc[i][2] &&
+ mouse.y > dialogOptLoc[i][1] && mouse.y < dialogOptLoc[i][1] + 16 ){
+ setFontColor(255, 255, 0);
+ putStringCentered(offset.x, dialogOptLoc[i][1], dialogOptText[i]);
}
}
- setFontColor(255,255,255);
- }else{
+
+ setFontColor(255, 255, 255);
+ }else{ //normal dialog box
x=offset.x-SCREEN_WIDTH/2+HLINE*8;
y=(offset.y+SCREEN_HEIGHT/2)-HLINE*8;
-
-
- glColor3ub(255,255,255);
+
+ // draw white border
+ glColor3ub(255, 255, 255);
glBegin(GL_LINE_STRIP);
- glVertex2f(x-1 ,y+1);
- glVertex2f(x+1+SCREEN_WIDTH-HLINE*16,y+1);
- glVertex2f(x+1+SCREEN_WIDTH-HLINE*16,y-1-SCREEN_HEIGHT/4);
- glVertex2f(x-1 ,y-1-SCREEN_HEIGHT/4);
- glVertex2f(x ,y+1);
+ glVertex2f(x - 1 , y + 1 );
+ glVertex2f(x + 1 + SCREEN_WIDTH - HLINE * 16, y + 1 );
+ glVertex2f(x + 1 + SCREEN_WIDTH - HLINE * 16, y - 1 - SCREEN_HEIGHT / 4 );
+ glVertex2f(x - 1 , y - 1 - SCREEN_HEIGHT / 4 );
+ glVertex2f(x , y + 1 );
glEnd();
- glColor3ub(0,0,0);
- glRectf(x,y,x+SCREEN_WIDTH-HLINE*16,y-SCREEN_HEIGHT/4);
-
- rtext=typeOut(dialogBoxText);
+ // draw black box
+ glColor3ub(0, 0, 0);
+ glRectf(x, y, x + SCREEN_WIDTH - HLINE * 16, y - SCREEN_HEIGHT / 4);
- putString(x+HLINE,y-fontSize-HLINE,rtext);
+ // draw typeOut'd text
+ putString(x + HLINE, y - fontSize - HLINE, (rtext = typeOut(dialogBoxText)));
- for(i=0;i<dialogOptCount;i++){
- setFontColor(255,255,255);
- tmp = putStringCentered(offset.x,dialogOptLoc[i][1],dialogOptText[i]);
+ // draw / handle dialog options if they exist
+ for(i = 0; i < dialogOptCount; i++){
+ setFontColor(255, 255, 255);
+
+ // draw option
+ tmp = putStringCentered(offset.x, dialogOptLoc[i][1], dialogOptText[i]);
+
+ // get coordinate information on option
dialogOptLoc[i][2] = offset.x + tmp;
dialogOptLoc[i][0] = offset.x - tmp;
dialogOptLoc[i][1] = y - SCREEN_HEIGHT / 4 + (fontSize + HLINE) * (i + 1);
- if(mouse.x > dialogOptLoc[i][0] &&
- mouse.x < dialogOptLoc[i][2] &&
- mouse.y > dialogOptLoc[i][1] &&
- mouse.y < dialogOptLoc[i][1] + 16 ){ // fontSize
- setFontColor(255,255,0);
- putStringCentered(offset.x,dialogOptLoc[i][1],dialogOptText[i]);
+
+ // make text yellow if the mouse hovers over the text
+ if(mouse.x > dialogOptLoc[i][0] && mouse.x < dialogOptLoc[i][2] &&
+ mouse.y > dialogOptLoc[i][1] && mouse.y < dialogOptLoc[i][1] + 16 ){
+ setFontColor(255, 255, 0);
+ putStringCentered(offset.x, dialogOptLoc[i][1], dialogOptText[i]);
}
}
- setFontColor(255,255,255);
+
+ setFontColor(255, 255, 255);
}
- if(strcmp(rtext,dialogBoxText)){
- Mix_PlayChannel(1,dialogClick,0);
- }
+ // make click for each character update
+ if(strcmp(rtext, dialogBoxText))
+ Mix_PlayChannel(1, dialogClick, 0);
+
+ }
+
+ // draw information stuffs
+ if(!fadeIntensity){
- }if(!fadeIntensity){
vec2 hub = {
- (SCREEN_WIDTH/2+offset.x)-fontSize*10,
- (offset.y+SCREEN_HEIGHT/2)-fontSize
+ (offset.x + SCREEN_WIDTH / 2) - fontSize * 10,
+ (offset.y + SCREEN_HEIGHT / 2) - fontSize
};
- putText(hub.x,hub.y,"Health: %u/%u",player->health>0?(unsigned)player->health:0,
- (unsigned)player->maxHealth
- );
+ // health text
+ putText(hub.x,
+ hub.y,
+ "Health: %.0f/%.0f",
+ player->health > 0 ? player->health : 0,
+ player->maxHealth
+ );
+
+ // health bar
if(player->alive){
- glColor3ub(150,0,0);
- hub.y-=fontSize*1.15;
+ hub.y -= fontSize * 1.15;
+
+ glColor3ub(150, 0, 0);
glRectf(hub.x,
hub.y,
- hub.x+150,
- hub.y+12);
+ hub.x + 150,
+ hub.y + 12
+ );
+
glColor3ub(255,0,0);
glRectf(hub.x,
hub.y,
- hub.x+(player->health/player->maxHealth * 150),
- hub.y+12);
+ hub.x + (player->health / player->maxHealth * 150),
+ hub.y + 12
+ );
}
- /*
- * Lists all of the quests the player is currently taking.
- */
-
+ // inventory
if(player->inv->invOpen){
hub.y = player->loc.y + fontSize * 8;
- hub.x = player->loc.x;// + player->width / 2;
+ hub.x = player->loc.x;
- putStringCentered(hub.x,hub.y,"Current Quests:");
+ putStringCentered(hub.x, hub.y, "Current Quests:");
- for(auto &c : player->qh.current){
- hub.y -= fontSize * 1.15;
- putStringCentered(hub.x,hub.y,c.title.c_str());
- }
+ for(auto &q : player->qh.current)
+ putStringCentered(hub.x, (hub.y -= fontSize * 1.15), q.title.c_str());
}
}
- uiLoop = false;
}
+ /**
+ * Safely exits the game.
+ */
+
void quitGame(){
- dialogBoxExists = false;
+ // clean menu stuff
currentMenu = NULL;
delete[] currentMenu;
- gameRunning = false;
+
+ // save options
updateConfig();
saveConfig();
+
+ // tell main loop to exit
+ gameRunning = false;
}
menuItem createButton(vec2 l, dim2 d, Color c, const char* t, menuFunc f){
@@ -1117,30 +1232,38 @@ namespace ui {
fclose(bmp);
}
+ /**
+ * Handles dialog box closing, option selecting and stuff.
+ */
+
void dialogAdvance(void){
unsigned char i;
+
+ // if typeOut hasn't finished, tell it to then try again
if(!typeOutDone){
typeOutDone = true;
return;
}
- for(i=0;i<dialogOptCount;i++){
- if(mouse.x > dialogOptLoc[i][0] &&
- mouse.x < dialogOptLoc[i][2] &&
- mouse.y > dialogOptLoc[i][1] &&
- mouse.y < dialogOptLoc[i][1] + 16 ){ // fontSize
+ // check for selected option
+ for(i = 0; i < dialogOptCount; i++){
+ if(mouse.x > dialogOptLoc[i][0] && mouse.x < dialogOptLoc[i][2] &&
+ mouse.y > dialogOptLoc[i][1] && mouse.y < dialogOptLoc[i][1] + 16 ){
dialogOptChosen = i + 1;
- goto DONE;
+ break;
+ }
+ }
+
+ if(dialogMerchant){
+ for(i = 0; i < 2; i++){
}
}
-DONE:
+
+ // handle important text
if(dialogImportant){
dialogImportant = false;
setFontSize(16);
- //toggleBlack();
}
- /*if(ui::fontSize != 16)
- setFontSize(16);*/
if(dialogMerchant) dialogMerchant = false;
dialogBoxExists = false;
@@ -1286,7 +1409,6 @@ DONE:
if(debug)posFlag ^= true;
break;
case SDLK_e:
- edown=true;
if(!heyOhLetsGo){
heyOhLetsGo = loops;
player->inv->mouseSel = false;
@@ -1334,7 +1456,6 @@ DONE:
player->speed = 1;
break;
case SDLK_e:
- edown=false;
if(player->inv->invHover){
player->inv->invHover = false;
}else{
@@ -1378,10 +1499,10 @@ DONE:
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);
+ //static std::thread scr;
+ //scr = std::thread(takeScreenshot,pixels);
+ //scr.detach();
+ takeScreenshot(pixels);
std::cout << "Took screenshot" << std::endl;
break;
@@ -1407,25 +1528,46 @@ DONE:
}
}
+ /**
+ * Toggle a slow fade to/from black.
+ */
+
void toggleBlack(void){
fadeEnable ^= true;
fadeWhite = false;
fadeFast = false;
}
+
+ /**
+ * Toggle a fast fade to/from black.
+ */
+
void toggleBlackFast(void){
fadeEnable ^= true;
fadeWhite = false;
fadeFast = true;
}
+
+ /**
+ * Toggle a slow fade to/from white.
+ */
+
void toggleWhite(void){
fadeEnable ^= true;
fadeWhite = true;
fadeFast = false;
}
+
+ /**
+ * Toggle a fast fade to/from white. This should only be used for battle
+ * initiations, so the 'battle start' sound is played as well.
+ */
+
void toggleWhiteFast(void){
fadeEnable ^= true;
fadeWhite = true;
fadeFast = true;
- Mix_PlayChannel(1,battleStart,0);
+
+ Mix_PlayChannel(1, battleStart, 0);
}
}
diff --git a/src/world.cpp b/src/world.cpp
index 306933d..9e891fb 100644
--- a/src/world.cpp
+++ b/src/world.cpp
@@ -118,14 +118,13 @@ void World::deleteEntities(void){
delete mob.back();
mob.pop_back();
}
+ while(!merchant.empty()){
+ merchant.pop_back();
+ }
while(!npc.empty()){
delete npc.back();
npc.pop_back();
}
- while(!merchant.empty()){
- delete merchant.back();
- merchant.pop_back();
- }
while(!build.empty()){
delete build.back();
build.pop_back();
@@ -144,6 +143,10 @@ void World::deleteEntities(void){
while(!light.empty()){
light.pop_back();
}
+ while(!village.empty()){
+ delete village.back();
+ village.pop_back();
+ }
}
World::~World(void){
@@ -1004,13 +1007,13 @@ void World::detect(Player *p){
}
for(auto &v : village){
- if(p->loc.x > v.start.x && p->loc.x < v.end.x){
- if(!v.in){
- ui::passiveImportantText(5000,"Welcome to %s",v.name.c_str());
- v.in = true;
+ if(p->loc.x > v->start.x && p->loc.x < v->end.x){
+ if(!v->in){
+ ui::passiveImportantText(5000,"Welcome to %s",v->name.c_str());
+ v->in = true;
}
}else{
- v.in = false;
+ v->in = false;
}
}
@@ -1545,14 +1548,15 @@ World *loadWorldFromXMLNoSave(const char *path){
xml.LoadFile(currentXML);
wxml = xml.FirstChildElement("World");
- vil = xml.FirstChildElement("World")->FirstChildElement("village");
if(wxml){
wxml = wxml->FirstChildElement();
+ vil = xml.FirstChildElement("World")->FirstChildElement("village");
Indoor = false;
tmp = new World();
}else if((wxml = xml.FirstChildElement("IndoorWorld"))){
wxml = wxml->FirstChildElement();
+ vil = NULL;
Indoor = true;
tmp = new IndoorWorld();
}
@@ -1627,8 +1631,11 @@ World *loadWorldFromXMLNoSave(const char *path){
wxml = wxml->NextSiblingElement();
}
+ Village *vptr;
+
if(vil){
- tmp->village.push_back(vil->Attribute("name"));
+ tmp->village.push_back(new Village(vil->Attribute("name"), tmp));
+ vptr = tmp->village.back();
vil = vil->FirstChildElement();
}
@@ -1640,11 +1647,11 @@ World *loadWorldFromXMLNoSave(const char *path){
/**
* READS DATA ABOUT STRUCTURE CONTAINED IN VILLAGE
*/
+
if(!strcmp(name,"structure")){
ptr = vil->Attribute("inside");
tmp->addStructure((BUILD_SUB)vil->UnsignedAttribute("type"),
- vil->QueryFloatAttribute("x", &spawnx) != XML_NO_ERROR ?
- randx : spawnx,
+ vil->QueryFloatAttribute("x", &spawnx) != XML_NO_ERROR ? randx : spawnx,
100,
(char*)vil->Attribute("texture"),
ptr);
@@ -1659,6 +1666,11 @@ World *loadWorldFromXMLNoSave(const char *path){
(char*)vil->Attribute("texture"),
ptr);
tmp->addMerchant(0,100);
+ if(!strcmp(name,"buy")){
+ std::cout << "Buying";
+ }else if(!strcmp(name,"sell")){
+ std::cout << "Selling";
+ }
strcpy(tmp->merchant.back()->name,"meme");
}else if(!strcmp(vil->Attribute("type"),"trader")){
@@ -1671,12 +1683,12 @@ World *loadWorldFromXMLNoSave(const char *path){
ptr);
}
}
- tmp->village.back().build.push_back(tmp->build.back());
- if(tmp->village.back().build.back()->loc.x < tmp->village.back().start.x){
- tmp->village.back().start.x = tmp->village.back().build.back()->loc.x;
+ vptr->build.push_back(tmp->build.back());
+ if(vptr->build.back()->loc.x < vptr->start.x){
+ vptr->start.x = vptr->build.back()->loc.x;
}
- if(tmp->village.back().build.back()->loc.x + tmp->village.back().build.back()->width > tmp->village.back().end.x){
- tmp->village.back().end.x = tmp->village.back().build.back()->loc.x + tmp->village.back().build.back()->width;
+ if(vptr->build.back()->loc.x + vptr->build.back()->width > vptr->end.x){
+ vptr->end.x = vptr->build.back()->loc.x + vptr->build.back()->width;
}
//go to the next element in the village block
@@ -1691,3 +1703,10 @@ World *loadWorldFromXMLNoSave(const char *path){
return tmp;
}
+
+Village::Village(const char *meme, World *w){
+ name = meme;
+ start.x = w->getTheWidth() / 2.0f;
+ end.x = -start.x;
+ in = false;
+}