? Stunning 3D Fish Animation | Interactive Underwater Scene with Three.js

Experience an interactive 3D fish animation built with Three.js & WebGL! Watch realistic koi fish swim through an immersive underwater world with dynamic lighting and floating lanterns. Perfect for web animations, aquarium simulations, and educational applications.

Sunday, March 9, 2025
? Stunning 3D Fish Animation | Interactive Underwater Scene with Three.js

3D Fish Animation Using Three.js – A Stunning Underwater Experience

Project Overview

This 3D Fish Animation project is an interactive web-based experience created using Three.js, a powerful JavaScript library for rendering 3D graphics. The animation features beautifully simulated koi fish swimming underwater, along with mesmerizing lanterns that add depth and ambiance to the scene. The script dynamically renders objects, applies movement physics, and incorporates shaders to create a realistic and immersive underwater world.

Key Features

Realistic 3D Fish Animation – The fish models follow a smooth Catmull-Rom curve path, mimicking natural swimming movements.
Dynamic Lantern Effects – Floating lanterns with shader-based lighting effects enhance the underwater ambiance.
Custom Shader Implementation – Uses vertex and fragment shaders to simulate light flickering and realistic fish reflections.
Orbit Controls for User Interaction – Allows users to zoom, pan, and rotate the scene effortlessly.
Performance Optimized – Efficient use of InstancedBufferGeometry for handling multiple objects smoothly.
WebGL Rendering – Powered by Three.js WebGLRenderer, ensuring high performance across modern browsers.

How It Works

  1. Scene Setup: A Three.js scene, camera, and renderer are initialized to create a 3D environment.
  2. Floating Lanterns: Custom geometry and shaders simulate flickering lanterns that float dynamically.
  3. Fish Movement: A Catmull-Rom curve path is used for realistic fish swimming, with movement driven by STL models and shader-based transformations.
  4. Lighting and Effects: Custom shader effects control the glowing lanterns, fish shading, and light scattering, giving the scene an authentic underwater feel.
  5. Interactivity: Users can navigate the scene using OrbitControls, zoom in on the fish, and explore the lantern-lit underwater world.

SEO-Optimized Benefits

? Web-Based & Interactive: This animation is fully browser-compatible, making it accessible on any device without additional plugins.
? Fast Loading & Performance Optimized: Uses efficient rendering techniques to ensure smooth animations even on low-end devices.
? Engaging User Experience: A beautifully crafted real-time 3D visualization that increases user engagement and session duration.
? Ideal for Educational & Entertainment Purposes: Can be used for interactive websites, aquarium simulations, games, and educational applications.
? Keyword-Rich Optimization: "3D Fish Animation," "Three.js Underwater Scene," "WebGL 3D Graphics," "Interactive Aquarium Simulation" – Perfect for boosting SEO rankings.

Use Cases

Interactive Websites – Enhance user experience with engaging 3D visuals.
Aquarium Simulations – Ideal for creating virtual fish tanks.
Game Development – Can be integrated into 3D game environments.
Educational Applications – Helps students and professionals learn about WebGL, shaders, and 3D rendering techniques.

Conclusion

This Three.js 3D Fish Animation project is a visually stunning and technically sophisticated web-based experience. It combines advanced geometry processing, custom shaders, and physics-based animation to create a realistic underwater world. Whether for entertainment, education, or website enhancement, this interactive animation is a perfect blend of art and technology.

? Explore the magic of 3D fish animation today! ??✨


? Want to see the animation live? Try it now & experience the magic of 3D fish animation! ?✨

HTML Code

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <title>Fish Animation</title>
  <link rel="stylesheet" href="./style.css">

</head>

<body>
  <!-- partial:index.partial.html -->
  <script src="https://cdn.jsdelivr.net/npm/three@0.115.0/build/three.min.js"></script>
  <script src="https://cdn.jsdelivr.net/npm/three@0.115.0/examples/js/utils/BufferGeometryUtils.js"></script>
  <script src="https://cdn.jsdelivr.net/npm/three@0.115.0/examples/js/controls/OrbitControls.js"></script>
  <script src="https://cdn.jsdelivr.net/npm/three@0.115.0/examples/js/loaders/STLLoader.js"></script>
  <!-- partial -->
  <script src="./script.js"></script>

