import * as React from 'react'; import * as PropTypes from 'prop-types'; import { cloneElement } from 'react'; import classNames from 'classnames'; import BreadcrumbItem from './BreadcrumbItem'; import Menu from '../menu'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; import warning from '../_util/warning'; import { Omit } from '../_util/type'; export interface Route { path: string; breadcrumbName: string; children?: Omit[]; } export interface BreadcrumbProps { prefixCls?: string; routes?: Route[]; params?: any; separator?: React.ReactNode; itemRender?: ( route: Route, params: any, routes: Array, paths: Array, ) => React.ReactNode; style?: React.CSSProperties; className?: string; } function getBreadcrumbName(route: Route, params: any) { if (!route.breadcrumbName) { return null; } const paramsKeys = Object.keys(params).join('|'); const name = route.breadcrumbName.replace( new RegExp(`:(${paramsKeys})`, 'g'), (replacement, key) => params[key] || replacement, ); return name; } function defaultItemRender(route: Route, params: any, routes: Route[], paths: string[]) { const isLastItem = routes.indexOf(route) === routes.length - 1; const name = getBreadcrumbName(route, params); return isLastItem ? {name} : {name}; } export default class Breadcrumb extends React.Component { static Item: typeof BreadcrumbItem; static defaultProps = { separator: '/', }; static propTypes = { prefixCls: PropTypes.string, separator: PropTypes.node, routes: PropTypes.array, params: PropTypes.object, }; componentDidMount() { const props = this.props; warning( !('linkRender' in props || 'nameRender' in props), 'Breadcrumb', '`linkRender` and `nameRender` are removed, please use `itemRender` instead, ' + 'see: https://u.ant.design/item-render.', ); } getPath = (path: string, params: any) => { path = (path || '').replace(/^\//, ''); Object.keys(params).forEach(key => { path = path.replace(`:${key}`, params[key]); }); return path; }; addChildPath = (paths: string[], childPath: string = '', params: any) => { const originalPaths = [...paths]; const path = this.getPath(childPath, params); if (path) { originalPaths.push(path); } return originalPaths; }; genForRoutes = ({ routes = [], params = {}, separator, itemRender = defaultItemRender, }: BreadcrumbProps) => { const paths: string[] = []; return routes.map(route => { const path = this.getPath(route.path, params); if (path) { paths.push(path); } // generated overlay by route.children let overlay = null; if (route.children && route.children.length) { overlay = ( {route.children.map(child => ( {itemRender(child, params, routes, this.addChildPath(paths, child.path, params))} ))} ); } return ( {itemRender(route, params, routes, paths)} ); }); }; renderBreadcrumb = ({ getPrefixCls }: ConfigConsumerProps) => { let crumbs; const { prefixCls: customizePrefixCls, separator, style, className, routes, children, } = this.props; const prefixCls = getPrefixCls('breadcrumb', customizePrefixCls); if (routes && routes.length > 0) { // generated by route crumbs = this.genForRoutes(this.props); } else if (children) { crumbs = React.Children.map(children, (element: any, index) => { if (!element) { return element; } warning( element.type && element.type.__ANT_BREADCRUMB_ITEM, 'Breadcrumb', "Only accepts Breadcrumb.Item as it's children", ); return cloneElement(element, { separator, key: index, }); }); } return (
{crumbs}
); }; render() { return {this.renderBreadcrumb}; } }