161 lines
4.8 KiB
Haskell
161 lines
4.8 KiB
Haskell
{-# LANGUAGE TemplateHaskell #-}
|
|
{-# LANGUAGE RecordWildCards #-}
|
|
{-# LANGUAGE TypeSynonymInstances #-}
|
|
{-# LANGUAGE FlexibleInstances #-}
|
|
module Tetris where
|
|
|
|
import Data.Map (Map)
|
|
import qualified Data.Map as M
|
|
import Data.Sequence (ViewL(..), (<|), (><))
|
|
import qualified Data.Sequence as Seq
|
|
import Lens.Micro
|
|
import Lens.Micro.TH
|
|
import System.Random (getStdRandom, randomR)
|
|
|
|
import Prelude hiding (Left, Right)
|
|
|
|
-- Types and instances
|
|
|
|
-- | Tetris shape types
|
|
data Tetrimino = I | O | T | S | Z | J | L
|
|
deriving (Eq, Show, Enum)
|
|
|
|
-- | Coordinates
|
|
type Coord = (Int, Int)
|
|
|
|
-- | Tetris shape in coordinate context
|
|
data Block = Block
|
|
{ _shape :: Tetrimino -- ^ block type
|
|
, _origin :: Coord -- ^ origin
|
|
, _extra :: [Coord] -- ^ extraneous cells
|
|
} deriving (Eq, Show)
|
|
|
|
makeLenses ''Block
|
|
|
|
data Direction = Left | Right | Down
|
|
deriving (Eq, Show)
|
|
|
|
-- | Cell state within a tetris board
|
|
data Cell = Filled Tetrimino | Empty
|
|
deriving (Eq, Show)
|
|
|
|
-- | Board of cells
|
|
type Board = Map Coord Cell
|
|
|
|
-- | Game state
|
|
data Game = Game
|
|
{ _level :: Int
|
|
, _currBlock :: Block
|
|
, _nextShape :: Tetrimino
|
|
, _nextShapeBag :: Seq.Seq Tetrimino
|
|
, _score :: Int
|
|
, _board :: Board
|
|
} deriving (Eq, Show)
|
|
|
|
makeLenses ''Game
|
|
|
|
|
|
-- Translate class for direct translations, without concern for boundaries
|
|
-- Shiftable concerns safe translations with boundaries
|
|
class Translatable s where
|
|
translate :: Direction -> s -> s
|
|
|
|
instance Translatable Coord where
|
|
translate Left (x, y) = (x-1, y)
|
|
translate Right (x, y) = (x+1, y)
|
|
translate Down (x,y) = (x, y-1)
|
|
|
|
instance Translatable Block where
|
|
translate d b =
|
|
b & origin %~ translate d
|
|
& extra %~ fmap (translate d)
|
|
|
|
-- Low level functions on blocks, cells, and coordinates
|
|
|
|
initBlock :: Tetrimino -> Block
|
|
initBlock I = Block I startOrigin [(-2,0), (-1,0), (1,0)]
|
|
initBlock O = Block O startOrigin [(-1,0), (-1,-1), (0,-1)]
|
|
initBlock S = Block S startOrigin [(-1,-1), (0,-1), (1,0)]
|
|
initBlock Z = Block Z startOrigin [(-1,0), (0,-1), (1,-1)]
|
|
initBlock L = Block L startOrigin [(-1,-1), (-1,0), (1,0)]
|
|
initBlock J = Block J startOrigin [(-1,0), (1,0), (1,-1)]
|
|
initBlock T = Block T startOrigin [(-1,0), (0,-1), (1,0)]
|
|
|
|
-- | Visible, active board size
|
|
boardWidth, boardHeight :: Int
|
|
boardWidth = 10
|
|
boardHeight = 20
|
|
|
|
-- | Starting block origin cell
|
|
startOrigin :: Coord
|
|
startOrigin = (6, 21)
|
|
|
|
-- | Rotate block counter clockwise about origin
|
|
-- *Note*: Strict unsafe rotation not respecting boundaries
|
|
-- Safety can only be assured within Game context
|
|
rotate' :: Block -> Block
|
|
rotate' b@(Block s o@(xo,yo) cs)
|
|
| s == O = b -- O doesn't need rotation
|
|
| s == I && (xo,yo+1) `elem` cs = rotateWith clockwise b -- I only has two orientations
|
|
| otherwise = rotateWith counterclockwise b
|
|
|
|
rotateWith :: (Coord -> Coord -> Coord) -> Block -> Block
|
|
rotateWith dir b = let o = b ^. origin
|
|
in b & extra %~ fmap (dir o)
|
|
|
|
clockwise :: Coord -- ^ origin
|
|
-> Coord -- ^ point to rotate around origin
|
|
-> Coord
|
|
clockwise (xo, yo) (x, y) = (xo + y - yo, xo + y - x)
|
|
|
|
counterclockwise :: Coord -- ^ origin
|
|
-> Coord -- ^ point to rotate around origin
|
|
-> Coord
|
|
counterclockwise (xo, yo) (x, y) = (xo + yo - y, x + yo - xo)
|
|
|
|
-- | Get coordinates of all block cells
|
|
occupiedCells :: Block -> [Coord]
|
|
occupiedCells b = b ^. origin : b ^. extra
|
|
|
|
-- Higher level functions on game and board
|
|
|
|
bagFourTetriminoEach :: IO (Seq.Seq Tetrimino)
|
|
bagFourTetriminoEach =
|
|
shuffle $ Seq.cycleTaking 28 $ Seq.fromList [(I)..]
|
|
|
|
-- | Initialize a game with a given level
|
|
initGame :: Int -> IO Game
|
|
initGame lvl = do
|
|
initBag <- bagFourTetriminoEach
|
|
let (fstShape :< fstBag) = Seq.viewl initBag
|
|
(sndShape :< sndBag) = Seq.viewl fstBag
|
|
return
|
|
Game { _level = lvl
|
|
, _currBlock = initBlock fstShape
|
|
, _nextShape = sndShape
|
|
, _nextShapeBag = sndBag
|
|
, _score = 0
|
|
, _board = mempty }
|
|
|
|
-- TODO check if mapKeysMonotonic works
|
|
clearFullRows :: Game -> Game
|
|
clearFullRows g = g & board %~ clearBoard
|
|
where clearBoard = M.mapKeys shiftRowsAbove . M.filterWithKey isInFullRow
|
|
isInFullRow (_,y) _ = y `elem` fullRowIndices
|
|
fullRowIndices = filter isFullRow [1..boardHeight]
|
|
isFullRow r = boardWidth == (length . M.filterWithKey (inRow r) $ g ^. board)
|
|
inRow r (_, y) _ = r == y
|
|
shiftRowsAbove (x,y) =
|
|
let offset = length . filter (< y) $ fullRowIndices
|
|
in (x, y - offset)
|
|
|
|
-- TODO wallkicks http://tetris.wikia.com/wiki/TGM_rotation
|
|
|
|
shuffle :: Seq.Seq a -> IO (Seq.Seq a)
|
|
shuffle xs
|
|
| null xs = mempty
|
|
| otherwise = do
|
|
randomPosition <- getStdRandom (randomR (0, length xs - 1))
|
|
let (left, right) = Seq.splitAt randomPosition xs
|
|
(y :< ys) = Seq.viewl right
|
|
fmap (y <|) (shuffle $ left >< ys)
|