93 lines
3 KiB
Haskell
93 lines
3 KiB
Haskell
|
{- This file is part of Vervis.
|
||
|
-
|
||
|
- Written in 2016 by fr33domlover <fr33domlover@riseup.net>.
|
||
|
-
|
||
|
- ♡ Copying is an act of love. Please copy, reuse and share.
|
||
|
-
|
||
|
- The author(s) have dedicated all copyright and related and neighboring
|
||
|
- rights to this software to the public domain worldwide. This software is
|
||
|
- distributed without any warranty.
|
||
|
-
|
||
|
- You should have received a copy of the CC0 Public Domain Dedication along
|
||
|
- with this software. If not, see
|
||
|
- <http://creativecommons.org/publicdomain/zero/1.0/>.
|
||
|
-}
|
||
|
|
||
|
-- | Layer an intransitive DAG and build a diagram of it. The layering
|
||
|
-- algorithm currently used here is trivial, and doesn't try to minimize
|
||
|
-- crossing edges. This will hopefully change in the future.
|
||
|
module Diagrams.IntransitiveDAG
|
||
|
( intransDag
|
||
|
)
|
||
|
where
|
||
|
|
||
|
import Prelude
|
||
|
|
||
|
import Data.Graph.Inductive.Graph
|
||
|
import Data.Graph.Inductive.Query.Layer (rlayerWith)
|
||
|
import Data.HashMap.Lazy (HashMap)
|
||
|
import Data.IntMap.Lazy (IntMap)
|
||
|
import Data.Tuple.Select (sel2)
|
||
|
import Diagrams.Combinators (atop)
|
||
|
import Diagrams.TwoD.Attributes (fc, lc)
|
||
|
import Diagrams.TwoD.Combinators (hcat, vcat)
|
||
|
import Diagrams.TwoD.Path (stroke)
|
||
|
import Diagrams.TwoD.Shapes (roundedRect)
|
||
|
import Diagrams.TwoD.Size (width)
|
||
|
import Diagrams.Util ((#), with)
|
||
|
import Graphics.SVGFonts.Fonts (lin2)
|
||
|
import Graphics.SVGFonts.Text (textSVG', TextOpts (..))
|
||
|
|
||
|
import qualified Data.HashMap.Lazy as H
|
||
|
import qualified Data.IntMap.Lazy as I
|
||
|
|
||
|
import Vervis.Colour
|
||
|
|
||
|
-- TODO how do I connect the layers?
|
||
|
--
|
||
|
-- Here's a suggestion. I can use rlayerWith to specify the result, and then in
|
||
|
-- addition to the Node, also return its out-edges. Now what remains is to
|
||
|
-- efficiently determine in which layer that node lives. That can be done by
|
||
|
-- keeping the node-to-layer map I built. But that may make the *layer*
|
||
|
-- function type sigs uglier, so instead I could also avoid relying on the
|
||
|
-- existence of such a map, and build it externally from the list of layers.
|
||
|
-- Yeah, sounds good to me.
|
||
|
--
|
||
|
-- However in order to find the layer quickly, it may be a good idea to put the
|
||
|
-- layer lists into an IntMap or HashMap for fast queries.
|
||
|
|
||
|
attachNumbers :: [a] -> IntMap a
|
||
|
attachNumbers = I.fromList . zip [1..]
|
||
|
|
||
|
nodeToLayerMap :: (a -> Node) -> IntMap [a] -> HashMap Node Int
|
||
|
nodeToLayerMap f =
|
||
|
H.fromList . concatMap (\ (l, xs) -> zip (map f xs) (repeat l)) . I.toList
|
||
|
|
||
|
layers :: Graph g => g a b -> [[(Node, a, [Node])]]
|
||
|
layers = rlayerWith $ \ c -> (node' c, lab' c, pre' c)
|
||
|
|
||
|
box w h =
|
||
|
let golden = 0.618 * h
|
||
|
w' = golden + w + golden
|
||
|
h' = golden + h + golden
|
||
|
r = golden
|
||
|
in roundedRect w' h' r
|
||
|
# fc black
|
||
|
# lc plain
|
||
|
|
||
|
roleBox s =
|
||
|
let h = 1
|
||
|
t = stroke $ textSVG' with {textFont = lin2, textHeight = h} s
|
||
|
t' = t # lc plain
|
||
|
w = width t'
|
||
|
b = box w h
|
||
|
in t' `atop` b
|
||
|
|
||
|
-- intransDag :: Graph g => g a b -> QDiagram
|
||
|
intransDag disp = --TODO connect the layers
|
||
|
vcat .
|
||
|
I.elems .
|
||
|
fmap (hcat . map (roleBox . disp . sel2)) .
|
||
|
attachNumbers .
|
||
|
layers
|