2005-07-24 15:12:37 +01:00
|
|
|
/* $Id$ */
|
|
|
|
|
2009-08-21 21:21:05 +01: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 01:06:17 +01:00
|
|
|
/** @file station_base.h Base classes/functions for stations. */
|
2007-04-04 02:35:16 +01:00
|
|
|
|
2008-03-31 01:06:17 +01:00
|
|
|
#ifndef STATION_BASE_H
|
|
|
|
#define STATION_BASE_H
|
2004-08-09 18:04:08 +01:00
|
|
|
|
2009-07-22 12:35:35 +01:00
|
|
|
#include "base_station_base.h"
|
2010-02-22 16:09:26 +00:00
|
|
|
#include "newgrf_airport.h"
|
2007-06-22 12:58:59 +01:00
|
|
|
#include "cargopacket.h"
|
2008-11-19 23:55:34 +00:00
|
|
|
#include "industry_type.h"
|
2010-08-05 13:06:13 +01:00
|
|
|
#include "newgrf_storage.h"
|
2004-08-09 18:04:08 +01:00
|
|
|
|
2009-07-22 09:59:57 +01:00
|
|
|
typedef Pool<BaseStation, StationID, 32, 64000> StationPool;
|
2009-05-22 16:13:50 +01:00
|
|
|
extern StationPool _station_pool;
|
2007-08-02 09:47:56 +01:00
|
|
|
|
2007-03-08 20:50:27 +00:00
|
|
|
static const byte INITIAL_STATION_RATING = 175;
|
2007-01-14 20:00:25 +00:00
|
|
|
|
2011-06-04 22:19:00 +01:00
|
|
|
/**
|
|
|
|
* Stores station stats for a single cargo.
|
|
|
|
*/
|
2007-03-07 12:11:48 +00:00
|
|
|
struct GoodsEntry {
|
2011-06-04 22:19:00 +01:00
|
|
|
/** Status of this cargo for the station. */
|
2011-06-04 22:17:07 +01:00
|
|
|
enum GoodsEntryStatus {
|
2011-06-04 22:21:00 +01:00
|
|
|
GES_ACCEPTANCE, ///< This cargo is currently being accepted by the station.
|
|
|
|
GES_PICKUP, ///< This cargo has been picked up at this station at least once.
|
|
|
|
GES_EVER_ACCEPTED, ///< The cargo has been accepted at least once.
|
|
|
|
GES_LAST_MONTH, ///< The cargo was accepted last month.
|
|
|
|
GES_CURRENT_MONTH, ///< The cargo was accepted this month.
|
|
|
|
GES_ACCEPTED_BIGTICK, ///< The cargo has been accepted since the last periodic processing.
|
2007-08-26 14:55:36 +01:00
|
|
|
};
|
|
|
|
|
2007-01-14 20:00:25 +00:00
|
|
|
GoodsEntry() :
|
2007-08-26 14:55:36 +01:00
|
|
|
acceptance_pickup(0),
|
2007-06-22 18:34:04 +01:00
|
|
|
days_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),
|
2007-06-22 12:58:59 +01:00
|
|
|
last_age(255)
|
2007-01-14 20:00:25 +00:00
|
|
|
{}
|
|
|
|
|
2011-06-04 22:19:00 +01:00
|
|
|
byte acceptance_pickup; ///< Status of this cargo, see #GoodsEntryStatus.
|
|
|
|
byte days_since_pickup; ///< Number of days since the last pickup for this cargo (up to 255).
|
|
|
|
byte rating; ///< Station rating for this cargo.
|
|
|
|
byte last_speed; ///< Maximum speed of the last vehicle that picked up this cargo (up to 255).
|
|
|
|
byte last_age; ///< Age in years of the last vehicle that picked up this cargo.
|
2010-09-28 23:00:24 +01:00
|
|
|
byte amount_fract; ///< Fractional part of the amount in the cargo list
|
2009-10-18 14:39:00 +01:00
|
|
|
StationCargoList cargo; ///< The cargo packets of cargo waiting in this station
|
2007-03-07 12:11:48 +00:00
|
|
|
};
|
2004-08-09 18:04:08 +01: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 13:00:09 +01:00
|
|
|
uint64 flags; ///< stores which blocks on the airport are taken. was 16 bit earlier on, then 32
|
|
|
|
byte type; ///< Type of this airport, @see AirportTypes.
|
|
|
|
byte layout; ///< Airport layout number.
|
|
|
|
Direction rotation; ///< How this airport is rotated.
|
2011-06-12 21:47:45 +01:00
|
|
|
|
|
|
|
PersistentStorage *psa; ///< Persistent storage for NewGRF airports.
|
2010-08-02 22:11:52 +01: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 22:11:52 +01: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 22:11:52 +01: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 13:00:09 +01: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 13:00:09 +01: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 22:11:52 +01: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 13:00:09 +01:00
|
|
|
return this->GetRotatedTileFromOffset(as->depot_table[i].ti);
|
2010-03-19 11:17:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
2010-08-02 22:11:52 +01:00
|
|
|
/**
|
2011-08-13 13:44:42 +01: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 13:44:42 +01: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 22:11:52 +01: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 13:44:42 +01:00
|
|
|
const HangarTileTable *htt = GetHangarDataByTile(tile);
|
|
|
|
return htt->hangar_num;
|
2010-03-19 11:17:52 +00:00
|
|
|
}
|
|
|
|
|
2010-08-02 22:11:52 +01: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 13:44:42 +01: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 13:44:42 +01: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 19:39:17 +01:00
|
|
|
|
2009-06-25 16:42:03 +01:00
|
|
|
typedef SmallVector<Industry *, 2> IndustryVector;
|
|
|
|
|
2008-10-19 16:39:12 +01:00
|
|
|
/** Station data structure */
|
2011-12-18 17:17:18 +00:00
|
|
|
struct Station FINAL : SpecializedStation<Station, false> {
|
2007-08-24 20:19:18 +01:00
|
|
|
public:
|
2008-03-31 01:06:17 +01:00
|
|
|
RoadStop *GetPrimaryRoadStop(RoadStopType type) const
|
2007-08-24 20:19:18 +01:00
|
|
|
{
|
2008-03-31 01:06:17 +01:00
|
|
|
return type == ROADSTOP_BUS ? bus_stops : truck_stops;
|
2007-08-24 20:19:18 +01:00
|
|
|
}
|
|
|
|
|
2009-05-22 21:18:45 +01:00
|
|
|
RoadStop *GetPrimaryRoadStop(const struct RoadVehicle *v) const;
|
2007-08-24 20:19:18 +01:00
|
|
|
|
2009-07-25 09:54:19 +01: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 09:54:19 +01: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
|
|
|
|
|
2010-03-18 21:02:20 +00:00
|
|
|
Airport airport; ///< Tile area the airport covers
|
2009-07-25 09:54:19 +01:00
|
|
|
TileIndex dock_tile; ///< The location of the dock
|
2008-11-19 23:55:34 +00:00
|
|
|
|
|
|
|
IndustryType indtype; ///< Industry type to get the name from
|
2004-08-09 18:04:08 +01:00
|
|
|
|
2009-07-04 12:26:57 +01:00
|
|
|
StationHadVehicleOfTypeByte had_vehicle_of_type;
|
2004-09-10 20:02:27 +01:00
|
|
|
|
2004-08-09 18:04:08 +01:00
|
|
|
byte time_since_load;
|
|
|
|
byte time_since_unload;
|
|
|
|
|
2006-04-18 09:50:17 +01:00
|
|
|
byte last_vehicle_type;
|
2007-04-20 09:00:30 +01:00
|
|
|
std::list<Vehicle *> loading_vehicles;
|
2008-10-19 16:39:12 +01:00
|
|
|
GoodsEntry goods[NUM_CARGO]; ///< Goods at this station
|
2009-09-20 19:52:12 +01:00
|
|
|
uint32 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
|
|
|
|
2009-06-25 16:42:03 +01:00
|
|
|
IndustryVector industries_near; ///< Cached list of industries near the station that can accept cargo, @see DeliverGoodsToIndustry()
|
|
|
|
|
2008-12-26 18:01:15 +00:00
|
|
|
Station(TileIndex tile = INVALID_TILE);
|
2009-05-22 16:13:50 +01:00
|
|
|
~Station();
|
2007-01-14 19:18:50 +00:00
|
|
|
|
2009-07-04 12:26:57 +01:00
|
|
|
void AddFacility(StationFacility new_facility_bit, TileIndex facil_xy);
|
2007-09-09 11:13:17 +01:00
|
|
|
|
2007-06-08 10:35:39 +01:00
|
|
|
void MarkTilesDirty(bool cargo_change) const;
|
2009-06-26 00:49:59 +01:00
|
|
|
|
2009-07-07 17:43:58 +01:00
|
|
|
void UpdateVirtCoord();
|
|
|
|
|
2009-07-24 16:18:25 +01:00
|
|
|
/* virtual */ uint GetPlatformLength(TileIndex tile, DiagDirection dir) const;
|
|
|
|
/* virtual */ uint GetPlatformLength(TileIndex tile) const;
|
2009-06-25 16:42:03 +01:00
|
|
|
void RecomputeIndustriesNear();
|
|
|
|
static void RecomputeIndustriesNearForAll();
|
|
|
|
|
2008-10-25 15:19:09 +01:00
|
|
|
uint GetCatchmentRadius() const;
|
2009-08-07 22:11:58 +01:00
|
|
|
Rect GetCatchmentRect() const;
|
2009-06-24 18:39:54 +01:00
|
|
|
|
2011-12-20 17:57:56 +00:00
|
|
|
/* virtual */ inline bool TileBelongsToRailStation(TileIndex tile) const
|
2009-06-26 00:49:59 +01:00
|
|
|
{
|
2009-07-24 12:47:12 +01:00
|
|
|
return IsRailStationTile(tile) && GetStationIndex(tile) == this->index;
|
2009-06-26 00:49:59 +01:00
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2009-07-17 20:44:13 +01:00
|
|
|
/* virtual */ uint32 GetNewGRFVariable(const ResolverObject *object, byte variable, byte parameter, bool *available) const;
|
|
|
|
|
2009-07-21 12:11:05 +01:00
|
|
|
/* virtual */ void GetTileArea(TileArea *ta, StationType type) const;
|
2004-08-09 18:04:08 +01:00
|
|
|
};
|
|
|
|
|
2009-07-22 09:59:57 +01:00
|
|
|
#define FOR_ALL_STATIONS(var) FOR_ALL_BASE_STATIONS_OF_TYPE(Station, var)
|
2005-01-06 22:31:58 +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.
|
|
|
|
* @param ta Area, i.e. begin point and width/height of to-be-iterated area.
|
|
|
|
*/
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual TileIterator *Clone() const
|
|
|
|
{
|
|
|
|
return new AirportTileIterator(*this);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2008-03-31 01:06:17 +01:00
|
|
|
#endif /* STATION_BASE_H */
|