#include #include #define getWidth(w) ((w->lineCount-GEN_INC)*HLINE) // Calculates the width of world 'w' #define GEN_INIT 60 #define GRASS_HEIGHT 4 // Defines how long the grass layer of a line should be in multiples of HLINE. #define DRAW_Y_OFFSET 50 // Defines how many pixels each layer should be offset from each other on the y axis when drawn. #define DRAW_SHADE 30 // Defines a shade increment for draw() #define INDOOR_FLOOR_HEIGHT 100 // Defines how high the base floor of an IndoorWorld should be bool worldInside = false; // True if player is inside a structure WEATHER weather = SUNNY; const char *bgPaths[2][7]={ {"assets/bg.png", // Daytime background "assets/bgn.png", // Nighttime background "assets/bgFarMountain.png", // Furthest layer "assets/forestTileBack.png", // Closer layer "assets/forestTileMid.png", // Near layer "assets/forestTileFront.png", // Closest layer "assets/dirt.png"}, // Dirt {"assets/bgWoodTile.png", NULL, NULL, NULL, NULL, NULL} }; Texturec *grassT; const float bgDraw[3][3]={ {100,240,.6 }, {150,250,.4 }, {255,255,.25} }; float worldGetYBase(World *w){ // World *tmp = w; float base = GEN_MIN; /* while(tmp->infront){ tmp = tmp->infront; base -= DRAW_Y_OFFSET; }*/ if(!w)return 0; return base; } void World::setBackground(WORLD_BG_TYPE bgt){ bgType = bgt; switch(bgt){ case BG_FOREST: bgTex = new Texturec(7,bgPaths[0]); break; case BG_WOODHOUSE: bgTex = new Texturec(1,bgPaths[1]); break; } } World::World(void){ bgm = NULL; bgmObj = NULL; toLeft = toRight = NULL; /* * Allocate and clear an array for star coordinates. */ star = new vec2[100]; memset(star,0,100 * sizeof(vec2)); grassT = new Texturec(1,"assets/grass.png"); } void World::deleteEntities(void){ while(!mob.empty()){ delete mob.back(); mob.pop_back(); } while(!npc.empty()){ delete npc.back(); npc.pop_back(); } while(!build.empty()){ delete build.back(); build.pop_back(); } while(!object.empty()){ delete object.back(); object.pop_back(); } while(!entity.empty()){ entity.pop_back(); } while(!particles.empty()){ delete particles.back(); particles.pop_back(); } while(!light.empty()){ light.pop_back(); } } World::~World(void){ /*if(behind != NULL) delete behind;*/ if(bgmObj) Mix_FreeMusic(bgmObj); if(bgm) delete[] bgm; delete bgTex; delete[] star; delete[] line; delete[] toLeft; delete[] toRight; deleteEntities(); } void World::generate(unsigned int width){ // Generates the world and sets all variables contained in the World class. unsigned int i; float inc; /* * Calculate the world's real width. The current form of generation fails to generate * the last GEN_INC lines, so we offset those making the real real width what was passed * to this function. * * Abort if the width is invalid. * */ if(width <= 0) abort(); lineCount = width + GEN_INC; /* * Allocate enough memory for the world to be stored. */ line = new struct line_t[lineCount]; memset(line,0,lineCount * sizeof(struct line_t)); /* * Set an initial y to base generation off of, as generation references previous lines. */ line[0].y = GEN_INIT; /* * Populate every GEN_INCth line structure. The remaining lines will be based off of these. */ for(i = GEN_INC;i < lineCount;i += GEN_INC){ /* * Generate a y value, ensuring it stays within a reasonable range. */ line[i].y = rand() % 8 - 4 + line[i - GEN_INC].y; // Add +/- 4 to the previous line if(line[i].y < GEN_MIN)line[i].y = GEN_MIN; // Minimum bound else if(line[i].y > GEN_MAX)line[i].y = GEN_MAX; // Maximum bound } /* * Generate values for the remaining lines here. */ for(i = 0;i < lineCount - GEN_INC;i++){ /* * Every GEN_INCth line calculate the slope between the current line and the one * GEN_INC lines before it. This value is then divided into an increment that is * added to lines between these two points resulting in a smooth slope. * */ if(!(i % GEN_INC)) inc = (line[i + GEN_INC].y - line[i].y) / (float)GEN_INC; /* * Add the increment to create the smooth slope. */ else line[i].y = line[i - 1].y + inc; /* * Generate a color value for the line. This will be referenced in World->draw(), * by setting an RGB value of color (red), color - 50 (green), color - 100 (blue). */ line[i].color = rand() % 32 / 8; // 100 to 120 /* * Each line has two 'blades' of grass, here we generate random heights for them. */ line[i].gh[0] = (getRand() % 16) / 3.5 + 2; // Not sure what the range resolves to here... line[i].gh[1] = (getRand() % 16) / 3.5 + 2; // line[i].gs = true; // Show the blades of grass (modified by the player) } /* * Calculate the x coordinate to start drawing this world from so that it is centered at (0,0). */ x_start = 0 - getWidth(this) / 2; /* * Assign coordinates for the stars, seen at nighttime. */ for(i = 0;i < 100;i++){ star[i].x = getRand() % getTheWidth() - getTheWidth() / 2; star[i].y = getRand() % SCREEN_HEIGHT + 100; } } void World::generateFunc(unsigned int width,float(*func)(float)){ unsigned int i; /* * Check for a valid width. */ if((lineCount = width) <= 0) abort(); /* * Allocate memory for the line array. */ line = new struct line_t[lineCount]; memset(line,0,lineCount*sizeof(struct line_t)); /* * Populate entries in the line array, using `func` to get y values. */ for(i = 0;i < lineCount;i++){ line[i].y = func(i); if(line[i].y < 0)line[i].y = 0; // Minimum bound if(line[i].y > 2000)line[i].y = 2000; // Maximum bound line[i].color = rand() % 20 + 100; line[i].gh[0] = (getRand() % 16) / 3.5 + 2; line[i].gh[1] = (getRand() % 16) / 3.5 + 2; line[i].gs = true; } x_start = 0 - getWidth(this) / 2; for(i = 0;i < 100;i++){ star[i].x = getRand() % getTheWidth() - getTheWidth() / 2; star[i].y = getRand() % SCREEN_HEIGHT + 100; } } void World::update(Player *p,unsigned int delta){ /* * Update the player's coordinates. */ p->loc.y += p->vel.y * delta; p->loc.x +=(p->vel.x * p->speed) * delta; if(p->inv->usingi){ p->inv->useItem(); } /* * Update coordinates of all entities except for structures. */ for(auto &e : entity){ e->loc.y += e->vel.y * delta; if(e->type != STRUCTURET && e->canMove){ e->loc.x += e->vel.x * delta; if(e->vel.x < 0)e->left = true; else if(e->vel.x > 0)e->left = false; } } for(unsigned int i=0;ikill(deltaTime)){ delete particles[i]; particles.erase(particles.begin()+i); }else if(particles[i]->canMove){ particles[i]->loc.y += particles[i]->vely * deltaTime; particles[i]->loc.x += particles[i]->velx * deltaTime; for(auto &b : build){ if(b->bsubtype==FOUNTAIN){ if(particles[i]->loc.x >= b->loc.x && particles[i]->loc.x <= b->loc.x + b->width){ if(particles[i]->loc.y <= b->loc.y + b->height * .25){ delete particles[i]; particles.erase(particles.begin()+i); } } } } } } if(ui::dialogImportant){ Mix_FadeOutMusic(2000); }else if(!Mix_PlayingMusic()){ Mix_FadeInMusic(bgmObj,-1,2000); } } void World::setBGM(const char *path){ if(!bgm) delete[] bgm; if(path != NULL){ bgm = new char[strlen(path) + 1]; strcpy(bgm,path); bgmObj = Mix_LoadMUS(bgm); }else{ bgm = new char[1]; bgm[0] = '\0'; } } void World::bgmPlay(World *prev){ if(prev && strcmp(bgm,prev->bgm)){ Mix_FadeOutMusic(800); Mix_VolumeMusic(50); Mix_PlayMusic(bgmObj,-1); // Loop infinitely } } int worldShade = 0; extern vec2 offset; extern unsigned int tickCount; void World::draw(Player *p){ static float yoff=DRAW_Y_OFFSET; // Initialize stuff static int shade,bgshade; static World *current=this; unsigned int i; int is,ie,v_offset,cx_start,width; struct line_t *cline; float base; bgshade = worldShade << 1; // *2 base = worldGetYBase(this); /* * Draw the background images in the appropriate order. */ //LLLOOP: /*if(current->infront){ current=current->infront; goto LLLOOP; }*/ cx_start = current->x_start * 1.5; width = (-x_start) << 1; glEnable(GL_TEXTURE_2D); bgTex->bind(0); safeSetColorA(255,255,255,255 - worldShade * 4); glBegin(GL_QUADS); glTexCoord2i(0,0);glVertex2i( cx_start,SCREEN_HEIGHT); glTexCoord2i(1,0);glVertex2i(-cx_start,SCREEN_HEIGHT); glTexCoord2i(1,1);glVertex2i(-cx_start,0); glTexCoord2i(0,1);glVertex2i( cx_start,0); glEnd(); bgTex->bindNext(); safeSetColorA(255,255,255,worldShade * 4); glBegin(GL_QUADS); glTexCoord2i(0,0);glVertex2i( cx_start,SCREEN_HEIGHT); glTexCoord2i(1,0);glVertex2i(-cx_start,SCREEN_HEIGHT); glTexCoord2i(1,1);glVertex2i(-cx_start,0); glTexCoord2i(0,1);glVertex2i( cx_start,0); glEnd(); glDisable(GL_TEXTURE_2D); /* * Draws stars if it is an appropriate time of day for them. */ if((((weather==DARK )&(tickCount%DAY_CYCLE))DAY_CYCLE*.75) ){ if(tickCount % DAY_CYCLE){ // The above if statement doesn't check for exact midnight. safeSetColorA(255,255,255,bgshade + getRand() % 30 - 15); for(i = 0; i < 100; i++){ glRectf(star[i].x+offset.x*.9, star[i].y, star[i].x+offset.x*.9+HLINE, star[i].y+HLINE ); } } } glEnable(GL_TEXTURE_2D); /* * Draw the mountains. */ bgTex->bindNext(); safeSetColorA(150-bgshade,150-bgshade,150-bgshade,220); glBegin(GL_QUADS); for(int i = 0; i <= width/1920; i++){ glTexCoord2i(0,1);glVertex2i(width/-2+(1920*i )+offset.x*.85,base); glTexCoord2i(1,1);glVertex2i(width/-2+(1920*(i+1))+offset.x*.85,base); glTexCoord2i(1,0);glVertex2i(width/-2+(1920*(i+1))+offset.x*.85,base+1080); glTexCoord2i(0,0);glVertex2i(width/-2+(1920*i )+offset.x*.85,base+1080); } glEnd(); /* * Draw three layers of trees. */ for(i = 0; i < 3; i++){ bgTex->bindNext(); safeSetColorA(bgDraw[i][0]-bgshade,bgDraw[i][0]-bgshade,bgDraw[i][0]-bgshade,bgDraw[i][1]); glBegin(GL_QUADS); for(int j = cx_start; j <= -cx_start; j += 600){ glTexCoord2i(0,1);glVertex2i( j +offset.x*bgDraw[i][2],base); glTexCoord2i(1,1);glVertex2i((j+600)+offset.x*bgDraw[i][2],base); glTexCoord2i(1,0);glVertex2i((j+600)+offset.x*bgDraw[i][2],base+400); glTexCoord2i(0,0);glVertex2i( j +offset.x*bgDraw[i][2],base+400); } glEnd(); } glDisable(GL_TEXTURE_2D); glColor3ub(0,0,0); glRectf(cx_start,GEN_MIN,-cx_start,0); /* * World drawing is done recursively, meaning that this function jumps * back as many 'layers' as it can and then draws, eventually coming * back to the initial or 'root' layer. LOOP1 does the recursion back * to the furthest behind layer, modifying shade and y offsets as it * does. * */ current=this; shade=worldShade; //LOOP1: //if(current->behind){ /* * Add to the y offset and shade values (explained further below) * and recurse. * */ /*yoff+=DRAW_Y_OFFSET; shade+=DRAW_SHADE; current=current->behind; goto LOOP1; }*/ /* * Here is where the actual world drawing begins. A goto is made to * LOOP2 once the current layer is drawn and the function shifts to * draw the next closest layer. */ //LOOP2: /* * Calculate the offset in the line array that the player is (or would) * currently be on. This function then calculates reasonable values for * the 'for' loop below that draws the layer. */ v_offset=(offset.x + p->width / 2 - current->x_start) / HLINE; // is -> i start is=v_offset - (SCREEN_WIDTH / 2 / HLINE) - GEN_INC; if(is<0)is=0; // Minimum bound // ie -> i end ie=v_offset + (SCREEN_WIDTH / 2 / HLINE) + GEN_INC + HLINE; if(ie>(int)current->lineCount)ie=current->lineCount; // Maximum bound else if(ie < GEN_INC)ie = GEN_INC; /* * Make more direct variables for quicker referencing. */ cline =current->line; cx_start=current->x_start; /* * Invert shading if desired. */ shade*=-1; /* * Draw structures. We draw structures behind the dirt/grass so that the building's * corners don't stick out. */ for(auto &part : particles){if(part->behind)part->draw();} for(auto &b : current->build){ b->draw(); } /* * Draw the layer up until the grass portion, which is done later. */ bool hey=false; glEnable(GL_TEXTURE_2D); glActiveTexture(GL_TEXTURE0); bgTex->bindNext(); GLfloat pointArray[light.size() + (int)p->light][2]; for(uint w = 0; w < light.size(); w++){ pointArray[w][0] = light[w].loc.x - offset.x; pointArray[w][1] = light[w].loc.y; } glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT); //for the s direction glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT); //for the t direction glUseProgram(shaderProgram); glUniform1i(glGetUniformLocation(shaderProgram, "sampler"), 0); glUniform1f(glGetUniformLocation(shaderProgram, "amb"), float(shade+50.0f)/100.0f); if(p->light){ //glUniform1i(glGetUniformLocation(shaderProgram, "numLight"), 1); //glUniform2f(glGetUniformLocation(shaderProgram, "lightLocation"), p->loc.x - offset.x+SCREEN_WIDTH/2, p->loc.y); //glUniform3f(glGetUniformLocation(shaderProgram, "lightColor"), 1.0f,1.0f,1.0f); pointArray[light.size()+1][0] = (float)(p->loc.x + SCREEN_WIDTH/2); pointArray[light.size()+1][1] = (float)(p->loc.y); } if(light.size()+(int)p->light == 0){ glUniform1i(glGetUniformLocation(shaderProgram, "numLight"), 0); }else{ glUniform1i (glGetUniformLocation(shaderProgram, "numLight"), light.size()+(int)p->light); glUniform2fv(glGetUniformLocation(shaderProgram, "lightLocation"), light.size()+(int)p->light, (GLfloat *)&pointArray); glUniform3f (glGetUniformLocation(shaderProgram, "lightColor"), 1.0f,1.0f,1.0f); } glBegin(GL_QUADS); for(i=is;i<(unsigned)ie-GEN_INC;i++){ cline[i].y+=(yoff-DRAW_Y_OFFSET); // Add the y offset if(!cline[i].y){ cline[i].y=base; hey=true; glColor4ub(0,0,0,255); }else safeSetColorA(150+shade*2,150+shade*2,150+shade*2,255); glTexCoord2i(0,0);glVertex2i(cx_start+i*HLINE ,cline[i].y-GRASS_HEIGHT); glTexCoord2i(1,0);glVertex2i(cx_start+i*HLINE+HLINE,cline[i].y-GRASS_HEIGHT); glTexCoord2i(1,(int)(cline[i].y/64)+cline[i].color);glVertex2i(cx_start+i*HLINE+HLINE,0); glTexCoord2i(0,(int)(cline[i].y/64)+cline[i].color);glVertex2i(cx_start+i*HLINE ,0); cline[i].y-=(yoff-DRAW_Y_OFFSET); // Restore the line's y value if(hey){ hey=false; cline[i].y=0; } } glEnd(); glUseProgram(0); glDisable(GL_TEXTURE_2D); /* * Draw grass on every line. */ float cgh[2]; glEnable(GL_TEXTURE_2D); glActiveTexture(GL_TEXTURE0); grassT->bind(0); glUseProgram(shaderProgram); glUniform1i(glGetUniformLocation(shaderProgram, "sampler"), 0); //glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT); //for the s direction //glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT); //for the t direction //glBegin(GL_QUADS); for(i=is;i<(unsigned)ie-GEN_INC;i++){ /* * Load the current line's grass values */ if(cline[i].y)memcpy(cgh,cline[i].gh,2*sizeof(float)); else memset(cgh,0 ,2*sizeof(float)); /* * Flatten the grass if the player is standing on it. */ if(!cline[i].gs){ cgh[0]/=4; cgh[1]/=4; } /* * Actually draw the grass. */ cline[i].y+=(yoff-DRAW_Y_OFFSET); safeSetColorA(255,255,255,255); glBegin(GL_QUADS); glTexCoord2i(0,0);glVertex2i(cx_start+i*HLINE ,cline[i].y+cgh[0]); glTexCoord2i(1,0);glVertex2i(cx_start+i*HLINE+HLINE/2,cline[i].y+cgh[0]); glTexCoord2i(1,1);glVertex2i(cx_start+i*HLINE+HLINE/2,cline[i].y-GRASS_HEIGHT); glTexCoord2i(0,1);glVertex2i(cx_start+i*HLINE ,cline[i].y-GRASS_HEIGHT); glEnd(); glBegin(GL_QUADS); glTexCoord2i(0,0);glVertex2i(cx_start+i*HLINE+HLINE/2,cline[i].y+cgh[1]); glTexCoord2i(1,0);glVertex2i(cx_start+i*HLINE+HLINE ,cline[i].y+cgh[1]); glTexCoord2i(1,1);glVertex2i(cx_start+i*HLINE+HLINE ,cline[i].y-GRASS_HEIGHT); glTexCoord2i(0,1);glVertex2i(cx_start+i*HLINE+HLINE/2,cline[i].y-GRASS_HEIGHT); glEnd(); cline[i].y-=(yoff-DRAW_Y_OFFSET); } //glEnd(); glUseProgram(0); glDisable(GL_TEXTURE_2D); /* * Draw non-structure entities. */ for(auto &part : particles){if(!part->behind)part->draw();} for(auto &n : current->npc){ n->loc.y+=(yoff-DRAW_Y_OFFSET); n->draw(); n->loc.y-=(yoff-DRAW_Y_OFFSET); } for(auto &m : current->mob){ m->loc.y+=(yoff-DRAW_Y_OFFSET); m->draw(); m->loc.y-=(yoff-DRAW_Y_OFFSET); } for(auto &o : current->object){ if(o->alive){ o->loc.y+=(yoff-DRAW_Y_OFFSET); o->draw(); o->loc.y-=(yoff-DRAW_Y_OFFSET); } } /* * If we're drawing the closest/last world, handle and draw the player. */ if(current==this){ /* * Calculate the line that the player is on */ unsigned int ph = (p->loc.x + p->width / 2 - x_start) / HLINE; /* * If the player is on the ground, flatten the grass where the player is standing * by setting line.gs to false. */ if(p->ground==1){ for(i=0;i ph - 6 ) cline[i].gs=false; else cline[i].gs=true; } }else{ for(i=0;idraw(); } /* * Restore the inverted shading if it was inverted above. */ shade*=-1; /* * Draw the next closest world if it exists. */ /*if(current->infront){ yoff -= DRAW_Y_OFFSET; shade -= DRAW_SHADE; current=current->infront; goto LOOP2; }else{*/ /* * If finished, reset the yoff and shade variables for the next call. */ yoff=DRAW_Y_OFFSET; shade=0; //} } void World::singleDetect(Entity *e){ unsigned int i,j; int l; /* * Kill any dead entities. */ if(!e->alive||e->health<=0){ for(i=0;itype){ case STRUCTURET: std::cout<<"Killed a building..."<name<<"."<alive){ if(e->type == MOBT && Mobp(e)->subtype == MS_TRIGGER)return; /* * Calculate the line that this entity is currently standing on. */ l=(e->loc.x + e->width / 2 - x_start) / HLINE; if(l < 0) l=0; i = l; if(i > lineCount-1) i=lineCount-1; /* * If the entity is under the world/line, pop it back to the surface. */ if(e->loc.y < line[i].y){ /* * Check that the entity isn't trying to run through a wall. */ //if(e->loc.y + e->height > line[i-(int)e->width/2/HLINE].y && // e->loc.y + e->height > line[i+(int)e->width/2/HLINE].y ){ e->loc.y=line[i].y - .001 * deltaTime; e->ground=true; e->vel.y=0; //}else{ /* * Push the entity out of the wall if it's trying to go through it. */ /*do{ e->loc.x+=.001 * e->vel.x>0?-1:1; l=(e->loc.x - e->width / 2 - x_start) / HLINE; if(l < 0){ std::cout<<"push kill lol "<type<alive = false; return; } i = l; if(i > lineCount-1){ std::cout<<"push kill lol "<type<alive = false; return; } }while(line[i].y>e->loc.y+ e->height); }*/ /* * Handle gravity if the entity is above the line. */ }else{ if(e->type == STRUCTURET && e->loc.y > 2000){ e->loc.y = line[i].y; e->vel.y = 0; e->ground = true; return; }else if(e->vel.y > -2)e->vel.y-=.003 * deltaTime; } /* * Insure that the entity doesn't fall off either edge of the world. */ if(e->loc.xvel.x=0; e->loc.x=x_start + HLINE / 2; }else if(e->loc.x + e->width + HLINE > x_start + getWidth(this)){ // Right bound e->vel.x=0; e->loc.x=x_start + getWidth(this) - e->width - HLINE; } } } void World::detect(Player *p){ World *hey = this; /* * Handle the player. */ singleDetect(p); /* * Handle all remaining entities in this world. */ //LOOOOP: static int what = 0; for(auto &e : hey->entity) hey->singleDetect(e); for(auto &part : particles){ int l; unsigned int i; l=(part->loc.x + part->width / 2 - x_start) / HLINE; if(l < 0) l=0; i = l; if(i > lineCount-1) i=lineCount-1; if(part->loc.y < line[i].y){ part->loc.y = line[i].y; part->vely = 0; part->velx = 0; part->canMove = false; }else{ if(part->gravity && part->vely > -2)part->vely-=.003 * deltaTime; } what++; }what=0; /*if(hey->infront){ hey = hey->infront; goto LOOOOP; }*/ } void World::addStructure(BUILD_SUB sub, float x,float y,const char *inside){ build.push_back(new Structures()); build.back()->spawn(sub,x,y); if(inside) strcpy((build.back()->inside = new char[1 + strlen(inside)]),inside); else strcpy((build.back()->inside = new char[1]),"\0"); //strcpy((build.back()->outside = new char[1 + strlen((char *)(currentXML+4))]),(char *)(currentXML+4)); entity.push_back(build.back()); } void World::addVillage(int bCount, int npcMin, int npcMax,const char *inside){ std::cout << npcMin << ", " << npcMax << std::endl; //int xwasd; for(int i = 0; i < bCount; i++){ addStructure(HOUSE,x_start + (i * 300),100,inside); /*std::cout<<"1\n"; HERE: xwasd = (rand()%(int)x+1000*HLINE); for(auto &bu : build){ if(xwasd > bu->loc.x && xwasd < bu->loc.x+bu->width)goto HERE; } std::cout<<"2\n"; addStructure(t,HOUSE,xwasd,y,inside); std::cout<<"3\n";*/ } } void World::addMob(int t,float x,float y){ mob.push_back(new Mob(t)); mob.back()->spawn(x,y); entity.push_back(mob.back()); } void World::addMob(int t,float x,float y,void (*hey)(Mob *)){ mob.push_back(new Mob(t)); mob.back()->spawn(x,y); mob.back()->hey = hey; entity.push_back(mob.back()); } void World::addNPC(float x,float y){ npc.push_back(new NPC()); npc.back()->spawn(x,y); entity.push_back(npc.back()); } void World::addObject(ITEM_ID i, bool q, const char *p, float x, float y){ object.push_back(new Object(i,q, p)); object.back()->spawn(x,y); entity.push_back(object.back()); } void World::addParticle(float x, float y, float w, float h, float vx, float vy, Color color, int d){ particles.push_back(new Particles(x,y,w,h,vx,vy,color,d)); particles.back()->canMove = true; } void World::addLight(vec2 loc, Color color){ if(light.size() < 64){ light.push_back(Light()); light.back().loc = loc; light.back().color = color; } } /*void World::removeObject(Object i){ object.delete[](i); }*/ /* * The rest of these functions are explained well enough in world.h ;) */ /*void World::addLayer(unsigned int width){ if(behind){ behind->addLayer(width); return; } behind=new World(); behind->generate(width); behind->infront=this; behind->star=star; behind->bgmObj=bgmObj; behind->bgTex=bgTex; }*/ NPC *World::getAvailableNPC(void){ for(auto &n : npc){ if(n->aiFunc.empty()) return n; } return (NPC *)NULL; } char *World::setToLeft(const char *file){ if(toLeft) delete[] toLeft; if(!file) return (toLeft = NULL); strcpy((toLeft = new char[strlen(file) + 1]),file); return toLeft; } char *World::setToRight(const char *file){ if(toRight) delete[] toRight; if(!file) return (toRight = NULL); strcpy((toRight = new char[strlen(file) + 1]),file); return toRight; } World *World::goWorldLeft(Player *p){ World *tmp; if(toLeft && p->loc.x < x_start + HLINE * 15){ tmp = loadWorldFromXML(toLeft); p->loc.x = -tmp->x_start - HLINE * 10; p->loc.y = tmp->line[tmp->lineCount - 1].y; return tmp; } return this; } World *World::goWorldRight(Player *p){ World *tmp; if(toRight && p->loc.x + p->width > -x_start - HLINE * 15){ tmp = loadWorldFromXML(toRight); p->loc.x = tmp->x_start + HLINE * 10; p->loc.y = tmp->line[0].y; return tmp; } return this; } /*World *World::goWorldBack(Player *p){ if(behind&&p->loc.x>(int)(0-getWidth(behind)/2)&&p->loc.xloc.x>(int)(0-getWidth(infront)/2)&&p->loc.xinside; World *World::goInsideStructure(Player *p){ World *tmp; char *current; if(inside.empty()){ for(auto &b : build){ if(p->loc.x > b->loc.x && p->loc.x + p->width < b->loc.x + b->width ){ inside.push_back(new char[1 + strlen(currentXML)]); strcpy(inside.back(),(char *)(currentXML+4)); tmp = loadWorldFromXML(b->inside); ui::toggleBlackFast(); ui::waitForCover(); ui::toggleBlackFast(); return tmp; } } }else{ strcpy((current = new char[strlen((char *)(currentXML + 4)) + 1]),(char *)(currentXML + 4)); tmp = loadWorldFromXML(inside.back()); for(auto &b : tmp->build){ if(!strcmp(current,b->inside)){ p->loc.x = b->loc.x + (b->width / 2); delete[] inside.back(); inside.pop_back(); ui::toggleBlackFast(); ui::waitForCover(); ui::toggleBlackFast(); return tmp; } } delete[] current; } return this; } void World::addHole(unsigned int start,unsigned int end){ unsigned int i; for(i=start;iinfront){ hey=hey->infront; goto LOOP; }*/ return -hey->x_start*2; } void World::save(void){ static std::string data; std::string save = (std::string)currentXML + ".dat"; std::ofstream out (save,std::ios::out | std::ios::binary); std::cout<<"Saving to "<dialogIndex) + "\n"); data.append("dOnE\0"); out.write(data.c_str(),data.size()); out.close(); } #include extern int commonAIFunc(NPC *); void World::load(void){ std::string save,data,line; save = (std::string)currentXML + ".dat"; data = readFile(save.c_str()); std::istringstream iss (data); for(auto &n : npc){ std::getline(iss,line); if((n->dialogIndex = std::stoi(line)) != -1) n->addAIFunc(commonAIFunc,false); } while(std::getline(iss,line)){ if(line == "dOnE") break; std::cout<light){ glUniform1i(glGetUniformLocation(shaderProgram, "numLight"), 1); glUniform2f(glGetUniformLocation(shaderProgram, "lightLocation"), p->loc.x - offset.x+SCREEN_WIDTH/2, p->loc.y); glUniform3f(glGetUniformLocation(shaderProgram, "lightColor"), 1.0f,1.0f,1.0f); }else if(!light.size()){ glUniform1i(glGetUniformLocation(shaderProgram, "numLight"), 0); }else{ glUniform1i(glGetUniformLocation(shaderProgram, "numLight"), light.size()); glUniform2fv(glGetUniformLocation(shaderProgram, "lightLocation"), light.size(), (GLfloat *)&pointArray); glUniform3f(glGetUniformLocation(shaderProgram, "lightColor"), 1.0f,1.0f,1.0f); } bgTex->bind(0); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT); //for the s direction glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT); //for the t direction glColor4ub(255,255,255,255); glBegin(GL_QUADS); //for(j = x_start - SCREEN_WIDTH / 2;j < -x_start + SCREEN_WIDTH / 2; j += 512){ glTexCoord2i(0,1); glVertex2i( x_start - SCREEN_WIDTH / 2,0); glTexCoord2i((-x_start*2+SCREEN_WIDTH)/512,1);glVertex2i(-x_start + SCREEN_WIDTH / 2,0); glTexCoord2i((-x_start*2+SCREEN_WIDTH)/512,0);glVertex2i(-x_start + SCREEN_WIDTH / 2,SCREEN_HEIGHT); glTexCoord2i(0,0); glVertex2i( x_start - SCREEN_WIDTH / 2,SCREEN_HEIGHT); //} glEnd(); glUseProgram(0); glDisable(GL_TEXTURE_2D); /* * Calculate the starting and ending points to draw the ground from. */ /*v_offset = (p->loc.x - x_start) / HLINE; j = v_offset - (SCREEN_WIDTH / 2 / HLINE) - GEN_INC; if(j < 0)j = 0; i = j; ie = v_offset + (SCREEN_WIDTH / 2 / HLINE) - GEN_INC; if(ie > lineCount)ie = lineCount;*/ i = 0; ie = lineCount; /* * Draw the ground. */ glUseProgram(shaderProgram); glUniform1i(glGetUniformLocation(shaderProgram, "sampler"), 0); glBegin(GL_QUADS); for(;i < ie - GEN_INC;i++){ safeSetColor(150,100,50); x = x_start + i * HLINE; glVertex2i(x ,line[i].y); glVertex2i(x + HLINE,line[i].y); glVertex2i(x + HLINE,line[i].y - 50); glVertex2i(x ,line[i].y - 50); } glEnd(); glUseProgram(0); /* * Draw all entities. */ for(auto &part : particles) part->draw(); for(auto &e : entity) e->draw(); p->draw(); } extern bool inBattle; Arena::Arena(World *leave,Player *p){ generate(300); addMob(MS_DOOR,100,100); inBattle = true; exit = leave; pxy = p->loc; star = new vec2[100]; memset(star,0,100 * sizeof(vec2)); } Arena::~Arena(void){ delete bgTex; delete[] star; delete[] line; deleteEntities(); } World *Arena::exitArena(Player *p){ if(p->loc.x + p->width / 2 > mob[0]->loc.x && p->loc.x + p->width / 2 < mob[0]->loc.x + HLINE * 12 ){ inBattle = false; ui::toggleBlackFast(); ui::waitForCover(); p->loc = pxy; return exit; }else{ return this; } } #include using namespace tinyxml2; char *currentXML = NULL; extern World *currentWorld; World *loadWorldFromXML(const char *path){ XMLDocument xml; XMLElement *wxml; World *tmp; float spawnx; bool dialog,Indoor; const char *ptr,*name; unsigned int size = 5 + strlen(path); if(currentXML){ currentWorld->save(); delete[] currentXML; } memset((currentXML = new char[size]),0,size); strcpy(currentXML,"xml/"); strcat(currentXML,path); //std::cout<FirstChildElement(); Indoor = false; tmp = new World(); }else if((wxml = xml.FirstChildElement("IndoorWorld"))){ wxml = wxml->FirstChildElement(); Indoor = true; tmp = new IndoorWorld(); } while(wxml){ name = wxml->Name(); if(!strcmp(name,"link")){ if((ptr = wxml->Attribute("left"))) tmp->setToLeft(ptr); else if((ptr = wxml->Attribute("right"))) tmp->setToRight(ptr); else abort(); }else if(!strcmp(name,"style")){ tmp->setBackground((WORLD_BG_TYPE)wxml->UnsignedAttribute("background")); tmp->setBGM(wxml->Attribute("bgm")); }else if(!strcmp(name,"generation")){ if(!strcmp(wxml->Attribute("type"),"Random")){ if(Indoor) ((IndoorWorld *)tmp)->generate(wxml->UnsignedAttribute("width")); else tmp->generate(wxml->UnsignedAttribute("width")); }else if(Indoor){ abort(); } }else if(!strcmp(name,"mob")){ unsigned int type; type = wxml->UnsignedAttribute("type"); if(wxml->QueryFloatAttribute("x",&spawnx) != XML_NO_ERROR) tmp->addMob(type,getRand() % tmp->getTheWidth() / 2,100); else tmp->addMob(type,spawnx,wxml->FloatAttribute("y")); }else if(!strcmp(name,"npc")){ const char *npcname; if(wxml->QueryFloatAttribute("x",&spawnx) != XML_NO_ERROR) tmp->addNPC(getRand() % tmp->getTheWidth() / 2.0f,100); else tmp->addNPC(spawnx,wxml->FloatAttribute("y")); if((npcname = wxml->Attribute("name"))){ delete[] tmp->npc.back()->name; tmp->npc.back()->name = new char[strlen(npcname) + 1]; strcpy(tmp->npc.back()->name,npcname); } dialog = false; if(wxml->QueryBoolAttribute("hasDialog",&dialog) == XML_NO_ERROR && dialog) tmp->npc.back()->addAIFunc(commonAIFunc,false); else tmp->npc.back()->dialogIndex = -1; }else if(!strcmp(name,"structure")){ ptr = wxml->Attribute("inside"); if(wxml->QueryFloatAttribute("x",&spawnx) != XML_NO_ERROR) tmp->addStructure((BUILD_SUB)wxml->UnsignedAttribute("type"),getRand() % tmp->getTheWidth() / 2.0f,100,ptr); else tmp->addStructure((BUILD_SUB)wxml->UnsignedAttribute("type"),spawnx,wxml->FloatAttribute("y"),ptr); } wxml = wxml->NextSiblingElement(); } std::ifstream dat (((std::string)currentXML + ".dat").c_str()); if(dat.good()){ dat.close(); tmp->load(); } return tmp; }