import {ReactNode, useCallback} from 'react';
import styled from '@emotion/styled';
import debounce from 'lodash/debounce';
import useSplitPanelTracking from 'sentry/utils/replays/hooks/useSplitPanelTracking';
import {useResizableDrawer} from 'sentry/utils/useResizableDrawer';
import SplitDivider from 'sentry/views/replays/detail/layout/splitDivider';
type Side = {
content: ReactNode;
default: number;
max: number;
min: number;
};
type Props =
| {
availableSize: number;
/**
* Content on the right side of the split
*/
left: Side;
/**
* Content on the left side of the split
*/
right: ReactNode;
}
| {
availableSize: number;
/**
* Content below the split
*/
bottom: ReactNode;
/**
* Content above of the split
*/
top: Side;
};
function SplitPanel(props: Props) {
const isLeftRight = 'left' in props;
const initialSize = isLeftRight ? props.left.default : props.top.default;
const min = isLeftRight ? props.left.min : props.top.min;
const max = isLeftRight ? props.left.max : props.top.max;
const {setStartPosition, logEndPosition} = useSplitPanelTracking({
slideDirection: isLeftRight ? 'leftright' : 'updown',
});
// eslint-disable-next-line react-hooks/exhaustive-deps
const onResize = useCallback(
debounce(newSize => logEndPosition(`${(newSize / props.availableSize) * 100}%`), 750),
[debounce, logEndPosition, props.availableSize]
);
const {
isHeld,
onDoubleClick,
onMouseDown: onDragStart,
size: containerSize,
} = useResizableDrawer({
direction: isLeftRight ? 'left' : 'down',
initialSize,
min,
onResize,
});
const sizePct = `${
(Math.min(containerSize, max) / props.availableSize) * 100
}%` as `${number}%`;
const onMouseDown = useCallback(
event => {
setStartPosition(sizePct);
onDragStart(event);
},
[setStartPosition, onDragStart, sizePct]
);
if (isLeftRight) {
const {left: a, right: b} = props;
return (
{a.content}
{b}
);
}
const {top: a, bottom: b} = props;
return (
{a.content}
{b}
);
}
const SplitPanelContainer = styled('div')<{
orientation: 'rows' | 'columns';
size: `${number}px` | `${number}%`;
}>`
width: 100%;
height: 100%;
position: relative;
display: grid;
overflow: auto;
grid-template-${p => p.orientation}: ${p => p.size} auto 1fr;
&.disable-iframe-pointer iframe {
pointer-events: none !important;
}
`;
const Panel = styled('div')`
overflow: hidden;
`;
export default SplitPanel;