#include /* ---------------------------------------------------------------------------- ** Includes section ** --------------------------------------------------------------------------*/ // standard library headers #include #include #include #include #include #include using namespace std::literals::chrono_literals; // local game headers #include #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 ** --------------------------------------------------------------------------*/ extern std::string xmlFolder; // wait static bool waitToSwap = false; // 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 const unsigned int GRASS_HEIGHT = HLINES(4); // the path of the currently loaded XML file, externally referenced in places std::string currentXML; // pathnames of images for world themes using StyleList = std::array; static const std::vector bgPaths = { { // Forest "bg.png", // daytime background "bgn.png", // nighttime background "bgFarMountain.png", // layer 1 (furthest) "forestTileFar.png", // layer 2 "forestTileBack.png", // layer 3 "forestTileMid.png", // layer 4 "forestTileFront.png", // layer 5 (closest) "dirt.png", // ground texture "grass.png" // grass (ground-top) texture } }; // 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 } }; /* ---------------------------------------------------------------------------- ** Functions section ** --------------------------------------------------------------------------*/ void WorldSystem::generate(int width) { float geninc = 0; // allocate space for world world.data = std::vector (width + GROUND_HILLINESS, WorldData { false, {0, 0}, 0, 0 }); // prepare for generation world.data[0].groundHeight = GROUND_HEIGHT_INITIAL; auto wditer = std::begin(world.data) + GROUND_HILLINESS; if (world.indoor) { for (auto &l : world.data) { l.groundHeight = GROUND_HEIGHT_MINIMUM + 5; l.groundColor = 4; } } else { // give every GROUND_HILLINESSth entry a groundHeight value for (; wditer < std::end(world.data); 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(world.data) + 1; wditer < std::end(world.data); 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 + HLINES(2); w->grassHeight[1] = (randGet() % 16) / 3 + HLINES(2); } } // define x-coordinate of world's leftmost 'line' world.startX = HLINES(width * -0.5); } static Color ambient; bool WorldSystem::save(void) { if (world.indoor) return false; std::ofstream save (xmlFolder + currentXMLFile + ".dat"); // signature? save << "831998 "; game::entities.each([&](entityx::Entity entity, Position& pos) { // save position save << "p " << pos.x << ' ' << pos.y << ' '; // save dialog, if exists if (entity.has_component()) save << "d " << entity.component()->index << ' '; }); save.close(); return true; } void WorldSystem::load(const std::string& file) { auto& render = *game::engine.getSystem(); auto str2coord = [](std::string s) -> vec2 { auto cpos = s.find(','); s[cpos] = '\0'; return vec2 (std::stof(s), std::stof(s.substr(cpos + 1))); }; entityx::Entity entity; std::string xmlRaw; std::string xmlPath; // check for empty file name if (file.empty()) return; // load file data to string xmlPath = xmlFolder + file; auto xmlRawData = readFile(xmlPath.c_str()); xmlRaw = xmlRawData; delete[] xmlRawData; // let tinyxml parse the file if (xmlDoc.Parse(xmlRaw.data()) != XML_NO_ERROR) UserError("XML Error: Failed to parse file (not your fault though..?)"); // include headers auto ixml = xmlDoc.FirstChildElement("include"); while (ixml) { auto file = ixml->Attribute("file"); if (file != nullptr) { DEBUG_printf("Including file: %s\n", file); auto include = readFile((xmlFolder + file).c_str()); xmlRaw.append(include); delete[] include; } ixml = ixml->NextSiblingElement(); } xmlDoc.Parse(xmlRaw.data()); // look for an opening world tag auto wxml = xmlDoc.FirstChildElement("World"); if (wxml != nullptr) { wxml = wxml->FirstChildElement(); world.indoor = false; } else { wxml = xmlDoc.FirstChildElement("IndoorWorld"); if (wxml != nullptr) { wxml = wxml->FirstChildElement(); world.indoor = true; } else { UserError("XML Error: Cannot find a or tag in " + xmlPath); } } world.toLeft = world.toRight = ""; currentXMLFile = file; game::entities.reset(); game::engine.getSystem()->create(); // iterate through tags while (wxml) { std::string tagName = wxml->Name(); // style tag if (tagName == "style") { world.styleFolder = wxml->StrAttribute("folder"); unsigned int styleNo; if (wxml->QueryUnsignedAttribute("background", &styleNo) != XML_NO_ERROR) UserError("XML Error: No background given in