OpenTTD Source 20250717-master-g55605ae8f2
strings.cpp
Go to the documentation of this file.
1/*
2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
6 */
7
10#include "stdafx.h"
11#include "currency.h"
12#include "station_base.h"
13#include "town.h"
14#include "waypoint_base.h"
15#include "depot_base.h"
16#include "industry.h"
17#include "newgrf_text.h"
18#include "fileio_func.h"
19#include "signs_base.h"
20#include "fontdetection.h"
21#include "error.h"
22#include "error_func.h"
23#include "strings_func.h"
24#include "rev.h"
25#include "core/endian_func.hpp"
27#include "vehicle_base.h"
28#include "engine_base.h"
29#include "language.h"
30#include "townname_func.h"
31#include "string_func.h"
32#include "company_base.h"
33#include "smallmap_gui.h"
34#include "window_func.h"
35#include "debug.h"
36#include "game/game_text.hpp"
38#include "newgrf_engine.h"
39#include "core/backup_type.hpp"
40#include "gfx_layout.h"
41#include "core/utf8.hpp"
43#include <stack>
44
45#include "table/strings.h"
46#include "table/control_codes.h"
47#include "3rdparty/fmt/std.h"
48
49#include "strings_internal.h"
50
51#include "safeguards.h"
52
56
58
59#ifdef WITH_ICU_I18N
60std::unique_ptr<icu::Collator> _current_collator;
61#endif /* WITH_ICU_I18N */
62
70{
71 assert(this->next_type == 0 || (SCC_CONTROL_START <= this->next_type && this->next_type <= SCC_CONTROL_END));
72 if (this->offset >= this->parameters.size()) {
73 throw std::out_of_range("Trying to read invalid string parameter");
74 }
75
76 auto &param = this->parameters[this->offset++];
77 if (param.type != 0 && param.type != this->next_type) {
78 this->next_type = 0;
79 throw std::out_of_range("Trying to read string parameter with wrong type");
80 }
81 param.type = this->next_type;
82 this->next_type = 0;
83 return param;
84}
85
95
103EncodedString GetEncodedStringWithArgs(StringID str, std::span<const StringParameter> params)
104{
105 std::string result;
106 StringBuilder builder(result);
108 builder.PutIntegerBase(str, 16);
109
110 struct visitor {
111 StringBuilder &builder;
112
113 void operator()(const std::monostate &) {}
114
115 void operator()(const uint64_t &arg)
117 this->builder.PutUtf8(SCC_ENCODED_NUMERIC);
118 this->builder.PutIntegerBase(arg, 16);
119 }
120
121 void operator()(const std::string &value)
122 {
123#ifdef WITH_ASSERT
124 /* Don't allow an encoded string to contain another encoded string. */
125 {
126 auto [len, c] = DecodeUtf8(value);
127 assert(len == 0 || (c != SCC_ENCODED && c != SCC_ENCODED_INTERNAL && c != SCC_RECORD_SEPARATOR));
128 }
129#endif /* WITH_ASSERT */
130 this->builder.PutUtf8(SCC_ENCODED_STRING);
131 this->builder += value;
132 }
133 };
134
135 visitor v{builder};
136 for (const auto &param : params) {
137 builder.PutUtf8(SCC_RECORD_SEPARATOR);
138 std::visit(v, param.data);
139 }
140
141 return EncodedString{std::move(result)};
142}
143
152{
153 if (this->empty()) return {};
154
155 std::vector<StringParameter> params;
156 StringConsumer consumer(this->string);
157
158 if (!consumer.ReadUtf8If(SCC_ENCODED_INTERNAL)) return {};
159
160 StringID str;
161 if (auto r = consumer.TryReadIntegerBase<uint32_t>(16); r.has_value()) {
162 str = *r;
163 } else {
164 return {};
165 }
166 if (consumer.AnyBytesLeft() && !consumer.ReadUtf8If(SCC_RECORD_SEPARATOR)) return {};
167
168 while (consumer.AnyBytesLeft()) {
169 StringConsumer record(consumer.ReadUntilUtf8(SCC_RECORD_SEPARATOR, StringConsumer::SKIP_ONE_SEPARATOR));
170
171 if (!record.AnyBytesLeft()) {
172 /* This is an empty parameter. */
173 params.emplace_back(std::monostate{});
174 continue;
175 }
176
177 /* Get the parameter type. */
178 char32_t parameter_type = record.ReadUtf8();
179 switch (parameter_type) {
180 case SCC_ENCODED_NUMERIC: {
181 uint64_t value = record.ReadIntegerBase<uint64_t>(16);
182 assert(!record.AnyBytesLeft());
183 params.emplace_back(value);
184 break;
185 }
186
187 case SCC_ENCODED_STRING: {
188 params.emplace_back(record.Read(StringConsumer::npos));
189 break;
190 }
191
192 default:
193 /* Unknown parameter, make it blank. */
194 params.emplace_back(std::monostate{});
195 break;
196 }
197 }
198
199 if (param >= std::size(params)) return {};
200 params[param] = data;
201 return GetEncodedStringWithArgs(str, params);
202}
203
209{
210 return GetString(STR_JUST_RAW_STRING, this->string);
211}
212
219uint64_t GetParamMaxDigits(uint count, FontSize size)
220{
221 auto [front, next] = GetBroadestDigit(size);
222 uint64_t val = count > 1 ? front : next;
223 for (; count > 1; count--) {
224 val = 10 * val + next;
225 }
226 return val;
227}
228
237uint64_t GetParamMaxValue(uint64_t max_value, uint min_count, FontSize size)
238{
239 uint num_digits = 1;
240 while (max_value >= 10) {
241 num_digits++;
242 max_value /= 10;
243 }
244 return GetParamMaxDigits(std::max(min_count, num_digits), size);
245}
246
247static void StationGetSpecialString(StringBuilder &builder, StationFacilities x);
248static bool GetSpecialNameString(StringBuilder &builder, StringID string, StringParameters &args);
249
250static void FormatString(StringBuilder &builder, std::string_view str, StringParameters &args, uint case_index = 0, bool game_script = false, bool dry_run = false);
251
262static void FormatString(StringBuilder &builder, std::string_view str, std::span<StringParameter> params, uint case_index = 0, bool game_script = false, bool dry_run = false)
263{
264 StringParameters tmp_params{params};
265 FormatString(builder, str, tmp_params, case_index, game_script, dry_run);
266}
267
269 char data[]; // list of strings
270};
271
273 void operator()(LanguagePack *langpack)
274 {
275 /* LanguagePack is in fact reinterpreted char[], we need to reinterpret it back to free it properly. */
276 delete[] reinterpret_cast<char *>(langpack);
277 }
278};
279
281 std::unique_ptr<LanguagePack, LanguagePackDeleter> langpack;
282
283 std::vector<std::string_view> strings;
284
285 std::array<uint, TEXT_TAB_END> langtab_num;
286 std::array<uint, TEXT_TAB_END> langtab_start;
287
288 std::string list_separator;
289 std::string ellipsis;
290};
291
292static LoadedLanguagePack _langpack;
293
294static bool _scan_for_gender_data = false;
295
300std::string_view GetListSeparator()
301{
302 return _langpack.list_separator;
303}
304
309std::string_view GetEllipsis()
310{
311 return _langpack.ellipsis;
312}
313
314std::string_view GetStringPtr(StringID string)
315{
316 switch (GetStringTab(string)) {
318 /* 0xD0xx and 0xD4xx IDs have been converted earlier. */
319 case TEXT_TAB_OLD_NEWGRF: NOT_REACHED();
321 default: {
322 const size_t offset = _langpack.langtab_start[GetStringTab(string)] + GetStringIndex(string).base();
323 if (offset < _langpack.strings.size()) return _langpack.strings[offset];
324 return "(undefined string)";
325 }
326 }
327}
328
337void GetStringWithArgs(StringBuilder &builder, StringID string, StringParameters &args, uint case_index, bool game_script)
338{
339 if (string == 0) {
340 GetStringWithArgs(builder, STR_UNDEFINED, args);
341 return;
342 }
343
344 StringIndexInTab index = GetStringIndex(string);
345 StringTab tab = GetStringTab(string);
346
347 switch (tab) {
348 case TEXT_TAB_TOWN:
349 if (IsInsideMM(string, SPECSTR_TOWNNAME_START, SPECSTR_TOWNNAME_END) && !game_script) {
350 try {
351 GenerateTownNameString(builder, string - SPECSTR_TOWNNAME_START, args.GetNextParameter<uint32_t>());
352 } catch (const std::runtime_error &e) {
353 Debug(misc, 0, "GetStringWithArgs: {}", e.what());
354 builder += "(invalid string parameter)";
355 }
356 return;
357 }
358 break;
359
360 case TEXT_TAB_SPECIAL:
361 if (!game_script) {
362 try {
363 if (GetSpecialNameString(builder, string, args)) return;
364 } catch (const std::runtime_error &e) {
365 Debug(misc, 0, "GetStringWithArgs: {}", e.what());
366 builder += "(invalid string parameter)";
367 return;
368 }
369 }
370 break;
371
372 case TEXT_TAB_OLD_CUSTOM:
373 /* Old table for custom names. This is no longer used */
374 if (!game_script) {
375 FatalError("Incorrect conversion of custom name string.");
376 }
377 break;
378
380 FormatString(builder, GetGameStringPtr(index), args, case_index, true);
381 return;
382 }
383
384 case TEXT_TAB_OLD_NEWGRF:
385 NOT_REACHED();
386
388 FormatString(builder, GetGRFStringPtr(index), args, case_index);
389 return;
390 }
391
392 default:
393 break;
394 }
395
396 if (index >= _langpack.langtab_num[tab]) {
397 if (game_script) {
398 return GetStringWithArgs(builder, STR_UNDEFINED, args);
399 }
400 FatalError("String 0x{:X} is invalid. You are probably using an old version of the .lng file.\n", string);
401 }
402
403 FormatString(builder, GetStringPtr(string), args, case_index);
404}
405
414void GetStringWithArgs(StringBuilder &builder, StringID string, std::span<StringParameter> params, uint case_index, bool game_script)
415{
416 StringParameters tmp_params{params};
417 GetStringWithArgs(builder, string, tmp_params, case_index, game_script);
418}
419
425std::string GetString(StringID string)
426{
427 return GetStringWithArgs(string, {});
428}
429
435void AppendStringInPlace(std::string &result, StringID string)
436{
437 StringBuilder builder(result);
438 GetStringWithArgs(builder, string, {});
439}
440
441void AppendStringWithArgsInPlace(std::string &result, StringID string, std::span<StringParameter> params)
442{
443 StringParameters tmp_params{params};
444 StringBuilder builder(result);
445 GetStringWithArgs(builder, string, tmp_params);
446}
447
455{
456 std::string result;
457 StringBuilder builder(result);
458 GetStringWithArgs(builder, string, args);
459 return result;
460}
461
462std::string GetStringWithArgs(StringID string, std::span<StringParameter> args)
463{
464 std::string result;
465 StringBuilder builder(result);
466 GetStringWithArgs(builder, string, args);
467 return result;
468}
469
470static std::string_view GetDecimalSeparator()
471{
472 std::string_view decimal_separator = _settings_game.locale.digit_decimal_separator;
473 if (decimal_separator.empty()) decimal_separator = _langpack.langpack->digit_decimal_separator;
474 return decimal_separator;
475}
476
483static void FormatNumber(StringBuilder &builder, int64_t number, std::string_view separator)
484{
485 static const int max_digits = 20;
486 uint64_t divisor = 10000000000000000000ULL;
487 int thousands_offset = (max_digits - 1) % 3;
488
489 if (number < 0) {
490 builder.PutChar('-');
491 number = -number;
492 }
493
494 uint64_t num = number;
495 uint64_t tot = 0;
496 for (int i = 0; i < max_digits; i++) {
497 uint64_t quot = 0;
498 if (num >= divisor) {
499 quot = num / divisor;
500 num = num % divisor;
501 }
502 if ((tot |= quot) || i == max_digits - 1) {
503 builder.PutChar('0' + quot); // quot is a single digit
504 if ((i % 3) == thousands_offset && i < max_digits - 1) builder += separator;
505 }
506
507 divisor /= 10;
508 }
509}
510
511static void FormatCommaNumber(StringBuilder &builder, int64_t number)
512{
513 std::string_view separator = _settings_game.locale.digit_group_separator;
514 if (separator.empty()) separator = _langpack.langpack->digit_group_separator;
515 FormatNumber(builder, number, separator);
516}
517
518static void FormatNoCommaNumber(StringBuilder &builder, int64_t number)
519{
520 fmt::format_to(builder.back_inserter(), "{}", number);
521}
522
523static void FormatZerofillNumber(StringBuilder &builder, int64_t number, int count)
524{
525 fmt::format_to(builder.back_inserter(), "{:0{}d}", number, count);
526}
527
528static void FormatHexNumber(StringBuilder &builder, uint64_t number)
529{
530 fmt::format_to(builder.back_inserter(), "0x{:X}", number);
531}
532
538static void FormatBytes(StringBuilder &builder, int64_t number)
539{
540 assert(number >= 0);
541
542 /* 1 2^10 2^20 2^30 2^40 2^50 2^60 */
543 static const std::string_view iec_prefixes[] = {"", "Ki", "Mi", "Gi", "Ti", "Pi", "Ei"};
544 uint id = 1;
545 while (number >= 1024 * 1024) {
546 number /= 1024;
547 id++;
548 }
549
550 if (number < 1024) {
551 id = 0;
552 fmt::format_to(builder.back_inserter(), "{}", number);
553 } else if (number < 1024 * 10) {
554 fmt::format_to(builder.back_inserter(), "{}{}{:02}", number / 1024, GetDecimalSeparator(), (number % 1024) * 100 / 1024);
555 } else if (number < 1024 * 100) {
556 fmt::format_to(builder.back_inserter(), "{}{}{:01}", number / 1024, GetDecimalSeparator(), (number % 1024) * 10 / 1024);
557 } else {
558 assert(number < 1024 * 1024);
559 fmt::format_to(builder.back_inserter(), "{}", number / 1024);
560 }
561
562 assert(id < lengthof(iec_prefixes));
563 fmt::format_to(builder.back_inserter(), NBSP "{}B", iec_prefixes[id]);
564}
565
566static void FormatYmdString(StringBuilder &builder, TimerGameCalendar::Date date, uint case_index)
567{
568 TimerGameCalendar::YearMonthDay ymd = TimerGameCalendar::ConvertDateToYMD(date);
569
570 auto tmp_params = MakeParameters(STR_DAY_NUMBER_1ST + ymd.day - 1, STR_MONTH_ABBREV_JAN + ymd.month, ymd.year);
571 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_LONG), tmp_params, case_index);
572}
573
574static void FormatMonthAndYear(StringBuilder &builder, TimerGameCalendar::Date date, uint case_index)
575{
576 TimerGameCalendar::YearMonthDay ymd = TimerGameCalendar::ConvertDateToYMD(date);
577
578 auto tmp_params = MakeParameters(STR_MONTH_JAN + ymd.month, ymd.year);
579 FormatString(builder, GetStringPtr(STR_FORMAT_DATE_SHORT), tmp_params, case_index);
580}
581
582static void FormatTinyOrISODate(StringBuilder &builder, TimerGameCalendar::Date date, StringID str)
583{
584 TimerGameCalendar::YearMonthDay ymd = TimerGameCalendar::ConvertDateToYMD(date);
585
586 /* Day and month are zero-padded with ZEROFILL_NUM, hence the two 2s. */
587 auto tmp_params = MakeParameters(ymd.day, 2, ymd.month + 1, 2, ymd.year);
588 FormatString(builder, GetStringPtr(str), tmp_params);
589}
590
591static void FormatGenericCurrency(StringBuilder &builder, const CurrencySpec *spec, Money number, bool compact)
592{
593 /* We are going to make number absolute for printing, so
594 * keep this piece of data as we need it later on */
595 bool negative = number < 0;
596
597 number *= spec->rate;
598
599 /* convert from negative */
600 if (number < 0) {
601 builder.PutUtf8(SCC_PUSH_COLOUR);
602 builder.PutUtf8(SCC_RED);
603 builder.PutChar('-');
604 number = -number;
605 }
606
607 /* Add prefix part, following symbol_pos specification.
608 * Here, it can can be either 0 (prefix) or 2 (both prefix and suffix).
609 * The only remaining value is 1 (suffix), so everything that is not 1 */
610 if (spec->symbol_pos != 1) builder += spec->prefix;
611
612 StringID number_str = STR_NULL;
613
614 /* For huge numbers, compact the number. */
615 if (compact) {
616 /* Take care of the thousand rounding. Having 1 000 000 k
617 * and 1 000 M is inconsistent, so always use 1 000 M. */
618 if (number >= Money(1'000'000'000'000'000) - 500'000'000) {
619 number = (number + Money(500'000'000'000)) / Money(1'000'000'000'000);
620 number_str = STR_CURRENCY_SHORT_TERA;
621 } else if (number >= Money(1'000'000'000'000) - 500'000) {
622 number = (number + 500'000'000) / 1'000'000'000;
623 number_str = STR_CURRENCY_SHORT_GIGA;
624 } else if (number >= 1'000'000'000 - 500) {
625 number = (number + 500'000) / 1'000'000;
626 number_str = STR_CURRENCY_SHORT_MEGA;
627 } else if (number >= 1'000'000) {
628 number = (number + 500) / 1'000;
629 number_str = STR_CURRENCY_SHORT_KILO;
630 }
631 }
632
633 std::string_view separator = _settings_game.locale.digit_group_separator_currency;
634 if (separator.empty()) separator = GetCurrency().separator;
635 if (separator.empty()) separator = _langpack.langpack->digit_group_separator_currency;
636 FormatNumber(builder, number, separator);
637 if (number_str != STR_NULL) {
638 FormatString(builder, GetStringPtr(number_str), {});
639 }
640
641 /* Add suffix part, following symbol_pos specification.
642 * Here, it can can be either 1 (suffix) or 2 (both prefix and suffix).
643 * The only remaining value is 1 (prefix), so everything that is not 0 */
644 if (spec->symbol_pos != 0) builder += spec->suffix;
645
646 if (negative) {
647 builder.PutUtf8(SCC_POP_COLOUR);
648 }
649}
650
657static int DeterminePluralForm(int64_t count, uint plural_form)
658{
659 /* The absolute value determines plurality */
660 uint64_t n = abs(count);
661
662 switch (plural_form) {
663 default:
664 NOT_REACHED();
665
666 /* Two forms: singular used for one only.
667 * Used in:
668 * Danish, Dutch, English, German, Norwegian, Swedish, Estonian, Finnish,
669 * Greek, Hebrew, Italian, Portuguese, Spanish, Esperanto */
670 case 0:
671 return n != 1 ? 1 : 0;
672
673 /* Only one form.
674 * Used in:
675 * Hungarian, Japanese, Turkish */
676 case 1:
677 return 0;
678
679 /* Two forms: singular used for 0 and 1.
680 * Used in:
681 * French, Brazilian Portuguese */
682 case 2:
683 return n > 1 ? 1 : 0;
684
685 /* Three forms: special cases for 0, and numbers ending in 1 except when ending in 11.
686 * Note: Cases are out of order for hysterical reasons. '0' is last.
687 * Used in:
688 * Latvian */
689 case 3:
690 return n % 10 == 1 && n % 100 != 11 ? 0 : n != 0 ? 1 : 2;
691
692 /* Five forms: special cases for 1, 2, 3 to 6, and 7 to 10.
693 * Used in:
694 * Gaelige (Irish) */
695 case 4:
696 return n == 1 ? 0 : n == 2 ? 1 : n < 7 ? 2 : n < 11 ? 3 : 4;
697
698 /* Three forms: special cases for numbers ending in 1 except when ending in 11, and 2 to 9 except when ending in 12 to 19.
699 * Used in:
700 * Lithuanian */
701 case 5:
702 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
703
704 /* Three forms: special cases for numbers ending in 1 except when ending in 11, and 2 to 4 except when ending in 12 to 14.
705 * Used in:
706 * Croatian, Russian, Ukrainian */
707 case 6:
708 return n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
709
710 /* Three forms: special cases for 1, and numbers ending in 2 to 4 except when ending in 12 to 14.
711 * Used in:
712 * Polish */
713 case 7:
714 return n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
715
716 /* Four forms: special cases for numbers ending in 01, 02, and 03 to 04.
717 * Used in:
718 * Slovenian */
719 case 8:
720 return n % 100 == 1 ? 0 : n % 100 == 2 ? 1 : n % 100 == 3 || n % 100 == 4 ? 2 : 3;
721
722 /* Two forms: singular used for numbers ending in 1 except when ending in 11.
723 * Used in:
724 * Icelandic */
725 case 9:
726 return n % 10 == 1 && n % 100 != 11 ? 0 : 1;
727
728 /* Three forms: special cases for 1, and 2 to 4
729 * Used in:
730 * Czech, Slovak */
731 case 10:
732 return n == 1 ? 0 : n >= 2 && n <= 4 ? 1 : 2;
733
734 /* Two forms: cases for numbers ending with a consonant, and with a vowel.
735 * Korean doesn't have the concept of plural, but depending on how a
736 * number is pronounced it needs another version of a particle.
737 * As such the plural system is misused to give this distinction.
738 */
739 case 11:
740 switch (n % 10) {
741 case 0: // yeong
742 case 1: // il
743 case 3: // sam
744 case 6: // yuk
745 case 7: // chil
746 case 8: // pal
747 return 0;
748
749 case 2: // i
750 case 4: // sa
751 case 5: // o
752 case 9: // gu
753 return 1;
754
755 default:
756 NOT_REACHED();
757 }
758
759 /* Four forms: special cases for 1, 0 and numbers ending in 02 to 10, and numbers ending in 11 to 19.
760 * Used in:
761 * Maltese */
762 case 12:
763 return (n == 1 ? 0 : n == 0 || (n % 100 > 1 && n % 100 < 11) ? 1 : (n % 100 > 10 && n % 100 < 20) ? 2 : 3);
764 /* Four forms: special cases for 1 and 11, 2 and 12, 3 .. 10 and 13 .. 19, other
765 * Used in:
766 * Scottish Gaelic */
767 case 13:
768 return ((n == 1 || n == 11) ? 0 : (n == 2 || n == 12) ? 1 : ((n > 2 && n < 11) || (n > 12 && n < 20)) ? 2 : 3);
769
770 /* Three forms: special cases for 1, 0 and numbers ending in 01 to 19.
771 * Used in:
772 * Romanian */
773 case 14:
774 return n == 1 ? 0 : (n == 0 || (n % 100 > 0 && n % 100 < 20)) ? 1 : 2;
775 }
776}
777
778static void SkipStringChoice(StringConsumer &consumer)
779{
780 uint n = consumer.ReadUint8();
781 uint len = 0;
782 for (uint i = 0; i != n; i++) {
783 len += consumer.ReadUint8();
784 }
785 consumer.Skip(len);
786}
787
788static void ParseStringChoice(StringConsumer &consumer, uint form, StringBuilder &builder)
789{
790 /* <NUM> {Length of each string} {each string} */
791 uint n = consumer.ReadUint8();
792 size_t form_pre = 0, form_len = 0, form_post = 0;
793 for (uint i = 0; i != n; i++) {
794 uint len = consumer.ReadUint8();
795 if (i < form) {
796 form_pre += len;
797 } else if (i > form) {
798 form_post += len;
799 } else {
800 form_len = len;
801 }
802 }
803
804 consumer.Skip(form_pre);
805 builder += consumer.Read(form_len);
806 consumer.Skip(form_post);
807}
808
811 double factor;
812
819 int64_t ToDisplay(int64_t input, bool round = true) const
820 {
821 return round
822 ? (int64_t)std::round(input * this->factor)
823 : (int64_t)(input * this->factor);
824 }
825
833 int64_t FromDisplay(int64_t input, bool round = true, int64_t divider = 1) const
834 {
835 return round
836 ? (int64_t)std::round(input / this->factor / divider)
837 : (int64_t)(input / this->factor / divider);
838 }
839};
840
847
855
858 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
859 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
860 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
861 { { 0.578125 }, STR_UNITS_VELOCITY_GAMEUNITS_DAY, 1 },
862 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
863};
864
867 { { 1.0 }, STR_UNITS_VELOCITY_IMPERIAL, 0 },
868 { { 1.609344 }, STR_UNITS_VELOCITY_METRIC, 0 },
869 { { 0.44704 }, STR_UNITS_VELOCITY_SI, 0 },
870 { { 0.289352 }, STR_UNITS_VELOCITY_GAMEUNITS_SEC, 1 },
871 { { 0.868976 }, STR_UNITS_VELOCITY_KNOTS, 0 },
872};
873
875static const Units _units_power[] = {
876 { { 1.0 }, STR_UNITS_POWER_IMPERIAL, 0 },
877 { { 1.01387 }, STR_UNITS_POWER_METRIC, 0 },
878 { { 0.745699 }, STR_UNITS_POWER_SI, 0 },
879};
880
883 { { 0.907185 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_IMPERIAL, 1 },
884 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_METRIC, 1 },
885 { { 1.0 }, STR_UNITS_POWER_IMPERIAL_TO_WEIGHT_SI, 1 },
886 { { 0.919768 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_IMPERIAL, 1 },
887 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_METRIC, 1 },
888 { { 1.01387 }, STR_UNITS_POWER_METRIC_TO_WEIGHT_SI, 1 },
889 { { 0.676487 }, STR_UNITS_POWER_SI_TO_WEIGHT_IMPERIAL, 1 },
890 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_METRIC, 1 },
891 { { 0.745699 }, STR_UNITS_POWER_SI_TO_WEIGHT_SI, 1 },
892};
893
895static const UnitsLong _units_weight[] = {
896 { { 1.102311 }, STR_UNITS_WEIGHT_SHORT_IMPERIAL, STR_UNITS_WEIGHT_LONG_IMPERIAL, 0 },
897 { { 1.0 }, STR_UNITS_WEIGHT_SHORT_METRIC, STR_UNITS_WEIGHT_LONG_METRIC, 0 },
898 { { 1000.0 }, STR_UNITS_WEIGHT_SHORT_SI, STR_UNITS_WEIGHT_LONG_SI, 0 },
899};
900
902static const UnitsLong _units_volume[] = {
903 { { 264.172 }, STR_UNITS_VOLUME_SHORT_IMPERIAL, STR_UNITS_VOLUME_LONG_IMPERIAL, 0 },
904 { { 1000.0 }, STR_UNITS_VOLUME_SHORT_METRIC, STR_UNITS_VOLUME_LONG_METRIC, 0 },
905 { { 1.0 }, STR_UNITS_VOLUME_SHORT_SI, STR_UNITS_VOLUME_LONG_SI, 0 },
906};
907
909static const Units _units_force[] = {
910 { { 0.224809 }, STR_UNITS_FORCE_IMPERIAL, 0 },
911 { { 0.101972 }, STR_UNITS_FORCE_METRIC, 0 },
912 { { 0.001 }, STR_UNITS_FORCE_SI, 0 },
913};
914
916static const Units _units_height[] = {
917 { { 3.0 }, STR_UNITS_HEIGHT_IMPERIAL, 0 }, // "Wrong" conversion factor for more nicer GUI values
918 { { 1.0 }, STR_UNITS_HEIGHT_METRIC, 0 },
919 { { 1.0 }, STR_UNITS_HEIGHT_SI, 0 },
920};
921
924 { { 1 }, STR_UNITS_DAYS, 0 },
925 { { 2 }, STR_UNITS_SECONDS, 0 },
926};
927
930 { { 1 }, STR_UNITS_MONTHS, 0 },
931 { { 1 }, STR_UNITS_MINUTES, 0 },
932};
933
936 { { 1 }, STR_UNITS_YEARS, 0 },
937 { { 1 }, STR_UNITS_PERIODS, 0 },
938};
939
942 { { 1 }, STR_UNITS_YEARS, 0 },
943 { { 12 }, STR_UNITS_MINUTES, 0 },
944};
945
952{
954
955 assert(setting < lengthof(_units_velocity_calendar));
956 assert(setting < lengthof(_units_velocity_realtime));
957
959
960 return _units_velocity_calendar[setting];
961}
962
969{
970 /* For historical reasons we don't want to mess with the
971 * conversion for speed. So, don't round it and keep the
972 * original conversion factors instead of the real ones. */
973 return GetVelocityUnits(type).c.ToDisplay(speed, false);
974}
975
982{
983 return GetVelocityUnits(type).c.FromDisplay(speed);
984}
985
992{
993 return GetVelocityUnits(type).c.ToDisplay(speed * 10, false) / 16;
994}
995
1002{
1003 return GetVelocityUnits(type).c.FromDisplay(speed * 16, true, 10);
1004}
1005
1012static void DecodeEncodedString(StringConsumer &consumer, bool game_script, StringBuilder &builder)
1013{
1014 std::vector<StringParameter> sub_args;
1015
1016 StringIndexInTab id(consumer.ReadIntegerBase<uint32_t>(16));
1017 if (consumer.AnyBytesLeft() && !consumer.PeekUtf8If(SCC_RECORD_SEPARATOR)) {
1018 consumer.SkipAll();
1019 builder += "(invalid SCC_ENCODED)";
1020 return;
1021 }
1022 if (game_script && id >= TAB_SIZE_GAMESCRIPT) {
1023 consumer.SkipAll();
1024 builder += "(invalid StringID)";
1025 return;
1026 }
1027
1028 while (consumer.AnyBytesLeft()) {
1029 consumer.SkipUtf8If(SCC_RECORD_SEPARATOR);
1030 StringConsumer record(consumer.ReadUntilUtf8(SCC_RECORD_SEPARATOR, StringConsumer::KEEP_SEPARATOR));
1031
1032 if (!record.AnyBytesLeft()) {
1033 /* This is an empty parameter. */
1034 sub_args.emplace_back(std::monostate{});
1035 continue;
1036 }
1037
1038 /* Get the parameter type. */
1039 char32_t parameter_type = record.ReadUtf8();
1040 switch (parameter_type) {
1041 case SCC_ENCODED: {
1042 uint64_t param = record.ReadIntegerBase<uint64_t>(16);
1043 if (param >= TAB_SIZE_GAMESCRIPT) {
1044 builder += "(invalid sub-StringID)";
1045 return;
1046 }
1047 assert(!record.AnyBytesLeft());
1049 sub_args.emplace_back(param);
1050 break;
1051 }
1052
1053 case SCC_ENCODED_NUMERIC: {
1054 uint64_t param = record.ReadIntegerBase<uint64_t>(16);
1055 assert(!record.AnyBytesLeft());
1056 sub_args.emplace_back(param);
1057 break;
1058 }
1059
1060 case SCC_ENCODED_STRING: {
1061 sub_args.emplace_back(record.Read(StringConsumer::npos));
1062 break;
1063 }
1064
1065 default:
1066 /* Unknown parameter, make it blank. */
1067 sub_args.emplace_back(std::monostate{});
1068 break;
1069 }
1070 }
1071
1072 StringID stringid = game_script ? MakeStringID(TEXT_TAB_GAMESCRIPT_START, id) : StringID{id.base()};
1073 GetStringWithArgs(builder, stringid, sub_args, true);
1074}
1075
1083static void FormatString(StringBuilder &builder, std::string_view str_arg, StringParameters &args, uint orig_case_index, bool game_script, bool dry_run)
1084{
1085 size_t orig_first_param_offset = args.GetOffset();
1086
1087 if (!dry_run) {
1088 /*
1089 * This function is normally called with `dry_run` false, then we call this function again
1090 * with `dry_run` being true. The dry run is required for the gender formatting. For the
1091 * gender determination we need to format a sub string to get the gender, but for that we
1092 * need to know as what string control code type the specific parameter is encoded. Since
1093 * gendered words can be before the "parameter" words, this needs to be determined before
1094 * the actual formatting.
1095 */
1096 std::string buffer;
1097 StringBuilder dry_run_builder(buffer);
1098 FormatString(dry_run_builder, str_arg, args, orig_case_index, game_script, true);
1099 /* We have to restore the original offset here to to read the correct values. */
1100 args.SetOffset(orig_first_param_offset);
1101 }
1102 uint next_substr_case_index = 0;
1103 struct StrStackItem {
1104 StringConsumer consumer;
1105 size_t first_param_offset;
1106 uint case_index;
1107
1108 StrStackItem(std::string_view view, size_t first_param_offset, uint case_index)
1109 : consumer(view), first_param_offset(first_param_offset), case_index(case_index)
1110 {}
1111 };
1112 std::stack<StrStackItem, std::vector<StrStackItem>> str_stack;
1113 str_stack.emplace(str_arg, orig_first_param_offset, orig_case_index);
1114
1115 for (;;) {
1116 try {
1117 while (!str_stack.empty() && !str_stack.top().consumer.AnyBytesLeft()) {
1118 str_stack.pop();
1119 }
1120 if (str_stack.empty()) break;
1121 StringConsumer &consumer = str_stack.top().consumer;
1122 const size_t ref_param_offset = str_stack.top().first_param_offset;
1123 const uint case_index = str_stack.top().case_index;
1124 char32_t b = consumer.ReadUtf8();
1125 assert(b != 0);
1126 if (b == 0) {
1127 /* A NUL character should never be encountered, but for non-debug builds handle it gracefully. */
1128 builder += "(unexpected NUL)";
1129 continue;
1130 }
1131
1132 if (SCC_NEWGRF_FIRST <= b && b <= SCC_NEWGRF_LAST) {
1133 /* We need to pass some stuff as it might be modified. */
1134 b = RemapNewGRFStringControlCode(b, consumer);
1135 if (b == 0) continue;
1136 }
1137
1138 if (b < SCC_CONTROL_START || b > SCC_CONTROL_END) {
1139 builder.PutUtf8(b);
1140 continue;
1141 }
1142
1143 args.SetTypeOfNextParameter(b);
1144 switch (b) {
1145 case SCC_ENCODED:
1147 DecodeEncodedString(consumer, b == SCC_ENCODED, builder);
1148 break;
1149
1150 case SCC_NEWGRF_STRINL: {
1151 StringID substr = consumer.ReadUtf8(STR_NULL);
1152 std::string_view ptr = GetStringPtr(substr);
1153 str_stack.emplace(ptr, args.GetOffset(), next_substr_case_index); // this may invalidate "consumer"
1154 next_substr_case_index = 0;
1155 break;
1156 }
1157
1159 StringID substr = args.GetNextParameter<StringID>();
1160 std::string_view ptr = GetStringPtr(substr);
1161 str_stack.emplace(ptr, args.GetOffset(), next_substr_case_index); // this may invalidate "consumer"
1162 next_substr_case_index = 0;
1163 break;
1164 }
1165
1166 case SCC_GENDER_LIST: { // {G 0 Der Die Das}
1167 /* First read the meta data from the language file. */
1168 size_t offset = ref_param_offset + consumer.ReadUint8();
1169 uint8_t gender = 0;
1170 if (offset >= args.GetNumParameters()) {
1171 /* The offset may come from an external NewGRF, and be invalid. */
1172 builder += "(invalid GENDER parameter)";
1173 } else if (!dry_run && args.GetTypeAtOffset(offset) != 0) {
1174 /* Now we need to figure out what text to resolve, i.e.
1175 * what do we need to draw? So get the actual raw string
1176 * first using the control code to get said string. */
1177 std::string input;
1178 {
1179 StringBuilder tmp_builder(input);
1180 tmp_builder.PutUtf8(args.GetTypeAtOffset(offset));
1181 }
1182
1183 std::string buffer;
1184 {
1185 AutoRestoreBackup sgd_backup(_scan_for_gender_data, true);
1186 StringBuilder tmp_builder(buffer);
1187 StringParameters tmp_params = args.GetRemainingParameters(offset);
1188 FormatString(tmp_builder, input, tmp_params);
1189 }
1190
1191 /* The gender is stored at the start of the formatted string.
1192 * Does this string have a gender, if so, set it. */
1193 StringConsumer gender_consumer(buffer);
1194 if (gender_consumer.ReadUtf8If(SCC_GENDER_INDEX)) {
1195 gender = gender_consumer.ReadUint8();
1196 }
1197 }
1198 ParseStringChoice(consumer, gender, builder);
1199 break;
1200 }
1201
1202 /* This sets up the gender for the string.
1203 * We just ignore this one. It's used in {G 0 Der Die Das} to determine the case. */
1204 case SCC_GENDER_INDEX: { // {GENDER 0}
1205 uint8_t gender = consumer.ReadUint8();
1207 builder.PutUtf8(SCC_GENDER_INDEX);
1208 builder.PutUint8(gender);
1209 }
1210 break;
1211 }
1212
1213 case SCC_PLURAL_LIST: { // {P}
1214 uint8_t plural_form = consumer.ReadUint8(); // contains the plural form for this string
1215 size_t offset = ref_param_offset + consumer.ReadUint8();
1216 const uint64_t *v = nullptr;
1217 /* The offset may come from an external NewGRF, and be invalid. */
1218 if (offset < args.GetNumParameters()) {
1219 v = std::get_if<uint64_t>(&args.GetParam(offset)); // contains the number that determines plural
1220 }
1221 if (v != nullptr) {
1222 ParseStringChoice(consumer, DeterminePluralForm(static_cast<int64_t>(*v), plural_form), builder);
1223 } else {
1224 SkipStringChoice(consumer);
1225 builder += "(invalid PLURAL parameter)";
1226 }
1227 break;
1228 }
1229
1230 case SCC_ARG_INDEX: { // Move argument pointer
1231 args.SetOffset(ref_param_offset + consumer.ReadUint8());
1232 break;
1233 }
1234
1235 case SCC_SET_CASE: { // {SET_CASE}
1236 /* This is a pseudo command, it's outputted when someone does {STRING.ack}
1237 * The modifier is added to all subsequent GetStringWithArgs that accept the modifier. */
1238 next_substr_case_index = consumer.ReadUint8();
1239 break;
1240 }
1241
1242 case SCC_SWITCH_CASE: { // {Used to implement case switching}
1243 /* <0x9E> <NUM CASES> <CASE1> <LEN1> <STRING1> <CASE2> <LEN2> <STRING2> <CASE3> <LEN3> <STRING3> <LENDEFAULT> <STRINGDEFAULT>
1244 * Each LEN is printed using 2 bytes in little endian order. */
1245 uint num = consumer.ReadUint8();
1246 std::optional<std::string_view> found;
1247 for (; num > 0; --num) {
1248 uint8_t index = consumer.ReadUint8();
1249 uint16_t len = consumer.ReadUint16LE();
1250 auto case_str = consumer.Read(len);
1251 if (index == case_index) {
1252 /* Found the case */
1253 found = case_str;
1254 }
1255 }
1256 uint16_t default_len = consumer.ReadUint16LE();
1257 auto default_str = consumer.Read(default_len);
1258 if (!found.has_value()) found = default_str;
1259 str_stack.emplace(*found, ref_param_offset, case_index); // this may invalidate "consumer"
1260 break;
1261 }
1262
1263 case SCC_REVISION: // {REV}
1264 builder += _openttd_revision;
1265 break;
1266
1267 case SCC_RAW_STRING_POINTER: // {RAW_STRING}
1268 FormatString(builder, args.GetNextParameterString(), args);
1269 break;
1270
1271 case SCC_STRING: {// {STRING}
1272 StringID string_id = args.GetNextParameter<StringID>();
1273 if (game_script && GetStringTab(string_id) != TEXT_TAB_GAMESCRIPT_START) break;
1274 /* It's prohibited for the included string to consume any arguments. */
1275 StringParameters tmp_params(args, game_script ? args.GetDataLeft() : 0);
1276 GetStringWithArgs(builder, string_id, tmp_params, next_substr_case_index, game_script);
1277 next_substr_case_index = 0;
1278 break;
1279 }
1280
1281 case SCC_STRING1:
1282 case SCC_STRING2:
1283 case SCC_STRING3:
1284 case SCC_STRING4:
1285 case SCC_STRING5:
1286 case SCC_STRING6:
1287 case SCC_STRING7: { // {STRING1..7}
1288 /* Strings that consume arguments */
1289 StringID string_id = args.GetNextParameter<StringID>();
1290 if (game_script && GetStringTab(string_id) != TEXT_TAB_GAMESCRIPT_START) break;
1291 uint size = b - SCC_STRING1 + 1;
1292 if (size > args.GetDataLeft()) {
1293 builder += "(consumed too many parameters)";
1294 } else {
1295 StringParameters sub_args(args, game_script ? args.GetDataLeft() : size);
1296 GetStringWithArgs(builder, string_id, sub_args, next_substr_case_index, game_script);
1297 args.AdvanceOffset(size);
1298 }
1299 next_substr_case_index = 0;
1300 break;
1301 }
1302
1303 case SCC_COMMA: // {COMMA}
1304 FormatCommaNumber(builder, args.GetNextParameter<int64_t>());
1305 break;
1306
1307 case SCC_DECIMAL: { // {DECIMAL}
1308 int64_t number = args.GetNextParameter<int64_t>();
1309 int digits = args.GetNextParameter<int>();
1310 if (digits == 0) {
1311 FormatCommaNumber(builder, number);
1312 break;
1313 }
1314
1315 int64_t divisor = PowerOfTen(digits);
1316 int64_t fractional = number % divisor;
1317 number /= divisor;
1318 FormatCommaNumber(builder, number);
1319 fmt::format_to(builder.back_inserter(), "{}{:0{}d}", GetDecimalSeparator(), fractional, digits);
1320 break;
1321 }
1322
1323 case SCC_NUM: // {NUM}
1324 FormatNoCommaNumber(builder, args.GetNextParameter<int64_t>());
1325 break;
1326
1327 case SCC_ZEROFILL_NUM: { // {ZEROFILL_NUM}
1328 int64_t num = args.GetNextParameter<int64_t>();
1329 FormatZerofillNumber(builder, num, args.GetNextParameter<int>());
1330 break;
1331 }
1332
1333 case SCC_HEX: // {HEX}
1334 FormatHexNumber(builder, args.GetNextParameter<uint64_t>());
1335 break;
1336
1337 case SCC_BYTES: // {BYTES}
1338 FormatBytes(builder, args.GetNextParameter<int64_t>());
1339 break;
1340
1341 case SCC_CARGO_TINY: { // {CARGO_TINY}
1342 /* Tiny description of cargotypes. Layout:
1343 * param 1: cargo type
1344 * param 2: cargo count */
1345 CargoType cargo = args.GetNextParameter<CargoType>();
1346 int64_t amount = args.GetNextParameter<int64_t>();
1347
1348 if (cargo >= CargoSpec::GetArraySize()) {
1349 builder += "(invalid cargo type)";
1350 break;
1351 }
1352
1353 switch (CargoSpec::Get(cargo)->units_volume) {
1354 case STR_TONS:
1356 break;
1357
1358 case STR_LITERS:
1360 break;
1361
1362 default:
1363 break;
1364 }
1365
1366 FormatCommaNumber(builder, amount);
1367 break;
1368 }
1369
1370 case SCC_CARGO_SHORT: { // {CARGO_SHORT}
1371 /* Short description of cargotypes. Layout:
1372 * param 1: cargo type
1373 * param 2: cargo count */
1374 CargoType cargo = args.GetNextParameter<CargoType>();
1375 int64_t amount = args.GetNextParameter<int64_t>();
1376
1377 if (cargo >= CargoSpec::GetArraySize()) {
1378 builder += "(invalid cargo type)";
1379 break;
1380 }
1381
1382 StringID cargo_str = CargoSpec::Get(cargo)->units_volume;
1383 switch (cargo_str) {
1384 case STR_TONS: {
1387 auto tmp_params = MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1388 FormatString(builder, GetStringPtr(x.l), tmp_params);
1389 break;
1390 }
1391
1392 case STR_LITERS: {
1395 auto tmp_params = MakeParameters(x.c.ToDisplay(amount), x.decimal_places);
1396 FormatString(builder, GetStringPtr(x.l), tmp_params);
1397 break;
1398 }
1399
1400 default: {
1401 auto tmp_params = MakeParameters(amount);
1402 GetStringWithArgs(builder, cargo_str, tmp_params);
1403 break;
1404 }
1405 }
1406 break;
1407 }
1408
1409 case SCC_CARGO_LONG: { // {CARGO_LONG}
1410 /* First parameter is cargo type, second parameter is cargo count */
1411 CargoType cargo = args.GetNextParameter<CargoType>();
1412 int64_t amount = args.GetNextParameter<int64_t>();
1413 if (cargo < CargoSpec::GetArraySize()) {
1414 auto tmp_args = MakeParameters(amount);
1415 GetStringWithArgs(builder, CargoSpec::Get(cargo)->quantifier, tmp_args);
1416 } else if (!IsValidCargoType(cargo)) {
1417 GetStringWithArgs(builder, STR_QUANTITY_N_A, {});
1418 } else {
1419 builder += "(invalid cargo type)";
1420 }
1421 break;
1422 }
1423
1424 case SCC_CARGO_LIST: { // {CARGO_LIST}
1425 CargoTypes cmask = args.GetNextParameter<CargoTypes>();
1426 bool first = true;
1427
1428 std::string_view list_separator = GetListSeparator();
1429 for (const auto &cs : _sorted_cargo_specs) {
1430 if (!HasBit(cmask, cs->Index())) continue;
1431
1432 if (first) {
1433 first = false;
1434 } else {
1435 /* Add a comma if this is not the first item */
1436 builder += list_separator;
1437 }
1438
1439 GetStringWithArgs(builder, cs->name, args, next_substr_case_index, game_script);
1440 }
1441
1442 /* If first is still true then no cargo is accepted */
1443 if (first) GetStringWithArgs(builder, STR_JUST_NOTHING, args, next_substr_case_index, game_script);
1444
1445 next_substr_case_index = 0;
1446 break;
1447 }
1448
1449 case SCC_CURRENCY_SHORT: // {CURRENCY_SHORT}
1450 FormatGenericCurrency(builder, &GetCurrency(), args.GetNextParameter<int64_t>(), true);
1451 break;
1452
1453 case SCC_CURRENCY_LONG: // {CURRENCY_LONG}
1454 FormatGenericCurrency(builder, &GetCurrency(), args.GetNextParameter<int64_t>(), false);
1455 break;
1456
1457 case SCC_DATE_TINY: // {DATE_TINY}
1458 FormatTinyOrISODate(builder, args.GetNextParameter<TimerGameCalendar::Date>(), STR_FORMAT_DATE_TINY);
1459 break;
1460
1461 case SCC_DATE_SHORT: // {DATE_SHORT}
1462 FormatMonthAndYear(builder, args.GetNextParameter<TimerGameCalendar::Date>(), next_substr_case_index);
1463 next_substr_case_index = 0;
1464 break;
1465
1466 case SCC_DATE_LONG: // {DATE_LONG}
1467 FormatYmdString(builder, args.GetNextParameter<TimerGameCalendar::Date>(), next_substr_case_index);
1468 next_substr_case_index = 0;
1469 break;
1470
1471 case SCC_DATE_ISO: // {DATE_ISO}
1472 FormatTinyOrISODate(builder, args.GetNextParameter<TimerGameCalendar::Date>(), STR_FORMAT_DATE_ISO);
1473 break;
1474
1475 case SCC_FORCE: { // {FORCE}
1478 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1479 FormatString(builder, GetStringPtr(x.s), tmp_params);
1480 break;
1481 }
1482
1483 case SCC_HEIGHT: { // {HEIGHT}
1486 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1487 FormatString(builder, GetStringPtr(x.s), tmp_params);
1488 break;
1489 }
1490
1491 case SCC_POWER: { // {POWER}
1494 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1495 FormatString(builder, GetStringPtr(x.s), tmp_params);
1496 break;
1497 }
1498
1499 case SCC_POWER_TO_WEIGHT: { // {POWER_TO_WEIGHT}
1501 assert(setting < lengthof(_units_power_to_weight));
1502 const auto &x = _units_power_to_weight[setting];
1503 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1504 FormatString(builder, GetStringPtr(x.s), tmp_params);
1505 break;
1506 }
1507
1508 case SCC_VELOCITY: { // {VELOCITY}
1509 int64_t arg = args.GetNextParameter<int64_t>();
1510 /* Unpack vehicle type from packed argument to get desired units. */
1511 VehicleType vt = static_cast<VehicleType>(GB(arg, 56, 8));
1512 const auto &x = GetVelocityUnits(vt);
1513 auto tmp_params = MakeParameters(ConvertKmhishSpeedToDisplaySpeed(GB(arg, 0, 56), vt), x.decimal_places);
1514 FormatString(builder, GetStringPtr(x.s), tmp_params);
1515 break;
1516 }
1517
1518 case SCC_VOLUME_SHORT: { // {VOLUME_SHORT}
1521 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1522 FormatString(builder, GetStringPtr(x.s), tmp_params);
1523 break;
1524 }
1525
1526 case SCC_VOLUME_LONG: { // {VOLUME_LONG}
1529 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1530 FormatString(builder, GetStringPtr(x.l), tmp_params);
1531 break;
1532 }
1533
1534 case SCC_WEIGHT_SHORT: { // {WEIGHT_SHORT}
1537 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1538 FormatString(builder, GetStringPtr(x.s), tmp_params);
1539 break;
1540 }
1541
1542 case SCC_WEIGHT_LONG: { // {WEIGHT_LONG}
1545 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1546 FormatString(builder, GetStringPtr(x.l), tmp_params);
1547 break;
1548 }
1549
1550 case SCC_UNITS_DAYS_OR_SECONDS: { // {UNITS_DAYS_OR_SECONDS}
1551 uint8_t realtime = TimerGameEconomy::UsingWallclockUnits(_game_mode == GM_MENU);
1552 const auto &x = _units_time_days_or_seconds[realtime];
1553 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1554 FormatString(builder, GetStringPtr(x.s), tmp_params);
1555 break;
1556 }
1557
1558 case SCC_UNITS_MONTHS_OR_MINUTES: { // {UNITS_MONTHS_OR_MINUTES}
1559 uint8_t realtime = TimerGameEconomy::UsingWallclockUnits(_game_mode == GM_MENU);
1560 const auto &x = _units_time_months_or_minutes[realtime];
1561 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1562 FormatString(builder, GetStringPtr(x.s), tmp_params);
1563 break;
1564 }
1565
1566 case SCC_UNITS_YEARS_OR_PERIODS: { // {UNITS_YEARS_OR_PERIODS}
1567 uint8_t realtime = TimerGameEconomy::UsingWallclockUnits(_game_mode == GM_MENU);
1568 const auto &x = _units_time_years_or_periods[realtime];
1569 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1570 FormatString(builder, GetStringPtr(x.s), tmp_params);
1571 break;
1572 }
1573
1574 case SCC_UNITS_YEARS_OR_MINUTES: { // {UNITS_YEARS_OR_MINUTES}
1575 uint8_t realtime = TimerGameEconomy::UsingWallclockUnits(_game_mode == GM_MENU);
1576 const auto &x = _units_time_years_or_minutes[realtime];
1577 auto tmp_params = MakeParameters(x.c.ToDisplay(args.GetNextParameter<int64_t>()), x.decimal_places);
1578 FormatString(builder, GetStringPtr(x.s), tmp_params);
1579 break;
1580 }
1581
1582 case SCC_COMPANY_NAME: { // {COMPANY}
1584 if (c == nullptr) break;
1585
1586 if (!c->name.empty()) {
1587 auto tmp_params = MakeParameters(c->name);
1588 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1589 } else {
1590 auto tmp_params = MakeParameters(c->name_2);
1591 GetStringWithArgs(builder, c->name_1, tmp_params);
1592 }
1593 break;
1594 }
1595
1596 case SCC_COMPANY_NUM: { // {COMPANY_NUM}
1597 CompanyID company = args.GetNextParameter<CompanyID>();
1598
1599 /* Nothing is added for AI or inactive companies */
1600 if (Company::IsValidHumanID(company)) {
1601 auto tmp_params = MakeParameters(company + 1);
1602 GetStringWithArgs(builder, STR_FORMAT_COMPANY_NUM, tmp_params);
1603 }
1604 break;
1605 }
1606
1607 case SCC_DEPOT_NAME: { // {DEPOT}
1609 if (vt == VEH_AIRCRAFT) {
1610 auto tmp_params = MakeParameters(args.GetNextParameter<StationID>());
1611 GetStringWithArgs(builder, STR_FORMAT_DEPOT_NAME_AIRCRAFT, tmp_params);
1612 break;
1613 }
1614
1615 const Depot *d = Depot::Get(args.GetNextParameter<DepotID>());
1616 if (!d->name.empty()) {
1617 auto tmp_params = MakeParameters(d->name);
1618 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1619 } else {
1620 auto tmp_params = MakeParameters(d->town->index, d->town_cn + 1);
1621 GetStringWithArgs(builder, STR_FORMAT_DEPOT_NAME_TRAIN + 2 * vt + (d->town_cn == 0 ? 0 : 1), tmp_params);
1622 }
1623 break;
1624 }
1625
1626 case SCC_ENGINE_NAME: { // {ENGINE}
1627 int64_t arg = args.GetNextParameter<int64_t>();
1628 const Engine *e = Engine::GetIfValid(static_cast<EngineID>(arg));
1629 if (e == nullptr) break;
1630
1631 if (!e->name.empty() && e->IsEnabled()) {
1632 auto tmp_params = MakeParameters(e->name);
1633 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1634 break;
1635 }
1636
1637 if (e->info.callback_mask.Test(VehicleCallbackMask::Name)) {
1638 std::array<int32_t, 16> regs100;
1639 uint16_t callback = GetVehicleCallback(CBID_VEHICLE_NAME, static_cast<uint32_t>(arg >> 32), 0, e->index, nullptr, regs100);
1640 /* Not calling ErrorUnknownCallbackResult due to being inside string processing. */
1641 if (callback == 0x40F) {
1642 const GRFFile *grffile = e->GetGRF();
1643 assert(grffile != nullptr);
1644
1645 builder += GetGRFStringWithTextStack(grffile, static_cast<GRFStringID>(regs100[0]), std::span{regs100}.subspan(1));
1646 break;
1647 } else if (callback < 0x400) {
1648 const GRFFile *grffile = e->GetGRF();
1649 assert(grffile != nullptr);
1650
1651 builder += GetGRFStringWithTextStack(grffile, GRFSTR_MISC_GRF_TEXT + callback, regs100);
1652 break;
1653 }
1654 }
1655
1656 GetStringWithArgs(builder, e->info.string_id, {});
1657 break;
1658 }
1659
1660 case SCC_GROUP_NAME: { // {GROUP}
1661 const Group *g = Group::GetIfValid(args.GetNextParameter<GroupID>());
1662 if (g == nullptr) break;
1663
1664 if (!g->name.empty()) {
1665 auto tmp_params = MakeParameters(g->name);
1666 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1667 } else {
1668 auto tmp_params = MakeParameters(g->number);
1669 GetStringWithArgs(builder, STR_FORMAT_GROUP_NAME, tmp_params);
1670 }
1671 break;
1672 }
1673
1674 case SCC_INDUSTRY_NAME: { // {INDUSTRY}
1676 if (i == nullptr) break;
1677
1678 static bool use_cache = true;
1680 /* Gender is defined by the industry type.
1681 * STR_FORMAT_INDUSTRY_NAME may have the town first, so it would result in the gender of the town name */
1682 FormatString(builder, GetStringPtr(GetIndustrySpec(i->type)->name), {}, next_substr_case_index);
1683 } else if (use_cache) { // Use cached version if first call
1684 AutoRestoreBackup cache_backup(use_cache, false);
1685 builder += i->GetCachedName();
1686 } else {
1687 /* First print the town name and the industry type name. */
1688 auto tmp_params = MakeParameters(i->town->index, GetIndustrySpec(i->type)->name);
1689 FormatString(builder, GetStringPtr(STR_FORMAT_INDUSTRY_NAME), tmp_params, next_substr_case_index);
1690 }
1691 next_substr_case_index = 0;
1692 break;
1693 }
1694
1695 case SCC_PRESIDENT_NAME: { // {PRESIDENT_NAME}
1697 if (c == nullptr) break;
1698
1699 if (!c->president_name.empty()) {
1700 auto tmp_params = MakeParameters(c->president_name);
1701 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1702 } else {
1703 auto tmp_params = MakeParameters(c->president_name_2);
1704 GetStringWithArgs(builder, c->president_name_1, tmp_params);
1705 }
1706 break;
1707 }
1708
1709 case SCC_STATION_NAME: { // {STATION}
1710 StationID sid = args.GetNextParameter<StationID>();
1711 const Station *st = Station::GetIfValid(sid);
1712
1713 if (st == nullptr) {
1714 /* The station doesn't exist anymore. The only place where we might
1715 * be "drawing" an invalid station is in the case of cargo that is
1716 * in transit. */
1717 GetStringWithArgs(builder, STR_UNKNOWN_STATION, {});
1718 break;
1719 }
1720
1721 static bool use_cache = true;
1722 if (use_cache) { // Use cached version if first call
1723 AutoRestoreBackup cache_backup(use_cache, false);
1724 builder += st->GetCachedName();
1725 } else if (!st->name.empty()) {
1726 auto tmp_params = MakeParameters(st->name);
1727 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1728 } else {
1729 StringID string_id = st->string_id;
1730 if (st->indtype != IT_INVALID) {
1731 /* Special case where the industry provides the name for the station */
1732 const IndustrySpec *indsp = GetIndustrySpec(st->indtype);
1733
1734 /* Industry GRFs can change which might remove the station name and
1735 * thus cause very strange things. Here we check for that before we
1736 * actually set the station name. */
1737 if (indsp->station_name != STR_NULL && indsp->station_name != STR_UNDEFINED) {
1738 string_id = indsp->station_name;
1739 }
1740 }
1741
1742 auto tmp_params = MakeParameters(STR_TOWN_NAME, st->town->index, st->index);
1743 GetStringWithArgs(builder, string_id, tmp_params);
1744 }
1745 break;
1746 }
1747
1748 case SCC_TOWN_NAME: { // {TOWN}
1749 const Town *t = Town::GetIfValid(args.GetNextParameter<TownID>());
1750 if (t == nullptr) break;
1751
1752 static bool use_cache = true;
1753 if (use_cache) { // Use cached version if first call
1754 AutoRestoreBackup cache_backup(use_cache, false);
1755 builder += t->GetCachedName();
1756 } else if (!t->name.empty()) {
1757 auto tmp_params = MakeParameters(t->name);
1758 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1759 } else {
1760 GetTownName(builder, t);
1761 }
1762 break;
1763 }
1764
1765 case SCC_WAYPOINT_NAME: { // {WAYPOINT}
1767 if (wp == nullptr) break;
1768
1769 if (!wp->name.empty()) {
1770 auto tmp_params = MakeParameters(wp->name);
1771 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1772 } else {
1773 auto tmp_params = MakeParameters(wp->town->index, wp->town_cn + 1);
1774 StringID string_id = ((wp->string_id == STR_SV_STNAME_BUOY) ? STR_FORMAT_BUOY_NAME : STR_FORMAT_WAYPOINT_NAME);
1775 if (wp->town_cn != 0) string_id++;
1776 GetStringWithArgs(builder, string_id, tmp_params);
1777 }
1778 break;
1779 }
1780
1781 case SCC_VEHICLE_NAME: { // {VEHICLE}
1783 if (v == nullptr) break;
1784
1785 if (!v->name.empty()) {
1786 auto tmp_params = MakeParameters(v->name);
1787 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1788 } else if (v->group_id != DEFAULT_GROUP) {
1789 /* The vehicle has no name, but is member of a group, so print group name */
1790 auto tmp_params = MakeParameters(v->group_id, v->unitnumber);
1791 GetStringWithArgs(builder, STR_FORMAT_GROUP_VEHICLE_NAME, tmp_params);
1792 } else {
1793 auto tmp_params = MakeParameters(v->unitnumber);
1794
1795 StringID string_id;
1796 switch (v->type) {
1797 default: string_id = STR_INVALID_VEHICLE; break;
1798 case VEH_TRAIN: string_id = STR_SV_TRAIN_NAME; break;
1799 case VEH_ROAD: string_id = STR_SV_ROAD_VEHICLE_NAME; break;
1800 case VEH_SHIP: string_id = STR_SV_SHIP_NAME; break;
1801 case VEH_AIRCRAFT: string_id = STR_SV_AIRCRAFT_NAME; break;
1802 }
1803
1804 GetStringWithArgs(builder, string_id, tmp_params);
1805 }
1806 break;
1807 }
1808
1809 case SCC_SIGN_NAME: { // {SIGN}
1810 const Sign *si = Sign::GetIfValid(args.GetNextParameter<SignID>());
1811 if (si == nullptr) break;
1812
1813 if (!si->name.empty()) {
1814 auto tmp_params = MakeParameters(si->name);
1815 GetStringWithArgs(builder, STR_JUST_RAW_STRING, tmp_params);
1816 } else {
1817 GetStringWithArgs(builder, STR_DEFAULT_SIGN_NAME, {});
1818 }
1819 break;
1820 }
1821
1822 case SCC_STATION_FEATURES: { // {STATIONFEATURES}
1823 StationGetSpecialString(builder, args.GetNextParameter<StationFacilities>());
1824 break;
1825 }
1826
1827 case SCC_COLOUR: { // {COLOUR}
1828 StringControlCode scc = (StringControlCode)(SCC_BLUE + args.GetNextParameter<Colours>());
1829 if (IsInsideMM(scc, SCC_BLUE, SCC_COLOUR)) builder.PutUtf8(scc);
1830 break;
1831 }
1832
1833 default:
1834 builder.PutUtf8(b);
1835 break;
1836 }
1837 } catch (std::out_of_range &e) {
1838 Debug(misc, 0, "FormatString: {}", e.what());
1839 builder += "(invalid parameter)";
1840 }
1841 }
1842}
1843
1844
1845static void StationGetSpecialString(StringBuilder &builder, StationFacilities x)
1846{
1847 if (x.Test(StationFacility::Train)) builder.PutUtf8(SCC_TRAIN);
1848 if (x.Test(StationFacility::TruckStop)) builder.PutUtf8(SCC_LORRY);
1849 if (x.Test(StationFacility::BusStop)) builder.PutUtf8(SCC_BUS);
1850 if (x.Test(StationFacility::Dock)) builder.PutUtf8(SCC_SHIP);
1851 if (x.Test(StationFacility::Airport)) builder.PutUtf8(SCC_PLANE);
1852}
1853
1854static const std::string_view _silly_company_names[] = {
1855 "Bloggs Brothers",
1856 "Tiny Transport Ltd.",
1857 "Express Travel",
1858 "Comfy-Coach & Co.",
1859 "Crush & Bump Ltd.",
1860 "Broken & Late Ltd.",
1861 "Sam Speedy & Son",
1862 "Supersonic Travel",
1863 "Mike's Motors",
1864 "Lightning International",
1865 "Pannik & Loozit Ltd.",
1866 "Inter-City Transport",
1867 "Getout & Pushit Ltd."
1868};
1869
1870static const std::string_view _surname_list[] = {
1871 "Adams",
1872 "Allan",
1873 "Baker",
1874 "Bigwig",
1875 "Black",
1876 "Bloggs",
1877 "Brown",
1878 "Campbell",
1879 "Gordon",
1880 "Hamilton",
1881 "Hawthorn",
1882 "Higgins",
1883 "Green",
1884 "Gribble",
1885 "Jones",
1886 "McAlpine",
1887 "MacDonald",
1888 "McIntosh",
1889 "Muir",
1890 "Murphy",
1891 "Nelson",
1892 "O'Donnell",
1893 "Parker",
1894 "Phillips",
1895 "Pilkington",
1896 "Quigley",
1897 "Sharkey",
1898 "Thomson",
1899 "Watkins"
1900};
1901
1902static const std::string_view _silly_surname_list[] = {
1903 "Grumpy",
1904 "Dozy",
1905 "Speedy",
1906 "Nosey",
1907 "Dribble",
1908 "Mushroom",
1909 "Cabbage",
1910 "Sniffle",
1911 "Fishy",
1912 "Swindle",
1913 "Sneaky",
1914 "Nutkins"
1915};
1916
1917static const char _initial_name_letters[] = {
1918 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',
1919 'K', 'L', 'M', 'N', 'P', 'R', 'S', 'T', 'W',
1920};
1921
1922static std::span<const std::string_view> GetSurnameOptions()
1923{
1924 if (_settings_game.game_creation.landscape == LandscapeType::Toyland) return _silly_surname_list;
1925 return _surname_list;
1926}
1927
1933static std::string_view GetSurname(uint32_t seed)
1934{
1935 auto surname_options = GetSurnameOptions();
1936 return surname_options[surname_options.size() * GB(seed, 16, 8) >> 8];
1937}
1938
1939static void GenAndCoName(StringBuilder &builder, uint32_t seed)
1940{
1941 builder += GetSurname(seed);
1942 builder += " & Co.";
1943}
1944
1945static void GenPresidentName(StringBuilder &builder, uint32_t seed)
1946{
1947 builder.PutChar(_initial_name_letters[std::size(_initial_name_letters) * GB(seed, 0, 8) >> 8]);
1948 builder += ". ";
1949
1950 /* The second initial is optional. */
1951 size_t index = (std::size(_initial_name_letters) + 35) * GB(seed, 8, 8) >> 8;
1952 if (index < std::size(_initial_name_letters)) {
1953 builder.PutChar(_initial_name_letters[index]);
1954 builder += ". ";
1955 }
1956
1957 builder += GetSurname(seed);
1958}
1959
1960static bool GetSpecialNameString(StringBuilder &builder, StringID string, StringParameters &args)
1961{
1962 switch (string) {
1963 case SPECSTR_SILLY_NAME: // Not used in new companies, but retained for old-loader savegames
1964 builder += _silly_company_names[std::min<size_t>(args.GetNextParameter<uint16_t>(), std::size(_silly_company_names) - 1)];
1965 return true;
1966
1967 case SPECSTR_ANDCO_NAME: // used for Foobar & Co company names
1968 GenAndCoName(builder, args.GetNextParameter<uint32_t>());
1969 return true;
1970
1971 case SPECSTR_PRESIDENT_NAME: // President name
1972 GenPresidentName(builder, args.GetNextParameter<uint32_t>());
1973 return true;
1974 }
1975
1976 /* TownName Transport company names, with the appropriate town name. */
1977 if (IsInsideMM(string, SPECSTR_COMPANY_NAME_START, SPECSTR_COMPANY_NAME_END)) {
1978 GenerateTownNameString(builder, string - SPECSTR_COMPANY_NAME_START, args.GetNextParameter<uint32_t>());
1979 builder += " Transport";
1980 return true;
1981 }
1982
1983 return false;
1984}
1985
1991{
1992 return this->ident == TO_LE32(LanguagePackHeader::IDENT) &&
1993 this->version == TO_LE32(LANGUAGE_PACK_VERSION) &&
1994 this->plural_form < LANGUAGE_MAX_PLURAL &&
1995 this->text_dir <= 1 &&
1996 this->newgrflangid < MAX_LANG &&
1997 this->num_genders < MAX_NUM_GENDERS &&
1998 this->num_cases < MAX_NUM_CASES &&
1999 StrValid(this->name) &&
2000 StrValid(this->own_name) &&
2001 StrValid(this->isocode) &&
2005}
2006
2011{
2012 /* "Less than 25% missing" is "sufficiently finished". */
2013 return 4 * this->missing < LANGUAGE_TOTAL_STRINGS;
2014}
2015
2022{
2023 /* Current language pack */
2024 size_t total_len = 0;
2025 std::unique_ptr<LanguagePack, LanguagePackDeleter> lang_pack(reinterpret_cast<LanguagePack *>(ReadFileToMem(FS2OTTD(lang->file.native()), total_len, 1U << 20).release()));
2026 if (!lang_pack) return false;
2027
2028 /* End of read data (+ terminating zero added in ReadFileToMem()) */
2029 const char *end = (char *)lang_pack.get() + total_len + 1;
2030
2031 /* We need at least one byte of lang_pack->data */
2032 if (end <= lang_pack->data || !lang_pack->IsValid()) {
2033 return false;
2034 }
2035
2036 std::array<uint, TEXT_TAB_END> tab_start, tab_num;
2037
2038 uint count = 0;
2039 for (uint i = 0; i < TEXT_TAB_END; i++) {
2040 uint16_t num = FROM_LE16(lang_pack->offsets[i]);
2041 if (num > TAB_SIZE) return false;
2042
2043 tab_start[i] = count;
2044 tab_num[i] = num;
2045 count += num;
2046 }
2047
2048 /* Allocate offsets */
2049 std::vector<std::string_view> strings;
2050
2051 /* Fill offsets */
2052 char *s = lang_pack->data;
2053 for (uint i = 0; i < count; i++) {
2054 size_t len = static_cast<uint8_t>(*s++);
2055 if (s + len >= end) return false;
2056
2057 if (len >= 0xC0) {
2058 len = ((len & 0x3F) << 8) + static_cast<uint8_t>(*s++);
2059 if (s + len >= end) return false;
2060 }
2061 strings.emplace_back(s, len);
2062 s += len;
2063 }
2064 assert(strings.size() == count);
2065
2066 _langpack.langpack = std::move(lang_pack);
2067 _langpack.strings = std::move(strings);
2068 _langpack.langtab_num = tab_num;
2069 _langpack.langtab_start = tab_start;
2070
2071 _current_language = lang;
2073 _config_language_file = FS2OTTD(_current_language->file.filename().native());
2075 _langpack.list_separator = GetString(STR_LIST_SEPARATOR);
2076 _langpack.ellipsis = GetString(STR_TRUNCATION_ELLIPSIS);
2077
2078#ifdef _WIN32
2079 extern void Win32SetCurrentLocaleName(std::string iso_code);
2080 Win32SetCurrentLocaleName(_current_language->isocode);
2081#endif
2082
2083#ifdef WITH_COCOA
2084 extern void MacOSSetCurrentLocaleName(std::string_view iso_code);
2086#endif
2087
2088#ifdef WITH_ICU_I18N
2089 /* Create a collator instance for our current locale. */
2090 UErrorCode status = U_ZERO_ERROR;
2091 _current_collator.reset(icu::Collator::createInstance(icu::Locale(_current_language->isocode), status));
2092 /* Sort number substrings by their numerical value. */
2093 if (_current_collator) _current_collator->setAttribute(UCOL_NUMERIC_COLLATION, UCOL_ON, status);
2094 /* Avoid using the collator if it is not correctly set. */
2095 if (U_FAILURE(status)) {
2096 _current_collator.reset();
2097 }
2098#endif /* WITH_ICU_I18N */
2099
2101
2102 /* Some lists need to be sorted again after a language change. */
2108 InvalidateWindowClassesData(WC_BUILD_VEHICLE); // Build vehicle window.
2109 InvalidateWindowClassesData(WC_TRAINS_LIST); // Train group window.
2110 InvalidateWindowClassesData(WC_ROADVEH_LIST); // Road vehicle group window.
2111 InvalidateWindowClassesData(WC_SHIPS_LIST); // Ship group window.
2112 InvalidateWindowClassesData(WC_AIRCRAFT_LIST); // Aircraft group window.
2113 InvalidateWindowClassesData(WC_INDUSTRY_DIRECTORY); // Industry directory window.
2114 InvalidateWindowClassesData(WC_STATION_LIST); // Station list window.
2115
2116 return true;
2117}
2118
2119/* Win32 implementation in win32.cpp.
2120 * OS X implementation in os/macosx/macos.mm. */
2121#if !(defined(_WIN32) || defined(__APPLE__))
2130std::optional<std::string> GetCurrentLocale(const char *param)
2131{
2132 auto env = GetEnv("LANGUAGE");
2133 if (env.has_value()) return std::string{*env};
2134
2135 env = GetEnv("LC_ALL");
2136 if (env.has_value()) return std::string{*env};
2137
2138 if (param != nullptr) {
2139 env = GetEnv(param);
2140 if (env.has_value()) return std::string{*env};
2141 }
2142
2143 env = GetEnv("LANG");
2144 if (env.has_value()) return std::string{*env};
2145
2146 return std::nullopt;
2147}
2148#else
2149std::optional<std::string> GetCurrentLocale(const char *param);
2150#endif /* !(defined(_WIN32) || defined(__APPLE__)) */
2151
2157const LanguageMetadata *GetLanguage(uint8_t newgrflangid)
2158{
2159 for (const LanguageMetadata &lang : _languages) {
2160 if (newgrflangid == lang.newgrflangid) return &lang;
2161 }
2162
2163 return nullptr;
2164}
2165
2172static bool GetLanguageFileHeader(const std::string &file, LanguagePackHeader *hdr)
2173{
2174 auto f = FileHandle::Open(file, "rb");
2175 if (!f.has_value()) return false;
2176
2177 size_t read = fread(hdr, sizeof(*hdr), 1, *f);
2178
2179 bool ret = read == 1 && hdr->IsValid();
2180
2181 /* Convert endianness for the windows language ID */
2182 if (ret) {
2183 hdr->missing = FROM_LE16(hdr->missing);
2184 hdr->winlangid = FROM_LE16(hdr->winlangid);
2185 }
2186 return ret;
2187}
2188
2193static void FillLanguageList(const std::string &path)
2194{
2195 std::error_code error_code;
2196 for (const auto &dir_entry : std::filesystem::directory_iterator(OTTD2FS(path), error_code)) {
2197 if (!dir_entry.is_regular_file()) continue;
2198 if (dir_entry.path().extension() != ".lng") continue;
2199
2200 LanguageMetadata lmd;
2201 lmd.file = dir_entry.path();
2202
2203 /* Check whether the file is of the correct version */
2204 std::string file = FS2OTTD(lmd.file.native());
2205 if (!GetLanguageFileHeader(file, &lmd)) {
2206 Debug(misc, 3, "{} is not a valid language file", file);
2207 } else if (GetLanguage(lmd.newgrflangid) != nullptr) {
2208 Debug(misc, 3, "{}'s language ID is already known", file);
2209 } else {
2210 _languages.push_back(std::move(lmd));
2211 }
2212 }
2213 if (error_code) {
2214 Debug(misc, 9, "Unable to open directory {}: {}", path, error_code.message());
2215 }
2216}
2217
2223{
2224 for (Searchpath sp : _valid_searchpaths) {
2225 FillLanguageList(FioGetDirectory(sp, LANG_DIR));
2226 }
2227 if (_languages.empty()) UserError("No available language packs (invalid versions?)");
2228
2229 /* Acquire the locale of the current system */
2230 auto str_lang = GetCurrentLocale("LC_MESSAGES");
2231 std::string_view lang = str_lang.has_value() ? std::string_view{*str_lang} : "en_GB";
2232
2233 const LanguageMetadata *chosen_language = nullptr;
2234 const LanguageMetadata *language_fallback = nullptr;
2235 const LanguageMetadata *en_GB_fallback = _languages.data();
2236
2237 /* Find a proper language. */
2238 for (const LanguageMetadata &lng : _languages) {
2239 /* We are trying to find a default language. The priority is by
2240 * configuration file, local environment and last, if nothing found,
2241 * English. */
2242 if (_config_language_file == FS2OTTD(lng.file.filename().native())) {
2243 chosen_language = &lng;
2244 break;
2245 }
2246
2247 std::string_view iso_code = lng.isocode;
2248 if (iso_code == "en_GB") en_GB_fallback = &lng;
2249
2250 /* Only auto-pick finished translations */
2251 if (!lng.IsReasonablyFinished()) continue;
2252
2253 if (iso_code.starts_with(lang.substr(0, 5))) chosen_language = &lng;
2254 if (iso_code.starts_with(lang.substr(0, 2))) language_fallback = &lng;
2255 }
2256
2257 /* We haven't found the language in the config nor the one in the locale.
2258 * Now we set it to one of the fallback languages */
2259 if (chosen_language == nullptr) {
2260 chosen_language = (language_fallback != nullptr) ? language_fallback : en_GB_fallback;
2261 }
2262
2263 if (!ReadLanguagePack(chosen_language)) UserError("Can't read language pack '{}'", FS2OTTD(chosen_language->file.native()));
2264}
2265
2271{
2272 return _langpack.langpack->isocode;
2273}
2274
2280{
2281 InitFontCache(this->Monospace());
2282
2283 this->Reset();
2284 for (auto text = this->NextString(); text.has_value(); text = this->NextString()) {
2285 FontSize size = this->DefaultSize();
2286 FontCache *fc = FontCache::Get(size);
2287 for (char32_t c : Utf8View(*text)) {
2288 if (c >= SCC_FIRST_FONT && c <= SCC_LAST_FONT) {
2289 size = (FontSize)(c - SCC_FIRST_FONT);
2290 fc = FontCache::Get(size);
2291 } else if (!IsInsideMM(c, SCC_SPRITE_START, SCC_SPRITE_END) && IsPrintable(c) && !IsTextDirectionChar(c) && fc->MapCharToGlyph(c, false) == 0) {
2292 /* The character is printable, but not in the normal font. This is the case we were testing for. */
2293 Debug(fontcache, 0, "Font is missing glyphs to display char 0x{:X} in {} font size", static_cast<uint32_t>(c), FontSizeToName(size));
2294 return true;
2295 }
2296 }
2297 }
2298 return false;
2299}
2300
2303 uint i;
2304 uint j;
2305
2306 void Reset() override
2307 {
2308 this->i = 0;
2309 this->j = 0;
2310 }
2311
2313 {
2314 return FS_NORMAL;
2315 }
2316
2317 std::optional<std::string_view> NextString() override
2318 {
2319 if (this->i >= TEXT_TAB_END) return std::nullopt;
2320
2321 std::string_view ret = _langpack.strings[_langpack.langtab_start[this->i] + this->j];
2322
2323 this->j++;
2324 while (this->i < TEXT_TAB_END && this->j >= _langpack.langtab_num[this->i]) {
2325 this->i++;
2326 this->j = 0;
2327 }
2328
2329 return ret;
2330 }
2331
2332 bool Monospace() override
2333 {
2334 return false;
2335 }
2336
2337 void SetFontNames([[maybe_unused]] FontCacheSettings *settings, [[maybe_unused]] std::string_view font_name, [[maybe_unused]] const void *os_data) override
2338 {
2339#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2340 settings->small.font = font_name;
2341 settings->medium.font = font_name;
2342 settings->large.font = font_name;
2343
2344 settings->small.os_handle = os_data;
2345 settings->medium.os_handle = os_data;
2346 settings->large.os_handle = os_data;
2347#endif
2348 }
2349};
2350
2364void CheckForMissingGlyphs(bool base_font, MissingGlyphSearcher *searcher)
2365{
2366 static LanguagePackGlyphSearcher pack_searcher;
2367 if (searcher == nullptr) searcher = &pack_searcher;
2368 bool bad_font = !base_font || searcher->FindMissingGlyphs();
2369#if defined(WITH_FREETYPE) || defined(_WIN32) || defined(WITH_COCOA)
2370 if (bad_font) {
2371 /* We found an unprintable character... lets try whether we can find
2372 * a fallback font that can print the characters in the current language. */
2373 bool any_font_configured = !_fcsettings.medium.font.empty();
2374 FontCacheSettings backup = _fcsettings;
2375
2376 _fcsettings.mono.os_handle = nullptr;
2377 _fcsettings.medium.os_handle = nullptr;
2378
2379 bad_font = !SetFallbackFont(&_fcsettings, _langpack.langpack->isocode, searcher);
2380
2381 _fcsettings = std::move(backup);
2382
2383 if (!bad_font && any_font_configured) {
2384 /* If the user configured a bad font, and we found a better one,
2385 * show that we loaded the better font instead of the configured one.
2386 */
2387 std::string err_str;
2388 StringBuilder builder(err_str);
2389 builder.PutUtf8(SCC_YELLOW);
2390 builder.Put("The current font is missing some of the characters used in the texts for this language. Using system fallback font instead.");
2391 ShowErrorMessage(GetEncodedString(STR_JUST_RAW_STRING, std::move(err_str)), {}, WL_WARNING);
2392 }
2393
2394 if (bad_font && base_font) {
2395 /* Our fallback font does miss characters too, so keep the
2396 * user chosen font as that is more likely to be any good than
2397 * the wild guess we made */
2398 InitFontCache(searcher->Monospace());
2399 }
2400 }
2401#endif
2402
2403 if (bad_font) {
2404 /* All attempts have failed. Display an error. As we do not want the string to be translated by
2405 * the translators, we 'force' it into the binary and 'load' it via a BindCString. To do this
2406 * properly we have to set the colour of the string, otherwise we end up with a lot of artifacts.
2407 */
2408 std::string err_str;
2409 StringBuilder builder(err_str);
2410 builder.PutUtf8(SCC_YELLOW);
2411 builder.Put("The current font is missing some of the characters used in the texts for this language. Go to Help & Manuals > Fonts, or read the file docs/fonts.md in your OpenTTD directory, to see how to solve this.");
2412 ShowErrorMessage(GetEncodedString(STR_JUST_RAW_STRING, std::move(err_str)), {}, WL_WARNING);
2413
2414 /* Reset the font width */
2415 LoadStringWidthTable(searcher->Monospace());
2416 return;
2417 }
2418
2419 /* Update the font with cache */
2420 LoadStringWidthTable(searcher->Monospace());
2421
2422#if !(defined(WITH_ICU_I18N) && defined(WITH_HARFBUZZ)) && !defined(WITH_UNISCRIBE) && !defined(WITH_COCOA)
2423 /*
2424 * For right-to-left languages we need the ICU library. If
2425 * we do not have support for that library we warn the user
2426 * about it with a message. As we do not want the string to
2427 * be translated by the translators, we 'force' it into the
2428 * binary and 'load' it via a BindCString. To do this
2429 * properly we have to set the colour of the string,
2430 * otherwise we end up with a lot of artifacts.
2431 */
2432 if (_current_text_dir != TD_LTR) {
2433 std::string err_str;
2434 StringBuilder builder(err_str);
2435 builder.PutUtf8(SCC_YELLOW);
2436 builder.Put("This version of OpenTTD does not support right-to-left languages. Recompile with ICU + Harfbuzz enabled.");
2437 ShowErrorMessage(GetEncodedString(STR_JUST_RAW_STRING, std::move(err_str)), {}, WL_ERROR);
2438 }
2439#endif /* !(WITH_ICU_I18N && WITH_HARFBUZZ) && !WITH_UNISCRIBE && !WITH_COCOA */
2440}
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.
Definition cargo_type.h:23
bool IsValidCargoType(CargoType cargo)
Test whether cargo type is not INVALID_CARGO.
Definition cargo_type.h:106
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.
Definition strings.cpp:103
std::string GetDecodedString() const
Decode the encoded string.
Definition strings.cpp:208
EncodedString ReplaceParam(size_t param, StringParameter &&value) const
Replace a parameter of this EncodedString.
Definition strings.cpp:151
static std::optional< FileHandle > Open(const std::string &filename, std::string_view mode)
Open an RAII file handle if possible.
Definition fileio.cpp:1168
Font cache for basic fonts.
Definition fontcache.h:21
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.
Definition fontcache.h:134
Helper for searching through the language pack.
Definition strings.cpp:2302
bool Monospace() override
Whether to search for a monospace font or not.
Definition strings.cpp:2332
void SetFontNames(FontCacheSettings *settings, std::string_view font_name, const void *os_data) override
Set the right font names.
Definition strings.cpp:2337
uint j
Iterator for the secondary language tables.
Definition strings.cpp:2304
std::optional< std::string_view > NextString() override
Get the next string to search through.
Definition strings.cpp:2317
FontSize DefaultSize() override
Get the default (font) size of the string.
Definition strings.cpp:2312
uint i
Iterator for the primary language tables.
Definition strings.cpp:2303
void Reset() override
Reset the search, i.e.
Definition strings.cpp:2306
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.
Definition strings.cpp:2279
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.
Definition strings.cpp:69
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 utf8.hpp:30
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.
Definition utf8.cpp:48
Functions to handle different currencies.
const CurrencySpec & GetCurrency()
Get the currently selected currency.
Definition currency.h:118
Functions related to debugging.
#define Debug(category, level, format_string,...)
Output a line of debugging information.
Definition debug.h:37
Base for all depots (except hangars)
Function to handling different endian machines.
Base class for engines.
Functions related to errors.
@ WL_WARNING
Other information.
Definition error.h:25
@ WL_ERROR
Errors (eg. saving/loading failed)
Definition error.h:26
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.
Definition fileio.cpp:1023
Functions for Standard In/Out file operations.
Searchpath
Types of searchpaths OpenTTD might use.
@ LANG_DIR
Subdirectory for all translation files.
Definition fileio_type.h:98
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.
Definition font_osx.cpp:27
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.
Definition gfx.cpp:1292
void LoadStringWidthTable(bool monospace)
Initialize _stringwidth_table cache.
Definition gfx.cpp:1246
Functions related to laying out the texts.
FontSize
Available font sizes.
Definition gfx_type.h:250
@ FS_NORMAL
Index of the normal font in the font tables.
Definition gfx_type.h:251
static constexpr GroupID DEFAULT_GROUP
Ungrouped vehicles are in this group.
Definition group_type.h:18
Base of all industries.
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.
Definition language.h:20
std::vector< LanguageMetadata > LanguageList
Type for the list of language meta data.
Definition language.h:98
static const uint8_t MAX_NUM_CASES
Maximum number of supported cases.
Definition language.h:21
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.
Definition math_func.hpp:23
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.
@ Name
Engine name.
@ CBID_VEHICLE_NAME
Called to determine the engine name to show.
uint16_t GetVehicleCallback(CallbackID callback, uint32_t param1, uint32_t param2, EngineID engine, const Vehicle *v, std::span< int32_t > regs100)
Evaluate a newgrf callback for vehicles.
Functions for NewGRF engines.
std::string_view GetGRFStringPtr(StringIndexInTab stringid)
Get a C-string from a stringid set by a newgrf.
std::string GetGRFStringWithTextStack(const struct GRFFile *grffile, GRFStringID grfstringid, std::span< const int32_t > textstack)
Format a GRF string using the text ref stack for parameters.
void SetCurrentGrfLangID(uint8_t language_id)
Equivalence Setter function between game and newgrf langID.
char32_t RemapNewGRFStringControlCode(char32_t scc, StringConsumer &consumer)
Emit OpenTTD's internal string code for the different NewGRF string codes.
Header of Action 04 "universal holder" structure and functions.
static constexpr GRFStringID GRFSTR_MISC_GRF_TEXT
Miscellaneous GRF text range.
declaration of OTTD revision dependent variables
A number of safeguards to prevent using unsafe methods.
GameSettings _settings_game
Game settings of a running game or the scenario editor.
Definition settings.cpp:61
Base class for signs.
void BuildIndustriesLegend()
Fills an array for the industries legends.
Smallmap GUI functions.
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.
Definition stdafx.h:271
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),...
Definition string.cpp:857
bool StrValid(std::span< const char > str)
Checks whether the given string is valid, i.e.
Definition string.cpp:203
Parse strings.
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.
Definition string_type.h:16
uint64_t GetParamMaxValue(uint64_t max_value, uint min_count, FontSize size)
Get some number that is suitable for string size computations.
Definition strings.cpp:237
LanguageList _languages
The actual list of language meta data.
Definition strings.cpp:54
static void FormatNumber(StringBuilder &builder, int64_t number, std::string_view separator)
Format a number into a string.
Definition strings.cpp:483
EncodedString GetEncodedStringWithArgs(StringID str, std::span< const StringParameter > params)
Encode a string with its parameters into an encoded string.
Definition strings.cpp:103
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.
Definition strings.cpp:337
std::optional< std::string > GetCurrentLocale(const char *param)
Determine the current charset based on the environment First check some default values,...
Definition strings.cpp:2130
std::string_view GetListSeparator()
Get the list separator string for the current language.
Definition strings.cpp:300
static const Units _units_height[]
Unit conversions for height.
Definition strings.cpp:916
const LanguageMetadata * _current_language
The currently loaded language.
Definition strings.cpp:55
static void FormatBytes(StringBuilder &builder, int64_t number)
Format a given number as a number of bytes with the SI prefix.
Definition strings.cpp:538
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...
Definition strings.cpp:435
uint ConvertSpeedToDisplaySpeed(uint speed, VehicleType type)
Convert the given (internal) speed to the display speed.
Definition strings.cpp:968
void CheckForMissingGlyphs(bool base_font, MissingGlyphSearcher *searcher)
Check whether the currently loaded language pack uses characters that the currently loaded font does ...
Definition strings.cpp:2364
std::string _config_language_file
The file (name) stored in the configuration.
Definition strings.cpp:53
static const Units GetVelocityUnits(VehicleType type)
Get the correct velocity units depending on the vehicle type and whether we're using real-time units.
Definition strings.cpp:951
static int DeterminePluralForm(int64_t count, uint plural_form)
Determine the "plural" index given a plural form and a number.
Definition strings.cpp:657
EncodedString GetEncodedString(StringID str)
Encode a string with no parameters into an encoded string.
Definition strings.cpp:91
uint ConvertDisplaySpeedToKmhishSpeed(uint speed, VehicleType type)
Convert the given display speed to the km/h-ish speed.
Definition strings.cpp:1001
static const Units _units_power[]
Unit conversions for power.
Definition strings.cpp:875
static std::string_view GetSurname(uint32_t seed)
Get the surname of the president with the given seed.
Definition strings.cpp:1933
std::string GetString(StringID string)
Resolve the given StringID into a std::string with formatting but no parameters.
Definition strings.cpp:425
std::string_view GetCurrentLanguageIsoCode()
Get the ISO language code of the currently loaded language.
Definition strings.cpp:2270
static bool _scan_for_gender_data
Are we scanning for the gender of the current string? (instead of formatting it)
Definition strings.cpp:294
static const UnitsLong _units_volume[]
Unit conversions for volume.
Definition strings.cpp:902
static void DecodeEncodedString(StringConsumer &consumer, bool game_script, StringBuilder &builder)
Decodes an encoded string during FormatString.
Definition strings.cpp:1012
static const UnitsLong _units_weight[]
Unit conversions for weight.
Definition strings.cpp:895
static const Units _units_velocity_calendar[]
Unit conversions for velocity.
Definition strings.cpp:857
std::unique_ptr< icu::Collator > _current_collator
Collator for the language currently in use.
Definition strings.cpp:60
static void FillLanguageList(const std::string &path)
Search for the languages in the given directory and add them to the _languages list.
Definition strings.cpp:2193
uint ConvertKmhishSpeedToDisplaySpeed(uint speed, VehicleType type)
Convert the given km/h-ish speed to the display speed.
Definition strings.cpp:991
static const Units _units_force[]
Unit conversions for force.
Definition strings.cpp:909
void InitializeLanguagePacks()
Make a list of the available language packs.
Definition strings.cpp:2222
const LanguageMetadata * GetLanguage(uint8_t newgrflangid)
Get the language with the given NewGRF language ID.
Definition strings.cpp:2157
static const Units _units_time_days_or_seconds[]
Unit conversions for time in calendar days or wallclock seconds.
Definition strings.cpp:923
uint ConvertDisplaySpeedToSpeed(uint speed, VehicleType type)
Convert the given display speed to the (internal) speed.
Definition strings.cpp:981
TextDirection _current_text_dir
Text direction of the currently selected language.
Definition strings.cpp:57
static const Units _units_velocity_realtime[]
Unit conversions for velocity.
Definition strings.cpp:866
bool ReadLanguagePack(const LanguageMetadata *lang)
Read a particular language.
Definition strings.cpp:2021
static bool GetLanguageFileHeader(const std::string &file, LanguagePackHeader *hdr)
Reads the language file header and checks compatibility.
Definition strings.cpp:2172
static const Units _units_time_months_or_minutes[]
Unit conversions for time in calendar months or wallclock minutes.
Definition strings.cpp:929
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.
Definition strings.cpp:1083
static const Units _units_time_years_or_minutes[]
Unit conversions for time in calendar years or wallclock minutes.
Definition strings.cpp:941
static const Units _units_power_to_weight[]
Unit conversions for power to weight.
Definition strings.cpp:882
uint64_t GetParamMaxDigits(uint count, FontSize size)
Get some number that is suitable for string size computations.
Definition strings.cpp:219
std::string_view GetEllipsis()
Get the ellipsis string for the current language.
Definition strings.cpp:309
static const Units _units_time_years_or_periods[]
Unit conversions for time in calendar years or economic periods.
Definition strings.cpp:935
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.
Definition cargotype.h:93
static CargoSpec * Get(size_t index)
Retrieve cargo details for the given cargo type.
Definition cargotype.h:137
static size_t GetArraySize()
Total number of cargospecs, both valid and invalid.
Definition cargotype.h:127
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.
Definition currency.h:76
std::string separator
The thousands separator for this currency.
Definition currency.h:78
std::string prefix
Prefix to apply when formatting money in this currency.
Definition currency.h:80
std::string suffix
Suffix to apply when formatting money in this currency.
Definition currency.h:81
uint16_t rate
The conversion rate compared to the base currency.
Definition currency.h:77
uint8_t symbol_pos
The currency symbol is represented by two possible values, prefix and suffix Usage of one or the othe...
Definition currency.h:92
uint16_t town_cn
The N-1th depot for this town (consecutive number)
Definition depot_base.h:22
Settings for the four different fonts.
Definition fontcache.h:205
FontCacheSubSetting mono
The mono space font used for license/readme viewers.
Definition fontcache.h:209
FontCacheSubSetting medium
The normal font size.
Definition fontcache.h:207
std::string font
The name of the font, or path to the font.
Definition fontcache.h:198
const void * os_handle
Optional native OS font info. Only valid during font search.
Definition fontcache.h:201
Dynamic data of a loaded NewGRF.
Definition newgrf.h:115
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)
Group data.
Definition group.h:73
std::string name
Group Name.
Definition group.h:74
uint16_t number
Per-company group number.
Definition group.h:86
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.
Definition industry.h:62
IndustryType type
type of industry.
Definition industry.h:98
Town * town
Nearest town.
Definition industry.h:90
Make sure the size is right.
Definition language.h:93
std::filesystem::path file
Name of the file we read this data from.
Definition language.h:94
Header of a language file.
Definition language.h:24
uint8_t plural_form
plural form index
Definition language.h:41
bool IsValid() const
Check whether the header is a valid header for OpenTTD.
Definition strings.cpp:1990
bool IsReasonablyFinished() const
Check whether a translation is sufficiently finished to offer it to the public.
Definition strings.cpp:2010
uint32_t version
32-bits of auto generated version info which is basically a hash of strings.h
Definition language.h:28
char isocode[16]
the ISO code for the language (not country code)
Definition language.h:31
char own_name[32]
the localized name of this language
Definition language.h:30
char name[32]
the international name of this language
Definition language.h:29
uint16_t winlangid
Windows language ID: Windows cannot and will not convert isocodes to something it can use to determin...
Definition language.h:51
uint8_t num_genders
the number of genders of this language
Definition language.h:53
uint16_t missing
number of missing strings.
Definition language.h:40
char digit_group_separator[8]
Thousand separator used for anything not currencies.
Definition language.h:35
uint8_t newgrflangid
newgrf language id
Definition language.h:52
uint8_t text_dir
default direction of the text
Definition language.h:42
uint8_t num_cases
the number of cases of this language
Definition language.h:54
uint32_t ident
32-bits identifier
Definition language.h:27
char digit_decimal_separator[8]
Decimal separator.
Definition language.h:39
char digit_group_separator_currency[8]
Thousand separator used for currencies.
Definition language.h:37
static const uint32_t IDENT
Identifier for OpenTTD language files, big endian for "LANG".
Definition language.h:25
std::string list_separator
Current list separator string.
Definition strings.cpp:288
std::string ellipsis
Current ellipsis string.
Definition strings.cpp:289
std::array< uint, TEXT_TAB_END > langtab_num
Offset into langpack offs.
Definition strings.cpp:285
std::array< uint, TEXT_TAB_END > langtab_start
Offset into langpack offs.
Definition strings.cpp:286
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.
Station data structure.
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.
Town data structure.
Definition town.h:52
std::string name
Custom town name. If empty, the town was not renamed and uses the generated name.
Definition town.h:61
Helper for unit conversion.
Definition strings.cpp:810
double factor
Amount to multiply or divide upon conversion.
Definition strings.cpp:811
int64_t ToDisplay(int64_t input, bool round=true) const
Convert value from OpenTTD's internal unit into the displayed value.
Definition strings.cpp:819
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.
Definition strings.cpp:833
Information about a specific unit system with a long variant.
Definition strings.cpp:849
StringID s
String for the short variant of the unit.
Definition strings.cpp:851
StringID l
String for the long variant of the unit.
Definition strings.cpp:852
unsigned int decimal_places
Number of decimal places embedded in the value. For example, 1 if the value is in tenths,...
Definition strings.cpp:853
UnitConversion c
Conversion.
Definition strings.cpp:850
Information about a specific unit system.
Definition strings.cpp:842
StringID s
String for the unit.
Definition strings.cpp:844
unsigned int decimal_places
Number of decimal places embedded in the value. For example, 1 if the value is in tenths,...
Definition strings.cpp:845
UnitConversion c
Conversion.
Definition strings.cpp:843
Vehicle data structure.
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.
Base of the town class.
Town name generator stuff.
Handling of UTF-8 encoded data.
Base class for all vehicles.
VehicleType
Available vehicle types.
@ VEH_ROAD
Road vehicle type.
@ VEH_AIRCRAFT
Aircraft vehicle type.
@ VEH_SHIP
Ship vehicle type.
@ VEH_TRAIN
Train vehicle type.
Base of waypoints.
std::wstring OTTD2FS(std::string_view name)
Convert from OpenTTD's encoding to a wide string.
Definition win32.cpp:354
std::string FS2OTTD(std::wstring_view name)
Convert to OpenTTD's encoding from a wide string.
Definition win32.cpp:337
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...
Definition window.cpp:3282
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: