ant-design/components/carousel/index.tsx
Tom Xu 16093118a4
refactor(carousel): rewirte with hook (#27694)
* refactor(Carousel): rewirte with hook

* add coverage
2020-11-11 17:23:05 +08:00

124 lines
3.4 KiB
TypeScript

import * as React from 'react';
import debounce from 'lodash/debounce';
import SlickCarousel, { Settings } from '@ant-design/react-slick';
import classNames from 'classnames';
import { ConfigContext } from '../config-provider';
export type CarouselEffect = 'scrollx' | 'fade';
export type DotPosition = 'top' | 'bottom' | 'left' | 'right';
// Carousel
export interface CarouselProps extends Omit<Settings, 'dots' | 'dotsClass'> {
effect?: CarouselEffect;
style?: React.CSSProperties;
prefixCls?: string;
slickGoTo?: number;
dotPosition?: DotPosition;
children?: React.ReactNode;
dots?:
| boolean
| {
className?: string;
};
}
const Carousel = React.forwardRef<unknown, CarouselProps>(
({ dots = true, arrows = false, draggable = false, dotPosition = 'bottom', ...props }, ref) => {
const { getPrefixCls, direction } = React.useContext(ConfigContext);
const slickRef = React.useRef<any>();
const goTo = (slide: number, dontAnimate = false) => {
slickRef.current.slickGoTo(slide, dontAnimate);
};
React.useImperativeHandle(
ref,
() => ({
goTo,
autoPlay: slickRef.current.innerSlider.autoPlay,
innerSlider: slickRef.current.innerSlider,
prev: slickRef.current.slickPrev,
next: slickRef.current.slickNext,
}),
[slickRef.current],
);
React.useEffect(() => {
const func = () => {
// Fix https://github.com/ant-design/ant-design/issues/2550
const { autoplay } = props;
if (
autoplay &&
slickRef.current &&
slickRef.current.innerSlider &&
slickRef.current.innerSlider.autoPlay
) {
slickRef.current.innerSlider.autoPlay();
}
};
const onWindowResized = debounce(func, 500, {
leading: false,
});
if (props.autoplay) {
window.addEventListener('resize', onWindowResized);
}
return () => {
if (props.autoplay) {
window.removeEventListener('resize', onWindowResized);
(onWindowResized as any).cancel();
}
};
}, [slickRef.current, props.autoplay]);
const prevCount = React.useRef(React.Children.count(props.children));
React.useEffect(() => {
if (prevCount.current !== React.Children.count(props.children)) {
goTo(props.initialSlide || 0, false);
prevCount.current = React.Children.count(props.children);
}
}, [props.children]);
const newProps = {
...props,
};
if (newProps.effect === 'fade') {
newProps.fade = true;
}
const prefixCls = getPrefixCls('carousel', newProps.prefixCls);
const dotsClass = 'slick-dots';
newProps.vertical = dotPosition === 'left' || dotPosition === 'right';
const enableDots = !!dots;
const dsClass = classNames(
dotsClass,
`${dotsClass}-${dotPosition}`,
typeof dots === 'boolean' ? false : dots?.className,
);
const className = classNames(prefixCls, {
[`${prefixCls}-rtl`]: direction === 'rtl',
[`${prefixCls}-vertical`]: newProps.vertical,
});
return (
<div className={className}>
<SlickCarousel
ref={slickRef}
{...newProps}
dots={enableDots}
dotsClass={dsClass}
arrows={arrows}
draggable={draggable}
/>
</div>
);
},
);
export default Carousel;