From 6e29b9d674a36ba19e3b121466b212c168627fd8 Mon Sep 17 00:00:00 2001 From: Moshe Date: Thu, 16 Nov 2023 16:10:43 +0200 Subject: [PATCH] feat: collapsible menu (#215) menu can be collapsed in a way it shows only icons --- src/App.scss | 6 - src/App.tsx | 5 +- .../components => layout}/header/Header.tsx | 0 .../sidebar/GitHubLink/GitHubLink.scss | 0 .../sidebar/GitHubLink/GitHubLink.tsx | 0 .../header => layout}/sidebar/SideBar.tsx | 27 +++-- src/layout/sidebar/menu/Menu.tsx | 67 +++++++++++ .../header => layout}/sidebar/menu/menu.scss | 2 +- src/layout/sidebar/sidebar.scss | 72 ++++++++++++ src/locale/he.json | 2 +- .../components/header/sidebar/menu/Menu.tsx | 47 -------- .../components/header/sidebar/sidebar.scss | 109 ------------------ src/routes/index.tsx | 18 +-- tests/about.spec.ts | 4 +- tests/menu.spec.ts | 4 +- 15 files changed, 176 insertions(+), 187 deletions(-) rename src/{pages/components => layout}/header/Header.tsx (100%) rename src/{pages/components/header => layout}/sidebar/GitHubLink/GitHubLink.scss (100%) rename src/{pages/components/header => layout}/sidebar/GitHubLink/GitHubLink.tsx (100%) rename src/{pages/components/header => layout}/sidebar/SideBar.tsx (55%) create mode 100644 src/layout/sidebar/menu/Menu.tsx rename src/{pages/components/header => layout}/sidebar/menu/menu.scss (93%) create mode 100644 src/layout/sidebar/sidebar.scss delete mode 100644 src/pages/components/header/sidebar/menu/Menu.tsx delete mode 100644 src/pages/components/header/sidebar/sidebar.scss diff --git a/src/App.scss b/src/App.scss index 0cb361bd..4ce8e644 100644 --- a/src/App.scss +++ b/src/App.scss @@ -5,10 +5,4 @@ .main { flex-direction: row; font-family: Heebo; - - > .sidebar { - height: 100vh; - background-color: white; - padding: 10px; - } } \ No newline at end of file diff --git a/src/App.tsx b/src/App.tsx index b5bcc506..788d867f 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -9,7 +9,7 @@ import { BrowserRouter as Router, useSearchParams } from 'react-router-dom' import { PageSearchState, SearchContext } from './model/pageState' import moment from 'moment' import { useSessionStorage } from 'usehooks-ts' -import SideBar from './pages/components/header/sidebar/SideBar' + import { useLocation } from 'react-router-dom' import ReactGA from 'react-ga4' import { CacheProvider } from '@emotion/react' @@ -22,7 +22,8 @@ import { AdapterMoment } from '@mui/x-date-pickers/AdapterMoment' import { LocalizationProvider } from '@mui/x-date-pickers' import RoutesList, { PAGES } from './routes' -import MainHeader from './pages/components/header/Header' +import MainHeader from './layout/header/Header' +import SideBar from './layout/sidebar/SideBar' import LayoutContext from './layout/LayoutContext' import { EasterEgg } from './pages/EasterEgg/EasterEgg' const { Content } = Layout diff --git a/src/pages/components/header/Header.tsx b/src/layout/header/Header.tsx similarity index 100% rename from src/pages/components/header/Header.tsx rename to src/layout/header/Header.tsx diff --git a/src/pages/components/header/sidebar/GitHubLink/GitHubLink.scss b/src/layout/sidebar/GitHubLink/GitHubLink.scss similarity index 100% rename from src/pages/components/header/sidebar/GitHubLink/GitHubLink.scss rename to src/layout/sidebar/GitHubLink/GitHubLink.scss diff --git a/src/pages/components/header/sidebar/GitHubLink/GitHubLink.tsx b/src/layout/sidebar/GitHubLink/GitHubLink.tsx similarity index 100% rename from src/pages/components/header/sidebar/GitHubLink/GitHubLink.tsx rename to src/layout/sidebar/GitHubLink/GitHubLink.tsx diff --git a/src/pages/components/header/sidebar/SideBar.tsx b/src/layout/sidebar/SideBar.tsx similarity index 55% rename from src/pages/components/header/sidebar/SideBar.tsx rename to src/layout/sidebar/SideBar.tsx index 18885514..fcb85521 100644 --- a/src/pages/components/header/sidebar/SideBar.tsx +++ b/src/layout/sidebar/SideBar.tsx @@ -1,19 +1,22 @@ import Menu from './menu/Menu' import './sidebar.scss' -import { Drawer } from 'antd' -import { useContext } from 'react' -import { LayoutContextInterface, LayoutCtx } from 'src/layout/LayoutContext' +import { Drawer, Layout } from 'antd' +import { useContext, useState } from 'react' +import { LayoutContextInterface, LayoutCtx } from '../LayoutContext' import GitHubLink from './GitHubLink/GitHubLink' +const { Sider } = Layout + const Logo = () => (

דאטאבוס

) +const CollapsedLogo = () =>

