2016-12-30 21:41:28 +08:00
|
|
|
import classNames from 'classnames';
|
2019-08-20 15:58:37 +08:00
|
|
|
import toArray from 'rc-util/lib/Children/toArray';
|
2023-03-05 20:57:49 +08:00
|
|
|
import pickAttrs from 'rc-util/lib/pickAttrs';
|
2022-06-22 14:57:09 +08:00
|
|
|
import * as React from 'react';
|
2020-05-28 15:22:00 +08:00
|
|
|
import { ConfigContext } from '../config-provider';
|
2020-05-14 20:54:49 +08:00
|
|
|
import { cloneElement } from '../_util/reactNode';
|
2022-06-22 14:57:09 +08:00
|
|
|
import warning from '../_util/warning';
|
2022-11-15 15:10:47 +08:00
|
|
|
import type { BreadcrumbItemProps } from './BreadcrumbItem';
|
2022-06-22 14:57:09 +08:00
|
|
|
import BreadcrumbItem from './BreadcrumbItem';
|
|
|
|
import BreadcrumbSeparator from './BreadcrumbSeparator';
|
2016-03-31 17:46:35 +08:00
|
|
|
|
2022-05-24 01:13:36 +08:00
|
|
|
import useStyle from './style';
|
2023-03-05 20:57:49 +08:00
|
|
|
import useItems from './useItems';
|
2016-03-31 17:46:35 +08:00
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
/** @deprecated New of Breadcrumb using `items` instead of `routes` */
|
2017-11-22 11:10:33 +08:00
|
|
|
export interface Route {
|
|
|
|
path: string;
|
|
|
|
breadcrumbName: string;
|
2019-05-13 17:54:18 +08:00
|
|
|
children?: Omit<Route, 'children'>[];
|
2017-11-22 11:10:33 +08:00
|
|
|
}
|
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
export interface BreadcrumbItemType {
|
|
|
|
key?: React.Key;
|
|
|
|
/**
|
|
|
|
* Different with `path`. Directly set the link of this item.
|
|
|
|
*/
|
|
|
|
href?: string;
|
|
|
|
/**
|
|
|
|
* Different with `href`. It will concat all prev `path` to the current one.
|
|
|
|
*/
|
|
|
|
path?: string;
|
|
|
|
title: React.ReactNode;
|
|
|
|
menu?: BreadcrumbItemProps['menu'];
|
|
|
|
/** @deprecated Please use `menu` instead */
|
|
|
|
overlay?: React.ReactNode;
|
|
|
|
}
|
|
|
|
export interface BreadcrumbSeparatorType {
|
|
|
|
type: 'separator';
|
|
|
|
separator?: React.ReactNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
export type ItemType = BreadcrumbItemType | BreadcrumbSeparatorType;
|
|
|
|
|
|
|
|
type InternalRouteType = Partial<BreadcrumbItemType & BreadcrumbSeparatorType>;
|
|
|
|
|
|
|
|
export interface BaseBreadcrumbProps {
|
2016-08-29 16:52:35 +08:00
|
|
|
prefixCls?: string;
|
2019-06-24 11:29:58 +08:00
|
|
|
params?: any;
|
2017-01-20 20:10:50 +08:00
|
|
|
separator?: React.ReactNode;
|
2016-08-29 16:52:35 +08:00
|
|
|
style?: React.CSSProperties;
|
2016-12-30 21:41:28 +08:00
|
|
|
className?: string;
|
2023-01-20 11:03:50 +08:00
|
|
|
rootClassName?: string;
|
2022-04-08 22:55:42 +08:00
|
|
|
children?: React.ReactNode;
|
2017-05-25 16:54:15 +08:00
|
|
|
}
|
2016-08-29 16:52:35 +08:00
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
export interface LegacyBreadcrumbProps extends BaseBreadcrumbProps {
|
|
|
|
/** @deprecated Please use `items` instead */
|
|
|
|
routes: Route[];
|
|
|
|
|
|
|
|
itemRender?: (route: Route, params: any, routes: Route[], paths: string[]) => React.ReactNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface NewBreadcrumbProps extends BaseBreadcrumbProps {
|
|
|
|
items: ItemType[];
|
|
|
|
|
|
|
|
itemRender?: (
|
|
|
|
route: ItemType,
|
|
|
|
params: any,
|
|
|
|
routes: ItemType[],
|
|
|
|
paths: string[],
|
|
|
|
) => React.ReactNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
export type BreadcrumbProps = BaseBreadcrumbProps | LegacyBreadcrumbProps | NewBreadcrumbProps;
|
|
|
|
|
|
|
|
function getBreadcrumbName(route: InternalRouteType, params: any) {
|
|
|
|
if (route.title === undefined) {
|
2016-06-10 21:12:52 +08:00
|
|
|
return null;
|
|
|
|
}
|
2016-06-11 17:05:19 +08:00
|
|
|
const paramsKeys = Object.keys(params).join('|');
|
2023-03-05 20:57:49 +08:00
|
|
|
return typeof route.title === 'object'
|
|
|
|
? route.title
|
|
|
|
: String(route.title).replace(
|
|
|
|
new RegExp(`:(${paramsKeys})`, 'g'),
|
|
|
|
(replacement, key) => params[key] || replacement,
|
|
|
|
);
|
2016-07-14 13:29:50 +08:00
|
|
|
}
|
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
const getPath = (params: any, path?: string) => {
|
|
|
|
if (path === undefined) {
|
|
|
|
return path;
|
|
|
|
}
|
2016-10-24 12:04:26 +08:00
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
let mergedPath = (path || '').replace(/^\//, '');
|
2022-11-15 15:10:47 +08:00
|
|
|
Object.keys(params).forEach((key) => {
|
2023-03-05 20:57:49 +08:00
|
|
|
mergedPath = mergedPath.replace(`:${key}`, params[key]!);
|
2020-05-28 15:22:00 +08:00
|
|
|
});
|
2023-03-05 20:57:49 +08:00
|
|
|
return mergedPath;
|
2020-05-28 15:22:00 +08:00
|
|
|
};
|
2016-07-14 13:29:50 +08:00
|
|
|
|
2022-11-19 16:56:23 +08:00
|
|
|
type CompoundedComponent = React.FC<BreadcrumbProps> & {
|
2020-05-28 15:22:00 +08:00
|
|
|
Item: typeof BreadcrumbItem;
|
|
|
|
Separator: typeof BreadcrumbSeparator;
|
2022-11-19 16:56:23 +08:00
|
|
|
};
|
2016-03-31 17:46:35 +08:00
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
const Breadcrumb: CompoundedComponent = (props) => {
|
|
|
|
const {
|
|
|
|
prefixCls: customizePrefixCls,
|
|
|
|
separator = '/',
|
|
|
|
style,
|
|
|
|
className,
|
|
|
|
rootClassName,
|
|
|
|
routes: legacyRoutes,
|
|
|
|
items,
|
|
|
|
children,
|
|
|
|
itemRender,
|
|
|
|
params = {},
|
|
|
|
...restProps
|
|
|
|
} = props as LegacyBreadcrumbProps & NewBreadcrumbProps;
|
|
|
|
|
2020-05-28 15:22:00 +08:00
|
|
|
const { getPrefixCls, direction } = React.useContext(ConfigContext);
|
|
|
|
|
2022-10-14 11:37:48 +08:00
|
|
|
let crumbs: React.ReactNode;
|
2020-05-28 15:22:00 +08:00
|
|
|
const prefixCls = getPrefixCls('breadcrumb', customizePrefixCls);
|
2022-05-24 01:13:36 +08:00
|
|
|
const [wrapSSR, hashId] = useStyle(prefixCls);
|
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
const mergedItems = useItems(items, legacyRoutes);
|
|
|
|
|
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
warning(!legacyRoutes, 'Breadcrumb', '`routes` is deprecated. Please use `items` instead.');
|
|
|
|
}
|
|
|
|
|
|
|
|
const mergedItemRender =
|
|
|
|
itemRender ||
|
|
|
|
((route: BreadcrumbItemType) => {
|
|
|
|
const name = getBreadcrumbName(route, params);
|
|
|
|
|
|
|
|
return name;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (mergedItems && mergedItems.length > 0) {
|
2020-05-28 15:22:00 +08:00
|
|
|
// generated by route
|
2019-05-06 12:04:39 +08:00
|
|
|
const paths: string[] = [];
|
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
const itemRenderRoutes: any = items || legacyRoutes;
|
|
|
|
|
|
|
|
crumbs = mergedItems.map((item, index) => {
|
|
|
|
const { path, key, type, menu, overlay, separator: itemSeparator } = item;
|
|
|
|
const mergedPath = getPath(params, path);
|
|
|
|
|
|
|
|
if (mergedPath !== undefined) {
|
|
|
|
paths.push(mergedPath);
|
2019-05-06 12:04:39 +08:00
|
|
|
}
|
2023-03-05 20:57:49 +08:00
|
|
|
|
|
|
|
const mergedKey = key ?? index;
|
|
|
|
|
|
|
|
if (type === 'separator') {
|
|
|
|
return <BreadcrumbSeparator key={mergedKey}>{itemSeparator}</BreadcrumbSeparator>;
|
2019-05-06 12:04:39 +08:00
|
|
|
}
|
2016-09-09 13:55:46 +08:00
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
const itemProps: BreadcrumbItemProps = {};
|
|
|
|
const isLastItem = index === mergedItems.length - 1;
|
2022-11-15 15:10:47 +08:00
|
|
|
|
2023-03-05 20:57:49 +08:00
|
|
|
if (menu) {
|
|
|
|
itemProps.menu = menu;
|
|
|
|
} else if (overlay) {
|
|
|
|
itemProps.overlay = overlay as any;
|
|
|
|
}
|
|
|
|
|
|
|
|
let { href } = item;
|
|
|
|
if (paths.length && mergedPath !== undefined) {
|
|
|
|
href = `#/${paths.join('/')}`;
|
2022-11-15 15:10:47 +08:00
|
|
|
}
|
|
|
|
|
2019-05-06 12:04:39 +08:00
|
|
|
return (
|
2023-03-05 20:57:49 +08:00
|
|
|
<BreadcrumbItem
|
|
|
|
key={mergedKey}
|
|
|
|
{...itemProps}
|
|
|
|
{...pickAttrs(item, {
|
|
|
|
data: true,
|
|
|
|
aria: true,
|
|
|
|
})}
|
|
|
|
href={href}
|
|
|
|
separator={isLastItem ? '' : separator}
|
|
|
|
>
|
|
|
|
{mergedItemRender(item as BreadcrumbItemType, params, itemRenderRoutes, paths)}
|
2019-05-06 12:04:39 +08:00
|
|
|
</BreadcrumbItem>
|
|
|
|
);
|
|
|
|
});
|
2020-05-28 15:22:00 +08:00
|
|
|
} else if (children) {
|
2023-02-24 18:20:28 +08:00
|
|
|
const childrenLength = toArray(children).length;
|
2020-05-28 15:22:00 +08:00
|
|
|
crumbs = toArray(children).map((element: any, index) => {
|
|
|
|
if (!element) {
|
|
|
|
return element;
|
|
|
|
}
|
2023-03-05 20:57:49 +08:00
|
|
|
// =================== Warning =====================
|
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
warning(
|
|
|
|
!element,
|
|
|
|
'Breadcrumb',
|
|
|
|
'`Breadcrumb.Item and Breadcrumb.Separator` is deprecated. Please use `items` instead.',
|
|
|
|
);
|
|
|
|
}
|
2022-05-10 15:43:29 +08:00
|
|
|
warning(
|
2020-05-28 15:22:00 +08:00
|
|
|
element.type &&
|
|
|
|
(element.type.__ANT_BREADCRUMB_ITEM === true ||
|
|
|
|
element.type.__ANT_BREADCRUMB_SEPARATOR === true),
|
|
|
|
'Breadcrumb',
|
|
|
|
"Only accepts Breadcrumb.Item and Breadcrumb.Separator as it's children",
|
|
|
|
);
|
2023-02-24 18:20:28 +08:00
|
|
|
const isLastItem = index === childrenLength - 1;
|
2020-05-28 15:22:00 +08:00
|
|
|
return cloneElement(element, {
|
2023-02-24 18:20:28 +08:00
|
|
|
separator: isLastItem ? '' : separator,
|
2020-05-28 15:22:00 +08:00
|
|
|
key: index,
|
2016-03-31 17:46:35 +08:00
|
|
|
});
|
2020-01-21 22:01:53 +08:00
|
|
|
});
|
2018-12-05 19:12:18 +08:00
|
|
|
}
|
2020-05-28 15:22:00 +08:00
|
|
|
|
2020-09-06 13:07:39 +08:00
|
|
|
const breadcrumbClassName = classNames(
|
|
|
|
prefixCls,
|
|
|
|
{
|
|
|
|
[`${prefixCls}-rtl`]: direction === 'rtl',
|
|
|
|
},
|
|
|
|
className,
|
2023-01-20 11:03:50 +08:00
|
|
|
rootClassName,
|
2022-05-24 01:13:36 +08:00
|
|
|
hashId,
|
2020-09-06 13:07:39 +08:00
|
|
|
);
|
2020-05-28 15:22:00 +08:00
|
|
|
|
2022-05-24 01:13:36 +08:00
|
|
|
return wrapSSR(
|
2022-03-10 11:46:42 +08:00
|
|
|
<nav className={breadcrumbClassName} style={style} {...restProps}>
|
2022-03-18 15:20:35 +08:00
|
|
|
<ol>{crumbs}</ol>
|
2022-05-24 01:13:36 +08:00
|
|
|
</nav>,
|
2020-05-28 15:22:00 +08:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
Breadcrumb.Item = BreadcrumbItem;
|
|
|
|
Breadcrumb.Separator = BreadcrumbSeparator;
|
|
|
|
|
2023-01-08 21:30:41 +08:00
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
Breadcrumb.displayName = 'Breadcrumb';
|
|
|
|
}
|
|
|
|
|
2020-05-28 15:22:00 +08:00
|
|
|
export default Breadcrumb;
|