#include // standard library headers #include #include #include #include #include using namespace std::literals::chrono_literals; // local library headers #include using namespace tinyxml2; // game headers #include #include #include #include #include #include #include #include #include #include #include #include #include WorldData2 WorldSystem::world; Mix_Music* WorldSystem::bgmObj; std::string WorldSystem::bgmCurrent; std::vector WorldSystem::bgFiles; TextureIterator WorldSystem::bgTex; XMLDocument WorldSystem::xmlDoc; std::string WorldSystem::currentXMLFile; std::thread WorldSystem::thAmbient; std::vector WorldSystem::stars; std::string WorldSystem::toLoad; // wait static bool waitToSwap = false; // 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 static std::string currentXML; // pathnames of images for world themes //using StyleList = std::string[8]; constexpr const char* bgPaths[1][8] = { { // Forest "bg.png", // sky/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 } }; /* ---------------------------------------------------------------------------- ** Functions section ** --------------------------------------------------------------------------*/ /*int generateString(const std::string& s) { int mag = 0; std::stringstream ss; ss.str(s); ss >> mag >> ':'; }*/ int WorldSystem::getLineIndex(float x) { return std::clamp(static_cast((x - world.startX) / game::HLINE), 0, static_cast(world.data.size())); } void WorldSystem::generate(int width) { float geninc = 0; // allocate space for world world.data = std::vector (width + GROUND_HILLINESS); // prepare for generation world.data[0].groundHeight = GROUND_HEIGHT_INITIAL; auto wditer = std::begin(world.data) + GROUND_HILLINESS; if (world.indoor) { std::fill(world.data.begin(), world.data.end(), WorldData {true, {0, 0}, GROUND_HEIGHT_MINIMUM + 5, 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); // gen. star coordinates if (stars.empty()) { stars.resize(game::SCREEN_WIDTH / 30); for (auto& s : stars) { s.x = world.startX + (randGet() % (int)HLINES(width)); s.y = game::SCREEN_HEIGHT - (randGet() % (int)HLINES(game::SCREEN_HEIGHT / 1.3f)); } } } float WorldSystem::isAboveGround(const vec2& p) { const auto& gh = world.data[getLineIndex(p.x)].groundHeight; return p.y >= gh ? 0 : gh; } static Color ambient; bool WorldSystem::save(void) { if (world.indoor) return false; std::ofstream save (game::config::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 << ' '; if (entity.has_component()) save << "h " << entity.component()->health << ' '; }, true); save.close(); return true; } static std::vector savedEntities; void WorldSystem::fight(entityx::Entity entity) { std::string exit = currentXMLFile; savedEntities.emplace_back(entity.id()); load(entity.component()->arena); savedEntities.clear(); entity.component()->health = entity.component()->maxHealth; entity.remove(); auto door = game::entities.create(); door.assign(0, 100); door.assign(); door.assign(-5); door.assign(exit); auto sprite = door.assign(); auto dtex = RenderSystem::loadTexture("assets/style/classic/door.png"); sprite->addSpriteSegment(SpriteData(dtex), 0); auto dim = HLINES(sprite->getSpriteSize()); door.assign(dim.x, dim.y); } void WorldSystem::load(const std::string& file) { toLoad = file; } void WorldSystem::loader(void) { entityx::Entity entity; // check for empty file name if (toLoad.empty()) return; // save the current world's data if (!currentXMLFile.empty()) save(); // load file data to string auto xmlPath = game::config::xmlFolder + toLoad; auto xmlRaw = readFile(xmlPath); // let tinyxml parse the file UserAssert(xmlDoc.Parse(xmlRaw.data()) == XML_NO_ERROR, "XML Error: Failed to parse file (not your fault though..?)"); // include headers std::vector toAdd; auto ixml = xmlDoc.FirstChildElement("include"); while (ixml != nullptr) { auto file = ixml->Attribute("file"); UserAssert(file != nullptr, "XML Error: tag file not given"); if (file != nullptr) { DEBUG_printf("Including file: %s\n", file); toAdd.emplace_back(game::config::xmlFolder + file); //xmlRaw.append(readFile(xmlFolder + file)); } else { UserError("XML Error: tag file not given"); } ixml = ixml->NextSiblingElement("include"); } for (const auto& f : toAdd) xmlRaw.append(readFile(f)); UserAssert(xmlDoc.Parse(xmlRaw.data()) == XML_NO_ERROR, "XML Error: failed to append includes"); // 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"); UserAssert(wxml != nullptr, "XML Error: Cannot find a or tag in " + xmlPath); wxml = wxml->FirstChildElement(); world.indoor = true; if (world.outdoor.empty()) { world.outdoor = currentXMLFile; world.outdoorCoords = vec2(0, 100); } } world.toLeft = world.toRight = ""; currentXMLFile = toLoad; //game::entities.reset(); if (!savedEntities.empty()) { savedEntities.emplace_back(PlayerSystem::getId()); game::entities.each( [&](entityx::Entity entity, Position& p) { (void)p; if (std::find(savedEntities.begin(), savedEntities.end(), entity.id()) == savedEntities.end()) entity.destroy(); }, true); } else { game::entities.reset(); PlayerSystem::create(); } // iterate through tags while (wxml != nullptr) { 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