diff --git a/src/PCBViewer.tsx b/src/PCBViewer.tsx index d9cb64a..80ef800 100644 --- a/src/PCBViewer.tsx +++ b/src/PCBViewer.tsx @@ -23,6 +23,7 @@ type Props = { editEvents?: EditEvent[] initialState?: Partial onEditEventsChanged?: (editEvents: EditEvent[]) => void + focusOnHover?: boolean } export const PCBViewer = ({ @@ -34,6 +35,7 @@ export const PCBViewer = ({ allowEditing = true, editEvents: editEventsProp, onEditEventsChanged, + focusOnHover = false, }: Props) => { circuitJson ??= soup const { @@ -123,6 +125,7 @@ export const PCBViewer = ({ height={height} width={refDimensions.width} allowEditing={allowEditing} + focusOnHover={focusOnHover} cancelPanDrag={cancelPanDrag} onCreateEditEvent={onCreateEditEvent} onModifyEditEvent={onModifyEditEvent} diff --git a/src/components/CanvasElementsRenderer.tsx b/src/components/CanvasElementsRenderer.tsx index 5c881b3..5a62138 100644 --- a/src/components/CanvasElementsRenderer.tsx +++ b/src/components/CanvasElementsRenderer.tsx @@ -3,14 +3,14 @@ import { CanvasPrimitiveRenderer } from "./CanvasPrimitiveRenderer" import { pcb_port, type AnyCircuitElement } from "circuit-json" import { useMemo } from "react" import { convertElementToPrimitives } from "../lib/convert-element-to-primitive" -import { Matrix } from "transformation-matrix" -import { GridConfig, Primitive } from "lib/types" +import type { Matrix } from "transformation-matrix" +import type { GridConfig, Primitive } from "lib/types" import { MouseElementTracker } from "./MouseElementTracker" import { DimensionOverlay } from "./DimensionOverlay" import { ToolbarOverlay } from "./ToolbarOverlay" import { ErrorOverlay } from "./ErrorOverlay" import { EditPlacementOverlay } from "./EditPlacementOverlay" -import { EditEvent } from "lib/edit-events" +import type { EditEvent } from "lib/edit-events" import { EditTraceHintOverlay } from "./EditTraceHintOverlay" import { RatsNestOverlay } from "./RatsNestOverlay" import { getFullConnectivityMapFromCircuitJson } from "circuit-json-to-connectivity-map" @@ -23,7 +23,8 @@ export interface CanvasElementsRendererProps { height?: number grid?: GridConfig allowEditing: boolean - cancelPanDrag: Function + focusOnHover: boolean + cancelPanDrag: () => void onCreateEditEvent: (event: EditEvent) => void onModifyEditEvent: (event: Partial) => void } @@ -95,7 +96,7 @@ export const CanvasElementsRenderer = (props: CanvasElementsRendererProps) => { onCreateEditEvent={props.onCreateEditEvent as any} onModifyEditEvent={props.onModifyEditEvent as any} > - + diff --git a/src/components/DimensionOverlay.tsx b/src/components/DimensionOverlay.tsx index 7a8acf2..22225ac 100644 --- a/src/components/DimensionOverlay.tsx +++ b/src/components/DimensionOverlay.tsx @@ -1,13 +1,15 @@ import { zIndexMap } from "lib/util/z-index-map" import { useEffect, useRef, useState } from "react" -import { Matrix, applyToPoint, identity, inverse } from "transformation-matrix" +import type { Matrix } from "transformation-matrix" +import { applyToPoint, identity, inverse } from "transformation-matrix" interface Props { transform?: Matrix children: any + focusOnHover?: boolean } -export const DimensionOverlay = ({ children, transform }: Props) => { +export const DimensionOverlay = ({ children, transform, focusOnHover = false }: Props) => { if (!transform) transform = identity() const [dimensionToolVisible, setDimensionToolVisible] = useState(false) const [dimensionToolStretching, setDimensionToolStretching] = useState(false) @@ -83,10 +85,11 @@ export const DimensionOverlay = ({ children, transform }: Props) => { return (
tabIndex={0} style={{ position: "relative" }} onMouseEnter={() => { - if (containerRef.current) { + if (focusOnHover && containerRef.current) { containerRef.current.focus() } }} @@ -162,6 +165,7 @@ export const DimensionOverlay = ({ children, transform }: Props) => { {Math.abs(dStart.y - dEnd.y).toFixed(2)}
+ {/* biome-ignore lint/a11y/noSvgWithoutTitle: */} { {dEnd.x.toFixed(2)},{dEnd.y.toFixed(2)})
dist:{" "} {Math.sqrt( - Math.pow(dEnd.x - dStart.x, 2) + Math.pow(dEnd.y - dStart.y, 2), + ((dEnd.x - dStart.x) ** 2) + ((dEnd.y - dStart.y) ** 2) ).toFixed(2)} diff --git a/src/components/MouseElementTracker.tsx b/src/components/MouseElementTracker.tsx index 176afac..c354068 100644 --- a/src/components/MouseElementTracker.tsx +++ b/src/components/MouseElementTracker.tsx @@ -1,9 +1,9 @@ -import React, { useState } from "react" -import { useMemo } from "react" -import { Matrix, applyToPoint, inverse } from "transformation-matrix" -import { Primitive } from "lib/types" +import React, { useState, useMemo } from "react" +import type { Matrix } from "transformation-matrix" +import { applyToPoint, inverse } from "transformation-matrix" +import type { Primitive } from "lib/types" import { ElementOverlayBox } from "./ElementOverlayBox" -import { AnyCircuitElement } from "circuit-json" +import type { AnyCircuitElement } from "circuit-json" import { ifSetsMatchExactly } from "lib/util/if-sets-match-exactly" import { pointToSegmentDistance } from "@tscircuit/math-utils" @@ -43,7 +43,7 @@ export const MouseElementTracker = ({ // FANCY: If 2+ highlighted primitives inhabit the same space, give // them an incrementing same_space_index - let same_space_index = highlightedPrimitives.filter( + const same_space_index = highlightedPrimitives.filter( (hp) => screenPos.x === hp.screen_x && screenPos.y === hp.screen_y && @@ -96,8 +96,8 @@ export const MouseElementTracker = ({ if (!("x" in primitive && "y" in primitive)) continue // Handle different primitive types - let w = 0, - h = 0 + let w = 0 + let h = 0 if ("w" in primitive && "h" in primitive) { w = primitive.w diff --git a/src/stories/hover-behavior.stories.tsx b/src/stories/hover-behavior.stories.tsx new file mode 100644 index 0000000..852873b --- /dev/null +++ b/src/stories/hover-behavior.stories.tsx @@ -0,0 +1,95 @@ +import type { Meta, StoryObj } from "@storybook/react" +import { Circuit } from "@tscircuit/core" +import { PCBViewer } from "../PCBViewer" + +const HoverBehaviorDemo: React.FC<{ focusOnHover?: boolean }> = ({ + focusOnHover, +}) => { + const circuit = new Circuit() + + circuit.add( + + + + + + .right", "net.Ground"]} /> + .right", "net.Ground"]} /> + .left"]} /> + .left", ".R2 > .right"]} /> + , + ) + + const soup = circuit.getCircuitJson() + + return ( +
+
+ +
+
+ ) +} + +// Story with hover focus disabled (default behavior) +export const HoverDisabled: Story = { + args: { + focusOnHover: false, + }, + parameters: { + docs: { + description: { + story: + "Hover focus is disabled by default. Hovering over the PCB viewer will not cause the page to scroll or adjust focus position.", + }, + }, + }, +} + +// Story with hover focus enabled +export const HoverEnabled: Story = { + args: { + focusOnHover: true, + }, + parameters: { + docs: { + description: { + story: + "Hover focus is enabled. Hovering over the PCB viewer will cause it to receive focus and may adjust the scroll position.", + }, + }, + }, +} + +const meta: Meta = { + title: "Hover Behavior", + component: HoverBehaviorDemo, +} + +export default meta + +type Story = StoryObj