44#include "table/strings.h"
46#include "3rdparty/fmt/std.h"
48#include "strings_internal.h"
72 throw std::out_of_range(
"Trying to read invalid string parameter");
76 if (param.type != 0 && param.type != this->next_type) {
78 throw std::out_of_range(
"Trying to read string parameter with wrong type");
112 void operator()(
const std::monostate &) {}
114 void operator()(
const uint64_t &arg)
120 void operator()(
const std::string &value)
130 this->builder += value;
135 for (
const auto ¶m : params) {
136 builder.
PutUtf8(SCC_RECORD_SEPARATOR);
137 std::visit(v, param.data);
152 if (this->empty())
return {};
154 std::vector<StringParameter> params;
172 params.emplace_back(std::monostate{});
177 char32_t parameter_type = record.
ReadUtf8();
178 switch (parameter_type) {
182 params.emplace_back(value);
193 params.emplace_back(std::monostate{});
198 if (param >= std::size(params))
return {};
199 params[param] = data;
209 return GetString(STR_JUST_RAW_STRING, this->
string);
221 uint64_t val = count > 1 ? front : next;
222 for (; count > 1; count--) {
223 val = 10 * val + next;
239 while (max_value >= 10) {
261static void FormatString(
StringBuilder &builder, std::string_view str, std::span<StringParameter> params, uint case_index = 0,
bool game_script =
false,
bool dry_run =
false)
264 FormatString(builder, str, tmp_params, case_index, game_script, dry_run);
261static void FormatString(
StringBuilder &builder, std::string_view str, std::span<StringParameter> params, uint case_index = 0,
bool game_script =
false,
bool dry_run =
false) {
…}
275 delete[]
reinterpret_cast<char *
>(langpack);
280 std::unique_ptr<LanguagePack, LanguagePackDeleter> langpack;
282 std::vector<std::string_view> strings;
313std::string_view GetStringPtr(
StringID string)
318 case TEXT_TAB_OLD_NEWGRF: NOT_REACHED();
322 if (offset < _langpack.strings.size())
return _langpack.strings[offset];
323 return "(undefined string)";
351 }
catch (
const std::runtime_error &e) {
352 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
353 builder +=
"(invalid string parameter)";
359 case TEXT_TAB_SPECIAL:
362 if (GetSpecialNameString(builder,
string, args))
return;
363 }
catch (
const std::runtime_error &e) {
364 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
365 builder +=
"(invalid string parameter)";
371 case TEXT_TAB_OLD_CUSTOM:
374 FatalError(
"Incorrect conversion of custom name string.");
383 case TEXT_TAB_OLD_NEWGRF:
399 FatalError(
"String 0x{:X} is invalid. You are probably using an old version of the .lng file.\n",
string);
402 FormatString(builder, GetStringPtr(
string), args, case_index);
440void AppendStringWithArgsInPlace(std::string &result,
StringID string, std::span<StringParameter> params)
469static std::string_view GetDecimalSeparator()
472 if (decimal_separator.empty()) decimal_separator = _langpack.langpack->digit_decimal_separator;
473 return decimal_separator;
484 static const int max_digits = 20;
485 uint64_t divisor = 10000000000000000000ULL;
486 int thousands_offset = (max_digits - 1) % 3;
493 uint64_t num = number;
495 for (
int i = 0; i < max_digits; i++) {
497 if (num >= divisor) {
498 quot = num / divisor;
501 if ((tot |= quot) || i == max_digits - 1) {
503 if ((i % 3) == thousands_offset && i < max_digits - 1) builder += separator;
510static void FormatCommaNumber(
StringBuilder &builder, int64_t number)
513 if (separator.empty()) separator = _langpack.langpack->digit_group_separator;
517static void FormatNoCommaNumber(
StringBuilder &builder, int64_t number)
522static void FormatZerofillNumber(
StringBuilder &builder, int64_t number,
int count)
524 fmt::format_to(builder.
back_inserter(),
"{:0{}d}", number, count);
527static void FormatHexNumber(
StringBuilder &builder, uint64_t number)
542 static const std::string_view iec_prefixes[] = {
"",
"Ki",
"Mi",
"Gi",
"Ti",
"Pi",
"Ei"};
544 while (number >= 1024 * 1024) {
552 }
else if (number < 1024 * 10) {
553 fmt::format_to(builder.
back_inserter(),
"{}{}{:02}", number / 1024, GetDecimalSeparator(), (number % 1024) * 100 / 1024);
554 }
else if (number < 1024 * 100) {
555 fmt::format_to(builder.
back_inserter(),
"{}{}{:01}", number / 1024, GetDecimalSeparator(), (number % 1024) * 10 / 1024);
557 assert(number < 1024 * 1024);
558 fmt::format_to(builder.
back_inserter(),
"{}", number / 1024);
561 assert(
id <
lengthof(iec_prefixes));
569 auto tmp_params =
MakeParameters(STR_DAY_NUMBER_1ST + ymd.day - 1, STR_MONTH_ABBREV_JAN + ymd.month, ymd.year);
570 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_LONG), tmp_params, case_index);
577 auto tmp_params =
MakeParameters(STR_MONTH_JAN + ymd.month, ymd.year);
578 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_SHORT), tmp_params, case_index);
586 auto tmp_params =
MakeParameters(ymd.day, 2, ymd.month + 1, 2, ymd.year);
594 bool negative = number < 0;
596 number *= spec->
rate;
600 builder.
PutUtf8(SCC_PUSH_COLOUR);
617 if (number >=
Money(1'000'000'000'000'000) - 500'000'000) {
618 number = (number +
Money(500'000'000'000)) /
Money(1'000'000'000'000);
619 number_str = STR_CURRENCY_SHORT_TERA;
620 }
else if (number >=
Money(1'000'000'000'000) - 500'000) {
621 number = (number + 500'000'000) / 1'000'000'000;
622 number_str = STR_CURRENCY_SHORT_GIGA;
623 }
else if (number >= 1'000'000'000 - 500) {
624 number = (number + 500'000) / 1'000'000;
625 number_str = STR_CURRENCY_SHORT_MEGA;
626 }
else if (number >= 1'000'000) {
627 number = (number + 500) / 1'000;
628 number_str = STR_CURRENCY_SHORT_KILO;
634 if (separator.empty()) separator = _langpack.langpack->digit_group_separator_currency;
636 if (number_str != STR_NULL) {
646 builder.
PutUtf8(SCC_POP_COLOUR);
659 uint64_t n =
abs(count);
661 switch (plural_form) {
670 return n != 1 ? 1 : 0;
682 return n > 1 ? 1 : 0;
689 return n % 10 == 1 && n % 100 != 11 ? 0 : n != 0 ? 1 : 2;
695 return n == 1 ? 0 : n == 2 ? 1 : n < 7 ? 2 : n < 11 ? 3 : 4;
701 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
707 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
713 return n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
719 return n % 100 == 1 ? 0 : n % 100 == 2 ? 1 : n % 100 == 3 || n % 100 == 4 ? 2 : 3;
725 return n % 10 == 1 && n % 100 != 11 ? 0 : 1;
731 return n == 1 ? 0 : n >= 2 && n <= 4 ? 1 : 2;
762 return (n == 1 ? 0 : n == 0 || (n % 100 > 1 && n % 100 < 11) ? 1 : (n % 100 > 10 && n % 100 < 20) ? 2 : 3);
767 return ((n == 1 || n == 11) ? 0 : (n == 2 || n == 12) ? 1 : ((n > 2 && n < 11) || (n > 12 && n < 20)) ? 2 : 3);
773 return n == 1 ? 0 : (n == 0 || (n % 100 > 0 && n % 100 < 20)) ? 1 : 2;
781 for (uint i = 0; i != n; i++) {
791 size_t form_pre = 0, form_len = 0, form_post = 0;
792 for (uint i = 0; i != n; i++) {
796 }
else if (i > form) {
803 consumer.
Skip(form_pre);
804 builder += consumer.
Read(form_len);
805 consumer.
Skip(form_post);
818 int64_t
ToDisplay(int64_t input,
bool round =
true)
const
821 ? (int64_t)std::round(input * this->factor)
822 : (int64_t)(input * this->factor);
818 int64_t
ToDisplay(int64_t input,
bool round =
true)
const {
…}
832 int64_t
FromDisplay(int64_t input,
bool round =
true, int64_t divider = 1)
const
835 ? (int64_t)std::round(input / this->factor / divider)
836 : (int64_t)(input / this->factor / divider);
832 int64_t
FromDisplay(int64_t input,
bool round =
true, int64_t divider = 1)
const {
…}
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.578125 }, STR_UNITS_VELOCITY_GAMEUNITS_DAY, 1 },
861 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
866 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
867 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
868 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
869 { { 0.289352 }, STR_UNITS_VELOCITY_GAMEUNITS_SEC, 1 },
870 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
875 { { 1.0 }, STR_UNITS_POWER_IMPERIAL, 0 },
876 { { 1.01387 }, STR_UNITS_POWER_METRIC, 0 },
877 { { 0.745699 }, STR_UNITS_POWER_SI, 0 },
882 { { 0.907185 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_IMPERIAL, 1 },
883 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_METRIC, 1 },
884 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_SI, 1 },
885 { { 0.919768 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_IMPERIAL, 1 },
886 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_METRIC, 1 },
887 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_SI, 1 },
888 { { 0.676487 }, STR_UNITS_POWER_SI_TO_WEIGHT_IMPERIAL, 1 },
889 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_METRIC, 1 },
890 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_SI, 1 },
895 { { 1.102311 }, STR_UNITS_WEIGHT_SHORT_IMPERIAL, STR_UNITS_WEIGHT_LONG_IMPERIAL, 0 },
896 { { 1.0 }, STR_UNITS_WEIGHT_SHORT_METRIC, STR_UNITS_WEIGHT_LONG_METRIC, 0 },
897 { { 1000.0 }, STR_UNITS_WEIGHT_SHORT_SI, STR_UNITS_WEIGHT_LONG_SI, 0 },
902 { { 264.172 }, STR_UNITS_VOLUME_SHORT_IMPERIAL, STR_UNITS_VOLUME_LONG_IMPERIAL, 0 },
903 { { 1000.0 }, STR_UNITS_VOLUME_SHORT_METRIC, STR_UNITS_VOLUME_LONG_METRIC, 0 },
904 { { 1.0 }, STR_UNITS_VOLUME_SHORT_SI, STR_UNITS_VOLUME_LONG_SI, 0 },
909 { { 0.224809 }, STR_UNITS_FORCE_IMPERIAL, 0 },
910 { { 0.101972 }, STR_UNITS_FORCE_METRIC, 0 },
911 { { 0.001 }, STR_UNITS_FORCE_SI, 0 },
916 { { 3.0 }, STR_UNITS_HEIGHT_IMPERIAL, 0 },
917 { { 1.0 }, STR_UNITS_HEIGHT_METRIC, 0 },
918 { { 1.0 }, STR_UNITS_HEIGHT_SI, 0 },
923 { { 1 }, STR_UNITS_DAYS, 0 },
924 { { 2 }, STR_UNITS_SECONDS, 0 },
929 { { 1 }, STR_UNITS_MONTHS, 0 },
930 { { 1 }, STR_UNITS_MINUTES, 0 },
935 { { 1 }, STR_UNITS_YEARS, 0 },
936 { { 1 }, STR_UNITS_PERIODS, 0 },
941 { { 1 }, STR_UNITS_YEARS, 0 },
942 { { 12 }, STR_UNITS_MINUTES, 0 },
1013 std::vector<StringParameter> sub_args;
1018 builder +=
"(invalid SCC_ENCODED)";
1023 builder +=
"(invalid StringID)";
1033 sub_args.emplace_back(std::monostate{});
1038 char32_t parameter_type = record.
ReadUtf8();
1039 switch (parameter_type) {
1043 builder +=
"(invalid sub-StringID)";
1048 sub_args.emplace_back(param);
1055 sub_args.emplace_back(param);
1066 sub_args.emplace_back(std::monostate{});
1084 size_t orig_first_param_offset = args.
GetOffset();
1097 FormatString(dry_run_builder, str_arg, args, orig_case_index, game_script,
true);
1099 args.
SetOffset(orig_first_param_offset);
1101 uint next_substr_case_index = 0;
1102 struct StrStackItem {
1104 size_t first_param_offset;
1107 StrStackItem(std::string_view view,
size_t first_param_offset, uint case_index)
1108 : consumer(view), first_param_offset(first_param_offset), case_index(case_index)
1111 std::stack<StrStackItem, std::vector<StrStackItem>> str_stack;
1112 str_stack.emplace(str_arg, orig_first_param_offset, orig_case_index);
1116 while (!str_stack.empty() && !str_stack.top().consumer.AnyBytesLeft()) {
1119 if (str_stack.empty())
break;
1121 const size_t ref_param_offset = str_stack.top().first_param_offset;
1122 const uint case_index = str_stack.top().case_index;
1127 builder +=
"(unexpected NUL)";
1134 if (b == 0)
continue;
1137 if (b < SCC_CONTROL_START || b > SCC_CONTROL_END) {
1142 args.SetTypeOfNextParameter(b);
1151 std::string_view ptr = GetStringPtr(substr);
1152 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1153 next_substr_case_index = 0;
1159 std::string_view ptr = GetStringPtr(substr);
1160 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1161 next_substr_case_index = 0;
1165 case SCC_GENDER_LIST: {
1167 size_t offset = ref_param_offset + consumer.
ReadUint8();
1171 builder +=
"(invalid GENDER parameter)";
1193 if (gender_consumer.
ReadUtf8If(SCC_GENDER_INDEX)) {
1197 ParseStringChoice(consumer, gender, builder);
1203 case SCC_GENDER_INDEX: {
1206 builder.
PutUtf8(SCC_GENDER_INDEX);
1212 case SCC_PLURAL_LIST: {
1213 uint8_t plural_form = consumer.
ReadUint8();
1214 size_t offset = ref_param_offset + consumer.
ReadUint8();
1215 const uint64_t *v =
nullptr;
1218 v = std::get_if<uint64_t>(&args.GetParam(offset));
1221 ParseStringChoice(consumer,
DeterminePluralForm(
static_cast<int64_t
>(*v), plural_form), builder);
1223 SkipStringChoice(consumer);
1224 builder +=
"(invalid PLURAL parameter)";
1229 case SCC_ARG_INDEX: {
1234 case SCC_SET_CASE: {
1237 next_substr_case_index = consumer.
ReadUint8();
1241 case SCC_SWITCH_CASE: {
1245 std::optional<std::string_view> found;
1246 for (; num > 0; --num) {
1249 auto case_str = consumer.
Read(len);
1250 if (index == case_index) {
1256 auto default_str = consumer.
Read(default_len);
1257 if (!found.has_value()) found = default_str;
1258 str_stack.emplace(*found, ref_param_offset, case_index);
1263 builder += _openttd_revision;
1266 case SCC_RAW_STRING_POINTER:
1275 GetStringWithArgs(builder, string_id, tmp_params, next_substr_case_index, game_script);
1276 next_substr_case_index = 0;
1290 uint size = b - SCC_STRING1 + 1;
1292 builder +=
"(consumed too many parameters)";
1295 GetStringWithArgs(builder, string_id, sub_args, next_substr_case_index, game_script);
1298 next_substr_case_index = 0;
1310 FormatCommaNumber(builder, number);
1315 int64_t fractional = number % divisor;
1317 FormatCommaNumber(builder, number);
1318 fmt::format_to(builder.
back_inserter(),
"{}{:0{}d}", GetDecimalSeparator(), fractional, digits);
1326 case SCC_ZEROFILL_NUM: {
1340 case SCC_CARGO_TINY: {
1348 builder +=
"(invalid cargo type)";
1365 FormatCommaNumber(builder, amount);
1369 case SCC_CARGO_SHORT: {
1377 builder +=
"(invalid cargo type)";
1382 switch (cargo_str) {
1386 auto tmp_params =
MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1394 auto tmp_params =
MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1408 case SCC_CARGO_LONG: {
1418 builder +=
"(invalid cargo type)";
1423 case SCC_CARGO_LIST: {
1429 if (!
HasBit(cmask, cs->Index()))
continue;
1435 builder += list_separator;
1438 GetStringWithArgs(builder, cs->name, args, next_substr_case_index, game_script);
1442 if (first)
GetStringWithArgs(builder, STR_JUST_NOTHING, args, next_substr_case_index, game_script);
1444 next_substr_case_index = 0;
1448 case SCC_CURRENCY_SHORT:
1452 case SCC_CURRENCY_LONG:
1460 case SCC_DATE_SHORT:
1462 next_substr_case_index = 0;
1467 next_substr_case_index = 0;
1498 case SCC_POWER_TO_WEIGHT: {
1507 case SCC_VELOCITY: {
1517 case SCC_VOLUME_SHORT: {
1525 case SCC_VOLUME_LONG: {
1533 case SCC_WEIGHT_SHORT: {
1541 case SCC_WEIGHT_LONG: {
1549 case SCC_UNITS_DAYS_OR_SECONDS: {
1557 case SCC_UNITS_MONTHS_OR_MINUTES: {
1565 case SCC_UNITS_YEARS_OR_PERIODS: {
1573 case SCC_UNITS_YEARS_OR_MINUTES: {
1581 case SCC_COMPANY_NAME: {
1583 if (c ==
nullptr)
break;
1585 if (!c->
name.empty()) {
1595 case SCC_COMPANY_NUM: {
1606 case SCC_DEPOT_NAME: {
1615 if (!d->name.empty()) {
1625 case SCC_ENGINE_NAME: {
1628 if (e ==
nullptr)
break;
1630 if (!e->name.empty() && e->IsEnabled()) {
1637 std::array<int32_t, 16> regs100;
1640 if (callback == 0x40F) {
1641 const GRFFile *grffile = e->GetGRF();
1642 assert(grffile !=
nullptr);
1646 }
else if (callback < 0x400) {
1647 const GRFFile *grffile = e->GetGRF();
1648 assert(grffile !=
nullptr);
1659 case SCC_GROUP_NAME: {
1661 if (g ==
nullptr)
break;
1663 if (!g->
name.empty()) {
1673 case SCC_INDUSTRY_NAME: {
1675 if (i ==
nullptr)
break;
1677 static bool use_cache =
true;
1682 }
else if (use_cache) {
1684 builder += i->GetCachedName();
1688 FormatString(builder, GetStringPtr(STR_FORMAT_INDUSTRY_NAME), tmp_params, next_substr_case_index);
1690 next_substr_case_index = 0;
1694 case SCC_PRESIDENT_NAME: {
1696 if (c ==
nullptr)
break;
1708 case SCC_STATION_NAME: {
1712 if (st ==
nullptr) {
1720 static bool use_cache =
true;
1723 builder += st->GetCachedName();
1724 }
else if (!st->name.empty()) {
1728 StringID string_id = st->string_id;
1729 if (st->indtype != IT_INVALID) {
1741 auto tmp_params =
MakeParameters(STR_TOWN_NAME, st->town->index, st->index);
1747 case SCC_TOWN_NAME: {
1749 if (t ==
nullptr)
break;
1751 static bool use_cache =
true;
1754 builder += t->GetCachedName();
1755 }
else if (!t->
name.empty()) {
1759 GetTownName(builder, t);
1764 case SCC_WAYPOINT_NAME: {
1766 if (wp ==
nullptr)
break;
1768 if (!wp->
name.empty()) {
1773 StringID string_id = ((wp->
string_id == STR_SV_STNAME_BUOY) ? STR_FORMAT_BUOY_NAME : STR_FORMAT_WAYPOINT_NAME);
1774 if (wp->
town_cn != 0) string_id++;
1780 case SCC_VEHICLE_NAME: {
1782 if (v ==
nullptr)
break;
1784 if (!v->
name.empty()) {
1796 default: string_id = STR_INVALID_VEHICLE;
break;
1797 case VEH_TRAIN: string_id = STR_SV_TRAIN_NAME;
break;
1798 case VEH_ROAD: string_id = STR_SV_ROAD_VEHICLE_NAME;
break;
1799 case VEH_SHIP: string_id = STR_SV_SHIP_NAME;
break;
1800 case VEH_AIRCRAFT: string_id = STR_SV_AIRCRAFT_NAME;
break;
1808 case SCC_SIGN_NAME: {
1810 if (si ==
nullptr)
break;
1812 if (!si->name.empty()) {
1821 case SCC_STATION_FEATURES: {
1836 }
catch (std::out_of_range &e) {
1837 Debug(misc, 0,
"FormatString: {}", e.what());
1838 builder +=
"(invalid parameter)";
1853static const std::string_view _silly_company_names[] = {
1855 "Tiny Transport Ltd.",
1857 "Comfy-Coach & Co.",
1858 "Crush & Bump Ltd.",
1859 "Broken & Late Ltd.",
1861 "Supersonic Travel",
1863 "Lightning International",
1864 "Pannik & Loozit Ltd.",
1865 "Inter-City Transport",
1866 "Getout & Pushit Ltd."
1869static const std::string_view _surname_list[] = {
1901static const std::string_view _silly_surname_list[] = {
1916static const char _initial_name_letters[] = {
1917 'A',
'B',
'C',
'D',
'E',
'F',
'G',
'H',
'I',
'J',
1918 'K',
'L',
'M',
'N',
'P',
'R',
'S',
'T',
'W',
1921static std::span<const std::string_view> GetSurnameOptions()
1924 return _surname_list;
1934 auto surname_options = GetSurnameOptions();
1935 return surname_options[surname_options.size() *
GB(seed, 16, 8) >> 8];
1938static void GenAndCoName(
StringBuilder &builder, uint32_t seed)
1941 builder +=
" & Co.";
1944static void GenPresidentName(
StringBuilder &builder, uint32_t seed)
1946 builder.
PutChar(_initial_name_letters[std::size(_initial_name_letters) *
GB(seed, 0, 8) >> 8]);
1950 size_t index = (std::size(_initial_name_letters) + 35) *
GB(seed, 8, 8) >> 8;
1951 if (index < std::size(_initial_name_letters)) {
1952 builder.
PutChar(_initial_name_letters[index]);
1963 builder += _silly_company_names[std::min<size_t>(args.
GetNextParameter<uint16_t>(), std::size(_silly_company_names) - 1)];
1978 builder +=
" Transport";
1992 this->
version == TO_LE32(LANGUAGE_PACK_VERSION) &&
2012 return 4 * this->
missing < LANGUAGE_TOTAL_STRINGS;
2023 size_t total_len = 0;
2025 if (!lang_pack)
return false;
2028 const char *end = (
char *)lang_pack.get() + total_len + 1;
2031 if (end <= lang_pack->data || !lang_pack->IsValid()) {
2035 std::array<uint, TEXT_TAB_END> tab_start, tab_num;
2039 uint16_t num = FROM_LE16(lang_pack->offsets[i]);
2042 tab_start[i] = count;
2048 std::vector<std::string_view> strings;
2051 char *s = lang_pack->data;
2052 for (uint i = 0; i < count; i++) {
2053 size_t len =
static_cast<uint8_t
>(*s++);
2054 if (s + len >= end)
return false;
2057 len = ((len & 0x3F) << 8) +
static_cast<uint8_t
>(*s++);
2058 if (s + len >= end)
return false;
2060 strings.emplace_back(s, len);
2063 assert(strings.size() == count);
2065 _langpack.langpack = std::move(lang_pack);
2066 _langpack.strings = std::move(strings);
2078 extern void Win32SetCurrentLocaleName(std::string iso_code);
2089 UErrorCode status = U_ZERO_ERROR;
2094 if (U_FAILURE(status)) {
2120#if !(defined(_WIN32) || defined(__APPLE__))
2131 auto env =
GetEnv(
"LANGUAGE");
2132 if (env.has_value())
return std::string{*env};
2135 if (env.has_value())
return std::string{*env};
2137 if (param !=
nullptr) {
2139 if (env.has_value())
return std::string{*env};
2143 if (env.has_value())
return std::string{*env};
2145 return std::nullopt;
2159 if (newgrflangid == lang.newgrflangid)
return ⟨
2174 if (!f.has_value())
return false;
2176 size_t read = fread(hdr,
sizeof(*hdr), 1, *f);
2178 bool ret = read == 1 && hdr->
IsValid();
2194 std::error_code error_code;
2195 for (
const auto &dir_entry : std::filesystem::directory_iterator(
OTTD2FS(path), error_code)) {
2196 if (!dir_entry.is_regular_file())
continue;
2197 if (dir_entry.path().extension() !=
".lng")
continue;
2200 lmd.
file = dir_entry.path();
2205 Debug(misc, 3,
"{} is not a valid language file", file);
2207 Debug(misc, 3,
"{}'s language ID is already known", file);
2213 Debug(misc, 9,
"Unable to open directory {}: {}", path, error_code.message());
2226 if (
_languages.empty()) UserError(
"No available language packs (invalid versions?)");
2230 std::string_view lang = str_lang.has_value() ? std::string_view{*str_lang} :
"en_GB";
2242 chosen_language = &lng;
2246 std::string_view iso_code = lng.
isocode;
2247 if (iso_code ==
"en_GB") en_GB_fallback = &lng;
2250 if (!lng.IsReasonablyFinished())
continue;
2252 if (iso_code.starts_with(lang.substr(0, 5))) chosen_language = &lng;
2253 if (iso_code.starts_with(lang.substr(0, 2))) language_fallback = &lng;
2258 if (chosen_language ==
nullptr) {
2259 chosen_language = (language_fallback !=
nullptr) ? language_fallback : en_GB_fallback;
2271 return _langpack.langpack->isocode;
2286 for (
char32_t c :
Utf8View(*text)) {
2287 if (c >= SCC_FIRST_FONT && c <= SCC_LAST_FONT) {
2288 size = (
FontSize)(c - SCC_FIRST_FONT);
2292 Debug(fontcache, 0,
"Font is missing glyphs to display char 0x{:X} in {} font size",
static_cast<uint32_t
>(c), FontSizeToName(size));
2320 std::string_view ret = _langpack.strings[_langpack.
langtab_start[this->
i] + this->
j];
2323 while (this->i < TEXT_TAB_END && this->
j >= _langpack.
langtab_num[this->i]) {
2338#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2343 settings->small.os_handle = os_data;
2344 settings->medium.os_handle = os_data;
2345 settings->large.os_handle = os_data;
2366 if (searcher ==
nullptr) searcher = &pack_searcher;
2368#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2372 bool any_font_configured = !_fcsettings.
medium.
font.empty();
2380 _fcsettings = std::move(backup);
2382 if (!bad_font && any_font_configured) {
2386 std::string err_str;
2389 builder.
Put(
"The current font is missing some of the characters used in the texts for this language. Using system fallback font instead.");
2393 if (bad_font && base_font) {
2407 std::string err_str;
2410 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.");
2421#if !(defined(WITH_ICU_I18N) && defined(WITH_HARFBUZZ)) && !defined(WITH_UNISCRIBE) && !defined(WITH_COCOA)
2432 std::string err_str;
2435 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 void LoadFontCaches(FontSizes fontsizes)
(Re)initialize the font cache related things, i.e.
static FontCache * Get(FontSize fs)
Get the font cache of a given font size.
static bool FindFallbackFont(FontCacheSettings *settings, const std::string &language_isocode, MissingGlyphSearcher *callback)
We would like to have a fallback font as the current one doesn't contain all characters we need.
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 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(FontSizes fontsizes)
Initialize _stringwidth_table cache for the specified font sizes.
Functions related to laying out the texts.
FontSize
Available font sizes.
@ FS_MONO
Index of the monospaced font in the font tables.
@ FS_NORMAL
Index of the normal font in the font tables.
constexpr FontSizes FONTSIZES_REQUIRED
Mask of font sizes required to be present.
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.
std::string_view GetEllipsis()
Get the ellipsis string for the current language.
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.
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::string ellipsis
Current ellipsis 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: