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;
314std::string_view GetStringPtr(
StringID string)
319 case TEXT_TAB_OLD_NEWGRF: NOT_REACHED();
323 if (offset < _langpack.strings.size())
return _langpack.strings[offset];
324 return "(undefined string)";
352 }
catch (
const std::runtime_error &e) {
353 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
354 builder +=
"(invalid string parameter)";
360 case TEXT_TAB_SPECIAL:
363 if (GetSpecialNameString(builder,
string, args))
return;
364 }
catch (
const std::runtime_error &e) {
365 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
366 builder +=
"(invalid string parameter)";
372 case TEXT_TAB_OLD_CUSTOM:
375 FatalError(
"Incorrect conversion of custom name string.");
384 case TEXT_TAB_OLD_NEWGRF:
400 FatalError(
"String 0x{:X} is invalid. You are probably using an old version of the .lng file.\n",
string);
403 FormatString(builder, GetStringPtr(
string), args, case_index);
441void AppendStringWithArgsInPlace(std::string &result,
StringID string, std::span<StringParameter> params)
470static std::string_view GetDecimalSeparator()
473 if (decimal_separator.empty()) decimal_separator = _langpack.langpack->digit_decimal_separator;
474 return decimal_separator;
485 static const int max_digits = 20;
486 uint64_t divisor = 10000000000000000000ULL;
487 int thousands_offset = (max_digits - 1) % 3;
494 uint64_t num = number;
496 for (
int i = 0; i < max_digits; i++) {
498 if (num >= divisor) {
499 quot = num / divisor;
502 if ((tot |= quot) || i == max_digits - 1) {
504 if ((i % 3) == thousands_offset && i < max_digits - 1) builder += separator;
511static void FormatCommaNumber(
StringBuilder &builder, int64_t number)
514 if (separator.empty()) separator = _langpack.langpack->digit_group_separator;
518static void FormatNoCommaNumber(
StringBuilder &builder, int64_t number)
523static void FormatZerofillNumber(
StringBuilder &builder, int64_t number,
int count)
525 fmt::format_to(builder.
back_inserter(),
"{:0{}d}", number, count);
528static void FormatHexNumber(
StringBuilder &builder, uint64_t number)
543 static const std::string_view iec_prefixes[] = {
"",
"Ki",
"Mi",
"Gi",
"Ti",
"Pi",
"Ei"};
545 while (number >= 1024 * 1024) {
553 }
else if (number < 1024 * 10) {
554 fmt::format_to(builder.
back_inserter(),
"{}{}{:02}", number / 1024, GetDecimalSeparator(), (number % 1024) * 100 / 1024);
555 }
else if (number < 1024 * 100) {
556 fmt::format_to(builder.
back_inserter(),
"{}{}{:01}", number / 1024, GetDecimalSeparator(), (number % 1024) * 10 / 1024);
558 assert(number < 1024 * 1024);
559 fmt::format_to(builder.
back_inserter(),
"{}", number / 1024);
562 assert(
id <
lengthof(iec_prefixes));
570 auto tmp_params =
MakeParameters(STR_DAY_NUMBER_1ST + ymd.day - 1, STR_MONTH_ABBREV_JAN + ymd.month, ymd.year);
571 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_LONG), tmp_params, case_index);
578 auto tmp_params =
MakeParameters(STR_MONTH_JAN + ymd.month, ymd.year);
579 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_SHORT), tmp_params, case_index);
587 auto tmp_params =
MakeParameters(ymd.day, 2, ymd.month + 1, 2, ymd.year);
595 bool negative = number < 0;
597 number *= spec->
rate;
601 builder.
PutUtf8(SCC_PUSH_COLOUR);
618 if (number >=
Money(1'000'000'000'000'000) - 500'000'000) {
619 number = (number +
Money(500'000'000'000)) /
Money(1'000'000'000'000);
620 number_str = STR_CURRENCY_SHORT_TERA;
621 }
else if (number >=
Money(1'000'000'000'000) - 500'000) {
622 number = (number + 500'000'000) / 1'000'000'000;
623 number_str = STR_CURRENCY_SHORT_GIGA;
624 }
else if (number >= 1'000'000'000 - 500) {
625 number = (number + 500'000) / 1'000'000;
626 number_str = STR_CURRENCY_SHORT_MEGA;
627 }
else if (number >= 1'000'000) {
628 number = (number + 500) / 1'000;
629 number_str = STR_CURRENCY_SHORT_KILO;
635 if (separator.empty()) separator = _langpack.langpack->digit_group_separator_currency;
637 if (number_str != STR_NULL) {
647 builder.
PutUtf8(SCC_POP_COLOUR);
660 uint64_t n =
abs(count);
662 switch (plural_form) {
671 return n != 1 ? 1 : 0;
683 return n > 1 ? 1 : 0;
690 return n % 10 == 1 && n % 100 != 11 ? 0 : n != 0 ? 1 : 2;
696 return n == 1 ? 0 : n == 2 ? 1 : n < 7 ? 2 : n < 11 ? 3 : 4;
702 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
708 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
714 return n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
720 return n % 100 == 1 ? 0 : n % 100 == 2 ? 1 : n % 100 == 3 || n % 100 == 4 ? 2 : 3;
726 return n % 10 == 1 && n % 100 != 11 ? 0 : 1;
732 return n == 1 ? 0 : n >= 2 && n <= 4 ? 1 : 2;
763 return (n == 1 ? 0 : n == 0 || (n % 100 > 1 && n % 100 < 11) ? 1 : (n % 100 > 10 && n % 100 < 20) ? 2 : 3);
768 return ((n == 1 || n == 11) ? 0 : (n == 2 || n == 12) ? 1 : ((n > 2 && n < 11) || (n > 12 && n < 20)) ? 2 : 3);
774 return n == 1 ? 0 : (n == 0 || (n % 100 > 0 && n % 100 < 20)) ? 1 : 2;
782 for (uint i = 0; i != n; i++) {
792 size_t form_pre = 0, form_len = 0, form_post = 0;
793 for (uint i = 0; i != n; i++) {
797 }
else if (i > form) {
804 consumer.
Skip(form_pre);
805 builder += consumer.
Read(form_len);
806 consumer.
Skip(form_post);
819 int64_t
ToDisplay(int64_t input,
bool round =
true)
const
822 ? (int64_t)std::round(input * this->factor)
823 : (int64_t)(input * this->factor);
819 int64_t
ToDisplay(int64_t input,
bool round =
true)
const {
…}
833 int64_t
FromDisplay(int64_t input,
bool round =
true, int64_t divider = 1)
const
836 ? (int64_t)std::round(input / this->factor / divider)
837 : (int64_t)(input / this->factor / divider);
833 int64_t
FromDisplay(int64_t input,
bool round =
true, int64_t divider = 1)
const {
…}
858 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
859 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
860 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
861 { { 0.578125 }, STR_UNITS_VELOCITY_GAMEUNITS_DAY, 1 },
862 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
867 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
868 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
869 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
870 { { 0.289352 }, STR_UNITS_VELOCITY_GAMEUNITS_SEC, 1 },
871 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
876 { { 1.0 }, STR_UNITS_POWER_IMPERIAL, 0 },
877 { { 1.01387 }, STR_UNITS_POWER_METRIC, 0 },
878 { { 0.745699 }, STR_UNITS_POWER_SI, 0 },
883 { { 0.907185 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_IMPERIAL, 1 },
884 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_METRIC, 1 },
885 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_SI, 1 },
886 { { 0.919768 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_IMPERIAL, 1 },
887 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_METRIC, 1 },
888 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_SI, 1 },
889 { { 0.676487 }, STR_UNITS_POWER_SI_TO_WEIGHT_IMPERIAL, 1 },
890 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_METRIC, 1 },
891 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_SI, 1 },
896 { { 1.102311 }, STR_UNITS_WEIGHT_SHORT_IMPERIAL, STR_UNITS_WEIGHT_LONG_IMPERIAL, 0 },
897 { { 1.0 }, STR_UNITS_WEIGHT_SHORT_METRIC, STR_UNITS_WEIGHT_LONG_METRIC, 0 },
898 { { 1000.0 }, STR_UNITS_WEIGHT_SHORT_SI, STR_UNITS_WEIGHT_LONG_SI, 0 },
903 { { 264.172 }, STR_UNITS_VOLUME_SHORT_IMPERIAL, STR_UNITS_VOLUME_LONG_IMPERIAL, 0 },
904 { { 1000.0 }, STR_UNITS_VOLUME_SHORT_METRIC, STR_UNITS_VOLUME_LONG_METRIC, 0 },
905 { { 1.0 }, STR_UNITS_VOLUME_SHORT_SI, STR_UNITS_VOLUME_LONG_SI, 0 },
910 { { 0.224809 }, STR_UNITS_FORCE_IMPERIAL, 0 },
911 { { 0.101972 }, STR_UNITS_FORCE_METRIC, 0 },
912 { { 0.001 }, STR_UNITS_FORCE_SI, 0 },
917 { { 3.0 }, STR_UNITS_HEIGHT_IMPERIAL, 0 },
918 { { 1.0 }, STR_UNITS_HEIGHT_METRIC, 0 },
919 { { 1.0 }, STR_UNITS_HEIGHT_SI, 0 },
924 { { 1 }, STR_UNITS_DAYS, 0 },
925 { { 2 }, STR_UNITS_SECONDS, 0 },
930 { { 1 }, STR_UNITS_MONTHS, 0 },
931 { { 1 }, STR_UNITS_MINUTES, 0 },
936 { { 1 }, STR_UNITS_YEARS, 0 },
937 { { 1 }, STR_UNITS_PERIODS, 0 },
942 { { 1 }, STR_UNITS_YEARS, 0 },
943 { { 12 }, STR_UNITS_MINUTES, 0 },
1014 std::vector<StringParameter> sub_args;
1019 builder +=
"(invalid SCC_ENCODED)";
1024 builder +=
"(invalid StringID)";
1034 sub_args.emplace_back(std::monostate{});
1039 char32_t parameter_type = record.
ReadUtf8();
1040 switch (parameter_type) {
1044 builder +=
"(invalid sub-StringID)";
1049 sub_args.emplace_back(param);
1056 sub_args.emplace_back(param);
1067 sub_args.emplace_back(std::monostate{});
1085 size_t orig_first_param_offset = args.
GetOffset();
1098 FormatString(dry_run_builder, str_arg, args, orig_case_index, game_script,
true);
1100 args.
SetOffset(orig_first_param_offset);
1102 uint next_substr_case_index = 0;
1103 struct StrStackItem {
1105 size_t first_param_offset;
1108 StrStackItem(std::string_view view,
size_t first_param_offset, uint case_index)
1109 : consumer(view), first_param_offset(first_param_offset), case_index(case_index)
1112 std::stack<StrStackItem, std::vector<StrStackItem>> str_stack;
1113 str_stack.emplace(str_arg, orig_first_param_offset, orig_case_index);
1117 while (!str_stack.empty() && !str_stack.top().consumer.AnyBytesLeft()) {
1120 if (str_stack.empty())
break;
1122 const size_t ref_param_offset = str_stack.top().first_param_offset;
1123 const uint case_index = str_stack.top().case_index;
1128 builder +=
"(unexpected NUL)";
1135 if (b == 0)
continue;
1138 if (b < SCC_CONTROL_START || b > SCC_CONTROL_END) {
1143 args.SetTypeOfNextParameter(b);
1152 std::string_view ptr = GetStringPtr(substr);
1153 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1154 next_substr_case_index = 0;
1160 std::string_view ptr = GetStringPtr(substr);
1161 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1162 next_substr_case_index = 0;
1166 case SCC_GENDER_LIST: {
1168 size_t offset = ref_param_offset + consumer.
ReadUint8();
1172 builder +=
"(invalid GENDER parameter)";
1194 if (gender_consumer.
ReadUtf8If(SCC_GENDER_INDEX)) {
1198 ParseStringChoice(consumer, gender, builder);
1204 case SCC_GENDER_INDEX: {
1207 builder.
PutUtf8(SCC_GENDER_INDEX);
1213 case SCC_PLURAL_LIST: {
1214 uint8_t plural_form = consumer.
ReadUint8();
1215 size_t offset = ref_param_offset + consumer.
ReadUint8();
1216 const uint64_t *v =
nullptr;
1219 v = std::get_if<uint64_t>(&args.GetParam(offset));
1222 ParseStringChoice(consumer,
DeterminePluralForm(
static_cast<int64_t
>(*v), plural_form), builder);
1224 SkipStringChoice(consumer);
1225 builder +=
"(invalid PLURAL parameter)";
1230 case SCC_ARG_INDEX: {
1235 case SCC_SET_CASE: {
1238 next_substr_case_index = consumer.
ReadUint8();
1242 case SCC_SWITCH_CASE: {
1246 std::optional<std::string_view> found;
1247 for (; num > 0; --num) {
1250 auto case_str = consumer.
Read(len);
1251 if (index == case_index) {
1257 auto default_str = consumer.
Read(default_len);
1258 if (!found.has_value()) found = default_str;
1259 str_stack.emplace(*found, ref_param_offset, case_index);
1264 builder += _openttd_revision;
1267 case SCC_RAW_STRING_POINTER:
1276 GetStringWithArgs(builder, string_id, tmp_params, next_substr_case_index, game_script);
1277 next_substr_case_index = 0;
1291 uint size = b - SCC_STRING1 + 1;
1293 builder +=
"(consumed too many parameters)";
1296 GetStringWithArgs(builder, string_id, sub_args, next_substr_case_index, game_script);
1299 next_substr_case_index = 0;
1311 FormatCommaNumber(builder, number);
1316 int64_t fractional = number % divisor;
1318 FormatCommaNumber(builder, number);
1319 fmt::format_to(builder.
back_inserter(),
"{}{:0{}d}", GetDecimalSeparator(), fractional, digits);
1327 case SCC_ZEROFILL_NUM: {
1341 case SCC_CARGO_TINY: {
1349 builder +=
"(invalid cargo type)";
1366 FormatCommaNumber(builder, amount);
1370 case SCC_CARGO_SHORT: {
1378 builder +=
"(invalid cargo type)";
1383 switch (cargo_str) {
1387 auto tmp_params =
MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1395 auto tmp_params =
MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1409 case SCC_CARGO_LONG: {
1419 builder +=
"(invalid cargo type)";
1424 case SCC_CARGO_LIST: {
1430 if (!
HasBit(cmask, cs->Index()))
continue;
1436 builder += list_separator;
1439 GetStringWithArgs(builder, cs->name, args, next_substr_case_index, game_script);
1443 if (first)
GetStringWithArgs(builder, STR_JUST_NOTHING, args, next_substr_case_index, game_script);
1445 next_substr_case_index = 0;
1449 case SCC_CURRENCY_SHORT:
1453 case SCC_CURRENCY_LONG:
1461 case SCC_DATE_SHORT:
1463 next_substr_case_index = 0;
1468 next_substr_case_index = 0;
1499 case SCC_POWER_TO_WEIGHT: {
1508 case SCC_VELOCITY: {
1518 case SCC_VOLUME_SHORT: {
1526 case SCC_VOLUME_LONG: {
1534 case SCC_WEIGHT_SHORT: {
1542 case SCC_WEIGHT_LONG: {
1550 case SCC_UNITS_DAYS_OR_SECONDS: {
1558 case SCC_UNITS_MONTHS_OR_MINUTES: {
1566 case SCC_UNITS_YEARS_OR_PERIODS: {
1574 case SCC_UNITS_YEARS_OR_MINUTES: {
1582 case SCC_COMPANY_NAME: {
1584 if (c ==
nullptr)
break;
1586 if (!c->
name.empty()) {
1596 case SCC_COMPANY_NUM: {
1607 case SCC_DEPOT_NAME: {
1616 if (!d->name.empty()) {
1626 case SCC_ENGINE_NAME: {
1629 if (e ==
nullptr)
break;
1631 if (!e->name.empty() && e->IsEnabled()) {
1638 std::array<int32_t, 16> regs100;
1641 if (callback == 0x40F) {
1642 const GRFFile *grffile = e->GetGRF();
1643 assert(grffile !=
nullptr);
1647 }
else if (callback < 0x400) {
1648 const GRFFile *grffile = e->GetGRF();
1649 assert(grffile !=
nullptr);
1660 case SCC_GROUP_NAME: {
1662 if (g ==
nullptr)
break;
1664 if (!g->
name.empty()) {
1674 case SCC_INDUSTRY_NAME: {
1676 if (i ==
nullptr)
break;
1678 static bool use_cache =
true;
1683 }
else if (use_cache) {
1685 builder += i->GetCachedName();
1689 FormatString(builder, GetStringPtr(STR_FORMAT_INDUSTRY_NAME), tmp_params, next_substr_case_index);
1691 next_substr_case_index = 0;
1695 case SCC_PRESIDENT_NAME: {
1697 if (c ==
nullptr)
break;
1709 case SCC_STATION_NAME: {
1713 if (st ==
nullptr) {
1721 static bool use_cache =
true;
1724 builder += st->GetCachedName();
1725 }
else if (!st->name.empty()) {
1729 StringID string_id = st->string_id;
1730 if (st->indtype != IT_INVALID) {
1742 auto tmp_params =
MakeParameters(STR_TOWN_NAME, st->town->index, st->index);
1748 case SCC_TOWN_NAME: {
1750 if (t ==
nullptr)
break;
1752 static bool use_cache =
true;
1755 builder += t->GetCachedName();
1756 }
else if (!t->
name.empty()) {
1760 GetTownName(builder, t);
1765 case SCC_WAYPOINT_NAME: {
1767 if (wp ==
nullptr)
break;
1769 if (!wp->
name.empty()) {
1774 StringID string_id = ((wp->
string_id == STR_SV_STNAME_BUOY) ? STR_FORMAT_BUOY_NAME : STR_FORMAT_WAYPOINT_NAME);
1775 if (wp->
town_cn != 0) string_id++;
1781 case SCC_VEHICLE_NAME: {
1783 if (v ==
nullptr)
break;
1785 if (!v->
name.empty()) {
1797 default: string_id = STR_INVALID_VEHICLE;
break;
1798 case VEH_TRAIN: string_id = STR_SV_TRAIN_NAME;
break;
1799 case VEH_ROAD: string_id = STR_SV_ROAD_VEHICLE_NAME;
break;
1800 case VEH_SHIP: string_id = STR_SV_SHIP_NAME;
break;
1801 case VEH_AIRCRAFT: string_id = STR_SV_AIRCRAFT_NAME;
break;
1809 case SCC_SIGN_NAME: {
1811 if (si ==
nullptr)
break;
1813 if (!si->name.empty()) {
1822 case SCC_STATION_FEATURES: {
1837 }
catch (std::out_of_range &e) {
1838 Debug(misc, 0,
"FormatString: {}", e.what());
1839 builder +=
"(invalid parameter)";
1854static const std::string_view _silly_company_names[] = {
1856 "Tiny Transport Ltd.",
1858 "Comfy-Coach & Co.",
1859 "Crush & Bump Ltd.",
1860 "Broken & Late Ltd.",
1862 "Supersonic Travel",
1864 "Lightning International",
1865 "Pannik & Loozit Ltd.",
1866 "Inter-City Transport",
1867 "Getout & Pushit Ltd."
1870static const std::string_view _surname_list[] = {
1902static const std::string_view _silly_surname_list[] = {
1917static const char _initial_name_letters[] = {
1918 'A',
'B',
'C',
'D',
'E',
'F',
'G',
'H',
'I',
'J',
1919 'K',
'L',
'M',
'N',
'P',
'R',
'S',
'T',
'W',
1922static std::span<const std::string_view> GetSurnameOptions()
1925 return _surname_list;
1935 auto surname_options = GetSurnameOptions();
1936 return surname_options[surname_options.size() *
GB(seed, 16, 8) >> 8];
1939static void GenAndCoName(
StringBuilder &builder, uint32_t seed)
1942 builder +=
" & Co.";
1945static void GenPresidentName(
StringBuilder &builder, uint32_t seed)
1947 builder.
PutChar(_initial_name_letters[std::size(_initial_name_letters) *
GB(seed, 0, 8) >> 8]);
1951 size_t index = (std::size(_initial_name_letters) + 35) *
GB(seed, 8, 8) >> 8;
1952 if (index < std::size(_initial_name_letters)) {
1953 builder.
PutChar(_initial_name_letters[index]);
1964 builder += _silly_company_names[std::min<size_t>(args.
GetNextParameter<uint16_t>(), std::size(_silly_company_names) - 1)];
1979 builder +=
" Transport";
1993 this->
version == TO_LE32(LANGUAGE_PACK_VERSION) &&
2013 return 4 * this->
missing < LANGUAGE_TOTAL_STRINGS;
2024 size_t total_len = 0;
2026 if (!lang_pack)
return false;
2029 const char *end = (
char *)lang_pack.get() + total_len + 1;
2032 if (end <= lang_pack->data || !lang_pack->IsValid()) {
2036 std::array<uint, TEXT_TAB_END> tab_start, tab_num;
2040 uint16_t num = FROM_LE16(lang_pack->offsets[i]);
2043 tab_start[i] = count;
2049 std::vector<std::string_view> strings;
2052 char *s = lang_pack->data;
2053 for (uint i = 0; i < count; i++) {
2054 size_t len =
static_cast<uint8_t
>(*s++);
2055 if (s + len >= end)
return false;
2058 len = ((len & 0x3F) << 8) +
static_cast<uint8_t
>(*s++);
2059 if (s + len >= end)
return false;
2061 strings.emplace_back(s, len);
2064 assert(strings.size() == count);
2066 _langpack.langpack = std::move(lang_pack);
2067 _langpack.strings = std::move(strings);
2079 extern void Win32SetCurrentLocaleName(std::string iso_code);
2090 UErrorCode status = U_ZERO_ERROR;
2095 if (U_FAILURE(status)) {
2121#if !(defined(_WIN32) || defined(__APPLE__))
2132 auto env =
GetEnv(
"LANGUAGE");
2133 if (env.has_value())
return std::string{*env};
2136 if (env.has_value())
return std::string{*env};
2138 if (param !=
nullptr) {
2140 if (env.has_value())
return std::string{*env};
2144 if (env.has_value())
return std::string{*env};
2146 return std::nullopt;
2160 if (newgrflangid == lang.newgrflangid)
return ⟨
2175 if (!f.has_value())
return false;
2177 size_t read = fread(hdr,
sizeof(*hdr), 1, *f);
2179 bool ret = read == 1 && hdr->
IsValid();
2195 std::error_code error_code;
2196 for (
const auto &dir_entry : std::filesystem::directory_iterator(
OTTD2FS(path), error_code)) {
2197 if (!dir_entry.is_regular_file())
continue;
2198 if (dir_entry.path().extension() !=
".lng")
continue;
2201 lmd.
file = dir_entry.path();
2206 Debug(misc, 3,
"{} is not a valid language file", file);
2208 Debug(misc, 3,
"{}'s language ID is already known", file);
2214 Debug(misc, 9,
"Unable to open directory {}: {}", path, error_code.message());
2227 if (
_languages.empty()) UserError(
"No available language packs (invalid versions?)");
2231 std::string_view lang = str_lang.has_value() ? std::string_view{*str_lang} :
"en_GB";
2243 chosen_language = &lng;
2247 std::string_view iso_code = lng.
isocode;
2248 if (iso_code ==
"en_GB") en_GB_fallback = &lng;
2251 if (!lng.IsReasonablyFinished())
continue;
2253 if (iso_code.starts_with(lang.substr(0, 5))) chosen_language = &lng;
2254 if (iso_code.starts_with(lang.substr(0, 2))) language_fallback = &lng;
2259 if (chosen_language ==
nullptr) {
2260 chosen_language = (language_fallback !=
nullptr) ? language_fallback : en_GB_fallback;
2272 return _langpack.langpack->isocode;
2287 for (
char32_t c :
Utf8View(*text)) {
2288 if (c >= SCC_FIRST_FONT && c <= SCC_LAST_FONT) {
2289 size = (
FontSize)(c - SCC_FIRST_FONT);
2293 Debug(fontcache, 0,
"Font is missing glyphs to display char 0x{:X} in {} font size",
static_cast<uint32_t
>(c), FontSizeToName(size));
2321 std::string_view ret = _langpack.strings[_langpack.
langtab_start[this->
i] + this->
j];
2324 while (this->i < TEXT_TAB_END && this->
j >= _langpack.
langtab_num[this->i]) {
2339#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2344 settings->small.os_handle = os_data;
2345 settings->medium.os_handle = os_data;
2346 settings->large.os_handle = os_data;
2367 if (searcher ==
nullptr) searcher = &pack_searcher;
2369#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2373 bool any_font_configured = !_fcsettings.
medium.
font.empty();
2379 bad_font = !
SetFallbackFont(&_fcsettings, _langpack.langpack->isocode, searcher);
2381 _fcsettings = std::move(backup);
2383 if (!bad_font && any_font_configured) {
2387 std::string err_str;
2390 builder.
Put(
"The current font is missing some of the characters used in the texts for this language. Using system fallback font instead.");
2394 if (bad_font && base_font) {
2408 std::string err_str;
2411 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.");
2422#if !(defined(WITH_ICU_I18N) && defined(WITH_HARFBUZZ)) && !defined(WITH_UNISCRIBE) && !defined(WITH_COCOA)
2433 std::string err_str;
2436 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_NORMAL
Index of the normal 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.
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: