module Penny.Cabin.Posts.Growers (
GrowOpts(..),
growCells, Fields(..), grownWidth,
eFields, EFields(..), pairWithSpacer) where
import Control.Applicative((<$>), Applicative(pure, (<*>)))
import qualified Data.Foldable as Fdbl
import Data.Map (elems)
import qualified Data.Map as Map
import qualified Data.Semigroup as Semi
import Data.Semigroup ((<>))
import Data.Text (Text, pack, empty)
import qualified Data.Text as X
import qualified Penny.Cabin.Posts.Fields as F
import qualified Penny.Cabin.Posts.Meta as M
import Penny.Cabin.Posts.Meta (Box)
import qualified Penny.Cabin.Posts.Spacers as S
import qualified Penny.Cabin.Row as R
import qualified Penny.Cabin.Scheme as E
import qualified Penny.Liberty as Ly
import qualified Penny.Lincoln as L
import qualified Penny.Lincoln.Queries as Q
data GrowOpts = GrowOpts
{ dateFormat :: Box -> X.Text
, qtyFormat :: Box -> X.Text
, balanceFormat :: L.Commodity -> L.Qty -> X.Text
, fields :: F.Fields Bool
}
growCells ::
GrowOpts
-> [Box]
-> Fields (Maybe ([R.ColumnSpec], Int))
growCells o infos = toPair <$> wanted <*> growers where
toPair b gwr
| b =
let cs = map (gwr o) infos
w = Fdbl.foldl' f 0 cs where
f acc c = max acc (widestLine c)
cs' = map (sizer (R.Width w)) cs
in if w > 0 then Just (cs', w) else Nothing
| otherwise = Nothing
wanted = growingFields . fields $ o
widestLine :: PreSpec -> Int
widestLine (PreSpec _ _ bs) =
case bs of
[] -> 0
xs -> maximum . map (R.unWidth . E.width) $ xs
data PreSpec = PreSpec {
_justification :: R.Justification
, _padSpec :: (E.Label, E.EvenOdd)
, _bits :: [E.PreChunk] }
sizer :: R.Width -> PreSpec -> R.ColumnSpec
sizer w (PreSpec j ts bs) = R.ColumnSpec j w ts bs
oneLine :: Text -> E.Label -> Box -> PreSpec
oneLine t lbl b =
let eo = E.fromVisibleNum . M.visibleNum . L.boxMeta $ b
j = R.LeftJustify
pcs = E.PreChunk lbl eo t
in PreSpec j (lbl, eo) [pcs]
growers :: Fields (GrowOpts -> Box -> PreSpec)
growers = Fields
{ globalTransaction = const getGlobalTransaction
, revGlobalTransaction = const getRevGlobalTransaction
, globalPosting = const getGlobalPosting
, revGlobalPosting = const getRevGlobalPosting
, fileTransaction = const getFileTransaction
, revFileTransaction = const getRevFileTransaction
, filePosting = const getFilePosting
, revFilePosting = const getRevFilePosting
, filtered = const getFiltered
, revFiltered = const getRevFiltered
, sorted = const getSorted
, revSorted = const getRevSorted
, visible = const getVisible
, revVisible = const getRevVisible
, lineNum = const getLineNum
, date = \o -> getDate (dateFormat o)
, flag = const getFlag
, number = const getNumber
, postingDrCr = const getPostingDrCr
, postingCmdty = const getPostingCmdty
, postingQty = \o -> getPostingQty (qtyFormat o)
, totalDrCr = const getTotalDrCr
, totalCmdty = const getTotalCmdty
, totalQty = \o -> getTotalQty (balanceFormat o)
}
serialCellMaybe ::
(L.PostFam -> Maybe Int)
-> Box -> PreSpec
serialCellMaybe f b = oneLine t E.Other b
where
t = case f (L.boxPostFam b) of
Nothing -> X.empty
Just i -> X.pack . show $ i
serialCell ::
(M.PostMeta -> Int)
-> Box -> PreSpec
serialCell f b = oneLine t E.Other b
where
t = pack . show . f . L.boxMeta $ b
getGlobalTransaction :: Box -> PreSpec
getGlobalTransaction =
serialCellMaybe (fmap (L.forward . L.unGlobalTransaction)
. Q.globalTransaction)
getRevGlobalTransaction :: Box -> PreSpec
getRevGlobalTransaction =
serialCellMaybe (fmap (L.backward . L.unGlobalTransaction)
. Q.globalTransaction)
getGlobalPosting :: Box -> PreSpec
getGlobalPosting =
serialCellMaybe (fmap (L.forward . L.unGlobalPosting)
. Q.globalPosting)
getRevGlobalPosting :: Box -> PreSpec
getRevGlobalPosting =
serialCellMaybe (fmap (L.backward . L.unGlobalPosting)
. Q.globalPosting)
getFileTransaction :: Box -> PreSpec
getFileTransaction =
serialCellMaybe (fmap (L.forward . L.unFileTransaction)
. Q.fileTransaction)
getRevFileTransaction :: Box -> PreSpec
getRevFileTransaction =
serialCellMaybe (fmap (L.backward . L.unFileTransaction)
. Q.fileTransaction)
getFilePosting :: Box -> PreSpec
getFilePosting =
serialCellMaybe (fmap (L.forward . L.unFilePosting)
. Q.filePosting)
getRevFilePosting :: Box -> PreSpec
getRevFilePosting =
serialCellMaybe (fmap (L.backward . L.unFilePosting)
. Q.filePosting)
getSorted :: Box -> PreSpec
getSorted =
serialCell (L.forward . Ly.unSortedNum . M.sortedNum)
getRevSorted :: Box -> PreSpec
getRevSorted =
serialCell (L.backward . Ly.unSortedNum . M.sortedNum)
getFiltered :: Box -> PreSpec
getFiltered =
serialCell (L.forward . Ly.unFilteredNum . M.filteredNum)
getRevFiltered :: Box -> PreSpec
getRevFiltered =
serialCell (L.backward . Ly.unFilteredNum . M.filteredNum)
getVisible :: Box -> PreSpec
getVisible =
serialCell (L.forward . M.unVisibleNum . M.visibleNum)
getRevVisible :: Box -> PreSpec
getRevVisible =
serialCell (L.backward . M.unVisibleNum . M.visibleNum)
getLineNum :: Box -> PreSpec
getLineNum b = oneLine t E.Other b where
lineTxt = pack . show . L.unPostingLine
t = maybe empty lineTxt (Q.postingLine . L.boxPostFam $ b)
getDate :: (Box -> X.Text) -> Box -> PreSpec
getDate gd b = oneLine (gd b) E.Other b
getFlag :: Box -> PreSpec
getFlag i = oneLine t E.Other i where
t = maybe empty L.text (Q.flag . L.boxPostFam $ i)
getNumber :: Box -> PreSpec
getNumber i = oneLine t E.Other i where
t = maybe empty L.text (Q.number . L.boxPostFam $ i)
dcTxt :: L.DrCr -> Text
dcTxt L.Debit = X.singleton '<'
dcTxt L.Credit = X.singleton '>'
coloredPostingCell :: Text -> Box -> PreSpec
coloredPostingCell t i = PreSpec j (lbl, eo) [bit] where
j = R.LeftJustify
lbl = case Q.drCr . L.boxPostFam $ i of
L.Debit -> E.Debit
L.Credit -> E.Credit
eo = E.fromVisibleNum . M.visibleNum . L.boxMeta $ i
bit = E.PreChunk lbl eo t
getPostingDrCr :: Box -> PreSpec
getPostingDrCr i = coloredPostingCell t i where
t = dcTxt . Q.drCr . L.boxPostFam $ i
getPostingCmdty :: Box -> PreSpec
getPostingCmdty i = coloredPostingCell t i where
t = L.unCommodity . Q.commodity . L.boxPostFam $ i
getPostingQty :: (Box -> X.Text) -> Box -> PreSpec
getPostingQty qf i = coloredPostingCell (qf i) i
getTotalDrCr :: Box -> PreSpec
getTotalDrCr i =
let vn = M.visibleNum . L.boxMeta $ i
ps = (lbl, eo)
dc = Q.drCr . L.boxPostFam $ i
lbl = E.dcToLbl dc
eo = E.fromVisibleNum vn
bal = L.unBalance . M.balance . L.boxMeta $ i
bits =
if Map.null bal
then [E.PreChunk E.Zero eo (pack "--")]
else fmap (flip E.bottomLineToDrCr eo) . elems $ bal
j = R.LeftJustify
in PreSpec j ps bits
getTotalCmdty :: Box -> PreSpec
getTotalCmdty i =
let vn = M.visibleNum . L.boxMeta $ i
j = R.RightJustify
ps = (lbl, eo)
dc = Q.drCr . L.boxPostFam $ i
eo = E.fromVisibleNum vn
lbl = E.dcToLbl dc
bal = Map.toList . L.unBalance . M.balance . L.boxMeta $ i
preChunks = E.balancesToCmdtys eo bal
in PreSpec j ps preChunks
getTotalQty ::
(L.Commodity -> L.Qty -> X.Text)
-> Box
-> PreSpec
getTotalQty balFmt i =
let vn = M.visibleNum . L.boxMeta $ i
j = R.LeftJustify
dc = Q.drCr . L.boxPostFam $ i
ps = (E.dcToLbl dc, eo)
eo = E.fromVisibleNum vn
bal = Map.toList . L.unBalance . M.balance . L.boxMeta $ i
preChunks = E.balanceToQtys balFmt eo bal
in PreSpec j ps preChunks
growingFields :: F.Fields Bool -> Fields Bool
growingFields f = Fields
{ globalTransaction = F.globalTransaction f
, revGlobalTransaction = F.revGlobalTransaction f
, globalPosting = F.globalPosting f
, revGlobalPosting = F.revGlobalPosting f
, fileTransaction = F.fileTransaction f
, revFileTransaction = F.revFileTransaction f
, filePosting = F.filePosting f
, revFilePosting = F.revFilePosting f
, filtered = F.filtered f
, revFiltered = F.revFiltered f
, sorted = F.sorted f
, revSorted = F.revSorted f
, visible = F.visible f
, revVisible = F.revVisible f
, lineNum = F.lineNum f
, date = F.date f
, flag = F.flag f
, number = F.number f
, postingDrCr = F.postingDrCr f
, postingCmdty = F.postingCmdty f
, postingQty = F.postingQty f
, totalDrCr = F.totalDrCr f
, totalCmdty = F.totalCmdty f
, totalQty = F.totalQty f }
data EFields =
EGlobalTransaction
| ERevGlobalTransaction
| EGlobalPosting
| ERevGlobalPosting
| EFileTransaction
| ERevFileTransaction
| EFilePosting
| ERevFilePosting
| EFiltered
| ERevFiltered
| ESorted
| ERevSorted
| EVisible
| ERevVisible
| ELineNum
| EDate
| EFlag
| ENumber
| EPostingDrCr
| EPostingCmdty
| EPostingQty
| ETotalDrCr
| ETotalCmdty
| ETotalQty
deriving (Show, Eq, Ord, Enum)
eFields :: Fields EFields
eFields = Fields
{ globalTransaction = EGlobalTransaction
, revGlobalTransaction = ERevGlobalTransaction
, globalPosting = EGlobalPosting
, revGlobalPosting = ERevGlobalPosting
, fileTransaction = EFileTransaction
, revFileTransaction = ERevFileTransaction
, filePosting = EFilePosting
, revFilePosting = ERevFilePosting
, filtered = EFiltered
, revFiltered = ERevFiltered
, sorted = ESorted
, revSorted = ERevSorted
, visible = EVisible
, revVisible = ERevVisible
, lineNum = ELineNum
, date = EDate
, flag = EFlag
, number = ENumber
, postingDrCr = EPostingDrCr
, postingCmdty = EPostingCmdty
, postingQty = EPostingQty
, totalDrCr = ETotalDrCr
, totalCmdty = ETotalCmdty
, totalQty = ETotalQty }
data Fields a = Fields
{ globalTransaction :: a
, revGlobalTransaction :: a
, globalPosting :: a
, revGlobalPosting :: a
, fileTransaction :: a
, revFileTransaction :: a
, filePosting :: a
, revFilePosting :: a
, filtered :: a
, revFiltered :: a
, sorted :: a
, revSorted :: a
, visible :: a
, revVisible :: a
, lineNum :: a
, date :: a
, flag :: a
, number :: a
, postingDrCr :: a
, postingCmdty :: a
, postingQty :: a
, totalDrCr :: a
, totalCmdty :: a
, totalQty :: a }
deriving (Show, Eq)
instance Fdbl.Foldable Fields where
foldr f z i =
f (globalTransaction i)
(f (revGlobalTransaction i)
(f (globalPosting i)
(f (revGlobalPosting i)
(f (fileTransaction i)
(f (revFileTransaction i)
(f (filePosting i)
(f (revFilePosting i)
(f (filtered i)
(f (revFiltered i)
(f (sorted i)
(f (revSorted i)
(f (visible i)
(f (revVisible i)
(f (lineNum i)
(f (date i)
(f (flag i)
(f (number i)
(f (postingDrCr i)
(f (postingCmdty i)
(f (postingQty i)
(f (totalDrCr i)
(f (totalCmdty i)
(f (totalQty i) z)))))))))))))))))))))))
instance Functor Fields where
fmap f i = Fields
{ globalTransaction = f (globalTransaction i)
, revGlobalTransaction = f (revGlobalTransaction i)
, globalPosting = f (globalPosting i)
, revGlobalPosting = f (revGlobalPosting i)
, fileTransaction = f (fileTransaction i)
, revFileTransaction = f (revFileTransaction i)
, filePosting = f (filePosting i)
, revFilePosting = f (revFilePosting i)
, filtered = f (filtered i)
, revFiltered = f (revFiltered i)
, sorted = f (sorted i)
, revSorted = f (revSorted i)
, visible = f (visible i)
, revVisible = f (revVisible i)
, lineNum = f (lineNum i)
, date = f (date i)
, flag = f (flag i)
, number = f (number i)
, postingDrCr = f (postingDrCr i)
, postingCmdty = f (postingCmdty i)
, postingQty = f (postingQty i)
, totalDrCr = f (totalDrCr i)
, totalCmdty = f (totalCmdty i)
, totalQty = f (totalQty i) }
instance Applicative Fields where
pure a = Fields
{ globalTransaction = a
, revGlobalTransaction = a
, globalPosting = a
, revGlobalPosting = a
, fileTransaction = a
, revFileTransaction = a
, filePosting = a
, revFilePosting = a
, filtered = a
, revFiltered = a
, sorted = a
, revSorted = a
, visible = a
, revVisible = a
, lineNum = a
, date = a
, flag = a
, number = a
, postingDrCr = a
, postingCmdty = a
, postingQty = a
, totalDrCr = a
, totalCmdty = a
, totalQty = a }
fl <*> fa = Fields
{ globalTransaction = globalTransaction fl (globalTransaction fa)
, revGlobalTransaction = revGlobalTransaction fl (revGlobalTransaction fa)
, globalPosting = globalPosting fl (globalPosting fa)
, revGlobalPosting = revGlobalPosting fl (revGlobalPosting fa)
, fileTransaction = fileTransaction fl (fileTransaction fa)
, revFileTransaction = revFileTransaction fl (revFileTransaction fa)
, filePosting = filePosting fl (filePosting fa)
, revFilePosting = revFilePosting fl (revFilePosting fa)
, filtered = filtered fl (filtered fa)
, revFiltered = revFiltered fl (revFiltered fa)
, sorted = sorted fl (sorted fa)
, revSorted = revSorted fl (revSorted fa)
, visible = visible fl (visible fa)
, revVisible = revVisible fl (revVisible fa)
, lineNum = lineNum fl (lineNum fa)
, date = date fl (date fa)
, flag = flag fl (flag fa)
, number = number fl (number fa)
, postingDrCr = postingDrCr fl (postingDrCr fa)
, postingCmdty = postingCmdty fl (postingCmdty fa)
, postingQty = postingQty fl (postingQty fa)
, totalDrCr = totalDrCr fl (totalDrCr fa)
, totalCmdty = totalCmdty fl (totalCmdty fa)
, totalQty = totalQty fl (totalQty fa) }
pairWithSpacer :: Fields a -> S.Spacers b -> Fields (a, Maybe b)
pairWithSpacer f s = Fields {
globalTransaction = (globalTransaction f, Just (S.globalTransaction s))
, revGlobalTransaction = (revGlobalTransaction f, Just (S.revGlobalTransaction s))
, globalPosting = (globalPosting f, Just (S.globalPosting s))
, revGlobalPosting = (revGlobalPosting f, Just (S.revGlobalPosting s))
, fileTransaction = (fileTransaction f, Just (S.fileTransaction s))
, revFileTransaction = (revFileTransaction f, Just (S.revFileTransaction s))
, filePosting = (filePosting f, Just (S.filePosting s))
, revFilePosting = (revFilePosting f, Just (S.revFilePosting s))
, filtered = (filtered f, Just (S.filtered s))
, revFiltered = (revFiltered f, Just (S.revFiltered s))
, sorted = (sorted f, Just (S.sorted s))
, revSorted = (revSorted f, Just (S.revSorted s))
, visible = (visible f, Just (S.visible s))
, revVisible = (revVisible f, Just (S.revVisible s))
, lineNum = (lineNum f, Just (S.lineNum s))
, date = (date f, Just (S.date s))
, flag = (flag f, Just (S.flag s))
, number = (number f, Just (S.number s))
, postingDrCr = (postingDrCr f, Just (S.postingDrCr s))
, postingCmdty = (postingCmdty f, Just (S.postingCmdty s))
, postingQty = (postingQty f, Just (S.postingQty s))
, totalDrCr = (totalDrCr f, Just (S.totalDrCr s))
, totalCmdty = (totalCmdty f, Just (S.totalCmdty s))
, totalQty = (totalQty f, Nothing ) }
reduce :: Semi.Semigroup s => Fields s -> s
reduce f =
globalTransaction f
<> revGlobalTransaction f
<> globalPosting f
<> revGlobalPosting f
<> fileTransaction f
<> revFileTransaction f
<> filePosting f
<> revFilePosting f
<> filtered f
<> revFiltered f
<> sorted f
<> revSorted f
<> visible f
<> revVisible f
<> lineNum f
<> date f
<> flag f
<> number f
<> postingDrCr f
<> postingCmdty f
<> postingQty f
<> totalDrCr f
<> totalCmdty f
<> totalQty f
grownWidth ::
Fields (Maybe Int)
-> S.Spacers Int
-> Int
grownWidth fs ss =
Semi.getSum
. reduce
. fmap Semi.Sum
. fmap fieldWidth
$ pairWithSpacer fs ss
fieldWidth :: (Maybe Int, Maybe Int) -> Int
fieldWidth (m1, m2) = case m1 of
Nothing -> 0
Just i1 -> case m2 of
Just i2 -> if i2 > 0 then i1 + i2 else i1
Nothing -> i1