Add pause feature, closes #5

This commit is contained in:
Sam Tay 2018-12-27 13:12:28 -05:00
parent b3d812bee5
commit 33f5175595

View file

@ -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")
] ]