/***************************************************************************** * Copyright (c) 2014 Ted John * OpenRCT2, an open source clone of Roller Coaster Tycoon 2. * * This file is part of 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. * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * You should have received a copy of the GNU General Public License * along with this program. If not, see . *****************************************************************************/ #include "addresses.h" #include "finance.h" #include "map.h" #include "park.h" #include "peep.h" #include "ride.h" #include "scenario.h" #include "sprite.h" #include "strings.h" #include "window.h" int park_is_open() { return (RCT2_GLOBAL(RCT2_ADDRESS_PARK_FLAGS, uint32) & PARK_FLAGS_PARK_OPEN) != 0; } /** * * rct2: 0x00667132 */ void park_init() { int i; RCT2_GLOBAL(0x013CA740, uint8) = 0; RCT2_GLOBAL(0x013573D4, uint16) = 777; RCT2_GLOBAL(RCT2_ADDRESS_HANDYMAN_COLOUR, uint8) = 28; RCT2_GLOBAL(RCT2_ADDRESS_MECHANIC_COLOUR, uint8) = 28; RCT2_GLOBAL(RCT2_ADDRESS_SECURITY_COLOUR, uint8) = 28; RCT2_GLOBAL(RCT2_ADDRESS_GUESTS_IN_PARK, uint16) = 0; RCT2_GLOBAL(0x01357BC8, uint16) = 0; RCT2_GLOBAL(0x01357846, uint16) = 0; RCT2_GLOBAL(0x013573FE, uint16) = 0; RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_PARK_RATING, uint16) = 0; RCT2_GLOBAL(0x013580EC, uint16) = 0; RCT2_GLOBAL(0x013580EE, uint16) = 0; RCT2_GLOBAL(0x01357CF4, sint32) = -1; for (i = 0; i < 20; i++) RCT2_ADDRESS(0x01358102, uint8)[i] = 0; RCT2_GLOBAL(0x01358844, uint32) = 0xFFFFFFFF; RCT2_GLOBAL(0x01358849, uint32) = 0xFFFFFFFE; RCT2_GLOBAL(0x0135884E, uint32) = 0xFFFFFFFD; finance_init(); for (i = 0; i < 2; i++) RCT2_ADDRESS(0x01357404, uint32)[i] = 0; for (i = 0; i < 56; i++) RCT2_ADDRESS(0x01357BD0, sint32)[i] = -1; RCT2_GLOBAL(RCT2_ADDRESS_PARK_ENTRANCE_FEE, uint16) = CURRENCY(10, 00); RCT2_GLOBAL(0x013573F2, sint16) = -1; RCT2_GLOBAL(0x013573F8, sint16) = -1; RCT2_GLOBAL(0x01357CF2, uint16) = 127; RCT2_GLOBAL(0x013573FF, uint8) = 2; RCT2_GLOBAL(0x013580F4, uint16) = 500; RCT2_GLOBAL(0x013580E9, uint8) = 128; RCT2_GLOBAL(0x013580F6, uint8) = 200; RCT2_GLOBAL(0x013580F7, uint8) = 200; RCT2_GLOBAL(RCT2_ADDRESS_OBJECTIVE_TYPE, uint8) = 1; RCT2_GLOBAL(RCT2_ADDRESS_OBJECTIVE_YEAR, uint8) = 4; RCT2_GLOBAL(RCT2_ADDRESS_OBJECTIVE_NUM_GUESTS, uint16) = 1000; RCT2_GLOBAL(0x01358770, uint16) = 900; RCT2_GLOBAL(0x01358772, uint16) = 400; RCT2_GLOBAL(0x01358774, uint16) = 0; RCT2_GLOBAL(RCT2_ADDRESS_PARK_FLAGS, uint32) = PARK_FLAGS_11 | PARK_FLAGS_SHOW_REAL_GUEST_NAMES; RCT2_CALLPROC_EBPSAFE(0x0066729F); // reset history / finance / awards rct_s6_info *info = 0x0141F570; info->name[0] = '\0'; format_string(info->details, STR_NO_DETAILS_YET, NULL); } /** * * rct2: 0x0066A348 */ int park_calculate_size() { int tiles, x, y; rct_map_element *mapElement; tiles = 0; for (y = 0; y < 256; y++) { for (x = 0; x < 256; x++) { mapElement = RCT2_ADDRESS(RCT2_ADDRESS_TILE_MAP_ELEMENT_POINTERS, rct_map_element*)[y * 256 + x]; while (mapElement->type & MAP_ELEMENT_TYPE_MASK) { mapElement++; } if (mapElement->properties.surface.ownership & 0x30) tiles++; } } if (tiles != RCT2_GLOBAL(RCT2_ADDRESS_PARK_SIZE, sint16)) { RCT2_GLOBAL(RCT2_ADDRESS_PARK_SIZE, sint16) = tiles; window_invalidate_by_id(WC_PARK_INFORMATION, 0); } return tiles; } /** * * rct2: 0x00669EAA */ int calculate_park_rating() { int result; result = 1150; if (RCT2_GLOBAL(RCT2_ADDRESS_PARK_FLAGS, uint32) & 0x4000) result = 1050; // Guests { rct_peep* peep; uint16 sprite_idx; int num_happy_peeps; short _bp; // -150 to +3 based on a range of guests from 0 to 2000 result -= 150 - (min(2000, RCT2_GLOBAL(RCT2_ADDRESS_GUESTS_IN_PARK, uint16)) / 13); // Guests, happiness, ? num_happy_peeps = 0; _bp = 0; for (sprite_idx = RCT2_GLOBAL(RCT2_ADDRESS_SPRITES_START_PEEP, uint16); sprite_idx != SPRITE_INDEX_NULL; sprite_idx = peep->next) { peep = &(RCT2_ADDRESS(RCT2_ADDRESS_SPRITE_LIST, rct_sprite)[sprite_idx].peep); if (peep->type != PEEP_TYPE_GUEST) continue; if (peep->var_2A != 0) continue; if (peep->happiness > 128) num_happy_peeps++; if (!(peep->flags & PEEP_FLAGS_LEAVING_PARK)) continue; if (peep->var_C6 <= 89) _bp++; } // Peep happiness -500 to +0 result -= 500; if (RCT2_GLOBAL(RCT2_ADDRESS_GUESTS_IN_PARK, uint16) > 0) result += 2 * min(250, (num_happy_peeps * 300) / RCT2_GLOBAL(RCT2_ADDRESS_GUESTS_IN_PARK, uint16)); // ? _bp -= 25; if (_bp >= 0) result -= _bp * 7; } // Rides { int i; short _ax, total_ride_intensity = 0, total_ride_excitement = 0, average_intensity, average_excitement; int num_rides, num_exciting_rides = 0; rct_ride* ride; // _ax = 0; num_rides = 0; for (i = 0; i < 255; i++) { ride = &(RCT2_ADDRESS(RCT2_ADDRESS_RIDE_LIST, rct_ride)[i]); if (ride->type == RIDE_TYPE_NULL) continue; _ax += 100 - ride->var_199; if (ride->excitement != -1){ total_ride_excitement += ride->excitement / 8; total_ride_intensity += ride->intensity / 8; num_exciting_rides++; } num_rides++; } result -= 200; if (num_rides > 0) result += (_ax / num_rides) * 2; result -= 100; if (num_exciting_rides>0){ average_excitement = total_ride_excitement / num_exciting_rides; average_intensity = total_ride_intensity / num_exciting_rides; average_excitement -= 46; if (average_excitement < 0){ average_excitement = -average_excitement; } average_intensity -= 65; if (average_intensity < 0){ average_intensity = -average_intensity; } average_excitement = min(average_excitement / 2, 50); average_intensity = min(average_intensity / 2, 50); result += 100 - average_excitement - average_intensity; } total_ride_excitement = min(1000, total_ride_excitement); total_ride_intensity = min(1000, total_ride_intensity); result -= 200 - ((total_ride_excitement + total_ride_intensity) / 10); } // Litter { rct_litter* litter; uint16 sprite_idx; short num_litter; num_litter = 0; for (sprite_idx = RCT2_GLOBAL(RCT2_ADDRESS_SPRITES_START_LITTER, uint16); sprite_idx != SPRITE_INDEX_NULL; sprite_idx = litter->next) { litter = &(RCT2_ADDRESS(RCT2_ADDRESS_SPRITE_LIST, rct_sprite)[sprite_idx].litter); // Guessing this eliminates recently dropped litter if (litter->var_24 - RCT2_GLOBAL(0x00F663AC, uint32) >= 7680) num_litter++; } result -= 600 - (4 * (150 - min(150, num_litter))); } result -= RCT2_GLOBAL(0x0135882E, sint16); result = clamp(0, result, 999); return result; } /** * * rct2: 0x0066A3F6 */ int calculate_park_value() { int result, value, i; rct_ride* ride; result = 0; for (i = 0; i < 255; i++) { ride = &(RCT2_ADDRESS(RCT2_ADDRESS_RIDE_LIST, rct_ride)[i]); if (ride->type == RIDE_TYPE_NULL) continue; if (ride->reliability == 0xFFFF) continue; value = 0; value += ride->var_124 + ride->var_126 + ride->var_128 + ride->var_12A; value += ride->var_12C + ride->var_12E + ride->age + ride->running_cost; value += ride->var_134 + ride->var_136; value += *((uint8*)(0x0097D21E + (ride->type * 8))) * 4; value *= ride->reliability * 10; result += value; } result += RCT2_GLOBAL(RCT2_ADDRESS_GUESTS_IN_PARK, uint16) * 70; return result; } /** * * rct2: 0x0066A498 */ int calculate_company_value() { int result; result = DECRYPT_MONEY(RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_MONEY_ENCRYPTED, sint32)); result += RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_PARK_VALUE, sint32); result -= RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_LOAN, sint32); return result; } /** * * rct2: 0x00667104 */ void reset_park_entrances() { RCT2_GLOBAL(0x013573D4, uint16) = 0; for (short i = 0; i < 4; i++) { RCT2_ADDRESS(RCT2_ADDRESS_PARK_ENTRANCE_X, uint16)[i] = 0x8000; } RCT2_GLOBAL(0x013573F2, uint16) = 0xFFFF; RCT2_GLOBAL(0x013573F8, uint16) = 0xFFFF; }