🚌

export default function SideBar() { const { drawerOpen, setDrawerOpen } = useContext(LayoutCtx) - + const [collapsed, setCollapsed] = useState(false) return ( <> setDrawerOpen(false)} open={drawerOpen} - className="hideOnDesktop"> + className="hideOnDesktop" + bodyStyle={{ padding: '0' }}>
- + ) } diff --git a/src/layout/sidebar/menu/Menu.tsx b/src/layout/sidebar/menu/Menu.tsx new file mode 100644 index 00000000..dfe95114 --- /dev/null +++ b/src/layout/sidebar/menu/Menu.tsx @@ -0,0 +1,67 @@ +import React, { useEffect, useState } from 'react' +import { Link, useLocation } from 'react-router-dom' +import './menu.scss' +import { useTranslation } from 'react-i18next' +import { PAGES as pages } from 'src/routes' + +import type { MenuProps } from 'antd' +import { Menu } from 'antd' + +type MenuItem = Required['items'][number] +function getItem( + label: React.ReactNode, + key: React.Key, + icon?: React.ReactNode, + children?: MenuItem[], +): MenuItem { + return { + key, + icon, + children, + label, + } as MenuItem +} + +const MainMenu = () => { + const { t, i18n } = useTranslation() + const items: MenuItem[] = pages.map((itm) => { + return getItem({t(itm.label)}, itm.path, itm.icon) + }) + const [currentLanguage, setCurrentLanguage] = useState('en') + + const handleChangeLanguage = () => { + const newLanguage = currentLanguage === 'en' ? 'he' : 'en' + setCurrentLanguage(newLanguage) + i18n.changeLanguage(newLanguage) + } + const location = useLocation() + const [current, setCurrent] = useState( + location.pathname === '/' || location.pathname === '' ? '/dashboard' : location.pathname, + ) + + useEffect(() => { + if (location) { + if (current !== location.pathname) { + setCurrent(location.pathname) + } + } + }, [location, current]) + + const handleClick: MenuProps['onClick'] = ({ key }) => { + setCurrent(key) + } + return ( + <> + + {null && } + + ) +} + +export default MainMenu diff --git a/src/pages/components/header/sidebar/menu/menu.scss b/src/layout/sidebar/menu/menu.scss similarity index 93% rename from src/pages/components/header/sidebar/menu/menu.scss rename to src/layout/sidebar/menu/menu.scss index f219e8db..7362e77d 100644 --- a/src/pages/components/header/sidebar/menu/menu.scss +++ b/src/layout/sidebar/menu/menu.scss @@ -1,4 +1,4 @@ -@import '../../../../../resources/variables'; +@import '../../../resources/variables'; .menu { flex-direction: column; diff --git a/src/layout/sidebar/sidebar.scss b/src/layout/sidebar/sidebar.scss new file mode 100644 index 00000000..2a821218 --- /dev/null +++ b/src/layout/sidebar/sidebar.scss @@ -0,0 +1,72 @@ +@import '../../resources/variables'; + + +.hideOnMobile { + display: none; +} +@media only screen and (min-width: 768px) { + .hideOnMobile { + display: block; + } +} +.hideOnDesktop { + display: block; +} +@media only screen and (min-width: 768px) { + .hideOnDesktop { + display: none; + } +} + +.sidebar-logo { + position: relative; + margin: 20px auto; + width: 119px; + font-size: 30px; + border-top: 6px solid gray; + border-bottom: 8px solid gray; + padding-bottom: 0; + line-height: 25px; + &::before { + height: 10px; + width: 10px; + border-radius: 5px; + position: absolute; + background: gray; + top: 28px; + right: 24px; + content: ' ' + } + &::after { + height: 10px; + width: 10px; + border-radius: 5px; + position: absolute; + background: gray; + top: 28px; + left: 17px; + content: ' ' + } +} + +.sidebar-logo-collapsed { + position: relative; + margin: 20px auto; + text-align: center; + font-size: 22px; + border-top: 6px solid gray; + border-bottom: 8px solid gray; + padding-bottom: 0; + line-height: 25px; +} + +.sidebar-section { + display: flex; + flex-direction: column; +} + +.sidebar-divider { + height: 1px; + background-color: #ccc; + margin: 8px 0; +} \ No newline at end of file diff --git a/src/locale/he.json b/src/locale/he.json index d788d706..2dfd2081 100644 --- a/src/locale/he.json +++ b/src/locale/he.json @@ -33,7 +33,7 @@ "ride_extra": "נסיעה שלא תוכננה 🧐", "ride_duped": "נסיעה כפולה ❇️", "checkbox_only_gaps": "רק פערים", - "dashboard_page_title": "מפעילי תח\"צ לפי קיום נסיעות מתוכננות", + "dashboard_page_title": "קיום נסיעות", "dashboard_tooltip_content": "על כל קו בישראל מוצמד GPS שמדווח את מיקום האוטובוס כל כמה רגעים.\nאז מה היא נסיעה שלא בוצעה? זאת נסיעה שתוכננה, אבל לא דווח שיצאה בנתוני הGPS. תוכלו לראות אותה באפליקציה למשל, אבל כשתחכו בתחנה, היא לעולם לא תגיע", "worst_lines_page_title": "הקווים הגרועים ביותר של 5 המפעילות הגדולות", "rides_planned": "נסיעות שתוכננו", diff --git a/src/pages/components/header/sidebar/menu/Menu.tsx b/src/pages/components/header/sidebar/menu/Menu.tsx deleted file mode 100644 index 378951e4..00000000 --- a/src/pages/components/header/sidebar/menu/Menu.tsx +++ /dev/null @@ -1,47 +0,0 @@ -import React, { useState } from 'react' -import { useLocation, useNavigate } from 'react-router-dom' -import cn from 'classnames' -import './menu.scss' -import { useTranslation } from 'react-i18next' -import { PAGES as pages } from 'src/routes' - -const Menu = () => { - const { t, i18n } = useTranslation() - - const [currentLanguage, setCurrentLanguage] = useState('en') - - const navigate = useNavigate() - const { pathname: currpage } = useLocation() - - const handleChangeLanguage = () => { - const newLanguage = currentLanguage === 'en' ? 'he' : 'en' - setCurrentLanguage(newLanguage) - i18n.changeLanguage(newLanguage) - } - - return ( -
    - {pages.map((page) => ( -
  • - page.path[0] === '/' ? navigate(page.path) : window.open(page.path, '_blank') - }> - {React.createElement(page.icon)} - { - - } - {t(page.label)} -
  • - ))} - {null && } -
- ) -} - -export default Menu diff --git a/src/pages/components/header/sidebar/sidebar.scss b/src/pages/components/header/sidebar/sidebar.scss deleted file mode 100644 index b3af711f..00000000 --- a/src/pages/components/header/sidebar/sidebar.scss +++ /dev/null @@ -1,109 +0,0 @@ -@import '../../../../resources/variables'; - -.sidebar-menu-toggle { - display: none; -} - -@media screen and (max-width: $mobile-max-width) { - - .main > .sidebar { - max-width: 0; - padding: 0; - transition: all 0.2s ease-in-out; - .sidebar-logo { - display: none; - } - .sidebar-menu-toggle { - display: block; - position: absolute; - top: 0; - right: 0; - margin: 10px; - cursor: pointer; - font-size: 20px; - } - } - - .main > .sidebar.open{ - .sidebar-logo { - display: block; - } - display: block; - position: absolute; - z-index: 500; - max-width: 100%; - padding: 30px; - margin-right: 1px; - } - -} -.hideOnMobile { - display: none; - } - @media only screen and (min-width: 768px) { - .hideOnMobile { - display: block; - } - } - - .hideOnDesktop { - display: block; - } - @media only screen and (min-width: 768px) { - .hideOnDesktop { - display: none; - } - } - -.sidebar-logo { - position: relative; - margin: 20px auto; - width: 119px; - font-size: 30px; - border-top: 6px solid gray; - border-bottom: 8px solid gray; - padding-bottom: 0; - line-height: 25px; - &::before { - height: 10px; - width: 10px; - border-radius: 5px; - position: absolute; - background: gray; - top: 28px; - right: 24px; - content: ' ' - } - &::after { - height: 10px; - width: 10px; - border-radius: 5px; - position: absolute; - background: gray; - top: 28px; - left: 17px; - content: ' ' - } -} - -.sidebar-logo-collapsed { - position: relative; - margin: 20px auto; - text-align: center; - font-size: 30px; - border-top: 6px solid gray; - border-bottom: 8px solid gray; - padding-bottom: 0; - line-height: 25px; -} - -.sidebar-section { - display: flex; - flex-direction: column; -} - -.sidebar-divider { - height: 1px; - background-color: #ccc; - margin: 8px 0; -} \ No newline at end of file diff --git a/src/routes/index.tsx b/src/routes/index.tsx index 4cade905..c5b01395 100644 --- a/src/routes/index.tsx +++ b/src/routes/index.tsx @@ -26,58 +26,58 @@ export const PAGES = [ { label: TEXT_KEYS.dashboard_page_title, path: '/dashboard', - icon: LaptopOutlined, + icon: , element: , }, { label: TEXT_KEYS.timeline_page_title, path: '/timeline', searchParamsRequired: true, - icon: FieldTimeOutlined, + icon: , element: , }, { label: TEXT_KEYS.gaps_page_title, path: '/gaps', searchParamsRequired: true, - icon: BarChartOutlined, + icon: , element: , }, { label: TEXT_KEYS.gaps_patterns_page_title, path: '/gaps_patterns', - icon: LineChartOutlined, + icon: , element: , }, { label: TEXT_KEYS.realtime_map_page_title, path: '/map', - icon: HeatMapOutlined, + icon: , element: , }, { label: TEXT_KEYS.singleline_map_page_title, path: '/single-line-map', searchParamsRequired: true, - icon: RadarChartOutlined, + icon: , element: , }, { label: TEXT_KEYS.about_title, path: '/about', - icon: BellOutlined, + icon: , element: , }, { label: TEXT_KEYS.report_a_bug_title, path: 'https://github.com/hasadna/open-bus-map-search/issues', - icon: BugOutlined, + icon: , element: null, }, { label: TEXT_KEYS.donate_title, path: 'https://www.jgive.com/new/he/ils/donation-targets/3268#donation-modal', - icon: DollarOutlined, + icon: , element: null, }, ] diff --git a/tests/about.spec.ts b/tests/about.spec.ts index e0585ce2..543f9c87 100644 --- a/tests/about.spec.ts +++ b/tests/about.spec.ts @@ -4,8 +4,8 @@ test.describe('About Page Tests', () => { await page.goto('/') await page.getByText('אודות').click() await expect(page).toHaveURL('http://localhost:3000/about') - const locator = await page.getByText('אודות') - await expect(locator).toHaveClass('menu-item active') + const locator = await page.locator('li:has-text("אודות")') + await expect(locator).toHaveClass(/menu-item-selected/) }) test('page display title `מהו אתר “דאטאבוס”?`', async ({ page }) => { await page.goto('/about') diff --git a/tests/menu.spec.ts b/tests/menu.spec.ts index f8bc623c..e1249cbb 100644 --- a/tests/menu.spec.ts +++ b/tests/menu.spec.ts @@ -4,7 +4,7 @@ test('menu', async ({ page }) => { await page.goto('/') await expect(page.locator('h1')).toContainText('דאטאבוס') const menuItemsInOrder = [ - 'מפעילי תח"צ לפי קיום נסיעות מתוכננות', + 'קיום נסיעות', 'לוח זמנים היסטורי', 'נסיעות שלא יצאו', 'דפוסי נסיעות שלא יצאו', @@ -14,5 +14,5 @@ test('menu', async ({ page }) => { 'דיווח על באג', 'לתרומות', ] - await expect(page.locator('ul > li')).toContainText(menuItemsInOrder) + await expect(page.locator('ul > li a')).toContainText(menuItemsInOrder) })