OpenTTD Source  20240917-master-g9ab0a47812
strings_internal.h
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 #ifndef STRINGS_INTERNAL_H
11 #define STRINGS_INTERNAL_H
12 
13 #include "strings_func.h"
14 #include "string_func.h"
15 
18  StringParameterData data;
19  char32_t type;
20 };
21 
23 protected:
24  StringParameters *parent = nullptr;
25  std::span<StringParameter> parameters = {};
26 
27  size_t offset = 0;
28  char32_t next_type = 0;
29 
30  StringParameters(std::span<StringParameter> parameters = {}) :
32  {}
33 
35 
36 public:
42  parent(&parent),
43  parameters(parent.parameters.subspan(parent.offset, size))
44  {}
45 
46  void PrepareForNextRun();
47  void SetTypeOfNextParameter(char32_t type) { this->next_type = type; }
48 
54  size_t GetOffset() { return this->offset; }
55 
61  void SetOffset(size_t offset)
62  {
63  /*
64  * The offset must be fewer than the number of parameters when it is
65  * being set. Unless restoring a backup, then the original value is
66  * correct as well as long as the offset was not changed. In other
67  * words, when the offset was already at the end of the parameters and
68  * the string did not consume any parameters.
69  */
70  assert(offset < this->parameters.size() || this->offset == offset);
71  this->offset = offset;
72  }
73 
79  void AdvanceOffset(size_t advance)
80  {
81  this->offset += advance;
82  assert(this->offset <= this->parameters.size());
83  }
84 
91  template <typename T>
93  {
94  const auto &param = GetNextParameterReference();
95  const uint64_t *data = std::get_if<uint64_t>(&param.data);
96  if (data != nullptr) return static_cast<T>(*data);
97  throw std::runtime_error("Attempt to read string parameter as integer");
98  }
99 
107  {
108  const auto &param = GetNextParameterReference();
109  const std::string *data = std::get_if<std::string>(&param.data);
110  if (data != nullptr) return data->c_str();
111  throw std::runtime_error("Attempt to read integer parameter as string");
112  }
113 
123 
135  {
136  return StringParameters(this->parameters.subspan(offset, this->parameters.size() - offset));
137  }
138 
140  size_t GetDataLeft() const
141  {
142  return this->parameters.size() - this->offset;
143  }
144 
146  char32_t GetTypeAtOffset(size_t offset) const
147  {
148  assert(offset < this->parameters.size());
149  return this->parameters[offset].type;
150  }
151 
152  void SetParam(size_t n, const StringParameterData &v)
153  {
154  assert(n < this->parameters.size());
155  this->parameters[n].data = v;
156  }
157 
158  void SetParam(size_t n, uint64_t v)
159  {
160  assert(n < this->parameters.size());
161  this->parameters[n].data = v;
162  }
163 
164  template <typename T, std::enable_if_t<std::is_base_of<StrongTypedefBase, T>::value, int> = 0>
165  void SetParam(size_t n, T v)
166  {
167  SetParam(n, v.base());
168  }
169 
170  void SetParam(size_t n, const char *str)
171  {
172  assert(n < this->parameters.size());
173  this->parameters[n].data = str;
174  }
175 
176  void SetParam(size_t n, const std::string &str) { this->SetParam(n, str.c_str()); }
177 
178  void SetParam(size_t n, std::string &&str)
179  {
180  assert(n < this->parameters.size());
181  this->parameters[n].data = std::move(str);
182  }
183 
184  const StringParameterData &GetParam(size_t n) const
185  {
186  assert(n < this->parameters.size());
187  return this->parameters[n].data;
188  }
189 };
190 
195 template <size_t N>
197  std::array<StringParameter, N> params{};
198 
199 public:
201  {
202  this->parameters = std::span(params.data(), params.size());
203  }
204 
206  {
207  *this = std::move(other);
208  }
209 
210  ArrayStringParameters& operator=(ArrayStringParameters &&other) noexcept
211  {
212  this->offset = other.offset;
213  this->next_type = other.next_type;
214  this->params = std::move(other.params);
215  this->parameters = std::span(params.data(), params.size());
216  return *this;
217  }
218 
219  ArrayStringParameters(const ArrayStringParameters &other) = delete;
220  ArrayStringParameters& operator=(const ArrayStringParameters &other) = delete;
221 };
222 
229 template <typename... Args>
230 static auto MakeParameters(const Args&... args)
231 {
232  ArrayStringParameters<sizeof...(args)> parameters;
233  size_t index = 0;
234  (parameters.SetParam(index++, std::forward<const Args&>(args)), ...);
235  return parameters;
236 }
237 
243  std::string *string;
244 
245 public:
246  /* Required type for this to be an output_iterator; mimics std::back_insert_iterator. */
247  using value_type = void;
248  using difference_type = void;
249  using iterator_category = std::output_iterator_tag;
250  using pointer = void;
251  using reference = void;
252 
258  StringBuilder(std::string &string) : string(&string) {}
259 
260  /* Required operators for this to be an output_iterator; mimics std::back_insert_iterator, which has no-ops. */
261  StringBuilder &operator++() { return *this; }
262  StringBuilder operator++(int) { return *this; }
263  StringBuilder &operator*() { return *this; }
264 
272  StringBuilder &operator=(const char value)
273  {
274  return this->operator+=(value);
275  }
276 
282  StringBuilder &operator+=(const char value)
283  {
284  this->string->push_back(value);
285  return *this;
286  }
287 
293  StringBuilder &operator+=(std::string_view str)
294  {
295  *this->string += str;
296  return *this;
297  }
298 
303  void Utf8Encode(char32_t c)
304  {
305  auto iterator = std::back_inserter(*this->string);
306  ::Utf8Encode(iterator, c);
307  }
308 
314  void RemoveElementsFromBack(size_t amount)
315  {
316  this->string->erase(this->string->size() - std::min(amount, this->string->size()));
317  }
318 
323  size_t CurrentIndex()
324  {
325  return this->string->size();
326  }
327 
332  char &operator[](size_t index)
333  {
334  return (*this->string)[index];
335  }
336 };
337 
338 void GetStringWithArgs(StringBuilder &builder, StringID string, StringParameters &args, uint case_index = 0, bool game_script = false);
339 std::string GetStringWithArgs(StringID string, StringParameters &args);
340 
341 /* Do not leak the StringBuilder to everywhere. */
342 void GenerateTownNameString(StringBuilder &builder, size_t lang, uint32_t seed);
343 void GetTownName(StringBuilder &builder, const struct Town *t);
344 void GRFTownNameGenerate(StringBuilder &builder, uint32_t grfid, uint16_t gen, uint32_t seed);
345 
346 uint RemapNewGRFStringControlCode(uint scc, const char **str, StringParameters &parameters, bool modify_parameters);
347 
348 #endif /* STRINGS_INTERNAL_H */
StringParameters::SetOffset
void SetOffset(size_t offset)
Set the offset within the string from where to return the next result of GetInt64 or GetInt32.
Definition: strings_internal.h:61
StringBuilder
Equivalent to the std::back_insert_iterator in function, with some convenience helpers for string con...
Definition: strings_internal.h:242
StringParameters::GetOffset
size_t GetOffset()
Get the current offset, so it can be backed up for certain processing steps, or be used to offset the...
Definition: strings_internal.h:54
ArrayStringParameters
Extension of StringParameters with its own statically sized buffer for the parameters.
Definition: strings_internal.h:196
StringParameters::PrepareForNextRun
void PrepareForNextRun()
Prepare the string parameters for the next formatting run.
Definition: strings.cpp:68
StringParameters::parent
StringParameters * parent
If not nullptr, this instance references data from this parent instance.
Definition: strings_internal.h:24
StringID
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
Definition: strings_type.h:16
StringParameters::GetNextParameter
T GetNextParameter()
Get the next parameter from our parameters.
Definition: strings_internal.h:92
GenerateTownNameString
void GenerateTownNameString(StringBuilder &builder, size_t lang, uint32_t seed)
Generates town name from given seed.
Definition: townname.cpp:1013
StringParameter
The data required to format and validate a single parameter of a string.
Definition: strings_internal.h:17
StringParameters::GetNextParameterString
const char * GetNextParameterString()
Get the next string parameter from our parameters.
Definition: strings_internal.h:106
StringBuilder::RemoveElementsFromBack
void RemoveElementsFromBack(size_t amount)
Remove the given amount of characters from the back of the string.
Definition: strings_internal.h:314
GetStringWithArgs
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:243
StringParameters::GetRemainingParameters
StringParameters GetRemainingParameters()
Get a new instance of StringParameters that is a "range" into the remaining existing parameters.
Definition: strings_internal.h:122
RemapNewGRFStringControlCode
uint RemapNewGRFStringControlCode(uint scc, const char **str, StringParameters &parameters, bool modify_parameters)
FormatString for NewGRF specific "magic" string control codes.
Definition: newgrf_text.cpp:828
StringParameters::AdvanceOffset
void AdvanceOffset(size_t advance)
Advance the offset within the string from where to return the next result of GetInt64 or GetInt32.
Definition: strings_internal.h:79
StringParameters::GetTypeAtOffset
char32_t GetTypeAtOffset(size_t offset) const
Get the type of a specific element.
Definition: strings_internal.h:146
StringParameters::GetDataLeft
size_t GetDataLeft() const
Return the amount of elements which can still be read.
Definition: strings_internal.h:140
StringBuilder::operator+=
StringBuilder & operator+=(const char value)
Operator to add a character to the end of the buffer.
Definition: strings_internal.h:282
ArrayStringParameters::params
std::array< StringParameter, N > params
The actual parameters.
Definition: strings_internal.h:197
StringBuilder::operator[]
char & operator[](size_t index)
Get the reference to the character at the given index.
Definition: strings_internal.h:332
StringBuilder::operator+=
StringBuilder & operator+=(std::string_view str)
Operator to append the given string to the output buffer.
Definition: strings_internal.h:293
StringBuilder::StringBuilder
StringBuilder(std::string &string)
Create the builder of an external buffer.
Definition: strings_internal.h:258
string_func.h
strings_func.h
StringParameters
Definition: strings_internal.h:22
StringParameters::StringParameters
StringParameters(StringParameters &parent, size_t size)
Create a new StringParameters instance that can reference part of the data of the given parent instan...
Definition: strings_internal.h:41
StringParameters::offset
size_t offset
Current offset in the parameters span.
Definition: strings_internal.h:27
GetTownName
static void GetTownName(StringBuilder &builder, const TownNameParams *par, uint32_t townnameparts)
Fills builder with specified town name.
Definition: townname.cpp:48
StringBuilder::CurrentIndex
size_t CurrentIndex()
Get the current index in the string.
Definition: strings_internal.h:323
StringParameters::parameters
std::span< StringParameter > parameters
Array with the actual parameters.
Definition: strings_internal.h:25
Town
Town data structure.
Definition: town.h:54
StringParameter::type
char32_t type
The StringControlCode to interpret this data with when it's the first parameter, otherwise '\0'.
Definition: strings_internal.h:19
StringBuilder::operator=
StringBuilder & operator=(const char value)
Operator to add a character to the end of the buffer.
Definition: strings_internal.h:272
StringParameters::GetRemainingParameters
StringParameters GetRemainingParameters(size_t offset)
Get a new instance of StringParameters that is a "range" into the remaining existing parameters from ...
Definition: strings_internal.h:134
StringParameters::GetNextParameterReference
const StringParameter & GetNextParameterReference()
Get the next parameter from our parameters.
Definition: strings.cpp:81
StringBuilder::Utf8Encode
void Utf8Encode(char32_t c)
Encode the given Utf8 character into the output buffer.
Definition: strings_internal.h:303
StringParameters::next_type
char32_t next_type
The type of the next data that is retrieved.
Definition: strings_internal.h:28
StringParameter::data
StringParameterData data
The data of the parameter.
Definition: strings_internal.h:18