textual/tests/test_query.py

213 lines
6.6 KiB
Python

import pytest
from textual.app import App, ComposeResult
from textual.containers import Container
from textual.css.query import InvalidQueryFormat, NoMatches, TooManyMatches, WrongType
from textual.widget import Widget
def test_query():
class View(Widget):
pass
class View2(View):
pass
class App(Widget):
pass
app = App()
main_view = View(id="main")
help_view = View2(id="help")
app._add_child(main_view)
app._add_child(help_view)
widget1 = Widget(id="widget1")
widget2 = Widget(id="widget2")
sidebar = Widget(id="sidebar")
sidebar.add_class("float")
helpbar = Widget(id="helpbar")
helpbar.add_class("float")
main_view._add_child(widget1)
main_view._add_child(widget2)
main_view._add_child(sidebar)
sub_view = View(id="sub")
sub_view.add_class("-subview")
main_view._add_child(sub_view)
tooltip = Widget(id="tooltip")
tooltip.add_class("float", "transient")
sub_view._add_child(tooltip)
help = Widget(id="markdown")
help_view._add_child(help)
help_view._add_child(helpbar)
# repeat tests to account for caching
for repeat in range(3):
assert list(app.query("Frob")) == []
assert list(app.query(".frob")) == []
assert list(app.query("#frob")) == []
assert not app.query("NotAnApp")
assert list(app.query("App")) == [] # Root is not included in queries
assert list(app.query("#main")) == [main_view]
assert list(app.query("View#main")) == [main_view]
assert list(app.query("View2#help")) == [help_view]
assert list(app.query("#widget1")) == [widget1]
assert list(app.query("#Widget1")) == [] # Note case.
assert list(app.query("#widget2")) == [widget2]
assert list(app.query("#Widget2")) == [] # Note case.
assert list(app.query("Widget.float")) == [sidebar, tooltip, helpbar]
assert list(app.query(Widget).filter(".float")) == [sidebar, tooltip, helpbar]
assert list(
app.query(Widget)
.exclude("App")
.exclude("#sub")
.exclude("#markdown")
.exclude("#main")
.exclude("#help")
.exclude("#widget1")
.exclude("#widget2")
) == [sidebar, tooltip, helpbar]
assert list(reversed(app.query("Widget.float"))) == [helpbar, tooltip, sidebar]
assert list(app.query("Widget.float").results(Widget)) == [
sidebar,
tooltip,
helpbar,
]
assert list(app.query("Widget.float").results()) == [
sidebar,
tooltip,
helpbar,
]
assert list(app.query("Widget.float").results(View)) == []
assert app.query_one("#widget1") == widget1
assert app.query_one("#widget1", Widget) == widget1
with pytest.raises(TooManyMatches):
_ = app.query_one(Widget)
assert app.query("Widget.float")[0] == sidebar
assert app.query("Widget.float")[0:2] == [sidebar, tooltip]
assert list(app.query("Widget.float.transient")) == [tooltip]
assert list(app.query("App > View")) == [main_view, help_view]
assert list(app.query("App > View#help")) == [help_view]
assert list(app.query("App > View#main .float ")) == [sidebar, tooltip]
assert list(app.query("View > View")) == [sub_view]
assert list(app.query("#help *")) == [help, helpbar]
assert list(app.query("#main *")) == [
widget1,
widget2,
sidebar,
sub_view,
tooltip,
]
assert list(app.query("View")) == [main_view, sub_view, help_view]
assert list(app.query("#widget1, #widget2")) == [widget1, widget2]
assert list(app.query("#widget1 , #widget2")) == [widget1, widget2]
assert list(app.query("#widget1, #widget2, App")) == [widget1, widget2]
assert app.query(".float").first() == sidebar
assert app.query(".float").last() == helpbar
with pytest.raises(NoMatches):
_ = app.query(".no_such_class").first()
with pytest.raises(NoMatches):
_ = app.query(".no_such_class").last()
with pytest.raises(WrongType):
_ = app.query(".float").first(View)
with pytest.raises(WrongType):
_ = app.query(".float").last(View)
def test_query_classes():
class App(Widget):
pass
class ClassTest(Widget):
pass
CHILD_COUNT = 100
# Create a fake app to hold everything else.
app = App()
# Now spin up a bunch of children.
for n in range(CHILD_COUNT):
app._add_child(ClassTest(id=f"child{n}"))
# Let's just be 100% sure everything was created fine.
assert len(app.query(ClassTest)) == CHILD_COUNT
# Now, let's check there are *no* children with the test class.
assert len(app.query(".test")) == 0
# Add the test class to everything and then check again.
app.query(ClassTest).add_class("test")
assert len(app.query(".test")) == CHILD_COUNT
# Remove the test class from everything then try again.
app.query(ClassTest).remove_class("test")
assert len(app.query(".test")) == 0
# Add the test class to everything using set_class.
app.query(ClassTest).set_class(True, "test")
assert len(app.query(".test")) == CHILD_COUNT
# Remove the test class from everything using set_class.
app.query(ClassTest).set_class(False, "test")
assert len(app.query(".test")) == 0
# Add the test class to everything using toggle_class.
app.query(ClassTest).toggle_class("test")
assert len(app.query(".test")) == CHILD_COUNT
# Remove the test class from everything using toggle_class.
app.query(ClassTest).toggle_class("test")
assert len(app.query(".test")) == 0
def test_invalid_query():
class App(Widget):
pass
app = App()
with pytest.raises(InvalidQueryFormat):
app.query("#3")
with pytest.raises(InvalidQueryFormat):
app.query("#foo").exclude("#2")
async def test_universal_selector_doesnt_select_self():
class ExampleApp(App):
def compose(self) -> ComposeResult:
yield Container(
Widget(
Widget(),
Widget(
Widget(),
),
),
Widget(),
id="root-container",
)
app = ExampleApp()
async with app.run_test():
container = app.query_one("#root-container", Container)
query = container.query("*")
results = list(query.results())
assert len(results) == 5
assert not any(node.id == "root-container" for node in results)