OpenRCT2/src/openrct2/object/RideObject.cpp

1043 lines
42 KiB
C++
Raw Normal View History

#pragma region Copyright (c) 2014-2017 OpenRCT2 Developers
2016-06-25 23:07:01 +02:00
/*****************************************************************************
* OpenRCT2, an open source clone of Roller Coaster Tycoon 2.
*
* OpenRCT2 is the work of many authors, a full list can be found in contributors.md
* For more information, visit https://github.com/OpenRCT2/OpenRCT2
*
* OpenRCT2 is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* A full copy of the GNU General Public License can be found in licence.txt
*****************************************************************************/
#pragma endregion
2017-12-20 17:31:06 +01:00
#pragma warning(disable : 4706) // assignment within conditional expression
2017-12-07 13:53:54 +01:00
#include <algorithm>
#include <unordered_map>
2016-06-25 23:07:01 +02:00
#include "../core/IStream.hpp"
#include "../core/Math.hpp"
2016-06-25 23:07:01 +02:00
#include "../core/Memory.hpp"
#include "../core/String.hpp"
2017-12-20 22:47:57 +01:00
#include "../core/Util.hpp"
2017-12-07 13:53:54 +01:00
#include "../ride/RideGroupManager.h"
2018-01-05 22:17:33 +01:00
#include "../drawing/Drawing.h"
2018-01-06 18:32:25 +01:00
#include "../localisation/Language.h"
2018-01-09 10:40:42 +01:00
#include "../rct2/RCT2.h"
2017-12-31 13:21:34 +01:00
#include "../ride/Ride.h"
2017-10-17 13:51:47 +02:00
#include "../ride/Track.h"
2017-12-07 13:53:54 +01:00
#include "../OpenRCT2.h"
#include "ObjectJsonHelpers.h"
2017-12-07 13:53:54 +01:00
#include "ObjectRepository.h"
#include "RideObject.h"
using namespace OpenRCT2;
2016-06-25 23:07:01 +02:00
2016-07-01 22:01:06 +02:00
void RideObject::ReadLegacy(IReadObjectContext * context, IStream * stream)
2016-06-25 23:07:01 +02:00
{
stream->Seek(8, STREAM_SEEK_CURRENT);
_legacyType.flags = stream->ReadValue<uint32>();
for (auto &rideType : _legacyType.ride_type)
{
rideType = stream->ReadValue<uint8>();
}
_legacyType.min_cars_in_train = stream->ReadValue<uint8>();
_legacyType.max_cars_in_train = stream->ReadValue<uint8>();
_legacyType.cars_per_flat_ride = stream->ReadValue<uint8>();
_legacyType.zero_cars = stream->ReadValue<uint8>();
_legacyType.tab_vehicle = stream->ReadValue<uint8>();
_legacyType.default_vehicle = stream->ReadValue<uint8>();
_legacyType.front_vehicle = stream->ReadValue<uint8>();
_legacyType.second_vehicle = stream->ReadValue<uint8>();
_legacyType.rear_vehicle = stream->ReadValue<uint8>();
_legacyType.third_vehicle = stream->ReadValue<uint8>();
_legacyType.pad_019 = stream->ReadValue<uint8>();
for (auto &vehicleEntry : _legacyType.vehicles)
{
ReadLegacyVehicle(context, stream, &vehicleEntry);
}
stream->Seek(4, STREAM_SEEK_CURRENT);
2017-11-13 16:22:36 +01:00
_legacyType.excitement_multiplier = stream->ReadValue<sint8>();
_legacyType.intensity_multiplier = stream->ReadValue<sint8>();
_legacyType.nausea_multiplier = stream->ReadValue<sint8>();
_legacyType.max_height = stream->ReadValue<uint8>();
_legacyType.enabledTrackPieces = stream->ReadValue<uint64>();
_legacyType.category[0] = stream->ReadValue<uint8>();
_legacyType.category[1] = stream->ReadValue<uint8>();
_legacyType.shop_item = stream->ReadValue<uint8>();
_legacyType.shop_item_secondary = stream->ReadValue<uint8>();
2016-06-25 23:07:01 +02:00
GetStringTable().Read(context, stream, OBJ_STRING_ID_NAME);
GetStringTable().Read(context, stream, OBJ_STRING_ID_DESCRIPTION);
GetStringTable().Read(context, stream, OBJ_STRING_ID_CAPACITY);
2016-06-25 23:07:01 +02:00
// Read preset colours, by default there are 32
_presetColours.count = stream->ReadValue<uint8>();
2016-11-13 13:37:30 +01:00
2017-01-04 22:17:08 +01:00
sint32 coloursCount = _presetColours.count;
2016-11-13 20:17:49 +01:00
// To indicate a ride has different colours each train the count
// is set to 255. There are only actually 32 colours though.
if (coloursCount == 255)
2016-06-25 23:07:01 +02:00
{
coloursCount = 32;
2016-06-25 23:07:01 +02:00
}
for (uint8 i = 0; i < coloursCount; i++)
2016-06-25 23:07:01 +02:00
{
_presetColours.list[i] = stream->ReadValue<vehicle_colour>();
}
// Read peep loading positions
for (sint32 i = 0; i < RCT2_MAX_VEHICLES_PER_RIDE_ENTRY; i++)
2016-06-25 23:07:01 +02:00
{
uint16 numPeepLoadingPositions = stream->ReadValue<uint8>();
if (numPeepLoadingPositions == 255)
{
numPeepLoadingPositions = stream->ReadValue<uint16>();
}
auto data = stream->ReadArray<sint8>(numPeepLoadingPositions);
_peepLoadingPositions[i] = std::vector<sint8>(data, data + numPeepLoadingPositions);
Memory::Free(data);
2016-06-25 23:07:01 +02:00
}
GetImageTable().Read(context, stream);
2016-07-02 18:23:06 +02:00
// Validate properties
2017-11-13 16:22:36 +01:00
if (_legacyType.excitement_multiplier > 75)
2016-07-02 18:23:06 +02:00
{
context->LogError(OBJECT_ERROR_INVALID_PROPERTY, "Excitement multiplier too high.");
}
2017-11-13 16:22:36 +01:00
if (_legacyType.intensity_multiplier > 75)
2016-07-02 18:23:06 +02:00
{
context->LogError(OBJECT_ERROR_INVALID_PROPERTY, "Intensity multiplier too high.");
}
2017-11-13 16:22:36 +01:00
if (_legacyType.nausea_multiplier > 75)
2016-07-02 18:23:06 +02:00
{
context->LogError(OBJECT_ERROR_INVALID_PROPERTY, "Nausea multiplier too high.");
}
PerformFixes();
2016-06-25 23:07:01 +02:00
}
void RideObject::Load()
{
GetStringTable().Sort();
2017-09-19 12:18:17 +02:00
_legacyType.naming.name = language_allocate_object_string(GetName());
_legacyType.naming.description = language_allocate_object_string(GetDescription());
_legacyType.capacity = language_allocate_object_string(GetCapacity());
_legacyType.images_offset = gfx_object_allocate_images(GetImageTable().GetImages(), GetImageTable().GetCount());
2016-07-02 22:52:17 +02:00
_legacyType.vehicle_preset_list = &_presetColours;
2016-06-25 23:07:01 +02:00
sint32 cur_vehicle_images_offset = _legacyType.images_offset + MAX_RIDE_TYPES_PER_RIDE_ENTRY;
for (sint32 i = 0; i < RCT2_MAX_VEHICLES_PER_RIDE_ENTRY; i++)
2016-06-25 23:07:01 +02:00
{
rct_ride_entry_vehicle * vehicleEntry = &_legacyType.vehicles[i];
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_FLAT)
{
// RCT2 calculates num_vertical_frames and num_horizontal_frames and overwrites these properties on the vehicle entry.
// Immediately afterwards, the two were multiplied in order to calculate base_num_frames and were never used again.
// This has been changed to use the calculation results directly -
// num_vertical_frames and num_horizontal_frames are no longer set on the vehicle entry.
2016-06-25 23:07:01 +02:00
// 0x6DE946
vehicleEntry->base_num_frames = CalculateNumVerticalFrames(vehicleEntry) * CalculateNumHorizontalFrames(vehicleEntry);
2016-06-25 23:07:01 +02:00
vehicleEntry->base_image_id = cur_vehicle_images_offset;
2017-01-04 22:17:08 +01:00
sint32 image_index = vehicleEntry->base_image_id;
2016-06-25 23:07:01 +02:00
if (vehicleEntry->car_visual != VEHICLE_VISUAL_RIVER_RAPIDS)
{
sint32 b = vehicleEntry->base_num_frames * 32;
2016-06-25 23:07:01 +02:00
2017-07-31 12:27:04 +02:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_11) b /= 2;
2016-06-25 23:07:01 +02:00
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_15) b /= 8;
image_index += b;
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_GENTLE_SLOPES)
{
vehicleEntry->gentle_slope_image_id = image_index;
b = vehicleEntry->base_num_frames * 72;
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_SPINNING_ADDITIONAL_FRAMES)
2016-06-25 23:07:01 +02:00
{
b = vehicleEntry->base_num_frames * 16;
2016-06-25 23:07:01 +02:00
}
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_STEEP_SLOPES)
{
vehicleEntry->steep_slope_image_id = image_index;
b = vehicleEntry->base_num_frames * 80;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_VERTICAL_SLOPES)
{
vehicleEntry->vertical_slope_image_id = image_index;
b = vehicleEntry->base_num_frames * 116;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_DIAGONAL_SLOPES)
{
vehicleEntry->diagonal_slope_image_id = image_index;
b = vehicleEntry->base_num_frames * 24;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_FLAT_BANKED)
{
vehicleEntry->banked_image_id = image_index;
b = vehicleEntry->base_num_frames * 80;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_INLINE_TWISTS)
{
vehicleEntry->inline_twist_image_id = image_index;
b = vehicleEntry->base_num_frames * 40;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_FLAT_TO_GENTLE_SLOPE_BANKED_TRANSITIONS)
{
vehicleEntry->flat_to_gentle_bank_image_id = image_index;
b = vehicleEntry->base_num_frames * 128;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_DIAGONAL_GENTLE_SLOPE_BANKED_TRANSITIONS)
{
vehicleEntry->diagonal_to_gentle_slope_bank_image_id = image_index;
b = vehicleEntry->base_num_frames * 16;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_GENTLE_SLOPE_BANKED_TRANSITIONS)
{
vehicleEntry->gentle_slope_to_bank_image_id = image_index;
b = vehicleEntry->base_num_frames * 16;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_GENTLE_SLOPE_BANKED_TURNS)
{
vehicleEntry->gentle_slope_bank_turn_image_id = image_index;
b = vehicleEntry->base_num_frames * 128;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_FLAT_TO_GENTLE_SLOPE_WHILE_BANKED_TRANSITIONS)
{
vehicleEntry->flat_bank_to_gentle_slope_image_id = image_index;
b = vehicleEntry->base_num_frames * 16;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_CORKSCREWS)
{
vehicleEntry->corkscrew_image_id = image_index;
b = vehicleEntry->base_num_frames * 80;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_RESTRAINT_ANIMATION)
{
vehicleEntry->restraint_image_id = image_index;
b = vehicleEntry->base_num_frames * 12;
2016-06-25 23:07:01 +02:00
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_CURVED_LIFT_HILL)
2016-06-25 23:07:01 +02:00
{
// Same offset as corkscrew
vehicleEntry->curved_lift_hill_image_id = image_index;
b = vehicleEntry->base_num_frames * 32;
2016-06-25 23:07:01 +02:00
image_index += b;
}
}
else
{
image_index += vehicleEntry->base_num_frames * 36;
2016-06-25 23:07:01 +02:00
}
// No vehicle images
vehicleEntry->no_vehicle_images = image_index - cur_vehicle_images_offset;
// Move the offset over this vehicles images. Including peeps
cur_vehicle_images_offset = image_index + vehicleEntry->no_seating_rows * vehicleEntry->no_vehicle_images;
// 0x6DEB0D
2017-07-31 12:27:04 +02:00
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_10))
2016-06-25 23:07:01 +02:00
{
2017-01-04 22:17:08 +01:00
sint32 num_images = cur_vehicle_images_offset - vehicleEntry->base_image_id;
2017-07-31 12:27:04 +02:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_13)
2016-06-25 23:07:01 +02:00
{
num_images *= 2;
}
2017-06-08 00:17:19 +02:00
if (!gOpenRCT2NoGraphics) {
set_vehicle_type_image_max_sizes(vehicleEntry, num_images);
}
2016-06-25 23:07:01 +02:00
}
vehicleEntry->peep_loading_positions = _peepLoadingPositions[i].data();
vehicleEntry->peep_loading_positions_count = (uint16)_peepLoadingPositions[i].size();
2016-06-25 23:07:01 +02:00
}
}
}
void RideObject::Unload()
{
2017-09-19 12:18:17 +02:00
language_free_object_string(_legacyType.naming.name);
language_free_object_string(_legacyType.naming.description);
language_free_object_string(_legacyType.capacity);
gfx_object_free_images(_legacyType.images_offset, GetImageTable().GetCount());
2017-09-19 12:18:17 +02:00
_legacyType.naming.name = 0;
_legacyType.naming.description = 0;
_legacyType.capacity = 0;
_legacyType.images_offset = 0;
2016-07-02 13:06:29 +02:00
}
void RideObject::DrawPreview(rct_drawpixelinfo * dpi, sint32 width, sint32 height) const
2016-07-02 13:06:29 +02:00
{
uint32 imageId = _legacyType.images_offset;
for (auto rideType : _legacyType.ride_type)
2016-07-02 13:06:29 +02:00
{
if (rideType != RIDE_TYPE_NULL)
break;
else
2016-07-02 13:06:29 +02:00
imageId++;
}
2016-07-02 13:06:29 +02:00
gfx_draw_sprite(dpi, imageId, 0, 0, 0);
2016-06-25 23:07:01 +02:00
}
2017-12-05 13:55:33 +01:00
std::string RideObject::GetDescription() const
2016-06-25 23:07:01 +02:00
{
2016-07-04 19:53:35 +02:00
return GetString(OBJ_STRING_ID_DESCRIPTION);
2016-06-25 23:07:01 +02:00
}
2017-12-05 13:55:33 +01:00
std::string RideObject::GetCapacity() const
2016-06-25 23:07:01 +02:00
{
2016-07-04 19:53:35 +02:00
return GetString(OBJ_STRING_ID_CAPACITY);
2016-06-25 23:07:01 +02:00
}
2016-06-29 23:15:38 +02:00
void RideObject::SetRepositoryItem(ObjectRepositoryItem * item) const
{
for (sint32 i = 0; i < RCT2_MAX_RIDE_TYPES_PER_RIDE_ENTRY; i++)
2016-06-29 23:15:38 +02:00
{
item->RideType[i] = _legacyType.ride_type[i];
}
for (sint32 i = 0; i < RCT2_MAX_CATEGORIES_PER_RIDE; i++)
2016-06-29 23:15:38 +02:00
{
item->RideCategory[i] = _legacyType.category[i];
}
uint8 flags = 0;
item->RideFlags = flags;
// Find the first non-null ride type, to be used when checking the ride group
uint8 rideTypeIdx = ride_entry_get_first_non_null_ride_type(&_legacyType);
// Determines the ride group. Will fall back to 0 if there is none found.
uint8 rideGroupIndex = 0;
const RideGroup * rideGroup = RideGroupManager::GetRideGroup(rideTypeIdx, &_legacyType);
2017-08-15 10:07:44 +02:00
// If the ride group is nullptr, the track type does not have ride groups.
if (rideGroup != nullptr)
{
for (uint8 i = rideGroupIndex + 1; i < MAX_RIDE_GROUPS_PER_RIDE_TYPE; i++)
{
2017-10-06 10:29:24 +02:00
const RideGroup * irg = RideGroupManager::RideGroupFind(rideTypeIdx, i);
2017-08-15 10:07:44 +02:00
if (irg != nullptr)
{
2017-10-06 10:29:24 +02:00
if (RideGroupManager::RideGroupsAreEqual(irg, rideGroup))
{
rideGroupIndex = i;
break;
}
}
}
}
item->RideGroupIndex = rideGroupIndex;
2016-06-29 23:15:38 +02:00
}
void RideObject::ReadLegacyVehicle(IReadObjectContext * context, IStream * stream, rct_ride_entry_vehicle * vehicle)
{
vehicle->rotation_frame_mask = stream->ReadValue<uint16>();
vehicle->num_vertical_frames = stream->ReadValue<uint8>();
vehicle->num_horizontal_frames = stream->ReadValue<uint8>();
vehicle->spacing = stream->ReadValue<uint32>();
2017-12-17 18:00:42 +01:00
vehicle->car_mass = stream->ReadValue<uint16>();
vehicle->tab_height = stream->ReadValue<sint8>();
vehicle->num_seats = stream->ReadValue<uint8>();
vehicle->sprite_flags = stream->ReadValue<uint16>();
vehicle->sprite_width = stream->ReadValue<uint8>();
vehicle->sprite_height_negative = stream->ReadValue<uint8>();
vehicle->sprite_height_positive = stream->ReadValue<uint8>();
vehicle->animation = stream->ReadValue<uint8>();
2017-07-31 12:27:04 +02:00
vehicle->flags = stream->ReadValue<uint32>();
vehicle->base_num_frames = stream->ReadValue<uint16>();
stream->Seek(4, STREAM_SEEK_CURRENT);
vehicle->restraint_image_id = stream->ReadValue<uint32>();
vehicle->gentle_slope_image_id = stream->ReadValue<uint32>();
vehicle->steep_slope_image_id = stream->ReadValue<uint32>();
vehicle->vertical_slope_image_id = stream->ReadValue<uint32>();
vehicle->diagonal_slope_image_id = stream->ReadValue<uint32>();
vehicle->banked_image_id = stream->ReadValue<uint32>();
vehicle->inline_twist_image_id = stream->ReadValue<uint32>();
vehicle->flat_to_gentle_bank_image_id = stream->ReadValue<uint32>();
vehicle->diagonal_to_gentle_slope_bank_image_id = stream->ReadValue<uint32>();
vehicle->gentle_slope_to_bank_image_id = stream->ReadValue<uint32>();
vehicle->gentle_slope_bank_turn_image_id = stream->ReadValue<uint32>();
vehicle->flat_bank_to_gentle_slope_image_id = stream->ReadValue<uint32>();
vehicle->corkscrew_image_id = stream->ReadValue<uint32>();
vehicle->no_vehicle_images = stream->ReadValue<uint32>();
vehicle->no_seating_rows = stream->ReadValue<uint8>();
vehicle->spinning_inertia = stream->ReadValue<uint8>();
vehicle->spinning_friction = stream->ReadValue<uint8>();
vehicle->friction_sound_id = stream->ReadValue<uint8>();
vehicle->log_flume_reverser_vehicle_type = stream->ReadValue<uint8>();
vehicle->sound_range = stream->ReadValue<uint8>();
vehicle->double_sound_frequency = stream->ReadValue<uint8>();
vehicle->powered_acceleration = stream->ReadValue<uint8>();
vehicle->powered_max_speed = stream->ReadValue<uint8>();
vehicle->car_visual = stream->ReadValue<uint8>();
vehicle->effect_visual = stream->ReadValue<uint8>();
vehicle->draw_order = stream->ReadValue<uint8>();
vehicle->num_vertical_frames_override = stream->ReadValue<uint8>();
stream->Seek(4, STREAM_SEEK_CURRENT);
}
void RideObject::PerformFixes()
{
std::string identifier = GetIdentifier();
// Add boosters if the track type is eligible
for (auto rideType : _legacyType.ride_type)
{
if (ride_type_supports_boosters(rideType))
{
_legacyType.enabledTrackPieces |= (1ULL << TRACK_BOOSTER);
}
}
// The rocket cars could take 3 cars per train in RCT1. Restore this.
if (String::Equals(identifier, "RCKC "))
{
_legacyType.max_cars_in_train = 3 + _legacyType.zero_cars;
}
// The Wooden Roller Coaster could take 7 cars per train in RCT1.
else if (String::Equals(identifier, "PTCT1 "))
{
_legacyType.max_cars_in_train = 7 + _legacyType.zero_cars;
}
// The Looping Roller Coaster could take 8 cars per train in RCT1.
else if (String::Equals(identifier, "SCHT1 "))
{
_legacyType.max_cars_in_train = 8 + _legacyType.zero_cars;
}
2017-11-13 20:37:13 +01:00
// The Steel Twister could take 8 cars per train in RCT1. (The other two vehicles are already correct.)
else if (String::Equals(identifier, "BMSD ") || String::Equals(identifier, "BMSU "))
{
_legacyType.max_cars_in_train = 8 + _legacyType.zero_cars;
}
// Wacky Worlds' Crocodile Ride (a log flume vehicle) is incorrectly locked to 5 cars.
else if (String::Equals(identifier, "CROCFLUM"))
{
_legacyType.cars_per_flat_ride = 0xFF;
}
// All vanilla/WW/OCC Junior RC vehicles incorrectly have this flag set
else if (String::Equals(identifier, "ZLDB ") ||
String::Equals(identifier, "ZLOG ") ||
String::Equals(identifier, "ZPANDA ") ||
String::Equals(identifier, "WHICGRUB"))
{
_legacyType.enabledTrackPieces &= ~(1ULL << TRACK_SLOPE_STEEP);
}
}
uint8 RideObject::CalculateNumVerticalFrames(const rct_ride_entry_vehicle * vehicleEntry)
{
// 0x6DE90B
uint8 numVerticalFrames;
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_OVERRIDE_NUM_VERTICAL_FRAMES)
{
numVerticalFrames = vehicleEntry->num_vertical_frames_override;
}
else
{
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_SPINNING_ADDITIONAL_FRAMES))
{
2018-01-17 22:56:22 +01:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_VEHICLE_ANIMATION && vehicleEntry->animation != VEHICLE_ENTRY_ANIMATION_OBSERVATION_TOWER)
{
2018-01-13 10:07:52 +01:00
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_DODGEM_INUSE_LIGHTS))
{
numVerticalFrames = 4;
}
else
{
numVerticalFrames = 2;
}
}
else
{
numVerticalFrames = 1;
}
}
else
{
numVerticalFrames = 32;
}
}
return numVerticalFrames;
}
uint8 RideObject::CalculateNumHorizontalFrames(const rct_ride_entry_vehicle * vehicleEntry)
{
uint8 numHorizontalFrames;
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_SWINGING)
{
2018-01-17 22:49:38 +01:00
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_21) && !(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_SLIDE_SWING))
{
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_25)
{
numHorizontalFrames = 3;
}
else
{
numHorizontalFrames = 5;
}
}
2018-01-17 22:49:38 +01:00
else if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_21) || !(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_SLIDE_SWING))
{
numHorizontalFrames = 7;
}
else
{
numHorizontalFrames = 13;
}
}
else
{
numHorizontalFrames = 1;
}
return numHorizontalFrames;
}
2017-12-04 22:56:06 +01:00
2017-12-20 17:31:06 +01:00
void RideObject::ReadJson(IReadObjectContext * context, const json_t * root)
{
auto properties = json_object_get(root, "properties");
auto rideTypes = ObjectJsonHelpers::GetJsonStringArray(json_object_get(properties, "type"));
for (size_t i = 0; i < std::min<size_t>(MAX_RIDE_TYPES_PER_RIDE_ENTRY, rideTypes.size()); i++)
{
_legacyType.ride_type[i] = ParseRideType(rideTypes[i]);
}
auto rideCategories = ObjectJsonHelpers::GetJsonStringArray(json_object_get(properties, "category"));
for (size_t i = 0; i < std::min<size_t>(MAX_CATEGORIES_PER_RIDE, rideCategories.size()); i++)
{
_legacyType.category[0] = ParseRideCategory(rideCategories[i]);
}
_legacyType.max_height = ObjectJsonHelpers::GetInteger(properties, "maxHeight");
if (IsRideTypeShopOrFacility(_legacyType.ride_type[0]))
{
// Standard car info for a shop
auto &car = _legacyType.vehicles[0];
car.spacing = 544;
car.sprite_flags = VEHICLE_SPRITE_FLAG_FLAT;
car.sprite_width = 1;
car.sprite_height_negative = 1;
car.sprite_height_positive = 1;
car.flags = VEHICLE_ENTRY_FLAG_SPINNING;
car.car_visual = VEHICLE_VISUAL_FLAT_RIDE_OR_CAR_RIDE;
car.friction_sound_id = 0xFF;
car.sound_range = 0xFF;
car.draw_order = 6;
// Shop item
auto rideSells = ObjectJsonHelpers::GetJsonStringArray(json_object_get(json_object_get(root, "properties"), "sells"));
_legacyType.shop_item = SHOP_ITEM_NONE;
_legacyType.shop_item_secondary = SHOP_ITEM_NONE;
if (rideSells.size() >= 1)
{
_legacyType.shop_item = ParseShopItem(rideSells[0]);
}
if (rideSells.size() >= 2)
{
_legacyType.shop_item_secondary = ParseShopItem(rideSells[1]);
}
}
else
{
ReadJsonVehicleInfo(context, properties);
2017-12-22 23:58:02 +01:00
auto swingMode = ObjectJsonHelpers::GetInteger(properties, "swingMode");
if (swingMode == 1)
{
_legacyType.flags |= RIDE_ENTRY_FLAG_ALTERNATIVE_SWING_MODE_1;
}
else if (swingMode == 2)
{
_legacyType.flags |= RIDE_ENTRY_FLAG_ALTERNATIVE_SWING_MODE_1;
_legacyType.flags |= RIDE_ENTRY_FLAG_ALTERNATIVE_SWING_MODE_2;
}
auto rotationMode = ObjectJsonHelpers::GetInteger(properties, "rotationMode");
if (rotationMode == 1)
{
_legacyType.flags |= RIDE_ENTRY_FLAG_ALTERNATIVE_ROTATION_MODE_1;
}
else if (rotationMode == 2)
{
_legacyType.flags |= RIDE_ENTRY_FLAG_ALTERNATIVE_ROTATION_MODE_2;
}
2017-12-20 17:31:06 +01:00
auto ratingMultiplier = json_object_get(properties, "ratingMultipler");
if (ratingMultiplier != nullptr)
{
_legacyType.excitement_multiplier = ObjectJsonHelpers::GetInteger(ratingMultiplier, "excitement");
_legacyType.intensity_multiplier = ObjectJsonHelpers::GetInteger(ratingMultiplier, "intensity");
_legacyType.nausea_multiplier = ObjectJsonHelpers::GetInteger(ratingMultiplier, "nausea");
}
auto availableTrackPieces = ObjectJsonHelpers::GetJsonStringArray(json_object_get(properties, "availableTrackPieces"));
2017-12-20 22:47:57 +01:00
_presetColours = ReadJsonCarColours(json_object_get(properties, "carColours"));
// Get loading positions
auto loadingPositions = json_object_get(properties, "loadingPositions");
auto numLoadingPositions = std::min<size_t>(json_array_size(loadingPositions), 4);
for (size_t i = 0; i < numLoadingPositions; i++)
{
auto positions = json_array_get(loadingPositions, i);
auto numPositions = json_array_size(positions);
if (numPositions > 0 && numPositions <= std::numeric_limits<uint16>::max())
{
std::vector<sint8> positionData;
if (numPositions < 255)
{
positionData.push_back(static_cast<sint8>(numPositions));
}
else
{
positionData.push_back(-1);
positionData.push_back(static_cast<sint8>(numPositions & 0xFF));
positionData.push_back(static_cast<sint8>(numPositions >> 8));
}
for (size_t j = 0; j < numPositions; j++)
{
auto pos = json_integer_value(json_array_get(positions, j));
pos = Math::Clamp<sint64>(std::numeric_limits<sint8>::min(), pos, std::numeric_limits<sint8>::max());
positionData.push_back(pos);
}
_peepLoadingPositions[i] = std::move(positionData);
}
}
2017-12-20 17:31:06 +01:00
}
_legacyType.flags |= ObjectJsonHelpers::GetFlags<uint32>(properties, {
{ "noInversions", RIDE_ENTRY_FLAG_NO_INVERSIONS },
{ "noBanking", RIDE_ENTRY_FLAG_NO_BANKED_TRACK },
{ "playDepartSound", RIDE_ENTRY_FLAG_PLAY_DEPART_SOUND },
2017-12-22 23:58:02 +01:00
{ "RIDE_ENTRY_FLAG_7", RIDE_ENTRY_FLAG_7 },
2017-12-20 17:31:06 +01:00
{ "playSplashSound", RIDE_ENTRY_FLAG_PLAY_SPLASH_SOUND },
{ "playSplashSoundSlide", RIDE_ENTRY_FLAG_PLAY_SPLASH_SOUND_SLIDE },
{ "hasShelter", RIDE_ENTRY_FLAG_COVERED_RIDE },
{ "limitAirTimeBonus", RIDE_ENTRY_FLAG_LIMIT_AIRTIME_BONUS },
{ "separateRide", RIDE_ENTRY_FLAG_SEPARATE_RIDE },
{ "disableBreakdown", RIDE_ENTRY_FLAG_CANNOT_BREAK_DOWN },
2017-12-22 23:58:02 +01:00
{ "RIDE_ENTRY_FLAG_16", RIDE_ENTRY_FLAG_16 },
{ "RIDE_ENTRY_FLAG_18", RIDE_ENTRY_FLAG_18 },
{ "disablePainting", RIDE_ENTRY_FLAG_DISABLE_COLOUR_TAB } });
2017-12-20 17:31:06 +01:00
ObjectJsonHelpers::LoadStrings(root, GetStringTable());
ObjectJsonHelpers::LoadImages(root, GetImageTable());
}
void RideObject::ReadJsonVehicleInfo(IReadObjectContext * context, const json_t * properties)
{
_legacyType.min_cars_in_train = json_integer_value(json_object_get(properties, "minCarsPerTrain"));
_legacyType.max_cars_in_train = json_integer_value(json_object_get(properties, "maxCarsPerTrain"));
_legacyType.cars_per_flat_ride = json_integer_value(json_object_get(properties, "carsPerFlatRide"));
_legacyType.zero_cars = json_integer_value(json_object_get(properties, "numEmptyCars"));
// Train formation from car indices
_legacyType.default_vehicle = json_integer_value(json_object_get(properties, "defaultCar"));
_legacyType.tab_vehicle = json_integer_value(json_object_get(properties, "tabCar"));
auto tabScale = ObjectJsonHelpers::GetFloat(properties, "tabScale");
if (tabScale != 0 && ObjectJsonHelpers::GetFloat(properties, "tabScale") <= 0.5f)
{
_legacyType.flags |= RIDE_ENTRY_FLAG_VEHICLE_TAB_SCALE_HALF;
}
auto headCars = ObjectJsonHelpers::GetJsonIntegerArray(json_object_get(properties, "headCars"));
if (headCars.size() >= 1)
{
_legacyType.front_vehicle = headCars[0];
}
if (headCars.size() >= 2)
{
_legacyType.second_vehicle = headCars[1];
}
if (headCars.size() >= 3)
{
_legacyType.third_vehicle = headCars[2];
}
if (headCars.size() >= 4)
{
// More than 3 head cars not supported yet!
}
auto tailCars = ObjectJsonHelpers::GetJsonIntegerArray(json_object_get(properties, "tailCars"));
if (tailCars.size() >= 1)
{
_legacyType.rear_vehicle = tailCars[0];
}
if (tailCars.size() >= 2)
{
// More than 1 tail car not supported yet!
}
2017-12-20 22:47:57 +01:00
auto cars = ReadJsonCars(json_object_get(properties, "cars"));
auto numCars = std::min(Util::CountOf(_legacyType.vehicles), cars.size());
for (size_t i = 0; i < numCars; i++)
{
_legacyType.vehicles[i] = cars[i];
}
2017-12-20 17:31:06 +01:00
}
std::vector<rct_ride_entry_vehicle> RideObject::ReadJsonCars(const json_t * jCars)
{
std::vector<rct_ride_entry_vehicle> cars;
json_t * jCar;
size_t index;
json_array_foreach(jCars, index, jCar)
{
auto car = ReadJsonCar(jCar);
cars.push_back(car);
}
return cars;
}
rct_ride_entry_vehicle RideObject::ReadJsonCar(const json_t * jCar)
{
rct_ride_entry_vehicle car = { 0 };
car.rotation_frame_mask = ObjectJsonHelpers::GetInteger(jCar, "rotationFrameMask");
car.spacing = ObjectJsonHelpers::GetInteger(jCar, "spacing");
car.car_friction = ObjectJsonHelpers::GetInteger(jCar, "friction");
2017-12-20 22:47:57 +01:00
car.tab_height = ObjectJsonHelpers::GetInteger(jCar, "tabOffset");
2017-12-20 17:31:06 +01:00
car.num_seats = ObjectJsonHelpers::GetInteger(jCar, "numSeats");
car.no_seating_rows = ObjectJsonHelpers::GetInteger(jCar, "numSeatRows");
2017-12-20 22:47:57 +01:00
car.spinning_inertia = ObjectJsonHelpers::GetInteger(jCar, "spinningInertia");
car.spinning_friction = ObjectJsonHelpers::GetInteger(jCar, "spinningFriction");
2017-12-20 17:31:06 +01:00
car.powered_acceleration = ObjectJsonHelpers::GetInteger(jCar, "poweredAcceleration");
car.powered_max_speed = ObjectJsonHelpers::GetInteger(jCar, "poweredMaxSpeed");
2017-12-20 22:47:57 +01:00
car.car_visual = ObjectJsonHelpers::GetInteger(jCar, "carVisual");
2017-12-20 17:31:06 +01:00
car.effect_visual = ObjectJsonHelpers::GetInteger(jCar, "effectVisual");
car.draw_order = ObjectJsonHelpers::GetInteger(jCar, "drawOrder");
return car;
}
2017-12-20 22:47:57 +01:00
vehicle_colour_preset_list RideObject::ReadJsonCarColours(const json_t * jCarColours)
{
// The JSON supports multiple configurations of per car colours, but
// the ride entry structure currently doesn't allow for it. Assume that
// a single configuration with multiple colour entries is per car scheme.
if (json_array_size(jCarColours) == 1)
{
auto firstElement = json_array_get(jCarColours, 0);
auto numColours = json_array_size(firstElement);
if (numColours >= 2)
{
// Read all colours from first config
auto config = ReadJsonColourConfiguration(firstElement);
vehicle_colour_preset_list list = { 0 };
list.count = 255;
std::copy_n(config.data(), std::min<size_t>(numColours, 32), list.list);
return list;
}
}
// Read first colour for each config
vehicle_colour_preset_list list = { 0 };
size_t index;
const json_t * jConfiguration;
json_array_foreach(jCarColours, index, jConfiguration)
{
auto config = ReadJsonColourConfiguration(jConfiguration);
if (config.size() >= 1)
{
list.list[index] = config[0];
list.count++;
if (list.count == 254)
{
// Reached maximum number of configurations
break;
}
}
}
return list;
}
std::vector<vehicle_colour> RideObject::ReadJsonColourConfiguration(const json_t * jColourConfig)
{
std::vector<vehicle_colour> config;
size_t index;
const json_t * jColours;
json_array_foreach(jColourConfig, index, jColours)
{
vehicle_colour carColour = { 0 };
auto colours = ObjectJsonHelpers::GetJsonStringArray(jColours);
if (colours.size() >= 1)
{
carColour.main = ParseColour(colours[0]);
carColour.additional_1 = carColour.main;
carColour.additional_2 = carColour.main;
if (colours.size() >= 2)
{
carColour.additional_1 = ParseColour(colours[1]);
}
if (colours.size() >= 3)
{
carColour.additional_2 = ParseColour(colours[2]);
}
}
config.push_back(carColour);
}
return config;
}
2017-12-20 17:31:06 +01:00
bool RideObject::IsRideTypeShopOrFacility(uint8 rideType)
{
switch (rideType)
{
case RIDE_TYPE_TOILETS:
case RIDE_TYPE_SHOP:
case RIDE_TYPE_DRINK_STALL:
case RIDE_TYPE_FOOD_STALL:
case RIDE_TYPE_INFORMATION_KIOSK:
case RIDE_TYPE_CASH_MACHINE:
case RIDE_TYPE_FIRST_AID:
return true;
default:
return false;
}
}
uint8 RideObject::ParseRideType(const std::string &s)
2017-12-07 13:53:54 +01:00
{
static const std::unordered_map<std::string, uint8> LookupTable
{
{ "spiral_rc", RIDE_TYPE_SPIRAL_ROLLER_COASTER },
{ "stand_up_rc", RIDE_TYPE_STAND_UP_ROLLER_COASTER },
{ "suspended_swinging_rc", RIDE_TYPE_SUSPENDED_SWINGING_COASTER },
{ "inverted_rc", RIDE_TYPE_INVERTED_ROLLER_COASTER },
{ "junior_rc", RIDE_TYPE_JUNIOR_ROLLER_COASTER },
{ "miniature_railway", RIDE_TYPE_MINIATURE_RAILWAY },
{ "monorail", RIDE_TYPE_MONORAIL },
{ "mini_suspended_rc", RIDE_TYPE_MINI_SUSPENDED_COASTER },
2017-12-08 08:08:46 +01:00
{ "boat_hire", RIDE_TYPE_BOAT_HIRE },
2017-12-07 13:53:54 +01:00
{ "wooden_wild_mouse", RIDE_TYPE_WOODEN_WILD_MOUSE },
{ "steeplechase", RIDE_TYPE_STEEPLECHASE },
{ "car_ride", RIDE_TYPE_CAR_RIDE },
{ "launched_freefall", RIDE_TYPE_LAUNCHED_FREEFALL },
{ "bobsleigh_rc", RIDE_TYPE_BOBSLEIGH_COASTER },
{ "observation_tower", RIDE_TYPE_OBSERVATION_TOWER },
{ "looping_rc", RIDE_TYPE_LOOPING_ROLLER_COASTER },
{ "dinghy_slide", RIDE_TYPE_DINGHY_SLIDE },
{ "mine_train_rc", RIDE_TYPE_MINE_TRAIN_COASTER },
{ "chairlift", RIDE_TYPE_CHAIRLIFT },
{ "corkscrew_rc", RIDE_TYPE_CORKSCREW_ROLLER_COASTER },
{ "maze", RIDE_TYPE_MAZE },
{ "spiral_slide", RIDE_TYPE_SPIRAL_SLIDE },
{ "go_karts", RIDE_TYPE_GO_KARTS },
{ "log_flume", RIDE_TYPE_LOG_FLUME },
{ "river_rapids", RIDE_TYPE_RIVER_RAPIDS },
{ "dodgems", RIDE_TYPE_DODGEMS },
{ "swinging_ship", RIDE_TYPE_SWINGING_SHIP },
{ "swinging_inverter_ship", RIDE_TYPE_SWINGING_INVERTER_SHIP },
{ "food_stall", RIDE_TYPE_FOOD_STALL },
{ "drink_stall", RIDE_TYPE_DRINK_STALL },
{ "shop", RIDE_TYPE_SHOP },
{ "merry_go_round", RIDE_TYPE_MERRY_GO_ROUND },
{ "information_kiosk", RIDE_TYPE_INFORMATION_KIOSK },
{ "toilets", RIDE_TYPE_TOILETS },
{ "ferris_wheel", RIDE_TYPE_FERRIS_WHEEL },
{ "motion_simulator", RIDE_TYPE_MOTION_SIMULATOR },
{ "3d_cinema", RIDE_TYPE_3D_CINEMA },
{ "top_spin", RIDE_TYPE_TOP_SPIN },
{ "space_rings", RIDE_TYPE_SPACE_RINGS },
{ "reverse_freefall_rc", RIDE_TYPE_REVERSE_FREEFALL_COASTER },
{ "lift", RIDE_TYPE_LIFT },
{ "vertical_drop_rc", RIDE_TYPE_VERTICAL_DROP_ROLLER_COASTER },
{ "cash_machine", RIDE_TYPE_CASH_MACHINE },
{ "twist", RIDE_TYPE_TWIST },
{ "haunted_house", RIDE_TYPE_HAUNTED_HOUSE },
{ "first_aid", RIDE_TYPE_FIRST_AID },
{ "circus", RIDE_TYPE_CIRCUS },
{ "ghost_train", RIDE_TYPE_GHOST_TRAIN },
{ "twister_rc", RIDE_TYPE_TWISTER_ROLLER_COASTER },
{ "wooden_rc", RIDE_TYPE_WOODEN_ROLLER_COASTER },
{ "side_friction", RIDE_TYPE_SIDE_FRICTION_ROLLER_COASTER },
{ "steel_wild_mouse", RIDE_TYPE_STEEL_WILD_MOUSE },
{ "multi_dimension_rc", RIDE_TYPE_MULTI_DIMENSION_ROLLER_COASTER },
{ "flying_rc", RIDE_TYPE_FLYING_ROLLER_COASTER },
{ "virginia_reel", RIDE_TYPE_VIRGINIA_REEL },
{ "splash_boats", RIDE_TYPE_SPLASH_BOATS },
{ "mini_helicopters", RIDE_TYPE_MINI_HELICOPTERS },
{ "lay_down_rc", RIDE_TYPE_LAY_DOWN_ROLLER_COASTER },
{ "suspended_monorail", RIDE_TYPE_SUSPENDED_MONORAIL },
{ "reverser_rc", RIDE_TYPE_REVERSER_ROLLER_COASTER },
{ "heartline_twister_rc", RIDE_TYPE_HEARTLINE_TWISTER_COASTER },
{ "mini_golf", RIDE_TYPE_MINI_GOLF },
{ "giga_rc", RIDE_TYPE_GIGA_COASTER },
{ "roto_drop", RIDE_TYPE_ROTO_DROP },
{ "flying_saucers", RIDE_TYPE_FLYING_SAUCERS },
{ "crooked_house", RIDE_TYPE_CROOKED_HOUSE },
{ "monorail_cycles", RIDE_TYPE_MONORAIL_CYCLES },
{ "compact_inverted", RIDE_TYPE_COMPACT_INVERTED_COASTER },
{ "water_coaster", RIDE_TYPE_WATER_COASTER },
{ "air_powered_vertical_rc", RIDE_TYPE_AIR_POWERED_VERTICAL_COASTER },
{ "inverted_hairpin_rc", RIDE_TYPE_INVERTED_HAIRPIN_COASTER },
{ "magic_carpet", RIDE_TYPE_MAGIC_CARPET },
{ "submarine_ride", RIDE_TYPE_SUBMARINE_RIDE },
{ "river_rafts", RIDE_TYPE_RIVER_RAFTS },
{ "enterprise", RIDE_TYPE_ENTERPRISE },
{ "inverted_impulse_rc", RIDE_TYPE_INVERTED_IMPULSE_COASTER },
{ "mini_rc", RIDE_TYPE_MINI_ROLLER_COASTER },
{ "mine_ride", RIDE_TYPE_MINE_RIDE },
{ "lim_launched_rc", RIDE_TYPE_LIM_LAUNCHED_ROLLER_COASTER },
// TEMPORARY:
{ "restroom", RIDE_TYPE_TOILETS },
};
auto result = LookupTable.find(s);
return (result != LookupTable.end()) ?
result->second :
RIDE_TYPE_NULL;
}
2017-12-20 17:31:06 +01:00
uint8 RideObject::ParseRideCategory(const std::string &s)
2017-12-07 23:33:41 +01:00
{
static const std::unordered_map<std::string, uint8> LookupTable
{
{ "transport", RIDE_CATEGORY_TRANSPORT },
{ "gentle", RIDE_CATEGORY_GENTLE },
{ "rollercoaster", RIDE_CATEGORY_ROLLERCOASTER },
{ "thrill", RIDE_CATEGORY_THRILL },
{ "water", RIDE_CATEGORY_WATER },
{ "stall", RIDE_CATEGORY_SHOP },
};
auto result = LookupTable.find(s);
return (result != LookupTable.end()) ?
result->second :
RIDE_CATEGORY_TRANSPORT;
}
2017-12-20 17:31:06 +01:00
uint8 RideObject::ParseShopItem(const std::string &s)
2017-12-07 23:33:41 +01:00
{
static const std::unordered_map<std::string, uint8> LookupTable
{
{ "burger", SHOP_ITEM_BURGER },
{ "fries", SHOP_ITEM_CHIPS },
{ "icecream", SHOP_ITEM_ICE_CREAM },
{ "cottoncandy", SHOP_ITEM_CANDYFLOSS },
{ "pizza", SHOP_ITEM_PIZZA },
{ "popcorn", SHOP_ITEM_POPCORN },
{ "hotdog", SHOP_ITEM_HOT_DOG },
{ "seafood", SHOP_ITEM_TENTACLE },
2017-12-08 08:08:46 +01:00
{ "candyapple", SHOP_ITEM_TOFFEE_APPLE },
{ "donut", SHOP_ITEM_DOUGHNUT },
2017-12-07 23:33:41 +01:00
{ "chicken", SHOP_ITEM_CHICKEN },
{ "pretzel", SHOP_ITEM_PRETZEL },
{ "funnelcake", SHOP_ITEM_FUNNEL_CAKE },
{ "beefnoodles", SHOP_ITEM_BEEF_NOODLES },
{ "friednoodles", SHOP_ITEM_FRIED_RICE_NOODLES },
{ "wontonsoup", SHOP_ITEM_WONTON_SOUP },
{ "meatballsoup", SHOP_ITEM_MEATBALL_SOUP },
{ "subsandwich", SHOP_ITEM_SUB_SANDWICH },
{ "cookies", SHOP_ITEM_COOKIE },
{ "roastsausage", SHOP_ITEM_ROAST_SAUSAGE },
{ "cola", SHOP_ITEM_DRINK },
{ "coffee", SHOP_ITEM_COFFEE },
{ "lemonade", SHOP_ITEM_LEMONADE },
{ "hotchocolate", SHOP_ITEM_CHOCOLATE },
{ "icedtea", SHOP_ITEM_ICED_TEA },
{ "fruitjuice", SHOP_ITEM_FRUIT_JUICE },
{ "soybeanmilk", SHOP_ITEM_SOYBEAN_MILK },
{ "sujongkwa", SHOP_ITEM_SU_JONGKWA },
{ "balloon", SHOP_ITEM_BALLOON },
{ "plushtoy", SHOP_ITEM_TOY },
{ "map", SHOP_ITEM_MAP },
{ "onridephoto", SHOP_ITEM_PHOTO },
{ "umbrella", SHOP_ITEM_UMBRELLA },
{ "voucher", SHOP_ITEM_VOUCHER },
{ "hat", SHOP_ITEM_HAT },
{ "tshirt", SHOP_ITEM_TSHIRT },
{ "sunglasses", SHOP_ITEM_SUNGLASSES },
};
auto result = LookupTable.find(s);
return (result != LookupTable.end()) ?
result->second :
SHOP_ITEM_NONE;
}
2017-12-20 22:47:57 +01:00
colour_t RideObject::ParseColour(const std::string &s)
{
static const std::unordered_map<std::string, colour_t> LookupTable
{
{ "black", COLOUR_BLACK },
{ "grey", COLOUR_GREY },
{ "white", COLOUR_WHITE },
{ "dark_purple", COLOUR_DARK_PURPLE },
{ "light_purple", COLOUR_LIGHT_PURPLE },
{ "bright_purple", COLOUR_BRIGHT_PURPLE },
{ "dark_blue", COLOUR_DARK_BLUE },
{ "light_blue", COLOUR_LIGHT_BLUE },
{ "icy_blue", COLOUR_ICY_BLUE },
{ "teal", COLOUR_TEAL },
{ "aquamarine", COLOUR_AQUAMARINE },
{ "saturated_green", COLOUR_SATURATED_GREEN },
{ "dark_green", COLOUR_DARK_GREEN },
{ "moss_green", COLOUR_MOSS_GREEN },
{ "bright_green", COLOUR_BRIGHT_GREEN },
{ "olive_green", COLOUR_OLIVE_GREEN },
{ "dark_olive_green", COLOUR_DARK_OLIVE_GREEN },
{ "bright_yellow", COLOUR_BRIGHT_YELLOW },
{ "yellow", COLOUR_YELLOW },
{ "dark_yellow", COLOUR_DARK_YELLOW },
{ "light_orange", COLOUR_LIGHT_ORANGE },
{ "dark_orange", COLOUR_DARK_ORANGE },
{ "light_brown", COLOUR_LIGHT_BROWN },
{ "saturated_brown", COLOUR_SATURATED_BROWN },
{ "dark_brown", COLOUR_DARK_BROWN },
{ "salmon_pink", COLOUR_SALMON_PINK },
{ "bordeaux_red", COLOUR_BORDEAUX_RED },
{ "saturated_red", COLOUR_SATURATED_RED },
{ "bright_red", COLOUR_BRIGHT_RED },
{ "dark_pink", COLOUR_DARK_PINK },
{ "bright_pink", COLOUR_BRIGHT_PINK },
{ "light_pink", COLOUR_LIGHT_PINK },
};
auto result = LookupTable.find(s);
return (result != LookupTable.end()) ?
result->second :
COLOUR_BLACK;
}