1
0
mirror of https://github.com/ascribe/onion.git synced 2025-01-05 11:25:09 +01:00

Control expanded state of Navbar.Collapse

Controlling the expanded state allows us to close the collapse after
items are selected (as recommended by react-bootstrap maintainers).

Also makes changes to the nav components to pass through props
(required by react-bootstrap to correctly wire everything together).
This commit is contained in:
Brett Sun 2016-06-17 15:07:32 +02:00
parent f775222768
commit f9dcdb0318
4 changed files with 69 additions and 40 deletions

View File

@ -21,6 +21,7 @@ import NavRoutesLinks from './nav_routes_links';
import { currentUserShape, whitelabelShape } from './prop_types'; import { currentUserShape, whitelabelShape } from './prop_types';
import { constructHead } from '../utils/dom'; import { constructHead } from '../utils/dom';
import { safeMerge } from '../utils/general';
import { getLangText } from '../utils/lang'; import { getLangText } from '../utils/lang';
@ -35,7 +36,12 @@ let Header = React.createClass({
}, },
getInitialState() { getInitialState() {
return PieceListStore.getState(); return safeMerge(
PieceListStore.getState(),
{
expandedCollapse: false
}
);
}, },
componentDidMount() { componentDidMount() {
@ -52,7 +58,17 @@ let Header = React.createClass({
this.setState(state); this.setState(state);
}, },
getLogo() { collapseNav() {
if (this.state.expandedCollapse) {
this.onToggleCollapse(false);
}
},
onToggleCollapse(expandedCollapse) {
this.setState({ expandedCollapse });
},
renderLogo() {
const { whitelabel } = this.props; const { whitelabel } = this.props;
if (whitelabel.head) { if (whitelabel.head) {
@ -74,7 +90,7 @@ let Header = React.createClass({
} }
}, },
getPoweredBy() { renderPoweredBy() {
return ( return (
<a className="pull-left ascribe-powered-by" href="https://www.ascribe.io/" target="_blank"> <a className="pull-left ascribe-powered-by" href="https://www.ascribe.io/" target="_blank">
<span>{getLangText('powered by')} </span> <span>{getLangText('powered by')} </span>
@ -129,6 +145,7 @@ let Header = React.createClass({
navbar navbar
pullRight pullRight
hasPieces={!!unfilteredPieceListCount} hasPieces={!!unfilteredPieceListCount}
onSelect={this.collapseNav}
routes={routes} routes={routes}
userAcl={currentUser.acl} /> userAcl={currentUser.acl} />
); );
@ -154,20 +171,22 @@ let Header = React.createClass({
<Navbar <Navbar
ref="navbar" ref="navbar"
fixedTop fixedTop
className="hidden-print"> className="hidden-print"
expanded={this.state.expandedCollapse}
onToggle={this.onToggleCollapse}>
<Navbar.Header> <Navbar.Header>
<Navbar.Brand> <Navbar.Brand>
{this.getLogo()} {this.renderLogo()}
</Navbar.Brand> </Navbar.Brand>
<AclProxy <AclProxy
aclName="acl_view_powered_by" aclName="acl_view_powered_by"
aclObject={whitelabel}> aclObject={whitelabel}>
{this.getPoweredBy()} {this.renderPoweredBy()}
</AclProxy> </AclProxy>
<Navbar.Toggle /> <Navbar.Toggle />
</Navbar.Header> </Navbar.Header>
<Navbar.Collapse> <Navbar.Collapse>
<Nav navbar pullRight> <Nav navbar pullRight onSelect={this.collapseNav}>
<HeaderNotificationDebug show={false} /> <HeaderNotificationDebug show={false} />
<HeaderNotifications /> <HeaderNotifications />
{account} {account}

View File

@ -12,6 +12,7 @@ import NotificationStore from '../stores/notification_store';
import withContext from './context/with_context'; import withContext from './context/with_context';
import { currentUserShape } from './prop_types'; import { currentUserShape } from './prop_types';
import { omitFromObject } from '../utils/general';
import { getLangText } from '../utils/lang'; import { getLangText } from '../utils/lang';
@ -98,6 +99,8 @@ const HeaderNotifications = React.createClass({
// Injected through HOCs // Injected through HOCs
currentUser: currentUserShape.isRequired, currentUser: currentUserShape.isRequired,
isLoggedIn: bool.isRequired isLoggedIn: bool.isRequired
// All other props are passed down to the backing NavDropdown
}, },
getInitialState() { getInitialState() {
@ -133,6 +136,8 @@ const HeaderNotifications = React.createClass({
render() { render() {
const { editionListNotifications, pieceListNotifications } = this.state; const { editionListNotifications, pieceListNotifications } = this.state;
const dropdownProps = omitFromObject(this.props, ['currentUser'], ['isLoggedIn']);
if (pieceListNotifications.length || editionListNotifications.length) { if (pieceListNotifications.length || editionListNotifications.length) {
let numNotifications = 0; let numNotifications = 0;
@ -145,6 +150,7 @@ const HeaderNotifications = React.createClass({
return ( return (
<NavDropdown <NavDropdown
{...dropdownProps}
ref="dropdownButton" ref="dropdownButton"
className="notification-menu" className="notification-menu"
id="header-notification-dropdown" id="header-notification-dropdown"

View File

@ -1,5 +1,3 @@
'use strict';
import React from 'react'; import React from 'react';
import Nav from 'react-bootstrap/lib/Nav'; import Nav from 'react-bootstrap/lib/Nav';
@ -12,26 +10,30 @@ import { sanitizeList } from '../utils/general';
const DISABLE_ENUM = ['hasPieces', 'noPieces']; const DISABLE_ENUM = ['hasPieces', 'noPieces'];
let NavRoutesLinks = React.createClass({ const NavRoutesLinks = React.createClass({
propTypes: { propTypes: {
hasPieces: React.PropTypes.bool, hasPieces: React.PropTypes.bool,
routes: React.PropTypes.arrayOf(React.PropTypes.object), routes: React.PropTypes.arrayOf(React.PropTypes.object),
userAcl: React.PropTypes.object userAcl: React.PropTypes.object
// All other props are passed to the backing Nav
}, },
isRouteDisabled(disableOn) { isRouteDisabled(disableOn) {
const { hasPieces } = this.props; const { hasPieces } = this.props;
if (disableOn) { if (disableOn && !DISABLE_ENUM.includes(disableOn)) {
if (!DISABLE_ENUM.includes(disableOn)) { throw new Error(
throw new Error(`"disableOn" must be one of: [${DISABLE_ENUM.join(', ')}] got "${disableOn}" instead`); `"disableOn" must be one of: [${DISABLE_ENUM.join(', ')}] got "${disableOn}" instead`
);
} }
if (disableOn === 'hasPieces') { if (disableOn === 'hasPieces') {
return hasPieces; return hasPieces;
} else if (disableOn === 'noPieces') { } else if (disableOn === 'noPieces') {
return !hasPieces; return !hasPieces;
} } else {
return false;
} }
}, },
@ -48,11 +50,11 @@ let NavRoutesLinks = React.createClass({
*/ */
extractLinksFromRoutes(node, userAcl, i) { extractLinksFromRoutes(node, userAcl, i) {
if (!node) { if (!node) {
return; return null;
} }
const links = node.childRoutes.map((child, j) => { const links = node.childRoutes.map((child, j) => {
const { aclName, disableOn, headerTitle, path, childRoutes } = child; const { aclName, disableOn, headerTitle, path } = child;
// We validate if the user has set the title correctly, // We validate if the user has set the title correctly,
// otherwise we're not going to render his route // otherwise we're not going to render his route
@ -88,15 +90,12 @@ let NavRoutesLinks = React.createClass({
); );
} else { } else {
return ( return (
<NavRoutesLinksLink <NavRoutesLinksLink key={j} {...navLinkProps} />
key={j}
{...navLinkProps} />
); );
} }
} else { } else {
return null; return null;
} }
}); });
// remove all nulls from the list of generated links // remove all nulls from the list of generated links
@ -104,10 +103,15 @@ let NavRoutesLinks = React.createClass({
}, },
render() { render() {
const {routes, userAcl} = this.props; const {
routes,
userAcl,
hasPieces: ignoredHasPieces,
...props
} = this.props;
return ( return (
<Nav {...this.props}> <Nav {...props}>
{this.extractLinksFromRoutes(routes[0], userAcl, 0)} {this.extractLinksFromRoutes(routes[0], userAcl, 0)}
</Nav> </Nav>
); );

View File

@ -1,46 +1,45 @@
'use strict';
import React from 'react'; import React from 'react';
import DropdownButton from 'react-bootstrap/lib/DropdownButton';
import MenuItem from 'react-bootstrap/lib/MenuItem'; import MenuItem from 'react-bootstrap/lib/MenuItem';
import NavItem from 'react-bootstrap/lib/NavItem'; import NavItem from 'react-bootstrap/lib/NavItem';
import NavDropdown from 'react-bootstrap/lib/NavDropdown';
import LinkContainer from 'react-router-bootstrap/lib/LinkContainer'; import LinkContainer from 'react-router-bootstrap/lib/LinkContainer';
let NavRoutesLinksLink = React.createClass({ const NavRoutesLinksLink = React.createClass({
propTypes: { propTypes: {
children: React.PropTypes.oneOfType([ children: React.PropTypes.node,
React.PropTypes.arrayOf(React.PropTypes.element),
React.PropTypes.element
]),
disabled: React.PropTypes.bool,
depth: React.PropTypes.number, depth: React.PropTypes.number,
disabled: React.PropTypes.bool,
headerTitle: React.PropTypes.string, headerTitle: React.PropTypes.string,
routePath: React.PropTypes.string routePath: React.PropTypes.string
// All other props are passed through to the backing NavItem, or NavDropdown
}, },
render() { render() {
const { children, headerTitle, depth, disabled, routePath } = this.props; const { children, headerTitle, depth, disabled, routePath, ...props } = this.props;
// if the route has children, we're returning a DropdownButton that will get filled // if the route has children, we're returning a DropdownButton that will get filled
// with MenuItems // with MenuItems
if (children) { if (children) {
return ( return (
<DropdownButton <NavDropdown
{...props}
disabled={disabled} disabled={disabled}
id={`nav-route-${headerTitle.toLowerCase()}-dropdown`} id={`nav-route-${headerTitle.toLowerCase()}-dropdown`}
title={headerTitle}> title={headerTitle}>
{children} {children}
</DropdownButton> </NavDropdown>
); );
} else { } else {
if (depth === 1) { if (depth === 1) {
// if the node's child is actually a node of level one (a child of a node), we're // if the node's child is actually a node of level one (a child of a node), we're
// returning a DropdownButton matching MenuItem // returning a MenuItem for the containing NavDropdown
return ( return (
<LinkContainer <LinkContainer
{...props}
disabled={disabled} disabled={disabled}
to={routePath}> to={routePath}>
<MenuItem>{headerTitle}</MenuItem> <MenuItem>{headerTitle}</MenuItem>
@ -49,6 +48,7 @@ let NavRoutesLinksLink = React.createClass({
} else if (depth === 0) { } else if (depth === 0) {
return ( return (
<LinkContainer <LinkContainer
{...props}
disabled={disabled} disabled={disabled}
to={routePath}> to={routePath}>
<NavItem>{headerTitle}</NavItem> <NavItem>{headerTitle}</NavItem>