Merge pull request #79 from marshmalien/org-breadcrumb-tabs

Org Breadcrumb and Tabs
This commit is contained in:
Marliana Lara
2019-01-07 11:23:51 -05:00
committed by GitHub
11 changed files with 154 additions and 108 deletions

View File

@@ -3,9 +3,8 @@ import getTabName from '../../../src/pages/Organizations/utils';
describe('getTabName', () => { describe('getTabName', () => {
test('returns tab name', () => { test('returns tab name', () => {
expect(getTabName('details')).toBe('Details'); expect(getTabName('details')).toBe('Details');
expect(getTabName('users')).toBe('Users'); expect(getTabName('access')).toBe('Access');
expect(getTabName('teams')).toBe('Teams'); expect(getTabName('teams')).toBe('Teams');
expect(getTabName('admins')).toBe('Admins');
expect(getTabName('notifications')).toBe('Notifications'); expect(getTabName('notifications')).toBe('Notifications');
expect(getTabName('unknown')).toBe(''); expect(getTabName('unknown')).toBe('');
expect(getTabName()).toBe(''); expect(getTabName()).toBe('');

View File

@@ -56,7 +56,7 @@
// page header overrides // page header overrides
// //
.pf-l-page__main-section.pf-m-condensed { .pf-c-page__main-section.pf-m-condensed {
padding-top: 16px; padding-top: 16px;
padding-bottom: 16px; padding-bottom: 16px;
} }

View File

@@ -0,0 +1,40 @@
import React from 'react';
import { Link } from 'react-router-dom';
import './tabs.scss';
const Tab = ({ location, match, tab, currentTab, children, breadcrumb }) => {
const tabClasses = () => {
let classes = 'pf-c-tabs__item';
if (tab === currentTab) {
classes += ' pf-m-current';
}
return classes;
};
const tabParams = () => {
const params = new URLSearchParams(location.search);
if (params.get('tab') !== undefined) {
params.set('tab', tab);
} else {
params.append('tab', tab);
}
return `?${params.toString()}`;
};
return (
<li className={tabClasses()}>
<Link
className={'pf-c-tabs__button'}
to={{ pathname: `${match.url}`, search: tabParams(), state: { breadcrumb } }}
replace={tab === currentTab}>
{children}
</Link>
</li>
)
}
export default Tab;

View File

@@ -0,0 +1,13 @@
import React from 'react';
import './tabs.scss';
const Tabs = ({ children, labelText }) => (
<div className="pf-c-tabs" aria-label={labelText}>
<ul className="pf-c-tabs__list">
{children}
</ul>
</div>
);
export default Tabs;

View File

@@ -0,0 +1,50 @@
.at-c-orgPane {
a {
display: block;
}
}
.pf-c-card__header {
--pf-c-card__header--PaddingBottom: 0;
--pf-c-card__header--PaddingX: 0;
--pf-c-card__header--PaddingTop: 0;
}
.pf-c-tabs {
--pf-global--link--Color: #484848;
--pf-global--link--Color--hover: #484848;
--pf-global--link--TextDecoration--hover: none;
&:before {
border-bottom: 1px solid var(--pf-c-tabs__item--BorderColor);
border-top: 1px solid var(--pf-c-tabs__item--BorderColor);
bottom: 0;
content: " ";
left: 0;
position: absolute;
right: 0;
top: 0;
}
.pf-c-tabs__button {
--pf-c-tabs__button--PaddingLeft: 20px;
--pf-c-tabs__button--PaddingRight: 20px;
}
.pf-c-tabs__item.pf-m-current
.pf-c-tabs__button::after {
border-bottom: 3px solid var(--pf-c-tabs__item--m-current--Color);
border-top: none;
}
.pf-c-tabs__item:not(.pf-m-current):hover
.pf-c-tabs__button::after {
border-bottom: 3px solid var(--pf-global--Color--dark-200);
border-top: none;
}
}
.pf-c-breadcrumb__item.heading {
flex: 100%;
font-size: 20px;
}

View File

@@ -3,7 +3,9 @@ import { Trans } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
Title, Breadcrumb,
BreadcrumbItem,
BreadcrumbHeading
} from '@patternfly/react-core'; } from '@patternfly/react-core';
import { import {
Link Link
@@ -21,20 +23,22 @@ const OrganizationBreadcrumb = ({ parentObj, organization, currentTab, location
.map(({ url, name }, index) => { .map(({ url, name }, index) => {
let elem; let elem;
if (noLastLink && parentObj.length - 1 === index) { if (noLastLink && parentObj.length - 1 === index) {
elem = (<Fragment key={name}>{name}</Fragment>); elem = (<BreadcrumbHeading className="heading" key={name}>{name}</BreadcrumbHeading>);
} else { } else {
elem = ( elem = (
<Link <BreadcrumbItem key={name}>
key={name} <Link
to={{ pathname: url, state: { breadcrumb: parentObj, organization } }} key={name}
> to={{ pathname: url, state: { breadcrumb: parentObj, organization } }}
{name} >
</Link> {name}
</Link>
</BreadcrumbItem>
); );
} }
return elem; return elem;
}) })
.reduce((prev, curr) => [prev, ' > ', curr])} .reduce((prev, curr) => [prev, curr])}
</Fragment> </Fragment>
); );
@@ -42,25 +46,31 @@ const OrganizationBreadcrumb = ({ parentObj, organization, currentTab, location
breadcrumb = ( breadcrumb = (
<Fragment> <Fragment>
{generateCrumb()} {generateCrumb()}
{' > '} <BreadcrumbHeading className="heading">
{getTabName(currentTab)} {getTabName(currentTab)}
</BreadcrumbHeading>
</Fragment> </Fragment>
); );
} else if (location.pathname.indexOf('edit') > -1) { } else if (location.pathname.indexOf('edit') > -1) {
breadcrumb = ( breadcrumb = (
<Fragment> <Fragment>
{generateCrumb()} {generateCrumb()}
<Trans>{' > edit'}</Trans> <BreadcrumbHeading className="heading">
<Trans>Edit</Trans>
</BreadcrumbHeading>
</Fragment> </Fragment>
); );
} else if (location.pathname.indexOf('add') > -1) { } else if (location.pathname.indexOf('add') > -1) {
breadcrumb = ( breadcrumb = (
<Fragment> <Fragment>
{generateCrumb()} {generateCrumb()}
<Trans>{' > add'}</Trans> <BreadcrumbHeading className="heading">
<Trans>Add</Trans>
</BreadcrumbHeading>
</Fragment> </Fragment>
); );
} else { } else {
breadcrumb = ( breadcrumb = (
<Fragment> <Fragment>
{generateCrumb(true)} {generateCrumb(true)}
@@ -71,7 +81,7 @@ const OrganizationBreadcrumb = ({ parentObj, organization, currentTab, location
return ( return (
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl">{breadcrumb}</Title> <Breadcrumb>{breadcrumb}</Breadcrumb>
</PageSection> </PageSection>
); );
}; };

View File

@@ -6,10 +6,7 @@ import {
CardHeader, CardHeader,
CardBody, CardBody,
PageSection, PageSection,
PageSectionVariants, PageSectionVariants
ToolbarGroup,
ToolbarItem,
ToolbarSection,
} from '@patternfly/react-core'; } from '@patternfly/react-core';
import { import {
Switch, Switch,
@@ -17,39 +14,10 @@ import {
Route Route
} from 'react-router-dom'; } from 'react-router-dom';
import Tab from '../../../components/Tabs/Tab';
import Tabs from '../../../components/Tabs/Tabs';
import getTabName from '../utils'; import getTabName from '../utils';
import '../tabs.scss';
const DetailTab = ({ location, match, tab, currentTab, children, breadcrumb }) => {
const tabClasses = () => {
let classes = 'at-c-tabs__tab';
if (tab === currentTab) {
classes += ' at-m-selected';
}
return classes;
};
const updateTab = () => {
const params = new URLSearchParams(location.search);
if (params.get('tab') !== undefined) {
params.set('tab', tab);
} else {
params.append('tab', tab);
}
return `?${params.toString()}`;
};
return (
<ToolbarItem className={tabClasses()}>
<Link to={{ pathname: `${match.url}`, search: updateTab(), state: { breadcrumb } }} replace={tab === currentTab}>
{children}
</Link>
</ToolbarItem>
);
};
const OrganizationDetail = ({ const OrganizationDetail = ({
location, location,
@@ -61,6 +29,7 @@ const OrganizationDetail = ({
}) => { }) => {
// TODO: set objectName by param or through grabbing org detail get from api // TODO: set objectName by param or through grabbing org detail get from api
const { medium } = PageSectionVariants; const { medium } = PageSectionVariants;
const tabList=['details', 'access', 'teams', 'notifications'];
const deleteResourceView = () => ( const deleteResourceView = () => (
<Fragment> <Fragment>
@@ -93,34 +62,29 @@ const OrganizationDetail = ({
</Fragment> </Fragment>
); );
const detailTabs = (tabs) => (
<I18n>
{({ i18n }) => (
<ToolbarSection aria-label={i18n._(t`Organization detail tabs`)}>
<ToolbarGroup className="at-c-tabs">
{tabs.map(tab => (
<DetailTab
key={tab}
tab={tab}
location={location}
match={match}
currentTab={currentTab}
breadcrumb={parentBreadcrumbObj}
>
{getTabName(tab)}
</DetailTab>
))}
</ToolbarGroup>
</ToolbarSection>
)}
</I18n>
);
return ( return (
<PageSection variant={medium}> <PageSection variant={medium}>
<Card className="at-c-orgPane"> <Card className="at-c-orgPane">
<CardHeader> <CardHeader>
{detailTabs(['details', 'users', 'teams', 'admins', 'notifications'])} <I18n>
{({ i18n }) => (
<Tabs labelText={i18n._(t`Organization detail tabs`)}>
{tabList.map(tab => (
<Tab
key={tab}
tab={tab}
location={location}
match={match}
currentTab={currentTab}
breadcrumb={parentBreadcrumbObj}
>
<Trans>{getTabName(tab)}</Trans>
</Tab>
))}
</Tabs>
)}
</I18n>
</CardHeader> </CardHeader>
<CardBody> <CardBody>
{(currentTab && currentTab !== 'details') ? ( {(currentTab && currentTab !== 'details') ? (

View File

@@ -14,7 +14,6 @@ export default ({
name, name,
userCount, userCount,
teamCount, teamCount,
adminCount,
isSelected, isSelected,
onSelect, onSelect,
detailUrl, detailUrl,
@@ -46,7 +45,7 @@ export default ({
</span> </span>
</div> </div>
<div className="pf-c-data-list__cell"> <div className="pf-c-data-list__cell">
<Link to={`${detailUrl}?tab=users`}> <Link to={`${detailUrl}?tab=access`}>
<Trans>Users</Trans> <Trans>Users</Trans>
</Link> </Link>
<Badge isRead> <Badge isRead>
@@ -62,14 +61,6 @@ export default ({
{teamCount} {teamCount}
{' '} {' '}
</Badge> </Badge>
<Link to={`${detailUrl}?tab=admins`}>
<Trans>Admins</Trans>
</Link>
<Badge isRead>
{' '}
{adminCount}
{' '}
</Badge>
</div> </div>
<div className="pf-c-data-list__cell" /> <div className="pf-c-data-list__cell" />
</li> </li>

View File

@@ -1,18 +0,0 @@
.at-c-tabs {
padding: 0 5px !important;
margin: 0 -10px !important;
.at-c-tabs__tab {
margin: 0 5px;
}
.at-c-tabs__tab.at-m-selected {
text-decoration: underline;
}
}
.at-c-orgPane {
a {
display: block;
}
}

View File

@@ -2,12 +2,10 @@ const getTabName = (tab) => {
let tabName = ''; let tabName = '';
if (tab === 'details') { if (tab === 'details') {
tabName = 'Details'; tabName = 'Details';
} else if (tab === 'users') { } else if (tab === 'access') {
tabName = 'Users'; tabName = 'Access';
} else if (tab === 'teams') { } else if (tab === 'teams') {
tabName = 'Teams'; tabName = 'Teams';
} else if (tab === 'admins') {
tabName = 'Admins';
} else if (tab === 'notifications') { } else if (tab === 'notifications') {
tabName = 'Notifications'; tabName = 'Notifications';
} }

View File

@@ -225,7 +225,6 @@ class Organizations extends Component {
parentBreadcrumb={parentBreadcrumb} parentBreadcrumb={parentBreadcrumb}
userCount={o.summary_fields.related_field_counts.users} userCount={o.summary_fields.related_field_counts.users}
teamCount={o.summary_fields.related_field_counts.teams} teamCount={o.summary_fields.related_field_counts.teams}
adminCount={o.summary_fields.related_field_counts.admins}
isSelected={selected.includes(o.id)} isSelected={selected.includes(o.id)}
onSelect={() => this.onSelect(o.id)} onSelect={() => this.onSelect(o.id)}
/> />