Elm Tetris is a game of Tetris implemented in the Elm programming language.
See a live demo here.
See the official Tetris Glossary for definitions of any unknown terms.
- Arrow Left: Move left
- Arrow Right: Move right
- Arrow Down: Move down (also called soft drop)
- Space: Drop (also called hard drop)
- Arrow Up or X: Rotate clockwise
- Ctrl or Z: Rotate counterclockwise
- Esc or P: Pause
- Q: Quit game
- G: Toggle ghost piece
- V: Toggle vertical stripes
This list is also available in the Help dialog (H key).
A level represents the speed by which the falling piece drops to the bottom. There are 15 levels (16, if counting Level 0, see below).
The Tetris Guideline formula for the interval (in milliseconds) between two successive drops is:
1000 * (0.8 - (level - 1) * 0.007) ^ (level - 1)
The formula yields:
- Level 1: 1000 ms
- Level 2: 793 ms
- Level 3: 617.8 ms
- Level 4: 472.73 ms
- Level 5: 355.2 ms
- Level 6: 262 ms
- Level 7: 189.68 ms
- Level 8: 134.73 ms
- Level 9: 93.88 ms
- Level 10: 64.15 ms
- Level 11: 42.98 ms
- Level 12: 28.22 ms
- Level 13: 18.15 ms
- Level 14: 11.44 ms
- Level 15: 7.06 ms
For intervals shorter than the frame rate (16 ms), the falling piece will drop several rows at a time to achieve that interval on average. For example, if the interval is 7 ms, the falling piece will drop 2 or 3 rows per frame (16 / 7 = 2.29).
The level can be changed directly by the player in the Start dialog (after loading the page, or after the Quit or Game Over dialogs). While playing the game, the level will be increased automatically for every 10 lines cleared.
Level 0 is a non-standard level (no other games implement it), and does not follow the formula above. This level has no gravity (the falling piece does not drop by itself), and will not increase automatically (stays 0 forever). When the piece reaches the bottom (the player moves it there with the Arrow Down key), it locks normally, with lock delay. Hard drop (Space key) also works normally. Scoring for level 0 is the same as for level 1.
Scoring follows the Tetris Guideline:
- Soft drop: 1 × distance (regardless of level)
- Hard drop: 2 × distance (regardless of level)
- Single line clear: 100 × level
- Double line clear: 300 × level
- Triple line clear: 500 × level
- Tetris (4) line clear: 800 × level
- Back-to-back bonus (Tetrises): 0.5 × action total
A back-to-back bonus is awarded for two or more Tetris line clears in a row, uninterrupted by single/double/triple line clears. For example, 2 Tetrises in a row will be awarded a back-to-back bonus of (800 + 800) × 0.5 = 800 points. This will be added to the normal points for 2 Tetrises, 800 + 800 = 1600, for a total of 2400 (assuming level 1). The timing will be: 800 points after the first Tetris (not back-to-back yet), and 1600 points after the second.
The Guideline scoring has been only partially implemented (no points for T-Spins or Combos).
The implemented random generator generates a sequence of all seven tetrominoes (Tetris pieces: I, J, L, O, S, T, Z) permuted randomly, as if drawn from a bag. Then it deals all seven to the game before generating another bag. This algorithm makes it much less likely that the player will get an obscenely long run without a desired tetromino. It can produce a maximum of 12 tetrominoes between one I and the next, and a run of S and Z tetrominoes is limited to a maximum of 4.
Long "droughts" of I tetrominoes and long sequences of S and Z tetrominoes are undesirable because they increase the probability of prematurely ending the game. It has been demonstrated that long enough sequences of S and Z pieces make infinite gameplay impossible (although a good player can survive over 150 consecutive S and Z tetrominoes).
Using a 7-bag random generator is one of Tetris Guideline's indispensable rules (see subsection Random Generator), to be followed by all licensed games.
Lock delay is the time interval between the falling piece reaching the bottom and its locking to the bottom. It's normally 500 ms, but can be extended by player moves or rotations. The goal is to allow the player a few correction moves after reaching the bottom, but not so many that it makes the game unchallenging (and certainly not infinite spin, floating the piece indefinitely). The current implementation might be called limited move reset lock delay. The gory details:
- every time the deepest row reached changes, lock delay is reset to 500 ms, and the allowed moves counter is reset to 15
- when the player makes a move but the deepest row reached does not change, the allowed moves counter is decreased by 1, and lock delay is reset to 500 ms (only actual moves count; trying to move into a wall does not)
- when the player makes a move but the allowed moves counter is 0, lock delay is not extended anymore
- after the end of lock delay (when the 500 ms interval ends), the piece will lock as soon as it touches the bottom
- if the deepest row reached changes again at any point, lock delay is again reset to 500 ms and the allowed moves counter is reset to 15
The values of 500 ms for lock delay and 15 for the reset limit are standard for existing Tetris implementations. See subsection Lock Down of the Tetris Guideline for details, and for a description of the 3 types of lock delay.
A hard drop (Space key) instantly moves the falling piece to the bottom, then instantly locks it (zero lock delay). Useful when the player wants to move the game faster. Standard feature in most games.
To be contrasted with soft drop (Arrow Down key), which drops the piece one row at a time, and locks it to the bottom after a lock delay.
Falling piece rotation is specified by a Tetris Guideline standard bombastically named Super Rotation System (SRS), which also describes wall kicks.
When implementing rotation, each tetromino (Tetris piece) has a pivot around which it rotates, and which it carries along as it rotates or moves. See a visual representation here, where the pivot is shown as a white circle. Notice that for the I and O tetrominoes, the pivot is at the intersection of gridlines, whereas for the J, L, S, T and Z tetrominoes, the pivot is at the center of a square block.
Early games (before the first Tetris Guideline in 1996) each used a different rotation system, usually simpler but slightly awkward (see comparison). SRS standardized rotation for later games, and made it easier to implement wall kicks.
Wall kicks are specified by the Super Rotation System (SRS) standard, similarly to rotation.
When the player attempts to rotate a tetromino (Tetris piece), but the position it would normally occupy after basic rotation is obstructed (either by the wall or floor of the playfield, or by the stack), the game will attempt to "kick" the tetromino into an alternative position nearby. In SRS, 4 alternative positions are sequentially tested, and if none is available, the rotation fails completely.
See details on how to implement wall kicks here, with examples.
A common criticism of SRS wall kicks is that the algorithm doesn't make intuitive sense. It just supplies a list of 4 alternatives, with no explanation of why those specific 4. This is a valid criticism.
There are two conditions in which the game ends (also called top out conditions):
- Block out: when part of a newly-generated piece overlaps with an existing block on the playfield
- Lock out: when a piece locks entirely above the ceiling
The following statistics are shown on the side panel:
- Score
- Level
- Lines: number of lines cleared
- Time: time actually played (not including time paused); it shows minutes and seconds, and will also show hours and days, if the game continues
Statistics are cleared when a new game is started.
The piece preview is an area that displays the next piece that will enter the playfield. It is positioned to the right of the playfield, under the statistics. Standard feature in most games.
At this moment only one piece is shown in the preview, but the game is already capable of showing more. The decision to show only one piece comes from wanting a minimalist look. Commercial games show between 1 to 6 preview pieces.
The ghost piece is a visualization of where the falling piece will land if allowed to drop to the bottom. Standard feature in most games.
It is shown in gray color, and can be switched on/off by pressing G (on by default).
Vertical stripes are alternating white-gray stripes on the background of the playfield, designed to help the player track where the falling piece will land if allowed to drop (similar in purpose to the ghost piece). Non-standard feature.
Vertical stripes can be switched on/off by pressing V (off by default).
The game consists of the play screen and several modal dialogs:
- Start dialog: displayed on loading the page, and after the Quit and Game Over confirmation dialogs; pressing +/- will change the start level (0-15, default 1); pressing S or Esc will start the game
- Pause dialog: can be opened from the play screen by pressing P or Esc; it simply pauses the game
- Quit dialog: can be opened from the play screen or the Pause dialog by pressing Q (Quit game); its role is to ask for confirmation (Y/N); on Yes, it goes to the Start dialog
- Game Over dialog: triggered by Game Over conditions during play; its role is to ask for confirmation, "Start new game? (Y/N)"; on Yes, it goes to the Start dialog
- Help dialog: can be opened from the play screen and from all other dialogs by pressing H; shows the keyboard shortcuts
All dialogs pause the game. They (and the game in general) can only be controlled by keyboard, not by mouse. All dialogs can be closed with the Esc key (except the Game Over dialog, which only accepts Y, to start a new game).
When control goes from a dialog back to the play screen, a countdown screen is displayed, counting down 3-2-1, for 1 second each. This is to allow the player to position their hands over the keyboard in anticipation of the game. Standard feature in most games.
The countdown screen can be interrupted with the P key (or Esc), which brings up the Pause dialog.
There is a short delay (200 ms) between the moment the player completely fills one or more lines, and the moment the game clears those lines off the playfield. This is meant to let the player "take in" the line clearing event, while not slowing down the game too much. A common way for games to mark this event is to show little explosion animations for a brief moment.
When the browser window is resized, the game will resize within it, so that no scrollbars are needed and no game areas are hidden.
Game rendering is implemented using SVG (Scalable Vector Graphics). The whole game is contained within one <svg>
element, and there is no significant HTML outside it. One property of SVG is that it scales up/down well, without the image becoming pixelated.
This is how game size reacts to browser window size:
- If the window is large enough (for example, a maximized window on a large monitor), the game will display in its default size (about 620×730 pixels).
- If the player resizes the window to a smaller size, the game will also resize, while keeping aspect ratio constant. Therefore, the whole game will always be visible, without needing scrollbars, or game areas being hidden. A resize also happens when the maximized window size is smaller than the game default size (such as on a small laptop or tablet).
- The player can use the Zoom In/Zoom Out functionality of the browser (Ctrl + and Ctrl -) to increase/decrease the size of the game. If zooming in (Ctrl +), the game will only become as large as the window (no scrollbars, no game areas hidden).
As far as alignment within the window is concerned, the game is always horizontally centered and vertically top-aligned.
When the player minimizes the browser window or switches to another browser tab, the game pauses automatically (opens the Pause dialog). This has to do with the way browsers deal with these events: they stop sending animation frame updates for the duration the window is minimized, or the tab inactive. On reactivating the window/tab, the first frame update will report a very large time delta (e.g., 5000 ms, as opposed to the usual 16 ms), which the application can then use to "catch up" with the time lost. In our case, this is not really useful, and means the game is neither fully playing nor paused, so we switch to paused mode to make it official. This also has the advantage of making time calculations more simple, precise and deterministic.
- Mouse-based dialogs instead of keyboard-based ones
- Settings dialog:
- reassigning keyboard shortcuts (Move left, Move right, Move down, Rotate clockwise, Rotate counterclockwise, Drop)
- Lock Delay: Limited Spin (default) / Infinite Spin / Step Reset
- Random Generator: 7-Bag (default) / Simple
- Piece Preview: 1 to 6 Pieces
- Ghost Piece: On (default) / Off
- Vertical Stripes : On / Off (default)
- Pause/Play button, mouse-based alternative to keyboard shortcut P
- Hold piece, storing a falling piece for later use
- Award score points for:
- Levels above 15: For now, the higher the level, the higher the speed of the drop animation. Above level 15, the falling piece drops to the bottom almost instantly, so increasing the speed would not increase difficulty. Lock delay (now 500 ms) and maximum lock delay moves (now 15) will have to gradually decrease in order to make the game more challenging.
- Game modes, for example (in Tetris Zone):
- Marathon (normal): level will increase for every 10 lines cleared
- Challenge: highest score in 10 minutes
- Sprint: fastest time to clear 40 lines
- Master: same as Marathon, but with instant lock
- Smoother movement of the falling piece, with less animation flicker, especially on levels 7-12 (if SVG allows it)
- Play by mouse, not just by keyboard (as on tetris.com)
- Save settings locally between visits (into browser localStorage)
- Play with pentominoes instead of tetrominoes (option in the Settings dialog)
Elm Tetris was developed with Elm version 0.19.1.
Instructions for installing the latest Elm can be found here. Historical versions can be installed here.
It's also useful to install the Elm extension for Visual Studio Code. It has syntax highlighting, points out syntax errors, and formats code on Save.
For day-to-day development, start the reactor
server:
elm reactor
Then go to http://localhost:8000
to see the project dashboard (file viewer). Inside it, navigate to src/Main.elm
. This will show the full working version of the application (minus the styling in index.html
).
Every time you edit Main.elm
or one of its dependencies, just refresh the browser and reactor
will recompile everything.
Occasionally it can be useful to rewind and replay events in order to fix tricky bugs. To start the time-travelling debugger, first run:
elm make src/Main.elm --output=main.js --debug
Then load index.html
in the browser, which will show the running application with a little Elm icon in the bottom right corner of the page. Click it to open the time-travelling debugger.
Unfortunatelly, this application is subscribing to Browser.Events.onAnimationFrameDelta
, which will flood the message queue with one message every 16 ms (60 times/s), thus making the use of the time-travelling debugger a bit difficult.
Another useful tool is the Elm REPL (read–eval–print loop):
elm repl
To see a list of all Elm commands (install, etc.), use:
elm --help
The most useful links:
For a quick way to run the application through index.html
instead of reactor
:
elm make src/Main.elm --output=main.js
If needing to deploy, copy index.html
, main.js
and favicon.ico
to the destination folder.
For a fully optimized/minified version of the application:
elm make src/Main.elm --output=main.js --optimize
uglifyjs main.js --compress "pure_funcs=[F2,F3,F4,F5,F6,F7,F8,F9,A2,A3,A4,A5,A6,A7,A8,A9],pure_getters,keep_fargs=false,unsafe_comps,unsafe" | uglifyjs --mangle --output=main.min.js
As an example, this would decrease .js file size from 171 kB to 33 kB.
To deploy, copy index.html
, main.min.js
and favicon.ico
to the destination folder. Inside index.html
, change the <script>
tag to point to main.min.js
instead of main.js
.
To install uglifyjs
:
npm install uglify-js --global
See here for details on how to optimize asset size in Elm.