File: Tree.hs

package info (click to toggle)
bali-phy 4.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 15,392 kB
  • sloc: cpp: 120,442; xml: 13,966; haskell: 9,975; python: 2,936; yacc: 1,328; perl: 1,169; lex: 912; sh: 343; makefile: 26
file content (89 lines) | stat: -rw-r--r-- 3,040 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
module Tree (module Tree, module Forest, module Graph) where

import Forest
import Graph
import Data.Foldable
import Data.Array
import Data.List (lookup)

import Data.IntMap (IntMap)
import qualified Data.IntMap as IntMap
import Data.IntSet (IntSet)
import qualified Data.IntSet as IntSet
import Data.Text (Text)
import qualified Data.Text as T
import Control.DeepSeq

-- NOTE: Data.Tree (Rose trees) and Data.Forest (collections of Data.Tree) exist, but are records, not classes.

class IsForest t => IsTree t

class (HasRoots t, IsTree t) => HasRoot t where
    root :: t -> NodeId
    setRoot :: NodeId -> t -> t
    setRoot r t = setRoots [r] t

instance (HasRoots t, IsTree t) => HasRoot t where
    root tree = case roots tree of [] -> error "root: Tree has no roots!"
                                   [r] -> r
                                   _ -> error "root: Tree has multiple roots!"

-- OK, so should we store attributes inside the tree?
-- 

data Tree l  = Tree (Forest l)

instance NFData (Forest l) => NFData (Tree l) where
    rnf (Tree forest) = rnf forest

instance IsGraph (Tree l) where
    getNodesSet (Tree f) = getNodesSet f
    getEdgesSet (Tree f) = getEdgesSet f

    edgesOutOfNodeSet (Tree f) nodeId = edgesOutOfNodeSet f nodeId
    sourceNode (Tree f) edge = sourceNode f edge
    targetNode (Tree f) edge = targetNode f edge

    getNodeAttributes (Tree f) node = getNodeAttributes f node
    getEdgeAttributes (Tree f) edge = getEdgeAttributes f edge
    getAttributes (Tree f) = getAttributes f

    setNodeAttributes (Tree f) as = Tree (setNodeAttributes f as)
    setEdgeAttributes (Tree f) as = Tree (setEdgeAttributes f as)
    setAttributes (Tree f) as = Tree (setAttributes f as)

    type instance LabelType (Tree l) = l
    type instance NewLabelType (Tree l) a = Tree a
    getLabel (Tree f) node = getLabel f node
    getLabels (Tree f) = getLabels f
    relabel newLabels (Tree f) = Tree (relabel newLabels f)

instance IsForest (Tree l) where
    type instance Rooted (Tree l) = WithRoots (Tree l)

    makeRooted t = addRoot root t where root = head $ (internalNodes t ++ leafNodes t)
    isRooted (Tree f) = Unrooted

instance IsTree (Tree l)

instance IsTree t => IsTree (WithRoots t)

instance IsTree t => IsTree (WithBranchLengths t)

instance IsTree t => IsTree (WithNodeTimes t)

instance IsTree t => IsTree (WithBranchRates t)

treeFromEdges nodes edges = Tree $ forestFromEdges nodes edges

allEdgesFromRoot tree = concatMap (allEdgesAfterEdge tree) (edgesOutOfNode tree (root tree))

-- addRoot :: IsTree t => NodeId -> t -> Rooted t
addRoot r t = addRoots [r] t

-- Should this go somewhere else?
weightedAverage weights values | length weights == length values = go weights values 0 0
                               | otherwise                       = error $ "weightedAverage: |weights| = " ++ show (length weights) ++ "  |values| = " ++ show (length values)
    where
      go (w:ws) (v:vs) top bot = go ws vs (top + (w * v)) (bot + w)
      go []     []     top bot = top / bot