#pragma region Copyright (c) 2014-2017 OpenRCT2 Developers /***************************************************************************** * 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 #include #include "SegmentSupportHeightCall.hpp" #include static bool SortSegmentSupportCalls(SegmentSupportCall lhs, SegmentSupportCall rhs) { if (lhs.height != rhs.height) { return lhs.height < rhs.height; } if (lhs.slope != rhs.slope) { return lhs.slope < rhs.slope; } return lhs.segments < rhs.segments; } std::vector SegmentSupportHeightCall::getSegmentCalls(support_height *supports, uint8 rotation) { uint16 positionsRemaining = SEGMENTS_ALL; for (int i = 0; i < 9; i++) { if (supports[i].height == 0 && supports[i].slope == 0xFF) { positionsRemaining &= ~segment_offsets[i]; } } std::vector calls; while (positionsRemaining != 0) { SegmentSupportCall call = {0}; call.height = -1; call.slope = -1; support_height referenceSupport = { 0 }; for (int i = 0; i < 9; i++) { if (positionsRemaining & segment_offsets[i]) { referenceSupport = supports[i]; if (supports[i].height != 0) { call.height = supports[i].height; } if (supports[i].slope != 0xFF) { call.slope = supports[i].slope; } break; } } uint16 positionsMatched = 0; for (int i = 0; i < 9; i++) { if (supports[i].height == referenceSupport.height && supports[i].slope == referenceSupport.slope) { positionsMatched |= segment_offsets[i]; } } positionsRemaining &= ~positionsMatched; call.segments = paint_util_rotate_segments(positionsMatched, (4 - rotation) % 4); calls.push_back(call); } if (calls.size() > 1) { std::sort(calls.begin(), calls.end(), SortSegmentSupportCalls); } return calls; } bool SegmentSupportHeightCall::CallsMatch(std::vector tileSegmentSupportCalls[4]) { std::vector baseCallList = tileSegmentSupportCalls[0]; for (int i = 1; i < 4; i++) { if (!CallsEqual(baseCallList, tileSegmentSupportCalls[i])) { return false; } } return true; } bool SegmentSupportHeightCall::CallsEqual(std::vector lhs, std::vector rhs) { if (lhs.size() != rhs.size()) return false; for (size_t i = 0; i < lhs.size(); ++i) { if (lhs[i].segments != rhs[i].segments) return false; if (lhs[i].height != rhs[i].height) return false; if (lhs[i].slope != rhs[i].slope) return false; } return true; } bool SegmentSupportHeightCall::FindMostCommonSupportCall(std::vector calls[4], std::vector *out) { std::map, int> map; for (int i = 0; i < 4; ++i) { map[calls[i]] += 1; } if (map.size() == 1) { (*out) = calls[0]; return true; } if (map.size() == 2) { for (auto &&item : map) { if (item.second == 3) { (*out) = item.first; return true; } } return false; } return false; }