Skip to content

Commit

Permalink
Fix SVG pack_untracked_polytomies bug
Browse files Browse the repository at this point in the history
Fixes #3049
  • Loading branch information
hyanwong committed Nov 6, 2024
1 parent 73ef4cc commit b07bee5
Show file tree
Hide file tree
Showing 3 changed files with 35 additions and 13 deletions.
6 changes: 6 additions & 0 deletions python/CHANGELOG.rst
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,12 @@
[0.6.1] - 2024-XX-XX
--------------------

**Bugfixes**

- Fix to ``Tree.draw_svg()`` with the (undocumented) experimental
``pack_untracked_polytomies=True`` option. (:user:`hyanwong`,
:issue:`3049`, :pr:`3050`)


--------------------
[0.6.0] - 2024-10-16
Expand Down
13 changes: 13 additions & 0 deletions python/tests/test_drawing.py
Original file line number Diff line number Diff line change
Expand Up @@ -2602,6 +2602,19 @@ def test_only_subset_nodes_in_rank(self, caplog):
)
assert "Ticks {10: '10'} lie outside the plotted axis" in caplog.text

def test_polytomy_collapsing(self):
tree = tskit.Tree.generate_balanced(
20, arity=4, tracked_samples=np.arange(2, 8)
)
svg = tree.draw_svg(pack_untracked_polytomies=True)
# Should have one collapsed node (untracked samples 8 and 9)
# and two "polytomy lines" (from nodes 21 and 28 (the root))
assert svg.count('class="polytomy"') == 2 # poolytomy lines
collapsed_symbol = re.search("<polygon[^>]*>", svg)
assert collapsed_symbol is not None
assert collapsed_symbol.group(0).count("sym") == 1
assert collapsed_symbol.group(0).count("multi") == 1


class TestDrawKnownSvg(TestDrawSvgBase):
"""
Expand Down
29 changes: 16 additions & 13 deletions python/tskit/drawing.py
Original file line number Diff line number Diff line change
Expand Up @@ -1827,33 +1827,36 @@ def assign_x_coordinates(self):
prev = tree.virtual_root
for u in self.postorder_nodes:
parent = tree.parent(u)
omit = self.pack_untracked_polytomies and tree.num_tracked_samples(u) == 0
if parent == prev:
raise ValueError("Nodes must be passed in postorder to Tree.draw_svg()")
is_tip = tree.parent(prev) != u
if is_tip:
if self.pack_untracked_polytomies and tree.num_tracked_samples(u) == 0:
untracked_children[parent].append(u)
else:
if not omit:
leaf_x += 1
node_xpos[u] = leaf_x
else:
# Concatenate all the untracked children
num_untracked_children = len(untracked_children[u])
elif not omit:
# Untracked children are available for packing into a polytomy summary
untracked_children = []
if self.pack_untracked_polytomies:
untracked_children += [
c for c in tree.children(u) if tree.num_tracked_samples(c) == 0
]
child_x = [node_xpos[c] for c in tree.children(u) if c in node_xpos]
if num_untracked_children > 0:
if num_untracked_children <= 1:
# If only a single non-focal lineage, we might as well show it
for child in untracked_children[u]:
if len(untracked_children) > 0:
if len(untracked_children) <= 1:
# If only a single non-focal lineage, treat it as a condensed tip
for child in untracked_children:
leaf_x += 1
node_xpos[child] = leaf_x
child_x.append(leaf_x)
else:
# Otherwise show a horizontal line with the number of lineages
# Extra length of line is equal to log of the polytomy size
self.extra_line[u] = self.PolytomyLine(
num_untracked_children,
sum(tree.num_samples(v) for v in untracked_children[u]),
[leaf_x, leaf_x + 1 + np.log(num_untracked_children)],
len(untracked_children),
sum(tree.num_samples(v) for v in untracked_children),
[leaf_x, leaf_x + 1 + np.log(len(untracked_children))],
)
child_x.append(leaf_x + 1)
leaf_x = self.extra_line[u].line_pos[1]
Expand Down

0 comments on commit b07bee5

Please sign in to comment.