44#include "table/strings.h"
46#include "3rdparty/fmt/std.h"
48#include "strings_internal.h"
72 throw std::out_of_range(
"Trying to read invalid string parameter");
76 if (param.type != 0 && param.type != this->next_type) {
78 throw std::out_of_range(
"Trying to read string parameter with wrong type");
105 auto output = std::back_inserter(result);
107 fmt::format_to(output,
"{:X}", str);
110 std::back_insert_iterator<std::string> &output;
112 void operator()(
const std::monostate &) {}
114 void operator()(
const uint64_t &arg)
117 fmt::format_to(this->output,
"{:X}", arg);
120 void operator()(
const std::string &value)
124 if (!value.empty()) {
126 const char *p = value.data();
133 fmt::format_to(this->output,
"{}", value);
138 for (
const auto ¶m : params) {
139 *output = SCC_RECORD_SEPARATOR;
140 std::visit(v, param.data);
155 if (this->empty())
return {};
157 std::vector<StringParameter> params;
160 const char *p = this->
string.data();
161 const char *e = this->
string.data() + this->
string.length();
163 char32_t c = Utf8Consume(p);
167 auto result = std::from_chars(p, e, str, 16);
168 if (result.ec != std::errc())
return {};
169 if (result.ptr != e && *result.ptr != SCC_RECORD_SEPARATOR)
return {};
176 for (; p != e && *p != SCC_RECORD_SEPARATOR; ++p) {}
180 params.emplace_back(std::monostate{});
185 char32_t parameter_type;
189 switch (parameter_type) {
192 result = std::from_chars(s, p, value, 16);
193 if (result.ec != std::errc() || result.ptr != p)
return {};
194 params.emplace_back(value);
199 params.emplace_back(std::string(s, p));
205 params.emplace_back(std::monostate{});
210 if (param >= std::size(params))
return {};
211 params[param] = data;
221 return GetString(STR_JUST_RAW_STRING, this->
string);
233 uint64_t val = count > 1 ? front : next;
234 for (; count > 1; count--) {
235 val = 10 * val + next;
251 while (max_value >= 10) {
273static void FormatString(
StringBuilder &builder,
const char *str, std::span<StringParameter> params, uint case_index = 0,
bool game_script =
false,
bool dry_run =
false)
276 FormatString(builder, str, tmp_params, case_index, game_script, dry_run);
287 delete[]
reinterpret_cast<char*
>(langpack);
292 std::unique_ptr<LanguagePack, LanguagePackDeleter> langpack;
294 std::vector<char *> offsets;
315const char *GetStringPtr(
StringID string)
320 case TEXT_TAB_OLD_NEWGRF: NOT_REACHED();
324 if (offset < _langpack.offsets.size())
return _langpack.offsets[offset];
353 }
catch (
const std::runtime_error &e) {
354 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
355 builder +=
"(invalid string parameter)";
361 case TEXT_TAB_SPECIAL:
364 if (GetSpecialNameString(builder,
string, args))
return;
365 }
catch (
const std::runtime_error &e) {
366 Debug(misc, 0,
"GetStringWithArgs: {}", e.what());
367 builder +=
"(invalid string parameter)";
373 case TEXT_TAB_OLD_CUSTOM:
376 FatalError(
"Incorrect conversion of custom name string.");
385 case TEXT_TAB_OLD_NEWGRF:
401 FatalError(
"String 0x{:X} is invalid. You are probably using an old version of the .lng file.\n",
string);
404 FormatString(builder, GetStringPtr(
string), args, case_index);
442void AppendStringWithArgsInPlace(std::string &result,
StringID string, std::span<StringParameter> params)
471static const char *GetDecimalSeparator()
474 if (
StrEmpty(decimal_separator)) decimal_separator = _langpack.langpack->digit_decimal_separator;
475 return decimal_separator;
486 static const int max_digits = 20;
487 uint64_t divisor = 10000000000000000000ULL;
488 int thousands_offset = (max_digits - 1) % 3;
495 uint64_t num = number;
497 for (
int i = 0; i < max_digits; i++) {
499 if (num >= divisor) {
500 quot = num / divisor;
503 if ((tot |= quot) || i == max_digits - 1) {
504 builder +=
'0' + quot;
505 if ((i % 3) == thousands_offset && i < max_digits - 1) builder += separator;
512static void FormatCommaNumber(
StringBuilder &builder, int64_t number)
515 if (
StrEmpty(separator)) separator = _langpack.langpack->digit_group_separator;
519static void FormatNoCommaNumber(
StringBuilder &builder, int64_t number)
521 fmt::format_to(builder,
"{}", number);
524static void FormatZerofillNumber(
StringBuilder &builder, int64_t number,
int count)
526 fmt::format_to(builder,
"{:0{}d}", number, count);
529static void FormatHexNumber(
StringBuilder &builder, uint64_t number)
531 fmt::format_to(builder,
"0x{:X}", number);
544 const char *
const iec_prefixes[] = {
"",
"Ki",
"Mi",
"Gi",
"Ti",
"Pi",
"Ei"};
546 while (number >= 1024 * 1024) {
553 fmt::format_to(builder,
"{}", number);
554 }
else if (number < 1024 * 10) {
555 fmt::format_to(builder,
"{}{}{:02}", number / 1024, GetDecimalSeparator(), (number % 1024) * 100 / 1024);
556 }
else if (number < 1024 * 100) {
557 fmt::format_to(builder,
"{}{}{:01}", number / 1024, GetDecimalSeparator(), (number % 1024) * 10 / 1024);
559 assert(number < 1024 * 1024);
560 fmt::format_to(builder,
"{}", number / 1024);
563 assert(
id <
lengthof(iec_prefixes));
564 fmt::format_to(builder,
NBSP "{}B", iec_prefixes[
id]);
567static void FormatYmdString(
StringBuilder &builder, TimerGameCalendar::Date date, uint case_index)
571 auto tmp_params =
MakeParameters(STR_DAY_NUMBER_1ST + ymd.day - 1, STR_MONTH_ABBREV_JAN + ymd.month, ymd.year);
572 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_LONG), tmp_params, case_index);
575static void FormatMonthAndYear(
StringBuilder &builder, TimerGameCalendar::Date date, uint case_index)
579 auto tmp_params =
MakeParameters(STR_MONTH_JAN + ymd.month, ymd.year);
580 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_SHORT), tmp_params, case_index);
588 auto tmp_params =
MakeParameters(ymd.day, 2, ymd.month + 1, 2, ymd.year);
596 bool negative = number < 0;
598 number *= spec->
rate;
619 if (number >=
Money(1'000'000'000'000'000) - 500'000'000) {
620 number = (number +
Money(500'000'000'000)) /
Money(1'000'000'000'000);
621 number_str = STR_CURRENCY_SHORT_TERA;
622 }
else if (number >=
Money(1'000'000'000'000) - 500'000) {
623 number = (number + 500'000'000) / 1'000'000'000;
624 number_str = STR_CURRENCY_SHORT_GIGA;
625 }
else if (number >= 1'000'000'000 - 500) {
626 number = (number + 500'000) / 1'000'000;
627 number_str = STR_CURRENCY_SHORT_MEGA;
628 }
else if (number >= 1'000'000) {
629 number = (number + 500) / 1'000;
630 number_str = STR_CURRENCY_SHORT_KILO;
636 if (
StrEmpty(separator)) separator = _langpack.langpack->digit_group_separator_currency;
638 if (number_str != STR_NULL) {
661 uint64_t n =
abs(count);
663 switch (plural_form) {
672 return n != 1 ? 1 : 0;
684 return n > 1 ? 1 : 0;
691 return n % 10 == 1 && n % 100 != 11 ? 0 : n != 0 ? 1 : 2;
697 return n == 1 ? 0 : n == 2 ? 1 : n < 7 ? 2 : n < 11 ? 3 : 4;
703 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
709 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
715 return n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
721 return n % 100 == 1 ? 0 : n % 100 == 2 ? 1 : n % 100 == 3 || n % 100 == 4 ? 2 : 3;
727 return n % 10 == 1 && n % 100 != 11 ? 0 : 1;
733 return n == 1 ? 0 : n >= 2 && n <= 4 ? 1 : 2;
764 return (n == 1 ? 0 : n == 0 || (n % 100 > 1 && n % 100 < 11) ? 1 : (n % 100 > 10 && n % 100 < 20) ? 2 : 3);
769 return ((n == 1 || n == 11) ? 0 : (n == 2 || n == 12) ? 1 : ((n > 2 && n < 11) || (n > 12 && n < 20)) ? 2 : 3);
775 return n == 1 ? 0 : (n == 0 || (n % 100 > 0 && n % 100 < 20)) ? 1 : 2;
779static const char *ParseStringChoice(
const char *b, uint form,
StringBuilder &builder)
782 uint n = (uint8_t)*b++;
783 uint pos, i, mypos = 0;
785 for (i = pos = 0; i != n; i++) {
786 uint len = (uint8_t)*b++;
787 if (i == form) mypos = pos;
791 builder += b + mypos;
805 int64_t
ToDisplay(int64_t input,
bool round =
true)
const
808 ? (int64_t)std::round(input * this->factor)
809 : (int64_t)(input * this->factor);
819 int64_t
FromDisplay(int64_t input,
bool round =
true, int64_t divider = 1)
const
822 ? (int64_t)std::round(input / this->factor / divider)
823 : (int64_t)(input / this->factor / divider);
844 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
845 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
846 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
847 { { 0.578125 }, STR_UNITS_VELOCITY_GAMEUNITS_DAY, 1 },
848 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
853 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
854 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
855 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
856 { { 0.289352 }, STR_UNITS_VELOCITY_GAMEUNITS_SEC, 1 },
857 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
862 { { 1.0 }, STR_UNITS_POWER_IMPERIAL, 0 },
863 { { 1.01387 }, STR_UNITS_POWER_METRIC, 0 },
864 { { 0.745699 }, STR_UNITS_POWER_SI, 0 },
869 { { 0.907185 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_IMPERIAL, 1 },
870 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_METRIC, 1 },
871 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_SI, 1 },
872 { { 0.919768 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_IMPERIAL, 1 },
873 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_METRIC, 1 },
874 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_SI, 1 },
875 { { 0.676487 }, STR_UNITS_POWER_SI_TO_WEIGHT_IMPERIAL, 1 },
876 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_METRIC, 1 },
877 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_SI, 1 },
882 { { 1.102311 }, STR_UNITS_WEIGHT_SHORT_IMPERIAL, STR_UNITS_WEIGHT_LONG_IMPERIAL, 0 },
883 { { 1.0 }, STR_UNITS_WEIGHT_SHORT_METRIC, STR_UNITS_WEIGHT_LONG_METRIC, 0 },
884 { { 1000.0 }, STR_UNITS_WEIGHT_SHORT_SI, STR_UNITS_WEIGHT_LONG_SI, 0 },
889 { { 264.172 }, STR_UNITS_VOLUME_SHORT_IMPERIAL, STR_UNITS_VOLUME_LONG_IMPERIAL, 0 },
890 { { 1000.0 }, STR_UNITS_VOLUME_SHORT_METRIC, STR_UNITS_VOLUME_LONG_METRIC, 0 },
891 { { 1.0 }, STR_UNITS_VOLUME_SHORT_SI, STR_UNITS_VOLUME_LONG_SI, 0 },
896 { { 0.224809 }, STR_UNITS_FORCE_IMPERIAL, 0 },
897 { { 0.101972 }, STR_UNITS_FORCE_METRIC, 0 },
898 { { 0.001 }, STR_UNITS_FORCE_SI, 0 },
903 { { 3.0 }, STR_UNITS_HEIGHT_IMPERIAL, 0 },
904 { { 1.0 }, STR_UNITS_HEIGHT_METRIC, 0 },
905 { { 1.0 }, STR_UNITS_HEIGHT_SI, 0 },
910 { { 1 }, STR_UNITS_DAYS, 0 },
911 { { 2 }, STR_UNITS_SECONDS, 0 },
916 { { 1 }, STR_UNITS_MONTHS, 0 },
917 { { 1 }, STR_UNITS_MINUTES, 0 },
922 { { 1 }, STR_UNITS_YEARS, 0 },
923 { { 1 }, STR_UNITS_PERIODS, 0 },
928 { { 1 }, STR_UNITS_YEARS, 0 },
929 { { 12 }, STR_UNITS_MINUTES, 0 },
1001 std::vector<StringParameter> sub_args;
1005 if (*p != SCC_RECORD_SEPARATOR && *p !=
'\0') {
1006 while (*p !=
'\0') p++;
1007 builder +=
"(invalid SCC_ENCODED)";
1011 while (*p !=
'\0') p++;
1012 builder +=
"(invalid StringID)";
1016 while (*p !=
'\0') {
1018 const char *s = ++p;
1021 for (; *p !=
'\0' && *p != SCC_RECORD_SEPARATOR; ++p) {}
1025 sub_args.emplace_back(std::monostate{});
1030 char32_t parameter_type;
1034 switch (parameter_type) {
1036 uint64_t param = std::strtoull(s, &p, 16);
1038 while (*p !=
'\0') p++;
1039 builder +=
"(invalid sub-StringID)";
1043 sub_args.emplace_back(param);
1048 uint64_t param = std::strtoull(s, &p, 16);
1049 sub_args.emplace_back(param);
1054 sub_args.emplace_back(std::string(s, p - s));
1060 sub_args.emplace_back(std::monostate{});
1093 FormatString(dry_run_builder, str_arg, args, case_index, game_script,
true);
1098 uint next_substr_case_index = 0;
1099 std::stack<const char *, std::vector<const char *>> str_stack;
1100 str_stack.push(str_arg);
1104 while (!str_stack.empty() && (b = Utf8Consume(&str_stack.top())) ==
'\0') {
1107 if (str_stack.empty())
break;
1108 const char *&str = str_stack.top();
1113 if (b == 0)
continue;
1116 if (b < SCC_CONTROL_START || b > SCC_CONTROL_END) {
1121 args.SetTypeOfNextParameter(b);
1129 StringID substr = Utf8Consume(&str);
1130 const char *ptr = GetStringPtr(substr);
1131 if (ptr ==
nullptr) {
1132 builder +=
"(invalid NewGRF string)";
1134 str_stack.push(ptr);
1141 const char *ptr = GetStringPtr(substr);
1142 if (ptr ==
nullptr) {
1143 builder +=
"(invalid NewGRF string)";
1145 str_stack.push(ptr);
1147 case_index = next_substr_case_index;
1148 next_substr_case_index = 0;
1153 case SCC_GENDER_LIST: {
1155 size_t offset = orig_offset + (uint8_t)*str++;
1175 const char *s = buffer.c_str();
1176 char32_t c = Utf8Consume(&s);
1178 if (c == SCC_GENDER_INDEX) gender = (uint8_t)s[0];
1180 str = ParseStringChoice(str, gender, builder);
1186 case SCC_GENDER_INDEX:
1195 case SCC_PLURAL_LIST: {
1196 int plural_form = *str++;
1197 size_t offset = orig_offset + (uint8_t)*str++;
1198 const uint64_t *v = std::get_if<uint64_t>(&args.GetParam(offset));
1200 str = ParseStringChoice(str,
DeterminePluralForm(
static_cast<int64_t
>(*v), plural_form), builder);
1202 builder +=
"(invalid PLURAL parameter)";
1207 case SCC_ARG_INDEX: {
1208 args.
SetOffset(orig_offset + (uint8_t)*str++);
1212 case SCC_SET_CASE: {
1215 next_substr_case_index = (uint8_t)*str++;
1219 case SCC_SWITCH_CASE: {
1222 uint num = (uint8_t)*str++;
1224 if ((uint8_t)str[0] == case_index) {
1230 str += 3 + (str[1] << 8) + str[2];
1237 builder += _openttd_revision;
1240 case SCC_RAW_STRING_POINTER: {
1243 if (raw_string ==
nullptr) {
1244 builder +=
"(invalid RAW_STRING parameter)";
1256 GetStringWithArgs(builder, string_id, tmp_params, next_substr_case_index, game_script);
1257 next_substr_case_index = 0;
1271 uint size = b - SCC_STRING1 + 1;
1273 builder +=
"(consumed too many parameters)";
1276 GetStringWithArgs(builder, string_id, sub_args, next_substr_case_index, game_script);
1279 next_substr_case_index = 0;
1291 FormatCommaNumber(builder, number);
1296 int64_t fractional = number % divisor;
1298 FormatCommaNumber(builder, number);
1299 fmt::format_to(builder,
"{}{:0{}d}", GetDecimalSeparator(), fractional, digits);
1307 case SCC_ZEROFILL_NUM: {
1321 case SCC_CARGO_TINY: {
1330 switch (cargo_str) {
1345 FormatCommaNumber(builder, amount);
1349 case SCC_CARGO_SHORT: {
1357 switch (cargo_str) {
1383 case SCC_CARGO_LONG: {
1394 case SCC_CARGO_LIST: {
1400 if (!
HasBit(cmask, cs->Index()))
continue;
1406 builder += list_separator;
1409 GetStringWithArgs(builder, cs->name, args, next_substr_case_index, game_script);
1413 if (first)
GetStringWithArgs(builder, STR_JUST_NOTHING, args, next_substr_case_index, game_script);
1415 next_substr_case_index = 0;
1419 case SCC_CURRENCY_SHORT:
1423 case SCC_CURRENCY_LONG:
1428 FormatTinyOrISODate(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), STR_FORMAT_DATE_TINY);
1431 case SCC_DATE_SHORT:
1432 FormatMonthAndYear(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), next_substr_case_index);
1433 next_substr_case_index = 0;
1437 FormatYmdString(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), next_substr_case_index);
1438 next_substr_case_index = 0;
1442 FormatTinyOrISODate(builder, args.
GetNextParameter<TimerGameCalendar::Date>(), STR_FORMAT_DATE_ISO);
1469 case SCC_POWER_TO_WEIGHT: {
1478 case SCC_VELOCITY: {
1488 case SCC_VOLUME_SHORT: {
1496 case SCC_VOLUME_LONG: {
1504 case SCC_WEIGHT_SHORT: {
1512 case SCC_WEIGHT_LONG: {
1520 case SCC_UNITS_DAYS_OR_SECONDS: {
1528 case SCC_UNITS_MONTHS_OR_MINUTES: {
1536 case SCC_UNITS_YEARS_OR_PERIODS: {
1544 case SCC_UNITS_YEARS_OR_MINUTES: {
1552 case SCC_COMPANY_NAME: {
1554 if (c ==
nullptr)
break;
1556 if (!c->
name.empty()) {
1566 case SCC_COMPANY_NUM: {
1577 case SCC_DEPOT_NAME: {
1586 if (!d->name.empty()) {
1596 case SCC_ENGINE_NAME: {
1599 if (e ==
nullptr)
break;
1601 if (!e->name.empty() && e->IsEnabled()) {
1611 const GRFFile *grffile = e->GetGRF();
1612 assert(grffile !=
nullptr);
1623 case SCC_GROUP_NAME: {
1625 if (g ==
nullptr)
break;
1627 if (!g->
name.empty()) {
1637 case SCC_INDUSTRY_NAME: {
1639 if (i ==
nullptr)
break;
1641 static bool use_cache =
true;
1646 }
else if (use_cache) {
1648 builder += i->GetCachedName();
1652 FormatString(builder, GetStringPtr(STR_FORMAT_INDUSTRY_NAME), tmp_params, next_substr_case_index);
1654 next_substr_case_index = 0;
1658 case SCC_PRESIDENT_NAME: {
1660 if (c ==
nullptr)
break;
1672 case SCC_STATION_NAME: {
1676 if (st ==
nullptr) {
1684 static bool use_cache =
true;
1687 builder += st->GetCachedName();
1688 }
else if (!st->name.empty()) {
1692 StringID string_id = st->string_id;
1693 if (st->indtype != IT_INVALID) {
1705 auto tmp_params =
MakeParameters(STR_TOWN_NAME, st->town->index, st->index);
1711 case SCC_TOWN_NAME: {
1713 if (t ==
nullptr)
break;
1715 static bool use_cache =
true;
1718 builder += t->GetCachedName();
1719 }
else if (!t->
name.empty()) {
1723 GetTownName(builder, t);
1728 case SCC_WAYPOINT_NAME: {
1730 if (wp ==
nullptr)
break;
1732 if (!wp->
name.empty()) {
1737 StringID string_id = ((wp->
string_id == STR_SV_STNAME_BUOY) ? STR_FORMAT_BUOY_NAME : STR_FORMAT_WAYPOINT_NAME);
1738 if (wp->
town_cn != 0) string_id++;
1744 case SCC_VEHICLE_NAME: {
1746 if (v ==
nullptr)
break;
1748 if (!v->
name.empty()) {
1760 default: string_id = STR_INVALID_VEHICLE;
break;
1761 case VEH_TRAIN: string_id = STR_SV_TRAIN_NAME;
break;
1762 case VEH_ROAD: string_id = STR_SV_ROAD_VEHICLE_NAME;
break;
1763 case VEH_SHIP: string_id = STR_SV_SHIP_NAME;
break;
1764 case VEH_AIRCRAFT: string_id = STR_SV_AIRCRAFT_NAME;
break;
1772 case SCC_SIGN_NAME: {
1774 if (si ==
nullptr)
break;
1776 if (!si->name.empty()) {
1785 case SCC_STATION_FEATURES: {
1800 }
catch (std::out_of_range &e) {
1801 Debug(misc, 0,
"FormatString: {}", e.what());
1802 builder +=
"(invalid parameter)";
1817static const char *
const _silly_company_names[] = {
1819 "Tiny Transport Ltd.",
1821 "Comfy-Coach & Co.",
1822 "Crush & Bump Ltd.",
1823 "Broken & Late Ltd.",
1825 "Supersonic Travel",
1827 "Lightning International",
1828 "Pannik & Loozit Ltd.",
1829 "Inter-City Transport",
1830 "Getout & Pushit Ltd."
1833static const char *
const _surname_list[] = {
1865static const char *
const _silly_surname_list[] = {
1880static const char _initial_name_letters[] = {
1881 'A',
'B',
'C',
'D',
'E',
'F',
'G',
'H',
'I',
'J',
1882 'K',
'L',
'M',
'N',
'P',
'R',
'S',
'T',
'W',
1885static std::span<const char * const> GetSurnameOptions()
1888 return _surname_list;
1898 auto surname_options = GetSurnameOptions();
1899 return surname_options[surname_options.size() *
GB(seed, 16, 8) >> 8];
1902static void GenAndCoName(
StringBuilder &builder, uint32_t seed)
1905 builder +=
" & Co.";
1908static void GenPresidentName(
StringBuilder &builder, uint32_t seed)
1910 builder += _initial_name_letters[std::size(_initial_name_letters) *
GB(seed, 0, 8) >> 8];
1914 size_t index = (std::size(_initial_name_letters) + 35) *
GB(seed, 8, 8) >> 8;
1915 if (index < std::size(_initial_name_letters)) {
1916 builder += _initial_name_letters[index];
1927 builder += _silly_company_names[std::min<size_t>(args.
GetNextParameter<uint16_t>(), std::size(_silly_company_names) - 1)];
1942 builder +=
" Transport";
1956 this->
version == TO_LE32(LANGUAGE_PACK_VERSION) &&
1976 return 4 * this->
missing < LANGUAGE_TOTAL_STRINGS;
1989 if (!lang_pack)
return false;
1992 const char *end = (
char *)lang_pack.get() + len + 1;
1995 if (end <= lang_pack->data || !lang_pack->IsValid()) {
1999 std::array<uint, TEXT_TAB_END> tab_start, tab_num;
2003 uint16_t num = FROM_LE16(lang_pack->offsets[i]);
2006 tab_start[i] = count;
2012 std::vector<char *> offs(count);
2015 char *s = lang_pack->data;
2016 len = (uint8_t)*s++;
2017 for (uint i = 0; i < count; i++) {
2018 if (s + len >= end)
return false;
2021 len = ((len & 0x3F) << 8) + (uint8_t)*s++;
2022 if (s + len >= end)
return false;
2030 _langpack.langpack = std::move(lang_pack);
2031 _langpack.offsets = std::move(offs);
2042 extern void Win32SetCurrentLocaleName(std::string iso_code);
2053 UErrorCode status = U_ZERO_ERROR;
2058 if (U_FAILURE(status)) {
2084#if !(defined(_WIN32) || defined(__APPLE__))
2097 env = std::getenv(
"LANGUAGE");
2098 if (env !=
nullptr)
return env;
2100 env = std::getenv(
"LC_ALL");
2101 if (env !=
nullptr)
return env;
2103 if (param !=
nullptr) {
2104 env = std::getenv(param);
2105 if (env !=
nullptr)
return env;
2108 return std::getenv(
"LANG");
2122 if (newgrflangid == lang.newgrflangid)
return ⟨
2137 if (!f.has_value())
return false;
2139 size_t read = fread(hdr,
sizeof(*hdr), 1, *f);
2141 bool ret = read == 1 && hdr->
IsValid();
2157 std::error_code error_code;
2158 for (
const auto &dir_entry : std::filesystem::directory_iterator(
OTTD2FS(path), error_code)) {
2159 if (!dir_entry.is_regular_file())
continue;
2160 if (dir_entry.path().extension() !=
".lng")
continue;
2163 lmd.
file = dir_entry.path();
2175 Debug(misc, 9,
"Unable to open directory {}: {}", path, error_code.message());
2188 if (
_languages.empty()) UserError(
"No available language packs (invalid versions?)");
2192 if (lang ==
nullptr) lang =
"en_GB";
2204 chosen_language = &lng;
2208 if (strcmp (lng.isocode,
"en_GB") == 0) en_GB_fallback = &lng;
2211 if (!lng.IsReasonablyFinished())
continue;
2213 if (strncmp(lng.isocode, lang, 5) == 0) chosen_language = &lng;
2214 if (strncmp(lng.isocode, lang, 2) == 0) language_fallback = &lng;
2219 if (chosen_language ==
nullptr) {
2220 chosen_language = (language_fallback !=
nullptr) ? language_fallback : en_GB_fallback;
2232 return _langpack.langpack->isocode;
2245 auto src = text->cbegin();
2249 while (src != text->cend()) {
2250 char32_t c = Utf8Consume(src);
2252 if (c >= SCC_FIRST_FONT && c <= SCC_LAST_FONT) {
2253 size = (
FontSize)(c - SCC_FIRST_FONT);
2257 std::string size_name;
2260 case FS_NORMAL: size_name =
"medium";
break;
2261 case FS_SMALL: size_name =
"small";
break;
2262 case FS_LARGE: size_name =
"large";
break;
2263 case FS_MONO: size_name =
"mono";
break;
2264 default: NOT_REACHED();
2267 Debug(fontcache, 0,
"Font is missing glyphs to display char 0x{:X} in {} font size", (
int)c, size_name);
2295 const char *ret = _langpack.offsets[_langpack.
langtab_start[this->
i] + this->
j];
2298 while (this->i < TEXT_TAB_END && this->
j >= _langpack.
langtab_num[this->i]) {
2313#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2318 settings->small.os_handle = os_data;
2319 settings->medium.os_handle = os_data;
2320 settings->large.os_handle = os_data;
2341 if (searcher ==
nullptr) searcher = &pack_searcher;
2343#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2347 bool any_font_configured = !_fcsettings.
medium.
font.empty();
2353 bad_font = !
SetFallbackFont(&_fcsettings, _langpack.langpack->isocode, searcher);
2355 _fcsettings = backup;
2357 if (!bad_font && any_font_configured) {
2364 static std::string err_str(
"XXXThe current font is missing some of the characters used in the texts for this language. Using system fallback font instead.");
2369 if (bad_font && base_font) {
2384 static std::string err_str(
"XXXThe 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.");
2396#if !(defined(WITH_ICU_I18N) && defined(WITH_HARFBUZZ)) && !defined(WITH_UNISCRIBE) && !defined(WITH_COCOA)
2411 static std::string err_str(
"XXXThis 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 t)
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.
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, const std::string &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.
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.
void SetFontNames(FontCacheSettings *settings, const char *font_name, const void *os_data) override
Set the right font names.
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.
Equivalent to the std::back_insert_iterator in function, with some convenience helpers for string con...
void Utf8Encode(char32_t c)
Encode the given Utf8 character into the output buffer.
void SetOffset(size_t offset)
Set the offset within the string from where to return the next result of GetInt64 or GetInt32.
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...
const char * GetNextParameterString()
Get the next string parameter from our 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.
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.
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, const 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.
const char * GetGameStringPtr(StringIndexInTab id)
Get the string pointer of a particular game string.
Base functions regarding game texts.
std::pair< uint8_t, uint8_t > GetBroadestDigit(FontSize size)
Determine the broadest digits for guessing the maximum width of a n-digit number.
void LoadStringWidthTable(bool monospace)
Initialize _stringwidth_table cache.
Functions related to laying out the texts.
FontSize
Available font sizes.
@ FS_MONO
Index of the monospaced font in the font tables.
@ FS_SMALL
Index of the small font in the font tables.
@ FS_NORMAL
Index of the normal font in the font tables.
@ FS_LARGE
Index of the large font in the font tables.
static constexpr GroupID DEFAULT_GROUP
Ungrouped vehicles are in this group.
const IndustrySpec * GetIndustrySpec(IndustryType thistype)
Accessor for array _industry_specs.
void SortIndustryTypes()
Initialize the list of sorted industry types.
Information about languages and their files.
static const uint8_t MAX_NUM_GENDERS
Maximum number of supported genders.
std::vector< LanguageMetadata > LanguageList
Type for the list of language meta data.
static const uint8_t MAX_NUM_CASES
Maximum number of supported cases.
constexpr bool IsInsideMM(const size_t x, const size_t min, const size_t max) noexcept
Checks if a value is in an interval.
constexpr T abs(const T a)
Returns the absolute value of (scalar) variable.
constexpr uint64_t PowerOfTen(int power)
Computes ten to the given power.
void BuildContentTypeStringList()
Build array of all strings corresponding to the content types.
User interface for downloading files.
@ CBID_VEHICLE_NAME
Called to determine the engine name to show.
static const uint CALLBACK_FAILED
Different values for Callback result evaluations.
uint16_t GetVehicleCallback(CallbackID callback, uint32_t param1, uint32_t param2, EngineID engine, const Vehicle *v)
Evaluate a newgrf callback for vehicles.
Functions for NewGRF engines.
const char * GetGRFStringPtr(StringIndexInTab stringid)
Get a C-string from a stringid set by a newgrf.
static void RemapNewGRFStringControlCode(char32_t scc, const char **str, TextRefStack &stack, std::vector< StringParameter > ¶ms)
Process NewGRF string control code instructions.
void SetCurrentGrfLangID(uint8_t language_id)
Equivalence Setter function between game and newgrf langID.
std::string GetGRFStringWithTextStack(const struct GRFFile *grffile, GRFStringID grfstringid, uint8_t num_entries)
Format a GRF string using the text ref stack for parameters.
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.
bool StrValid(std::span< const char > str)
Checks whether the given string is valid, i.e.
size_t Utf8Decode(char32_t *c, const char *s)
Decode and consume the next UTF-8 encoded character.
size_t Utf8Encode(T buf, char32_t c)
Encode a unicode character and place it in the buffer.
Functions related to low-level strings.
bool StrEmpty(const char *s)
Check if a string buffer is empty.
bool IsTextDirectionChar(char32_t c)
Is the given character a text direction character.
void MacOSSetCurrentLocaleName(const char *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.
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::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.
const char * GetCurrentLanguageIsoCode()
Get the ISO language code of the currently loaded language.
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.
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 void FormatNumber(StringBuilder &builder, int64_t number, const char *separator)
Format a number into a string.
std::string GetString(StringID string)
Resolve the given StringID into a std::string with formatting but no parameters.
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 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 FormatString(StringBuilder &builder, const char *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 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 char * GetSurname(uint32_t seed)
Get the surname of the president with the given seed.
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 const char * DecodeEncodedString(const char *str, bool game_script, StringBuilder &builder)
Decodes an encoded string during FormatString.
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.
const char * GetCurrentLocale(const char *param)
Determine the current charset based on the environment First check some default values,...
uint64_t GetParamMaxDigits(uint count, FontSize size)
Get some number that is suitable for string size computations.
static const Units _units_time_years_or_periods[]
Unit conversions for time in calendar years or economic periods.
static int DeterminePluralForm(int64_t count, int plural_form)
Determine the "plural" index given a plural form and a number.
Functions related to OTTD's strings.
StringTab GetStringTab(StringID str)
Extract the StringTab from a StringID.
auto MakeParameters(Args &&... args)
Helper to create the StringParameters with its own buffer with the given parameter values.
StringID MakeStringID(StringTab tab, StringIndexInTab index)
Create a StringID.
StringIndexInTab GetStringIndex(StringID str)
Extract the StringIndex from a StringID.
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
StrongType::Typedef< uint32_t, struct StringIndexInTabTag, StrongType::Compare, StrongType::Integer > StringIndexInTab
The index/offset of a string within a StringTab.
static constexpr StringID SPECSTR_COMPANY_NAME_START
Special strings for company names on the form "TownName transport".
static constexpr StringID SPECSTR_SILLY_NAME
Special string for silly company names.
static const uint TAB_SIZE_GAMESCRIPT
Number of strings for GameScripts.
static const uint MAX_LANG
Maximum number of languages supported by the game, and the NewGRF specs.
static constexpr StringID SPECSTR_ANDCO_NAME
Special string for Surname & Co company names.
static constexpr StringID SPECSTR_PRESIDENT_NAME
Special string for the president's name.
static constexpr StringID SPECSTR_TOWNNAME_START
Special strings for town names.
static const uint TAB_SIZE
Number of strings per StringTab.
StringTab
StringTabs to group StringIDs.
@ TEXT_TAB_NEWGRF_START
Start of NewGRF supplied strings.
@ TEXT_TAB_GAMESCRIPT_START
Start of GameScript supplied strings.
@ TEXT_TAB_END
End of language files.
TextDirection
Directions a text can go to.
@ TD_LTR
Text is written left-to-right by default.
Class to backup a specific variable and restore it upon destruction of this object to prevent stack v...
std::string name
Name of vehicle.
StringID string_id
Default name (town area) of station.
Town * town
The town this station is associated with.
std::string name
Custom name.
VehicleType type
Type of vehicle.
StringID units_volume
Name of a single unit of cargo of this type.
static CargoSpec * Get(size_t index)
Retrieve cargo details for the given cargo type.
StringID quantifier
Text for multiple units of cargo of this type.
static size_t GetArraySize()
Total number of cargospecs, both valid and invalid.
std::string president_name
Name of the president if the user changed it.
uint32_t name_2
Parameter of name_1.
uint32_t president_name_2
Parameter of president_name_1.
StringID name_1
Name of the company if the user did not change it.
StringID president_name_1
Name of the president if the user did not change it.
std::string name
Name of the company if the user changed it.
static bool IsValidHumanID(auto index)
Is this company a valid company, not controlled by a NoAI program?
Specification of a currency.
std::string separator
The thousands separator for this currency.
std::string prefix
Prefix to apply when formatting money in this currency.
std::string suffix
Suffix to apply when formatting money in this currency.
uint16_t rate
The conversion rate compared to the base currency.
uint8_t symbol_pos
The currency symbol is represented by two possible values, prefix and suffix Usage of one or the othe...
uint16_t town_cn
The N-1th depot for this town (consecutive number)
Settings for the four different fonts.
FontCacheSubSetting mono
The mono space font used for license/readme viewers.
FontCacheSubSetting medium
The normal font size.
std::string font
The name of the font, or path to the font.
const void * os_handle
Optional native OS font info. Only valid during font search.
Dynamic data of a loaded NewGRF.
LandscapeType landscape
the landscape we're currently in
LocaleSettings locale
settings related to used currency/unit system in the current game
GameCreationSettings game_creation
settings used during the creation of a game (map)
std::string name
Group Name.
uint16_t number
Per-company group number.
Defines the data structure for constructing industry.
StringID name
Displayed name of the industry.
StringID station_name
Default name for nearby station.
Defines the internal data of a functional industry.
IndustryType type
type of industry.
std::string list_separator
Current list separator string.
std::array< uint, TEXT_TAB_END > langtab_num
Offset into langpack offs.
std::array< uint, TEXT_TAB_END > langtab_start
Offset into langpack offs.
uint8_t units_velocity_nautical
unit system for velocity of ships and aircraft
uint8_t units_velocity
unit system for velocity of trains and road vehicles
std::string digit_decimal_separator
decimal separator
std::string digit_group_separator_currency
thousand separator for currencies
uint8_t units_power
unit system for power
uint8_t units_height
unit system for height
uint8_t units_force
unit system for force
uint8_t units_volume
unit system for volume
uint8_t units_weight
unit system for weight
std::string digit_group_separator
thousand separator for non-currencies
static Titem * Get(auto index)
Returns Titem with given index.
Tindex index
Index of this pool item.
static Titem * GetIfValid(auto index)
Returns Titem with given index.
static Station * GetIfValid(auto index)
Returns station if the index is a valid index for this station type.
The data required to format and validate a single parameter of a string.
Templated helper to make a type-safe 'typedef' representing a single POD value.
std::string name
Custom town name. If empty, the town was not renamed and uses the generated name.
Helper for unit conversion.
double factor
Amount to multiply or divide upon conversion.
int64_t ToDisplay(int64_t input, bool round=true) const
Convert value from OpenTTD's internal unit into the displayed value.
int64_t FromDisplay(int64_t input, bool round=true, int64_t divider=1) const
Convert the displayed value back into a value of OpenTTD's internal unit.
Information about a specific unit system with a long variant.
StringID s
String for the short variant of the unit.
StringID l
String for the long variant of the unit.
unsigned int decimal_places
Number of decimal places embedded in the value. For example, 1 if the value is in tenths,...
UnitConversion c
Conversion.
Information about a specific unit system.
StringID s
String for the unit.
unsigned int decimal_places
Number of decimal places embedded in the value. For example, 1 if the value is in tenths,...
UnitConversion c
Conversion.
GroupID group_id
Index of group Pool array.
UnitID unitnumber
unit number, for display purposes only
Representation of a waypoint.
uint16_t town_cn
The N-1th waypoint for this town (consecutive number)
Definition of the game-calendar-timer.
Town name generator stuff.
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(const std::string &name)
Convert from OpenTTD's encoding to a wide string.
std::string FS2OTTD(const std::wstring &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: