#include /* ---------------------------------------------------------------------------- ** Includes section ** --------------------------------------------------------------------------*/ // standard library headers #include #include #include #include // local game headers #include #include // local library headers #include using namespace tinyxml2; /* ---------------------------------------------------------------------------- ** 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; // 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 pathnames of XML file'd worlds the player has left to enter structures static std::vector inside; // 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 } }; static std::string currentXMLRaw; XMLDocument currentXMLDoc; /* ---------------------------------------------------------------------------- ** Functions section ** --------------------------------------------------------------------------*/ /** * Creates a world object. * Note that all this does is nullify a pointer... */ World:: World(void) { bgmObj = nullptr; 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) { // SDL2_mixer's object if (bgmObj != nullptr) Mix_FreeMusic(bgmObj); 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 mobs clearPointerVector(mob); // free npcs merchant.clear(); // TODO clearPointerVector(npc); // free structures clearPointerVector(build); // free objects object.clear(); // free particles particles.clear(); // clear light array light.clear(); // free villages village.clear(); // clear entity array entity.clear(); } /** * 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; } weather = WorldWeather::Sunny; } /** * The world draw function. * This function will draw the background layers, entities, and player to the * screen. */ void World::drawBackgrounds(void) { auto SCREEN_WIDTH = game::SCREEN_WIDTH; auto SCREEN_HEIGHT = game::SCREEN_HEIGHT; auto HLINE = game::HLINE; const ivec2 backgroundOffset = ivec2 { static_cast(SCREEN_WIDTH) / 2, static_cast(SCREEN_HEIGHT) / 2 }; // world width in pixels int width = worldData.size() * HLINE; // used for alpha values of background textures int alpha; // shade value for GLSL float shadeAmbient = std::max(0.0f, static_cast(-worldShade) / 50 + 0.5f); // 0 to 1.5f if (shadeAmbient > 0.9f) shadeAmbient = 1; // the sunny wallpaper is faded with the night depending on tickCount switch (weather) { case WorldWeather::Snowy: alpha = 150; break; case WorldWeather::Rain: alpha = 0; break; default: alpha = 255 - worldShade * 4; break; } glActiveTexture(GL_TEXTURE0); glUniform1i(worldShader_uniform_texture, 0); // draw background images. GLfloat tex_coord[] = { 0.0f, 1.0f, 1.0f, 1.0f, 1.0f, 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f,}; vec2 bg_tex_coord[] = { vec2(0.0f, 0.0f), vec2(1.0f, 0.0f), vec2(1.0f, 1.0f), vec2(1.0f, 1.0f), vec2(0.0f, 1.0f), vec2(0.0f, 0.0f)}; GLfloat back_tex_coord[] = {offset.x - backgroundOffset.x - 5, offset.y + backgroundOffset.y, 9.9f, offset.x + backgroundOffset.x + 5, offset.y + backgroundOffset.y, 9.9f, offset.x + backgroundOffset.x + 5, offset.y - backgroundOffset.y, 9.9f, offset.x + backgroundOffset.x + 5, offset.y - backgroundOffset.y, 9.9f, offset.x - backgroundOffset.x - 5, offset.y - backgroundOffset.y, 9.9f, offset.x - backgroundOffset.x - 5, offset.y + backgroundOffset.y, 9.9f}; GLfloat fron_tex_coord[] = {offset.x - backgroundOffset.x - 5, offset.y + backgroundOffset.y, 9.8f, offset.x + backgroundOffset.x + 5, offset.y + backgroundOffset.y, 9.8f, offset.x + backgroundOffset.x + 5, offset.y - backgroundOffset.y, 9.8f, offset.x + backgroundOffset.x + 5, offset.y - backgroundOffset.y, 9.8f, offset.x - backgroundOffset.x - 5, offset.y - backgroundOffset.y, 9.8f, offset.x - backgroundOffset.x - 5, offset.y + backgroundOffset.y, 9.8f}; glUseProgram(worldShader); glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); bgTex(0); glUniform4f(worldShader_uniform_color, 1.0, 1.0, 1.0, 1.0); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, back_tex_coord); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, tex_coord); glDrawArrays(GL_TRIANGLES, 0 , 6); bgTex++; glUniform4f(worldShader_uniform_color, .8, .8, .8, 1.3 - static_cast(alpha)/255.0f); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, fron_tex_coord); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, tex_coord); glDrawArrays(GL_TRIANGLES, 0 , 6); static GLuint starTex = Texture::genColor(Color(255, 255, 255)); constexpr const static float stardim = 2; GLfloat star_coord[star.size() * 5 * 6 + 1]; GLfloat *si = &star_coord[0]; if (worldShade > 0) { auto xcoord = offset.x * 0.9f; for (auto &s : star) { *(si++) = s.x + xcoord; *(si++) = s.y, *(si++) = 9.8; *(si++) = 0.0; *(si++) = 0.0; *(si++) = s.x + xcoord + stardim; *(si++) = s.y, *(si++) = 9.8; *(si++) = 1.0; *(si++) = 0.0; *(si++) = s.x + xcoord + stardim; *(si++) = s.y + stardim, *(si++) = 9.8; *(si++) = 1.0; *(si++) = 1.0; *(si++) = s.x + xcoord + stardim; *(si++) = s.y + stardim, *(si++) = 9.8; *(si++) = 1.0; *(si++) = 1.0; *(si++) = s.x + xcoord; *(si++) = s.y + stardim, *(si++) = 9.8; *(si++) = 0.0; *(si++) = 1.0; *(si++) = s.x + xcoord; *(si++) = s.y, *(si++) = 9.8; *(si++) = 0.0; *(si++) = 0.0; } glBindTexture(GL_TEXTURE_2D, starTex); glUniform4f(worldShader_uniform_color, 1.0, 1.0, 1.0, (255.0f - (randGet() % 200 - 100)) / 255.0f); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &star_coord[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &star_coord[3]); glDrawArrays(GL_TRIANGLES, 0, star.size() * 6); } glDisableVertexAttribArray(worldShader_attribute_coord); glDisableVertexAttribArray(worldShader_attribute_tex); glUniform4f(worldShader_uniform_color, 1.0, 1.0, 1.0, 1.0); glUseProgram(0); std::vector bg_items; bgTex++; auto xcoord = width / 2 * -1 + offset.x * 0.85f; for (unsigned int i = 0; i <= worldData.size() * HLINE / 1920; i++) { bg_items.push_back(vec3(1920 * i + xcoord, GROUND_HEIGHT_MINIMUM, 8.0f)); bg_items.push_back(vec3(1920 * (i + 1) + xcoord, GROUND_HEIGHT_MINIMUM, 8.0f)); bg_items.push_back(vec3(1920 * (i + 1) + xcoord, GROUND_HEIGHT_MINIMUM + 1080, 8.0f)); bg_items.push_back(vec3(1920 * (i + 1) + xcoord, GROUND_HEIGHT_MINIMUM + 1080, 8.0f)); bg_items.push_back(vec3(1920 * i + xcoord, GROUND_HEIGHT_MINIMUM + 1080, 8.0f)); bg_items.push_back(vec3(1920 * i + xcoord, GROUND_HEIGHT_MINIMUM, 8.0f)); } std::vector bg_i; std::vector bg_tx; for (auto &v : bg_items) { bg_i.push_back(v.x); bg_i.push_back(v.y); bg_i.push_back(v.z); } for (uint i = 0; i < bg_items.size()/6; i++) { for (auto &v : bg_tex_coord) { bg_tx.push_back(v.x); bg_tx.push_back(v.y); } } glUseProgram(worldShader); glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, &bg_i[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, &bg_tx[0]); glDrawArrays(GL_TRIANGLES, 0 , bg_items.size()); glDisableVertexAttribArray(worldShader_attribute_tex); glDisableVertexAttribArray(worldShader_attribute_coord); glUseProgram(0); // draw the remaining layers for (unsigned int i = 0; i < 4; i++) { std::vectorc; bgTex++; auto xcoord = offset.x * bgDraw[i][2]; for (int j = worldStart; j <= -worldStart; j += 600) { c.push_back(vec3(j + xcoord, GROUND_HEIGHT_MINIMUM, 7-(i*.1))); c.push_back(vec3(j + 600 + xcoord, GROUND_HEIGHT_MINIMUM, 7-(i*.1))); c.push_back(vec3(j + 600 + xcoord, GROUND_HEIGHT_MINIMUM + 400, 7-(i*.1))); c.push_back(vec3(j + 600 + xcoord, GROUND_HEIGHT_MINIMUM + 400, 7-(i*.1))); c.push_back(vec3(j + xcoord, GROUND_HEIGHT_MINIMUM + 400, 7-(i*.1))); c.push_back(vec3(j + xcoord, GROUND_HEIGHT_MINIMUM, 7-(i*.1))); } bg_i.clear(); bg_tx.clear(); for (auto &v : c) { bg_i.push_back(v.x); bg_i.push_back(v.y); bg_i.push_back(v.z); } for (uint i = 0; i < c.size()/6; i++) { for (auto &v : bg_tex_coord) { bg_tx.push_back(v.x); bg_tx.push_back(v.y); } } glUseProgram(worldShader); glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, &bg_i[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, &bg_tx[0]); glDrawArrays(GL_TRIANGLES, 0 , c.size()); glDisableVertexAttribArray(worldShader_attribute_tex); glDisableVertexAttribArray(worldShader_attribute_coord); glUseProgram(0); } } void World::draw(Player *p) { int iStart, iEnd, pOffset; auto SCREEN_WIDTH = game::SCREEN_WIDTH; auto HLINE = game::HLINE; drawBackgrounds(); for (auto &l : light) { if (l.belongsTo) { l.loc.x = l.following->loc.x + SCREEN_WIDTH / 2; 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; } } glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT); // get the line that the player is currently standing on pOffset = (offset.x + p->width / 2 - worldStart) / HLINE; // only draw world within player vision iStart = std::clamp(static_cast(pOffset - (SCREEN_WIDTH / 2 / HLINE) - GROUND_HILLINESS), 0, static_cast(lineCount)); iEnd = std::clamp(static_cast(pOffset + (SCREEN_WIDTH / 2 / HLINE)), 0, static_cast(lineCount)); // draw the dirt bgTex++; std::vector> c; for (int i = iStart; i < iEnd; i++) { if (worldData[i].groundHeight <= 0) { // TODO holes (andy) worldData[i].groundHeight = GROUND_HEIGHT_MINIMUM - 1; glColor4ub(0, 0, 0, 255); } else { safeSetColorA(150, 150, 150, 255); } int ty = worldData[i].groundHeight / 64 + worldData[i].groundColor; // glTexCoord2i(0, 0); glVertex2i(worldStart + i * HLINE , worldData[i].groundHeight - GRASS_HEIGHT); // glTexCoord2i(1, 0); glVertex2i(worldStart + i * HLINE + HLINE , worldData[i].groundHeight - GRASS_HEIGHT); // glTexCoord2i(1, ty); glVertex2i(worldStart + i * HLINE + HLINE, 0); // glTexCoord2i(0, ty); glVertex2i(worldStart + i * HLINE , 0); c.push_back(std::make_pair(vec2(0, 0), vec3(worldStart + HLINES(i), worldData[i].groundHeight - GRASS_HEIGHT, -4.0f))); c.push_back(std::make_pair(vec2(1, 0), vec3(worldStart + HLINES(i) + HLINE, worldData[i].groundHeight - GRASS_HEIGHT, -4.0f))); c.push_back(std::make_pair(vec2(1, ty),vec3(worldStart + HLINES(i) + HLINE, 0, -4.0f))); c.push_back(std::make_pair(vec2(1, ty),vec3(worldStart + HLINES(i) + HLINE, 0, -4.0f))); c.push_back(std::make_pair(vec2(0, ty),vec3(worldStart + HLINES(i), 0, -4.0f))); c.push_back(std::make_pair(vec2(0, 0), vec3(worldStart + HLINES(i), worldData[i].groundHeight - GRASS_HEIGHT, -4.0f))); if (worldData[i].groundHeight == GROUND_HEIGHT_MINIMUM - 1) worldData[i].groundHeight = 0; } std::vector dirtc; std::vector dirtt; for (auto &v : c) { dirtc.push_back(v.second.x); dirtc.push_back(v.second.y); dirtc.push_back(v.second.z); dirtt.push_back(v.first.x); dirtt.push_back(v.first.y); } glUseProgram(worldShader); glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, &dirtc[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, &dirtt[0]); glDrawArrays(GL_TRIANGLES, 0 , c.size()); glDisableVertexAttribArray(worldShader_attribute_tex); glDisableVertexAttribArray(worldShader_attribute_coord); glUseProgram(0); //glEnd(); //glUseProgram(0); // draw the grass //glEnable(GL_TEXTURE_2D); //glActiveTexture(GL_TEXTURE0); bgTex++; //glUseProgram(shaderProgram); //glUniform1i(glGetUniformLocation(shaderProgram, "sampler"), 0); safeSetColorA(255, 255, 255, 255); c.clear(); std::vector grassc; std::vector grasst; for (int i = iStart; i < iEnd; i++) { auto wd = worldData[i]; auto gh = wd.grassHeight; // flatten the grass if the player is standing on it. if (!wd.grassUnpressed) { gh[0] /= 4; gh[1] /= 4; } // actually draw the grass. if (wd.groundHeight) { //glBegin(GL_QUADS); /*glTexCoord2i(0, 0); glVertex2i(worldStart + i * HLINE , wd.groundHeight + gh[0]); glTexCoord2i(1, 0); glVertex2i(worldStart + i * HLINE + HLINE / 2, wd.groundHeight + gh[0]); glTexCoord2i(1, 1); glVertex2i(worldStart + i * HLINE + HLINE / 2, wd.groundHeight - GRASS_HEIGHT); glTexCoord2i(0, 1); glVertex2i(worldStart + i * HLINE , wd.groundHeight - GRASS_HEIGHT); glTexCoord2i(0, 0); glVertex2i(worldStart + i * HLINE + HLINE / 2, wd.groundHeight + gh[1]); glTexCoord2i(1, 0); glVertex2i(worldStart + i * HLINE + HLINE , wd.groundHeight + gh[1]); glTexCoord2i(1, 1); glVertex2i(worldStart + i * HLINE + HLINE , wd.groundHeight - GRASS_HEIGHT); glTexCoord2i(0, 1); glVertex2i(worldStart + i * HLINE + HLINE / 2, wd.groundHeight - GRASS_HEIGHT);*/ c.push_back(std::make_pair(vec2(0, 0),vec3(worldStart + HLINES(i) , wd.groundHeight + gh[0], -3))); c.push_back(std::make_pair(vec2(1, 0),vec3(worldStart + HLINES(i) + HLINE / 2, wd.groundHeight + gh[0], -3))); c.push_back(std::make_pair(vec2(1, 1),vec3(worldStart + HLINES(i) + HLINE / 2, wd.groundHeight - GRASS_HEIGHT, -3))); c.push_back(std::make_pair(vec2(1, 1),vec3(worldStart + HLINES(i) + HLINE / 2, wd.groundHeight - GRASS_HEIGHT, -3))); c.push_back(std::make_pair(vec2(0, 1),vec3(worldStart + HLINES(i) , wd.groundHeight - GRASS_HEIGHT, -3))); c.push_back(std::make_pair(vec2(0, 0),vec3(worldStart + HLINES(i) , wd.groundHeight + gh[0], -3))); c.push_back(std::make_pair(vec2(0, 0),vec3(worldStart + HLINES(i) + HLINE / 2, wd.groundHeight + gh[1], -3))); c.push_back(std::make_pair(vec2(1, 0),vec3(worldStart + HLINES(i) + HLINE , wd.groundHeight + gh[1], -3))); c.push_back(std::make_pair(vec2(1, 1),vec3(worldStart + HLINES(i) + HLINE , wd.groundHeight - GRASS_HEIGHT, -3))); c.push_back(std::make_pair(vec2(1, 1),vec3(worldStart + HLINES(i) + HLINE , wd.groundHeight - GRASS_HEIGHT, -3))); c.push_back(std::make_pair(vec2(0, 1),vec3(worldStart + HLINES(i) + HLINE / 2, wd.groundHeight - GRASS_HEIGHT, -3))); c.push_back(std::make_pair(vec2(0, 0),vec3(worldStart + HLINES(i) + HLINE / 2, wd.groundHeight + gh[1], -3))); //glEnd(); } } for (auto &v : c) { grassc.push_back(v.second.x); grassc.push_back(v.second.y); grassc.push_back(v.second.z); grasst.push_back(v.first.x); grasst.push_back(v.first.y); } glUseProgram(worldShader); glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, &grassc[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, &grasst[0]); glDrawArrays(GL_TRIANGLES, 0 , c.size()); glDisableVertexAttribArray(worldShader_attribute_tex); glDisableVertexAttribArray(worldShader_attribute_coord); glUseProgram(0); //glUseProgram(0); //glDisable(GL_TEXTURE_2D); // draw particles glBindTexture(GL_TEXTURE_2D, colorIndex); glUniform1i(worldShader_uniform_texture, 0); glUseProgram(worldShader); glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); /*GLfloat *pIndexT = &partVec[0]; for (auto &p : particles) { if (!p.behind) p.draw(pIndexT); } glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &partVec[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &partVec[3]); glDrawArrays(GL_TRIANGLES, 0, ps * 6); */ glDisableVertexAttribArray(worldShader_attribute_tex); glDisableVertexAttribArray(worldShader_attribute_coord); glUseProgram(0); // draw the buildings for (auto &b : build) { b->draw(); } // draw remaining entities for (auto &n : npc) { n->draw(); } for (auto &m : mob) m->draw(); for (auto &o : object) o.draw(); // flatten grass under the player if the player is on the ground if (p->ground) { 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(worldShader_uniform_texture, 0); glUseProgram(worldShader); glUniform4f(worldShader_uniform_color, 1.0, 1.0, 1.0, .8); glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); uint ps = particles.size(); uint pss = ps * 6 * 5; uint pc = 0; std::vector partVec(pss); GLfloat *pIndex = &partVec[0]; for (const auto &p : particles) { pc += 30; if (pc > pss) { // TODO resize the vector or something better than breaking //std::cout << "Whoops:" << pc << "," << partVec.size() << std::endl; break; } p.draw(pIndex); } glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &partVec[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &partVec[3]); glDrawArrays(GL_TRIANGLES, 0, ps * 6); glDisableVertexAttribArray(worldShader_attribute_tex); glDisableVertexAttribArray(worldShader_attribute_coord); glUniform4f(worldShader_uniform_color, 1.0, 1.0, 1.0, 1.0); glUseProgram(0); } /** * Handles physics and such for a single entity. * This function is kept private, as World::detect() should be used instead to * handle stuffs for all entities at once. */ void World:: singleDetect(Entity *e) { std::string killed; unsigned int i; int l; auto deltaTime = game::time::getDeltaTime(); if (e == nullptr) return; // kill dead entities if (!e->isAlive()) { return; } else if (e->health <= 0) { // die e->die(); if (inBattle && e->type == MOBT) Mobp(e)->onDeath(); // delete the entity for (i = 0; i < entity.size(); i++) { if (entity[i] == e) { switch (e->type) { case STRUCTURET: killed = "structure"; build.erase(std::find(std::begin(build), std::end(build), e)); break; case NPCT: killed = "NPC"; npc.erase(std::find(std::begin(npc), std::end(npc), e)); break; case MOBT: killed = "mob"; mob.erase(std::find(std::begin(mob), std::end(mob), e)); break; case OBJECTT: killed = "object"; object.erase(std::find(std::begin(object), std::end(object), *Objectp(e))); break; default: break; } std::cout << "Killed a " << killed << "...\n"; entity.erase(entity.begin() + i); return; } } // exit on player death std::cout << "RIP " << e->name << ".\n"; exit(0); } // collision / gravity: handle only living entities else { // forced movement gravity (sword hits) e->handleHits(); // calculate the line that this entity is currently standing on l = std::clamp(static_cast((e->loc.x + e->width / 2 - worldStart) / game::HLINE), 0, static_cast(lineCount)); // if the entity is under the world/line, pop it back to the surface if (e->loc.y < worldData[l].groundHeight) { int dir = e->vel.x < 0 ? -1 : 1; if (l + (dir * 2) < static_cast(worldData.size()) && worldData[l + (dir * 2)].groundHeight - 30 > worldData[l + dir].groundHeight) { e->loc.x -= (PLAYER_SPEED_CONSTANT + 2.7f) * e->speed * 2 * dir; e->vel.x = 0; } else { e->loc.y = worldData[l].groundHeight - 0.001f * deltaTime; e->ground = true; e->vel.y = 0; } } // handle gravity if the entity is above the line else { if (e->type == STRUCTURET) { e->loc.y = worldData[l].groundHeight; e->vel.y = 0; e->ground = true; return; } else if (e->vel.y > -2) { e->vel.y -= GRAVITY_CONSTANT * deltaTime; } } // insure that the entity doesn't fall off either edge of the world. if (e->loc.x < worldStart) { e->vel.x = 0; e->loc.x = worldStart + HLINES(0.5f); } else if (e->loc.x + e->width + game::HLINE > worldStart + worldStart * -2) { e->vel.x = 0; e->loc.x = worldStart + worldStart * -2 - e->width - game::HLINE; } } } /** * Handle entity logic for the world. * * This function runs World::singleDetect() for the player and every entity * currently in a vector of this world. Particles and village entrance/exiting * are also handled here. */ void World:: detect(Player *p) { int l; // handle the player singleDetect(p); //std::thread(&World::singleDetect, this, p).detach(); // handle other entities for (auto &e : entity) singleDetect(e); //std::thread(&World::singleDetect, this, e).detach(); // handle particles for (auto &part : particles) { // get particle's current world line l = std::clamp(static_cast((part.loc.x + part.width / 2 - worldStart) / game::HLINE), 0, static_cast(lineCount - 1)); part.update(GRAVITY_CONSTANT, worldData[l].groundHeight); } // handle particle creation for (auto &b : build) { switch (b->bsubtype) { case FOUNTAIN: for (unsigned int r = (randGet() % 25) + 11; r--;) { addParticle(randGet() % HLINES(3) + b->loc.x + b->width / 2, // x b->loc.y + b->height, // y HLINES(1.25), // width HLINES(1.25), // height randGet() % 7 * .01 * (randGet() % 2 == 0 ? -1 : 1), // vel.x randGet() % 1 ? (8 + randGet() % 6) * .05 : (4 + randGet() % 6) * .05, // vel.y { 0, 0, 255 }, // RGB color 2500 // duration (ms) ); particles.back().fountain = true; particles.back().stu = b; } break; case FIRE_PIT: for(unsigned int r = (randGet() % 20) + 11; r--;) { addParticle(randGet() % (int)(b->width / 2) + b->loc.x + b->width / 4, // x b->loc.y + HLINES(3), // y game::HLINE, // width game::HLINE, // height randGet() % 3 * .01 * (randGet() % 2 == 0 ? -1 : 1), // vel.x (4 + randGet() % 6) * .005, // vel.y { 255, 0, 0 }, // RGB color 400 // duration (ms) ); particles.back().gravity = false; particles.back().behind = true; particles.back().stu = b; } break; default: break; } } // draws the village welcome message if the player enters the village bounds 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; } } else { v.in = false; } } } /** * Updates all entity and player coordinates with their velocities. * Also handles music fading, although that could probably be placed elsewhere. */ void World:: update(Player *p, unsigned int delta, unsigned int ticks) { // update day/night time if (!(ticks % DAY_CYCLE) || ticks == 0) { if (weather == WorldWeather::Sunny) weather = WorldWeather::Dark; else if (weather == WorldWeather::Dark) weather = WorldWeather::Sunny; } // update player coords p->loc.y += p->vel.y * delta; p->loc.x +=(p->vel.x * p->speed) * delta; // handle high-ness if (p->loc.y > 5000) UserError("Too high for me m8."); // update entity coords for (auto &e : entity) { // dont let structures move? if (e->type != STRUCTURET && e->canMove) { e->loc.x += e->vel.x * delta; e->loc.y += e->vel.y * delta; // update boolean directions e->left = e->vel.x ? (e->vel.x < 0) : e->left; } else if (e->vel.y < 0) { e->loc.y += e->vel.y * delta; } } // iterate through particles particles.remove_if([](const Particles &part) { return part.duration <= 0; }); for (auto &pa : particles) { if (pa.canMove) { // causes overhead pa.loc.y += pa.vel.y * delta; pa.loc.x += pa.vel.x * delta; if (pa.stu != nullptr) { if (pa.loc.x >= pa.stu->loc.x && pa.loc.x <= pa.stu->loc.x + pa.stu->width && pa.loc.y <= pa.stu->loc.y + pa.stu->height * 0.25f) pa.duration = 0; } } } // handle music fades if (!Mix_PlayingMusic()) Mix_FadeInMusic(bgmObj, -1, 2000); } /** * 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) { return mob.back(); } /** * 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(mob), std::end(mob), [&](Mob *&a) { return 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(mob) ? nullptr : *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 (build.empty() || (unsigned)index >= build.size()) return vec2 {0, 0}; if (index < 0) return build.back()->loc; return build[index]->loc; } /** * Saves world data to a file. */ void World::save(void) { for (const auto &e : entity) e->saveToXML(); currentXMLDoc.SaveFile(currentXML.c_str(), false); } /** * Toggle play/stop of the background music. * If new music is to be played a crossfade will occur, otherwise... uhm. */ void World::bgmPlay(World *prev) const { if (prev == nullptr || bgm != prev->bgm) { Mix_FadeOutMusic(800); Mix_PlayMusic(bgmObj, -1); } } /** * Set the world's BGM. * This will load a sound file to be played while the player is in this world. * If no file is found, no music should play. */ void World::setBGM(std::string path) { if (!path.empty()) bgmObj = Mix_LoadMUS((bgm = path).c_str()); } /** * 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) { // load textures with a limit check switch ((bgType = bgt)) { case WorldBGType::Forest: bgTex = TextureIterator(bgFiles); break; case WorldBGType::WoodHouse: bgTex = TextureIterator(bgFilesIndoors); break; default: UserError("Invalid world background type"); break; } } /** * 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; for (const auto &s : buildPaths) sTexLoc.push_back(prefix + s); prefix += "bg/"; for (const auto &s : bgPaths[0]) bgFiles.push_back(prefix + s); for (const auto &s : bgPaths[1]) bgFilesIndoors.push_back(prefix + s); } /** * Pretty self-explanatory. */ std::string World::getWeatherStr(void) const { switch (weather) { case WorldWeather::Sunny: return "Sunny"; break; case WorldWeather::Dark: return "Dark"; break; case WorldWeather::Rain: return "Rainy"; break; case WorldWeather::Snowy: return "Snowy"; break; } return "???"; } const WorldWeather& World::getWeatherId(void) const { return weather; } /** * 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); npc.push_back(e); } void World::adoptMob(Mob *e) { entity.push_back(e); mob.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); npc.erase(std::find(std::begin(npc), std::end(npc), 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); npc.erase(std::find(std::begin(npc), std::end(npc), 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; // enter a building if (inside.empty()) { auto d = std::find_if(std::begin(build), std::end(build), [p](const Structures *s) { return ((p->loc.x > s->loc.x) && (p->loc.x + p->width < s->loc.x + s->width)); }); auto b = *d; if ((d == std::end(build)) || b->inside.empty()) return std::make_pair(this, vec2 {0, 0}); // +size cuts folder prefix inside.push_back(¤tXML[xmlFolder.size()]); tmp = loadWorldFromXML(b->inside); return std::make_pair(tmp, vec2 {0, 100}); } // exit the building else { std::string current = ¤tXML[xmlFolder.size()]; tmp = loadWorldFromXML(inside.back()); inside.clear(); Structures *b = nullptr; for (auto &s : tmp->build) { if (s->inside == current) { b = 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) { build.push_back(s); entity.push_back(build.back()); } Village *World:: addVillage(std::string name, World *world) { village.emplace_back(name, world); return &village.back(); } void World::addMob(Mob *m, vec2 coord) { mob.push_back(m); mob.back()->spawn(coord.x, coord.y); entity.push_back(mob.back()); } void World:: addNPC(NPC *n) { npc.push_back(n); entity.push_back(npc.back()); } void World:: addMerchant(float x, float y, bool housed) { merchant.push_back(new Merchant()); merchant.back()->spawn(x, y); if (housed) { merchant.back()->inside = build.back(); merchant.back()->z = build.back()->z+.1; } npc.push_back(merchant.back()); entity.push_back(npc.back()); } void World:: addObject(std::string in, std::string p, float x, float y) { object.emplace_back(in, 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(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, Color color) { if (light.size() < 64) light.emplace_back(loc, color, 1); } 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); } */ glUseProgram(worldShader); glActiveTexture(GL_TEXTURE0); bgTex(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(worldShader_uniform_texture, 0); glUniform4f(worldShader_uniform_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}; glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, backTile); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, backTile_tex); glDrawArrays(GL_TRIANGLES, 0, 6); glDisableVertexAttribArray(worldShader_attribute_coord); glDisableVertexAttribArray(worldShader_attribute_tex); glUseProgram(0); /* * Draw the ground. */ // TODO make floor texture static GLuint floorTex = Texture::genColor(Color(150, 100, 50)); glUseProgram(worldShader); glBindTexture(GL_TEXTURE_2D, floorTex); std::vectorf; for (fl = 0; fl < floor.size(); fl++) { i = 0; for (const auto &h : floor[fl]) { x = worldStart + fstart[fl] * HLINE + HLINES(i); f.emplace_back(x); f.emplace_back(h); f.emplace_back(-3); f.emplace_back(0); f.emplace_back(0); f.emplace_back(x + HLINE); f.emplace_back(h); f.emplace_back(-3); f.emplace_back(1); f.emplace_back(0); f.emplace_back(x + HLINE); f.emplace_back(h - INDOOR_FLOOR_THICKNESS); f.emplace_back(-3); f.emplace_back(1); f.emplace_back(1); f.emplace_back(x + HLINE); f.emplace_back(h - INDOOR_FLOOR_THICKNESS); f.emplace_back(-3); f.emplace_back(1); f.emplace_back(1); f.emplace_back(x); f.emplace_back(h - INDOOR_FLOOR_THICKNESS); f.emplace_back(-3); f.emplace_back(0); f.emplace_back(1); f.emplace_back(x); f.emplace_back(h); f.emplace_back(-3); f.emplace_back(0); f.emplace_back(0); i++; } } glEnableVertexAttribArray(worldShader_attribute_coord); glEnableVertexAttribArray(worldShader_attribute_tex); glVertexAttribPointer(worldShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &f[0]); glVertexAttribPointer(worldShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), &f[3]); glDrawArrays(GL_TRIANGLES, 0, floor.size() * 6); glDisableVertexAttribArray(worldShader_attribute_coord); glDisableVertexAttribArray(worldShader_attribute_tex); glUseProgram(0); /* * 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; mob.push_back((mmob = m)); entity.push_back(mmob); mmob->aggressive = false; arenaNest.emplace_back(leave, p->loc); } WorldSwitchInfo Arena::exitArena(Player *p) { if (!mmob->isAlive() && p->loc.x + p->width / 2 > mob[0]->loc.x && p->loc.x + p->width / 2 < mob[0]->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) { World *tmp = ptr; loadedLeft = true; currentWorldToLeft = loadWorldFromXML(tmp->getToLeft()); loadedLeft = false; loadedRight = true; currentWorldToRight = loadWorldFromXML(tmp->getToRight()); loadedRight = false; return tmp; } /** * Loads a world from the given XML file. */ World * loadWorldFromXMLNoSave(std::string path) { XMLDocument *_currentXMLDoc; 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; _currentXMLRaw = readFile(_currentXML.c_str()); // 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"))) 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