import * as React from 'react'; import isPlainObject from 'lodash/isPlainObject'; import debounce from 'lodash/debounce'; import SlickCarousel, { Settings } from '@ant-design/react-slick'; import classNames from 'classnames'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; export type CarouselEffect = 'scrollx' | 'fade'; export type DotPosition = 'top' | 'bottom' | 'left' | 'right'; // Carousel export interface CarouselProps extends Omit { effect?: CarouselEffect; style?: React.CSSProperties; prefixCls?: string; slickGoTo?: number; dotPosition?: DotPosition; children?: React.ReactNode; dots?: | boolean | { className?: string; }; } export default class Carousel extends React.Component { static defaultProps = { dots: true, arrows: false, draggable: false, }; innerSlider: any; private slick: any; constructor(props: CarouselProps) { super(props); this.onWindowResized = debounce(this.onWindowResized, 500, { leading: false, }); } componentDidMount() { const { autoplay } = this.props; if (autoplay) { window.addEventListener('resize', this.onWindowResized); } // https://github.com/ant-design/ant-design/issues/7191 this.innerSlider = this.slick && this.slick.innerSlider; } componentDidUpdate(prevProps: CarouselProps) { if (React.Children.count(this.props.children) !== React.Children.count(prevProps.children)) { this.goTo(this.props.initialSlide || 0, false); } } componentWillUnmount() { const { autoplay } = this.props; if (autoplay) { window.removeEventListener('resize', this.onWindowResized); (this.onWindowResized as any).cancel(); } } getDotPosition(): DotPosition { const { dotPosition = 'bottom' } = this.props; return dotPosition; } saveSlick = (node: any) => { this.slick = node; }; onWindowResized = () => { // Fix https://github.com/ant-design/ant-design/issues/2550 const { autoplay } = this.props; if (autoplay && this.slick && this.slick.innerSlider && this.slick.innerSlider.autoPlay) { this.slick.innerSlider.autoPlay(); } }; next() { this.slick.slickNext(); } prev() { this.slick.slickPrev(); } goTo(slide: number, dontAnimate = false) { this.slick.slickGoTo(slide, dontAnimate); } renderCarousel = ({ getPrefixCls, direction }: ConfigConsumerProps) => { const props = { ...this.props, }; if (props.effect === 'fade') { props.fade = true; } const prefixCls = getPrefixCls('carousel', props.prefixCls); const dotsClass = 'slick-dots'; const dotPosition = this.getDotPosition(); props.vertical = dotPosition === 'left' || dotPosition === 'right'; const enableDots = props.dots === true || isPlainObject(props.dots); const dsClass = classNames( dotsClass, `${dotsClass}-${dotPosition || 'bottom'}`, typeof props.dots === 'boolean' ? false : props.dots?.className, ); const className = classNames(prefixCls, { [`${prefixCls}-rtl`]: direction === 'rtl', [`${prefixCls}-vertical`]: props.vertical, }); return (
); }; render() { return {this.renderCarousel}; } }