176 lines
7.2 KiB
JavaScript
176 lines
7.2 KiB
JavaScript
"use client";
|
|
|
|
/* eslint-disable react/no-array-index-key */
|
|
import React, { useState } from 'react';
|
|
import classNames from 'classnames';
|
|
import ResizeObserver from 'rc-resize-observer';
|
|
import useEvent from "rc-util/es/hooks/useEvent";
|
|
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 { InternalPanel } from './Panel';
|
|
import SplitBar from './SplitBar';
|
|
import useStyle from './style';
|
|
const Splitter = props => {
|
|
const {
|
|
prefixCls: customizePrefixCls,
|
|
className,
|
|
style,
|
|
layout = 'horizontal',
|
|
children,
|
|
rootClassName,
|
|
onResizeStart,
|
|
onResize,
|
|
onResizeEnd,
|
|
lazy
|
|
} = props;
|
|
const {
|
|
getPrefixCls,
|
|
direction,
|
|
className: contextClassName,
|
|
style: contextStyle
|
|
} = useComponentConfig('splitter');
|
|
const prefixCls = getPrefixCls('splitter', customizePrefixCls);
|
|
const rootCls = useCSSVarCls(prefixCls);
|
|
const [wrapCSSVar, hashId, cssVarCls] = useStyle(prefixCls, rootCls);
|
|
// ======================== Direct ========================
|
|
const isVertical = layout === 'vertical';
|
|
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) {
|
|
process.env.NODE_ENV !== "production" ? warning(false, 'usage', 'When part of `Splitter.Panel` has `size`, `onResize` is required or change `size` to `defaultSize`.') : void 0;
|
|
}
|
|
}
|
|
// ====================== Container =======================
|
|
const [containerSize, setContainerSize] = useState();
|
|
const onContainerResize = 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, isRTL);
|
|
const [onOffsetStart, onOffsetUpdate, onOffsetEnd, onCollapse, movingIndex] = useResize(items, resizableInfos, itemPtgSizes, containerSize, updateSizes, isRTL);
|
|
// ======================== Events ========================
|
|
const onInternalResizeStart = useEvent(index => {
|
|
onOffsetStart(index);
|
|
onResizeStart === null || onResizeStart === void 0 ? void 0 : onResizeStart(itemPxSizes);
|
|
});
|
|
const onInternalResizeUpdate = useEvent((index, offset, lazyEnd) => {
|
|
const nextSizes = onOffsetUpdate(index, offset);
|
|
if (lazyEnd) {
|
|
onResizeEnd === null || onResizeEnd === void 0 ? void 0 : onResizeEnd(nextSizes);
|
|
} else {
|
|
onResize === null || onResize === void 0 ? void 0 : onResize(nextSizes);
|
|
}
|
|
});
|
|
const onInternalResizeEnd = useEvent(lazyEnd => {
|
|
onOffsetEnd();
|
|
if (!lazyEnd) {
|
|
onResizeEnd === null || onResizeEnd === void 0 ? void 0 : onResizeEnd(itemPxSizes);
|
|
}
|
|
});
|
|
const onInternalCollapse = useEvent((index, type) => {
|
|
var _a;
|
|
const nextSizes = onCollapse(index, type);
|
|
onResize === null || onResize === void 0 ? void 0 : onResize(nextSizes);
|
|
onResizeEnd === null || onResizeEnd === void 0 ? void 0 : onResizeEnd(nextSizes);
|
|
const collapsed = nextSizes.map(size => Math.abs(size) < Number.EPSILON);
|
|
(_a = props.onCollapse) === null || _a === void 0 ? void 0 : _a.call(props, collapsed, nextSizes);
|
|
});
|
|
// ======================== Styles ========================
|
|
const containerClassName = classNames(prefixCls, className, `${prefixCls}-${layout}`, {
|
|
[`${prefixCls}-rtl`]: isRTL
|
|
}, rootClassName, contextClassName, cssVarCls, rootCls, hashId);
|
|
// ======================== Render ========================
|
|
const maskCls = `${prefixCls}-mask`;
|
|
const stackSizes = React.useMemo(() => {
|
|
const mergedSizes = [];
|
|
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 = Object.assign(Object.assign({}, contextStyle), style);
|
|
return wrapCSSVar(/*#__PURE__*/React.createElement(ResizeObserver, {
|
|
onResize: onContainerResize
|
|
}, /*#__PURE__*/React.createElement("div", {
|
|
style: mergedStyle,
|
|
className: containerClassName
|
|
}, items.map((item, idx) => {
|
|
// Panel
|
|
const panel = /*#__PURE__*/React.createElement(InternalPanel, Object.assign({}, item, {
|
|
prefixCls: prefixCls,
|
|
size: panelSizes[idx]
|
|
}));
|
|
// Split Bar
|
|
let splitBar = 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 = /*#__PURE__*/React.createElement(SplitBar, {
|
|
lazy: lazy,
|
|
index: idx,
|
|
active: movingIndex === idx,
|
|
prefixCls: prefixCls,
|
|
vertical: isVertical,
|
|
resizable: resizableInfo.resizable,
|
|
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 /*#__PURE__*/React.createElement(React.Fragment, {
|
|
key: `split-panel-${idx}`
|
|
}, panel, splitBar);
|
|
}), typeof movingIndex === 'number' && (/*#__PURE__*/React.createElement("div", {
|
|
"aria-hidden": true,
|
|
className: classNames(maskCls, `${maskCls}-${layout}`)
|
|
})))));
|
|
};
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
Splitter.displayName = 'Splitter';
|
|
}
|
|
export default Splitter; |