/* * This file is part of OpenTTD. * OpenTTD 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, version 2. * OpenTTD 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 OpenTTD. If not, see . */ /** @file script_bridge.cpp Implementation of ScriptBridge. */ #include "../../stdafx.h" #include "script_bridge.hpp" #include "script_rail.hpp" #include "../script_instance.hpp" #include "../../bridge_map.h" #include "../../strings_func.h" #include "../../landscape_cmd.h" #include "../../road_cmd.h" #include "../../tunnelbridge_cmd.h" #include "../../timer/timer_game_calendar.h" #include "table/strings.h" #include "../../safeguards.h" /* static */ bool ScriptBridge::IsValidBridge(BridgeType bridge_type) { return bridge_type < MAX_BRIDGES && ::GetBridgeSpec(bridge_type)->avail_year <= TimerGameCalendar::year; } /* static */ bool ScriptBridge::IsBridgeTile(TileIndex tile) { if (!::IsValidTile(tile)) return false; return ::IsBridgeTile(tile); } /* static */ BridgeType ScriptBridge::GetBridgeType(TileIndex tile) { if (!IsBridgeTile(tile)) return (BridgeType)-1; return (BridgeType)::GetBridgeType(tile); } /** * Helper function to connect a just built bridge to nearby roads. * @param instance The script instance we have to built the road for. */ static void _DoCommandReturnBuildBridge2(class ScriptInstance *instance) { if (!ScriptBridge::_BuildBridgeRoad2()) { ScriptInstance::DoCommandReturn(instance); return; } /* This can never happen, as in test-mode this callback is never executed, * and in execute-mode, the other callback is called. */ NOT_REACHED(); } /** * Helper function to connect a just built bridge to nearby roads. * @param instance The script instance we have to built the road for. */ static void _DoCommandReturnBuildBridge1(class ScriptInstance *instance) { if (!ScriptBridge::_BuildBridgeRoad1()) { ScriptInstance::DoCommandReturn(instance); return; } /* This can never happen, as in test-mode this callback is never executed, * and in execute-mode, the other callback is called. */ NOT_REACHED(); } /* static */ bool ScriptBridge::BuildBridge(ScriptVehicle::VehicleType vehicle_type, BridgeType bridge_type, TileIndex start, TileIndex end) { EnforceDeityOrCompanyModeValid(false); EnforcePrecondition(false, start != end); EnforcePrecondition(false, ::IsValidTile(start) && ::IsValidTile(end)); EnforcePrecondition(false, TileX(start) == TileX(end) || TileY(start) == TileY(end)); EnforcePrecondition(false, vehicle_type == ScriptVehicle::VT_ROAD || vehicle_type == ScriptVehicle::VT_RAIL || vehicle_type == ScriptVehicle::VT_WATER); EnforcePrecondition(false, vehicle_type != ScriptVehicle::VT_RAIL || ScriptRail::IsRailTypeAvailable(ScriptRail::GetCurrentRailType())); EnforcePrecondition(false, vehicle_type != ScriptVehicle::VT_ROAD || ScriptRoad::IsRoadTypeAvailable(ScriptRoad::GetCurrentRoadType())); EnforcePrecondition(false, ScriptCompanyMode::IsValid() || vehicle_type == ScriptVehicle::VT_ROAD); switch (vehicle_type) { case ScriptVehicle::VT_ROAD: ScriptObject::SetCallbackVariable(0, start.base()); ScriptObject::SetCallbackVariable(1, end.base()); return ScriptObject::Command::Do(&::_DoCommandReturnBuildBridge1, end, start, TRANSPORT_ROAD, bridge_type, ScriptRoad::GetCurrentRoadType()); case ScriptVehicle::VT_RAIL: return ScriptObject::Command::Do(end, start, TRANSPORT_RAIL, bridge_type, ScriptRail::GetCurrentRailType()); case ScriptVehicle::VT_WATER: return ScriptObject::Command::Do(end, start, TRANSPORT_WATER, bridge_type, 0); default: NOT_REACHED(); } } /* static */ bool ScriptBridge::_BuildBridgeRoad1() { EnforceDeityOrCompanyModeValid(false); /* Build the piece of road on the 'start' side of the bridge */ TileIndex end(ScriptObject::GetCallbackVariable(0)); TileIndex start(ScriptObject::GetCallbackVariable(1)); DiagDirection dir_1 = ::DiagdirBetweenTiles(end, start); DiagDirection dir_2 = ::ReverseDiagDir(dir_1); return ScriptObject::Command::Do(&::_DoCommandReturnBuildBridge2, start + ::TileOffsByDiagDir(dir_1), ::DiagDirToRoadBits(dir_2), (::RoadType)ScriptRoad::GetCurrentRoadType(), DRD_NONE, TownID::Invalid()); } /* static */ bool ScriptBridge::_BuildBridgeRoad2() { EnforceDeityOrCompanyModeValid(false); /* Build the piece of road on the 'end' side of the bridge */ TileIndex end(ScriptObject::GetCallbackVariable(0)); TileIndex start(ScriptObject::GetCallbackVariable(1)); DiagDirection dir_1 = ::DiagdirBetweenTiles(end, start); DiagDirection dir_2 = ::ReverseDiagDir(dir_1); return ScriptObject::Command::Do(end + ::TileOffsByDiagDir(dir_2), ::DiagDirToRoadBits(dir_1), (::RoadType)ScriptRoad::GetCurrentRoadType(), DRD_NONE, TownID::Invalid()); } /* static */ bool ScriptBridge::RemoveBridge(TileIndex tile) { EnforceCompanyModeValid(false); EnforcePrecondition(false, IsBridgeTile(tile)); return ScriptObject::Command::Do(tile); } /* static */ std::optional ScriptBridge::GetName(BridgeType bridge_type, ScriptVehicle::VehicleType vehicle_type) { EnforcePrecondition(std::nullopt, vehicle_type == ScriptVehicle::VT_ROAD || vehicle_type == ScriptVehicle::VT_RAIL || vehicle_type == ScriptVehicle::VT_WATER); if (!IsValidBridge(bridge_type)) return std::nullopt; return ::StrMakeValid(::GetString(vehicle_type == ScriptVehicle::VT_WATER ? STR_LAI_BRIDGE_DESCRIPTION_AQUEDUCT : ::GetBridgeSpec(bridge_type)->transport_name[vehicle_type])); } /* static */ SQInteger ScriptBridge::GetMaxSpeed(BridgeType bridge_type) { if (!IsValidBridge(bridge_type)) return -1; return ::GetBridgeSpec(bridge_type)->speed; // km-ish/h } /* static */ Money ScriptBridge::GetPrice(BridgeType bridge_type, SQInteger length) { if (!IsValidBridge(bridge_type)) return -1; length = Clamp(length, 0, INT32_MAX); return ::CalcBridgeLenCostFactor(length) * _price[PR_BUILD_BRIDGE] * ::GetBridgeSpec(bridge_type)->price >> 8; } /* static */ SQInteger ScriptBridge::GetMaxLength(BridgeType bridge_type) { if (!IsValidBridge(bridge_type)) return -1; return std::min(::GetBridgeSpec(bridge_type)->max_length, _settings_game.construction.max_bridge_length) + 2; } /* static */ SQInteger ScriptBridge::GetMinLength(BridgeType bridge_type) { if (!IsValidBridge(bridge_type)) return -1; return static_cast(::GetBridgeSpec(bridge_type)->min_length) + 2; } /* static */ TileIndex ScriptBridge::GetOtherBridgeEnd(TileIndex tile) { if (!::IsValidTile(tile)) return INVALID_TILE; if (!IsBridgeTile(tile)) return INVALID_TILE; return ::GetOtherBridgeEnd(tile); }