Files @ r28800:381a234fb097
Branch filter:

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

translators
Update: Translations from eints
english (au): 2 changes by krysclarke
spanish (mexican): 149 changes by Can202
estonian: 11 changes by RM87
chinese (simplified): 18 changes by WenSimEHRP
hungarian: 2 changes by PstasDev
italian: 195 changes by Rivarossi
serbian: 42 changes by nkrs
german: 2 changes by Wuzzy2
belarusian: 537 changes by KorneySan
russian: 25 changes by KorneySan
ukrainian: 21 changes by StepanIvasyn
turkish: 14 changes by jnmbk
latvian: 2 changes by lexuslatvia
dutch: 1 change by iamthedutchdude
spanish: 15 changes by MontyMontana
french: 2 changes by ottdfevr
portuguese: 2 changes by jcteotonio, 2 changes by azulcosta
portuguese (brazilian): 149 changes by pasantoro
polish: 2 changes by pAter-exe
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r18668:b656d614c4fd
r6123:049e9624d068
r6091:2faa7d307565
r6091:2faa7d307565
r28219:a652c2aea845
r13871:7588cd2474cc
r17611:e84f76d88e45
r14922:51cba7fe7b9b
r26524:5061ea344fd8
r6091:2faa7d307565
r8264:d493cb51fe8a
r8264:d493cb51fe8a
r6091:2faa7d307565
r6091:2faa7d307565
r21383:942c32fb8b0e
r21383:942c32fb8b0e
r12408:d7d52eed8c09
r28642:6e2af53a89ea
r6091:2faa7d307565
r15610:623a23fb6560
r19893:0fdf665730bf
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r22867:9bff1c966805
r6113:c77736e8b915
r15610:623a23fb6560
r19893:0fdf665730bf
r19893:0fdf665730bf
r22867:9bff1c966805
r19893:0fdf665730bf
r19893:0fdf665730bf
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r15610:623a23fb6560
r13874:a3b7b89e1abf
r13874:a3b7b89e1abf
r6091:2faa7d307565
r6091:2faa7d307565
r6091:2faa7d307565
r6091:2faa7d307565
r6113:c77736e8b915
r28672:8ec1f5f8a9d1
r6113:c77736e8b915
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r28672:8ec1f5f8a9d1
r6091:2faa7d307565
r27934:39f0f7c399a7
r28672:8ec1f5f8a9d1
r27934:39f0f7c399a7
r28672:8ec1f5f8a9d1
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r6640:772ee7bcd404
r6640:772ee7bcd404
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r6113:c77736e8b915
r27934:39f0f7c399a7
r27934:39f0f7c399a7
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r28672:8ec1f5f8a9d1
r6091:2faa7d307565
r6091:2faa7d307565
r15610:623a23fb6560
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r28219:a652c2aea845
r15610:623a23fb6560
r6460:74b53af67ae0
r28422:76582babb47f
r6460:74b53af67ae0
r27737:728d55b97775
r6460:74b53af67ae0
r28422:76582babb47f
r6460:74b53af67ae0
r25310:e304033a69f3
r12408:d7d52eed8c09
r6460:74b53af67ae0
r6460:74b53af67ae0
r6460:74b53af67ae0
r28422:76582babb47f
r6460:74b53af67ae0
r6460:74b53af67ae0
r15610:623a23fb6560
r15610:623a23fb6560
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r13871:7588cd2474cc
r27939:8e2e1ddcb130
r25783:02166e47d365
r28492:0d3d4fd5e362
r14955:fecfdf8f80c1
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14922:51cba7fe7b9b
r27193:e816892ba57b
r27193:e816892ba57b
r14922:51cba7fe7b9b
r27193:e816892ba57b
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14955:fecfdf8f80c1
r14955:fecfdf8f80c1
r14955:fecfdf8f80c1
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r28642:6e2af53a89ea
r23628:b9e9809abe89
r28219:a652c2aea845
r25310:e304033a69f3
r23628:b9e9809abe89
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r23628:b9e9809abe89
r14955:fecfdf8f80c1
r27939:8e2e1ddcb130
r27939:8e2e1ddcb130
r27939:8e2e1ddcb130
r27939:8e2e1ddcb130
r27939:8e2e1ddcb130
r25783:02166e47d365
r19893:0fdf665730bf
r27737:728d55b97775
r25301:13cfd266320c
r28642:6e2af53a89ea
r28642:6e2af53a89ea
r14955:fecfdf8f80c1
r25783:02166e47d365
r19893:0fdf665730bf
r14955:fecfdf8f80c1
r25783:02166e47d365
r25783:02166e47d365
r25783:02166e47d365
r14922:51cba7fe7b9b
r14922:51cba7fe7b9b
r27737:728d55b97775
r26524:5061ea344fd8
r26524:5061ea344fd8
r26524:5061ea344fd8
r26524:5061ea344fd8
/*
 * 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.cpp Implementation of cargoes. */

