mirror of
https://github.com/ant-design/ant-design.git
synced 2024-12-28 19:18:40 +08:00
371 lines
11 KiB
TypeScript
371 lines
11 KiB
TypeScript
import * as React from 'react';
|
|
import classNames from 'classnames';
|
|
import useEvent from 'rc-util/lib/hooks/useEvent';
|
|
import scrollIntoView from 'scroll-into-view-if-needed';
|
|
|
|
import getScroll from '../_util/getScroll';
|
|
import scrollTo from '../_util/scrollTo';
|
|
import { devUseWarning } from '../_util/warning';
|
|
import Affix from '../affix';
|
|
import type { AffixProps } from '../affix';
|
|
import type { ConfigConsumerProps } from '../config-provider';
|
|
import { ConfigContext } from '../config-provider';
|
|
import useCSSVarCls from '../config-provider/hooks/useCSSVarCls';
|
|
import type { AnchorLinkBaseProps } from './AnchorLink';
|
|
import AnchorLink from './AnchorLink';
|
|
import AnchorContext from './context';
|
|
import useStyle from './style';
|
|
|
|
export interface AnchorLinkItemProps extends AnchorLinkBaseProps {
|
|
key: React.Key;
|
|
children?: AnchorLinkItemProps[];
|
|
}
|
|
|
|
export type AnchorContainer = HTMLElement | Window;
|
|
|
|
function getDefaultContainer() {
|
|
return window;
|
|
}
|
|
|
|
function getOffsetTop(element: HTMLElement, container: AnchorContainer): number {
|
|
if (!element.getClientRects().length) {
|
|
return 0;
|
|
}
|
|
|
|
const rect = element.getBoundingClientRect();
|
|
|
|
if (rect.width || rect.height) {
|
|
if (container === window) {
|
|
return rect.top - element.ownerDocument!.documentElement!.clientTop;
|
|
}
|
|
return rect.top - (container as HTMLElement).getBoundingClientRect().top;
|
|
}
|
|
|
|
return rect.top;
|
|
}
|
|
|
|
const sharpMatcherRegex = /#([\S ]+)$/;
|
|
|
|
interface Section {
|
|
link: string;
|
|
top: number;
|
|
}
|
|
|
|
export interface AnchorProps {
|
|
prefixCls?: string;
|
|
className?: string;
|
|
rootClassName?: string;
|
|
style?: React.CSSProperties;
|
|
/**
|
|
* @deprecated Please use `items` instead.
|
|
*/
|
|
children?: React.ReactNode;
|
|
offsetTop?: number;
|
|
bounds?: number;
|
|
affix?: boolean | Omit<AffixProps, 'offsetTop' | 'target' | 'children'>;
|
|
showInkInFixed?: boolean;
|
|
getContainer?: () => AnchorContainer;
|
|
/** Return customize highlight anchor */
|
|
getCurrentAnchor?: (activeLink: string) => string;
|
|
onClick?: (
|
|
e: React.MouseEvent<HTMLElement>,
|
|
link: { title: React.ReactNode; href: string },
|
|
) => void;
|
|
/** Scroll to target offset value, if none, it's offsetTop prop value or 0. */
|
|
targetOffset?: number;
|
|
/** Listening event when scrolling change active link */
|
|
onChange?: (currentActiveLink: string) => void;
|
|
items?: AnchorLinkItemProps[];
|
|
direction?: AnchorDirection;
|
|
replace?: boolean;
|
|
}
|
|
|
|
export interface AnchorState {
|
|
activeLink: null | string;
|
|
}
|
|
|
|
export interface AnchorDefaultProps extends AnchorProps {
|
|
prefixCls: string;
|
|
affix: boolean;
|
|
showInkInFixed: boolean;
|
|
getContainer: () => AnchorContainer;
|
|
}
|
|
|
|
export type AnchorDirection = 'vertical' | 'horizontal';
|
|
|
|
export interface AntAnchor {
|
|
registerLink: (link: string) => void;
|
|
unregisterLink: (link: string) => void;
|
|
activeLink: string | null;
|
|
scrollTo: (link: string) => void;
|
|
onClick?: (
|
|
e: React.MouseEvent<HTMLAnchorElement, MouseEvent>,
|
|
link: { title: React.ReactNode; href: string },
|
|
) => void;
|
|
direction: AnchorDirection;
|
|
}
|
|
|
|
const Anchor: React.FC<AnchorProps> = (props) => {
|
|
const {
|
|
rootClassName,
|
|
prefixCls: customPrefixCls,
|
|
className,
|
|
style,
|
|
offsetTop,
|
|
affix = true,
|
|
showInkInFixed = false,
|
|
children,
|
|
items,
|
|
direction: anchorDirection = 'vertical',
|
|
bounds,
|
|
targetOffset,
|
|
onClick,
|
|
onChange,
|
|
getContainer,
|
|
getCurrentAnchor,
|
|
replace,
|
|
} = props;
|
|
|
|
// =================== Warning =====================
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
const warning = devUseWarning('Anchor');
|
|
|
|
warning.deprecated(!children, 'Anchor children', 'items');
|
|
|
|
warning(
|
|
!(anchorDirection === 'horizontal' && items?.some((n) => 'children' in n)),
|
|
'usage',
|
|
'`Anchor items#children` is not supported when `Anchor` direction is horizontal.',
|
|
);
|
|
}
|
|
|
|
const [links, setLinks] = React.useState<string[]>([]);
|
|
const [activeLink, setActiveLink] = React.useState<string | null>(null);
|
|
const activeLinkRef = React.useRef<string | null>(activeLink);
|
|
|
|
const wrapperRef = React.useRef<HTMLDivElement>(null);
|
|
const spanLinkNode = React.useRef<HTMLSpanElement>(null);
|
|
const animating = React.useRef<boolean>(false);
|
|
|
|
const { direction, anchor, getTargetContainer, getPrefixCls } =
|
|
React.useContext<ConfigConsumerProps>(ConfigContext);
|
|
|
|
const prefixCls = getPrefixCls('anchor', customPrefixCls);
|
|
|
|
const rootCls = useCSSVarCls(prefixCls);
|
|
const [wrapCSSVar, hashId, cssVarCls] = useStyle(prefixCls, rootCls);
|
|
|
|
const getCurrentContainer = getContainer ?? getTargetContainer ?? getDefaultContainer;
|
|
|
|
const dependencyListItem: React.DependencyList[number] = JSON.stringify(links);
|
|
|
|
const registerLink = useEvent<AntAnchor['registerLink']>((link) => {
|
|
if (!links.includes(link)) {
|
|
setLinks((prev) => [...prev, link]);
|
|
}
|
|
});
|
|
|
|
const unregisterLink = useEvent<AntAnchor['unregisterLink']>((link) => {
|
|
if (links.includes(link)) {
|
|
setLinks((prev) => prev.filter((i) => i !== link));
|
|
}
|
|
});
|
|
|
|
const updateInk = () => {
|
|
const linkNode = wrapperRef.current?.querySelector<HTMLElement>(
|
|
`.${prefixCls}-link-title-active`,
|
|
);
|
|
if (linkNode && spanLinkNode.current) {
|
|
const { style: inkStyle } = spanLinkNode.current;
|
|
const horizontalAnchor = anchorDirection === 'horizontal';
|
|
inkStyle.top = horizontalAnchor ? '' : `${linkNode.offsetTop + linkNode.clientHeight / 2}px`;
|
|
inkStyle.height = horizontalAnchor ? '' : `${linkNode.clientHeight}px`;
|
|
inkStyle.left = horizontalAnchor ? `${linkNode.offsetLeft}px` : '';
|
|
inkStyle.width = horizontalAnchor ? `${linkNode.clientWidth}px` : '';
|
|
if (horizontalAnchor) {
|
|
scrollIntoView(linkNode, { scrollMode: 'if-needed', block: 'nearest' });
|
|
}
|
|
}
|
|
};
|
|
|
|
const getInternalCurrentAnchor = (_links: string[], _offsetTop = 0, _bounds = 5): string => {
|
|
const linkSections: Section[] = [];
|
|
const container = getCurrentContainer();
|
|
_links.forEach((link) => {
|
|
const sharpLinkMatch = sharpMatcherRegex.exec(link?.toString());
|
|
if (!sharpLinkMatch) {
|
|
return;
|
|
}
|
|
const target = document.getElementById(sharpLinkMatch[1]);
|
|
if (target) {
|
|
const top = getOffsetTop(target, container);
|
|
if (top <= _offsetTop + _bounds) {
|
|
linkSections.push({ link, top });
|
|
}
|
|
}
|
|
});
|
|
|
|
if (linkSections.length) {
|
|
const maxSection = linkSections.reduce((prev, curr) => (curr.top > prev.top ? curr : prev));
|
|
return maxSection.link;
|
|
}
|
|
return '';
|
|
};
|
|
|
|
const setCurrentActiveLink = useEvent((link: string) => {
|
|
// FIXME: Seems a bug since this compare is not equals
|
|
// `activeLinkRef` is parsed value which will always trigger `onChange` event.
|
|
if (activeLinkRef.current === link) {
|
|
return;
|
|
}
|
|
|
|
// https://github.com/ant-design/ant-design/issues/30584
|
|
const newLink = typeof getCurrentAnchor === 'function' ? getCurrentAnchor(link) : link;
|
|
setActiveLink(newLink);
|
|
activeLinkRef.current = newLink;
|
|
|
|
// onChange should respect the original link (which may caused by
|
|
// window scroll or user click), not the new link
|
|
onChange?.(link);
|
|
});
|
|
|
|
const handleScroll = React.useCallback(() => {
|
|
if (animating.current) {
|
|
return;
|
|
}
|
|
|
|
const currentActiveLink = getInternalCurrentAnchor(
|
|
links,
|
|
targetOffset !== undefined ? targetOffset : offsetTop || 0,
|
|
bounds,
|
|
);
|
|
|
|
setCurrentActiveLink(currentActiveLink);
|
|
}, [dependencyListItem, targetOffset, offsetTop]);
|
|
|
|
const handleScrollTo = React.useCallback<(link: string) => void>(
|
|
(link) => {
|
|
setCurrentActiveLink(link);
|
|
const sharpLinkMatch = sharpMatcherRegex.exec(link);
|
|
if (!sharpLinkMatch) {
|
|
return;
|
|
}
|
|
const targetElement = document.getElementById(sharpLinkMatch[1]);
|
|
if (!targetElement) {
|
|
return;
|
|
}
|
|
|
|
const container = getCurrentContainer();
|
|
const scrollTop = getScroll(container);
|
|
const eleOffsetTop = getOffsetTop(targetElement, container);
|
|
let y = scrollTop + eleOffsetTop;
|
|
y -= targetOffset !== undefined ? targetOffset : offsetTop || 0;
|
|
animating.current = true;
|
|
scrollTo(y, {
|
|
getContainer: getCurrentContainer,
|
|
callback() {
|
|
animating.current = false;
|
|
},
|
|
});
|
|
},
|
|
[targetOffset, offsetTop],
|
|
);
|
|
|
|
const wrapperClass = classNames(
|
|
hashId,
|
|
cssVarCls,
|
|
rootCls,
|
|
rootClassName,
|
|
`${prefixCls}-wrapper`,
|
|
{
|
|
[`${prefixCls}-wrapper-horizontal`]: anchorDirection === 'horizontal',
|
|
[`${prefixCls}-rtl`]: direction === 'rtl',
|
|
},
|
|
className,
|
|
anchor?.className,
|
|
);
|
|
|
|
const anchorClass = classNames(prefixCls, {
|
|
[`${prefixCls}-fixed`]: !affix && !showInkInFixed,
|
|
});
|
|
|
|
const inkClass = classNames(`${prefixCls}-ink`, {
|
|
[`${prefixCls}-ink-visible`]: activeLink,
|
|
});
|
|
|
|
const wrapperStyle: React.CSSProperties = {
|
|
maxHeight: offsetTop ? `calc(100vh - ${offsetTop}px)` : '100vh',
|
|
...anchor?.style,
|
|
...style,
|
|
};
|
|
|
|
const createNestedLink = (options?: AnchorLinkItemProps[]) =>
|
|
Array.isArray(options)
|
|
? options.map((item) => (
|
|
<AnchorLink replace={replace} {...item} key={item.key}>
|
|
{anchorDirection === 'vertical' && createNestedLink(item.children)}
|
|
</AnchorLink>
|
|
))
|
|
: null;
|
|
|
|
const anchorContent = (
|
|
<div ref={wrapperRef} className={wrapperClass} style={wrapperStyle}>
|
|
<div className={anchorClass}>
|
|
<span className={inkClass} ref={spanLinkNode} />
|
|
{'items' in props ? createNestedLink(items) : children}
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
React.useEffect(() => {
|
|
const scrollContainer = getCurrentContainer();
|
|
handleScroll();
|
|
scrollContainer?.addEventListener('scroll', handleScroll);
|
|
return () => {
|
|
scrollContainer?.removeEventListener('scroll', handleScroll);
|
|
};
|
|
}, [dependencyListItem]);
|
|
|
|
React.useEffect(() => {
|
|
if (typeof getCurrentAnchor === 'function') {
|
|
setCurrentActiveLink(getCurrentAnchor(activeLinkRef.current || ''));
|
|
}
|
|
}, [getCurrentAnchor]);
|
|
|
|
React.useEffect(() => {
|
|
updateInk();
|
|
}, [anchorDirection, getCurrentAnchor, dependencyListItem, activeLink]);
|
|
|
|
const memoizedContextValue = React.useMemo<AntAnchor>(
|
|
() => ({
|
|
registerLink,
|
|
unregisterLink,
|
|
scrollTo: handleScrollTo,
|
|
activeLink,
|
|
onClick,
|
|
direction: anchorDirection,
|
|
}),
|
|
[activeLink, onClick, handleScrollTo, anchorDirection],
|
|
);
|
|
|
|
const affixProps = affix && typeof affix === 'object' ? affix : undefined;
|
|
|
|
return wrapCSSVar(
|
|
<AnchorContext.Provider value={memoizedContextValue}>
|
|
{affix ? (
|
|
<Affix offsetTop={offsetTop} target={getCurrentContainer} {...affixProps}>
|
|
{anchorContent}
|
|
</Affix>
|
|
) : (
|
|
anchorContent
|
|
)}
|
|
</AnchorContext.Provider>,
|
|
);
|
|
};
|
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
Anchor.displayName = 'Anchor';
|
|
}
|
|
|
|
export default Anchor;
|