2009-08-21 20:21:05 +00:00
|
|
|
/*
|
|
|
|
* This file is part of OpenTTD.
|
|
|
|
* OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
|
|
|
|
* OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
* See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2008-03-31 00:06:17 +00:00
|
|
|
/** @file station_base.h Base classes/functions for stations. */
|
2007-04-04 01:35:16 +00:00
|
|
|
|
2008-03-31 00:06:17 +00:00
|
|
|
#ifndef STATION_BASE_H
|
|
|
|
#define STATION_BASE_H
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2013-06-09 12:48:42 +00:00
|
|
|
#include "core/random_func.hpp"
|
2009-07-22 11:35:35 +00:00
|
|
|
#include "base_station_base.h"
|
2010-02-22 16:09:26 +00:00
|
|
|
#include "newgrf_airport.h"
|
2007-06-22 11:58:59 +00:00
|
|
|
#include "cargopacket.h"
|
2008-11-19 23:55:34 +00:00
|
|
|
#include "industry_type.h"
|
2013-05-19 14:11:20 +00:00
|
|
|
#include "linkgraph/linkgraph_type.h"
|
2010-08-05 12:06:13 +00:00
|
|
|
#include "newgrf_storage.h"
|
2019-02-14 21:07:15 +00:00
|
|
|
#include "bitmap_type.h"
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2007-03-08 20:50:27 +00:00
|
|
|
static const byte INITIAL_STATION_RATING = 175;
|
2007-01-14 20:00:25 +00:00
|
|
|
|
2013-06-09 12:48:42 +00:00
|
|
|
/**
|
|
|
|
* Flow statistics telling how much flow should be sent along a link. This is
|
|
|
|
* done by creating "flow shares" and using std::map's upper_bound() method to
|
|
|
|
* look them up with a random number. A flow share is the difference between a
|
|
|
|
* key in a map and the previous key. So one key in the map doesn't actually
|
|
|
|
* mean anything by itself.
|
|
|
|
*/
|
|
|
|
class FlowStat {
|
|
|
|
public:
|
|
|
|
typedef std::map<uint32, StationID> SharesMap;
|
|
|
|
|
2015-03-07 18:27:01 +00:00
|
|
|
static const SharesMap empty_sharesmap;
|
|
|
|
|
2013-06-09 12:48:42 +00:00
|
|
|
/**
|
|
|
|
* Invalid constructor. This can't be called as a FlowStat must not be
|
|
|
|
* empty. However, the constructor must be defined and reachable for
|
2020-05-28 03:15:59 +00:00
|
|
|
* FlowStat to be used in a std::map.
|
2013-06-09 12:48:42 +00:00
|
|
|
*/
|
|
|
|
inline FlowStat() {NOT_REACHED();}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a FlowStat with an initial entry.
|
|
|
|
* @param st Station the initial entry refers to.
|
|
|
|
* @param flow Amount of flow for the initial entry.
|
2014-08-22 17:11:59 +00:00
|
|
|
* @param restricted If the flow to be added is restricted.
|
2013-06-09 12:48:42 +00:00
|
|
|
*/
|
2014-08-22 17:11:59 +00:00
|
|
|
inline FlowStat(StationID st, uint flow, bool restricted = false)
|
2013-06-09 12:48:42 +00:00
|
|
|
{
|
|
|
|
assert(flow > 0);
|
|
|
|
this->shares[flow] = st;
|
2014-08-22 17:11:59 +00:00
|
|
|
this->unrestricted = restricted ? 0 : flow;
|
2013-06-09 12:48:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add some flow to the end of the shares map. Only do that if you know
|
|
|
|
* that the station isn't in the map yet. Anything else may lead to
|
|
|
|
* inconsistencies.
|
|
|
|
* @param st Remote station.
|
|
|
|
* @param flow Amount of flow to be added.
|
2013-10-22 16:13:54 +00:00
|
|
|
* @param restricted If the flow to be added is restricted.
|
2013-06-09 12:48:42 +00:00
|
|
|
*/
|
2013-10-22 16:13:54 +00:00
|
|
|
inline void AppendShare(StationID st, uint flow, bool restricted = false)
|
2013-06-09 12:48:42 +00:00
|
|
|
{
|
|
|
|
assert(flow > 0);
|
|
|
|
this->shares[(--this->shares.end())->first + flow] = st;
|
2013-10-22 16:13:54 +00:00
|
|
|
if (!restricted) this->unrestricted += flow;
|
2013-06-09 12:48:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint GetShare(StationID st) const;
|
|
|
|
|
|
|
|
void ChangeShare(StationID st, int flow);
|
|
|
|
|
2013-10-22 16:13:54 +00:00
|
|
|
void RestrictShare(StationID st);
|
|
|
|
|
|
|
|
void ReleaseShare(StationID st);
|
|
|
|
|
2013-12-20 14:57:44 +00:00
|
|
|
void ScaleToMonthly(uint runtime);
|
|
|
|
|
2013-06-09 12:48:42 +00:00
|
|
|
/**
|
|
|
|
* Get the actual shares as a const pointer so that they can be iterated
|
|
|
|
* over.
|
|
|
|
* @return Actual shares.
|
|
|
|
*/
|
|
|
|
inline const SharesMap *GetShares() const { return &this->shares; }
|
|
|
|
|
2013-10-22 16:13:54 +00:00
|
|
|
/**
|
|
|
|
* Return total amount of unrestricted shares.
|
|
|
|
* @return Amount of unrestricted shares.
|
|
|
|
*/
|
|
|
|
inline uint GetUnrestricted() const { return this->unrestricted; }
|
|
|
|
|
2013-06-17 20:38:11 +00:00
|
|
|
/**
|
|
|
|
* Swap the shares maps, and thus the content of this FlowStat with the
|
|
|
|
* other one.
|
|
|
|
* @param other FlowStat to swap with.
|
|
|
|
*/
|
2013-10-22 16:13:54 +00:00
|
|
|
inline void SwapShares(FlowStat &other)
|
|
|
|
{
|
|
|
|
this->shares.swap(other.shares);
|
|
|
|
Swap(this->unrestricted, other.unrestricted);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a station a package can be routed to. This done by drawing a
|
|
|
|
* random number between 0 and sum_shares and then looking that up in
|
|
|
|
* the map with lower_bound. So each share gets selected with a
|
|
|
|
* probability dependent on its flow. Do include restricted flows here.
|
|
|
|
* @param is_restricted Output if a restricted flow was chosen.
|
|
|
|
* @return A station ID from the shares map.
|
|
|
|
*/
|
|
|
|
inline StationID GetViaWithRestricted(bool &is_restricted) const
|
|
|
|
{
|
|
|
|
assert(!this->shares.empty());
|
|
|
|
uint rand = RandomRange((--this->shares.end())->first);
|
|
|
|
is_restricted = rand >= this->unrestricted;
|
|
|
|
return this->shares.upper_bound(rand)->second;
|
|
|
|
}
|
2013-06-17 20:38:11 +00:00
|
|
|
|
2013-06-09 12:48:42 +00:00
|
|
|
/**
|
|
|
|
* Get a station a package can be routed to. This done by drawing a
|
|
|
|
* random number between 0 and sum_shares and then looking that up in
|
|
|
|
* the map with lower_bound. So each share gets selected with a
|
2013-10-22 16:13:54 +00:00
|
|
|
* probability dependent on its flow. Don't include restricted flows.
|
2013-06-09 12:48:42 +00:00
|
|
|
* @return A station ID from the shares map.
|
|
|
|
*/
|
|
|
|
inline StationID GetVia() const
|
|
|
|
{
|
|
|
|
assert(!this->shares.empty());
|
2013-10-22 16:13:54 +00:00
|
|
|
return this->unrestricted > 0 ?
|
|
|
|
this->shares.upper_bound(RandomRange(this->unrestricted))->second :
|
|
|
|
INVALID_STATION;
|
2013-06-09 12:48:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
StationID GetVia(StationID excluded, StationID excluded2 = INVALID_STATION) const;
|
|
|
|
|
2013-06-17 20:38:11 +00:00
|
|
|
void Invalidate();
|
|
|
|
|
2013-06-09 12:48:42 +00:00
|
|
|
private:
|
|
|
|
SharesMap shares; ///< Shares of flow to be sent via specified station (or consumed locally).
|
2013-10-22 16:13:54 +00:00
|
|
|
uint unrestricted; ///< Limit for unrestricted shares.
|
2013-06-09 12:48:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/** Flow descriptions by origin stations. */
|
|
|
|
class FlowStatMap : public std::map<StationID, FlowStat> {
|
|
|
|
public:
|
2014-09-21 16:19:52 +00:00
|
|
|
uint GetFlow() const;
|
|
|
|
uint GetFlowVia(StationID via) const;
|
|
|
|
uint GetFlowFrom(StationID from) const;
|
|
|
|
uint GetFlowFromVia(StationID from, StationID via) const;
|
|
|
|
|
2013-06-09 12:48:42 +00:00
|
|
|
void AddFlow(StationID origin, StationID via, uint amount);
|
|
|
|
void PassOnFlow(StationID origin, StationID via, uint amount);
|
2013-11-10 15:18:49 +00:00
|
|
|
StationIDStack DeleteFlows(StationID via);
|
2013-10-22 16:13:54 +00:00
|
|
|
void RestrictFlows(StationID via);
|
|
|
|
void ReleaseFlows(StationID via);
|
2013-06-09 12:48:42 +00:00
|
|
|
void FinalizeLocalConsumption(StationID self);
|
|
|
|
};
|
|
|
|
|
2011-06-04 21:19:00 +00:00
|
|
|
/**
|
|
|
|
* Stores station stats for a single cargo.
|
|
|
|
*/
|
2007-03-07 12:11:48 +00:00
|
|
|
struct GoodsEntry {
|
2011-06-04 21:19:00 +00:00
|
|
|
/** Status of this cargo for the station. */
|
2011-06-04 21:17:07 +00:00
|
|
|
enum GoodsEntryStatus {
|
2012-11-12 18:11:05 +00:00
|
|
|
/**
|
|
|
|
* Set when the station accepts the cargo currently for final deliveries.
|
|
|
|
* It is updated every STATION_ACCEPTANCE_TICKS ticks by checking surrounding tiles for acceptance >= 8/8.
|
|
|
|
*/
|
|
|
|
GES_ACCEPTANCE,
|
|
|
|
|
|
|
|
/**
|
2014-05-11 18:35:34 +00:00
|
|
|
* This indicates whether a cargo has a rating at the station.
|
|
|
|
* Set when cargo was ever waiting at the station.
|
2012-11-12 18:11:05 +00:00
|
|
|
* It is set when cargo supplied by surrounding tiles is moved to the station, or when
|
|
|
|
* arriving vehicles unload/transfer cargo without it being a final delivery.
|
2014-05-11 18:35:34 +00:00
|
|
|
*
|
|
|
|
* This flag is cleared after 255 * STATION_RATING_TICKS of not having seen a pickup.
|
2012-11-12 18:11:05 +00:00
|
|
|
*/
|
2014-05-11 18:02:11 +00:00
|
|
|
GES_RATING,
|
2012-11-12 18:11:05 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Set when a vehicle ever delivered cargo to the station for final delivery.
|
|
|
|
* This flag is never cleared.
|
|
|
|
*/
|
|
|
|
GES_EVER_ACCEPTED,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set when cargo was delivered for final delivery last month.
|
|
|
|
* This flag is set to the value of GES_CURRENT_MONTH at the start of each month.
|
|
|
|
*/
|
|
|
|
GES_LAST_MONTH,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set when cargo was delivered for final delivery this month.
|
|
|
|
* This flag is reset on the beginning of every month.
|
|
|
|
*/
|
|
|
|
GES_CURRENT_MONTH,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set when cargo was delivered for final delivery during the current STATION_ACCEPTANCE_TICKS interval.
|
|
|
|
* This flag is reset every STATION_ACCEPTANCE_TICKS ticks.
|
|
|
|
*/
|
|
|
|
GES_ACCEPTED_BIGTICK,
|
2007-08-26 13:55:36 +00:00
|
|
|
};
|
|
|
|
|
2007-01-14 20:00:25 +00:00
|
|
|
GoodsEntry() :
|
2014-05-11 18:02:11 +00:00
|
|
|
status(0),
|
2012-11-12 21:59:02 +00:00
|
|
|
time_since_pickup(255),
|
2007-03-08 20:50:27 +00:00
|
|
|
rating(INITIAL_STATION_RATING),
|
2007-01-14 20:00:25 +00:00
|
|
|
last_speed(0),
|
2013-05-19 14:11:20 +00:00
|
|
|
last_age(255),
|
2013-11-23 13:12:19 +00:00
|
|
|
amount_fract(0),
|
2013-05-19 14:11:20 +00:00
|
|
|
link_graph(INVALID_LINK_GRAPH),
|
2013-06-09 13:04:32 +00:00
|
|
|
node(INVALID_NODE),
|
|
|
|
max_waiting_cargo(0)
|
2007-01-14 20:00:25 +00:00
|
|
|
{}
|
|
|
|
|
2014-05-11 18:02:11 +00:00
|
|
|
byte status; ///< Status of this cargo, see #GoodsEntryStatus.
|
2012-11-12 18:11:05 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of rating-intervals (up to 255) since the last vehicle tried to load this cargo.
|
|
|
|
* The unit used is STATION_RATING_TICKS.
|
|
|
|
* This does not imply there was any cargo to load.
|
|
|
|
*/
|
2012-11-12 21:59:02 +00:00
|
|
|
byte time_since_pickup;
|
2012-11-12 18:11:05 +00:00
|
|
|
|
2012-09-21 09:47:21 +00:00
|
|
|
byte rating; ///< %Station rating for this cargo.
|
2012-11-12 18:11:05 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Maximum speed (up to 255) of the last vehicle that tried to load this cargo.
|
|
|
|
* This does not imply there was any cargo to load.
|
|
|
|
* The unit used is a special vehicle-specific speed unit for station ratings.
|
|
|
|
* - Trains: km-ish/h
|
|
|
|
* - RV: km-ish/h
|
|
|
|
* - Ships: 0.5 * km-ish/h
|
|
|
|
* - Aircraft: 8 * mph
|
|
|
|
*/
|
|
|
|
byte last_speed;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Age in years (up to 255) of the last vehicle that tried to load this cargo.
|
|
|
|
* This does not imply there was any cargo to load.
|
|
|
|
*/
|
|
|
|
byte last_age;
|
|
|
|
|
2010-09-28 22:00:24 +00:00
|
|
|
byte amount_fract; ///< Fractional part of the amount in the cargo list
|
2009-10-18 13:39:00 +00:00
|
|
|
StationCargoList cargo; ///< The cargo packets of cargo waiting in this station
|
2012-11-12 18:11:26 +00:00
|
|
|
|
2013-05-19 14:11:20 +00:00
|
|
|
LinkGraphID link_graph; ///< Link graph this station belongs to.
|
|
|
|
NodeID node; ///< ID of node in link graph referring to this goods entry.
|
2013-06-09 12:49:47 +00:00
|
|
|
FlowStatMap flows; ///< Planned flows through this station.
|
2013-06-09 13:04:32 +00:00
|
|
|
uint max_waiting_cargo; ///< Max cargo from this station waiting at any station.
|
2013-05-19 14:11:20 +00:00
|
|
|
|
2012-11-12 18:11:26 +00:00
|
|
|
/**
|
|
|
|
* Reports whether a vehicle has ever tried to load the cargo at this station.
|
2014-05-11 18:02:11 +00:00
|
|
|
* This does not imply that there was cargo available for loading. Refer to GES_RATING for that.
|
2012-11-12 18:11:26 +00:00
|
|
|
* @return true if vehicle tried to load.
|
|
|
|
*/
|
|
|
|
bool HasVehicleEverTriedLoading() const { return this->last_speed != 0; }
|
2013-04-06 12:10:16 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Does this cargo have a rating at this station?
|
2014-05-11 18:02:11 +00:00
|
|
|
* @return true if the cargo has a rating, i.e. cargo has been moved to the station.
|
2013-04-06 12:10:16 +00:00
|
|
|
*/
|
|
|
|
inline bool HasRating() const
|
|
|
|
{
|
2014-05-11 18:02:11 +00:00
|
|
|
return HasBit(this->status, GES_RATING);
|
2013-04-06 12:10:16 +00:00
|
|
|
}
|
2013-06-09 12:49:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the best next hop for a cargo packet from station source.
|
|
|
|
* @param source Source of the packet.
|
|
|
|
* @return The chosen next hop or INVALID_STATION if none was found.
|
|
|
|
*/
|
|
|
|
inline StationID GetVia(StationID source) const
|
|
|
|
{
|
|
|
|
FlowStatMap::const_iterator flow_it(this->flows.find(source));
|
|
|
|
return flow_it != this->flows.end() ? flow_it->second.GetVia() : INVALID_STATION;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the best next hop for a cargo packet from station source, optionally
|
|
|
|
* excluding one or two stations.
|
|
|
|
* @param source Source of the packet.
|
|
|
|
* @param excluded If this station would be chosen choose the second best one instead.
|
|
|
|
* @param excluded2 Second station to be excluded, if != INVALID_STATION.
|
|
|
|
* @return The chosen next hop or INVALID_STATION if none was found.
|
|
|
|
*/
|
|
|
|
inline StationID GetVia(StationID source, StationID excluded, StationID excluded2 = INVALID_STATION) const
|
|
|
|
{
|
|
|
|
FlowStatMap::const_iterator flow_it(this->flows.find(source));
|
|
|
|
return flow_it != this->flows.end() ? flow_it->second.GetVia(excluded, excluded2) : INVALID_STATION;
|
|
|
|
}
|
2007-03-07 12:11:48 +00:00
|
|
|
};
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2010-03-18 21:02:20 +00:00
|
|
|
/** All airport-related information. Only valid if tile != INVALID_TILE. */
|
|
|
|
struct Airport : public TileArea {
|
|
|
|
Airport() : TileArea(INVALID_TILE, 0, 0) {}
|
|
|
|
|
2010-08-05 12:00:09 +00:00
|
|
|
uint64 flags; ///< stores which blocks on the airport are taken. was 16 bit earlier on, then 32
|
2012-09-21 09:47:21 +00:00
|
|
|
byte type; ///< Type of this airport, @see AirportTypes
|
2010-08-05 12:00:09 +00:00
|
|
|
byte layout; ///< Airport layout number.
|
2019-04-22 07:11:28 +00:00
|
|
|
Direction rotation; ///< How this airport is rotated.
|
2011-06-12 20:47:45 +00:00
|
|
|
|
|
|
|
PersistentStorage *psa; ///< Persistent storage for NewGRF airports.
|
2010-08-02 21:11:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the AirportSpec that from the airport type of this airport. If there
|
|
|
|
* is no airport (\c tile == INVALID_TILE) then return the dummy AirportSpec.
|
|
|
|
* @return The AirportSpec for this airport.
|
|
|
|
*/
|
2010-03-18 21:02:20 +00:00
|
|
|
const AirportSpec *GetSpec() const
|
|
|
|
{
|
|
|
|
if (this->tile == INVALID_TILE) return &AirportSpec::dummy;
|
|
|
|
return AirportSpec::Get(this->type);
|
|
|
|
}
|
|
|
|
|
2010-08-02 21:11:52 +00:00
|
|
|
/**
|
|
|
|
* Get the finite-state machine for this airport or the finite-state machine
|
|
|
|
* for the dummy airport in case this isn't an airport.
|
|
|
|
* @pre this->type < NEW_AIRPORT_OFFSET.
|
|
|
|
* @return The state machine for this airport.
|
|
|
|
*/
|
2010-03-18 21:02:20 +00:00
|
|
|
const AirportFTAClass *GetFTA() const
|
|
|
|
{
|
|
|
|
return this->GetSpec()->fsm;
|
|
|
|
}
|
2010-03-19 09:48:44 +00:00
|
|
|
|
2010-08-02 21:11:52 +00:00
|
|
|
/** Check if this airport has at least one hangar. */
|
2011-12-20 17:57:56 +00:00
|
|
|
inline bool HasHangar() const
|
2010-03-19 09:48:44 +00:00
|
|
|
{
|
|
|
|
return this->GetSpec()->nof_depots > 0;
|
|
|
|
}
|
2010-03-19 09:58:46 +00:00
|
|
|
|
2010-08-05 12:00:09 +00:00
|
|
|
/**
|
|
|
|
* Add the tileoffset to the base tile of this airport but rotate it first.
|
|
|
|
* The base tile is the northernmost tile of this airport. This function
|
|
|
|
* helps to make sure that getting the tile of a hangar works even for
|
|
|
|
* rotated airport layouts without requiring a rotated array of hangar tiles.
|
|
|
|
* @param tidc The tilediff to add to the airport tile.
|
|
|
|
* @return The tile of this airport plus the rotated offset.
|
|
|
|
*/
|
2011-12-20 17:57:56 +00:00
|
|
|
inline TileIndex GetRotatedTileFromOffset(TileIndexDiffC tidc) const
|
2010-08-05 12:00:09 +00:00
|
|
|
{
|
|
|
|
const AirportSpec *as = this->GetSpec();
|
|
|
|
switch (this->rotation) {
|
|
|
|
case DIR_N: return this->tile + ToTileIndexDiff(tidc);
|
|
|
|
|
|
|
|
case DIR_E: return this->tile + TileDiffXY(tidc.y, as->size_x - 1 - tidc.x);
|
|
|
|
|
|
|
|
case DIR_S: return this->tile + TileDiffXY(as->size_x - 1 - tidc.x, as->size_y - 1 - tidc.y);
|
|
|
|
|
|
|
|
case DIR_W: return this->tile + TileDiffXY(as->size_y - 1 - tidc.y, tidc.x);
|
|
|
|
|
|
|
|
default: NOT_REACHED();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-02 21:11:52 +00:00
|
|
|
/**
|
|
|
|
* Get the first tile of the given hangar.
|
|
|
|
* @param hangar_num The hangar to get the location of.
|
|
|
|
* @pre hangar_num < GetNumHangars().
|
|
|
|
* @return A tile with the given hangar.
|
|
|
|
*/
|
2011-12-20 17:57:56 +00:00
|
|
|
inline TileIndex GetHangarTile(uint hangar_num) const
|
2010-03-19 09:58:46 +00:00
|
|
|
{
|
2010-03-19 11:17:52 +00:00
|
|
|
const AirportSpec *as = this->GetSpec();
|
|
|
|
for (uint i = 0; i < as->nof_depots; i++) {
|
|
|
|
if (as->depot_table[i].hangar_num == hangar_num) {
|
2010-08-05 12:00:09 +00:00
|
|
|
return this->GetRotatedTileFromOffset(as->depot_table[i].ti);
|
2010-03-19 11:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
2010-08-02 21:11:52 +00:00
|
|
|
/**
|
2011-08-13 12:44:42 +00:00
|
|
|
* Get the exit direction of the hangar at a specific tile.
|
|
|
|
* @param tile The tile to query.
|
|
|
|
* @pre IsHangarTile(tile).
|
|
|
|
* @return The exit direction of the hangar, taking airport rotation into account.
|
|
|
|
*/
|
2011-12-20 17:57:56 +00:00
|
|
|
inline Direction GetHangarExitDirection(TileIndex tile) const
|
2011-08-13 12:44:42 +00:00
|
|
|
{
|
|
|
|
const AirportSpec *as = this->GetSpec();
|
|
|
|
const HangarTileTable *htt = GetHangarDataByTile(tile);
|
|
|
|
return ChangeDir(htt->dir, DirDifference(this->rotation, as->rotation[0]));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the hangar number of the hangar at a specific tile.
|
2010-08-02 21:11:52 +00:00
|
|
|
* @param tile The tile to query.
|
|
|
|
* @pre IsHangarTile(tile).
|
|
|
|
* @return The hangar number of the hangar at the given tile.
|
|
|
|
*/
|
2011-12-20 17:57:56 +00:00
|
|
|
inline uint GetHangarNum(TileIndex tile) const
|
2010-03-19 11:17:52 +00:00
|
|
|
{
|
2011-08-13 12:44:42 +00:00
|
|
|
const HangarTileTable *htt = GetHangarDataByTile(tile);
|
|
|
|
return htt->hangar_num;
|
2010-03-19 11:17:52 +00:00
|
|
|
}
|
|
|
|
|
2010-08-02 21:11:52 +00:00
|
|
|
/** Get the number of hangars on this airport. */
|
2011-12-20 17:57:56 +00:00
|
|
|
inline uint GetNumHangars() const
|
2010-03-19 11:17:52 +00:00
|
|
|
{
|
|
|
|
uint num = 0;
|
|
|
|
uint counted = 0;
|
|
|
|
const AirportSpec *as = this->GetSpec();
|
|
|
|
for (uint i = 0; i < as->nof_depots; i++) {
|
|
|
|
if (!HasBit(counted, as->depot_table[i].hangar_num)) {
|
|
|
|
num++;
|
|
|
|
SetBit(counted, as->depot_table[i].hangar_num);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return num;
|
2010-03-19 09:58:46 +00:00
|
|
|
}
|
2011-08-13 12:44:42 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* Retrieve hangar information of a hangar at a given tile.
|
|
|
|
* @param tile %Tile containing the hangar.
|
|
|
|
* @return The requested hangar information.
|
|
|
|
* @pre The \a tile must be at a hangar tile at an airport.
|
|
|
|
*/
|
2011-12-20 17:57:56 +00:00
|
|
|
inline const HangarTileTable *GetHangarDataByTile(TileIndex tile) const
|
2011-08-13 12:44:42 +00:00
|
|
|
{
|
|
|
|
const AirportSpec *as = this->GetSpec();
|
|
|
|
for (uint i = 0; i < as->nof_depots; i++) {
|
|
|
|
if (this->GetRotatedTileFromOffset(as->depot_table[i].ti) == tile) {
|
|
|
|
return as->depot_table + i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NOT_REACHED();
|
|
|
|
}
|
2010-03-18 21:02:20 +00:00
|
|
|
};
|
2009-07-18 18:39:17 +00:00
|
|
|
|
2022-02-19 18:08:23 +00:00
|
|
|
struct IndustryListEntry {
|
|
|
|
uint distance;
|
|
|
|
Industry *industry;
|
|
|
|
|
|
|
|
bool operator== (const IndustryListEntry &other) const { return this->distance == other.distance && this->industry == other.industry; };
|
|
|
|
};
|
|
|
|
|
2019-02-24 19:16:24 +00:00
|
|
|
struct IndustryCompare {
|
2022-02-19 18:08:23 +00:00
|
|
|
bool operator() (const IndustryListEntry &lhs, const IndustryListEntry &rhs) const;
|
2019-02-24 19:16:24 +00:00
|
|
|
};
|
|
|
|
|
2022-02-19 18:08:23 +00:00
|
|
|
typedef std::set<IndustryListEntry, IndustryCompare> IndustryList;
|
2009-06-25 15:42:03 +00:00
|
|
|
|
2008-10-19 15:39:12 +00:00
|
|
|
/** Station data structure */
|
2011-12-18 17:17:18 +00:00
|
|
|
struct Station FINAL : SpecializedStation<Station, false> {
|
2007-08-24 19:19:18 +00:00
|
|
|
public:
|
2008-03-31 00:06:17 +00:00
|
|
|
RoadStop *GetPrimaryRoadStop(RoadStopType type) const
|
2007-08-24 19:19:18 +00:00
|
|
|
{
|
2008-03-31 00:06:17 +00:00
|
|
|
return type == ROADSTOP_BUS ? bus_stops : truck_stops;
|
2007-08-24 19:19:18 +00:00
|
|
|
}
|
|
|
|
|
2009-05-22 20:18:45 +00:00
|
|
|
RoadStop *GetPrimaryRoadStop(const struct RoadVehicle *v) const;
|
2007-08-24 19:19:18 +00:00
|
|
|
|
2009-07-25 08:54:19 +00:00
|
|
|
RoadStop *bus_stops; ///< All the road stops
|
2009-12-02 16:20:44 +00:00
|
|
|
TileArea bus_station; ///< Tile area the bus 'station' part covers
|
2009-07-25 08:54:19 +00:00
|
|
|
RoadStop *truck_stops; ///< All the truck stops
|
2009-12-02 16:20:44 +00:00
|
|
|
TileArea truck_station; ///< Tile area the truck 'station' part covers
|
|
|
|
|
2019-03-11 10:37:47 +00:00
|
|
|
Airport airport; ///< Tile area the airport covers
|
|
|
|
TileArea ship_station; ///< Tile area the ship 'station' part covers
|
|
|
|
TileArea docking_station; ///< Tile area the docking tiles cover
|
2008-11-19 23:55:34 +00:00
|
|
|
|
|
|
|
IndustryType indtype; ///< Industry type to get the name from
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2019-02-14 21:07:15 +00:00
|
|
|
BitmapTileArea catchment_tiles; ///< NOSAVE: Set of individual tiles covered by catchment area
|
|
|
|
|
2019-04-21 22:13:27 +00:00
|
|
|
StationHadVehicleOfType had_vehicle_of_type;
|
2004-09-10 19:02:27 +00:00
|
|
|
|
2004-08-09 17:04:08 +00:00
|
|
|
byte time_since_load;
|
|
|
|
byte time_since_unload;
|
|
|
|
|
2006-04-18 08:50:17 +00:00
|
|
|
byte last_vehicle_type;
|
2007-04-20 08:00:30 +00:00
|
|
|
std::list<Vehicle *> loading_vehicles;
|
2008-10-19 15:39:12 +00:00
|
|
|
GoodsEntry goods[NUM_CARGO]; ///< Goods at this station
|
2018-05-21 21:08:39 +00:00
|
|
|
CargoTypes always_accepted; ///< Bitmask of always accepted cargo types (by houses, HQs, industry tiles when industry doesn't accept cargo)
|
2005-01-29 19:41:44 +00:00
|
|
|
|
2019-02-24 19:16:24 +00:00
|
|
|
IndustryList industries_near; ///< Cached list of industries near the station that can accept cargo, @see DeliverGoodsToIndustry()
|
|
|
|
Industry *industry; ///< NOSAVE: Associated industry for neutral stations. (Rebuilt on load from Industry->st)
|
2009-06-25 15:42:03 +00:00
|
|
|
|
2008-12-26 18:01:15 +00:00
|
|
|
Station(TileIndex tile = INVALID_TILE);
|
2009-05-22 15:13:50 +00:00
|
|
|
~Station();
|
2007-01-14 19:18:50 +00:00
|
|
|
|
2009-07-04 11:26:57 +00:00
|
|
|
void AddFacility(StationFacility new_facility_bit, TileIndex facil_xy);
|
2007-09-09 10:13:17 +00:00
|
|
|
|
2007-06-08 09:35:39 +00:00
|
|
|
void MarkTilesDirty(bool cargo_change) const;
|
2009-06-25 23:49:59 +00:00
|
|
|
|
2019-03-03 22:25:13 +00:00
|
|
|
void UpdateVirtCoord() override;
|
2009-07-07 16:43:58 +00:00
|
|
|
|
2019-03-23 11:39:13 +00:00
|
|
|
void MoveSign(TileIndex new_xy) override;
|
|
|
|
|
2012-12-31 15:34:21 +00:00
|
|
|
void AfterStationTileSetChange(bool adding, StationType type);
|
|
|
|
|
2019-03-03 22:25:13 +00:00
|
|
|
uint GetPlatformLength(TileIndex tile, DiagDirection dir) const override;
|
|
|
|
uint GetPlatformLength(TileIndex tile) const override;
|
2019-09-23 17:45:37 +00:00
|
|
|
void RecomputeCatchment(bool no_clear_nearby_lists = false);
|
2019-02-14 21:07:15 +00:00
|
|
|
static void RecomputeCatchmentForAll();
|
2009-06-25 15:42:03 +00:00
|
|
|
|
2008-10-25 14:19:09 +00:00
|
|
|
uint GetCatchmentRadius() const;
|
2009-08-07 21:11:58 +00:00
|
|
|
Rect GetCatchmentRect() const;
|
2019-02-14 21:07:15 +00:00
|
|
|
bool CatchmentCoversTown(TownID t) const;
|
2022-02-19 18:08:23 +00:00
|
|
|
void AddIndustryToDeliver(Industry *ind, TileIndex tile);
|
|
|
|
void RemoveIndustryToDeliver(Industry *ind);
|
2019-02-14 21:07:15 +00:00
|
|
|
void RemoveFromAllNearbyLists();
|
|
|
|
|
|
|
|
inline bool TileIsInCatchment(TileIndex tile) const
|
|
|
|
{
|
|
|
|
return this->catchment_tiles.HasTile(tile);
|
|
|
|
}
|
2009-06-24 17:39:54 +00:00
|
|
|
|
2019-03-03 22:25:13 +00:00
|
|
|
inline bool TileBelongsToRailStation(TileIndex tile) const override
|
2009-06-25 23:49:59 +00:00
|
|
|
{
|
2009-07-24 11:47:12 +00:00
|
|
|
return IsRailStationTile(tile) && GetStationIndex(tile) == this->index;
|
2009-06-25 23:49:59 +00:00
|
|
|
}
|
|
|
|
|
2022-11-06 15:01:27 +00:00
|
|
|
inline bool TileBelongsToRoadStop(TileIndex tile) const
|
|
|
|
{
|
|
|
|
return IsRoadStopTile(tile) && GetStationIndex(tile) == this->index;
|
|
|
|
}
|
|
|
|
|
2011-12-20 17:57:56 +00:00
|
|
|
inline bool TileBelongsToAirport(TileIndex tile) const
|
2010-01-18 12:32:50 +00:00
|
|
|
{
|
|
|
|
return IsAirportTile(tile) && GetStationIndex(tile) == this->index;
|
|
|
|
}
|
|
|
|
|
2019-03-03 22:25:13 +00:00
|
|
|
uint32 GetNewGRFVariable(const ResolverObject &object, byte variable, byte parameter, bool *available) const override;
|
2009-07-17 19:44:13 +00:00
|
|
|
|
2019-03-03 22:25:13 +00:00
|
|
|
void GetTileArea(TileArea *ta, StationType type) const override;
|
2004-08-09 17:04:08 +00:00
|
|
|
};
|
|
|
|
|
2011-12-09 19:30:30 +00:00
|
|
|
/** Iterator to iterate over all tiles belonging to an airport. */
|
|
|
|
class AirportTileIterator : public OrthogonalTileIterator {
|
|
|
|
private:
|
|
|
|
const Station *st; ///< The station the airport is a part of.
|
|
|
|
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Construct the iterator.
|
2018-10-28 02:17:36 +00:00
|
|
|
* @param st Station the airport is part of.
|
2011-12-09 19:30:30 +00:00
|
|
|
*/
|
|
|
|
AirportTileIterator(const Station *st) : OrthogonalTileIterator(st->airport), st(st)
|
|
|
|
{
|
|
|
|
if (!st->TileBelongsToAirport(this->tile)) ++(*this);
|
|
|
|
}
|
|
|
|
|
2011-12-20 17:57:56 +00:00
|
|
|
inline TileIterator& operator ++()
|
2011-12-09 19:30:30 +00:00
|
|
|
{
|
|
|
|
(*this).OrthogonalTileIterator::operator++();
|
|
|
|
while (this->tile != INVALID_TILE && !st->TileBelongsToAirport(this->tile)) {
|
|
|
|
(*this).OrthogonalTileIterator::operator++();
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2023-01-13 16:54:53 +00:00
|
|
|
virtual std::unique_ptr<TileIterator> Clone() const
|
2011-12-09 19:30:30 +00:00
|
|
|
{
|
2023-01-13 16:54:53 +00:00
|
|
|
return std::make_unique<AirportTileIterator>(*this);
|
2011-12-09 19:30:30 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-02-18 20:14:52 +00:00
|
|
|
void RebuildStationKdtree();
|
|
|
|
|
2020-05-11 22:36:28 +00:00
|
|
|
/**
|
|
|
|
* Call a function on all stations that have any part of the requested area within their catchment.
|
2020-05-11 23:21:14 +00:00
|
|
|
* @tparam Func The type of funcion to call
|
|
|
|
* @param area The TileArea to check
|
|
|
|
* @param func The function to call, must take two parameters: Station* and TileIndex and return true
|
|
|
|
* if coverage of that tile is acceptable for a given station or false if search should continue
|
2020-05-11 22:36:28 +00:00
|
|
|
*/
|
|
|
|
template<typename Func>
|
|
|
|
void ForAllStationsAroundTiles(const TileArea &ta, Func func)
|
|
|
|
{
|
2021-08-31 12:31:37 +00:00
|
|
|
/* There are no stations, so we will never find anything. */
|
|
|
|
if (Station::GetNumItems() == 0) return;
|
|
|
|
|
2020-05-11 22:36:28 +00:00
|
|
|
/* Not using, or don't have a nearby stations list, so we need to scan. */
|
|
|
|
std::set<StationID> seen_stations;
|
|
|
|
|
|
|
|
/* Scan an area around the building covering the maximum possible station
|
|
|
|
* to find the possible nearby stations. */
|
|
|
|
uint max_c = _settings_game.station.modified_catchment ? MAX_CATCHMENT : CA_UNMODIFIED;
|
|
|
|
TileArea ta_ext = TileArea(ta).Expand(max_c);
|
2021-05-12 14:45:28 +00:00
|
|
|
for (TileIndex tile : ta_ext) {
|
2020-05-11 22:36:28 +00:00
|
|
|
if (IsTileType(tile, MP_STATION)) seen_stations.insert(GetStationIndex(tile));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (StationID stationid : seen_stations) {
|
|
|
|
Station *st = Station::GetIfValid(stationid);
|
|
|
|
if (st == nullptr) continue; /* Waypoint */
|
|
|
|
|
|
|
|
/* Check if station is attached to an industry */
|
|
|
|
if (!_settings_game.station.serve_neutral_industries && st->industry != nullptr) continue;
|
|
|
|
|
|
|
|
/* Test if the tile is within the station's catchment */
|
2021-05-12 14:45:28 +00:00
|
|
|
for (TileIndex tile : ta) {
|
2020-05-11 22:36:28 +00:00
|
|
|
if (st->TileIsInCatchment(tile)) {
|
2020-05-11 23:21:14 +00:00
|
|
|
if (func(st, tile)) break;
|
2020-05-11 22:36:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-31 00:06:17 +00:00
|
|
|
#endif /* STATION_BASE_H */
|