Add pause feature, closes #5
This commit is contained in:
parent
b3d812bee5
commit
33f5175595
1 changed files with 38 additions and 14 deletions
|
@ -29,6 +29,7 @@ data UI = UI
|
||||||
{ _game :: Game -- ^ tetris game
|
{ _game :: Game -- ^ tetris game
|
||||||
, _preview :: Maybe String -- ^ hard drop preview cell
|
, _preview :: Maybe String -- ^ hard drop preview cell
|
||||||
, _locked :: Bool -- ^ lock after hard drop before time step
|
, _locked :: Bool -- ^ lock after hard drop before time step
|
||||||
|
, _paused :: Bool -- ^ game paused
|
||||||
}
|
}
|
||||||
|
|
||||||
makeLenses ''UI
|
makeLenses ''UI
|
||||||
|
@ -61,8 +62,12 @@ playGame lvl mp = do
|
||||||
writeBChan chan Tick
|
writeBChan chan Tick
|
||||||
threadDelay delay
|
threadDelay delay
|
||||||
initialGame <- initGame lvl
|
initialGame <- initGame lvl
|
||||||
let initialUI = UI initialGame mp False
|
ui <- customMain (V.mkVty V.defaultConfig) (Just chan) app $ UI
|
||||||
ui <- customMain (V.mkVty V.defaultConfig) (Just chan) app initialUI
|
{ _game = initialGame
|
||||||
|
, _preview = mp
|
||||||
|
, _locked = False
|
||||||
|
, _paused = False
|
||||||
|
}
|
||||||
return $ ui ^. game
|
return $ ui ^. game
|
||||||
|
|
||||||
levelToDelay :: Int -> Int
|
levelToDelay :: Int -> Int
|
||||||
|
@ -81,25 +86,41 @@ handleEvent ui (VtyEvent (V.EvKey (V.KChar 'j') [])) = exec (shift Down) ui
|
||||||
handleEvent ui (VtyEvent (V.EvKey V.KUp [])) = exec rotate ui
|
handleEvent ui (VtyEvent (V.EvKey V.KUp [])) = exec rotate ui
|
||||||
handleEvent ui (VtyEvent (V.EvKey (V.KChar 'k') [])) = exec rotate ui
|
handleEvent ui (VtyEvent (V.EvKey (V.KChar 'k') [])) = exec rotate ui
|
||||||
handleEvent ui (VtyEvent (V.EvKey (V.KChar ' ') [])) =
|
handleEvent ui (VtyEvent (V.EvKey (V.KChar ' ') [])) =
|
||||||
continue $ ui & game %~ execTetris hardDrop & locked .~ True
|
guarded
|
||||||
|
(not . view paused)
|
||||||
|
(over game (execTetris hardDrop) . set locked True)
|
||||||
|
ui
|
||||||
|
handleEvent ui (VtyEvent (V.EvKey (V.KChar 'p') [])) =
|
||||||
|
guarded
|
||||||
|
(not . view locked)
|
||||||
|
(over paused not)
|
||||||
|
ui
|
||||||
handleEvent ui (VtyEvent (V.EvKey (V.KChar 'r') [])) = restart ui
|
handleEvent ui (VtyEvent (V.EvKey (V.KChar 'r') [])) = restart ui
|
||||||
handleEvent ui (VtyEvent (V.EvKey (V.KChar 'q') [])) = halt ui
|
handleEvent ui (VtyEvent (V.EvKey (V.KChar 'q') [])) = halt ui
|
||||||
handleEvent ui (VtyEvent (V.EvKey V.KEsc [])) = halt ui
|
handleEvent ui (VtyEvent (V.EvKey V.KEsc [])) = halt ui
|
||||||
handleEvent ui _ = continue ui
|
handleEvent ui _ = continue ui
|
||||||
|
|
||||||
-- | This common execution function is used for all game input except hard
|
-- | This common execution function is used for all game user input except hard
|
||||||
-- drop. If locked (from hard drop) do nothing, else execute the state
|
-- drop and pause. If paused or locked (from hard drop) do nothing, else
|
||||||
-- computation and unlock.
|
-- execute the state computation.
|
||||||
exec :: Tetris () -> UI -> EventM Name (Next UI)
|
exec :: Tetris () -> UI -> EventM Name (Next UI)
|
||||||
exec op ui = continue
|
exec op =
|
||||||
$ if ui ^. locked || ui ^. game . to isGameOver
|
guarded
|
||||||
then ui
|
(not . \ui -> ui ^. paused || ui ^. locked)
|
||||||
else ui & game %~ execTetris op
|
(game %~ execTetris op)
|
||||||
|
|
||||||
-- | Handles time steps, does nothing if game is over
|
-- | This base execution function takes a predicate and only issues UI
|
||||||
|
-- modification when predicate passes and game is not over.
|
||||||
|
guarded :: (UI -> Bool) -> (UI -> UI) -> UI -> EventM Name (Next UI)
|
||||||
|
guarded p f ui = continue
|
||||||
|
$ if not (p ui) || ui ^. game . to isGameOver
|
||||||
|
then ui
|
||||||
|
else f ui
|
||||||
|
|
||||||
|
-- | Handles time steps, does nothing if game is over or paused
|
||||||
handleTick :: UI -> EventM Name (Next UI)
|
handleTick :: UI -> EventM Name (Next UI)
|
||||||
handleTick ui =
|
handleTick ui =
|
||||||
if ui ^. game . to isGameOver
|
if ui ^. paused || ui ^. game . to isGameOver
|
||||||
then continue ui
|
then continue ui
|
||||||
else do
|
else do
|
||||||
next <- execStateT timeStep $ ui ^. game
|
next <- execStateT timeStep $ ui ^. game
|
||||||
|
@ -130,7 +151,9 @@ drawGrid ui =
|
||||||
hLimit 22
|
hLimit 22
|
||||||
$ withBorderStyle BS.unicodeBold
|
$ withBorderStyle BS.unicodeBold
|
||||||
$ B.borderWithLabel (str "Tetris")
|
$ B.borderWithLabel (str "Tetris")
|
||||||
$ vBox rows
|
$ case ui ^. paused of
|
||||||
|
False -> vBox rows
|
||||||
|
True -> C.center $ str "Paused"
|
||||||
where
|
where
|
||||||
rows =
|
rows =
|
||||||
[ foldr (<+>) emptyWidget $ M.filterWithKey (inRow r) gmap
|
[ foldr (<+>) emptyWidget $ M.filterWithKey (inRow r) gmap
|
||||||
|
@ -210,7 +233,7 @@ drawInfo :: Game -> Widget Name
|
||||||
drawInfo g = hLimit 18 -- size of next piece box
|
drawInfo g = hLimit 18 -- size of next piece box
|
||||||
$ vBox
|
$ vBox
|
||||||
[ drawNextShape (g ^. nextShape)
|
[ drawNextShape (g ^. nextShape)
|
||||||
, padTop (Pad 2) $ drawHelp
|
, padTop (Pad 1) $ drawHelp
|
||||||
, padTop (Pad 1) $ drawGameOver g
|
, padTop (Pad 1) $ drawGameOver g
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -248,6 +271,7 @@ drawHelp =
|
||||||
, ("Rotate" , "k, ↑")
|
, ("Rotate" , "k, ↑")
|
||||||
, ("Drop" , "space")
|
, ("Drop" , "space")
|
||||||
, ("Restart", "r")
|
, ("Restart", "r")
|
||||||
|
, ("Pause" , "p")
|
||||||
, ("Quit" , "q")
|
, ("Quit" , "q")
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue