OpenRCT2/src/openrct2/object/SceneryGroupObject.cpp

203 lines
6.3 KiB
C++
Raw Normal View History

2016-06-26 01:06:55 +02:00
/*****************************************************************************
2020-07-21 15:04:34 +02:00
* Copyright (c) 2014-2020 OpenRCT2 developers
2016-06-26 01:06:55 +02:00
*
* For a complete list of all authors, please refer to contributors.md
* Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2
2016-06-26 01:06:55 +02:00
*
* OpenRCT2 is licensed under the GNU General Public License version 3.
2016-06-26 01:06:55 +02:00
*****************************************************************************/
2017-12-14 19:43:59 +01:00
#pragma warning(disable : 4706) // assignment within conditional expression
2018-06-22 23:03:20 +02:00
#include "SceneryGroupObject.h"
#include "../Context.h"
2016-06-26 01:06:55 +02:00
#include "../core/IStream.hpp"
#include "../core/Json.hpp"
2016-06-26 01:06:55 +02:00
#include "../core/Memory.hpp"
2017-12-14 19:43:59 +01:00
#include "../core/String.hpp"
#include "../drawing/Drawing.h"
#include "../localisation/Language.h"
#include "../peep/Staff.h"
2016-07-09 03:01:27 +02:00
#include "ObjectManager.h"
#include "ObjectRepository.h"
2018-06-22 23:03:20 +02:00
#include <unordered_map>
2016-06-26 01:06:55 +02:00
using namespace OpenRCT2;
2018-06-22 23:03:20 +02:00
void SceneryGroupObject::ReadLegacy(IReadObjectContext* context, IStream* stream)
2016-06-26 01:06:55 +02:00
{
stream->Seek(6, STREAM_SEEK_CURRENT);
2016-06-26 01:06:55 +02:00
stream->Seek(0x80 * 2, STREAM_SEEK_CURRENT);
_legacyType.entry_count = stream->ReadValue<uint8_t>();
2020-03-13 14:44:34 +01:00
stream->Seek(1, STREAM_SEEK_CURRENT); // pad_107;
_legacyType.priority = stream->ReadValue<uint8_t>();
2020-03-13 14:44:34 +01:00
stream->Seek(1, STREAM_SEEK_CURRENT); // pad_109;
_legacyType.entertainer_costumes = stream->ReadValue<uint32_t>();
2016-06-26 01:06:55 +02:00
GetStringTable().Read(context, stream, ObjectStringID::NAME);
_items = ReadItems(stream);
2018-02-07 21:37:20 +01:00
GetImageTable().Read(context, stream);
2016-06-26 01:06:55 +02:00
}
void SceneryGroupObject::Load()
{
GetStringTable().Sort();
2016-06-26 01:06:55 +02:00
_legacyType.name = language_allocate_object_string(GetName());
_legacyType.image = gfx_object_allocate_images(GetImageTable().GetImages(), GetImageTable().GetCount());
2016-06-26 01:06:55 +02:00
_legacyType.entry_count = 0;
}
void SceneryGroupObject::Unload()
{
language_free_object_string(_legacyType.name);
gfx_object_free_images(_legacyType.image, GetImageTable().GetCount());
_legacyType.name = 0;
_legacyType.image = 0;
2016-06-26 01:06:55 +02:00
}
2018-06-22 23:03:20 +02:00
void SceneryGroupObject::DrawPreview(rct_drawpixelinfo* dpi, int32_t width, int32_t height) const
2016-07-02 13:29:13 +02:00
{
auto screenCoords = ScreenCoordsXY{ width / 2, height / 2 };
2016-07-02 13:29:13 +02:00
uint32_t imageId = _legacyType.image + 0x20600001;
gfx_draw_sprite(dpi, imageId, screenCoords - ScreenCoordsXY{ 15, 14 }, 0);
2016-07-02 13:29:13 +02:00
}
2021-09-27 22:54:41 +02:00
static std::optional<uint8_t> GetSceneryType(const ObjectType type)
{
switch (type)
{
case ObjectType::SmallScenery:
return SCENERY_TYPE_SMALL;
case ObjectType::LargeScenery:
return SCENERY_TYPE_LARGE;
case ObjectType::Walls:
return SCENERY_TYPE_WALL;
case ObjectType::Banners:
return SCENERY_TYPE_BANNER;
case ObjectType::PathBits:
return SCENERY_TYPE_PATH_ITEM;
default:
return std::nullopt;
}
}
2016-07-09 03:01:27 +02:00
void SceneryGroupObject::UpdateEntryIndexes()
{
auto context = GetContext();
auto& objectRepository = context->GetObjectRepository();
auto& objectManager = context->GetObjectManager();
2016-07-09 03:01:27 +02:00
_legacyType.entry_count = 0;
2018-06-22 23:03:20 +02:00
for (const auto& objectEntry : _items)
2016-07-09 03:01:27 +02:00
{
auto ori = objectRepository.FindObject(objectEntry);
2018-06-22 23:03:20 +02:00
if (ori == nullptr)
continue;
if (ori->LoadedObject == nullptr)
continue;
2016-07-09 03:01:27 +02:00
2021-09-03 12:43:06 +02:00
auto entryIndex = objectManager.GetLoadedObjectEntryIndex(ori->LoadedObject.get());
Guard::Assert(entryIndex != OBJECT_ENTRY_INDEX_NULL, GUARD_LINE);
2016-07-09 03:01:27 +02:00
2021-09-27 22:54:41 +02:00
auto sceneryType = GetSceneryType(ori->Type);
2021-09-13 18:47:13 +02:00
if (sceneryType.has_value())
2017-12-14 19:43:59 +01:00
{
2021-09-13 18:47:13 +02:00
_legacyType.scenery_entries[_legacyType.entry_count] = { sceneryType.value(), entryIndex };
2017-12-14 19:43:59 +01:00
_legacyType.entry_count++;
2016-07-09 03:01:27 +02:00
}
}
}
2018-06-22 23:03:20 +02:00
void SceneryGroupObject::SetRepositoryItem(ObjectRepositoryItem* item) const
{
2018-05-14 14:05:59 +02:00
item->SceneryGroupInfo.Entries = _items;
}
std::vector<ObjectEntryDescriptor> SceneryGroupObject::ReadItems(IStream* stream)
2016-06-26 01:06:55 +02:00
{
auto items = std::vector<ObjectEntryDescriptor>();
while (stream->ReadValue<uint8_t>() != 0xFF)
2016-06-26 01:06:55 +02:00
{
stream->Seek(-1, STREAM_SEEK_CURRENT);
2017-12-14 19:43:59 +01:00
auto entry = stream->ReadValue<rct_object_entry>();
items.emplace_back(entry);
2016-06-26 01:06:55 +02:00
}
return items;
2016-06-26 01:06:55 +02:00
}
2017-12-14 19:43:59 +01:00
void SceneryGroupObject::ReadJson(IReadObjectContext* context, json_t& root)
2017-12-14 19:43:59 +01:00
{
Guard::Assert(root.is_object(), "SceneryGroupObject::ReadJson expects parameter root to be object");
2017-12-14 19:43:59 +01:00
auto properties = root["properties"];
2017-12-14 19:43:59 +01:00
if (properties.is_object())
2017-12-14 19:43:59 +01:00
{
_legacyType.priority = Json::GetNumber<uint8_t>(properties["priority"]);
_legacyType.entertainer_costumes = ReadJsonEntertainerCostumes(properties["entertainerCostumes"]);
_items = ReadJsonEntries(properties["entries"]);
2017-12-14 19:43:59 +01:00
}
PopulateTablesFromJson(context, root);
2017-12-14 19:43:59 +01:00
}
uint32_t SceneryGroupObject::ReadJsonEntertainerCostumes(json_t& jCostumes)
2017-12-14 19:43:59 +01:00
{
uint32_t costumes = 0;
for (auto& jCostume : jCostumes)
2017-12-14 19:43:59 +01:00
{
auto entertainer = ParseEntertainerCostume(Json::GetString(jCostume));
auto peepSprite = EntertainerCostumeToSprite(entertainer);
costumes |= 1 << (static_cast<uint8_t>(peepSprite));
2017-12-14 19:43:59 +01:00
}
return costumes;
}
EntertainerCostume SceneryGroupObject::ParseEntertainerCostume(const std::string& s)
2017-12-14 19:43:59 +01:00
{
2018-06-22 23:03:20 +02:00
if (s == "panda")
return EntertainerCostume::Panda;
2018-06-22 23:03:20 +02:00
if (s == "tiger")
return EntertainerCostume::Tiger;
2018-06-22 23:03:20 +02:00
if (s == "elephant")
return EntertainerCostume::Elephant;
2018-06-22 23:03:20 +02:00
if (s == "roman")
return EntertainerCostume::Roman;
2018-06-22 23:03:20 +02:00
if (s == "gorilla")
return EntertainerCostume::Gorilla;
2018-06-22 23:03:20 +02:00
if (s == "snowman")
return EntertainerCostume::Snowman;
2018-06-22 23:03:20 +02:00
if (s == "knight")
return EntertainerCostume::Knight;
2018-06-22 23:03:20 +02:00
if (s == "astronaut")
return EntertainerCostume::Astronaut;
2018-06-22 23:03:20 +02:00
if (s == "bandit")
return EntertainerCostume::Bandit;
2018-06-22 23:03:20 +02:00
if (s == "sheriff")
return EntertainerCostume::Sheriff;
2018-06-22 23:03:20 +02:00
if (s == "pirate")
return EntertainerCostume::Pirate;
return EntertainerCostume::Panda;
2017-12-14 19:43:59 +01:00
}
std::vector<ObjectEntryDescriptor> SceneryGroupObject::ReadJsonEntries(json_t& jEntries)
2017-12-14 19:43:59 +01:00
{
std::vector<ObjectEntryDescriptor> entries;
for (const auto& jEntry : jEntries)
2017-12-14 19:43:59 +01:00
{
entries.emplace_back(Json::GetString(jEntry));
2017-12-14 19:43:59 +01:00
}
return entries;
}
uint16_t SceneryGroupObject::GetNumIncludedObjects() const
{
return static_cast<uint16_t>(_items.size());
}