Skip to content

Commit

Permalink
Auto-close modal before evaluating ModalCallback
Browse files Browse the repository at this point in the history
  • Loading branch information
grancalavera committed Jan 12, 2019
1 parent 8ecb87e commit ed544c6
Show file tree
Hide file tree
Showing 10 changed files with 55 additions and 49 deletions.
27 changes: 14 additions & 13 deletions app/Main.hs
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,16 @@ import Graphics.Vty ( Vty )
import qualified Graphics.Vty as V
import Lens.Micro ( (^.) )
import Data.Maybe ( fromMaybe )
-- import Control.Monad ( void )

import qualified Labyrinth.UI as UI
import qualified Labyrinth.UI.Widget as UI
import Labyrinth.UI ( Name )
import Labyrinth.UI.Widget

import qualified Labyrinth.UI.Debug as Debug

import qualified Labyrinth.UI.Modal as Modal
import qualified Labyrinth.Store.Event.Modal as Modal

import qualified Labyrinth.UI.Screen.Splash as Splash
import qualified Labyrinth.Store.Event.Splash as Splash

Expand All @@ -21,8 +25,6 @@ import qualified Labyrinth.Store as Store
import Labyrinth.Store ( Store
, State(..)
, Ev
, state
, modal
)

main :: IO ()
Expand All @@ -39,21 +41,21 @@ app = App { appDraw = draw
}

draw :: Store e -> [Widget Name]
draw store = [showModals, maybe drawScreen Modal.draw (Store.nextModal store)]
draw store =
[Debug.draw store, maybe drawScreen Modal.draw (Store.nextModal store)]
where
showModals = str $ show (store ^. modal)
drawScreen = appContainer 50 $ case store ^. state of
drawScreen = UI.appContainer 50 $ case store ^. Store.state of
Splash s -> Splash.draw s
Setup s -> Setup.draw s
_ -> emptyWidget
_ -> txt "Screen not implemented"

handleEvent
:: Ord e => Store e -> BrickEvent Name e -> EventM Name (Next (Store e))
handleEvent store ev = handle store ev
where
handle = if Store.isModalEvent store ev
then Modal.handle
else case store ^. state of
else case store ^. Store.state of
Splash s -> Splash.handle s
Setup s -> Setup.handle s
_ -> \_ _ -> halt store
Expand All @@ -68,7 +70,6 @@ chooseCursor :: Store e -> [CursorLocation Name] -> Maybe (CursorLocation Name)
chooseCursor store =
fromMaybe (neverShowCursor store) $ if Store.isShowingModal store
then Store.nextModal store >>= Modal.chooseCursor
else case store ^. state of
Splash s -> Splash.chooseCursor s
Setup s -> Setup.chooseCursor s
_ -> Nothing
else case store ^. Store.state of
Setup s -> Setup.chooseCursor s
_ -> Nothing
3 changes: 2 additions & 1 deletion labyrinth.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
--
-- see: https://github.com/sol/hpack
--
-- hash: c4ecded4a317515194ffec780399ead781fb6980bb83fc4782fe5aa4ec1d9811
-- hash: 6cbcea1f571593f670a43c58a89b3e50fcc26e29adcc1665118ae3f7c365c410

name: labyrinth
version: 0.1.0.0
Expand Down Expand Up @@ -30,6 +30,7 @@ library
Labyrinth.Game.Players
Labyrinth.UI
Labyrinth.UI.Modal
Labyrinth.UI.Debug
Labyrinth.UI.Screen.Splash
Labyrinth.UI.Screen.Setup
Labyrinth.UI.Screen.Game
Expand Down
6 changes: 3 additions & 3 deletions lib/Labyrinth/Store.hs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ module Labyrinth.Store
, State(..)
, Ev
, state
, modal
, modals
, initial
, isModalEvent
, isShowingModal
Expand All @@ -19,10 +19,10 @@ import Labyrinth.UI ( Name )
import qualified Labyrinth.UI.Screen.Splash as Splash

initial :: Store e
initial = Store { _state = Splash Splash.initial, _modal = [] }
initial = Store { _state = Splash Splash.initial, _modals = [] }

isModalEvent :: Ord e => Store e -> BrickEvent Name e -> Bool
isModalEvent store ev = isShowingModal store || Modal.isModalEvent ev

isShowingModal :: Store e -> Bool
isShowingModal = not . null . (^. modal)
isShowingModal = not . null . (^. modals)
21 changes: 16 additions & 5 deletions lib/Labyrinth/Store/Event/Modal.hs
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
module Labyrinth.Store.Event.Modal
( handle
, isModalEvent
, showModal
, nextModal
)
where

