Files
grafana/public/app/plugins/panel/canvas/components/connections/ConnectionAnchors.tsx

149 lines
4.6 KiB
TypeScript
Raw Normal View History

import { css } from '@emotion/css';
import React, { useRef } from 'react';
import { GrafanaTheme2 } from '@grafana/data';
import { useStyles2 } from '@grafana/ui';
import { ConnectionCoordinates } from 'app/features/canvas';
type Props = {
setRef: (anchorElement: HTMLDivElement) => void;
handleMouseLeave: (
event: React.MouseEvent<Element, MouseEvent> | React.FocusEvent<HTMLDivElement, Element>
) => boolean;
};
export const CONNECTION_ANCHOR_DIV_ID = 'connectionControl';
export const CONNECTION_ANCHOR_ALT = 'connection anchor';
export const CONNECTION_ANCHOR_HIGHLIGHT_OFFSET = 8;
const ANCHOR_PADDING = 3;
export const ConnectionAnchors = ({ setRef, handleMouseLeave }: Props) => {
const highlightEllipseRef = useRef<HTMLDivElement>(null);
const styles = useStyles2(getStyles);
const halfSize = 2.5;
const halfSizeHighlightEllipse = 5.5;
const anchorImage =
'';
const onMouseEnterAnchor = (event: React.MouseEvent) => {
if (!(event.target instanceof HTMLImageElement)) {
return;
}
if (highlightEllipseRef.current && event.target.style) {
highlightEllipseRef.current.style.display = 'block';
highlightEllipseRef.current.style.top = `calc(${event.target.style.top} - ${halfSizeHighlightEllipse}px + ${ANCHOR_PADDING}px)`;
highlightEllipseRef.current.style.left = `calc(${event.target.style.left} - ${halfSizeHighlightEllipse}px + ${ANCHOR_PADDING}px)`;
}
};
const onMouseLeaveHighlightElement = () => {
if (highlightEllipseRef.current) {
highlightEllipseRef.current.style.display = 'none';
}
};
const handleMouseLeaveAnchors = (
event: React.MouseEvent<Element, MouseEvent> | React.FocusEvent<HTMLDivElement, Element>
) => {
const didHideAnchors = handleMouseLeave(event);
if (didHideAnchors) {
onMouseLeaveHighlightElement();
}
};
// Unit is percentage from the middle of the element
// 0, 0 middle; -1, -1 bottom left; 1, 1 top right
const ANCHORS = [
{ x: -1, y: 1 },
{ x: -0.5, y: 1 },
{ x: 0, y: 1 },
{ x: 0.5, y: 1 },
{ x: 1, y: 1 },
{ x: 1, y: 0.5 },
{ x: 1, y: 0 },
{ x: 1, y: -0.5 },
{ x: 1, y: -1 },
{ x: 0.5, y: -1 },
{ x: 0, y: -1 },
{ x: -0.5, y: -1 },
{ x: -1, y: -1 },
{ x: -1, y: -0.5 },
{ x: -1, y: 0 },
{ x: -1, y: 0.5 },
];
const generateAnchors = (anchors: ConnectionCoordinates[] = ANCHORS) => {
return anchors.map((anchor) => {
const id = `${anchor.x},${anchor.y}`;
// Convert anchor coords to relative percentage
const style = {
top: `calc(${-anchor.y * 50 + 50}% - ${halfSize}px - ${ANCHOR_PADDING}px)`,
left: `calc(${anchor.x * 50 + 50}% - ${halfSize}px - ${ANCHOR_PADDING}px)`,
};
return (
<img
id={id}
key={id}
alt={CONNECTION_ANCHOR_ALT}
className={styles.anchor}
style={style}
src={anchorImage}
onMouseEnter={onMouseEnterAnchor}
/>
);
});
};
return (
<div className={styles.root} ref={setRef}>
<div className={styles.mouseoutDiv} onMouseOut={handleMouseLeaveAnchors} onBlur={handleMouseLeaveAnchors} />
<div
id={CONNECTION_ANCHOR_DIV_ID}
ref={highlightEllipseRef}
className={styles.highlightElement}
onMouseLeave={onMouseLeaveHighlightElement}
/>
{generateAnchors()}
</div>
);
};
const getStyles = (theme: GrafanaTheme2) => ({
root: css({
position: 'absolute',
display: 'none',
}),
mouseoutDiv: css({
position: 'absolute',
margin: '-30px',
width: 'calc(100% + 60px)',
height: 'calc(100% + 60px)',
}),
anchor: css({
padding: `${ANCHOR_PADDING}px`,
position: 'absolute',
cursor: 'cursor',
width: `calc(5px + 2 * ${ANCHOR_PADDING}px)`,
height: `calc(5px + 2 * ${ANCHOR_PADDING}px)`,
zIndex: 100,
pointerEvents: 'auto',
}),
highlightElement: css({
backgroundColor: '#00ff00',
opacity: 0.3,
position: 'absolute',
cursor: 'cursor',
pointerEvents: 'auto',
width: '16px',
height: '16px',
borderRadius: theme.shape.radius.circle,
display: 'none',
zIndex: 110,
}),
});