Skip to content

Commit

Permalink
Merge branch 'main' of https://github.com/mckinsey/vizro into release…
Browse files Browse the repository at this point in the history
…_tests_fix
  • Loading branch information
l0uden committed Oct 2, 2023
2 parents 0ee0ec5 + 320eb28 commit 8ee61a4
Show file tree
Hide file tree
Showing 11 changed files with 122 additions and 93 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
<!--
A new scriv changelog fragment.
Uncomment the section that is right (remove the HTML comment wrapper).
-->

<!--
### Removed
- A bullet item for the Removed category.
-->
<!--
### Added
- A bullet item for the Added category.
-->
<!--
### Changed
- A bullet item for the Changed category.
-->
<!--
### Deprecated
- A bullet item for the Deprecated category.
-->

### Fixed

- Fix unit test interdependence issue due to shared dash.page_registry ([#84](https://github.com/mckinsey/vizro/pull/84))

<!--
### Security
- A bullet item for the Security category.
-->
1 change: 1 addition & 0 deletions vizro-core/docs/pages/development/authors.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@

## Previous team members and code contributors

[axa99](https://github.com/axa99),
[Jo Stichbury](https://github.com/stichbury),
[Juan Luis Cano Rodríguez](https://github.com/astrojuanlu),
[Denis Lebedev](https://github.com/DenisLebedevMcK),
Expand Down
2 changes: 1 addition & 1 deletion vizro-core/src/vizro/_constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,4 +6,4 @@
ON_PAGE_LOAD_ACTION_PREFIX = "on_page_load_action"
FILTER_ACTION_PREFIX = "filter_action"
PARAMETER_ACTION_PREFIX = "parameter_action"
ACCORDION_TITLE = "SELECT PAGE"
ACCORDION_DEFAULT_TITLE = "SELECT PAGE"
68 changes: 21 additions & 47 deletions vizro-core/src/vizro/models/_navigation/_accordion.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
from dash import html
from pydantic import validator

from vizro._constants import ACCORDION_TITLE, MODULE_PAGE_404
from vizro._constants import ACCORDION_DEFAULT_TITLE
from vizro.models import VizroBaseModel
from vizro.models._models_utils import _log_call
from vizro.models._navigation.navigation import _validate_pages
Expand All @@ -27,11 +27,13 @@ class Accordion(VizroBaseModel):

@_log_call
def build(self):
if self.pages:
return self._create_accordion()
return self._create_default_accordion()
return self._create_accordion()

def _create_accordion_buttons(self, accordion_pages):
def _create_accordion_buttons(self, pages):
"""Creates a button for each provided page."""
# TODO: Better if we loop through pages from MM so the Accordion.build does not depend on dashboard build.
# However, this would require that only pages used in the Dashboard are registered in the MM.
# Note: Relative path currently deviates from page.path for first page.
return [
dbc.Button(
children=[page["name"]],
Expand All @@ -40,33 +42,20 @@ def _create_accordion_buttons(self, accordion_pages):
href=page["relative_path"],
)
for page in dash.page_registry.values()
for accordion_page in accordion_pages
if accordion_page == page["module"] and page["module"] != MODULE_PAGE_404
if page["module"] in pages
]

def _create_accordion_item(self, accordion_buttons, title=ACCORDION_TITLE):
def _create_accordion_item(self, accordion_buttons, title=ACCORDION_DEFAULT_TITLE):
"""Creates an accordion item for each sub-group of pages."""
return dbc.AccordionItem(
children=accordion_buttons,
title=title.upper(),
class_name="accordion_item",
)

def _create_default_accordion(self):
accordion_buttons = [
dbc.Button(
children=[page["name"]],
key=page["relative_path"],
className="accordion-item-button",
href=page["relative_path"],
)
for page in dash.page_registry.values()
if page["module"] != MODULE_PAGE_404
]

accordion_items = [self._create_accordion_item(accordion_buttons=accordion_buttons)]

# Don't create accordion navigation if there is only one page and one accordion item
if len(accordion_buttons) == len(accordion_items) == 1:
def _get_accordion_container(self, accordion_items, accordion_buttons):
# Return no container if there is only one page in the dashboard or no pages exist
if (len(accordion_buttons) == len(accordion_items) == 1) or not accordion_buttons:
return None

return html.Div(
Expand All @@ -85,31 +74,16 @@ def _create_default_accordion(self):
)

def _create_accordion(self):
"""Creates a custom accordion only with user-provided pages."""
accordion_items = []

if isinstance(self.pages, dict):
for title, accordion_pages in self.pages.items():
accordion_buttons = self._create_accordion_buttons(accordion_pages=accordion_pages)
accordion_items.append(self._create_accordion_item(accordion_buttons=accordion_buttons, title=title))
for page_group, page_members in self.pages.items():
accordion_buttons = self._create_accordion_buttons(pages=page_members)
accordion_items.append(
self._create_accordion_item(accordion_buttons=accordion_buttons, title=page_group)
)

if isinstance(self.pages, list):
accordion_buttons = self._create_accordion_buttons(accordion_pages=self.pages)
accordion_buttons = self._create_accordion_buttons(pages=self.pages)
accordion_items.append(self._create_accordion_item(accordion_buttons=accordion_buttons))

if len(accordion_buttons) == len(accordion_items) == 1:
return None

return html.Div(
children=[
dbc.Accordion(
id=self.id,
children=accordion_items,
class_name="accordion",
persistence=True,
persistence_type="session",
),
html.Div(className="keyline"),
],
className="nav_panel",
id=f"{self.id}_outer",
)
return self._get_accordion_container(accordion_items=accordion_items, accordion_buttons=accordion_buttons)
5 changes: 5 additions & 0 deletions vizro-core/src/vizro/models/_navigation/navigation.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,10 @@
import warnings
from typing import TYPE_CHECKING, Optional

import dash
from pydantic import PrivateAttr, validator

from vizro._constants import MODULE_PAGE_404
from vizro.managers import model_manager
from vizro.models import VizroBaseModel
from vizro.models._models_utils import _log_call
Expand Down Expand Up @@ -45,6 +47,9 @@ def _validate_pages(pages):
f"Unknown page ID or page title provided to Navigation 'pages'. " f"Unknown pages: {unknown_pages}"
)

if pages is None:
return [page for page in dash.page_registry.keys() if page != MODULE_PAGE_404]

return pages


Expand Down
26 changes: 18 additions & 8 deletions vizro-core/tests/unit/vizro/conftest.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
"""Fixtures to be shared across several tests."""

import dash
import pytest

import vizro.models as vm
Expand All @@ -24,15 +26,23 @@ def standard_px_chart(gapminder):


@pytest.fixture()
def two_pages():
two_pages = [
vm.Page(title="Page 1", components=[vm.Button(), vm.Button()]),
vm.Page(title="Page 2", components=[vm.Button(), vm.Button()]),
]
return two_pages
def page1():
return vm.Page(title="Page 1", components=[vm.Button(), vm.Button()])


@pytest.fixture()
def page2():
return vm.Page(title="Page 2", components=[vm.Button(), vm.Button()])


@pytest.fixture()
def dashboard(two_pages):
dashboard = vm.Dashboard(pages=two_pages)
def dashboard(page1, page2):
dashboard = vm.Dashboard(pages=[page1, page2])
return dashboard


@pytest.fixture()
def dashboard_build(dashboard):
yield dashboard.build()
del dash.page_registry["Page 1"]
del dash.page_registry["Page 2"]
10 changes: 3 additions & 7 deletions vizro-core/tests/unit/vizro/models/_navigation/conftest.py
Original file line number Diff line number Diff line change
@@ -1,14 +1,10 @@
"""Fixtures to be shared across several tests."""

import dash_bootstrap_components as dbc
import pytest
from dash import html

from vizro import Vizro


@pytest.fixture
def dashboard_build(dashboard):
return Vizro().build(dashboard)
from vizro._constants import ACCORDION_DEFAULT_TITLE


@pytest.fixture()
Expand Down Expand Up @@ -40,7 +36,7 @@ def accordion_from_page_as_list():
accordion_items = [
dbc.AccordionItem(
children=[*accordion_buttons],
title="SELECT PAGE",
title=ACCORDION_DEFAULT_TITLE,
class_name="accordion_item",
)
]
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ class TestAccordionInstantiation:
def test_create_accordion_default(self):
accordion = Accordion(id="accordion_id")
assert accordion.id == "accordion_id"
assert accordion.pages is None
assert accordion.pages == ["Page 1", "Page 2"]

def test_create_accordion_pages_as_list(self, pages_as_list):
accordion = Accordion(pages=pages_as_list, id="accordion_id")
Expand Down Expand Up @@ -64,6 +64,6 @@ def test_accordion_build_single_page_accordion(self):
accordion = Accordion(pages=["Page 1"], id="single_accordion").build()
assert accordion is None

def test_navigation_not_all_pages_included(self, dashboard_build):
def test_navigation_not_all_pages_included(self):
with pytest.warns(UserWarning):
Accordion(pages=["Page 1"])
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ class TestNavigationInstantiation:
def test_navigation_default(self):
navigation = vm.Navigation(id="navigation")
assert navigation.id == "navigation"
assert navigation.pages is None
assert navigation.pages == ["Page 1", "Page 2"]

def test_navigation_pages_as_list(self, pages_as_list):
navigation = vm.Navigation(pages=pages_as_list, id="navigation")
Expand All @@ -28,7 +28,7 @@ def test_navigation_pages_as_dict(self, pages_as_dict):
assert navigation.id == "navigation"
assert navigation.pages == pages_as_dict

def test_navigation_not_all_pages_included(self, dashboard_build):
def test_navigation_not_all_pages_included(self):
with pytest.warns(UserWarning):
vm.Navigation(pages=["Page 1"])

Expand Down
Loading

0 comments on commit 8ee61a4

Please sign in to comment.