import * as React from 'react'; import classNames from 'classnames'; import RcSegmented from 'rc-segmented'; import type { SegmentedProps as RCSegmentedProps, SegmentedRawOption, SegmentedLabeledOption as RcSegmentedLabeledOption, } from 'rc-segmented'; import { ConfigContext } from '../config-provider'; import type { SizeType } from '../config-provider/SizeContext'; import SizeContext from '../config-provider/SizeContext'; export type { SegmentedValue } from 'rc-segmented'; interface SegmentedLabeledOptionWithoutIcon extends RcSegmentedLabeledOption { label: RcSegmentedLabeledOption['label']; } interface SegmentedLabeledOptionWithIcon extends Omit { 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 interface SegmentedProps extends Omit { options: (SegmentedRawOption | SegmentedLabeledOption)[]; /** Option to fit width to its parent's width */ block?: boolean; /** Option to control the display size */ size?: SizeType; } const Segmented = React.forwardRef((props, ref) => { const { prefixCls: customizePrefixCls, className, block, options, size: customSize = 'middle', ...restProps } = props; const { getPrefixCls, direction } = React.useContext(ConfigContext); const prefixCls = getPrefixCls('segmented', customizePrefixCls); // ===================== Size ===================== const size = React.useContext(SizeContext); const mergedSize = customSize || size; // 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], ); return ( ); }); Segmented.displayName = 'Segmented'; Segmented.defaultProps = { options: [], }; export default Segmented;