Expand description
§Falling Tetromino Engine
A tetromino stacker engine in Rust, with the goals of being featureful, efficient and elegant.
use falling_tetromino_engine::*;
// Initialize a game. In-game time starts at 0s.
let mut game = Game::builder()
.seed(1234)
/* Further customization possible here. */
.build();
// Update the game with info that 'left' is activated at second 4.2 (i.e. piece starts moving left).
let input = Input::Activate(Button::MoveLeft);
game.update(InGameTime::from_secs_f64(4.2), Some(input));
// Update the game with info that no input changes up to second 6.79 (e.g. piece falls).
game.update(InGameTime::from_secs_f64(6.79), None);
// Read game state (for rendering etc.)
let State { board, .. } = game.state();§Features Overview
Fundamental points to note:
- The engine implements the pure game logic/backend, i.e. it accurately and only simulates a virtual board with tetromino pieces spawning/moving/locking, lines clearing etc.
- The engine is frontend-agnostic and by itself does not prescribe how to interact with the real world player (it does not know about the keyboard, refresh-/framerate etc.)
Internally, the game processes a pure timeline like so:
Piece spawns e.g. Game state viewed here
| Piece falls |
| | Piece falls |
v v v v
|--------¦--|----¦-------¦-------¦----+--¦------->
^
|
"RotateLeft" player input:
Piece rotatesI.e. running a game at 60 Hz just means that Game::update is called 60 times in one second to determine the state in the timeline and show it.
(The precision used internally is currently based on std::time::Duration which goes down to nanoseconds.)
Depending on configuration, calls to Game::update and Game::forfeit can return additional information (Notification) which can facilitate frontend implementation (e.g. hard drop, piece lock, line clears and other visual feedback).
The engine provides possibilities for compile-time modding. Mods may arbitrarily access and modify game state when called on given engine hooks.
In terms of advanced game mechanics the engine aims to compare with other modern tetromino stackers. It should already incorporate many features desired by familiar/experienced players, such as:
- Available player actions:
- Move left/right,
- Rotate left/right/180°
- Drop soft/hard
- Teleport down(=‘Sonic drop’) and left/right
- Hold piece,
- Tetromino randomizers: ‘Uniform’, ‘Stock’ (generalized Bag), ‘Recency’ (history), ‘Balance-out’,
- Piece preview (arbitrary size),
- Spawn delay (ARE),
- Spawn actions (IRS/IHS; by keeping rotate/hold pressed during spawn),
- Rotation systems: ‘Ocular’ (engine-specific, playtested), ‘Classic’, ‘Super’,
- Delayed auto-move (DAS),
- Auto-move rate (ARR),
- Soft drop factor (SDF),
- Customizable gravity/fall and lock delay curves (exponential and/or linear; also, ‘20G’ (fall rate of ≥1200 Hz) just becomes ≤00083s fall delay),
- Ensure move delay less than lock delay toggle (i.e. DAS/ARR are automatically shortened when lock delay is very low),
- Allow lenient lock-reset toggle (i.e. reset lock delay even if rotate/move fails),
- Lock-reset cap factor (i.e. maximum time before lock delay cannot be reset),
- Line clear duration (LCD),
- Customizable win/loss conditions based on the time, pieces, lines, points,
- Score more points for larger lineclears, spins (‘allspin’), perfect clear, combo,
- Game reproducibility (PRNG/determinism).
The basics seem to have been figured out through many iterative improvements. Ongoing areas of investigation (to improve generalization) are:
- Choice of
Notifications provided to frontend clients; - Choice of update
Hooks for modding clients; - Choice of
Stats to query game with or make game automatically halt; - Various engine generalizations for engine clients that want to plug custom behavior for currently-hardcoded structures.
§Implementation Idea
The game keeps:
- Configuration to read from, which determines game behavior.
- State values which persist throughout the game.
- A dedicated ‘phase’-state field:
- This represents the macro-scale state machine and can store values specific to separate stages during the game.
- E.g., ‘Spawning’ (no piece) vs. ‘Piece-is-in-play’ (with piece data to keep track of).
During each update, the game looks at the (very limited) number of upcoming in-game ‘events’ and processes them.
The only complicated phase is Phase::PieceInPlay { .. }, which encapsulates several types of upcoming events (priority in the given order):
- Action by player: Player input which causes e.g. the piece to move, makes it lock immediately or cancels auto-movement.
- Autonomous movement: While move buttons are active (‘held down’), the piece may move autonomously.
- Falling or locking: Whenever the piece is airborne or grounded, there is an upcoming fall or lock scheduled.
§FIXME
Current documentation is lacking and sometimes slightly outdated. All features should be commented in detail (including IRS, etc., cargo feature serde etc.)]]
Re-exports§
pub use core::Board;pub use core::Button;pub use core::ButtonsState;pub use core::CoordAdd;pub use core::Coordinate;pub use core::DelayCurve;pub use core::DelayParameters;pub use core::DelayTable;pub use core::GameEndCause;pub use core::GameLimits;pub use core::GameRng;pub use core::HEIGHT;pub use core::InGameTime;pub use core::Input;pub use core::LOCK_OUT_HEIGHT;pub use core::Line;pub use core::Notification;pub use core::NotificationFeed;pub use core::Offset;pub use core::Orientation;pub use core::Piece;pub use core::SoftDropRate;pub use core::Stat;pub use core::Tetromino;pub use core::TileID;pub use core::UpdateGameError;pub use core::WIDTH;pub use game_building::GameBuilder;pub use game_modding::GameModifier;pub use helper_types::extduration::ExtDuration;pub use helper_types::extnonnegf64::ExtNonNegF64;pub use piece_rotation::PieceRotator;pub use piece_rotation::StdPceRot;pub use tetromino_generation::StdTetGen;pub use tetromino_generation::TetrominoGenerator;
Modules§
- core
- The core
Gametype and types for its fields. - game_
building - Customizing, templating and constructing
Games. - game_
modding - Modding facilities for the engine.
- game_
update - Handles what happens when
Game::updateis called. - helper_
types - General helper types.
- piece_
rotation - Rotation of tetromino
Pieces. - tetromino_
generation - Random generation of
Tetrominos.
Type Aliases§
- Configuration
- Standard export of the more generic
core::Configurationtype. - Game
- Standard export of the more generic
core::Gametype. - Game
Access - Standard export of the more generic
core::Gametype. - Phase
- Standard export of the more generic
core::Phasetype. - State
- Standard export of the more generic
core::Statetype. - State
Initialization - Standard export of the more generic
core::StateInitializationtype.