mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
149 lines
4.6 KiB
TypeScript
149 lines
4.6 KiB
TypeScript
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,
|
|
}),
|
|
});
|