diff --git a/src/script/api/script_gamesettings.cpp b/src/script/api/script_gamesettings.cpp index c6da9d1822..e5aa8c306b 100644 --- a/src/script/api/script_gamesettings.cpp +++ b/src/script/api/script_gamesettings.cpp @@ -12,31 +12,83 @@ #include "../../settings_internal.h" #include "../../settings_type.h" #include "../../command_type.h" +#include "../../command_func.h" +#include "../../economy_func.h" + +#include #include "../../safeguards.h" +struct CargoScalingProxy { + bool is_industry; + + CargoScalingProxy(bool is_industry) : is_industry(is_industry) {} + + SQInteger ReadValue() const + { + uint64_t scale = this->is_industry ? _settings_game.economy.industry_cargo_scale : _settings_game.economy.town_cargo_scale; + CargoScalingMode mode = this->is_industry ? _settings_game.economy.industry_cargo_scale_mode : _settings_game.economy.town_cargo_scale_mode; + if (mode == CSM_DAYLENGTH) { + scale *= _settings_game.economy.day_length_factor; + } + return PercentageToScaleQuantityFactor(scale); + } + + bool SetValue(SQInteger value) const + { + CargoScalingMode mode = this->is_industry ? _settings_game.economy.industry_cargo_scale_mode : _settings_game.economy.town_cargo_scale_mode; + if (mode == CSM_DAYLENGTH) { + /* Asynchronous free command, don't bother halting the script or saving the result */ + ::DoCommandPScript(0, 0, (uint32_t)CSM_MONTHLY, 0, CMD_CHANGE_SETTING, nullptr, + this->is_industry ? "economy.industry_cargo_scale_mode" : "economy.town_cargo_scale_mode", false, false, true, nullptr); + } + + return ScriptGameSettings::SetValue(this->is_industry ? "economy.industry_cargo_scale" : "economy.town_cargo_scale", ScaleQuantity(100, (int)value)); + } + + static std::optional Get(const std::string &setting) + { + std::string_view prefix = "economy."; + std::string_view str = setting; + if (str.starts_with(prefix)) str.remove_prefix(prefix.size()); + if (str == "town_cargo_scale_factor") return CargoScalingProxy(false); + if (str == "industry_cargo_scale_factor") return CargoScalingProxy(true); + return std::nullopt; + } +}; + /* static */ bool ScriptGameSettings::IsValid(const std::string &setting) { + std::optional csproxy = CargoScalingProxy::Get(setting); + if (csproxy.has_value()) return true; + const SettingDesc *sd = GetSettingFromName(setting); return sd != nullptr && sd->IsIntSetting(); } /* static */ SQInteger ScriptGameSettings::GetValue(const std::string &setting) { - if (!IsValid(setting)) return -1; + std::optional csproxy = CargoScalingProxy::Get(setting); + if (csproxy.has_value()) { + return csproxy->ReadValue(); + } const SettingDesc *sd = GetSettingFromName(setting); - assert(sd != nullptr); + if (sd == nullptr || !sd->IsIntSetting()) return -1; return sd->AsIntSetting()->Read(&_settings_game); } /* static */ bool ScriptGameSettings::SetValue(const std::string &setting, SQInteger value) { EnforceDeityOrCompanyModeValid(false); - if (!IsValid(setting)) return false; + + std::optional csproxy = CargoScalingProxy::Get(setting); + if (csproxy.has_value()) { + return csproxy->SetValue(value); + } const SettingDesc *sd = GetSettingFromName(setting); - assert(sd != nullptr); + if (sd == nullptr || !sd->IsIntSetting()) return false; if ((sd->flags & SF_NO_NETWORK_SYNC) != 0) return false;