Files
ant-design/components/splitter/Splitter.tsx
lijianan d937a3930f refactor(Splitter): use genCssVar hook to generate CSS variables (#56555)
* refactor(Splitter): use genCssVar hook to generate CSS variables

* Update components/splitter/style/index.ts

Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com>
Signed-off-by: lijianan <574980606@qq.com>

---------

Signed-off-by: lijianan <574980606@qq.com>
Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com>
2026-01-10 11:03:22 +08:00

292 lines
9.0 KiB
TypeScript

/* eslint-disable react/no-array-index-key */
import React, { useState } from 'react';
import ResizeObserver from '@rc-component/resize-observer';
import { useEvent } from '@rc-component/util';
import { clsx } from 'clsx';
import { useMergeSemantic, useOrientation } from '../_util/hooks';
import type { GetProp } from '../_util/type';
import { devUseWarning } from '../_util/warning';
import { useComponentConfig } from '../config-provider/context';
import useCSSVarCls from '../config-provider/hooks/useCSSVarCls';
import useItems from './hooks/useItems';
import useResizable from './hooks/useResizable';
import useResize from './hooks/useResize';
import useSizes from './hooks/useSizes';
import type {
SplitterClassNamesType,
SplitterProps,
SplitterSemanticDraggerClassNames,
SplitterStylesType,
} from './interface';
import { InternalPanel } from './Panel';
import SplitBar from './SplitBar';
import useStyle from './style';
const Splitter: React.FC<React.PropsWithChildren<SplitterProps>> = (props) => {
const {
prefixCls: customizePrefixCls,
className,
classNames,
style,
styles,
layout,
orientation,
vertical,
children,
draggerIcon,
collapsibleIcon,
rootClassName,
onResizeStart,
onResize,
onResizeEnd,
lazy,
} = props;
const {
getPrefixCls,
direction,
className: contextClassName,
style: contextStyle,
classNames: contextClassNames,
styles: contextStyles,
} = useComponentConfig('splitter');
const prefixCls = getPrefixCls('splitter', customizePrefixCls);
const rootPrefixCls = getPrefixCls();
const rootCls = useCSSVarCls(prefixCls);
const [hashId, cssVarCls] = useStyle(prefixCls, rootCls);
// ======================== Direct ========================
const [mergedOrientation, isVertical] = useOrientation(orientation, vertical, layout);
const isRTL = direction === 'rtl';
const reverse = !isVertical && isRTL;
// ====================== Items Data ======================
const items = useItems(children);
// >>> Warning for uncontrolled
if (process.env.NODE_ENV !== 'production') {
const warning = devUseWarning('Splitter');
const existSize = items.some((item) => item.size !== undefined);
const existUndefinedSize = items.some((item) => item.size === undefined);
if (existSize && existUndefinedSize && !onResize) {
warning(
false,
'usage',
'When part of `Splitter.Panel` has `size`, `onResize` is required or change `size` to `defaultSize`.',
);
}
warning.deprecated(!layout, 'layout', 'orientation');
}
// ====================== Container =======================
const [containerSize, setContainerSize] = useState<number | undefined>();
const onContainerResize: GetProp<typeof ResizeObserver, 'onResize'> = (size) => {
const { offsetWidth, offsetHeight } = size;
const containerSize = isVertical ? offsetHeight : offsetWidth;
// Skip when container has no size, Such as nested in a hidden tab panel
// to fix: https://github.com/ant-design/ant-design/issues/51106
if (containerSize === 0) {
return;
}
setContainerSize(containerSize);
};
// ========================= Size =========================
const [panelSizes, itemPxSizes, itemPtgSizes, itemPtgMinSizes, itemPtgMaxSizes, updateSizes] =
useSizes(items, containerSize);
// ====================== Resizable =======================
const resizableInfos = useResizable(items, itemPxSizes, reverse);
const [onOffsetStart, onOffsetUpdate, onOffsetEnd, onCollapse, movingIndex] = useResize(
items,
resizableInfos,
itemPtgSizes,
containerSize,
updateSizes,
reverse,
);
// ======================== Events ========================
const onInternalResizeStart = useEvent((index: number) => {
onOffsetStart(index);
onResizeStart?.(itemPxSizes);
});
const onInternalResizeUpdate = useEvent((index: number, offset: number, lazyEnd?: boolean) => {
const nextSizes = onOffsetUpdate(index, offset);
if (lazyEnd) {
onResizeEnd?.(nextSizes);
} else {
onResize?.(nextSizes);
}
});
const onInternalResizeEnd = useEvent((lazyEnd?: boolean) => {
onOffsetEnd();
if (!lazyEnd) {
onResizeEnd?.(itemPxSizes);
}
});
const onInternalCollapse = useEvent((index: number, type: 'start' | 'end') => {
const nextSizes = onCollapse(index, type);
onResize?.(nextSizes);
onResizeEnd?.(nextSizes);
const collapsed = nextSizes.map((size) => Math.abs(size) < Number.EPSILON);
props.onCollapse?.(collapsed, nextSizes);
});
// =========== Merged Props for Semantic ==========
const mergedProps: SplitterProps = {
...props,
vertical: isVertical,
orientation: mergedOrientation,
};
// ======================== Styles ========================
const [mergedClassNames, mergedStyles] = useMergeSemantic<
SplitterClassNamesType,
SplitterStylesType,
SplitterProps
>(
[contextClassNames, classNames],
[contextStyles, styles],
{ props: mergedProps },
{
// Convert `classNames.dragger: 'a'` to
// `classNames.dragger: { default: 'a' }`
dragger: {
_default: 'default',
},
},
);
const containerClassName = clsx(
prefixCls,
className,
`${prefixCls}-${mergedOrientation}`,
{
[`${prefixCls}-rtl`]: isRTL,
},
rootClassName,
mergedClassNames.root,
contextClassName,
cssVarCls,
rootCls,
hashId,
);
// ======================== Render ========================
const maskCls = `${prefixCls}-mask`;
const stackSizes = React.useMemo(() => {
const mergedSizes: number[] = [];
let stack = 0;
const len = items.length;
for (let i = 0; i < len; i += 1) {
stack += itemPtgSizes[i];
mergedSizes.push(stack);
}
return mergedSizes;
}, [itemPtgSizes, items.length]);
const mergedStyle: React.CSSProperties = {
...mergedStyles.root,
...contextStyle,
...style,
};
return (
<ResizeObserver onResize={onContainerResize}>
<div style={mergedStyle} className={containerClassName}>
{items.map((item, idx) => {
const panelProps = {
...item,
className: clsx(mergedClassNames.panel, item.className),
style: { ...mergedStyles.panel, ...item.style },
};
// Panel
const panel = (
<InternalPanel {...panelProps} prefixCls={prefixCls} size={panelSizes[idx]} />
);
// Split Bar
let splitBar: React.ReactElement | null = null;
const resizableInfo = resizableInfos[idx];
if (resizableInfo) {
const ariaMinStart = (stackSizes[idx - 1] || 0) + itemPtgMinSizes[idx];
const ariaMinEnd = (stackSizes[idx + 1] || 100) - itemPtgMaxSizes[idx + 1];
const ariaMaxStart = (stackSizes[idx - 1] || 0) + itemPtgMaxSizes[idx];
const ariaMaxEnd = (stackSizes[idx + 1] || 100) - itemPtgMinSizes[idx + 1];
splitBar = (
<SplitBar
lazy={lazy}
index={idx}
active={movingIndex === idx}
prefixCls={prefixCls}
rootPrefixCls={rootPrefixCls}
vertical={isVertical}
resizable={resizableInfo.resizable}
draggerStyle={mergedStyles.dragger}
draggerClassName={mergedClassNames.dragger as SplitterSemanticDraggerClassNames}
draggerIcon={draggerIcon}
collapsibleIcon={collapsibleIcon}
ariaNow={stackSizes[idx] * 100}
ariaMin={Math.max(ariaMinStart, ariaMinEnd) * 100}
ariaMax={Math.min(ariaMaxStart, ariaMaxEnd) * 100}
startCollapsible={resizableInfo.startCollapsible}
endCollapsible={resizableInfo.endCollapsible}
showStartCollapsibleIcon={resizableInfo.showStartCollapsibleIcon}
showEndCollapsibleIcon={resizableInfo.showEndCollapsibleIcon}
onOffsetStart={onInternalResizeStart}
onOffsetUpdate={(index, offsetX, offsetY, lazyEnd) => {
let offset = isVertical ? offsetY : offsetX;
if (reverse) {
offset = -offset;
}
onInternalResizeUpdate(index, offset, lazyEnd);
}}
onOffsetEnd={onInternalResizeEnd}
onCollapse={onInternalCollapse}
containerSize={containerSize || 0}
/>
);
}
return (
<React.Fragment key={`split-panel-${idx}`}>
{panel}
{splitBar}
</React.Fragment>
);
})}
{/* Fake mask for cursor */}
{typeof movingIndex === 'number' && (
<div aria-hidden className={clsx(maskCls, `${maskCls}-${mergedOrientation}`)} />
)}
</div>
</ResizeObserver>
);
};
if (process.env.NODE_ENV !== 'production') {
Splitter.displayName = 'Splitter';
}
export default Splitter;