Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add read-only access to the children of a TreeNode #1495

Merged
merged 10 commits into from
Jan 6, 2023
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ and this project adheres to [Semantic Versioning](http://semver.org/).
### Added

- Added public `TreeNode` label access via `TreeNode.label` https://github.com/Textualize/textual/issues/1396
- Added read-only public access to the children of a `TreeNode` via `TreeNode.children` https://github.com/Textualize/textual/issues/1398

### Changed

Expand Down
65 changes: 65 additions & 0 deletions src/textual/_collections.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
"""Provides collection-based utility code."""

from __future__ import annotations
from typing import Generic, TypeVar, Iterator, overload, Iterable, Sequence

T = TypeVar("T")


class ImmutableSequence(Generic[T]):
"""Class to wrap a sequence of some sort, but not allow modification."""

def __init__(self, wrap: Iterable[T]) -> None:
"""Initialise the immutable sequence.

Args:
wrap (Iterable[T]): The iterable value being wrapped.
"""
self._wrap = wrap if isinstance(wrap, Sequence) else tuple(wrap)

@overload
def __getitem__(self, index: int) -> T:
...

@overload
def __getitem__(self, index: slice) -> ImmutableSequence[T]:
...

def __getitem__(self, index: int | slice) -> T | ImmutableSequence[T]:
return (
self._wrap[index]
if isinstance(index, int)
else ImmutableSequence[T](self._wrap[index])
)

def __iter__(self) -> Iterator[T]:
return iter(self._wrap)

def __len__(self) -> int:
return len(self._wrap)

def __length_hint__(self) -> int:
return len(self)

def __bool__(self) -> bool:
return bool(self._wrap)

def __contains__(self, item: T) -> bool:
return item in self._wrap

def index(self, item: T) -> int:
davep marked this conversation as resolved.
Show resolved Hide resolved
"""Return the index of the given item.

Args:
item (T): The item to find in the sequence.

Returns:
int: The index of the item in the sequence.

Raises:
ValueError: If the item is not in the sequence.
"""
return self._wrap.index(item)

def __reversed__(self) -> Iterator[T]:
return reversed(self._wrap)
12 changes: 11 additions & 1 deletion src/textual/widgets/_tree.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@
from .._segment_tools import line_crop, line_pad
from .._types import MessageTarget
from .._typing import TypeAlias
from .._collections import ImmutableSequence
from ..binding import Binding
from ..geometry import Region, Size, clamp
from ..message import Message
Expand Down Expand Up @@ -53,6 +54,10 @@ def _get_guide_width(self, guide_depth: int, show_root: bool) -> int:
return guides


class TreeNodes(ImmutableSequence["TreeNode[TreeDataType]"]):
"""An immutable collection of `TreeNode`."""


@rich.repr.auto
class TreeNode(Generic[TreeDataType]):
"""An object that represents a "node" in a tree control."""
Expand All @@ -74,7 +79,7 @@ def __init__(
self._label = tree.process_label(label)
self.data = data
self._expanded = expanded
self._children: list[TreeNode] = []
self._children: list[TreeNode[TreeDataType]] = []

self._hover_ = False
self._selected_ = False
Expand All @@ -91,6 +96,11 @@ def _reset(self) -> None:
self._selected_ = False
self._updates += 1

@property
def children(self) -> TreeNodes[TreeDataType]:
"""TreeNodes[TreeDataType]: The child nodes of a TreeNode."""
return TreeNodes(self._children)

@property
def line(self) -> int:
"""int: Get the line number for this node, or -1 if it is not displayed."""
Expand Down
82 changes: 82 additions & 0 deletions tests/test_collections.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
import pytest

from typing import Iterable
from textual._collections import ImmutableSequence

def wrap(source: Iterable[int]) -> ImmutableSequence[int]:
"""Wrap an itertable of integers inside an immutable sequence."""
return ImmutableSequence[int](source)


def test_empty_immutable_sequence() -> None:
"""An empty immutable sequence should act as anticipated."""
assert len(wrap([])) == 0
assert bool(wrap([])) is False
assert list(wrap([])) == []


def test_non_empty_immutable_sequence() -> None:
"""A non-empty immutable sequence should act as anticipated."""
assert len(wrap([0])) == 1
assert bool(wrap([0])) is True
assert list(wrap([0])) == [0]


def test_immutable_sequence_from_empty_iter() -> None:
"""An immutable sequence around an empty iterator should act as anticipated."""
assert len(wrap(iter([]))) == 0
assert bool(wrap(iter([]))) is False
assert list(wrap(iter([]))) == []


def test_immutable_sequence_from_non_empty_iter() -> None:
"""An immutable sequence around a non-empty iterator should act as anticipated."""
assert len(wrap(iter(range(23)))) == 23
assert bool(wrap(iter(range(23)))) is True
assert list(wrap(iter(range(23)))) == list(range(23))


def test_no_assign_to_immutable_sequence() -> None:
"""It should not be possible to assign into an immutable sequence."""
tester = wrap([1,2,3,4,5])
with pytest.raises(TypeError):
tester[0] = 23
with pytest.raises(TypeError):
tester[0:3] = 23


def test_no_del_from_iummutable_sequence() -> None:
"""It should not be possible delete an item from an immutable sequence."""
tester = wrap([1,2,3,4,5])
with pytest.raises(TypeError):
del tester[0]


def test_get_item_from_immutable_sequence() -> None:
"""It should be possible to get an item from an immutable sequence."""
assert wrap(range(10))[0] == 0
assert wrap(range(10))[-1] == 9


def test_get_slice_from_immutable_sequence() -> None:
"""It should be possible to get a slice from an immutable sequence."""
assert list(wrap(range(10))[0:2]) == [0,1]
assert list(wrap(range(10))[0:-1]) == [0,1,2,3,4,5,6,7,8]


def test_immutable_sequence_contains() -> None:
"""It should be possible to see if an immutable sequence contains a value."""
tester = wrap([1,2,3,4,5])
assert 1 in tester
assert 11 not in tester


def test_immutable_sequence_index() -> None:
tester = wrap([1,2,3,4,5])
assert tester.index(1) == 0
with pytest.raises(ValueError):
_ = tester.index(11)


def test_reverse_immutable_sequence() -> None:
assert list(reversed(wrap([1,2]))) == [2,1]
27 changes: 27 additions & 0 deletions tests/tree/test_tree_node_children.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import pytest
from textual.widgets import Tree, TreeNode

def label_of(node: TreeNode[None]):
"""Get the label of a node as a string"""
return str(node.label)


def test_tree_node_children() -> None:
"""A node's children property should act like an immutable list."""
CHILDREN=23
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Did you run Black on this?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not by hand on the tests, no. black is run by pre-commit of course so if there's a non-black-happy-ness in the above, that would suggest that both pre-commit and CI black checks avoid the unit tests?

Checking:

$ poetry run black --check .
would reformat devtools/__init__.py
would reformat snapshot_tests/snapshot_apps/horizontal_auto_width.py
would reformat snapshot_tests/snapshot_apps/multiple_css/multiple_css.py
would reformat renderables/test_sparkline.py
would reformat snapshot_tests/snapshot_apps/key_display.py
would reformat renderables/test_text_opacity.py
would reformat test_path.py
would reformat test_focus.py
would reformat test_immutable_sequence_view.py
would reformat test_node_list.py
would reformat renderables/test_underline_bar.py
would reformat test_arrange.py
would reformat test_unmount.py
would reformat test_table.py
would reformat test_dom.py
would reformat test_screens.py
would reformat tree/test_tree_node_label.py
would reformat tree/test_tree_node_children.py
would reformat test_text_backend.py
would reformat test_widget_child_moving.py
would reformat test_reactive.py
would reformat test_widget_mounting.py
would reformat test_widget_removing.py
would reformat test_geometry.py

Oh no! 💥 💔 💥
24 files would be reformatted, 66 files would be left unchanged.

Guess none of us are?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we may be excluding tests.

Can you check you are also running the latest black specified in dev dependencies.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

$ poetry run black --version
black, 22.10.0 (compiled: yes)
Python (CPython) 3.11.1

Which seems to be a later version that specified in poetry.lock?

[[package]]
name = "black"
version = "22.8.0"

Although weirdly:

black = "^22.3.0,<22.10.0"  # macos wheel issue on 22.10

from pyproject.toml

tree = Tree[None]("Root")
for child in range(CHILDREN):
tree.root.add(str(child))
assert len(tree.root.children)==CHILDREN
for child in range(CHILDREN):
assert label_of(tree.root.children[child]) == str(child)
assert label_of(tree.root.children[0]) == "0"
assert label_of(tree.root.children[-1]) == str(CHILDREN-1)
assert [label_of(node) for node in tree.root.children] == [str(n) for n in range(CHILDREN)]
assert [label_of(node) for node in tree.root.children[:2]] == [str(n) for n in range(2)]
with pytest.raises(TypeError):
tree.root.children[0] = tree.root.children[1]
with pytest.raises(TypeError):
del tree.root.children[0]
with pytest.raises(TypeError):
del tree.root.children[0:2]