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 getScroll from '../_util/getScroll'; import raf from 'raf'; 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; } function easeInOutCubic(t: number, b: number, c: number, d: number) { const cc = c - b; t /= d / 2; if (t < 1) { return (cc / 2) * t * t * t + b; } return (cc / 2) * ((t -= 2) * t * t + 2) + b; } const sharpMatcherRegx = /#([^#]+)$/; function scrollTo( href: string, offsetTop = 0, getContainer: () => AnchorContainer, callback = () => {}, ) { const container = getContainer(); const scrollTop = getScroll(container, true); const sharpLinkMatch = sharpMatcherRegx.exec(href); if (!sharpLinkMatch) { return; } const targetElement = document.getElementById(sharpLinkMatch[1]); if (!targetElement) { return; } const eleOffsetTop = getOffsetTop(targetElement, container); const targetScrollTop = scrollTop + eleOffsetTop - offsetTop; const startTime = Date.now(); const frameFunc = () => { const timestamp = Date.now(); const time = timestamp - startTime; const nextScrollTop = easeInOutCubic(time, scrollTop, targetScrollTop, 450); if (container === window) { window.scrollTo(window.pageXOffset, nextScrollTop); } else { (container as HTMLElement).scrollTop = nextScrollTop; } if (time < 450) { raf(frameFunc); } else { callback(); } }; raf(frameFunc); } 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; onClick?: ( e: React.MouseEvent, link: { title: React.ReactNode; href: 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, link: { title: React.ReactNode; href: string }, ) => void; } export default class Anchor extends React.Component { 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(); } componentWillUnmount() { if (this.scrollEvent) { this.scrollEvent.remove(); } } 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(); } handleScroll = () => { if (this.animating) { return; } const { activeLink } = this.state; const { offsetTop, bounds } = this.props; const currentActiveLink = this.getCurrentAnchor(offsetTop, bounds); if (activeLink !== currentActiveLink) { this.setState({ activeLink: currentActiveLink, }); } }; handleScrollTo = (link: string) => { const { offsetTop, getContainer } = this.props as AnchorDefaultProps; this.animating = true; this.setState({ activeLink: link }); scrollTo(link, offsetTop, getContainer, () => { this.animating = false; }); }; getCurrentAnchor(offsetTop = 0, bounds = 5): string { const activeLink = ''; if (typeof document === 'undefined') { return activeLink; } const linkSections: Array
= []; 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 ''; } updateInk = () => { if (typeof document === 'undefined') { return; } const prefixCls = this.prefixCls; 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`; } }; saveInkNode = (node: HTMLSpanElement) => { this.inkNode = node; }; renderAnchor = ({ getPrefixCls }: 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`); const anchorClass = classNames(prefixCls, { fixed: !affix && !showInkInFixed, }); const wrapperStyle = { maxHeight: offsetTop ? `calc(100vh - ${offsetTop}px)` : '100vh', ...style, }; const anchorContent = (
{children}
); return !affix ? ( anchorContent ) : ( {anchorContent} ); }; render() { return {this.renderAnchor}; } }