45#include "table/strings.h"
47#include "3rdparty/fmt/std.h"
49#include "strings_internal.h"
73 throw std::out_of_range(
"Trying to read invalid string parameter");
77 if (param.type != 0 && param.type != this->next_type) {
79 throw std::out_of_range(
"Trying to read string parameter with wrong type");
113 void operator()(
const std::monostate &) {}
115 void operator()(
const uint64_t &arg)
121 void operator()(
const std::string &value)
131 this->builder += value;
136 for (
const auto ¶m : params) {
137 builder.
PutUtf8(SCC_RECORD_SEPARATOR);
138 std::visit(v, param.data);
153 if (this->empty())
return {};
155 std::vector<StringParameter> params;
173 params.emplace_back(std::monostate{});
178 char32_t parameter_type = record.
ReadUtf8();
179 switch (parameter_type) {
183 params.emplace_back(value);
194 params.emplace_back(std::monostate{});
199 if (param >= std::size(params))
return {};
200 params[param] = data;
210 return GetString(STR_JUST_RAW_STRING, this->
string);
222 uint64_t val = count > 1 ? front : next;
223 for (; count > 1; count--) {
224 val = 10 * val + next;
240 while (max_value >= 10) {
262static void FormatString(
StringBuilder &builder, std::string_view str, std::span<StringParameter> params, uint case_index = 0,
bool game_script =
false,
bool dry_run =
false)
265 FormatString(builder, str, tmp_params, case_index, game_script, dry_run);
262static void FormatString(
StringBuilder &builder, std::string_view str, std::span<StringParameter> params, uint case_index = 0,
bool game_script =
false,
bool dry_run =
false) {
…}
276 delete[]
reinterpret_cast<char *
>(langpack);
281 std::unique_ptr<LanguagePack, LanguagePackDeleter> langpack;
283 std::vector<std::string_view> strings;
304std::string_view GetStringPtr(
StringID string)
309 case TEXT_TAB_OLD_NEWGRF: NOT_REACHED();
313 if (offset < _langpack.strings.size())
return _langpack.strings[offset];
314 return "(undefined string)";
342 }
catch (
const std::runtime_error &e) {
343 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
344 builder +=
"(invalid string parameter)";
350 case TEXT_TAB_SPECIAL:
353 if (GetSpecialNameString(builder,
string, args))
return;
354 }
catch (
const std::runtime_error &e) {
355 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
356 builder +=
"(invalid string parameter)";
362 case TEXT_TAB_OLD_CUSTOM:
365 FatalError(
"Incorrect conversion of custom name string.");
374 case TEXT_TAB_OLD_NEWGRF:
390 FatalError(
"String 0x{:X} is invalid. You are probably using an old version of the .lng file.\n",
string);
393 FormatString(builder, GetStringPtr(
string), args, case_index);
431void AppendStringWithArgsInPlace(std::string &result,
StringID string, std::span<StringParameter> params)
460static std::string_view GetDecimalSeparator()
463 if (decimal_separator.empty()) decimal_separator = _langpack.langpack->digit_decimal_separator;
464 return decimal_separator;
475 static const int max_digits = 20;
476 uint64_t divisor = 10000000000000000000ULL;
477 int thousands_offset = (max_digits - 1) % 3;
484 uint64_t num = number;
486 for (
int i = 0; i < max_digits; i++) {
488 if (num >= divisor) {
489 quot = num / divisor;
492 if ((tot |= quot) || i == max_digits - 1) {
494 if ((i % 3) == thousands_offset && i < max_digits - 1) builder += separator;
501static void FormatCommaNumber(
StringBuilder &builder, int64_t number)
504 if (separator.empty()) separator = _langpack.langpack->digit_group_separator;
508static void FormatNoCommaNumber(
StringBuilder &builder, int64_t number)
513static void FormatZerofillNumber(
StringBuilder &builder, int64_t number,
int count)
515 fmt::format_to(builder.
back_inserter(),
"{:0{}d}", number, count);
518static void FormatHexNumber(
StringBuilder &builder, uint64_t number)
533 static const std::string_view iec_prefixes[] = {
"",
"Ki",
"Mi",
"Gi",
"Ti",
"Pi",
"Ei"};
535 while (number >= 1024 * 1024) {
543 }
else if (number < 1024 * 10) {
544 fmt::format_to(builder.
back_inserter(),
"{}{}{:02}", number / 1024, GetDecimalSeparator(), (number % 1024) * 100 / 1024);
545 }
else if (number < 1024 * 100) {
546 fmt::format_to(builder.
back_inserter(),
"{}{}{:01}", number / 1024, GetDecimalSeparator(), (number % 1024) * 10 / 1024);
548 assert(number < 1024 * 1024);
549 fmt::format_to(builder.
back_inserter(),
"{}", number / 1024);
552 assert(
id <
lengthof(iec_prefixes));
560 auto tmp_params =
MakeParameters(STR_DAY_NUMBER_1ST + ymd.day - 1, STR_MONTH_ABBREV_JAN + ymd.month, ymd.year);
561 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_LONG), tmp_params, case_index);
568 auto tmp_params =
MakeParameters(STR_MONTH_JAN + ymd.month, ymd.year);
569 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_SHORT), tmp_params, case_index);
577 auto tmp_params =
MakeParameters(ymd.day, 2, ymd.month + 1, 2, ymd.year);
585 bool negative = number < 0;
587 number *= spec->
rate;
591 builder.
PutUtf8(SCC_PUSH_COLOUR);
608 if (number >=
Money(1'000'000'000'000'000) - 500'000'000) {
609 number = (number +
Money(500'000'000'000)) /
Money(1'000'000'000'000);
610 number_str = STR_CURRENCY_SHORT_TERA;
611 }
else if (number >=
Money(1'000'000'000'000) - 500'000) {
612 number = (number + 500'000'000) / 1'000'000'000;
613 number_str = STR_CURRENCY_SHORT_GIGA;
614 }
else if (number >= 1'000'000'000 - 500) {
615 number = (number + 500'000) / 1'000'000;
616 number_str = STR_CURRENCY_SHORT_MEGA;
617 }
else if (number >= 1'000'000) {
618 number = (number + 500) / 1'000;
619 number_str = STR_CURRENCY_SHORT_KILO;
625 if (separator.empty()) separator = _langpack.langpack->digit_group_separator_currency;
627 if (number_str != STR_NULL) {
637 builder.
PutUtf8(SCC_POP_COLOUR);
650 uint64_t n =
abs(count);
652 switch (plural_form) {
661 return n != 1 ? 1 : 0;
673 return n > 1 ? 1 : 0;
680 return n % 10 == 1 && n % 100 != 11 ? 0 : n != 0 ? 1 : 2;
686 return n == 1 ? 0 : n == 2 ? 1 : n < 7 ? 2 : n < 11 ? 3 : 4;
692 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
698 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
704 return n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
710 return n % 100 == 1 ? 0 : n % 100 == 2 ? 1 : n % 100 == 3 || n % 100 == 4 ? 2 : 3;
716 return n % 10 == 1 && n % 100 != 11 ? 0 : 1;
722 return n == 1 ? 0 : n >= 2 && n <= 4 ? 1 : 2;
753 return (n == 1 ? 0 : n == 0 || (n % 100 > 1 && n % 100 < 11) ? 1 : (n % 100 > 10 && n % 100 < 20) ? 2 : 3);
758 return ((n == 1 || n == 11) ? 0 : (n == 2 || n == 12) ? 1 : ((n > 2 && n < 11) || (n > 12 && n < 20)) ? 2 : 3);
764 return n == 1 ? 0 : (n == 0 || (n % 100 > 0 && n % 100 < 20)) ? 1 : 2;
772 for (uint i = 0; i != n; i++) {
782 size_t form_pre = 0, form_len = 0, form_post = 0;
783 for (uint i = 0; i != n; i++) {
787 }
else if (i > form) {
794 consumer.
Skip(form_pre);
795 builder += consumer.
Read(form_len);
796 consumer.
Skip(form_post);
809 int64_t
ToDisplay(int64_t input,
bool round =
true)
const
812 ? (int64_t)std::round(input * this->factor)
813 : (int64_t)(input * this->factor);
809 int64_t
ToDisplay(int64_t input,
bool round =
true)
const {
…}
823 int64_t
FromDisplay(int64_t input,
bool round =
true, int64_t divider = 1)
const
826 ? (int64_t)std::round(input / this->factor / divider)
827 : (int64_t)(input / this->factor / divider);
823 int64_t
FromDisplay(int64_t input,
bool round =
true, int64_t divider = 1)
const {
…}
848 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
849 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
850 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
851 { { 0.578125 }, STR_UNITS_VELOCITY_GAMEUNITS_DAY, 1 },
852 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
857 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
858 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
859 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
860 { { 0.289352 }, STR_UNITS_VELOCITY_GAMEUNITS_SEC, 1 },
861 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
866 { { 1.0 }, STR_UNITS_POWER_IMPERIAL, 0 },
867 { { 1.01387 }, STR_UNITS_POWER_METRIC, 0 },
868 { { 0.745699 }, STR_UNITS_POWER_SI, 0 },
873 { { 0.907185 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_IMPERIAL, 1 },
874 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_METRIC, 1 },
875 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_SI, 1 },
876 { { 0.919768 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_IMPERIAL, 1 },
877 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_METRIC, 1 },
878 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_SI, 1 },
879 { { 0.676487 }, STR_UNITS_POWER_SI_TO_WEIGHT_IMPERIAL, 1 },
880 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_METRIC, 1 },
881 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_SI, 1 },
886 { { 1.102311 }, STR_UNITS_WEIGHT_SHORT_IMPERIAL, STR_UNITS_WEIGHT_LONG_IMPERIAL, 0 },
887 { { 1.0 }, STR_UNITS_WEIGHT_SHORT_METRIC, STR_UNITS_WEIGHT_LONG_METRIC, 0 },
888 { { 1000.0 }, STR_UNITS_WEIGHT_SHORT_SI, STR_UNITS_WEIGHT_LONG_SI, 0 },
893 { { 264.172 }, STR_UNITS_VOLUME_SHORT_IMPERIAL, STR_UNITS_VOLUME_LONG_IMPERIAL, 0 },
894 { { 1000.0 }, STR_UNITS_VOLUME_SHORT_METRIC, STR_UNITS_VOLUME_LONG_METRIC, 0 },
895 { { 1.0 }, STR_UNITS_VOLUME_SHORT_SI, STR_UNITS_VOLUME_LONG_SI, 0 },
900 { { 0.224809 }, STR_UNITS_FORCE_IMPERIAL, 0 },
901 { { 0.101972 }, STR_UNITS_FORCE_METRIC, 0 },
902 { { 0.001 }, STR_UNITS_FORCE_SI, 0 },
907 { { 3.0 }, STR_UNITS_HEIGHT_IMPERIAL, 0 },
908 { { 1.0 }, STR_UNITS_HEIGHT_METRIC, 0 },
909 { { 1.0 }, STR_UNITS_HEIGHT_SI, 0 },
914 { { 1 }, STR_UNITS_DAYS, 0 },
915 { { 2 }, STR_UNITS_SECONDS, 0 },
920 { { 1 }, STR_UNITS_MONTHS, 0 },
921 { { 1 }, STR_UNITS_MINUTES, 0 },
926 { { 1 }, STR_UNITS_YEARS, 0 },
927 { { 1 }, STR_UNITS_PERIODS, 0 },
932 { { 1 }, STR_UNITS_YEARS, 0 },
933 { { 12 }, STR_UNITS_MINUTES, 0 },
1004 std::vector<StringParameter> sub_args;
1009 builder +=
"(invalid SCC_ENCODED)";
1014 builder +=
"(invalid StringID)";
1024 sub_args.emplace_back(std::monostate{});
1029 char32_t parameter_type = record.
ReadUtf8();
1030 switch (parameter_type) {
1034 builder +=
"(invalid sub-StringID)";
1039 sub_args.emplace_back(param);
1046 sub_args.emplace_back(param);
1057 sub_args.emplace_back(std::monostate{});
1075 size_t orig_first_param_offset = args.
GetOffset();
1088 FormatString(dry_run_builder, str_arg, args, orig_case_index, game_script,
true);
1090 args.
SetOffset(orig_first_param_offset);
1092 uint next_substr_case_index = 0;
1093 struct StrStackItem {
1095 size_t first_param_offset;
1098 StrStackItem(std::string_view view,
size_t first_param_offset, uint case_index)
1099 : consumer(view), first_param_offset(first_param_offset), case_index(case_index)
1102 std::stack<StrStackItem, std::vector<StrStackItem>> str_stack;
1103 str_stack.emplace(str_arg, orig_first_param_offset, orig_case_index);
1107 while (!str_stack.empty() && !str_stack.top().consumer.AnyBytesLeft()) {
1110 if (str_stack.empty())
break;
1112 const size_t ref_param_offset = str_stack.top().first_param_offset;
1113 const uint case_index = str_stack.top().case_index;
1118 builder +=
"(unexpected NUL)";
1125 if (b == 0)
continue;
1128 if (b < SCC_CONTROL_START || b > SCC_CONTROL_END) {
1133 args.SetTypeOfNextParameter(b);
1142 std::string_view ptr = GetStringPtr(substr);
1143 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1144 next_substr_case_index = 0;
1150 std::string_view ptr = GetStringPtr(substr);
1151 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1152 next_substr_case_index = 0;
1156 case SCC_GENDER_LIST: {
1158 size_t offset = ref_param_offset + consumer.
ReadUint8();
1162 builder +=
"(invalid GENDER parameter)";
1184 if (gender_consumer.
ReadUtf8If(SCC_GENDER_INDEX)) {
1188 ParseStringChoice(consumer, gender, builder);
1194 case SCC_GENDER_INDEX: {
1197 builder.
PutUtf8(SCC_GENDER_INDEX);
1203 case SCC_PLURAL_LIST: {
1204 uint8_t plural_form = consumer.
ReadUint8();
1205 size_t offset = ref_param_offset + consumer.
ReadUint8();
1206 const uint64_t *v =
nullptr;
1209 v = std::get_if<uint64_t>(&args.GetParam(offset));
1212 ParseStringChoice(consumer,
DeterminePluralForm(
static_cast<int64_t
>(*v), plural_form), builder);
1214 SkipStringChoice(consumer);
1215 builder +=
"(invalid PLURAL parameter)";
1220 case SCC_ARG_INDEX: {
1225 case SCC_SET_CASE: {
1228 next_substr_case_index = consumer.
ReadUint8();
1232 case SCC_SWITCH_CASE: {
1236 std::optional<std::string_view> found;
1237 for (; num > 0; --num) {
1240 auto case_str = consumer.
Read(len);
1241 if (index == case_index) {
1247 auto default_str = consumer.
Read(default_len);
1248 if (!found.has_value()) found = default_str;
1249 str_stack.emplace(*found, ref_param_offset, case_index);
1254 builder += _openttd_revision;
1257 case SCC_RAW_STRING_POINTER:
1266 GetStringWithArgs(builder, string_id, tmp_params, next_substr_case_index, game_script);
1267 next_substr_case_index = 0;
1281 uint size = b - SCC_STRING1 + 1;
1283 builder +=
"(consumed too many parameters)";
1286 GetStringWithArgs(builder, string_id, sub_args, next_substr_case_index, game_script);
1289 next_substr_case_index = 0;
1301 FormatCommaNumber(builder, number);
1306 int64_t fractional = number % divisor;
1308 FormatCommaNumber(builder, number);
1309 fmt::format_to(builder.
back_inserter(),
"{}{:0{}d}", GetDecimalSeparator(), fractional, digits);
1317 case SCC_ZEROFILL_NUM: {
1331 case SCC_CARGO_TINY: {
1340 switch (cargo_str) {
1355 FormatCommaNumber(builder, amount);
1359 case SCC_CARGO_SHORT: {
1367 switch (cargo_str) {
1393 case SCC_CARGO_LONG: {
1404 case SCC_CARGO_LIST: {
1410 if (!
HasBit(cmask, cs->Index()))
continue;
1416 builder += list_separator;
1419 GetStringWithArgs(builder, cs->name, args, next_substr_case_index, game_script);
1423 if (first)
GetStringWithArgs(builder, STR_JUST_NOTHING, args, next_substr_case_index, game_script);
1425 next_substr_case_index = 0;
1429 case SCC_CURRENCY_SHORT:
1433 case SCC_CURRENCY_LONG:
1441 case SCC_DATE_SHORT:
1443 next_substr_case_index = 0;
1448 next_substr_case_index = 0;
1479 case SCC_POWER_TO_WEIGHT: {
1488 case SCC_VELOCITY: {
1498 case SCC_VOLUME_SHORT: {
1506 case SCC_VOLUME_LONG: {
1514 case SCC_WEIGHT_SHORT: {
1522 case SCC_WEIGHT_LONG: {
1530 case SCC_UNITS_DAYS_OR_SECONDS: {
1538 case SCC_UNITS_MONTHS_OR_MINUTES: {
1546 case SCC_UNITS_YEARS_OR_PERIODS: {
1554 case SCC_UNITS_YEARS_OR_MINUTES: {
1562 case SCC_COMPANY_NAME: {
1564 if (c ==
nullptr)
break;
1566 if (!c->
name.empty()) {
1576 case SCC_COMPANY_NUM: {
1587 case SCC_DEPOT_NAME: {
1596 if (!d->name.empty()) {
1606 case SCC_ENGINE_NAME: {
1609 if (e ==
nullptr)
break;
1611 if (!e->name.empty() && e->IsEnabled()) {
1618 std::array<int32_t, 16> regs100;
1621 if (callback == 0x40F) {
1622 const GRFFile *grffile = e->GetGRF();
1623 assert(grffile !=
nullptr);
1627 }
else if (callback < 0x400) {
1628 const GRFFile *grffile = e->GetGRF();
1629 assert(grffile !=
nullptr);
1640 case SCC_GROUP_NAME: {
1642 if (g ==
nullptr)
break;
1644 if (!g->
name.empty()) {
1654 case SCC_INDUSTRY_NAME: {
1656 if (i ==
nullptr)
break;
1658 static bool use_cache =
true;
1663 }
else if (use_cache) {
1665 builder += i->GetCachedName();
1669 FormatString(builder, GetStringPtr(STR_FORMAT_INDUSTRY_NAME), tmp_params, next_substr_case_index);
1671 next_substr_case_index = 0;
1675 case SCC_PRESIDENT_NAME: {
1677 if (c ==
nullptr)
break;
1689 case SCC_STATION_NAME: {
1693 if (st ==
nullptr) {
1701 static bool use_cache =
true;
1704 builder += st->GetCachedName();
1705 }
else if (!st->name.empty()) {
1709 StringID string_id = st->string_id;
1710 if (st->indtype != IT_INVALID) {
1722 auto tmp_params =
MakeParameters(STR_TOWN_NAME, st->town->index, st->index);
1728 case SCC_TOWN_NAME: {
1730 if (t ==
nullptr)
break;
1732 static bool use_cache =
true;
1735 builder += t->GetCachedName();
1736 }
else if (!t->
name.empty()) {
1740 GetTownName(builder, t);
1745 case SCC_WAYPOINT_NAME: {
1747 if (wp ==
nullptr)
break;
1749 if (!wp->
name.empty()) {
1754 StringID string_id = ((wp->
string_id == STR_SV_STNAME_BUOY) ? STR_FORMAT_BUOY_NAME : STR_FORMAT_WAYPOINT_NAME);
1755 if (wp->
town_cn != 0) string_id++;
1761 case SCC_VEHICLE_NAME: {
1763 if (v ==
nullptr)
break;
1765 if (!v->
name.empty()) {
1777 default: string_id = STR_INVALID_VEHICLE;
break;
1778 case VEH_TRAIN: string_id = STR_SV_TRAIN_NAME;
break;
1779 case VEH_ROAD: string_id = STR_SV_ROAD_VEHICLE_NAME;
break;
1780 case VEH_SHIP: string_id = STR_SV_SHIP_NAME;
break;
1781 case VEH_AIRCRAFT: string_id = STR_SV_AIRCRAFT_NAME;
break;
1789 case SCC_SIGN_NAME: {
1791 if (si ==
nullptr)
break;
1793 if (!si->name.empty()) {
1802 case SCC_STATION_FEATURES: {
1817 }
catch (std::out_of_range &e) {
1818 Debug(misc, 0,
"FormatString: {}", e.what());
1819 builder +=
"(invalid parameter)";
1834static const std::string_view _silly_company_names[] = {
1836 "Tiny Transport Ltd.",
1838 "Comfy-Coach & Co.",
1839 "Crush & Bump Ltd.",
1840 "Broken & Late Ltd.",
1842 "Supersonic Travel",
1844 "Lightning International",
1845 "Pannik & Loozit Ltd.",
1846 "Inter-City Transport",
1847 "Getout & Pushit Ltd."
1850static const std::string_view _surname_list[] = {
1882static const std::string_view _silly_surname_list[] = {
1897static const char _initial_name_letters[] = {
1898 'A',
'B',
'C',
'D',
'E',
'F',
'G',
'H',
'I',
'J',
1899 'K',
'L',
'M',
'N',
'P',
'R',
'S',
'T',
'W',
1902static std::span<const std::string_view> GetSurnameOptions()
1905 return _surname_list;
1915 auto surname_options = GetSurnameOptions();
1916 return surname_options[surname_options.size() *
GB(seed, 16, 8) >> 8];
1919static void GenAndCoName(
StringBuilder &builder, uint32_t seed)
1922 builder +=
" & Co.";
1925static void GenPresidentName(
StringBuilder &builder, uint32_t seed)
1927 builder.
PutChar(_initial_name_letters[std::size(_initial_name_letters) *
GB(seed, 0, 8) >> 8]);
1931 size_t index = (std::size(_initial_name_letters) + 35) *
GB(seed, 8, 8) >> 8;
1932 if (index < std::size(_initial_name_letters)) {
1933 builder.
PutChar(_initial_name_letters[index]);
1944 builder += _silly_company_names[std::min<size_t>(args.
GetNextParameter<uint16_t>(), std::size(_silly_company_names) - 1)];
1959 builder +=
" Transport";
1973 this->
version == TO_LE32(LANGUAGE_PACK_VERSION) &&
1993 return 4 * this->
missing < LANGUAGE_TOTAL_STRINGS;
2004 size_t total_len = 0;
2006 if (!lang_pack)
return false;
2009 const char *end = (
char *)lang_pack.get() + total_len + 1;
2012 if (end <= lang_pack->data || !lang_pack->IsValid()) {
2016 std::array<uint, TEXT_TAB_END> tab_start, tab_num;
2020 uint16_t num = FROM_LE16(lang_pack->offsets[i]);
2023 tab_start[i] = count;
2029 std::vector<std::string_view> strings;
2032 char *s = lang_pack->data;
2033 for (uint i = 0; i < count; i++) {
2034 size_t len =
static_cast<uint8_t
>(*s++);
2035 if (s + len >= end)
return false;
2038 len = ((len & 0x3F) << 8) +
static_cast<uint8_t
>(*s++);
2039 if (s + len >= end)
return false;
2041 strings.emplace_back(s, len);
2044 assert(strings.size() == count);
2046 _langpack.langpack = std::move(lang_pack);
2047 _langpack.strings = std::move(strings);
2058 extern void Win32SetCurrentLocaleName(std::string iso_code);
2069 UErrorCode status = U_ZERO_ERROR;
2074 if (U_FAILURE(status)) {
2100#if !(defined(_WIN32) || defined(__APPLE__))
2111 auto env =
GetEnv(
"LANGUAGE");
2112 if (env.has_value())
return std::string{*env};
2115 if (env.has_value())
return std::string{*env};
2117 if (param !=
nullptr) {
2119 if (env.has_value())
return std::string{*env};
2123 if (env.has_value())
return std::string{*env};
2125 return std::nullopt;
2139 if (newgrflangid == lang.newgrflangid)
return ⟨
2154 if (!f.has_value())
return false;
2156 size_t read = fread(hdr,
sizeof(*hdr), 1, *f);
2158 bool ret = read == 1 && hdr->
IsValid();
2174 std::error_code error_code;
2175 for (
const auto &dir_entry : std::filesystem::directory_iterator(
OTTD2FS(path), error_code)) {
2176 if (!dir_entry.is_regular_file())
continue;
2177 if (dir_entry.path().extension() !=
".lng")
continue;
2180 lmd.
file = dir_entry.path();
2185 Debug(misc, 3,
"{} is not a valid language file", file);
2187 Debug(misc, 3,
"{}'s language ID is already known", file);
2193 Debug(misc, 9,
"Unable to open directory {}: {}", path, error_code.message());
2206 if (
_languages.empty()) UserError(
"No available language packs (invalid versions?)");
2210 std::string_view lang = str_lang.has_value() ? std::string_view{*str_lang} :
"en_GB";
2222 chosen_language = &lng;
2226 std::string_view iso_code = lng.
isocode;
2227 if (iso_code ==
"en_GB") en_GB_fallback = &lng;
2230 if (!lng.IsReasonablyFinished())
continue;
2232 if (iso_code.starts_with(lang.substr(0, 5))) chosen_language = &lng;
2233 if (iso_code.starts_with(lang.substr(0, 2))) language_fallback = &lng;
2238 if (chosen_language ==
nullptr) {
2239 chosen_language = (language_fallback !=
nullptr) ? language_fallback : en_GB_fallback;
2251 return _langpack.langpack->isocode;
2266 for (
char32_t c :
Utf8View(*text)) {
2267 if (c >= SCC_FIRST_FONT && c <= SCC_LAST_FONT) {
2268 size = (
FontSize)(c - SCC_FIRST_FONT);
2272 std::string size_name;
2275 case FS_NORMAL: size_name =
"medium";
break;
2276 case FS_SMALL: size_name =
"small";
break;
2277 case FS_LARGE: size_name =
"large";
break;
2278 case FS_MONO: size_name =
"mono";
break;
2279 default: NOT_REACHED();
2282 Debug(fontcache, 0,
"Font is missing glyphs to display char 0x{:X} in {} font size", (
int)c, size_name);
2310 std::string_view ret = _langpack.strings[_langpack.
langtab_start[this->
i] + this->
j];
2313 while (this->i < TEXT_TAB_END && this->
j >= _langpack.
langtab_num[this->i]) {
2328#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2333 settings->small.os_handle = os_data;
2334 settings->medium.os_handle = os_data;
2335 settings->large.os_handle = os_data;
2356 if (searcher ==
nullptr) searcher = &pack_searcher;
2358#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2362 bool any_font_configured = !_fcsettings.
medium.
font.empty();
2368 bad_font = !
SetFallbackFont(&_fcsettings, _langpack.langpack->isocode, searcher);
2370 _fcsettings = std::move(backup);
2372 if (!bad_font && any_font_configured) {
2376 std::string err_str;
2379 builder.
Put(
"The current font is missing some of the characters used in the texts for this language. Using system fallback font instead.");
2383 if (bad_font && base_font) {
2397 std::string err_str;
2400 builder.
Put(
"The current font is missing some of the characters used in the texts for this language. Go to Help & Manuals > Fonts, or read the file docs/fonts.md in your OpenTTD directory, to see how to solve this.");
2411#if !(defined(WITH_ICU_I18N) && defined(WITH_HARFBUZZ)) && !defined(WITH_UNISCRIBE) && !defined(WITH_COCOA)
2422 std::string err_str;
2425 builder.
Put(
"This version of OpenTTD does not support right-to-left languages. Recompile with ICU + Harfbuzz enabled.");
Class for backupping variables and making sure they are restored later.
debug_inline constexpr bool HasBit(const T x, const uint8_t y)
Checks if a bit in a value is set.
debug_inline static constexpr uint GB(const T x, const uint8_t s, const uint8_t n)
Fetch n bits from x, started at bit s.
uint8_t CargoType
Cargo slots to indicate a cargo type within a game.
bool IsValidCargoType(CargoType cargo)
Test whether cargo type is not INVALID_CARGO.
std::vector< const CargoSpec * > _sorted_cargo_specs
Cargo specifications sorted alphabetically by name.
void InitializeSortedCargoSpecs()
Initialize the list of sorted cargo specifications.
void PutUtf8(char32_t c)
Append UTF.8 char.
void Put(std::string_view str)
Append string.
void PutIntegerBase(T value, int base)
Append integer 'value' in given number 'base'.
void PutChar(char c)
Append 8-bit char.
void PutUint8(uint8_t value)
Append binary uint8.
Container for an encoded string, created by GetEncodedString.
friend EncodedString GetEncodedStringWithArgs(StringID str, std::span< const StringParameter > params)
Encode a string with its parameters into an encoded string.
std::string GetDecodedString() const
Decode the encoded string.
EncodedString ReplaceParam(size_t param, StringParameter &&value) const
Replace a parameter of this EncodedString.
static std::optional< FileHandle > Open(const std::string &filename, std::string_view mode)
Open an RAII file handle if possible.
Font cache for basic fonts.
virtual GlyphID MapCharToGlyph(char32_t key, bool fallback=true)=0
Map a character into a glyph.
static FontCache * Get(FontSize fs)
Get the font cache of a given font size.
Helper for searching through the language pack.
bool Monospace() override
Whether to search for a monospace font or not.
void SetFontNames(FontCacheSettings *settings, std::string_view font_name, const void *os_data) override
Set the right font names.
uint j
Iterator for the secondary language tables.
std::optional< std::string_view > NextString() override
Get the next string to search through.
FontSize DefaultSize() override
Get the default (font) size of the string.
uint i
Iterator for the primary language tables.
void Reset() override
Reset the search, i.e.
static void Initialize()
Perform initialization of layout engine.
A searcher for missing glyphs.
virtual void Reset()=0
Reset the search, i.e.
virtual FontSize DefaultSize()=0
Get the default (font) size of the string.
bool FindMissingGlyphs()
Check whether there are glyphs missing in the current language.
virtual bool Monospace()=0
Whether to search for a monospace font or not.
virtual std::optional< std::string_view > NextString()=0
Get the next string to search through.
Compose data into a growing std::string.
back_insert_iterator back_inserter()
Create a back-insert-iterator.
Parse data from a string / buffer.
std::optional< T > TryReadIntegerBase(int base, bool clamp=false)
Try to read and parse an integer in number 'base', and then advance the reader.
bool PeekUtf8If(char32_t c) const
Check whether the next UTF-8 char matches 'c'.
char32_t ReadUtf8(char32_t def='?')
Read UTF-8 character, and advance reader.
@ SKIP_ONE_SEPARATOR
Read and discard one separator, do not include it in the result.
@ KEEP_SEPARATOR
Keep the separator in the data as next value to be read.
bool AnyBytesLeft() const noexcept
Check whether any bytes left to read.
uint8_t ReadUint8(uint8_t def=0)
Read binary uint8, and advance reader.
uint16_t ReadUint16LE(uint16_t def=0)
Read binary uint16 using little endian, and advance reader.
void SkipAll()
Discard all remaining data.
T ReadIntegerBase(int base, T def=0, bool clamp=false)
Read and parse an integer in number 'base', and advance the reader.
std::string_view Read(size_type len)
Read the next 'len' bytes, and advance reader.
bool ReadUtf8If(char32_t c)
Check whether the next UTF-8 char matches 'c', and skip it.
static constexpr size_type npos
Special value for "end of data".
void Skip(size_type len)
Discard some bytes.
void SkipUtf8If(char32_t c)
If the next data matches the UTF-8 char 'c', then skip it.
std::string_view ReadUntilUtf8(char32_t c, SeparatorUsage sep)
Read data until the first occurrence of UTF-8 char 'c', and advance reader.
void SetOffset(size_t offset)
Set the offset within the string from where to return the next result of GetInt64 or GetInt32.
std::string_view GetNextParameterString()
Get the next string parameter from our parameters.
StringParameters GetRemainingParameters()
Get a new instance of StringParameters that is a "range" into the remaining existing parameters.
std::span< StringParameter > parameters
Array with the actual parameters.
const StringParameter & GetNextParameterReference()
Get the next parameter from our parameters.
void AdvanceOffset(size_t advance)
Advance the offset within the string from where to return the next result of GetInt64 or GetInt32.
uint64_t GetNextParameter()
Get the next parameter from our parameters.
size_t offset
Current offset in the parameters span.
char32_t next_type
The type of the next data that is retrieved.
size_t GetOffset()
Get the current offset, so it can be backed up for certain processing steps, or be used to offset the...
size_t GetNumParameters() const
Return the number of parameters.
size_t GetDataLeft() const
Return the amount of elements which can still be read.
char32_t GetTypeAtOffset(size_t offset) const
Get the type of a specific element.
static YearMonthDay ConvertDateToYMD(Date date)
Converts a Date to a Year, Month & Day.
static bool UsingWallclockUnits(bool newgame=false)
Check if we are using wallclock units.
Constant span of UTF-8 encoded data.
Definition of stuff that is very close to a company, like the company struct itself.
Control codes that are embedded in the translation strings.
StringControlCode
List of string control codes used for string formatting, displaying, and by strgen to generate the la...
@ SCC_ENCODED
Encoded string marker and sub-string parameter.
@ SCC_ENCODED_NUMERIC
Encoded numeric parameter.
@ SCC_ENCODED_STRING
Encoded string parameter.
@ SCC_NEWGRF_STRINL
Inline another string at the current position, StringID is encoded in the string.
@ SCC_NEWGRF_FIRST
The next variables are part of a NewGRF subsystem for creating text strings.
@ SCC_NEWGRF_PRINT_WORD_STRING_ID
81: Read 2 bytes from the stack as String ID
@ SCC_ENCODED_INTERNAL
Encoded text from OpenTTD.
std::pair< size_t, char32_t > DecodeUtf8(std::string_view buf)
Decode a character from UTF-8.
Functions to handle different currencies.
const CurrencySpec & GetCurrency()
Get the currently selected currency.
Functions related to debugging.
#define Debug(category, level, format_string,...)
Output a line of debugging information.
Base for all depots (except hangars)
Function to handling different endian machines.
Functions related to errors.
@ WL_WARNING
Other information.
@ WL_ERROR
Errors (eg. saving/loading failed)
void ShowErrorMessage(EncodedString &&summary_msg, int x, int y, CommandCost &cc)
Display an error message in a window.
Error reporting related functions.
std::unique_ptr< char[]> ReadFileToMem(const std::string &filename, size_t &lenp, size_t maxsize)
Load a file into memory.
Functions for Standard In/Out file operations.
Searchpath
Types of searchpaths OpenTTD might use.
@ LANG_DIR
Subdirectory for all translation files.
fluid_settings_t * settings
FluidSynth settings handle.
void InitFontCache(bool monospace)
(Re)initialize the font cache related things, i.e.
Functions related to detecting/finding the right font.
bool SetFallbackFont(struct FontCacheSettings *settings, const std::string &language_isocode, class MissingGlyphSearcher *callback)
We would like to have a fallback font as the current one doesn't contain all characters we need.
void ReconsiderGameScriptLanguage()
Reconsider the game script language, so we use the right one.
std::string_view GetGameStringPtr(StringIndexInTab id)
Get the string pointer of a particular game string.
Base functions regarding game texts.
std::pair< uint8_t, uint8_t > GetBroadestDigit(FontSize size)
Determine the broadest digits for guessing the maximum width of a n-digit number.
void LoadStringWidthTable(bool monospace)
Initialize _stringwidth_table cache.
Functions related to laying out the texts.
FontSize
Available font sizes.
@ FS_MONO
Index of the monospaced font in the font tables.
@ FS_SMALL
Index of the small font in the font tables.
@ FS_NORMAL
Index of the normal font in the font tables.
@ FS_LARGE
Index of the large font in the font tables.
static constexpr GroupID DEFAULT_GROUP
Ungrouped vehicles are in this group.
const IndustrySpec * GetIndustrySpec(IndustryType thistype)
Accessor for array _industry_specs.
void SortIndustryTypes()
Initialize the list of sorted industry types.
Information about languages and their files.
static const uint8_t MAX_NUM_GENDERS
Maximum number of supported genders.
std::vector< LanguageMetadata > LanguageList
Type for the list of language meta data.
static const uint8_t MAX_NUM_CASES
Maximum number of supported cases.
constexpr bool IsInsideMM(const size_t x, const size_t min, const size_t max) noexcept
Checks if a value is in an interval.
constexpr T abs(const T a)
Returns the absolute value of (scalar) variable.
constexpr uint64_t PowerOfTen(int power)
Computes ten to the given power.
void BuildContentTypeStringList()
Build array of all strings corresponding to the content types.
User interface for downloading files.
@ CBID_VEHICLE_NAME
Called to determine the engine name to show.
uint16_t GetVehicleCallback(CallbackID callback, uint32_t param1, uint32_t param2, EngineID engine, const Vehicle *v, std::span< int32_t > regs100)
Evaluate a newgrf callback for vehicles.
Functions for NewGRF engines.
std::string_view GetGRFStringPtr(StringIndexInTab stringid)
Get a C-string from a stringid set by a newgrf.
std::string GetGRFStringWithTextStack(const struct GRFFile *grffile, GRFStringID grfstringid, std::span< const int32_t > textstack)
Format a GRF string using the text ref stack for parameters.
void SetCurrentGrfLangID(uint8_t language_id)
Equivalence Setter function between game and newgrf langID.
char32_t RemapNewGRFStringControlCode(char32_t scc, StringConsumer &consumer)
Emit OpenTTD's internal string code for the different NewGRF string codes.
Header of Action 04 "universal holder" structure and functions.
static constexpr GRFStringID GRFSTR_MISC_GRF_TEXT
Miscellaneous GRF text range.
declaration of OTTD revision dependent variables
A number of safeguards to prevent using unsafe methods.
GameSettings _settings_game
Game settings of a running game or the scenario editor.
void BuildIndustriesLegend()
Fills an array for the industries legends.
Base classes/functions for stations.
@ Dock
Station with a dock.
@ TruckStop
Station with truck stops.
@ Train
Station with train station.
@ Airport
Station with an airport.
@ BusStop
Station with bus stops.
Definition of base types and functions in a cross-platform compatible way.
#define lengthof(array)
Return the length of an fixed size array.
std::optional< std::string_view > GetEnv(const char *variable)
Get the environment variable using std::getenv and when it is an empty string (or nullptr),...
bool StrValid(std::span< const char > str)
Checks whether the given string is valid, i.e.
Functions related to low-level strings.
bool IsTextDirectionChar(char32_t c)
Is the given character a text direction character.
void MacOSSetCurrentLocaleName(std::string_view iso_code)
Store current language locale as a CoreFoundation locale.
#define NBSP
A non-breaking space.
uint64_t GetParamMaxValue(uint64_t max_value, uint min_count, FontSize size)
Get some number that is suitable for string size computations.
LanguageList _languages
The actual list of language meta data.
static void FormatNumber(StringBuilder &builder, int64_t number, std::string_view separator)
Format a number into a string.
EncodedString GetEncodedStringWithArgs(StringID str, std::span< const StringParameter > params)
Encode a string with its parameters into an encoded string.
void GetStringWithArgs(StringBuilder &builder, StringID string, StringParameters &args, uint case_index, bool game_script)
Get a parsed string with most special stringcodes replaced by the string parameters.
std::optional< std::string > GetCurrentLocale(const char *param)
Determine the current charset based on the environment First check some default values,...
std::string_view GetListSeparator()
Get the list separator string for the current language.
static const Units _units_height[]
Unit conversions for height.
const LanguageMetadata * _current_language
The currently loaded language.
static void FormatBytes(StringBuilder &builder, int64_t number)
Format a given number as a number of bytes with the SI prefix.
void AppendStringInPlace(std::string &result, StringID string)
Resolve the given StringID and append in place into an existing std::string with formatting but no pa...
uint ConvertSpeedToDisplaySpeed(uint speed, VehicleType type)
Convert the given (internal) speed to the display speed.
void CheckForMissingGlyphs(bool base_font, MissingGlyphSearcher *searcher)
Check whether the currently loaded language pack uses characters that the currently loaded font does ...
std::string _config_language_file
The file (name) stored in the configuration.
static const Units GetVelocityUnits(VehicleType type)
Get the correct velocity units depending on the vehicle type and whether we're using real-time units.
static int DeterminePluralForm(int64_t count, uint plural_form)
Determine the "plural" index given a plural form and a number.
EncodedString GetEncodedString(StringID str)
Encode a string with no parameters into an encoded string.
uint ConvertDisplaySpeedToKmhishSpeed(uint speed, VehicleType type)
Convert the given display speed to the km/h-ish speed.
static const Units _units_power[]
Unit conversions for power.
static std::string_view GetSurname(uint32_t seed)
Get the surname of the president with the given seed.
std::string GetString(StringID string)
Resolve the given StringID into a std::string with formatting but no parameters.
std::string_view GetCurrentLanguageIsoCode()
Get the ISO language code of the currently loaded language.
static bool _scan_for_gender_data
Are we scanning for the gender of the current string? (instead of formatting it)
static const UnitsLong _units_volume[]
Unit conversions for volume.
static void DecodeEncodedString(StringConsumer &consumer, bool game_script, StringBuilder &builder)
Decodes an encoded string during FormatString.
static const UnitsLong _units_weight[]
Unit conversions for weight.
static const Units _units_velocity_calendar[]
Unit conversions for velocity.
std::unique_ptr< icu::Collator > _current_collator
Collator for the language currently in use.
static void FillLanguageList(const std::string &path)
Search for the languages in the given directory and add them to the _languages list.
uint ConvertKmhishSpeedToDisplaySpeed(uint speed, VehicleType type)
Convert the given km/h-ish speed to the display speed.
static const Units _units_force[]
Unit conversions for force.
void InitializeLanguagePacks()
Make a list of the available language packs.
const LanguageMetadata * GetLanguage(uint8_t newgrflangid)
Get the language with the given NewGRF language ID.
static const Units _units_time_days_or_seconds[]
Unit conversions for time in calendar days or wallclock seconds.
uint ConvertDisplaySpeedToSpeed(uint speed, VehicleType type)
Convert the given display speed to the (internal) speed.
TextDirection _current_text_dir
Text direction of the currently selected language.
static const Units _units_velocity_realtime[]
Unit conversions for velocity.
bool ReadLanguagePack(const LanguageMetadata *lang)
Read a particular language.
static bool GetLanguageFileHeader(const std::string &file, LanguagePackHeader *hdr)
Reads the language file header and checks compatibility.
static const Units _units_time_months_or_minutes[]
Unit conversions for time in calendar months or wallclock minutes.
static void FormatString(StringBuilder &builder, std::string_view str, StringParameters &args, uint case_index=0, bool game_script=false, bool dry_run=false)
Parse most format codes within a string and write the result to a buffer.
static const Units _units_time_years_or_minutes[]
Unit conversions for time in calendar years or wallclock minutes.
static const Units _units_power_to_weight[]
Unit conversions for power to weight.
uint64_t GetParamMaxDigits(uint count, FontSize size)
Get some number that is suitable for string size computations.
static const Units _units_time_years_or_periods[]
Unit conversions for time in calendar years or economic periods.
Functions related to OTTD's strings.
StringTab GetStringTab(StringID str)
Extract the StringTab from a StringID.
auto MakeParameters(Args &&... args)
Helper to create the StringParameters with its own buffer with the given parameter values.
StringID MakeStringID(StringTab tab, StringIndexInTab index)
Create a StringID.
StringIndexInTab GetStringIndex(StringID str)
Extract the StringIndex from a StringID.
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
StrongType::Typedef< uint32_t, struct StringIndexInTabTag, StrongType::Compare, StrongType::Integer > StringIndexInTab
The index/offset of a string within a StringTab.
static constexpr StringID SPECSTR_COMPANY_NAME_START
Special strings for company names on the form "TownName transport".
static constexpr StringID SPECSTR_SILLY_NAME
Special string for silly company names.
static const uint TAB_SIZE_GAMESCRIPT
Number of strings for GameScripts.
static const uint MAX_LANG
Maximum number of languages supported by the game, and the NewGRF specs.
static constexpr StringID SPECSTR_ANDCO_NAME
Special string for Surname & Co company names.
static constexpr StringID SPECSTR_PRESIDENT_NAME
Special string for the president's name.
static constexpr StringID SPECSTR_TOWNNAME_START
Special strings for town names.
static const uint TAB_SIZE
Number of strings per StringTab.
StringTab
StringTabs to group StringIDs.
@ TEXT_TAB_NEWGRF_START
Start of NewGRF supplied strings.
@ TEXT_TAB_GAMESCRIPT_START
Start of GameScript supplied strings.
@ TEXT_TAB_END
End of language files.
TextDirection
Directions a text can go to.
@ TD_LTR
Text is written left-to-right by default.
Class to backup a specific variable and restore it upon destruction of this object to prevent stack v...
std::string name
Name of vehicle.
StringID string_id
Default name (town area) of station.
Town * town
The town this station is associated with.
std::string name
Custom name.
VehicleType type
Type of vehicle.
StringID units_volume
Name of a single unit of cargo of this type.
static CargoSpec * Get(size_t index)
Retrieve cargo details for the given cargo type.
StringID quantifier
Text for multiple units of cargo of this type.
static size_t GetArraySize()
Total number of cargospecs, both valid and invalid.
std::string president_name
Name of the president if the user changed it.
uint32_t name_2
Parameter of name_1.
uint32_t president_name_2
Parameter of president_name_1.
StringID name_1
Name of the company if the user did not change it.
StringID president_name_1
Name of the president if the user did not change it.
std::string name
Name of the company if the user changed it.
static bool IsValidHumanID(auto index)
Is this company a valid company, not controlled by a NoAI program?
Specification of a currency.
std::string separator
The thousands separator for this currency.
std::string prefix
Prefix to apply when formatting money in this currency.
std::string suffix
Suffix to apply when formatting money in this currency.
uint16_t rate
The conversion rate compared to the base currency.
uint8_t symbol_pos
The currency symbol is represented by two possible values, prefix and suffix Usage of one or the othe...
uint16_t town_cn
The N-1th depot for this town (consecutive number)
Settings for the four different fonts.
FontCacheSubSetting mono
The mono space font used for license/readme viewers.
FontCacheSubSetting medium
The normal font size.
std::string font
The name of the font, or path to the font.
const void * os_handle
Optional native OS font info. Only valid during font search.
Dynamic data of a loaded NewGRF.
LandscapeType landscape
the landscape we're currently in
LocaleSettings locale
settings related to used currency/unit system in the current game
GameCreationSettings game_creation
settings used during the creation of a game (map)
std::string name
Group Name.
uint16_t number
Per-company group number.
Defines the data structure for constructing industry.
StringID name
Displayed name of the industry.
StringID station_name
Default name for nearby station.
Defines the internal data of a functional industry.
IndustryType type
type of industry.
std::string list_separator
Current list separator string.
std::array< uint, TEXT_TAB_END > langtab_num
Offset into langpack offs.
std::array< uint, TEXT_TAB_END > langtab_start
Offset into langpack offs.
uint8_t units_velocity_nautical
unit system for velocity of ships and aircraft
uint8_t units_velocity
unit system for velocity of trains and road vehicles
std::string digit_decimal_separator
decimal separator
std::string digit_group_separator_currency
thousand separator for currencies
uint8_t units_power
unit system for power
uint8_t units_height
unit system for height
uint8_t units_force
unit system for force
uint8_t units_volume
unit system for volume
uint8_t units_weight
unit system for weight
std::string digit_group_separator
thousand separator for non-currencies
static Titem * Get(auto index)
Returns Titem with given index.
Tindex index
Index of this pool item.
static Titem * GetIfValid(auto index)
Returns Titem with given index.
static Station * GetIfValid(auto index)
Returns station if the index is a valid index for this station type.
The data required to format and validate a single parameter of a string.
Templated helper to make a type-safe 'typedef' representing a single POD value.
std::string name
Custom town name. If empty, the town was not renamed and uses the generated name.
Helper for unit conversion.
double factor
Amount to multiply or divide upon conversion.
int64_t ToDisplay(int64_t input, bool round=true) const
Convert value from OpenTTD's internal unit into the displayed value.
int64_t FromDisplay(int64_t input, bool round=true, int64_t divider=1) const
Convert the displayed value back into a value of OpenTTD's internal unit.
Information about a specific unit system with a long variant.
StringID s
String for the short variant of the unit.
StringID l
String for the long variant of the unit.
unsigned int decimal_places
Number of decimal places embedded in the value. For example, 1 if the value is in tenths,...
UnitConversion c
Conversion.
Information about a specific unit system.
StringID s
String for the unit.
unsigned int decimal_places
Number of decimal places embedded in the value. For example, 1 if the value is in tenths,...
UnitConversion c
Conversion.
GroupID group_id
Index of group Pool array.
UnitID unitnumber
unit number, for display purposes only
Representation of a waypoint.
uint16_t town_cn
The N-1th waypoint for this town (consecutive number)
Definition of the game-calendar-timer.
Town name generator stuff.
Handling of UTF-8 encoded data.
Base class for all vehicles.
VehicleType
Available vehicle types.
@ VEH_ROAD
Road vehicle type.
@ VEH_AIRCRAFT
Aircraft vehicle type.
@ VEH_SHIP
Ship vehicle type.
@ VEH_TRAIN
Train vehicle type.
std::wstring OTTD2FS(std::string_view name)
Convert from OpenTTD's encoding to a wide string.
std::string FS2OTTD(std::wstring_view name)
Convert to OpenTTD's encoding from a wide string.
void InvalidateWindowClassesData(WindowClass cls, int data, bool gui_scope)
Mark window data of all windows of a given class as invalid (in need of re-computing) Note that by de...
Window functions not directly related to making/drawing windows.
@ WC_STATION_LIST
Station list; Window numbers:
@ WC_ROADVEH_LIST
Road vehicle list; Window numbers:
@ WC_INDUSTRY_DIRECTORY
Industry directory; Window numbers:
@ WC_SHIPS_LIST
Ships list; Window numbers:
@ WC_TRAINS_LIST
Trains list; Window numbers:
@ WC_BUILD_VEHICLE
Build vehicle; Window numbers:
@ WC_AIRCRAFT_LIST
Aircraft list; Window numbers: