Created
March 13, 2025 07:29
-
-
Save anon987654321/03af706a0303a9af4522f64d148ca030 to your computer and use it in GitHub Desktop.
WebGL Wireframe Tunnel – Custom Shaders, Perlin Noise and GSAP
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import * as THREE from "https://esm.sh/three"; | |
// Scene, Camera, and Renderer Setup | |
const scene = new THREE.Scene(); | |
const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000); | |
camera.position.z = 0; | |
const renderer = new THREE.WebGLRenderer({ antialias: true }); | |
renderer.setSize(window.innerWidth, window.innerHeight); | |
renderer.setClearColor(0x141414); | |
document.body.appendChild(renderer.domElement); | |
// Shader Uniforms | |
const uniforms = { | |
uSmoothness: { value: 1.0 }, | |
uGridDensity: { value: 26.0 }, | |
uNoiseScale: { value: 10.0 }, // Frequency of Perlin noise | |
uNoiseSpeed: { value: 0.5 }, // Speed of noise animation | |
uNoiseStrength: { value: 0.15 }, // Strength of noise effect | |
uEnableDisplacement: { value: true }, // Enable/disable displacement | |
uTime: { value: 0.0 }, // Time for noise animation | |
uWireColor: { value: new THREE.Color(0xFFFFFF) }, // Wireframe color | |
uBaseColor: { value: new THREE.Color(0x141414) } // Background color | |
}; | |
// Wireframe Shader Material with Toggleable Perlin Noise | |
const wireframeMaterial = new THREE.ShaderMaterial({ | |
uniforms: uniforms, | |
vertexShader: ` | |
varying vec2 vUv; | |
void main() { | |
vUv = uv; | |
gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0); | |
} | |
`, | |
fragmentShader: ` | |
uniform float uSmoothness; | |
uniform float uGridDensity; | |
uniform float uNoiseScale; | |
uniform float uNoiseSpeed; | |
uniform float uNoiseStrength; | |
uniform bool uEnableDisplacement; // Enable/disable flag | |
uniform float uTime; | |
uniform vec3 uWireColor; | |
uniform vec3 uBaseColor; | |
varying vec2 vUv; | |
// Simple Perlin Noise Function | |
float random(vec2 st) { | |
return fract(sin(dot(st.xy, vec2(12.9898, 78.233))) * 43758.5453123); | |
} | |
float noise(vec2 st) { | |
vec2 i = floor(st); | |
vec2 f = fract(st); | |
float a = random(i); | |
float b = random(i + vec2(1.0, 0.0)); | |
float c = random(i + vec2(0.0, 1.0)); | |
float d = random(i + vec2(1.0, 1.0)); | |
vec2 u = f * f * (3.0 - 2.0 * f); | |
return mix(a, b, u.x) + (c - a) * u.y * (1.0 - u.x) + (d - b) * u.x * u.y; | |
} | |
void main() { | |
// Generate grid lines | |
vec2 grid = abs(fract(vUv * uGridDensity - 0.5) - 0.5); | |
vec2 gridWidth = fwidth(vUv * uGridDensity); | |
float lineX = smoothstep(0.0, gridWidth.x * uSmoothness, grid.x); | |
float lineY = smoothstep(0.0, gridWidth.y * uSmoothness, grid.y); | |
float line = 1.0 - min(lineX, lineY); | |
// Perlin noise for displacement | |
float noiseValue = 0.0; | |
if (uEnableDisplacement) { | |
noiseValue = noise(vUv * uNoiseScale + uTime * uNoiseSpeed) * uNoiseStrength; | |
} | |
// Combine base color and wireframe with noise distortion | |
vec3 finalColor = mix(uBaseColor, uWireColor, line); | |
finalColor += noiseValue; // Add noise if enabled | |
gl_FragColor = vec4(finalColor, 1.0); | |
} | |
`, | |
side: THREE.BackSide | |
}); | |
// Tunnel Path and Tube | |
const path = new THREE.CatmullRomCurve3([ | |
new THREE.Vector3(0, 0, 0), | |
new THREE.Vector3(0, 0, -10), | |
new THREE.Vector3(3, 2, -20), | |
new THREE.Vector3(-3, -2, -30), | |
new THREE.Vector3(0, 0, -40), | |
new THREE.Vector3(2, 1, -50), | |
new THREE.Vector3(-2, -1, -60), | |
new THREE.Vector3(0, 0, -70), | |
]); | |
const geometry = new THREE.TubeGeometry(path, 300, 2, 32, false); | |
const tube = new THREE.Mesh(geometry, wireframeMaterial); | |
scene.add(tube); | |
// Mouse Movement - Camera Shake | |
const mouse = { x: 0, y: 0 }; | |
window.addEventListener("mousemove", (e) => { | |
mouse.x = (e.clientX / window.innerWidth) * 2 - 1; // Normalize mouse X to -1 -> 1 | |
mouse.y = -(e.clientY / window.innerHeight) * 2 + 1; // Normalize mouse Y to -1 -> 1 | |
}); | |
// GSAP Camera Animation with Mouse Shake | |
let percentage = { value: 0 }; | |
gsap.to(percentage, { | |
value: 1, | |
duration: 10, | |
ease: "linear", | |
repeat: -1, | |
onUpdate: () => { | |
const p1 = path.getPointAt(percentage.value); | |
const p2 = path.getPointAt((percentage.value + 0.01) % 1); | |
// Add a small shake effect based on mouse movement | |
const shakeX = mouse.x * 0.3; // Subtle shake | |
const shakeY = mouse.y * 0.3; | |
camera.position.set(p1.x + shakeX, p1.y + shakeY, p1.z); | |
camera.lookAt(p2); | |
} | |
}); | |
// Animation Loop | |
function render() { | |
uniforms.uTime.value += 0.01; // Increment time for animated noise | |
renderer.render(scene, camera); | |
requestAnimationFrame(render); | |
} | |
render(); | |
// Window Resize Handling | |
window.addEventListener("resize", () => { | |
camera.aspect = window.innerWidth / window.innerHeight; | |
camera.updateProjectionMatrix(); | |
renderer.setSize(window.innerWidth, window.innerHeight); | |
}); | |
// Enable/Disable Displacement | |
function toggleDisplacement(enable) { | |
uniforms.uEnableDisplacement.value = enable; | |
console.log("Displacement Enabled:", enable); | |
} | |
// Enable Displacement | |
toggleDisplacement(false) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/gsap.min.js"></script> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
* { | |
margin:0; | |
padding:0; | |
} | |
body { | |
margin: 0; | |
overflow: hidden; | |
background-color: #141414; | |
} | |
canvas { | |
display: block; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment