import * as React from 'react'; import RcSlider from 'rc-slider/lib/Slider'; import RcRange from 'rc-slider/lib/Range'; import RcHandle from 'rc-slider/lib/Handle'; import classNames from 'classnames'; import { TooltipPlacement } from '../tooltip'; import SliderTooltip from './SliderTooltip'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; export interface SliderMarks { [key: number]: | React.ReactNode | { style: React.CSSProperties; label: React.ReactNode; }; } export type SliderValue = number | [number, number]; interface HandleGeneratorInfo { value: number; dragging: boolean; index: number; rest: any[]; } export type HandleGeneratorFn = (config: { tooltipPrefixCls?: string; prefixCls?: string; info: HandleGeneratorInfo; }) => React.ReactNode; export interface SliderProps { prefixCls?: string; tooltipPrefixCls?: string; range?: boolean; reverse?: boolean; min?: number; max?: number; step?: number | null; marks?: SliderMarks; dots?: boolean; value?: SliderValue; defaultValue?: SliderValue; included?: boolean; disabled?: boolean; vertical?: boolean; onChange?: (value: SliderValue) => void; onAfterChange?: (value: SliderValue) => void; tipFormatter?: null | ((value: number) => React.ReactNode); className?: string; id?: string; style?: React.CSSProperties; tooltipVisible?: boolean; tooltipPlacement?: TooltipPlacement; getTooltipPopupContainer?: (triggerNode: HTMLElement) => HTMLElement; } export interface SliderState { visibles: { [index: number]: boolean }; } export default class Slider extends React.Component { static defaultProps = { tipFormatter(value: number) { return typeof value === 'number' ? value.toString() : ''; }, }; rcSlider: any; constructor(props: SliderProps) { super(props); this.state = { visibles: {}, }; } toggleTooltipVisible = (index: number, visible: boolean) => { this.setState(({ visibles }) => ({ visibles: { ...visibles, [index]: visible, }, })); }; handleWithTooltip: HandleGeneratorFn = ({ tooltipPrefixCls, prefixCls, info: { value, dragging, index, ...restProps }, }) => { const { tipFormatter, tooltipVisible, tooltipPlacement, getTooltipPopupContainer, vertical, } = this.props; const { visibles } = this.state; const isTipFormatter = tipFormatter ? visibles[index] || dragging : false; const visible = tooltipVisible || (tooltipVisible === undefined && isTipFormatter); return ( document.body)} > this.toggleTooltipVisible(index, true)} onMouseLeave={() => this.toggleTooltipVisible(index, false)} /> ); }; saveSlider = (node: any) => { this.rcSlider = node; }; focus() { this.rcSlider.focus(); } blur() { this.rcSlider.blur(); } renderSlider = ({ getPrefixCls, direction }: ConfigConsumerProps) => { const { prefixCls: customizePrefixCls, tooltipPrefixCls: customizeTooltipPrefixCls, range, className, ...restProps } = this.props; const prefixCls = getPrefixCls('slider', customizePrefixCls); const tooltipPrefixCls = getPrefixCls('tooltip', customizeTooltipPrefixCls); const cls = classNames(className, { [`${prefixCls}-rtl`]: direction === 'rtl', }); // make reverse default on rtl direction if (direction === 'rtl' && !restProps.vertical) { restProps.reverse = !restProps.reverse; } if (range) { return ( this.handleWithTooltip({ tooltipPrefixCls, prefixCls, info, }) } prefixCls={prefixCls} tooltipPrefixCls={tooltipPrefixCls} /> ); } return ( this.handleWithTooltip({ tooltipPrefixCls, prefixCls, info, }) } prefixCls={prefixCls} tooltipPrefixCls={tooltipPrefixCls} /> ); }; render() { return {this.renderSlider}; } }