#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 "../Game.h" #include "../interface/Window.h" #include "../localisation/Date.h" #include "../localisation/Localisation.h" #include "../peep/Peep.h" #include "../peep/Staff.h" #include "../ride/Ride.h" #include "../util/Util.h" #include "../world/Park.h" #include "../world/sprite.h" #include "Finance.h" /** * Monthly staff wages * * rct2: 0x00992A00 */ const money32 wage_table[STAFF_TYPE_COUNT] = { MONEY(50, 00), // Handyman MONEY(80, 00), // Mechanic MONEY(60, 00), // Security guard MONEY(55, 00) // Entertainer }; // Monthly research funding costs const money32 research_cost_table[RESEARCH_FUNDING_COUNT] = { MONEY(0, 00), // No funding MONEY(100, 00), // Minimum funding MONEY(200, 00), // Normal funding MONEY(400, 00) // Maximum funding }; static const sint32 dword_988E60[RCT_EXPENDITURE_TYPE_COUNT] = {1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 0}; money32 gInitialCash; money32 gCash; money32 gBankLoan; uint8 gBankLoanInterestRate; money32 gMaxBankLoan; money32 gCurrentExpenditure; money32 gCurrentProfit; money32 gHistoricalProfit; money32 gWeeklyProfitAverageDividend; uint16 gWeeklyProfitAverageDivisor; money32 gCashHistory[FINANCE_GRAPH_SIZE]; money32 gWeeklyProfitHistory[FINANCE_GRAPH_SIZE]; money32 gParkValueHistory[FINANCE_GRAPH_SIZE]; money32 gExpenditureTable[EXPENDITURE_TABLE_MONTH_COUNT][RCT_EXPENDITURE_TYPE_COUNT]; uint8 gCommandExpenditureType; /** * Pay an amount of money. * rct2: 0x069C674 * @param amount (eax) * @param type passed via global var 0x0141F56C (RCT2_ADDRESS_NEXT_EXPENDITURE_TYPE), our type is that var/4. */ void finance_payment(money32 amount, rct_expenditure_type type) { //overflow check gCash = add_clamp_money32(gCash, -amount); gExpenditureTable[0][type] -= amount; if (dword_988E60[type] & 1) { // Cumulative amount of money spent this day gCurrentExpenditure -= amount; } gToolbarDirtyFlags |= BTM_TB_DIRTY_FLAG_MONEY; window_invalidate_by_class(WC_FINANCES); } /** * Pays the wages of all active staff members in the park. * rct2: 0x006C18A9 */ void finance_pay_wages() { rct_peep * peep; uint16 spriteIndex; if (gParkFlags & PARK_FLAGS_NO_MONEY) { return; } FOR_ALL_STAFF(spriteIndex, peep) { finance_payment(wage_table[peep->staff_type] / 4, RCT_EXPENDITURE_TYPE_WAGES); } } /** * Pays the current research level's cost. * rct2: 0x00684DA5 **/ void finance_pay_research() { uint8 level; if (gParkFlags & PARK_FLAGS_NO_MONEY) { return; } level = gResearchFundingLevel; finance_payment(research_cost_table[level] / 4, RCT_EXPENDITURE_TYPE_RESEARCH); } /** * Pay interest on current loans. * rct2: 0x0069E092 */ void finance_pay_interest() { // This variable uses the 64-bit type as the computation below can involve multiplying very large numbers // that will overflow money32 if the loan is greater than (1 << 31) / (5 * current_interest_rate) money64 current_loan = gBankLoan; uint8 current_interest_rate = gBankLoanInterestRate; money32 interest_to_pay; if (gParkFlags & PARK_FLAGS_NO_MONEY) { return; } interest_to_pay = (current_loan * 5 * current_interest_rate) >> 14; finance_payment(interest_to_pay, RCT_EXPENDITURE_TYPE_INTEREST); } /** * * rct2: 0x006AC885 */ void finance_pay_ride_upkeep() { sint32 i; Ride * ride; FOR_ALL_RIDES(i, ride) { if (!(ride->lifecycle_flags & RIDE_LIFECYCLE_EVER_BEEN_OPENED)) { ride->build_date = gDateMonthsElapsed; ride->reliability = RIDE_INITIAL_RELIABILITY; } if (ride->status != RIDE_STATUS_CLOSED && !(gParkFlags & PARK_FLAGS_NO_MONEY)) { sint16 upkeep = ride->upkeep_cost; if (upkeep != -1) { ride->total_profit -= upkeep; ride->window_invalidate_flags |= RIDE_INVALIDATE_RIDE_INCOME; finance_payment(upkeep, RCT_EXPENDITURE_TYPE_RIDE_RUNNING_COSTS); } } if (ride->last_crash_type != RIDE_CRASH_TYPE_NONE) { ride->last_crash_type--; } } } void finance_reset_history() { for (sint32 i = 0; i < FINANCE_GRAPH_SIZE; i++) { gCashHistory[i] = MONEY32_UNDEFINED; gWeeklyProfitHistory[i] = MONEY32_UNDEFINED; gParkValueHistory[i] = MONEY32_UNDEFINED; } } /** * * rct2: 0x0069DEFB */ void finance_init() { // It only initialises the first month for (uint32 i = 0; i < RCT_EXPENDITURE_TYPE_COUNT; i++) { gExpenditureTable[0][i] = 0; } gCurrentExpenditure = 0; gCurrentProfit = 0; gWeeklyProfitAverageDividend = 0; gWeeklyProfitAverageDivisor = 0; gInitialCash = MONEY(10000, 00); // Cheat detection gCash = MONEY(10000, 00); gBankLoan = MONEY(10000, 00); gMaxBankLoan = MONEY(20000, 00); gHistoricalProfit = 0; gBankLoanInterestRate = 10; gParkValue = 0; gCompanyValue = 0; gScenarioCompletedCompanyValue = MONEY32_UNDEFINED; gTotalAdmissions = 0; gTotalIncomeFromAdmissions = 0; safe_strcpy(gScenarioCompletedBy, "?", sizeof(gScenarioCompletedBy)); } /** * * rct2: 0x0069E79A */ void finance_update_daily_profit() { gCurrentProfit = 7 * gCurrentExpenditure; gCurrentExpenditure = 0; // Reset daily expenditure money32 current_profit = 0; if (!(gParkFlags & PARK_FLAGS_NO_MONEY)) { // Staff costs uint16 sprite_index; rct_peep * peep; FOR_ALL_STAFF(sprite_index, peep) { current_profit -= wage_table[peep->staff_type]; } // Research costs uint8 level = gResearchFundingLevel; current_profit -= research_cost_table[level]; // Loan costs money32 current_loan = gBankLoan; current_profit -= current_loan / 600; // Ride costs Ride * ride; sint32 i; FOR_ALL_RIDES(i, ride) { if (ride->status != RIDE_STATUS_CLOSED && ride->upkeep_cost != MONEY16_UNDEFINED) { current_profit -= 2 * ride->upkeep_cost; } } } // This is not equivalent to / 4 due to rounding of negative numbers current_profit = current_profit >> 2; gCurrentProfit += current_profit; // These are related to weekly profit graph gWeeklyProfitAverageDividend += gCurrentProfit; gWeeklyProfitAverageDivisor += 1; window_invalidate_by_class(WC_FINANCES); } void finance_set_loan(money32 loan) { game_do_command(0, GAME_COMMAND_FLAG_APPLY, 0, loan, GAME_COMMAND_SET_CURRENT_LOAN, 0, 0); } money32 finance_get_initial_cash() { return gInitialCash; } money32 finance_get_current_loan() { return gBankLoan; } money32 finance_get_maximum_loan() { return gMaxBankLoan; } money32 finance_get_current_cash() { return gCash; } /** * * rct2: 0x0069DFB3 */ void game_command_set_current_loan(sint32 * eax, sint32 * ebx, sint32 * ecx, sint32 * edx, sint32 * esi, sint32 * edi, sint32 * ebp) { money32 loanDifference, currentLoan; money32 newLoan = *edx; currentLoan = gBankLoan; loanDifference = currentLoan - newLoan; gCommandExpenditureType = RCT_EXPENDITURE_TYPE_INTEREST; if (newLoan > currentLoan) { if (newLoan > gMaxBankLoan) { gGameCommandErrorText = STR_BANK_REFUSES_TO_INCREASE_LOAN; *ebx = MONEY32_UNDEFINED; return; } } else { if (loanDifference > gCash) { gGameCommandErrorText = STR_NOT_ENOUGH_CASH_AVAILABLE; *ebx = MONEY32_UNDEFINED; return; } } if (*ebx & GAME_COMMAND_FLAG_APPLY) { gCash -= loanDifference; gBankLoan = newLoan; gInitialCash = gCash; window_invalidate_by_class(WC_FINANCES); gToolbarDirtyFlags |= BTM_TB_DIRTY_FLAG_MONEY; } *ebx = 0; } /** * Shift the expenditure table history one month to the left * If the table is full, accumulate the sum of the oldest month first * rct2: 0x0069DEAD */ void finance_shift_expenditure_table() { // If EXPENDITURE_TABLE_MONTH_COUNT months have passed then is full, sum the oldest month if (gDateMonthsElapsed >= EXPENDITURE_TABLE_MONTH_COUNT) { money32 sum = 0; for (uint32 i = 0; i < RCT_EXPENDITURE_TYPE_COUNT; i++) { sum += gExpenditureTable[EXPENDITURE_TABLE_MONTH_COUNT - 1][i]; } gHistoricalProfit += sum; } // Shift the table for (size_t i = EXPENDITURE_TABLE_MONTH_COUNT - 1; i >= 1; i--) { for (size_t j = 0; j < RCT_EXPENDITURE_TYPE_COUNT; j++) { gExpenditureTable[i][j] = gExpenditureTable[i - 1][j]; } } // Zero the beginning of the table, which is the new month for (uint32 i = 0; i < RCT_EXPENDITURE_TYPE_COUNT; i++) { gExpenditureTable[0][i] = 0; } window_invalidate_by_class(WC_FINANCES); } /** * * rct2: 0x0069E89B */ void finance_reset_cash_to_initial() { gCash = gInitialCash; } /** * Gets the last month's profit from food, drink and merchandise. */ money32 finance_get_last_month_shop_profit() { money32 profit = 0; if (gDateMonthsElapsed != 0) { money32 * lastMonthExpenditure = gExpenditureTable[1]; profit += lastMonthExpenditure[RCT_EXPENDITURE_TYPE_SHOP_SHOP_SALES]; profit += lastMonthExpenditure[RCT_EXPENDITURE_TYPE_SHOP_STOCK]; profit += lastMonthExpenditure[RCT_EXPENDITURE_TYPE_FOODDRINK_SALES]; profit += lastMonthExpenditure[RCT_EXPENDITURE_TYPE_FOODDRINK_STOCK]; } return profit; }