OpenTTD Source 20250523-master-g321f7e8683
elrail_data.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 <http://www.gnu.org/licenses/>.
6 */
7
13#ifndef ELRAIL_DATA_H
14#define ELRAIL_DATA_H
15
20enum TileLocationGroup : uint8_t {
21 XEVEN_YEVEN = 0,
22 XEVEN_YODD = 1,
23 XODD_YEVEN = 2,
24 XODD_YODD = 3,
25 TLG_END
26};
27
33enum TileSource : uint8_t {
34 TS_HOME = 0,
35 TS_NEIGHBOUR = 1,
36
37 TS_END
38};
39
40static const uint NUM_TRACKS_AT_PCP = 6;
41
49
61
64 {DIAGDIR_NE, DIAGDIR_SW}, // X
65 {DIAGDIR_SE, DIAGDIR_NW}, // Y
66 {DIAGDIR_NW, DIAGDIR_NE}, // UPPER
67 {DIAGDIR_SE, DIAGDIR_SW}, // LOWER
68 {DIAGDIR_SW, DIAGDIR_NW}, // LEFT
69 {DIAGDIR_NE, DIAGDIR_SE}, // RIGHT
70};
71
79 { // X
80 {DIR_NE, DIR_SE, DIR_NW}, // NE
81 DIRECTIONS_ALL, // SE
82 {DIR_SE, DIR_SW, DIR_NW}, // SW
83 DIRECTIONS_ALL // NE
84 },
85 { // Y
90 },
91 { // UPPER
92 {DIR_E, DIR_N, DIR_S},
95 {DIR_W, DIR_N, DIR_S},
96 },
97 { // LOWER
99 {DIR_E, DIR_N, DIR_S},
100 {DIR_W, DIR_N, DIR_S},
102 },
103 { // LEFT
106 {DIR_S, DIR_E, DIR_W},
107 {DIR_N, DIR_E, DIR_W},
108 },
109 { // RIGHT
110 {DIR_N, DIR_E, DIR_W},
111 {DIR_S, DIR_E, DIR_W},
114 },
115};
116
117#define NUM_IGNORE_GROUPS 3
123static const Directions _ignored_pcp[NUM_IGNORE_GROUPS][TLG_END][DIAGDIR_END] = {
124 { // Ignore group 1, X and Y tracks
125 { // X even, Y even
127 {DIR_NE, DIR_SW},
128 {DIR_NW, DIR_SE},
130 }, { // X even, Y odd
133 {DIR_NW, DIR_SE},
134 {DIR_NE, DIR_SW},
135 }, { // X odd, Y even
136 {DIR_NW, DIR_SE},
137 {DIR_NE, DIR_SW},
140 }, { // X odd, Y odd
141 {DIR_NW, DIR_SE},
144 {DIR_NE, DIR_SW},
145 }
146 },
147 { // Ignore group 2, LEFT and RIGHT tracks
148 {
149 {DIR_E, DIR_W},
152 {DIR_E, DIR_W},
153 }, {
155 {DIR_E, DIR_W},
156 {DIR_E, DIR_W},
158 }, {
160 {DIR_E, DIR_W},
161 {DIR_E, DIR_W},
163 }, {
164 {DIR_E, DIR_W},
167 {DIR_E, DIR_W},
168 }
169 },
170 { // Ignore group 3, UPPER and LOWER tracks
171 {
172 {DIR_N, DIR_S},
173 {DIR_N, DIR_S},
176 }, {
179 {DIR_N, DIR_S},
180 {DIR_N, DIR_S},
181 }, {
184 {DIR_N, DIR_S},
185 {DIR_N, DIR_S},
186 }, {
187 {DIR_N, DIR_S},
188 {DIR_N, DIR_S},
191 }
192 }
193};
194
197 {{DIR_SW, DIR_NE}, {}, {DIR_SW, DIR_NE}, {} }, // X
198 {{}, {DIR_NW, DIR_SE}, {}, {DIR_NW, DIR_SE}}, // Y
199 {{DIR_W, DIR_E}, {}, {}, {DIR_W, DIR_E} }, // UPPER
200 {{}, {DIR_W, DIR_E}, {DIR_W, DIR_E}, {} }, // LOWER
201 {{}, {}, {DIR_S, DIR_N}, {DIR_N, DIR_S} }, // LEFT
202 {{DIR_S, DIR_N}, {DIR_S, DIR_N}, {}, {}, }, // RIGHT
203};
204
205/* This array stores which track bits can meet at a tile edge */
206static const Track _tracks_at_pcp[DIAGDIR_END][NUM_TRACKS_AT_PCP] = {
211};
212
213/* takes each of the 6 track bits from the array above and
214 * assigns it to the home tile or neighbour tile */
215static const TileSource _track_source_tile[DIAGDIR_END][NUM_TRACKS_AT_PCP] = {
216 {TS_HOME, TS_NEIGHBOUR, TS_HOME , TS_NEIGHBOUR, TS_NEIGHBOUR, TS_HOME },
217 {TS_HOME, TS_NEIGHBOUR, TS_NEIGHBOUR, TS_HOME , TS_NEIGHBOUR, TS_HOME },
218 {TS_HOME, TS_NEIGHBOUR, TS_NEIGHBOUR, TS_HOME , TS_HOME , TS_NEIGHBOUR},
219 {TS_HOME, TS_NEIGHBOUR, TS_HOME , TS_NEIGHBOUR, TS_HOME , TS_NEIGHBOUR},
220};
221
222/* Several PPPs maybe exist, here they are sorted in order of preference. */
223static const Direction _ppp_order[DIAGDIR_END][TLG_END][DIR_END] = { // X - Y
224 { // PCP 0
225 {DIR_NE, DIR_NW, DIR_SE, DIR_SW, DIR_N, DIR_E, DIR_S, DIR_W}, // evn - evn
226 {DIR_NE, DIR_SE, DIR_SW, DIR_NW, DIR_S, DIR_W, DIR_N, DIR_E}, // evn - odd
227 {DIR_SW, DIR_NW, DIR_NE, DIR_SE, DIR_S, DIR_W, DIR_N, DIR_E}, // odd - evn
228 {DIR_SW, DIR_SE, DIR_NE, DIR_NW, DIR_N, DIR_E, DIR_S, DIR_W}, // odd - odd
229 }, {// PCP 1
230 {DIR_NE, DIR_NW, DIR_SE, DIR_SW, DIR_S, DIR_E, DIR_N, DIR_W}, // evn - evn
231 {DIR_NE, DIR_SE, DIR_SW, DIR_NW, DIR_N, DIR_W, DIR_S, DIR_E}, // evn - odd
232 {DIR_SW, DIR_NW, DIR_NE, DIR_SE, DIR_N, DIR_W, DIR_S, DIR_E}, // odd - evn
233 {DIR_SW, DIR_SE, DIR_NE, DIR_NW, DIR_S, DIR_E, DIR_N, DIR_W}, // odd - odd
234 }, {// PCP 2
235 {DIR_NE, DIR_NW, DIR_SE, DIR_SW, DIR_S, DIR_W, DIR_N, DIR_E}, // evn - evn
236 {DIR_NE, DIR_SE, DIR_SW, DIR_NW, DIR_N, DIR_E, DIR_S, DIR_W}, // evn - odd
237 {DIR_SW, DIR_NW, DIR_NE, DIR_SE, DIR_N, DIR_E, DIR_S, DIR_W}, // odd - evn
238 {DIR_SW, DIR_SE, DIR_NE, DIR_NW, DIR_S, DIR_W, DIR_N, DIR_E}, // odd - odd
239 }, {// PCP 3
240 {DIR_NE, DIR_NW, DIR_SE, DIR_SW, DIR_N, DIR_W, DIR_S, DIR_E}, // evn - evn
241 {DIR_NE, DIR_SE, DIR_SW, DIR_NW, DIR_S, DIR_E, DIR_N, DIR_W}, // evn - odd
242 {DIR_SW, DIR_NW, DIR_NE, DIR_SE, DIR_S, DIR_E, DIR_N, DIR_W}, // odd - evn
243 {DIR_SW, DIR_SE, DIR_NE, DIR_NW, DIR_N, DIR_W, DIR_S, DIR_E}, // odd - odd
244 }
245};
246/* Geometric placement of the PCP relative to the tile origin */
247static const int8_t _x_pcp_offsets[DIAGDIR_END] = {0, 8, 16, 8};
248static const int8_t _y_pcp_offsets[DIAGDIR_END] = {8, 16, 8, 0};
249/* Geometric placement of the PPP relative to the PCP*/
250static const int8_t _x_ppp_offsets[DIR_END] = {-2, -4, -2, 0, 2, 4, 2, 0};
251static const int8_t _y_ppp_offsets[DIR_END] = {-2, 0, 2, 4, 2, 0, -2, -4};
252
256enum PylonSpriteOffset : uint8_t {
257 PSO_Y_NE,
258 PSO_Y_SW,
259 PSO_X_NW,
260 PSO_X_SE,
261 PSO_EW_N,
262 PSO_EW_S,
263 PSO_NS_W,
264 PSO_NS_E,
265};
266
267/* The type of pylon to draw at each PPP */
268static const uint8_t _pylon_sprites[] = {
269 PSO_EW_N,
270 PSO_Y_NE,
271 PSO_NS_E,
272 PSO_X_SE,
273 PSO_EW_S,
274 PSO_Y_SW,
275 PSO_NS_W,
276 PSO_X_NW,
277};
278
282enum WireSpriteOffset : uint8_t {
283 WSO_X_SHORT,
284 WSO_Y_SHORT,
285 WSO_EW_SHORT,
286 WSO_NS_SHORT,
287 WSO_X_SHORT_DOWN,
288 WSO_Y_SHORT_UP,
289 WSO_X_SHORT_UP,
290 WSO_Y_SHORT_DOWN,
291
292 WSO_X_SW,
293 WSO_Y_SE,
294 WSO_EW_E,
295 WSO_NS_S,
296 WSO_X_SW_DOWN,
297 WSO_Y_SE_UP,
298 WSO_X_SW_UP,
299 WSO_Y_SE_DOWN,
300
301 WSO_X_NE,
302 WSO_Y_NW,
303 WSO_EW_W,
304 WSO_NS_N,
305 WSO_X_NE_DOWN,
306 WSO_Y_NW_UP,
307 WSO_X_NE_UP,
308 WSO_Y_NW_DOWN,
309
310 WSO_ENTRANCE_SW,
311 WSO_ENTRANCE_NW,
312 WSO_ENTRANCE_NE,
313 WSO_ENTRANCE_SE,
314};
315
317 uint8_t image_offset;
318 int8_t x_offset;
319 int8_t y_offset;
320 int8_t x_size;
321 int8_t y_size;
322 int8_t z_size;
323 int8_t z_offset;
324};
325
327static const uint ELRAIL_ELEVATION = 10;
331static const uint ELRAIL_ELEVLOWER = ELRAIL_ELEVATION - 1;
332
333static const SortableSpriteStruct _rail_catenary_sprite_data[] = {
334/* X direction
335 * Flat tiles:
336 * Wires */
337 { WSO_X_SW, 0, 7, 15, 1, 1, ELRAIL_ELEVATION },
338 { WSO_X_NE, 0, 7, 15, 1, 1, ELRAIL_ELEVATION },
339 { WSO_X_SHORT, 0, 7, 15, 1, 1, ELRAIL_ELEVATION },
340
341 /* "up" tiles
342 * Wires */
343 { WSO_X_SW_UP, 0, 7, 15, 8, 1, ELRAIL_ELEVRAISE },
344 { WSO_X_NE_UP, 0, 7, 15, 8, 1, ELRAIL_ELEVRAISE },
345 { WSO_X_SHORT_UP, 0, 7, 15, 8, 1, ELRAIL_ELEVRAISE },
346
347 /* "down" tiles
348 * Wires */
349 { WSO_X_SW_DOWN, 0, 7, 15, 8, 1, ELRAIL_ELEVLOWER },
350 { WSO_X_NE_DOWN, 0, 7, 15, 8, 1, ELRAIL_ELEVLOWER },
351 { WSO_X_SHORT_DOWN, 0, 7, 15, 8, 1, ELRAIL_ELEVLOWER },
352
353
354/* Y direction
355 * Flat tiles:
356 * Wires */
357 { WSO_Y_SE, 7, 0, 1, 15, 1, ELRAIL_ELEVATION },
358 { WSO_Y_NW, 7, 0, 1, 15, 1, ELRAIL_ELEVATION },
359 { WSO_Y_SHORT, 7, 0, 1, 15, 1, ELRAIL_ELEVATION },
360
361 /* "up" tiles
362 * Wires */
363 { WSO_Y_SE_UP, 7, 0, 8, 15, 1, ELRAIL_ELEVRAISE },
364 { WSO_Y_NW_UP, 7, 0, 8, 15, 1, ELRAIL_ELEVRAISE },
365 { WSO_Y_SHORT_UP, 7, 0, 8, 15, 1, ELRAIL_ELEVRAISE },
366
367 /* "down" tiles
368 * Wires */
369 { WSO_Y_SE_DOWN, 7, 0, 8, 15, 1, ELRAIL_ELEVLOWER },
370 { WSO_Y_NW_DOWN, 7, 0, 8, 15, 1, ELRAIL_ELEVLOWER },
371 { WSO_Y_SHORT_DOWN, 7, 0, 8, 15, 1, ELRAIL_ELEVLOWER },
372
373/* NS Direction */
374 { WSO_NS_SHORT, 8, 0, 8, 8, 1, ELRAIL_ELEVATION },
375 { WSO_NS_SHORT, 0, 8, 8, 8, 1, ELRAIL_ELEVATION },
376
377 { WSO_NS_N, 8, 0, 8, 8, 1, ELRAIL_ELEVATION },
378 { WSO_NS_N, 0, 8, 8, 8, 1, ELRAIL_ELEVATION },
379
380 { WSO_NS_S, 8, 0, 8, 8, 1, ELRAIL_ELEVATION },
381 { WSO_NS_S, 0, 8, 8, 8, 1, ELRAIL_ELEVATION },
382
383/* EW Direction */
384 { WSO_EW_SHORT, 7, 0, 1, 1, 1, ELRAIL_ELEVATION },
385 { WSO_EW_SHORT, 15, 8, 3, 3, 1, ELRAIL_ELEVATION },
386
387 { WSO_EW_W, 7, 0, 1, 1, 1, ELRAIL_ELEVATION },
388 { WSO_EW_W, 15, 8, 3, 3, 1, ELRAIL_ELEVATION },
389
390 { WSO_EW_E, 7, 0, 1, 1, 1, ELRAIL_ELEVATION },
391 { WSO_EW_E, 15, 8, 3, 3, 1, ELRAIL_ELEVATION }
392};
393
394static const SortableSpriteStruct _rail_catenary_sprite_data_depot[] = {
395 { WSO_ENTRANCE_NE, 0, 7, 15, 1, 1, ELRAIL_ELEVATION },
396 { WSO_ENTRANCE_SE, 7, 0, 1, 15, 1, ELRAIL_ELEVATION },
397 { WSO_ENTRANCE_SW, 0, 7, 15, 1, 1, ELRAIL_ELEVATION },
398 { WSO_ENTRANCE_NW, 7, 0, 1, 15, 1, ELRAIL_ELEVATION }
399};
400
401static const SortableSpriteStruct _rail_catenary_sprite_data_tunnel[] = {
402 { WSO_ENTRANCE_SW, 0, 7, 15, 1, 1, ELRAIL_ELEVATION },
403 { WSO_ENTRANCE_NW, 7, 0, 1, 15, 1, ELRAIL_ELEVATION },
404 { WSO_ENTRANCE_NE, 0, 7, 15, 1, 1, ELRAIL_ELEVATION },
405 { WSO_ENTRANCE_SE, 7, 0, 1, 15, 1, ELRAIL_ELEVATION }
406};
407
408
421enum RailCatenarySprite : uint8_t {
422 WIRE_X_FLAT_SW,
423 WIRE_X_FLAT_NE,
424 WIRE_X_FLAT_BOTH,
425
426 WIRE_X_UP_SW,
427 WIRE_X_UP_NE,
428 WIRE_X_UP_BOTH,
429
430 WIRE_X_DOWN_SW,
431 WIRE_X_DOWN_NE,
432 WIRE_X_DOWN_BOTH,
433
434 WIRE_Y_FLAT_SE,
435 WIRE_Y_FLAT_NW,
436 WIRE_Y_FLAT_BOTH,
437
438 WIRE_Y_UP_SE,
439 WIRE_Y_UP_NW,
440 WIRE_Y_UP_BOTH,
441
442 WIRE_Y_DOWN_SE,
443 WIRE_Y_DOWN_NW,
444 WIRE_Y_DOWN_BOTH,
445
446 WIRE_NS_W_BOTH,
447 WIRE_NS_E_BOTH,
448
449 WIRE_NS_W_N,
450 WIRE_NS_E_N,
451
452 WIRE_NS_W_S,
453 WIRE_NS_E_S,
454
455 WIRE_EW_N_BOTH,
456 WIRE_EW_S_BOTH,
457
458 WIRE_EW_N_W,
459 WIRE_EW_S_W,
460
461 WIRE_EW_N_E,
462 WIRE_EW_S_E,
463
464 INVALID_CATENARY = 0xFF
465};
466
467/* Selects a Wire (with white and grey ends) depending on whether:
468 * a) none (should never happen)
469 * b) the first
470 * c) the second
471 * d) both
472 * PCP exists.*/
473static const RailCatenarySprite _rail_wires[5][TRACK_END][4] = {
474 { // Tileh == 0
475 {INVALID_CATENARY, WIRE_X_FLAT_NE, WIRE_X_FLAT_SW, WIRE_X_FLAT_BOTH},
476 {INVALID_CATENARY, WIRE_Y_FLAT_SE, WIRE_Y_FLAT_NW, WIRE_Y_FLAT_BOTH},
477 {INVALID_CATENARY, WIRE_EW_N_W, WIRE_EW_N_E, WIRE_EW_N_BOTH},
478 {INVALID_CATENARY, WIRE_EW_S_E, WIRE_EW_S_W, WIRE_EW_S_BOTH},
479 {INVALID_CATENARY, WIRE_NS_W_S, WIRE_NS_W_N, WIRE_NS_W_BOTH},
480 {INVALID_CATENARY, WIRE_NS_E_N, WIRE_NS_E_S, WIRE_NS_E_BOTH},
481 }, { // Tileh == 3
482 {INVALID_CATENARY, WIRE_X_UP_NE, WIRE_X_UP_SW, WIRE_X_UP_BOTH},
483 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
484 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
485 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
486 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
487 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
488 }, { // Tileh == 6
489 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
490 {INVALID_CATENARY, WIRE_Y_UP_SE, WIRE_Y_UP_NW, WIRE_Y_UP_BOTH},
491 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
492 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
493 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
494 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
495 }, { // Tileh == 9
496 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
497 {INVALID_CATENARY, WIRE_Y_DOWN_SE, WIRE_Y_DOWN_NW, WIRE_Y_DOWN_BOTH},
498 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
499 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
500 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
501 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
502 }, { // Tileh == 12
503 {INVALID_CATENARY, WIRE_X_DOWN_NE, WIRE_X_DOWN_SW, WIRE_X_DOWN_BOTH},
504 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
505 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
506 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
507 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
508 {INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY, INVALID_CATENARY},
509 }
510};
511
512#endif /* ELRAIL_DATA_H */
Enum-as-bit-set wrapper.
Direction
Defines the 8 directions on the map.
@ DIR_SW
Southwest.
@ DIR_NW
Northwest.
@ DIR_N
North.
@ DIR_SE
Southeast.
@ DIR_S
South.
@ DIR_NE
Northeast.
@ DIR_END
Used to iterate.
@ DIR_W
West.
@ DIR_E
East.
DiagDirection
Enumeration for diagonal directions.
@ DIAGDIR_NE
Northeast, upper right on your monitor.
@ DIAGDIR_NW
Northwest.
@ DIAGDIR_SE
Southeast.
@ DIAGDIR_END
Used for iterations.
@ DIAGDIR_SW
Southwest.
static constexpr Directions DIRECTIONS_ALL
All possible directions.
static const uint ELRAIL_ELEVRAISE
Wires that a draw one level higher than the north corner.
PylonSpriteOffset
Offset for pylon sprites from the base pylon sprite.
static const uint ELRAIL_ELEVLOWER
Wires that a draw one level lower than the north corner.
TileSource
When determining the pylon configuration on the edge, two tiles are taken into account: the tile bein...
Definition elrail_data.h:33
RailCatenarySprite
Refers to a certain element of the catenary.
static const Directions _owned_ppp_on_pcp[DIAGDIR_END]
Which of the PPPs are inside the tile.
Definition elrail_data.h:55
WireSpriteOffset
Offset for wire sprites from the base wire sprite.
static const Directions _disallowed_ppp_of_track_at_pcp[TRACK_END][DIAGDIR_END]
Which pylons can definitely NOT be built.
static const DiagDirection _pcp_positions[TRACK_END][2]
Maps a track bit onto two PCP positions.
Definition elrail_data.h:63
static const Directions _ignored_pcp[NUM_IGNORE_GROUPS][TLG_END][DIAGDIR_END]
In case we have a straight line, we place pylon only every two tiles, so there are certain tiles whic...
static const Directions _preferred_ppp_of_track_at_pcp[TRACK_END][DIAGDIR_END]
Preferred points of each trackbit.
Definition elrail_data.h:78
static const Directions _allowed_ppp_on_pcp[DIAGDIR_END]
Which PPPs are possible at all on a given PCP.
Definition elrail_data.h:43
TileLocationGroup
Tile Location group.
Definition elrail_data.h:20
static const uint ELRAIL_ELEVATION
Distance between wire and rail.
static const uint TILE_HEIGHT
Height of a height level in world coordinate AND in pixels in ZOOM_BASE.
Definition tile_type.h:18
Track
These are used to specify a single track.
Definition track_type.h:19
@ TRACK_Y
Track along the y-axis (north-west to south-east)
Definition track_type.h:22
@ TRACK_LOWER
Track in the lower corner of the tile (south)
Definition track_type.h:24
@ TRACK_END
Used for iterations.
Definition track_type.h:27
@ TRACK_LEFT
Track in the left corner of the tile (west)
Definition track_type.h:25
@ TRACK_RIGHT
Track in the right corner of the tile (east)
Definition track_type.h:26
@ TRACK_X
Track along the x-axis (north-east to south-west)
Definition track_type.h:21
@ TRACK_UPPER
Track in the upper corner of the tile (north)
Definition track_type.h:23