import * as React from 'react'; import classNames from 'classnames'; import type { SegmentedLabeledOption as RcSegmentedLabeledOption, SegmentedProps as RCSegmentedProps, SegmentedValue as RcSegmentedValue, SegmentedRawOption, } from 'rc-segmented'; import RcSegmented from 'rc-segmented'; import { ConfigContext } from '../config-provider'; import useSize from '../config-provider/hooks/useSize'; import type { SizeType } from '../config-provider/SizeContext'; import useStyle from './style'; export type { SegmentedValue } from 'rc-segmented'; interface SegmentedLabeledOptionWithoutIcon extends RcSegmentedLabeledOption { label: RcSegmentedLabeledOption['label']; } interface SegmentedLabeledOptionWithIcon extends Omit, 'label'> { label?: RcSegmentedLabeledOption['label']; /** Set icon for Segmented item */ icon: React.ReactNode; } function isSegmentedLabeledOptionWithIcon( option: SegmentedRawOption | SegmentedLabeledOptionWithIcon | SegmentedLabeledOptionWithoutIcon, ): option is SegmentedLabeledOptionWithIcon { return typeof option === 'object' && !!(option as SegmentedLabeledOptionWithIcon)?.icon; } export type SegmentedLabeledOption = | SegmentedLabeledOptionWithIcon | SegmentedLabeledOptionWithoutIcon; export type SegmentedOptions = (T | SegmentedLabeledOption)[]; export interface SegmentedProps extends Omit, 'size' | 'options'> { rootClassName?: string; options: SegmentedOptions; /** Option to fit width to its parent's width */ block?: boolean; /** Option to control the display size */ size?: SizeType; vertical?: boolean; } const InternalSegmented = React.forwardRef((props, ref) => { const { prefixCls: customizePrefixCls, className, rootClassName, block, options = [], size: customSize = 'middle', style, vertical, ...restProps } = props; const { getPrefixCls, direction, segmented } = React.useContext(ConfigContext); const prefixCls = getPrefixCls('segmented', customizePrefixCls); // Style const [wrapCSSVar, hashId, cssVarCls] = useStyle(prefixCls); // ===================== Size ===================== const mergedSize = useSize(customSize); // syntactic sugar to support `icon` for Segmented Item const extendedOptions = React.useMemo( () => options.map((option) => { if (isSegmentedLabeledOptionWithIcon(option)) { const { icon, label, ...restOption } = option; return { ...restOption, label: ( <> {icon} {label && {label}} ), }; } return option; }), [options, prefixCls], ); const cls = classNames( className, rootClassName, segmented?.className, { [`${prefixCls}-block`]: block, [`${prefixCls}-sm`]: mergedSize === 'small', [`${prefixCls}-lg`]: mergedSize === 'large', [`${prefixCls}-vertical`]: vertical, }, hashId, cssVarCls, ); const mergedStyle: React.CSSProperties = { ...segmented?.style, ...style }; return wrapCSSVar( , ); }); const Segmented = InternalSegmented as (( props: SegmentedProps & React.RefAttributes, ) => ReturnType) & Pick; if (process.env.NODE_ENV !== 'production') { Segmented.displayName = 'Segmented'; } export default Segmented;