1
0
Fork 0
mirror of https://code.naskya.net/repos/ndqEd synced 2025-01-10 20:36:47 +09:00

Generelize DAG tree view node from Int to any Eq+Ord+Hashable

This commit is contained in:
fr33domlover 2016-08-03 23:25:35 +00:00
parent e7abd8eb9d
commit 3807a02daf

View file

@ -24,6 +24,7 @@ import Prelude
import Control.Arrow ((***)) import Control.Arrow ((***))
import Data.Function (on) import Data.Function (on)
import Data.Hashable (Hashable)
import Data.HashMap.Lazy (HashMap) import Data.HashMap.Lazy (HashMap)
import Data.HashSet (HashSet) import Data.HashSet (HashSet)
import Data.List (groupBy, sortOn) import Data.List (groupBy, sortOn)
@ -32,17 +33,18 @@ import Data.Monoid (Endo (..))
import qualified Data.HashMap.Lazy as M import qualified Data.HashMap.Lazy as M
import qualified Data.HashSet as S import qualified Data.HashSet as S
type Graph a b = HashMap Int (a, [(Int, b)]) type Graph n a b = HashMap n (a, [(n, b)])
data DagViewTree a b = FullNode a [DagViewTree a b] | LinkNode b data DagViewTree a b = FullNode a [DagViewTree a b] | LinkNode b
-- | Update the map according to a choice of a full parent for a given child. -- | Update the map according to a choice of a full parent for a given child.
-- Also specifies whether the choice was sucessfully applied. -- Also specifies whether the choice was sucessfully applied.
chooseParent chooseParent
:: Int :: (Eq n, Hashable n)
-> Int => n
-> HashMap Int [(Int, Bool)] -> n
-> Maybe (HashMap Int [(Int, Bool)]) -> HashMap n [(n, Bool)]
-> Maybe (HashMap n [(n, Bool)])
chooseParent c p h = chooseParent c p h =
case M.lookup c h of case M.lookup c h of
Nothing -> Nothing Nothing -> Nothing
@ -70,28 +72,30 @@ markFst :: [a] -> [(a, Bool)]
markFst [] = [] markFst [] = []
markFst (x:xs) = (x, True) : map (, False) xs markFst (x:xs) = (x, True) : map (, False) xs
labeledDeps :: HashMap Int [(Int, b)] -> [(Int, Int, b)] labeledDeps :: Hashable n => HashMap n [(n, b)] -> [(n, n, b)]
labeledDeps = labeledDeps =
let mk c (p, full) = (c, p, full) let mk c (p, full) = (c, p, full)
in concatMap (\ (c, ps) -> map (mk c) ps) . M.toList in concatMap (\ (c, ps) -> map (mk c) ps) . M.toList
edgeView edgeView
:: HashMap Int Int :: (Eq n, Hashable n)
=> HashMap n n
-- ^ Full parent user choices -- ^ Full parent user choices
-> (Int, Int, Bool) -> (n, n, Bool)
-- ^ Child, parent, and whether the parent is full -- ^ Child, parent, and whether the parent is full
-> Maybe (HashMap Int Int) -> Maybe (HashMap n n)
-- ^ New edge label. For a full edge, 'Nothing'. For a link edge, 'Just' an -- ^ New edge label. For a full edge, 'Nothing'. For a link edge, 'Just' an
-- updated choice map that chooses this edge as the new full edge for the -- updated choice map that chooses this edge as the new full edge for the
-- child. -- child.
edgeView _ (_, _, False) = Nothing edgeView _ (_, _, False) = Nothing
edgeView choices (child, parent, True) = Just $ M.insert child parent choices edgeView choices (child, parent, True) = Just $ M.insert child parent choices
reverseEdge :: (Int, Int, a) -> (Int, Int, a) reverseEdge :: (n, n, a) -> (n, n, a)
reverseEdge (x, y, l) = (y, x, l) reverseEdge (x, y, l) = (y, x, l)
-- | Given labeled nodes and labeled edges, prepare a hashmap. -- | Given labeled nodes and labeled edges, prepare a hashmap.
mkGraph :: HashMap Int a -> [(Int, Int, b)] -> Graph a b mkGraph
:: (Eq n, Ord n, Hashable n) => HashMap n a -> [(n, n, b)] -> Graph n a b
mkGraph nodeMap edges = mkGraph nodeMap edges =
let pair23 (x, y, z) = (x, (y, z)) let pair23 (x, y, z) = (x, (y, z))
edgeMap = M.fromList $ groupSnd $ sortOn fst $ map pair23 edges edgeMap = M.fromList $ groupSnd $ sortOn fst $ map pair23 edges
@ -102,7 +106,11 @@ mkGraph nodeMap edges =
keySet :: HashMap k v -> HashSet k keySet :: HashMap k v -> HashSet k
keySet = S.fromMap . M.map (const ()) keySet = S.fromMap . M.map (const ())
buildTree :: [(Int, Maybe b)] -> Graph a (Maybe b) -> [DagViewTree a (a, b)] buildTree
:: (Eq n, Hashable n)
=> [(n, Maybe b)]
-> Graph n a (Maybe b)
-> [DagViewTree a (a, b)]
buildTree nodes graph = go nodes buildTree nodes graph = go nodes
where where
go [] = [] go [] = []
@ -120,15 +128,16 @@ buildTree nodes graph = go nodes
in LinkNode (fst c, info) : ts in LinkNode (fst c, info) : ts
dagViewTree dagViewTree
:: [(Int, a)] :: (Eq n, Ord n, Hashable n)
=> [(n, a)]
-- ^ Nodes: Numbers and details -- ^ Nodes: Numbers and details
-> [(Int, Int)] -> [(n, n)]
-- ^ Edges: Child-parent pairs -- ^ Edges: Child-parent pairs
-> [(Int, Int)] -> [(n, n)]
-- ^ Full parent choices as child-parent pairs. This is whatever user input -- ^ Full parent choices as child-parent pairs. This is whatever user input
-- has been received, even if it includes duplicates or nonexistent node -- has been received, even if it includes duplicates or nonexistent node
-- numbers. So just pass the user input directly here. -- numbers. So just pass the user input directly here.
-> [DagViewTree a (a, HashMap Int Int)] -> [DagViewTree a (a, HashMap n n)]
dagViewTree nodes deps choices = dagViewTree nodes deps choices =
let choose ns@(c, p) acc@(h, l) = let choose ns@(c, p) acc@(h, l) =
case chooseParent c p h of case chooseParent c p h of