RigsofRods
Soft-body Physics Simulation
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
AddonPartFileFormat.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-2023 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 "AddonPartFileFormat.h"
23 
24 #include "Actor.h"
25 #include "Application.h"
26 #include "CacheSystem.h"
27 #include "Console.h"
28 #include "GameContext.h"
29 #include "GenericFileFormat.h"
30 #include "GUI_MessageBox.h"
31 #include "RigDef_Parser.h"
32 #include "TuneupFileFormat.h"
33 
34 #include <Ogre.h>
35 
36 using namespace RoR;
37 using namespace RigDef;
38 
40  :m_silent_mode(silent_mode)
41 {
42  // Inits `RefCountingObjectPtr<>` (CacheEntryPtr, GenericDocumentPtr) - shouldn't be in header.
43 }
44 
46 {
47  // Destroys `RefCountingObjectPtr<>` (CacheEntryPtr, GenericDocumentPtr) - shouldn't be in header.
48 }
49 
50 std::shared_ptr<Document::Module> AddonPartUtility::TransformToRigDefModule(CacheEntryPtr& entry)
51 {
53  m_addonpart_entry = entry;
54  try
55  {
56  Ogre::DataStreamPtr datastream = Ogre::ResourceGroupManager::getSingleton().openResource(entry->fname, entry->resource_group);
57 
60  m_document->loadFromDataStream(datastream, options);
61 
62  m_module = std::shared_ptr<Document::Module>(new Document::Module(entry->dname));
63  m_module->origin_addonpart = entry;
65  Keyword keyword = Keyword::INVALID;
66  Keyword block = Keyword::INVALID;
67 
68  while (!m_context->endOfFile())
69  {
70  if (m_context->isTokKeyword())
71  {
72  // (ignore 'addonpart_*' directives)
73  if (m_context->getTokKeyword().find("addonpart_") != std::string::npos)
74  {
76  continue;
77  }
78 
79  keyword = Parser::IdentifyKeyword(m_context->getTokKeyword());
80  if (keyword == Keyword::INVALID && m_context->getTokKeyword() == "set_managedmaterials_options")
81  {
82  keyword = Keyword::SET_MANAGEDMATERIALS_OPTIONS; // Workaround, don't ask me why the regex doesn't match this.
83  }
84  if (keyword != Keyword::INVALID)
85  {
86  if (keyword == Keyword::SET_MANAGEDMATERIALS_OPTIONS)
87  {
90  continue;
91  }
92  else if (keyword == Keyword::MANAGEDMATERIALS
93  || keyword == Keyword::PROPS
94  || keyword == Keyword::FLEXBODIES
95  || keyword == Keyword::FLARES
96  || keyword == Keyword::FLARES2)
97  {
98  block = keyword;
100  continue;
101  }
102 
103  }
104  }
105 
106  if (block != Keyword::INVALID && !m_context->isTokComment() && !m_context->isTokLineBreak())
107  {
108  switch (block)
109  {
110  case Keyword::MANAGEDMATERIALS: this->ProcessManagedMaterial(); break;
111  case Keyword::PROPS: this->ProcessProp(); break;
112  case Keyword::FLEXBODIES: this->ProcessFlexbody(); break;
113  case Keyword::FLARES: this->ProcessFlare(); break;
114  case Keyword::FLARES2: this->ProcessFlare2(); break;
115  default: break;
116  }
117  }
118 
120  }
121  return m_module;
122  }
123  catch (Ogre::Exception& e)
124  {
127  fmt::format("Could not use addonpart: Error parsing file '{}', message: {}",
128  entry->fname, e.getFullDescription()));
129  return nullptr;
130  }
131 }
132 
134 {
135  // Evaluates 'addonpart_unwanted_*' directives, respecting 'protected_*' directives in the tuneup.
136  // Also handles 'addonpart_tweak_*' directives, resolving possible conflicts among used addonparts.
137  // ---------------------------------------------------------------------------------------------
138 
139  App::GetCacheSystem()->LoadResource(addonpart_entry);
140  m_addonpart_entry = addonpart_entry;
141  m_tuneup = tuneup;
142 
143  try
144  {
145  Ogre::DataStreamPtr datastream = Ogre::ResourceGroupManager::getSingleton().openResource(addonpart_entry->fname, addonpart_entry->resource_group);
146 
147  m_document = new GenericDocument();
149  m_document->loadFromDataStream(datastream, options);
151 
152  while (!m_context->endOfFile())
153  {
154  if (m_context->isTokKeyword())
155  {
156  if (m_context->getTokKeyword() == "addonpart_unwanted_prop" )
157  this->ProcessUnwantedProp();
158  else if (m_context->getTokKeyword() == "addonpart_unwanted_flexbody" )
159  this->ProcessUnwantedFlexbody();
160  else if (m_context->getTokKeyword() == "addonpart_unwanted_flare" )
161  this->ProcessUnwantedFlare();
162  else if (m_context->getTokKeyword() == "addonpart_unwanted_exhaust" )
163  this->ProcessUnwantedExhaust();
164  else if (m_context->getTokKeyword() == "addonpart_unwanted_managedmaterial")
166  else if (m_context->getTokKeyword() == "addonpart_tweak_wheel")
167  this->ProcessTweakWheel();
168  else if (m_context->getTokKeyword() == "addonpart_tweak_node")
169  this->ProcessTweakNode();
170  else if (m_context->getTokKeyword() == "addonpart_tweak_prop")
171  this->ProcessTweakProp();
172  else if (m_context->getTokKeyword() == "addonpart_tweak_flexbody")
173  this->ProcessTweakFlexbody();
174  else if (m_context->getTokKeyword() == "addonpart_tweak_managedmaterial")
175  this->ProcessTweakManagedMat();
176  }
177 
179  }
180 
181  }
182  catch (Ogre::Exception& e)
183  {
186  fmt::format("Addonpart unwanted elements check: Error parsing file '{}', message: {}",
187  addonpart_entry->fname, e.getFullDescription()));
188  }
189 }
190 
192 {
193  ROR_ASSERT(tuneup);
194 
195  // Unwanted
196  tuneup->unwanted_flexbodies.clear();
197  tuneup->unwanted_props.clear();
198  tuneup->unwanted_flares.clear();
199 
200  // Tweaked
201  tuneup->node_tweaks.clear();
202  tuneup->wheel_tweaks.clear();
203  tuneup->prop_tweaks.clear();
204  tuneup->flexbody_tweaks.clear();
205 }
206 
207 
208 // Helpers of `TransformToRigDefModule()`, they expect `m_context` to be in position:
209 // These expect `m_context` to be in position:
210 
212 {
213  ManagedMaterial def;
214  int n = m_context->countLineArgs();
215 
216  // Name:
217  def.name = m_context->getStringData(0); // It may be a STRING (if with quotes), or KEYWORD (if without quotes - because it's at start of line).
218 
219  // Type:
220  std::string str = m_context->getTokString(1);
221  if (str == "mesh_standard") def.type = ManagedMaterialType::MESH_STANDARD;
222  if (str == "mesh_transparent") def.type = ManagedMaterialType::MESH_TRANSPARENT;
223  if (str == "flexmesh_standard") def.type = ManagedMaterialType::FLEXMESH_STANDARD;
224  if (str == "flexmesh_transparent") def.type = ManagedMaterialType::FLEXMESH_TRANSPARENT;
225 
226  // Textures:
228  if (n > 3) def.specular_map = m_context->getTokString(3);
229  if (n > 4) def.damaged_diffuse_map = m_context->getTokString(4);
230  // (placeholders)
231  if (def.specular_map == "-") def.specular_map = "";
232  if (def.damaged_diffuse_map == "-") def.damaged_diffuse_map = "";
233 
234  // Options:
236 
237  m_module->managedmaterials.push_back(def);
238 }
239 
241 {
242  int n = m_context->countLineArgs();
243  if (n > 1)
244  {
246  }
247 }
248 
250 {
251  RigDef::Prop def;
252  int n = m_context->countLineArgs();
253  if (n < 10)
254  {
257  fmt::format("Error parsing addonpart file '{}': 'install_prop' has only {} arguments, expected {}",
258  m_addonpart_entry->fname, n, 10));
259  return;
260  }
261 
262  int importflags = Node::Ref::REGULAR_STATE_IS_VALID | Node::Ref::REGULAR_STATE_IS_NUMBERED;
263  def.reference_node = Node::Ref("", (unsigned int)m_context->getTokFloat(0), importflags, 0);
264  def.x_axis_node = Node::Ref("", (unsigned int)m_context->getTokFloat(1), importflags, 0);
265  def.y_axis_node = Node::Ref("", (unsigned int)m_context->getTokFloat(2), importflags, 0);
266 
267  def.offset.x = m_context->getTokNumeric(3);
268  def.offset.y = m_context->getTokNumeric(4);
269  def.offset.z = m_context->getTokNumeric(5);
270 
271  def.rotation.x = m_context->getTokNumeric(6);
272  def.rotation.y = m_context->getTokNumeric(7);
273  def.rotation.z = m_context->getTokNumeric(8);
274 
275  def.mesh_name = m_context->getTokString(9);
277 
278  switch (def.special)
279  {
280  case SpecialProp::BEACON:
281  if (n >= 14)
282  {
284  Ogre::StringUtil::trim(def.special_prop_beacon.flare_material_name);
285 
286  def.special_prop_beacon.color = Ogre::ColourValue(
288  }
289  break;
290 
291  case SpecialProp::DASHBOARD_LEFT:
292  case SpecialProp::DASHBOARD_RIGHT:
293  if (n > 10)
294  {
296  }
297  if (n > 13)
298  {
301  }
302  if (n > 14)
303  {
305  }
306  break;
307 
308  default:
309  break;
310  }
311 
312  m_module->props.push_back(def);
313 }
314 
316 {
317  Flexbody def;
318  int n = m_context->countLineArgs();
319  if (n < 10)
320  {
323  fmt::format("Error parsing addonpart file '{}': flexbody has only {} arguments, expected {}", m_addonpart_entry->fname, n, 10));
324  return;
325  }
326 
327  int importflags = Node::Ref::REGULAR_STATE_IS_VALID | Node::Ref::REGULAR_STATE_IS_NUMBERED;
328  def.reference_node = Node::Ref("", (unsigned int)m_context->getTokInt(0), importflags, 0);
329  def.x_axis_node = Node::Ref("", (unsigned int)m_context->getTokInt(1), importflags, 0);
330  def.y_axis_node = Node::Ref("", (unsigned int)m_context->getTokInt(2), importflags, 0);
331 
332  def.offset.x = m_context->getTokNumeric(3);
333  def.offset.y = m_context->getTokNumeric(4);
334  def.offset.z = m_context->getTokNumeric(5);
335 
336  def.rotation.x = m_context->getTokNumeric(6);
337  def.rotation.y = m_context->getTokNumeric(7);
338  def.rotation.z = m_context->getTokNumeric(8);
339 
340  def.mesh_name = m_context->getTokString(9);
341 
343 
344  if (!m_context->isTokString())
345  {
348  fmt::format("Error parsing addonpart file '{}': flexbody is not followed by 'forset'!", m_addonpart_entry->fname));
349  return;
350  }
351 
352  Parser::ProcessForsetLine(def, m_context->getTokString());
353 
354  // Resolve `forset` ranges:
355  for (RigDef::Node::Range const& range: def.node_list_to_import)
356  {
357  for (unsigned int i = range.start.Num(); i <= range.end.Num(); ++i)
358  {
359  Node::Ref ref("", i, Node::Ref::REGULAR_STATE_IS_VALID | Node::Ref::REGULAR_STATE_IS_NUMBERED, 0);
360  def.node_list.push_back(ref);
361  }
362  }
363 
364  m_module->flexbodies.push_back(def);
365 }
366 
368 {
369  int n = m_context->countLineArgs();
370  if (n < 5)
371  {
374  fmt::format("Error parsing addonpart file '{}': flare has only {} arguments, expected {}", m_addonpart_entry->fname, n, 5));
375  return;
376  }
377 
378  Flare2 def; // We auto-import 'flares' as 'flares2', leaving the `offset.z` at 0.
379  int importflags = Node::Ref::REGULAR_STATE_IS_VALID | Node::Ref::REGULAR_STATE_IS_NUMBERED;
380  def.reference_node = Node::Ref("", (unsigned int)m_context->getTokInt(0), importflags, 0);
381  def.node_axis_x = Node::Ref("", (unsigned int)m_context->getTokInt(1), importflags, 0);
382  def.node_axis_y = Node::Ref("", (unsigned int)m_context->getTokInt(2), importflags, 0);
383  def.offset.x = m_context->getTokNumeric(3);
384  def.offset.y = m_context->getTokNumeric(4);
385 
386  if (n > 5) def.type = (FlareType)m_context->getTokString(5)[0];
387 
388  if (n > 6)
389  {
390  switch (def.type)
391  {
392  case FlareType::USER: def.control_number = m_context->getTokInt(6); break;
394  default: break;
395  }
396  }
397 
398  if (n > 7) { def.blink_delay_milis = m_context->getTokInt(7); }
399  if (n > 8) { def.size = m_context->getTokNumeric(8); }
400  if (n > 9) { def.material_name = m_context->getTokString(9); }
401 
402  m_module->flares2.push_back(def);
403 }
404 
406 {
407  int n = m_context->countLineArgs();
408  if (n < 6)
409  {
412  fmt::format("Error parsing addonpart file '{}': flare2 has only {} arguments, expected {}", m_addonpart_entry->fname, n, 6));
413  return;
414  }
415 
416  Flare2 def;
417  int importflags = Node::Ref::REGULAR_STATE_IS_VALID | Node::Ref::REGULAR_STATE_IS_NUMBERED;
418  def.reference_node = Node::Ref("", (unsigned int)m_context->getTokInt(0), importflags, 0);
419  def.node_axis_x = Node::Ref("", (unsigned int)m_context->getTokInt(1), importflags, 0);
420  def.node_axis_y = Node::Ref("", (unsigned int)m_context->getTokInt(2), importflags, 0);
421  def.offset.x = m_context->getTokNumeric(3);
422  def.offset.y = m_context->getTokNumeric(4);
423  def.offset.z = m_context->getTokNumeric(5); // <-- Specific to 'flares2' (the only difference)
424 
425  if (n > 6) def.type = (FlareType)m_context->getTokString(6)[0];
426 
427  if (n > 7)
428  {
429  switch (def.type)
430  {
431  case FlareType::USER: def.control_number = m_context->getTokInt(7); break;
433  default: break;
434  }
435  }
436 
437  if (n > 8) { def.blink_delay_milis = m_context->getTokInt(8); }
438  if (n > 9) { def.size = m_context->getTokNumeric(9); }
439  if (n > 10) { def.material_name = m_context->getTokString(10); }
440 
441  m_module->flares2.push_back(def);
442 }
443 
444 // Helpers of `ResolveUnwantedAndTweakedElements()`, they expect `m_context` to be in position:
445 
447 {
448  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_unwanted_prop"); // also asserts !EOF and TokenType::KEYWORD
449 
450  if (m_context->isTokInt(1))
451  {
453  {
455  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': marking prop '{}' as UNWANTED",
457  }
458  else
459  {
460  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping prop '{}' because it's marked PROTECTED",
462  }
463  }
464  else
465  {
466  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
467  }
468 }
469 
471 {
472  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_unwanted_flexbody"); // also asserts !EOF and TokenType::KEYWORD
473 
474  if (m_context->isTokInt(1))
475  {
477  {
479  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': marking flexbody '{}' as UNWANTED",
481  }
482  else
483  {
484  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping flexbody '{}' because it's marked PROTECTED",
486  }
487  }
488  else
489  {
490  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
491  }
492 }
493 
495 {
496  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_unwanted_flare"); // also asserts !EOF and TokenType::KEYWORD
497 
498  if (m_context->isTokInt(1))
499  {
501  {
503  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': marking flare '{}' as UNWANTED",
505  }
506  else
507  {
508  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping flare '{}' because it's marked PROTECTED",
510  }
511  }
512  else
513  {
514  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
515  }
516 }
517 
519 {
520  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_unwanted_exhaust"); // also asserts !EOF and TokenType::KEYWORD
521 
522  if (m_context->isTokInt(1))
523  {
525  {
527  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': marking exhaust '{}' as UNWANTED",
529  }
530  else
531  {
532  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping exhaust '{}' because it's marked PROTECTED",
534  }
535  }
536  else
537  {
538  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
539  }
540 }
541 
543 {
544  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_unwanted_managedmaterial"); // also asserts !EOF and TokenType::KEYWORD
545 
546  if (m_context->isTokString(1))
547  {
548  std::string mat_name = m_context->getTokString(1);
549  if (!m_tuneup->isManagedMatProtected(mat_name))
550  {
551  m_tuneup->unwanted_managedmats.insert(mat_name);
552  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': marking managedmaterial '{}' as UNWANTED",
554  }
555  else
556  {
557  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping managedmaterial '{}' because it's marked PROTECTED",
559  }
560  }
561  else
562  {
563  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
564  }
565 }
566 
568 {
569  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_tweak_wheel"); // also asserts !EOF and TokenType::KEYWORD
570 
571  if (m_context->isTokInt(1) && m_context->isTokString(2))
572  {
573  const int wheel_id = m_context->getTokInt(1);
574  if (!m_tuneup->isWheelProtected(wheel_id))
575  {
576  if (m_tuneup->wheel_tweaks.find(wheel_id) == m_tuneup->wheel_tweaks.end())
577  {
578  TuneupWheelTweak data;
579  bool stop = false;
581  data.twt_wheel_id = wheel_id;
582  data.twt_media[0] = m_context->getTokString(2);
583  if (!stop && m_context->isTokString(3)) { data.twt_media[1] = m_context->getTokString(3); } else { stop=true; }
584  if (!stop && m_context->isTokString(4)) { data.twt_side = (m_context->getTokString(4)[0] == 'l') ? WheelSide::LEFT : WheelSide::RIGHT; } else { stop=true; }
585  if (!stop && m_context->isTokFloat(5)) { data.twt_tire_radius = m_context->getTokFloat(5); } else { stop=true; }
586  if (!stop && m_context->isTokFloat(6)) { data.twt_rim_radius = m_context->getTokFloat(6); } else { stop=true; }
587  m_tuneup->wheel_tweaks.insert(std::make_pair(wheel_id, data));
588 
589  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': Sheduling tweak for wheel '{}'"
590  " with params {{ media1={}, media2={}, side={}, tire_radius={}, rim_radius={} }}",
592  data.twt_media[0], data.twt_media[1], (char)data.twt_side, data.twt_tire_radius, data.twt_rim_radius));
593  }
594  else if (m_tuneup->wheel_tweaks[wheel_id].twt_origin != m_addonpart_entry->fname)
595  {
596  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': Resetting tweaks for wheel '{}' due to conflict with '{}'",
598  m_tuneup->wheel_tweaks[wheel_id].twt_origin));
599 
600  m_tuneup->wheel_tweaks.erase(wheel_id);
601  }
602  }
603  else
604  {
605  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping wheel '{}' because it's marked PROTECTED",
607  }
608  }
609  else
610  {
611  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
612  }
613 }
614 
616 {
617  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_tweak_node"); // also asserts !EOF and TokenType::KEYWORD
618 
620  {
621  NodeNum_t nodenum = (NodeNum_t)m_context->getTokInt(1);
622  if (!m_tuneup->isNodeProtected(nodenum))
623  {
624  if (m_tuneup->node_tweaks.find(nodenum) == m_tuneup->node_tweaks.end())
625  {
626  TuneupNodeTweak data;
628  data.tnt_nodenum = nodenum;
629  data.tnt_pos.x = m_context->getTokNumeric(2);
630  data.tnt_pos.y = m_context->getTokNumeric(3);
631  data.tnt_pos.z = m_context->getTokNumeric(4);
632  m_tuneup->node_tweaks.insert(std::make_pair(nodenum, data));
633 
634  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': Scheduling tweak for node '{}'"
635  " with params {{ x={}, y={}, z={} }}",
637  data.tnt_pos.x, data.tnt_pos.y, data.tnt_pos.z));
638  }
639  else if (m_tuneup->node_tweaks[nodenum].tnt_origin != m_addonpart_entry->fname)
640  {
641  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': Resetting tweaks for node '{}' due to conflict with '{}'",
643  m_tuneup->node_tweaks[nodenum].tnt_origin));
644 
645  m_tuneup->node_tweaks.erase(nodenum);
646  }
647  }
648  else
649  {
650  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping node '{}' because it's marked PROTECTED",
652  }
653  }
654  else
655  {
656  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
657  }
658 }
659 
661 {
662  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_tweak_flexbody"); // also asserts !EOF and TokenType::KEYWORD
663 
664  // TBD: add `null` token type to GenericDocument, so these params can be made optional
665  if (m_context->isTokInt(1) && // ID
666  m_context->isTokNumeric(2) && m_context->isTokNumeric(3) && m_context->isTokNumeric(4) && // offset
667  m_context->isTokNumeric(5) && m_context->isTokNumeric(6) && m_context->isTokNumeric(7) && // rotation
668  m_context->isTokString(8)) // media
669  {
670  const int flexbody_id = m_context->getTokInt(1);
671  if (!m_tuneup->isFlexbodyProtected(flexbody_id))
672  {
673  if (m_tuneup->flexbody_tweaks.find(flexbody_id) == m_tuneup->flexbody_tweaks.end())
674  {
675  TuneupFlexbodyTweak data;
677  data.tft_flexbody_id = flexbody_id;
678  data.tft_offset.x = m_context->getTokNumeric(2);
679  data.tft_offset.y = m_context->getTokNumeric(3);
680  data.tft_offset.z = m_context->getTokNumeric(4);
681  data.tft_rotation.x = m_context->getTokNumeric(5);
682  data.tft_rotation.y = m_context->getTokNumeric(6);
683  data.tft_rotation.z = m_context->getTokNumeric(7);
684  data.tft_media = m_context->getTokString(8);
685  m_tuneup->flexbody_tweaks.insert(std::make_pair(flexbody_id, data));
686 
687  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': Scheduling tweak for flexbody '{}'"
688  " with params {{ offsetX={}, offsetY={}, offsetZ={}, rotX={}, rotY={}, rotZ={}, media={} }}",
689  m_addonpart_entry->fname, m_context->getTokKeyword(), flexbody_id,
690  data.tft_offset.x, data.tft_offset.y, data.tft_offset.z,
691  data.tft_rotation.x, data.tft_rotation.y, data.tft_rotation.z, data.tft_media[0]));
692  }
693  else if (m_tuneup->flexbody_tweaks[flexbody_id].tft_origin != m_addonpart_entry->fname)
694  {
695  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': Resetting tweaks for flexbody '{}' due to conflict with '{}'",
697  m_tuneup->flexbody_tweaks[flexbody_id].tft_origin));
698 
699  m_tuneup->flexbody_tweaks.erase(flexbody_id);
700  }
701  }
702  else
703  {
704  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping flexbody '{}' because it's marked PROTECTED",
706  }
707  }
708  else
709  {
710  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
711  }
712 }
713 
715 {
716  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_tweak_prop"); // also asserts !EOF and TokenType::KEYWORD
717 
718  // TBD: add `null` token type to GenericDocument, so these params can be made optional
719  if (m_context->isTokInt(1) && // ID
720  m_context->isTokNumeric(2) && m_context->isTokNumeric(3) && m_context->isTokNumeric(4) && // offset
721  m_context->isTokNumeric(5) && m_context->isTokNumeric(6) && m_context->isTokNumeric(7) && // rotation
722  m_context->isTokString(8)) // media
723  {
724  const int prop_id = m_context->getTokInt(1);
725  if (!m_tuneup->isPropProtected(prop_id))
726  {
727  if (m_tuneup->prop_tweaks.find(prop_id) == m_tuneup->prop_tweaks.end())
728  {
729  TuneupPropTweak data;
731  data.tpt_prop_id = prop_id;
732 
733  data.tpt_offset.x = m_context->getTokNumeric(2);
734  data.tpt_offset.y = m_context->getTokNumeric(3);
735  data.tpt_offset.z = m_context->getTokNumeric(4);
736  data.tpt_rotation.x = m_context->getTokNumeric(5);
737  data.tpt_rotation.y = m_context->getTokNumeric(6);
738  data.tpt_rotation.z = m_context->getTokNumeric(7);
739  data.tpt_media[0] = m_context->getTokString(8);
740  if (m_context->isTokString(9)) data.tpt_media[1] = m_context->getTokString(9); // <== Optional Media2 is specific for prop
741  m_tuneup->prop_tweaks.insert(std::make_pair(prop_id, data));
742 
743  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': Scheduling tweak for prop '{}'"
744  " with params {{ media1={}, offsetX={}, offsetY={}, offsetZ={}, rotX={}, rotY={}, rotZ={}, media2={} }}",
745  m_addonpart_entry->fname, m_context->getTokKeyword(), prop_id, data.tpt_media[0],
746  data.tpt_offset.x, data.tpt_offset.y, data.tpt_offset.z,
747  data.tpt_rotation.x, data.tpt_rotation.y, data.tpt_rotation.z,
748  data.tpt_media[1]));
749  }
750  else if (m_tuneup->prop_tweaks[prop_id].tpt_origin != m_addonpart_entry->fname)
751  {
752  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': Resetting tweaks for prop '{}' due to conflict with '{}'",
754  m_tuneup->prop_tweaks[prop_id].tpt_origin));
755 
756  m_tuneup->prop_tweaks.erase(prop_id);
757  }
758  }
759  else
760  {
761  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping prop '{}' because it's marked PROTECTED",
763  }
764  }
765  else
766  {
767  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': bad arguments", m_addonpart_entry->fname, m_context->getTokKeyword()));
768  }
769 }
770 
772 {
773  ROR_ASSERT(m_context->getTokKeyword() == "addonpart_tweak_managedmaterial"); // also asserts !EOF and TokenType::KEYWORD
774 
776  {
777  const std::string& mat_name = m_context->getTokString(1);
778  if (!m_tuneup->isManagedMatProtected(mat_name))
779  {
780  if (m_tuneup->managedmat_tweaks.find(mat_name) == m_tuneup->managedmat_tweaks.end())
781  {
783  bool stop=false;
785  data.tmt_name = mat_name;
786  data.tmt_type = m_context->getTokString(2);
787  if (!stop && m_context->isTokString(3)) { data.tmt_media[0] = m_context->getTokString(3); } else {stop=true;}
788  if (!stop && m_context->isTokString(4)) { data.tmt_media[1] = m_context->getTokString(4); } else {stop=true;}
789  if (!stop && m_context->isTokString(5)) { data.tmt_media[2] = m_context->getTokString(5); } else {stop=true;}
790  m_tuneup->managedmat_tweaks.insert(std::make_pair(mat_name, data));
791 
792  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': Scheduling tweak for managed material '{}'"
793  " with params {{ type={}, media1={}, media2={}, media3={} }}",
794  m_addonpart_entry->fname, m_context->getTokKeyword(), mat_name, data.tmt_type, data.tmt_media[0], data.tmt_media[1], data.tmt_media[2]));
795  }
796  else if (m_tuneup->managedmat_tweaks[mat_name].tmt_origin != m_addonpart_entry->fname)
797  {
798  this->Log(fmt::format("[RoR|Addonpart] WARNING: file '{}', directive '{}': Resetting tweaks for managed material '{}' due to conflict with '{}'",
800  m_tuneup->managedmat_tweaks[mat_name].tmt_origin));
801 
802  m_tuneup->managedmat_tweaks.erase(mat_name);
803  }
804  }
805  else
806  {
807  this->Log(fmt::format("[RoR|Addonpart] INFO: file '{}', directive '{}': skipping managed material '{}' because it's marked PROTECTED",
809  }
810  }
811 }
812 
814 {
815  LOG(fmt::format("[RoR|Addonpart] -- Performing `RecordAddonpartConflicts()` between '{}' and '{}' ~ this involves generating dummy tuneups (hence messages below) --", addonpart1->fname, addonpart2->fname));
816 
817  // Make sure both addonparts are loaded and cached.
818  App::GetCacheSystem()->LoadResource(addonpart1);
819  if (!addonpart1->addonpart_data_only)
820  {
821  addonpart1->addonpart_data_only = new TuneupDef();
822  AddonPartUtility util(/*silent mode:*/true);
823  util.ResolveUnwantedAndTweakedElements(addonpart1->addonpart_data_only, addonpart1);
824  }
825 
826  App::GetCacheSystem()->LoadResource(addonpart2);
827  if (!addonpart2->addonpart_data_only)
828  {
829  addonpart2->addonpart_data_only = new TuneupDef();
830  AddonPartUtility util(/*silent mode:*/true);
831  util.ResolveUnwantedAndTweakedElements(addonpart2->addonpart_data_only, addonpart2);
832  }
833 
834  // NODE TWEAKS:
835  for (auto& i_pair: addonpart1->addonpart_data_only->node_tweaks)
836  {
837  NodeNum_t suspect = i_pair.second.tnt_nodenum;
838  TuneupNodeTweak* offender = nullptr;
839  if (TuneupUtil::isNodeTweaked(addonpart2->addonpart_data_only, suspect, offender))
840  {
841  conflicts.push_back(AddonPartConflict{addonpart1, addonpart2, "addonpart_tweak_node", (int)suspect});
842  LOG(fmt::format("[RoR|Addonpart] Found conflict between '{}' and '{}' - node {} is tweaked by both", addonpart1->fname, addonpart2->fname, (int)suspect));
843  }
844  }
845 
846  // WHEEL TWEAKS:
847  for (auto& i_pair: addonpart1->addonpart_data_only->wheel_tweaks)
848  {
849  WheelID_t suspect = i_pair.second.twt_wheel_id;
850  TuneupWheelTweak* offender = nullptr;
851  if (TuneupUtil::isWheelTweaked(addonpart2->addonpart_data_only, suspect, offender))
852  {
853  conflicts.push_back(AddonPartConflict{addonpart1, addonpart2, "addonpart_tweak_wheel", (int)suspect});
854  LOG(fmt::format("[RoR|Addonpart] Found conflict between '{}' and '{}' - wheel {} is tweaked by both", addonpart1->fname, addonpart2->fname, (int)suspect));
855  }
856  }
857 
858  // PROP TWEAKS:
859  for (auto& i_pair:addonpart1->addonpart_data_only->prop_tweaks)
860  {
861  PropID_t suspect = i_pair.second.tpt_prop_id;
862  TuneupPropTweak* offender = nullptr;
863  if (TuneupUtil::isPropTweaked(addonpart2->addonpart_data_only, suspect, offender))
864  {
865  conflicts.push_back(AddonPartConflict{addonpart1, addonpart2, "addonpart_tweak_prop", (int)suspect});
866  LOG(fmt::format("[RoR|Addonpart] Found conflict between '{}' and '{}' - prop {} is tweaked by both", addonpart1->fname, addonpart2->fname, (int)suspect));
867  }
868  }
869 
870  // FLEXBODY TWEAKS:
871  for (auto& i_pair: addonpart1->addonpart_data_only->flexbody_tweaks)
872  {
873  FlexbodyID_t suspect = i_pair.second.tft_flexbody_id;
874  TuneupFlexbodyTweak* offender = nullptr;
875  if (TuneupUtil::isFlexbodyTweaked(addonpart2->addonpart_data_only, suspect, offender))
876  {
877  conflicts.push_back(AddonPartConflict{addonpart1, addonpart2, "addonpart_tweak_flexbody", (int)suspect});
878  LOG(fmt::format("[RoR|Addonpart] Found conflict between '{}' and '{}' - flexbody {} is tweaked by both", addonpart1->fname, addonpart2->fname, (int)suspect));
879  }
880  }
881 
882  LOG(fmt::format("[RoR|Addonpart] -- Done with `RecordAddonpartConflicts()` between '{}' and '{}' --", addonpart1->fname, addonpart2->fname));
883 }
884 
886 {
887  if (!addonpart1 || !addonpart2)
888  {
889  return false;
890  }
891 
892  for (AddonPartConflict& conflict: conflicts)
893  {
894  if ((conflict.atc_addonpart1 == addonpart1 && conflict.atc_addonpart2 == addonpart2) ||
895  (conflict.atc_addonpart1 == addonpart2 && conflict.atc_addonpart2 == addonpart1))
896  {
897  return true;
898  }
899  }
900  return false;
901 }
902 
903 void AddonPartUtility::Log(const std::string& text)
904 {
905  if (!m_silent_mode)
906  {
907  LOG(text);
908  }
909 }
910 
912 {
913  // Re-check conflicts (request may come from 'Browse all' button or script).
914  // -------------------------------------------------------------------------
915 
916  AddonPartConflictVec conflicts;
917  for (const std::string& use_addonpart: target_actor->getWorkingTuneupDef()->use_addonparts)
918  {
919  CacheEntryPtr use_entry = App::GetCacheSystem()->FindEntryByFilename(LT_AddonPart, /*partial=*/false, use_addonpart);
920  AddonPartUtility::RecordAddonpartConflicts(addonpart_entry, use_entry, conflicts);
921  }
922 
923  if (conflicts.size() > 0)
924  {
925  // Messagebox text
927  dialog->mbc_content_width = 700.f;
928  dialog->mbc_title = _LC("Tuning", "Cannot install addon part, conflicts were detected.");
929  dialog->mbc_text = fmt::format(_LC("Tuning", "Requested addon part: '{}' (file '{}')."), addonpart_entry->dname, addonpart_entry->fname);
930  dialog->mbc_text += "\n";
931  dialog->mbc_text += fmt::format(_LC("Tuning", "Total conflicts: {}."), conflicts.size());
932  dialog->mbc_text += "\n";
933  for (size_t i=0; i < conflicts.size(); i++)
934  {
935  dialog->mbc_text += "\n";
936  dialog->mbc_text += fmt::format(_LC("Tuning", "[{}/{}] '{}' (file '{}') conflicts with '{}' #{}."),
937  i+1, conflicts.size(),
938  conflicts[i].atc_addonpart2->dname, conflicts[i].atc_addonpart2->fname,
939  conflicts[i].atc_keyword, conflicts[i].atc_element_id);
940  }
941 
942  // Messagebox OK button
943  GUI::MessageBoxButton ok_btn;
944  ok_btn.mbb_caption = _LC("Tuning", "OK");
946  dialog->mbc_buttons.push_back(ok_btn);
947 
948  // Show the messagebox
950  }
951  return conflicts.size() > 0;
952 }
RoR::TuneupDef
Dual purpose:
Definition: TuneupFileFormat.h:93
RoR::GUI::MessageBoxConfig::mbc_buttons
std::vector< MessageBoxButton > mbc_buttons
Definition: GUI_MessageBox.h:59
ROR_ASSERT
#define ROR_ASSERT(_EXPR)
Definition: Application.h:40
GameContext.h
Game state manager and message-queue provider.
RoR::AddonPartUtility::ResolveUnwantedAndTweakedElements
void ResolveUnwantedAndTweakedElements(TuneupDefPtr &tuneup, CacheEntryPtr &addonpart_entry)
Evaluates 'addonpart_unwanted_*' elements, respecting 'protected_*' directives in the tuneup.
Definition: AddonPartFileFormat.cpp:133
RoR::AddonPartUtility::m_module
std::shared_ptr< RigDef::Document::Module > m_module
Definition: AddonPartFileFormat.h:106
RoR::MSG_GUI_SHOW_MESSAGE_BOX_REQUESTED
@ MSG_GUI_SHOW_MESSAGE_BOX_REQUESTED
Payload = MessageBoxConfig* (owner)
Definition: Application.h:142
RoR::AddonPartUtility::m_document
GenericDocumentPtr m_document
Definition: AddonPartFileFormat.h:102
RigDef::Prop::DashboardSpecial::offset
Ogre::Vector3 offset
Definition: RigDef_File.h:1096
RoR::TuneupWheelTweak::twt_origin
std::string twt_origin
Addonpart filename.
Definition: TuneupFileFormat.h:60
RoR::CacheEntry::dname
Ogre::String dname
name parsed from the file
Definition: CacheSystem.h:70
RigDef::Node::Ref
Legacy parser resolved references on-the-fly and the condition to check named nodes was "are there an...
Definition: RigDef_Node.h:77
RigDef::Prop
Definition: RigDef_File.h:1085
RoR::TuneupNodeTweak::tnt_nodenum
NodeNum_t tnt_nodenum
Arg#1, required.
Definition: TuneupFileFormat.h:46
RoR::GenericDocContext::isTokInt
bool isTokInt(int offset=0) const
Definition: GenericFileFormat.h:123
RigDef::Prop::offset
Ogre::Vector3 offset
Definition: RigDef_File.h:1116
RoR::TuneupUtil::isWheelTweaked
static bool isWheelTweaked(TuneupDefPtr &tuneup_entry, WheelID_t wheel_id, TuneupWheelTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:272
RoR::TuneupDef::isManagedMatProtected
bool isManagedMatProtected(const std::string &matname) const
Definition: TuneupFileFormat.h:156
RoR::TuneupWheelTweak::twt_tire_radius
float twt_tire_radius
Arg#5, optional.
Definition: TuneupFileFormat.h:58
RigDef::ManagedMaterial::diffuse_map
Ogre::String diffuse_map
Definition: RigDef_File.h:1033
RigDef::Flare2::type
RoR::FlareType type
Definition: RigDef_File.h:883
RoR::TuneupDef::unwanted_props
std::set< PropID_t > unwanted_props
'addonpart_unwanted_prop' directives.
Definition: TuneupFileFormat.h:116
RoR::LT_AddonPart
@ LT_AddonPart
Definition: Application.h:321
RoR::TuneupPropTweak
< Data of 'addonpart_tweak_prop <prop ID> <offsetX> <offsetY> <offsetZ> <rotX> <rotY> <rotZ> <media1>...
Definition: TuneupFileFormat.h:64
RoR::TuneupWheelTweak::twt_rim_radius
float twt_rim_radius
Arg#6, optional, only applies to some wheel types.
Definition: TuneupFileFormat.h:59
RoR::TuneupFlexbodyTweak::tft_rotation
Ogre::Vector3 tft_rotation
Definition: TuneupFileFormat.h:78
RoR::GenericDocContext::getTokNumeric
float getTokNumeric(int offset=0) const
Definition: GenericFileFormat.h:116
RoR::GenericDocContext::isTokKeyword
bool isTokKeyword(int offset=0) const
Definition: GenericFileFormat.h:125
GUI_MessageBox.h
Generic UI dialog (not modal). Invocable from scripting. Any number of buttons. Configurable to fire ...
RigDef::Node::Range::start
Node::Ref start
Definition: RigDef_Node.h:134
RoR::AddonPartUtility::ProcessUnwantedFlare
void ProcessUnwantedFlare()
Definition: AddonPartFileFormat.cpp:494
RoR::TuneupUtil::isPropTweaked
static bool isPropTweaked(TuneupDefPtr &tuneup_entry, PropID_t flexbody_id, TuneupPropTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:394
RoR::AddonPartUtility::ProcessUnwantedExhaust
void ProcessUnwantedExhaust()
Definition: AddonPartFileFormat.cpp:518
RoR::TuneupFlexbodyTweak::tft_origin
std::string tft_origin
Addonpart filename.
Definition: TuneupFileFormat.h:79
RoR::TuneupManagedMatTweak::tmt_type
std::string tmt_type
Arg#2, required.
Definition: TuneupFileFormat.h:85
RigDef::Prop::DashboardSpecial::mesh_name
Ogre::String mesh_name
Definition: RigDef_File.h:1099
RoR::TuneupFlexbodyTweak::tft_media
std::string tft_media
Definition: TuneupFileFormat.h:76
RoR::TuneupDef::unwanted_managedmats
std::set< std::string > unwanted_managedmats
'addonpart_unwanted_managedmaterial' directives.
Definition: TuneupFileFormat.h:120
RigDef::Prop::rotation
Ogre::Vector3 rotation
Definition: RigDef_File.h:1117
RigDef::ManagedMaterial::damaged_diffuse_map
Ogre::String damaged_diffuse_map
Definition: RigDef_File.h:1034
RoR::TuneupDef::prop_tweaks
std::map< PropID_t, TuneupPropTweak > prop_tweaks
Mesh name(s), offset and rotation overrides via 'addonpart_tweak_prop'.
Definition: TuneupFileFormat.h:113
format
Truck file format(technical spec)
RoR::TuneupDef::unwanted_flexbodies
std::set< FlexbodyID_t > unwanted_flexbodies
'addonpart_unwanted_flexbody' directives.
Definition: TuneupFileFormat.h:117
RoR::GenericDocument::OPTION_ALLOW_SLASH_COMMENTS
static const BitMask_t OPTION_ALLOW_SLASH_COMMENTS
Allow comments starting with //.
Definition: GenericFileFormat.h:69
RigDef::Flexbody::rotation
Ogre::Vector3 rotation
Definition: RigDef_File.h:908
RoR::GenericDocContext::isTokLineBreak
bool isTokLineBreak(int offset=0) const
Definition: GenericFileFormat.h:127
RigDef::Flexbody::offset
Ogre::Vector3 offset
Definition: RigDef_File.h:907
RoR::FlareType::USER
@ USER
RigDef::Flare2
Definition: RigDef_File.h:877
RoR::AddonPartUtility::m_context
GenericDocContextPtr m_context
Definition: AddonPartFileFormat.h:103
Console.h
RoR::GenericDocContext::getTokString
std::string getTokString(int offset=0) const
Definition: GenericFileFormat.h:113
RoR::AddonPartUtility::ProcessUnwantedManagedMat
void ProcessUnwantedManagedMat()
Definition: AddonPartFileFormat.cpp:542
RoR::TuneupDef::isFlexbodyProtected
bool isFlexbodyProtected(FlexbodyID_t flexbodyid) const
Definition: TuneupFileFormat.h:151
RoR::Console::putMessage
void putMessage(MessageArea area, MessageType type, std::string const &msg, std::string icon="")
Definition: Console.cpp:103
RigDef::Flare2::node_axis_y
Node::Ref node_axis_y
Definition: RigDef_File.h:881
RoR::GenericDocContext::isTokFloat
bool isTokFloat(int offset=0) const
Definition: GenericFileFormat.h:122
RoR::AddonPartUtility::TransformToRigDefModule
std::shared_ptr< RigDef::Document::Module > TransformToRigDefModule(CacheEntryPtr &addonpart_entry)
transforms the addonpart to RigDef::File::Module (fake 'section/end_section') used for spawning.
Definition: AddonPartFileFormat.cpp:50
RoR::TuneupPropTweak::tpt_prop_id
PropID_t tpt_prop_id
Definition: TuneupFileFormat.h:66
RigDef::Flare2::blink_delay_milis
int blink_delay_milis
Definition: RigDef_File.h:886
RigDef::Document::Module
Definition: RigDef_File.h:1484
RoR::CacheEntry::resource_group
Ogre::String resource_group
Resource group of the loaded bundle. Empty if not loaded yet.
Definition: CacheSystem.h:89
RoR::GUI::MessageBoxButton::mbb_caption
std::string mbb_caption
Definition: GUI_MessageBox.h:43
RoR::GenericDocContext::seekNextLine
bool seekNextLine()
Definition: GenericFileFormat.cpp:1129
RoR::FlareType
FlareType
Definition: SimData.h:228
TuneupFileFormat.h
The vehicle tuning system; applies addonparts and user overrides to vehicles.
RoR::AddonPartUtility::m_silent_mode
bool m_silent_mode
To block logging during conflict resolution (which works by generating dummy tuneups - would confuse ...
Definition: AddonPartFileFormat.h:110
RoR::TuneupDef::use_addonparts
std::set< std::string > use_addonparts
Addonpart filenames.
Definition: TuneupFileFormat.h:109
RigDef::Node::Range
Definition: RigDef_Node.h:114
RoR::GenericDocContext::getTokFloat
float getTokFloat(int offset=0) const
Definition: GenericFileFormat.h:114
RoR::CacheEntry::addonpart_data_only
TuneupDefPtr addonpart_data_only
Cached addonpart data (dummy tuneup), only used for evaluating conflicts, see AddonPartUtility::Recor...
Definition: CacheSystem.h:94
RoR::AddonPartUtility::ProcessTweakProp
void ProcessTweakProp()
Definition: AddonPartFileFormat.cpp:714
RigDef::Flexbody::reference_node
Node::Ref reference_node
Definition: RigDef_File.h:904
RoR::TuneupPropTweak::tpt_rotation
Ogre::Vector3 tpt_rotation
Definition: TuneupFileFormat.h:69
RoR::AddonPartConflictVec
std::vector< AddonPartConflict > AddonPartConflictVec
Definition: AddonPartFileFormat.h:53
RefCountingObjectPtr< CacheEntry >
RoR::AddonPartUtility::ProcessTweakNode
void ProcessTweakNode()
Definition: AddonPartFileFormat.cpp:615
RoR::AddonPartUtility::ProcessManagedMaterial
void ProcessManagedMaterial()
Definition: AddonPartFileFormat.cpp:211
Actor.h
RoR::ExhaustID_t
int ExhaustID_t
Index into GfxActor::m_exhausts, use RoR::EXHAUSTID_INVALID as empty value.
Definition: ForwardDeclarations.h:73
RoR::TuneupWheelTweak::twt_wheel_id
WheelID_t twt_wheel_id
Arg#1, required.
Definition: TuneupFileFormat.h:53
RoR::AddonPartUtility::m_tuneup
TuneupDefPtr m_tuneup
Definition: AddonPartFileFormat.h:109
RoR::TuneupDef::isPropProtected
bool isPropProtected(PropID_t propid) const
Definition: TuneupFileFormat.h:150
RoR::GenericDocContext::isTokString
bool isTokString(int offset=0) const
Definition: GenericFileFormat.h:121
RoR::AddonPartUtility::ProcessDirectiveSetManagedMaterialsOptions
void ProcessDirectiveSetManagedMaterialsOptions()
Definition: AddonPartFileFormat.cpp:240
RoR::TuneupUtil::isNodeTweaked
static bool isNodeTweaked(TuneupDefPtr &tuneup_entry, NodeNum_t nodenum, TuneupNodeTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:303
RigDef::ManagedMaterial::type
ManagedMaterialType type
Definition: RigDef_File.h:1031
RoR::GenericDocContext::countLineArgs
int countLineArgs()
Definition: GenericFileFormat.cpp:1147
RoR::TuneupFlexbodyTweak::tft_flexbody_id
FlexbodyID_t tft_flexbody_id
Definition: TuneupFileFormat.h:75
RigDef::ManagedMaterial::options
ManagedMaterialsOptions options
Definition: RigDef_File.h:1032
RigDef::ManagedMaterial::name
Ogre::String name
Definition: RigDef_File.h:1030
RigDef::Prop::special
SpecialProp special
Definition: RigDef_File.h:1121
RoR::AddonPartUtility::ProcessUnwantedProp
void ProcessUnwantedProp()
Definition: AddonPartFileFormat.cpp:446
RoR::FlareType::DASHBOARD
@ DASHBOARD
RoR::AddonPartUtility::ResetUnwantedAndTweakedElements
static void ResetUnwantedAndTweakedElements(TuneupDefPtr &tuneup)
Definition: AddonPartFileFormat.cpp:191
RoR::TuneupDef::managedmat_tweaks
std::map< std::string, TuneupManagedMatTweak > managedmat_tweaks
Managed material overrides via 'addonpart_tweak_managedmaterial'.
Definition: TuneupFileFormat.h:115
RigDef::Flare2::material_name
Ogre::String material_name
Definition: RigDef_File.h:888
RoR::TuneupDef::node_tweaks
std::map< NodeNum_t, TuneupNodeTweak > node_tweaks
Node position overrides via 'addonpart_tweak_node'.
Definition: TuneupFileFormat.h:111
RoR::TuneupDef::unwanted_flares
std::set< FlareID_t > unwanted_flares
'addonpart_unwanted_flare' directives.
Definition: TuneupFileFormat.h:118
RoR::TuneupManagedMatTweak
< Data of 'addonpart_tweak_managedmaterial <name> <type> <media1> <media2> [<media3>]'
Definition: TuneupFileFormat.h:82
RigDef::Flare2::dashboard_link
std::string dashboard_link
Only 'd' type flares.
Definition: RigDef_File.h:885
RoR::TuneupManagedMatTweak::tmt_origin
std::string tmt_origin
Addonpart filename.
Definition: TuneupFileFormat.h:87
RoR::NodeNum_t
uint16_t NodeNum_t
Node position within Actor::ar_nodes; use RoR::NODENUM_INVALID as empty value.
Definition: ForwardDeclarations.h:54
RoR::CacheSystem::FindEntryByFilename
CacheEntryPtr FindEntryByFilename(RoR::LoaderType type, bool partial, const std::string &_filename_maybe_bundlequalified)
Returns NULL if none found; "Bundle-qualified" format also specifies the ZIP/directory in modcache,...
Definition: CacheSystem.cpp:186
RoR::GUI::MessageBoxButton
Definition: GUI_MessageBox.h:41
RoR::TuneupNodeTweak
< Data of 'addonpart_tweak_node <nodenum> <posX> <posY> <posZ>'
Definition: TuneupFileFormat.h:44
RoR::AddonPartUtility::RecordAddonpartConflicts
static void RecordAddonpartConflicts(CacheEntryPtr addonpart1, CacheEntryPtr addonpart2, AddonPartConflictVec &conflicts)
Definition: AddonPartFileFormat.cpp:813
CacheSystem.h
A database of user-installed content alias 'mods' (vehicles, terrains...)
keyword
static int keyword
Definition: Bench_TruckParser_IdentifyKeyword.cpp:1448
RigDef::Parser::IdentifySpecialProp
static SpecialProp IdentifySpecialProp(const std::string &str)
Definition: RigDef_Parser.cpp:3325
RigDef
Definition: ForwardDeclarations.h:276
RigDef::Keyword
Keyword
Definition: RigDef_File.h:65
RigDef::Node::Range::end
Node::Ref end
Definition: RigDef_Node.h:135
RoR::TuneupWheelTweak::twt_side
WheelSide twt_side
Arg#4, optional, default LEFT (Only applicable to mesh/flexbody wheels)
Definition: TuneupFileFormat.h:57
RoR::GenericDocContext::isTokNumeric
bool isTokNumeric(int offset=0) const
Definition: GenericFileFormat.h:128
RoR::TuneupDef::flexbody_tweaks
std::map< FlexbodyID_t, TuneupFlexbodyTweak > flexbody_tweaks
Mesh name, offset and rotation overrides via 'addonpart_tweak_flexbody'.
Definition: TuneupFileFormat.h:114
RoR::AddonPartUtility::ProcessTweakWheel
void ProcessTweakWheel()
Definition: AddonPartFileFormat.cpp:567
RoR::GameContext::PushMessage
void PushMessage(Message m)
Doesn't guarantee order! Use ChainMessage() if order matters.
Definition: GameContext.cpp:66
RigDef::Flare2::reference_node
Node::Ref reference_node
Definition: RigDef_File.h:879
RoR::WheelID_t
int WheelID_t
Index to Actor::ar_wheels, use RoR::WHEELID_INVALID as empty value.
Definition: ForwardDeclarations.h:58
RoR::GUI::MessageBoxConfig::mbc_title
std::string mbc_title
Definition: GUI_MessageBox.h:52
RigDef::ManagedMaterial::specular_map
Ogre::String specular_map
Definition: RigDef_File.h:1035
RoR::TuneupNodeTweak::tnt_pos
Ogre::Vector3 tnt_pos
Args#234, required.
Definition: TuneupFileFormat.h:47
RigDef::Flare2::offset
Ogre::Vector3 offset
Definition: RigDef_File.h:882
RigDef::Prop::DashboardSpecial::_offset_is_set
bool _offset_is_set
Definition: RigDef_File.h:1097
RoR::AddonPartUtility::m_addonpart_entry
CacheEntryPtr m_addonpart_entry
Definition: AddonPartFileFormat.h:104
RoR::TuneupDef::isWheelProtected
bool isWheelProtected(WheelID_t wheelid) const
Definition: TuneupFileFormat.h:152
RigDef::Flexbody::node_list_to_import
std::vector< Node::Range > node_list_to_import
Definition: RigDef_File.h:911
RoR::WheelSide::RIGHT
@ RIGHT
RigDef::ManagedMaterial
Definition: RigDef_File.h:1028
RoR::WheelSide::LEFT
@ LEFT
Application.h
Central state/object manager and communications hub.
RoR::App::GetConsole
Console * GetConsole()
Definition: Application.cpp:274
RoR::TuneupPropTweak::tpt_origin
std::string tpt_origin
Addonpart filename.
Definition: TuneupFileFormat.h:70
RigDef::Flexbody::mesh_name
Ogre::String mesh_name
Definition: RigDef_File.h:909
RoR::AddonPartConflict
< Conflict between two addonparts tweaking the same element
Definition: AddonPartFileFormat.h:38
RoR::AddonPartUtility::DoubleCheckForAddonpartConflict
static bool DoubleCheckForAddonpartConflict(ActorPtr target_actor, CacheEntryPtr addonpart_entry)
Definition: AddonPartFileFormat.cpp:911
RoR::App::GetGameContext
GameContext * GetGameContext()
Definition: Application.cpp:284
RoR::GUI::MessageBoxConfig::mbc_text
std::string mbc_text
Definition: GUI_MessageBox.h:53
RoR::TuneupDef::unwanted_exhausts
std::set< ExhaustID_t > unwanted_exhausts
'addonpart_unwanted_exhaust' directives.
Definition: TuneupFileFormat.h:119
RigDef::Prop::mesh_name
Ogre::String mesh_name
Definition: RigDef_File.h:1118
RoR::AddonPartUtility::Log
void Log(const std::string &text)
Definition: AddonPartFileFormat.cpp:903
RigDef::Flare2::node_axis_x
Node::Ref node_axis_x
Definition: RigDef_File.h:880
_LC
#define _LC(ctx, str)
Definition: Language.h:38
RigDef::Flexbody
Definition: RigDef_File.h:902
RoR::GUI::MessageBoxButton::mbb_mq_message
MsgType mbb_mq_message
Message to queue on click.
Definition: GUI_MessageBox.h:44
RoR::GenericDocContext::getStringData
const char * getStringData(int offset=0) const
Definition: GenericFileFormat.h:154
RoR::GenericDocContext::endOfFile
bool endOfFile(int offset=0) const
Definition: GenericFileFormat.h:110
RoR::AddonPartUtility::ProcessTweakFlexbody
void ProcessTweakFlexbody()
Definition: AddonPartFileFormat.cpp:660
RigDef::Prop::y_axis_node
Node::Ref y_axis_node
Definition: RigDef_File.h:1115
RoR::AddonPartUtility::CheckForAddonpartConflict
static bool CheckForAddonpartConflict(CacheEntryPtr addonpart1, CacheEntryPtr addonpart2, AddonPartConflictVec &conflicts)
Definition: AddonPartFileFormat.cpp:885
RoR::TuneupWheelTweak::twt_media
std::array< std::string, 2 > twt_media
twt_media[0] Arg#2, required ('wheels[2]': face material, 'meshwheels[2]/flexbodywheels': rim mesh) t...
Definition: TuneupFileFormat.h:56
RoR::GenericDocContext::getFloatData
float getFloatData(int offset=0) const
Definition: GenericFileFormat.h:155
RoR::App::GetCacheSystem
CacheSystem * GetCacheSystem()
Definition: Application.cpp:276
RigDef::ManagedMaterialsOptions::double_sided
bool double_sided
Definition: RigDef_File.h:1025
RoR::TuneupPropTweak::tpt_media
std::array< std::string, 2 > tpt_media
Media1 = prop mesh; Media2: Steering wheel mesh or beacon flare material.
Definition: TuneupFileFormat.h:67
RigDef::Flexbody::node_list
std::vector< Node::Ref > node_list
Definition: RigDef_File.h:912
RoR::TuneupNodeTweak::tnt_origin
std::string tnt_origin
Addonpart filename.
Definition: TuneupFileFormat.h:48
RoR::TuneupPropTweak::tpt_offset
Ogre::Vector3 tpt_offset
Definition: TuneupFileFormat.h:68
RoR::GUI::MessageBoxConfig
Definition: GUI_MessageBox.h:50
RigDef::Flare2::size
float size
Definition: RigDef_File.h:887
RoR::CacheSystem::LoadResource
void LoadResource(CacheEntryPtr &t)
Loads the associated resource bundle if not already done.
Definition: CacheSystem.cpp:1538
RoR::GenericDocument
Definition: GenericFileFormat.h:66
RoR::Message
Unified game event system - all requests and state changes are reported using a message.
Definition: GameContext.h:51
RoR::AddonPartUtility::m_managedmaterials_options
RigDef::ManagedMaterialsOptions m_managedmaterials_options
Definition: AddonPartFileFormat.h:107
RigDef_Parser.h
Checks the rig-def file syntax and loads data to memory.
RoR::FlareID_t
int FlareID_t
Index into Actor::ar_flares, use RoR::FLAREID_INVALID as empty value.
Definition: ForwardDeclarations.h:70
RoR::AddonPartUtility::ProcessFlare2
void ProcessFlare2()
Definition: AddonPartFileFormat.cpp:405
RoR::TuneupDef::isFlareProtected
bool isFlareProtected(FlareID_t flareid) const
Definition: TuneupFileFormat.h:154
RoR::GenericDocContext::getTokKeyword
std::string getTokKeyword(int offset=0) const
Definition: GenericFileFormat.h:118
RigDef::Prop::special_prop_dashboard
DashboardSpecial special_prop_dashboard
Definition: RigDef_File.h:1123
RoR::TuneupManagedMatTweak::tmt_name
std::string tmt_name
Arg#1, required.
Definition: TuneupFileFormat.h:84
RoR::TuneupWheelTweak
< Data of 'addonpart_tweak_wheel <wheel ID> <media1> <media2> <side flag> <tire radius> <rim radius>'
Definition: TuneupFileFormat.h:51
RoR::GUI::MessageBoxConfig::mbc_content_width
float mbc_content_width
Parameter to ImGui::SetContentWidth() - hard limit on content size.
Definition: GUI_MessageBox.h:57
RigDef::Flare2::control_number
int control_number
Only 'u' type flares.
Definition: RigDef_File.h:884
BitMask_t
uint32_t BitMask_t
Definition: BitFlags.h:7
RoR::GenericDocContext
Definition: GenericFileFormat.h:89
RoR::AddonPartUtility::ProcessUnwantedFlexbody
void ProcessUnwantedFlexbody()
Definition: AddonPartFileFormat.cpp:470
RoR::AddonPartUtility::ProcessProp
void ProcessProp()
Definition: AddonPartFileFormat.cpp:249
RoR::Console::CONSOLE_MSGTYPE_ACTOR
@ CONSOLE_MSGTYPE_ACTOR
Parsing/spawn/simulation messages for actors.
Definition: Console.h:63
AddonPartFileFormat.h
RoR::GenericDocument::loadFromDataStream
virtual void loadFromDataStream(Ogre::DataStreamPtr datastream, BitMask_t options=0)
Definition: GenericFileFormat.cpp:1007
RoR::Console::CONSOLE_SYSTEM_WARNING
@ CONSOLE_SYSTEM_WARNING
Definition: Console.h:53
RoR::TuneupDef::isExhaustProtected
bool isExhaustProtected(ExhaustID_t exhaustid) const
Definition: TuneupFileFormat.h:155
RoR::AddonPartUtility::AddonPartUtility
AddonPartUtility(bool silent_mode=false)
Definition: AddonPartFileFormat.cpp:39
RoR::AddonPartUtility
NOTE: Modcache processes this format directly using RoR::GenericDocument, see RoR::CacheSystem::FillA...
Definition: AddonPartFileFormat.h:56
RoR::FlexbodyID_t
int FlexbodyID_t
Index to GfxActor::m_flexbodies, use RoR::FLEXBODYID_INVALID as empty value.
Definition: ForwardDeclarations.h:64
RigDef::Prop::reference_node
Node::Ref reference_node
Definition: RigDef_File.h:1113
RoR::TuneupUtil::isFlexbodyTweaked
static bool isFlexbodyTweaked(TuneupDefPtr &tuneup_entry, FlexbodyID_t flexbody_id, TuneupFlexbodyTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:485
RigDef::Flexbody::x_axis_node
Node::Ref x_axis_node
Definition: RigDef_File.h:905
RoR::GenericDocContext::isTokComment
bool isTokComment(int offset=0) const
Definition: GenericFileFormat.h:126
RoR::TuneupFlexbodyTweak::tft_offset
Ogre::Vector3 tft_offset
Definition: TuneupFileFormat.h:77
RoR::MSG_GUI_HIDE_MESSAGE_BOX_REQUESTED
@ MSG_GUI_HIDE_MESSAGE_BOX_REQUESTED
Definition: Application.h:143
RoR::AddonPartUtility::~AddonPartUtility
~AddonPartUtility()
Definition: AddonPartFileFormat.cpp:45
GenericFileFormat.h
Generic text file parser.
RoR::TuneupFlexbodyTweak
< Data of 'addonpart_tweak_flexbody <flexbody ID> <offsetX> <offsetY> <offsetZ> <rotX> <rotY> <rotZ> ...
Definition: TuneupFileFormat.h:73
RigDef::Node::Ref::Num
unsigned int Num() const
Definition: RigDef_Node.h:95
RoR::GenericDocument::OPTION_ALLOW_NAKED_STRINGS
static const BitMask_t OPTION_ALLOW_NAKED_STRINGS
Allow strings without quotes, for backwards compatibility.
Definition: GenericFileFormat.h:68
RigDef::Prop::x_axis_node
Node::Ref x_axis_node
Definition: RigDef_File.h:1114
RigDef::Prop::DashboardSpecial::rotation_angle
float rotation_angle
Definition: RigDef_File.h:1098
RoR::Actor::getWorkingTuneupDef
TuneupDefPtr & getWorkingTuneupDef()
Definition: Actor.cpp:4673
RoR
Definition: AppContext.h:36
RoR::TuneupManagedMatTweak::tmt_media
std::array< std::string, 3 > tmt_media
Arg#3, required, Arg#4, optional, Arg#5, optional.
Definition: TuneupFileFormat.h:86
RoR::TuneupDef::wheel_tweaks
std::map< WheelID_t, TuneupWheelTweak > wheel_tweaks
Mesh name and radius overrides via 'addonpart_tweak_wheel'.
Definition: TuneupFileFormat.h:112
RigDef::Prop::special_prop_beacon
BeaconSpecial special_prop_beacon
Definition: RigDef_File.h:1122
RigDef::Flexbody::y_axis_node
Node::Ref y_axis_node
Definition: RigDef_File.h:906
RoR::PropID_t
int PropID_t
Index to GfxActor::m_props, use RoR::PROPID_INVALID as empty value.
Definition: ForwardDeclarations.h:61
RoR::AddonPartUtility::ProcessFlexbody
void ProcessFlexbody()
Definition: AddonPartFileFormat.cpp:315
RoR::TuneupDef::isNodeProtected
bool isNodeProtected(NodeNum_t nodenum) const
Definition: TuneupFileFormat.h:153
RigDef::Prop::BeaconSpecial::flare_material_name
Ogre::String flare_material_name
Definition: RigDef_File.h:1109
RoR::AddonPartUtility::ProcessFlare
void ProcessFlare()
Definition: AddonPartFileFormat.cpp:367
RoR::AddonPartUtility::ProcessTweakManagedMat
void ProcessTweakManagedMat()
Definition: AddonPartFileFormat.cpp:771
RoR::CacheEntry::fname
Ogre::String fname
filename
Definition: CacheSystem.h:67
RoR::GenericDocContext::getTokInt
int getTokInt(int offset=0) const
Definition: GenericFileFormat.h:115
RigDef::Prop::BeaconSpecial::color
Ogre::ColourValue color
Definition: RigDef_File.h:1110