</body>

</html>

CSS Code

body {
  overflow: hidden;
  margin: 0;
}

JavaScript Code

var scene = new THREE.Scene();
var camera = new THREE.PerspectiveCamera(60, innerWidth / innerHeight, 0.1, 500);
camera.position.set(0, -25, 80);
var renderer = new THREE.WebGLRenderer({antialias: true});
renderer.setClearColor(0x181005);
renderer.setSize(innerWidth, innerHeight);
document.body.appendChild(renderer.domElement);

var controls = new THREE.OrbitControls(camera, renderer.domElement);
controls.maxDistance = 150;

// lanterns

let geoms = [];
let pts = [
  new THREE.Vector2(0, 1. - 0),
  new THREE.Vector2(0.25, 1. - 0),
  new THREE.Vector2(0.25, 1. - 0.125),
  new THREE.Vector2(0.45, 1. - 0.125),
  new THREE.Vector2(0.45, 1. - 0.95)
];
var geom = new THREE.LatheBufferGeometry(pts, 20);
geoms.push(geom);

var geomLight = new THREE.CylinderBufferGeometry(0.1, 0.1, 0.05, 20);
//geomLight.rotateX(Math.PI * 0.5);
geoms.push(geomLight);

var fullGeom = THREE.BufferGeometryUtils.mergeBufferGeometries(geoms);

var instGeom = new THREE.InstancedBufferGeometry().copy(fullGeom);

var num = 500;
let instPos = []; //3
let instSpeed = []; //1
let instLight = []; // 2 (initial intensity, frequency)
for (let i = 0; i < num; i++){
  instPos.push( Math.random() - 0.5, Math.random() - 0.5, Math.random() - 0.5 );
  instSpeed.push( Math.random() * 0.25 + 1);
  instLight.push( Math.PI + (Math.PI * Math.random()), Math.random() + 5);
}
instGeom.setAttribute("instPos", new THREE.InstancedBufferAttribute(new Float32Array(instPos), 3));
instGeom.setAttribute("instSpeed", new THREE.InstancedBufferAttribute(new Float32Array(instSpeed), 1));
instGeom.setAttribute("instLight", new THREE.InstancedBufferAttribute(new Float32Array(instPos), 2));

var mat = new THREE.ShaderMaterial({
  uniforms: {
    uTime: {value: 0},
    uLight: {value: new THREE.Color("red").multiplyScalar(1.5)},
    uColor: {value: new THREE.Color("maroon").multiplyScalar(1)},
    uFire: {value: new THREE.Color(1, 0.75, 0)}
  },
  vertexShader:`
    uniform float uTime;

    attribute vec3 instPos;
    attribute float instSpeed;
    attribute vec2 instLight;

    varying vec2 vInstLight;
    varying float vY;
    
    void main() {
      
      vInstLight = instLight;
      vY = position.y;

      vec3 pos = vec3(position) * 2.;
      vec3 iPos = instPos * 200.;
      
      iPos.xz += vec2(
        cos(instLight.x + instLight.y * uTime),
        sin(instLight.x + instLight.y * uTime * fract(sin(instLight.x)))
      );

      iPos.y = mod(iPos.y + 100. + (uTime * instSpeed), 200.) - 100.;
      pos += iPos;
      gl_Position = projectionMatrix * modelViewMatrix * vec4(pos,1.0);
    }
`,
  fragmentShader: `
    uniform float uTime;
    uniform vec3 uLight;
    uniform vec3 uColor;
    uniform vec3 uFire;

    varying vec2 vInstLight;
    varying float vY;
    
    void main() {
      
      vec3 col = vec3(0);
      float t = vInstLight.x + (vInstLight.y * uTime * 10.);
      float ts = sin(t * 3.14) * 0.5 + 0.5;
      float tc = cos(t * 2.7) * 0.5 + 0.5;
      float f = smoothstep(0.12, 0.12 + (ts + tc) * 0.25, vY);
      float li = (0.5 + smoothstep(0., 1., ts * ts + tc * tc) * 0.5);
      col = mix(uLight * li, uColor * (0.75 + li * 0.25), f);

      col = mix(col, uFire, step(vY, 0.05) * (0.75 + li * 0.25));

      gl_FragColor = vec4(col, 1);
    }
`,
  side: THREE.DoubleSide
});

