|
<!--////////////////////////////////////////////////////////////////////////////////////////
|
|
/// ///
|
|
/// Example Using Three.js Library, HTML, CSS & JavaScript ///
|
|
// 3D Interactive Web Apps & Games 2021-2024 ///
|
|
/// Contact Shane Brumback https://www.shanebrumback.com ///
|
|
/// Send a message if you have questions about this code ///
|
|
/// I am a freelance developer. I develop any and all web. ///
|
|
/// Apps Websites 3D 2D CMS Systems etc. Contact me anytime :) ///
|
|
/// ///
|
|
////////////////////////////////////////////////////////////////////////////////////////////-->
|
|
|
|
|
|
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Three.js Examples - Comet Tail Particle System</title>
|
|
<style>
|
|
body {
|
|
color: white;
|
|
text-align: center;
|
|
margin: 0;
|
|
background-color: black
|
|
}
|
|
|
|
a {
|
|
text-decoration: none;
|
|
color: white;
|
|
}
|
|
|
|
h1 {
|
|
padding: 10px;
|
|
}
|
|
</style>
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<a href="http://www.shanebrumback.com/threejs-examples/comet-particle-system.html">
|
|
<h1>Three.js Examples - Comet Tail Particle System</h1>
|
|
</a>
|
|
<!--Load the latest Version of Three.js from CDN-->
|
|
<script src="https://cdn.jsdelivr.net/npm/three@latest/build/three.min.js"></script>
|
|
<script src="https://cdn.jsdelivr.net/npm/three@latest/examples/js/controls/OrbitControls.js"></script>
|
|
|
|
<script type="module">
|
|
|
|
function init() {
|
|
// Define global variables
|
|
let scene, camera;
|
|
|
|
// Set up Three.js scene
|
|
scene = new THREE.Scene();
|
|
|
|
// Set up the camera
|
|
camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
|
|
camera.position.x = 20;
|
|
camera.position.z = 20;
|
|
camera.position.y = 5;
|
|
|
|
// Set up the renderer and configure settings
|
|
let renderer = new THREE.WebGLRenderer({ antialias: true, alpha: true });
|
|
renderer.setPixelRatio(window.devicePixelRatio);
|
|
renderer.setSize(window.innerWidth, window.innerHeight);
|
|
renderer.toneMapping = THREE.ReinhardToneMapping;
|
|
renderer.setClearColor(0x000000); // Set background color to black
|
|
renderer.domElement.style.position = 'fixed';
|
|
renderer.domElement.style.zIndex = '-1';
|
|
renderer.domElement.style.left = '0';
|
|
renderer.domElement.style.top = '0';
|
|
document.body.appendChild(renderer.domElement);
|
|
|
|
camera.aspect = window.innerWidth / window.innerHeight;
|
|
camera.updateProjectionMatrix();
|
|
scene.scale.normalize().multiplyScalar(1);
|
|
|
|
// Create the ambient light
|
|
const ambientLight = new THREE.AmbientLight(0xffffff, 10); // Parameters: color, intensity
|
|
scene.add(ambientLight);
|
|
|
|
|
|
// Set up the orbital controls
|
|
const controls = new THREE.OrbitControls(camera, renderer.domElement);
|
|
controls.dampingFactor = 0.1; // Reduce camera damping for smoother movement
|
|
controls.autoRotate = true; // Make the camera rotate sinuously around the spheres
|
|
|
|
window.addEventListener('resize', onWindowResize);
|
|
|
|
function onWindowResize() {
|
|
// Update camera aspect ratio and renderer size on window resize
|
|
camera.aspect = window.innerWidth / window.innerHeight;
|
|
camera.updateProjectionMatrix();
|
|
renderer.setSize(window.innerWidth, window.innerHeight);
|
|
}
|
|
|
|
// Create a grid helper
|
|
var grid = new THREE.GridHelper(100, 40);
|
|
scene.add(grid);
|
|
|
|
// Create the spotlight with shadows
|
|
const spotLight = new THREE.SpotLight(0xffffff);
|
|
spotLight.position.set(10, 20, 30);
|
|
spotLight.castShadow = true;
|
|
scene.add(spotLight);
|
|
|
|
// Create the sphere
|
|
let sphereGeometry = new THREE.SphereGeometry(1.5, 32, 32);
|
|
let sphereMaterial = new THREE.MeshPhongMaterial({ color: 'white' });
|
|
let sphere = new THREE.Mesh(sphereGeometry, sphereMaterial);
|
|
|
|
// Create the particle system
|
|
const particleCount = 2500;
|
|
let sphereRadius = 1;
|
|
const particleGeometry = new THREE.BufferGeometry();
|
|
const particlePositions = new Float32Array(particleCount * 3);
|
|
const particleColors = new Float32Array(particleCount * 3);
|
|
|
|
for (let i = 0; i < particleCount; i++) {
|
|
// Generate random values within the sphere-shaped area
|
|
const r = sphereRadius * Math.sqrt(Math.random());
|
|
const theta = Math.random() * Math.PI * 2;
|
|
const phi = Math.acos(2 * Math.random() - 1);
|
|
|
|
// Convert from spherical to Cartesian coordinates
|
|
const x = r * Math.sin(phi) * Math.cos(theta);
|
|
const y = r * Math.sin(phi) * Math.sin(theta);
|
|
const z = r * Math.cos(phi);
|
|
|
|
// Set the positions and colors for each particle
|
|
particlePositions[i * 3] = x;
|
|
particlePositions[i * 3 + 1] = y;
|
|
particlePositions[i * 3 + 2] = z;
|
|
particleColors[i * 3] = 1.0;
|
|
particleColors[i * 3 + 1] = 0.5;
|
|
particleColors[i * 3 + 2] = 0.2;
|
|
}
|
|
|
|
// Add the position and color attributes to the geometry
|
|
particleGeometry.setAttribute('position', new THREE.BufferAttribute(particlePositions, 3));
|
|
particleGeometry.setAttribute('color', new THREE.BufferAttribute(particleColors, 3));
|
|
|
|
// Create the particle material and system
|
|
const particleMaterial = new THREE.PointsMaterial({ size: .10, vertexColors: THREE.VertexColors });
|
|
const particleSystem = new THREE.Points(particleGeometry, particleMaterial);
|
|
|
|
// Create a group to hold the particle system and sphere
|
|
let cometGroup = new THREE.Group();
|
|
cometGroup.add(particleSystem);
|
|
cometGroup.add(sphere);
|
|
scene.add(cometGroup);
|
|
|
|
const animate = function () {
|
|
requestAnimationFrame(animate);
|
|
|
|
for (let i = 0; i < particleCount; i++) {
|
|
// Move the particles down the -z axis
|
|
particlePositions[i * 3 + 2] -= 0.4;
|
|
|
|
// Randomly move some particles out a greater distance from the center
|
|
if (Math.random() < 0.009) {
|
|
particlePositions[i * 3] += (Math.random() * 2 - 1) * sphereRadius * 4;
|
|
particlePositions[i * 3 + 1] += (Math.random() * 2 - 1) * sphereRadius * 4;
|
|
}
|
|
|
|
if (particlePositions[i * 3 + 2] < -25) {
|
|
// Randomly reset some of the particles back to the sphere area
|
|
if (Math.random() < 0.01) {
|
|
particlePositions[i * 3] = (Math.random() * 1.5 - .75) * sphereRadius * 1.5;
|
|
particlePositions[i * 3 + 1] = (Math.random() * 1.5 - .75) * sphereRadius * 1.5;
|
|
particlePositions[i * 3 + 2] = (Math.random() * 1.5 - .75) * sphereRadius * 1.5;
|
|
}
|
|
}
|
|
|
|
// Check if the particles should move randomly or not
|
|
if (Math.random() < 0.05) {
|
|
// Move the particles randomly within the larger sphere area
|
|
particlePositions[i * 3] += (Math.random() * 2 - 1) * sphereRadius * 2;
|
|
particlePositions[i * 3 + 1] += (Math.random() * 2 - 1) * sphereRadius * 2;
|
|
particlePositions[i * 3 + 2] += (Math.random() * 2 - 1) * sphereRadius * 2;
|
|
}
|
|
|
|
// Update particle colors based on position
|
|
if (Math.sqrt(particlePositions[i * 3] * particlePositions[i * 3] + particlePositions[i * 3 + 1] * particlePositions[i * 3 + 1] + particlePositions[i * 3 + 2] * particlePositions[i * 3 + 2]) < sphereRadius) {
|
|
particleColors[i * 3] = 1.0;
|
|
particleColors[i * 3 + 1] = 1.0;
|
|
particleColors[i * 3 + 2] = 1.0;
|
|
} else {
|
|
particleColors[i * 3] = 0.0;
|
|
particleColors[i * 3 + 1] = 0.0;
|
|
particleColors[i * 3 + 2] = 1.0;
|
|
}
|
|
|
|
// Update particle color attribute
|
|
particleGeometry.attributes.color.setXYZ(i, particleColors[i * 3], particleColors[i * 3 + 1], particleColors[i * 3 + 2]);
|
|
}
|
|
|
|
particleGeometry.attributes.position.needsUpdate = true;
|
|
|
|
controls.update();
|
|
|
|
renderer.render(scene, camera);
|
|
};
|
|
|
|
animate();
|
|
}
|
|
|
|
init();
|
|
|
|
|
|
</script>
|
|
</body>
|
|
</html> |