2016-06-11 03:30:45 +03:00
|
|
|
{- | Rendering & misc. helpers. -}
|
|
|
|
|
2021-05-09 03:12:06 +03:00
|
|
|
{-# LANGUAGE CPP #-}
|
2015-08-18 03:44:18 +03:00
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
2021-11-18 20:53:02 +03:00
|
|
|
{-# LANGUAGE NamedFieldPuns #-}
|
2022-10-29 05:30:26 +03:00
|
|
|
{-# LANGUAGE LambdaCase #-}
|
2015-08-18 03:44:18 +03:00
|
|
|
|
2018-10-23 14:56:29 +03:00
|
|
|
module Hledger.UI.UIUtils (
|
|
|
|
borderDepthStr
|
|
|
|
,borderKeysStr
|
|
|
|
,borderKeysStr'
|
|
|
|
,borderPeriodStr
|
|
|
|
,borderQueryStr
|
|
|
|
,defaultLayout
|
|
|
|
,helpDialog
|
|
|
|
,helpHandle
|
|
|
|
,minibuffer
|
|
|
|
,moveDownEvents
|
|
|
|
,moveLeftEvents
|
|
|
|
,moveRightEvents
|
|
|
|
,moveUpEvents
|
|
|
|
,normaliseMovementKeys
|
2018-11-05 00:27:50 +03:00
|
|
|
,renderToggle
|
2020-02-16 16:27:09 +03:00
|
|
|
,renderToggle1
|
2018-10-23 14:56:29 +03:00
|
|
|
,replaceHiddenAccountsNameWith
|
|
|
|
,scrollSelectionToMiddle
|
2022-08-23 01:55:14 +03:00
|
|
|
,get'
|
|
|
|
,put'
|
|
|
|
,modify'
|
2022-10-29 05:30:26 +03:00
|
|
|
,mapScreens
|
|
|
|
,screenId
|
2019-01-19 03:29:56 +03:00
|
|
|
,suspend
|
2019-01-19 03:25:26 +03:00
|
|
|
,redraw
|
2022-09-08 21:53:42 +03:00
|
|
|
,reportSpecAddQuery
|
2021-08-24 08:50:32 +03:00
|
|
|
,reportSpecSetFutureAndForecast
|
2021-11-18 20:53:02 +03:00
|
|
|
,listScrollPushingSelection
|
2022-08-23 01:55:14 +03:00
|
|
|
,dlogUiTrace
|
2022-10-29 05:30:26 +03:00
|
|
|
,dlogUiTraceIO
|
2022-08-23 01:55:14 +03:00
|
|
|
,dlogUiTraceM
|
2022-10-29 05:30:26 +03:00
|
|
|
,dlogUiScreenStack
|
2022-08-23 01:55:14 +03:00
|
|
|
,uiDebugLevel
|
|
|
|
,uiNumBlankItems
|
2021-11-18 20:53:02 +03:00
|
|
|
)
|
2016-06-11 03:30:45 +03:00
|
|
|
where
|
2015-08-18 03:44:18 +03:00
|
|
|
|
2015-08-20 14:54:23 +03:00
|
|
|
import Brick
|
2015-08-23 03:46:57 +03:00
|
|
|
import Brick.Widgets.Border
|
|
|
|
import Brick.Widgets.Border.Style
|
2016-06-11 03:30:45 +03:00
|
|
|
import Brick.Widgets.Dialog
|
|
|
|
import Brick.Widgets.Edit
|
2022-08-17 13:04:50 +03:00
|
|
|
import Brick.Widgets.List (List, listSelectedL, listNameL, listItemHeightL, listSelected, listMoveDown, listMoveUp, GenericList)
|
2017-12-30 23:59:25 +03:00
|
|
|
import Control.Monad.IO.Class
|
2021-08-16 09:09:55 +03:00
|
|
|
import Data.Bifunctor (second)
|
2016-06-11 03:30:45 +03:00
|
|
|
import Data.List
|
2020-11-05 04:58:04 +03:00
|
|
|
import qualified Data.Text as T
|
2022-09-09 00:56:20 +03:00
|
|
|
import Data.Time (addDays)
|
2019-01-19 03:25:26 +03:00
|
|
|
import Graphics.Vty
|
2021-11-21 04:33:28 +03:00
|
|
|
(Event(..),Key(..),Modifier(..),Vty(..),Color,Attr,currentAttr,refresh, displayBounds
|
2019-01-19 03:25:26 +03:00
|
|
|
-- ,Output(displayBounds,mkDisplayContext),DisplayContext(..)
|
|
|
|
)
|
2016-06-11 03:30:45 +03:00
|
|
|
import Lens.Micro.Platform
|
2015-08-18 03:44:18 +03:00
|
|
|
|
2021-08-30 10:43:14 +03:00
|
|
|
import Hledger
|
2022-09-11 01:41:37 +03:00
|
|
|
-- import Hledger.Cli.CliOptions (CliOpts(reportspec_))
|
2017-12-30 23:59:25 +03:00
|
|
|
import Hledger.Cli.DocFiles
|
2022-09-11 01:41:37 +03:00
|
|
|
-- import Hledger.UI.UIOptions (UIOpts(uoCliOpts))
|
2015-08-18 03:44:18 +03:00
|
|
|
import Hledger.UI.UITypes
|
2022-09-02 23:36:05 +03:00
|
|
|
|
2022-09-11 01:39:32 +03:00
|
|
|
import Data.Vector (Vector)
|
2015-08-18 03:44:18 +03:00
|
|
|
|
2019-01-24 04:25:20 +03:00
|
|
|
-- | On posix platforms, send the system STOP signal to suspend the
|
|
|
|
-- current program. On windows, does nothing.
|
2022-09-11 01:39:32 +03:00
|
|
|
-- (Though, currently hledger-ui is not built on windows.)
|
2019-01-19 03:29:56 +03:00
|
|
|
#ifdef mingw32_HOST_OS
|
2019-01-24 04:25:20 +03:00
|
|
|
suspendSignal :: IO ()
|
|
|
|
suspendSignal = return ()
|
2019-01-19 03:29:56 +03:00
|
|
|
#else
|
|
|
|
import System.Posix.Signals
|
2019-01-24 04:25:20 +03:00
|
|
|
suspendSignal :: IO ()
|
2019-07-15 13:28:52 +03:00
|
|
|
suspendSignal = raiseSignal sigSTOP
|
2019-01-19 03:29:56 +03:00
|
|
|
#endif
|
|
|
|
|
2022-08-23 01:55:14 +03:00
|
|
|
-- Debug logging for UI state changes.
|
2022-10-29 05:30:26 +03:00
|
|
|
-- A good place to log things of interest while debugging, see commented examples below.
|
2022-08-23 01:55:14 +03:00
|
|
|
|
|
|
|
get' = do
|
|
|
|
x <- get
|
|
|
|
dlogUiTraceM $ "getting state: " ++ (head $ lines $ pshow $ aScreen x)
|
2022-09-11 01:41:37 +03:00
|
|
|
-- dlogUiTraceM $ ("query: "++) $ pshow' $ x & aopts & uoCliOpts & reportspec_ & _rsQuery
|
2022-10-29 05:30:26 +03:00
|
|
|
-- dlogUiScreenStack "" "" screenId x
|
|
|
|
-- dlogUiScreenStack "getting " "with register descriptions: " showscreenregisterdescriptions x
|
2022-08-23 01:55:14 +03:00
|
|
|
return x
|
|
|
|
|
|
|
|
put' x = do
|
|
|
|
dlogUiTraceM $ "putting state: " ++ (head $ lines $ pshow $ aScreen x)
|
2022-09-11 01:41:37 +03:00
|
|
|
-- dlogUiTraceM $ ("query: "++) $ pshow' $ x & aopts & uoCliOpts & reportspec_ & _rsQuery
|
2022-10-29 05:30:26 +03:00
|
|
|
-- dlogUiScreenStack "" "" screenId x
|
|
|
|
-- dlogUiScreenStack "putting " "with register descriptions: " showscreenregisterdescriptions x
|
2022-08-23 01:55:14 +03:00
|
|
|
put x
|
|
|
|
|
|
|
|
modify' f = do
|
|
|
|
x <- get
|
|
|
|
let x' = f x
|
|
|
|
dlogUiTraceM $ "modifying state: " ++ (head $ lines $ pshow $ aScreen x')
|
2022-09-11 01:41:37 +03:00
|
|
|
-- dlogUiTraceM $ ("from: "++) $ pshow' $ x & aopts & uoCliOpts & reportspec_ & _rsQuery
|
|
|
|
-- dlogUiTraceM $ ("to: "++) $ pshow' $ x' & aopts & uoCliOpts & reportspec_ & _rsQuery
|
2022-10-29 05:30:26 +03:00
|
|
|
-- dlogUiScreenStack "getting" "" screenId x
|
|
|
|
-- dlogUiScreenStack "putting" "" screenId x'
|
|
|
|
-- dlogUiScreenStack "getting " "with register descriptions: " showscreenregisterdescriptions x
|
|
|
|
-- dlogUiScreenStack "putting " "with register descriptions: " showscreenregisterdescriptions x'
|
2022-08-23 01:55:14 +03:00
|
|
|
modify f
|
|
|
|
|
2022-10-29 05:30:26 +03:00
|
|
|
-- showscreenregisterdescriptions :: Screen -> String
|
|
|
|
-- showscreenregisterdescriptions scr = case scr of
|
|
|
|
-- MS _ -> "M" -- menu
|
|
|
|
-- AS _ -> "A" -- all accounts
|
|
|
|
-- BS _ -> "B" -- bs accounts
|
|
|
|
-- IS _ -> "I" -- is accounts
|
|
|
|
-- RS sst -> ("R:" ++) $ -- menu
|
|
|
|
-- intercalate "," $ map (T.unpack . rsItemDescription) $
|
|
|
|
-- takeWhile (not . T.null . rsItemDate) $ Data.Vector.toList $ listElements $ _rssList sst
|
|
|
|
-- TS _ -> "T" -- transaction
|
|
|
|
-- ES _ -> "E" -- error
|
|
|
|
|
|
|
|
-- | Run a function on each screen in a UIState's screen "stack",
|
|
|
|
-- from topmost screen down to currently-viewed screen.
|
|
|
|
mapScreens :: (Screen -> a) -> UIState -> [a]
|
|
|
|
mapScreens f UIState{aPrevScreens, aScreen} = map f $ reverse $ aScreen : aPrevScreens
|
|
|
|
|
2019-01-24 04:25:20 +03:00
|
|
|
-- | On posix platforms, suspend the program using the STOP signal,
|
|
|
|
-- like control-z in bash, returning to the original shell prompt,
|
|
|
|
-- and when resumed, continue where we left off.
|
|
|
|
-- On windows, does nothing.
|
2022-08-17 13:04:50 +03:00
|
|
|
suspend :: Ord a => s -> EventM a s ()
|
2019-01-24 04:25:20 +03:00
|
|
|
suspend st = suspendAndResume $ suspendSignal >> return st
|
2016-06-20 21:09:12 +03:00
|
|
|
|
2022-08-17 13:04:50 +03:00
|
|
|
-- | Tell vty to redraw the whole screen.
|
|
|
|
redraw :: EventM a s ()
|
|
|
|
redraw = getVtyHandle >>= liftIO . refresh
|
2019-01-19 03:25:26 +03:00
|
|
|
|
2018-11-05 00:27:50 +03:00
|
|
|
-- | Wrap a widget in the default hledger-ui screen layout.
|
|
|
|
defaultLayout :: Widget Name -> Widget Name -> Widget Name -> Widget Name
|
|
|
|
defaultLayout toplabel bottomlabel =
|
|
|
|
topBottomBorderWithLabels (str " "<+>toplabel<+>str " ") (str " "<+>bottomlabel<+>str " ") .
|
|
|
|
margin 1 0 Nothing
|
2022-09-02 23:36:05 +03:00
|
|
|
-- topBottomBorderWithLabel label .
|
2018-11-05 00:27:50 +03:00
|
|
|
-- padLeftRight 1 -- XXX should reduce inner widget's width by 2, but doesn't
|
|
|
|
-- "the layout adjusts... if you use the core combinators"
|
|
|
|
|
2016-06-10 21:50:57 +03:00
|
|
|
-- | Draw the help dialog, called when help mode is active.
|
2022-09-09 02:42:29 +03:00
|
|
|
helpDialog :: Widget Name
|
|
|
|
helpDialog =
|
2016-06-10 21:50:57 +03:00
|
|
|
Widget Fixed Fixed $ do
|
|
|
|
c <- getContext
|
|
|
|
render $
|
2022-08-17 13:04:50 +03:00
|
|
|
withDefAttr (attrName "help") $
|
2020-07-18 18:59:01 +03:00
|
|
|
renderDialog (dialog (Just "Help (LEFT/ESC/?/q to close help)") Nothing (c^.availWidthL)) $ -- (Just (0,[("ok",())]))
|
2020-07-18 01:17:21 +03:00
|
|
|
padTop (Pad 0) $ padLeft (Pad 1) $ padRight (Pad 1) $
|
2016-06-20 21:09:12 +03:00
|
|
|
vBox [
|
|
|
|
hBox [
|
2018-10-17 00:16:58 +03:00
|
|
|
padRight (Pad 1) $
|
2016-06-20 21:09:12 +03:00
|
|
|
vBox [
|
2022-08-17 13:04:50 +03:00
|
|
|
withAttr (attrName "help" <> attrName "heading") $ str "Navigation"
|
2020-07-18 18:39:23 +03:00
|
|
|
,renderKey ("UP/DOWN/PUP/PDN/HOME/END/k/j/C-p/C-n", "")
|
2020-07-18 22:24:37 +03:00
|
|
|
,str " move selection up/down"
|
|
|
|
,renderKey ("RIGHT/l/C-f", "show txns, or txn detail")
|
2020-07-18 18:39:23 +03:00
|
|
|
,renderKey ("LEFT/h/C-b ", "go back")
|
2020-07-18 22:30:37 +03:00
|
|
|
,renderKey ("ESC ", "cancel, or reset app state")
|
2020-07-18 18:59:01 +03:00
|
|
|
|
2018-10-17 00:16:58 +03:00
|
|
|
,str " "
|
2022-08-17 13:04:50 +03:00
|
|
|
,withAttr (attrName "help" <> attrName "heading") $ str "Accounts screen"
|
2020-07-18 01:17:21 +03:00
|
|
|
,renderKey ("1234567890-+ ", "set/adjust depth limit")
|
2020-07-18 18:39:23 +03:00
|
|
|
,renderKey ("t ", "toggle accounts tree/list mode")
|
|
|
|
,renderKey ("H ", "toggle historical balance/change")
|
2018-10-17 00:16:58 +03:00
|
|
|
,str " "
|
2022-08-17 13:04:50 +03:00
|
|
|
,withAttr (attrName "help" <> attrName "heading") $ str "Register screen"
|
2020-07-18 18:39:23 +03:00
|
|
|
,renderKey ("t ", "toggle subaccount txns\n(and accounts tree/list mode)")
|
|
|
|
,renderKey ("H ", "toggle historical/period total")
|
2016-08-13 03:44:55 +03:00
|
|
|
,str " "
|
2022-08-17 13:04:50 +03:00
|
|
|
,withAttr (attrName "help" <> attrName "heading") $ str "Help"
|
2020-07-18 18:59:01 +03:00
|
|
|
,renderKey ("? ", "toggle this help")
|
|
|
|
,renderKey ("p/m/i", "while help is open:\nshow manual in pager/man/info")
|
|
|
|
,str " "
|
2016-08-13 03:44:55 +03:00
|
|
|
]
|
2018-10-17 00:16:58 +03:00
|
|
|
,padLeft (Pad 1) $ padRight (Pad 0) $
|
2016-06-20 21:09:12 +03:00
|
|
|
vBox [
|
2022-08-17 13:04:50 +03:00
|
|
|
withAttr (attrName "help" <> attrName "heading") $ str "Filtering"
|
2018-10-17 00:16:58 +03:00
|
|
|
,renderKey ("/ ", "set a filter query")
|
2020-02-16 16:27:09 +03:00
|
|
|
,renderKey ("F ", "show future & periodic txns")
|
2018-10-17 00:16:58 +03:00
|
|
|
,renderKey ("R ", "show real/all postings")
|
2021-11-23 23:16:32 +03:00
|
|
|
,renderKey ("z ", "show nonzero/all amounts")
|
2020-07-19 00:26:07 +03:00
|
|
|
,renderKey ("U/P/C ", "show unmarked/pending/cleared")
|
2020-07-18 18:59:01 +03:00
|
|
|
,renderKey ("S-DOWN /S-UP ", "shrink/grow period")
|
|
|
|
,renderKey ("S-RIGHT/S-LEFT", "next/previous period")
|
|
|
|
,renderKey ("T ", "set period to today")
|
2020-07-18 22:30:37 +03:00
|
|
|
,renderKey ("DEL ", "reset filters")
|
2016-08-10 01:31:35 +03:00
|
|
|
,str " "
|
2022-08-17 13:04:50 +03:00
|
|
|
,withAttr (attrName "help" <> attrName "heading") $ str "Other"
|
2018-10-17 00:16:58 +03:00
|
|
|
,renderKey ("a ", "add transaction (hledger add)")
|
|
|
|
,renderKey ("A ", "add transaction (hledger-iadd)")
|
2020-07-19 00:26:07 +03:00
|
|
|
,renderKey ("B ", "show amounts/costs")
|
2018-10-17 00:16:58 +03:00
|
|
|
,renderKey ("E ", "open editor")
|
|
|
|
,renderKey ("I ", "toggle balance assertions")
|
2020-07-19 00:26:07 +03:00
|
|
|
,renderKey ("V ", "show amounts/market values")
|
2018-10-17 00:16:58 +03:00
|
|
|
,renderKey ("g ", "reload data")
|
2019-01-19 03:29:56 +03:00
|
|
|
,renderKey ("C-l ", "redraw & recenter")
|
|
|
|
,renderKey ("C-z ", "suspend")
|
2018-10-17 00:16:58 +03:00
|
|
|
,renderKey ("q ", "quit")
|
2016-06-20 21:09:12 +03:00
|
|
|
]
|
|
|
|
]
|
2016-08-13 03:44:55 +03:00
|
|
|
-- ,vBox [
|
|
|
|
-- str " "
|
|
|
|
-- ,hCenter $ padLeftRight 1 $
|
|
|
|
-- hCenter (str "MANUAL")
|
|
|
|
-- <=>
|
|
|
|
-- hCenter (hBox [
|
|
|
|
-- renderKey ("t", "text")
|
|
|
|
-- ,str " "
|
|
|
|
-- ,renderKey ("m", "man page")
|
|
|
|
-- ,str " "
|
|
|
|
-- ,renderKey ("i", "info")
|
|
|
|
-- ])
|
|
|
|
-- ]
|
2016-06-20 21:09:12 +03:00
|
|
|
]
|
2016-06-10 21:50:57 +03:00
|
|
|
where
|
2022-08-17 13:04:50 +03:00
|
|
|
renderKey (key,desc) = withAttr (attrName "help" <> attrName "key") (str key) <+> str " " <+> str desc
|
2016-06-10 21:50:57 +03:00
|
|
|
|
|
|
|
-- | Event handler used when help mode is active.
|
2017-12-30 23:59:25 +03:00
|
|
|
-- May invoke $PAGER, less, man or info, which is likely to fail on MS Windows, TODO.
|
2022-08-17 13:04:50 +03:00
|
|
|
helpHandle :: BrickEvent Name AppEvent -> EventM Name UIState ()
|
|
|
|
helpHandle ev = do
|
|
|
|
ui <- get
|
2022-09-02 23:36:05 +03:00
|
|
|
let ui' = ui{aMode=Normal}
|
2016-06-10 21:50:57 +03:00
|
|
|
case ev of
|
2022-08-23 01:55:14 +03:00
|
|
|
VtyEvent e | e `elem` closeHelpEvents -> put' ui'
|
2022-08-23 03:54:16 +03:00
|
|
|
VtyEvent (EvKey (KChar 'p') []) -> suspendAndResume (runPagerForTopic "hledger-ui" Nothing >> return ui')
|
|
|
|
VtyEvent (EvKey (KChar 'm') []) -> suspendAndResume (runManForTopic "hledger-ui" Nothing >> return ui')
|
|
|
|
VtyEvent (EvKey (KChar 'i') []) -> suspendAndResume (runInfoForTopic "hledger-ui" Nothing >> return ui')
|
2022-08-17 13:04:50 +03:00
|
|
|
_ -> return ()
|
2016-08-13 03:44:55 +03:00
|
|
|
where
|
2020-07-18 22:24:37 +03:00
|
|
|
closeHelpEvents = moveLeftEvents ++ [EvKey KEsc [], EvKey (KChar '?') [], EvKey (KChar 'q') []]
|
2016-06-10 21:50:57 +03:00
|
|
|
|
2021-11-16 04:22:05 +03:00
|
|
|
-- | Draw the minibuffer with the given label.
|
|
|
|
minibuffer :: T.Text -> Editor String Name -> Widget Name
|
|
|
|
minibuffer string ed =
|
2022-08-17 13:04:50 +03:00
|
|
|
forceAttr (attrName "border" <> attrName "minibuffer") $
|
2021-11-16 04:22:05 +03:00
|
|
|
hBox [txt $ string <> ": ", renderEditor (str . unlines) True ed]
|
2015-08-21 18:40:05 +03:00
|
|
|
|
2016-07-25 04:06:49 +03:00
|
|
|
borderQueryStr :: String -> Widget Name
|
2016-06-11 03:30:45 +03:00
|
|
|
borderQueryStr "" = str ""
|
2022-08-17 13:04:50 +03:00
|
|
|
borderQueryStr qry = str " matching " <+> withAttr (attrName "border" <> attrName "query") (str qry)
|
2016-06-11 03:30:45 +03:00
|
|
|
|
2016-07-25 04:06:49 +03:00
|
|
|
borderDepthStr :: Maybe Int -> Widget Name
|
2016-06-11 03:30:45 +03:00
|
|
|
borderDepthStr Nothing = str ""
|
2022-08-17 13:04:50 +03:00
|
|
|
borderDepthStr (Just d) = str " to depth " <+> withAttr (attrName "border" <> attrName "query") (str $ show d)
|
2016-06-11 03:30:45 +03:00
|
|
|
|
2016-08-13 03:44:55 +03:00
|
|
|
borderPeriodStr :: String -> Period -> Widget Name
|
|
|
|
borderPeriodStr _ PeriodAll = str ""
|
2022-08-17 13:04:50 +03:00
|
|
|
borderPeriodStr preposition p = str (" "++preposition++" ") <+> withAttr (attrName "border" <> attrName "query") (str . T.unpack $ showPeriod p)
|
2016-08-02 18:22:21 +03:00
|
|
|
|
2016-07-25 04:06:49 +03:00
|
|
|
borderKeysStr :: [(String,String)] -> Widget Name
|
2021-08-16 09:09:55 +03:00
|
|
|
borderKeysStr = borderKeysStr' . map (second str)
|
2016-08-13 03:44:55 +03:00
|
|
|
|
|
|
|
borderKeysStr' :: [(String,Widget Name)] -> Widget Name
|
|
|
|
borderKeysStr' keydescs =
|
2016-06-11 03:30:45 +03:00
|
|
|
hBox $
|
|
|
|
intersperse sep $
|
2022-08-17 13:04:50 +03:00
|
|
|
[withAttr (attrName "border" <> attrName "key") (str keys) <+> str ":" <+> desc | (keys, desc) <- keydescs]
|
2016-06-11 03:30:45 +03:00
|
|
|
where
|
|
|
|
-- sep = str " | "
|
|
|
|
sep = str " "
|
|
|
|
|
2020-02-16 16:27:09 +03:00
|
|
|
-- | Show both states of a toggle ("aaa/bbb"), highlighting the active one.
|
2018-11-05 00:27:50 +03:00
|
|
|
renderToggle :: Bool -> String -> String -> Widget Name
|
|
|
|
renderToggle isright l r =
|
2022-08-17 13:04:50 +03:00
|
|
|
let bold = withAttr (attrName "border" <> attrName "selected") in
|
2018-11-05 00:27:50 +03:00
|
|
|
if isright
|
2019-07-15 13:28:52 +03:00
|
|
|
then str (l++"/") <+> bold (str r)
|
2018-11-05 00:27:50 +03:00
|
|
|
else bold (str l) <+> str ("/"++r)
|
|
|
|
|
2020-02-16 16:27:09 +03:00
|
|
|
-- | Show a toggle's label, highlighted (bold) when the toggle is active.
|
|
|
|
renderToggle1 :: Bool -> String -> Widget Name
|
|
|
|
renderToggle1 isactive l =
|
2022-08-17 13:04:50 +03:00
|
|
|
let bold = withAttr (attrName "border" <> attrName "selected") in
|
2020-02-16 16:27:09 +03:00
|
|
|
if isactive
|
|
|
|
then bold (str l)
|
|
|
|
else str l
|
|
|
|
|
2016-06-12 20:55:43 +03:00
|
|
|
-- temporary shenanigans:
|
|
|
|
|
|
|
|
-- | Convert the special account name "*" (from balance report with depth limit 0) to something clearer.
|
|
|
|
replaceHiddenAccountsNameWith :: AccountName -> AccountName -> AccountName
|
|
|
|
replaceHiddenAccountsNameWith anew a | a == hiddenAccountsName = anew
|
|
|
|
| a == "*" = anew
|
|
|
|
| otherwise = a
|
|
|
|
|
|
|
|
hiddenAccountsName = "..." -- for now
|
|
|
|
|
2016-06-11 03:30:45 +03:00
|
|
|
-- generic
|
|
|
|
|
2018-10-23 14:56:29 +03:00
|
|
|
--topBottomBorderWithLabel :: Widget Name -> Widget Name -> Widget Name
|
|
|
|
--topBottomBorderWithLabel label = \wrapped ->
|
|
|
|
-- Widget Greedy Greedy $ do
|
|
|
|
-- c <- getContext
|
|
|
|
-- let (_w,h) = (c^.availWidthL, c^.availHeightL)
|
|
|
|
-- h' = h - 2
|
|
|
|
-- wrapped' = vLimit (h') wrapped
|
|
|
|
-- debugmsg =
|
|
|
|
-- ""
|
|
|
|
-- -- " debug: "++show (_w,h')
|
|
|
|
-- render $
|
|
|
|
-- hBorderWithLabel (label <+> str debugmsg)
|
|
|
|
-- <=>
|
|
|
|
-- wrapped'
|
|
|
|
-- <=>
|
|
|
|
-- hBorder
|
2015-08-23 03:46:57 +03:00
|
|
|
|
2016-07-25 04:06:49 +03:00
|
|
|
topBottomBorderWithLabels :: Widget Name -> Widget Name -> Widget Name -> Widget Name
|
2018-10-23 14:56:29 +03:00
|
|
|
topBottomBorderWithLabels toplabel bottomlabel body =
|
2015-08-28 08:45:49 +03:00
|
|
|
Widget Greedy Greedy $ do
|
|
|
|
c <- getContext
|
|
|
|
let (_w,h) = (c^.availWidthL, c^.availHeightL)
|
|
|
|
h' = h - 2
|
2018-10-23 14:56:29 +03:00
|
|
|
body' = vLimit (h') body
|
2015-08-28 08:45:49 +03:00
|
|
|
debugmsg =
|
|
|
|
""
|
|
|
|
-- " debug: "++show (_w,h')
|
|
|
|
render $
|
2022-08-17 13:04:50 +03:00
|
|
|
hBorderWithLabel (withAttr (attrName "border") $ toplabel <+> str debugmsg)
|
2015-08-28 08:45:49 +03:00
|
|
|
<=>
|
2018-10-23 14:56:29 +03:00
|
|
|
body'
|
2015-08-28 08:45:49 +03:00
|
|
|
<=>
|
2022-08-17 13:04:50 +03:00
|
|
|
hBorderWithLabel (withAttr (attrName "border") bottomlabel)
|
2015-08-28 08:45:49 +03:00
|
|
|
|
2018-10-23 14:56:29 +03:00
|
|
|
---- XXX should be equivalent to the above, but isn't (page down goes offscreen)
|
2022-09-02 23:36:05 +03:00
|
|
|
--_topBottomBorderWithLabel :: Widget Name -> Widget Name -> Widget Name
|
|
|
|
--_topBottomBorderWithLabel label = \wrapped ->
|
2018-10-23 14:56:29 +03:00
|
|
|
-- let debugmsg = ""
|
|
|
|
-- in hBorderWithLabel (label <+> str debugmsg)
|
|
|
|
-- <=>
|
|
|
|
-- wrapped
|
|
|
|
-- <=>
|
|
|
|
-- hBorder
|
2015-08-25 02:22:09 +03:00
|
|
|
|
|
|
|
-- XXX superseded by pad, in theory
|
2015-08-23 03:46:57 +03:00
|
|
|
-- | Wrap a widget in a margin with the given horizontal and vertical
|
|
|
|
-- thickness, using the current background colour or the specified
|
2015-08-25 02:22:09 +03:00
|
|
|
-- colour.
|
|
|
|
-- XXX May disrupt border style of inner widgets.
|
2022-09-02 23:36:05 +03:00
|
|
|
-- XXX Should reduce the available size visible to inner widget, but doesn't seem to (cf rsDraw).
|
2016-07-25 04:06:49 +03:00
|
|
|
margin :: Int -> Int -> Maybe Color -> Widget Name -> Widget Name
|
2021-08-16 08:25:49 +03:00
|
|
|
margin h v mcolour w = Widget Greedy Greedy $ do
|
2022-08-23 04:55:37 +03:00
|
|
|
ctx <- getContext
|
|
|
|
let w' = vLimit (ctx^.availHeightL - v*2) $ hLimit (ctx^.availWidthL - h*2) w
|
2015-08-23 03:46:57 +03:00
|
|
|
attr = maybe currentAttr (\c -> c `on` c) mcolour
|
|
|
|
render $
|
|
|
|
withBorderAttr attr $
|
|
|
|
withBorderStyle (borderStyleFromChar ' ') $
|
|
|
|
applyN v (hBorder <=>) $
|
|
|
|
applyN h (vBorder <+>) $
|
|
|
|
applyN v (<=> hBorder) $
|
|
|
|
applyN h (<+> vBorder) $
|
|
|
|
w'
|
|
|
|
|
|
|
|
-- withBorderAttr attr .
|
|
|
|
-- withBorderStyle (borderStyleFromChar ' ') .
|
|
|
|
-- applyN n border
|
|
|
|
|
2016-07-25 04:06:49 +03:00
|
|
|
withBorderAttr :: Attr -> Widget Name -> Widget Name
|
2022-08-17 13:04:50 +03:00
|
|
|
withBorderAttr attr = updateAttrMap (applyAttrMappings [(attrName "border", attr)])
|
2015-08-23 03:46:57 +03:00
|
|
|
|
2018-10-23 14:56:29 +03:00
|
|
|
---- | Like brick's continue, but first run some action to modify brick's state.
|
|
|
|
---- This action does not affect the app state, but might eg adjust a widget's scroll position.
|
|
|
|
--continueWith :: EventM n () -> ui -> EventM n (Next ui)
|
|
|
|
--continueWith brickaction ui = brickaction >> continue ui
|
2017-06-30 17:51:08 +03:00
|
|
|
|
2018-10-23 14:56:29 +03:00
|
|
|
---- | Scroll a list's viewport so that the selected item is at the top
|
|
|
|
---- of the display area.
|
|
|
|
--scrollToTop :: List Name e -> EventM Name ()
|
|
|
|
--scrollToTop list = do
|
|
|
|
-- let vpname = list^.listNameL
|
2019-07-15 13:28:52 +03:00
|
|
|
-- setTop (viewportScroll vpname) 0
|
2017-06-30 17:51:08 +03:00
|
|
|
|
|
|
|
-- | Scroll a list's viewport so that the selected item is centered in the
|
|
|
|
-- middle of the display area.
|
2022-08-17 13:04:50 +03:00
|
|
|
scrollSelectionToMiddle :: List Name item -> EventM Name UIState ()
|
2017-06-30 17:51:08 +03:00
|
|
|
scrollSelectionToMiddle list = do
|
2021-11-21 04:33:28 +03:00
|
|
|
case list^.listSelectedL of
|
|
|
|
Nothing -> return ()
|
|
|
|
Just selectedrow -> do
|
|
|
|
Vty{outputIface} <- getVtyHandle
|
|
|
|
pageheight <- dbg4 "pageheight" . snd <$> liftIO (displayBounds outputIface)
|
2017-06-30 17:51:08 +03:00
|
|
|
let
|
|
|
|
itemheight = dbg4 "itemheight" $ list^.listItemHeightL
|
2021-11-21 04:33:28 +03:00
|
|
|
itemsperpage = dbg4 "itemsperpage" $ pageheight `div` itemheight
|
2017-06-30 17:51:08 +03:00
|
|
|
toprow = dbg4 "toprow" $ max 0 (selectedrow - (itemsperpage `div` 2)) -- assuming ViewportScroll's row offset is measured in list items not screen rows
|
2021-11-21 04:33:28 +03:00
|
|
|
setTop (viewportScroll $ list^.listNameL) toprow
|
2017-06-30 18:37:10 +03:00
|
|
|
|
2017-06-30 19:18:54 +03:00
|
|
|
-- arrow keys vi keys emacs keys
|
2020-07-18 18:39:23 +03:00
|
|
|
moveUpEvents = [EvKey KUp [] , EvKey (KChar 'k') [], EvKey (KChar 'p') [MCtrl]]
|
|
|
|
moveDownEvents = [EvKey KDown [] , EvKey (KChar 'j') [], EvKey (KChar 'n') [MCtrl]]
|
|
|
|
moveLeftEvents = [EvKey KLeft [] , EvKey (KChar 'h') [], EvKey (KChar 'b') [MCtrl]]
|
|
|
|
moveRightEvents = [EvKey KRight [], EvKey (KChar 'l') [], EvKey (KChar 'f') [MCtrl]]
|
2017-06-30 18:37:10 +03:00
|
|
|
|
|
|
|
normaliseMovementKeys ev
|
|
|
|
| ev `elem` moveUpEvents = EvKey KUp []
|
|
|
|
| ev `elem` moveDownEvents = EvKey KDown []
|
|
|
|
| ev `elem` moveLeftEvents = EvKey KLeft []
|
|
|
|
| ev `elem` moveRightEvents = EvKey KRight []
|
|
|
|
| otherwise = ev
|
2021-08-24 08:50:32 +03:00
|
|
|
|
2022-09-08 21:53:42 +03:00
|
|
|
-- | Restrict the ReportSpec's query by adding the given additional query.
|
|
|
|
reportSpecAddQuery :: Query -> ReportSpec -> ReportSpec
|
|
|
|
reportSpecAddQuery q rspec =
|
|
|
|
rspec{_rsQuery=simplifyQuery $ And [_rsQuery rspec, q]}
|
|
|
|
|
2022-09-09 00:56:20 +03:00
|
|
|
-- | Update the ReportSpec's query to exclude future transactions (later than its "today" date)
|
2021-08-24 08:50:32 +03:00
|
|
|
-- and forecast transactions (generated by --forecast), if the given forecast DateSpan is Nothing,
|
|
|
|
-- and include them otherwise.
|
2022-09-09 00:56:20 +03:00
|
|
|
reportSpecSetFutureAndForecast :: Maybe DateSpan -> ReportSpec -> ReportSpec
|
|
|
|
reportSpecSetFutureAndForecast fcast rspec =
|
2022-08-23 04:55:37 +03:00
|
|
|
rspec{_rsQuery=simplifyQuery $ And [_rsQuery rspec, periodq, excludeforecastq fcast]}
|
2021-08-24 08:50:32 +03:00
|
|
|
where
|
|
|
|
periodq = Date . periodAsDateSpan . period_ $ _rsReportOpts rspec
|
|
|
|
-- Except in forecast mode, exclude future/forecast transactions.
|
|
|
|
excludeforecastq (Just _) = Any
|
|
|
|
excludeforecastq Nothing = -- not:date:tomorrow- not:tag:generated-transaction
|
|
|
|
And [
|
2022-09-09 00:56:20 +03:00
|
|
|
Not (Date $ DateSpan (Just $ addDays 1 $ _rsDay rspec) Nothing)
|
2021-08-24 08:50:32 +03:00
|
|
|
,Not generatedTransactionTag
|
|
|
|
]
|
2021-11-18 20:53:02 +03:00
|
|
|
|
2021-11-18 21:27:55 +03:00
|
|
|
-- Vertically scroll the named list's viewport with the given number of non-empty items
|
2021-11-18 20:53:02 +03:00
|
|
|
-- by the given positive or negative number of items (usually 1 or -1).
|
|
|
|
-- The selection will be moved when necessary to keep it visible and allow the scroll.
|
2022-08-17 13:04:50 +03:00
|
|
|
listScrollPushingSelection :: Name -> Int -> Int -> EventM Name (List Name item) (GenericList Name Vector item)
|
|
|
|
listScrollPushingSelection name listheight scrollamt = do
|
|
|
|
list <- get
|
2021-11-18 21:27:55 +03:00
|
|
|
viewportScroll name `vScrollBy` scrollamt
|
2021-11-18 20:53:02 +03:00
|
|
|
mvp <- lookupViewport name
|
2021-11-18 21:27:55 +03:00
|
|
|
case mvp of
|
|
|
|
Just VP{_vpTop, _vpSize=(_,vpheight)} -> do
|
|
|
|
let mselidx = listSelected list
|
|
|
|
case mselidx of
|
|
|
|
Just selidx -> return $ pushsel list
|
|
|
|
where
|
|
|
|
pushsel
|
|
|
|
| scrollamt > 0, selidx <= _vpTop && selidx < (listheight-1) = listMoveDown
|
|
|
|
| scrollamt < 0, selidx >= _vpTop + vpheight - 1 && selidx > 0 = listMoveUp
|
|
|
|
| otherwise = id
|
|
|
|
_ -> return list
|
2021-11-18 20:53:02 +03:00
|
|
|
_ -> return list
|
2022-08-23 01:55:14 +03:00
|
|
|
|
|
|
|
-- | Log a string to ./debug.log before returning the second argument,
|
|
|
|
-- if the global debug level is at or above a standard hledger-ui debug level.
|
|
|
|
-- Uses unsafePerformIO.
|
|
|
|
dlogUiTrace :: String -> a -> a
|
|
|
|
dlogUiTrace = dlogTraceAt uiDebugLevel
|
|
|
|
|
2022-10-29 05:30:26 +03:00
|
|
|
-- | Like dlogUiTrace, but convenient in IO.
|
|
|
|
dlogUiTraceIO :: String -> IO ()
|
|
|
|
dlogUiTraceIO s = dlogUiTrace s $ return ()
|
|
|
|
|
|
|
|
-- | Like dlogUiTrace, but convenient in event handlers.
|
2022-08-23 01:55:14 +03:00
|
|
|
dlogUiTraceM :: String -> EventM Name UIState ()
|
|
|
|
dlogUiTraceM s = dlogUiTrace s $ return ()
|
|
|
|
|
2022-10-29 05:30:26 +03:00
|
|
|
-- | Like dlogUiTraceM, but log a prefix, "screen stack", a postfix,
|
|
|
|
-- and a compact view of the current screen stack,
|
|
|
|
-- from topmost screen to currently-viewed screen,
|
|
|
|
-- with each screen rendered by the given rendering function.
|
|
|
|
-- Useful for inspecting states across the whole screen stack.
|
|
|
|
-- To just show the stack: @dlogUiScreenStack "" "" screenId ui@
|
|
|
|
dlogUiScreenStack :: String -> String -> (Screen -> String) -> UIState -> EventM Name UIState ()
|
|
|
|
dlogUiScreenStack prefix postfix showscr ui =
|
|
|
|
dlogUiTraceM $ prefix ++ "screen stack: " ++ postfix ++ (unwords $ mapScreens showscr ui)
|
|
|
|
|
|
|
|
-- Show a screen's compact id (first letter of its constructor).
|
|
|
|
screenId :: Screen -> String
|
|
|
|
screenId = \case
|
|
|
|
MS _ -> "M" -- menu
|
|
|
|
AS _ -> "A" -- all accounts
|
|
|
|
BS _ -> "B" -- bs accounts
|
|
|
|
IS _ -> "I" -- is accounts
|
|
|
|
RS _ -> "R" -- menu
|
|
|
|
TS _ -> "T" -- transaction
|
|
|
|
ES _ -> "E" -- error
|
|
|
|
|
2022-08-23 01:55:14 +03:00
|
|
|
-- | Log hledger-ui events at this debug level.
|
|
|
|
uiDebugLevel :: Int
|
|
|
|
uiDebugLevel = 2
|
|
|
|
|
|
|
|
-- | How many blank items to add to lists to fill the full window height.
|
|
|
|
uiNumBlankItems :: Int
|
|
|
|
uiNumBlankItems
|
|
|
|
-- | debugLevel >= uiDebugLevel = 0 -- suppress to improve debug output.
|
|
|
|
-- | otherwise
|
|
|
|
= 100 -- 100 ought to be enough for anyone
|