Update: Translations from eints
[openttd-github.git] / src / newgrf_station.cpp
blob86dbe8bf329254e8710c8aefbcd70221ff27638c
1 /*
2 * This file is part of OpenTTD.
3 * 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.
4 * 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.
5 * 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 <http://www.gnu.org/licenses/>.
6 */
8 /** @file newgrf_station.cpp Functions for dealing with station classes and custom stations. */
10 #include "stdafx.h"
11 #include "debug.h"
12 #include "station_base.h"
13 #include "waypoint_base.h"
14 #include "roadstop_base.h"
15 #include "newgrf_cargo.h"
16 #include "newgrf_station.h"
17 #include "newgrf_spritegroup.h"
18 #include "newgrf_sound.h"
19 #include "newgrf_railtype.h"
20 #include "town.h"
21 #include "newgrf_town.h"
22 #include "company_func.h"
23 #include "tunnelbridge_map.h"
24 #include "newgrf_animation_base.h"
25 #include "newgrf_class_func.h"
26 #include "timer/timer_game_calendar.h"
28 #include "safeguards.h"
31 template <>
32 /* static */ void StationClass::InsertDefaults()
34 /* Set up initial data */
35 StationClass::Get(StationClass::Allocate(STATION_CLASS_LABEL_DEFAULT))->name = STR_STATION_CLASS_DFLT;
36 StationClass::Get(StationClass::Allocate(STATION_CLASS_LABEL_DEFAULT))->Insert(nullptr);
37 StationClass::Get(StationClass::Allocate(STATION_CLASS_LABEL_WAYPOINT))->name = STR_STATION_CLASS_WAYP;
38 StationClass::Get(StationClass::Allocate(STATION_CLASS_LABEL_WAYPOINT))->Insert(nullptr);
41 template <>
42 bool StationClass::IsUIAvailable(uint) const
44 return true;
47 /* Instantiate StationClass. */
48 template class NewGRFClass<StationSpec, StationClassID, STAT_CLASS_MAX>;
50 static const uint NUM_STATIONSSPECS_PER_STATION = 255; ///< Maximum number of parts per station.
52 enum TriggerArea {
53 TA_TILE,
54 TA_PLATFORM,
55 TA_WHOLE,
58 struct ETileArea : TileArea {
59 ETileArea(const BaseStation *st, TileIndex tile, TriggerArea ta)
61 switch (ta) {
62 default: NOT_REACHED();
64 case TA_TILE:
65 this->tile = tile;
66 this->w = 1;
67 this->h = 1;
68 break;
70 case TA_PLATFORM: {
71 TileIndex start, end;
72 Axis axis = GetRailStationAxis(tile);
73 TileIndexDiff delta = TileOffsByAxis(axis);
75 for (end = tile; IsRailStationTile(end + delta) && IsCompatibleTrainStationTile(end + delta, tile); end += delta) { /* Nothing */ }
76 for (start = tile; IsRailStationTile(start - delta) && IsCompatibleTrainStationTile(start - delta, tile); start -= delta) { /* Nothing */ }
78 this->tile = start;
79 this->w = TileX(end) - TileX(start) + 1;
80 this->h = TileY(end) - TileY(start) + 1;
81 break;
84 case TA_WHOLE:
85 st->GetTileArea(this, Station::IsExpected(st) ? STATION_RAIL : STATION_WAYPOINT);
86 break;
92 /**
93 * Evaluate a tile's position within a station, and return the result in a bit-stuffed format.
94 * if not centered: .TNLcCpP, if centered: .TNL..CP
95 * - T = Tile layout number (#GetStationGfx)
96 * - N = Number of platforms
97 * - L = Length of platforms
98 * - C = Current platform number from start, c = from end
99 * - P = Position along platform from start, p = from end
101 * if centered, C/P start from the centre and c/p are not available.
102 * @return Platform information in bit-stuffed format.
104 uint32_t GetPlatformInfo(Axis axis, uint8_t tile, int platforms, int length, int x, int y, bool centred)
106 uint32_t retval = 0;
108 if (axis == AXIS_X) {
109 Swap(platforms, length);
110 Swap(x, y);
113 if (centred) {
114 x -= platforms / 2;
115 y -= length / 2;
116 x = Clamp(x, -8, 7);
117 y = Clamp(y, -8, 7);
118 SB(retval, 0, 4, y & 0xF);
119 SB(retval, 4, 4, x & 0xF);
120 } else {
121 SB(retval, 0, 4, std::min(15, y));
122 SB(retval, 4, 4, std::min(15, length - y - 1));
123 SB(retval, 8, 4, std::min(15, x));
124 SB(retval, 12, 4, std::min(15, platforms - x - 1));
126 SB(retval, 16, 4, std::min(15, length));
127 SB(retval, 20, 4, std::min(15, platforms));
128 SB(retval, 24, 8, tile);
130 return retval;
135 * Find the end of a railway station, from the \a tile, in the direction of \a delta.
136 * @param tile Start tile.
137 * @param delta Movement direction.
138 * @param check_type Stop when the custom station type changes.
139 * @param check_axis Stop when the station direction changes.
140 * @return Found end of the railway station.
142 static TileIndex FindRailStationEnd(TileIndex tile, TileIndexDiff delta, bool check_type, bool check_axis)
144 uint8_t orig_type = 0;
145 Axis orig_axis = AXIS_X;
146 StationID sid = GetStationIndex(tile);
148 if (check_type) orig_type = GetCustomStationSpecIndex(tile);
149 if (check_axis) orig_axis = GetRailStationAxis(tile);
151 for (;;) {
152 TileIndex new_tile = TileAdd(tile, delta);
154 if (!IsTileType(new_tile, MP_STATION) || GetStationIndex(new_tile) != sid) break;
155 if (!HasStationRail(new_tile)) break;
156 if (check_type && GetCustomStationSpecIndex(new_tile) != orig_type) break;
157 if (check_axis && GetRailStationAxis(new_tile) != orig_axis) break;
159 tile = new_tile;
161 return tile;
165 static uint32_t GetPlatformInfoHelper(TileIndex tile, bool check_type, bool check_axis, bool centred)
167 int tx = TileX(tile);
168 int ty = TileY(tile);
169 int sx = TileX(FindRailStationEnd(tile, TileDiffXY(-1, 0), check_type, check_axis));
170 int sy = TileY(FindRailStationEnd(tile, TileDiffXY( 0, -1), check_type, check_axis));
171 int ex = TileX(FindRailStationEnd(tile, TileDiffXY( 1, 0), check_type, check_axis)) + 1;
172 int ey = TileY(FindRailStationEnd(tile, TileDiffXY( 0, 1), check_type, check_axis)) + 1;
174 tx -= sx; ex -= sx;
175 ty -= sy; ey -= sy;
177 return GetPlatformInfo(GetRailStationAxis(tile), GetStationGfx(tile), ex, ey, tx, ty, centred);
181 static uint32_t GetRailContinuationInfo(TileIndex tile)
183 /* Tile offsets and exit dirs for X axis */
184 static const Direction x_dir[8] = { DIR_SW, DIR_NE, DIR_SE, DIR_NW, DIR_S, DIR_E, DIR_W, DIR_N };
185 static const DiagDirection x_exits[8] = { DIAGDIR_SW, DIAGDIR_NE, DIAGDIR_SE, DIAGDIR_NW, DIAGDIR_SW, DIAGDIR_NE, DIAGDIR_SW, DIAGDIR_NE };
187 /* Tile offsets and exit dirs for Y axis */
188 static const Direction y_dir[8] = { DIR_SE, DIR_NW, DIR_SW, DIR_NE, DIR_S, DIR_W, DIR_E, DIR_N };
189 static const DiagDirection y_exits[8] = { DIAGDIR_SE, DIAGDIR_NW, DIAGDIR_SW, DIAGDIR_NE, DIAGDIR_SE, DIAGDIR_NW, DIAGDIR_SE, DIAGDIR_NW };
191 Axis axis = GetRailStationAxis(tile);
193 /* Choose appropriate lookup table to use */
194 const Direction *dir = axis == AXIS_X ? x_dir : y_dir;
195 const DiagDirection *diagdir = axis == AXIS_X ? x_exits : y_exits;
197 uint32_t res = 0;
198 uint i;
200 for (i = 0; i < lengthof(x_dir); i++, dir++, diagdir++) {
201 TileIndex neighbour_tile = tile + TileOffsByDir(*dir);
202 TrackBits trackbits = TrackStatusToTrackBits(GetTileTrackStatus(neighbour_tile, TRANSPORT_RAIL, 0));
203 if (trackbits != TRACK_BIT_NONE) {
204 /* If there is any track on the tile, set the bit in the second byte */
205 SetBit(res, i + 8);
207 /* With tunnels and bridges the tile has tracks, but they are not necessarily connected
208 * with the next tile because the ramp is not going in the right direction. */
209 if (IsTileType(neighbour_tile, MP_TUNNELBRIDGE) && GetTunnelBridgeDirection(neighbour_tile) != *diagdir) {
210 continue;
213 /* If any track reaches our exit direction, set the bit in the lower byte */
214 if (trackbits & DiagdirReachesTracks(*diagdir)) SetBit(res, i);
218 return res;
222 /* Station Resolver Functions */
223 /* virtual */ uint32_t StationScopeResolver::GetRandomBits() const
225 return (this->st == nullptr ? 0 : this->st->random_bits) | (this->tile == INVALID_TILE ? 0 : GetStationTileRandomBits(this->tile) << 16);
229 /* virtual */ uint32_t StationScopeResolver::GetTriggers() const
231 return this->st == nullptr ? 0 : this->st->waiting_triggers;
236 * Station variable cache
237 * This caches 'expensive' station variable lookups which iterate over
238 * several tiles that may be called multiple times per Resolve().
240 static struct {
241 uint32_t v40;
242 uint32_t v41;
243 uint32_t v45;
244 uint32_t v46;
245 uint32_t v47;
246 uint32_t v49;
247 uint8_t valid; ///< Bits indicating what variable is valid (for each bit, \c 0 is invalid, \c 1 is valid).
248 } _svc;
251 * Get the town scope associated with a station, if it exists.
252 * On the first call, the town scope is created (if possible).
253 * @return Town scope, if available.
255 TownScopeResolver *StationResolverObject::GetTown()
257 if (!this->town_scope.has_value()) {
258 Town *t = nullptr;
259 if (this->station_scope.st != nullptr) {
260 t = this->station_scope.st->town;
261 } else if (this->station_scope.tile != INVALID_TILE) {
262 t = ClosestTownFromTile(this->station_scope.tile, UINT_MAX);
264 if (t == nullptr) return nullptr;
265 this->town_scope.emplace(*this, t, this->station_scope.st == nullptr);
267 return &*this->town_scope;
270 /* virtual */ uint32_t StationScopeResolver::GetVariable(uint8_t variable, [[maybe_unused]] uint32_t parameter, bool &available) const
272 if (this->st == nullptr) {
273 /* Station does not exist, so we're in a purchase list or the land slope check callback. */
274 switch (variable) {
275 case 0x40:
276 case 0x41:
277 case 0x46:
278 case 0x47:
279 case 0x49: return 0x2110000; // Platforms, tracks & position
280 case 0x42: return 0; // Rail type (XXX Get current type from GUI?)
281 case 0x43: return GetCompanyInfo(_current_company); // Station owner
282 case 0x44: return 2; // PBS status
283 case 0x67: // Land info of nearby tile
284 if (this->axis != INVALID_AXIS && this->tile != INVALID_TILE) {
285 TileIndex tile = this->tile;
286 if (parameter != 0) tile = GetNearbyTile(parameter, tile, true, this->axis); // only perform if it is required
288 Slope tileh = GetTileSlope(tile);
289 bool swap = (this->axis == AXIS_Y && HasBit(tileh, CORNER_W) != HasBit(tileh, CORNER_E));
291 return GetNearbyTileInformation(tile, this->ro.grffile->grf_version >= 8) ^ (swap ? SLOPE_EW : 0);
293 break;
295 case 0xFA: return ClampTo<uint16_t>(TimerGameCalendar::date - CalendarTime::DAYS_TILL_ORIGINAL_BASE_YEAR); // Build date, clamped to a 16 bit value
298 available = false;
299 return UINT_MAX;
302 switch (variable) {
303 /* Calculated station variables */
304 case 0x40:
305 if (!HasBit(_svc.valid, 0)) { _svc.v40 = GetPlatformInfoHelper(this->tile, false, false, false); SetBit(_svc.valid, 0); }
306 return _svc.v40;
308 case 0x41:
309 if (!HasBit(_svc.valid, 1)) { _svc.v41 = GetPlatformInfoHelper(this->tile, true, false, false); SetBit(_svc.valid, 1); }
310 return _svc.v41;
312 case 0x42: return GetTerrainType(this->tile) | (GetReverseRailTypeTranslation(GetRailType(this->tile), this->statspec->grf_prop.grffile) << 8);
313 case 0x43: return GetCompanyInfo(this->st->owner); // Station owner
314 case 0x44: return HasStationReservation(this->tile) ? 7 : 4; // PBS status
315 case 0x45:
316 if (!HasBit(_svc.valid, 2)) { _svc.v45 = GetRailContinuationInfo(this->tile); SetBit(_svc.valid, 2); }
317 return _svc.v45;
319 case 0x46:
320 if (!HasBit(_svc.valid, 3)) { _svc.v46 = GetPlatformInfoHelper(this->tile, false, false, true); SetBit(_svc.valid, 3); }
321 return _svc.v46;
323 case 0x47:
324 if (!HasBit(_svc.valid, 4)) { _svc.v47 = GetPlatformInfoHelper(this->tile, true, false, true); SetBit(_svc.valid, 4); }
325 return _svc.v47;
327 case 0x49:
328 if (!HasBit(_svc.valid, 5)) { _svc.v49 = GetPlatformInfoHelper(this->tile, false, true, false); SetBit(_svc.valid, 5); }
329 return _svc.v49;
331 case 0x4A: // Animation frame of tile
332 return GetAnimationFrame(this->tile);
334 /* Variables which use the parameter */
335 /* Variables 0x60 to 0x65 and 0x69 are handled separately below */
336 case 0x66: { // Animation frame of nearby tile
337 TileIndex tile = this->tile;
338 if (parameter != 0) tile = GetNearbyTile(parameter, tile);
339 return this->st->TileBelongsToRailStation(tile) ? GetAnimationFrame(tile) : UINT_MAX;
342 case 0x67: { // Land info of nearby tile
343 Axis axis = GetRailStationAxis(this->tile);
344 TileIndex tile = this->tile;
345 if (parameter != 0) tile = GetNearbyTile(parameter, tile); // only perform if it is required
347 Slope tileh = GetTileSlope(tile);
348 bool swap = (axis == AXIS_Y && HasBit(tileh, CORNER_W) != HasBit(tileh, CORNER_E));
350 return GetNearbyTileInformation(tile, this->ro.grffile->grf_version >= 8) ^ (swap ? SLOPE_EW : 0);
353 case 0x68: { // Station info of nearby tiles
354 TileIndex nearby_tile = GetNearbyTile(parameter, this->tile);
356 if (!HasStationTileRail(nearby_tile)) return 0xFFFFFFFF;
358 uint32_t grfid = this->st->speclist[GetCustomStationSpecIndex(this->tile)].grfid;
359 bool perpendicular = GetRailStationAxis(this->tile) != GetRailStationAxis(nearby_tile);
360 bool same_station = this->st->TileBelongsToRailStation(nearby_tile);
361 uint32_t res = GB(GetStationGfx(nearby_tile), 1, 2) << 12 | !!perpendicular << 11 | !!same_station << 10;
363 if (IsCustomStationSpecIndex(nearby_tile)) {
364 const auto &sm = BaseStation::GetByTile(nearby_tile)->speclist[GetCustomStationSpecIndex(nearby_tile)];
365 res |= 1 << (sm.grfid != grfid ? 9 : 8) | ClampTo<uint8_t>(sm.localidx);
367 return res;
370 case 0x6A: { // GRFID of nearby station tiles
371 TileIndex nearby_tile = GetNearbyTile(parameter, this->tile);
373 if (!HasStationTileRail(nearby_tile)) return 0xFFFFFFFF;
374 if (!IsCustomStationSpecIndex(nearby_tile)) return 0;
376 const auto &sm = BaseStation::GetByTile(nearby_tile)->speclist[GetCustomStationSpecIndex(nearby_tile)];
377 return sm.grfid;
380 case 0x6B: { // 16 bit Station ID of nearby tiles
381 TileIndex nearby_tile = GetNearbyTile(parameter, this->tile);
383 if (!HasStationTileRail(nearby_tile)) return 0xFFFFFFFF;
384 if (!IsCustomStationSpecIndex(nearby_tile)) return 0xFFFE;
386 uint32_t grfid = this->st->speclist[GetCustomStationSpecIndex(this->tile)].grfid;
388 const auto &sm = BaseStation::GetByTile(nearby_tile)->speclist[GetCustomStationSpecIndex(nearby_tile)];
389 if (sm.grfid == grfid) {
390 return sm.localidx;
393 return 0xFFFE;
396 /* General station variables */
397 case 0x82: return 50;
398 case 0x84: return this->st->string_id;
399 case 0x86: return 0;
400 case 0xF0: return this->st->facilities;
401 case 0xFA: return ClampTo<uint16_t>(this->st->build_date - CalendarTime::DAYS_TILL_ORIGINAL_BASE_YEAR);
404 return this->st->GetNewGRFVariable(this->ro, variable, parameter, available);
407 uint32_t Station::GetNewGRFVariable(const ResolverObject &object, uint8_t variable, uint8_t parameter, bool &available) const
409 switch (variable) {
410 case 0x48: { // Accepted cargo types
411 uint32_t value = GetAcceptanceMask(this);
412 return value;
415 case 0x8A: return this->had_vehicle_of_type;
416 case 0xF1: return (this->airport.tile != INVALID_TILE) ? this->airport.GetSpec()->ttd_airport_type : ATP_TTDP_LARGE;
417 case 0xF2: return (this->truck_stops != nullptr) ? this->truck_stops->status : 0;
418 case 0xF3: return (this->bus_stops != nullptr) ? this->bus_stops->status : 0;
419 case 0xF6: return this->airport.flags;
420 case 0xF7: return GB(this->airport.flags, 8, 8);
423 /* Handle cargo variables with parameter, 0x60 to 0x65 and 0x69 */
424 if ((variable >= 0x60 && variable <= 0x65) || variable == 0x69) {
425 CargoID c = GetCargoTranslation(parameter, object.grffile);
427 if (!IsValidCargoID(c)) {
428 switch (variable) {
429 case 0x62: return 0xFFFFFFFF;
430 case 0x64: return 0xFF00;
431 default: return 0;
434 const GoodsEntry *ge = &this->goods[c];
436 switch (variable) {
437 case 0x60: return std::min(ge->cargo.TotalCount(), 4095u);
438 case 0x61: return ge->HasVehicleEverTriedLoading() ? ge->time_since_pickup : 0;
439 case 0x62: return ge->HasRating() ? ge->rating : 0xFFFFFFFF;
440 case 0x63: return ge->cargo.PeriodsInTransit();
441 case 0x64: return ge->HasVehicleEverTriedLoading() ? ge->last_speed | (ge->last_age << 8) : 0xFF00;
442 case 0x65: return GB(ge->status, GoodsEntry::GES_ACCEPTANCE, 1) << 3;
443 case 0x69: {
444 static_assert((int)GoodsEntry::GES_EVER_ACCEPTED + 1 == (int)GoodsEntry::GES_LAST_MONTH);
445 static_assert((int)GoodsEntry::GES_EVER_ACCEPTED + 2 == (int)GoodsEntry::GES_CURRENT_MONTH);
446 static_assert((int)GoodsEntry::GES_EVER_ACCEPTED + 3 == (int)GoodsEntry::GES_ACCEPTED_BIGTICK);
447 return GB(ge->status, GoodsEntry::GES_EVER_ACCEPTED, 4);
452 /* Handle cargo variables (deprecated) */
453 if (variable >= 0x8C && variable <= 0xEC) {
454 const GoodsEntry *g = &this->goods[GB(variable - 0x8C, 3, 4)];
455 switch (GB(variable - 0x8C, 0, 3)) {
456 case 0: return g->cargo.TotalCount();
457 case 1: return GB(std::min(g->cargo.TotalCount(), 4095u), 0, 4) | (GB(g->status, GoodsEntry::GES_ACCEPTANCE, 1) << 7);
458 case 2: return g->time_since_pickup;
459 case 3: return g->rating;
460 case 4: return g->cargo.GetFirstStation();
461 case 5: return g->cargo.PeriodsInTransit();
462 case 6: return g->last_speed;
463 case 7: return g->last_age;
467 Debug(grf, 1, "Unhandled station variable 0x{:X}", variable);
469 available = false;
470 return UINT_MAX;
473 uint32_t Waypoint::GetNewGRFVariable(const ResolverObject &, uint8_t variable, [[maybe_unused]] uint8_t parameter, bool &available) const
475 switch (variable) {
476 case 0x48: return 0; // Accepted cargo types
477 case 0x8A: return HVOT_WAYPOINT;
478 case 0xF1: return 0; // airport type
479 case 0xF2: return 0; // truck stop status
480 case 0xF3: return 0; // bus stop status
481 case 0xF6: return 0; // airport flags
482 case 0xF7: return 0; // airport flags cont.
485 /* Handle cargo variables with parameter, 0x60 to 0x65 */
486 if (variable >= 0x60 && variable <= 0x65) {
487 return 0;
490 /* Handle cargo variables (deprecated) */
491 if (variable >= 0x8C && variable <= 0xEC) {
492 switch (GB(variable - 0x8C, 0, 3)) {
493 case 3: return INITIAL_STATION_RATING;
494 case 4: return INVALID_STATION;
495 default: return 0;
499 Debug(grf, 1, "Unhandled station variable 0x{:X}", variable);
501 available = false;
502 return UINT_MAX;
505 /* virtual */ const SpriteGroup *StationResolverObject::ResolveReal(const RealSpriteGroup *group) const
507 if (this->station_scope.st == nullptr || !Station::IsExpected(this->station_scope.st)) {
508 return group->loading[0];
511 uint cargo = 0;
512 const Station *st = Station::From(this->station_scope.st);
514 switch (this->station_scope.cargo_type) {
515 case INVALID_CARGO:
516 case SpriteGroupCargo::SG_DEFAULT_NA:
517 case SpriteGroupCargo::SG_PURCHASE:
518 cargo = 0;
519 break;
521 case SpriteGroupCargo::SG_DEFAULT:
522 for (const GoodsEntry &ge : st->goods) {
523 cargo += ge.cargo.TotalCount();
525 break;
527 default:
528 cargo = st->goods[this->station_scope.cargo_type].cargo.TotalCount();
529 break;
532 if (HasBit(this->station_scope.statspec->flags, SSF_DIV_BY_STATION_SIZE)) cargo /= (st->train_station.w + st->train_station.h);
533 cargo = std::min(0xfffu, cargo);
535 if (cargo > this->station_scope.statspec->cargo_threshold) {
536 if (!group->loading.empty()) {
537 uint set = ((cargo - this->station_scope.statspec->cargo_threshold) * (uint)group->loading.size()) / (4096 - this->station_scope.statspec->cargo_threshold);
538 return group->loading[set];
540 } else {
541 if (!group->loaded.empty()) {
542 uint set = (cargo * (uint)group->loaded.size()) / (this->station_scope.statspec->cargo_threshold + 1);
543 return group->loaded[set];
547 return group->loading[0];
550 GrfSpecFeature StationResolverObject::GetFeature() const
552 return GSF_STATIONS;
555 uint32_t StationResolverObject::GetDebugID() const
557 return this->station_scope.statspec->grf_prop.local_id;
561 * Resolver for stations.
562 * @param statspec Station (type) specification.
563 * @param base_station Instance of the station.
564 * @param tile %Tile of the station.
565 * @param callback Callback ID.
566 * @param callback_param1 First parameter (var 10) of the callback.
567 * @param callback_param2 Second parameter (var 18) of the callback.
569 StationResolverObject::StationResolverObject(const StationSpec *statspec, BaseStation *base_station, TileIndex tile,
570 CallbackID callback, uint32_t callback_param1, uint32_t callback_param2)
571 : ResolverObject(statspec->grf_prop.grffile, callback, callback_param1, callback_param2),
572 station_scope(*this, statspec, base_station, tile)
574 /* Invalidate all cached vars */
575 _svc.valid = 0;
577 CargoID ctype = SpriteGroupCargo::SG_DEFAULT_NA;
579 if (this->station_scope.st == nullptr) {
580 /* No station, so we are in a purchase list */
581 ctype = SpriteGroupCargo::SG_PURCHASE;
582 } else if (Station::IsExpected(this->station_scope.st)) {
583 const Station *st = Station::From(this->station_scope.st);
584 /* Pick the first cargo that we have waiting */
585 for (const CargoSpec *cs : CargoSpec::Iterate()) {
586 if (this->station_scope.statspec->grf_prop.spritegroup[cs->Index()] != nullptr &&
587 st->goods[cs->Index()].cargo.TotalCount() > 0) {
588 ctype = cs->Index();
589 break;
594 if (this->station_scope.statspec->grf_prop.spritegroup[ctype] == nullptr) {
595 ctype = SpriteGroupCargo::SG_DEFAULT;
598 /* Remember the cargo type we've picked */
599 this->station_scope.cargo_type = ctype;
600 this->root_spritegroup = this->station_scope.statspec->grf_prop.spritegroup[this->station_scope.cargo_type];
604 * Resolve sprites for drawing a station tile.
605 * @param statspec Station spec
606 * @param st Station (nullptr in GUI)
607 * @param tile Station tile being drawn (INVALID_TILE in GUI)
608 * @param var10 Value to put in variable 10; normally 0; 1 when resolving the groundsprite and SSF_SEPARATE_GROUND is set.
609 * @return First sprite of the Action 1 spriteset to use, minus an offset of 0x42D to accommodate for weird NewGRF specs.
611 SpriteID GetCustomStationRelocation(const StationSpec *statspec, BaseStation *st, TileIndex tile, uint32_t var10)
613 StationResolverObject object(statspec, st, tile, CBID_NO_CALLBACK, var10);
614 const SpriteGroup *group = object.Resolve();
615 if (group == nullptr || group->type != SGT_RESULT) return 0;
616 return group->GetResult() - 0x42D;
620 * Resolve the sprites for custom station foundations.
621 * @param statspec Station spec
622 * @param st Station
623 * @param tile Station tile being drawn
624 * @param layout Spritelayout as returned by previous callback
625 * @param edge_info Information about northern tile edges; whether they need foundations or merge into adjacent tile's foundations.
626 * @return First sprite of a set of foundation sprites for various slopes, or 0 if default foundations shall be drawn.
628 SpriteID GetCustomStationFoundationRelocation(const StationSpec *statspec, BaseStation *st, TileIndex tile, uint layout, uint edge_info)
630 /* callback_param1 == 2 means we are resolving the foundation sprites. */
631 StationResolverObject object(statspec, st, tile, CBID_NO_CALLBACK, 2, layout | (edge_info << 16));
633 const SpriteGroup *group = object.Resolve();
634 if (group == nullptr || group->type != SGT_RESULT) return 0;
636 /* Note: SpriteGroup::Resolve zeroes all registers, so register 0x100 is initialised to 0. (compatibility) */
637 return group->GetResult() + GetRegister(0x100);
641 uint16_t GetStationCallback(CallbackID callback, uint32_t param1, uint32_t param2, const StationSpec *statspec, BaseStation *st, TileIndex tile)
643 StationResolverObject object(statspec, st, tile, callback, param1, param2);
644 return object.ResolveCallback();
648 * Check the slope of a tile of a new station.
649 * @param north_tile Norther tile of the station rect.
650 * @param cur_tile Tile to check.
651 * @param statspec Station spec.
652 * @param axis Axis of the new station.
653 * @param plat_len Platform length.
654 * @param numtracks Number of platforms.
655 * @return Succeeded or failed command.
657 CommandCost PerformStationTileSlopeCheck(TileIndex north_tile, TileIndex cur_tile, const StationSpec *statspec, Axis axis, uint8_t plat_len, uint8_t numtracks)
659 TileIndex diff = cur_tile - north_tile;
660 Slope slope = GetTileSlope(cur_tile);
662 StationResolverObject object(statspec, nullptr, cur_tile, CBID_STATION_LAND_SLOPE_CHECK,
663 (slope << 4) | (slope ^ (axis == AXIS_Y && HasBit(slope, CORNER_W) != HasBit(slope, CORNER_E) ? SLOPE_EW : 0)),
664 (numtracks << 24) | (plat_len << 16) | (axis == AXIS_Y ? TileX(diff) << 8 | TileY(diff) : TileY(diff) << 8 | TileX(diff)));
665 object.station_scope.axis = axis;
667 uint16_t cb_res = object.ResolveCallback();
669 /* Failed callback means success. */
670 if (cb_res == CALLBACK_FAILED) return CommandCost();
672 /* The meaning of bit 10 is inverted for a grf version < 8. */
673 if (statspec->grf_prop.grffile->grf_version < 8) ToggleBit(cb_res, 10);
674 return GetErrorMessageFromLocationCallbackResult(cb_res, statspec->grf_prop.grffile, STR_ERROR_LAND_SLOPED_IN_WRONG_DIRECTION);
679 * Allocate a StationSpec to a Station. This is called once per build operation.
680 * @param statspec StationSpec to allocate.
681 * @param st Station to allocate it to.
682 * @param exec Whether to actually allocate the spec.
683 * @return Index within the Station's spec list, or -1 if the allocation failed.
685 int AllocateSpecToStation(const StationSpec *statspec, BaseStation *st, bool exec)
687 uint i;
689 if (statspec == nullptr || st == nullptr) return 0;
691 for (i = 1; i < st->speclist.size() && i < NUM_STATIONSSPECS_PER_STATION; i++) {
692 if (st->speclist[i].spec == nullptr && st->speclist[i].grfid == 0) break;
695 if (i == NUM_STATIONSSPECS_PER_STATION) {
696 /* As final effort when the spec list is already full...
697 * try to find the same spec and return that one. This might
698 * result in slightly "wrong" (as per specs) looking stations,
699 * but it's fairly unlikely that one reaches the limit anyways.
701 for (i = 1; i < st->speclist.size() && i < NUM_STATIONSSPECS_PER_STATION; i++) {
702 if (st->speclist[i].spec == statspec) return i;
705 return -1;
708 if (exec) {
709 if (i >= st->speclist.size()) st->speclist.resize(i + 1);
710 st->speclist[i].spec = statspec;
711 st->speclist[i].grfid = statspec->grf_prop.grffile->grfid;
712 st->speclist[i].localidx = statspec->grf_prop.local_id;
714 StationUpdateCachedTriggers(st);
717 return i;
722 * Deallocate a StationSpec from a Station. Called when removing a single station tile.
723 * @param st Station to work with.
724 * @param specindex Index of the custom station within the Station's spec list.
725 * @return Indicates whether the StationSpec was deallocated.
727 void DeallocateSpecFromStation(BaseStation *st, uint8_t specindex)
729 /* specindex of 0 (default) is never freeable */
730 if (specindex == 0) return;
732 ETileArea area = ETileArea(st, INVALID_TILE, TA_WHOLE);
733 /* Check all tiles over the station to check if the specindex is still in use */
734 for (TileIndex tile : area) {
735 if (st->TileBelongsToRailStation(tile) && GetCustomStationSpecIndex(tile) == specindex) {
736 return;
740 /* This specindex is no longer in use, so deallocate it */
741 st->speclist[specindex].spec = nullptr;
742 st->speclist[specindex].grfid = 0;
743 st->speclist[specindex].localidx = 0;
745 /* If this was the highest spec index, reallocate */
746 if (specindex == st->speclist.size() - 1) {
747 size_t num_specs;
748 for (num_specs = st->speclist.size() - 1; num_specs > 0; num_specs--) {
749 if (st->speclist[num_specs].grfid != 0) break;
752 if (num_specs > 0) {
753 st->speclist.resize(num_specs + 1);
754 } else {
755 st->speclist.clear();
756 st->cached_anim_triggers = 0;
757 st->cached_cargo_triggers = 0;
758 return;
762 StationUpdateCachedTriggers(st);
766 * Draw representation of a station tile for GUI purposes.
767 * @param x Position x of image.
768 * @param y Position y of image.
769 * @param axis Axis.
770 * @param railtype Rail type.
771 * @param sclass, station Type of station.
772 * @param station station ID
773 * @return True if the tile was drawn (allows for fallback to default graphic)
775 bool DrawStationTile(int x, int y, RailType railtype, Axis axis, StationClassID sclass, uint station)
777 const DrawTileSprites *sprites = nullptr;
778 const RailTypeInfo *rti = GetRailTypeInfo(railtype);
779 PaletteID palette = COMPANY_SPRITE_COLOUR(_local_company);
780 uint tile = 2;
782 const StationSpec *statspec = StationClass::Get(sclass)->GetSpec(station);
783 if (statspec == nullptr) return false;
785 if (HasBit(statspec->callback_mask, CBM_STATION_DRAW_TILE_LAYOUT)) {
786 uint16_t callback = GetStationCallback(CBID_STATION_DRAW_TILE_LAYOUT, 0, 0, statspec, nullptr, INVALID_TILE);
787 if (callback != CALLBACK_FAILED) tile = callback & ~1;
790 uint32_t total_offset = rti->GetRailtypeSpriteOffset();
791 uint32_t relocation = 0;
792 uint32_t ground_relocation = 0;
793 const NewGRFSpriteLayout *layout = nullptr;
794 DrawTileSprites tmp_rail_layout;
796 if (statspec->renderdata.empty()) {
797 sprites = GetStationTileLayout(STATION_RAIL, tile + axis);
798 } else {
799 layout = &statspec->renderdata[(tile < statspec->renderdata.size()) ? tile + axis : (uint)axis];
800 if (!layout->NeedsPreprocessing()) {
801 sprites = layout;
802 layout = nullptr;
806 if (layout != nullptr) {
807 /* Sprite layout which needs preprocessing */
808 bool separate_ground = HasBit(statspec->flags, SSF_SEPARATE_GROUND);
809 uint32_t var10_values = layout->PrepareLayout(total_offset, rti->fallback_railtype, 0, 0, separate_ground);
810 for (uint8_t var10 : SetBitIterator(var10_values)) {
811 uint32_t var10_relocation = GetCustomStationRelocation(statspec, nullptr, INVALID_TILE, var10);
812 layout->ProcessRegisters(var10, var10_relocation, separate_ground);
815 tmp_rail_layout.seq = layout->GetLayout(&tmp_rail_layout.ground);
816 sprites = &tmp_rail_layout;
817 total_offset = 0;
818 } else {
819 /* Simple sprite layout */
820 ground_relocation = relocation = GetCustomStationRelocation(statspec, nullptr, INVALID_TILE, 0);
821 if (HasBit(sprites->ground.sprite, SPRITE_MODIFIER_CUSTOM_SPRITE)) {
822 ground_relocation = GetCustomStationRelocation(statspec, nullptr, INVALID_TILE, 1);
824 ground_relocation += rti->fallback_railtype;
827 SpriteID image = sprites->ground.sprite;
828 PaletteID pal = sprites->ground.pal;
829 RailTrackOffset overlay_offset;
830 if (rti->UsesOverlay() && SplitGroundSpriteForOverlay(nullptr, &image, &overlay_offset)) {
831 SpriteID ground = GetCustomRailSprite(rti, INVALID_TILE, RTSG_GROUND);
832 DrawSprite(image, PAL_NONE, x, y);
833 DrawSprite(ground + overlay_offset, PAL_NONE, x, y);
834 } else {
835 image += HasBit(image, SPRITE_MODIFIER_CUSTOM_SPRITE) ? ground_relocation : total_offset;
836 if (HasBit(pal, SPRITE_MODIFIER_CUSTOM_SPRITE)) pal += ground_relocation;
837 DrawSprite(image, GroundSpritePaletteTransform(image, pal, palette), x, y);
840 DrawRailTileSeqInGUI(x, y, sprites, total_offset, relocation, palette);
842 return true;
846 const StationSpec *GetStationSpec(TileIndex t)
848 if (!IsCustomStationSpecIndex(t)) return nullptr;
850 const BaseStation *st = BaseStation::GetByTile(t);
851 uint specindex = GetCustomStationSpecIndex(t);
852 return specindex < st->speclist.size() ? st->speclist[specindex].spec : nullptr;
855 /** Wrapper for animation control, see GetStationCallback. */
856 uint16_t GetAnimStationCallback(CallbackID callback, uint32_t param1, uint32_t param2, const StationSpec *statspec, BaseStation *st, TileIndex tile, int)
858 return GetStationCallback(callback, param1, param2, statspec, st, tile);
861 /** Helper class for animation control. */
862 struct StationAnimationBase : public AnimationBase<StationAnimationBase, StationSpec, BaseStation, int, GetAnimStationCallback, TileAnimationFrameAnimationHelper<BaseStation> > {
863 static const CallbackID cb_animation_speed = CBID_STATION_ANIMATION_SPEED;
864 static const CallbackID cb_animation_next_frame = CBID_STATION_ANIM_NEXT_FRAME;
866 static const StationCallbackMask cbm_animation_speed = CBM_STATION_ANIMATION_SPEED;
867 static const StationCallbackMask cbm_animation_next_frame = CBM_STATION_ANIMATION_NEXT_FRAME;
870 void AnimateStationTile(TileIndex tile)
872 const StationSpec *ss = GetStationSpec(tile);
873 if (ss == nullptr) return;
875 StationAnimationBase::AnimateTile(ss, BaseStation::GetByTile(tile), tile, HasBit(ss->flags, SSF_CB141_RANDOM_BITS));
878 void TriggerStationAnimation(BaseStation *st, TileIndex trigger_tile, StationAnimationTrigger trigger, CargoID cargo_type)
880 /* List of coverage areas for each animation trigger */
881 static const TriggerArea tas[] = {
882 TA_TILE, TA_WHOLE, TA_WHOLE, TA_PLATFORM, TA_PLATFORM, TA_PLATFORM, TA_WHOLE
885 /* Get Station if it wasn't supplied */
886 if (st == nullptr) st = BaseStation::GetByTile(trigger_tile);
888 /* Check the cached animation trigger bitmask to see if we need
889 * to bother with any further processing. */
890 if (!HasBit(st->cached_anim_triggers, trigger)) return;
892 uint16_t random_bits = Random();
893 ETileArea area = ETileArea(st, trigger_tile, tas[trigger]);
895 /* Check all tiles over the station to check if the specindex is still in use */
896 for (TileIndex tile : area) {
897 if (st->TileBelongsToRailStation(tile)) {
898 const StationSpec *ss = GetStationSpec(tile);
899 if (ss != nullptr && HasBit(ss->animation.triggers, trigger)) {
900 CargoID cargo;
901 if (!IsValidCargoID(cargo_type)) {
902 cargo = INVALID_CARGO;
903 } else {
904 cargo = ss->grf_prop.grffile->cargo_map[cargo_type];
906 StationAnimationBase::ChangeAnimationFrame(CBID_STATION_ANIM_START_STOP, ss, st, tile, (random_bits << 16) | GB(Random(), 0, 16), (uint8_t)trigger | (cargo << 8));
913 * Trigger station randomisation
914 * @param st station being triggered
915 * @param trigger_tile specific tile of platform to trigger
916 * @param trigger trigger type
917 * @param cargo_type cargo type causing trigger
919 void TriggerStationRandomisation(Station *st, TileIndex trigger_tile, StationRandomTrigger trigger, CargoID cargo_type)
921 /* List of coverage areas for each animation trigger */
922 static const TriggerArea tas[] = {
923 TA_WHOLE, TA_WHOLE, TA_PLATFORM, TA_PLATFORM, TA_PLATFORM, TA_PLATFORM
926 /* Get Station if it wasn't supplied */
927 if (st == nullptr) st = Station::GetByTile(trigger_tile);
929 /* Check the cached cargo trigger bitmask to see if we need
930 * to bother with any further processing. */
931 if (st->cached_cargo_triggers == 0) return;
932 if (IsValidCargoID(cargo_type) && !HasBit(st->cached_cargo_triggers, cargo_type)) return;
934 uint32_t whole_reseed = 0;
935 ETileArea area = ETileArea(st, trigger_tile, tas[trigger]);
937 /* Bitmask of completely empty cargo types to be matched. */
938 CargoTypes empty_mask = (trigger == SRT_CARGO_TAKEN) ? GetEmptyMask(st) : 0;
940 /* Store triggers now for var 5F */
941 SetBit(st->waiting_triggers, trigger);
942 uint32_t used_triggers = 0;
944 /* Check all tiles over the station to check if the specindex is still in use */
945 for (TileIndex tile : area) {
946 if (st->TileBelongsToRailStation(tile)) {
947 const StationSpec *ss = GetStationSpec(tile);
948 if (ss == nullptr) continue;
950 /* Cargo taken "will only be triggered if all of those
951 * cargo types have no more cargo waiting." */
952 if (trigger == SRT_CARGO_TAKEN) {
953 if ((ss->cargo_triggers & ~empty_mask) != 0) continue;
956 if (!IsValidCargoID(cargo_type) || HasBit(ss->cargo_triggers, cargo_type)) {
957 StationResolverObject object(ss, st, tile, CBID_RANDOM_TRIGGER, 0);
958 object.waiting_triggers = st->waiting_triggers;
960 const SpriteGroup *group = object.Resolve();
961 if (group == nullptr) continue;
963 used_triggers |= object.used_triggers;
965 uint32_t reseed = object.GetReseedSum();
966 if (reseed != 0) {
967 whole_reseed |= reseed;
968 reseed >>= 16;
970 /* Set individual tile random bits */
971 uint8_t random_bits = GetStationTileRandomBits(tile);
972 random_bits &= ~reseed;
973 random_bits |= Random() & reseed;
974 SetStationTileRandomBits(tile, random_bits);
976 MarkTileDirtyByTile(tile);
982 /* Update whole station random bits */
983 st->waiting_triggers &= ~used_triggers;
984 if ((whole_reseed & 0xFFFF) != 0) {
985 st->random_bits &= ~whole_reseed;
986 st->random_bits |= Random() & whole_reseed;
991 * Update the cached animation trigger bitmask for a station.
992 * @param st Station to update.
994 void StationUpdateCachedTriggers(BaseStation *st)
996 st->cached_anim_triggers = 0;
997 st->cached_cargo_triggers = 0;
999 /* Combine animation trigger bitmask for all station specs
1000 * of this station. */
1001 for (const auto &sm : GetStationSpecList<StationSpec>(st)) {
1002 if (sm.spec == nullptr) continue;
1003 st->cached_anim_triggers |= sm.spec->animation.triggers;
1004 st->cached_cargo_triggers |= sm.spec->cargo_triggers;