Add workaround for token bar token placement

This commit is contained in:
Mitchell McCaffrey 2021-05-28 15:11:46 +10:00
parent 57aafce938
commit f84baab6fb

View File

@ -1,4 +1,4 @@
import React, { useState } from "react"; import React, { useState, useRef } from "react";
import { createPortal } from "react-dom"; import { createPortal } from "react-dom";
import { Box, Flex } from "theme-ui"; import { Box, Flex } from "theme-ui";
import SimpleBar from "simplebar-react"; import SimpleBar from "simplebar-react";
@ -14,6 +14,7 @@ import useSetting from "../../hooks/useSetting";
import { useTokenData } from "../../contexts/TokenDataContext"; import { useTokenData } from "../../contexts/TokenDataContext";
import { useAuth } from "../../contexts/AuthContext"; import { useAuth } from "../../contexts/AuthContext";
import { useMapStage } from "../../contexts/MapStageContext";
import { createTokenState } from "../../helpers/token"; import { createTokenState } from "../../helpers/token";
@ -24,18 +25,56 @@ function TokenBar({ onMapTokenStateCreate }) {
const [dragId, setDragId] = useState(); const [dragId, setDragId] = useState();
const mapStageRef = useMapStage();
// Use a ref to the drag overlay to get it's position on dragEnd
// TODO: use active.rect when dnd-kit bug is fixed
// https://github.com/clauderic/dnd-kit/issues/238
const dragOverlayRef = useRef();
function handleDragStart({ active }) { function handleDragStart({ active }) {
setDragId(active.id); setDragId(active.id);
} }
function handleDragEnd({ active }) { function handleDragEnd({ active }) {
setDragId(null); setDragId(null);
const token = tokensById[active.id];
console.log("Drag", active); const mapStage = mapStageRef.current;
if (token) { const dragOverlay = dragOverlayRef.current;
// TODO: Get drag position if (mapStage && dragOverlay) {
const tokenState = createTokenState(token, { x: 0, y: 0 }, userId); const mapImage = mapStage.findOne("#mapImage");
onMapTokenStateCreate(tokenState); const map = document.querySelector(".map");
const mapRect = map.getBoundingClientRect();
const dragRect = dragOverlay.getBoundingClientRect();
const dragPosition = {
x: dragRect.left + dragRect.width / 2,
y: dragRect.top + dragRect.height / 2,
};
// Check map bounds
if (dragPosition.x < mapRect.left || dragPosition.x > mapRect.right) {
return;
}
// Convert relative to map rect
const mapPosition = {
x: dragPosition.x - mapRect.left,
y: dragPosition.y - mapRect.top,
};
// Convert relative to map image
const transform = mapImage.getAbsoluteTransform().copy().invert();
const relativePosition = transform.point(mapPosition);
const normalizedPosition = {
x: relativePosition.x / mapImage.width(),
y: relativePosition.y / mapImage.height(),
};
const token = tokensById[active.id];
if (token) {
const tokenState = createTokenState(token, normalizedPosition, userId);
onMapTokenStateCreate(tokenState);
}
} }
} }
@ -107,8 +146,19 @@ function TokenBar({ onMapTokenStateCreate }) {
<SelectTokensButton /> <SelectTokensButton />
</Flex> </Flex>
{createPortal( {createPortal(
<DragOverlay> <DragOverlay
{dragId && <TokenBarToken token={tokensById[dragId]} />} // Ensure a drop animation plays to allow us to get the position of the drag overlay in drag end
dropAnimation={{
dragSourceOpacity: 0,
duration: 1,
easing: "ease",
}}
>
{dragId && (
<div ref={dragOverlayRef}>
<TokenBarToken token={tokensById[dragId]} />
</div>
)}
</DragOverlay>, </DragOverlay>,
document.body document.body
)} )}