44#include "table/strings.h"
46#include "3rdparty/fmt/std.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) {
246static void StationGetSpecialString(
StringBuilder &builder, StationFacilities x);
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);
275 delete[]
reinterpret_cast<char *
>(langpack);
280 std::unique_ptr<LanguagePack, LanguagePackDeleter> langpack;
282 std::vector<std::string_view> strings;
301 return _langpack.list_separator;
310 return _langpack.ellipsis;
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:
395 if (index >= _langpack.langtab_num[tab]) {
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()
471 std::string_view decimal_separator =
_settings_game.locale.digit_decimal_separator;
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)
512 std::string_view separator =
_settings_game.locale.digit_group_separator;
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));
565static void FormatYmdString(
StringBuilder &builder, TimerGameCalendar::Date date, uint case_index)
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);
573static void FormatMonthAndYear(
StringBuilder &builder, TimerGameCalendar::Date date, uint 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;
632 std::string_view separator =
_settings_game.locale.digit_group_separator_currency;
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);
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);
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 },
919 { { .02 }, STR_UNITS_HEIGHT_GAMEUNITS,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 },
1018 std::vector<StringParameter> sub_args;
1023 builder +=
"(invalid SCC_ENCODED)";
1028 builder +=
"(invalid StringID)";
1038 sub_args.emplace_back(std::monostate{});
1043 char32_t parameter_type = record.
ReadUtf8();
1044 switch (parameter_type) {
1048 builder +=
"(invalid sub-StringID)";
1053 sub_args.emplace_back(param);
1060 sub_args.emplace_back(param);
1071 sub_args.emplace_back(std::monostate{});
1088 for (
char32_t ch :
Utf8View(buffer)) {
1089 if (ch == SCC_PUSH_COLOUR) {
1091 }
else if (ch == SCC_POP_COLOUR) {
1093 if (safety < 0)
return false;
1094 }
else if ((ch >= SCC_BLUE && ch <= SCC_BLACK) || ch == SCC_COLOUR) {
1095 if (safety == 0)
return false;
1112 size_t orig_first_param_offset = args.
GetOffset();
1113 bool emit_automatic_push_pop =
false;
1126 FormatString(dry_run_builder, str_arg, args, orig_case_index, game_script,
true);
1129 args.
SetOffset(orig_first_param_offset);
1131 uint next_substr_case_index = 0;
1132 struct StrStackItem {
1134 size_t first_param_offset;
1137 StrStackItem(std::string_view view,
size_t first_param_offset, uint case_index)
1138 : consumer(view), first_param_offset(first_param_offset), case_index(case_index)
1141 std::stack<StrStackItem, std::vector<StrStackItem>> str_stack;
1142 str_stack.emplace(str_arg, orig_first_param_offset, orig_case_index);
1144 if (emit_automatic_push_pop) builder.
PutUtf8(SCC_PUSH_COLOUR);
1148 while (!str_stack.empty() && !str_stack.top().consumer.AnyBytesLeft()) {
1151 if (str_stack.empty())
break;
1153 const size_t ref_param_offset = str_stack.top().first_param_offset;
1154 const uint case_index = str_stack.top().case_index;
1159 builder +=
"(unexpected NUL)";
1166 if (b == 0)
continue;
1169 if (b < SCC_CONTROL_START || b > SCC_CONTROL_END) {
1174 args.SetTypeOfNextParameter(b);
1183 std::string_view ptr = GetStringPtr(substr);
1184 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1185 next_substr_case_index = 0;
1191 std::string_view ptr = GetStringPtr(substr);
1192 str_stack.emplace(ptr, args.
GetOffset(), next_substr_case_index);
1193 next_substr_case_index = 0;
1197 case SCC_GENDER_LIST: {
1199 size_t offset = ref_param_offset + consumer.
ReadUint8();
1203 builder +=
"(invalid GENDER parameter)";
1225 if (gender_consumer.
ReadUtf8If(SCC_GENDER_INDEX)) {
1229 ParseStringChoice(consumer, gender, builder);
1235 case SCC_GENDER_INDEX: {
1238 builder.
PutUtf8(SCC_GENDER_INDEX);
1244 case SCC_PLURAL_LIST: {
1245 uint8_t plural_form = consumer.
ReadUint8();
1246 size_t offset = ref_param_offset + consumer.
ReadUint8();
1247 const uint64_t *v =
nullptr;
1250 v = std::get_if<uint64_t>(&args.GetParam(offset));
1253 ParseStringChoice(consumer,
DeterminePluralForm(
static_cast<int64_t
>(*v), plural_form), builder);
1255 SkipStringChoice(consumer);
1256 builder +=
"(invalid PLURAL parameter)";
1261 case SCC_ARG_INDEX: {
1266 case SCC_SET_CASE: {
1269 next_substr_case_index = consumer.
ReadUint8();
1273 case SCC_SWITCH_CASE: {
1277 std::optional<std::string_view> found;
1278 for (; num > 0; --num) {
1281 auto case_str = consumer.
Read(len);
1282 if (index == case_index) {
1288 auto default_str = consumer.
Read(default_len);
1289 if (!found.has_value()) found = default_str;
1290 str_stack.emplace(*found, ref_param_offset, case_index);
1295 builder += _openttd_revision;
1298 case SCC_RAW_STRING_POINTER:
1307 GetStringWithArgs(builder, string_id, tmp_params, next_substr_case_index, game_script);
1308 next_substr_case_index = 0;
1322 uint size = b - SCC_STRING1 + 1;
1324 builder +=
"(consumed too many parameters)";
1327 GetStringWithArgs(builder, string_id, sub_args, next_substr_case_index, game_script);
1330 next_substr_case_index = 0;
1342 FormatCommaNumber(builder, number);
1347 int64_t fractional = number % divisor;
1349 FormatCommaNumber(builder, number);
1350 fmt::format_to(builder.
back_inserter(),
"{}{:0{}d}", GetDecimalSeparator(), fractional, digits);
1358 case SCC_ZEROFILL_NUM: {
1372 case SCC_CARGO_TINY: {
1380 builder +=
"(invalid cargo type)";
1397 FormatCommaNumber(builder, amount);
1401 case SCC_CARGO_SHORT: {
1409 builder +=
"(invalid cargo type)";
1414 switch (cargo_str) {
1418 auto tmp_params =
MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1426 auto tmp_params =
MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1440 case SCC_CARGO_LONG: {
1450 builder +=
"(invalid cargo type)";
1455 case SCC_CARGO_LIST: {
1461 if (!
HasBit(cmask, cs->Index()))
continue;
1467 builder += list_separator;
1470 GetStringWithArgs(builder, cs->name, args, next_substr_case_index, game_script);
1474 if (first)
GetStringWithArgs(builder, STR_JUST_NOTHING, args, next_substr_case_index, game_script);
1476 next_substr_case_index = 0;
1480 case SCC_CURRENCY_SHORT:
1484 case SCC_CURRENCY_LONG:
1489 FormatTinyOrISODate(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), STR_FORMAT_DATE_TINY);
1492 case SCC_DATE_SHORT:
1493 FormatMonthAndYear(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), next_substr_case_index);
1494 next_substr_case_index = 0;
1498 FormatYmdString(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), next_substr_case_index);
1499 next_substr_case_index = 0;
1503 FormatTinyOrISODate(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), STR_FORMAT_DATE_ISO);
1530 case SCC_POWER_TO_WEIGHT: {
1539 case SCC_VELOCITY: {
1549 case SCC_VOLUME_SHORT: {
1557 case SCC_VOLUME_LONG: {
1565 case SCC_WEIGHT_SHORT: {
1573 case SCC_WEIGHT_LONG: {
1581 case SCC_UNITS_DAYS_OR_SECONDS: {
1589 case SCC_UNITS_MONTHS_OR_MINUTES: {
1597 case SCC_UNITS_YEARS_OR_PERIODS: {
1605 case SCC_UNITS_YEARS_OR_MINUTES: {
1613 case SCC_COMPANY_NAME: {
1615 if (c ==
nullptr)
break;
1617 if (!c->
name.empty()) {
1627 case SCC_COMPANY_NUM: {
1638 case SCC_DEPOT_NAME: {
1647 if (!d->name.empty()) {
1657 case SCC_ENGINE_NAME: {
1660 if (e ==
nullptr)
break;
1662 if (!e->name.empty() && e->IsEnabled()) {
1669 std::array<int32_t, 16> regs100;
1672 if (callback == 0x40F) {
1673 const GRFFile *grffile = e->GetGRF();
1674 assert(grffile !=
nullptr);
1678 }
else if (callback < 0x400) {
1679 const GRFFile *grffile = e->GetGRF();
1680 assert(grffile !=
nullptr);
1691 case SCC_GROUP_NAME: {
1693 if (g ==
nullptr)
break;
1695 if (!g->
name.empty()) {
1705 case SCC_INDUSTRY_NAME: {
1707 if (i ==
nullptr)
break;
1709 static bool use_cache =
true;
1714 }
else if (use_cache) {
1716 builder += i->GetCachedName();
1720 FormatString(builder, GetStringPtr(STR_FORMAT_INDUSTRY_NAME), tmp_params, next_substr_case_index);
1722 next_substr_case_index = 0;
1726 case SCC_PRESIDENT_NAME: {
1728 if (c ==
nullptr)
break;
1740 case SCC_STATION_NAME: {
1744 if (st ==
nullptr) {
1752 static bool use_cache =
true;
1755 builder += st->GetCachedName();
1756 }
else if (!st->name.empty()) {
1760 StringID string_id = st->string_id;
1761 if (st->indtype != IT_INVALID) {
1773 auto tmp_params =
MakeParameters(STR_TOWN_NAME, st->town->index, st->index);
1779 case SCC_TOWN_NAME: {
1781 if (t ==
nullptr)
break;
1783 static bool use_cache =
true;
1786 builder += t->GetCachedName();
1787 }
else if (!t->
name.empty()) {
1791 GetTownName(builder, t);
1796 case SCC_WAYPOINT_NAME: {
1798 if (wp ==
nullptr)
break;
1800 if (!wp->
name.empty()) {
1805 StringID string_id = ((wp->
string_id == STR_SV_STNAME_BUOY) ? STR_FORMAT_BUOY_NAME : STR_FORMAT_WAYPOINT_NAME);
1806 if (wp->
town_cn != 0) string_id++;
1812 case SCC_VEHICLE_NAME: {
1814 if (v ==
nullptr)
break;
1816 if (!v->
name.empty()) {
1828 default: string_id = STR_INVALID_VEHICLE;
break;
1829 case VEH_TRAIN: string_id = STR_SV_TRAIN_NAME;
break;
1830 case VEH_ROAD: string_id = STR_SV_ROAD_VEHICLE_NAME;
break;
1831 case VEH_SHIP: string_id = STR_SV_SHIP_NAME;
break;
1832 case VEH_AIRCRAFT: string_id = STR_SV_AIRCRAFT_NAME;
break;
1840 case SCC_SIGN_NAME: {
1842 if (si ==
nullptr)
break;
1844 if (!si->name.empty()) {
1853 case SCC_STATION_FEATURES: {
1854 StationGetSpecialString(builder, args.
GetNextParameter<StationFacilities>());
1868 }
catch (std::out_of_range &e) {
1869 Debug(misc, 0,
"FormatString: {}", e.what());
1870 builder +=
"(invalid parameter)";
1874 if (emit_automatic_push_pop) builder.
PutUtf8(SCC_POP_COLOUR);
1878static void StationGetSpecialString(
StringBuilder &builder, StationFacilities x)
1887static const std::string_view _silly_company_names[] = {
1889 "Tiny Transport Ltd.",
1891 "Comfy-Coach & Co.",
1892 "Crush & Bump Ltd.",
1893 "Broken & Late Ltd.",
1895 "Supersonic Travel",
1897 "Lightning International",
1898 "Pannik & Loozit Ltd.",
1899 "Inter-City Transport",
1900 "Getout & Pushit Ltd."
1903static const std::string_view _surname_list[] = {
1935static const std::string_view _silly_surname_list[] = {
1950static const char _initial_name_letters[] = {
1951 'A',
'B',
'C',
'D',
'E',
'F',
'G',
'H',
'I',
'J',
1952 'K',
'L',
'M',
'N',
'P',
'R',
'S',
'T',
'W',
1955static std::span<const std::string_view> GetSurnameOptions()
1958 return _surname_list;
1968 auto surname_options = GetSurnameOptions();
1969 return surname_options[surname_options.size() *
GB(seed, 16, 8) >> 8];
1972static void GenAndCoName(
StringBuilder &builder, uint32_t seed)
1975 builder +=
" & Co.";
1978static void GenPresidentName(
StringBuilder &builder, uint32_t seed)
1980 builder.
PutChar(_initial_name_letters[std::size(_initial_name_letters) *
GB(seed, 0, 8) >> 8]);
1984 size_t index = (std::size(_initial_name_letters) + 35) *
GB(seed, 8, 8) >> 8;
1985 if (index < std::size(_initial_name_letters)) {
1986 builder.
PutChar(_initial_name_letters[index]);
1997 builder += _silly_company_names[std::min<size_t>(args.
GetNextParameter<uint16_t>(), std::size(_silly_company_names) - 1)];
2012 builder +=
" Transport";
2026 this->
version == TO_LE32(LANGUAGE_PACK_VERSION) &&
2046 return 4 * this->
missing < LANGUAGE_TOTAL_STRINGS;
2057 size_t total_len = 0;
2059 if (!lang_pack)
return false;
2062 const char *end = (
char *)lang_pack.get() + total_len + 1;
2065 if (end <= lang_pack->data || !lang_pack->IsValid()) {
2069 std::array<uint, TEXT_TAB_END> tab_start, tab_num;
2073 uint16_t num = FROM_LE16(lang_pack->offsets[i]);
2076 tab_start[i] = count;
2082 std::vector<std::string_view> strings;
2085 char *s = lang_pack->data;
2086 for (uint i = 0; i < count; i++) {
2087 size_t len =
static_cast<uint8_t
>(*s++);
2088 if (s + len >= end)
return false;
2091 len = ((len & 0x3F) << 8) +
static_cast<uint8_t
>(*s++);
2092 if (s + len >= end)
return false;
2094 strings.emplace_back(s, len);
2097 assert(strings.size() == count);
2099 _langpack.langpack = std::move(lang_pack);
2100 _langpack.strings = std::move(strings);
2101 _langpack.langtab_num = tab_num;
2102 _langpack.langtab_start = tab_start;
2108 _langpack.list_separator =
GetString(STR_LIST_SEPARATOR);
2109 _langpack.ellipsis =
GetString(STR_TRUNCATION_ELLIPSIS);
2112 extern void Win32SetCurrentLocaleName(std::string iso_code);
2123 UErrorCode status = U_ZERO_ERROR;
2128 if (U_FAILURE(status)) {
2154#if !(defined(_WIN32) || defined(__APPLE__))
2165 auto env =
GetEnv(
"LANGUAGE");
2166 if (env.has_value())
return std::string{*env};
2169 if (env.has_value())
return std::string{*env};
2171 if (param !=
nullptr) {
2173 if (env.has_value())
return std::string{*env};
2177 if (env.has_value())
return std::string{*env};
2179 return std::nullopt;
2193 if (newgrflangid == lang.newgrflangid)
return ⟨
2208 if (!f.has_value())
return false;
2210 size_t read = fread(hdr,
sizeof(*hdr), 1, *f);
2212 bool ret = read == 1 && hdr->
IsValid();
2228 std::error_code error_code;
2229 for (
const auto &dir_entry : std::filesystem::directory_iterator(
OTTD2FS(path), error_code)) {
2230 if (!dir_entry.is_regular_file())
continue;
2231 if (dir_entry.path().extension() !=
".lng")
continue;
2234 lmd.
file = dir_entry.path();
2239 Debug(misc, 3,
"{} is not a valid language file", file);
2241 Debug(misc, 3,
"{}'s language ID is already known", file);
2247 Debug(misc, 9,
"Unable to open directory {}: {}", path, error_code.message());
2260 if (
_languages.empty()) UserError(
"No available language packs (invalid versions?)");
2264 std::string_view lang = str_lang.has_value() ? std::string_view{*str_lang} :
"en_GB";
2276 chosen_language = &lng;
2280 std::string_view iso_code = lng.isocode;
2281 if (iso_code ==
"en_GB") en_GB_fallback = &lng;
2284 if (!lng.IsReasonablyFinished())
continue;
2286 if (iso_code.starts_with(lang.substr(0, 5))) chosen_language = &lng;
2287 if (iso_code.starts_with(lang.substr(0, 2))) language_fallback = &lng;
2292 if (chosen_language ==
nullptr) {
2293 chosen_language = (language_fallback !=
nullptr) ? language_fallback : en_GB_fallback;
2305 return _langpack.langpack->isocode;
2320 for (
char32_t c :
Utf8View(*text)) {
2321 if (c >= SCC_FIRST_FONT && c <= SCC_LAST_FONT) {
2322 size = (
FontSize)(c - SCC_FIRST_FONT);
2326 Debug(fontcache, 0,
"Font is missing glyphs to display char 0x{:X} in {} font size",
static_cast<uint32_t
>(c), FontSizeToName(size));
2354 std::string_view ret = _langpack.strings[_langpack.langtab_start[this->i] + this->j];
2372#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2377 settings->small.os_handle = os_data;
2378 settings->medium.os_handle = os_data;
2379 settings->large.os_handle = os_data;
2399 if (searcher ==
nullptr) searcher = &pack_searcher;
2401#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2405 bool any_font_configured = !_fcsettings.medium.font.empty();
2409 _fcsettings.medium.os_handle =
nullptr;
2413 _fcsettings = std::move(backup);
2415 if (!bad_font && any_font_configured) {
2419 std::string err_str;
2422 builder.
Put(
"The current font is missing some of the characters used in the texts for this language. Using system fallback font instead.");
2440 std::string err_str;
2443 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.");
2454#if !(defined(WITH_ICU_I18N) && defined(WITH_HARFBUZZ)) && !defined(WITH_UNISCRIBE) && !defined(WITH_COCOA)
2465 std::string err_str;
2468 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.
static constexpr uint GB(const T x, const uint8_t s, const uint8_t n)
Fetch n bits from x, started at bit s.
constexpr bool HasBit(const T x, const uint8_t y)
Checks if a bit in a value is set.
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).
PoolID< uint16_t, struct DepotIDTag, 64000, 0xFFFF > DepotID
Type for the unique identifier of depots.
Function to handling different endian machines.
PoolID< uint16_t, struct EngineIDTag, 64000, 0xFFFF > EngineID
Unique identification number of an engine.
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.
@ Toyland
Landscape with funky industries and vehicles.
Information about languages and their files.
LanguageList _languages
The actual list of language meta data.
const LanguageMetadata * _current_language
The currently loaded language.
static const uint8_t MAX_NUM_GENDERS
Maximum number of supported genders.
std::unique_ptr< icu::Collator > _current_collator
Collator for the language currently in use.
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.
StrongType::Typedef< uint32_t, struct GRFStringIDTag, StrongType::Compare, StrongType::Integer > GRFStringID
Type for GRF-internal string IDs.
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.
PoolID< uint16_t, struct SignIDTag, 64000, 0xFFFF > SignID
The type of the IDs of signs.
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.
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.
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.
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.
void CheckForMissingGlyphs(MissingGlyphSearcher *searcher)
Check whether the currently loaded language pack uses characters that the currently loaded font does ...
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.
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 bool IsColourSafe(std::string_view buffer)
Test if a string contains colour codes, and is not wrapped by push/pop codes.
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.
Types and functions related to the internal workings of formatting OpenTTD's strings.
void GenerateTownNameString(StringBuilder &builder, size_t lang, uint32_t seed)
Generates town name from given seed.
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.
const void * os_handle
Optional native OS font info. Only valid during font search.
Dynamic data of a loaded NewGRF.
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.
static Depot * Get(auto index)
static Company * GetIfValid(auto index)
static Station * GetIfValid(auto index)
The data required to format and validate a single parameter of a string.
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.
PoolID< uint32_t, struct VehicleIDTag, 0xFF000, 0xFFFFF > VehicleID
The type all our vehicle IDs have.
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: