module ProjectM36.Relation.Show.Term where
import ProjectM36.Base
import ProjectM36.Atom
import ProjectM36.AtomType
import ProjectM36.Tuple
import ProjectM36.Relation
import ProjectM36.Attribute
import qualified Data.List as L
import qualified Data.Text as T
import qualified Data.Vector as V
boxV :: StringType
boxV = "│"
boxH :: StringType
boxH = "─"
boxTL :: StringType
boxTL = "┌"
boxTR :: StringType
boxTR = "┐"
boxBL :: StringType
boxBL = "└"
boxBR :: StringType
boxBR = "┘"
boxLB :: StringType
boxLB = "├"
boxRB :: StringType
boxRB = "┤"
boxTB :: StringType
boxTB = "┬"
boxBB :: StringType
boxBB = "┴"
boxC :: StringType
boxC = "┼"
dboxH :: StringType
dboxH = "═"
dboxL :: StringType
dboxL = "╞"
dboxR :: StringType
dboxR = "╡"
class TermSize a where
termLength :: a -> Int
type Cell = StringType
type Table = ([Cell], [[Cell]])
addRow :: [Cell] -> Table -> Table
addRow cells (header,body) = (header, body ++ [cells])
cellLocations :: Table -> ([Int],[Int])
cellLocations tab@(header, _) = (maxWidths, maxHeights)
where
cellSizeMatrix = cellSizes tab
maxWidths = foldl mergeMax (baseSize (length header)) (map fst cellSizeMatrix)
baseSize num = take num (repeat 0)
rowHeights = map snd cellSizeMatrix
maxHeights = map (\l -> if length l == 0 then 0 else L.maximumBy compare l) rowHeights
mergeMax a b = map (\(c,d) -> max c d) (zip a b)
breakLines :: StringType -> [StringType]
breakLines "" = [""]
breakLines x = T.lines x
cellSizes :: Table -> [([Int], [Int])]
cellSizes (header, body) = map (\row -> (map maxRowWidth row, map (length . breakLines) row)) allRows
where
maxRowWidth row = if length (lengths row) == 0 then
0
else
L.maximumBy compare (lengths row)
lengths row = map T.length (breakLines row)
allRows = [header] ++ body
relationAsTable :: Relation -> Table
relationAsTable rel@(Relation _ tupleSet) = (header, body)
where
oAttrs = orderedAttributes rel
oAttrNames = orderedAttributeNames rel
header = map prettyAttribute oAttrs
body :: [[Cell]]
body = L.foldl' tupleFolder [] (asList tupleSet)
tupleFolder acc tuple = acc ++ [map (\attrName -> case atomForAttributeName attrName tuple of
Left _ -> "?"
Right atom -> showAtom 0 atom
) oAttrNames]
showParens :: Bool -> StringType -> StringType
showParens predicate f = if predicate then
"(" `T.append` f `T.append` ")"
else
f
showAtom :: Int -> Atom -> StringType
showAtom _ (RelationAtom rel) = renderTable $ relationAsTable rel
showAtom level (ConstructedAtom dConsName _ atoms) = showParens (level >= 1 && length atoms >= 1) $ T.concat (L.intersperse " " (dConsName : (map (showAtom 1) atoms)))
showAtom _ atom = atomToText atom
renderTable :: Table -> StringType
renderTable table = renderHeader table (fst cellLocs) `T.append` renderBody (snd table) cellLocs
where
cellLocs = cellLocations table
renderHeader :: Table -> [Int] -> StringType
renderHeader (header, body) columnLocations = renderTopBar `T.append` renderHeaderNames `T.append` renderBottomBar
where
renderTopBar = boxTL `T.append` T.concat (L.intersperse boxTB (map (\x -> repeatString x boxH) columnLocations)) `T.append` boxTR `T.append` "\n"
renderHeaderNames = renderRow header columnLocations 1 boxV
renderBottomBar = if length body == 0 then ""
else renderHBar boxLB boxC boxRB columnLocations `T.append` "\n"
renderHBar :: StringType -> StringType -> StringType -> [Int] -> StringType
renderHBar left middle end columnLocations = left `T.append` T.concat (L.intersperse middle (map (\x -> repeatString x boxH) columnLocations)) `T.append` end
leftPaddedString :: Int -> Int -> StringType -> StringType
leftPaddedString lineNum size str = if lineNum > length paddedLines 1 then
repeatString size " "
else
paddedLines !! lineNum
where
paddedLines = map (\line -> line `T.append` repeatString (size T.length line) " ") (breakLines str)
renderRow :: [Cell] -> [Int] -> Int -> StringType -> StringType
renderRow cells columnLocations rowHeight interspersed = T.unlines $ map renderOneLine [0..rowHeight1]
where
renderOneLine lineNum = boxV `T.append` T.concat (L.intersperse interspersed (map (\(size, value) -> leftPaddedString lineNum size value) (zip columnLocations cells))) `T.append` boxV
renderBody :: [[Cell]] -> ([Int],[Int]) -> StringType
renderBody cellMatrix cellLocs = renderRows `T.append` renderBottomBar
where
columnLocations = fst cellLocs
rowLocations = snd cellLocs
renderRows = T.concat (map (\(row, rowHeight)-> renderRow row columnLocations rowHeight boxV) rowHeightMatrix)
rowHeightMatrix = zip cellMatrix (tail rowLocations)
renderBottomBar = renderHBar boxBL boxBB boxBR columnLocations
orderedAttributes :: Relation -> [Attribute]
orderedAttributes rel = L.sortBy (\a b -> attributeName a `compare` attributeName b) (V.toList (attributes rel))
orderedAttributeNames :: Relation -> [AttributeName]
orderedAttributeNames rel = map attributeName (orderedAttributes rel)
repeatString :: Int -> StringType -> StringType
repeatString c s = T.concat (take c (repeat s))
showRelation :: Relation -> StringType
showRelation rel = renderTable (relationAsTable rel)