25 #include <Overlay/OgreOverlayManager.h>
26 #include <Overlay/OgreOverlay.h>
27 #include <Plugins/ParticleFX/OgreBoxEmitterFactory.h>
47 #ifdef USE_ANGELSCRIPT
50 #endif // USE_ANGELSCRIPT
54 #include <OgreFileSystem.h>
56 #include <rapidjson/stringbuffer.h>
57 #include <rapidjson/writer.h>
59 #include <OgreMeshLodGenerator.h>
68 #define DECLARE_RESOURCE_PACK(_FIELD_, _NAME_, _RESOURCE_GROUP_) \
69 const ContentManager::ResourcePack ContentManager::ResourcePack::_FIELD_(_NAME_, _RESOURCE_GROUP_);
99 void ContentManager::AddResourcePack(
ResourcePack const& resource_pack, std::string
const& override_rgn)
101 Ogre::ResourceGroupManager& rgm = Ogre::ResourceGroupManager::getSingleton();
103 Ogre::String rg_name;
104 if (!override_rgn.empty())
106 rg_name = override_rgn;
117 std::stringstream log_msg;
118 log_msg <<
"[RoR|ContentManager] Loading resource pack \"" << resource_pack.
name <<
"\" to group \"" << rg_name <<
"\"";
120 std::string zip_path = dir_path +
".zip";
123 log_msg <<
" (ZIP archive)";
125 rgm.addResourceLocation(zip_path,
"Zip", rg_name);
131 log_msg <<
" (directory)";
133 rgm.addResourceLocation(dir_path,
"FileSystem", rg_name);
137 log_msg <<
" failed, data not found.";
138 throw std::runtime_error(log_msg.str());
142 if (override_rgn.empty())
144 rgm.initialiseResourceGroup(rg_name);
148 void ContentManager::InitContentManager()
150 ResourceGroupManager::getSingleton().addResourceLocation(
152 ResourceGroupManager::getSingleton().addResourceLocation(
154 ResourceGroupManager::getSingleton().addResourceLocation(
156 ResourceGroupManager::getSingleton().addResourceLocation(
159 Ogre::ScriptCompilerManager::getSingleton().setListener(
this);
168 if (!Ogre::ResourceGroupManager::getSingleton().getLoadingListener())
169 Ogre::ResourceGroupManager::getSingleton().setLoadingListener(
this);
175 this->AddResourcePack(ResourcePack::MYGUI);
176 this->AddResourcePack(ResourcePack::DASHBOARDS);
182 LOG(
"RoR|ContentManager: Registering Particle Box Emitter");
184 ParticleSystemManager::getSingleton().addRendererFactory(mParticleSystemRendererFact);
193 #ifdef USE_ANGELSCRIPT
196 ParticleSystemManager::getSingleton().addAffectorFactory(pAffFact);
200 ParticleSystemManager::getSingleton().addAffectorFactory(pAffFact);
201 #endif // USE_ANGELSCRIPT
205 LOG(
"RoR|ContentManager: Creating Sound Manager");
210 AddResourcePack(ResourcePack::SOUNDS);
213 LOG(
"RoR|ContentManager: Loading filesystems");
215 LOG(
"RoR|ContentManager: Registering colored text overlay factory");
217 OverlayManager::getSingleton().addOverlayElementFactory(pCT);
220 if (TextureManager::getSingletonPtr())
221 TextureManager::getSingleton().setDefaultNumMipmaps(5);
223 TextureFilterOptions tfo = TFO_NONE;
226 case GfxTexFilter::ANISOTROPIC: tfo = TFO_ANISOTROPIC;
break;
227 case GfxTexFilter::TRILINEAR: tfo = TFO_TRILINEAR;
break;
228 case GfxTexFilter::BILINEAR: tfo = TFO_BILINEAR;
break;
229 case GfxTexFilter::NONE: tfo = TFO_NONE;
break;
231 MaterialManager::getSingleton().setDefaultAnisotropy(Math::Clamp(
App::gfx_anisotropy->getInt(), 1, 16));
232 MaterialManager::getSingleton().setDefaultTextureFiltering(tfo);
235 LOG(
"RoR|ContentManager: Calling initialiseAllResourceGroups()");
238 ResourceGroupManager::getSingleton().initialiseAllResourceGroups();
240 catch (Ogre::Exception& e)
242 LOG(
"RoR|ContentManager: catched error while initializing Resource groups: " + e.getFullDescription());
248 new Ogre::MeshLodGenerator();
262 ResourceGroupManager::getSingleton().addResourceLocation(
264 ResourceGroupManager::getSingleton().addResourceLocation(
272 ResourceGroupManager::getSingleton().addResourceLocation(extra_mod_path ,
"FileSystem",
RGN_CONTENT);
279 std::string objects =
PathCombine(
"resources",
"beamobjects.zip");
281 std::string dashboards =
PathCombine(
"resources",
"dashboards.zip");
283 std::string gadgets =
PathCombine(
"resources",
"gadgets.zip");
288 ResourceGroupManager::getSingleton().createResourceGroup(
RGN_TEMP,
false);
292 ResourceGroupManager::getSingleton().addResourceLocation(extra_mod_path ,
"FileSystem",
RGN_TEMP,
true);
303 FileInfoListPtr dirs = ResourceGroupManager::getSingleton().findResourceFileInfo(
RGN_TEMP,
"*",
true);
304 for (
const auto& dir_fileinfo : *dirs)
306 if (!dir_fileinfo.archive)
308 String fullpath =
PathCombine(dir_fileinfo.archive->getName(), dir_fileinfo.filename);
309 ResourceGroupManager::getSingleton().addResourceLocation(fullpath,
"FileSystem",
RGN_CONTENT,
false,
false);
311 ResourceGroupManager::getSingleton().destroyResourceGroup(
RGN_TEMP);
315 if (validity == CacheValidity::UNKNOWN)
321 ResourceGroupManager::getSingleton().destroyResourceGroup(
RGN_CONTENT);
325 Ogre::DataStreamPtr ContentManager::resourceLoading(
const Ogre::String& name,
const Ogre::String& group, Ogre::Resource* resource)
327 return Ogre::DataStreamPtr();
330 void ContentManager::resourceStreamOpened(
const Ogre::String& name,
const Ogre::String& group, Ogre::Resource* resource, Ogre::DataStreamPtr& dataStream)
334 bool ContentManager::resourceCollision(Ogre::Resource* resource, Ogre::ResourceManager* resourceManager)
341 RoR::LogFormat(
"[RoR|ContentManager] Skipping resource with duplicate name: '%s' (origin: '%s')",
342 resource->getName().c_str(), resource->getOrigin().c_str());
346 bool ContentManager::handleEvent(ScriptCompiler *compiler, ScriptCompilerEvent *evt,
void *retval)
348 if (evt->mType == CreateMaterialScriptCompilerEvent::eventType)
352 auto* matEvent =
static_cast<CreateMaterialScriptCompilerEvent*
>(evt);
353 if (matEvent->mName.empty())
355 RoR::LogFormat(
"[RoR] Got malformed material (empty name) from file: '%s' - forcing OGRE to fail loading.",
356 matEvent->mFile.c_str());
362 else if (evt->mType == CreateParticleSystemScriptCompilerEvent::eventType)
366 auto* particleEvent =
static_cast<CreateParticleSystemScriptCompilerEvent*
>(evt);
367 if (Ogre::ParticleSystemManager::getSingleton().getTemplate(particleEvent->mName) !=
nullptr)
370 RoR::LogFormat(
"[RoR] Duplicate particle system name '%s' in file: '%s' - forcing OGRE to fail loading.",
371 particleEvent->mName.c_str(), particleEvent->mFile.c_str());
379 void ContentManager::InitManagedMaterials(std::string
const & rg_name)
388 ResourceGroupManager::getSingleton().addResourceLocation(
PathCombine(managed_materials_dir,
"shadows/pssm/on/shared"),
"FileSystem", rg_name);
390 ResourceGroupManager::getSingleton().addResourceLocation(
PathCombine(managed_materials_dir,
"shadows/pssm/on"),
"FileSystem", rg_name);
394 ResourceGroupManager::getSingleton().addResourceLocation(
PathCombine(managed_materials_dir,
"shadows/pssm/off"),
"FileSystem", rg_name);
397 ResourceGroupManager::getSingleton().addResourceLocation(
PathCombine(managed_materials_dir,
"texture"),
"FileSystem", rg_name);
400 ResourceGroupManager::getSingleton().addResourceLocation(managed_materials_dir,
"FileSystem", rg_name);
403 ResourceGroupManager::getSingleton().initialiseResourceGroup(rg_name);
406 void ContentManager::LoadGameplayResources()
408 if (!m_base_resource_loaded)
410 this->AddResourcePack(ContentManager::ResourcePack::AIRFOILS);
411 this->AddResourcePack(ContentManager::ResourcePack::TEXTURES);
412 this->AddResourcePack(ContentManager::ResourcePack::FAMICONS);
413 this->AddResourcePack(ContentManager::ResourcePack::MATERIALS);
414 this->AddResourcePack(ContentManager::ResourcePack::MESHES);
415 this->AddResourcePack(ContentManager::ResourcePack::OVERLAYS);
416 this->AddResourcePack(ContentManager::ResourcePack::PARTICLES);
418 m_base_resource_loaded =
true;
422 this->AddResourcePack(ContentManager::ResourcePack::HYDRAX);
425 this->AddResourcePack(ContentManager::ResourcePack::CAELUM);
428 this->AddResourcePack(ContentManager::ResourcePack::SKYX);
431 this->AddResourcePack(ContentManager::ResourcePack::PAGED);
434 std::string ContentManager::ListAllUserContent()
436 std::stringstream buf;
438 auto dir_list = Ogre::ResourceGroupManager::getSingleton().listResourceFileInfo(
RGN_CONTENT,
true);
439 for (
auto dir: *dir_list)
441 buf << dir.filename << std::endl;
445 std::regex file_whitelist(
"^.\\.(airplane|boat|car|fixed|load|machine|skin|terrn2|train|truck)$", std::regex::icase);
447 auto file_list = Ogre::ResourceGroupManager::getSingleton().listResourceFileInfo(
RGN_CONTENT,
false);
448 for (
auto file: *file_list)
450 if ((
file.archive !=
nullptr) || std::regex_match(
file.filename, file_whitelist))
452 buf <<
file.filename << std::endl;
459 bool ContentManager::LoadAndParseJson(std::string
const& filename, std::string
const& rg_name, rapidjson::Document& j_doc)
463 Ogre::DataStreamPtr stream = Ogre::ResourceGroupManager::getSingleton().openResource(filename, rg_name);
464 Ogre::String json_str = stream->getAsString();
465 rapidjson::MemoryStream j_stream(json_str.data(), json_str.length());
466 j_doc.ParseStream<rapidjson::kParseNanAndInfFlag>(j_stream);
468 catch (Ogre::FileNotFoundException)
472 catch (std::exception& e)
474 RoR::LogFormat(
"[RoR] Failed to open or read json file '%s' (resource group '%s'), message: '%s'",
475 filename.c_str(), rg_name.c_str(), e.what());
479 if (j_doc.HasParseError())
481 RoR::LogFormat(
"[RoR] Error parsing JSON file '%s' (resource group '%s')",
482 filename.c_str(), rg_name.c_str());
489 bool ContentManager::SerializeAndWriteJson(std::string
const& filename, std::string
const& rg_name, rapidjson::Document& j_doc)
492 rapidjson::StringBuffer buffer;
493 rapidjson::Writer<rapidjson::StringBuffer, rapidjson::UTF8<>, rapidjson::UTF8<>,
494 rapidjson::CrtAllocator, rapidjson::kWriteNanAndInfFlag>
496 j_doc.Accept(writer);
501 Ogre::DataStreamPtr stream
502 = Ogre::ResourceGroupManager::getSingleton().createResource(
503 filename, rg_name,
true);
504 size_t written = stream->write(buffer.GetString(), buffer.GetSize());
505 if (written < buffer.GetSize())
507 RoR::LogFormat(
"[RoR] Error writing JSON file '%s' (resource group '%s'), ",
508 "only written %u out of %u bytes!",
509 filename.c_str(), rg_name.c_str(), written, buffer.GetSize());
514 catch (std::exception& e)
516 RoR::LogFormat(
"[RoR] Error writing JSON file '%s' (resource group '%s'), message: '%s'",
517 filename.c_str(), rg_name.c_str(), e.what());
522 bool ContentManager::DeleteDiskFile(std::string
const& filename, std::string
const& rg_name)
526 Ogre::ResourceGroupManager::getSingleton().deleteResource(filename, rg_name);
529 catch (std::exception& e)
531 RoR::LogFormat(
"[RoR|ModCache] Error deleting file '%s' (resource group '%s'), message: '%s'",
532 filename.c_str(), rg_name.c_str(), e.what());