mirror of https://github.com/OpenRCT2/OpenRCT2.git
154 lines
4.4 KiB
C++
154 lines
4.4 KiB
C++
#pragma region Copyright (c) 2014-2016 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 <map>
|
|
#include <algorithm>
|
|
|
|
#include "SegmentSupportHeightCall.hpp"
|
|
|
|
extern "C" {
|
|
#include "../../src/paint/map_element/map_element.h"
|
|
}
|
|
|
|
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<SegmentSupportCall> 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<SegmentSupportCall> 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<SegmentSupportCall> tileSegmentSupportCalls[4])
|
|
{
|
|
std::vector<SegmentSupportCall> baseCallList = tileSegmentSupportCalls[0];
|
|
for (int i = 1; i < 4; i++) {
|
|
if (!CallsEqual(baseCallList, tileSegmentSupportCalls[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SegmentSupportHeightCall::CallsEqual(std::vector<SegmentSupportCall> lhs, std::vector<SegmentSupportCall> 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;
|
|
}
|
|
|
|
std::vector<SegmentSupportCall>* SegmentSupportHeightCall::FindMostCommonSupportCall(std::vector<SegmentSupportCall> *calls) {
|
|
std::map<std::vector<SegmentSupportCall>, int> map;
|
|
|
|
for (int i = 0; i < 4; ++i) {
|
|
if (map.count(calls[i]) == 0) {
|
|
map[calls[i]] = 1;
|
|
} else {
|
|
map[calls[i]] += 1;
|
|
}
|
|
}
|
|
|
|
if (map.size() == 1) {
|
|
return &calls[0];
|
|
}
|
|
|
|
if (map.size() == 2) {
|
|
for (auto &&item : map) {
|
|
if (item.second == 3) {
|
|
return (std::vector<SegmentSupportCall> *)&item.first;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
if (map.size() == 3) {
|
|
for (auto &&item : map) {
|
|
if (item.second == 2) {
|
|
return (std::vector<SegmentSupportCall> *)&item.first;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|