22#include "strings_internal.h"
33#include "table/strings.h"
44 GRFLB_AMERICAN = 0x01,
52enum GRFExtendedLanguages : uint8_t {
53 GRFLX_AMERICAN = 0x00,
58 GRFLX_UNSPECIFIED = 0x7F,
88 if (m.newgrf_id == newgrf_id)
return m.openttd_id;
103 if (m.openttd_id == openttd_id)
return m.newgrf_id;
133 if (this->strings.find(0) == this->strings.end()) {
136 GrfMsg(1,
"choice list misses default value");
137 this->strings[0] = std::stringstream();
140 std::ostreambuf_iterator<char> d(dest);
146 dest << this->strings[0].rdbuf();
152 if (this->type == SCC_SWITCH_CASE) {
163 if (this->strings.find(lm->
GetReverseMapping(i,
false)) != this->strings.end()) count++;
170 if (this->strings.find(idx) == this->strings.end())
continue;
171 auto str = this->strings[idx].str();
177 size_t len = std::min<size_t>(0xFFFE, str.size());
178 *d++ =
GB(len + 1, 8, 8);
179 *d++ =
GB(len + 1, 0, 8);
182 dest.write(str.c_str(), len);
187 dest << this->strings[0].rdbuf() <<
'\0';
189 if (this->type == SCC_PLURAL_LIST) {
199 *d++ = this->offset - 0x80;
206 for (
int i = 0; i < count; i++) {
207 int idx = (this->type == SCC_GENDER_LIST ? lm->
GetReverseMapping(i,
true) : i + 1);
208 const auto &str = this->strings[this->strings.find(idx) != this->strings.end() ? idx : 0].str();
209 size_t len = str.size() + 1;
210 if (len > 0xFF) GrfMsg(1,
"choice list string is too long");
211 *d++ =
GB(len, 0, 8);
215 for (
int i = 0; i < count; i++) {
216 int idx = (this->type == SCC_GENDER_LIST ? lm->
GetReverseMapping(i,
true) : i + 1);
217 const auto &str = this->strings[this->strings.find(idx) != this->strings.end() ? idx : 0].str();
220 size_t len = std::min<size_t>(0xFE, str.size());
221 dest.write(str.c_str(), len);
240 if (str.empty())
return {};
242 std::string_view::const_iterator src = str.cbegin();
245 bool unicode =
false;
257 std::ostringstream dest;
258 std::ostreambuf_iterator<char> d(dest);
259 while (src != str.cend()) {
263 c = Utf8Consume(src);
265 if (
GB(c, 8, 8) == 0xE0) {
267 }
else if (c >= 0x20) {
273 c =
static_cast<uint8_t
>(*src++);
276 if (c ==
'\0')
break;
280 if (*src ==
'\0')
goto string_end;
286 if (allow_newlines) {
289 GrfMsg(1,
"Detected newline in string that does not allow one");
295 if (src[0] ==
'\0' || src[1] ==
'\0')
goto string_end;
307 if (src[0] ==
'\0' || src[1] ==
'\0')
goto string_end;
309 string =
static_cast<uint8_t
>(*src++);
310 string |=
static_cast<uint8_t
>(*src++) << 8;
342 case 0x00:
goto string_end;
353 if (src[0] ==
'\0' || src[1] ==
'\0')
goto string_end;
354 uint16_t tmp =
static_cast<uint8_t
>(*src++);
355 tmp |=
static_cast<uint8_t
>(*src++) << 8;
370 if (str[0] ==
'\0')
goto string_end;
373 int mapped = lm !=
nullptr ? lm->
GetMapping(index, code == 0x0E) : -1;
375 Utf8Encode(d, code == 0x0E ? SCC_GENDER_INDEX : SCC_SET_CASE);
376 Utf8Encode(d, code == 0x0E ? mapped : mapped + 1);
383 if (str[0] ==
'\0')
goto string_end;
384 if (mapping ==
nullptr) {
385 if (code == 0x10) src++;
386 GrfMsg(1,
"choice list {} marker found when not expected", code == 0x10 ?
"next" :
"default");
389 int index = (code == 0x10 ? *src++ : 0);
391 GrfMsg(1,
"duplicate choice list string, ignoring");
393 d = std::ostreambuf_iterator<char>(mapping->
strings[index]);
399 if (mapping ==
nullptr) {
400 GrfMsg(1,
"choice list end marker found when not expected");
407 d = std::ostreambuf_iterator<char>(dest);
414 if (src[0] ==
'\0')
goto string_end;
415 if (mapping !=
nullptr) {
416 GrfMsg(1,
"choice lists can't be stacked, it's going to get messy now...");
417 if (code != 0x14) src++;
419 static const StringControlCode mp[] = { SCC_GENDER_LIST, SCC_SWITCH_CASE, SCC_PLURAL_LIST };
436 case 0x1F:
Utf8Encode(d, SCC_PUSH_COLOUR);
break;
437 case 0x20:
Utf8Encode(d, SCC_POP_COLOUR);
break;
442 GrfMsg(1,
"missing handler for extended format code");
450 case 0xA0:
Utf8Encode(d, SCC_UP_ARROW);
break;
451 case 0xAA:
Utf8Encode(d, SCC_DOWN_ARROW);
break;
452 case 0xAC:
Utf8Encode(d, SCC_CHECKMARK);
break;
454 case 0xAF:
Utf8Encode(d, SCC_RIGHT_ARROW);
break;
460 case 0xB9:
Utf8Encode(d, SCC_SUPERSCRIPT_M1);
break;
461 case 0xBC:
Utf8Encode(d, SCC_SMALL_UP_ARROW);
break;
462 case 0xBD:
Utf8Encode(d, SCC_SMALL_DOWN_ARROW);
break;
472 if (mapping !=
nullptr) {
473 GrfMsg(1,
"choice list was incomplete, the whole list is ignored");
489 for (
auto &text : list) {
490 if (text.langid == langid) {
491 text.text = text_to_add;
497 list.push_back(
GRFText{ langid, std::string(text_to_add) });
525 if (list ==
nullptr) list = std::make_shared<GRFTextList>();
537 if (list ==
nullptr) list = std::make_shared<GRFTextList>();
553 if (langid_to_add & (GRFLB_AMERICAN | GRFLB_ENGLISH)) {
554 langid_to_add = GRFLX_ENGLISH;
557 if (langid_to_add & GRFLB_GERMAN) ret =
AddGRFString(grfid, stringid, GRFLX_GERMAN,
true, allow_newlines, text_to_add, def_string);
558 if (langid_to_add & GRFLB_FRENCH) ret =
AddGRFString(grfid, stringid, GRFLX_FRENCH,
true, allow_newlines, text_to_add, def_string);
559 if (langid_to_add & GRFLB_SPANISH) ret =
AddGRFString(grfid, stringid, GRFLX_SPANISH,
true, allow_newlines, text_to_add, def_string);
564 auto it = std::ranges::find_if(_grf_text, [&grfid, &stringid](
const GRFTextEntry &grf_text) {
return grf_text.grfid == grfid && grf_text.stringid == stringid; });
565 if (it == std::end(_grf_text)) {
570 it = _grf_text.emplace(std::end(_grf_text));
572 it->stringid = stringid;
573 it->def_string = def_string;
580 GrfMsg(3,
"Added 0x{:X} grfid {:08X} string 0x{:X} lang 0x{:X} string '{}' ({:X})",
id, grfid, stringid, langid_to_add, newtext,
MakeStringID(
TEXT_TAB_NEWGRF_START,
id));
590 auto it = std::ranges::find_if(_grf_text, [&grfid, &stringid](
const GRFTextEntry &grf_text) {
return grf_text.grfid == grfid && grf_text.stringid == stringid; });
591 if (it != std::end(_grf_text)) {
596 return STR_UNDEFINED;
609 const char *default_text =
nullptr;
612 for (
const auto &text : text_list) {
617 if (text.langid == GRFLX_UNSPECIFIED || (default_text ==
nullptr && (text.langid == GRFLX_ENGLISH || text.langid == GRFLX_AMERICAN))) {
618 default_text = text.text.c_str();
642 assert(stringid.base() < _grf_text.size());
643 assert(_grf_text[stringid].grfid != 0);
646 if (str !=
nullptr)
return str;
649 return GetStringPtr(_grf_text[stringid].def_string);
665bool CheckGrfLangID(uint8_t lang_id, uint8_t grf_version)
667 if (grf_version < 7) {
669 case GRFLX_GERMAN:
return (lang_id & GRFLB_GERMAN) != 0;
670 case GRFLX_FRENCH:
return (lang_id & GRFLB_FRENCH) != 0;
671 case GRFLX_SPANISH:
return (lang_id & GRFLB_SPANISH) != 0;
672 default:
return (lang_id & (GRFLB_ENGLISH | GRFLB_AMERICAN)) != 0;
676 return (lang_id ==
_currentLangID || lang_id == GRFLX_UNSPECIFIED);
689 std::array<uint8_t, 0x30> stack{};
690 uint8_t position = 0;
691 const GRFFile *grffile =
nullptr;
697 assert(num_entries <
sizeof(uint32_t) * std::size(stack));
699 auto stack_it = this->stack.begin();
700 for (uint i = 0; i < num_entries; i++) {
701 uint32_t value = _temp_store.GetValue(0x100 + i);
702 for (uint j = 0; j < 32; j += 8) {
703 *stack_it++ =
GB(value, j, 8);
708 uint8_t PopUnsignedByte() { assert(this->position < this->stack.size());
return this->stack[this->position++]; }
709 int8_t PopSignedByte() {
return (int8_t)this->PopUnsignedByte(); }
711 uint16_t PopUnsignedWord()
713 uint16_t val = this->PopUnsignedByte();
714 return val | (this->PopUnsignedByte() << 8);
716 int16_t PopSignedWord() {
return (int32_t)this->PopUnsignedWord(); }
718 uint32_t PopUnsignedDWord()
720 uint32_t val = this->PopUnsignedWord();
721 return val | (this->PopUnsignedWord() << 16);
723 int32_t PopSignedDWord() {
return (int32_t)this->PopUnsignedDWord(); }
725 uint64_t PopUnsignedQWord()
727 uint64_t val = this->PopUnsignedDWord();
728 return val | (((uint64_t)this->PopUnsignedDWord()) << 32);
730 int64_t PopSignedQWord() {
return (int64_t)this->PopUnsignedQWord(); }
736 for (
int i = 0; i < 2; i++) tmp[i] = this->stack[this->position + i + 6];
737 for (
int i = 5; i >= 0; i--) this->stack[this->position + i + 2] = this->stack[this->position + i];
738 for (
int i = 0; i < 2; i++) this->stack[this->position + i] = tmp[i];
741 void PushWord(uint16_t word)
743 if (this->position >= 2) {
747 std::rotate(this->stack.rbegin(), this->stack.rbegin() + 2, this->stack.rend());
749 this->stack[this->position] =
GB(word, 0, 8);
750 this->stack[this->position + 1] =
GB(word, 8, 8);
808 params.emplace_back(stack.PopUnsignedWord());
813 params.emplace_back(stringid);
821 params.emplace_back(cargo <
NUM_CARGO ? 1ULL << cargo : 0);
853 return SCC_CURRENCY_LONG;
860 return SCC_DATE_LONG;
864 return SCC_DATE_SHORT;
870 return SCC_VOLUME_LONG;
873 return SCC_VOLUME_SHORT;
876 return SCC_WEIGHT_LONG;
879 return SCC_WEIGHT_SHORT;
888 return SCC_CARGO_LONG;
891 return SCC_CARGO_SHORT;
894 return SCC_CARGO_TINY;
897 return SCC_CARGO_LIST;
900 return SCC_STATION_NAME;
921 if (str ==
nullptr)
return;
923 for (
const char *p = str; *p !=
'\0'; ) {
941 const char *str = GetStringPtr(stringid);
942 if (str ==
nullptr)
return {};
944 std::vector<StringParameter> params;
Helper types related to the allocation of memory.
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.
static const CargoType NUM_CARGO
Maximum number of cargo types in a game.
A sort-of mixin that adds 'at(pos)' and 'operator[](pos)' implementations for 'ConvertibleThroughBase...
static constexpr TimerGame< struct Calendar >::Date DAYS_TILL_ORIGINAL_BASE_YEAR
The date of the first day of the original base year.
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_NEWGRF_PRINT_WORD_VOLUME_SHORT
9A 19: Read 2 bytes from the stack as short signed volume
@ SCC_NEWGRF_PRINT_DWORD_CURRENCY
8F: Read 4 bytes from the stack as currency
@ SCC_NEWGRF_PRINT_WORD_HEX
9A 07: Read 2 bytes from the stack and print it as hex
@ SCC_NEWGRF_PRINT_WORD_WEIGHT_SHORT
9A 1A: Read 2 bytes from the stack as short unsigned weight
@ SCC_NEWGRF_ROTATE_TOP_4_WORDS
86: Rotate the top 4 words of the stack (W4 W1 W2 W3)
@ SCC_TINYFONT
Switch to small font.
@ SCC_NEWGRF_PRINT_DWORD_DATE_LONG
9A 16: Read 4 bytes from the stack as base 0 date
@ SCC_NEWGRF_PRINT_DWORD_HEX
9A 08: Read 4 bytes from the stack and print it as hex
@ SCC_NEWGRF_PRINT_WORD_DATE_SHORT
83: Read 2 bytes from the stack as base 1920 date
@ SCC_BIGFONT
Switch to large font.
@ SCC_NEWGRF_PRINT_BYTE_HEX
9A 06: Read 1 byte from the stack and print it as hex
@ SCC_NEWGRF_PRINT_WORD_UNSIGNED
7E: Read 2 bytes from the stack as unsigned value
@ SCC_NEWGRF_PRINT_BYTE_SIGNED
7D: Read 1 byte from the stack as signed value
@ SCC_NEWGRF_PRINT_WORD_CARGO_NAME
9A 1E: Read 2 bytes from the stack as cargo name
@ SCC_NEWGRF_PRINT_DWORD_SIGNED
7B: Read 4 bytes from the stack
@ SCC_NEWGRF_STRINL
Inline another string at the current position, StringID is encoded in the string.
@ SCC_NEWGRF_PRINT_WORD_CARGO_SHORT
9A 1C: Read 2 + 2 bytes from the stack as cargo type (translated) and unsigned cargo amount
@ SCC_NEWGRF_PRINT_WORD_STATION_NAME
9A 0C: Read 2 bytes from the stack as station name
@ SCC_NEWGRF_PRINT_WORD_DATE_LONG
82: Read 2 bytes from the stack as base 1920 date
@ SCC_NEWGRF_PRINT_WORD_SIGNED
7C: Read 2 bytes from the stack as signed value
@ SCC_NEWGRF_PRINT_WORD_POWER
9A 18: Read 2 bytes from the stack as unsigned power
@ SCC_NEWGRF_PUSH_WORD
9A 03: Pushes 2 bytes onto the stack
@ SCC_NEWGRF_PRINT_WORD_STRING_ID
81: Read 2 bytes from the stack as String ID
@ SCC_NEWGRF_PRINT_WORD_VOLUME_LONG
87: Read 2 bytes from the stack as long signed volume
@ SCC_NEWGRF_DISCARD_WORD
85: Discard the next two bytes
@ SCC_NEWGRF_PRINT_DWORD_FORCE
9A 21: Read 4 bytes from the stack as unsigned force
@ SCC_NEWGRF_PRINT_WORD_CARGO_TINY
9A 1D: Read 2 + 2 bytes from the stack as cargo type (translated) and unsigned cargo amount
@ SCC_NEWGRF_PRINT_QWORD_HEX
9A 0B: Read 8 bytes from the stack and print it as hex
@ SCC_NEWGRF_PRINT_WORD_CARGO_LONG
9A 1B: Read 2 + 2 bytes from the stack as cargo type (translated) and unsigned cargo amount
@ SCC_NEWGRF_PRINT_WORD_SPEED
84: Read 2 bytes from the stack as signed speed
@ SCC_NEWGRF_PRINT_DWORD_DATE_SHORT
9A 17: Read 4 bytes from the stack as base 0 date
@ SCC_NEWGRF_PRINT_WORD_WEIGHT_LONG
9A 0D: Read 2 bytes from the stack as long unsigned weight
@ SCC_NEWGRF_PRINT_QWORD_CURRENCY
9A 01: Read 8 bytes from the stack as currency
Functions related to debugging.
Information about languages and their files.
const LanguageMetadata * _current_language
The currently loaded language.
StringID MapGRFStringID(uint32_t grfid, GRFStringID str)
Used when setting an object's property to map to the GRF's strings while taking in consideration the ...
Base for the NewGRF implementation.
CargoType GetCargoTranslation(uint8_t cargo, const GRFFile *grffile, bool usebit)
Translate a GRF-local cargo slot/bitnum into a CargoType.
Cargo support for NewGRFs.
Functionality related to the temporary and persistent storage arrays for NewGRFs.
static void HandleNewGRFStringControlCodes(const char *str, TextRefStack &stack, std::vector< StringParameter > ¶ms)
Handle control codes in a NewGRF string, processing the stack and filling parameters.
void CleanUpStrings()
House cleaning.
static void AddGRFTextToList(GRFTextList &list, uint8_t langid, std::string_view text_to_add)
Add a new text to a GRFText list.
GRFBaseLanguages
Explains the newgrf shift bit positioning.
std::string TranslateTTDPatchCodes(uint32_t grfid, uint8_t language_id, bool allow_newlines, std::string_view str, StringControlCode byte80)
Translate TTDPatch string codes into something OpenTTD can handle (better).
const char * GetGRFStringPtr(StringIndexInTab stringid)
Get a C-string from a stringid set by a newgrf.
static void RemapNewGRFStringControlCode(char32_t scc, const char **str, TextRefStack &stack, std::vector< StringParameter > ¶ms)
Process NewGRF string control code instructions.
void SetCurrentGrfLangID(uint8_t language_id)
Equivalence Setter function between game and newgrf langID.
StringID GetGRFStringID(uint32_t grfid, GRFStringID stringid)
Returns the index for this stringid associated with its grfID.
const char * GetGRFStringFromGRFText(const GRFTextList &text_list)
Get a C-string from a GRFText-list.
static uint8_t _currentLangID
by default, english is used.
std::vector< StringParameter > GetGRFSringTextStackParameters(const GRFFile *grffile, StringID stringid, uint8_t num_entries)
Process the text ref stack for a GRF String and return its parameters.
StringID AddGRFString(uint32_t grfid, GRFStringID stringid, uint8_t langid_to_add, bool new_scheme, bool allow_newlines, std::string_view text_to_add, StringID def_string)
Add the new read string into our structure.
std::string GetGRFStringWithTextStack(const struct GRFFile *grffile, GRFStringID grfstringid, uint8_t num_entries)
Format a GRF string using the text ref stack for parameters.
Header of Action 04 "universal holder" structure and functions.
std::shared_ptr< GRFTextList > GRFTextWrapper
Reference counted wrapper around a GRFText pointer.
std::vector< GRFText > GRFTextList
A GRF text with a list of translations.
static const char32_t NFO_UTF8_IDENTIFIER
This character (thorn) indicates a unicode string to NFO.
A number of safeguards to prevent using unsafe methods.
Definition of base types and functions in a cross-platform compatible way.
bool IsValidChar(char32_t key, CharSetFilter afilter)
Only allow certain keys.
size_t Utf8Decode(char32_t *c, const char *s)
Decode and consume the next UTF-8 encoded character.
size_t Utf8Encode(T buf, char32_t c)
Encode a unicode character and place it in the buffer.
Functions related to low-level strings.
int8_t Utf8EncodedCharLen(char c)
Return the length of an UTF-8 encoded value based on a single char.
@ CS_ALPHANUMERAL
Both numeric and alphabetic and spaces and stuff.
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.
StringID MakeStringID(StringTab tab, StringIndexInTab index)
Create a StringID.
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
static const StringID INVALID_STRING_ID
Constant representing an invalid string (16bit in case it is used in savegames)
static const uint TAB_SIZE_NEWGRF
Number of strings for NewGRFs.
@ TEXT_TAB_NEWGRF_START
Start of NewGRF supplied strings.
Dynamic data of a loaded NewGRF.
Holder of the above structure.
A GRF text with associated language ID.
Mapping between NewGRF and OpenTTD IDs.
Mapping of language data between a NewGRF and OpenTTD.
std::vector< Mapping > gender_map
Mapping of NewGRF and OpenTTD IDs for genders.
int GetReverseMapping(int openttd_id, bool gender) const
Get the mapping from OpenTTD's internal ID to the NewGRF supplied ID.
int plural_form
The plural form used for this language.
static const LanguageMap * GetLanguageMap(uint32_t grfid, uint8_t language_id)
Get the language map associated with a given NewGRF and language.
std::vector< Mapping > case_map
Mapping of NewGRF and OpenTTD IDs for cases.
int GetMapping(int newgrf_id, bool gender) const
Get the mapping from the NewGRF supplied ID to OpenTTD's internal ID.
Class for temporary storage of data.
void RotateTop4Words()
Rotate the top four words down: W1, W2, W3, W4 -> W4, W1, W2, W3.
Helper structure for mapping choice lists.
std::map< uint8_t, std::stringstream > strings
Mapping of NewGRF supplied ID to the different strings in the choice list.
StringControlCode type
The type of choice list.
int offset
The offset for the plural/gender form.
UnmappedChoiceList(StringControlCode type, int offset)
Initialise the mapping.
void Flush(const LanguageMap *lm, std::ostringstream &dest)
Flush this choice list into the destination string.
Definition of the game-calendar-timer.