From 130250ef94084ae2558b8dd03868a944dd796a5d Mon Sep 17 00:00:00 2001 From: gruhn Date: Thu, 31 Oct 2024 11:17:31 +0100 Subject: [PATCH 01/24] add: highlightening selected menu item --- web_ui_src/main.css | 6 ++++++ web_ui_src/shc_base.js | 25 +++++++++++++++++++++++++ 2 files changed, 31 insertions(+) diff --git a/web_ui_src/main.css b/web_ui_src/main.css index 8fe08850..cb7cedd1 100644 --- a/web_ui_src/main.css +++ b/web_ui_src/main.css @@ -22,6 +22,12 @@ body.pushable>.pusher, body:not(.pushable) { line-height: 36px; } +/* highlight selected menu item */ +.main-menu a.active { + background-color: #f0f0f0; + color: #000; +} + /* ************************** * * Semantic UI extensions * diff --git a/web_ui_src/shc_base.js b/web_ui_src/shc_base.js index bc1d634c..3a136c0b 100644 --- a/web_ui_src/shc_base.js +++ b/web_ui_src/shc_base.js @@ -54,6 +54,8 @@ export let WIDGET_TYPES = new Map(); if (widgetElements.length > 0) { openWebsocket(); } + + highlightSelectedPage(); } function openWebsocket() { @@ -137,6 +139,29 @@ export let WIDGET_TYPES = new Map(); } } + function highlightSelectedPage() { + // Get the current URL path + const currentPath = window.location.pathname; + + // Select all the menu items + const menuItems = document.querySelectorAll('.main-menu a'); + + // Loop through each menu item and add active class to current page + menuItems.forEach(function(item) { + if (item.href) { + itemPath = new URL(item.href).pathname; + if (currentPath.endsWith(itemPath)) { + if (item.parentElement.classList.contains("menu")) { // special case submenu + item.parentElement.parentElement.classList.add('active'); + } else { + item.classList.add('active'); + } + } + } + }); + + } + document.addEventListener('DOMContentLoaded', (event) => init()); })(); From 8e903891b0f83d6fadd12b9baa60ada9dba487c2 Mon Sep 17 00:00:00 2001 From: gruhn Date: Thu, 31 Oct 2024 11:25:24 +0100 Subject: [PATCH 02/24] remove css part, now using semantic defaults --- web_ui_src/main.css | 6 ------ 1 file changed, 6 deletions(-) diff --git a/web_ui_src/main.css b/web_ui_src/main.css index cb7cedd1..8fe08850 100644 --- a/web_ui_src/main.css +++ b/web_ui_src/main.css @@ -22,12 +22,6 @@ body.pushable>.pusher, body:not(.pushable) { line-height: 36px; } -/* highlight selected menu item */ -.main-menu a.active { - background-color: #f0f0f0; - color: #000; -} - /* ************************** * * Semantic UI extensions * From 2a309c4a332ffa639b53a021f89c7a38968c2d80 Mon Sep 17 00:00:00 2001 From: gruhn Date: Thu, 31 Oct 2024 11:49:42 +0100 Subject: [PATCH 03/24] nop: try trigger github actions --- web_ui_src/shc_base.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/web_ui_src/shc_base.js b/web_ui_src/shc_base.js index 3a136c0b..03eb04e0 100644 --- a/web_ui_src/shc_base.js +++ b/web_ui_src/shc_base.js @@ -146,7 +146,7 @@ export let WIDGET_TYPES = new Map(); // Select all the menu items const menuItems = document.querySelectorAll('.main-menu a'); - // Loop through each menu item and add active class to current page + // Loop through each menu item and add the active class to current page menuItems.forEach(function(item) { if (item.href) { itemPath = new URL(item.href).pathname; From 67d03df5394288b06cd0a42c986a010a991604e2 Mon Sep 17 00:00:00 2001 From: gruhn Date: Thu, 31 Oct 2024 18:40:32 +0100 Subject: [PATCH 04/24] add: test for menu selection --- example/ui_showcase.py | 10 +++++++-- test/test_web.py | 46 ++++++++++++++++++++++++++++++++++++++++++ web_ui_src/main.css | 4 ++++ web_ui_src/shc_base.js | 32 +++++++++++++++++------------ 4 files changed, 77 insertions(+), 15 deletions(-) diff --git a/example/ui_showcase.py b/example/ui_showcase.py index 922f87ac..9dbb5c88 100644 --- a/example/ui_showcase.py +++ b/example/ui_showcase.py @@ -149,11 +149,17 @@ async def do_magic(_v, _o) -> None: ############################################################################################# # Overview page # -# Here we show the ImageMap and HideRows # +# Here we show the ImageMap, HideRows and sub menus. # ############################################################################################# overview_page = web_server.page('overview', "Overview", menu_entry='Some Submenu', menu_icon='tachometer alternate', - menu_sub_label="Overview") + menu_sub_label="Overview", menu_sub_icon='tachometer alternate') + +submenu_page2 = web_server.page('submenu2', "Nothing here", menu_entry='Some Submenu', menu_sub_icon='couch', + menu_sub_label="Empty Submenu") + +submenu_page3 = web_server.page('submenu3', "Nothing here either", menu_entry='Some Submenu', menu_sub_icon='motorcycle', + menu_sub_label="Empty Submenu 2") # ImageMap supports all the different Buttons as items, as well as the special ImageMapLabel # The optional fourth entry of each item is a list of WebPageItems (everything we have shown so far – even an ImageMap)) diff --git a/test/test_web.py b/test/test_web.py index 377c79bc..e62544a9 100644 --- a/test/test_web.py +++ b/test/test_web.py @@ -148,6 +148,52 @@ def test_main_menu(self) -> None: self.assertEqual(submenu_entry_href.strip(), "http://localhost:42080/page/another_page/") submenu_entry.find_element(By.CSS_SELECTOR, 'i.bars.icon') + def test_main_menu_selection(self) -> None: + """Test that the clicked menu item is selected thus highlighted.""" + self.server.page('index', menu_entry="Home", menu_icon='home') + self.server.page('overview', menu_entry="Foo", menu_icon='info') + + # add submenu + self.server.page('submenu1', "Sub1", menu_entry='Some Submenu', menu_icon='bell', + menu_sub_label="Overview") + self.server.page('submenu2', "Sub2", menu_entry='Some Submenu', + menu_sub_label="Empty Submenu") + self.server.page('submenu3', "Sub3", menu_entry='Some Submenu', + menu_sub_label="Empty Submenu 2") + + self.server_runner.start() + self.driver.get("http://localhost:42080") + + # test on startup only 1st item is selected + container = self.driver.find_element(By.CSS_SELECTOR, '.pusher') + selected_menus = container.find_elements(By.CLASS_NAME, 'selected') + self.assertEqual(len(selected_menus), 1) + self.assertIn("Home", selected_menus[0].text) + + # test after click on foo only the clicked item is selected + foo_link = container.find_element(By.CSS_SELECTOR, 'i.info.icon').find_element(By.XPATH, '..') + foo_link.click() + + container = self.driver.find_element(By.CSS_SELECTOR, '.pusher') + selected_menus = container.find_elements(By.CLASS_NAME, 'selected') + self.assertEqual(len(selected_menus), 1) + self.assertIn("Foo", selected_menus[0].text) + + # click top level submenu item 1st to open submenu + submenu = container.find_element(By.CSS_SELECTOR, 'i.bell.icon').find_element(By.XPATH, '..') + submenu.click() + + # now select submenu item + submenu_entry = submenu.find_element(By.XPATH, './/a[contains(@class, "item")]') + submenu_entry.click() + + # test after selecting a submenu both are selected the submenu item and the menu item + container = self.driver.find_element(By.CSS_SELECTOR, '.pusher') + selected_menus = container.find_elements(By.CLASS_NAME, 'selected') + self.assertEqual(len(selected_menus), 2) + self.assertIn("Some Submenu", selected_menus[0].text) + self.assertTrue(selected_menus[1].get_attribute("href").endswith("/page/submenu1/")) + class MonitoringTest(unittest.TestCase): def setUp(self) -> None: diff --git a/web_ui_src/main.css b/web_ui_src/main.css index 8fe08850..e81a015f 100644 --- a/web_ui_src/main.css +++ b/web_ui_src/main.css @@ -22,6 +22,10 @@ body.pushable>.pusher, body:not(.pushable) { line-height: 36px; } +a.mobile.hidden.item.selected, +div.mobile.hidden.dropdown.item.selected { + background: url("prism.png") !important; +} /* ************************** * * Semantic UI extensions * diff --git a/web_ui_src/shc_base.js b/web_ui_src/shc_base.js index 03eb04e0..74f0832a 100644 --- a/web_ui_src/shc_base.js +++ b/web_ui_src/shc_base.js @@ -142,26 +142,32 @@ export let WIDGET_TYPES = new Map(); function highlightSelectedPage() { // Get the current URL path const currentPath = window.location.pathname; - // Select all the menu items - const menuItems = document.querySelectorAll('.main-menu a'); - - // Loop through each menu item and add the active class to current page - menuItems.forEach(function(item) { - if (item.href) { - itemPath = new URL(item.href).pathname; - if (currentPath.endsWith(itemPath)) { - if (item.parentElement.classList.contains("menu")) { // special case submenu - item.parentElement.parentElement.classList.add('active'); - } else { - item.classList.add('active'); + const menuItems = $('.main-menu a'); + + // Loop through each menu item and add the selected class to the current page + menuItems.each(function() { + let $item = $(this); + $item.removeClass('selected'); + + if ($item.attr('href')) { + const href = $item.attr('href'); + + // Add selected class + if (currentPath.endsWith(href)) { + $item.addClass('selected'); + + // Special case for submenu + const dropdownMenu = $item.parents('.dropdown'); + if (dropdownMenu.length) { + dropdownMenu.addClass('selected'); } } } }); - } + document.addEventListener('DOMContentLoaded', (event) => init()); })(); From 97ed155f06abf36c9aa624272b471ed5b1179866 Mon Sep 17 00:00:00 2001 From: gruhn Date: Thu, 31 Oct 2024 19:01:56 +0100 Subject: [PATCH 05/24] fix: flake and mypy issues --- example/ui_showcase.py | 4 ++-- test/test_web.py | 3 ++- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/example/ui_showcase.py b/example/ui_showcase.py index 9dbb5c88..fea5e1fa 100644 --- a/example/ui_showcase.py +++ b/example/ui_showcase.py @@ -158,8 +158,8 @@ async def do_magic(_v, _o) -> None: submenu_page2 = web_server.page('submenu2', "Nothing here", menu_entry='Some Submenu', menu_sub_icon='couch', menu_sub_label="Empty Submenu") -submenu_page3 = web_server.page('submenu3', "Nothing here either", menu_entry='Some Submenu', menu_sub_icon='motorcycle', - menu_sub_label="Empty Submenu 2") +submenu_page3 = web_server.page('submenu3', "Nothing here either", menu_entry='Some Submenu', + menu_sub_icon='motorcycle', menu_sub_label="Empty Submenu 2") # ImageMap supports all the different Buttons as items, as well as the special ImageMapLabel # The optional fourth entry of each item is a list of WebPageItems (everything we have shown so far – even an ImageMap)) diff --git a/test/test_web.py b/test/test_web.py index e62544a9..2d07845b 100644 --- a/test/test_web.py +++ b/test/test_web.py @@ -192,7 +192,8 @@ def test_main_menu_selection(self) -> None: selected_menus = container.find_elements(By.CLASS_NAME, 'selected') self.assertEqual(len(selected_menus), 2) self.assertIn("Some Submenu", selected_menus[0].text) - self.assertTrue(selected_menus[1].get_attribute("href").endswith("/page/submenu1/")) + submenu_href: str = str(selected_menus[1].get_attribute("href")) + self.assertTrue(submenu_href.endswith("/page/submenu1/")) class MonitoringTest(unittest.TestCase): From 1d1d5c57accd23791b1947e6b60668b958b53570 Mon Sep 17 00:00:00 2001 From: gruhn Date: Sun, 3 Nov 2024 09:30:08 +0100 Subject: [PATCH 06/24] fix: flake and mypy issues --- shc/web/interface.py | 5 +++-- shc/web/templates/base.htm | 14 +++++++++++--- web_ui_src/main.css | 3 +-- web_ui_src/main.js | 3 ++- web_ui_src/shc_base.js | 31 ------------------------------- 5 files changed, 17 insertions(+), 39 deletions(-) diff --git a/shc/web/interface.py b/shc/web/interface.py index 1a1f7415..82c45a1e 100644 --- a/shc/web/interface.py +++ b/shc/web/interface.py @@ -292,8 +292,9 @@ async def _page_handler(self, request: aiohttp.web.Request) -> aiohttp.web.Respo html_title = self.title_formatter(page.title) template = jinja_env.get_template('page.htm') body = await template.render_async(title=page.title, segments=page.segments, menu=self.ui_menu_entries, - root_url=self.root_url, js_files=self._js_files, css_files=self._css_files, - server_token=id(self), html_title=html_title) + qroot_url=self.root_url, js_files=self._js_files, css_files=self._css_files, + server_token=id(self), html_title=html_title, + current_url=request.path) return aiohttp.web.Response(body=body, content_type="text/html", charset='utf-8') async def _ui_websocket_handler(self, request: aiohttp.web.Request) -> aiohttp.web.WebSocketResponse: diff --git a/shc/web/templates/base.htm b/shc/web/templates/base.htm index 414f011b..85155f00 100644 --- a/shc/web/templates/base.htm +++ b/shc/web/templates/base.htm @@ -61,18 +61,26 @@
{% for label, icon, link in menu %} {% if link is string %} - {% else %} -