Fix #10490: Allow ships to exit depots if another is not moving at the exit point...
[openttd-github.git] / src / newgrf_town.cpp
blob0bd4924e34de02c96ca2cfb9b83beeb0746a4867
1 /*
2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
6 */
8 /** @file newgrf_town.cpp Implementation of the town part of NewGRF houses. */
10 #include "stdafx.h"
11 #include "debug.h"
12 #include "town.h"
13 #include "newgrf_town.h"
14 #include "timer/timer_game_tick.h"
16 #include "safeguards.h"
18 /* virtual */ uint32_t TownScopeResolver::GetVariable(byte variable, [[maybe_unused]] uint32_t parameter, bool *available) const
20 CargoID cid;
21 switch (variable) {
22 /* Larger towns */
23 case 0x40:
24 if (_settings_game.economy.larger_towns == 0) return 2;
25 if (this->t->larger_town) return 1;
26 return 0;
28 /* Town index */
29 case 0x41: return this->t->index;
31 /* Get a variable from the persistent storage */
32 case 0x7C: {
33 /* Check the persistent storage for the GrfID stored in register 100h. */
34 uint32_t grfid = GetRegister(0x100);
35 if (grfid == 0xFFFFFFFF) {
36 if (this->ro.grffile == nullptr) return 0;
37 grfid = this->ro.grffile->grfid;
40 for (auto &it : this->t->psa_list) {
41 if (it->grfid == grfid) return it->GetValue(parameter);
44 return 0;
47 /* Town properties */
48 case 0x80: return this->t->xy.base();
49 case 0x81: return GB(this->t->xy.base(), 8, 8);
50 case 0x82: return ClampTo<uint16_t>(this->t->cache.population);
51 case 0x83: return GB(ClampTo<uint16_t>(this->t->cache.population), 8, 8);
52 case 0x8A: return this->t->grow_counter / Ticks::TOWN_GROWTH_TICKS;
53 case 0x92: return this->t->flags; // In original game, 0x92 and 0x93 are really one word. Since flags is a byte, this is to adjust
54 case 0x93: return 0;
55 case 0x94: return ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_EDGE]);
56 case 0x95: return GB(ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_EDGE]), 8, 8);
57 case 0x96: return ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_OUTSKIRT]);
58 case 0x97: return GB(ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_OUTSKIRT]), 8, 8);
59 case 0x98: return ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_OUTER_SUBURB]);
60 case 0x99: return GB(ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_OUTER_SUBURB]), 8, 8);
61 case 0x9A: return ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_INNER_SUBURB]);
62 case 0x9B: return GB(ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_INNER_SUBURB]), 8, 8);
63 case 0x9C: return ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_CENTRE]);
64 case 0x9D: return GB(ClampTo<uint16_t>(this->t->cache.squared_town_zone_radius[HZB_TOWN_CENTRE]), 8, 8);
65 case 0x9E: return this->t->ratings[0];
66 case 0x9F: return GB(this->t->ratings[0], 8, 8);
67 case 0xA0: return this->t->ratings[1];
68 case 0xA1: return GB(this->t->ratings[1], 8, 8);
69 case 0xA2: return this->t->ratings[2];
70 case 0xA3: return GB(this->t->ratings[2], 8, 8);
71 case 0xA4: return this->t->ratings[3];
72 case 0xA5: return GB(this->t->ratings[3], 8, 8);
73 case 0xA6: return this->t->ratings[4];
74 case 0xA7: return GB(this->t->ratings[4], 8, 8);
75 case 0xA8: return this->t->ratings[5];
76 case 0xA9: return GB(this->t->ratings[5], 8, 8);
77 case 0xAA: return this->t->ratings[6];
78 case 0xAB: return GB(this->t->ratings[6], 8, 8);
79 case 0xAC: return this->t->ratings[7];
80 case 0xAD: return GB(this->t->ratings[7], 8, 8);
81 case 0xAE: return this->t->have_ratings;
82 case 0xB2: return this->t->statues;
83 case 0xB6: return ClampTo<uint16_t>(this->t->cache.num_houses);
84 case 0xB9: return this->t->growth_rate / Ticks::TOWN_GROWTH_TICKS;
85 case 0xBA: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].new_max) : 0;
86 case 0xBB: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].new_max), 8, 8) : 0;
87 case 0xBC: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].new_max) : 0;
88 case 0xBD: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].new_max), 8, 8) : 0;
89 case 0xBE: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].new_act) : 0;
90 case 0xBF: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].new_act), 8, 8) : 0;
91 case 0xC0: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].new_act) : 0;
92 case 0xC1: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].new_act), 8, 8) : 0;
93 case 0xC2: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].old_max) : 0;
94 case 0xC3: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].old_max), 8, 8) : 0;
95 case 0xC4: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].old_max) : 0;
96 case 0xC5: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].old_max), 8, 8) : 0;
97 case 0xC6: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].old_act) : 0;
98 case 0xC7: cid = GetCargoIDByLabel(CT_PASSENGERS); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].old_act), 8, 8) : 0;
99 case 0xC8: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? ClampTo<uint16_t>(this->t->supplied[cid].old_act) : 0;
100 case 0xC9: cid = GetCargoIDByLabel(CT_MAIL); return IsValidCargoID(cid) ? GB(ClampTo<uint16_t>(this->t->supplied[cid].old_act), 8, 8) : 0;
101 case 0xCA: return this->t->GetPercentTransported(GetCargoIDByLabel(CT_PASSENGERS));
102 case 0xCB: return this->t->GetPercentTransported(GetCargoIDByLabel(CT_MAIL));
103 case 0xCC: return this->t->received[TAE_FOOD].new_act;
104 case 0xCD: return GB(this->t->received[TAE_FOOD].new_act, 8, 8);
105 case 0xCE: return this->t->received[TAE_WATER].new_act;
106 case 0xCF: return GB(this->t->received[TAE_WATER].new_act, 8, 8);
107 case 0xD0: return this->t->received[TAE_FOOD].old_act;
108 case 0xD1: return GB(this->t->received[TAE_FOOD].old_act, 8, 8);
109 case 0xD2: return this->t->received[TAE_WATER].old_act;
110 case 0xD3: return GB(this->t->received[TAE_WATER].old_act, 8, 8);
111 case 0xD4: return this->t->road_build_months;
112 case 0xD5: return this->t->fund_buildings_months;
115 Debug(grf, 1, "Unhandled town variable 0x{:X}", variable);
117 *available = false;
118 return UINT_MAX;
121 /* virtual */ void TownScopeResolver::StorePSA(uint pos, int32_t value)
123 if (this->readonly) return;
125 assert(this->t != nullptr);
126 /* We can't store anything if the caller has no #GRFFile. */
127 if (this->ro.grffile == nullptr) return;
129 /* Check the persistent storage for the GrfID stored in register 100h. */
130 uint32_t grfid = GetRegister(0x100);
132 /* A NewGRF can only write in the persistent storage associated to its own GRFID. */
133 if (grfid == 0xFFFFFFFF) grfid = this->ro.grffile->grfid;
134 if (grfid != this->ro.grffile->grfid) return;
136 /* Check if the storage exists. */
137 for (auto &it : t->psa_list) {
138 if (it->grfid == grfid) {
139 it->StoreValue(pos, value);
140 return;
144 /* Create a new storage. */
145 assert(PersistentStorage::CanAllocateItem());
146 PersistentStorage *psa = new PersistentStorage(grfid, GSF_FAKE_TOWNS, this->t->xy);
147 psa->StoreValue(pos, value);
148 t->psa_list.push_back(psa);
152 * Resolver for a town.
153 * @param grffile NewGRF file associated with the town.
154 * @param t %Town of the scope.
155 * @param readonly Scope may change persistent storage of the town.
157 TownResolverObject::TownResolverObject(const struct GRFFile *grffile, Town *t, bool readonly)
158 : ResolverObject(grffile), town_scope(*this, t, readonly)