#include /* ---------------------------------------------------------------------------- ** Includes section ** --------------------------------------------------------------------------*/ // standard library headers #include #include #include #include #include // local game headers #include #include #include #include #include // local library headers #include using namespace tinyxml2; void makeWorldDrawingSimplerEvenThoughAndyDoesntThinkWeCanMakeItIntoFunctions( unsigned size, void *coordAddr, void *texAddr, unsigned triCount ) { glVertexAttribPointer(Render::worldShader.coord, 3, GL_FLOAT, GL_FALSE, size, coordAddr); glVertexAttribPointer(Render::worldShader.tex , 2, GL_FLOAT, GL_FALSE, size, texAddr ); glDrawArrays(GL_TRIANGLES, 0, triCount); } void makeWorldDrawingSimplerEvenThoughAndyDoesntThinkWeCanMakeItIntoFunctions_JustDrawThis( unsigned size, void *coordAddr, void *texAddr, unsigned triCount ) { Render::worldShader.enable(); makeWorldDrawingSimplerEvenThoughAndyDoesntThinkWeCanMakeItIntoFunctions(size, coordAddr, texAddr, triCount); Render::worldShader.disable(); } /* ---------------------------------------------------------------------------- ** Variables section ** --------------------------------------------------------------------------*/ // external variables extern Player *player; // main.cpp? extern World *currentWorld; // main.cpp extern World *currentWorldToLeft; // main.cpp extern World *currentWorldToRight; // main.cpp extern bool inBattle; // ui.cpp? extern std::string xmlFolder; // particle mutex std::mutex partMutex; // externally referenced in main.cpp int worldShade = 0; // ground-generating constants constexpr const float GROUND_HEIGHT_INITIAL = 80.0f; constexpr const float GROUND_HEIGHT_MINIMUM = 60.0f; constexpr const float GROUND_HEIGHT_MAXIMUM = 110.0f; constexpr const float GROUND_HILLINESS = 10.0f; // defines grass height in HLINEs constexpr const unsigned int GRASS_HEIGHT = 4; // the path of the currently loaded XML file, externally referenced in places std::string currentXML; // keeps track of information of worlds the player has left to enter arenas static std::vector arenaNest; // pathnames of images for world themes constexpr const unsigned int BG_PATHS_ENTRY_SIZE = 9; static const std::string bgPaths[][BG_PATHS_ENTRY_SIZE] = { {"bg.png", // Daytime background "bgn.png", // Nighttime background "bgFarMountain.png", // Furthest layer "forestTileFar.png", // Furthest away Tree Layer "forestTileBack.png", // Closer layer "forestTileMid.png", // Near layer "forestTileFront.png", // Closest layer "dirt.png", // Dirt "grass.png"}, // Grass {"bgWoodTile.png", "bgWoodTile.png", "bgWoodTile.png", "bgWoodTile.png", "bgWoodTile.png", "bgWoodTile.png", "bgWoodTile.png", "bgWoodTile.png"} }; // pathnames of structure textures static const std::string buildPaths[] = { "townhall.png", "house1.png", "house2.png", "house1.png", "house1.png", "fountain1.png", "lampPost1.png", "brazzier.png" }; // alpha-related values used for world drawing? nobody knows... static const float bgDraw[4][3]={ { 100, 240, 0.6 }, { 150, 250, 0.4 }, { 200, 255, 0.25 }, { 255, 255, 0.1 } }; std::string currentXMLRaw; XMLDocument currentXMLDoc; /* ---------------------------------------------------------------------------- ** Functions section ** --------------------------------------------------------------------------*/ /** * Creates a world object. * Note that all this does is nullify a pointer... */ World:: World(void) { worldStart = 0; lineCount = 0; } /** * The world destructor. * This will free objects used by the world itself, then free the vectors of * entity-related objects. */ World:: ~World(void) { deleteEntities(); } /** * The entity vector destroyer. * This function will free all memory used by all entities, and then empty the * vectors they were stored in. */ template void clearPointerVector(T &vec) { while (!vec.empty()) { delete vec.back(); vec.pop_back(); } } void World:: deleteEntities(void) { // free particles particles.clear(); // clear light array light.clear(); // free villages village.clear(); // clear entity array clearPointerVector(entity); } /** * Generates a world of the specified width. * This will mainly populate the WorldData array, preparing most of the world * object for usage. */ void World:: generate(int width) { float geninc = 0; // check for valid width if (width <= 0) UserError("Invalid world dimensions"); // allocate space for world worldData = std::vector (width + GROUND_HILLINESS, WorldData { false, {0, 0}, 0, 0 }); lineCount = worldData.size(); // prepare for generation worldData.front().groundHeight = GROUND_HEIGHT_INITIAL; auto wditer = std::begin(worldData) + GROUND_HILLINESS; // give every GROUND_HILLINESSth entry a groundHeight value for (; wditer < std::end(worldData); wditer += GROUND_HILLINESS) wditer[-static_cast(GROUND_HILLINESS)].groundHeight = wditer[0].groundHeight + (randGet() % 8 - 4); // create slopes from the points that were just defined, populate the rest of the WorldData structure for (wditer = std::begin(worldData) + 1; wditer < std::end(worldData); wditer++){ auto w = &*(wditer); if (w->groundHeight != 0) geninc = (w[static_cast(GROUND_HILLINESS)].groundHeight - w->groundHeight) / GROUND_HILLINESS; w->groundHeight = std::clamp(w[-1].groundHeight + geninc, GROUND_HEIGHT_MINIMUM, GROUND_HEIGHT_MAXIMUM); w->groundColor = randGet() % 32 / 8; w->grassUnpressed = true; w->grassHeight[0] = (randGet() % 16) / 3 + 2; w->grassHeight[1] = (randGet() % 16) / 3 + 2; } // define x-coordinate of world's leftmost 'line' worldStart = (width - GROUND_HILLINESS) * game::HLINE / 2 * -1; // create empty star array, should be filled here as well... star = std::vector (100, vec2 { 0, 400 }); for (auto &s : star) { s.x = (randGet() % (static_cast(-worldStart) * 2)) + worldStart; s.y = (randGet() % game::SCREEN_HEIGHT) + 100; } } static Color ambient; void World::draw(Player *p) { auto HLINE = game::HLINE; uint ls = light.size(); GLfloat *lightCoords = new GLfloat[ls * 4]; GLfloat *lightColors = new GLfloat[ls * 4]; uint lpIndex = 0; uint lcIndex = 0; static bool ambientUpdaterStarted = false; if (!ambientUpdaterStarted) { ambientUpdaterStarted = true; std::thread([&](void) { while (true) { float v = 75 * sin((game::time::getTickCount() + (DAY_CYCLE / 2)) / (DAY_CYCLE / PI)); float rg = std::clamp(.5f + (-v / 100.0f), 0.01f, .9f); float b = std::clamp(.5f + (-v / 80.0f), 0.03f, .9f); ambient = Color(rg, rg, b, 1.0f); std::this_thread::sleep_for(std::chrono::milliseconds(1)); } }).detach(); } for (uint i = 0; i < ls; i++) { auto &l = light[i]; if (l.belongsTo) { l.loc.x = l.following->loc.x; l.loc.y = l.following->loc.y; } if (l.flame) { l.fireFlicker = 0.9f + ((rand()% 2 ) / 10.0f); l.fireLoc.x = l.loc.x + (rand() % 2 - 1) * 3; l.fireLoc.y = l.loc.y + (rand() % 2 - 1) * 3; } else { l.fireFlicker = 1; } lightCoords[lpIndex++] = l.loc.x; lightCoords[lpIndex++] = l.loc.y; lightCoords[lpIndex++] = 0.0; lightCoords[lpIndex++] = l.radius; lightColors[lcIndex++] = l.color.red; lightColors[lcIndex++] = l.color.green; lightColors[lcIndex++] = l.color.blue; lightColors[lcIndex++] = 1.0; } Render::worldShader.use(); glUniform4fv(Render::worldShader.uniform[WU_light], ls, lightCoords); glUniform4fv(Render::worldShader.uniform[WU_light_color], ls, lightColors); glUniform1i(Render::worldShader.uniform[WU_light_size], ls); Render::worldShader.unuse(); for (auto &e :entity) e->draw(); // flatten grass under the player if the player is on the ground if (p->ground) { int pOffset = (p->loc.x + p->width / 2 - worldStart) / HLINE; for (unsigned int i = 0; i < worldData.size(); i++) worldData[i].grassUnpressed = !(i < static_cast(pOffset + 6) && i > static_cast(pOffset - 6)); } else { for (auto &wd : worldData) wd.grassUnpressed = true; } // draw the player //p->draw(); // draw particles like a MASTAH glBindTexture(GL_TEXTURE_2D, colorIndex); glUniform1i(Render::worldShader.uniform[WU_texture], 0); Render::worldShader.use(); glUniform4f(Render::worldShader.uniform[WU_tex_color], 1.0, 1.0, 1.0, .8); Render::worldShader.enable(); partMutex.lock(); uint ps = particles.size(); uint pss = ps * 6 * 5; uint pc = 0; std::vector partVec(pss); auto *pIndex = &partVec[0]; for (uint i = 0; i < ps; i++) { pc += 30; if (pc > pss) { // TODO resize the vector or something better than breaking break; } particles[i].draw(pIndex); } partMutex.unlock(); glVertexAttribPointer(Render::worldShader.coord, 3, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &partVec[0]); glVertexAttribPointer(Render::worldShader.tex, 2, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &partVec[3]); glDrawArrays(GL_TRIANGLES, 0, ps * 6); Render::worldShader.disable(); glUniform4f(Render::worldShader.uniform[WU_tex_color], 1.0, 1.0, 1.0, 1.0); Render::worldShader.unuse(); } /** * Get's the world's width in pixels. */ int World:: getTheWidth(void) const { return (worldStart * -2); } float World:: getWorldStart(void) const { return static_cast(worldStart); } /** * Get a pointer to the most recently created light. * Meant to be non-constant. */ Light& World:: getLastLight(void) { return light.back(); } /** * Get a pointer to the most recently created mob. * Meant to be non-constant. */ Mob* World:: getLastMob(void) { for (auto e = entity.rbegin(); e != entity.rend(); ++e) { if ((*e)->type == MOBT) return dynamic_cast(*e); } return nullptr; } /** * Get the interactable entity that is closest to the entity provided. */ Entity* World:: getNearInteractable(Entity &e) { auto n = std::find_if(std::begin(entity), std::end(entity), [&](Entity *&a) { return ((a->type == MOBT) || (a->type == NPCT) || a->type == MERCHT) && e.isNear(a) && (e.left ? (a->loc.x < e.loc.x) : (a->loc.x > e.loc.x)); }); return n == std::end(entity) ? nullptr : *n; } Mob* World:: getNearMob(Entity &e) { auto n = std::find_if(std::begin(entity), std::end(entity), [&](Entity *a) { return (a->type == MOBT && e.isNear(a) && (e.left ? (a->loc.x < e.loc.x + e.width / 2) : (a->loc.x + a->width > e.loc.x + e.width / 2))); }); return (n == std::end(entity)) ? nullptr : dynamic_cast(*n); } /** * Get the file path for the `index`th building. */ std::string World:: getSTextureLocation(unsigned int index) const { return index < sTexLoc.size() ? sTexLoc[ index ] : ""; } /** * Get the coordinates of the `index`th building, with -1 meaning the last building. */ vec2 World:: getStructurePos(int index) { if (index < 0) { for (auto e = entity.rbegin(); e != entity.rend(); ++e) { if ((*e)->type == STRUCTURET) return (*e)->loc; } return vec2 {0, 0}; } int nth = 0; for (const auto &e : entity) { if (e->type == STRUCTURET) { if (index == nth) return e->loc; else ++nth; } } return vec2 {0, 0}; } /** * Saves world data to a file. */ void World::save(const std::string& s) { for (const auto &e : entity) e->saveToXML(); currentXMLDoc.SaveFile((s.empty() ? currentXML : xmlFolder + s).c_str(), false); } /** * Sets the desired theme for the world's background. * The images chosen for the background layers are selected depending on the * world's background type. */ void World::setBackground(WorldBGType bgt) { bgType = bgt; } /** * Sets the world's style. * The world's style will determine what sprites are used for things like\ * generic structures. */ void World::setStyle(std::string pre) { // get folder prefix std::string prefix = pre.empty() ? "assets/style/classic/" : pre; styleFolder = prefix + "bg/"; for (const auto &s : buildPaths) sTexLoc.push_back(prefix + s); } /** * Pretty self-explanatory. */ std::string World::setToLeft(std::string file) { return (toLeft = file); } /** * Pretty self-explanatory. */ std::string World::setToRight(std::string file) { return (toRight = file); } /** * Pretty self-explanatory. */ std::string World::getToLeft(void) const { return toLeft; } /** * Pretty self-explanatory. */ std::string World::getToRight(void) const { return toRight; } /** * Attempts to go to the left world, returning either that world or itself. */ WorldSwitchInfo World::goWorldLeft(Player *p) { World *tmp; // check if player is at world edge if (!toLeft.empty() && p->loc.x < worldStart + HLINES(15)) { // load world (`toLeft` conditional confirms existance) tmp = loadWorldFromPtr(currentWorldToLeft); // return pointer and new player coords return std::make_pair(tmp, vec2 {tmp->worldStart + tmp->getTheWidth() - (float)HLINES(15), tmp->worldData[tmp->lineCount - 1].groundHeight}); } return std::make_pair(this, vec2 {0, 0}); } /** * Attempts to go to the right world, returning either that world or itself. */ WorldSwitchInfo World::goWorldRight(Player *p) { World *tmp; if (!toRight.empty() && p->loc.x + p->width > -worldStart - HLINES(15)) { tmp = loadWorldFromPtr(currentWorldToRight); return std::make_pair(tmp, vec2 {tmp->worldStart + (float)HLINES(15.0), GROUND_HEIGHT_MINIMUM} ); } return std::make_pair(this, vec2 {0, 0}); } void World::adoptNPC(NPC *e) { entity.push_back(e); } void World::adoptMob(Mob* e) { entity.push_back(e); } /** * Acts like goWorldLeft(), but takes an NPC; returning true on success. */ bool World::goWorldLeft(NPC *e) { // check if entity is at world edge if (!toLeft.empty() && e->loc.x < worldStart + HLINES(15)) { currentWorldToLeft->adoptNPC(e); entity.erase(std::find(std::begin(entity), std::end(entity), e)); e->loc.x = currentWorldToLeft->worldStart + currentWorldToLeft->getTheWidth() - HLINES(15); e->loc.y = GROUND_HEIGHT_MAXIMUM; ++e->outnabout; return true; } return false; } bool World::goWorldRight(NPC *e) { if (!toRight.empty() && e->loc.x + e->width > -worldStart - HLINES(15)) { currentWorldToRight->adoptNPC(e); entity.erase(std::find(std::begin(entity), std::end(entity), e)); e->loc.x = currentWorldToRight->worldStart + HLINES(15); e->loc.y = GROUND_HEIGHT_MINIMUM; --e->outnabout; return true; } return false; } /** * Attempts to enter a building that the player is standing in front of. */ WorldSwitchInfo World::goInsideStructure(Player *p) { World *tmp; static std::string outdoorData, outdoorName; // enter a building if (outdoorName.empty()) { auto d = std::find_if(std::begin(entity), std::end(entity), [p](const Entity *s) { return ((p->loc.x > s->loc.x) && (p->loc.x + p->width < s->loc.x + s->width)); }); if ((d == std::end(entity)) || dynamic_cast(*d)->inside.empty()) return std::make_pair(this, vec2 {0, 0}); outdoorData = currentXMLRaw; outdoorName = currentXML; currentXML = xmlFolder + dynamic_cast(*d)->inside; const char *buf = readFile(currentXML.c_str()); currentXMLRaw = buf; delete[] buf; tmp = dynamic_cast(*d)->insideWorld; return std::make_pair(tmp, vec2 {0, 100}); } // exit the building else { std::string current = ¤tXML[xmlFolder.size()]; currentXML = outdoorName; currentXMLRaw = outdoorData; outdoorName.clear(); outdoorData.clear(); tmp = dynamic_cast(currentWorld)->outside; //loadWorldFromXML(inside.back()); Structures *b = nullptr; for (auto &s : tmp->entity) { if (s->type == STRUCTURET && dynamic_cast(s)->inside == current) { b = dynamic_cast(s); break; } } if (b == nullptr) return std::make_pair(this, vec2 {0, 0}); return std::make_pair(tmp, vec2 {b->loc.x + (b->width / 2), 0}); } return std::make_pair(this, vec2 {0, 0}); } void World:: addStructure(Structures *s) { entityPending.push_back(s); } Village *World:: addVillage(std::string name, World *world) { village.emplace_back(name, world); return &village.back(); } void World::addMob(Mob *m, vec2 coord) { m->spawn(coord.x, coord.y); entityPending.push_back(m); } void World:: addNPC(NPC *n) { entityPending.push_back(n); } void World:: addMerchant(float x, float y, bool housed) { Merchant *tmp = new Merchant(); tmp->spawn(x, y); if (housed) { tmp->inside = dynamic_cast(*std::find_if(entity.rbegin(), entity.rend(), [&](Entity *e){ return (e->type == STRUCTURET); })); tmp->z = tmp->inside->z + 0.1f; } entityPending.push_back(tmp); } void World:: addObject(std::string in, std::string p, float x, float y) { Object *tmp = new Object(in, p); tmp->spawn(x, y); entityPending.push_back(tmp); } void World:: addParticle(float x, float y, float w, float h, float vx, float vy, Color color, int d) { particles.push_back(Particles(x, y, w, h, vx, vy, color, d)); particles.back().canMove = true; } void World:: addParticle(float x, float y, float w, float h, float vx, float vy, Color color, int d, unsigned char flags) { particles.push_back(Particles(x, y, w, h, vx, vy, color, d)); particles.back().canMove = true; particles.back().gravity = flags & (1 << 0); particles.back().bounce = flags & (1 << 1); } void World:: addLight(vec2 loc, float radius, Color color) { if (light.size() < 128) light.emplace_back(loc, radius, color); } void World:: addHole(unsigned int start, unsigned int end) { end = fmin(worldData.size(), end); 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 = 0; const float thing = peak.y - worldData[std::clamp(start, 0, static_cast(lineCount))].groundHeight; const float period = PI / width; if (start < 0) { offset = -start; start = 0; } end = fmin(worldData.size(), end); for (int i = start; i < end; i++) { worldData[i].groundHeight += thing * sin((i - start + offset) * period); if (worldData[i].groundHeight > peak.y) worldData[i].groundHeight = peak.y; } } IndoorWorld::IndoorWorld(void) { } IndoorWorld::~IndoorWorld(void) { deleteEntities(); } void IndoorWorld:: addFloor(unsigned int width) { if (floor.empty()) generate(width); floor.emplace_back(width, floor.size() * INDOOR_FLOOR_HEIGHT); fstart.push_back(0); } void IndoorWorld:: addFloor(unsigned int width, unsigned int start) { if (floor.empty()) generate(width); floor.emplace_back(width, floor.size() * INDOOR_FLOOR_HEIGHT); fstart.push_back(start); } bool IndoorWorld:: moveToFloor(Entity *e, unsigned int _floor) { if (_floor > floor.size()) return false; e->loc.y = floor[_floor - 1][0]; return true; } bool IndoorWorld:: isFloorAbove(Entity *e) { for (unsigned int i = 0; i < floor.size(); i++) { if (floor[i][0] + INDOOR_FLOOR_HEIGHTT - 100 > e->loc.y) return (i + 1) != floor.size(); } return false; } bool IndoorWorld:: isFloorBelow(Entity *e) { for (unsigned int i = 0; i < floor.size(); i++) { if (floor[i][0] + INDOOR_FLOOR_HEIGHTT - 100 > e->loc.y) return i > 0; } return false; } void IndoorWorld:: singleDetect(Entity *e) { unsigned int floornum = 0; float start, end; if (!e->isAlive()) return; for (; floornum < floor.size(); floornum++) { if (floor[floornum][0] + INDOOR_FLOOR_HEIGHTT - 100 > e->loc.y) { if (e->loc.y < floor[floornum][0]) { e->loc.y = floor[floornum][0]; e->vel.y = 0; e->ground = true; } break; } } if (e->vel.y > -2) e->vel.y -= GRAVITY_CONSTANT * game::time::getDeltaTime(); if (e->ground) { e->loc.y = ceil(e->loc.y); e->vel.y = 0; } start = worldStart + HLINES(fstart[floornum]); end = start + HLINES(floor[floornum].size()); if (e->loc.x < start) { e->vel.x = 0; e->loc.x = start + game::HLINE / 2; } else if (e->loc.x + e->width + game::HLINE > end) { e->vel.x = 0; e->loc.x = end - e->width - game::HLINE; } } void IndoorWorld:: draw(Player *p) { unsigned int i,fl; int x; auto SCREEN_WIDTH = game::SCREEN_WIDTH; auto SCREEN_HEIGHT = game::SCREEN_HEIGHT; auto HLINE = game::HLINE; // draw lights for (auto &l : light) { if (l.belongsTo) { l.loc.x = l.following->loc.x + SCREEN_WIDTH / 2; l.loc.y = (l.following->loc.y > SCREEN_HEIGHT / 2) ? SCREEN_HEIGHT / 2 : l.following->loc.y; } if (l.flame) { l.fireFlicker = .9 + ((rand() % 2) / 10.0f); l.fireLoc.x = l.loc.x + (rand() % 2 - 1) * 3; l.fireLoc.y = l.loc.y + (rand() % 2 - 1) * 3; } else l.fireFlicker = 1.0f; } /* std::unique_ptr pointArrayBuf = std::make_unique (2 * (light.size())); auto pointArray = pointArrayBuf.get(); GLfloat flameArray[64]; for (i = 0; i < light.size(); i++) { if (light[i].flame) { pointArray[2 * i ] = light[i].fireLoc.x - offset.x; pointArray[2 * i + 1] = light[i].fireLoc.y; }else{ pointArray[2 * i ] = light[i].loc.x - offset.x; pointArray[2 * i + 1] = light[i].loc.y; } } for(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); glUseProgram(shaderProgram); glUniform1i(glGetUniformLocation(shaderProgram, "sampler"), 0); glUniform1f(glGetUniformLocation(shaderProgram, "amb"), 0.02f + light.size()/50.0f); if (light.empty()) glUniform1i(glGetUniformLocation(shaderProgram, "numLight"), 0); else { glUniform1i (glGetUniformLocation(shaderProgram, "numLight"), light.size()); glUniform2fv(glGetUniformLocation(shaderProgram, "lightLocation"), light.size(), pointArray); glUniform3f (glGetUniformLocation(shaderProgram, "lightColor"), 1.0f, 1.0f, 1.0f); glUniform1fv(glGetUniformLocation(shaderProgram, "fireFlicker"), light.size(), flameArray); } */ Render::worldShader.use(); glActiveTexture(GL_TEXTURE0); //game::engine.getSystembgTex(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 glUniform1i(Render::worldShader.uniform[WU_texture], 0); glUniform4f(Render::worldShader.uniform[WU_tex_color], 1.0, 1.0, 1.0, 1.0); GLfloat backTile[] = {worldStart - SCREEN_WIDTH / 2, 0, 9.9, -worldStart + SCREEN_WIDTH / 2, 0, 9.9, -worldStart + SCREEN_WIDTH / 2, static_cast(SCREEN_HEIGHT), 9.9, -worldStart + SCREEN_WIDTH / 2, static_cast(SCREEN_HEIGHT), 9.9, worldStart - SCREEN_WIDTH / 2, static_cast(SCREEN_HEIGHT), 9.9, worldStart - SCREEN_WIDTH / 2, 0, 9.9}; GLfloat backTile_tex[] = {0, 1, (-worldStart*2+SCREEN_WIDTH)/512, 1, (-worldStart*2+SCREEN_WIDTH)/512, 0, (-worldStart*2+SCREEN_WIDTH)/512, 0, 0, 0, 0, 1}; makeWorldDrawingSimplerEvenThoughAndyDoesntThinkWeCanMakeItIntoFunctions_JustDrawThis(0, backTile, backTile_tex, 6); glUseProgram(0); /* * Draw the ground. */ // TODO make floor texture static GLuint floorTex = Texture::genColor(Color(150, 100, 50)); glBindTexture(GL_TEXTURE_2D, floorTex); Render::worldShader.use(); Render::useShader(&Render::worldShader); for (fl = 0; fl < floor.size(); fl++) { i = 0; for (const auto &h : floor[fl]) { x = worldStart + HLINES(fstart[fl] + i); Render::drawRect(vec2 {(float)x, h}, vec2 {(float)(x + HLINE), h - INDOOR_FLOOR_THICKNESS}, -3.0f); i++; } } Render::worldShader.unuse(); /* * Draw all entities. */ // TODO draw particles // glBindTexture(GL_TEXTURE_2D, colorIndex); /*for (auto &part : particles) part.draw();*/ for (auto &e : entity) e->draw(); p->draw(); } Arena::Arena(void) { generate(800); addMob(new Door(), vec2 {100, 100}); mmob = nullptr; } Arena::~Arena(void) { if (mmob != nullptr) mmob->die(); deleteEntities(); } void Arena::fight(World *leave, const Player *p, Mob *m) { inBattle = true; entity.push_back((mmob = m)); mmob->aggressive = false; arenaNest.emplace_back(leave, p->loc); } WorldSwitchInfo Arena::exitArena(Player *p) { if (!mmob->isAlive() && p->loc.x + p->width / 2 > mmob->loc.x && p->loc.x + p->width / 2 < mmob->loc.x + HLINES(12)) { auto ret = arenaNest.back(); arenaNest.pop_back(); inBattle = !(arenaNest.empty()); return ret; } return std::make_pair(this, vec2 {0, 0}); } static bool loadedLeft = false; static bool loadedRight = false; World *loadWorldFromXML(std::string path) { if (!currentXML.empty()) currentWorld->save(); return loadWorldFromXMLNoSave(path); } World *loadWorldFromPtr(World *ptr) { currentWorld->save(); // save the current world to the current xml path if (ptr->getToLeft() == currentXML) { currentWorldToLeft = currentWorld; loadedRight = true; currentWorldToRight = loadWorldFromXMLNoSave(ptr->getToRight()); loadedRight = false; } else if (ptr->getToRight() == currentXML) { currentWorldToRight = currentWorld; loadedLeft = true; currentWorldToLeft = loadWorldFromXMLNoSave(ptr->getToLeft()); loadedLeft = false; } return ptr; } /** * Loads a world from the given XML file. */ World * loadWorldFromXMLNoTakeover(std::string path) { loadedLeft = true, loadedRight = true; auto ret = loadWorldFromXMLNoSave(path); loadedLeft = false, loadedRight = false; return ret; } World * loadWorldFromXMLNoSave(std::string path) { XMLDocument *_currentXMLDoc; static std::string _currentXML, _currentXMLRaw; XMLElement *wxml; XMLElement *vil; World *tmp; Entity *newEntity; float spawnx; bool Indoor; const char *ptr; std::string name, sptr; // no file? -> no world if (path.empty()) return nullptr; _currentXML = xmlFolder + path; const char *worthless = readFile(_currentXML.c_str()); _currentXMLRaw = worthless; delete[] worthless; // create a temporary XMLDocument if this isn't the main world if (!loadedLeft && !loadedRight) _currentXMLDoc = ¤tXMLDoc; else _currentXMLDoc = new XMLDocument(); // parse the file if (_currentXMLDoc->Parse(_currentXMLRaw.data()) != XML_NO_ERROR) UserError("XML Error: Failed to parse file (not your fault though..?)"); // attempt to load a tag if ((wxml = _currentXMLDoc->FirstChildElement("World"))) { wxml = wxml->FirstChildElement(); vil = _currentXMLDoc->FirstChildElement("World")->FirstChildElement("village"); tmp = new World(); Indoor = false; } // attempt to load an tag else if ((wxml = _currentXMLDoc->FirstChildElement("IndoorWorld"))) { wxml = wxml->FirstChildElement(); vil = NULL; tmp = new IndoorWorld(); Indoor = true; } // error: can't load a world... else UserError("XML Error: Cannot find a or tag in " + _currentXML + "!"); // iterate through world tags while (wxml) { newEntity = nullptr; name = wxml->Name(); // world linkage if (name == "link") { // links world to the left if ((ptr = wxml->Attribute("left"))) { tmp->setToLeft(ptr); // load the left world if it isn't if (!loadedLeft) { loadedRight = true; currentWorldToLeft = loadWorldFromXMLNoSave(ptr); loadedRight = false; } else { currentWorldToLeft = nullptr; } } // links world to the right else if ((ptr = wxml->Attribute("right"))) { tmp->setToRight(ptr); // load the right world if it isn't if (!loadedRight) { loadedLeft = true; currentWorldToRight = loadWorldFromXMLNoSave(ptr); loadedLeft = false; } else { currentWorldToRight = nullptr; } } // tells what world is outside, if in a structure else if (Indoor && (ptr = wxml->Attribute("outside"))) { // if (!loadedLeft && !loadedRight) // inside.push_back(ptr); } // error, invalid link tag else { UserError("XML Error: Invalid tag in " + _currentXML + "!"); } } // style tags else if (name == "style") { // set style folder tmp->setStyle(wxml->StrAttribute("folder")); // set background folder unsigned int bgt; if (wxml->QueryUnsignedAttribute("background", &bgt) != XML_NO_ERROR) UserError("XML Error: No background given in