import {useEffect, useRef, useState} from 'react'; import styled from '@emotion/styled'; import {Button} from 'sentry/components/button'; import {IconPanel} from 'sentry/icons'; import {IconClose} from 'sentry/icons/iconClose'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import useKeyPress from 'sentry/utils/useKeyPress'; import {useLocalStorageState} from 'sentry/utils/useLocalStorageState'; import useOnClickOutside from 'sentry/utils/useOnClickOutside'; import SlideOverPanel from 'sentry/views/starfish/components/slideOverPanel'; type DetailProps = { children: React.ReactNode; detailKey?: string; onClose?: () => void; onOpen?: () => void; }; type DetailState = { collapsed: boolean; }; const SLIDEOUT_STORAGE_KEY = 'starfish-panel-slideout-direction'; export default function Detail({children, detailKey, onClose, onOpen}: DetailProps) { const [state, setState] = useState({collapsed: true}); const [slidePosition, setSlidePosition] = useLocalStorageState<'right' | 'bottom'>( SLIDEOUT_STORAGE_KEY, 'right' ); const escapeKeyPressed = useKeyPress('Escape'); // Any time the key prop changes (due to user interaction), we want to open the panel useEffect(() => { if (detailKey) { setState({collapsed: false}); } else { setState({collapsed: true}); } }, [detailKey]); const panelRef = useRef(null); useOnClickOutside(panelRef, () => { if (!state.collapsed) { onClose?.(); setState({collapsed: true}); } }); useEffect(() => { if (escapeKeyPressed) { if (!state.collapsed) { onClose?.(); setState({collapsed: true}); } } // eslint-disable-next-line react-hooks/exhaustive-deps }, [escapeKeyPressed]); return ( } onClick={() => { setSlidePosition('bottom'); }} /> } onClick={() => { setSlidePosition('right'); }} /> } onClick={() => { setState({collapsed: true}); onClose?.(); }} /> {children} ); } const CloseButton = styled(Button)` color: ${p => p.theme.gray300}; &:hover { color: ${p => p.theme.gray400}; } z-index: 100; `; const PanelButton = styled(Button)` color: ${p => p.theme.gray300}; &:hover { color: ${p => p.theme.gray400}; } z-index: 100; `; const CloseButtonWrapper = styled('div')` justify-content: flex-end; display: flex; padding: ${space(2)}; `; const DetailWrapper = styled('div')` padding: 0 ${space(4)}; `;