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
|
||||
, _preview :: Maybe String -- ^ hard drop preview cell
|
||||
, _locked :: Bool -- ^ lock after hard drop before time step
|
||||
, _paused :: Bool -- ^ game paused
|
||||
}
|
||||
|
||||
makeLenses ''UI
|
||||
|
@ -61,8 +62,12 @@ playGame lvl mp = do
|
|||
writeBChan chan Tick
|
||||
threadDelay delay
|
||||
initialGame <- initGame lvl
|
||||
let initialUI = UI initialGame mp False
|
||||
ui <- customMain (V.mkVty V.defaultConfig) (Just chan) app initialUI
|
||||
ui <- customMain (V.mkVty V.defaultConfig) (Just chan) app $ UI
|
||||
{ _game = initialGame
|
||||
, _preview = mp
|
||||
, _locked = False
|
||||
, _paused = False
|
||||
}
|
||||
return $ ui ^. game
|
||||
|
||||
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.KChar 'k') [])) = exec rotate ui
|
||||
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 'q') [])) = halt ui
|
||||
handleEvent ui (VtyEvent (V.EvKey V.KEsc [])) = halt ui
|
||||
handleEvent ui _ = continue ui
|
||||
|
||||
-- | This common execution function is used for all game input except hard
|
||||
-- drop. If locked (from hard drop) do nothing, else execute the state
|
||||
-- computation and unlock.
|
||||
-- | This common execution function is used for all game user input except hard
|
||||
-- drop and pause. If paused or locked (from hard drop) do nothing, else
|
||||
-- execute the state computation.
|
||||
exec :: Tetris () -> UI -> EventM Name (Next UI)
|
||||
exec op ui = continue
|
||||
$ if ui ^. locked || ui ^. game . to isGameOver
|
||||
then ui
|
||||
else ui & game %~ execTetris op
|
||||
exec op =
|
||||
guarded
|
||||
(not . \ui -> ui ^. paused || ui ^. locked)
|
||||
(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 =
|
||||
if ui ^. game . to isGameOver
|
||||
if ui ^. paused || ui ^. game . to isGameOver
|
||||
then continue ui
|
||||
else do
|
||||
next <- execStateT timeStep $ ui ^. game
|
||||
|
@ -130,7 +151,9 @@ drawGrid ui =
|
|||
hLimit 22
|
||||
$ withBorderStyle BS.unicodeBold
|
||||
$ B.borderWithLabel (str "Tetris")
|
||||
$ vBox rows
|
||||
$ case ui ^. paused of
|
||||
False -> vBox rows
|
||||
True -> C.center $ str "Paused"
|
||||
where
|
||||
rows =
|
||||
[ foldr (<+>) emptyWidget $ M.filterWithKey (inRow r) gmap
|
||||
|
@ -210,7 +233,7 @@ drawInfo :: Game -> Widget Name
|
|||
drawInfo g = hLimit 18 -- size of next piece box
|
||||
$ vBox
|
||||
[ drawNextShape (g ^. nextShape)
|
||||
, padTop (Pad 2) $ drawHelp
|
||||
, padTop (Pad 1) $ drawHelp
|
||||
, padTop (Pad 1) $ drawGameOver g
|
||||
]
|
||||
|
||||
|
@ -248,6 +271,7 @@ drawHelp =
|
|||
, ("Rotate" , "k, ↑")
|
||||
, ("Drop" , "space")
|
||||
, ("Restart", "r")
|
||||
, ("Pause" , "p")
|
||||
, ("Quit" , "q")
|
||||
]
|
||||
|
||||
|
|
Loading…
Add table
Reference in a new issue