mirror of
https://github.com/Textualize/textual.git
synced 2025-10-17 02:38:12 +03:00
Until now the Tree.NodeExpanded and Tree.NodeCollapsed messages were only sent out when changes were made to the tree by user interaction. This meant that if any changes were made with the TreeNode expand, expand_all, collapse, collapse_all, toggle or toggle_all API calls no messages would be sent. This PR corrects this. The work here is, in part, required for #2456 (DirectoryTree lazy-loads directory information on node expansion so if someone is expanding nodes under code control the DirectoryTree never gets to know that it should load a directory's content) and will build on #1644, essentially adding a missing aspect to the latter PR.
156 lines
5.5 KiB
Python
156 lines
5.5 KiB
Python
from __future__ import annotations
|
|
|
|
from typing import Any
|
|
|
|
from textual.app import App, ComposeResult
|
|
from textual.message import Message
|
|
from textual.widgets import Tree
|
|
|
|
|
|
class MyTree(Tree[None]):
|
|
pass
|
|
|
|
|
|
class TreeApp(App[None]):
|
|
"""Test tree app."""
|
|
|
|
def __init__(self, *args: Any, **kwargs: Any) -> None:
|
|
super().__init__(*args, **kwargs)
|
|
self.messages: list[tuple[str, str]] = []
|
|
|
|
def compose(self) -> ComposeResult:
|
|
"""Compose the child widgets."""
|
|
yield MyTree("Root", id="test-tree")
|
|
|
|
def on_mount(self) -> None:
|
|
self.query_one(MyTree).root.add("Child")
|
|
self.query_one(MyTree).focus()
|
|
|
|
def record(
|
|
self,
|
|
event: Tree.NodeSelected[None]
|
|
| Tree.NodeExpanded[None]
|
|
| Tree.NodeCollapsed[None]
|
|
| Tree.NodeHighlighted[None],
|
|
) -> None:
|
|
self.messages.append(
|
|
(event.__class__.__name__, event.node.tree.id or "Unknown")
|
|
)
|
|
|
|
def on_tree_node_selected(self, event: Tree.NodeSelected[None]) -> None:
|
|
self.record(event)
|
|
|
|
def on_tree_node_expanded(self, event: Tree.NodeExpanded[None]) -> None:
|
|
self.record(event)
|
|
|
|
def on_tree_node_collapsed(self, event: Tree.NodeCollapsed[None]) -> None:
|
|
self.record(event)
|
|
|
|
def on_tree_node_highlighted(self, event: Tree.NodeHighlighted[None]) -> None:
|
|
self.record(event)
|
|
|
|
|
|
async def test_tree_node_selected_message() -> None:
|
|
"""Selecting a node should result in a selected message being emitted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
await pilot.press("enter")
|
|
await pilot.pause()
|
|
assert pilot.app.messages == [
|
|
("NodeExpanded", "test-tree"),
|
|
("NodeSelected", "test-tree"),
|
|
]
|
|
|
|
|
|
async def test_tree_node_selected_message_no_auto() -> None:
|
|
"""Selecting a node should result in only a selected message being emitted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
pilot.app.query_one(MyTree).auto_expand = False
|
|
await pilot.press("enter")
|
|
await pilot.pause()
|
|
assert pilot.app.messages == [("NodeSelected", "test-tree")]
|
|
|
|
|
|
async def test_tree_node_expanded_message() -> None:
|
|
"""Expanding a node should result in an expanded message being emitted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
await pilot.press("space")
|
|
await pilot.pause()
|
|
assert pilot.app.messages == [("NodeExpanded", "test-tree")]
|
|
|
|
|
|
async def tree_node_expanded_by_code_message() -> None:
|
|
"""Expanding a node via the API should result in an expanded message being posted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
pilot.app.query_one(Tree).root.children[0].expand()
|
|
assert pilot.app.messages == [("NodeExpanded", "test-tree")]
|
|
|
|
|
|
async def tree_node_all_expanded_by_code_message() -> None:
|
|
"""Expanding all nodes via the API should result in expanded messages being posted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
pilot.app.query_one(Tree).root.children[0].expand_all()
|
|
assert pilot.app.messages == [("NodeExpanded", "test-tree")]
|
|
|
|
|
|
async def test_tree_node_collapsed_message() -> None:
|
|
"""Collapsing a node should result in a collapsed message being emitted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
await pilot.press("space", "space")
|
|
await pilot.pause()
|
|
assert pilot.app.messages == [
|
|
("NodeExpanded", "test-tree"),
|
|
("NodeCollapsed", "test-tree"),
|
|
]
|
|
|
|
|
|
async def tree_node_collapsed_by_code_message() -> None:
|
|
"""Collapsing a node via the API should result in a collapsed message being posted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
pilot.app.query_one(Tree).root.children[0].expand().collapse()
|
|
assert pilot.app.messages == [
|
|
("NodeExpanded", "test-tree"),
|
|
("NodeCollapsed", "test-tree"),
|
|
]
|
|
|
|
|
|
async def tree_node_all_collapsed_by_code_message() -> None:
|
|
"""Collapsing all nodes via the API should result in collapsed messages being posted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
pilot.app.query_one(Tree).root.children[0].expand_all().collapse_all()
|
|
assert pilot.app.messages == [
|
|
("NodeExpanded", "test-tree"),
|
|
("NodeCollapsed", "test-tree"),
|
|
]
|
|
|
|
|
|
async def tree_node_toggled_by_code_message() -> None:
|
|
"""Toggling a node twice via the API should result in expanded and collapsed messages."""
|
|
async with TreeApp().run_test() as pilot:
|
|
pilot.app.query_one(Tree).root.children[0].toggle().toggle()
|
|
assert pilot.app.messages == [
|
|
("NodeExpanded", "test-tree"),
|
|
("NodeCollapsed", "test-tree"),
|
|
]
|
|
|
|
|
|
async def tree_node_all_toggled_by_code_message() -> None:
|
|
"""Toggling all nodes twice via the API should result in expanded and collapsed messages."""
|
|
async with TreeApp().run_test() as pilot:
|
|
pilot.app.query_one(Tree).root.children[0].toggle_all().toggle_all()
|
|
assert pilot.app.messages == [
|
|
("NodeExpanded", "test-tree"),
|
|
("NodeCollapsed", "test-tree"),
|
|
]
|
|
|
|
|
|
async def test_tree_node_highlighted_message() -> None:
|
|
"""Highlighting a node should result in a highlighted message being emitted."""
|
|
async with TreeApp().run_test() as pilot:
|
|
await pilot.press("enter", "down")
|
|
await pilot.pause()
|
|
assert pilot.app.messages == [
|
|
("NodeExpanded", "test-tree"),
|
|
("NodeSelected", "test-tree"),
|
|
("NodeHighlighted", "test-tree"),
|
|
]
|