#include "stdafx.h"
#include "cargotype.h"
#include "core/geometry_func.hpp"
#include "newgrf_cargo.h"
#include "string_func.h"
#include "strings_func.h"
#include "settings_type.h"

#include "table/sprites.h"
#include "table/strings.h"
#include "table/cargo_const.h"

#include "safeguards.h"

CargoSpec CargoSpec::array[NUM_CARGO];
std::array<std::vector<const CargoSpec *>, NUM_TPE> CargoSpec::town_production_cargoes{};

/**
 * Bitmask of cargo types available. This includes phony cargoes like regearing cargoes.
 * Initialized during a call to #SetupCargoForClimate.
 */
CargoTypes _cargo_mask;

/**
 * Bitmask of real cargo types available. Phony cargoes like regearing cargoes are excluded.
 */
CargoTypes _standard_cargo_mask;

/**
 * List of default cargo labels, used when setting up cargo types for default vehicles.
 * This is done by label so that a cargo label can be redefined in a different slot.
 */
static std::vector<CargoLabel> _default_cargo_labels;

/**
 * Set up the default cargo types for the given landscape type.
 * @param l Landscape
 */
void SetupCargoForClimate(LandscapeID l)
{
	assert(l < lengthof(_default_climate_cargo));

	_cargo_mask = 0;
	_default_cargo_labels.clear();

	/* Copy from default cargo by label or index. */
	auto insert = std::begin(CargoSpec::array);
	for (const auto &cl : _default_climate_cargo[l]) {

		/* Check if value is an index into the cargo table */
		if (std::holds_alternative<int>(cl)) {
			/* Copy the default cargo by index. */
			*insert = _default_cargo[std::get<int>(cl)];
		} else {
			/* Search for label in default cargo types and copy if found. */
			CargoLabel label = std::get<CargoLabel>(cl);
			auto found = std::find_if(std::begin(_default_cargo), std::end(_default_cargo), [&label](const CargoSpec &cs) { return cs.label == label; });
			if (found != std::end(_default_cargo)) {
				*insert = *found;
			} else {
				/* Index or label is invalid, this should not happen. */
				NOT_REACHED();
			}
		}

		if (insert->IsValid()) {
			SetBit(_cargo_mask, insert->Index());
			_default_cargo_labels.push_back(insert->label);
		}
		++insert;
	}

	/* Reset and disable remaining cargo types. */
	std::fill(insert, std::end(CargoSpec::array), CargoSpec{});

	BuildCargoLabelMap();
}

/**
 * Build cargo label map.
 * This is called multiple times during NewGRF initialization as cargos are defined, so that TranslateRefitMask() and
 * GetCargoTranslation(), also used during initialization, get the correct information.
 */
void BuildCargoLabelMap()
{
	CargoSpec::label_map.clear();
	for (const CargoSpec &cs : CargoSpec::array) {
		/* During initialization, CargoSpec can be marked valid before the label has been set. */
		if (!cs.IsValid() || cs.label == CargoLabel{0}) continue;
		/* Label already exists, don't addd again. */
		if (CargoSpec::label_map.count(cs.label) != 0) continue;

		CargoSpec::label_map.insert(std::make_pair(cs.label, cs.Index()));
	}
}

/**
 * Test if a cargo is a default cargo type.
 * @param cid Cargo ID.
 * @returns true iff the cargo type is a default cargo type.
 */
bool IsDefaultCargo(CargoID cid)
{
	auto cs = CargoSpec::Get(cid);
	if (!cs->IsValid()) return false;

	CargoLabel label = cs->label;
	return std::any_of(std::begin(_default_cargo_labels), std::end(_default_cargo_labels), [&label](const CargoLabel &cl) { return cl == label; });
}

/**
 * Get dimensions of largest cargo icon.
 * @return Dimensions of largest cargo icon.
 */
Dimension GetLargestCargoIconSize()
{
	Dimension size = {0, 0};
	for (const CargoSpec *cs : _sorted_cargo_specs) {
		size = maxdim(size, GetSpriteSize(cs->GetCargoIcon()));
	}
	return size;
}

