More work

This commit is contained in:
Julian Ospald 2022-12-06 00:25:10 +08:00
parent e6ce466700
commit f564b1b533
Signed by: hasufell
GPG Key ID: 3786C5262ECB4A3F

View File

@ -3,6 +3,7 @@
{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TypeApplications #-} {-# LANGUAGE TypeApplications #-}
{-# LANGUAGE RankNTypes #-} {-# LANGUAGE RankNTypes #-}
{-# LANGUAGE MultiWayIf #-}
module AnsiMain where module AnsiMain where
@ -26,7 +27,11 @@ import System.Console.ANSI.Types
import Terminal.Game import Terminal.Game
import Text.PrettyPrint.HughesPJClass ( prettyShow ) import Text.PrettyPrint.HughesPJClass ( prettyShow )
import Control.Monad ( join, when )
import Control.Monad.ST
import Control.Monad.Reader ( ReaderT(runReaderT) ) import Control.Monad.Reader ( ReaderT(runReaderT) )
import Data.Bifunctor
import Data.STRef
import Data.IORef import Data.IORef
import Data.Maybe ( fromMaybe ) import Data.Maybe ( fromMaybe )
import qualified Data.Text as Tx import qualified Data.Text as Tx
@ -37,6 +42,9 @@ import Haskus.Utils.Variant.Excepts
import System.Exit import System.Exit
import Data.Versions (prettyVer) import Data.Versions (prettyVer)
data Direction = Up
| Down
deriving (Show, Eq)
data BrickData = BrickData data BrickData = BrickData
{ lr :: [ListResult] { lr :: [ListResult]
@ -56,11 +64,12 @@ data BrickInternalState = BrickInternalState
deriving Show deriving Show
data BrickState = BrickState data BrickState = BrickState
{ appData :: BrickData { appData :: BrickData
, appSettings :: BrickSettings , appSettings :: BrickSettings
, appState :: BrickInternalState , appState :: BrickInternalState
, appKeys :: KeyBindings , appKeys :: KeyBindings
, appQuit :: Bool , appQuit :: Bool
, appMoreInput :: Maybe String
} }
deriving Show deriving Show
@ -76,9 +85,9 @@ ansiMain s = do
(constructList ad defaultAppSettings Nothing) (constructList ad defaultAppSettings Nothing)
(keyBindings (s :: AppState)) (keyBindings (s :: AppState))
False False
Nothing
sizeCheck
errorPress $ playGame (ghcupGame g) errorPress $ playGame (ghcupGame g)
Left e -> do Left e -> do
flip runReaderT s $ logError $ "Error building app state: " <> Tx.pack flip runReaderT s $ logError $ "Error building app state: " <> Tx.pack
@ -94,7 +103,7 @@ ansiMain s = do
ghcupGame :: BrickState -> Game BrickState ghcupGame :: BrickState -> Game BrickState
ghcupGame bs = Game 13 ghcupGame bs = Game 13
bs -- ticks per second bs -- ticks per second
(\_ s e -> logicFun s e) -- logic function (\ge s e -> logicFun ge s e) -- logic function
(\r s -> centerFull r $ drawFun s r) -- draw function (\r s -> centerFull r $ drawFun s r) -- draw function
appQuit -- quit function appQuit -- quit function
@ -102,15 +111,22 @@ ghcupGame bs = Game 13
drawFun :: BrickState -> GEnv -> Plane drawFun :: BrickState -> GEnv -> Plane
drawFun (BrickState {..}) GEnv{..} = drawFun (BrickState {..}) GEnv{..} =
blankPlane mw mh blankPlane mw mh
& (1, 1) % box mw mh '.' & (1, 1) % box 1 1 '┌'
& (2, 2) % box (mw-2) (mh-2) ' ' & (2, 1) % box 1 (mh - 2) '│'
& (2, 2) % renderItems (renderItem <$> lr appData) & (1, 2) % box (mw - 2) 1 '─'
& (2, mw) % box 1 (mh - 2) '│'
& (1, mw) % box 1 1 '┐'
& (mh, 2) % box (mw - 2) 1 '─'
& (mh, 1) % box 1 1 '└'
& (mh, mw) % box 1 1 '┘'
& (2, 2) % box (mw - 2) (mh - 2) ' '
& (2, 2) % renderItems
where where
mh :: Height mh :: Height
mw :: Width mw :: Width
(mh, mw) = snd boundaries (mh, mw) = T.swap eTermDims
renderItems = foldr1 (===) renderItems = drawListElements renderItem True appState
renderItem ListResult{..} = renderItem _ b ListResult{..} =
let marks = if let marks = if
| lSet -> (stringPlane "✔✔") | lSet -> (stringPlane "✔✔")
| lInstalled -> (stringPlane "") | lInstalled -> (stringPlane "")
@ -129,11 +145,81 @@ drawFun (BrickState {..}) GEnv{..} =
space = stringPlane " " space = stringPlane " "
logicFun :: BrickState -> Event -> BrickState -- | Draws the list elements.
logicFun gs (KeyPress 'q') = gs { appQuit = True } --
logicFun gs Tick = gs -- Evaluates the underlying container up to, and a bit beyond, the
logicFun gs (KeyPress c) = gs -- selected element. The exact amount depends on available height
-- for drawing and 'listItemHeight'. At most, it will evaluate up to
-- element @(i + h + 1)@ where @i@ is the selected index and @h@ is the
-- available height.
drawListElements :: (Int -> Bool -> ListResult -> Plane)
-> Bool
-> BrickInternalState
-> Plane
drawListElements drawElem foc is@(BrickInternalState clr _) =
let es = clr
listSelected = fmap fst $ listSelectedElement' is
(drawnElements, selIx) = runST $ do
ref <- newSTRef (Nothing :: Maybe Int)
elem' <- newSTRef 0
arr <- fmap join $ flip V.imapM es $ \i' e -> do
let isSelected = Just i' == listSelected
elemWidget = drawElem i' isSelected e
selItemAttr = if foc
then listSelectedFocusedAttr
else listSelectedAttr
markSelected = if isSelected then selItemAttr else id
case es V.!? (i' - 1) of
Just e' | lTool e' /= lTool e -> do
modifySTRef elem' (+2)
i <- readSTRef elem'
when isSelected $ writeSTRef ref (Just i)
pure $ V.fromList [hBorder, markSelected elemWidget] -- add separator
_ -> do
modifySTRef elem' (+1)
i <- readSTRef elem'
when isSelected $ writeSTRef ref (Just i)
pure $ V.fromList [markSelected elemWidget]
i <- readSTRef ref
pure (arr, i)
in vcat $ V.toList (makeVisible drawnElements (mh - 2) selIx)
where
makeVisible :: V.Vector Plane -> Height -> Maybe Int -> V.Vector Plane
makeVisible listElements drawableHeight (Just ix) =
let listHeight = V.length listElements
in if | listHeight <= 0 -> listElements
| listHeight > drawableHeight ->
if | ix <= drawableHeight -> makeVisible (V.init listElements) drawableHeight (Just ix)
| otherwise -> makeVisible (V.tail listElements) drawableHeight (Just (ix - 1))
| otherwise -> listElements
makeVisible listElements _ Nothing = listElements
listSelectedFocusedAttr = invert
listSelectedAttr = invert
hBorder = box (mw - 2) 1 '─'
logicFun :: GEnv -> BrickState -> Event -> BrickState
logicFun _ gs (KeyPress 'q') = gs { appQuit = True }
logicFun _ gs Tick = gs
logicFun _ gs@BrickState{appMoreInput = Nothing} (KeyPress '\ESC') = gs { appMoreInput = Just "\ESC" }
logicFun _ gs@BrickState{appMoreInput = Just "\ESC"} (KeyPress '[') = gs { appMoreInput = Just "\ESC[" }
logicFun _ gs@BrickState{appMoreInput = Just "\ESC[", appState = s'} (KeyPress 'A')
= gs { appMoreInput = Nothing, appState = moveCursor 1 s' Up }
logicFun _ gs@BrickState{appMoreInput = Just "\ESC[", appState = s'} (KeyPress 'B')
= gs { appMoreInput = Nothing, appState = moveCursor 1 s' Down }
logicFun _ gs@BrickState{appMoreInput = Just _} _ = gs { appMoreInput = Nothing }
logicFun _ gs (KeyPress c) = gs
moveCursor :: Int -> BrickInternalState -> Direction -> BrickInternalState
moveCursor steps ais@BrickInternalState{..} direction =
let newIx = if direction == Down then ix + steps else ix - steps
in case clr V.!? newIx of
Just _ -> BrickInternalState { ix = newIx, .. }
Nothing -> ais
defaultAppSettings :: BrickSettings defaultAppSettings :: BrickSettings
defaultAppSettings = defaultAppSettings =