diff --git a/CHANGELOG.md b/CHANGELOG.md index 497ae2bc23..f7172fcff9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -52,6 +52,7 @@ and this project adheres to [Semantic Versioning](http://semver.org/). - When the terminal window loses focus, the currently-focused widget will also lose focus. - When the terminal window regains focus, the previously-focused widget will regain focus. - TextArea binding for ctrl+k will now delete the line if the line is empty https://github.com/Textualize/textual/issues/4277 +- The active tab (in `Tabs`) / tab pane (in `TabbedContent`) can now be unset https://github.com/Textualize/textual/issues/4241 ## [0.52.1] - 2024-02-20 diff --git a/docs/widgets/tabbed_content.md b/docs/widgets/tabbed_content.md index 15164e7907..e6edb899cc 100644 --- a/docs/widgets/tabbed_content.md +++ b/docs/widgets/tabbed_content.md @@ -127,6 +127,7 @@ For example, to create a `TabbedContent` that has red and green labels: ## Messages +- [TabbedContent.Cleared][textual.widgets.TabbedContent.Cleared] - [TabbedContent.TabActivated][textual.widgets.TabbedContent.TabActivated] ## Bindings diff --git a/src/textual/widgets/_tabbed_content.py b/src/textual/widgets/_tabbed_content.py index 61a2f29a4e..54b0c38799 100644 --- a/src/textual/widgets/_tabbed_content.py +++ b/src/textual/widgets/_tabbed_content.py @@ -276,7 +276,11 @@ def __rich_repr__(self) -> Result: yield self.pane class Cleared(Message): - """Posted when there are no more tab panes.""" + """Posted when no tab pane is active. + + This can happen if all tab panes are removed or if the currently active tab + pane is unset. + """ def __init__(self, tabbed_content: TabbedContent) -> None: """Initialize message. @@ -329,22 +333,6 @@ def active_pane(self) -> TabPane | None: return None return self.get_pane(self.active) - def validate_active(self, active: str) -> str: - """It doesn't make sense for `active` to be an empty string. - - Args: - active: Attribute to be validated. - - Returns: - Value of `active`. - - Raises: - ValueError: If the active attribute is set to empty string when there are tabs available. - """ - if not active and self.get_child_by_type(ContentSwitcher).current: - raise ValueError("'active' tab must not be empty string.") - return active - @staticmethod def _set_id(content: TabPane, new_id: int) -> TabPane: """Set an id on the content, if not already present. @@ -467,8 +455,6 @@ def remove_pane(self, pane_id: str) -> AwaitComplete: async def _remove_content() -> None: await gather(*removal_awaitables) - if self.tab_count == 0: - self.post_message(self.Cleared(self).set_sender(self)) return AwaitComplete(_remove_content()) @@ -486,7 +472,6 @@ def clear_panes(self) -> AwaitComplete: async def _clear_content() -> None: await await_clear - self.post_message(self.Cleared(self).set_sender(self)) return AwaitComplete(_clear_content()) @@ -547,7 +532,7 @@ def _is_associated_tabs(self, tabs: Tabs) -> bool: def _watch_active(self, active: str) -> None: """Switch tabs when the active attributes changes.""" - with self.prevent(Tabs.TabActivated): + with self.prevent(Tabs.TabActivated, Tabs.Cleared): self.get_child_by_type(ContentTabs).active = ContentTab.add_prefix(active) self.get_child_by_type(ContentSwitcher).current = active if active: @@ -557,6 +542,10 @@ def _watch_active(self, active: str) -> None: tab=self.get_child_by_type(ContentTabs).get_content_tab(active), ) ) + else: + self.post_message( + TabbedContent.Cleared(tabbed_content=self).set_sender(self) + ) @property def tab_count(self) -> int: diff --git a/src/textual/widgets/_tabs.py b/src/textual/widgets/_tabs.py index e46c972362..df2076e5d8 100644 --- a/src/textual/widgets/_tabs.py +++ b/src/textual/widgets/_tabs.py @@ -285,7 +285,8 @@ class TabShown(TabMessage): class Cleared(Message): """Sent when there are no active tabs. - This can occur when Tabs are cleared, or if all tabs are hidden. + This can occur when Tabs are cleared, if all tabs are hidden, or if the + currently active tab is unset. """ def __init__(self, tabs: Tabs) -> None: @@ -527,9 +528,10 @@ def remove_tab(self, tab_or_id: Tab | str | None) -> AwaitComplete: async def do_remove() -> None: """Perform the remove after refresh so the underline bar gets new positions.""" await remove_await - if next_tab is None: + if next_tab is None or (removing_active_tab and next_tab.id is None): self.active = "" elif removing_active_tab: + assert next_tab.id is not None self.active = next_tab.id next_tab.add_class("-active") @@ -575,12 +577,12 @@ def compose(self) -> ComposeResult: def watch_active(self, previously_active: str, active: str) -> None: """Handle a change to the active tab.""" + self.query("#tabs-list > Tab.-active").remove_class("-active") if active: try: active_tab = self.query_one(f"#tabs-list > #{active}", Tab) except NoMatches: return - self.query("#tabs-list > Tab.-active").remove_class("-active") active_tab.add_class("-active") self._highlight_active(animate=previously_active != "") self._scroll_active_tab() @@ -699,17 +701,21 @@ def action_previous_tab(self) -> None: self._move_tab(-1) def _move_tab(self, direction: int) -> None: - """Activate the next tab. + """Activate the next enabled tab in the given direction. + + Tab selection wraps around. If no tab is currently active, the "next" + tab is set to be the first and the "previous" tab is the last one. Args: direction: +1 for the next tab, -1 for the previous. """ active_tab = self.active_tab - if active_tab is None: - return tabs = self._potentially_active_tabs if not tabs: return + if not active_tab: + self.active = tabs[0 if direction == 1 else -1].id or "" + return tab_count = len(tabs) new_tab_index = (tabs.index(active_tab) + direction) % tab_count self.active = tabs[new_tab_index].id or "" diff --git a/tests/test_tabbed_content.py b/tests/test_tabbed_content.py index 92d525f698..57a8874ebf 100644 --- a/tests/test_tabbed_content.py +++ b/tests/test_tabbed_content.py @@ -108,9 +108,33 @@ def compose(self) -> ComposeResult: with pytest.raises(ValueError): tabbed_content.active = "X" - # Check fail with empty tab - with pytest.raises(ValueError): - tabbed_content.active = "" + +async def test_unsetting_tabbed_content_active(): + """Check that setting `TabbedContent.active = ""` unsets active tab.""" + + messages = [] + + class TabbedApp(App[None]): + def compose(self) -> ComposeResult: + with TabbedContent(initial="bar"): + with TabPane("foo", id="foo"): + yield Label("Foo", id="foo-label") + with TabPane("bar", id="bar"): + yield Label("Bar", id="bar-label") + with TabPane("baz", id="baz"): + yield Label("Baz", id="baz-label") + + def on_tabbed_content_cleared(self, event: TabbedContent.Cleared) -> None: + messages.append(event) + + app = TabbedApp() + async with app.run_test() as pilot: + tabbed_content = app.query_one(TabbedContent) + assert bool(tabbed_content.active) + tabbed_content.active = "" + await pilot.pause() + assert len(messages) == 1 + assert isinstance(messages[0], TabbedContent.Cleared) async def test_tabbed_content_initial(): diff --git a/tests/test_tabs.py b/tests/test_tabs.py index 32e626cefb..8d865552d2 100644 --- a/tests/test_tabs.py +++ b/tests/test_tabs.py @@ -316,15 +316,8 @@ def compose(self) -> ComposeResult: assert tabs.active_tab.id == "tab-2" assert tabs.active == tabs.active_tab.id - # TODO: This one is questionable. It seems Tabs has been designed so - # that you can set the active tab to an empty string, and it remains - # so, and just removes the underline; no other changes. So active - # will be an empty string while active_tab will be a tab. This feels - # like an oversight. Need to investigate and possibly modify this - # behaviour unless there's a good reason for this. tabs.active = "" - assert tabs.active_tab is not None - assert tabs.active_tab.id == "tab-2" + assert tabs.active_tab is None async def test_navigate_tabs_with_keyboard():