var lantern = new THREE.Mesh(instGeom, mat);
scene.add(lantern);

// Koi

let oUs = [];

let loader = new THREE.STLLoader();
//https://clara.io/view/b47726c8-02cf-4eb5-b275-d9b2be591bad
loader.load("https://cywarr.github.io/small-shop/fish.stl", objGeom => {
  console.log(objGeom);
  //objGeom.rotateX(-MathPI * 0.5);

  // path
  let baseVector = new THREE.Vector3(40, 0, 0);
  let axis = new THREE.Vector3(0, 1, 0);
  let cPts = [];
  let cSegments = 6;
  let cStep = Math.PI * 2 / cSegments;
  for (let i = 0; i < cSegments; i++){
    cPts.push(
      new THREE.Vector3().copy(baseVector)
      //.setLength(35 + (Math.random() - 0.5) * 5)
      .applyAxisAngle(axis, cStep * i).setY(THREE.MathUtils.randFloat(-10, 10))
    );
  }
  let curve = new THREE.CatmullRomCurve3(cPts);
  curve.closed = true;

  console.log(curve);

  let numPoints = 511;
  let cPoints = curve.getSpacedPoints(numPoints);
  let cObjects = curve.computeFrenetFrames(numPoints, true);
  console.log(cObjects);
  let pGeom = new THREE.BufferGeometry().setFromPoints(cPoints);
  let pMat = new THREE.LineBasicMaterial({color: "yellow"});
  let pathLine = new THREE.Line(pGeom, pMat);
  //scene.add(pathLine);

  // data texture
  let data = [];
  cPoints.forEach( v => { data.push(v.x, v.y, v.z);} );
  cObjects.binormals.forEach( v => { data.push(v.x, v.y, v.z);} );
  cObjects.normals.forEach( v => { data.push(v.x, v.y, v.z);} );
  cObjects.tangents.forEach( v => { data.push(v.x, v.y, v.z);} );

  let dataArray = new Float32Array(data);

  let tex = new THREE.DataTexture(dataArray, numPoints + 1, 4, THREE.RGBFormat, THREE.FloatType);
  tex.magFilter = THREE.NearestFilter;
  console.log(tex);
  
  objGeom.center();
  objGeom.rotateX(-Math.PI * 0.5);
  objGeom.scale(0.5, 0.5, 0.5);
  let objBox = new THREE.Box3().setFromBufferAttribute(objGeom.getAttribute("position"));
  let objSize = new THREE.Vector3();
  objBox.getSize(objSize);
  //objGeom.translate(0, 0, objBox.z);

  objUniforms = {
    uSpatialTexture: {value: tex},
    uTextureSize: {value: new THREE.Vector2(numPoints + 1, 4)},
    uTime: {value: 0},
    uLengthRatio: {value: objSize.z / curve.cacheArcLengths[200]}, // more or less real lenght along the path
    uObjSize: {value: objSize} // lenght
  }
  oUs.push(objUniforms);

  let objMat = new THREE.MeshBasicMaterial({color: 0xff6600, wireframe: true});
  objMat.onBeforeCompile = shader => {
    shader.uniforms.uSpatialTexture = objUniforms.uSpatialTexture;
    shader.uniforms.uTextureSize = objUniforms.uTextureSize;
    shader.uniforms.uTime = objUniforms.uTime;
    shader.uniforms.uLengthRatio = objUniforms.uLengthRatio;
    shader.uniforms.uObjSize = objUniforms.uObjSize;

    shader.vertexShader = `
      uniform sampler2D uSpatialTexture;
      uniform vec2 uTextureSize;
      uniform float uTime;
      uniform float uLengthRatio;
      uniform vec3 uObjSize;

      struct splineData {
        vec3 point;
        vec3 binormal;
        vec3 normal;
      };

      splineData getSplineData(float t){
        float step = 1. / uTextureSize.y;
        float halfStep = step * 0.5;
        splineData sd;
        sd.point    = texture2D(uSpatialTexture, vec2(t, step * 0. + halfStep)).rgb;
        sd.binormal = texture2D(uSpatialTexture, vec2(t, step * 1. + halfStep)).rgb;
        sd.normal   = texture2D(uSpatialTexture, vec2(t, step * 2. + halfStep)).rgb;
        return sd;
      }
  ` + shader.vertexShader;
    shader.vertexShader = shader.vertexShader.replace(
      `#include <begin_vertex>`,
      `#include <begin_vertex>

      vec3 pos = position;

      float wStep = 1. / uTextureSize.x;
      float hWStep = wStep * 0.5;

      float d = pos.z / uObjSize.z;
      float t = fract((uTime * 0.1) + (d * uLengthRatio));
      float numPrev = floor(t / wStep);
      float numNext = numPrev + 1.;
      //numNext = numNext > (uTextureSize.x - 1.) ? 0. : numNext;
      float tPrev = numPrev * wStep + hWStep;
      float tNext = numNext * wStep + hWStep;
      //float tDiff = tNext - tPrev;
      splineData splinePrev = getSplineData(tPrev);
      splineData splineNext = getSplineData(tNext);

      float f = (t - tPrev) / wStep;
      vec3 P = mix(splinePrev.point, splineNext.point, f);
      vec3 B = mix(splinePrev.binormal, splineNext.binormal, f);
      vec3 N = mix(splinePrev.normal, splineNext.normal, f);

      transformed = P + (N * pos.x) + (B * pos.y);
  `
    );
    //console.log(shader.vertexShader);
  }
  let obj = new THREE.Mesh(objGeom, objMat);
  scene.add(obj);
});


