mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-10 14:10:15 +08:00
338 lines
8.6 KiB
TypeScript
338 lines
8.6 KiB
TypeScript
import * as React from 'react';
|
|
import * as ReactDOM from 'react-dom';
|
|
import * as PropTypes from 'prop-types';
|
|
import classNames from 'classnames';
|
|
import addEventListener from 'rc-util/lib/Dom/addEventListener';
|
|
import Affix from '../affix';
|
|
import AnchorLink from './AnchorLink';
|
|
import { ConfigConsumer, ConfigConsumerProps } from '../config-provider';
|
|
import scrollTo from '../_util/scrollTo';
|
|
import getScroll from '../_util/getScroll';
|
|
|
|
function getDefaultContainer() {
|
|
return window;
|
|
}
|
|
|
|
function getOffsetTop(element: HTMLElement, container: AnchorContainer): number {
|
|
if (!element) {
|
|
return 0;
|
|
}
|
|
|
|
if (!element.getClientRects().length) {
|
|
return 0;
|
|
}
|
|
|
|
const rect = element.getBoundingClientRect();
|
|
|
|
if (rect.width || rect.height) {
|
|
if (container === window) {
|
|
container = element.ownerDocument!.documentElement!;
|
|
return rect.top - container.clientTop;
|
|
}
|
|
return rect.top - (container as HTMLElement).getBoundingClientRect().top;
|
|
}
|
|
|
|
return rect.top;
|
|
}
|
|
|
|
const sharpMatcherRegx = /#([^#]+)$/;
|
|
|
|
type Section = {
|
|
link: string;
|
|
top: number;
|
|
};
|
|
|
|
export type AnchorContainer = HTMLElement | Window;
|
|
|
|
export interface AnchorProps {
|
|
prefixCls?: string;
|
|
className?: string;
|
|
style?: React.CSSProperties;
|
|
children?: React.ReactNode;
|
|
offsetTop?: number;
|
|
bounds?: number;
|
|
affix?: boolean;
|
|
showInkInFixed?: boolean;
|
|
getContainer?: () => AnchorContainer;
|
|
/** Return customize highlight anchor */
|
|
getCurrentAnchor?: () => 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;
|
|
}
|
|
|
|
export interface AnchorState {
|
|
activeLink: null | string;
|
|
}
|
|
|
|
export interface AnchorDefaultProps extends AnchorProps {
|
|
prefixCls: string;
|
|
affix: boolean;
|
|
showInkInFixed: boolean;
|
|
getContainer: () => AnchorContainer;
|
|
}
|
|
|
|
export interface AntAnchor {
|
|
registerLink: (link: string) => void;
|
|
unregisterLink: (link: string) => void;
|
|
activeLink: string | null;
|
|
scrollTo: (link: string) => void;
|
|
onClick?: (
|
|
e: React.MouseEvent<HTMLElement>,
|
|
link: { title: React.ReactNode; href: string },
|
|
) => void;
|
|
}
|
|
|
|
export default class Anchor extends React.Component<AnchorProps, AnchorState> {
|
|
static Link: typeof AnchorLink;
|
|
|
|
static defaultProps = {
|
|
affix: true,
|
|
showInkInFixed: false,
|
|
getContainer: getDefaultContainer,
|
|
};
|
|
|
|
static childContextTypes = {
|
|
antAnchor: PropTypes.object,
|
|
};
|
|
|
|
state = {
|
|
activeLink: null,
|
|
};
|
|
|
|
private inkNode: HTMLSpanElement;
|
|
|
|
// scroll scope's container
|
|
private scrollContainer: HTMLElement | Window;
|
|
|
|
private links: string[] = [];
|
|
|
|
private scrollEvent: any;
|
|
|
|
private animating: boolean;
|
|
|
|
private prefixCls?: string;
|
|
|
|
getChildContext() {
|
|
const antAnchor: AntAnchor = {
|
|
registerLink: (link: string) => {
|
|
if (!this.links.includes(link)) {
|
|
this.links.push(link);
|
|
}
|
|
},
|
|
unregisterLink: (link: string) => {
|
|
const index = this.links.indexOf(link);
|
|
if (index !== -1) {
|
|
this.links.splice(index, 1);
|
|
}
|
|
},
|
|
activeLink: this.state.activeLink,
|
|
scrollTo: this.handleScrollTo,
|
|
onClick: this.props.onClick,
|
|
};
|
|
return { antAnchor };
|
|
}
|
|
|
|
componentDidMount() {
|
|
const { getContainer } = this.props as AnchorDefaultProps;
|
|
this.scrollContainer = getContainer();
|
|
this.scrollEvent = addEventListener(this.scrollContainer, 'scroll', this.handleScroll);
|
|
this.handleScroll();
|
|
}
|
|
|
|
componentDidUpdate() {
|
|
if (this.scrollEvent) {
|
|
const { getContainer } = this.props as AnchorDefaultProps;
|
|
const currentContainer = getContainer();
|
|
if (this.scrollContainer !== currentContainer) {
|
|
this.scrollContainer = currentContainer;
|
|
this.scrollEvent.remove();
|
|
this.scrollEvent = addEventListener(this.scrollContainer, 'scroll', this.handleScroll);
|
|
this.handleScroll();
|
|
}
|
|
}
|
|
this.updateInk();
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
if (this.scrollEvent) {
|
|
this.scrollEvent.remove();
|
|
}
|
|
}
|
|
|
|
getCurrentAnchor(offsetTop = 0, bounds = 5): string {
|
|
const { getCurrentAnchor } = this.props;
|
|
|
|
if (typeof getCurrentAnchor === 'function') {
|
|
return getCurrentAnchor();
|
|
}
|
|
|
|
const activeLink = '';
|
|
if (typeof document === 'undefined') {
|
|
return activeLink;
|
|
}
|
|
|
|
const linkSections: Array<Section> = [];
|
|
const { getContainer } = this.props as AnchorDefaultProps;
|
|
const container = getContainer();
|
|
this.links.forEach(link => {
|
|
const sharpLinkMatch = sharpMatcherRegx.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 '';
|
|
}
|
|
|
|
handleScrollTo = (link: string) => {
|
|
const { offsetTop, getContainer, targetOffset } = this.props as AnchorDefaultProps;
|
|
|
|
this.setCurrentActiveLink(link);
|
|
const container = getContainer();
|
|
const scrollTop = getScroll(container, true);
|
|
const sharpLinkMatch = sharpMatcherRegx.exec(link);
|
|
if (!sharpLinkMatch) {
|
|
return;
|
|
}
|
|
const targetElement = document.getElementById(sharpLinkMatch[1]);
|
|
if (!targetElement) {
|
|
return;
|
|
}
|
|
|
|
const eleOffsetTop = getOffsetTop(targetElement, container);
|
|
let y = scrollTop + eleOffsetTop;
|
|
y -= targetOffset !== undefined ? targetOffset : offsetTop || 0;
|
|
this.animating = true;
|
|
|
|
scrollTo(y, {
|
|
callback: () => {
|
|
this.animating = false;
|
|
},
|
|
getContainer,
|
|
});
|
|
};
|
|
|
|
saveInkNode = (node: HTMLSpanElement) => {
|
|
this.inkNode = node;
|
|
};
|
|
|
|
setCurrentActiveLink = (link: string) => {
|
|
const { activeLink } = this.state;
|
|
const { onChange } = this.props;
|
|
|
|
if (activeLink !== link) {
|
|
this.setState({
|
|
activeLink: link,
|
|
});
|
|
if (onChange) {
|
|
onChange(link);
|
|
}
|
|
}
|
|
};
|
|
|
|
handleScroll = () => {
|
|
if (this.animating) {
|
|
return;
|
|
}
|
|
const { offsetTop, bounds, targetOffset } = this.props;
|
|
const currentActiveLink = this.getCurrentAnchor(
|
|
targetOffset !== undefined ? targetOffset : offsetTop || 0,
|
|
bounds,
|
|
);
|
|
this.setCurrentActiveLink(currentActiveLink);
|
|
};
|
|
|
|
updateInk = () => {
|
|
if (typeof document === 'undefined') {
|
|
return;
|
|
}
|
|
const { prefixCls } = this;
|
|
const anchorNode = ReactDOM.findDOMNode(this) as Element;
|
|
const linkNode = anchorNode.getElementsByClassName(`${prefixCls}-link-title-active`)[0];
|
|
if (linkNode) {
|
|
this.inkNode.style.top = `${(linkNode as any).offsetTop + linkNode.clientHeight / 2 - 4.5}px`;
|
|
}
|
|
};
|
|
|
|
renderAnchor = ({ getPrefixCls, direction }: ConfigConsumerProps) => {
|
|
const {
|
|
prefixCls: customizePrefixCls,
|
|
className = '',
|
|
style,
|
|
offsetTop,
|
|
affix,
|
|
showInkInFixed,
|
|
children,
|
|
getContainer,
|
|
} = this.props;
|
|
const { activeLink } = this.state;
|
|
|
|
const prefixCls = getPrefixCls('anchor', customizePrefixCls);
|
|
|
|
// To support old version react.
|
|
// Have to add prefixCls on the instance.
|
|
// https://github.com/facebook/react/issues/12397
|
|
this.prefixCls = prefixCls;
|
|
|
|
const inkClass = classNames(`${prefixCls}-ink-ball`, {
|
|
visible: activeLink,
|
|
});
|
|
|
|
const wrapperClass = classNames(className, `${prefixCls}-wrapper`, {
|
|
[`${prefixCls}-rtl`]: direction === 'rtl',
|
|
});
|
|
|
|
const anchorClass = classNames(prefixCls, {
|
|
fixed: !affix && !showInkInFixed,
|
|
});
|
|
|
|
const wrapperStyle = {
|
|
maxHeight: offsetTop ? `calc(100vh - ${offsetTop}px)` : '100vh',
|
|
...style,
|
|
};
|
|
|
|
const anchorContent = (
|
|
<div className={wrapperClass} style={wrapperStyle}>
|
|
<div className={anchorClass}>
|
|
<div className={`${prefixCls}-ink`}>
|
|
<span className={inkClass} ref={this.saveInkNode} />
|
|
</div>
|
|
{children}
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
return !affix ? (
|
|
anchorContent
|
|
) : (
|
|
<Affix offsetTop={offsetTop} target={getContainer}>
|
|
{anchorContent}
|
|
</Affix>
|
|
);
|
|
};
|
|
|
|
render() {
|
|
return <ConfigConsumer>{this.renderAnchor}</ConfigConsumer>;
|
|
}
|
|
}
|