Files @ r26179:ad5479cbfaa1
Branch filter:

Location: cpp/openttd-patchpack/source/src/cargotype.h - annotation

dP
Change: Deliver cargo to the closest industry first (#9536)
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r18668:b656d614c4fd
r6123:049e9624d068
r6091:2faa7d307565
r6091:2faa7d307565
r6091:2faa7d307565
r12932:8c04eed8b65b
r8119:8fdb3a371896
r8123:dde0a9a84019
r8959:484dbfc31293
r9126:35955a7b9d9e
r25785:f792cdc9a9ff
r25783:02166e47d365
r23628:b9e9809abe89
r6091:2faa7d307565
r13874:a3b7b89e1abf
r6091:2faa7d307565
r6091:2faa7d307565
r13874:a3b7b89e1abf
r26116:c6706853a5ab
r18444:c1e881f954d6
r18444:c1e881f954d6
r18444:c1e881f954d6
r18444:c1e881f954d6
r18444:c1e881f954d6
r18444:c1e881f954d6
r18444:c1e881f954d6
r18444:c1e881f954d6
r18444:c1e881f954d6
r6311:78f38fcd68ca
r6311:78f38fcd68ca
r13874:a3b7b89e1abf
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r13456:7f9200df57a3
r19944:25a78576fb5e
r18737:1ee35da7fa2f
r13456:7f9200df57a3
r6091:2faa7d307565
r13874:a3b7b89e1abf
r12236:ae676539361d
r13874:a3b7b89e1abf
r6248:b940b09d7ab8
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r6091:2faa7d307565
r6091:2faa7d307565
r13874:a3b7b89e1abf
r18994:b2513f4fbc6f
r25866:d941c036e59d
r6091:2faa7d307565
r6091:2faa7d307565
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r6091:2faa7d307565
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r6091:2faa7d307565
r13874:a3b7b89e1abf
r6091:2faa7d307565
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r6365:410001496130
r6122:508316e17f8e
r12932:8c04eed8b65b
r12932:8c04eed8b65b
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r18782:6453522c2154
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r18782:6453522c2154
r12236:ae676539361d
r12236:ae676539361d
r12236:ae676539361d
r12405:ba094e765533
r12405:ba094e765533
r12409:5f538763905f
r12409:5f538763905f
r12405:ba094e765533
r18782:6453522c2154
r12405:ba094e765533
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r18782:6453522c2154
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12408:d7d52eed8c09
r12405:ba094e765533
r12405:ba094e765533
r13871:7588cd2474cc
r13871:7588cd2474cc
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r25310:e304033a69f3
r12405:ba094e765533
r12409:5f538763905f
r12405:ba094e765533
r12405:ba094e765533
r6248:b940b09d7ab8
r6091:2faa7d307565
r22867:9bff1c966805
r22867:9bff1c966805
r6113:c77736e8b915
r6091:2faa7d307565
r6143:9ed364174dfd
r6460:74b53af67ae0
r25472:a02bcac2091b
r6113:c77736e8b915
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r25783:02166e47d365
r14922:51cba7fe7b9b
r15610:623a23fb6560
r15610:623a23fb6560
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r13456:7f9200df57a3
r12236:ae676539361d
r12405:ba094e765533
r12236:ae676539361d
r12236:ae676539361d
r25785:f792cdc9a9ff
r15151:8e6202164b81
r6091:2faa7d307565
/*
 * 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/>.
 */

/** @file cargotype.h Types/functions related to cargoes. */

#ifndef CARGOTYPE_H
#define CARGOTYPE_H

#include "economy_type.h"
#include "cargo_type.h"
#include "gfx_type.h"
#include "strings_type.h"
#include "landscape_type.h"
#include "core/bitmath_func.hpp"
#include "core/span_type.hpp"
#include <vector>

/** Globally unique label of a cargo type. */
typedef uint32 CargoLabel;

/** Town growth effect when delivering cargo. */
enum TownEffect : byte {
	TE_BEGIN = 0,
	TE_NONE = TE_BEGIN, ///< Cargo has no effect.
	TE_PASSENGERS,      ///< Cargo behaves passenger-like.
	TE_MAIL,            ///< Cargo behaves mail-like.
	TE_GOODS,           ///< Cargo behaves goods/candy-like.
	TE_WATER,           ///< Cargo behaves water-like.
	TE_FOOD,            ///< Cargo behaves food/fizzy-drinks-like.
	TE_END,             ///< End of town effects.
	NUM_TE = TE_END,    ///< Amount of town effects.
};

/** Cargo classes. */
enum CargoClass {
	CC_NOAVAILABLE  = 0,       ///< No cargo class has been specified
	CC_PASSENGERS   = 1 <<  0, ///< Passengers
	CC_MAIL         = 1 <<  1, ///< Mail
	CC_EXPRESS      = 1 <<  2, ///< Express cargo (Goods, Food, Candy, but also possible for passengers)
	CC_ARMOURED     = 1 <<  3, ///< Armoured cargo (Valuables, Gold, Diamonds)
	CC_BULK         = 1 <<  4, ///< Bulk cargo (Coal, Grain etc., Ores, Fruit)
	CC_PIECE_GOODS  = 1 <<  5, ///< Piece goods (Livestock, Wood, Steel, Paper)
	CC_LIQUID       = 1 <<  6, ///< Liquids (Oil, Water, Rubber)
	CC_REFRIGERATED = 1 <<  7, ///< Refrigerated cargo (Food, Fruit)
	CC_HAZARDOUS    = 1 <<  8, ///< Hazardous cargo (Nuclear Fuel, Explosives, etc.)
	CC_COVERED      = 1 <<  9, ///< Covered/Sheltered Freight (Transportation in Box Vans, Silo Wagons, etc.)
	CC_SPECIAL      = 1 << 15, ///< Special bit used for livery refit tricks instead of normal cargoes.
};

static const byte INVALID_CARGO = 0xFF; ///< Constant representing invalid cargo

/** Specification of a cargo type. */
struct CargoSpec {
	uint8 bitnum;                    ///< Cargo bit number, is #INVALID_CARGO for a non-used spec.
	CargoLabel label;                ///< Unique label of the cargo type.
	uint8 legend_colour;
	uint8 rating_colour;
	uint8 weight;                    ///< Weight of a single unit of this cargo type in 1/16 ton (62.5 kg).
	uint16 multiplier;               ///< Capacity multiplier for vehicles. (8 fractional bits)
	int32 initial_payment;           ///< Initial payment rate before inflation is applied.
	uint8 transit_days[2];

	bool is_freight;                 ///< Cargo type is considered to be freight (affects train freight multiplier).
	TownEffect town_effect;          ///< The effect that delivering this cargo type has on towns. Also affects destination of subsidies.
	uint8 callback_mask;             ///< Bitmask of cargo callbacks that have to be called

	StringID name;                   ///< Name of this type of cargo.
	StringID name_single;            ///< Name of a single entity of this type of cargo.
	StringID units_volume;           ///< Name of a single unit of cargo of this type.
	StringID quantifier;             ///< Text for multiple units of cargo of this type.
	StringID abbrev;                 ///< Two letter abbreviation for this cargo type.

	SpriteID sprite;                 ///< Icon to display this cargo type, may be \c 0xFFF (which means to resolve an action123 chain).

	uint16 classes;                  ///< Classes of this cargo type. @see CargoClass
	const struct GRFFile *grffile;   ///< NewGRF where #group belongs to.
	const struct SpriteGroup *group;

	Money current_payment;

	/**
	 * Determines index of this cargospec
	 * @return index (in the CargoSpec::array array)
	 */
	inline CargoID Index() const
	{
		return this - CargoSpec::array;
	}

	/**
	 * Tests for validity of this cargospec
	 * @return is this cargospec valid?
	 * @note assert(cs->IsValid()) can be triggered when GRF config is modified
	 */
	inline bool IsValid() const
	{
		return this->bitnum != INVALID_CARGO;
	}

	/**
	 * Total number of cargospecs, both valid and invalid
	 * @return length of CargoSpec::array
	 */
	static inline size_t GetArraySize()
	{
		return lengthof(CargoSpec::array);
	}

	/**
	 * Retrieve cargo details for the given cargo ID
	 * @param index ID of cargo
	 * @pre index is a valid cargo ID
	 */
	static inline CargoSpec *Get(size_t index)
	{
		assert(index < lengthof(CargoSpec::array));
		return &CargoSpec::array[index];
	}

	SpriteID GetCargoIcon() const;

	/**
	 * Iterator to iterate all valid CargoSpec
	 */
	struct Iterator {
		typedef CargoSpec value_type;
		typedef CargoSpec *pointer;
		typedef CargoSpec &reference;
		typedef size_t difference_type;
		typedef std::forward_iterator_tag iterator_category;

		explicit Iterator(size_t index) : index(index)
		{
			this->ValidateIndex();
		};

		bool operator==(const Iterator &other) const { return this->index == other.index; }
		bool operator!=(const Iterator &other) const { return !(*this == other); }
		CargoSpec * operator*() const { return CargoSpec::Get(this->index); }
		Iterator & operator++() { this->index++; this->ValidateIndex(); return *this; }

	private:
		size_t index;
		void ValidateIndex() { while (this->index < CargoSpec::GetArraySize() && !(CargoSpec::Get(this->index)->IsValid())) this->index++; }
	};

	/*
	 * Iterable ensemble of all valid CargoSpec
	 */
	struct IterateWrapper {
		size_t from;
		IterateWrapper(size_t from = 0) : from(from) {}
		Iterator begin() { return Iterator(this->from); }
		Iterator end() { return Iterator(CargoSpec::GetArraySize()); }
		bool empty() { return this->begin() == this->end(); }
	};

	/**
	 * Returns an iterable ensemble of all valid CargoSpec
	 * @param from index of the first CargoSpec to consider
	 * @return an iterable ensemble of all valid CargoSpec
	 */
	static IterateWrapper Iterate(size_t from = 0) { return IterateWrapper(from); }

private:
	static CargoSpec array[NUM_CARGO]; ///< Array holding all CargoSpecs

	friend void SetupCargoForClimate(LandscapeID l);
};

extern CargoTypes _cargo_mask;
extern CargoTypes _standard_cargo_mask;

void SetupCargoForClimate(LandscapeID l);
CargoID GetCargoIDByLabel(CargoLabel cl);
CargoID GetCargoIDByBitnum(uint8 bitnum);
CargoID GetDefaultCargoID(LandscapeID l, CargoType ct);

void InitializeSortedCargoSpecs();
extern std::vector<const CargoSpec *> _sorted_cargo_specs;
extern span<const CargoSpec *> _sorted_standard_cargo_specs;

/**
 * Does cargo \a c have cargo class \a cc?
 * @param c  Cargo type.
 * @param cc Cargo class.
 * @return The type fits in the class.
 */
static inline bool IsCargoInClass(CargoID c, CargoClass cc)
{
	return (CargoSpec::Get(c)->classes & cc) != 0;
}

using SetCargoBitIterator = SetBitIterator<CargoID, CargoTypes>;

#endif /* CARGOTYPE_H */