Files
filestash/client/components/breadcrumb.js
2018-04-10 14:51:52 +10:00

153 lines
4.6 KiB
JavaScript

import React from 'react';
import PropTypes from 'prop-types';
import { Link } from 'react-router-dom';
import { NgIf, Icon, EventEmitter, EventReceiver } from './';
import ReactCSSTransitionGroup from 'react-addons-css-transition-group';
import './breadcrumb.scss';
export class BreadCrumb extends React.Component {
constructor(props){
super(props);
this.state = {
path: this._formatPath(props.path)
};
}
componentWillReceiveProps(props){
this.setState({path: this._formatPath(props.path)});
}
_formatPath(full_path){
let paths = full_path.split("/");
if(paths.slice(-1)[0] === ''){
paths.pop();
}
paths = paths.map((path, index) => {
let sub_path = paths.slice(0, index+1).join('/'),
label = path === ''? 'Nuage' : path;
if(index === paths.length - 1){
return {full: null, label: label};
}else{
return {full: sub_path+'/', label: label};
}
});
return paths;
}
render(Element) {
const Path = Element? Element : PathElement;
return (
<div className="component_breadcrumb">
<BreadCrumbContainer className={this.props.className+' no-select'}>
<Logout />
<ReactCSSTransitionGroup transitionName="breadcrumb" transitionLeave={true} transitionEnter={true} transitionLeaveTimeout={150} transitionEnterTimeout={200} transitionAppear={false}>
{
this.state.path.map((path, index) => {
return (
<Path key={"breadcrumb_"+index} path={path} isLast={this.state.path.length === index + 1} needSaving={this.props.needSaving} />
);
})
}
</ReactCSSTransitionGroup>
</BreadCrumbContainer>
</div>
);
}
}
BreadCrumb.propTypes = {
path: PropTypes.string.isRequired,
needSaving: PropTypes.bool
}
const BreadCrumbContainer = (props) => {
return (
<div className={props.className}>
<ul>
{props.children}
</ul>
</div>
);
}
const Logout = (props) => {
return (
<li className="component_logout">
<Link to="/logout">
<Icon name="power"/>
</Link>
</li>
);
}
const Saving = (props) => {
return (
<NgIf className="component_saving" cond={props.needSaving === true}>
*
</NgIf>
);
}
const Separator = (props) => {
return (
<div className="component_separator">
<img width="16" height="16" src=""/>
</div>
);
}
@EventEmitter
export class PathElementWrapper extends React.Component {
constructor(props){
super(props);
}
limitSize(str){
if(str.length > 30){
return str.substring(0,23)+'...';
}
return str;
}
render(){
let className = "component_path-element-wrapper";
if(this.props.highlight) { className += " highlight";}
return (
<li className={className}>
<NgIf cond={this.props.isLast === false}>
<Link to={"/files" + this.props.path.full || "/"} className="label">
{this.limitSize(this.props.path.label)}
</Link>
<Separator/>
</NgIf>
<NgIf cond={this.props.isLast === true} className="label">
{this.limitSize(this.props.path.label)}
<Saving needSaving={this.props.needSaving} />
</NgIf>
</li>
);
}
}
// just a hack to make it play nicely with react-dnd as it refuses to use our custom component if it's not wrap by something it knows ...
export class PathElement extends PathElementWrapper {
constructor(props){
super(props);
}
render(highlight = false){
let className = "component_path-element";
if(this.props.isLast){
className += " is-last";
}
return (
<div className={className}>
<PathElementWrapper highlight={highlight} {...this.props} />
</div>
);
}
}