/**
 * Find the CargoID of a 'bitnum' value.
 * @param bitnum 'bitnum' to find.
 * @return First CargoID with the given bitnum, or #INVALID_CARGO if not found or if the provided \a bitnum is invalid.
 */
CargoID GetCargoIDByBitnum(uint8_t bitnum)
{
	if (bitnum == INVALID_CARGO_BITNUM) return INVALID_CARGO;

	for (const CargoSpec *cs : CargoSpec::Iterate()) {
		if (cs->bitnum == bitnum) return cs->Index();
	}

	/* No matching label was found, so it is invalid */
	return INVALID_CARGO;
}

/**
 * Get sprite for showing cargo of this type.
 * @return Sprite number to use.
 */
SpriteID CargoSpec::GetCargoIcon() const
{
	SpriteID sprite = this->sprite;
	if (sprite == 0xFFFF) {
		/* A value of 0xFFFF indicates we should draw a custom icon */
		sprite = GetCustomCargoSprite(this);
	}

	if (sprite == 0) sprite = SPR_CARGO_GOODS;

	return sprite;
}

std::array<uint8_t, NUM_CARGO> _sorted_cargo_types; ///< Sort order of cargoes by cargo ID.
std::vector<const CargoSpec *> _sorted_cargo_specs;   ///< Cargo specifications sorted alphabetically by name.
std::span<const CargoSpec *> _sorted_standard_cargo_specs; ///< Standard cargo specifications sorted alphabetically by name.

/** Sort cargo specifications by their name. */
static bool CargoSpecNameSorter(const CargoSpec * const &a, const CargoSpec * const &b)
{
	std::string a_name = GetString(a->name);
	std::string b_name = GetString(b->name);

	int res = StrNaturalCompare(a_name, b_name); // Sort by name (natural sorting).

	/* If the names are equal, sort by cargo bitnum. */
	return (res != 0) ? res < 0 : (a->bitnum < b->bitnum);
}

/** Sort cargo specifications by their cargo class. */
static bool CargoSpecClassSorter(const CargoSpec * const &a, const CargoSpec * const &b)
{
	int res = (b->classes & CC_PASSENGERS) - (a->classes & CC_PASSENGERS);
	if (res == 0) {
		res = (b->classes & CC_MAIL) - (a->classes & CC_MAIL);
		if (res == 0) {
			res = (a->classes & CC_SPECIAL) - (b->classes & CC_SPECIAL);
			if (res == 0) {
				return CargoSpecNameSorter(a, b);
			}
		}
	}

	return res < 0;
}

/** Initialize the list of sorted cargo specifications. */
void InitializeSortedCargoSpecs()
{
	for (auto &tpc : CargoSpec::town_production_cargoes) tpc.clear();
	_sorted_cargo_specs.clear();
	/* Add each cargo spec to the list, and determine the largest cargo icon size. */
	for (const CargoSpec *cargo : CargoSpec::Iterate()) {
		_sorted_cargo_specs.push_back(cargo);
	}

	/* Sort cargo specifications by cargo class and name. */
	std::sort(_sorted_cargo_specs.begin(), _sorted_cargo_specs.end(), &CargoSpecClassSorter);

	/* Populate */
	for (auto it = std::begin(_sorted_cargo_specs); it != std::end(_sorted_cargo_specs); ++it) {
		_sorted_cargo_types[(*it)->Index()] = static_cast<uint8_t>(it - std::begin(_sorted_cargo_specs));
	}

	/* Count the number of standard cargos and fill the mask. */
	_standard_cargo_mask = 0;
	uint8_t nb_standard_cargo = 0;
	for (const auto &cargo : _sorted_cargo_specs) {
		assert(cargo->town_production_effect != INVALID_TPE);
		CargoSpec::town_production_cargoes[cargo->town_production_effect].push_back(cargo);
		if (cargo->classes & CC_SPECIAL) break;
		nb_standard_cargo++;
		SetBit(_standard_cargo_mask, cargo->Index());
	}

	/* _sorted_standard_cargo_specs is a subset of _sorted_cargo_specs. */
	_sorted_standard_cargo_specs = { _sorted_cargo_specs.data(), nb_standard_cargo };
}

uint64_t CargoSpec::WeightOfNUnitsInTrain(uint32_t n) const
{
	if (this->is_freight) n *= _settings_game.vehicle.freight_trains;
	return this->WeightOfNUnits(n);
}