var clock = new THREE.Clock();

renderer.setAnimationLoop(() => {
  let t = clock.getElapsedTime();
  mat.uniforms.uTime.value = t;
  oUs.forEach(ou => {ou.uTime.value = t;});
  renderer.render(scene, camera);
});

PHP Code

NO Need

? FAQs ?

1️⃣ What is a 3D fish animation in Three.js? ??

A 3D fish animation in Three.js is a web-based simulation where realistic koi fish ? swim dynamically in an underwater scene ?. It uses custom shaders, smooth motion paths, and WebGL rendering for a lifelike experience.

2️⃣ How does Three.js help in creating 3D fish animations? ??

Three.js provides a powerful WebGL-based rendering engine ? that supports custom geometry, shaders, and animation paths, allowing developers to create interactive 3D fish simulations ? that run smoothly in browsers.

3️⃣ Can I add a 3D fish animation to my website? ?✨

✅ Yes! You can integrate a Three.js-based 3D fish animation into your website to enhance user engagement ?. It’s lightweight, mobile-friendly, and works on all modern browsers without extra plugins.

4️⃣ Does 3D fish animation affect website performance? ⚡?

No, when optimized properly! ✅ Using instanced rendering, efficient shaders, and low-poly models, Three.js ensures fast loading and smooth animations ?️ without slowing down the website.

5️⃣ Why should I use a 3D fish animation on my website? ??

A 3D fish animation ? makes your website visually appealing ✨, increases session duration ⏳, boosts user interaction ?, and can improve SEO rankings ? by lowering bounce rates.

6️⃣ Is Three.js good for SEO and fast indexing? ??

✅ Yes! A well-optimized Three.js project can load fast, be indexed by Google, and appear in search results, especially if combined with proper meta tags, structured data, and mobile optimization ?.






Leave a Comment: 👇