Also fix disconnected exits

This commit is contained in:
Michael Steenbeek 2017-10-25 15:05:10 +02:00
parent e53d76f7e1
commit e3b0409fbc
5 changed files with 89 additions and 30 deletions

View File

@ -1125,8 +1125,8 @@ void game_fix_save_vars()
// Fix gParkEntrance locations for which the map_element no longer exists // Fix gParkEntrance locations for which the map_element no longer exists
fix_park_entrance_locations(); fix_park_entrance_locations();
// Fix ride entrances that were moved without updating ride->entrances[] // Fix ride entrances and exits that were moved without updating ride->entrances[] / ride->exits[]
fix_ride_entrance_locations(); fix_ride_entrance_and_exit_locations();
} }
void handle_park_load_failure_with_title_opt(const ParkLoadResult * result, const utf8 * path, bool loadTitleFirst) void handle_park_load_failure_with_title_opt(const ParkLoadResult * result, const utf8 * path, bool loadTitleFirst)

View File

@ -8227,7 +8227,7 @@ LocationXY16 ride_get_rotated_coords(sint16 x, sint16 y, sint16 z)
return rotatedCoords; return rotatedCoords;
} }
void fix_ride_entrance_locations() void fix_ride_entrance_and_exit_locations()
{ {
sint32 rideIndex; sint32 rideIndex;
Ride * ride; Ride * ride;
@ -8236,25 +8236,43 @@ void fix_ride_entrance_locations()
{ {
for (sint32 stationIndex = 0; stationIndex < MAX_STATIONS; stationIndex++) for (sint32 stationIndex = 0; stationIndex < MAX_STATIONS; stationIndex++)
{ {
LocationXY8 entranceLoc = ride->entrances[stationIndex]; LocationXY8 entranceLoc = ride->entrances[stationIndex];
uint8 entranceHeight = ride->station_heights[stationIndex]; LocationXY8 exitLoc = ride->exits[stationIndex];
uint8 entranceExitHeight = ride->station_heights[stationIndex];
bool fixEntrance = false;
bool fixExit = false;
rct_map_element * mapElement;
// Skip if the station has no entrance // Skip if the station has no entrance
if (entranceLoc.xy == RCT_XY8_UNDEFINED) if (entranceLoc.xy != RCT_XY8_UNDEFINED)
{
mapElement = map_get_ride_entrance_element_at(entranceLoc.x * 32, entranceLoc.y * 32, entranceExitHeight, false);
if (mapElement == NULL || mapElement->properties.entrance.ride_index != rideIndex || map_element_get_station(mapElement) != stationIndex)
{
fixEntrance = true;
}
}
if (exitLoc.xy != RCT_XY8_UNDEFINED)
{
mapElement = map_get_ride_exit_element_at(exitLoc.x * 32, exitLoc.y * 32, entranceExitHeight, false);
if (mapElement == NULL || mapElement->properties.entrance.ride_index != rideIndex || map_element_get_station(mapElement) != stationIndex)
{
fixExit = true;
}
}
if (!fixEntrance && !fixExit)
{ {
continue; continue;
} }
rct_map_element * mapElement = map_get_ride_entrance_element_at(entranceLoc.x * 32, entranceLoc.y * 32, entranceHeight, false); // At this point, we know we have a disconnected entrance or exit.
if (mapElement != NULL && mapElement->properties.entrance.ride_index == rideIndex && map_element_get_station(mapElement) == stationIndex)
{
continue;
}
// At this point, we know we have a disconnected entrance.
// Search the map to find it. Skip the outer ring of invisible tiles. // Search the map to find it. Skip the outer ring of invisible tiles.
bool alreadyFoundOne = false; bool alreadyFoundEntrance = false;
bool alreadyFoundExit = false;
for (uint8 x = 1; x < MAXIMUM_MAP_SIZE_TECHNICAL - 1; x++) for (uint8 x = 1; x < MAXIMUM_MAP_SIZE_TECHNICAL - 1; x++)
{ {
for (uint8 y = 1; y < MAXIMUM_MAP_SIZE_TECHNICAL - 1; y++) for (uint8 y = 1; y < MAXIMUM_MAP_SIZE_TECHNICAL - 1; y++)
@ -8269,10 +8287,6 @@ void fix_ride_entrance_locations()
{ {
continue; continue;
} }
if (mapElement->properties.entrance.type != ENTRANCE_TYPE_RIDE_ENTRANCE)
{
continue;
}
if (mapElement->properties.entrance.ride_index != rideIndex) if (mapElement->properties.entrance.ride_index != rideIndex)
{ {
continue; continue;
@ -8281,21 +8295,41 @@ void fix_ride_entrance_locations()
{ {
continue; continue;
} }
// There are some cases (like Belmont Shores), where there is both a sunk and a disconnected entrance.
// Not sure why, but in this case, pick the highest one. if (fixEntrance && mapElement->properties.entrance.type == ENTRANCE_TYPE_RIDE_ENTRANCE)
if (alreadyFoundOne && ride->station_heights[stationIndex] > mapElement->base_height)
{ {
continue; // There are some cases (like Belmont Shores), where there is both a sunk and a disconnected entrance.
// Not sure why, but in this case, pick the highest one.
if (alreadyFoundEntrance && ride->station_heights[stationIndex] > mapElement->base_height)
{
continue;
}
// Found our entrance
ride->entrances[stationIndex].x = x;
ride->entrances[stationIndex].y = y;
ride->station_heights[stationIndex] = mapElement->base_height;
alreadyFoundEntrance = true;
log_info("Fixed disconnected entrance of ride %d, station %d to x = %d, y = %d and z = %d.", rideIndex, stationIndex, x, y, mapElement->base_height);
} }
else if (fixExit && mapElement->properties.entrance.type == ENTRANCE_TYPE_RIDE_EXIT)
{
if (alreadyFoundExit && ride->station_heights[stationIndex] > mapElement->base_height)
{
continue;
}
// Found our entrance // Found our exit
ride->entrances[stationIndex].x = x; ride->exits[stationIndex].x = x;
ride->entrances[stationIndex].y = y; ride->exits[stationIndex].y = y;
ride->station_heights[stationIndex] = mapElement->base_height; ride->station_heights[stationIndex] = mapElement->base_height;
alreadyFoundOne = true; alreadyFoundExit = true;
log_info("Fixed disconnected entrance of ride %d, station %d to x = %d, y = %d and z = %d.", rideIndex, stationIndex, x, y, mapElement->base_height); log_info("Fixed disconnected exit of ride %d, station %d to x = %d, y = %d and z = %d.", rideIndex, stationIndex, x, y, mapElement->base_height);
}
} }
while (!map_element_is_last_for_tile(mapElement++)); while (!map_element_is_last_for_tile(mapElement++));
} }

View File

@ -1214,7 +1214,7 @@ void ride_stop_peeps_queuing(sint32 rideIndex);
LocationXY16 ride_get_rotated_coords(sint16 x, sint16 y, sint16 z); LocationXY16 ride_get_rotated_coords(sint16 x, sint16 y, sint16 z);
void fix_ride_entrance_locations(); void fix_ride_entrance_and_exit_locations();
#ifdef __cplusplus #ifdef __cplusplus
} }

View File

@ -3856,6 +3856,30 @@ rct_map_element * map_get_ride_entrance_element_at(sint32 x, sint32 y, sint32 z,
return NULL; return NULL;
} }
rct_map_element * map_get_ride_exit_element_at(sint32 x, sint32 y, sint32 z, bool ghost)
{
rct_map_element * mapElement = map_get_first_element_at(x >> 5, y >> 5);
do
{
if (map_element_get_type(mapElement) != MAP_ELEMENT_TYPE_ENTRANCE)
continue;
if (mapElement->base_height != z)
continue;
if (mapElement->properties.entrance.type != ENTRANCE_TYPE_RIDE_EXIT)
continue;
if ((ghost == false) && (mapElement->flags & MAP_ELEMENT_FLAG_GHOST))
continue;
return mapElement;
}
while (!map_element_is_last_for_tile(mapElement++));
return NULL;
}
rct_map_element *map_get_small_scenery_element_at(sint32 x, sint32 y, sint32 z, sint32 type, uint8 quadrant) rct_map_element *map_get_small_scenery_element_at(sint32 x, sint32 y, sint32 z, sint32 type, uint8 quadrant)
{ {
rct_map_element *mapElement = map_get_first_element_at(x >> 5, y >> 5); rct_map_element *mapElement = map_get_first_element_at(x >> 5, y >> 5);

View File

@ -416,6 +416,7 @@ rct_map_element *map_get_wall_element_at(sint32 x, sint32 y, sint32 z, sint32 di
rct_map_element *map_get_small_scenery_element_at(sint32 x, sint32 y, sint32 z, sint32 type, uint8 quadrant); rct_map_element *map_get_small_scenery_element_at(sint32 x, sint32 y, sint32 z, sint32 type, uint8 quadrant);
rct_map_element *map_get_park_entrance_element_at(sint32 x, sint32 y, sint32 z, bool ghost); rct_map_element *map_get_park_entrance_element_at(sint32 x, sint32 y, sint32 z, bool ghost);
rct_map_element * map_get_ride_entrance_element_at(sint32 x, sint32 y, sint32 z, bool ghost); rct_map_element * map_get_ride_entrance_element_at(sint32 x, sint32 y, sint32 z, bool ghost);
rct_map_element * map_get_ride_exit_element_at(sint32 x, sint32 y, sint32 z, bool ghost);
sint32 map_element_height(sint32 x, sint32 y); sint32 map_element_height(sint32 x, sint32 y);
void sub_68B089(); void sub_68B089();
sint32 map_coord_is_connected(sint32 x, sint32 y, sint32 z, uint8 faceDirection); sint32 map_coord_is_connected(sint32 x, sint32 y, sint32 z, uint8 faceDirection);