File: trees.cpp

package info (click to toggle)
iqtree 2.0.7%2Bdfsg-1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, forky, sid, trixie
  • size: 14,620 kB
  • sloc: cpp: 142,571; ansic: 57,789; sh: 275; python: 242; makefile: 95
file content (93 lines) | stat: -rw-r--r-- 3,090 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
90
91
92
93
#include "trees_impl.hpp"
#include "utils.hpp"
#include <algorithm>
#include <ostream>

namespace terraces {

void check_rooted_tree(const tree& t) {
	// check edge cases
	utils::ensure<std::invalid_argument>(t.size() != 0, "tree is empty");
	utils::ensure<std::invalid_argument>((t.size() != 1 || (is_leaf(t[0]) && is_root(t[0]))),
	                                     "invalid trivial tree");

	// check if parent/child assignment is symmetric
	for (index i = 0; i < t.size(); ++i) {
		auto n = t[i];
		if (is_leaf(n)) {
			// leaf l: child of parent is correct
			auto p = n.parent();
			utils::ensure<std::invalid_argument>(p < t.size(), "parent overflow");
			utils::ensure<std::invalid_argument>(
			        (t[p].lchild() == i || t[p].rchild() == i),
			        "leaf's parent doesn't point to leaf");
		} else {
			// inner n: parent of lc and rc is correct
			auto lc = n.lchild();
			auto rc = n.rchild();
			utils::ensure<std::invalid_argument>(lc < t.size(), "lchild overflow");
			utils::ensure<std::invalid_argument>(rc < t.size(), "rchild overflow");
			utils::ensure<std::invalid_argument>(t[lc].parent() == i &&
			                                             t[rc].parent() == i,
			                                     "nodes children don't point to node");
			utils::ensure<std::invalid_argument>(lc != rc, "lchild == rchild");
		}
	}
	utils::ensure<std::invalid_argument>(is_root(t[0]), "first node is not the root");
}

std::ostream& operator<<(std::ostream& s, newick_t tree_pair) {
	const auto& t = *tree_pair.t;
	const auto& names = *tree_pair.names;
	auto pre_cb = [&](index) { s << '('; };
	auto post_cb = [&](index) { s << ')'; };
	auto leaf_cb = [&](index i) {
		if (t[i].taxon() != none)
			s << names[t[i].taxon()];
	};
	auto sibling_cb = [&](index) { s << ','; };
	index root = 0;
	tree_traversal(t, pre_cb, post_cb, sibling_cb, leaf_cb, root);
	s << ';';
	return s;
}

std::vector<index> preorder(const tree& t) {
	std::vector<index> result;
	foreach_preorder(t, [&](index i) { result.push_back(i); });
	return result;
}

std::vector<index> postorder(const tree& t) {
	std::vector<index> result;
	foreach_postorder(t, [&](index i) { result.push_back(i); });
	return result;
}

void print_tree_dot(const tree& t, const name_map& n, std::ostream& stream, bool rooted) {
	auto nop = [](index) {};
	std::string edge = rooted ? " -> " : " -- ";
	auto node_cb = [&](index node) {
		stream << node << " [shape=point];\n";
		stream << t[node].lchild() << edge << node << ";\n";
		stream << t[node].rchild() << edge << node << ";\n";
	};
	auto leaf_cb = [&](index node) {
		stream << node << " [label=\"" << n[t[node].taxon()] << "\"];\n";
	};
	stream << (rooted ? "digraph {\n" : "graph {\n");
	if (rooted) {
		tree_traversal(t, node_cb, nop, nop, leaf_cb, 0);
	} else {
		if (is_leaf(t[0])) {
			leaf_cb(0);
		} else {
			tree_traversal(t, node_cb, nop, nop, leaf_cb, t[0].lchild());
			tree_traversal(t, node_cb, nop, nop, leaf_cb, t[0].rchild());
			stream << t[0].lchild() << edge << t[0].rchild() << ";\n";
		}
	}
	stream << "}\n";
}

} // namespace terraces