RigsofRods
Soft-body Physics Simulation
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
TuneupFileFormat.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-2024 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 "TuneupFileFormat.h"
23 
24 #include "Actor.h"
25 #include "Application.h"
26 #include "CacheSystem.h"
27 #include "Console.h"
28 #include "Utils.h"
29 
30 #include <OgreEntity.h>
31 #include <OgreMaterialManager.h>
32 #include <OgrePass.h>
33 #include <OgreSubEntity.h>
34 #include <OgreTechnique.h>
35 
36 using namespace RoR;
37 
39 {
40  TuneupDefPtr ret = new TuneupDef();
41 
42  // General info
43  ret->name = this->name ; //std::string
44  ret->guid = this->guid ; //std::string
45  ret->thumbnail = this->thumbnail ; //std::string
46  ret->description = this->description ; //std::string
47  ret->author_name = this->author_name ; //std::string
48  ret->author_id = this->author_id ; //int
49  ret->category_id = this->category_id ; //CacheCategoryId
50  ret->filename = this->filename ; //std::string
51 
52  // addonparts
53  ret->use_addonparts = this->use_addonparts ;
54 
55  // props
56  ret->protected_props = this->protected_props;
57  ret->unwanted_props = this->unwanted_props;
59  ret->prop_tweaks = this->prop_tweaks;
60 
61  // flexbodies
65  ret->flexbody_tweaks = this->flexbody_tweaks;
66 
67  // wheels
70  ret->wheel_tweaks = this->wheel_tweaks;
71 
72  // nodes
73  ret->protected_nodes = this->protected_nodes;
74  ret->node_tweaks = this->node_tweaks;
75 
76  // video cameras
78 
79  // flares
81 
82  // exhausts
84 
85  // managed materials
87 
88  return ret;
89 }
90 
92 {
93  // addonparts
94  this->use_addonparts.clear();
95 
96  // props
97  this->protected_props.clear();
98  this->unwanted_props.clear();
99  this->force_remove_props.clear();
100  this->prop_tweaks.clear();
101 
102  // flexbodies
103  this->protected_flexbodies.clear();
104  this->unwanted_flexbodies.clear();
105  this->force_remove_flexbodies.clear();
106  this->flexbody_tweaks.clear();
107 
108  // wheels
109  this->protected_wheels.clear();
110  this->force_wheel_sides.clear();
111  this->wheel_tweaks.clear();
112 
113  // nodes
114  this->protected_nodes.clear();
115  this->node_tweaks.clear();
116 
117  // video cameras
118  this->force_video_cam_roles.clear();
119 
120  // flares
121  this->protected_flares.clear();
122 
123  // exhausts
124  this->protected_exhausts.clear();
125 
126  // managed materials
127  this->protected_managedmats.clear();
128 }
129 
130 bool TuneupDef::isWheelSideForced(WheelID_t wheelid, WheelSide& out_val) const
131 {
132  auto itor = force_wheel_sides.find(wheelid);
133  if (itor != force_wheel_sides.end())
134  {
135  out_val = itor->second;
136  return true;
137  }
138  else
139  {
140  return false;
141  }
142 }
143 
145 {
146  auto itor = force_video_cam_roles.find(camera_id);
147  if (itor != force_video_cam_roles.end())
148  {
149  out_val = itor->second;
150  return true;
151  }
152  else
153  {
154  return false;
155  }
156 }
157 
158  // Tweaking helpers
159 
160 float RoR::TuneupUtil::getTweakedWheelTireRadius(TuneupDefPtr& tuneup_def, WheelID_t wheel_id, float orig_val)
161 {
162  TuneupWheelTweak* tweak = nullptr;
163  if (TuneupUtil::isWheelTweaked(tuneup_def, wheel_id, /*out:*/ tweak))
164  {
165  ROR_ASSERT(tweak);
166  return (tweak->twt_tire_radius > 0) ? tweak->twt_tire_radius : orig_val;
167  }
168  else
169  {
170  return orig_val;
171  }
172 }
173 
174 float RoR::TuneupUtil::getTweakedWheelRimRadius(TuneupDefPtr& tuneup_def, WheelID_t wheel_id, float orig_val)
175 {
176  TuneupWheelTweak* tweak = nullptr;
177  if (TuneupUtil::isWheelTweaked(tuneup_def, wheel_id, tweak))
178  {
179  ROR_ASSERT(tweak);
180  return (tweak->twt_rim_radius > 0) ? tweak->twt_rim_radius : orig_val;
181  }
182  else
183  {
184  return orig_val;
185  }
186 }
187 
188 std::string RoR::TuneupUtil::getTweakedWheelMedia(TuneupDefPtr& tuneup_def, WheelID_t wheel_id, int media_idx, const std::string& orig_val)
189 {
190  TuneupWheelTweak* tweak = nullptr;
191  if (TuneupUtil::isWheelTweaked(tuneup_def, wheel_id, tweak))
192  {
193  ROR_ASSERT(tweak);
194  ROR_ASSERT(tweak->twt_media.size() > media_idx);
195  return (tweak->twt_media[media_idx] != "") ? tweak->twt_media[media_idx] : orig_val;
196  }
197  else
198  {
199  return orig_val;
200  }
201 }
202 
203 std::string RoR::TuneupUtil::getTweakedWheelMediaRG(TuneupDefPtr& tuneup_def, WheelID_t wheel_id, int media_idx, const std::string& orig_val)
204 {
205  TuneupWheelTweak* tweak = nullptr;
206  if (TuneupUtil::isWheelTweaked(tuneup_def, wheel_id, tweak))
207  {
208  ROR_ASSERT(tweak);
209  ROR_ASSERT(tweak->twt_media.size() > media_idx);
210  if (tweak->twt_media[media_idx] != "")
211  {
212  // Find the tweak addonpart
213  CacheEntryPtr addonpart_entry = App::GetCacheSystem()->FindEntryByFilename(LT_AddonPart, /*'partial':*/false, tweak->twt_origin);
214  if (addonpart_entry)
215  {
216  return addonpart_entry->resource_group;
217  }
218  else
219  {
220  LOG(fmt::format("[RoR|Tuneup] WARN Addonpart '{}' not found in modcache!", tweak->twt_origin));
221  return orig_val;
222  }
223  }
224  }
225 
226  return orig_val;
227 
228 }
229 
231 {
232 
233 
234  if (tuneup_def)
235  {
236  WheelSide forced_side = WheelSide::INVALID;
237  if (tuneup_def->isWheelSideForced(wheel_id, forced_side))
238  {
239  return forced_side;
240  }
241 
242  TuneupWheelTweak* tweak = nullptr;
243  if (TuneupUtil::isWheelTweaked(tuneup_def, wheel_id, tweak))
244  {
245  ROR_ASSERT(tweak);
246  if (tweak->twt_side != WheelSide::INVALID)
247  {
248  return tweak->twt_side;
249  }
250  }
251  }
252 
253  return orig_val;
254 }
255 
257 {
258  if (tuneup_def)
259  {
260  VideoCamRole forced_role = VCAM_ROLE_INVALID;
261  if (tuneup_def->isVideoCameraRoleForced(camera_id, forced_role))
262  {
263  return forced_role;
264  }
265 
266  // STUB: actual tweaking isn't implemented yet
267  }
268 
269  return orig_val;
270 }
271 
273 {
274 
275 
276  if (tuneup_def)
277  {
278  auto itor = tuneup_def->wheel_tweaks.find(wheel_id);
279  if (itor != tuneup_def->wheel_tweaks.end())
280  {
281  out_tweak = &itor->second;
282  return true;
283  }
284  }
285 
286  return false;
287 }
288 
289 Ogre::Vector3 RoR::TuneupUtil::getTweakedNodePosition(TuneupDefPtr& tuneup_def, NodeNum_t nodenum, Ogre::Vector3 orig_val)
290 {
291  TuneupNodeTweak* tweak = nullptr;
292  if (TuneupUtil::isNodeTweaked(tuneup_def, nodenum, tweak))
293  {
294  ROR_ASSERT(tweak);
295  return tweak->tnt_pos;
296  }
297  else
298  {
299  return orig_val;
300  }
301 }
302 
304 {
305  if (tuneup_def)
306  {
307  auto itor = tuneup_def->node_tweaks.find(nodenum);
308  if (itor != tuneup_def->node_tweaks.end())
309  {
310  out_tweak = &itor->second;
311  return true;
312  }
313  }
314 
315  return false;
316 }
317 
318 
319 // > prop
321 {
322  return tuneup_def
323  && (tuneup_def->isPropUnwanted(prop_id) || tuneup_def->isPropForceRemoved(prop_id));
324 }
325 
326 Ogre::Vector3 RoR::TuneupUtil::getTweakedPropOffset(TuneupDefPtr& tuneup_def, PropID_t prop_id, Ogre::Vector3 orig_val)
327 {
328  TuneupPropTweak* tweak = nullptr;
329  if (TuneupUtil::isPropTweaked(tuneup_def, prop_id, tweak))
330  {
331  ROR_ASSERT(tweak);
332  return tweak->tpt_offset;
333  }
334  else
335  {
336  return orig_val;
337  }
338 }
339 
340 Ogre::Vector3 RoR::TuneupUtil::getTweakedPropRotation(TuneupDefPtr& tuneup_def, PropID_t prop_id, Ogre::Vector3 orig_val)
341 {
342  TuneupPropTweak* tweak = nullptr;
343  if (TuneupUtil::isPropTweaked(tuneup_def, prop_id, tweak))
344  {
345  ROR_ASSERT(tweak);
346  return tweak->tpt_rotation;
347  }
348  else
349  {
350  return orig_val;
351  }
352 }
353 
354 std::string RoR::TuneupUtil::getTweakedPropMedia(TuneupDefPtr& tuneup_def, PropID_t prop_id, int media_idx, const std::string& orig_val)
355 {
356  TuneupPropTweak* tweak = nullptr;
357  if (TuneupUtil::isPropTweaked(tuneup_def, prop_id, tweak))
358  {
359  ROR_ASSERT(tweak);
360  ROR_ASSERT(tweak->tpt_media.size() > media_idx);
361  return (tweak->tpt_media[media_idx] != "") ? tweak->tpt_media[media_idx] : orig_val;
362  }
363  else
364  {
365  return orig_val;
366  }
367 }
368 
369 std::string RoR::TuneupUtil::getTweakedPropMediaRG(TuneupDefPtr& tuneup_def, PropID_t prop_id, int media_idx, const std::string& orig_val)
370 {
371  TuneupPropTweak* tweak = nullptr;
372  if (TuneupUtil::isPropTweaked(tuneup_def, prop_id, tweak))
373  {
374  ROR_ASSERT(tweak);
375  ROR_ASSERT(tweak->tpt_media.size() > media_idx);
376  if (tweak->tpt_media[media_idx] != "")
377  {
378  // Find the tweak addonpart
379  CacheEntryPtr addonpart_entry = App::GetCacheSystem()->FindEntryByFilename(LT_AddonPart, /*partial:*/false, tweak->tpt_origin);
380  if (addonpart_entry)
381  {
382  return addonpart_entry->resource_group;
383  }
384  else
385  {
386  LOG(fmt::format("[RoR|Tuneup] WARN Addonpart '{}' not found in modcache!", tweak->tpt_origin));
387  }
388  }
389  }
390 
391  return orig_val;
392 }
393 
395 {
396 
397 
398  if (tuneup_def)
399  {
400  auto itor = tuneup_def->prop_tweaks.find(prop_id);
401  if (itor != tuneup_def->prop_tweaks.end())
402  {
403  out_tweak = &itor->second;
404  return true;
405  }
406  }
407 
408  return false;
409 }
410 
411 // > flexbody
413 {
414  return tuneup_def &&
415  (tuneup_def->isFlexbodyUnwanted(flexbody_id) || tuneup_def->isFlexbodyForceRemoved(flexbody_id));
416 }
417 
418 Ogre::Vector3 RoR::TuneupUtil::getTweakedFlexbodyOffset(TuneupDefPtr& tuneup_def, FlexbodyID_t flexbody_id, Ogre::Vector3 orig_val)
419 {
420  TuneupFlexbodyTweak* tweak = nullptr;
421  if (TuneupUtil::isFlexbodyTweaked(tuneup_def, flexbody_id, tweak))
422  {
423  ROR_ASSERT(tweak);
424  return tweak->tft_offset;
425  }
426  else
427  {
428  return orig_val;
429  }
430 }
431 
432 Ogre::Vector3 RoR::TuneupUtil::getTweakedFlexbodyRotation(TuneupDefPtr& tuneup_def, FlexbodyID_t flexbody_id, Ogre::Vector3 orig_val)
433 {
434  TuneupFlexbodyTweak* tweak = nullptr;
435  if (TuneupUtil::isFlexbodyTweaked(tuneup_def, flexbody_id, tweak))
436  {
437  ROR_ASSERT(tweak);
438  return tweak->tft_rotation;
439  }
440  else
441  {
442  return orig_val;
443  }
444 }
445 
446 std::string RoR::TuneupUtil::getTweakedFlexbodyMedia(TuneupDefPtr& tuneup_def, FlexbodyID_t flexbody_id, int media_idx, const std::string& orig_val)
447 {
448  TuneupFlexbodyTweak* tweak = nullptr;
449  if (TuneupUtil::isFlexbodyTweaked(tuneup_def, flexbody_id, tweak))
450  {
451  ROR_ASSERT(tweak);
452  return (tweak->tft_media != "") ? tweak->tft_media : orig_val;
453  }
454  else
455  {
456  return orig_val;
457  }
458 }
459 
460 std::string RoR::TuneupUtil::getTweakedFlexbodyMediaRG(TuneupDefPtr& tuneup_def, FlexbodyID_t flexbody_id, int media_idx, const std::string& orig_val)
461 {
462  TuneupFlexbodyTweak* tweak = nullptr;
463  if (TuneupUtil::isFlexbodyTweaked(tuneup_def, flexbody_id, tweak))
464  {
465  ROR_ASSERT(tweak);
466  if (tweak->tft_media != "")
467  {
468  // Find the tweak addonpart
469  CacheEntryPtr addonpart_entry = App::GetCacheSystem()->FindEntryByFilename(LT_AddonPart, /*partial:*/false, tweak->tft_origin);
470  if (addonpart_entry)
471  {
472  return addonpart_entry->resource_group;
473  }
474  else
475  {
476  LOG(fmt::format("[RoR|Tuneup] WARN Addonpart '{}' not found in modcache!", tweak->tft_origin));
477  return orig_val;
478  }
479  }
480  }
481 
482  return orig_val;
483 }
484 
486 {
487 
488 
489  if (tuneup_def)
490  {
491  auto itor = tuneup_def->flexbody_tweaks.find(flexbody_id);
492  if (itor != tuneup_def->flexbody_tweaks.end())
493  {
494  out_tweak = &itor->second;
495  return true;
496  }
497  }
498 
499  return false;
500 }
501 
502 // > flare
504 {
505  return tuneup_def
506  && (tuneup_def->isFlareUnwanted(flare_id) || tuneup_def->isFlareForceRemoved(flare_id));
507 }
508 
509 // > exhaust
511 {
512  return tuneup_def
513  && (tuneup_def->isExhaustUnwanted(exhaust_id) || tuneup_def->isExhaustForceRemoved(exhaust_id));
514 }
515 
516 // > managedmaterials
517 bool RoR::TuneupUtil::isManagedMatAnyhowRemoved(TuneupDefPtr& tuneup_def, const std::string& material_name)
518 {
519  return tuneup_def
520  && (tuneup_def->isManagedMatUnwanted(material_name) || tuneup_def->isManagedMatForceRemoved(material_name));
521 }
522 
523 bool RoR::TuneupUtil::isManagedMatTweaked(TuneupDefPtr& tuneup_def, const std::string& material_name, TuneupManagedMatTweak*& out_tweak)
524 {
525  if (tuneup_def)
526  {
527  auto itor = tuneup_def->managedmat_tweaks.find(material_name);
528  if (itor != tuneup_def->managedmat_tweaks.end())
529  {
530  out_tweak = &itor->second;
531  return true;
532  }
533  }
534 
535  return false;
536 }
537 
538 std::string RoR::TuneupUtil::getTweakedManagedMatType(TuneupDefPtr& tuneup_def, const std::string& material_name, const std::string& orig_val)
539 {
540  TuneupManagedMatTweak* tweak = nullptr;
541  if (TuneupUtil::isManagedMatTweaked(tuneup_def, material_name, tweak))
542  {
543  ROR_ASSERT(tweak);
544  return (tweak->tmt_type != "") ? tweak->tmt_type : orig_val;
545  }
546  else
547  {
548  return orig_val;
549  }
550 }
551 
552 std::string RoR::TuneupUtil::getTweakedManagedMatMedia(TuneupDefPtr& tuneup_def, const std::string& material_name, int media_idx, const std::string& orig_val)
553 {
554  TuneupManagedMatTweak* tweak = nullptr;
555  if (TuneupUtil::isManagedMatTweaked(tuneup_def, material_name, tweak))
556  {
557  ROR_ASSERT(tweak);
558  ROR_ASSERT(tweak->tmt_media.size() > media_idx);
559  return (tweak->tmt_media[media_idx] != "") ? tweak->tmt_media[media_idx] : orig_val;
560  }
561  else
562  {
563  return orig_val;
564  }
565 }
566 
567 std::string RoR::TuneupUtil::getTweakedManagedMatMediaRG(TuneupDefPtr& tuneup_def, const std::string& material_name, int media_idx, const std::string& orig_val)
568 {
569  TuneupManagedMatTweak* tweak = nullptr;
570  if (TuneupUtil::isManagedMatTweaked(tuneup_def, material_name, tweak))
571  {
572  ROR_ASSERT(tweak);
573  ROR_ASSERT(tweak->tmt_media.size() > media_idx);
574  if (tweak->tmt_media[media_idx] != "")
575  {
576  // Find the tweak addonpart
577  CacheEntryPtr addonpart_entry = App::GetCacheSystem()->FindEntryByFilename(LT_AddonPart, /*partial:*/false, tweak->tmt_origin);
578  if (addonpart_entry)
579  {
580  return addonpart_entry->resource_group;
581  }
582  else
583  {
584  LOG(fmt::format("[RoR|Tuneup] WARN managedmaterial '{}': Addonpart '{}' not found in modcache!", material_name, tweak->tmt_origin));
585  return orig_val;
586  }
587  }
588  }
589 
590  return orig_val;
591 }
592 
593 bool RoR::TuneupUtil::isAddonPartUsed(TuneupDefPtr& tuneup_entry, const std::string& filename)
594 {
595  return tuneup_entry
596  && tuneup_entry->use_addonparts.find(filename) != tuneup_entry->use_addonparts.end();
597 }
598 
599 std::vector<TuneupDefPtr> RoR::TuneupUtil::ParseTuneups(Ogre::DataStreamPtr& stream)
600 {
601  std::vector<TuneupDefPtr> result;
602  TuneupDefPtr curr_tuneup;
603  try
604  {
605  while(!stream->eof())
606  {
607  std::string line = SanitizeUtf8String(stream->getLine());
608 
609  // Ignore blanks & comments
610  if (!line.length() || line.substr(0, 2) == "//")
611  {
612  continue;
613  }
614 
615  if (!curr_tuneup)
616  {
617  // No current tuneup -- So first valid data should be tuneup name
618  Ogre::StringUtil::trim(line);
619  curr_tuneup = new TuneupDef();
620  curr_tuneup->name = line;
621  stream->skipLine("{");
622  }
623  else
624  {
625  // Already in tuneup
626  if (line == "}")
627  {
628  result.push_back(curr_tuneup); // Finished
629  curr_tuneup = nullptr;
630  }
631  else
632  {
633  RoR::TuneupUtil::ParseTuneupAttribute(line, curr_tuneup);
634  }
635  }
636  }
637 
638  if (curr_tuneup)
639  {
642  fmt::format("Tuneup '{}' in file '{}' not properly closed with '}}'",
643  curr_tuneup->name, stream->getName()));
644  result.push_back(curr_tuneup); // Submit anyway
645  curr_tuneup = nullptr;
646  }
647  }
648  catch (Ogre::Exception& e)
649  {
652  fmt::format("Error parsing tuneup file '{}', message: {}",
653  stream->getName(), e.getFullDescription()));
654  }
655  return result;
656 }
657 
658 void RoR::TuneupUtil::ParseTuneupAttribute(const std::string& line, TuneupDefPtr& tuneup_def) // static
659 {
660  Ogre::StringVector params = Ogre::StringUtil::split(line, "\t=,;\n");
661  for (unsigned int i=0; i < params.size(); i++)
662  {
663  Ogre::StringUtil::trim(params[i]);
664  }
665  Ogre::String& attrib = params[0];
666  Ogre::StringUtil::toLowerCase(attrib);
667 
668  // General info
669  if (attrib == "preview" && params.size() >= 2) { tuneup_def->thumbnail = params[1]; return; }
670  if (attrib == "description" && params.size() >= 2) { tuneup_def->description = params[1]; return; }
671  if (attrib == "author_name" && params.size() >= 2) { tuneup_def->author_name = params[1]; return; }
672  if (attrib == "author_id" && params.size() == 2) { tuneup_def->author_id = PARSEINT(params[1]); return; }
673  if (attrib == "category_id" && params.size() == 2) { tuneup_def->category_id = (CacheCategoryId)PARSEINT(params[1]); return; }
674  if (attrib == "guid" && params.size() >= 2) { tuneup_def->guid = params[1]; Ogre::StringUtil::trim(tuneup_def->guid); Ogre::StringUtil::toLowerCase(tuneup_def->guid); return; }
675  if (attrib == "name" && params.size() >= 2) { tuneup_def->name = params[1]; Ogre::StringUtil::trim(tuneup_def->name); return; }
676  if (attrib == "filename" && params.size() >= 2) { tuneup_def->filename = params[1]; Ogre::StringUtil::trim(tuneup_def->filename); return; }
677 
678  // Addonparts and extracted data
679  if (attrib == "use_addonpart" && params.size() == 2) { tuneup_def->use_addonparts.insert(params[1]); return; }
680  if (attrib == "unwanted_prop" && params.size() == 2) { tuneup_def->unwanted_props.insert(PARSEINT(params[1])); return; }
681  if (attrib == "unwanted_flexbody" && params.size() == 2) { tuneup_def->unwanted_flexbodies.insert(PARSEINT(params[1])); return; }
682  if (attrib == "protected_prop" && params.size() == 2) { tuneup_def->protected_props.insert(PARSEINT(params[1])); return; }
683  if (attrib == "protected_flexbody" && params.size() == 2) { tuneup_def->protected_flexbodies.insert(PARSEINT(params[1])); return; }
684 
685  // UI overrides
686  if (attrib == "forced_wheel_side" && params.size() == 3) { tuneup_def->force_wheel_sides[PARSEINT(params[1])] = (WheelSide)PARSEINT(params[2]); return; }
687  if (attrib == "force_remove_prop" && params.size() == 2) { tuneup_def->force_remove_props.insert(PARSEINT(params[1])); return; }
688  if (attrib == "force_remove_flexbody" && params.size() == 2) { tuneup_def->force_remove_flexbodies.insert(PARSEINT(params[1])); return; }
689 }
690 
691 void RoR::TuneupUtil::ExportTuneup(Ogre::DataStreamPtr& stream, TuneupDefPtr& tuneup)
692 {
694  buf << tuneup->name << "\n";
695  buf << "{\n";
696 
697  // General info:
698  buf << "\tpreview = " << tuneup->thumbnail << "\n";
699  buf << "\tdescription = " << tuneup->description << "\n";
700  buf << "\tauthor_name = " << tuneup->author_name << "\n";
701  buf << "\tauthor_id = " << tuneup->author_id << "\n";
702  buf << "\tcategory_id = " << (int)tuneup->category_id << "\n";
703  buf << "\tguid = " << tuneup->guid << "\n";
704  buf << "\tfilename = " << tuneup->filename << "\n";
705  buf << "\n";
706 
707  // Addonparts and extracted data:
708  for (const std::string& addonpart: tuneup->use_addonparts)
709  {
710  buf << "\tuse_addonpart = " << addonpart << "\n";
711  }
712  for (PropID_t unwanted_prop: tuneup->unwanted_props)
713  {
714  buf << "\tunwanted_prop = " << (int)unwanted_prop << "\n";
715  }
716  for (FlexbodyID_t unwanted_flexbody: tuneup->unwanted_flexbodies)
717  {
718  buf << "\tunwanted_flexbody = " << (int)unwanted_flexbody << "\n";
719  }
720  for (PropID_t protected_prop: tuneup->protected_props)
721  {
722  buf << "\tprotected_prop = " << (int)protected_prop << "\n";
723  }
724  for (FlexbodyID_t protected_flexbody: tuneup->protected_flexbodies)
725  {
726  buf << "\tprotected_flexbody = " << (int)protected_flexbody << "\n";
727  }
728 
729  // UI overrides:
730  for (PropID_t prop: tuneup->force_remove_props)
731  {
732  buf << "\tforce_remove_prop = " << (int)prop << "\n";
733  }
734  for (FlexbodyID_t flexbody: tuneup->force_remove_flexbodies)
735  {
736  buf << "\tforce_remove_flexbody = " << (int)flexbody << "\n";
737  }
738  for (auto& pair: tuneup->force_wheel_sides)
739  {
740  buf << "\tforced_wheel_side = " << pair.first << ", " << (int)pair.second << "\n";
741  }
742  buf << "}\n\n";
743 
744  size_t written = stream->write(buf.GetBuffer(), buf.GetLength());
745  if (written < buf.GetLength())
746  {
748  fmt::format("Error writing file '{}': only written {}/{} bytes", stream->getName(), written, buf.GetLength()));
749  }
750 }
751 
RoR::TuneupDef
Dual purpose:
Definition: TuneupFileFormat.h:93
ROR_ASSERT
#define ROR_ASSERT(_EXPR)
Definition: Application.h:40
RoR::TuneupDef::isFlexbodyForceRemoved
bool isFlexbodyForceRemoved(FlexbodyID_t flexbodyid)
Definition: TuneupFileFormat.h:171
RoR::TuneupDef::clone
TuneupDefPtr clone()
Definition: TuneupFileFormat.cpp:38
RoR::TuneupDef::force_remove_flexbodies
std::set< FlexbodyID_t > force_remove_flexbodies
UI overrides.
Definition: TuneupFileFormat.h:126
RoR::TuneupWheelTweak::twt_origin
std::string twt_origin
Addonpart filename.
Definition: TuneupFileFormat.h:60
RoR::TuneupUtil::isAddonPartUsed
static bool isAddonPartUsed(TuneupDefPtr &tuneup_entry, const std::string &filename)
Definition: TuneupFileFormat.cpp:593
RoR::TuneupDef::protected_flares
std::set< FlareID_t > protected_flares
Flares which cannot be altered via 'addonpart_unwanted_flare' directive.
Definition: TuneupFileFormat.h:140
RoR::TuneupUtil::ParseTuneups
static std::vector< TuneupDefPtr > ParseTuneups(Ogre::DataStreamPtr &stream)
Definition: TuneupFileFormat.cpp:599
RoR::TuneupUtil::isWheelTweaked
static bool isWheelTweaked(TuneupDefPtr &tuneup_entry, WheelID_t wheel_id, TuneupWheelTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:272
RoR::TuneupWheelTweak::twt_tire_radius
float twt_tire_radius
Arg#5, optional.
Definition: TuneupFileFormat.h:58
RoR::TuneupDef::force_remove_props
std::set< PropID_t > force_remove_props
UI overrides.
Definition: TuneupFileFormat.h:125
RoR::Str::GetBuffer
char * GetBuffer()
Definition: Str.h:48
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::VCAM_ROLE_INVALID
@ VCAM_ROLE_INVALID
Definition: Application.h:389
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::TuneupUtil::isPropTweaked
static bool isPropTweaked(TuneupDefPtr &tuneup_entry, PropID_t flexbody_id, TuneupPropTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:394
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
RoR::TuneupFlexbodyTweak::tft_media
std::string tft_media
Definition: TuneupFileFormat.h:76
RoR::TuneupUtil::isPropAnyhowRemoved
static bool isPropAnyhowRemoved(TuneupDefPtr &tuneup_def, PropID_t prop_id)
Definition: TuneupFileFormat.cpp:320
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::SanitizeUtf8String
std::string SanitizeUtf8String(std::string const &str_in)
Definition: Utils.cpp:120
RoR::TuneupUtil::ExportTuneup
static void ExportTuneup(Ogre::DataStreamPtr &stream, TuneupDefPtr &tuneup)
Definition: TuneupFileFormat.cpp:691
Console.h
RoR::TuneupDef::name
std::string name
Definition: TuneupFileFormat.h:97
RoR::Console::putMessage
void putMessage(MessageArea area, MessageType type, std::string const &msg, std::string icon="")
Definition: Console.cpp:103
RoR::TuneupDef::protected_nodes
std::set< NodeNum_t > protected_nodes
Nodes that cannot be altered via 'addonpart_tweak_node'.
Definition: TuneupFileFormat.h:136
RoR::CacheEntry::resource_group
Ogre::String resource_group
Resource group of the loaded bundle. Empty if not loaded yet.
Definition: CacheSystem.h:89
TuneupFileFormat.h
The vehicle tuning system; applies addonparts and user overrides to vehicles.
RoR::TuneupDef::use_addonparts
std::set< std::string > use_addonparts
Addonpart filenames.
Definition: TuneupFileFormat.h:109
RoR::TuneupDef::isExhaustUnwanted
bool isExhaustUnwanted(ExhaustID_t exhaustid)
Definition: TuneupFileFormat.h:164
RoR::TuneupDef::isPropUnwanted
bool isPropUnwanted(PropID_t propid)
Definition: TuneupFileFormat.h:161
RoR::TuneupUtil::getTweakedPropOffset
static Ogre::Vector3 getTweakedPropOffset(TuneupDefPtr &tuneup_entry, PropID_t prop_id, Ogre::Vector3 orig_val)
Definition: TuneupFileFormat.cpp:326
RoR::TuneupUtil::getTweakedPropRotation
static Ogre::Vector3 getTweakedPropRotation(TuneupDefPtr &tuneup_entry, PropID_t prop_id, Ogre::Vector3 orig_val)
Definition: TuneupFileFormat.cpp:340
Utils.h
RoR::Str::GetLength
size_t GetLength() const
Definition: Str.h:51
RoR::TuneupPropTweak::tpt_rotation
Ogre::Vector3 tpt_rotation
Definition: TuneupFileFormat.h:69
RefCountingObjectPtr< TuneupDef >
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::TuneupDef::protected_props
std::set< PropID_t > protected_props
Props which cannot be altered via 'addonpart_tweak_prop' or 'addonpart_unwanted_prop' directive.
Definition: TuneupFileFormat.h:138
RoR::TuneupUtil::isNodeTweaked
static bool isNodeTweaked(TuneupDefPtr &tuneup_entry, NodeNum_t nodenum, TuneupNodeTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:303
RoR::TuneupDef::isVideoCameraRoleForced
bool isVideoCameraRoleForced(VideoCameraID_t camera_id, VideoCamRole &out_val) const
Definition: TuneupFileFormat.cpp:144
RoR::TuneupUtil::getTweakedFlexbodyMediaRG
static std::string getTweakedFlexbodyMediaRG(TuneupDefPtr &tuneup_def, FlexbodyID_t flexbody_id, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:460
PARSEINT
#define PARSEINT(x)
Definition: Application.h:57
RoR::TuneupDef::managedmat_tweaks
std::map< std::string, TuneupManagedMatTweak > managedmat_tweaks
Managed material overrides via 'addonpart_tweak_managedmaterial'.
Definition: TuneupFileFormat.h:115
RoR::TuneupDef::node_tweaks
std::map< NodeNum_t, TuneupNodeTweak > node_tweaks
Node position overrides via 'addonpart_tweak_node'.
Definition: TuneupFileFormat.h:111
RoR::TuneupManagedMatTweak
< Data of 'addonpart_tweak_managedmaterial <name> <type> <media1> <media2> [<media3>]'
Definition: TuneupFileFormat.h:82
RoR::VideoCameraID_t
int VideoCameraID_t
Index into GfxActor::m_videocameras, use RoR::VIDEOCAMERAID_INVALID as empty value.
Definition: ForwardDeclarations.h:85
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::TuneupUtil::isManagedMatAnyhowRemoved
static bool isManagedMatAnyhowRemoved(TuneupDefPtr &tuneup_def, const std::string &matname)
Definition: TuneupFileFormat.cpp:517
RoR::Str
Wrapper for classic c-string (local buffer) Refresher: strlen() excludes '\0' terminator; strncat() A...
Definition: Str.h:35
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::TuneupUtil::isFlexbodyAnyhowRemoved
static bool isFlexbodyAnyhowRemoved(TuneupDefPtr &tuneup_def, FlexbodyID_t flexbody_id)
Definition: TuneupFileFormat.cpp:412
RoR::TuneupDef::isManagedMatForceRemoved
bool isManagedMatForceRemoved(const std::string &matname)
Definition: TuneupFileFormat.h:175
RoR::TuneupNodeTweak
< Data of 'addonpart_tweak_node <nodenum> <posX> <posY> <posZ>'
Definition: TuneupFileFormat.h:44
CacheSystem.h
A database of user-installed content alias 'mods' (vehicles, terrains...)
RoR::TuneupDef::protected_wheels
std::set< WheelID_t > protected_wheels
Wheels that cannot be altered via 'addonpart_tweak_wheel'.
Definition: TuneupFileFormat.h:137
RoR::TuneupWheelTweak::twt_side
WheelSide twt_side
Arg#4, optional, default LEFT (Only applicable to mesh/flexbody wheels)
Definition: TuneupFileFormat.h:57
RoR::TuneupUtil::getTweakedPropMedia
static std::string getTweakedPropMedia(TuneupDefPtr &tuneup_entry, PropID_t prop_id, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:354
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::WheelID_t
int WheelID_t
Index to Actor::ar_wheels, use RoR::WHEELID_INVALID as empty value.
Definition: ForwardDeclarations.h:58
RoR::TuneupNodeTweak::tnt_pos
Ogre::Vector3 tnt_pos
Args#234, required.
Definition: TuneupFileFormat.h:47
RoR::TuneupDef::reset
void reset()
Definition: TuneupFileFormat.cpp:91
RoR::TuneupDef::thumbnail
std::string thumbnail
Definition: TuneupFileFormat.h:100
RoR::TuneupDef::protected_managedmats
std::set< std::string > protected_managedmats
Managed materials which cannot be altered via 'addonpart_tweak_managedmaterial' directive.
Definition: TuneupFileFormat.h:142
RoR::TuneupDef::isFlareUnwanted
bool isFlareUnwanted(FlareID_t flareid)
Definition: TuneupFileFormat.h:163
RoR::CacheCategoryId
CacheCategoryId
Definition: Application.h:328
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
RoR::TuneupDef::description
std::string description
Definition: TuneupFileFormat.h:101
RoR::TuneupUtil::getTweakedFlexbodyMedia
static std::string getTweakedFlexbodyMedia(TuneupDefPtr &tuneup_entry, FlexbodyID_t flexbody_id, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:446
RoR::TuneupUtil::getTweakedManagedMatMediaRG
static std::string getTweakedManagedMatMediaRG(TuneupDefPtr &tuneup_def, const std::string &matname, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:567
RoR::TuneupUtil::getTweakedWheelRimRadius
static float getTweakedWheelRimRadius(TuneupDefPtr &tuneup_entry, WheelID_t wheel_id, float orig_val)
Definition: TuneupFileFormat.cpp:174
RoR::TuneupDef::isFlareForceRemoved
bool isFlareForceRemoved(FlareID_t flareid)
Definition: TuneupFileFormat.h:173
RoR::TuneupUtil::isFlareAnyhowRemoved
static bool isFlareAnyhowRemoved(TuneupDefPtr &tuneup_def, FlareID_t flare_id)
Definition: TuneupFileFormat.cpp:503
RoR::TuneupUtil::getTweakedFlexbodyRotation
static Ogre::Vector3 getTweakedFlexbodyRotation(TuneupDefPtr &tuneup_entry, FlexbodyID_t flexbody_id, Ogre::Vector3 orig_val)
Definition: TuneupFileFormat.cpp:432
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::TuneupDef::author_id
int author_id
Definition: TuneupFileFormat.h:103
RoR::App::GetCacheSystem
CacheSystem * GetCacheSystem()
Definition: Application.cpp:276
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
RoR::TuneupDef::author_name
std::string author_name
Definition: TuneupFileFormat.h:102
RoR::TuneupDef::force_video_cam_roles
std::map< VideoCameraID_t, VideoCamRole > force_video_cam_roles
UI overrides.
Definition: TuneupFileFormat.h:128
RoR::TuneupDef::protected_exhausts
std::set< ExhaustID_t > protected_exhausts
Exhausts which cannot be altered via 'addonpart_unwanted_exhaust' directive.
Definition: TuneupFileFormat.h:141
RoR::TuneupUtil::getTweakedFlexbodyOffset
static Ogre::Vector3 getTweakedFlexbodyOffset(TuneupDefPtr &tuneup_entry, FlexbodyID_t flexbody_id, Ogre::Vector3 orig_val)
Definition: TuneupFileFormat.cpp:418
RoR::TuneupDef::protected_flexbodies
std::set< FlexbodyID_t > protected_flexbodies
Flexbodies which cannot be altered via 'addonpart_tweak_flexbody' or 'addonpart_unwanted_flexbody' di...
Definition: TuneupFileFormat.h:139
RoR::TuneupPropTweak::tpt_offset
Ogre::Vector3 tpt_offset
Definition: TuneupFileFormat.h:68
RoR::TuneupUtil::getTweakedPropMediaRG
static std::string getTweakedPropMediaRG(TuneupDefPtr &tuneup_def, PropID_t prop_id, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:369
RoR::VideoCamRole
VideoCamRole
Definition: Application.h:373
RoR::WheelSide::INVALID
@ INVALID
RoR::TuneupUtil::getTweakedManagedMatType
static std::string getTweakedManagedMatType(TuneupDefPtr &tuneup_def, const std::string &matname, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:538
RoR::TuneupDef::filename
std::string filename
target vehicle filename
Definition: TuneupFileFormat.h:99
RoR::TuneupUtil::isExhaustAnyhowRemoved
static bool isExhaustAnyhowRemoved(TuneupDefPtr &tuneup_def, ExhaustID_t exhaust_id)
Definition: TuneupFileFormat.cpp:510
RoR::TuneupUtil::isManagedMatTweaked
static bool isManagedMatTweaked(TuneupDefPtr &tuneup_def, const std::string &matname, TuneupManagedMatTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:523
RoR::TuneupUtil::getTweakedWheelMediaRG
static std::string getTweakedWheelMediaRG(TuneupDefPtr &tuneup_def, WheelID_t wheel_id, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:203
RoR::FlareID_t
int FlareID_t
Index into Actor::ar_flares, use RoR::FLAREID_INVALID as empty value.
Definition: ForwardDeclarations.h:70
RoR::TuneupUtil::getTweakedWheelSide
static WheelSide getTweakedWheelSide(TuneupDefPtr &tuneup_entry, WheelID_t wheel_id, WheelSide orig_val)
Definition: TuneupFileFormat.cpp:230
RoR::TuneupUtil::getTweakedWheelTireRadius
static float getTweakedWheelTireRadius(TuneupDefPtr &tuneup_entry, WheelID_t wheel_id, float orig_val)
Definition: TuneupFileFormat.cpp:160
RoR::TuneupDef::isExhaustForceRemoved
bool isExhaustForceRemoved(ExhaustID_t exhaustid)
Definition: TuneupFileFormat.h:174
RoR::TuneupUtil::getTweakedManagedMatMedia
static std::string getTweakedManagedMatMedia(TuneupDefPtr &tuneup_def, const std::string &matname, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:552
RoR::TuneupDef::category_id
CacheCategoryId category_id
Definition: TuneupFileFormat.h:104
RoR::TuneupWheelTweak
< Data of 'addonpart_tweak_wheel <wheel ID> <media1> <media2> <side flag> <tire radius> <rim radius>'
Definition: TuneupFileFormat.h:51
RoR::TuneupUtil::getTweakedNodePosition
static Ogre::Vector3 getTweakedNodePosition(TuneupDefPtr &tuneup_entry, NodeNum_t nodenum, Ogre::Vector3 orig_val)
Definition: TuneupFileFormat.cpp:289
RoR::Console::CONSOLE_MSGTYPE_ACTOR
@ CONSOLE_MSGTYPE_ACTOR
Parsing/spawn/simulation messages for actors.
Definition: Console.h:63
RoR::Console::CONSOLE_SYSTEM_WARNING
@ CONSOLE_SYSTEM_WARNING
Definition: Console.h:53
RoR::TuneupDef::force_wheel_sides
std::map< WheelID_t, WheelSide > force_wheel_sides
UI overrides.
Definition: TuneupFileFormat.h:127
RoR::TuneupDef::isFlexbodyUnwanted
bool isFlexbodyUnwanted(FlexbodyID_t flexbodyid)
Definition: TuneupFileFormat.h:162
RoR::Console::CONSOLE_MSGTYPE_INFO
@ CONSOLE_MSGTYPE_INFO
Generic message.
Definition: Console.h:60
RoR::WheelSide
WheelSide
Used by rig-def/addonpart/tuneup formats to specify wheel rim mesh orientation.
Definition: GfxData.h:115
RoR::FlexbodyID_t
int FlexbodyID_t
Index to GfxActor::m_flexbodies, use RoR::FLEXBODYID_INVALID as empty value.
Definition: ForwardDeclarations.h:64
RoR::TuneupDef::isWheelSideForced
bool isWheelSideForced(WheelID_t wheelid, WheelSide &out_val) const
Definition: TuneupFileFormat.cpp:130
RoR::TuneupUtil::isFlexbodyTweaked
static bool isFlexbodyTweaked(TuneupDefPtr &tuneup_entry, FlexbodyID_t flexbody_id, TuneupFlexbodyTweak *&out_tweak)
Definition: TuneupFileFormat.cpp:485
RoR::TuneupDef::guid
std::string guid
target vehicle GUID
Definition: TuneupFileFormat.h:98
RoR::TuneupUtil::ParseTuneupAttribute
static void ParseTuneupAttribute(const std::string &line, TuneupDefPtr &tuneup_def)
Definition: TuneupFileFormat.cpp:658
RoR::TuneupFlexbodyTweak::tft_offset
Ogre::Vector3 tft_offset
Definition: TuneupFileFormat.h:77
RoR::TuneupFlexbodyTweak
< Data of 'addonpart_tweak_flexbody <flexbody ID> <offsetX> <offsetY> <offsetZ> <rotX> <rotY> <rotZ> ...
Definition: TuneupFileFormat.h:73
RoR::TuneupUtil::getTweakedWheelMedia
static std::string getTweakedWheelMedia(TuneupDefPtr &tuneup_entry, WheelID_t wheel_id, int media_idx, const std::string &orig_val)
Definition: TuneupFileFormat.cpp:188
RoR::TuneupDef::isPropForceRemoved
bool isPropForceRemoved(PropID_t propid)
Definition: TuneupFileFormat.h:170
RoR::TuneupUtil::getTweakedVideoCameraRole
static VideoCamRole getTweakedVideoCameraRole(TuneupDefPtr &tuneup_def, VideoCameraID_t camera_id, VideoCamRole orig_val)
Definition: TuneupFileFormat.cpp:256
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
RoR::TuneupDef::isManagedMatUnwanted
bool isManagedMatUnwanted(const std::string &matname)
Definition: TuneupFileFormat.h:165
RoR::PropID_t
int PropID_t
Index to GfxActor::m_props, use RoR::PROPID_INVALID as empty value.
Definition: ForwardDeclarations.h:61