2025-01-14 13:59:44 +01:00
|
|
|
#ifndef _LCR_GAME_H
|
|
|
|
#define _LCR_GAME_H
|
|
|
|
|
2025-01-20 21:33:05 +01:00
|
|
|
/*
|
|
|
|
Licar: game module
|
|
|
|
|
|
|
|
This file implements the backend of a complete, actually playable game with
|
|
|
|
graphics, sound etc., and is meant to be included and used by specific
|
|
|
|
frontends (which will handle each platform's hardware details and I/O). See
|
|
|
|
the frontend info below for help with porting the game to a new platform.
|
2024-09-06 00:58:32 +02:00
|
|
|
|
2025-01-20 21:33:05 +01:00
|
|
|
The code uses LCR_ (or _LCR) prefix as a kind of namespace preventing
|
|
|
|
collision with 3rd party identifiers.
|
2024-09-06 00:58:32 +02:00
|
|
|
|
|
|
|
UNITS: There are various kinds of units used to ensure independence of the
|
2025-01-20 21:33:05 +01:00
|
|
|
modules. Here is a summary:
|
2024-09-06 00:58:32 +02:00
|
|
|
|
|
|
|
- LCR_GameUnit: data type, abstract unit of the game (racing module). One map
|
|
|
|
square is LCR_GAME_UNITs long, a full angle is also LCR_GAME_UNITs.
|
|
|
|
- LCR_GAME_UNIT: Size of one game square and full angle in LCR_GameUnits.
|
2024-11-21 21:33:36 +01:00
|
|
|
Square height is only half of this.
|
2024-09-06 00:58:32 +02:00
|
|
|
- S3L_Unit: data type, small3dlib's unit. May change with renderer change.
|
2024-09-09 19:16:51 +02:00
|
|
|
- S3L_F: small3dlib's value representing 1.0.
|
2024-09-06 00:58:32 +02:00
|
|
|
- LCR_RENDERER_UNIT: for the renderer one map square is this many S3L_Units.
|
|
|
|
- TPE_Unit: tinyphysicsengine's unit. May change with phys. engine change.
|
2024-09-09 19:16:51 +02:00
|
|
|
- TPE_F: tinyphysicsengine's value representing value 1.0.
|
2024-09-06 00:58:32 +02:00
|
|
|
- LCR_PHYSICS_UNIT: for the phys. eng. one map square is this many TPE_Units.
|
|
|
|
|
|
|
|
COORDINATE SYSTEM AND ROTATIONS: The game itself (racing module) is
|
|
|
|
independent of rendering and physics libraries, but out of convenient adopts
|
|
|
|
their coordinate system (X right, Y up, Z forward) and rotations (Euler
|
2024-09-10 15:30:07 +02:00
|
|
|
angles, by Z, then by X, then Y).
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-20 21:33:05 +01:00
|
|
|
DATA FILE: The game uses so called data file to store various resources,
|
2025-01-20 19:58:56 +01:00
|
|
|
mainly maps and replays. There is considered to be one abstract global file
|
|
|
|
which is just a long text string. Internally the global file is composed of
|
|
|
|
a hardcoded internal data file string (stored in assets) with very basic maps,
|
|
|
|
and an optional user file (appended to the internal file) that the frontend
|
|
|
|
may provide, allowing adding more resources without recompiling the game. The
|
|
|
|
user file may be disabled on platforms that e.g. don't have file systems, but
|
|
|
|
the internal file will be always present. The format of the data file is
|
|
|
|
following: it consists of data strings separated by '#' character (data
|
|
|
|
strings cannot contain this character). Each data string starts with a magic
|
|
|
|
number: a single character identifying the type of the resource (map, replay,
|
|
|
|
...), then the name of the data follows, then character ';' and then the data
|
|
|
|
string itself (up until next '#' or end of file). The format of the string
|
|
|
|
depends on the type of the data, i.e. the format of map string has a special
|
|
|
|
format (described in the map module) etc.
|
2023-09-10 14:43:20 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#define LCR_KEY_UP 0x00
|
|
|
|
#define LCR_KEY_RIGHT 0x01
|
|
|
|
#define LCR_KEY_DOWN 0x02
|
|
|
|
#define LCR_KEY_LEFT 0x03
|
|
|
|
#define LCR_KEY_A 0x04 ///< confirm, restart race
|
|
|
|
#define LCR_KEY_B 0x05 ///< cancel, open menu
|
|
|
|
|
|
|
|
#define LCR_KEYS_TOTAL 6
|
|
|
|
|
|
|
|
/*
|
2025-01-20 21:33:05 +01:00
|
|
|
FOR FRONTENDS (porting to other platforms):
|
2023-09-10 14:43:20 +02:00
|
|
|
- Implement the below described functions according to their description.
|
|
|
|
- Implement the main program and game loop.
|
|
|
|
- Call the below described functions as described.
|
2024-12-23 22:50:07 +01:00
|
|
|
- If you want to support music, make your frontend play music from the "music"
|
|
|
|
file in assets. It is in raw format, storing 8bit unsigned samples at 8000
|
2025-01-07 13:17:41 +01:00
|
|
|
Hz. Use the LCR_gameMusicOn to check what the music volume is. If you
|
2024-12-23 22:50:07 +01:00
|
|
|
don't support music, set LCR_SETTING_MUSIC to 0 in your frontend code so
|
|
|
|
that the game knows music is disabled.
|
2023-09-10 14:43:20 +02:00
|
|
|
*/
|
2023-08-08 20:34:21 +02:00
|
|
|
|
2023-09-10 14:43:20 +02:00
|
|
|
/**
|
|
|
|
Implement this in your frontend. Returns 1 if given key is pressed or 0
|
|
|
|
otherwise.
|
|
|
|
*/
|
2023-08-08 20:34:21 +02:00
|
|
|
uint8_t LCR_keyPressed(uint8_t key);
|
|
|
|
|
2023-09-10 14:43:20 +02:00
|
|
|
/**
|
|
|
|
Implement this in your frontend. This function pauses program execution for
|
|
|
|
given amount of milliseconds and relieves the CPU usage. On platforms that
|
|
|
|
don't support this the function may simply do nothing.
|
|
|
|
*/
|
2023-08-08 20:34:21 +02:00
|
|
|
void LCR_sleep(uint16_t timeMs);
|
|
|
|
|
2023-09-10 14:43:20 +02:00
|
|
|
/**
|
|
|
|
Implement this in your frontend. This function draws a pixel of given color
|
|
|
|
to the screen back buffer (i.e. NOT directly to screen, back buffer shall
|
|
|
|
only be copied to front buffer once the LCR_gameStep function finishes all
|
|
|
|
rendering). This function should NOT check for out-of-screen coordinates, this
|
|
|
|
is handled by the game internals and out-of-screen pixels will never be drawn.
|
|
|
|
The color value depends on game settings but is normally an RGB565 value.
|
|
|
|
*/
|
2024-07-22 01:16:16 +02:00
|
|
|
void LCR_drawPixel(unsigned long index, uint16_t color);
|
2023-08-08 20:34:21 +02:00
|
|
|
|
2024-08-02 00:05:03 +02:00
|
|
|
/**
|
|
|
|
Implement this in your frontend. This function will be called to log what the
|
|
|
|
program is doing. If you want to ignore logging, simply make the function do
|
|
|
|
nothing.
|
|
|
|
*/
|
|
|
|
void LCR_log(const char *str);
|
|
|
|
|
2024-12-30 00:49:41 +01:00
|
|
|
/**
|
|
|
|
Implement this in your frontend. This function serves for loading optional
|
2025-01-20 19:58:56 +01:00
|
|
|
data file that allows to add more maps, replays etc. If your frontend
|
2024-12-30 00:49:41 +01:00
|
|
|
won't support this, just make the function return 0. Otherwise it must return
|
2025-01-20 19:58:56 +01:00
|
|
|
characters from the data file one by one; after reaching the end of file
|
2024-12-30 00:49:41 +01:00
|
|
|
0 must be returned and the reading position will be reset to start again.
|
|
|
|
*/
|
2025-01-20 19:58:56 +01:00
|
|
|
char LCR_getNextDataFileChar(void);
|
2024-12-30 00:49:41 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
Implement this in your frontend. This serves to store data in the optional
|
2025-01-20 19:58:56 +01:00
|
|
|
data file, e.g. replays. If your frontend doesn't support this (e.g.
|
2024-12-30 00:49:41 +01:00
|
|
|
because the file is read only), the function may ignore the append, but if
|
|
|
|
the file is otherwise supported, a rewind of the read position must still be
|
|
|
|
done. If appending is supported, the function must append the provided string
|
2025-01-20 19:58:56 +01:00
|
|
|
to the data file AND then reset the data file reading position back to
|
2024-12-30 00:49:41 +01:00
|
|
|
the start.
|
|
|
|
*/
|
2025-01-20 19:58:56 +01:00
|
|
|
void LCR_appendDataStr(const char *str);
|
2024-12-30 00:49:41 +01:00
|
|
|
|
2023-08-08 20:34:21 +02:00
|
|
|
/**
|
2023-09-10 14:43:20 +02:00
|
|
|
Call this function in your frontend at the start of the program.
|
2023-08-08 20:34:21 +02:00
|
|
|
*/
|
|
|
|
void LCR_gameInit(void);
|
|
|
|
|
|
|
|
/**
|
2023-09-10 14:43:20 +02:00
|
|
|
Call this function in your frontend right before program end.
|
2023-08-08 20:34:21 +02:00
|
|
|
*/
|
|
|
|
void LCR_gameEnd(void);
|
|
|
|
|
|
|
|
/**
|
2023-09-10 14:43:20 +02:00
|
|
|
Call this function in your frontend repeatedly inside the main loop, pass the
|
|
|
|
current time as the number of milliseconds since program start. This function
|
|
|
|
will perform the game step AND other things such as checking the input states,
|
|
|
|
rendering or sleeping (all using the above functions you should implement).
|
|
|
|
Returns 0 if program should end, otherwise 1.
|
2023-08-08 20:34:21 +02:00
|
|
|
*/
|
|
|
|
uint8_t LCR_gameStep(uint32_t timeMs);
|
|
|
|
|
2024-12-23 22:50:07 +01:00
|
|
|
/**
|
|
|
|
Gets the current music volume;
|
|
|
|
*/
|
2025-01-07 13:17:41 +01:00
|
|
|
uint8_t LCR_gameMusicOn(void);
|
2024-12-23 22:50:07 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
Gets next audio sample (unsigned 8bit samples, 8 KHz).
|
|
|
|
*/
|
|
|
|
uint8_t LCR_gameGetNextAudioSample(void);
|
|
|
|
|
2023-08-08 20:34:21 +02:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
2024-09-27 00:08:52 +02:00
|
|
|
#define LCR_LOG0(s) ;
|
|
|
|
#define LCR_LOG1(s) ;
|
|
|
|
#define LCR_LOG2(s) ;
|
|
|
|
|
|
|
|
#if LCR_SETTING_LOG_LEVEL > 0
|
|
|
|
#undef LCR_LOG0
|
|
|
|
#define LCR_LOG0(s) LCR_log(s);
|
|
|
|
|
|
|
|
#if LCR_SETTING_LOG_LEVEL > 1
|
|
|
|
#undef LCR_LOG1
|
|
|
|
#define LCR_LOG1(s) LCR_log(s);
|
|
|
|
|
|
|
|
#if LCR_SETTING_LOG_LEVEL > 2
|
|
|
|
#undef LCR_LOG2
|
|
|
|
#define LCR_LOG2(s) LCR_log(s);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2025-01-07 20:48:04 +01:00
|
|
|
#define LCR_CAMERA_MODE_DRIVE 0x00
|
|
|
|
#define LCR_CAMERA_MODE_DRIVE2 0x01
|
|
|
|
#define LCR_CAMERA_MODE_INSIDE 0x02
|
|
|
|
#define LCR_CAMERA_MODE_FREE 0x03
|
2024-09-09 19:16:51 +02:00
|
|
|
|
2024-11-27 20:30:54 +01:00
|
|
|
#define LCR_GAME_STATE_MENU 0x00
|
|
|
|
#define LCR_GAME_STATE_RUN_STARTING 0x01
|
|
|
|
#define LCR_GAME_STATE_RUN 0x02
|
|
|
|
#define LCR_GAME_STATE_RUN_FINISHED 0x03
|
2025-01-19 22:19:44 +01:00
|
|
|
|
2025-01-22 00:06:15 +01:00
|
|
|
#define LCR_GAME_STATE_LOADING 0x04
|
2025-01-19 22:19:44 +01:00
|
|
|
|
2025-01-03 01:15:24 +01:00
|
|
|
#define LCR_GAME_STATE_END 0xff
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-08 22:05:54 +01:00
|
|
|
// forward decls of pixel drawing functions for the renderer
|
|
|
|
|
|
|
|
/**
|
|
|
|
Internal pixel drawing function that draws pixel at specified screen coords
|
|
|
|
without checking for safety (it's faster but can only be done if we know for
|
|
|
|
sure we're not drawing outside the screen).
|
|
|
|
*/
|
|
|
|
void LCR_drawPixelXYUnsafe(unsigned int x, unsigned int y, uint16_t color);
|
|
|
|
|
|
|
|
/**
|
|
|
|
Internal pixel drawing function that checks for out-of-screen coordinates. Use
|
|
|
|
this if the pixel can potentially lie of screen (however if you know it won't,
|
|
|
|
use the normal unsafe function in sake of performance).
|
|
|
|
*/
|
|
|
|
static inline void LCR_drawPixelXYSafe(unsigned int x, unsigned int y,
|
|
|
|
uint_fast16_t color);
|
|
|
|
|
2025-01-14 13:59:44 +01:00
|
|
|
#include "general.h"
|
2025-01-08 22:05:54 +01:00
|
|
|
#include "racing.h"
|
|
|
|
#include "audio.h"
|
|
|
|
#include "assets.h"
|
|
|
|
#include "renderer.h"
|
|
|
|
|
2025-01-16 15:00:13 +01:00
|
|
|
#define LCR_MENU_MAX_ITEMS 9 // don't change
|
|
|
|
#define LCR_RESOURCE_ITEM_CHUNK (LCR_MENU_MAX_ITEMS - 1)
|
|
|
|
#define LCR_MENU_TABS 4
|
2025-01-21 22:46:56 +01:00
|
|
|
|
|
|
|
#if LCR_SETTING_GHOST_MAX_SAMPLES == 0
|
|
|
|
#undef LCR_MENU_TABS
|
|
|
|
#define LCR_MENU_TABS 3
|
|
|
|
#endif
|
|
|
|
|
2025-01-16 15:00:13 +01:00
|
|
|
#define LCR_MENU_STRING_SIZE 16
|
|
|
|
|
|
|
|
#define LCR_RESOURCE_FILE_SEPARATOR '#'
|
|
|
|
#define LCR_RESOURCE_FILE_SEPARATOR2 ';'
|
|
|
|
|
|
|
|
#define LCR_FREE_CAMERA_STEP \
|
|
|
|
((LCR_SETTING_FREE_CAMERA_SPEED * LCR_GAME_UNIT / 8) \
|
|
|
|
/ LCR_SETTING_FPS)
|
|
|
|
|
|
|
|
#if LCR_FREE_CAMERA_STEP == 0
|
|
|
|
#define LCR_FREE_CAMERA_STEP 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define LCR_FREE_CAMERA_TURN_STEP \
|
|
|
|
((LCR_SETTING_FREE_CAMERA_TURN_SPEED * LCR_GAME_UNIT) \
|
|
|
|
/ (360 * LCR_SETTING_FPS))
|
|
|
|
|
|
|
|
#if LCR_FREE_CAMERA_TURN_STEP == 0
|
|
|
|
#define LCR_FREE_CAMERA_TURN_STEP 1
|
|
|
|
#endif
|
|
|
|
|
2024-09-05 22:51:11 +02:00
|
|
|
struct
|
|
|
|
{
|
2024-11-27 20:30:54 +01:00
|
|
|
uint8_t state;
|
|
|
|
uint32_t stateStartTime;
|
|
|
|
uint32_t time;
|
2024-12-30 00:49:41 +01:00
|
|
|
uint32_t frame;
|
2024-09-05 22:51:11 +02:00
|
|
|
uint32_t nextRenderFrameTime;
|
|
|
|
uint32_t nextRacingTickTime;
|
2025-01-07 20:48:04 +01:00
|
|
|
uint8_t cameraMode;
|
2024-09-09 19:16:51 +02:00
|
|
|
uint8_t debugDraw;
|
2025-01-07 13:17:41 +01:00
|
|
|
uint8_t musicOn;
|
2025-01-01 01:23:43 +01:00
|
|
|
uint8_t keyStates[LCR_KEYS_TOTAL]; /**< Assures unchanging key states
|
|
|
|
during a single frame, hold number of
|
|
|
|
frames for which the key has been
|
2025-01-08 00:10:27 +01:00
|
|
|
continuously held. */
|
|
|
|
uint32_t runTimeMS; /**< Current time of the run */
|
2025-01-07 23:21:08 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
uint8_t selectedTab;
|
|
|
|
uint8_t selectedItem;
|
|
|
|
uint8_t itemCount;
|
|
|
|
char itemNames[LCR_MENU_MAX_ITEMS][LCR_MENU_STRING_SIZE];
|
|
|
|
const char *itemNamePtrs[LCR_MENU_MAX_ITEMS]; ///< helper array
|
|
|
|
} menu;
|
2024-12-30 22:19:41 +01:00
|
|
|
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
int state; ///< -1 if reading external res. f., else pos.
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
// indices and counts are among the data of the same type
|
2025-01-03 01:15:24 +01:00
|
|
|
unsigned int firstItemIndex;
|
|
|
|
unsigned int itemsTotal;
|
2025-01-20 19:58:56 +01:00
|
|
|
} dataFile;
|
2025-01-21 22:46:56 +01:00
|
|
|
|
|
|
|
#define LCR_GHOST_SAMPLE_SIZE 5
|
|
|
|
|
|
|
|
#if LCR_SETTING_GHOST_MAX_SAMPLES != 0
|
|
|
|
struct
|
|
|
|
{
|
2025-01-22 00:18:28 +01:00
|
|
|
uint8_t active;
|
2025-01-21 22:46:56 +01:00
|
|
|
uint8_t samples[LCR_SETTING_GHOST_MAX_SAMPLES * LCR_GHOST_SAMPLE_SIZE];
|
2025-01-22 15:35:58 +01:00
|
|
|
/**< Samples, each 5 bytes: 9 bits for X and Z, 10 for Y,
|
2025-01-21 22:46:56 +01:00
|
|
|
4 for each rotation component. */
|
|
|
|
uint8_t stretch; /**< Stretch of the base sample step, as a bit shift
|
|
|
|
(i.e. 1 means the step will be 2x as long etc.). This
|
|
|
|
is to allow ghosts for even long replays. */
|
|
|
|
} ghost;
|
|
|
|
#endif
|
|
|
|
|
2024-09-05 22:51:11 +02:00
|
|
|
} LCR_game;
|
2023-09-17 13:21:19 +02:00
|
|
|
|
2025-01-07 13:17:41 +01:00
|
|
|
uint8_t LCR_gameMusicOn(void)
|
2024-12-23 22:50:07 +01:00
|
|
|
{
|
|
|
|
#if LCR_SETTING_MUSIC
|
2025-01-07 13:17:41 +01:00
|
|
|
return LCR_game.musicOn;
|
2024-12-23 22:50:07 +01:00
|
|
|
#else
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2024-07-22 01:16:16 +02:00
|
|
|
void LCR_drawPixelXYUnsafe(unsigned int x, unsigned int y,
|
|
|
|
uint16_t color)
|
2023-09-10 14:43:20 +02:00
|
|
|
{
|
2023-09-11 20:56:04 +02:00
|
|
|
#if LCR_SETTING_RESOLUTION_SUBDIVIDE == 1
|
2024-07-22 01:16:16 +02:00
|
|
|
LCR_drawPixel(y * LCR_SETTING_RESOLUTION_X + x,color);
|
2023-09-11 20:56:04 +02:00
|
|
|
#else
|
2024-07-22 01:16:16 +02:00
|
|
|
// TODO
|
2023-09-11 20:56:04 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2025-01-14 13:50:13 +01:00
|
|
|
void _LCR_physicdDebugDrawPixel(uint16_t x, uint16_t y, uint8_t color)
|
|
|
|
{
|
|
|
|
if (x > 1 && x < LCR_EFFECTIVE_RESOLUTION_X - 2 &&
|
|
|
|
y > 1 && y < LCR_EFFECTIVE_RESOLUTION_Y - 2)
|
|
|
|
{
|
|
|
|
uint16_t c = 0x8101 | (0x8f1f << (2 * color));
|
|
|
|
|
|
|
|
for (int j = -1; j <= 2; ++j)
|
|
|
|
for (int i = -1; i <= 2; ++i)
|
|
|
|
LCR_drawPixelXYUnsafe(x + i,y + j,c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-22 01:16:16 +02:00
|
|
|
static inline void LCR_drawPixelXYSafe(unsigned int x, unsigned int y,
|
2023-09-11 20:56:04 +02:00
|
|
|
uint_fast16_t color)
|
|
|
|
{
|
|
|
|
if (x < LCR_EFFECTIVE_RESOLUTION_X && y < LCR_EFFECTIVE_RESOLUTION_Y)
|
2024-07-22 01:16:16 +02:00
|
|
|
LCR_drawPixelXYUnsafe(x,y,color);
|
2023-09-10 14:43:20 +02:00
|
|
|
}
|
|
|
|
|
2024-11-27 20:30:54 +01:00
|
|
|
void LCR_gameSetState(uint8_t state)
|
|
|
|
{
|
|
|
|
LCR_LOG1("changing state");
|
|
|
|
LCR_game.state = state;
|
|
|
|
LCR_game.stateStartTime = LCR_game.time;
|
|
|
|
}
|
|
|
|
|
2025-01-22 00:18:28 +01:00
|
|
|
void LCR_gameResetRun(uint8_t replay, uint8_t ghost)
|
2024-11-24 21:21:29 +01:00
|
|
|
{
|
2024-12-18 00:18:31 +01:00
|
|
|
LCR_GameUnit carTransform[6];
|
|
|
|
|
2024-11-24 21:21:29 +01:00
|
|
|
LCR_LOG0("resetting run");
|
|
|
|
LCR_mapReset();
|
2025-01-13 15:20:30 +01:00
|
|
|
LCR_racingRestart(replay);
|
2024-11-24 21:21:29 +01:00
|
|
|
LCR_rendererUnmarkCPs();
|
2024-12-18 00:18:31 +01:00
|
|
|
LCR_racingGetCarTransform(carTransform,carTransform + 3,0);
|
|
|
|
LCR_rendererSetCarTransform(carTransform,carTransform + 3);
|
|
|
|
LCR_rendererCameraReset();
|
2025-01-22 00:18:28 +01:00
|
|
|
LCR_game.ghost.active = ghost;
|
2024-11-27 20:30:54 +01:00
|
|
|
LCR_gameSetState(LCR_GAME_STATE_RUN_STARTING);
|
2025-01-07 23:21:08 +01:00
|
|
|
LCR_game.runTimeMS = 0;
|
2024-11-27 20:30:54 +01:00
|
|
|
}
|
|
|
|
|
2025-01-22 15:35:58 +01:00
|
|
|
void _LCR_gameGetNthGhostSample(unsigned int n,
|
|
|
|
LCR_GameUnit position[3], LCR_GameUnit rotation[3])
|
|
|
|
{
|
|
|
|
n *= LCR_GHOST_SAMPLE_SIZE;
|
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
position[0] = LCR_game.ghost.samples[n];
|
|
|
|
n++;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
position[0] |= ((LCR_GameUnit) (LCR_game.ghost.samples[n] & 0x01)) << 8;
|
|
|
|
position[1] = LCR_game.ghost.samples[n] >> 1;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
n++;
|
|
|
|
position[1] |= ((LCR_GameUnit) (LCR_game.ghost.samples[n] & 0x07)) << 7;
|
|
|
|
position[2] = LCR_game.ghost.samples[n] >> 3;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
n++;
|
|
|
|
position[2] |= ((LCR_GameUnit) (LCR_game.ghost.samples[n] & 0x0f)) << 5;
|
|
|
|
rotation[0] = LCR_game.ghost.samples[n] >> 4;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
n++;
|
|
|
|
rotation[1] = LCR_game.ghost.samples[n] & 0x0f;
|
|
|
|
rotation[2] = LCR_game.ghost.samples[n] >> 4;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
for (int i = 0; i < 3; ++i)
|
|
|
|
{
|
|
|
|
if (i != 1)
|
|
|
|
position[i] <<= 1;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
position[i] = (position[i] * LCR_GAME_UNIT) / 16;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
position[i] -= (LCR_MAP_SIZE_BLOCKS / 2) *
|
|
|
|
(i == 1 ? LCR_GAME_UNIT / 2 : LCR_GAME_UNIT);
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
rotation[i] = (rotation[i] * LCR_GAME_UNIT) / 16;
|
|
|
|
}
|
2025-01-22 15:35:58 +01:00
|
|
|
}
|
|
|
|
|
2025-01-22 00:18:28 +01:00
|
|
|
void LCR_gameGhostGetTransform(uint32_t frame,
|
|
|
|
LCR_GameUnit position[3], LCR_GameUnit rotation[3])
|
2025-01-22 00:06:15 +01:00
|
|
|
{
|
2025-01-22 15:35:58 +01:00
|
|
|
unsigned int n = ((frame >> LCR_game.ghost.stretch) / LCR_SETTING_GHOST_STEP);
|
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
_LCR_gameGetNthGhostSample(n % 64,position,rotation);
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
if (n < LCR_SETTING_GHOST_MAX_SAMPLES - 1)
|
|
|
|
{
|
|
|
|
LCR_GameUnit carTransform[6];
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
// interpolate:
|
|
|
|
_LCR_gameGetNthGhostSample((n % 64) + 1,carTransform,carTransform + 3);
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
n = (frame >> LCR_game.ghost.stretch) % LCR_SETTING_GHOST_STEP;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
for (int i = 0; i < 3; ++i)
|
|
|
|
{
|
|
|
|
position[i] = position[i] + ((carTransform[i] - position[i]) * n) /
|
2025-01-22 15:35:58 +01:00
|
|
|
LCR_SETTING_GHOST_STEP;
|
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
// rotations are a bit harder to interpolate (e.g. 1 -> 359 deg.)
|
|
|
|
carTransform[3 + i] -= rotation[i];
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
if ((carTransform[3 + i] > LCR_GAME_UNIT / 2) ||
|
|
|
|
(carTransform[3 + i] < -1 * LCR_GAME_UNIT / 2))
|
|
|
|
carTransform[3 + i] = -1 *(
|
|
|
|
carTransform[3 + i] > 0 ?
|
|
|
|
LCR_GAME_UNIT - carTransform[3 + i] :
|
|
|
|
(-1 * LCR_GAME_UNIT - carTransform[3 + i]));
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
rotation[i] = (LCR_GAME_UNIT + (rotation[i] + (n * carTransform[3 + i])
|
|
|
|
/ LCR_SETTING_GHOST_STEP)) % LCR_GAME_UNIT;
|
|
|
|
}
|
|
|
|
}
|
2025-01-22 00:06:15 +01:00
|
|
|
}
|
|
|
|
|
2025-01-21 22:46:56 +01:00
|
|
|
void _LCR_gamePrepareGhost(void)
|
|
|
|
{
|
|
|
|
LCR_GameUnit carTransform[6];
|
|
|
|
LCR_LOG1("preparing ghost");
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 00:18:28 +01:00
|
|
|
LCR_gameResetRun(1,0);
|
|
|
|
|
|
|
|
uint8_t *currentSample = LCR_game.ghost.samples;
|
2025-01-21 22:46:56 +01:00
|
|
|
|
|
|
|
LCR_game.ghost.stretch = 0;
|
|
|
|
|
|
|
|
while (((int) LCR_replay.achievedTime) >
|
|
|
|
(LCR_SETTING_GHOST_STEP << LCR_game.ghost.stretch) *
|
|
|
|
LCR_SETTING_GHOST_MAX_SAMPLES)
|
|
|
|
{
|
|
|
|
LCR_LOG2("stretching replay step");
|
|
|
|
LCR_game.ghost.stretch++;
|
|
|
|
}
|
2025-01-22 21:39:14 +01:00
|
|
|
|
|
|
|
while (1)
|
2025-01-21 22:46:56 +01:00
|
|
|
{
|
|
|
|
if (LCR_racing.tick % (LCR_SETTING_GHOST_STEP << LCR_game.ghost.stretch)
|
2025-01-22 21:39:14 +01:00
|
|
|
== 0 || LCR_replayHasFinished())
|
2025-01-21 22:46:56 +01:00
|
|
|
{
|
|
|
|
LCR_racingGetCarTransform(carTransform,carTransform + 3,0);
|
2025-01-22 00:06:15 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
for (int i = 0; i < 3; ++i)
|
|
|
|
{
|
|
|
|
carTransform[i] += (LCR_MAP_SIZE_BLOCKS / 2) * (i == 1 ? LCR_GAME_UNIT
|
|
|
|
/ 2 : LCR_GAME_UNIT); // make non-negative
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
// convert to 10 bit value:
|
|
|
|
carTransform[i] = (carTransform[i] * 16) / LCR_GAME_UNIT;
|
2025-01-22 15:35:58 +01:00
|
|
|
|
2025-01-22 21:39:14 +01:00
|
|
|
// conv. rotations to 4 bits, we rely on them being non-negative!
|
|
|
|
carTransform[3 + i] = (carTransform[3 + i] * 16) / LCR_GAME_UNIT;
|
|
|
|
}
|
2025-01-22 15:35:58 +01:00
|
|
|
|
|
|
|
// format: XXXXXXXX YYYYYYYX ZZZZZYYY AAAAZZZZ CCCCBBBB
|
|
|
|
|
|
|
|
currentSample[0] = carTransform[0] >> 1;
|
2025-01-22 00:18:28 +01:00
|
|
|
currentSample[1] =
|
2025-01-22 21:39:14 +01:00
|
|
|
((carTransform[0] >> 9) & 0x01) | (carTransform[1] << 1);
|
2025-01-22 00:18:28 +01:00
|
|
|
currentSample[2] =
|
2025-01-22 21:39:14 +01:00
|
|
|
((carTransform[1] >> 7) & 0x07) | ((carTransform[2] << 2) & 0xf8);
|
2025-01-22 00:18:28 +01:00
|
|
|
currentSample[3] =
|
2025-01-22 21:39:14 +01:00
|
|
|
((carTransform[2] >> 6) & 0x0f) | (carTransform[3] << 4);
|
2025-01-22 00:18:28 +01:00
|
|
|
currentSample[4] =
|
2025-01-22 21:39:14 +01:00
|
|
|
(carTransform[4] & 0x0f) | (carTransform[5] << 4);
|
|
|
|
|
|
|
|
if (LCR_replayHasFinished())
|
|
|
|
break;
|
2025-01-22 00:06:15 +01:00
|
|
|
|
2025-01-22 00:18:28 +01:00
|
|
|
currentSample += LCR_GHOST_SAMPLE_SIZE;
|
2025-01-21 22:46:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
LCR_racingStep(0);
|
|
|
|
}
|
2025-01-22 15:35:58 +01:00
|
|
|
|
|
|
|
while ( // fill the rest with the last sample
|
|
|
|
currentSample >= LCR_game.ghost.samples + LCR_GHOST_SAMPLE_SIZE &&
|
|
|
|
currentSample < LCR_game.ghost.samples +
|
|
|
|
LCR_SETTING_GHOST_MAX_SAMPLES * LCR_GHOST_SAMPLE_SIZE)
|
|
|
|
{
|
|
|
|
*currentSample = *(currentSample - LCR_GHOST_SAMPLE_SIZE);
|
|
|
|
currentSample++;
|
|
|
|
}
|
2025-01-21 22:46:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
LCR_GameUnit LCR_carSpeedKMH(void)
|
|
|
|
{
|
|
|
|
return // we use 28/8 as an approximation of 3.6 to convers MPS to KMH
|
|
|
|
(28 * LCR_SETTING_CMS_PER_BLOCK * LCR_racingGetCarSpeedUnsigned() *
|
|
|
|
LCR_RACING_FPS) / (800 * LCR_GAME_UNIT);
|
|
|
|
}
|
|
|
|
|
2025-01-08 22:05:54 +01:00
|
|
|
/**
|
2025-01-20 19:58:56 +01:00
|
|
|
Rewinds the global data file reading head to the beginning.
|
2025-01-08 22:05:54 +01:00
|
|
|
*/
|
2025-01-20 19:58:56 +01:00
|
|
|
void LCR_gameRewindDataFile(void)
|
2024-12-30 00:49:41 +01:00
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_appendDataStr("");
|
|
|
|
LCR_game.dataFile.state = 0;
|
2024-12-30 00:49:41 +01:00
|
|
|
}
|
2024-12-27 09:06:21 +01:00
|
|
|
|
2024-12-30 00:49:41 +01:00
|
|
|
/**
|
2025-01-20 19:58:56 +01:00
|
|
|
Reads the next global data file character (merged internal data file
|
2025-01-08 22:05:54 +01:00
|
|
|
with the optional user file). First the internal file will be read,
|
|
|
|
immediately followed by the user file, then zero char will be returned and
|
|
|
|
then reading starts over.
|
2024-12-30 00:49:41 +01:00
|
|
|
*/
|
2025-01-20 19:58:56 +01:00
|
|
|
char LCR_gameGetNextDataFileChar(void)
|
2024-12-30 00:49:41 +01:00
|
|
|
{
|
|
|
|
#if LCR_SETTING_ENABLE_RESOURCE_FILE
|
|
|
|
char c;
|
2024-12-27 09:06:21 +01:00
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
if (LCR_game.dataFile.state < 0) // external file?
|
2024-12-30 00:49:41 +01:00
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_getNextDataFileChar();
|
2024-12-30 00:49:41 +01:00
|
|
|
|
|
|
|
if (c == 0)
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_game.dataFile.state = 0; // move to internal file next
|
2024-12-30 00:49:41 +01:00
|
|
|
}
|
|
|
|
else // internal file
|
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_internalDataFile[LCR_game.dataFile.state];
|
|
|
|
LCR_game.dataFile.state++;
|
2024-12-30 00:49:41 +01:00
|
|
|
|
|
|
|
if (c == 0)
|
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_getNextDataFileChar();
|
|
|
|
LCR_game.dataFile.state = c ? -1 : 0; // trust this
|
2024-12-30 00:49:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return c;
|
|
|
|
#else
|
2025-01-20 19:58:56 +01:00
|
|
|
if (LCR_internalDataFile[LCR_game.dataFile.state] == 0)
|
2024-12-30 00:49:41 +01:00
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_game.dataFile.state = 0;
|
2024-12-30 00:49:41 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
return LCR_internalDataFile[LCR_game.dataFile.state++];
|
2024-12-30 00:49:41 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2025-01-20 19:58:56 +01:00
|
|
|
Similar to LCR_gameGetNextDataFileChar, but returns 0 instead of the data
|
|
|
|
string separator character. This function is meant to be used by functions
|
|
|
|
that load something from a string while expecting a zero terminated string.
|
2024-12-30 00:49:41 +01:00
|
|
|
*/
|
2025-01-20 19:58:56 +01:00
|
|
|
char LCR_gameGetNextDataStrChar(void)
|
2024-11-27 20:30:54 +01:00
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
char c = LCR_gameGetNextDataFileChar();
|
2024-12-30 00:49:41 +01:00
|
|
|
return c != LCR_RESOURCE_FILE_SEPARATOR ? c : 0;
|
2024-12-27 09:06:21 +01:00
|
|
|
}
|
|
|
|
|
2024-12-30 00:49:41 +01:00
|
|
|
/**
|
2025-01-20 19:58:56 +01:00
|
|
|
Seeks to the Nth data string in the global data file, after the magic number,
|
|
|
|
so that the name is now available for reading.
|
2024-12-30 00:49:41 +01:00
|
|
|
*/
|
2025-01-20 19:58:56 +01:00
|
|
|
void LCR_seekDataByIndex(unsigned int index, char magicNumber)
|
2024-12-30 00:49:41 +01:00
|
|
|
{
|
|
|
|
char c;
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_LOG0("seeking data string");
|
2024-12-30 00:49:41 +01:00
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_gameRewindDataFile();
|
2024-12-27 09:06:21 +01:00
|
|
|
|
2025-01-14 13:40:22 +01:00
|
|
|
do
|
2024-12-30 00:49:41 +01:00
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_gameGetNextDataFileChar();
|
2025-01-04 20:49:46 +01:00
|
|
|
|
2025-01-03 01:15:24 +01:00
|
|
|
if (c == magicNumber)
|
2025-01-14 13:40:22 +01:00
|
|
|
{
|
|
|
|
if (index)
|
|
|
|
index--;
|
|
|
|
else
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (c != 0 && c != LCR_RESOURCE_FILE_SEPARATOR)
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_gameGetNextDataFileChar();
|
2025-01-14 13:40:22 +01:00
|
|
|
|
|
|
|
} while (c);
|
2024-12-30 00:49:41 +01:00
|
|
|
}
|
2024-12-27 09:06:21 +01:00
|
|
|
|
2025-01-19 22:19:44 +01:00
|
|
|
void LCR_gameLoadMap(unsigned int mapIndex)
|
2024-12-27 09:06:21 +01:00
|
|
|
{
|
2025-01-14 13:40:22 +01:00
|
|
|
char mapName[LCR_MAP_NAME_MAX_LEN];
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_seekDataByIndex(mapIndex,'M');
|
2025-01-14 13:40:22 +01:00
|
|
|
|
|
|
|
mapName[0] = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < LCR_MAP_NAME_MAX_LEN; ++i)
|
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
char c = LCR_gameGetNextDataFileChar();
|
2025-01-14 13:40:22 +01:00
|
|
|
|
|
|
|
if (c == LCR_RESOURCE_FILE_SEPARATOR2 ||
|
|
|
|
c == LCR_RESOURCE_FILE_SEPARATOR || c == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
mapName[i] = c;
|
|
|
|
mapName[i + 1] = 0;
|
|
|
|
}
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_mapLoadFromStr(LCR_gameGetNextDataStrChar,mapName);
|
2025-01-19 22:19:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Loads replay by its index, returns index of a map for the replay (and the map
|
|
|
|
will be loaded as with LCR_mapLoadFromStr) or -1 if the map wasn't found or -2
|
|
|
|
if the replay couldn't be loaded. This function potentially reloads current
|
|
|
|
map!
|
|
|
|
*/
|
|
|
|
unsigned int LCR_gameLoadReplay(unsigned int replayIndex)
|
|
|
|
{
|
|
|
|
uint32_t mapHash;
|
|
|
|
uint16_t nameHash;
|
|
|
|
char c;
|
|
|
|
|
|
|
|
LCR_LOG1("loading replay and map");
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_seekDataByIndex(replayIndex,'R');
|
2025-01-19 22:19:44 +01:00
|
|
|
|
|
|
|
do // skip name
|
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_gameGetNextDataFileChar();
|
2025-01-19 22:19:44 +01:00
|
|
|
}
|
|
|
|
while (c && c != LCR_RESOURCE_FILE_SEPARATOR2 &&
|
|
|
|
c != LCR_RESOURCE_FILE_SEPARATOR);
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
if (!LCR_replayLoadFromStr(LCR_gameGetNextDataStrChar,
|
2025-01-19 22:19:44 +01:00
|
|
|
&mapHash,&nameHash))
|
|
|
|
return -2;
|
|
|
|
|
|
|
|
// now try to find the map with given nameHash
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_gameRewindDataFile();
|
2025-01-19 22:19:44 +01:00
|
|
|
|
|
|
|
unsigned int skipTo = 0;
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
unsigned int mapIndex = 0;
|
|
|
|
|
|
|
|
while (1) // find first skipToth map
|
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_gameGetNextDataFileChar();
|
2025-01-19 22:19:44 +01:00
|
|
|
|
|
|
|
if (c == 0)
|
|
|
|
return -1;
|
|
|
|
else if (c == 'M')
|
|
|
|
{
|
|
|
|
if (mapIndex >= skipTo &&
|
2025-01-20 19:58:56 +01:00
|
|
|
nameHash == _LCR_simpleStrHash(LCR_gameGetNextDataStrChar,';'))
|
2025-01-19 22:19:44 +01:00
|
|
|
{
|
|
|
|
LCR_LOG2("map name hash matches");
|
|
|
|
LCR_gameLoadMap(mapIndex);
|
|
|
|
|
|
|
|
if (mapHash == LCR_currentMap.hash)
|
|
|
|
return mapIndex;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LCR_LOG2("map hash doesn't match");
|
|
|
|
// map hash doesn't match
|
|
|
|
skipTo = mapIndex + 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mapIndex++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (c != LCR_RESOURCE_FILE_SEPARATOR)
|
|
|
|
{
|
|
|
|
if (c == 0)
|
|
|
|
return -1;
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_gameGetNextDataFileChar();
|
2025-01-19 22:19:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2024-11-24 21:21:29 +01:00
|
|
|
}
|
|
|
|
|
2025-01-05 14:47:01 +01:00
|
|
|
void LCR_gameEraseMenuItemNames(void)
|
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
for (int i = 0; i < LCR_MENU_MAX_ITEMS; ++i)
|
|
|
|
for (int j = 0; j < LCR_MENU_STRING_SIZE; ++j)
|
|
|
|
LCR_game.menu.itemNames[i][j] = 0;
|
2025-01-05 14:47:01 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.itemCount = 0;
|
2025-01-05 14:47:01 +01:00
|
|
|
}
|
|
|
|
|
2025-01-07 13:17:41 +01:00
|
|
|
void LCR_gameSetMenuItemStr(uint8_t item, const char *str, char replaceChar)
|
2025-01-05 14:47:01 +01:00
|
|
|
{
|
|
|
|
for (int i = 0; i < LCR_MENU_STRING_SIZE - 1; ++i)
|
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.itemNames[item][i] = str[i] == '$' ? replaceChar : str[i];
|
|
|
|
LCR_game.menu.itemNames[item][i + 1] = 0;
|
2025-01-05 14:47:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LCR_gameLoadMainMenuItems(void)
|
|
|
|
{
|
2025-01-07 12:56:40 +01:00
|
|
|
for (int i = 0; i < 5; ++i)
|
2025-01-07 13:17:41 +01:00
|
|
|
{
|
2025-01-07 20:48:04 +01:00
|
|
|
char replaceChar = i == 0 ? '0' + LCR_game.cameraMode :
|
2025-01-07 13:17:41 +01:00
|
|
|
(i == 1 ? '0' + LCR_game.musicOn : ('0' + LCR_audio.on));
|
|
|
|
|
2025-01-08 21:28:01 +01:00
|
|
|
LCR_gameSetMenuItemStr(i,LCR_texts[LCR_TEXTS_MAIN_MENU + i],replaceChar);
|
2025-01-07 13:17:41 +01:00
|
|
|
}
|
2025-01-05 14:47:01 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.itemCount = 4;
|
2025-01-05 14:47:01 +01:00
|
|
|
}
|
|
|
|
|
2023-08-08 20:34:21 +02:00
|
|
|
void LCR_gameInit(void)
|
|
|
|
{
|
2024-09-27 00:08:52 +02:00
|
|
|
LCR_LOG0("initializing");
|
2024-08-02 00:05:03 +02:00
|
|
|
|
2023-09-10 14:43:20 +02:00
|
|
|
for (int i = 0; i < LCR_KEYS_TOTAL; ++i)
|
2025-01-01 01:23:43 +01:00
|
|
|
LCR_game.keyStates[i] = 0;
|
2024-07-22 01:16:16 +02:00
|
|
|
|
2023-09-16 20:35:01 +02:00
|
|
|
LCR_rendererInit();
|
2025-01-04 20:49:46 +01:00
|
|
|
|
2024-09-04 23:26:05 +02:00
|
|
|
LCR_racingInit();
|
2024-12-23 22:50:07 +01:00
|
|
|
LCR_audioInit();
|
2024-09-05 22:51:11 +02:00
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_game.dataFile.state = 0;
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-05 14:47:01 +01:00
|
|
|
for (int i = 0; i < LCR_MENU_MAX_ITEMS; ++i)
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.itemNamePtrs[i] = LCR_game.menu.itemNames[i];
|
2025-01-05 14:47:01 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedTab = 0;
|
|
|
|
LCR_game.menu.selectedItem = 0;
|
2025-01-02 20:17:15 +01:00
|
|
|
|
2024-12-30 00:49:41 +01:00
|
|
|
LCR_game.frame = 0;
|
2025-01-07 13:17:41 +01:00
|
|
|
LCR_game.musicOn = 1;
|
2024-09-05 22:51:11 +02:00
|
|
|
LCR_game.nextRenderFrameTime = 0;
|
|
|
|
LCR_game.nextRacingTickTime = 0;
|
2025-01-07 20:48:04 +01:00
|
|
|
LCR_game.cameraMode = LCR_CAMERA_MODE_DRIVE;
|
2025-01-04 20:49:46 +01:00
|
|
|
|
2025-01-07 13:17:41 +01:00
|
|
|
LCR_gameLoadMainMenuItems();
|
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
LCR_gameSetState(LCR_GAME_STATE_MENU);
|
|
|
|
|
|
|
|
LCR_currentMap.blockCount = 0; // means no map loaded
|
2024-12-30 22:19:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2025-01-03 01:15:24 +01:00
|
|
|
Loads up to LCR_RESOURCE_ITEM_CHUNK items of given type, starting at given
|
2025-01-08 22:05:54 +01:00
|
|
|
index (among items of the same type). This will also load the menu item names.
|
2024-12-30 22:19:41 +01:00
|
|
|
*/
|
2025-01-20 19:58:56 +01:00
|
|
|
void LCR_gameLoadDataFileChunk(unsigned int startIndex, char magicNumber)
|
2024-12-30 22:19:41 +01:00
|
|
|
{
|
2025-01-07 12:56:40 +01:00
|
|
|
char c;
|
|
|
|
unsigned char state = 0;
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-07 12:56:40 +01:00
|
|
|
LCR_gameEraseMenuItemNames();
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_game.dataFile.firstItemIndex = startIndex;
|
|
|
|
LCR_game.dataFile.itemsTotal = 0;
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_gameRewindDataFile();
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
/* 3 iterations: in first we seek to the start index, in second we load the
|
|
|
|
names, in third we just read the rest to get the total count. */
|
|
|
|
for (int i = 0; i < 3; ++i)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
2025-01-04 20:49:46 +01:00
|
|
|
while (1)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
2025-01-04 20:49:46 +01:00
|
|
|
if (i == 0 && !startIndex)
|
|
|
|
break;
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
c = LCR_gameGetNextDataFileChar();
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
if (c == 0)
|
|
|
|
return;
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
if (state == 0) // second magic char
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
|
|
|
state = 255;
|
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
if (c == magicNumber)
|
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_game.dataFile.itemsTotal++;
|
2025-01-04 20:49:46 +01:00
|
|
|
|
|
|
|
if (i == 0)
|
|
|
|
startIndex--;
|
|
|
|
else if (i == 1)
|
|
|
|
state = 1;
|
|
|
|
}
|
|
|
|
|
2025-01-03 01:15:24 +01:00
|
|
|
}
|
2025-01-04 20:49:46 +01:00
|
|
|
else if (i == 1 && state != 255)
|
|
|
|
{
|
2025-01-08 22:05:54 +01:00
|
|
|
if (c == LCR_RESOURCE_FILE_SEPARATOR ||
|
2025-01-04 20:49:46 +01:00
|
|
|
c == LCR_RESOURCE_FILE_SEPARATOR2 ||
|
|
|
|
state >= 1 + LCR_MENU_STRING_SIZE - 1)
|
|
|
|
{
|
|
|
|
state = 255;
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.itemCount++;
|
2025-01-04 20:49:46 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
if (LCR_game.menu.itemCount >= LCR_RESOURCE_ITEM_CHUNK)
|
2025-01-04 20:49:46 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.itemNames[LCR_game.menu.itemCount][state - 1] = c;
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
state++;
|
|
|
|
}
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
if (c == LCR_RESOURCE_FILE_SEPARATOR)
|
|
|
|
state = 0;
|
2025-01-03 01:15:24 +01:00
|
|
|
}
|
|
|
|
}
|
2023-08-08 20:34:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void LCR_gameEnd(void)
|
|
|
|
{
|
2024-09-27 00:08:52 +02:00
|
|
|
LCR_LOG0("ending");
|
2023-08-08 20:34:21 +02:00
|
|
|
}
|
|
|
|
|
2025-01-15 21:29:38 +01:00
|
|
|
|
|
|
|
|
|
|
|
void LCR_gameTimeToStr(uint32_t timeMS, char *str)
|
|
|
|
{
|
|
|
|
str[9] = 0;
|
|
|
|
|
|
|
|
str[6] = '0' + timeMS % 10; // milliseconds
|
|
|
|
timeMS /= 10;
|
|
|
|
str[7] = '0' + timeMS % 10;
|
|
|
|
timeMS /= 10;
|
|
|
|
str[8] = '0' + timeMS % 10;
|
|
|
|
timeMS /= 10;
|
|
|
|
|
|
|
|
str[3] = '0' + (timeMS % 60) / 10; // seconds
|
|
|
|
str[4] = '0' + timeMS % 10;
|
|
|
|
str[5] = '\'';
|
|
|
|
|
|
|
|
timeMS = (timeMS / 60) % 100; // minutes
|
|
|
|
|
|
|
|
str[0] = '0' + timeMS / 10;
|
|
|
|
str[1] = '0' + timeMS % 10;
|
|
|
|
str[2] = '\'';
|
|
|
|
}
|
|
|
|
|
2024-12-30 22:19:41 +01:00
|
|
|
void LCR_gameDraw3DView(void)
|
2023-08-08 20:34:21 +02:00
|
|
|
{
|
2024-12-25 14:21:28 +01:00
|
|
|
LCR_GameUnit carTransform[6];
|
|
|
|
|
2024-12-30 22:19:41 +01:00
|
|
|
LCR_GameUnit physicsInterpolationParam = LCR_GAME_UNIT -
|
|
|
|
((LCR_game.nextRacingTickTime - LCR_game.time) * LCR_GAME_UNIT)
|
|
|
|
/ LCR_RACING_TICK_MS;
|
|
|
|
|
|
|
|
LCR_racingGetCarTransform(carTransform,carTransform + 3,
|
|
|
|
physicsInterpolationParam);
|
|
|
|
|
|
|
|
LCR_rendererSetCarTransform(carTransform,carTransform + 3);
|
|
|
|
|
2025-01-22 00:18:28 +01:00
|
|
|
if (LCR_game.ghost.active)
|
|
|
|
{
|
|
|
|
LCR_rendererSetGhostVisibility(1);
|
|
|
|
LCR_gameGhostGetTransform(LCR_racing.tick,carTransform,carTransform + 3);
|
|
|
|
LCR_rendererSetGhostTransform(carTransform,carTransform + 3);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
LCR_rendererSetGhostVisibility(0);
|
|
|
|
|
2025-01-07 23:21:08 +01:00
|
|
|
if (LCR_game.cameraMode != LCR_CAMERA_MODE_FREE &&
|
|
|
|
LCR_game.state != LCR_GAME_STATE_RUN_FINISHED)
|
2025-01-07 20:48:04 +01:00
|
|
|
LCR_rendererCameraFollow(
|
|
|
|
(LCR_game.cameraMode != LCR_CAMERA_MODE_INSIDE) +
|
|
|
|
(LCR_game.cameraMode == LCR_CAMERA_MODE_DRIVE2));
|
2024-12-30 22:19:41 +01:00
|
|
|
|
|
|
|
#if LCR_ANIMATE_CAR
|
|
|
|
LCR_rendererSetWheelState(LCR_racingGetWheelRotation(),
|
|
|
|
LCR_racingGetWheelSteer() * 2);
|
|
|
|
#endif
|
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_rendererDraw3D();
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
#if LCR_SETTING_DEBUG_PHYSICS_DRAW
|
|
|
|
LCR_GameUnit camTr[7];
|
|
|
|
LCR_rendererGetCameraTransform(camTr,camTr + 3,camTr + 6);
|
2025-01-14 13:50:13 +01:00
|
|
|
LCR_physicsDebugDraw(camTr,camTr + 3,camTr[6],_LCR_physicdDebugDrawPixel);
|
2025-01-01 01:23:43 +01:00
|
|
|
#endif
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
// GUI/HUD:
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-07 23:21:08 +01:00
|
|
|
char str[10];
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
switch (LCR_game.state)
|
|
|
|
{
|
|
|
|
case LCR_GAME_STATE_RUN_STARTING:
|
|
|
|
str[0] = '0' + LCR_SETTING_COUNTDOWN_SECONDS -
|
|
|
|
(LCR_game.time - LCR_game.stateStartTime) / 1000;
|
|
|
|
str[1] = 0;
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
LCR_rendererDrawText(str,
|
|
|
|
(LCR_EFFECTIVE_RESOLUTION_X - LCR_rendererComputeTextWidth(str,8)) / 2,
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_EFFECTIVE_RESOLUTION_Y / 2,0x0707,8);
|
2025-01-01 01:23:43 +01:00
|
|
|
break;
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
default:
|
|
|
|
{
|
|
|
|
int val = LCR_carSpeedKMH();
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
if (val < 5) // don't show tiny oscillations when still
|
|
|
|
val = 0;
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
str[0] = val >= 100 ? '0' + (val / 100) % 10 : ' ';
|
|
|
|
str[1] = val >= 10 ? '0' + (val / 10) % 10 : ' ';
|
|
|
|
str[2] = '0' + val % 10;
|
|
|
|
str[3] = 0;
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
LCR_rendererDrawText(str,
|
|
|
|
LCR_EFFECTIVE_RESOLUTION_X -
|
|
|
|
LCR_rendererComputeTextWidth(str,2) - 20,
|
|
|
|
LCR_EFFECTIVE_RESOLUTION_Y -
|
|
|
|
LCR_rendererComputeTextHeight(2) - 20,0,2);
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-15 21:29:38 +01:00
|
|
|
LCR_gameTimeToStr(LCR_game.runTimeMS,str);
|
2025-01-07 23:21:08 +01:00
|
|
|
|
2025-01-15 21:29:38 +01:00
|
|
|
LCR_rendererDrawText(str,20,LCR_EFFECTIVE_RESOLUTION_Y -
|
|
|
|
LCR_rendererComputeTextHeight(2) - 45,0,2);
|
2025-01-01 01:23:43 +01:00
|
|
|
|
2025-01-15 21:29:38 +01:00
|
|
|
LCR_gameTimeToStr(LCR_currentMap.targetTime,str);
|
2025-01-01 01:23:43 +01:00
|
|
|
|
|
|
|
LCR_rendererDrawText(str,20,LCR_EFFECTIVE_RESOLUTION_Y -
|
2025-01-15 21:29:38 +01:00
|
|
|
LCR_rendererComputeTextHeight(2) - 20,0x4208,2);
|
2025-01-01 01:23:43 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2024-12-30 22:19:41 +01:00
|
|
|
}
|
|
|
|
|
2025-01-20 19:58:56 +01:00
|
|
|
void _LCR_gameDataCharWrite(char c)
|
2025-01-15 23:08:31 +01:00
|
|
|
{ // TODO
|
|
|
|
printf("%c",c);
|
|
|
|
}
|
|
|
|
|
2025-01-08 22:05:54 +01:00
|
|
|
/**
|
|
|
|
Helper subroutine, handles user input during main loop frame, EXCEPT for the
|
|
|
|
driving input (that is handled in the loop itself).
|
|
|
|
*/
|
2025-01-03 01:15:24 +01:00
|
|
|
void LCR_gameHandleInput(void)
|
2024-12-30 22:19:41 +01:00
|
|
|
{
|
2025-01-05 14:47:01 +01:00
|
|
|
int tabSwitchedTo = -1;
|
2025-01-03 01:15:24 +01:00
|
|
|
int scrolled = 0;
|
2025-01-01 01:23:43 +01:00
|
|
|
|
|
|
|
switch (LCR_game.state)
|
|
|
|
{
|
|
|
|
case LCR_GAME_STATE_MENU:
|
2025-01-02 20:17:15 +01:00
|
|
|
if (LCR_game.keyStates[LCR_KEY_RIGHT] == 1)
|
|
|
|
{
|
|
|
|
LCR_LOG1("menu tab right");
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedTab =
|
|
|
|
(LCR_game.menu.selectedTab + 1) % LCR_MENU_TABS;
|
|
|
|
tabSwitchedTo = LCR_game.menu.selectedTab;
|
|
|
|
LCR_game.menu.selectedItem = 0;
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
2025-01-02 20:17:15 +01:00
|
|
|
}
|
|
|
|
else if (LCR_game.keyStates[LCR_KEY_LEFT] == 1)
|
|
|
|
{
|
|
|
|
LCR_LOG1("menu tab left");
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedTab =
|
|
|
|
(LCR_game.menu.selectedTab + LCR_MENU_TABS - 1) % LCR_MENU_TABS;
|
|
|
|
tabSwitchedTo = LCR_game.menu.selectedTab;
|
|
|
|
LCR_game.menu.selectedItem = 0;
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
2025-01-02 20:17:15 +01:00
|
|
|
}
|
|
|
|
else if (LCR_game.keyStates[LCR_KEY_UP] == 1)
|
|
|
|
{
|
|
|
|
LCR_LOG1("menu item up");
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
if (LCR_game.menu.selectedItem != 0)
|
2025-01-07 20:01:36 +01:00
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedItem--;
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
|
|
|
}
|
2025-01-08 00:10:27 +01:00
|
|
|
else if (LCR_game.menu.selectedTab != 0 &&
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_game.dataFile.firstItemIndex != 0)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedItem = LCR_RESOURCE_ITEM_CHUNK - 1;
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
2025-01-03 01:15:24 +01:00
|
|
|
scrolled = -1;
|
|
|
|
}
|
2025-01-02 20:17:15 +01:00
|
|
|
}
|
|
|
|
else if (LCR_game.keyStates[LCR_KEY_DOWN] == 1)
|
|
|
|
{
|
|
|
|
LCR_LOG1("menu item down");
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
if (LCR_game.menu.selectedTab == 0)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
if (LCR_game.menu.selectedItem < 4)
|
2025-01-07 20:01:36 +01:00
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedItem++;
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
|
|
|
}
|
2025-01-03 01:15:24 +01:00
|
|
|
}
|
2025-01-08 00:10:27 +01:00
|
|
|
else if (LCR_game.menu.selectedItem < LCR_game.menu.itemCount - 1)
|
2025-01-07 20:01:36 +01:00
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedItem++;
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
|
|
|
}
|
2025-01-20 19:58:56 +01:00
|
|
|
else if (LCR_game.dataFile.firstItemIndex +
|
|
|
|
LCR_RESOURCE_ITEM_CHUNK < LCR_game.dataFile.itemsTotal)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedItem = 0;
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
2025-01-03 01:15:24 +01:00
|
|
|
scrolled = 1;
|
|
|
|
}
|
2025-01-02 20:17:15 +01:00
|
|
|
}
|
2025-01-04 20:49:46 +01:00
|
|
|
else if (LCR_game.keyStates[LCR_KEY_B] == 1 && LCR_currentMap.blockCount)
|
2025-01-02 20:17:15 +01:00
|
|
|
{
|
|
|
|
LCR_LOG1("menu quit");
|
2025-01-01 01:23:43 +01:00
|
|
|
LCR_gameSetState(LCR_GAME_STATE_RUN);
|
2025-01-02 20:17:15 +01:00
|
|
|
}
|
2025-01-03 01:15:24 +01:00
|
|
|
else if (LCR_game.keyStates[LCR_KEY_A] == 1)
|
|
|
|
{
|
|
|
|
LCR_LOG1("menu confirm");
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
2025-01-03 01:15:24 +01:00
|
|
|
|
2025-01-08 00:10:27 +01:00
|
|
|
switch (LCR_game.menu.selectedTab)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
|
|
|
case 0:
|
2025-01-08 00:10:27 +01:00
|
|
|
switch (LCR_game.menu.selectedItem)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
2025-01-07 13:17:41 +01:00
|
|
|
case 0:
|
2025-01-07 20:48:04 +01:00
|
|
|
LCR_game.cameraMode = (LCR_game.cameraMode + 1) % 4;
|
|
|
|
LCR_rendererSetCarVisibility(
|
|
|
|
LCR_game.cameraMode != LCR_CAMERA_MODE_INSIDE);
|
|
|
|
LCR_rendererCameraReset();
|
2025-01-07 13:17:41 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
LCR_game.musicOn = !LCR_game.musicOn;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
LCR_audio.on = !LCR_audio.on;
|
|
|
|
break;
|
|
|
|
|
2025-01-03 01:15:24 +01:00
|
|
|
case 4:
|
|
|
|
LCR_gameSetState(LCR_GAME_STATE_END);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
2025-01-07 13:17:41 +01:00
|
|
|
LCR_gameLoadMainMenuItems();
|
2025-01-03 01:15:24 +01:00
|
|
|
break;
|
|
|
|
|
2025-01-04 20:49:46 +01:00
|
|
|
case 1:
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_gameLoadMap(LCR_game.dataFile.firstItemIndex +
|
2025-01-19 22:19:44 +01:00
|
|
|
LCR_game.menu.selectedItem);
|
2025-01-22 00:06:15 +01:00
|
|
|
LCR_gameSetState(LCR_GAME_STATE_LOADING);
|
2025-01-04 20:49:46 +01:00
|
|
|
break;
|
|
|
|
|
2025-01-19 22:19:44 +01:00
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
{
|
2025-01-20 19:58:56 +01:00
|
|
|
int mapIndex = LCR_gameLoadReplay(LCR_game.dataFile.firstItemIndex +
|
2025-01-19 22:19:44 +01:00
|
|
|
LCR_game.menu.selectedItem);
|
|
|
|
|
|
|
|
if (mapIndex < -1)
|
|
|
|
{
|
|
|
|
LCR_LOG1("couldn't load replay");
|
|
|
|
}
|
|
|
|
else if (mapIndex == -1)
|
|
|
|
{
|
|
|
|
LCR_LOG1("couldn't load replay map");
|
|
|
|
}
|
|
|
|
else
|
2025-01-22 00:06:15 +01:00
|
|
|
LCR_gameSetState(LCR_GAME_STATE_LOADING);
|
2025-01-19 22:19:44 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2025-01-03 01:15:24 +01:00
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
}
|
2025-01-01 01:23:43 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
2025-01-07 23:21:08 +01:00
|
|
|
case LCR_GAME_STATE_RUN_FINISHED:
|
|
|
|
if (LCR_game.keyStates[LCR_KEY_A] == 1)
|
2025-01-22 00:18:28 +01:00
|
|
|
LCR_gameResetRun(LCR_racing.playingReplay,LCR_game.ghost.active);
|
2025-01-07 23:21:08 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
case LCR_GAME_STATE_RUN_STARTING:
|
|
|
|
if (LCR_game.time - LCR_game.stateStartTime
|
|
|
|
>= 1000 * LCR_SETTING_COUNTDOWN_SECONDS)
|
|
|
|
LCR_gameSetState(LCR_GAME_STATE_RUN);
|
|
|
|
|
|
|
|
// fall through
|
|
|
|
default:
|
|
|
|
if (LCR_game.keyStates[LCR_KEY_B] == 1)
|
2025-01-03 01:15:24 +01:00
|
|
|
{
|
|
|
|
LCR_LOG1("menu open");
|
2025-01-01 01:23:43 +01:00
|
|
|
LCR_gameSetState(LCR_GAME_STATE_MENU);
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedItem = 0;
|
2025-01-03 01:15:24 +01:00
|
|
|
}
|
2025-01-07 20:48:04 +01:00
|
|
|
else if (LCR_game.cameraMode == LCR_CAMERA_MODE_FREE)
|
|
|
|
{
|
|
|
|
LCR_GameUnit offsets[5];
|
|
|
|
|
|
|
|
for (int i = 0; i < 5; ++i)
|
|
|
|
offsets[i] = 0;
|
|
|
|
|
|
|
|
if (LCR_game.keyStates[LCR_KEY_A])
|
|
|
|
{
|
|
|
|
if (LCR_game.keyStates[LCR_KEY_UP])
|
|
|
|
offsets[4] = LCR_FREE_CAMERA_TURN_STEP;
|
|
|
|
else if (LCR_game.keyStates[LCR_KEY_DOWN])
|
|
|
|
offsets[4] -= LCR_FREE_CAMERA_TURN_STEP;
|
|
|
|
|
|
|
|
if (LCR_game.keyStates[LCR_KEY_RIGHT])
|
|
|
|
offsets[3] -= LCR_FREE_CAMERA_TURN_STEP;
|
|
|
|
else if (LCR_game.keyStates[LCR_KEY_LEFT])
|
|
|
|
offsets[3] = LCR_FREE_CAMERA_TURN_STEP;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (LCR_game.keyStates[LCR_KEY_UP])
|
|
|
|
offsets[0] = LCR_FREE_CAMERA_STEP;
|
|
|
|
else if (LCR_game.keyStates[LCR_KEY_DOWN])
|
|
|
|
offsets[0] -= LCR_FREE_CAMERA_STEP;
|
|
|
|
|
|
|
|
if (LCR_game.keyStates[LCR_KEY_RIGHT])
|
|
|
|
offsets[1] = LCR_FREE_CAMERA_STEP;
|
|
|
|
else if (LCR_game.keyStates[LCR_KEY_LEFT])
|
|
|
|
offsets[1] -= LCR_FREE_CAMERA_STEP;
|
|
|
|
}
|
|
|
|
|
|
|
|
LCR_rendererMoveCamera(offsets,offsets + 3);
|
|
|
|
}
|
2025-01-01 01:23:43 +01:00
|
|
|
else if (LCR_game.keyStates[LCR_KEY_A] == 1)
|
2025-01-22 00:18:28 +01:00
|
|
|
LCR_gameResetRun(LCR_racing.playingReplay,LCR_game.ghost.active);
|
2025-01-01 01:23:43 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2025-01-05 14:47:01 +01:00
|
|
|
if (tabSwitchedTo == 0)
|
|
|
|
LCR_gameLoadMainMenuItems();
|
|
|
|
else if (tabSwitchedTo > 0 || scrolled != 0)
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_gameLoadDataFileChunk(
|
|
|
|
(tabSwitchedTo > 0) ? 0 : (LCR_game.dataFile.firstItemIndex +
|
2025-01-04 19:51:33 +01:00
|
|
|
scrolled * LCR_RESOURCE_ITEM_CHUNK),
|
2025-01-08 00:10:27 +01:00
|
|
|
LCR_game.menu.selectedTab == 1 ? 'M' : 'R');
|
2025-01-03 01:15:24 +01:00
|
|
|
}
|
2025-01-01 01:23:43 +01:00
|
|
|
|
2025-01-03 01:15:24 +01:00
|
|
|
uint8_t LCR_gameStep(uint32_t time)
|
|
|
|
{
|
|
|
|
uint32_t sleep = 0;
|
2025-01-08 22:05:54 +01:00
|
|
|
int paused = LCR_game.state == LCR_GAME_STATE_MENU ||
|
2025-01-03 01:15:24 +01:00
|
|
|
LCR_game.state == LCR_GAME_STATE_RUN_STARTING;
|
2023-09-10 14:43:20 +02:00
|
|
|
|
2025-01-08 22:05:54 +01:00
|
|
|
LCR_LOG2("game step (start)");
|
2025-01-03 01:15:24 +01:00
|
|
|
|
|
|
|
LCR_game.time = time;
|
|
|
|
|
|
|
|
for (int i = 0; i < LCR_KEYS_TOTAL; ++i)
|
|
|
|
LCR_game.keyStates[i] = LCR_keyPressed(i) ?
|
|
|
|
(LCR_game.keyStates[i] < 255 ? LCR_game.keyStates[i] + 1 : 255) : 0;
|
2025-01-07 12:56:40 +01:00
|
|
|
|
2025-01-22 00:06:15 +01:00
|
|
|
if (LCR_game.state == LCR_GAME_STATE_LOADING)
|
2025-01-07 12:56:40 +01:00
|
|
|
{
|
|
|
|
LCR_rendererLoadMap();
|
2025-01-21 22:46:56 +01:00
|
|
|
|
2025-01-22 00:06:15 +01:00
|
|
|
if (LCR_game.menu.selectedTab == 3)
|
2025-01-21 22:46:56 +01:00
|
|
|
_LCR_gamePrepareGhost();
|
|
|
|
|
2025-01-22 00:18:28 +01:00
|
|
|
LCR_gameResetRun(
|
|
|
|
LCR_game.menu.selectedTab == 2,
|
|
|
|
LCR_game.menu.selectedTab == 3);
|
2025-01-07 12:56:40 +01:00
|
|
|
}
|
2025-01-03 01:15:24 +01:00
|
|
|
|
|
|
|
LCR_gameHandleInput();
|
2025-01-02 20:17:15 +01:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
// handle simulation:
|
2024-09-05 22:51:11 +02:00
|
|
|
while (time >= LCR_game.nextRacingTickTime)
|
2023-09-17 13:21:19 +02:00
|
|
|
{
|
2024-09-27 00:08:52 +02:00
|
|
|
LCR_LOG2("gonna step racing engine");
|
2025-01-07 20:48:04 +01:00
|
|
|
|
2025-01-07 23:21:08 +01:00
|
|
|
unsigned int input =
|
|
|
|
(LCR_game.cameraMode == LCR_CAMERA_MODE_FREE ||
|
|
|
|
LCR_game.state == LCR_GAME_STATE_RUN_FINISHED) ? 0 :
|
2025-01-07 20:48:04 +01:00
|
|
|
((LCR_game.keyStates[LCR_KEY_UP] ? LCR_RACING_INPUT_FORW : 0) |
|
|
|
|
(LCR_game.keyStates[LCR_KEY_RIGHT] ? LCR_RACING_INPUT_RIGHT : 0) |
|
|
|
|
(LCR_game.keyStates[LCR_KEY_DOWN] ? LCR_RACING_INPUT_BACK : 0) |
|
|
|
|
(LCR_game.keyStates[LCR_KEY_LEFT] ? LCR_RACING_INPUT_LEFT : 0));
|
2024-09-05 22:51:11 +02:00
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
uint32_t events = paused ? 0 : LCR_racingStep(input);
|
2024-11-21 21:33:36 +01:00
|
|
|
|
2024-12-25 14:21:28 +01:00
|
|
|
if (events & LCR_RACING_EVENT_CP_TAKEN)
|
2024-11-21 21:33:36 +01:00
|
|
|
{
|
2024-12-25 14:21:28 +01:00
|
|
|
int carBlock[3];
|
2024-11-24 21:21:29 +01:00
|
|
|
|
2024-12-25 14:21:28 +01:00
|
|
|
LCR_LOG1("CP taken");
|
2025-01-07 22:57:28 +01:00
|
|
|
LCR_racingGetCarBlockCoords(carBlock);
|
2024-12-25 14:21:28 +01:00
|
|
|
LCR_rendererMarkTakenCP(carBlock[0],carBlock[1],carBlock[2]);
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
2024-12-25 14:21:28 +01:00
|
|
|
}
|
2025-01-08 16:58:22 +01:00
|
|
|
else if (events & LCR_RACING_EVENT_FINISHED &&
|
|
|
|
LCR_game.state != LCR_GAME_STATE_RUN_FINISHED)
|
2024-12-25 14:21:28 +01:00
|
|
|
{
|
|
|
|
LCR_LOG1("finished");
|
2025-01-20 19:58:56 +01:00
|
|
|
LCR_replayOutputStr(_LCR_gameDataCharWrite);
|
2025-01-14 13:40:22 +01:00
|
|
|
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_audioPlaySound(LCR_SOUND_CLICK);
|
2025-01-07 23:21:08 +01:00
|
|
|
LCR_gameSetState(LCR_GAME_STATE_RUN_FINISHED);
|
2024-11-21 21:33:36 +01:00
|
|
|
}
|
|
|
|
|
2024-12-25 22:28:46 +01:00
|
|
|
if (events & LCR_RACING_EVENT_CRASH_SMALL)
|
|
|
|
{
|
|
|
|
LCR_audioPlaySound(LCR_SOUND_CRASH_SMALL);
|
|
|
|
LCR_LOG1("crash (small)");
|
|
|
|
}
|
|
|
|
else if (events & LCR_RACING_EVENT_CRASH_BIG)
|
|
|
|
{
|
|
|
|
LCR_audioPlaySound(LCR_SOUND_CRASH_BIG);
|
|
|
|
LCR_LOG1("crash (big)");
|
|
|
|
}
|
|
|
|
|
2024-12-30 22:19:41 +01:00
|
|
|
int engineIntensity = LCR_carSpeedKMH() * 2;
|
2025-01-08 22:05:54 +01:00
|
|
|
|
2025-01-07 12:56:40 +01:00
|
|
|
LCR_audioSetEngineIntensity(paused ? 0 :
|
|
|
|
(engineIntensity < 256 ? engineIntensity : 255));
|
2024-12-30 22:19:41 +01:00
|
|
|
|
2025-01-07 23:21:08 +01:00
|
|
|
if (LCR_game.state != LCR_GAME_STATE_RUN_FINISHED)
|
|
|
|
LCR_game.runTimeMS = LCR_racingGetRunTimeMS();
|
|
|
|
|
2024-09-05 22:51:11 +02:00
|
|
|
LCR_game.nextRacingTickTime += LCR_RACING_TICK_MS;
|
|
|
|
}
|
2024-09-04 23:26:05 +02:00
|
|
|
|
2024-09-10 21:49:23 +02:00
|
|
|
sleep = (3 * (LCR_game.nextRacingTickTime - time)) / 4;
|
|
|
|
|
2025-01-01 01:23:43 +01:00
|
|
|
// handle rendering:
|
2024-09-05 22:51:11 +02:00
|
|
|
if (time >= LCR_game.nextRenderFrameTime)
|
|
|
|
{
|
2025-01-08 22:05:54 +01:00
|
|
|
LCR_LOG2("rendering next frame");
|
2024-09-27 00:08:52 +02:00
|
|
|
|
2024-09-09 19:16:51 +02:00
|
|
|
while (time >= LCR_game.nextRenderFrameTime)
|
|
|
|
LCR_game.nextRenderFrameTime += 1000 / LCR_SETTING_FPS;
|
2025-01-01 21:04:05 +01:00
|
|
|
|
2025-01-22 00:06:15 +01:00
|
|
|
if ((LCR_game.state == LCR_GAME_STATE_MENU) ||
|
|
|
|
LCR_game.state == LCR_GAME_STATE_LOADING)
|
2025-01-08 21:28:01 +01:00
|
|
|
LCR_rendererDrawMenu(LCR_texts[LCR_TEXTS_TABS
|
|
|
|
+ LCR_game.menu.selectedTab],LCR_game.menu.itemNamePtrs,
|
|
|
|
LCR_game.menu.itemCount + 1,LCR_game.menu.selectedItem);
|
2025-01-02 20:17:15 +01:00
|
|
|
else
|
|
|
|
LCR_gameDraw3DView();
|
2023-09-17 13:21:19 +02:00
|
|
|
}
|
|
|
|
else
|
2024-09-10 21:49:23 +02:00
|
|
|
{
|
|
|
|
uint32_t tmp = (3 * (LCR_game.nextRenderFrameTime - time)) / 4;
|
|
|
|
sleep = tmp < sleep ? tmp : sleep;
|
|
|
|
}
|
2023-09-17 13:21:19 +02:00
|
|
|
|
2025-01-22 00:06:15 +01:00
|
|
|
if (LCR_game.state == LCR_GAME_STATE_LOADING)
|
2025-01-07 12:56:40 +01:00
|
|
|
{
|
2025-01-07 20:01:36 +01:00
|
|
|
// show the "loading" screen
|
|
|
|
|
2025-01-07 12:56:40 +01:00
|
|
|
LCR_rendererDrawRect(
|
|
|
|
LCR_EFFECTIVE_RESOLUTION_X / 8,
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_EFFECTIVE_RESOLUTION_Y / 3,
|
2025-01-07 12:56:40 +01:00
|
|
|
LCR_EFFECTIVE_RESOLUTION_X - LCR_EFFECTIVE_RESOLUTION_X / 4,
|
2025-01-07 20:01:36 +01:00
|
|
|
LCR_EFFECTIVE_RESOLUTION_Y - 2 * LCR_EFFECTIVE_RESOLUTION_Y / 3,
|
2025-01-07 12:56:40 +01:00
|
|
|
0xffff,0);
|
|
|
|
|
2025-01-08 21:28:01 +01:00
|
|
|
LCR_rendererDrawText(LCR_texts[LCR_TEXTS_LOADING],
|
2025-01-07 12:56:40 +01:00
|
|
|
(LCR_EFFECTIVE_RESOLUTION_X -
|
2025-01-08 21:28:01 +01:00
|
|
|
LCR_rendererComputeTextWidth(LCR_texts[LCR_TEXTS_LOADING],4)) / 2,
|
2025-01-07 12:56:40 +01:00
|
|
|
(LCR_EFFECTIVE_RESOLUTION_Y -
|
|
|
|
LCR_rendererComputeTextHeight(4)) / 2,0x0000,4);
|
|
|
|
}
|
|
|
|
|
2023-09-17 13:21:19 +02:00
|
|
|
if (sleep)
|
|
|
|
LCR_sleep(sleep);
|
2024-12-19 21:10:18 +01:00
|
|
|
else
|
|
|
|
{
|
2025-01-07 13:17:41 +01:00
|
|
|
LCR_LOG2("can't sleep");
|
2024-12-19 21:10:18 +01:00
|
|
|
}
|
2023-09-10 14:43:20 +02:00
|
|
|
|
2024-12-30 00:49:41 +01:00
|
|
|
LCR_game.frame++;
|
2025-01-08 22:05:54 +01:00
|
|
|
LCR_LOG2("game step (end)");
|
2024-09-27 00:08:52 +02:00
|
|
|
|
2025-01-03 01:15:24 +01:00
|
|
|
return LCR_game.state != LCR_GAME_STATE_END;
|
2023-08-08 20:34:21 +02:00
|
|
|
}
|
|
|
|
|
2024-12-23 22:50:07 +01:00
|
|
|
uint8_t LCR_gameGetNextAudioSample(void)
|
|
|
|
{
|
|
|
|
return LCR_audioGetNextSample();
|
|
|
|
}
|
|
|
|
|
2023-07-21 21:17:49 +02:00
|
|
|
#endif // guard
|