RigsofRods
Soft-body Physics Simulation
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
ContentManager.cpp
Go to the documentation of this file.
1 /*
2  This source file is part of Rigs of Rods
3  Copyright 2005-2012 Pierre-Michel Ricordel
4  Copyright 2007-2012 Thomas Fischer
5  Copyright 2013-2018 Petr Ohlidal
6 
7  For more information, see http://www.rigsofrods.org/
8 
9  Rigs of Rods is free software: you can redistribute it and/or modify
10  it under the terms of the GNU General Public License version 3, as
11  published by the Free Software Foundation.
12 
13  Rigs of Rods is distributed in the hope that it will be useful,
14  but WITHOUT ANY WARRANTY; without even the implied warranty of
15  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16  GNU General Public License for more details.
17 
18  You should have received a copy of the GNU General Public License
19  along with Rigs of Rods. If not, see <http://www.gnu.org/licenses/>.
20 */
21 
22 #include "ContentManager.h"
23 
24 
25 #include <Overlay/OgreOverlayManager.h>
26 #include <Overlay/OgreOverlay.h>
27 #include <Plugins/ParticleFX/OgreBoxEmitterFactory.h>
28 
29 
30 #include "Application.h"
32 #include "ErrorUtils.h"
33 #include "SoundScriptManager.h"
34 #include "SkinFileFormat.h"
35 #include "Language.h"
36 #include "PlatformUtils.h"
37 
38 #include "CacheSystem.h"
39 
41 
42 // Removed by Skybon as part of OGRE 1.9 port
43 // Disabling temporarily for 1.8.1 as well. ~ only_a_ptr, 2015-11
44 // TODO: Study the system, then re-enable or remove entirely.
45 //#include "OgreBoxEmitterFactory.h"
46 
47 #ifdef USE_ANGELSCRIPT
50 #endif // USE_ANGELSCRIPT
51 
52 #include "Utils.h"
53 
54 #include <OgreFileSystem.h>
55 #include <regex>
56 #include <rapidjson/stringbuffer.h>
57 #include <rapidjson/writer.h>
58 #include <sstream>
59 #include <OgreMeshLodGenerator.h>
60 
61 using namespace Ogre;
62 using namespace RoR;
63 
64 // ================================================================================
65 // Static variables
66 // ================================================================================
67 
68 #define DECLARE_RESOURCE_PACK(_FIELD_, _NAME_, _RESOURCE_GROUP_) \
69  const ContentManager::ResourcePack ContentManager::ResourcePack::_FIELD_(_NAME_, _RESOURCE_GROUP_);
70 
71 DECLARE_RESOURCE_PACK( OGRE_CORE, "OgreCore", "OgreCoreRG");
72 DECLARE_RESOURCE_PACK( WALLPAPERS, "wallpapers", "Wallpapers");
73 DECLARE_RESOURCE_PACK( AIRFOILS, "airfoils", "AirfoilsRG");
74 DECLARE_RESOURCE_PACK( CAELUM, "caelum", "CaelumRG");
75 DECLARE_RESOURCE_PACK( CUBEMAPS, "cubemaps", "CubemapsRG");
76 DECLARE_RESOURCE_PACK( DASHBOARDS, "dashboards", "DashboardsRG");
77 DECLARE_RESOURCE_PACK( FAMICONS, "famicons", "FamiconsRG");
78 DECLARE_RESOURCE_PACK( FLAGS, "flags", "FlagsRG");
79 DECLARE_RESOURCE_PACK( FONTS, "fonts", "FontsRG");
80 DECLARE_RESOURCE_PACK( HYDRAX, "hydrax", "HydraxRG");
81 DECLARE_RESOURCE_PACK( ICONS, "icons", "IconsRG");
82 DECLARE_RESOURCE_PACK( MATERIALS, "materials", "MaterialsRG");
83 DECLARE_RESOURCE_PACK( MESHES, "meshes", "MeshesRG");
84 DECLARE_RESOURCE_PACK( MYGUI, "mygui", "MyGuiRG");
85 DECLARE_RESOURCE_PACK( OVERLAYS, "overlays", "OverlaysRG");
86 DECLARE_RESOURCE_PACK( PAGED, "paged", "PagedRG");
87 DECLARE_RESOURCE_PACK( PARTICLES, "particles", "ParticlesRG");
88 DECLARE_RESOURCE_PACK( PSSM, "pssm", "PssmRG");
89 DECLARE_RESOURCE_PACK( RTSHADER, "rtshader", "RtShaderRG");
90 DECLARE_RESOURCE_PACK( SCRIPTS, "scripts", "ScriptsRG");
91 DECLARE_RESOURCE_PACK( SOUNDS, "sounds", "SoundsRG");
92 DECLARE_RESOURCE_PACK( TEXTURES, "textures", "TexturesRG");
93 DECLARE_RESOURCE_PACK( SKYX, "SkyX", "SkyXRG");
94 
95 // ================================================================================
96 // Functions
97 // ================================================================================
98 
99 void ContentManager::AddResourcePack(ResourcePack const& resource_pack, std::string const& override_rgn)
100 {
101  Ogre::ResourceGroupManager& rgm = Ogre::ResourceGroupManager::getSingleton();
102 
103  Ogre::String rg_name;
104  if (!override_rgn.empty()) // Custom RG defined?
105  {
106  rg_name = override_rgn;
107  }
108  else // Use default RG
109  {
110  if (rgm.resourceGroupExists(resource_pack.resource_group_name)) // Already loaded?
111  {
112  return; // Nothing to do, nothing to report
113  }
114  rg_name = resource_pack.resource_group_name;
115  }
116 
117  std::stringstream log_msg;
118  log_msg << "[RoR|ContentManager] Loading resource pack \"" << resource_pack.name << "\" to group \"" << rg_name << "\"";
119  std::string dir_path = PathCombine(App::sys_resources_dir->getStr(), resource_pack.name);
120  std::string zip_path = dir_path + ".zip";
121  if (FileExists(zip_path))
122  {
123  log_msg << " (ZIP archive)";
124  LOG(log_msg.str());
125  rgm.addResourceLocation(zip_path, "Zip", rg_name);
126  }
127  else
128  {
129  if (FolderExists(dir_path))
130  {
131  log_msg << " (directory)";
132  LOG(log_msg.str());
133  rgm.addResourceLocation(dir_path, "FileSystem", rg_name);
134  }
135  else
136  {
137  log_msg << " failed, data not found.";
138  throw std::runtime_error(log_msg.str());
139  }
140  }
141 
142  if (override_rgn.empty()) // Only init the default RG
143  {
144  rgm.initialiseResourceGroup(rg_name);
145  }
146 }
147 
148 void ContentManager::InitContentManager()
149 {
150  ResourceGroupManager::getSingleton().addResourceLocation(
151  App::sys_config_dir->getStr(), "FileSystem", RGN_CONFIG, /*recursive=*/false, /*readOnly=*/false);
152  ResourceGroupManager::getSingleton().addResourceLocation(
153  App::sys_savegames_dir->getStr(), "FileSystem", RGN_SAVEGAMES, /*recursive=*/false, /*readOnly=*/false);
154  ResourceGroupManager::getSingleton().addResourceLocation(
155  App::sys_scripts_dir->getStr(), "FileSystem", RGN_SCRIPTS, /*recursive:*/false, /*readonly:*/false);
156  ResourceGroupManager::getSingleton().addResourceLocation(
157  App::sys_logs_dir->getStr(), "FileSystem", RGN_LOGS, /*recursive:*/false, /*readonly:*/false);
158 
159  Ogre::ScriptCompilerManager::getSingleton().setListener(this);
160 
161  // Initialize "managed materials" first
162  // These are base materials referenced by user content
163  // They must be initialized before any content is loaded,
164  // otherwise material links are unresolved and loading ends with an exception
165  this->InitManagedMaterials(RGN_MANAGED_MATS);
166 
167  // set listener if none has already been set
168  if (!Ogre::ResourceGroupManager::getSingleton().getLoadingListener())
169  Ogre::ResourceGroupManager::getSingleton().setLoadingListener(this);
170 
171  // by default, display everything in the depth map
172  Ogre::MovableObject::setDefaultVisibilityFlags(DEPTHMAP_ENABLED);
173 
174 
175  this->AddResourcePack(ResourcePack::MYGUI);
176  this->AddResourcePack(ResourcePack::DASHBOARDS);
177 
178 
179 #ifdef _WIN32
180  // TODO: FIX UNDER LINUX!
181  // register particle classes
182  LOG("RoR|ContentManager: Registering Particle Box Emitter");
183  ParticleSystemRendererFactory* mParticleSystemRendererFact = OGRE_NEW ShaderParticleRendererFactory();
184  ParticleSystemManager::getSingleton().addRendererFactory(mParticleSystemRendererFact);
185 
186  // Removed by Skybon as part of OGRE 1.9 port
187  // Disabling temporarily for 1.8.1 as well. ~ only_a_ptr, 2015-11
188  //ParticleEmitterFactory *mParticleEmitterFact = OGRE_NEW BoxEmitterFactory();
189  //ParticleSystemManager::getSingleton().addEmitterFactory(mParticleEmitterFact);
190 
191 #endif // _WIN32
192 
193 #ifdef USE_ANGELSCRIPT
194  // FireExtinguisherAffector
195  ParticleAffectorFactory* pAffFact = OGRE_NEW FireExtinguisherAffectorFactory();
196  ParticleSystemManager::getSingleton().addAffectorFactory(pAffFact);
197 
198  // ExtinguishableFireAffector
199  pAffFact = OGRE_NEW ExtinguishableFireAffectorFactory();
200  ParticleSystemManager::getSingleton().addAffectorFactory(pAffFact);
201 #endif // USE_ANGELSCRIPT
202 
203  // sound is a bit special as we mark the base sounds so we don't clear them accidentally later on
204 #ifdef USE_OPENAL
205  LOG("RoR|ContentManager: Creating Sound Manager");
208 #endif // USE_OPENAL
209 
210  AddResourcePack(ResourcePack::SOUNDS);
211 
212  // streams path, to be processed later by the cache system
213  LOG("RoR|ContentManager: Loading filesystems");
214 
215  LOG("RoR|ContentManager: Registering colored text overlay factory");
217  OverlayManager::getSingleton().addOverlayElementFactory(pCT);
218 
219  // set default mipmap level (NB some APIs ignore this)
220  if (TextureManager::getSingletonPtr())
221  TextureManager::getSingleton().setDefaultNumMipmaps(5);
222 
223  TextureFilterOptions tfo = TFO_NONE;
224  switch (App::gfx_texture_filter->getEnum<GfxTexFilter>())
225  {
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;
230  }
231  MaterialManager::getSingleton().setDefaultAnisotropy(Math::Clamp(App::gfx_anisotropy->getInt(), 1, 16));
232  MaterialManager::getSingleton().setDefaultTextureFiltering(tfo);
233 
234  // load all resources now, so the zip files are also initiated
235  LOG("RoR|ContentManager: Calling initialiseAllResourceGroups()");
236  try
237  {
238  ResourceGroupManager::getSingleton().initialiseAllResourceGroups();
239  }
240  catch (Ogre::Exception& e)
241  {
242  LOG("RoR|ContentManager: catched error while initializing Resource groups: " + e.getFullDescription());
243  }
244 #ifdef USE_OPENAL
246 #endif // USE_OPENAL
247 
248  new Ogre::MeshLodGenerator();
249 }
250 
251 void ContentManager::InitModCache(CacheValidity validity)
252 {
253  // Sets up RGN_CONTENT which encompasses all mods, scans it for changes and deletes it again.
254  // IMPORTANT NOTE ON 'readOnly' FLAG:
255  // We need mods in subdirs to be writable for the Tuning menu to work.
256  // Apart from `Resources` and resource groups, OGRE also keeps `Archives` in `ArchiveManager`
257  // These aren't unloaded on destroying resource groups, and keep a 'readOnly' flag (defaults to true).
258  // Upon loading/creating new resource groups, OGRE complains (=assert on Debug, exception on Release) if the submitted flag doesn't match.
259  // It's possible to manually unload archives to reset the flag, but for simplicity we just always load subdirs as 'writable', even during modcache update.
260  // ------------------------------------------------------------------------------------------
261 
262  ResourceGroupManager::getSingleton().addResourceLocation(
263  App::sys_cache_dir->getStr(), "FileSystem", RGN_CACHE, /*recursive=*/false, /*readOnly=*/false);
264  ResourceGroupManager::getSingleton().addResourceLocation(
265  App::sys_thumbnails_dir->getStr(), "FileSystem", RGN_REPO, /*recursive=*/false, /*readOnly=*/false);
266 
267  // Add top-level ZIPs/directories to RGN_CONTENT (non-recursive)
268 
269  if (!App::app_extra_mod_path->getStr().empty())
270  {
271  std::string extra_mod_path = App::app_extra_mod_path->getStr();
272  ResourceGroupManager::getSingleton().addResourceLocation(extra_mod_path , "FileSystem", RGN_CONTENT);
273  }
274  for (const std::string& dirname : App::GetCacheSystem()->GetContentDirs())
275  {
276  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(App::sys_user_dir->getStr(), dirname), "FileSystem", RGN_CONTENT);
277  }
278  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(App::sys_process_dir->getStr(), "content") , "FileSystem", RGN_CONTENT);
279  std::string objects = PathCombine("resources", "beamobjects.zip");
280  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(App::sys_process_dir->getStr(), objects) , "Zip" , RGN_CONTENT);
281  std::string dashboards = PathCombine("resources", "dashboards.zip");
282  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(App::sys_process_dir->getStr(), dashboards), "Zip", RGN_CONTENT);
283  std::string gadgets = PathCombine("resources", "gadgets.zip");
284  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(App::sys_process_dir->getStr(), gadgets), "Zip", RGN_CONTENT);
285 
286  // Create RGN_TEMP in recursive mode to find all subdirectories.
287 
288  ResourceGroupManager::getSingleton().createResourceGroup(RGN_TEMP, false);
289  if (!App::app_extra_mod_path->getStr().empty())
290  {
291  std::string extra_mod_path = App::app_extra_mod_path->getStr();
292  ResourceGroupManager::getSingleton().addResourceLocation(extra_mod_path , "FileSystem", RGN_TEMP, true);
293  }
294  for (const std::string& dirname : App::GetCacheSystem()->GetContentDirs())
295  {
296  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(App::sys_user_dir->getStr(), dirname), "FileSystem", RGN_TEMP, true);
297  }
298  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(App::sys_process_dir->getStr(), "content") , "FileSystem", RGN_TEMP, true);
299 
300  // Traverse RGN_TEMP and add all subdirectories to RGN_CONTENT.
301  // (TBD: why not just make RGN_CONTENT itself recursive? -- ohlidalp, 10/2023)
302 
303  FileInfoListPtr dirs = ResourceGroupManager::getSingleton().findResourceFileInfo(RGN_TEMP, "*", /*dirs:*/true);
304  for (const auto& dir_fileinfo : *dirs)
305  {
306  if (!dir_fileinfo.archive)
307  continue;
308  String fullpath = PathCombine(dir_fileinfo.archive->getName(), dir_fileinfo.filename);
309  ResourceGroupManager::getSingleton().addResourceLocation(fullpath, "FileSystem", RGN_CONTENT, /*recursive:*/false, /*readonly:*/false);
310  }
311  ResourceGroupManager::getSingleton().destroyResourceGroup(RGN_TEMP);
312 
313  // Traverse RGN_CONTENT and detect updates
314 
315  if (validity == CacheValidity::UNKNOWN)
316  {
317  validity = App::GetCacheSystem()->EvaluateCacheValidity(); // Must be called while RGN_CONTENT is alive.
318  }
319  App::GetCacheSystem()->LoadModCache(validity);
320 
321  ResourceGroupManager::getSingleton().destroyResourceGroup(RGN_CONTENT);
322 
323 }
324 
325 Ogre::DataStreamPtr ContentManager::resourceLoading(const Ogre::String& name, const Ogre::String& group, Ogre::Resource* resource)
326 {
327  return Ogre::DataStreamPtr();
328 }
329 
330 void ContentManager::resourceStreamOpened(const Ogre::String& name, const Ogre::String& group, Ogre::Resource* resource, Ogre::DataStreamPtr& dataStream)
331 {
332 }
333 
334 bool ContentManager::resourceCollision(Ogre::Resource* resource, Ogre::ResourceManager* resourceManager)
335 {
336  // RoR loads each resource bundle (see CacheSystem.h for info)
337  // into dedicated resource group outside the global pool [see CacheSystem::LoadResource()]
338  // This means resource collision is pretty much content creator's fault, with 2 exceptions:
339  // * asset packs (introduced 2024) are mixed into the requesting mod's resource group.
340  // * bundled resources (e.g. beamobjects.zip) are also mixed into the mod's resource group.
341  RoR::LogFormat("[RoR|ContentManager] Skipping resource with duplicate name: '%s' (origin: '%s')",
342  resource->getName().c_str(), resource->getOrigin().c_str());
343  return false; // Instruct OGRE to drop the new resource and keep the original.
344 }
345 
346 bool ContentManager::handleEvent(ScriptCompiler *compiler, ScriptCompilerEvent *evt, void *retval)
347 {
348  if (evt->mType == CreateMaterialScriptCompilerEvent::eventType)
349  {
350  // Workaround for OGRE script compiler not properly checking that material name is not empty.
351  // See https://github.com/RigsOfRods/rigs-of-rods/issues/2349
352  auto* matEvent = static_cast<CreateMaterialScriptCompilerEvent*>(evt);
353  if (matEvent->mName.empty())
354  {
355  RoR::LogFormat("[RoR] Got malformed material (empty name) from file: '%s' - forcing OGRE to fail loading.",
356  matEvent->mFile.c_str());
357  // Report "handled" but create nothing -> OGRE will interrupt the loading
358  // with message "failed to find or create material" [in MaterialTranslator::translate()]
359  return true;
360  }
361  }
362  else if (evt->mType == CreateParticleSystemScriptCompilerEvent::eventType)
363  {
364  // Workaround for OGRE ignoring resource groups when registering particle templates
365  // See https://github.com/RigsOfRods/rigs-of-rods/pull/2398
366  auto* particleEvent = static_cast<CreateParticleSystemScriptCompilerEvent*>(evt);
367  if (Ogre::ParticleSystemManager::getSingleton().getTemplate(particleEvent->mName) != nullptr)
368  {
369  // Duplicate name -> OGRE would throw exception and fail initializing whole resource group
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());
372  return true; // Instruct OGRE to skip the particle system
373  }
374  }
375 
376  return false; // Report "not handled"
377 }
378 
379 void ContentManager::InitManagedMaterials(std::string const & rg_name)
380 {
381  Ogre::String managed_materials_dir = PathCombine(App::sys_resources_dir->getStr(), "managed_materials");
382 
383  //Dirty, needs to be improved
384  if (App::gfx_shadow_type->getEnum<GfxShadowType>() == GfxShadowType::PSSM)
385  {
386  if (rg_name == RGN_MANAGED_MATS) // Only load shared resources on startup
387  {
388  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(managed_materials_dir, "shadows/pssm/on/shared"), "FileSystem", rg_name);
389  }
390  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(managed_materials_dir, "shadows/pssm/on"), "FileSystem", rg_name);
391  }
392  else
393  {
394  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(managed_materials_dir,"shadows/pssm/off"), "FileSystem", rg_name);
395  }
396 
397  ResourceGroupManager::getSingleton().addResourceLocation(PathCombine(managed_materials_dir, "texture"), "FileSystem", rg_name);
398 
399  // Last
400  ResourceGroupManager::getSingleton().addResourceLocation(managed_materials_dir, "FileSystem", rg_name);
401 
402  if (rg_name == RGN_MANAGED_MATS) // Only initialize the global resource group
403  ResourceGroupManager::getSingleton().initialiseResourceGroup(rg_name);
404 }
405 
406 void ContentManager::LoadGameplayResources()
407 {
408  if (!m_base_resource_loaded)
409  {
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);
417 
418  m_base_resource_loaded = true;
419  }
420 
421  if (App::gfx_water_mode->getEnum<GfxWaterMode>() == GfxWaterMode::HYDRAX)
422  this->AddResourcePack(ContentManager::ResourcePack::HYDRAX);
423 
424  if (App::gfx_sky_mode->getEnum<GfxSkyMode>() == GfxSkyMode::CAELUM)
425  this->AddResourcePack(ContentManager::ResourcePack::CAELUM);
426 
427  if (App::gfx_sky_mode->getEnum<GfxSkyMode>() == GfxSkyMode::SKYX)
428  this->AddResourcePack(ContentManager::ResourcePack::SKYX);
429 
430  if (App::gfx_vegetation_mode->getEnum<GfxVegetation>() != RoR::GfxVegetation::NONE)
431  this->AddResourcePack(ContentManager::ResourcePack::PAGED);
432 }
433 
434 std::string ContentManager::ListAllUserContent()
435 {
436  std::stringstream buf;
437 
438  auto dir_list = Ogre::ResourceGroupManager::getSingleton().listResourceFileInfo(RGN_CONTENT, true);
439  for (auto dir: *dir_list)
440  {
441  buf << dir.filename << std::endl;
442  }
443 
444  // Any filename + listed extensions, ignore case
445  std::regex file_whitelist("^.\\.(airplane|boat|car|fixed|load|machine|skin|terrn2|train|truck)$", std::regex::icase);
446 
447  auto file_list = Ogre::ResourceGroupManager::getSingleton().listResourceFileInfo(RGN_CONTENT, false);
448  for (auto file: *file_list)
449  {
450  if ((file.archive != nullptr) || std::regex_match(file.filename, file_whitelist))
451  {
452  buf << file.filename << std::endl;
453  }
454  }
455 
456  return buf.str();
457 }
458 
459 bool ContentManager::LoadAndParseJson(std::string const& filename, std::string const& rg_name, rapidjson::Document& j_doc)
460 {
461  try
462  {
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);
467  }
468  catch (Ogre::FileNotFoundException)
469  {
470  return false; // Error already logged by OGRE
471  }
472  catch (std::exception& e)
473  {
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());
476  return false;
477  }
478 
479  if (j_doc.HasParseError())
480  {
481  RoR::LogFormat("[RoR] Error parsing JSON file '%s' (resource group '%s')",
482  filename.c_str(), rg_name.c_str());
483  return false;
484  }
485 
486  return true;
487 }
488 
489 bool ContentManager::SerializeAndWriteJson(std::string const& filename, std::string const& rg_name, rapidjson::Document& j_doc)
490 {
491  // Serialize JSON to string
492  rapidjson::StringBuffer buffer;
493  rapidjson::Writer<rapidjson::StringBuffer, rapidjson::UTF8<>, rapidjson::UTF8<>,
494  rapidjson::CrtAllocator, rapidjson::kWriteNanAndInfFlag>
495  writer(buffer);
496  j_doc.Accept(writer);
497 
498  // Write JSON to file
499  try
500  {
501  Ogre::DataStreamPtr stream
502  = Ogre::ResourceGroupManager::getSingleton().createResource(
503  filename, rg_name, /*overwrite=*/true);
504  size_t written = stream->write(buffer.GetString(), buffer.GetSize());
505  if (written < buffer.GetSize())
506  {
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());
510  return false;
511  }
512  return true;
513  }
514  catch (std::exception& e)
515  {
516  RoR::LogFormat("[RoR] Error writing JSON file '%s' (resource group '%s'), message: '%s'",
517  filename.c_str(), rg_name.c_str(), e.what());
518  return false;
519  }
520 }
521 
522 bool ContentManager::DeleteDiskFile(std::string const& filename, std::string const& rg_name)
523 {
524  try
525  {
526  Ogre::ResourceGroupManager::getSingleton().deleteResource(filename, rg_name);
527  return true;
528  }
529  catch (std::exception& e)
530  {
531  RoR::LogFormat("[RoR|ModCache] Error deleting file '%s' (resource group '%s'), message: '%s'",
532  filename.c_str(), rg_name.c_str(), e.what());
533  return false;
534  }
535 }
536 
RoR::App::sys_user_dir
CVar * sys_user_dir
Definition: Application.cpp:163
ColoredTextAreaOverlayElementFactory.h
RoR::App::GetSoundScriptManager
SoundScriptManager * GetSoundScriptManager()
Definition: Application.cpp:281
RGN_CONFIG
#define RGN_CONFIG
Definition: Application.h:48
RGN_CACHE
#define RGN_CACHE
Definition: Application.h:46
RoR::App::sys_resources_dir
CVar * sys_resources_dir
Definition: Application.cpp:168
RoR::App::sys_scripts_dir
CVar * sys_scripts_dir
Definition: Application.cpp:172
RGN_LOGS
#define RGN_LOGS
Definition: Application.h:53
file
This is a raw Ogre binding for Imgui No project cmake file
Definition: README-OgreImGui.txt:3
ContentManager.h
RoR::CacheSystem::LoadModCache
void LoadModCache(CacheValidity validity)
Definition: CacheSystem.cpp:153
RoR::App::gfx_shadow_type
CVar * gfx_shadow_type
Definition: Application.cpp:217
RoR::App::CreateSoundScriptManager
void CreateSoundScriptManager()
Definition: Application.cpp:338
RoR::LogFormat
void LogFormat(const char *format,...)
Improved logging utility. Uses fixed 2Kb buffer.
Definition: Application.cpp:428
RoR::FolderExists
bool FolderExists(const char *path)
Path must be UTF-8 encoded.
Definition: PlatformUtils.cpp:169
RoR::App::sys_logs_dir
CVar * sys_logs_dir
Definition: Application.cpp:167
Utils.h
RoR::App::gfx_texture_filter
CVar * gfx_texture_filter
Definition: Application.cpp:222
Language.h
RoR::App::sys_savegames_dir
CVar * sys_savegames_dir
Definition: Application.cpp:170
RoR::App::app_extra_mod_path
CVar * app_extra_mod_path
Definition: Application.cpp:87
RoR::ContentManager::ResourcePack
Definition: ContentManager.h:41
RoR::SoundScriptManager::setLoadingBaseSounds
void setLoadingBaseSounds(bool value)
Definition: SoundScriptManager.h:330
RGN_REPO
#define RGN_REPO
Definition: Application.h:47
RoR::ContentManager::ResourcePack::name
const char * name
Definition: ContentManager.h:72
OgreShaderParticleRenderer.h
RGN_MANAGED_MATS
#define RGN_MANAGED_MATS
Definition: Application.h:51
RoR::CVar::getStr
std::string const & getStr() const
Definition: CVar.h:95
RoR::PathCombine
std::string PathCombine(std::string a, std::string b)
Definition: PlatformUtils.h:48
CacheSystem.h
A database of user-installed content alias 'mods' (vehicles, terrains...)
RoR::CacheValidity
CacheValidity
Definition: CacheSystem.h:188
ErrorUtils.h
RGN_TEMP
#define RGN_TEMP
Definition: Application.h:45
Ogre::ShaderParticleRendererFactory
Factory class for ShaderParticleRenderer.
Definition: OgreShaderParticleRenderer.h:234
RoR::GfxVegetation::NONE
@ NONE
RoR::App::sys_thumbnails_dir
CVar * sys_thumbnails_dir
Definition: Application.cpp:166
RoR::App::sys_cache_dir
CVar * sys_cache_dir
Definition: Application.cpp:165
PlatformUtils.h
Platform-specific utilities. We use narrow UTF-8 encoded strings as paths. Inspired by http://utf8eve...
Application.h
Central state/object manager and communications hub.
SoundScriptManager.h
FireExtinguisherAffectorFactory.h
RGN_CONTENT
#define RGN_CONTENT
Definition: Application.h:49
RoR::App::gfx_water_mode
CVar * gfx_water_mode
Definition: Application.cpp:224
RoR::CacheSystem::EvaluateCacheValidity
CacheValidity EvaluateCacheValidity()
Definition: CacheSystem.cpp:259
RoR::App::gfx_sky_mode
CVar * gfx_sky_mode
Definition: Application.cpp:219
RoR::App::sys_config_dir
CVar * sys_config_dir
Definition: Application.cpp:164
ExtinguishableFireAffectorFactory.h
ColoredTextAreaOverlayElementFactory
Factory for creating TextAreaOverlayElement instances.
Definition: ColoredTextAreaOverlayElementFactory.h:31
RoR::App::GetCacheSystem
CacheSystem * GetCacheSystem()
Definition: Application.cpp:276
RoR::DEPTHMAP_ENABLED
@ DEPTHMAP_ENABLED
Definition: Application.h:301
Ogre::ExtinguishableFireAffectorFactory
Factory class for DeflectorPlaneAffector.
Definition: ExtinguishableFireAffectorFactory.h:37
SkinFileFormat.h
Ogre::FireExtinguisherAffectorFactory
Factory class for DeflectorPlaneAffector.
Definition: FireExtinguisherAffectorFactory.h:37
RoR::ContentManager::ResourcePack::resource_group_name
const char * resource_group_name
Definition: ContentManager.h:73
RGN_SCRIPTS
#define RGN_SCRIPTS
Definition: Application.h:52
RGN_SAVEGAMES
#define RGN_SAVEGAMES
Definition: Application.h:50
Ogre
Definition: ExtinguishableFireAffector.cpp:35
RoR::App::sys_process_dir
CVar * sys_process_dir
Definition: Application.cpp:162
RoR::App::gfx_vegetation_mode
CVar * gfx_vegetation_mode
Definition: Application.cpp:223
DECLARE_RESOURCE_PACK
#define DECLARE_RESOURCE_PACK(_FIELD_, _NAME_, _RESOURCE_GROUP_)
Definition: ContentManager.cpp:68
RoR::App::gfx_anisotropy
CVar * gfx_anisotropy
Definition: Application.cpp:225
RoR
Definition: AppContext.h:36
RoR::FileExists
bool FileExists(const char *path)
Path must be UTF-8 encoded.
Definition: PlatformUtils.cpp:163