diff options
-rw-r--r-- | src/train_cmd.cpp | 194 |
1 files changed, 148 insertions, 46 deletions
diff --git a/src/train_cmd.cpp b/src/train_cmd.cpp index d9e54e1f1..7f1f23747 100644 --- a/src/train_cmd.cpp +++ b/src/train_cmd.cpp @@ -53,6 +53,7 @@ static bool TrainCheckIfLineEnds(Vehicle *v); static void TrainController(Vehicle *v, bool update_image); +static TileIndex TrainApproachingCrossingTile(const Vehicle *v); static const byte _vehicle_initial_x_fract[4] = {10, 8, 4, 8}; static const byte _vehicle_initial_y_fract[4] = { 8, 4, 8, 10}; @@ -1624,23 +1625,81 @@ static void ReverseTrainSwapVeh(Vehicle *v, int l, int r) TrainPowerChanged(v); } -/* Check if the vehicle is a train and is on the tile we are testing */ -static void *TestTrainOnCrossing(Vehicle *v, void *data) + +/** + * Check if the vehicle is a train + * @param v vehicle on tile + * @return v if it is a train, NULL otherwise + */ +static void *TrainOnTileEnum(Vehicle *v, void *) { - if (v->type != VEH_TRAIN) return NULL; + return (v->type == VEH_TRAIN) ? v : NULL; +} + + +/** + * Checks if a train is approaching a rail-road crossing + * @param v vehicle on tile + * @param data tile with crossing we are testing + * @return v if it is approaching a crossing, NULL otherwise + */ +static void *TrainApproachingCrossingEnum(Vehicle *v, void *data) +{ + /* not a train || not front engine || crashed */ + if (v->type != VEH_TRAIN || !IsFrontEngine(v) || v->vehstatus & VS_CRASHED) return NULL; + + TileIndex tile = *(TileIndex*)data; + + if (TrainApproachingCrossingTile(v) != tile) return NULL; + return v; } -static void DisableTrainCrossing(TileIndex tile) + +/** + * Finds a vehicle approaching rail-road crossing + * @param tile tile to test + * @return pointer to vehicle approaching the crossing + * @pre tile is a rail-road crossing + */ +static Vehicle *TrainApproachingCrossing(TileIndex tile) { - if (IsLevelCrossingTile(tile) && - IsCrossingBarred(tile) && - VehicleFromPos(tile, NULL, &TestTrainOnCrossing) == NULL) { // empty? - UnbarCrossing(tile); - MarkTileDirtyByTile(tile); + assert(IsLevelCrossingTile(tile)); + + DiagDirection dir = AxisToDiagDir(OtherAxis(GetCrossingRoadAxis(tile))); + TileIndex tile_from = tile + TileOffsByDiagDir(dir); + + Vehicle *v = (Vehicle *)VehicleFromPos(tile_from, &tile, &TrainApproachingCrossingEnum); + + if (v != NULL) return v; + + dir = ReverseDiagDir(dir); + tile_from = tile + TileOffsByDiagDir(dir); + + return (Vehicle *)VehicleFromPos(tile_from, &tile, &TrainApproachingCrossingEnum); +} + + +/** + * Sets correct crossing state + * @param tile tile to update + * @pre tile is a rail-road crossing + */ +void UpdateTrainCrossing(TileIndex tile) +{ + assert(IsLevelCrossingTile(tile)); + + UnbarCrossing(tile); + + /* train on crossing || train approaching crossing */ + if (VehicleFromPos(tile, NULL, &TrainOnTileEnum) || TrainApproachingCrossing(tile)) { + BarCrossing(tile); } + + MarkTileDirtyByTile(tile); } + /** * Advances wagons for train reversing, needed for variable length wagons. * Needs to be called once before the train is reversed, and once after it. @@ -1688,14 +1747,7 @@ static void ReverseTrainDirection(Vehicle *v) } /* Check if we were approaching a rail/road-crossing */ - { - /* Determine the diagonal direction in which we will exit this tile */ - DiagDirection dir = TrainExitDir(v->direction, v->u.rail.track); - /* Calculate next tile */ - TileIndex tile = v->tile + TileOffsByDiagDir(dir); - /* Check if the train left a rail/road-crossing */ - DisableTrainCrossing(tile); - } + TileIndex crossing = TrainApproachingCrossingTile(v); /* count number of vehicles */ int r = 0; ///< number of vehicles - 1 @@ -1719,6 +1771,13 @@ static void ReverseTrainDirection(Vehicle *v) for (Vehicle *u = v; u != NULL; u = u->Next()) { u->cur_image = u->GetImage(u->direction); } ClrBit(v->u.rail.flags, VRF_REVERSING); + + /* update crossing we were approaching */ + if (crossing != INVALID_TILE) UpdateTrainCrossing(crossing); + + /* maybe we are approaching crossing now, after reversal */ + crossing = TrainApproachingCrossingTile(v); + if (crossing != INVALID_TILE) UpdateTrainCrossing(crossing); } /** Reverse train. @@ -2777,6 +2836,9 @@ static void SetVehicleCrashed(Vehicle *v) { if (v->u.rail.crash_anim_pos != 0) return; + /* we may need to update crossing we were approaching */ + TileIndex crossing = TrainApproachingCrossingTile(v); + v->u.rail.crash_anim_pos++; InvalidateWindowWidget(WC_VEHICLE_VIEW, v->index, STATUS_BAR); @@ -2792,6 +2854,9 @@ static void SetVehicleCrashed(Vehicle *v) v->vehstatus |= VS_CRASHED; MarkSingleVehicleDirty(v); END_ENUM_WAGONS(v) + + /* must be updated after the train has been marked crashed */ + if (crossing != INVALID_TILE) UpdateTrainCrossing(crossing); } static uint CountPassengersInTrain(const Vehicle* v) @@ -2899,7 +2964,7 @@ static void TrainController(Vehicle *v, bool update_image) /* For every vehicle after and including the given vehicle */ for (prev = v->Previous(); v != NULL; prev = v, v = v->Next()) { DiagDirection enterdir = DIAGDIR_BEGIN; - bool update_signals = false; + bool update_signals_crossing = false; // will we update signals or crossing state? BeginVehicleMove(v); GetNewVehiclePosResult gp = GetNewVehiclePos(v); @@ -3022,11 +3087,6 @@ static void TrainController(Vehicle *v, bool update_image) goto invalid_rail; } - if (IsLevelCrossingTile(v->tile) && v->Next() == NULL) { - UnbarCrossing(v->tile); - MarkTileDirtyByTile(v->tile); - } - if (IsFrontEngine(v)) v->load_unload_time_rem = 0; if (!HasBit(r, VETS_ENTERED_WORMHOLE)) { @@ -3042,7 +3102,7 @@ static void TrainController(Vehicle *v, bool update_image) /* We need to update signal status, but after the vehicle position hash * has been updated by AfterSetTrainPos() */ - update_signals = true; + update_signals_crossing = true; if (prev == NULL) AffectSpeedByDirChange(v, chosen_dir); @@ -3081,12 +3141,15 @@ static void TrainController(Vehicle *v, bool update_image) AffectSpeedByZChange(v, old_z); } - if (update_signals) { + if (update_signals_crossing) { if (IsFrontEngine(v)) TrainMovedChangeSignals(gp.new_tile, enterdir); /* Signals can only change when the first * (above) or the last vehicle moves. */ - if (v->Next() == NULL) TrainMovedChangeSignals(gp.old_tile, ReverseDiagDir(enterdir)); + if (v->Next() == NULL) { + TrainMovedChangeSignals(gp.old_tile, ReverseDiagDir(enterdir)); + if (IsLevelCrossingTile(gp.old_tile)) UpdateTrainCrossing(gp.old_tile); + } } } return; @@ -3147,9 +3210,8 @@ static void DeleteLastWagon(Vehicle *v) delete v; v = NULL; // make sure nobody will won't try to read 'v' anymore - /* Check if the wagon was on a road/rail-crossing and disable it if no - * others are on it */ - DisableTrainCrossing(tile); + /* check if the wagon was on a road/rail-crossing */ + if (IsLevelCrossingTile(tile)) UpdateTrainCrossing(tile); /* Update signals */ if (IsTileType(tile, MP_TUNNELBRIDGE) || IsTileDepotType(tile, TRANSPORT_RAIL)) { @@ -3296,6 +3358,61 @@ static bool TrainApproachingLineEnd(Vehicle *v, bool signal) /** + * Determines whether train would like to leave the tile + * @param v train to test + * @return true iff vehicle is NOT entering or inside a depot or tunnel/bridge + */ +static bool TrainCanLeaveTile(const Vehicle *v) +{ + /* Exit if inside a tunnel/bridge or a depot */ + if (v->u.rail.track == TRACK_BIT_WORMHOLE || v->u.rail.track == TRACK_BIT_DEPOT) return false; + + TileIndex tile = v->tile; + + /* entering a tunnel/bridge? */ + if (IsTileType(tile, MP_TUNNELBRIDGE)) { + DiagDirection dir = GetTunnelBridgeDirection(tile); + if (DiagDirToDir(dir) == v->direction) return false; + } + + /* entering a depot? */ + if (IsTileDepotType(tile, TRANSPORT_RAIL)) { + DiagDirection dir = ReverseDiagDir(GetRailDepotDirection(tile)); + if (DiagDirToDir(dir) == v->direction) return false; + } + + return true; +} + + +/** + * Determines whether train is approaching a rail-road crossing + * (thus making it barred) + * @param v front engine of train + * @return TileIndex of crossing the train is approaching, else INVALID_TILE + * @pre v in non-crashed front engine + */ +static TileIndex TrainApproachingCrossingTile(const Vehicle *v) +{ + assert(IsFrontEngine(v)); + assert(!(v->vehstatus & VS_CRASHED)); + + if (!TrainCanLeaveTile(v)) return INVALID_TILE; + + DiagDirection dir = TrainExitDir(v->direction, v->u.rail.track); + TileIndex tile = v->tile + TileOffsByDiagDir(dir); + + /* not a crossing || wrong axis || wrong railtype || wrong owner */ + if (!IsLevelCrossingTile(tile) || DiagDirToAxis(dir) == GetCrossingRoadAxis(tile) || + !CheckCompatibleRail(v, tile) || GetTileOwner(tile) != v->owner) { + return INVALID_TILE; + } + + return tile; +} + + +/** * Checks for line end. Also, bars crossing at next tile if needed * * @param v vehicle we are checking @@ -3315,27 +3432,12 @@ static bool TrainCheckIfLineEnds(Vehicle *v) v->vehstatus &= ~VS_TRAIN_SLOWING; } - /* Exit if inside a tunnel/bridge or a depot */ - if (v->u.rail.track == TRACK_BIT_WORMHOLE || v->u.rail.track == TRACK_BIT_DEPOT) return true; - - TileIndex tile = v->tile; - - /* entering a tunnel/bridge? */ - if (IsTileType(tile, MP_TUNNELBRIDGE)) { - DiagDirection dir = GetTunnelBridgeDirection(tile); - if (DiagDirToDir(dir) == v->direction) return true; - } - - /* entering a depot? */ - if (IsTileDepotType(tile, TRANSPORT_RAIL)) { - DiagDirection dir = ReverseDiagDir(GetRailDepotDirection(tile)); - if (DiagDirToDir(dir) == v->direction) return true; - } + if (!TrainCanLeaveTile(v)) return true; /* Determine the non-diagonal direction in which we will exit this tile */ DiagDirection dir = TrainExitDir(v->direction, v->u.rail.track); /* Calculate next tile */ - tile += TileOffsByDiagDir(dir); + TileIndex tile = v->tile + TileOffsByDiagDir(dir); /* Determine the track status on the next tile */ uint32 ts = GetTileTrackStatus(tile, TRANSPORT_RAIL, 0) & _reachable_tracks[dir]; |