OpenRCT2/src/openrct2/object/RideObject.cpp

507 lines
18 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
#include "../core/IStream.hpp"
#include "../core/Memory.hpp"
#include "../core/String.hpp"
2016-06-25 23:07:01 +02:00
#include "../core/Util.hpp"
2017-06-08 00:17:19 +02:00
#include "../OpenRCT2.h"
2016-06-29 23:15:38 +02:00
#include "ObjectRepository.h"
2016-06-25 23:07:01 +02:00
#include "RideObject.h"
#include "../ride/RideGroupManager.h"
2016-06-25 23:07:01 +02:00
extern "C"
{
2017-02-18 16:45:10 +01:00
#include "../config/Config.h"
2016-06-25 23:07:01 +02:00
#include "../drawing/drawing.h"
#include "../localisation/localisation.h"
#include "../rct1.h"
#include "../ride/ride.h"
2017-01-01 22:22:20 +01:00
#include "../ride/track.h"
2016-06-25 23:07:01 +02:00
}
RideObject::~RideObject()
{
2017-01-04 22:17:08 +01:00
for (sint32 i = 0; i < 4; i++)
2016-06-26 23:01:43 +02:00
{
Memory::Free(_peepLoadingPositions[i]);
}
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 (sint32 i = 0; i < RCT2_MAX_RIDE_TYPES_PER_RIDE_ENTRY; i++)
{
_legacyType.ride_type[i] = 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 (sint32 i = 0; i < RCT2_MAX_VEHICLES_PER_RIDE_ENTRY; i++)
{
rct_ride_entry_vehicle * entry = &_legacyType.vehicles[i];
ReadLegacyVehicle(context, stream, entry);
}
stream->Seek(4, STREAM_SEEK_CURRENT);
_legacyType.excitement_multipler = stream->ReadValue<sint8>();
_legacyType.intensity_multipler = stream->ReadValue<sint8>();
_legacyType.nausea_multipler = 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
2016-07-01 22:01:06 +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>();
}
_peepLoadingPositions[i] = stream->ReadArray<sint8>(numPeepLoadingPositions);
_peepLoadingPositionsCount[i] = numPeepLoadingPositions;
2016-06-25 23:07:01 +02:00
}
2016-07-01 22:01:06 +02:00
GetImageTable()->Read(context, stream);
2016-07-02 18:23:06 +02:00
// Validate properties
if (_legacyType.excitement_multipler > 75)
{
context->LogError(OBJECT_ERROR_INVALID_PROPERTY, "Excitement multiplier too high.");
}
if (_legacyType.intensity_multipler > 75)
{
context->LogError(OBJECT_ERROR_INVALID_PROPERTY, "Intensity multiplier too high.");
}
if (_legacyType.nausea_multipler > 75)
{
context->LogError(OBJECT_ERROR_INVALID_PROPERTY, "Nausea multiplier too high.");
}
PerformFixes();
2016-06-25 23:07:01 +02:00
}
void RideObject::Load()
{
2016-07-11 23:50:55 +02:00
GetStringTable()->Sort();
2016-06-25 23:07:01 +02:00
_legacyType.name = language_allocate_object_string(GetName());
_legacyType.description = language_allocate_object_string(GetDescription());
2016-06-26 22:43:30 +02:00
_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
2017-01-04 22:17:08 +01:00
sint32 cur_vehicle_images_offset = _legacyType.images_offset + 3;
for (sint32 i = 0; i < 4; 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)
{
2017-01-04 22:17:08 +01:00
sint32 al = 1;
2017-07-31 12:27:04 +02:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_SWINGING)
2016-06-25 23:07:01 +02:00
{
al = 13;
2017-07-31 12:27:04 +02:00
if ((vehicleEntry->flags & (VEHICLE_ENTRY_FLAG_21 | VEHICLE_ENTRY_FLAG_27)) != (VEHICLE_ENTRY_FLAG_21 | VEHICLE_ENTRY_FLAG_27))
2016-06-25 23:07:01 +02:00
{
al = 7;
2017-07-31 12:27:04 +02:00
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_21))
2016-06-25 23:07:01 +02:00
{
2017-07-31 12:27:04 +02:00
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_27))
2016-06-25 23:07:01 +02:00
{
al = 5;
2017-07-31 12:27:04 +02:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_25)
2016-06-25 23:07:01 +02:00
{
al = 3;
}
}
}
}
}
vehicleEntry->var_03 = al;
// 0x6DE90B
al = 0x20;
2017-07-31 12:27:04 +02:00
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_14))
2016-06-25 23:07:01 +02:00
{
al = 1;
2017-07-31 12:27:04 +02:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_23)
2016-06-25 23:07:01 +02:00
{
if (vehicleEntry->var_11 != 6)
{
al = 2;
2017-07-31 12:27:04 +02:00
if (!(vehicleEntry->flags & VEHICLE_ENTRY_FLAG_7))
2016-06-25 23:07:01 +02:00
{
al = 4;
}
}
}
}
2017-07-31 12:27:04 +02:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_12)
2016-06-25 23:07:01 +02:00
{
al = vehicleEntry->special_frames;
}
vehicleEntry->var_02 = al;
// 0x6DE946
vehicleEntry->var_16 = vehicleEntry->var_02 * vehicleEntry->var_03;
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)
{
2017-01-04 22:17:08 +01:00
sint32 b = vehicleEntry->var_16 * 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;
// Incline 25
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_GENTLE_SLOPES)
{
vehicleEntry->var_20 = image_index;
b = vehicleEntry->var_16 * 72;
2017-07-31 12:27:04 +02:00
if (vehicleEntry->flags & VEHICLE_ENTRY_FLAG_14)
2016-06-25 23:07:01 +02:00
{
b = vehicleEntry->var_16 * 16;
}
image_index += b;
}
// Incline 60
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_STEEP_SLOPES)
{
vehicleEntry->var_24 = image_index;
b = vehicleEntry->var_16 * 80;
image_index += b;
}
2017-04-30 02:00:31 +02:00
// Vertical
2016-06-25 23:07:01 +02:00
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_VERTICAL_SLOPES)
{
vehicleEntry->var_28 = image_index;
b = vehicleEntry->var_16 * 116;
image_index += b;
}
// Unknown
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_DIAGONAL_SLOPES)
{
vehicleEntry->var_2C = image_index;
b = vehicleEntry->var_16 * 24;
image_index += b;
}
// Bank
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_FLAT_BANKED)
{
vehicleEntry->var_30 = image_index;
b = vehicleEntry->var_16 * 80;
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_INLINE_TWISTS)
{
vehicleEntry->var_34 = image_index;
b = vehicleEntry->var_16 * 40;
image_index += b;
}
// Track half? Up/Down
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_FLAT_TO_GENTLE_SLOPE_BANKED_TRANSITIONS)
{
vehicleEntry->var_38 = image_index;
b = vehicleEntry->var_16 * 128;
image_index += b;
}
// Unknown
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_DIAGONAL_GENTLE_SLOPE_BANKED_TRANSITIONS)
{
vehicleEntry->var_3C = image_index;
b = vehicleEntry->var_16 * 16;
image_index += b;
}
// Unknown
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_GENTLE_SLOPE_BANKED_TRANSITIONS)
{
vehicleEntry->var_40 = image_index;
b = vehicleEntry->var_16 * 16;
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_GENTLE_SLOPE_BANKED_TURNS)
{
vehicleEntry->var_44 = image_index;
b = vehicleEntry->var_16 * 128;
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_FLAT_TO_GENTLE_SLOPE_WHILE_BANKED_TRANSITIONS)
{
vehicleEntry->var_48 = image_index;
b = vehicleEntry->var_16 * 16;
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_CORKSCREWS)
{
vehicleEntry->var_4C = image_index;
b = vehicleEntry->var_16 * 80;
image_index += b;
}
// Unknown
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_RESTRAINT_ANIMATION)
{
vehicleEntry->var_1C = image_index;
b = vehicleEntry->var_16 * 12;
image_index += b;
}
if (vehicleEntry->sprite_flags & VEHICLE_SPRITE_FLAG_14)
{
// Same offset as above???
vehicleEntry->var_4C = image_index;
b = vehicleEntry->var_16 * 32;
image_index += b;
}
}
else
{
image_index += vehicleEntry->var_16 * 36;
}
// 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];
#ifdef NO_RCT2
vehicleEntry->peep_loading_positions_count = _peepLoadingPositionsCount[i];
#endif
2016-06-25 23:07:01 +02:00
}
}
}
void RideObject::Unload()
{
language_free_object_string(_legacyType.name);
language_free_object_string(_legacyType.description);
2016-06-26 22:43:30 +02:00
gfx_object_free_images(_legacyType.images_offset, GetImageTable()->GetCount());
_legacyType.name = 0;
_legacyType.description = 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;
if (_legacyType.ride_type[0] == 0xFF)
{
imageId++;
if (_legacyType.ride_type[1] == 0xFF)
{
imageId++;
}
}
gfx_draw_sprite(dpi, imageId, 0, 0, 0);
2016-06-25 23:07:01 +02:00
}
const utf8 * 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
}
const utf8 * 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;
if (_legacyType.flags & RIDE_ENTRY_FLAG_SEPARATE_RIDE)
2016-06-29 23:15:38 +02:00
{
2016-07-04 18:55:04 +02:00
flags |= ORI_RIDE_FLAG_SEPARATE;
2016-06-29 23:15:38 +02:00
}
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((rct_ride_entry *)&_legacyType);
// Determines the ride group. Will fall back to 0 if there is none found.
uint8 rideGroupIndex = 0;
const ride_group * rideGroup = get_ride_group(rideTypeIdx, (rct_ride_entry *)&_legacyType);
// If the ride group is NULL, the track type does not have ride groups.
if (rideGroup != NULL)
{
for (uint8 i = rideGroupIndex + 1; i < MAX_RIDE_GROUPS_PER_RIDE_TYPE; i++)
{
ride_group * irg = ride_group_find(rideTypeIdx, i);
if (irg != NULL)
{
if (ride_groups_are_equal(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->var_02 = stream->ReadValue<uint8>();
vehicle->var_03 = stream->ReadValue<uint8>();
vehicle->spacing = stream->ReadValue<uint32>();
vehicle->car_friction = 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->var_11 = stream->ReadValue<uint8>();
2017-07-31 12:27:04 +02:00
vehicle->flags = stream->ReadValue<uint32>();
vehicle->var_16 = stream->ReadValue<uint16>();
stream->Seek(4, STREAM_SEEK_CURRENT);
vehicle->var_1C = stream->ReadValue<uint32>();
vehicle->var_20 = stream->ReadValue<uint32>();
vehicle->var_24 = stream->ReadValue<uint32>();
vehicle->var_28 = stream->ReadValue<uint32>();
vehicle->var_2C = stream->ReadValue<uint32>();
vehicle->var_30 = stream->ReadValue<uint32>();
vehicle->var_34 = stream->ReadValue<uint32>();
vehicle->var_38 = stream->ReadValue<uint32>();
vehicle->var_3C = stream->ReadValue<uint32>();
vehicle->var_40 = stream->ReadValue<uint32>();
vehicle->var_44 = stream->ReadValue<uint32>();
vehicle->var_48 = stream->ReadValue<uint32>();
vehicle->var_4C = 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->var_58 = stream->ReadValue<uint8>();
vehicle->sound_range = stream->ReadValue<uint8>();
vehicle->var_5A = 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->special_frames = 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 (sint32 i = 0; i < RCT2_MAX_RIDE_TYPES_PER_RIDE_ENTRY; i++)
{
if (ride_type_supports_boosters(_legacyType.ride_type[i]))
{
_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;
}
// 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);
}
}