From b380d71ad56c0400ba65ee0ad74f74472152190a Mon Sep 17 00:00:00 2001 From: ivan-aksamentov Date: Fri, 20 Oct 2023 12:39:16 +0200 Subject: [PATCH] feat(web): arrange nav items into breadcrumb --- .../components/Layout/LanguageSwitcher.tsx | 3 +- .../src/components/Layout/NavigationBar.tsx | 88 +++++++++++++++---- 2 files changed, 70 insertions(+), 21 deletions(-) diff --git a/packages_rs/nextclade-web/src/components/Layout/LanguageSwitcher.tsx b/packages_rs/nextclade-web/src/components/Layout/LanguageSwitcher.tsx index a788f45ec..8c72847ef 100644 --- a/packages_rs/nextclade-web/src/components/Layout/LanguageSwitcher.tsx +++ b/packages_rs/nextclade-web/src/components/Layout/LanguageSwitcher.tsx @@ -63,8 +63,7 @@ const LabelShortText = styled.span` ` const Dropdown = styled(DropdownBase)` - padding: 0; - margin: 0; + margin: 0 !important; ` const DropdownToggle = styled(DropdownToggleBase)` diff --git a/packages_rs/nextclade-web/src/components/Layout/NavigationBar.tsx b/packages_rs/nextclade-web/src/components/Layout/NavigationBar.tsx index ae494d31f..6cebb639d 100644 --- a/packages_rs/nextclade-web/src/components/Layout/NavigationBar.tsx +++ b/packages_rs/nextclade-web/src/components/Layout/NavigationBar.tsx @@ -7,11 +7,12 @@ import { NavItem as NavItemBase, } from 'reactstrap' import { useRecoilValue } from 'recoil' +import { BsCaretRightFill as ArrowRight } from 'react-icons/bs' import { Link } from 'src/components/Link/Link' import { FaDocker, FaGithub, FaXTwitter, FaDiscourse } from 'react-icons/fa6' import { LinkSmart } from 'src/components/Link/LinkSmart' import { hasRanAtom, hasTreeAtom } from 'src/state/results.state' -import styled from 'styled-components' +import styled, { useTheme } from 'styled-components' import { useTranslationSafe } from 'src/helpers/useTranslationSafe' import BrandLogoBase from 'src/assets/img/nextclade_logo.svg' import { CitationButton } from 'src/components/Citation/CitationButton' @@ -30,22 +31,10 @@ export const Navbar = styled(NavbarBase)` export const Nav = styled(NavBase)` display: flex; - vertical-align: middle; padding: 0 !important; margin: 0 !important; ` -export const NavItem = styled(NavItemBase)` - padding: 0 0.5rem; - flex-grow: 0; - flex-shrink: 0; - margin: auto; - - * { - vertical-align: middle; - } -` - const NavbarBrand = styled(NavbarBrandBase)` display: flex; flex: 1; @@ -65,16 +54,41 @@ const BrandText = styled(NextcladeTextLogo)` margin-right: 1rem; ` +export const NavItem = styled(NavItemBase)` + margin: auto; +` + export const NavLinkLocalStyle = styled(LinkSmart)<{ $active: boolean; disabled?: boolean }>` + padding: 0 0.5rem; color: ${({ $active, disabled, theme }) => (disabled ? theme.gray500 : $active ? theme.primary : theme.bodyColor)}; font-weight: ${({ $active }) => $active && 'bold'}; text-decoration: ${({ $active }) => $active && 'underline'}; cursor: ${({ disabled }) => (disabled ? 'not-allowed' : 'pointer')}; ` +export const NavItemBreadcrumb = styled(NavItem)<{ $active: boolean; disabled?: boolean }>` + min-width: 90px; + padding: 3px 0; + background: ${(props) => + props.$active ? props.theme.primary : props.disabled ? props.theme.gray250 : props.theme.gray150}; + text-align: center; + border: #0001 solid 1px; + border-radius: 3px; +` + +export const NavLinkBreadcrumbStyle = styled(NavLinkLocalStyle)<{ $active: boolean; disabled?: boolean }>` + text-decoration: none !important; + color: ${({ $active, disabled, theme }) => (disabled ? theme.gray500 : $active ? theme.white : theme.bodyColor)}; + + :hover { + color: ${({ $active, disabled, theme }) => (disabled ? theme.gray500 : $active ? theme.white : theme.bodyColor)}; + } +` + export interface NavLinkDesc { url?: string content?: ReactNode + component?: ReactNode title?: string } @@ -83,16 +97,35 @@ export interface NavLinkLocalProps { active?: boolean } -export function NavLinkImpl({ desc: { url, content, title }, active = false }: NavLinkLocalProps) { - return ( - +export function NavLinkImpl({ desc: { url, content, component, title }, active = false }: NavLinkLocalProps) { + const item = useMemo(() => { + if (component) { + return component + } + return ( {content} + ) + }, [active, component, content, url]) + + return ( + + {item} ) } +export function NavLinkBreadcrumb({ desc: { url, content, title }, active = false }: NavLinkLocalProps) { + return ( + + + {content} + + + ) +} + export function NavigationBar() { const { t } = useTranslationSafe() const { pathname } = useRouter() @@ -118,8 +151,13 @@ export function NavigationBar() { content: t('Export'), title: hasRan ? t('Export results') : t('Please run the analysis first.'), }, - ].map((desc) => { - return + ].map((desc, i) => { + const link = + if (i === 0) { + return [link] + } + const arrow = + return [arrow, link] }) }, [hasRan, hasTree, pathname, t]) @@ -166,7 +204,7 @@ export function NavigationBar() { }, { title: t('Change language'), - content: , + component: , }, ].map((desc) => { return @@ -187,3 +225,15 @@ export function NavigationBar() { ) } + +export function BreadcrumbArrow({ disabled }: { disabled?: boolean }) { + const theme = useTheme() + const color = disabled ? theme.gray500 : theme.bodyColor + return +} + +const BreadCrumbArrowIcon = styled(ArrowRight)<{ $color?: string }>` + margin: auto 0; + stroke: ${(props) => props.$color}; + fill: ${(props) => props.$color}; +`