Files
La-Fabrik/src/components/game/GameMap.tsx
T
math-pixel bfe8c49323 fix :editor
2026-04-27 17:25:56 +02:00

120 lines
3.1 KiB
TypeScript

import { useEffect, useState, useMemo, useRef } from "react";
import { useGLTF } from "@react-three/drei";
import * as THREE from "three";
import { useOctreeGraphNode } from "@/hooks/useOctreeGraphNode";
import type { OctreeReadyHandler } from "@/types/3d";
interface MapNode {
name: string;
type: string;
position: [number, number, number];
rotation: [number, number, number];
scale: [number, number, number];
}
const MAP_JSON_PATH = "/map.json";
interface GameMapProps {
onOctreeReady: OctreeReadyHandler;
}
export function GameMap({ onOctreeReady }: GameMapProps): React.JSX.Element {
const [mapNodes, setMapNodes] = useState<MapNode[]>([]);
const [availableModels, setAvailableModels] = useState<Set<string>>(
new Set(),
);
const [isLoading, setIsLoading] = useState(true);
const groupRef = useRef<THREE.Group>(null);
useOctreeGraphNode(groupRef, onOctreeReady);
useEffect(() => {
const loadMap = async () => {
try {
const nodesResponse = await fetch(MAP_JSON_PATH);
if (!nodesResponse.ok) {
console.warn("map.json not found");
setIsLoading(false);
return;
}
const nodes: MapNode[] = await nodesResponse.json();
setMapNodes(nodes);
const uniqueModelNames = [...new Set(nodes.map((n) => n.name))];
const available = new Set<string>();
for (const modelName of uniqueModelNames) {
try {
const modelUrl = `/models/${modelName}/model.gltf`;
const modelResponse = await fetch(modelUrl);
const contentType = modelResponse.headers.get("content-type") || "";
if (contentType.includes("gltf") || contentType.includes("model")) {
available.add(modelName);
}
} catch {
/* empty */
}
}
setAvailableModels(available);
} catch (error) {
console.error("Error loading map:", error);
} finally {
setIsLoading(false);
}
};
loadMap();
}, []);
const nodesToRender = useMemo(() => {
return mapNodes.filter((node) => availableModels.has(node.name));
}, [mapNodes, availableModels]);
if (isLoading) {
return <></>;
}
return (
<group ref={groupRef}>
{nodesToRender.map((node, index) => (
<ModelInstance key={index} node={node} />
))}
</group>
);
}
function ModelInstance({ node }: { node: MapNode }): React.JSX.Element {
const modelPath = `/models/${node.name}/model.gltf`;
const groupRef = useRef<THREE.Group>(null);
const { scene } = useGLTF(modelPath);
useEffect(() => {
if (groupRef.current) {
groupRef.current.position.set(...node.position);
groupRef.current.rotation.set(...node.rotation);
groupRef.current.scale.set(...node.scale);
}
}, [
node.position[0],
node.position[1],
node.position[2],
node.rotation[0],
node.rotation[1],
node.rotation[2],
node.scale[0],
node.scale[1],
node.scale[2],
]);
return (
<primitive
ref={groupRef}
object={scene}
position={node.position}
rotation={node.rotation}
scale={node.scale}
/>
);
}