function drawSpheres(spheres, canvas, options = {}) {
const { width, height } = canvas;
const { background = "lightgray" } = options;
const renderer = new THREE.WebGLRenderer({ canvas, antialias: true });
renderer.setSize(width, height);
renderer.shadowMap.enabled = true;
renderer.shadowMap.type = THREE.PCFSoftShadowMap;
const scene = new THREE.Scene();
scene.background = new THREE.Color(background);
const camera = new THREE.PerspectiveCamera(40, width / height, 1, 10000);
camera.position.set(0, 2, 3);
const controls = new THREE.OrbitControls(camera, renderer.domElement);
const light = new THREE.DirectionalLight(0xffffff, 1);
light.position.set(10, 10, 10);
const ambLight = new THREE.AmbientLight(0xffffff, 0.5);
camera.add(light, ambLight);
light.castShadow = true;
scene.add(camera);
const materials = [];
const colors = [];
const sphereGeom = new THREE.SphereGeometry(1, 32, 16);
for (let { center, radius, color } of spheres) {
if (radius < 0) continue;
color = new THREE.Color(color || "white");
let colorIndex = colors.findIndex((c) => color.equals(c));
let mat;
if (colorIndex == -1) {
colors.push(color);
mat = new THREE.MeshStandardMaterial({
metalness: 0.5,
roughness: 0.8,
color
});
materials.push(mat);
} else mat = materials[colorIndex];
const mesh = new THREE.Mesh(sphereGeom, mat);
mesh.position.set(...center);
mesh.scale.set(radius, radius, radius);
mesh.castShadow = true;
mesh.receiveShadow = true;
scene.add(mesh);
}
return {
draw: function () {
renderer.render(scene, camera);
},
dispose: function () {
renderer.dispose();
controls.dispose();
}
};
}