OpenTTD Source 20251213-master-g1091fa6071
gfx_layout.h
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 <https://www.gnu.org/licenses/old-licenses/gpl-2.0>.
6 */
7
10#ifndef GFX_LAYOUT_H
11#define GFX_LAYOUT_H
12
13#include "misc/lrucache.hpp"
14#include "fontcache.h"
15
16#include <string_view>
17
22struct FontState {
24 FontIndex font_index;
26 std::vector<TextColour> colour_stack;
27
28 FontState() : fontsize(FS_END), font_index(INVALID_FONT_INDEX), cur_colour(TC_INVALID) {}
30
31 auto operator<=>(const FontState &) const = default;
32
37 inline void SetColour(TextColour c)
38 {
39 assert(((c & TC_COLOUR_MASK) >= TC_BLUE && (c & TC_COLOUR_MASK) <= TC_BLACK) || (c & TC_COLOUR_MASK) == TC_INVALID);
40 assert((c & (TC_COLOUR_MASK | TC_FLAGS_MASK)) == c);
41 if ((this->cur_colour & TC_FORCED) == 0) this->cur_colour = c;
42 }
43
47 inline void PopColour()
48 {
49 if (colour_stack.empty()) return;
50 SetColour(colour_stack.back());
51 colour_stack.pop_back();
52 }
53
57 inline void PushColour()
58 {
59 colour_stack.push_back(this->cur_colour);
60 }
61
66 inline void SetFontSize(FontSize f)
67 {
68 this->fontsize = f;
69 this->font_index = FontCache::GetDefaultFontIndex(this->fontsize);
70 }
71};
72
73template <typename T> struct std::hash<std::vector<T>> {
74 size_t operator()(const std::vector<T> &vec) const
75 {
76 /* This is not an optimal hash algorithm but in most cases this is empty and therefore the same anyway. */
77 return std::transform_reduce(std::begin(vec), std::end(vec),
78 std::hash<size_t>{}(std::size(vec)),
79 [](const size_t &a, const size_t &b) -> size_t { return a ^ b; },
80 [](const T &x) -> size_t { return std::hash<T>{}(x); });
81 }
82};
83
84template <> struct std::hash<FontState> {
85 std::size_t operator()(const FontState &state) const noexcept
86 {
87 size_t h1 = std::hash<FontSize>{}(state.fontsize);
88 size_t h2 = std::hash<FontIndex>{}(state.font_index);
89 size_t h3 = std::hash<TextColour>{}(state.cur_colour);
90 size_t h4 = std::hash<std::vector<TextColour>>{}(state.colour_stack);
91 return h1 ^ (h2 << 1) ^ (h3 << 2) ^ (h4 << 3);
92 }
93};
94
98class Font {
99public:
100 FontIndex font_index = INVALID_FONT_INDEX;
101 TextColour colour = TC_INVALID;
102
103 inline FontCache &GetFontCache() const { return *FontCache::Get(this->font_index); }
104};
105
107using FontMap = std::vector<std::pair<int, Font>>;
108
113public:
114 virtual ~ParagraphLayouter() = default;
115
117 class Position {
118 public:
119 int16_t left;
120 int16_t right;
121 int16_t top;
122
123 constexpr inline Position(int16_t left, int16_t right, int16_t top) : left(left), right(right), top(top) { }
124
126 constexpr inline Position(const Point &pt) : left(pt.x), right(pt.x), top(pt.y) { }
127 };
128
130 class VisualRun {
131 public:
132 virtual ~VisualRun() = default;
133 virtual const Font &GetFont() const = 0;
134 virtual int GetGlyphCount() const = 0;
135 virtual std::span<const GlyphID> GetGlyphs() const = 0;
136 virtual std::span<const Position> GetPositions() const = 0;
137 virtual int GetLeading() const = 0;
138 virtual std::span<const int> GetGlyphToCharMap() const = 0;
139 };
140
142 class Line {
143 public:
144 virtual ~Line() = default;
145 virtual int GetLeading() const = 0;
146 virtual int GetWidth() const = 0;
147 virtual int CountRuns() const = 0;
148 virtual const VisualRun &GetVisualRun(int run) const = 0;
149 virtual int GetInternalCharLength(char32_t c) const = 0;
150 };
151
152 virtual void Reflow() = 0;
153 virtual std::unique_ptr<const Line> NextLine(int max_width) = 0;
154};
155
161class Layouter : public std::vector<const ParagraphLayouter::Line *> {
162 std::string_view string;
163
167 std::string str;
168 };
169
172 std::string_view str;
173 };
174
175 friend struct std::hash<Layouter::LineCacheQuery>;
176 struct LineCacheHash;
177
179 using is_transparent = void;
180
181 template <typename Tlhs, typename Trhs>
182 bool operator()(const Tlhs &lhs, const Trhs &rhs) const
183 {
184 return lhs.state_before == rhs.state_before && lhs.str == rhs.str;
185 }
186 };
187
188public:
191 /* Due to the type of data in the buffer differing depending on the Layouter, we need to pass our own deleter routine. */
192 using Buffer = std::unique_ptr<void, void(*)(void *)>;
193 /* Stuff that cannot be freed until the ParagraphLayout is freed */
194 Buffer buffer{nullptr, [](void *){}};
196
198 std::unique_ptr<ParagraphLayouter> layout = nullptr;
199
200 std::vector<std::unique_ptr<const ParagraphLayouter::Line>> cached_layout{};
201 int cached_width = 0;
202 };
203private:
205 static std::unique_ptr<LineCache> linecache;
206
207 static LineCacheItem &GetCachedParagraphLayout(std::string_view str, const FontState &state);
208
209public:
210 Layouter(std::string_view str, int maxw = INT32_MAX, FontSize fontsize = FS_NORMAL);
212 ParagraphLayouter::Position GetCharPosition(std::string_view::const_iterator ch) const;
213 ptrdiff_t GetCharAtPosition(int x, size_t line_index) const;
214
215 static void Initialize();
216 static void ResetFontCache(FontSize fs);
217 static void ResetLineCache();
218};
219
220ParagraphLayouter::Position GetCharPosInString(std::string_view str, size_t pos, FontSize start_fontsize = FS_NORMAL);
221ptrdiff_t GetCharAtPosition(std::string_view str, int x, FontSize start_fontsize = FS_NORMAL);
222
223template <> struct std::hash<Layouter::LineCacheQuery> {
224 std::size_t operator()(const Layouter::LineCacheQuery &state) const noexcept
225 {
226 size_t h1 = std::hash<std::string_view>{}(state.str);
227 size_t h2 = std::hash<FontState>{}(state.state_before);
228 return h1 ^ (h2 << 1);
229 }
230};
231
233 using is_transparent = void;
234
235 std::size_t operator()(const Layouter::LineCacheKey &query) const { return std::hash<Layouter::LineCacheQuery>{}(LineCacheQuery{query.state_before, query.str}); }
236 std::size_t operator()(const Layouter::LineCacheQuery &query) const { return std::hash<Layouter::LineCacheQuery>{}(query); }
237};
238
239#endif /* GFX_LAYOUT_H */
Font cache for basic fonts.
Definition fontcache.h:32
static std::span< const std::unique_ptr< FontCache > > Get()
Get span of all FontCaches.
Definition fontcache.h:177
Container with information about a font.
Definition gfx_layout.h:98
TextColour colour
The colour this font has to be.
Definition gfx_layout.h:101
FontIndex font_index
The font we are using.
Definition gfx_layout.h:100
Size limited cache with a least recently used eviction strategy.
Definition lrucache.hpp:23
The layouter performs all the layout work.
Definition gfx_layout.h:161
static void Initialize()
Perform initialization of layout engine.
static void ResetFontCache(FontSize fs)
Reset cached font information.
ptrdiff_t GetCharAtPosition(int x, size_t line_index) const
Get the character that is at a pixel position in the first line of the layouted text.
ParagraphLayouter::Position GetCharPosition(std::string_view::const_iterator ch) const
Get the position of a character in the layout.
static std::unique_ptr< LineCache > linecache
Cache of ParagraphLayout lines.
Definition gfx_layout.h:205
std::string_view string
Pointer to the original string.
Definition gfx_layout.h:162
static void ResetLineCache()
Clear line cache.
Dimension GetBounds()
Get the boundaries of this paragraph.
static LineCacheItem & GetCachedParagraphLayout(std::string_view str, const FontState &state)
Get reference to cache item.
A single line worth of VisualRuns.
Definition gfx_layout.h:142
Position of a glyph within a VisualRun.
Definition gfx_layout.h:117
int16_t right
Right-most position of glyph.
Definition gfx_layout.h:120
constexpr Position(const Point &pt)
Conversion from a single point to a Position.
Definition gfx_layout.h:126
int16_t left
Left-most position of glyph.
Definition gfx_layout.h:119
int16_t top
Top-most position of glyph.
Definition gfx_layout.h:121
Visual run contains data about the bit of text with the same font.
Definition gfx_layout.h:130
Interface to glue fallback and normal layouter into one.
Definition gfx_layout.h:112
Functions to read fonts from files and cache them.
std::vector< std::pair< int, Font > > FontMap
Mapping from index to font.
Definition gfx_layout.h:107
ParagraphLayouter::Position GetCharPosInString(std::string_view str, size_t pos, FontSize start_fontsize=FS_NORMAL)
Get the leading corner of a character in a single-line string relative to the start of the string.
ptrdiff_t GetCharAtPosition(std::string_view str, int x, FontSize start_fontsize=FS_NORMAL)
Get the character from a string that is drawn at a specific position.
FontSize
Available font sizes.
Definition gfx_type.h:248
@ FS_NORMAL
Index of the normal font in the font tables.
Definition gfx_type.h:249
TextColour
Colour of the strings, see _string_colourmap in table/string_colours.h or docs/ottd-colourtext-palett...
Definition gfx_type.h:307
@ TC_FORCED
Ignore colour changes from strings.
Definition gfx_type.h:332
@ TC_FLAGS_MASK
Mask to test if TextColour (with flags) is within limits.
Definition gfx_type.h:335
@ TC_COLOUR_MASK
Mask to test if TextColour (without flags) is within limits.
Definition gfx_type.h:334
Size limited cache map with a least recently used eviction strategy.
Dimensions (a width and height) of a rectangle in 2D.
Text drawing parameters, which can change while drawing a line, but are kept between multiple parts o...
Definition gfx_layout.h:22
void PushColour()
Push the current colour on to the stack.
Definition gfx_layout.h:57
void SetColour(TextColour c)
Switch to new colour c.
Definition gfx_layout.h:37
void PopColour()
Switch to and pop the last saved colour on the stack.
Definition gfx_layout.h:47
void SetFontSize(FontSize f)
Switch to using a new font f.
Definition gfx_layout.h:66
FontSize fontsize
Current font size.
Definition gfx_layout.h:23
FontIndex font_index
Current font index.
Definition gfx_layout.h:24
std::vector< TextColour > colour_stack
Stack of colours to assist with colour switching.
Definition gfx_layout.h:26
TextColour cur_colour
Current text colour.
Definition gfx_layout.h:25
Item in the linecache.
Definition gfx_layout.h:190
FontMap runs
Accessed by our ParagraphLayout::nextLine.
Definition gfx_layout.h:195
Buffer buffer
Accessed by our ParagraphLayout::nextLine.
Definition gfx_layout.h:194
int cached_width
Width used for the cached layout.
Definition gfx_layout.h:201
std::vector< std::unique_ptr< const ParagraphLayouter::Line > > cached_layout
Cached results of line layouting.
Definition gfx_layout.h:200
FontState state_after
Font state after the line.
Definition gfx_layout.h:197
std::unique_ptr< ParagraphLayouter > layout
Layout of the line.
Definition gfx_layout.h:198
Key into the linecache.
Definition gfx_layout.h:165
FontState state_before
Font state at the beginning of the line.
Definition gfx_layout.h:166
std::string str
Source string of the line (including colour and font size codes).
Definition gfx_layout.h:167
const FontState & state_before
Font state at the beginning of the line.
Definition gfx_layout.h:171
std::string_view str
Source string of the line (including colour and font size codes).
Definition gfx_layout.h:172