Expand All @@ -17,7 +19,9 @@ import Data.Map.Strict ( Map
, (!?)
)
import Data.Maybe ( fromMaybe )
import Labyrinth.UI ( Name )
import Labyrinth.UI ( Name
, ModalCallback
)
import Labyrinth.UI.Modal ( dialog
, onTrue
, onFalse
Expand All @@ -38,13 +42,15 @@ handleInModal :: GlobalEventHandler e
handleInModal store ev = maybe (continue store) withModal (nextModal store)
where
withModal m = case ev of
(VtyEvent (V.EvKey V.KEsc [])) -> (m ^. onFalse) store
(VtyEvent (V.EvKey V.KEsc [])) -> (hideModalAnd $ m ^. onFalse) store
(VtyEvent (V.EvKey V.KEnter [])) -> fromMaybe (continue store) $ do
sel <- D.dialogSelection (m ^. dialog)
return $ if sel then (m ^. onTrue) store else (m ^. onFalse) store
return $ if sel
then (hideModalAnd $ m ^. onTrue) store
else (hideModalAnd $ m ^. onFalse) store
(VtyEvent vtyEv) -> do
d <- D.handleDialogEvent vtyEv (m ^. dialog)
continue $ store & modal %~ \case
continue $ store & modals %~ \case
[] -> []
(_ : ms) -> (m & dialog .~ d) : ms
_ -> continue store
Expand All @@ -55,8 +61,13 @@ promptToQuit store _ = showModal store $ mkModal
(txt "Do you want to quit Labyrinth?")
(0, [("Stay", False), ("Quit", True)])
halt
hideModal
continue

eventMap :: Ord e => Map (BrickEvent Name e) (GlobalEventHandler e)
eventMap =
Map.fromList [(VtyEvent (V.EvKey (V.KChar 'q') [V.MCtrl]), promptToQuit)]

hideModalAnd :: ModalCallback Store e -> ModalCallback Store e
hideModalAnd f store = f $ store & modals %~ \case
[] -> []
(_ : ms) -> ms
7 changes: 4 additions & 3 deletions lib/Labyrinth/Store/Event/Setup.hs
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ play :: RegistrationEventHandler e
play s store _ = maybe (continue store) promptToPlay (S.setup s)
where
promptToPlay (p, ps) = showModal store
$ UI.mkOkModal "start" (UI.nextPlayerPrompt p) (toNewGame p ps)
$ UI.mkOkModal "start" (UI.nextPlayerPrompt p) (continueToNewGame p ps)

processInput :: RegistrationEventHandler e
processInput s store ev =
Expand All @@ -59,5 +59,6 @@ edit i s store _ =
update :: Store e -> SetupS e -> Store e
update store s = store & state .~ Setup s

toNewGame :: Player -> Players -> ModalCallback Store e
toNewGame p ps store = continue $ store & state .~ Plan (G.initial (Game ps p))
continueToNewGame :: Player -> Players -> ModalCallback Store e
continueToNewGame p ps store =
continue $ store & state .~ Plan (G.initial (Game ps p))
14 changes: 4 additions & 10 deletions lib/Labyrinth/Store/Internal.hs
Original file line number Diff line number Diff line change
Expand Up @@ -4,9 +4,8 @@ module Labyrinth.Store.Internal
, Ev
, EventHandler
, state
, modal
, modals
, showModal
, hideModal
, nextModal
)
where
Expand Down Expand Up @@ -36,19 +35,14 @@ data State e = Splash SplashS

data Store e = Store
{ _state :: State e
, _modal :: [Modal Store e]
, _modals :: [Modal Store e]
} deriving (Show)
makeLenses ''Store
type EventHandler s e
= s -> Store e -> BrickEvent Name e -> EventM Name (Next (Store e))

showModal :: Store e -> Modal Store e -> EventM Name (Next (Store e))
showModal store m = continue $ store & modal %~ (m :)

hideModal :: Store e -> EventM Name (Next (Store e))
hideModal store = continue $ store & modal %~ \case
[] -> []
(_ : ms) -> ms
showModal store m = continue $ store & modals %~ (m :)

nextModal :: Store e -> Maybe (Modal Store e)
nextModal = listToMaybe . (^. modal)
nextModal = listToMaybe . (^. modals)
18 changes: 10 additions & 8 deletions lib/Labyrinth/UI/Debug.hs
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,16 @@ where
import Brick
import Lens.Micro ( (^.) )
import qualified Labyrinth.Store as S
import Labyrinth.Store ( Store, State(..) )
import Labyrinth.Store ( Store
, State(..)
)

draw :: Store e -> Widget n
draw store = vBox [state, modals]
where
state = txt $ case (store ^. S.state) of
Splash _ -> "State: Splash"
Setup _ -> "State: Setup"
Plan _ -> "State: Plan"
_ -> "State: not implemented"
modals = str $ show $ store ^. S.modal
where
state = txt . ("State: " <>) $ case store ^. S.state of
Splash _ -> "Splash"
Setup _ -> "Setup"
Plan _ -> "Plan"
_ -> "not implemented"
modals = str $ show $ store ^. S.modals
2 changes: 1 addition & 1 deletion lib/Labyrinth/UI/Modal.hs
Original file line number Diff line number Diff line change
Expand Up @@ -60,4 +60,4 @@ mkModal desc body options onT onF = Modal
}

mkOkModal :: String -> Widget Name -> ModalCallback s e -> Modal s e
mkOkModal desc body onOk = mkModal desc body (0, [("OK", True)]) onOk onOk
mkOkModal desc body onOk = mkModal desc body (0, [("OK", True)]) onOk continue
5 changes: 0 additions & 5 deletions lib/Labyrinth/UI/Screen/Splash.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@ module Labyrinth.UI.Screen.Splash
( draw
, SplashS
, initial
, chooseCursor
)
where

Expand Down Expand Up @@ -30,7 +29,3 @@ draw _ =

initial :: SplashS
initial = SplashS

chooseCursor
:: SplashS -> Maybe ([CursorLocation Name] -> Maybe (CursorLocation Name))
chooseCursor _ = Nothing
1 change: 1 addition & 0 deletions package.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ library:
- Labyrinth.Game.Players
- Labyrinth.UI
- Labyrinth.UI.Modal
- Labyrinth.UI.Debug
- Labyrinth.UI.Screen.Splash
- Labyrinth.UI.Screen.Setup
- Labyrinth.UI.Screen.Game
Expand Down

0 comments on commit ed544c6

Please sign in to comment.