OpenTTD Source 20250328-master-gc3457cd4c0
vehicle.cpp
Go to the documentation of this file.
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 */
7
10#include "stdafx.h"
11#include "error.h"
12#include "roadveh.h"
13#include "ship.h"
14#include "spritecache.h"
15#include "timetable.h"
16#include "viewport_func.h"
17#include "news_func.h"
18#include "command_func.h"
19#include "company_func.h"
20#include "train.h"
21#include "aircraft.h"
22#include "newgrf_debug.h"
23#include "newgrf_sound.h"
24#include "newgrf_station.h"
25#include "group_gui.h"
26#include "strings_func.h"
27#include "zoom_func.h"
28#include "vehicle_func.h"
29#include "autoreplace_func.h"
30#include "autoreplace_gui.h"
31#include "station_base.h"
32#include "ai/ai.hpp"
33#include "depot_func.h"
34#include "network/network.h"
35#include "core/pool_func.hpp"
36#include "economy_base.h"
38#include "roadstop_base.h"
39#include "core/random_func.hpp"
40#include "core/backup_type.hpp"
42#include "order_backup.h"
43#include "sound_func.h"
44#include "effectvehicle_func.h"
45#include "effectvehicle_base.h"
46#include "vehiclelist.h"
47#include "bridge_map.h"
48#include "tunnel_map.h"
49#include "depot_map.h"
50#include "gamelog.h"
51#include "linkgraph/linkgraph.h"
52#include "linkgraph/refresh.h"
53#include "framerate_type.h"
54#include "autoreplace_cmd.h"
55#include "misc_cmd.h"
56#include "train_cmd.h"
57#include "vehicle_cmd.h"
58#include "newgrf_roadstop.h"
59#include "timer/timer.h"
63
64#include "table/strings.h"
65
66#include "safeguards.h"
67
68/* Number of bits in the hash to use from each vehicle coord */
69static const uint GEN_HASHX_BITS = 6;
70static const uint GEN_HASHY_BITS = 6;
71
72/* Size of each hash bucket */
73static const uint GEN_HASHX_BUCKET_BITS = 7;
74static const uint GEN_HASHY_BUCKET_BITS = 6;
75
76/* Compute hash for vehicle coord */
77#define GEN_HASHX(x) GB((x), GEN_HASHX_BUCKET_BITS + ZOOM_BASE_SHIFT, GEN_HASHX_BITS)
78#define GEN_HASHY(y) (GB((y), GEN_HASHY_BUCKET_BITS + ZOOM_BASE_SHIFT, GEN_HASHY_BITS) << GEN_HASHX_BITS)
79#define GEN_HASH(x, y) (GEN_HASHY(y) + GEN_HASHX(x))
80
81/* Maximum size until hash repeats */
82static const int GEN_HASHX_SIZE = 1 << (GEN_HASHX_BUCKET_BITS + GEN_HASHX_BITS + ZOOM_BASE_SHIFT);
83static const int GEN_HASHY_SIZE = 1 << (GEN_HASHY_BUCKET_BITS + GEN_HASHY_BITS + ZOOM_BASE_SHIFT);
84
85/* Increments to reach next bucket in hash table */
86static const int GEN_HASHX_INC = 1;
87static const int GEN_HASHY_INC = 1 << GEN_HASHX_BITS;
88
89/* Mask to wrap-around buckets */
90static const uint GEN_HASHX_MASK = (1 << GEN_HASHX_BITS) - 1;
91static const uint GEN_HASHY_MASK = ((1 << GEN_HASHY_BITS) - 1) << GEN_HASHX_BITS;
92
93
97
98
99
103void VehicleSpriteSeq::GetBounds(Rect *bounds) const
104{
105 bounds->left = bounds->top = bounds->right = bounds->bottom = 0;
106 for (uint i = 0; i < this->count; ++i) {
107 const Sprite *spr = GetSprite(this->seq[i].sprite, SpriteType::Normal);
108 if (i == 0) {
109 bounds->left = spr->x_offs;
110 bounds->top = spr->y_offs;
111 bounds->right = spr->width + spr->x_offs - 1;
112 bounds->bottom = spr->height + spr->y_offs - 1;
113 } else {
114 if (spr->x_offs < bounds->left) bounds->left = spr->x_offs;
115 if (spr->y_offs < bounds->top) bounds->top = spr->y_offs;
116 int right = spr->width + spr->x_offs - 1;
117 int bottom = spr->height + spr->y_offs - 1;
118 if (right > bounds->right) bounds->right = right;
119 if (bottom > bounds->bottom) bounds->bottom = bottom;
120 }
121 }
122}
123
131void VehicleSpriteSeq::Draw(int x, int y, PaletteID default_pal, bool force_pal) const
132{
133 for (uint i = 0; i < this->count; ++i) {
134 PaletteID pal = force_pal || !this->seq[i].pal ? default_pal : this->seq[i].pal;
135 DrawSprite(this->seq[i].sprite, pal, x, y);
136 }
137}
138
145bool Vehicle::NeedsAutorenewing(const Company *c, bool use_renew_setting) const
146{
147 /* We can always generate the Company pointer when we have the vehicle.
148 * However this takes time and since the Company pointer is often present
149 * when this function is called then it's faster to pass the pointer as an
150 * argument rather than finding it again. */
151 assert(c == Company::Get(this->owner));
152
153 if (use_renew_setting && !c->settings.engine_renew) return false;
154 if (this->age - this->max_age < (c->settings.engine_renew_months * 30)) return false;
155
156 /* Only engines need renewing */
157 if (this->type == VEH_TRAIN && !Train::From(this)->IsEngine()) return false;
158
159 return true;
160}
161
168{
169 assert(v != nullptr);
170 SetWindowDirty(WC_VEHICLE_DETAILS, v->index); // ensure that last service date and reliability are updated
171
172 do {
177 /* Prevent vehicles from breaking down directly after exiting the depot. */
178 v->breakdown_chance /= 4;
179 if (_settings_game.difficulty.vehicle_breakdowns == 1) v->breakdown_chance = 0; // on reduced breakdown
180 v = v->Next();
181 } while (v != nullptr && v->HasEngineType());
182}
183
191{
192 /* Stopped or crashed vehicles will not move, as such making unmovable
193 * vehicles to go for service is lame. */
194 if (this->vehstatus.Any({VehState::Stopped, VehState::Crashed})) return false;
195
196 /* Are we ready for the next service cycle? */
197 const Company *c = Company::Get(this->owner);
198
199 /* Service intervals can be measured in different units, which we handle individually. */
200 if (this->ServiceIntervalIsPercent()) {
201 /* Service interval is in percents. */
202 if (this->reliability >= this->GetEngine()->reliability * (100 - this->GetServiceInterval()) / 100) return false;
204 /* Service interval is in minutes. */
205 if (this->date_of_last_service + (this->GetServiceInterval() * EconomyTime::DAYS_IN_ECONOMY_MONTH) >= TimerGameEconomy::date) return false;
206 } else {
207 /* Service interval is in days. */
208 if (this->date_of_last_service + this->GetServiceInterval() >= TimerGameEconomy::date) return false;
209 }
210
211 /* If we're servicing anyway, because we have not disabled servicing when
212 * there are no breakdowns or we are playing with breakdowns, bail out. */
215 return true;
216 }
217
218 /* Test whether there is some pending autoreplace.
219 * Note: We do this after the service-interval test.
220 * There are a lot more reasons for autoreplace to fail than we can test here reasonably. */
221 bool pending_replace = false;
222 Money needed_money = c->settings.engine_renew_money;
223 if (needed_money > GetAvailableMoney(c->index)) return false;
224
225 for (const Vehicle *v = this; v != nullptr; v = (v->type == VEH_TRAIN) ? Train::From(v)->GetNextUnit() : nullptr) {
226 bool replace_when_old = false;
227 EngineID new_engine = EngineReplacementForCompany(c, v->engine_type, v->group_id, &replace_when_old);
228
229 /* Check engine availability */
230 if (new_engine == EngineID::Invalid() || !Engine::Get(new_engine)->company_avail.Test(v->owner)) continue;
231 /* Is the vehicle old if we are not always replacing? */
232 if (replace_when_old && !v->NeedsAutorenewing(c, false)) continue;
233
234 /* Check refittability */
235 CargoTypes available_cargo_types, union_mask;
236 GetArticulatedRefitMasks(new_engine, true, &union_mask, &available_cargo_types);
237 /* Is there anything to refit? */
238 if (union_mask != 0) {
240 CargoTypes cargo_mask = GetCargoTypesOfArticulatedVehicle(v, &cargo_type);
241 if (!HasAtMostOneBit(cargo_mask)) {
242 CargoTypes new_engine_default_cargoes = GetCargoTypesOfArticulatedParts(new_engine);
243 if ((cargo_mask & new_engine_default_cargoes) != cargo_mask) {
244 /* We cannot refit to mixed cargoes in an automated way */
245 continue;
246 }
247 /* engine_type is already a mixed cargo type which matches the incoming vehicle by default, no refit required */
248 } else {
249 /* Did the old vehicle carry anything? */
251 /* We can't refit the vehicle to carry the cargo we want */
252 if (!HasBit(available_cargo_types, cargo_type)) continue;
253 }
254 }
255 }
256
257 /* Check money.
258 * We want 2*(the price of the new vehicle) without looking at the value of the vehicle we are going to sell. */
259 pending_replace = true;
260 needed_money += 2 * Engine::Get(new_engine)->GetCost();
261 if (needed_money > GetAvailableMoney(c->index)) return false;
262 }
263
264 return pending_replace;
265}
266
273{
274 if (this->HasDepotOrder()) return false;
275 if (this->current_order.IsType(OT_LOADING)) return false;
276 if (this->current_order.IsType(OT_GOTO_DEPOT) && (this->current_order.GetDepotOrderType() & ODTFB_SERVICE) == 0) return false;
277 return NeedsServicing();
278}
279
281{
282 assert(!this->vehstatus.Test(VehState::Crashed));
283 assert(this->Previous() == nullptr); // IsPrimaryVehicle fails for free-wagon-chains
284
285 uint pass = 0;
286 /* Stop the vehicle. */
288 /* crash all wagons, and count passengers */
289 for (Vehicle *v = this; v != nullptr; v = v->Next()) {
290 /* We do not transfer reserver cargo back, so TotalCount() instead of StoredCount() */
291 if (IsCargoInClass(v->cargo_type, CargoClass::Passengers)) pass += v->cargo.TotalCount();
292 v->vehstatus.Set(VehState::Crashed);
293 v->MarkAllViewportsDirty();
294 }
295
296 /* Dirty some windows */
301
302 delete this->cargo_payment;
303 assert(this->cargo_payment == nullptr); // cleared by ~CargoPayment
304
305 return RandomRange(pass + 1); // Randomise deceased passengers.
306}
307
308
317void ShowNewGrfVehicleError(EngineID engine, StringID part1, StringID part2, GRFBug bug_type, bool critical)
318{
319 const Engine *e = Engine::Get(engine);
320 GRFConfig *grfconfig = GetGRFConfig(e->GetGRFID());
321
322 /* Missing GRF. Nothing useful can be done in this situation. */
323 if (grfconfig == nullptr) return;
324
325 if (!grfconfig->grf_bugs.Test(bug_type)) {
326 grfconfig->grf_bugs.Set(bug_type);
327 ShowErrorMessage(GetEncodedString(part1, grfconfig->GetName()),
328 GetEncodedString(part2, std::monostate{}, engine), WL_CRITICAL);
330 }
331
332 /* debug output */
333 Debug(grf, 0, "{}", StrMakeValid(GetString(part1, grfconfig->GetName())));
334
335 Debug(grf, 0, "{}", StrMakeValid(GetString(part2, std::monostate{}, engine)));
336}
337
344{
345 /* show a warning once for each engine in whole game and once for each GRF after each game load */
346 const Engine *engine = u->GetEngine();
347 uint32_t grfid = engine->grf_prop.grfid;
348 GRFConfig *grfconfig = GetGRFConfig(grfid);
349 if (_gamelog.GRFBugReverse(grfid, engine->grf_prop.local_id) || !grfconfig->grf_bugs.Test(GRFBug::VehLength)) {
350 ShowNewGrfVehicleError(u->engine_type, STR_NEWGRF_BROKEN, STR_NEWGRF_BROKEN_VEHICLE_LENGTH, GRFBug::VehLength, true);
351 }
352}
353
359{
360 this->type = type;
361 this->coord.left = INVALID_COORD;
363 this->group_id = DEFAULT_GROUP;
364 this->fill_percent_te_id = INVALID_TE_ID;
365 this->first = this;
366 this->colourmap = PAL_NONE;
367 this->cargo_age_counter = 1;
368 this->last_station_visited = StationID::Invalid();
369 this->last_loading_station = StationID::Invalid();
370}
371
372/* Size of the hash, 6 = 64 x 64, 7 = 128 x 128. Larger sizes will (in theory) reduce hash
373 * lookup times at the expense of memory usage. */
374const int HASH_BITS = 7;
375const int HASH_SIZE = 1 << HASH_BITS;
376const int HASH_MASK = HASH_SIZE - 1;
377const int TOTAL_HASH_SIZE = 1 << (HASH_BITS * 2);
378const int TOTAL_HASH_MASK = TOTAL_HASH_SIZE - 1;
379
380/* Resolution of the hash, 0 = 1*1 tile, 1 = 2*2 tiles, 2 = 4*4 tiles, etc.
381 * Profiling results show that 0 is fastest. */
382const int HASH_RES = 0;
383
384static std::array<Vehicle *, TOTAL_HASH_SIZE> _vehicle_tile_hash{};
385
386static Vehicle *VehicleFromTileHash(int xl, int yl, int xu, int yu, void *data, VehicleFromPosProc *proc, bool find_first)
387{
388 for (int y = yl; ; y = (y + (1 << HASH_BITS)) & (HASH_MASK << HASH_BITS)) {
389 for (int x = xl; ; x = (x + 1) & HASH_MASK) {
390 Vehicle *v = _vehicle_tile_hash[(x + y) & TOTAL_HASH_MASK];
391 for (; v != nullptr; v = v->hash_tile_next) {
392 Vehicle *a = proc(v, data);
393 if (find_first && a != nullptr) return a;
394 }
395 if (x == xu) break;
396 }
397 if (y == yu) break;
398 }
399
400 return nullptr;
401}
402
403
415static Vehicle *VehicleFromPosXY(int x, int y, void *data, VehicleFromPosProc *proc, bool find_first)
416{
417 const int COLL_DIST = 6;
418
419 /* Hash area to scan is from xl,yl to xu,yu */
420 int xl = GB((x - COLL_DIST) / TILE_SIZE, HASH_RES, HASH_BITS);
421 int xu = GB((x + COLL_DIST) / TILE_SIZE, HASH_RES, HASH_BITS);
422 int yl = GB((y - COLL_DIST) / TILE_SIZE, HASH_RES, HASH_BITS) << HASH_BITS;
423 int yu = GB((y + COLL_DIST) / TILE_SIZE, HASH_RES, HASH_BITS) << HASH_BITS;
424
425 return VehicleFromTileHash(xl, yl, xu, yu, data, proc, find_first);
426}
427
442void FindVehicleOnPosXY(int x, int y, void *data, VehicleFromPosProc *proc)
443{
444 VehicleFromPosXY(x, y, data, proc, false);
445}
446
458bool HasVehicleOnPosXY(int x, int y, void *data, VehicleFromPosProc *proc)
459{
460 return VehicleFromPosXY(x, y, data, proc, true) != nullptr;
461}
462
473static Vehicle *VehicleFromPos(TileIndex tile, void *data, VehicleFromPosProc *proc, bool find_first)
474{
475 int x = GB(TileX(tile), HASH_RES, HASH_BITS);
476 int y = GB(TileY(tile), HASH_RES, HASH_BITS) << HASH_BITS;
477
478 Vehicle *v = _vehicle_tile_hash[(x + y) & TOTAL_HASH_MASK];
479 for (; v != nullptr; v = v->hash_tile_next) {
480 if (v->tile != tile) continue;
481
482 Vehicle *a = proc(v, data);
483 if (find_first && a != nullptr) return a;
484 }
485
486 return nullptr;
487}
488
502void FindVehicleOnPos(TileIndex tile, void *data, VehicleFromPosProc *proc)
503{
504 VehicleFromPos(tile, data, proc, false);
505}
506
517bool HasVehicleOnPos(TileIndex tile, void *data, VehicleFromPosProc *proc)
518{
519 return VehicleFromPos(tile, data, proc, true) != nullptr;
520}
521
528static Vehicle *EnsureNoVehicleProcZ(Vehicle *v, void *data)
529{
530 int z = *(int*)data;
531
532 if (v->type == VEH_DISASTER || (v->type == VEH_AIRCRAFT && v->subtype == AIR_SHADOW)) return nullptr;
533 if (v->z_pos > z) return nullptr;
534
535 return v;
536}
537
544{
545 int z = GetTileMaxPixelZ(tile);
546
547 /* Value v is not safe in MP games, however, it is used to generate a local
548 * error message only (which may be different for different machines).
549 * Such a message does not affect MP synchronisation.
550 */
551 Vehicle *v = VehicleFromPos(tile, &z, &EnsureNoVehicleProcZ, true);
552 if (v != nullptr) return CommandCost(STR_ERROR_TRAIN_IN_THE_WAY + v->type);
553 return CommandCost();
554}
555
558{
559 if (v->type != VEH_TRAIN && v->type != VEH_ROAD && v->type != VEH_SHIP) return nullptr;
560 if (v == (const Vehicle *)data) return nullptr;
561
562 return v;
563}
564
573{
574 /* Value v is not safe in MP games, however, it is used to generate a local
575 * error message only (which may be different for different machines).
576 * Such a message does not affect MP synchronisation.
577 */
578 Vehicle *v = VehicleFromPos(tile, const_cast<Vehicle *>(ignore), &GetVehicleTunnelBridgeProc, true);
579 if (v == nullptr) v = VehicleFromPos(endtile, const_cast<Vehicle *>(ignore), &GetVehicleTunnelBridgeProc, true);
580
581 if (v != nullptr) return CommandCost(STR_ERROR_TRAIN_IN_THE_WAY + v->type);
582 return CommandCost();
583}
584
585static Vehicle *EnsureNoTrainOnTrackProc(Vehicle *v, void *data)
586{
587 TrackBits rail_bits = *(TrackBits *)data;
588
589 if (v->type != VEH_TRAIN) return nullptr;
590
591 Train *t = Train::From(v);
592 if ((t->track != rail_bits) && !TracksOverlap(t->track | rail_bits)) return nullptr;
593
594 return v;
595}
596
606{
607 /* Value v is not safe in MP games, however, it is used to generate a local
608 * error message only (which may be different for different machines).
609 * Such a message does not affect MP synchronisation.
610 */
611 Vehicle *v = VehicleFromPos(tile, &track_bits, &EnsureNoTrainOnTrackProc, true);
612 if (v != nullptr) return CommandCost(STR_ERROR_TRAIN_IN_THE_WAY + v->type);
613 return CommandCost();
614}
615
616static void UpdateVehicleTileHash(Vehicle *v, bool remove)
617{
618 Vehicle **old_hash = v->hash_tile_current;
619 Vehicle **new_hash;
620
621 if (remove) {
622 new_hash = nullptr;
623 } else {
624 int x = GB(TileX(v->tile), HASH_RES, HASH_BITS);
625 int y = GB(TileY(v->tile), HASH_RES, HASH_BITS) << HASH_BITS;
626 new_hash = &_vehicle_tile_hash[(x + y) & TOTAL_HASH_MASK];
627 }
628
629 if (old_hash == new_hash) return;
630
631 /* Remove from the old position in the hash table */
632 if (old_hash != nullptr) {
635 }
636
637 /* Insert vehicle at beginning of the new position in the hash table */
638 if (new_hash != nullptr) {
639 v->hash_tile_next = *new_hash;
640 if (v->hash_tile_next != nullptr) v->hash_tile_next->hash_tile_prev = &v->hash_tile_next;
641 v->hash_tile_prev = new_hash;
642 *new_hash = v;
643 }
644
645 /* Remember current hash position */
646 v->hash_tile_current = new_hash;
647}
648
649static std::array<Vehicle *, 1 << (GEN_HASHX_BITS + GEN_HASHY_BITS)> _vehicle_viewport_hash{};
650
651static void UpdateVehicleViewportHash(Vehicle *v, int x, int y, int old_x, int old_y)
652{
653 Vehicle **old_hash, **new_hash;
654
655 new_hash = (x == INVALID_COORD) ? nullptr : &_vehicle_viewport_hash[GEN_HASH(x, y)];
656 old_hash = (old_x == INVALID_COORD) ? nullptr : &_vehicle_viewport_hash[GEN_HASH(old_x, old_y)];
657
658 if (old_hash == new_hash) return;
659
660 /* remove from hash table? */
661 if (old_hash != nullptr) {
664 }
665
666 /* insert into hash table? */
667 if (new_hash != nullptr) {
668 v->hash_viewport_next = *new_hash;
670 v->hash_viewport_prev = new_hash;
671 *new_hash = v;
672 }
673}
674
675void ResetVehicleHash()
676{
677 for (Vehicle *v : Vehicle::Iterate()) { v->hash_tile_current = nullptr; }
678 _vehicle_viewport_hash.fill(nullptr);
679 _vehicle_tile_hash.fill(nullptr);
680}
681
682void ResetVehicleColourMap()
683{
684 for (Vehicle *v : Vehicle::Iterate()) { v->colourmap = PAL_NONE; }
685}
686
691using AutoreplaceMap = std::map<VehicleID, bool>;
692static AutoreplaceMap _vehicles_to_autoreplace;
693
694void InitializeVehicles()
695{
696 _vehicles_to_autoreplace.clear();
697 ResetVehicleHash();
698}
699
700uint CountVehiclesInChain(const Vehicle *v)
701{
702 uint count = 0;
703 do count++; while ((v = v->Next()) != nullptr);
704 return count;
705}
706
712{
713 switch (this->type) {
714 case VEH_AIRCRAFT: return Aircraft::From(this)->IsNormalAircraft(); // don't count plane shadows and helicopter rotors
715 case VEH_TRAIN:
716 return !this->IsArticulatedPart() && // tenders and other articulated parts
717 !Train::From(this)->IsRearDualheaded(); // rear parts of multiheaded engines
718 case VEH_ROAD: return RoadVehicle::From(this)->IsFrontEngine();
719 case VEH_SHIP: return true;
720 default: return false; // Only count company buildable vehicles
721 }
722}
723
729{
730 switch (this->type) {
731 case VEH_AIRCRAFT: return Aircraft::From(this)->IsNormalAircraft();
732 case VEH_TRAIN:
733 case VEH_ROAD:
734 case VEH_SHIP: return true;
735 default: return false;
736 }
737}
738
745{
746 return Engine::Get(this->engine_type);
747}
748
755{
756 return this->GetEngine()->GetGRF();
757}
758
764uint32_t Vehicle::GetGRFID() const
765{
766 return this->GetEngine()->GetGRFID();
767}
768
774void Vehicle::ShiftDates(TimerGameEconomy::Date interval)
775{
776 this->date_of_last_service = std::max(this->date_of_last_service + interval, TimerGameEconomy::Date(0));
777 /* date_of_last_service_newgrf is not updated here as it must stay stable
778 * for vehicles outside of a depot. */
779}
780
789{
790 if (path_found) {
791 /* Route found, is the vehicle marked with "lost" flag? */
793
794 /* Clear the flag as the PF's problem was solved. */
798 /* Delete the news item. */
800 return;
801 }
802
803 /* Were we already lost? */
805
806 /* It is first time the problem occurred, set the "lost" flag. */
810
811 /* Unbunching data is no longer valid. */
812 this->ResetDepotUnbunching();
813
814 /* Notify user about the event. */
815 AI::NewEvent(this->owner, new ScriptEventVehicleLost(this->index));
817 AddVehicleAdviceNewsItem(AdviceType::VehicleLost, GetEncodedString(STR_NEWS_VEHICLE_IS_LOST, this->index), this->index);
818 }
819}
820
823{
824 if (CleaningPool()) return;
825
828 st->loading_vehicles.remove(this);
829
831 this->CancelReservation(StationID::Invalid(), st);
832 delete this->cargo_payment;
833 assert(this->cargo_payment == nullptr); // cleared by ~CargoPayment
834 }
835
836 if (this->IsEngineCountable()) {
838 if (this->IsPrimaryVehicle()) GroupStatistics::CountVehicle(this, -1);
840
843 }
844
845 Company::Get(this->owner)->freeunits[this->type].ReleaseID(this->unitnumber);
846
847 if (this->type == VEH_AIRCRAFT && this->IsPrimaryVehicle()) {
848 Aircraft *a = Aircraft::From(this);
850 if (st != nullptr) {
851 const auto &layout = st->airport.GetFTA()->layout;
852 st->airport.blocks.Reset(layout[a->previous_pos].blocks | layout[a->pos].blocks);
853 }
854 }
855
856
857 if (this->type == VEH_ROAD && this->IsPrimaryVehicle()) {
859 if (!v->vehstatus.Test(VehState::Crashed) && IsInsideMM(v->state, RVSB_IN_DT_ROAD_STOP, RVSB_IN_DT_ROAD_STOP_END)) {
860 /* Leave the drive through roadstop, when you have not already left it. */
862 }
863
864 if (v->disaster_vehicle != VehicleID::Invalid()) ReleaseDisasterVehicle(v->disaster_vehicle);
865 }
866
867 if (this->Previous() == nullptr) {
869 }
870
871 if (this->IsPrimaryVehicle()) {
879 }
881
882 this->cargo.Truncate();
885
886 StopGlobalFollowVehicle(this);
887}
888
890{
891 if (CleaningPool()) {
892 this->cargo.OnCleanPool();
893 return;
894 }
895
896 /* sometimes, eg. for disaster vehicles, when company bankrupts, when removing crashed/flooded vehicles,
897 * it may happen that vehicle chain is deleted when visible */
899
900 Vehicle *v = this->Next();
901 this->SetNext(nullptr);
902
903 delete v;
904
905 UpdateVehicleTileHash(this, true);
906 UpdateVehicleViewportHash(this, INVALID_COORD, 0, this->sprite_cache.old_coord.left, this->sprite_cache.old_coord.top);
907 if (this->type != VEH_EFFECT) {
910 }
911}
912
918{
919 /* Vehicle should stop in the depot if it was in 'stopping' state */
920 _vehicles_to_autoreplace[v->index] = !v->vehstatus.Test(VehState::Stopped);
921
922 /* We ALWAYS set the stopped state. Even when the vehicle does not plan on
923 * stopping in the depot, so we stop it to ensure that it will not reserve
924 * the path out of the depot before we might autoreplace it to a different
925 * engine. The new engine would not own the reserved path we store that we
926 * stopped the vehicle, so autoreplace can start it again */
928}
929
934{
935 if (_game_mode != GM_NORMAL) return;
936
937 /* Run the calendar day proc for every DAY_TICKS vehicle starting at TimerGameCalendar::date_fract. */
939 Vehicle *v = Vehicle::Get(i);
940 if (v == nullptr) continue;
941 v->OnNewCalendarDay();
942 }
943}
944
951{
952 if (_game_mode != GM_NORMAL) return;
953
954 /* Run the economy day proc for every DAY_TICKS vehicle starting at TimerGameEconomy::date_fract. */
956 Vehicle *v = Vehicle::Get(i);
957 if (v == nullptr) continue;
958
959 /* Call the 32-day callback if needed */
960 if ((v->day_counter & 0x1F) == 0 && v->HasEngineType()) {
961 uint16_t callback = GetVehicleCallback(CBID_VEHICLE_32DAY_CALLBACK, 0, 0, v->engine_type, v);
962 if (callback != CALLBACK_FAILED) {
963 if (HasBit(callback, 0)) {
964 TriggerVehicle(v, VEHICLE_TRIGGER_CALLBACK_32); // Trigger vehicle trigger 10
965 }
966
967 /* After a vehicle trigger, the graphics and properties of the vehicle could change.
968 * Note: MarkDirty also invalidates the palette, which is the meaning of bit 1. So, nothing special there. */
969 if (callback != 0) v->First()->MarkDirty();
970
971 if (callback & ~3) ErrorUnknownCallbackResult(v->GetGRFID(), CBID_VEHICLE_32DAY_CALLBACK, callback);
972 }
973 }
974
975 /* This is called once per day for each vehicle, but not in the first tick of the day */
976 v->OnNewEconomyDay();
977 }
978}
979
980void CallVehicleTicks()
981{
982 _vehicles_to_autoreplace.clear();
983
985
986 {
988 for (Station *st : Station::Iterate()) LoadUnloadStation(st);
989 }
994
995 for (Vehicle *v : Vehicle::Iterate()) {
996 [[maybe_unused]] VehicleID vehicle_index = v->index;
997
998 /* Vehicle could be deleted in this tick */
999 if (!v->Tick()) {
1000 assert(Vehicle::Get(vehicle_index) == nullptr);
1001 continue;
1002 }
1003
1004 assert(Vehicle::Get(vehicle_index) == v);
1005
1006 switch (v->type) {
1007 default: break;
1008
1009 case VEH_TRAIN:
1010 case VEH_ROAD:
1011 case VEH_AIRCRAFT:
1012 case VEH_SHIP: {
1013 Vehicle *front = v->First();
1014
1015 if (v->vcache.cached_cargo_age_period != 0) {
1017 if (--v->cargo_age_counter == 0) {
1018 v->cargo.AgeCargo();
1020 }
1021 }
1022
1023 /* Do not play any sound when crashed */
1024 if (front->vehstatus.Test(VehState::Crashed)) continue;
1025
1026 /* Do not play any sound when in depot or tunnel */
1027 if (v->vehstatus.Test(VehState::Hidden)) continue;
1028
1029 /* Do not play any sound when stopped */
1030 if (front->vehstatus.Test(VehState::Stopped) && (front->type != VEH_TRAIN || front->cur_speed == 0)) continue;
1031
1032 /* Update motion counter for animation purposes. */
1033 v->motion_counter += front->cur_speed;
1034
1035 /* Check vehicle type specifics */
1036 switch (v->type) {
1037 case VEH_TRAIN:
1038 if (!Train::From(v)->IsEngine()) continue;
1039 break;
1040
1041 case VEH_ROAD:
1042 if (!RoadVehicle::From(v)->IsFrontEngine()) continue;
1043 break;
1044
1045 case VEH_AIRCRAFT:
1046 if (!Aircraft::From(v)->IsNormalAircraft()) continue;
1047 break;
1048
1049 default:
1050 break;
1051 }
1052
1053 /* Play a running sound if the motion counter passes 256 (Do we not skip sounds?) */
1054 if (GB(v->motion_counter, 0, 8) < front->cur_speed) PlayVehicleSound(v, VSE_RUNNING);
1055
1056 /* Play an alternating running sound every 16 ticks */
1057 if (GB(v->tick_counter, 0, 4) == 0) {
1058 /* Play running sound when speed > 0 and not braking */
1059 bool running = (front->cur_speed > 0) && !front->vehstatus.Any({VehState::Stopped, VehState::TrainSlowing});
1061 }
1062
1063 break;
1064 }
1065 }
1066 }
1067
1069 for (auto &it : _vehicles_to_autoreplace) {
1070 Vehicle *v = Vehicle::Get(it.first);
1071 /* Autoreplace needs the current company set as the vehicle owner */
1072 cur_company.Change(v->owner);
1073
1074 /* Start vehicle if we stopped them in VehicleEnteredDepotThisTick()
1075 * We need to stop them between VehicleEnteredDepotThisTick() and here or we risk that
1076 * they are already leaving the depot again before being replaced. */
1077 if (it.second) v->vehstatus.Reset(VehState::Stopped);
1078
1079 /* Store the position of the effect as the vehicle pointer will become invalid later */
1080 int x = v->x_pos;
1081 int y = v->y_pos;
1082 int z = v->z_pos;
1083
1088
1089 if (!IsLocalCompany()) continue;
1090
1091 if (res.Succeeded()) {
1092 ShowCostOrIncomeAnimation(x, y, z, res.GetCost());
1093 continue;
1094 }
1095
1096 StringID error_message = res.GetErrorMessage();
1097 if (error_message == STR_ERROR_AUTOREPLACE_NOTHING_TO_DO || error_message == INVALID_STRING_ID) continue;
1098
1099 if (error_message == STR_ERROR_NOT_ENOUGH_CASH_REQUIRES_CURRENCY) error_message = STR_ERROR_AUTOREPLACE_MONEY_LIMIT;
1100
1101 EncodedString headline;
1102 if (error_message == STR_ERROR_TRAIN_TOO_LONG_AFTER_REPLACEMENT) {
1103 headline = GetEncodedString(error_message, v->index);
1104 } else {
1105 headline = GetEncodedString(STR_NEWS_VEHICLE_AUTORENEW_FAILED, v->index, error_message, std::monostate{});
1106 }
1107
1109 }
1110
1111 cur_company.Restore();
1112}
1113
1118static void DoDrawVehicle(const Vehicle *v)
1119{
1120 PaletteID pal = PAL_NONE;
1121
1123
1124 /* Check whether the vehicle shall be transparent due to the game state */
1125 bool shadowed = v->vehstatus.Test(VehState::Shadow);
1126
1127 if (v->type == VEH_EFFECT) {
1128 /* Check whether the vehicle shall be transparent/invisible due to GUI settings.
1129 * However, transparent smoke and bubbles look weird, so always hide them. */
1131 if (to != TO_INVALID && (IsTransparencySet(to) || IsInvisibilitySet(to))) return;
1132 }
1133
1135 for (uint i = 0; i < v->sprite_cache.sprite_seq.count; ++i) {
1136 PaletteID pal2 = v->sprite_cache.sprite_seq.seq[i].pal;
1137 if (!pal2 || v->vehstatus.Test(VehState::Crashed)) pal2 = pal;
1138 AddSortableSpriteToDraw(v->sprite_cache.sprite_seq.seq[i].sprite, pal2, v->x_pos + v->x_offs, v->y_pos + v->y_offs,
1139 v->x_extent, v->y_extent, v->z_extent, v->z_pos, shadowed, v->x_bb_offs, v->y_bb_offs);
1140 }
1142}
1143
1149{
1150 /* The bounding rectangle */
1151 const int l = dpi->left;
1152 const int r = dpi->left + dpi->width;
1153 const int t = dpi->top;
1154 const int b = dpi->top + dpi->height;
1155
1156 /* Border size of MAX_VEHICLE_PIXEL_xy */
1157 const int xb = MAX_VEHICLE_PIXEL_X * ZOOM_BASE;
1158 const int yb = MAX_VEHICLE_PIXEL_Y * ZOOM_BASE;
1159
1160 /* The hash area to scan */
1161 int xl, xu, yl, yu;
1162
1163 if (dpi->width + xb < GEN_HASHX_SIZE) {
1164 xl = GEN_HASHX(l - xb);
1165 xu = GEN_HASHX(r);
1166 } else {
1167 /* scan whole hash row */
1168 xl = 0;
1169 xu = GEN_HASHX_MASK;
1170 }
1171
1172 if (dpi->height + yb < GEN_HASHY_SIZE) {
1173 yl = GEN_HASHY(t - yb);
1174 yu = GEN_HASHY(b);
1175 } else {
1176 /* scan whole column */
1177 yl = 0;
1178 yu = GEN_HASHY_MASK;
1179 }
1180
1181 for (int y = yl;; y = (y + GEN_HASHY_INC) & GEN_HASHY_MASK) {
1182 for (int x = xl;; x = (x + GEN_HASHX_INC) & GEN_HASHX_MASK) {
1183 const Vehicle *v = _vehicle_viewport_hash[x + y]; // already masked & 0xFFF
1184
1185 while (v != nullptr) {
1186
1187 if (!v->vehstatus.Test(VehState::Hidden) &&
1188 l <= v->coord.right + xb &&
1189 t <= v->coord.bottom + yb &&
1190 r >= v->coord.left - xb &&
1191 b >= v->coord.top - yb)
1192 {
1193 /*
1194 * This vehicle can potentially be drawn as part of this viewport and
1195 * needs to be revalidated, as the sprite may not be correct.
1196 */
1198 VehicleSpriteSeq seq;
1199 v->GetImage(v->direction, EIT_ON_MAP, &seq);
1200
1201 if (seq.IsValid() && v->sprite_cache.sprite_seq != seq) {
1202 v->sprite_cache.sprite_seq = seq;
1203 /*
1204 * A sprite change may also result in a bounding box change,
1205 * so we need to update the bounding box again before we
1206 * check to see if the vehicle should be drawn. Note that
1207 * we can't interfere with the viewport hash at this point,
1208 * so we keep the original hash on the assumption there will
1209 * not be a significant change in the top and left coordinates
1210 * of the vehicle.
1211 */
1213
1214 }
1215
1217 }
1218
1219 if (l <= v->coord.right &&
1220 t <= v->coord.bottom &&
1221 r >= v->coord.left &&
1222 b >= v->coord.top) DoDrawVehicle(v);
1223 }
1224
1225 v = v->hash_viewport_next;
1226 }
1227
1228 if (x == xu) break;
1229 }
1230
1231 if (y == yu) break;
1232 }
1233}
1234
1242Vehicle *CheckClickOnVehicle(const Viewport *vp, int x, int y)
1243{
1244 Vehicle *found = nullptr;
1245 uint dist, best_dist = UINT_MAX;
1246
1247 if ((uint)(x -= vp->left) >= (uint)vp->width || (uint)(y -= vp->top) >= (uint)vp->height) return nullptr;
1248
1249 x = ScaleByZoom(x, vp->zoom) + vp->virtual_left;
1250 y = ScaleByZoom(y, vp->zoom) + vp->virtual_top;
1251
1252 /* Border size of MAX_VEHICLE_PIXEL_xy */
1253 const int xb = MAX_VEHICLE_PIXEL_X * ZOOM_BASE;
1254 const int yb = MAX_VEHICLE_PIXEL_Y * ZOOM_BASE;
1255
1256 /* The hash area to scan */
1257 int xl = GEN_HASHX(x - xb);
1258 int xu = GEN_HASHX(x);
1259 int yl = GEN_HASHY(y - yb);
1260 int yu = GEN_HASHY(y);
1261
1262 for (int hy = yl;; hy = (hy + GEN_HASHY_INC) & GEN_HASHY_MASK) {
1263 for (int hx = xl;; hx = (hx + GEN_HASHX_INC) & GEN_HASHX_MASK) {
1264 Vehicle *v = _vehicle_viewport_hash[hx + hy]; // already masked & 0xFFF
1265
1266 while (v != nullptr) {
1268 x >= v->coord.left && x <= v->coord.right &&
1269 y >= v->coord.top && y <= v->coord.bottom) {
1270
1271 dist = std::max(
1272 abs(((v->coord.left + v->coord.right) >> 1) - x),
1273 abs(((v->coord.top + v->coord.bottom) >> 1) - y)
1274 );
1275
1276 if (dist < best_dist) {
1277 found = v;
1278 best_dist = dist;
1279 }
1280 }
1281 v = v->hash_viewport_next;
1282 }
1283 if (hx == xu) break;
1284 }
1285 if (hy == yu) break;
1286 }
1287
1288 return found;
1289}
1290
1296{
1297 v->value -= v->value >> 8;
1299}
1300
1301static const uint8_t _breakdown_chance[64] = {
1302 3, 3, 3, 3, 3, 3, 3, 3,
1303 4, 4, 5, 5, 6, 6, 7, 7,
1304 8, 8, 9, 9, 10, 10, 11, 11,
1305 12, 13, 13, 13, 13, 14, 15, 16,
1306 17, 19, 21, 25, 28, 31, 34, 37,
1307 40, 44, 48, 52, 56, 60, 64, 68,
1308 72, 80, 90, 100, 110, 120, 130, 140,
1309 150, 170, 190, 210, 230, 250, 250, 250,
1310};
1311
1312void CheckVehicleBreakdown(Vehicle *v)
1313{
1314 int rel, rel_old;
1315
1316 /* decrease reliability */
1319 v->reliability = rel = std::max((rel_old = v->reliability) - v->reliability_spd_dec, 0);
1320 if ((rel_old >> 8) != (rel >> 8)) SetWindowDirty(WC_VEHICLE_DETAILS, v->index);
1321 }
1322
1323 if (v->breakdown_ctr != 0 || v->vehstatus.Test(VehState::Stopped) ||
1325 v->cur_speed < 5 || _game_mode == GM_MENU) {
1326 return;
1327 }
1328
1329 uint32_t r = Random();
1330
1331 /* increase chance of failure */
1332 int chance = v->breakdown_chance + 1;
1333 if (Chance16I(1, 25, r)) chance += 25;
1334 v->breakdown_chance = ClampTo<uint8_t>(chance);
1335
1336 /* calculate reliability value to use in comparison */
1337 rel = v->reliability;
1338 if (v->type == VEH_SHIP) rel += 0x6666;
1339
1340 /* reduced breakdowns? */
1341 if (_settings_game.difficulty.vehicle_breakdowns == 1) rel += 0x6666;
1342
1343 /* check if to break down */
1344 if (_breakdown_chance[ClampTo<uint16_t>(rel) >> 10] <= v->breakdown_chance) {
1345 v->breakdown_ctr = GB(r, 16, 6) + 0x3F;
1346 v->breakdown_delay = GB(r, 24, 7) + 0x80;
1347 v->breakdown_chance = 0;
1348 }
1349}
1350
1358{
1359 /* Possible states for Vehicle::breakdown_ctr
1360 * 0 - vehicle is running normally
1361 * 1 - vehicle is currently broken down
1362 * 2 - vehicle is going to break down now
1363 * >2 - vehicle is counting down to the actual breakdown event */
1364 switch (this->breakdown_ctr) {
1365 case 0:
1366 return false;
1367
1368 case 2:
1369 this->breakdown_ctr = 1;
1370
1371 if (this->breakdowns_since_last_service != 255) {
1373 }
1374
1375 if (this->type == VEH_AIRCRAFT) {
1376 /* Aircraft just need this flag, the rest is handled elsewhere */
1378 } else {
1379 this->cur_speed = 0;
1380
1381 if (!PlayVehicleSound(this, VSE_BREAKDOWN)) {
1382 bool train_or_ship = this->type == VEH_TRAIN || this->type == VEH_SHIP;
1383 SndPlayVehicleFx((_settings_game.game_creation.landscape != LandscapeType::Toyland) ?
1386 }
1387
1388 if (!this->vehstatus.Test(VehState::Hidden) && !EngInfo(this->engine_type)->misc_flags.Test(EngineMiscFlag::NoBreakdownSmoke)) {
1390 if (u != nullptr) u->animation_state = this->breakdown_delay * 2;
1391 }
1392 }
1393
1394 this->MarkDirty(); // Update graphics after speed is zeroed
1397
1398 [[fallthrough]];
1399 case 1:
1400 /* Aircraft breakdowns end only when arriving at the airport */
1401 if (this->type == VEH_AIRCRAFT) return false;
1402
1403 /* For trains this function is called twice per tick, so decrease v->breakdown_delay at half the rate */
1404 if ((this->tick_counter & (this->type == VEH_TRAIN ? 3 : 1)) == 0) {
1405 if (--this->breakdown_delay == 0) {
1406 this->breakdown_ctr = 0;
1407 this->MarkDirty();
1409 }
1410 }
1411 return true;
1412
1413 default:
1414 if (!this->current_order.IsType(OT_LOADING)) this->breakdown_ctr--;
1415 return false;
1416 }
1417}
1418
1423void EconomyAgeVehicle(Vehicle *v)
1424{
1426 v->economy_age++;
1428 }
1429}
1430
1436{
1437 if (v->age < CalendarTime::MAX_DATE) v->age++;
1438
1439 if (!v->IsPrimaryVehicle() && (v->type != VEH_TRAIN || !Train::From(v)->IsEngine())) return;
1440
1441 auto age = v->age - v->max_age;
1442 for (int32_t i = 0; i <= 4; i++) {
1444 v->reliability_spd_dec <<= 1;
1445 break;
1446 }
1447 }
1448
1450
1451 /* Don't warn if warnings are disabled */
1453
1454 /* Don't warn about vehicles which are non-primary (e.g., part of an articulated vehicle), don't belong to us, are crashed, or are stopped */
1455 if (v->Previous() != nullptr || v->owner != _local_company || v->vehstatus.Any({VehState::Crashed, VehState::Stopped})) return;
1456
1457 const Company *c = Company::Get(v->owner);
1458 /* Don't warn if a renew is active */
1459 if (c->settings.engine_renew && v->GetEngine()->company_avail.Any()) return;
1460 /* Don't warn if a replacement is active */
1462
1463 StringID str;
1465 str = STR_NEWS_VEHICLE_IS_GETTING_OLD;
1467 str = STR_NEWS_VEHICLE_IS_GETTING_VERY_OLD;
1469 str = STR_NEWS_VEHICLE_IS_GETTING_VERY_OLD_AND;
1470 } else {
1471 return;
1472 }
1473
1475}
1476
1486uint8_t CalcPercentVehicleFilled(const Vehicle *front, StringID *colour)
1487{
1488 int count = 0;
1489 int max = 0;
1490 int cars = 0;
1491 int unloading = 0;
1492 bool loading = false;
1493
1494 bool is_loading = front->current_order.IsType(OT_LOADING);
1495
1496 /* The station may be nullptr when the (colour) string does not need to be set. */
1498 assert(colour == nullptr || (st != nullptr && is_loading));
1499
1500 bool order_no_load = is_loading && (front->current_order.GetLoadType() & OLFB_NO_LOAD);
1501 bool order_full_load = is_loading && (front->current_order.GetLoadType() & OLFB_FULL_LOAD);
1502
1503 /* Count up max and used */
1504 for (const Vehicle *v = front; v != nullptr; v = v->Next()) {
1505 count += v->cargo.StoredCount();
1506 max += v->cargo_cap;
1507 if (v->cargo_cap != 0 && colour != nullptr) {
1508 unloading += v->vehicle_flags.Test(VehicleFlag::CargoUnloading) ? 1 : 0;
1509 loading |= !order_no_load &&
1510 (order_full_load || st->goods[v->cargo_type].HasRating()) &&
1512 cars++;
1513 }
1514 }
1515
1516 if (colour != nullptr) {
1517 if (unloading == 0 && loading) {
1518 *colour = STR_PERCENT_UP;
1519 } else if (unloading == 0 && !loading) {
1520 *colour = STR_PERCENT_NONE;
1521 } else if (cars == unloading || !loading) {
1522 *colour = STR_PERCENT_DOWN;
1523 } else {
1524 *colour = STR_PERCENT_UP_DOWN;
1525 }
1526 }
1527
1528 /* Train without capacity */
1529 if (max == 0) return 100;
1530
1531 /* Return the percentage */
1532 if (count * 2 < max) {
1533 /* Less than 50%; round up, so that 0% means really empty. */
1534 return CeilDiv(count * 100, max);
1535 } else {
1536 /* More than 50%; round down, so that 100% means really full. */
1537 return (count * 100) / max;
1538 }
1539}
1540
1546{
1547 /* Always work with the front of the vehicle */
1548 assert(v == v->First());
1549
1550 switch (v->type) {
1551 case VEH_TRAIN: {
1552 Train *t = Train::From(v);
1554 /* Clear path reservation */
1555 SetDepotReservation(t->tile, false);
1557
1559 t->wait_counter = 0;
1560 t->force_proceed = TFP_NONE;
1561 ClrBit(t->flags, VRF_TOGGLE_REVERSE);
1563 break;
1564 }
1565
1566 case VEH_ROAD:
1568 break;
1569
1570 case VEH_SHIP: {
1572 Ship *ship = Ship::From(v);
1573 ship->state = TRACK_BIT_DEPOT;
1574 ship->UpdateCache();
1575 ship->UpdateViewport(true, true);
1577 break;
1578 }
1579
1580 case VEH_AIRCRAFT:
1583 break;
1584 default: NOT_REACHED();
1585 }
1587
1588 if (v->type != VEH_TRAIN) {
1589 /* Trains update the vehicle list when the first unit enters the depot and calls VehicleEnterDepot() when the last unit enters.
1590 * We only increase the number of vehicles when the first one enters, so we will not need to search for more vehicles in the depot */
1592 }
1594
1596 v->cur_speed = 0;
1597
1599
1600 /* After a vehicle trigger, the graphics and properties of the vehicle could change. */
1601 TriggerVehicle(v, VEHICLE_TRIGGER_DEPOT);
1602 v->MarkDirty();
1603
1605
1606 if (v->current_order.IsType(OT_GOTO_DEPOT)) {
1608
1609 const Order *real_order = v->GetOrder(v->cur_real_order_index);
1610
1611 /* Test whether we are heading for this depot. If not, do nothing.
1612 * Note: The target depot for nearest-/manual-depot-orders is only updated on junctions, but we want to accept every depot. */
1614 real_order != nullptr && !(real_order->GetDepotActionType() & ODATFB_NEAREST_DEPOT) &&
1616 /* We are heading for another depot, keep driving. */
1617 return;
1618 }
1619
1620 if (v->current_order.IsRefit()) {
1621 Backup<CompanyID> cur_company(_current_company, v->owner);
1622 CommandCost cost = std::get<0>(Command<CMD_REFIT_VEHICLE>::Do(DoCommandFlag::Execute, v->index, v->current_order.GetRefitCargo(), 0xFF, false, false, 0));
1623 cur_company.Restore();
1624
1625 if (cost.Failed()) {
1626 _vehicles_to_autoreplace[v->index] = false;
1627 if (v->owner == _local_company) {
1628 /* Notify the user that we stopped the vehicle */
1629 AddVehicleAdviceNewsItem(AdviceType::RefitFailed, GetEncodedString(STR_NEWS_ORDER_REFIT_FAILED, v->index), v->index);
1630 }
1631 } else if (cost.GetCost() != 0) {
1632 v->profit_this_year -= cost.GetCost() << 8;
1633 if (v->owner == _local_company) {
1635 }
1636 }
1637 }
1638
1640 /* Part of orders */
1642 UpdateVehicleTimetable(v, true);
1644 }
1646 /* Vehicles are always stopped on entering depots. Do not restart this one. */
1647 _vehicles_to_autoreplace[v->index] = false;
1648 /* Invalidate last_loading_station. As the link from the station
1649 * before the stop to the station after the stop can't be predicted
1650 * we shouldn't construct it when the vehicle visits the next stop. */
1651 v->last_loading_station = StationID::Invalid();
1652
1653 /* Clear unbunching data. */
1655
1656 /* Announce that the vehicle is waiting to players and AIs. */
1657 if (v->owner == _local_company) {
1658 AddVehicleAdviceNewsItem(AdviceType::VehicleWaiting, GetEncodedString(STR_NEWS_TRAIN_IS_WAITING + v->type, v->index), v->index);
1659 }
1660 AI::NewEvent(v->owner, new ScriptEventVehicleWaitingInDepot(v->index));
1661 }
1662
1663 /* If we've entered our unbunching depot, record the round trip duration. */
1666 if (v->round_trip_time == 0) {
1667 /* This might be our first round trip. */
1668 v->round_trip_time = measured_round_trip;
1669 } else {
1670 /* If we have a previous trip, smooth the effects of outlier trip calculations caused by jams or other interference. */
1671 v->round_trip_time = Clamp(measured_round_trip, (v->round_trip_time / 2), ClampTo<TimerGameTick::Ticks>(v->round_trip_time * 2));
1672 }
1673 }
1674
1676 }
1677}
1678
1679
1685{
1686 UpdateVehicleTileHash(this, false);
1687}
1688
1693void Vehicle::UpdateBoundingBoxCoordinates(bool update_cache) const
1694{
1695 Rect new_coord;
1696 this->sprite_cache.sprite_seq.GetBounds(&new_coord);
1697
1698 Point pt = RemapCoords(this->x_pos + this->x_offs, this->y_pos + this->y_offs, this->z_pos);
1699 new_coord.left += pt.x;
1700 new_coord.top += pt.y;
1701 new_coord.right += pt.x + 2 * ZOOM_BASE;
1702 new_coord.bottom += pt.y + 2 * ZOOM_BASE;
1703
1704 if (update_cache) {
1705 /*
1706 * If the old coordinates are invalid, set the cache to the new coordinates for correct
1707 * behaviour the next time the coordinate cache is checked.
1708 */
1709 this->sprite_cache.old_coord = this->coord.left == INVALID_COORD ? new_coord : this->coord;
1710 }
1711 else {
1712 /* Extend the bounds of the existing cached bounding box so the next dirty window is correct */
1713 this->sprite_cache.old_coord.left = std::min(this->sprite_cache.old_coord.left, this->coord.left);
1714 this->sprite_cache.old_coord.top = std::min(this->sprite_cache.old_coord.top, this->coord.top);
1715 this->sprite_cache.old_coord.right = std::max(this->sprite_cache.old_coord.right, this->coord.right);
1716 this->sprite_cache.old_coord.bottom = std::max(this->sprite_cache.old_coord.bottom, this->coord.bottom);
1717 }
1718
1719 this->coord = new_coord;
1720}
1721
1727{
1728 /* If the existing cache is invalid we should ignore it, as it will be set to the current coords by UpdateBoundingBoxCoordinates */
1729 bool ignore_cached_coords = this->sprite_cache.old_coord.left == INVALID_COORD;
1730
1731 this->UpdateBoundingBoxCoordinates(true);
1732
1733 if (ignore_cached_coords) {
1734 UpdateVehicleViewportHash(this, this->coord.left, this->coord.top, INVALID_COORD, INVALID_COORD);
1735 } else {
1736 UpdateVehicleViewportHash(this, this->coord.left, this->coord.top, this->sprite_cache.old_coord.left, this->sprite_cache.old_coord.top);
1737 }
1738
1739 if (dirty) {
1740 if (ignore_cached_coords) {
1742 } else {
1744 std::min(this->sprite_cache.old_coord.left, this->coord.left),
1745 std::min(this->sprite_cache.old_coord.top, this->coord.top),
1746 std::max(this->sprite_cache.old_coord.right, this->coord.right),
1747 std::max(this->sprite_cache.old_coord.bottom, this->coord.bottom));
1748 }
1749 }
1750}
1751
1756{
1757 this->UpdatePosition();
1758 this->UpdateViewport(true);
1759}
1760
1766{
1767 return ::MarkAllViewportsDirty(this->coord.left, this->coord.top, this->coord.right, this->coord.bottom);
1768}
1769
1776{
1777 static const int8_t _delta_coord[16] = {
1778 -1,-1,-1, 0, 1, 1, 1, 0, /* x */
1779 -1, 0, 1, 1, 1, 0,-1,-1, /* y */
1780 };
1781
1782 int x = v->x_pos + _delta_coord[v->direction];
1783 int y = v->y_pos + _delta_coord[v->direction + 8];
1784
1786 gp.x = x;
1787 gp.y = y;
1788 gp.old_tile = v->tile;
1789 gp.new_tile = TileVirtXY(x, y);
1790 return gp;
1791}
1792
1793static const Direction _new_direction_table[] = {
1794 DIR_N, DIR_NW, DIR_W,
1797};
1798
1799Direction GetDirectionTowards(const Vehicle *v, int x, int y)
1800{
1801 int i = 0;
1802
1803 if (y >= v->y_pos) {
1804 if (y != v->y_pos) i += 3;
1805 i += 3;
1806 }
1807
1808 if (x >= v->x_pos) {
1809 if (x != v->x_pos) i++;
1810 i++;
1811 }
1812
1813 Direction dir = v->direction;
1814
1815 DirDiff dirdiff = DirDifference(_new_direction_table[i], dir);
1816 if (dirdiff == DIRDIFF_SAME) return dir;
1817 return ChangeDir(dir, dirdiff > DIRDIFF_REVERSE ? DIRDIFF_45LEFT : DIRDIFF_45RIGHT);
1818}
1819
1830{
1831 return _tile_type_procs[GetTileType(tile)]->vehicle_enter_tile_proc(v, tile, x, y);
1832}
1833
1840{
1841 for (auto it = std::begin(this->used_bitmap); it != std::end(this->used_bitmap); ++it) {
1842 BitmapStorage available = ~(*it);
1843 if (available == 0) continue;
1844 return static_cast<UnitID>(std::distance(std::begin(this->used_bitmap), it) * BITMAP_SIZE + FindFirstBit(available) + 1);
1845 }
1846 return static_cast<UnitID>(this->used_bitmap.size() * BITMAP_SIZE + 1);
1847}
1848
1855{
1856 if (index == 0 || index == UINT16_MAX) return index;
1857
1858 index--;
1859
1860 size_t slot = index / BITMAP_SIZE;
1861 if (slot >= this->used_bitmap.size()) this->used_bitmap.resize(slot + 1);
1862 SetBit(this->used_bitmap[index / BITMAP_SIZE], index % BITMAP_SIZE);
1863
1864 return index + 1;
1865}
1866
1872{
1873 if (index == 0 || index == UINT16_MAX) return;
1874
1875 index--;
1876
1877 assert(index / BITMAP_SIZE < this->used_bitmap.size());
1878 ClrBit(this->used_bitmap[index / BITMAP_SIZE], index % BITMAP_SIZE);
1879}
1880
1887{
1888 /* Check whether it is allowed to build another vehicle. */
1889 uint max_veh;
1890 switch (type) {
1891 case VEH_TRAIN: max_veh = _settings_game.vehicle.max_trains; break;
1892 case VEH_ROAD: max_veh = _settings_game.vehicle.max_roadveh; break;
1893 case VEH_SHIP: max_veh = _settings_game.vehicle.max_ships; break;
1894 case VEH_AIRCRAFT: max_veh = _settings_game.vehicle.max_aircraft; break;
1895 default: NOT_REACHED();
1896 }
1897
1899 if (c->group_all[type].num_vehicle >= max_veh) return UINT16_MAX; // Currently already at the limit, no room to make a new one.
1900
1901 return c->freeunits[type].NextID();
1902}
1903
1904
1914{
1915 assert(IsCompanyBuildableVehicleType(type));
1916
1917 if (!Company::IsValidID(_local_company)) return false;
1918
1919 UnitID max;
1920 switch (type) {
1921 case VEH_TRAIN:
1922 if (!HasAnyRailTypesAvail(_local_company)) return false;
1924 break;
1925 case VEH_ROAD:
1926 if (!HasAnyRoadTypesAvail(_local_company, (RoadTramType)subtype)) return false;
1928 break;
1929 case VEH_SHIP: max = _settings_game.vehicle.max_ships; break;
1931 default: NOT_REACHED();
1932 }
1933
1934 /* We can build vehicle infrastructure when we may build the vehicle type */
1935 if (max > 0) {
1936 /* Can we actually build the vehicle type? */
1937 for (const Engine *e : Engine::IterateType(type)) {
1938 if (type == VEH_ROAD && GetRoadTramType(e->u.road.roadtype) != (RoadTramType)subtype) continue;
1939 if (e->company_avail.Test(_local_company)) return true;
1940 }
1941 return false;
1942 }
1943
1944 /* We should be able to build infrastructure when we have the actual vehicle type */
1945 for (const Vehicle *v : Vehicle::Iterate()) {
1946 if (v->type == VEH_ROAD && GetRoadTramType(RoadVehicle::From(v)->roadtype) != (RoadTramType)subtype) continue;
1947 if (v->owner == _local_company && v->type == type) return true;
1948 }
1949
1950 return false;
1951}
1952
1953
1961LiveryScheme GetEngineLiveryScheme(EngineID engine_type, EngineID parent_engine_type, const Vehicle *v)
1962{
1963 CargoType cargo_type = v == nullptr ? INVALID_CARGO : v->cargo_type;
1964 const Engine *e = Engine::Get(engine_type);
1965 switch (e->type) {
1966 default: NOT_REACHED();
1967 case VEH_TRAIN:
1968 if (v != nullptr && parent_engine_type != EngineID::Invalid() && (UsesWagonOverride(v) || (v->IsArticulatedPart() && e->u.rail.railveh_type != RAILVEH_WAGON))) {
1969 /* Wagonoverrides use the colour scheme of the front engine.
1970 * Articulated parts use the colour scheme of the first part. (Not supported for articulated wagons) */
1971 engine_type = parent_engine_type;
1972 e = Engine::Get(engine_type);
1973 /* Note: Luckily cargo_type is not needed for engines */
1974 }
1975
1976 if (!IsValidCargoType(cargo_type)) cargo_type = e->GetDefaultCargoType();
1977 if (!IsValidCargoType(cargo_type)) cargo_type = GetCargoTypeByLabel(CT_GOODS); // The vehicle does not carry anything, let's pick some freight cargo
1978 assert(IsValidCargoType(cargo_type));
1979 if (e->u.rail.railveh_type == RAILVEH_WAGON) {
1980 if (!CargoSpec::Get(cargo_type)->is_freight) {
1981 if (parent_engine_type == EngineID::Invalid()) {
1982 return LS_PASSENGER_WAGON_STEAM;
1983 } else {
1984 bool is_mu = EngInfo(parent_engine_type)->misc_flags.Test(EngineMiscFlag::RailIsMU);
1985 switch (RailVehInfo(parent_engine_type)->engclass) {
1986 default: NOT_REACHED();
1987 case EC_STEAM: return LS_PASSENGER_WAGON_STEAM;
1988 case EC_DIESEL: return is_mu ? LS_DMU : LS_PASSENGER_WAGON_DIESEL;
1989 case EC_ELECTRIC: return is_mu ? LS_EMU : LS_PASSENGER_WAGON_ELECTRIC;
1990 case EC_MONORAIL: return LS_PASSENGER_WAGON_MONORAIL;
1991 case EC_MAGLEV: return LS_PASSENGER_WAGON_MAGLEV;
1992 }
1993 }
1994 } else {
1995 return LS_FREIGHT_WAGON;
1996 }
1997 } else {
1998 bool is_mu = e->info.misc_flags.Test(EngineMiscFlag::RailIsMU);
1999
2000 switch (e->u.rail.engclass) {
2001 default: NOT_REACHED();
2002 case EC_STEAM: return LS_STEAM;
2003 case EC_DIESEL: return is_mu ? LS_DMU : LS_DIESEL;
2004 case EC_ELECTRIC: return is_mu ? LS_EMU : LS_ELECTRIC;
2005 case EC_MONORAIL: return LS_MONORAIL;
2006 case EC_MAGLEV: return LS_MAGLEV;
2007 }
2008 }
2009
2010 case VEH_ROAD:
2011 /* Always use the livery of the front */
2012 if (v != nullptr && parent_engine_type != EngineID::Invalid()) {
2013 engine_type = parent_engine_type;
2014 e = Engine::Get(engine_type);
2015 cargo_type = v->First()->cargo_type;
2016 }
2017 if (!IsValidCargoType(cargo_type)) cargo_type = e->GetDefaultCargoType();
2018 if (!IsValidCargoType(cargo_type)) cargo_type = GetCargoTypeByLabel(CT_GOODS); // The vehicle does not carry anything, let's pick some freight cargo
2019 assert(IsValidCargoType(cargo_type));
2020
2021 /* Important: Use Tram Flag of front part. Luckily engine_type refers to the front part here. */
2023 /* Tram */
2024 return IsCargoInClass(cargo_type, CargoClass::Passengers) ? LS_PASSENGER_TRAM : LS_FREIGHT_TRAM;
2025 } else {
2026 /* Bus or truck */
2027 return IsCargoInClass(cargo_type, CargoClass::Passengers) ? LS_BUS : LS_TRUCK;
2028 }
2029
2030 case VEH_SHIP:
2031 if (!IsValidCargoType(cargo_type)) cargo_type = e->GetDefaultCargoType();
2032 if (!IsValidCargoType(cargo_type)) cargo_type = GetCargoTypeByLabel(CT_GOODS); // The vehicle does not carry anything, let's pick some freight cargo
2033 assert(IsValidCargoType(cargo_type));
2034 return IsCargoInClass(cargo_type, CargoClass::Passengers) ? LS_PASSENGER_SHIP : LS_FREIGHT_SHIP;
2035
2036 case VEH_AIRCRAFT:
2037 switch (e->u.air.subtype) {
2038 case AIR_HELI: return LS_HELICOPTER;
2039 case AIR_CTOL: return LS_SMALL_PLANE;
2040 case AIR_CTOL | AIR_FAST: return LS_LARGE_PLANE;
2041 default: NOT_REACHED();
2042 }
2043 }
2044}
2045
2055const Livery *GetEngineLivery(EngineID engine_type, CompanyID company, EngineID parent_engine_type, const Vehicle *v, uint8_t livery_setting)
2056{
2057 const Company *c = Company::Get(company);
2058 LiveryScheme scheme = LS_DEFAULT;
2059
2060 if (livery_setting == LIT_ALL || (livery_setting == LIT_COMPANY && company == _local_company)) {
2061 if (v != nullptr) {
2062 const Group *g = Group::GetIfValid(v->First()->group_id);
2063 if (g != nullptr) {
2064 /* Traverse parents until we find a livery or reach the top */
2065 while (g->livery.in_use == 0 && g->parent != GroupID::Invalid()) {
2066 g = Group::Get(g->parent);
2067 }
2068 if (g->livery.in_use != 0) return &g->livery;
2069 }
2070 }
2071
2072 /* The default livery is always available for use, but its in_use flag determines
2073 * whether any _other_ liveries are in use. */
2074 if (c->livery[LS_DEFAULT].in_use != 0) {
2075 /* Determine the livery scheme to use */
2076 scheme = GetEngineLiveryScheme(engine_type, parent_engine_type, v);
2077 }
2078 }
2079
2080 return &c->livery[scheme];
2081}
2082
2083
2084static PaletteID GetEngineColourMap(EngineID engine_type, CompanyID company, EngineID parent_engine_type, const Vehicle *v)
2085{
2086 PaletteID map = (v != nullptr) ? v->colourmap : PAL_NONE;
2087
2088 /* Return cached value if any */
2089 if (map != PAL_NONE) return map;
2090
2091 const Engine *e = Engine::Get(engine_type);
2092
2093 /* Check if we should use the colour map callback */
2095 uint16_t callback = GetVehicleCallback(CBID_VEHICLE_COLOUR_MAPPING, 0, 0, engine_type, v);
2096 /* Failure means "use the default two-colour" */
2097 if (callback != CALLBACK_FAILED) {
2098 static_assert(PAL_NONE == 0); // Returning 0x4000 (resp. 0xC000) coincidences with default value (PAL_NONE)
2099 map = GB(callback, 0, 14);
2100 /* If bit 14 is set, then the company colours are applied to the
2101 * map else it's returned as-is. */
2102 if (!HasBit(callback, 14)) {
2103 /* Update cache */
2104 if (v != nullptr) const_cast<Vehicle *>(v)->colourmap = map;
2105 return map;
2106 }
2107 }
2108 }
2109
2110 bool twocc = e->info.misc_flags.Test(EngineMiscFlag::Uses2CC);
2111
2112 if (map == PAL_NONE) map = twocc ? (PaletteID)SPR_2CCMAP_BASE : (PaletteID)PALETTE_RECOLOUR_START;
2113
2114 /* Spectator has news shown too, but has invalid company ID - as well as dedicated server */
2115 if (!Company::IsValidID(company)) return map;
2116
2117 const Livery *livery = GetEngineLivery(engine_type, company, parent_engine_type, v, _settings_client.gui.liveries);
2118
2119 map += livery->colour1;
2120 if (twocc) map += livery->colour2 * 16;
2121
2122 /* Update cache */
2123 if (v != nullptr) const_cast<Vehicle *>(v)->colourmap = map;
2124 return map;
2125}
2126
2134{
2135 return GetEngineColourMap(engine_type, company, EngineID::Invalid(), nullptr);
2136}
2137
2144{
2145 if (v->IsGroundVehicle()) {
2146 return GetEngineColourMap(v->engine_type, v->owner, v->GetGroundVehicleCache()->first_engine, v);
2147 }
2148
2149 return GetEngineColourMap(v->engine_type, v->owner, EngineID::Invalid(), v);
2150}
2151
2156{
2157 if (this->IsGroundVehicle()) {
2158 uint16_t &gv_flags = this->GetGroundVehicleFlags();
2159 if (HasBit(gv_flags, GVF_SUPPRESS_IMPLICIT_ORDERS)) {
2160 /* Do not delete orders, only skip them */
2163 InvalidateVehicleOrder(this, 0);
2164 return;
2165 }
2166 }
2167
2168 const Order *order = this->GetOrder(this->cur_implicit_order_index);
2169 while (order != nullptr) {
2170 if (this->cur_implicit_order_index == this->cur_real_order_index) break;
2171
2172 if (order->IsType(OT_IMPLICIT)) {
2174 /* DeleteOrder does various magic with order_indices, so resync 'order' with 'cur_implicit_order_index' */
2175 order = this->GetOrder(this->cur_implicit_order_index);
2176 } else {
2177 /* Skip non-implicit orders, e.g. service-orders */
2178 order = order->next;
2180 }
2181
2182 /* Wrap around */
2183 if (order == nullptr) {
2184 order = this->GetOrder(0);
2185 this->cur_implicit_order_index = 0;
2186 }
2187 }
2188}
2189
2195{
2196 assert(IsTileType(this->tile, MP_STATION) || this->type == VEH_SHIP);
2197
2199 if (this->current_order.IsType(OT_GOTO_STATION) &&
2200 this->current_order.GetDestination() == this->last_station_visited) {
2202
2203 /* Now both order indices point to the destination station, and we can start loading */
2204 this->current_order.MakeLoading(true);
2205 UpdateVehicleTimetable(this, true);
2206
2207 /* Furthermore add the Non Stop flag to mark that this station
2208 * is the actual destination of the vehicle, which is (for example)
2209 * necessary to be known for HandleTrainLoading to determine
2210 * whether the train is lost or not; not marking a train lost
2211 * that arrives at random stations is bad. */
2213
2214 } else {
2215 /* We weren't scheduled to stop here. Insert an implicit order
2216 * to show that we are stopping here.
2217 * While only groundvehicles have implicit orders, e.g. aircraft might still enter
2218 * the 'wrong' terminal when skipping orders etc. */
2219 Order *in_list = this->GetOrder(this->cur_implicit_order_index);
2220 if (this->IsGroundVehicle() &&
2221 (in_list == nullptr || !in_list->IsType(OT_IMPLICIT) ||
2222 in_list->GetDestination() != this->last_station_visited)) {
2223 bool suppress_implicit_orders = HasBit(this->GetGroundVehicleFlags(), GVF_SUPPRESS_IMPLICIT_ORDERS);
2224 /* Do not create consecutive duplicates of implicit orders */
2225 Order *prev_order = this->cur_implicit_order_index > 0 ? this->GetOrder(this->cur_implicit_order_index - 1) : (this->GetNumOrders() > 1 ? this->GetLastOrder() : nullptr);
2226 if (prev_order == nullptr ||
2227 (!prev_order->IsType(OT_IMPLICIT) && !prev_order->IsType(OT_GOTO_STATION)) ||
2228 prev_order->GetDestination() != this->last_station_visited) {
2229
2230 /* Prefer deleting implicit orders instead of inserting new ones,
2231 * so test whether the right order follows later. In case of only
2232 * implicit orders treat the last order in the list like an
2233 * explicit one, except if the overall number of orders surpasses
2234 * IMPLICIT_ORDER_ONLY_CAP. */
2235 int target_index = this->cur_implicit_order_index;
2236 bool found = false;
2237 while (target_index != this->cur_real_order_index || this->GetNumManualOrders() == 0) {
2238 const Order *order = this->GetOrder(target_index);
2239 if (order == nullptr) break; // No orders.
2240 if (order->IsType(OT_IMPLICIT) && order->GetDestination() == this->last_station_visited) {
2241 found = true;
2242 break;
2243 }
2244 target_index++;
2245 if (target_index >= this->orders->GetNumOrders()) {
2246 if (this->GetNumManualOrders() == 0 &&
2248 break;
2249 }
2250 target_index = 0;
2251 }
2252 if (target_index == this->cur_implicit_order_index) break; // Avoid infinite loop.
2253 }
2254
2255 if (found) {
2256 if (suppress_implicit_orders) {
2257 /* Skip to the found order */
2258 this->cur_implicit_order_index = target_index;
2259 InvalidateVehicleOrder(this, 0);
2260 } else {
2261 /* Delete all implicit orders up to the station we just reached */
2262 const Order *order = this->GetOrder(this->cur_implicit_order_index);
2263 while (!order->IsType(OT_IMPLICIT) || order->GetDestination() != this->last_station_visited) {
2264 if (order->IsType(OT_IMPLICIT)) {
2266 /* DeleteOrder does various magic with order_indices, so resync 'order' with 'cur_implicit_order_index' */
2267 order = this->GetOrder(this->cur_implicit_order_index);
2268 } else {
2269 /* Skip non-implicit orders, e.g. service-orders */
2270 order = order->next;
2272 }
2273
2274 /* Wrap around */
2275 if (order == nullptr) {
2276 order = this->GetOrder(0);
2277 this->cur_implicit_order_index = 0;
2278 }
2279 assert(order != nullptr);
2280 }
2281 }
2282 } else if (!suppress_implicit_orders &&
2283 ((this->orders == nullptr ? OrderList::CanAllocateItem() : this->orders->GetNumOrders() < MAX_VEH_ORDER_ID)) &&
2285 /* Insert new implicit order */
2286 Order *implicit_order = new Order();
2287 implicit_order->MakeImplicit(this->last_station_visited);
2288 InsertOrder(this, implicit_order, this->cur_implicit_order_index);
2290
2291 /* InsertOrder disabled creation of implicit orders for all vehicles with the same implicit order.
2292 * Reenable it for this vehicle */
2293 uint16_t &gv_flags = this->GetGroundVehicleFlags();
2295 }
2296 }
2297 }
2298 this->current_order.MakeLoading(false);
2299 }
2300
2301 if (this->last_loading_station != StationID::Invalid() &&
2303 ((this->current_order.GetLoadType() & OLFB_NO_LOAD) == 0 ||
2304 (this->current_order.GetUnloadType() & OUFB_NO_UNLOAD) == 0)) {
2305 IncreaseStats(Station::Get(this->last_loading_station), this, this->last_station_visited, travel_time);
2306 }
2307
2308 PrepareUnload(this);
2309
2314
2316 this->cur_speed = 0;
2317 this->MarkDirty();
2318}
2319
2326{
2327 for (Vehicle *v = this; v != nullptr; v = v->next) {
2330 Debug(misc, 1, "cancelling cargo reservation");
2331 cargo.Return(UINT_MAX, &st->goods[v->cargo_type].GetOrCreateData().cargo, next, v->tile);
2332 }
2333 cargo.KeepAll();
2334 }
2335}
2336
2342{
2343 assert(this->current_order.IsType(OT_LOADING));
2344
2345 delete this->cargo_payment;
2346 assert(this->cargo_payment == nullptr); // cleared by ~CargoPayment
2347
2348 /* Only update the timetable if the vehicle was supposed to stop here. */
2350
2351 if ((this->current_order.GetLoadType() & OLFB_NO_LOAD) == 0 ||
2352 (this->current_order.GetUnloadType() & OUFB_NO_UNLOAD) == 0) {
2353 if (this->current_order.CanLeaveWithCargo(this->last_loading_station != StationID::Invalid())) {
2354 /* Refresh next hop stats to make sure we've done that at least once
2355 * during the stop and that refit_cap == cargo_cap for each vehicle in
2356 * the consist. */
2357 this->ResetRefitCaps();
2358 LinkRefresher::Run(this);
2359
2360 /* if the vehicle could load here or could stop with cargo loaded set the last loading station */
2363 } else {
2364 /* if the vehicle couldn't load and had to unload or transfer everything
2365 * set the last loading station to invalid as it will leave empty. */
2366 this->last_loading_station = StationID::Invalid();
2367 }
2368 }
2369
2372 this->CancelReservation(StationID::Invalid(), st);
2373 st->loading_vehicles.remove(this);
2374
2377
2378 if (this->type == VEH_TRAIN && !this->vehstatus.Test(VehState::Crashed)) {
2379 /* Trigger station animation (trains only) */
2380 if (IsTileType(this->tile, MP_STATION)) {
2382 TriggerStationAnimation(st, this->tile, SAT_TRAIN_DEPARTS);
2383 }
2384
2385 SetBit(Train::From(this)->flags, VRF_LEAVING_STATION);
2386 }
2387 if (this->type == VEH_ROAD && !this->vehstatus.Test(VehState::Crashed)) {
2388 /* Trigger road stop animation */
2389 if (IsStationRoadStopTile(this->tile)) {
2390 TriggerRoadStopRandomisation(st, this->tile, RSRT_VEH_DEPARTS);
2391 TriggerRoadStopAnimation(st, this->tile, SAT_TRAIN_DEPARTS);
2392 }
2393 }
2394
2395
2396 this->MarkDirty();
2397}
2398
2403{
2404 for (Vehicle *v = this; v != nullptr; v = v->Next()) v->refit_cap = v->cargo_cap;
2405}
2406
2411{
2412 Company::Get(this->owner)->freeunits[this->type].ReleaseID(this->unitnumber);
2413 this->unitnumber = 0;
2414}
2415
2422{
2423 switch (this->current_order.GetType()) {
2424 case OT_LOADING: {
2425 TimerGameTick::Ticks wait_time = std::max(this->current_order.GetTimetabledWait() - this->lateness_counter, 0);
2426
2427 /* Not the first call for this tick, or still loading */
2428 if (mode || !this->vehicle_flags.Test(VehicleFlag::LoadingFinished) || this->current_order_time < wait_time) return;
2429
2430 this->PlayLeaveStationSound();
2431
2432 this->LeaveStation();
2433
2434 /* Only advance to next order if we just loaded at the current one */
2435 const Order *order = this->GetOrder(this->cur_implicit_order_index);
2436 if (order == nullptr ||
2437 (!order->IsType(OT_IMPLICIT) && !order->IsType(OT_GOTO_STATION)) ||
2438 order->GetDestination() != this->last_station_visited) {
2439 return;
2440 }
2441 break;
2442 }
2443
2444 case OT_DUMMY: break;
2445
2446 default: return;
2447 }
2448
2450}
2451
2457{
2458 for (Order *o : this->Orders()) {
2459 if (o->IsType(OT_GOTO_STATION) && o->GetLoadType() & (OLFB_FULL_LOAD | OLF_FULL_LOAD_ANY)) return true;
2460 }
2461 return false;
2462}
2463
2469{
2470 for (Order *o : this->Orders()) {
2471 if (o->IsType(OT_CONDITIONAL)) return true;
2472 }
2473 return false;
2474}
2475
2481{
2482 for (Order *o : this->Orders()) {
2483 if (o->IsType(OT_GOTO_DEPOT) && o->GetDepotActionType() & ODATFB_UNBUNCH) return true;
2484 }
2485 return false;
2486}
2487
2493{
2494 /* If we are headed for the first order, we must wrap around back to the last order. */
2495 bool is_first_order = (v->GetOrder(v->cur_implicit_order_index) == v->GetFirstOrder());
2496 Order *previous_order = (is_first_order) ? v->GetLastOrder() : v->GetOrder(v->cur_implicit_order_index - 1);
2497
2498 if (previous_order == nullptr || !previous_order->IsType(OT_GOTO_DEPOT)) return false;
2499 return (previous_order->GetDepotActionType() & ODATFB_UNBUNCH) != 0;
2500}
2501
2506{
2507 /* Don't do anything if this is not our unbunching order. */
2508 if (!PreviousOrderIsUnbunching(this)) return;
2509
2510 /* Set the start point for this round trip time. */
2512
2513 /* Tell the timetable we are now "on time." */
2514 this->lateness_counter = 0;
2516
2517 /* Find the average travel time of vehicles that we share orders with. */
2518 int num_vehicles = 0;
2519 TimerGameTick::Ticks total_travel_time = 0;
2520
2521 Vehicle *u = this->FirstShared();
2522 for (; u != nullptr; u = u->NextShared()) {
2523 /* Ignore vehicles that are manually stopped or crashed. */
2524 if (u->vehstatus.Any({VehState::Stopped, VehState::Crashed})) continue;
2525
2526 num_vehicles++;
2527 total_travel_time += u->round_trip_time;
2528 }
2529
2530 /* Make sure we cannot divide by 0. */
2531 num_vehicles = std::max(num_vehicles, 1);
2532
2533 /* Calculate the separation by finding the average travel time, then calculating equal separation (minimum 1 tick) between vehicles. */
2534 TimerGameTick::Ticks separation = std::max((total_travel_time / num_vehicles / num_vehicles), 1);
2535 TimerGameTick::TickCounter next_departure = TimerGameTick::counter + separation;
2536
2537 /* Set the departure time of all vehicles that we share orders with. */
2538 u = this->FirstShared();
2539 for (; u != nullptr; u = u->NextShared()) {
2540 /* Ignore vehicles that are manually stopped or crashed. */
2541 if (u->vehstatus.Any({VehState::Stopped, VehState::Crashed})) continue;
2542
2543 u->depot_unbunching_next_departure = next_departure;
2545 }
2546}
2547
2553{
2554 assert(this->IsInDepot());
2555
2556 /* Don't bother if there are no vehicles sharing orders. */
2557 if (!this->IsOrderListShared()) return false;
2558
2559 /* Don't do anything if there aren't enough orders. */
2560 if (this->GetNumOrders() <= 1) return false;
2561
2562 /* Don't do anything if this is not our unbunching order. */
2563 if (!PreviousOrderIsUnbunching(this)) return false;
2564
2566};
2567
2575{
2576 CommandCost ret = CheckOwnership(this->owner);
2577 if (ret.Failed()) return ret;
2578
2579 if (this->vehstatus.Test(VehState::Crashed)) return CMD_ERROR;
2580 if (this->IsStoppedInDepot()) return CMD_ERROR;
2581
2582 /* No matter why we're headed to the depot, unbunching data is no longer valid. */
2584
2585 if (this->current_order.IsType(OT_GOTO_DEPOT)) {
2586 bool halt_in_depot = (this->current_order.GetDepotActionType() & ODATFB_HALT) != 0;
2587 if (command.Test(DepotCommandFlag::Service) == halt_in_depot) {
2588 /* We called with a different DEPOT_SERVICE setting.
2589 * Now we change the setting to apply the new one and let the vehicle head for the same depot.
2590 * Note: the if is (true for requesting service == true for ordered to stop in depot) */
2591 if (flags.Test(DoCommandFlag::Execute)) {
2595 }
2596 return CommandCost();
2597 }
2598
2599 if (command.Test(DepotCommandFlag::DontCancel)) return CMD_ERROR; // Requested no cancellation of depot orders
2600 if (flags.Test(DoCommandFlag::Execute)) {
2601 /* If the orders to 'goto depot' are in the orders list (forced servicing),
2602 * then skip to the next order; effectively cancelling this forced service */
2604
2605 if (this->IsGroundVehicle()) {
2606 uint16_t &gv_flags = this->GetGroundVehicleFlags();
2608 }
2609
2610 this->current_order.MakeDummy();
2612 }
2613 return CommandCost();
2614 }
2615
2616 ClosestDepot closestDepot = this->FindClosestDepot();
2617 static const StringID no_depot[] = {STR_ERROR_UNABLE_TO_FIND_ROUTE_TO, STR_ERROR_UNABLE_TO_FIND_LOCAL_DEPOT, STR_ERROR_UNABLE_TO_FIND_LOCAL_DEPOT, STR_ERROR_CAN_T_SEND_AIRCRAFT_TO_HANGAR};
2618 if (!closestDepot.found) return CommandCost(no_depot[this->type]);
2619
2620 if (flags.Test(DoCommandFlag::Execute)) {
2621 if (this->current_order.IsType(OT_LOADING)) this->LeaveStation();
2622
2623 if (this->IsGroundVehicle() && this->GetNumManualOrders() > 0) {
2624 uint16_t &gv_flags = this->GetGroundVehicleFlags();
2626 }
2627
2628 this->SetDestTile(closestDepot.location);
2629 this->current_order.MakeGoToDepot(closestDepot.destination.ToDepotID(), ODTF_MANUAL);
2632
2633 /* If there is no depot in front and the train is not already reversing, reverse automatically (trains only) */
2634 if (this->type == VEH_TRAIN && (closestDepot.reverse ^ HasBit(Train::From(this)->flags, VRF_REVERSING))) {
2636 }
2637
2638 if (this->type == VEH_AIRCRAFT) {
2639 Aircraft *a = Aircraft::From(this);
2640 if (a->state == FLYING && a->targetairport != closestDepot.destination) {
2641 /* The aircraft is now heading for a different hangar than the next in the orders */
2643 }
2644 }
2645 }
2646
2647 return CommandCost();
2648
2649}
2650
2655void Vehicle::UpdateVisualEffect(bool allow_power_change)
2656{
2657 bool powered_before = HasBit(this->vcache.cached_vis_effect, VE_DISABLE_WAGON_POWER);
2658 const Engine *e = this->GetEngine();
2659
2660 /* Evaluate properties */
2661 uint8_t visual_effect;
2662 switch (e->type) {
2663 case VEH_TRAIN: visual_effect = e->u.rail.visual_effect; break;
2664 case VEH_ROAD: visual_effect = e->u.road.visual_effect; break;
2665 case VEH_SHIP: visual_effect = e->u.ship.visual_effect; break;
2666 default: visual_effect = 1 << VE_DISABLE_EFFECT; break;
2667 }
2668
2669 /* Check powered wagon / visual effect callback */
2671 uint16_t callback = GetVehicleCallback(CBID_VEHICLE_VISUAL_EFFECT, 0, 0, this->engine_type, this);
2672
2673 if (callback != CALLBACK_FAILED) {
2674 if (callback >= 0x100 && e->GetGRF()->grf_version >= 8) ErrorUnknownCallbackResult(e->GetGRFID(), CBID_VEHICLE_VISUAL_EFFECT, callback);
2675
2676 callback = GB(callback, 0, 8);
2677 /* Avoid accidentally setting 'visual_effect' to the default value
2678 * Since bit 6 (disable effects) is set anyways, we can safely erase some bits. */
2679 if (callback == VE_DEFAULT) {
2680 assert(HasBit(callback, VE_DISABLE_EFFECT));
2681 SB(callback, VE_TYPE_START, VE_TYPE_COUNT, 0);
2682 }
2683 visual_effect = callback;
2684 }
2685 }
2686
2687 /* Apply default values */
2688 if (visual_effect == VE_DEFAULT ||
2689 (!HasBit(visual_effect, VE_DISABLE_EFFECT) && GB(visual_effect, VE_TYPE_START, VE_TYPE_COUNT) == VE_TYPE_DEFAULT)) {
2690 /* Only train engines have default effects.
2691 * Note: This is independent of whether the engine is a front engine or articulated part or whatever. */
2692 if (e->type != VEH_TRAIN || e->u.rail.railveh_type == RAILVEH_WAGON || !IsInsideMM(e->u.rail.engclass, EC_STEAM, EC_MONORAIL)) {
2693 if (visual_effect == VE_DEFAULT) {
2694 visual_effect = 1 << VE_DISABLE_EFFECT;
2695 } else {
2696 SetBit(visual_effect, VE_DISABLE_EFFECT);
2697 }
2698 } else {
2699 if (visual_effect == VE_DEFAULT) {
2700 /* Also set the offset */
2701 visual_effect = (VE_OFFSET_CENTRE - (e->u.rail.engclass == EC_STEAM ? 4 : 0)) << VE_OFFSET_START;
2702 }
2703 SB(visual_effect, VE_TYPE_START, VE_TYPE_COUNT, e->u.rail.engclass - EC_STEAM + VE_TYPE_STEAM);
2704 }
2705 }
2706
2707 this->vcache.cached_vis_effect = visual_effect;
2708
2709 if (!allow_power_change && powered_before != HasBit(this->vcache.cached_vis_effect, VE_DISABLE_WAGON_POWER)) {
2711 ShowNewGrfVehicleError(this->engine_type, STR_NEWGRF_BROKEN, STR_NEWGRF_BROKEN_POWERED_WAGON, GRFBug::VehPoweredWagon, false);
2712 }
2713}
2714
2715static const int8_t _vehicle_smoke_pos[8] = {
2716 1, 1, 1, 0, -1, -1, -1, 0
2717};
2718
2724{
2726 if (callback == CALLBACK_FAILED) return;
2727
2728 uint count = GB(callback, 0, 2);
2729 bool auto_center = HasBit(callback, 13);
2730 bool auto_rotate = !HasBit(callback, 14);
2731
2732 int8_t l_center = 0;
2733 if (auto_center) {
2734 /* For road vehicles: Compute offset from vehicle position to vehicle center */
2735 if (v->type == VEH_ROAD) l_center = -(int)(VEHICLE_LENGTH - RoadVehicle::From(v)->gcache.cached_veh_length) / 2;
2736 } else {
2737 /* For trains: Compute offset from vehicle position to sprite position */
2738 if (v->type == VEH_TRAIN) l_center = (VEHICLE_LENGTH - Train::From(v)->gcache.cached_veh_length) / 2;
2739 }
2740
2741 Direction l_dir = v->direction;
2742 if (v->type == VEH_TRAIN && HasBit(Train::From(v)->flags, VRF_REVERSE_DIRECTION)) l_dir = ReverseDir(l_dir);
2743 Direction t_dir = ChangeDir(l_dir, DIRDIFF_90RIGHT);
2744
2745 int8_t x_center = _vehicle_smoke_pos[l_dir] * l_center;
2746 int8_t y_center = _vehicle_smoke_pos[t_dir] * l_center;
2747
2748 for (uint i = 0; i < count; i++) {
2749 uint32_t reg = GetRegister(0x100 + i);
2750 uint type = GB(reg, 0, 8);
2751 int8_t x = GB(reg, 8, 8);
2752 int8_t y = GB(reg, 16, 8);
2753 int8_t z = GB(reg, 24, 8);
2754
2755 if (auto_rotate) {
2756 int8_t l = x;
2757 int8_t t = y;
2758 x = _vehicle_smoke_pos[l_dir] * l + _vehicle_smoke_pos[t_dir] * t;
2759 y = _vehicle_smoke_pos[t_dir] * l - _vehicle_smoke_pos[l_dir] * t;
2760 }
2761
2762 if (type >= 0xF0) {
2763 switch (type) {
2764 case 0xF1: CreateEffectVehicleRel(v, x_center + x, y_center + y, z, EV_STEAM_SMOKE); break;
2765 case 0xF2: CreateEffectVehicleRel(v, x_center + x, y_center + y, z, EV_DIESEL_SMOKE); break;
2766 case 0xF3: CreateEffectVehicleRel(v, x_center + x, y_center + y, z, EV_ELECTRIC_SPARK); break;
2767 case 0xFA: CreateEffectVehicleRel(v, x_center + x, y_center + y, z, EV_BREAKDOWN_SMOKE_AIRCRAFT); break;
2768 default: break;
2769 }
2770 }
2771 }
2772}
2773
2779{
2780 assert(this->IsPrimaryVehicle());
2781 bool sound = false;
2782
2783 /* Do not show any smoke when:
2784 * - vehicle smoke is disabled by the player
2785 * - the vehicle is slowing down or stopped (by the player)
2786 * - the vehicle is moving very slowly
2787 */
2789 this->vehstatus.Any({VehState::TrainSlowing, VehState::Stopped}) ||
2790 this->cur_speed < 2) {
2791 return;
2792 }
2793
2794 /* Use the speed as limited by underground and orders. */
2795 uint max_speed = this->GetCurrentMaxSpeed();
2796
2797 if (this->type == VEH_TRAIN) {
2798 const Train *t = Train::From(this);
2799 /* For trains, do not show any smoke when:
2800 * - the train is reversing
2801 * - is entering a station with an order to stop there and its speed is equal to maximum station entering speed
2802 */
2803 if (HasBit(t->flags, VRF_REVERSING) ||
2805 t->cur_speed >= max_speed)) {
2806 return;
2807 }
2808 }
2809
2810 const Vehicle *v = this;
2811
2812 do {
2815 VisualEffectSpawnModel effect_model = VESM_NONE;
2816 if (advanced) {
2817 effect_offset = VE_OFFSET_CENTRE;
2819 if (effect_model >= VESM_END) effect_model = VESM_NONE; // unknown spawning model
2820 } else {
2822 assert(effect_model != (VisualEffectSpawnModel)VE_TYPE_DEFAULT); // should have been resolved by UpdateVisualEffect
2823 static_assert((uint)VESM_STEAM == (uint)VE_TYPE_STEAM);
2824 static_assert((uint)VESM_DIESEL == (uint)VE_TYPE_DIESEL);
2825 static_assert((uint)VESM_ELECTRIC == (uint)VE_TYPE_ELECTRIC);
2826 }
2827
2828 /* Show no smoke when:
2829 * - Smoke has been disabled for this vehicle
2830 * - The vehicle is not visible
2831 * - The vehicle is under a bridge
2832 * - The vehicle is on a depot tile
2833 * - The vehicle is on a tunnel tile
2834 * - The vehicle is a train engine that is currently unpowered */
2835 if (effect_model == VESM_NONE ||
2837 IsBridgeAbove(v->tile) ||
2838 IsDepotTile(v->tile) ||
2839 IsTunnelTile(v->tile) ||
2840 (v->type == VEH_TRAIN &&
2841 !HasPowerOnRail(Train::From(v)->railtype, GetTileRailType(v->tile)))) {
2842 continue;
2843 }
2844
2845 EffectVehicleType evt = EV_END;
2846 switch (effect_model) {
2847 case VESM_STEAM:
2848 /* Steam smoke - amount is gradually falling until vehicle reaches its maximum speed, after that it's normal.
2849 * Details: while vehicle's current speed is gradually increasing, steam plumes' density decreases by one third each
2850 * third of its maximum speed spectrum. Steam emission finally normalises at very close to vehicle's maximum speed.
2851 * REGULATION:
2852 * - instead of 1, 4 / 2^smoke_amount (max. 2) is used to provide sufficient regulation to steam puffs' amount. */
2853 if (GB(v->tick_counter, 0, ((4 >> _settings_game.vehicle.smoke_amount) + ((this->cur_speed * 3) / max_speed))) == 0) {
2854 evt = EV_STEAM_SMOKE;
2855 }
2856 break;
2857
2858 case VESM_DIESEL: {
2859 /* Diesel smoke - thicker when vehicle is starting, gradually subsiding till it reaches its maximum speed
2860 * when smoke emission stops.
2861 * Details: Vehicle's (max.) speed spectrum is divided into 32 parts. When max. speed is reached, chance for smoke
2862 * emission erodes by 32 (1/4). For trains, power and weight come in handy too to either increase smoke emission in
2863 * 6 steps (1000HP each) if the power is low or decrease smoke emission in 6 steps (512 tonnes each) if the train
2864 * isn't overweight. Power and weight contributions are expressed in a way that neither extreme power, nor
2865 * extreme weight can ruin the balance (e.g. FreightWagonMultiplier) in the formula. When the vehicle reaches
2866 * maximum speed no diesel_smoke is emitted.
2867 * REGULATION:
2868 * - up to which speed a diesel vehicle is emitting smoke (with reduced/small setting only until 1/2 of max_speed),
2869 * - in Chance16 - the last value is 512 / 2^smoke_amount (max. smoke when 128 = smoke_amount of 2). */
2870 int power_weight_effect = 0;
2871 if (v->type == VEH_TRAIN) {
2872 power_weight_effect = (32 >> (Train::From(this)->gcache.cached_power >> 10)) - (32 >> (Train::From(this)->gcache.cached_weight >> 9));
2873 }
2874 if (this->cur_speed < (max_speed >> (2 >> _settings_game.vehicle.smoke_amount)) &&
2875 Chance16((64 - ((this->cur_speed << 5) / max_speed) + power_weight_effect), (512 >> _settings_game.vehicle.smoke_amount))) {
2876 evt = EV_DIESEL_SMOKE;
2877 }
2878 break;
2879 }
2880
2881 case VESM_ELECTRIC:
2882 /* Electric train's spark - more often occurs when train is departing (more load)
2883 * Details: Electric locomotives are usually at least twice as powerful as their diesel counterparts, so spark
2884 * emissions are kept simple. Only when starting, creating huge force are sparks more likely to happen, but when
2885 * reaching its max. speed, quarter by quarter of it, chance decreases until the usual 2,22% at train's top speed.
2886 * REGULATION:
2887 * - in Chance16 the last value is 360 / 2^smoke_amount (max. sparks when 90 = smoke_amount of 2). */
2888 if (GB(v->tick_counter, 0, 2) == 0 &&
2889 Chance16((6 - ((this->cur_speed << 2) / max_speed)), (360 >> _settings_game.vehicle.smoke_amount))) {
2890 evt = EV_ELECTRIC_SPARK;
2891 }
2892 break;
2893
2894 default:
2895 NOT_REACHED();
2896 }
2897
2898 if (evt != EV_END && advanced) {
2899 sound = true;
2901 } else if (evt != EV_END) {
2902 sound = true;
2903
2904 /* The effect offset is relative to a point 4 units behind the vehicle's
2905 * front (which is the center of an 8/8 vehicle). Shorter vehicles need a
2906 * correction factor. */
2907 if (v->type == VEH_TRAIN) effect_offset += (VEHICLE_LENGTH - Train::From(v)->gcache.cached_veh_length) / 2;
2908
2909 int x = _vehicle_smoke_pos[v->direction] * effect_offset;
2910 int y = _vehicle_smoke_pos[(v->direction + 2) % 8] * effect_offset;
2911
2912 if (v->type == VEH_TRAIN && HasBit(Train::From(v)->flags, VRF_REVERSE_DIRECTION)) {
2913 x = -x;
2914 y = -y;
2915 }
2916
2917 CreateEffectVehicleRel(v, x, y, 10, evt);
2918 }
2919 } while ((v = v->Next()) != nullptr);
2920
2921 if (sound) PlayVehicleSound(this, VSE_VISUAL_EFFECT);
2922}
2923
2929{
2930 assert(this != next);
2931
2932 if (this->next != nullptr) {
2933 /* We had an old next vehicle. Update the first and previous pointers */
2934 for (Vehicle *v = this->next; v != nullptr; v = v->Next()) {
2935 v->first = this->next;
2936 }
2937 this->next->previous = nullptr;
2938 }
2939
2940 this->next = next;
2941
2942 if (this->next != nullptr) {
2943 /* A new next vehicle. Update the first and previous pointers */
2944 if (this->next->previous != nullptr) this->next->previous->next = nullptr;
2945 this->next->previous = this;
2946 for (Vehicle *v = this->next; v != nullptr; v = v->Next()) {
2947 v->first = this->first;
2948 }
2949 }
2950}
2951
2957void Vehicle::AddToShared(Vehicle *shared_chain)
2958{
2959 assert(this->previous_shared == nullptr && this->next_shared == nullptr);
2960
2961 if (shared_chain->orders == nullptr) {
2962 assert(shared_chain->previous_shared == nullptr);
2963 assert(shared_chain->next_shared == nullptr);
2964 this->orders = shared_chain->orders = new OrderList(nullptr, shared_chain);
2965 }
2966
2967 this->next_shared = shared_chain->next_shared;
2968 this->previous_shared = shared_chain;
2969
2970 shared_chain->next_shared = this;
2971
2972 if (this->next_shared != nullptr) this->next_shared->previous_shared = this;
2973
2974 shared_chain->orders->AddVehicle(this);
2975}
2976
2981{
2982 /* Remember if we were first and the old window number before RemoveVehicle()
2983 * as this changes first if needed. */
2984 bool were_first = (this->FirstShared() == this);
2986
2987 this->orders->RemoveVehicle(this);
2988
2989 if (!were_first) {
2990 /* We are not the first shared one, so only relink our previous one. */
2991 this->previous_shared->next_shared = this->NextShared();
2992 }
2993
2994 if (this->next_shared != nullptr) this->next_shared->previous_shared = this->previous_shared;
2995
2996
2997 if (this->orders->GetNumVehicles() == 1) {
2998 /* When there is only one vehicle, remove the shared order list window. */
3001 } else if (were_first) {
3002 /* If we were the first one, update to the new first one.
3003 * Note: FirstShared() is already the new first */
3004 InvalidateWindowData(GetWindowClassForVehicleType(this->type), vli.ToWindowNumber(), this->FirstShared()->index.base() | (1U << 31));
3005 }
3006
3007 this->next_shared = nullptr;
3008 this->previous_shared = nullptr;
3009}
3010
3011static IntervalTimer<TimerGameEconomy> _economy_vehicles_yearly({TimerGameEconomy::YEAR, TimerGameEconomy::Priority::VEHICLE}, [](auto)
3012{
3013 for (Vehicle *v : Vehicle::Iterate()) {
3014 if (v->IsPrimaryVehicle()) {
3015 /* show warning if vehicle is not generating enough income last 2 years (corresponds to a red icon in the vehicle list) */
3016 Money profit = v->GetDisplayProfitThisYear();
3017 if (v->economy_age >= VEHICLE_PROFIT_MIN_AGE && profit < 0) {
3020 GetEncodedString(TimerGameEconomy::UsingWallclockUnits() ? STR_NEWS_VEHICLE_UNPROFITABLE_PERIOD : STR_NEWS_VEHICLE_UNPROFITABLE_YEAR, v->index, profit),
3021 v->index);
3022 }
3023 AI::NewEvent(v->owner, new ScriptEventVehicleUnprofitable(v->index));
3024 }
3025
3027 v->profit_this_year = 0;
3029 }
3030 }
3036});
3037
3047bool CanVehicleUseStation(EngineID engine_type, const Station *st)
3048{
3049 const Engine *e = Engine::GetIfValid(engine_type);
3050 assert(e != nullptr);
3051
3052 switch (e->type) {
3053 case VEH_TRAIN:
3055
3056 case VEH_ROAD:
3057 /* For road vehicles we need the vehicle to know whether it can actually
3058 * use the station, but if it doesn't have facilities for RVs it is
3059 * certainly not possible that the station can be used. */
3061
3062 case VEH_SHIP:
3064
3065 case VEH_AIRCRAFT:
3068
3069 default:
3070 return false;
3071 }
3072}
3073
3080bool CanVehicleUseStation(const Vehicle *v, const Station *st)
3081{
3082 if (v->type == VEH_ROAD) return st->GetPrimaryRoadStop(RoadVehicle::From(v)) != nullptr;
3083
3084 return CanVehicleUseStation(v->engine_type, st);
3085}
3086
3094{
3095 switch (v->type) {
3096 case VEH_TRAIN:
3097 return STR_ERROR_NO_RAIL_STATION;
3098
3099 case VEH_ROAD: {
3100 const RoadVehicle *rv = RoadVehicle::From(v);
3101 RoadStop *rs = st->GetPrimaryRoadStop(rv->IsBus() ? RoadStopType::Bus : RoadStopType::Truck);
3102
3103 StringID err = rv->IsBus() ? STR_ERROR_NO_BUS_STATION : STR_ERROR_NO_TRUCK_STATION;
3104
3105 for (; rs != nullptr; rs = rs->next) {
3106 /* Articulated vehicles cannot use bay road stops, only drive-through. Make sure the vehicle can actually use this bay stop */
3108 err = STR_ERROR_NO_STOP_ARTICULATED_VEHICLE;
3109 continue;
3110 }
3111
3112 /* Bay stop errors take precedence, but otherwise the vehicle may not be compatible with the roadtype/tramtype of this station tile.
3113 * We give bay stop errors precedence because they are usually a bus sent to a tram station or vice versa. */
3114 if (!HasTileAnyRoadType(rs->xy, rv->compatible_roadtypes) && err != STR_ERROR_NO_STOP_ARTICULATED_VEHICLE) {
3115 err = RoadTypeIsRoad(rv->roadtype) ? STR_ERROR_NO_STOP_COMPATIBLE_ROAD_TYPE : STR_ERROR_NO_STOP_COMPATIBLE_TRAM_TYPE;
3116 continue;
3117 }
3118 }
3119
3120 return err;
3121 }
3122
3123 case VEH_SHIP:
3124 return STR_ERROR_NO_DOCK;
3125
3126 case VEH_AIRCRAFT:
3127 if (!st->facilities.Test(StationFacility::Airport)) return STR_ERROR_NO_AIRPORT;
3128 if (v->GetEngine()->u.air.subtype & AIR_CTOL) {
3129 return STR_ERROR_AIRPORT_NO_PLANES;
3130 } else {
3131 return STR_ERROR_AIRPORT_NO_HELICOPTERS;
3132 }
3133
3134 default:
3135 return INVALID_STRING_ID;
3136 }
3137}
3138
3145{
3146 assert(this->IsGroundVehicle());
3147 if (this->type == VEH_TRAIN) {
3148 return &Train::From(this)->gcache;
3149 } else {
3150 return &RoadVehicle::From(this)->gcache;
3151 }
3152}
3153
3160{
3161 assert(this->IsGroundVehicle());
3162 if (this->type == VEH_TRAIN) {
3163 return &Train::From(this)->gcache;
3164 } else {
3165 return &RoadVehicle::From(this)->gcache;
3166 }
3167}
3168
3175{
3176 assert(this->IsGroundVehicle());
3177 if (this->type == VEH_TRAIN) {
3178 return Train::From(this)->gv_flags;
3179 } else {
3180 return RoadVehicle::From(this)->gv_flags;
3181 }
3182}
3183
3189const uint16_t &Vehicle::GetGroundVehicleFlags() const
3190{
3191 assert(this->IsGroundVehicle());
3192 if (this->type == VEH_TRAIN) {
3193 return Train::From(this)->gv_flags;
3194 } else {
3195 return RoadVehicle::From(this)->gv_flags;
3196 }
3197}
3198
3207void GetVehicleSet(VehicleSet &set, Vehicle *v, uint8_t num_vehicles)
3208{
3209 if (v->type == VEH_TRAIN) {
3210 Train *u = Train::From(v);
3211 /* Only include whole vehicles, so start with the first articulated part */
3212 u = u->GetFirstEnginePart();
3213
3214 /* Include num_vehicles vehicles, not counting articulated parts */
3215 for (; u != nullptr && num_vehicles > 0; num_vehicles--) {
3216 do {
3217 /* Include current vehicle in the selection. */
3218 include(set, u->index);
3219
3220 /* If the vehicle is multiheaded, add the other part too. */
3221 if (u->IsMultiheaded()) include(set, u->other_multiheaded_part->index);
3222
3223 u = u->Next();
3224 } while (u != nullptr && u->IsArticulatedPart());
3225 }
3226 }
3227}
3228
3234{
3235 uint32_t max_weight = 0;
3236
3237 for (const Vehicle *u = this; u != nullptr; u = u->Next()) {
3238 max_weight += u->GetMaxWeight();
3239 }
3240
3241 return max_weight;
3242}
3243
3249{
3250 uint32_t max_weight = GetDisplayMaxWeight();
3251 if (max_weight == 0) return 0;
3252 return GetGroundVehicleCache()->cached_power * 10u / max_weight;
3253}
3254
3262{
3263 while (true) {
3264 if (v1 == nullptr && v2 == nullptr) return true;
3265 if (v1 == nullptr || v2 == nullptr) return false;
3266 if (v1->GetEngine() != v2->GetEngine()) return false;
3267 v1 = v1->GetNextVehicle();
3268 v2 = v2->GetNextVehicle();
3269 }
3270}
3271
3279{
3280 const Order *o1 = v1->GetFirstOrder();
3281 const Order *o2 = v2->GetFirstOrder();
3282 while (true) {
3283 if (o1 == nullptr && o2 == nullptr) return true;
3284 if (o1 == nullptr || o2 == nullptr) return false;
3285 if (!o1->Equals(*o2)) return false;
3286 o1 = o1->next;
3287 o2 = o2->next;
3288 }
3289}
Base functions for all AIs.
Base for aircraft.
void AircraftNextAirportPos_and_Order(Aircraft *v)
set the right pos when heading to other airports after takeoff
Station * GetTargetAirportIfValid(const Aircraft *v)
Returns aircraft's target station if v->target_airport is a valid station with airport.
@ AIR_SHADOW
shadow of the aircraft
Definition aircraft.h:31
void HandleAircraftEnterHangar(Aircraft *v)
Handle Aircraft specific tasks when an Aircraft enters a hangar.
@ FLYING
Vehicle is flying in the air.
Definition airport.h:77
CargoTypes GetCargoTypesOfArticulatedVehicle(const Vehicle *v, CargoType *cargo_type)
Get cargo mask of all cargoes carried by an articulated vehicle.
void GetArticulatedRefitMasks(EngineID engine, bool include_initial_cargo_type, CargoTypes *union_mask, CargoTypes *intersection_mask)
Merges the refit_masks of all articulated parts.
CargoTypes GetCargoTypesOfArticulatedParts(EngineID engine)
Get the cargo mask of the parts of a given engine.
Functions related to articulated vehicles.
Command definitions related to autoreplace.
Functions related to autoreplacing.
EngineID EngineReplacementForCompany(const Company *c, EngineID engine, GroupID group, bool *replace_when_old=nullptr)
Retrieve the engine replacement for the given company and original engine type.
bool EngineHasReplacementForCompany(const Company *c, EngineID engine, GroupID group)
Check if a company has a replacement set up for the given engine.
void InvalidateAutoreplaceWindow(EngineID e, GroupID id_g)
Rebuild the left autoreplace list if an engine is removed or added.
Functions related to the autoreplace GUIs.
Class for backupping variables and making sure they are restored later.
@ PathfinderLost
Vehicle's pathfinder is lost.
@ StopLoading
Don't load anymore during the next load cycle.
@ LoadingFinished
Vehicle has finished loading.
@ CargoUnloading
Vehicle is unloading cargo.
debug_inline constexpr bool HasBit(const T x, const uint8_t y)
Checks if a bit in a value is set.
constexpr T SB(T &x, const uint8_t s, const uint8_t n, const U d)
Set n bits in x starting at bit s to d.
constexpr T SetBit(T &x, const uint8_t y)
Set a bit in a variable.
constexpr uint8_t FindFirstBit(T x)
Search the first set bit in a value.
constexpr bool HasAtMostOneBit(T value)
Test whether value has at most 1 bit set.
debug_inline static constexpr uint GB(const T x, const uint8_t s, const uint8_t n)
Fetch n bits from x, started at bit s.
constexpr T ToggleBit(T &x, const uint8_t y)
Toggles a bit in a variable.
constexpr T ClrBit(T &x, const uint8_t y)
Clears a bit in a variable.
Map accessor functions for bridges.
bool IsBridgeAbove(Tile t)
checks if a bridge is set above the ground of this tile
Definition bridge_map.h:45
uint8_t CargoType
Cargo slots to indicate a cargo type within a game.
Definition cargo_type.h:23
bool IsValidCargoType(CargoType cargo)
Test whether cargo type is not INVALID_CARGO.
Definition cargo_type.h:106
@ Passengers
Passengers.
bool IsCargoInClass(CargoType cargo, CargoClasses cc)
Does cargo c have cargo class cc?
Definition cargotype.h:236
static void NewEvent(CompanyID company, ScriptEvent *event)
Queue a new event for an AI.
Definition ai_core.cpp:235
constexpr bool Test(Tvalue_type value) const
Test if the value-th bit is set.
constexpr Timpl & Set()
Set all bits.
constexpr bool Any(const Timpl &other) const
Test if any of the given values are set.
constexpr Timpl & Reset(Tvalue_type value)
Reset the value-th bit.
void OnCleanPool()
Empty the cargo list, but don't free the cargo packets; the cargo packets are cleaned by CargoPacket'...
@ MTA_LOAD
Load the cargo from the station.
Common return value for all commands.
bool Succeeded() const
Did this command succeed?
Money GetCost() const
The costs as made up to this moment.
bool Failed() const
Did this command fail?
StringID GetErrorMessage() const
Returns the error message of a command.
static constexpr int DAYS_IN_ECONOMY_MONTH
Days in an economy month, when in wallclock timekeeping mode.
Container for an encoded string, created by GetEncodedString.
Enum-as-bit-set wrapper.
UnitID UseID(UnitID index)
Use a unit number.
Definition vehicle.cpp:1854
void ReleaseID(UnitID index)
Release a unit number.
Definition vehicle.cpp:1871
UnitID NextID() const
Find first unused unit number.
Definition vehicle.cpp:1839
bool GRFBugReverse(uint32_t grfid, uint16_t internal_id)
Logs GRF bug - rail vehicle has different length after reversing.
Definition gamelog.cpp:482
An interval timer will fire every interval, and will continue to fire until it is deleted.
Definition timer.h:76
static void Run(Vehicle *v, bool allow_merge=true, bool is_full_loading=false)
Refresh all links the given vehicle will visit.
Definition refresh.cpp:26
static void Reset(PerformanceElement elem)
Store the previous accumulator value and reset for a new cycle of accumulating measurements.
RAII class for measuring simple elements of performance.
static constexpr TimerGameTick::Ticks DAY_TICKS
1 day is 74 ticks; TimerGameCalendar::date_fract used to be uint16_t and incremented by 885.
static Date date
Current date in days (day counter).
static DateFract date_fract
Fractional part of the day.
static constexpr TimerGame< struct Economy >::Date MAX_DATE
The date of the last day of the max year.
static constexpr int DAYS_IN_LEAP_YEAR
sometimes, you need one day more...
static Date date
Current date in days (day counter).
static bool UsingWallclockUnits(bool newgame=false)
Check if we are using wallclock units.
static DateFract date_fract
Fractional part of the day.
uint64_t TickCounter
The type that the tick counter is stored in.
static TickCounter counter
Monotonic counter, in ticks, since start of game.
int32_t Ticks
The type to store ticks in.
static constexpr Date DateAtStartOfYear(Year year)
Calculate the date of the first day of a given year.
CargoList that is used for vehicles.
uint ActionCount(MoveToAction action) const
Returns the amount of cargo designated for a given purpose.
uint Truncate(uint max_move=UINT_MAX)
Truncates the cargo in this list to the given amount.
uint Return(uint max_move, StationCargoList *dest, StationID next_station, TileIndex current_tile)
Returns reserved cargo to the station and removes it from the cache.
void KeepAll()
Marks all cargo in the vehicle as to be kept.
void AgeCargo()
Ages the all cargo in this list.
uint StoredCount() const
Returns sum of cargo on board the vehicle (ie not only reserved).
Functions related to commands.
static const CommandCost CMD_ERROR
Define a default return value for a failed command.
@ Execute
execute the given command
CommandCost CheckOwnership(Owner owner, TileIndex tile)
Check whether the current owner owns something.
void SubtractMoneyFromCompany(const CommandCost &cost)
Subtract money from the _current_company, if the company is valid.
Money GetAvailableMoney(CompanyID company)
Get the amount of money that a company has available, or INT64_MAX if there is no such valid company.
CompanyID _local_company
Company controlled by the human player at this client. Can also be COMPANY_SPECTATOR.
CompanyID _current_company
Company currently doing an action.
Functions related to companies.
bool IsLocalCompany()
Is the current company the local company?
Some simple functions to help with accessing containers.
bool include(Container &container, typename Container::const_reference &item)
Helper function to append an item to a container if it is not already contained.
#define Debug(category, level, format_string,...)
Output a line of debugging information.
Definition debug.h:37
Functions related to depots.
void DeleteDepotHighlightOfVehicle(const Vehicle *v)
Removes the highlight of a vehicle in a depot window.
Map related accessors for depots.
bool IsDepotTile(Tile tile)
Is the given tile a tile with a depot on it?
Definition depot_map.h:42
DirDiff DirDifference(Direction d0, Direction d1)
Calculate the difference between two directions.
Direction ReverseDir(Direction d)
Return the reverse of a direction.
Direction ChangeDir(Direction d, DirDiff delta)
Change a direction by a given difference.
DirDiff
Allow incrementing of Direction variables.
@ DIRDIFF_45LEFT
Angle of 45 degrees left.
@ DIRDIFF_REVERSE
One direction is the opposite of the other one.
@ DIRDIFF_45RIGHT
Angle of 45 degrees right.
@ DIRDIFF_SAME
Both directions faces to the same direction.
@ DIRDIFF_90RIGHT
Angle of 90 degrees right.
Direction
Defines the 8 directions on the map.
@ DIR_SW
Southwest.
@ DIR_NW
Northwest.
@ DIR_N
North.
@ DIR_SE
Southeast.
@ DIR_S
South.
@ DIR_NE
Northeast.
@ DIR_W
West.
@ DIR_E
East.
@ INVALID_DIAGDIR
Flag for an invalid DiagDirection.
void ReleaseDisasterVehicle(VehicleID vehicle)
Notify disasters that we are about to delete a vehicle.
void LoadUnloadStation(Station *st)
Load/unload the vehicles in this station according to the order they entered.
Definition economy.cpp:1933
void PrepareUnload(Vehicle *front_v)
Prepare the vehicle to be unloaded.
Definition economy.cpp:1261
Base classes related to the economy.
@ EXPENSES_NEW_VEHICLES
New vehicles.
EffectVehicle * CreateEffectVehicleRel(const Vehicle *v, int x, int y, int z, EffectVehicleType type)
Create an effect vehicle above a particular vehicle.
Base class for all effect vehicles.
Functions related to effect vehicles.
EffectVehicleType
Effect vehicle types.
@ EV_BREAKDOWN_SMOKE
Smoke of broken vehicles except aircraft.
@ EV_STEAM_SMOKE
Smoke of steam engines.
@ EV_DIESEL_SMOKE
Smoke of diesel engines.
@ EV_BREAKDOWN_SMOKE_AIRCRAFT
Smoke of broken aircraft.
@ EV_ELECTRIC_SPARK
Sparcs of electric engines.
@ AIR_CTOL
Conventional Take Off and Landing, i.e. planes.
Definition engine_type.h:99
@ RoadIsTram
Road vehicle is a tram/light rail vehicle.
@ NoBreakdownSmoke
Do not show black smoke during a breakdown.
@ Uses2CC
Vehicle uses two company colours.
@ RailIsMU
Rail vehicle is a multiple-unit (DMU/EMU)
@ EC_DIESEL
Diesel rail engine.
Definition engine_type.h:39
@ EC_STEAM
Steam rail engine.
Definition engine_type.h:38
@ EC_MAGLEV
Maglev engine.
Definition engine_type.h:42
@ EC_ELECTRIC
Electric rail engine.
Definition engine_type.h:40
@ EC_MONORAIL
Mono rail engine.
Definition engine_type.h:41
@ RAILVEH_WAGON
simple wagon, not motorized
Definition engine_type.h:33
Functions related to errors.
@ WL_CRITICAL
Critical errors, the MessageBox is shown in all cases.
Definition error.h:27
void ShowErrorMessage(EncodedString &&summary_msg, int x, int y, CommandCost &cc)
Display an error message in a window.
Types for recording game performance data.
@ PFE_GL_SHIPS
Time spent processing ships.
@ PFE_GL_AIRCRAFT
Time spent processing aircraft.
@ PFE_GL_ECONOMY
Time spent processing cargo movement.
@ PFE_GL_ROADVEHS
Time spend processing road vehicles.
@ PFE_GL_TRAINS
Time spent processing trains.
Gamelog _gamelog
Gamelog instance.
Definition gamelog.cpp:31
Functions to be called to log fundamental changes to the game.
void DrawSprite(SpriteID img, PaletteID pal, int x, int y, const SubSprite *sub, ZoomLevel zoom)
Draw a sprite, not in a viewport.
Definition gfx.cpp:989
@ Normal
The most basic (normal) sprite.
uint32_t PaletteID
The number of the palette.
Definition gfx_type.h:18
@ GVF_SUPPRESS_IMPLICIT_ORDERS
Disable insertion and removal of automatic orders until the vehicle completes the real order.
void MarkTilesDirty(bool cargo_change) const
Marks the tiles of the station as dirty.
Definition station.cpp:246
void MarkTileDirtyByTile(TileIndex tile, int bridge_level_offset, int tile_height_override)
Mark a tile given by its index dirty for repaint.
virtual void MarkDirty()
Marks the vehicles to be redrawn and updates cached variables.
void DeleteGroupHighlightOfVehicle(const Vehicle *v)
Removes the highlight of a vehicle in a group window.
Functions/definitions that have something to do with groups.
static constexpr GroupID DEFAULT_GROUP
Ungrouped vehicles are in this group.
Definition group_type.h:18
const TileTypeProcs *const _tile_type_procs[16]
Tile callback functions for each type of tile.
Definition landscape.cpp:65
Point RemapCoords(int x, int y, int z)
Map 3D world or tile coordinate to equivalent 2D coordinate as used in the viewports and smallmap.
Definition landscape.h:79
@ Random
Randomise borders.
Declaration of link graph classes used for cargo distribution.
static const uint8_t LIT_ALL
Show the liveries of all companies.
Definition livery.h:18
LiveryScheme
List of different livery schemes.
Definition livery.h:21
static const uint8_t LIT_COMPANY
Show the liveries of your own company.
Definition livery.h:17
static debug_inline uint TileY(TileIndex tile)
Get the Y component of a tile.
Definition map_func.h:424
static debug_inline uint TileX(TileIndex tile)
Get the X component of a tile.
Definition map_func.h:414
static debug_inline TileIndex TileVirtXY(uint x, uint y)
Get a tile from the virtual XY-coordinate.
Definition map_func.h:403
constexpr bool IsInsideMM(const size_t x, const size_t min, const size_t max) noexcept
Checks if a value is in an interval.
constexpr T abs(const T a)
Returns the absolute value of (scalar) variable.
Definition math_func.hpp:23
constexpr uint CeilDiv(uint a, uint b)
Computes ceil(a / b) for non-negative a and b.
constexpr T Clamp(const T a, const T min, const T max)
Clamp a value between an interval.
Definition math_func.hpp:79
Miscellaneous command definitions.
void HideFillingPercent(TextEffectID *te_id)
Hide vehicle loading indicators.
Definition misc_gui.cpp:584
void ShowCostOrIncomeAnimation(int x, int y, int z, Money cost)
Display animated income or costs on the map.
Definition misc_gui.cpp:511
bool _networking
are we in networking mode?
Definition network.cpp:67
Basic functions/variables used all over the place.
@ SAT_TRAIN_DEPARTS
Trigger platform when train leaves.
@ VisualEffect
Visual effects and wagon power (trains, road vehicles and ships)
@ ColourRemap
Change colour mapping of vehicle.
@ CBID_VEHICLE_SPAWN_VISUAL_EFFECT
Called to spawn visual effects for vehicles.
@ CBID_VEHICLE_COLOUR_MAPPING
Called to determine if a specific colour map should be used for a vehicle instead of the default live...
@ CBID_VEHICLE_32DAY_CALLBACK
Called for every vehicle every 32 days (not all on same date though).
@ CBID_VEHICLE_VISUAL_EFFECT
Visual effects and wagon power.
static const uint CALLBACK_FAILED
Different values for Callback result evaluations.
void ErrorUnknownCallbackResult(uint32_t grfid, uint16_t cbid, uint16_t cb_res)
Record that a NewGRF returned an unknown/invalid callback result.
GRFConfig * GetGRFConfig(uint32_t grfid, uint32_t mask)
Retrieve a NewGRF from the current config by its grfid.
GRFBug
Encountered GRF bugs.
@ VehPoweredWagon
Powered wagon changed poweredness state when not inside a depot.
@ VehLength
Length of rail vehicle changes when not inside a depot.
Functions/types related to NewGRF debugging.
GrfSpecFeature GetGrfSpecFeature(TileIndex tile)
Get the GrfSpecFeature associated with the tile.
void DeleteNewGRFInspectWindow(GrfSpecFeature feature, uint index)
Delete inspect window for a given feature and index.
bool UsesWagonOverride(const Vehicle *v)
Check if a wagon is currently using a wagon override.
uint16_t GetVehicleCallback(CallbackID callback, uint32_t param1, uint32_t param2, EngineID engine, const Vehicle *v)
Evaluate a newgrf callback for vehicles.
NewGRF definitions and structures for road stops.
@ RSRT_VEH_DEPARTS
Trigger roadstop when road vehicle leaves.
bool PlayVehicleSound(const Vehicle *v, VehicleSoundEvent event, bool force)
Checks whether a NewGRF wants to play a different vehicle sound effect.
Functions related to NewGRF provided sounds.
@ VSE_VISUAL_EFFECT
Vehicle visual effect (steam, diesel smoke or electric spark) is shown.
@ VSE_RUNNING
Vehicle running normally.
@ VSE_STOPPED_16
Every 16 ticks while the vehicle is stopped (speed == 0).
@ VSE_RUNNING_16
Every 16 ticks while the vehicle is running (speed > 0).
@ VSE_BREAKDOWN
Vehicle breaking down.
uint32_t GetRegister(uint i)
Gets the value of a so-called newgrf "register".
void TriggerStationRandomisation(Station *st, TileIndex trigger_tile, StationRandomTrigger trigger, CargoType cargo_type)
Trigger station randomisation.
Header file for NewGRF stations.
@ SRT_TRAIN_DEPARTS
Trigger platform when train leaves.
Functions related to news.
void AddVehicleAdviceNewsItem(AdviceType advice_type, EncodedString &&headline, VehicleID vehicle)
Adds a vehicle-advice news item.
Definition news_func.h:40
void DeleteVehicleNews(VehicleID vid, AdviceType advice_type=AdviceType::Invalid)
Delete news with a given advice type about a vehicle.
@ VehicleLost
The vehicle has become lost.
@ VehicleWaiting
The vehicle is waiting in the depot.
@ AutorenewFailed
Autorenew or autoreplace failed.
@ VehicleOld
The vehicle is starting to get old.
@ VehicleUnprofitable
The vehicle is costing you money.
@ RefitFailed
The refit order failed to execute.
@ Error
A game paused because a (critical) error.
@ Normal
A game normally paused.
Functions related to order backups.
void InsertOrder(Vehicle *v, Order *new_o, VehicleOrderID sel_ord)
Insert a new order but skip the validation.
void DeleteOrder(Vehicle *v, VehicleOrderID sel_ord)
Delete an order but skip the parameter validation.
void InvalidateVehicleOrder(const Vehicle *v, int data)
Updates the widgets of a vehicle which contains the order-data.
void DeleteVehicleOrders(Vehicle *v, bool keep_orderlist, bool reset_order_indices)
Delete all orders from a vehicle.
@ OUFB_NO_UNLOAD
Totally no unloading will be done.
Definition order_type.h:72
@ ODTFB_PART_OF_ORDERS
This depot order is because of a regular order.
Definition order_type.h:112
@ ODTFB_SERVICE
This depot order is because of the servicing limit.
Definition order_type.h:111
@ ODTF_MANUAL
Manually initiated order.
Definition order_type.h:110
@ ODATFB_UNBUNCH
Service the vehicle and then unbunch it.
Definition order_type.h:122
@ ODATFB_NEAREST_DEPOT
Send the vehicle to the nearest depot.
Definition order_type.h:121
@ ODATFB_HALT
Service the vehicle and then halt it.
Definition order_type.h:120
@ ODATF_SERVICE_ONLY
Only service the vehicle.
Definition order_type.h:119
static const VehicleOrderID MAX_VEH_ORDER_ID
Last valid VehicleOrderID.
Definition order_type.h:42
@ OLFB_FULL_LOAD
Full load all cargoes of the consist.
Definition order_type.h:80
@ OLFB_NO_LOAD
Do not load anything.
Definition order_type.h:82
@ OLF_FULL_LOAD_ANY
Full load a single cargo of the consist.
Definition order_type.h:81
static const uint IMPLICIT_ORDER_ONLY_CAP
Maximum number of orders in implicit-only lists before we start searching harder for duplicates.
Definition order_type.h:48
@ ONSF_NO_STOP_AT_ANY_STATION
The vehicle will not stop at any stations it passes including the destination.
Definition order_type.h:92
@ ONSF_STOP_EVERYWHERE
The vehicle will stop at any station it passes and the destination.
Definition order_type.h:89
Some methods of Pool are placed here in order to reduce compilation time and binary size.
#define INSTANTIATE_POOL_METHODS(name)
Force instantiation of pool methods so we don't get linker errors.
bool HasAnyRailTypesAvail(const CompanyID company)
Test if any buildable railtype is available for a company.
Definition rail.cpp:196
RailType GetTileRailType(Tile tile)
Return the rail type of tile, or INVALID_RAILTYPE if this is no rail tile.
Definition rail.cpp:155
bool HasPowerOnRail(RailType enginetype, RailType tiletype)
Checks if an engine of the given RailType got power on a tile with a given RailType.
Definition rail.h:328
void SetDepotReservation(Tile t, bool b)
Set the reservation state of the depot.
Definition rail_map.h:269
Pseudo random number generator.
uint32_t RandomRange(uint32_t limit, const std::source_location location=std::source_location::current())
Pick a random number between 0 and limit - 1, inclusive.
bool Chance16I(const uint32_t a, const uint32_t b, const uint32_t r)
Checks if a given randomize-number is below a given probability.
bool Chance16(const uint32_t a, const uint32_t b, const std::source_location location=std::source_location::current())
Flips a coin with given probability.
Definition of link refreshing utility.
bool HasAnyRoadTypesAvail(CompanyID company, RoadTramType rtt)
Test if any buildable RoadType is available for a company.
Definition road.cpp:140
bool HasTileAnyRoadType(Tile t, RoadTypes rts)
Check if a tile has one of the specified road types.
Definition road_map.h:222
Base class for roadstops.
Road vehicle states.
@ RVSB_IN_DT_ROAD_STOP
The vehicle is in a drive-through road stop.
Definition roadveh.h:51
A number of safeguards to prevent using unsafe methods.
GameSettings _settings_game
Game settings of a running game or the scenario editor.
Definition settings.cpp:59
ClientSettings _settings_client
The current settings for this game.
Definition settings.cpp:58
Base for ships.
SigSegState UpdateSignalsOnSegment(TileIndex tile, DiagDirection side, Owner owner)
Update signals, starting at one side of a tile Will check tile next to this at opposite side too.
Definition signal.cpp:647
Functions related to sound.
@ SND_3A_BREAKDOWN_TRAIN_SHIP_TOYLAND
58 == 0x3A Breakdown: train or ship (toyland)
Definition sound_type.h:105
@ SND_10_BREAKDOWN_TRAIN_SHIP
14 == 0x0E Breakdown: train or ship (non-toyland)
Definition sound_type.h:61
@ SND_0F_BREAKDOWN_ROADVEHICLE
13 == 0x0D Breakdown: road vehicle (non-toyland)
Definition sound_type.h:60
@ SND_35_BREAKDOWN_ROADVEHICLE_TOYLAND
53 == 0x35 Breakdown: road vehicle (toyland)
Definition sound_type.h:100
Functions to cache sprites in memory.
static const PaletteID PALETTE_RECOLOUR_START
First recolour sprite for company colours.
Definition sprites.h:1578
static const PaletteID PALETTE_CRASH
Recolour sprite greying of crashed vehicles.
Definition sprites.h:1609
Base classes/functions for stations.
void IncreaseStats(Station *st, CargoType cargo, StationID next_station_id, uint capacity, uint usage, uint32_t time, EdgeUpdateModes modes)
Increase capacity for a link stat given by station cargo and next hop.
bool IsStationRoadStopTile(Tile t)
Is tile t a road stop station?
bool IsBayRoadStopTile(Tile t)
Is tile t a bay (non-drive through) road stop station?
bool IsRailStationTile(Tile t)
Is this tile a station tile and a rail station?
StationID GetStationIndex(Tile t)
Get StationID from a tile.
Definition station_map.h:28
RoadStopType GetRoadStopType(Tile t)
Get the road stop type of this tile.
Definition station_map.h:56
@ Bus
A standard stop for buses.
@ Truck
A standard stop for trucks.
@ Dock
Station with a dock.
@ TruckStop
Station with truck stops.
@ Train
Station with train station.
@ Airport
Station with an airport.
@ BusStop
Station with bus stops.
Definition of base types and functions in a cross-platform compatible way.
static void StrMakeValid(T &dst, const char *str, const char *last, StringValidationSettings settings)
Copies the valid (UTF-8) characters from str up to last to the dst.
Definition string.cpp:125
EncodedString GetEncodedString(StringID str)
Encode a string with no parameters into an encoded string.
Definition strings.cpp:90
std::string GetString(StringID string)
Resolve the given StringID into a std::string with formatting but no parameters.
Definition strings.cpp:426
Functions related to OTTD's strings.
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
static const StringID INVALID_STRING_ID
Constant representing an invalid string (16bit in case it is used in savegames)
uint8_t subtype
Type of aircraft.
Aircraft, helicopters, rotors and their shadows belong to this class.
Definition aircraft.h:72
uint8_t pos
Next desired position of the aircraft.
Definition aircraft.h:74
uint8_t state
State of the airport.
Definition aircraft.h:77
bool IsNormalAircraft() const
Check if the aircraft type is a normal flying device; eg not a rotor or a shadow.
Definition aircraft.h:121
uint8_t previous_pos
Previous desired position of the aircraft.
Definition aircraft.h:75
StationID targetairport
Airport to go to next.
Definition aircraft.h:76
@ Airplanes
Can planes land on this airport type?
@ Helicopters
Can helicopters land on this airport type?
std::vector< AirportFTA > layout
state machine for airport
Definition airport.h:190
Flags flags
Flags for this airport type.
Definition airport.h:193
AirportBlocks blocks
stores which blocks on the airport are taken. was 16 bit earlier on, then 32
const AirportFTAClass * GetFTA() const
Get the finite-state machine for this airport or the finite-state machine for the dummy airport in ca...
Class to backup a specific variable and restore it later.
void Restore()
Restore the variable.
TimerGameTick::TickCounter depot_unbunching_next_departure
When the vehicle will next try to leave its unbunching depot.
VehicleOrderID cur_real_order_index
The index to the current real (non-implicit) order.
TimerGameTick::Ticks round_trip_time
How many ticks for a single circumnavigation of the orders.
VehicleFlags vehicle_flags
Used for gradual loading and other miscellaneous things (.
TimerGameTick::TickCounter depot_unbunching_last_departure
When the vehicle last left its unbunching depot.
VehicleOrderID cur_implicit_order_index
The index to the current implicit order.
TimerGameTick::Ticks lateness_counter
How many ticks late (or early if negative) this vehicle is.
void ResetDepotUnbunching()
Resets all the data used for depot unbunching.
StationFacilities facilities
The facilities that this station has.
VehicleType type
Type of vehicle.
static CargoSpec * Get(size_t index)
Retrieve cargo details for the given cargo type.
Definition cargotype.h:137
GUISettings gui
settings related to the GUI
Structure to return information about the closest depot location, and whether it could be found.
DestinationID destination
The DestinationID as used for orders.
CompanySettings settings
settings specific for each company
uint32_t engine_renew_money
minimum amount of money before autorenew is used
int16_t engine_renew_months
months before/after the maximum vehicle age a vehicle should be renewed
bool engine_renew
is autorenew enabled
std::array< GroupStatistics, VEH_COMPANY_END > group_all
NOSAVE: Statistics for the ALL_GROUP group.
uint8_t vehicle_breakdowns
likelihood of vehicles breaking down
Data about how and where to blit pixels.
Definition gfx_type.h:158
A special vehicle is one of the following:
TransparencyOption GetTransparencyOption() const
Determines the transparency option affecting the effect.
uint16_t animation_state
State primarily used to change the graphics/behaviour.
EngineMiscFlags misc_flags
Miscellaneous flags.
VehicleCallbackMasks callback_mask
Bitmask of vehicle callbacks that have to be called.
uint32_t GetGRFID() const
Retrieve the GRF ID of the NewGRF the engine is tied to.
Definition engine.cpp:157
const GRFFile * GetGRF() const
Retrieve the NewGRF the engine is tied to.
static Pool::IterateWrapperFiltered< Engine, EngineTypeFilter > IterateType(VehicleType vt, size_t from=0)
Returns an iterable ensemble of all valid engines of the given type.
CompanyMask company_avail
Bit for each company whether the engine is available for that company.
Definition engine_base.h:39
VehicleType type
Vehicle type, ie VEH_ROAD, VEH_TRAIN, etc.
Definition engine_base.h:61
VariableGRFFileProps grf_prop
Properties related the the grf file.
Definition engine_base.h:83
CargoType GetDefaultCargoType() const
Determines the default cargo type of an engine.
uint16_t reliability
Current reliability of the engine.
Definition engine_base.h:48
Information about GRF, used in the game and (part of it) in savegames.
GRFBugs grf_bugs
NOSAVE: bugs in this GRF in this run,.
std::string GetName() const
Get the name of this grf.
uint16_t local_id
id defined by the grf file for this entity
uint32_t grfid
grfid that introduced this entity.
Dynamic data of a loaded NewGRF.
Definition newgrf.h:113
bool lost_vehicle_warn
if a vehicle can't find its destination, show a warning
bool vehicle_income_warn
if a vehicle isn't generating income, show a warning
bool show_track_reservation
highlight reserved tracks.
uint8_t liveries
options for displaying company liveries, 0=none, 1=self, 2=all
bool old_vehicle_warn
if a vehicle is getting old, show a warning
LandscapeType landscape
the landscape we're currently in
DifficultySettings difficulty
settings related to the difficulty
GameCreationSettings game_creation
settings used during the creation of a game (map)
VehicleSettings vehicle
options for vehicles
OrderSettings order
settings related to orders
Position information of a vehicle after it moved.
TileIndex new_tile
Tile of the vehicle after moving.
int y
x and y position of the vehicle after moving
TileIndex old_tile
Current tile of the vehicle.
Cached, frequently calculated values.
EngineID first_engine
Cached EngineID of the front vehicle. EngineID::Invalid() for the front vehicle itself.
uint32_t cached_power
Total power of the consist (valid only for the first engine).
bool IsMultiheaded() const
Check if the vehicle is a multiheaded engine.
static void CountVehicle(const Vehicle *v, int delta)
Update num_vehicle when adding or removing a vehicle.
static void VehicleReachedMinAge(const Vehicle *v)
Add a vehicle to the profit sum of its group.
static void CountEngine(const Vehicle *v, int delta)
Update num_engines when adding/removing an engine.
static void UpdateAutoreplace(CompanyID company)
Update autoreplace_defined and autoreplace_finished of all statistics of a company.
static void UpdateProfits()
Recompute the profits for all groups.
Group data.
Definition group.h:72
Livery livery
Custom colour scheme for vehicles in this group.
Definition group.h:78
GroupID parent
Parent group.
Definition group.h:83
Information about a particular livery.
Definition livery.h:78
Colours colour2
Second colour, for vehicles with 2CC support.
Definition livery.h:81
Colours colour1
First colour, for all vehicles.
Definition livery.h:80
uint8_t in_use
Bit 0 set if this livery should override the default livery first colour, Bit 1 for the second colour...
Definition livery.h:79
bool revalidate_before_draw
We need to do a GetImage() and check bounds before drawing this sprite.
VehicleSpriteSeq sprite_seq
Vehicle appearance.
bool is_viewport_candidate
This vehicle can potentially be drawn on a viewport.
Rect old_coord
Co-ordinates from the last valid bounding box.
static void ClearVehicle(const Vehicle *v)
Clear/update the (clone) vehicle from an order backup.
Shared order list linking together the linked list of orders and the list of vehicles sharing this or...
Definition order_base.h:258
uint GetNumVehicles() const
Return the number of vehicles that share this orders list.
Definition order_base.h:347
void RemoveVehicle(Vehicle *v)
Removes the vehicle from the shared order list.
void AddVehicle(Vehicle *v)
Adds the given vehicle to this shared order list.
Definition order_base.h:355
VehicleOrderID GetNumOrders() const
Get number of orders in the order list.
Definition order_base.h:316
bool no_servicing_if_no_breakdowns
don't send vehicles to depot when breakdowns are disabled
OrderDepotTypeFlags GetDepotOrderType() const
What caused us going to the depot?
Definition order_base.h:144
bool Equals(const Order &other) const
Does this order have the same type, flags and destination?
DestinationID GetDestination() const
Gets the destination of this order.
Definition order_base.h:103
bool IsType(OrderType type) const
Check whether this order is of the given type.
Definition order_base.h:70
void SetNonStopType(OrderNonStopFlags non_stop_type)
Set whether we must stop at stations or not.
Definition order_base.h:161
CargoType GetRefitCargo() const
Get the cargo to to refit to.
Definition order_base.h:131
OrderType GetType() const
Get the type of order of this order.
Definition order_base.h:76
void SetDepotOrderType(OrderDepotTypeFlags depot_order_type)
Set the cause to go to the depot.
Definition order_base.h:165
OrderLoadFlags GetLoadType() const
How must the consist be loaded?
Definition order_base.h:136
void MakeDummy()
Makes this order a Dummy order.
Order * next
Pointer to next order. If nullptr, end of list.
Definition order_base.h:59
void SetDepotActionType(OrderDepotActionFlags depot_service_type)
Set what we are going to do in the depot.
Definition order_base.h:167
OrderDepotActionFlags GetDepotActionType() const
What are we going to do when in the depot.
Definition order_base.h:146
void MakeLeaveStation()
Makes this order a Leave Station order.
bool CanLeaveWithCargo(bool has_cargo) const
A vehicle can leave the current station with cargo if:
uint16_t GetTimetabledWait() const
Get the time in ticks a vehicle should wait at the destination or 0 if it's not timetabled.
Definition order_base.h:188
bool ShouldStopAtStation(const Vehicle *v, StationID station) const
Check whether the given vehicle should stop at the given station based on this order and the non-stop...
void MakeImplicit(StationID destination)
Makes this order an implicit order.
OrderNonStopFlags GetNonStopType() const
At which stations must we stop?
Definition order_base.h:140
bool IsRefit() const
Is this order a refit order.
Definition order_base.h:117
void MakeLoading(bool ordered)
Makes this order a Loading order.
void MakeGoToDepot(DestinationID destination, OrderDepotTypeFlags order, OrderNonStopFlags non_stop_type=ONSF_NO_STOP_AT_INTERMEDIATE_STATIONS, OrderDepotActionFlags action=ODATF_SERVICE_ONLY, CargoType cargo=CARGO_NO_REFIT)
Makes this order a Go To Depot order.
Definition order_cmd.cpp:90
Coordinates of a point in 2D.
static Pool::IterateWrapper< Titem > Iterate(size_t from=0)
Returns an iterable ensemble of all valid Titem.
static size_t GetPoolSize()
Returns first unused index.
static Titem * Get(auto index)
Returns Titem with given index.
Tindex index
Index of this pool item.
static bool CanAllocateItem(size_t n=1)
Helper functions so we can use PoolItem::Function() instead of _poolitem_pool.Function()
static bool IsValidID(auto index)
Tests whether given index can be used to get valid (non-nullptr) Titem.
static bool CleaningPool()
Returns current state of pool cleaning - yes or no.
static Titem * GetIfValid(auto index)
Returns Titem with given index.
Base class for all pools.
uint8_t visual_effect
Bitstuffed NewGRF visual effect data.
Definition engine_type.h:62
EngineClass engclass
Class of engine for this vehicle.
Definition engine_type.h:58
Specification of a rectangle with absolute coordinates of all edges.
A Stop for a Road Vehicle.
void Leave(RoadVehicle *rv)
Leave the road stop.
Definition roadstop.cpp:214
RoadStop * next
Next stop of the given type at this station.
TileIndex xy
Position on the map.
static RoadStop * GetByTile(TileIndex tile, RoadStopType type)
Find a roadstop at given tile.
Definition roadstop.cpp:264
uint8_t visual_effect
Bitstuffed NewGRF visual effect data.
Buses, trucks and trams belong to this class.
Definition roadveh.h:98
uint8_t state
Definition roadveh.h:100
RoadTypes compatible_roadtypes
NOSAVE: Roadtypes this consist is powered on.
Definition roadveh.h:110
bool IsBus() const
Check whether a roadvehicle is a bus.
RoadType roadtype
NOSAVE: Roadtype of this vehicle.
Definition roadveh.h:108
VehicleID disaster_vehicle
NOSAVE: Disaster vehicle targetting this vehicle.
Definition roadveh.h:109
uint8_t visual_effect
Bitstuffed NewGRF visual effect data.
Definition engine_type.h:80
All ships have this type.
Definition ship.h:32
TrackBits state
The "track" the ship is following.
Definition ship.h:34
void UpdateCache()
Update the caches of this ship.
Definition ship_cmd.cpp:232
static bool IsValidID(auto index)
Tests whether given index is a valid index for station of this type.
static Station * Get(auto index)
Gets station with given index.
static Station * GetIfValid(auto index)
Returns station if the index is a valid index for this station type.
T * Next() const
Get next vehicle in the chain.
static T * From(Vehicle *v)
Converts a Vehicle to SpecializedVehicle with type checking.
void UpdateViewport(bool force_update, bool update_delta)
Update vehicle sprite- and position caches.
T * GetFirstEnginePart()
Get the first part of an articulated engine.
Data structure describing a sprite.
Definition spritecache.h:17
uint16_t width
Width of the sprite.
Definition spritecache.h:19
uint16_t height
Height of the sprite.
Definition spritecache.h:18
int16_t y_offs
Number of pixels to shift the sprite downwards.
Definition spritecache.h:21
int16_t x_offs
Number of pixels to shift the sprite to the right.
Definition spritecache.h:20
Station data structure.
std::array< GoodsEntry, NUM_CARGO > goods
Goods at this station.
Airport airport
Tile area the airport covers.
VehicleEnterTileProc * vehicle_enter_tile_proc
Called when a vehicle enters a tile.
Definition tile_cmd.h:171
'Train' is either a loco or a wagon.
Definition train.h:90
void ConsistChanged(ConsistChangeFlags allowed_changes)
Recalculates the cached stuff of a train.
uint16_t wait_counter
Ticks waiting in front of a signal, ticks being stuck or a counter for forced proceeding through sign...
Definition train.h:93
uint8_t cached_vis_effect
Visual effect to show (see VisualEffect)
uint16_t cached_cargo_age_period
Number of ticks before carried cargo is aged.
The information about a vehicle list.
Definition vehiclelist.h:32
WindowNumber ToWindowNumber() const
Pack a VehicleListIdentifier in 32 bits so it can be used as unique WindowNumber.
UnitID max_ships
max ships in game per company
UnitID max_trains
max trains in game per company
uint8_t smoke_amount
amount of smoke/sparks locomotives produce
UnitID max_aircraft
max planes in game per company
UnitID max_roadveh
max trucks in game per company
Sprite sequence for a vehicle part.
bool IsValid() const
Check whether the sequence contains any sprites.
void GetBounds(Rect *bounds) const
Determine shared bounds of all sprites.
Definition vehicle.cpp:103
void Draw(int x, int y, PaletteID default_pal, bool force_pal) const
Draw the sprite sequence.
Definition vehicle.cpp:131
Vehicle data structure.
Money GetDisplayProfitThisYear() const
Gets the profit vehicle had this year.
CargoPayment * cargo_payment
The cargo payment we're currently in.
EngineID engine_type
The type of engine used for this vehicle.
uint16_t cargo_age_counter
Ticks till cargo is aged next.
int32_t z_pos
z coordinate.
uint16_t & GetGroundVehicleFlags()
Access the ground vehicle flags of the vehicle.
Definition vehicle.cpp:3174
Direction direction
facing
void ShiftDates(TimerGameEconomy::Date interval)
Shift all dates by given interval.
Definition vehicle.cpp:774
TimerGameEconomy::Date economy_age
Age in economy days.
bool IsOrderListShared() const
Check if we share our orders with another vehicle.
const Engine * GetEngine() const
Retrieves the engine of the vehicle.
Definition vehicle.cpp:744
void IncrementRealOrderIndex()
Advanced cur_real_order_index to the next real order, keeps care of the wrap-around and invalidates t...
virtual uint Crash(bool flooded=false)
Crash the (whole) vehicle chain.
Definition vehicle.cpp:280
bool IsStoppedInDepot() const
Check whether the vehicle is in the depot and stopped.
Order * GetOrder(int index) const
Returns order 'index' of a vehicle or nullptr when it doesn't exists.
bool HasDepotOrder() const
Checks if a vehicle has a depot in its order list.
void LeaveStation()
Perform all actions when leaving a station.
Definition vehicle.cpp:2341
void AddToShared(Vehicle *shared_chain)
Adds this vehicle to a shared vehicle chain.
Definition vehicle.cpp:2957
VehicleCargoList cargo
The cargo this vehicle is carrying.
uint8_t x_extent
x-extent of vehicle bounding box
Vehicle ** hash_tile_prev
NOSAVE: Previous vehicle in the tile location hash.
bool HasUnbunchingOrder() const
Check if the current vehicle has an unbunching order.
Definition vehicle.cpp:2480
TimerGameEconomy::Date date_of_last_service
Last economy date the vehicle had a service at a depot.
uint16_t cargo_cap
total capacity
StationID last_loading_station
Last station the vehicle has stopped at and could possibly leave from with any cargo loaded.
VehicleOrderID GetNumOrders() const
Get the number of orders this vehicle has.
uint8_t subtype
subtype (Filled with values from AircraftSubType/DisasterSubType/EffectVehicleType/GroundVehicleSubty...
CommandCost SendToDepot(DoCommandFlags flags, DepotCommandFlags command)
Send this vehicle to the depot using the given command(s).
Definition vehicle.cpp:2574
void ReleaseUnitNumber()
Release the vehicle's unit number.
Definition vehicle.cpp:2410
void UpdateBoundingBoxCoordinates(bool update_cache) const
Update the bounding box co-ordinates of the vehicle.
Definition vehicle.cpp:1693
uint8_t day_counter
Increased by one for each day.
void HandleLoading(bool mode=false)
Handle the loading of the vehicle; when not it skips through dummy orders and does nothing in all oth...
Definition vehicle.cpp:2421
Money profit_this_year
Profit this year << 8, low 8 bits are fract.
bool HasArticulatedPart() const
Check if an engine has an articulated part.
SpriteID colourmap
NOSAVE: cached colour mapping.
uint8_t breakdown_ctr
Counter for managing breakdown events.
uint8_t breakdown_delay
Counter for managing breakdown length.
Vehicle * GetNextVehicle() const
Get the next real (non-articulated part) vehicle in the consist.
GroupID group_id
Index of group Pool array.
void IncrementImplicitOrderIndex()
Increments cur_implicit_order_index, keeps care of the wrap-around and invalidates the GUI.
uint8_t z_extent
z-extent of vehicle bounding box
VehStates vehstatus
Status.
TimerGameCalendar::Date date_of_last_service_newgrf
Last calendar date the vehicle had a service at a depot, unchanged by the date cheat to protect again...
Vehicle * first
NOSAVE: pointer to the first vehicle in the chain.
void CancelReservation(StationID next, Station *st)
Return all reserved cargo packets to the station and reset all packets staged for transfer.
Definition vehicle.cpp:2325
Money profit_last_year
Profit last year << 8, low 8 bits are fract.
Order * GetFirstOrder() const
Get the first order of the vehicles order list.
bool IsEngineCountable() const
Check if a vehicle is counted in num_engines in each company struct.
Definition vehicle.cpp:711
virtual ~Vehicle()
We want to 'destruct' the right class.
Definition vehicle.cpp:889
bool IsArticulatedPart() const
Check if the vehicle is an articulated part of an engine.
bool NeedsAutorenewing(const Company *c, bool use_renew_setting=true) const
Function to tell if a vehicle needs to be autorenewed.
Definition vehicle.cpp:145
void UpdateVisualEffect(bool allow_power_change=true)
Update the cached visual effect.
Definition vehicle.cpp:2655
void LeaveUnbunchingDepot()
Leave an unbunching depot and calculate the next departure time for shared order vehicles.
Definition vehicle.cpp:2505
int8_t y_offs
y offset for vehicle sprite
CargoType cargo_type
type of cargo this vehicle is carrying
debug_inline bool IsFrontEngine() const
Check if the vehicle is a front engine.
Vehicle * previous_shared
NOSAVE: pointer to the previous vehicle in the shared order chain.
VehicleOrderID GetNumManualOrders() const
Get the number of manually added orders this vehicle has.
Vehicle * First() const
Get the first vehicle of this vehicle chain.
int8_t trip_occupancy
NOSAVE: Occupancy of vehicle of the current trip (updated after leaving a station).
int8_t x_bb_offs
x offset of vehicle bounding box
Order current_order
The current order (+ status, like: loading)
void PreDestructor()
Destroy all stuff that (still) needs the virtual functions to work properly.
Definition vehicle.cpp:822
TimerGameTick::TickCounter last_loading_tick
Last TimerGameTick::counter tick that the vehicle has stopped at a station and could possibly leave w...
void HandlePathfindingResult(bool path_found)
Handle the pathfinding result, especially the lost status.
Definition vehicle.cpp:788
Vehicle * Next() const
Get the next vehicle of this vehicle.
int8_t x_offs
x offset for vehicle sprite
int32_t y_pos
y coordinate.
int32_t x_pos
x coordinate.
const GRFFile * GetGRF() const
Retrieve the NewGRF the vehicle is tied to.
Definition vehicle.cpp:754
OrderList * orders
Pointer to the order list for this vehicle.
uint32_t GetDisplayMinPowerToWeight() const
Calculates the minimum power-to-weight ratio using the maximum weight of the ground vehicle.
Definition vehicle.cpp:3248
uint8_t y_extent
y-extent of vehicle bounding box
virtual ClosestDepot FindClosestDepot()
Find the closest depot for this vehicle and tell us the location, DestinationID and whether we should...
void UpdateViewport(bool dirty)
Update the vehicle on the viewport, updating the right hash and setting the new coordinates.
Definition vehicle.cpp:1726
Money value
Value of the vehicle.
bool MarkAllViewportsDirty() const
Marks viewports dirty where the vehicle's image is.
Definition vehicle.cpp:1765
uint16_t refit_cap
Capacity left over from before last refit.
VehicleCache vcache
Cache of often used vehicle values.
uint32_t motion_counter
counter to occasionally play a vehicle sound.
uint32_t GetGRFID() const
Retrieve the GRF ID of the NewGRF the vehicle is tied to.
Definition vehicle.cpp:764
GroundVehicleCache * GetGroundVehicleCache()
Access the ground vehicle cache of the vehicle.
Definition vehicle.cpp:3144
virtual void OnNewEconomyDay()
Calls the new economy day handler of the vehicle.
Vehicle ** hash_tile_current
NOSAVE: Cache of the current hash chain.
virtual void OnNewCalendarDay()
Calls the new calendar day handler of the vehicle.
Vehicle * NextShared() const
Get the next vehicle of the shared vehicle chain.
int8_t y_bb_offs
y offset of vehicle bounding box
virtual int GetCurrentMaxSpeed() const
Calculates the maximum speed of the vehicle under its current conditions.
bool HasFullLoadOrder() const
Check if the current vehicle has a full load order.
Definition vehicle.cpp:2456
virtual bool IsPrimaryVehicle() const
Whether this is the primary vehicle in the chain.
void BeginLoading()
Prepare everything to begin the loading when arriving at a station.
Definition vehicle.cpp:2194
Order * GetLastOrder() const
Returns the last order of a vehicle, or nullptr if it doesn't exists.
Vehicle * hash_tile_next
NOSAVE: Next vehicle in the tile location hash.
uint16_t cur_speed
current speed
Vehicle * previous
NOSAVE: pointer to the previous vehicle in the chain.
bool HasEngineType() const
Check whether Vehicle::engine_type has any meaning.
Definition vehicle.cpp:728
TimerGameCalendar::Date age
Age in calendar days.
bool IsWaitingForUnbunching() const
Check whether a vehicle inside a depot is waiting for unbunching.
Definition vehicle.cpp:2552
TextEffectID fill_percent_te_id
a text-effect id to a loading indicator object
IterateWrapper Orders() const
Returns an iterable ensemble of orders of a vehicle.
void SetNext(Vehicle *next)
Set the next vehicle of this vehicle.
Definition vehicle.cpp:2928
uint8_t breakdowns_since_last_service
Counter for the amount of breakdowns.
Vehicle * next
pointer to the next vehicle in the chain
TimerGameCalendar::Date max_age
Maximum age.
MutableSpriteCache sprite_cache
Cache of sprites and values related to recalculating them, see MutableSpriteCache.
uint16_t reliability
Reliability.
uint32_t GetDisplayMaxWeight() const
Calculates the maximum weight of the ground vehicle when loaded.
Definition vehicle.cpp:3233
Vehicle * FirstShared() const
Get the first vehicle of this vehicle chain.
void RemoveFromShared()
Removes the vehicle from the shared order list.
Definition vehicle.cpp:2980
bool HandleBreakdown()
Handle all of the aspects of a vehicle breakdown This includes adding smoke and sounds,...
Definition vehicle.cpp:1357
debug_inline bool IsGroundVehicle() const
Check if the vehicle is a ground vehicle.
void UpdatePositionAndViewport()
Update the position of the vehicle, and update the viewport.
Definition vehicle.cpp:1755
Vehicle(VehicleType type=VEH_INVALID)
Vehicle constructor.
Definition vehicle.cpp:358
Vehicle * Previous() const
Get the previous vehicle of this vehicle.
virtual void PlayLeaveStationSound(bool force=false) const
Play the sound associated with leaving the station.
Rect coord
NOSAVE: Graphical bounding box of the vehicle, i.e. what to redraw on moves.
uint16_t reliability_spd_dec
Reliability decrease speed.
uint8_t tick_counter
Increased by one for each tick.
virtual bool IsInDepot() const
Check whether the vehicle is in the depot.
TileIndex tile
Current tile index.
virtual bool Tick()
Calls the tick handler of the vehicle.
TileIndex dest_tile
Heading for this tile.
bool NeedsServicing() const
Check if the vehicle needs to go to a depot in near future (if a opportunity presents itself) for ser...
Definition vehicle.cpp:190
bool HasConditionalOrder() const
Check if the current vehicle has a conditional order.
Definition vehicle.cpp:2468
void UpdatePosition()
Update the position of the vehicle.
Definition vehicle.cpp:1684
StationID last_station_visited
The last station we stopped at.
void ResetRefitCaps()
Reset all refit_cap in the consist to cargo_cap.
Definition vehicle.cpp:2402
uint8_t breakdown_chance
Current chance of breakdowns.
void ShowVisualEffect() const
Draw visual effects (smoke and/or sparks) for a vehicle chain.
Definition vehicle.cpp:2778
Owner owner
Which company owns the vehicle?
UnitID unitnumber
unit number, for display purposes only
Vehicle * next_shared
pointer to the next vehicle that shares the order
bool NeedsAutomaticServicing() const
Checks if the current order should be interrupted for a service-in-depot order.
Definition vehicle.cpp:272
void DeleteUnreachedImplicitOrders()
Delete all implicit orders which were not reached.
Definition vehicle.cpp:2155
Vehicle ** hash_viewport_prev
NOSAVE: Previous vehicle in the visual location hash.
virtual void GetImage(Direction direction, EngineImageType image_type, VehicleSpriteSeq *result) const
Gets the sprite to show for the given direction.
Vehicle * hash_viewport_next
NOSAVE: Next vehicle in the visual location hash.
Data structure for viewport, display of a part of the world.
int top
Screen coordinate top edge of the viewport.
int width
Screen width of the viewport.
ZoomLevel zoom
The zoom level of the viewport.
int virtual_top
Virtual top coordinate.
int virtual_left
Virtual left coordinate.
int left
Screen coordinate left edge of the viewport.
int height
Screen height of the viewport.
VehicleEnterTileStatus
The returned bits of VehicleEnterTile.
Definition tile_cmd.h:21
int GetTileMaxPixelZ(TileIndex tile)
Get top height of the tile.
Definition tile_map.h:312
static debug_inline TileType GetTileType(Tile tile)
Get the tiletype of a given tile.
Definition tile_map.h:96
static debug_inline bool IsTileType(Tile tile, TileType type)
Checks if a tile is a given tiletype.
Definition tile_map.h:150
static const int MAX_VEHICLE_PIXEL_Y
Maximum height of a vehicle in pixels in ZOOM_BASE.
Definition tile_type.h:22
static const uint TILE_SIZE
Tile size in world coordinates.
Definition tile_type.h:15
static const int MAX_VEHICLE_PIXEL_X
Maximum width of a vehicle in pixels in ZOOM_BASE.
Definition tile_type.h:21
@ MP_STATION
A tile of a station.
Definition tile_type.h:53
Definition of Interval and OneShot timers.
Definition of the game-calendar-timer.
Definition of the game-economy-timer.
Definition of the tick-based game-timer.
Functions related to time tabling.
void UpdateVehicleTimetable(Vehicle *v, bool travelling)
Update the timetable for the vehicle.
bool TracksOverlap(TrackBits bits)
Checks if the given tracks overlap, ie form a crossing.
Definition track_func.h:645
TrackBits
Allow incrementing of Track variables.
Definition track_type.h:35
@ TRACK_BIT_DEPOT
Bitflag for a depot.
Definition track_type.h:53
Base for the train class.
@ VRF_LEAVING_STATION
Train is just leaving a station.
Definition train.h:33
@ VRF_TOGGLE_REVERSE
Used for vehicle var 0xFE bit 8 (toggled each time the train is reversed, accurate for first vehicle ...
Definition train.h:31
@ VRF_REVERSE_DIRECTION
Reverse the visible direction of the vehicle.
Definition train.h:28
@ TFP_NONE
Normal operation.
Definition train.h:38
static constexpr ConsistChangeFlags CCF_ARRANGE
Valid changes for arranging the consist in a depot.
Definition train.h:55
Command definitions related to trains.
bool IsTransparencySet(TransparencyOption to)
Check if the transparency option bit is set and if we aren't in the game menu (there's never transpar...
bool IsInvisibilitySet(TransparencyOption to)
Check if the invisibility option bit is set and if we aren't in the game menu (there's never transpar...
TransparencyOption
Transparency option bits: which position in _transparency_opt stands for which transparency.
@ TO_INVALID
Invalid transparency option.
uint16_t UnitID
Type for the company global vehicle unit number.
Map accessors for tunnels.
bool IsTunnelTile(Tile t)
Is this a tunnel (entrance)?
Definition tunnel_map.h:34
PaletteID GetVehiclePalette(const Vehicle *v)
Get the colour map for a vehicle.
Definition vehicle.cpp:2143
void ShowNewGrfVehicleError(EngineID engine, StringID part1, StringID part2, GRFBug bug_type, bool critical)
Displays a "NewGrf Bug" error message for a engine, and pauses the game if not networking.
Definition vehicle.cpp:317
bool CanBuildVehicleInfrastructure(VehicleType type, uint8_t subtype)
Check whether we can build infrastructure for the given vehicle type.
Definition vehicle.cpp:1913
CommandCost EnsureNoVehicleOnGround(TileIndex tile)
Ensure there is no vehicle at the ground at the given position.
Definition vehicle.cpp:543
bool VehiclesHaveSameEngineList(const Vehicle *v1, const Vehicle *v2)
Checks if two vehicle chains have the same list of engines.
Definition vehicle.cpp:3261
bool VehiclesHaveSameOrderList(const Vehicle *v1, const Vehicle *v2)
Checks if two vehicles have the same list of orders.
Definition vehicle.cpp:3278
static void SpawnAdvancedVisualEffect(const Vehicle *v)
Call CBID_VEHICLE_SPAWN_VISUAL_EFFECT and spawn requested effects.
Definition vehicle.cpp:2723
static Vehicle * GetVehicleTunnelBridgeProc(Vehicle *v, void *data)
Procedure called for every vehicle found in tunnel/bridge in the hash map.
Definition vehicle.cpp:557
void VehicleEnterDepot(Vehicle *v)
Vehicle entirely entered the depot, update its status, orders, vehicle windows, service it,...
Definition vehicle.cpp:1545
void GetVehicleSet(VehicleSet &set, Vehicle *v, uint8_t num_vehicles)
Calculates the set of vehicles that will be affected by a given selection.
Definition vehicle.cpp:3207
bool HasVehicleOnPosXY(int x, int y, void *data, VehicleFromPosProc *proc)
Checks whether a vehicle in on a specific location.
Definition vehicle.cpp:458
UnitID GetFreeUnitNumber(VehicleType type)
Get an unused unit number for a vehicle (if allowed).
Definition vehicle.cpp:1886
void RunVehicleCalendarDayProc()
Age all vehicles, spreading out the action using the current TimerGameCalendar::date_fract.
Definition vehicle.cpp:933
static Vehicle * VehicleFromPos(TileIndex tile, void *data, VehicleFromPosProc *proc, bool find_first)
Helper function for FindVehicleOnPos/HasVehicleOnPos.
Definition vehicle.cpp:473
void VehicleLengthChanged(const Vehicle *u)
Logs a bug in GRF and shows a warning message if this is for the first time this happened.
Definition vehicle.cpp:343
const Livery * GetEngineLivery(EngineID engine_type, CompanyID company, EngineID parent_engine_type, const Vehicle *v, uint8_t livery_setting)
Determines the livery for a vehicle.
Definition vehicle.cpp:2055
VehiclePool _vehicle_pool("Vehicle")
The pool with all our precious vehicles.
void VehicleServiceInDepot(Vehicle *v)
Service a vehicle and all subsequent vehicles in the consist.
Definition vehicle.cpp:167
CommandCost EnsureNoTrainOnTrackBits(TileIndex tile, TrackBits track_bits)
Tests if a vehicle interacts with the specified track bits.
Definition vehicle.cpp:605
void FindVehicleOnPosXY(int x, int y, void *data, VehicleFromPosProc *proc)
Find a vehicle from a specific location.
Definition vehicle.cpp:442
static void RunEconomyVehicleDayProc()
Increases the day counter for all vehicles and calls 1-day and 32-day handlers.
Definition vehicle.cpp:950
LiveryScheme GetEngineLiveryScheme(EngineID engine_type, EngineID parent_engine_type, const Vehicle *v)
Determines the LiveryScheme for a vehicle.
Definition vehicle.cpp:1961
VehicleEnterTileStatus VehicleEnterTile(Vehicle *v, TileIndex tile, int x, int y)
Call the tile callback function for a vehicle entering a tile.
Definition vehicle.cpp:1829
void ViewportAddVehicles(DrawPixelInfo *dpi)
Add the vehicle sprites that should be drawn at a part of the screen.
Definition vehicle.cpp:1148
GetNewVehiclePosResult GetNewVehiclePos(const Vehicle *v)
Get position information of a vehicle when moving one pixel in the direction it is facing.
Definition vehicle.cpp:1775
bool HasVehicleOnPos(TileIndex tile, void *data, VehicleFromPosProc *proc)
Checks whether a vehicle is on a specific location.
Definition vehicle.cpp:517
void FindVehicleOnPos(TileIndex tile, void *data, VehicleFromPosProc *proc)
Find a vehicle from a specific location.
Definition vehicle.cpp:502
StringID GetVehicleCannotUseStationReason(const Vehicle *v, const Station *st)
Get reason string why this station can't be used by the given vehicle.
Definition vehicle.cpp:3093
void DecreaseVehicleValue(Vehicle *v)
Decrease the value of a vehicle.
Definition vehicle.cpp:1295
void EconomyAgeVehicle(Vehicle *v)
Update economy age of a vehicle.
Definition vehicle.cpp:1423
static Vehicle * EnsureNoVehicleProcZ(Vehicle *v, void *data)
Callback that returns 'real' vehicles lower or at height *(int*)data .
Definition vehicle.cpp:528
uint8_t CalcPercentVehicleFilled(const Vehicle *front, StringID *colour)
Calculates how full a vehicle is.
Definition vehicle.cpp:1486
Vehicle * CheckClickOnVehicle(const Viewport *vp, int x, int y)
Find the vehicle close to the clicked coordinates.
Definition vehicle.cpp:1242
CommandCost TunnelBridgeIsFree(TileIndex tile, TileIndex endtile, const Vehicle *ignore)
Finds vehicle in tunnel / bridge.
Definition vehicle.cpp:572
PaletteID GetEnginePalette(EngineID engine_type, CompanyID company)
Get the colour map for an engine.
Definition vehicle.cpp:2133
bool CanVehicleUseStation(EngineID engine_type, const Station *st)
Can this station be used by the given engine type?
Definition vehicle.cpp:3047
void AgeVehicle(Vehicle *v)
Update age of a vehicle.
Definition vehicle.cpp:1435
static Vehicle * VehicleFromPosXY(int x, int y, void *data, VehicleFromPosProc *proc, bool find_first)
Helper function for FindVehicleOnPos/HasVehicleOnPos.
Definition vehicle.cpp:415
void VehicleEnteredDepotThisTick(Vehicle *v)
Adds a vehicle to the list of vehicles that visited a depot this tick.
Definition vehicle.cpp:917
std::map< VehicleID, bool > AutoreplaceMap
List of vehicles that should check for autoreplace this tick.
Definition vehicle.cpp:691
static bool PreviousOrderIsUnbunching(const Vehicle *v)
Check if the previous order is a depot unbunching order.
Definition vehicle.cpp:2492
static void DoDrawVehicle(const Vehicle *v)
Add vehicle sprite for drawing to the screen.
Definition vehicle.cpp:1118
@ VE_TYPE_DEFAULT
Use default from engine class.
@ VE_TYPE_COUNT
Number of bits used for the effect type.
@ VE_OFFSET_CENTRE
Value of offset corresponding to a position above the centre of the vehicle.
@ VE_TYPE_ELECTRIC
Electric sparks.
@ VE_TYPE_START
First bit used for the type of effect.
@ VE_OFFSET_COUNT
Number of bits used for the offset.
@ VE_ADVANCED_EFFECT
Flag for advanced effects.
@ VE_DISABLE_EFFECT
Flag to disable visual effect.
@ VE_TYPE_STEAM
Steam plumes.
@ VE_TYPE_DIESEL
Diesel fumes.
@ VE_DEFAULT
Default value to indicate that visual effect should be based on engine class.
@ VE_OFFSET_START
First bit that contains the offset (0 = front, 8 = centre, 15 = rear)
@ VE_DISABLE_WAGON_POWER
Flag to disable wagon power.
@ Unclickable
Vehicle is not clickable by the user (shadow vehicles).
@ Crashed
Vehicle is crashed.
@ Shadow
Vehicle is a shadow vehicle.
@ TrainSlowing
Train is slowing down.
@ AircraftBroken
Aircraft is broken down.
@ Hidden
Vehicle is not visible.
@ DefaultPalette
Use default vehicle palette.
@ Stopped
Vehicle is stopped by the player.
VisualEffectSpawnModel
Models for spawning visual effects.
@ VESM_ELECTRIC
Electric model.
@ VESM_STEAM
Steam model.
@ VESM_DIESEL
Diesel model.
@ VESM_NONE
No visual effect.
static const int32_t INVALID_COORD
Sentinel for an invalid coordinate.
Command definitions for vehicles.
Functions related to vehicles.
static const TimerGameEconomy::Date VEHICLE_PROFIT_MIN_AGE
Only vehicles older than this have a meaningful profit.
bool IsCompanyBuildableVehicleType(VehicleType type)
Is the given vehicle type buildable by a company?
@ VIWD_MODIFY_ORDERS
Other order modifications.
Definition vehicle_gui.h:36
WindowClass GetWindowClassForVehicleType(VehicleType vt)
Get WindowClass for vehicle list of given vehicle type.
Definition vehicle_gui.h:97
@ EIT_ON_MAP
Vehicle drawn in viewport.
VehicleType
Available vehicle types.
@ VEH_ROAD
Road vehicle type.
@ VEH_DISASTER
Disaster vehicle type.
@ VEH_AIRCRAFT
Aircraft vehicle type.
@ VEH_SHIP
Ship vehicle type.
@ VEH_EFFECT
Effect vehicle type (smoke, explosions, sparks, bubbles)
@ VEH_TRAIN
Train vehicle type.
@ DontCancel
Don't cancel current goto depot command if any.
@ Service
The vehicle will leave the depot right after arrival (service only)
static const uint VEHICLE_LENGTH
The length of a vehicle in tile units.
@ WID_VV_START_STOP
Start or stop this vehicle, and show information about the current state.
Functions and type for generating vehicle lists.
@ VL_SHARED_ORDERS
Index is the first vehicle of the shared orders.
Definition vehiclelist.h:24
void StartSpriteCombine()
Starts a block of sprites, which are "combined" into a single bounding box.
Definition viewport.cpp:769
void AddSortableSpriteToDraw(SpriteID image, PaletteID pal, int x, int y, int w, int h, int dz, int z, bool transparent, int bb_offset_x, int bb_offset_y, int bb_offset_z, const SubSprite *sub)
Draw a (transparent) sprite at given coordinates with a given bounding box.
Definition viewport.cpp:673
void EndSpriteCombine()
Terminates a block of sprites started by StartSpriteCombine.
Definition viewport.cpp:779
Functions related to (drawing on) viewports.
void CloseWindowById(WindowClass cls, WindowNumber number, bool force, int data)
Close a window by its class and window number (if it is open).
Definition window.cpp:1145
void SetWindowClassesDirty(WindowClass cls)
Mark all windows of a particular class as dirty (in need of repainting)
Definition window.cpp:3134
void InvalidateWindowData(WindowClass cls, WindowNumber number, int data, bool gui_scope)
Mark window data of the window of a given class and specific window number as invalid (in need of re-...
Definition window.cpp:3226
void SetWindowWidgetDirty(WindowClass cls, WindowNumber number, WidgetID widget_index)
Mark a particular widget in a particular window as dirty (in need of repainting)
Definition window.cpp:3121
void SetWindowDirty(WindowClass cls, WindowNumber number)
Mark window as dirty (in need of repainting)
Definition window.cpp:3108
void InvalidateWindowClassesData(WindowClass cls, int data, bool gui_scope)
Mark window data of all windows of a given class as invalid (in need of re-computing) Note that by de...
Definition window.cpp:3243
@ WC_ROADVEH_LIST
Road vehicle list; Window numbers:
@ WC_VEHICLE_ORDERS
Vehicle orders; Window numbers:
@ WC_VEHICLE_DEPOT
Depot view; Window numbers:
@ WC_SHIPS_LIST
Ships list; Window numbers:
@ WC_STATION_VIEW
Station view; Window numbers:
@ WC_TRAINS_LIST
Trains list; Window numbers:
@ WC_VEHICLE_REFIT
Vehicle refit; Window numbers:
@ WC_VEHICLE_DETAILS
Vehicle details; Window numbers:
@ WC_COMPANY
Company view; Window numbers:
@ WC_VEHICLE_VIEW
Vehicle view; Window numbers:
@ WC_VEHICLE_TIMETABLE
Vehicle timetable; Window numbers:
@ WC_AIRCRAFT_LIST
Aircraft list; Window numbers:
Functions related to zooming.
int ScaleByZoom(int value, ZoomLevel zoom)
Scale by zoom level, usually shift left (when zoom > ZOOM_LVL_MIN) When shifting right,...
Definition zoom_func.h:22