<a href="https://twitter.com/nskargovskii" target="_blank">@nskargovskii</a>
<script id="fragShader" type="x-shader/x-fragment">
#extension GL_OES_standard_derivatives : enable
precision highp float;
varying vec3 vPos;
uniform sampler2D envmap;
varying vec2 vUv;
vec3 mod289(vec3 x) {
return x - floor(x * (1.0 / 289.0)) * 289.0;
}
vec4 mod289(vec4 x) {
return x - floor(x * (1.0 / 289.0)) * 289.0;
}
vec4 permute(vec4 x) {
return mod289(((x*34.0)+1.0)*x);
}
vec4 taylorInvSqrt(vec4 r)
{
return 1.79284291400159 - 0.85373472095314 * r;
}
float snoise(vec3 v)
{
const vec2 C = vec2(1.0/6.0, 1.0/3.0) ;
const vec4 D = vec4(0.0, 0.5, 1.0, 2.0);
// First corner
vec3 i = floor(v + dot(v, C.yyy) );
vec3 x0 = v - i + dot(i, C.xxx) ;
// Other corners
vec3 g = step(x0.yzx, x0.xyz);
vec3 l = 1.0 - g;
vec3 i1 = min( g.xyz, l.zxy );
vec3 i2 = max( g.xyz, l.zxy );
// x0 = x0 - 0.0 + 0.0 * C.xxx;
// x1 = x0 - i1 + 1.0 * C.xxx;
// x2 = x0 - i2 + 2.0 * C.xxx;
// x3 = x0 - 1.0 + 3.0 * C.xxx;
vec3 x1 = x0 - i1 + C.xxx;
vec3 x2 = x0 - i2 + C.yyy; // 2.0*C.x = 1/3 = C.y
vec3 x3 = x0 - D.yyy; // -1.0+3.0*C.x = -0.5 = -D.y
// Permutations
i = mod289(i);
vec4 p = permute( permute( permute(
i.z + vec4(0.0, i1.z, i2.z, 1.0 ))
+ i.y + vec4(0.0, i1.y, i2.y, 1.0 ))
+ i.x + vec4(0.0, i1.x, i2.x, 1.0 ));
// Gradients: 7x7 points over a square, mapped onto an octahedron.
// The ring size 17*17 = 289 is close to a multiple of 49 (49*6 = 294)
float n_ = 0.142857142857; // 1.0/7.0
vec3 ns = n_ * D.wyz - D.xzx;
vec4 j = p - 49.0 * floor(p * ns.z * ns.z); // mod(p,7*7)
vec4 x_ = floor(j * ns.z);
vec4 y_ = floor(j - 7.0 * x_ ); // mod(j,N)
vec4 x = x_ *ns.x + ns.yyyy;
vec4 y = y_ *ns.x + ns.yyyy;
vec4 h = 1.0 - abs(x) - abs(y);
vec4 b0 = vec4( x.xy, y.xy );
vec4 b1 = vec4( x.zw, y.zw );
//vec4 s0 = vec4(lessThan(b0,0.0))*2.0 - 1.0;
//vec4 s1 = vec4(lessThan(b1,0.0))*2.0 - 1.0;
vec4 s0 = floor(b0)*2.0 + 1.0;
vec4 s1 = floor(b1)*2.0 + 1.0;
vec4 sh = -step(h, vec4(0.0));
vec4 a0 = b0.xzyw + s0.xzyw*sh.xxyy ;
vec4 a1 = b1.xzyw + s1.xzyw*sh.zzww ;
vec3 p0 = vec3(a0.xy,h.x);
vec3 p1 = vec3(a0.zw,h.y);
vec3 p2 = vec3(a1.xy,h.z);
vec3 p3 = vec3(a1.zw,h.w);
//Normalise gradients
vec4 norm = taylorInvSqrt(vec4(dot(p0,p0), dot(p1,p1), dot(p2, p2), dot(p3,p3)));
p0 *= norm.x;
p1 *= norm.y;
p2 *= norm.z;
p3 *= norm.w;
// Mix final noise value
vec4 m = max(0.6 - vec4(dot(x0,x0), dot(x1,x1), dot(x2,x2), dot(x3,x3)), 0.0);
m = m * m;
return 42.0 * dot( m*m, vec4( dot(p0,x0), dot(p1,x1),
dot(p2,x2), dot(p3,x3) ) );
}
uniform mat4 modelViewMatrix;
uniform mat3 normalMatrix;
uniform sampler2D texture;
uniform float u_time;
void main() {
vec3 fdx = dFdx( vPos );
vec3 fdy = dFdy( vPos );
vec3 normal = normalize( cross( fdx, fdy ) );
vec3 e = normalize( vec3( modelViewMatrix * vec4(vPos,1.) ) );
vec3 n = normalize( normalMatrix * normal );
vec3 r = reflect( e, n );
float m = 3.*sqrt(
pow( r.x, 2. ) +
pow( r.y, 2. ) +
pow( r.z + 1., 2. )
);
vec2 vN = r.xy / m + .5;
float diffuse = dot( normal, vec3( 0., 0.,10. ) );
vec3 base = texture2D( envmap, vN ).rgb;
vec3 text = texture2D(texture, vUv).rgb;
gl_FragColor = vec4(base*diffuse*vec3(0.48,0.22,.98)*snoise(vPos*800.+u_time/600.),1.);
}
</script>
<script id="vertShader" type="x-shader/x-vertex">
#extension GL_OES_standard_derivatives : enable
uniform mat4 modelViewMatrix;
uniform mat4 projectionMatrix;
uniform mat3 normalMatrix;
vec3 mod289(vec3 x) {
return x - floor(x * (1.0 / 289.0)) * 289.0;
}
vec4 mod289(vec4 x) {
return x - floor(x * (1.0 / 289.0)) * 289.0;
}
vec4 permute(vec4 x) {
return mod289(((x*34.0)+1.0)*x);
}
vec4 taylorInvSqrt(vec4 r)
{
return 1.79284291400159 - 0.85373472095314 * r;
}
float snoise(vec3 v)
{
const vec2 C = vec2(1.0/6.0, 1.0/3.0) ;
const vec4 D = vec4(0.0, 0.5, 1.0, 2.0);
// First corner
vec3 i = floor(v + dot(v, C.yyy) );
vec3 x0 = v - i + dot(i, C.xxx) ;
// Other corners
vec3 g = step(x0.yzx, x0.xyz);
vec3 l = 1.0 - g;
vec3 i1 = min( g.xyz, l.zxy );
vec3 i2 = max( g.xyz, l.zxy );
// x0 = x0 - 0.0 + 0.0 * C.xxx;
// x1 = x0 - i1 + 1.0 * C.xxx;
// x2 = x0 - i2 + 2.0 * C.xxx;
// x3 = x0 - 1.0 + 3.0 * C.xxx;
vec3 x1 = x0 - i1 + C.xxx;
vec3 x2 = x0 - i2 + C.yyy; // 2.0*C.x = 1/3 = C.y
vec3 x3 = x0 - D.yyy; // -1.0+3.0*C.x = -0.5 = -D.y
// Permutations
i = mod289(i);
vec4 p = permute( permute( permute(
i.z + vec4(0.0, i1.z, i2.z, 1.0 ))
+ i.y + vec4(0.0, i1.y, i2.y, 1.0 ))
+ i.x + vec4(0.0, i1.x, i2.x, 1.0 ));
// Gradients: 7x7 points over a square, mapped onto an octahedron.
// The ring size 17*17 = 289 is close to a multiple of 49 (49*6 = 294)
float n_ = 0.142857142857; // 1.0/7.0
vec3 ns = n_ * D.wyz - D.xzx;
vec4 j = p - 49.0 * floor(p * ns.z * ns.z); // mod(p,7*7)
vec4 x_ = floor(j * ns.z);
vec4 y_ = floor(j - 7.0 * x_ ); // mod(j,N)
vec4 x = x_ *ns.x + ns.yyyy;
vec4 y = y_ *ns.x + ns.yyyy;
vec4 h = 1.0 - abs(x) - abs(y);
vec4 b0 = vec4( x.xy, y.xy );
vec4 b1 = vec4( x.zw, y.zw );
//vec4 s0 = vec4(lessThan(b0,0.0))*2.0 - 1.0;
//vec4 s1 = vec4(lessThan(b1,0.0))*2.0 - 1.0;
vec4 s0 = floor(b0)*2.0 + 1.0;
vec4 s1 = floor(b1)*2.0 + 1.0;
vec4 sh = -step(h, vec4(0.0));
vec4 a0 = b0.xzyw + s0.xzyw*sh.xxyy ;
vec4 a1 = b1.xzyw + s1.xzyw*sh.zzww ;
vec3 p0 = vec3(a0.xy,h.x);
vec3 p1 = vec3(a0.zw,h.y);
vec3 p2 = vec3(a1.xy,h.z);
vec3 p3 = vec3(a1.zw,h.w);
//Normalise gradients
vec4 norm = taylorInvSqrt(vec4(dot(p0,p0), dot(p1,p1), dot(p2, p2), dot(p3,p3)));
p0 *= norm.x;
p1 *= norm.y;
p2 *= norm.z;
p3 *= norm.w;
// Mix final noise value
vec4 m = max(0.6 - vec4(dot(x0,x0), dot(x1,x1), dot(x2,x2), dot(x3,x3)), 0.0);
m = m * m;
return 42.0 * dot( m*m, vec4( dot(p0,x0), dot(p1,x1),
dot(p2,x2), dot(p3,x3) ) );
}
attribute vec3 position;
attribute vec2 uv;
attribute vec3 translation;
attribute vec2 uvOffset;
attribute vec2 uvScale;
uniform float u_time;
varying vec3 vPos;
varying vec2 vUv;
varying vec2 vN;
vec3 transform(inout vec3 position ,vec3 T) {
position += T;
return position;
}
void main() {
vUv = uv * uvScale + uvOffset;
vec3 pos = position;
transform(pos, translation);
pos.z = pos.z + snoise(pos*10000.+u_time/600.)/1000.;
gl_Position = projectionMatrix * modelViewMatrix*vec4(pos, 1.);
vPos = pos;
}
</script>
a {
position: absolute;
color: white;
bottom : 15px;
left:10px;
font-family: arial;
text-decoration: none;
font-size: 18px;
}
canvas {
width: 100%;
height: 100%;
background: #000;
}
html, body {
margin: 0;
padding: 0;
overflow: hidden;
}
class SecondGrid {
constructor(opt) {
this.geometry = new THREE.InstancedBufferGeometry()
this.scene = opt.scene;
this.coords = opt.coords;
this.count = opt.count
this.screenRatio = opt.screenRatio;
this.createBlueprint()
this.instanceBlueprint()
}
createBlueprint() {
this.blueprint = this.coords;
this.sideLength = Math.sqrt((Math.pow(this.blueprint[0]-this.blueprint[3],2))+(Math.pow(this.blueprint[1]-this.blueprint[4],2)))
this.uv = [0,0.5,0.5,0.14,0.5,0.86,0.5,0.14,1,0.5,0.5,0.86]
console.log(this.coords)
let position = new THREE.BufferAttribute(new Float32Array(this.blueprint),3)
this.geometry.addAttribute('position', position)
let uv = new THREE.BufferAttribute(new Float32Array(this.uv),2)
this.geometry.addAttribute('uv', uv)
}
instanceBlueprint() {
var translation = new Float32Array( this.count * 3 );
var uvOffset = new Float32Array(this.count * 2);
var uvScales = new Float32Array(this.count * 2);
var uvOffsetIterator = 0;
var uvScalesIterator = 0;
//and iterators for convenience :)
var translationIterator = 0;
this.rank = -1;
let uvScale = new THREE.Vector2(1 / 60, 1 / 60);
for (let i = 0; i < 120; i++) {
for (let j = 0; j < 60; j++) {
this.rank++
uvScales[uvScalesIterator++] = uvScale.x;
uvScales[uvScalesIterator++] = uvScale.y;
if(i %2 ==0) {
translation[ translationIterator++ ] = 2*((Math.sin(Math.PI/3)*this.sideLength)*j) - Math.abs(((this.screenRatio.x*2)-2*((Math.sin(Math.PI/3)*this.sideLength))*60))/2
translation[ translationIterator++ ] = i*this.sideLength/2 - this.sideLength*60 + Math.abs(((-this.screenRatio.y*2)-this.sideLength*60))/2
translation[ translationIterator++ ] = 0
uvOffset[uvOffsetIterator++] = j * uvScale.x;
uvOffset[uvOffsetIterator++] = 0.36*i * uvScale.y;
} else {
translation[ translationIterator++ ] = 2*((Math.sin(Math.PI/3)*this.sideLength)*j)+(Math.sin(Math.PI/3)*this.sideLength) - Math.abs(((this.screenRatio.x*2)-2*((Math.sin(Math.PI/3)*this.sideLength))*60))/2
translation[ translationIterator++ ] = i*this.sideLength/2 - this.sideLength*60 + Math.abs(((-this.screenRatio.y*2)-this.sideLength*60))/2
translation[ translationIterator++ ] = 0
uvOffset[uvOffsetIterator++] = (j * uvScale.x)+(0.5/6) ;
uvOffset[uvOffsetIterator++] = 0.36*i * uvScale.y;
}
}
}
this.geometry.addAttribute( 'translation', new THREE.InstancedBufferAttribute( translation, 3, 1 ) );
this.geometry.addAttribute(
"uvOffset",
new THREE.InstancedBufferAttribute(uvOffset, 2, 1)
);
this.geometry.addAttribute(
"uvScale",
new THREE.InstancedBufferAttribute(uvScales, 2, 1)
);
// video = document.createElement( 'video' );
// // video.id = 'video';
// // video.type = ' video/ogg; codecs="theora, vorbis" ';
// video.src = "../Untitled.mp4";
// video.load(); // must call after setting/changing source
// video.play();
// var texture = new THREE.VideoTexture( video );
// texture.minFilter = THREE.LinearFilter;
// texture.magFilter = THREE.LinearFilter;
// texture.format = THREE.RGBFormat;
let material = new THREE.RawShaderMaterial(
{ uniforms: {
u_time: {
type:'f',
value:1.0,
},
envmap: { type: "t", value: null },
texture: { type: "t", value: null },
},
vertexShader: document.getElementById('vertShader').innerHTML,
fragmentShader: document.getElementById('fragShader').innerHTML,
side: THREE.DoubleSide,
wireframe: false,
}
)
var co = 'http://crossorigin.me/',
url = 'https://s3-us-west-2.amazonaws.com/s.cdpn.io/2017/leopard_stalking.jpg',
src = co + url;
var tl = new THREE.TextureLoader();
tl.setCrossOrigin( "Anonymous" );
tl.load('', function( t ) {
material.uniforms.envmap.value = t;
});
this.grid = new THREE.Mesh(this.geometry, material)
this.scene.add(this.grid)
}
}
class createApp {
constructor(opt) {
this.winWidth = window.innerWidth
this.winHeight = window.innerHeight
this.winRatio = this.winWidth/this.winHeight
this.camera = new THREE.PerspectiveCamera( 50,this.winRatio, 0.005, 1000 )
this.camera2 = new THREE.PerspectiveCamera( 50,this.winRatio, 0.005, 1000 )
this.camera.setFocalLength(50)
this.camera2.setFocalLength(50)
this.camera.position.z = 1
this.camera2.position.z = 0.015
this.controls = new THREE.OrbitControls(this.camera2);
this.controls.enableRotate = false;
this.target = new THREE.Vector3()
this.scene = new THREE.Scene();
this.renderer = new THREE.WebGLRenderer({antialias: true, alpha:true})
this.renderer.setSize(this.winWidth, this.winHeight)
document.body.appendChild(this.renderer.domElement)
window.addEventListener('resize', this.onResize.bind(this))
window.addEventListener('mousemove', this.onMouseMove.bind(this))
this.rawCoords = [
{
x:this.winWidth/101,
y:0,
},
{
x:Math.cos(2*Math.PI/3)*this.winWidth/101,
y:Math.sin(2*Math.PI/3)*this.winWidth/101
},
{
x:Math.cos((2*Math.PI/3)*2)*this.winWidth/101,
y:Math.sin((2*Math.PI/3)*2)*this.winWidth/101
},
]
this.rawCoords2 = [
{
x:-this.winWidth/60,
y:0,
},
{
x:- Math.cos(-2*Math.PI/3)*this.winWidth/60,
y:- Math.sin(-2*Math.PI/3)*this.winWidth/60
},
{
x:- Math.cos((-2*Math.PI/3)*2)*this.winWidth/60,
y:- Math.sin((-2*Math.PI/3)*2)*this.winWidth/60
},
{
x:- Math.cos(-2*Math.PI/3)*this.winWidth/60,
y:- Math.sin(-2*Math.PI/3)*this.winWidth/60
},
{
x:2*this.winWidth/60,
y:0,
},
{
x:- Math.cos((-2*Math.PI/3)*2)*this.winWidth/60,
y:- Math.sin((-2*Math.PI/3)*2)*this.winWidth/60
},
]
// let geo = new THREE.PlaneGeometry(10,10,120,120);
// let mat = new THREE.MeshBasicMaterial({color:"#ffffff",wireframe:true})
// let meshh = new THREE.Mesh(geo, mat)
// this.scene.add(meshh)
// meshh.rotation.z = Math.PI/4
this.treatedCoords = []
this.light = new THREE.PointLight(0xffffff)
this.light.position.set(0,0,0.6)
this.scene.add(this.light)
this.time = 0
this.initCoords()
this.animate()
}
initCoords() {
for (let i = 0; i < this.rawCoords2.length; i++) {
let treatedCoordsX = ((this.rawCoords2[i].x)/this.winWidth)*2-1
let treatedCoordsY = -((this.rawCoords2[i].y)/this.winHeight)*2+1
this.newPos = new THREE.Vector3(treatedCoordsX, treatedCoordsY,-1).unproject(this.camera)
this.treatedCoords.push(this.newPos.x, this.newPos.y, this.newPos.z)
}
this.grid2 = new SecondGrid({count:7200, scene: this.scene, coords: this.treatedCoords, screenRatio: new THREE.Vector3(1, -1,-1).unproject(this.camera) })
//this.grid = new Grid({count: 10201, scene: this.scene, coords:this.treatedCoords, screenRatio: new THREE.Vector3(1, -1,-1).unproject(this.camera)})
}
onMouseMove(e) {
let mouseX = e.clientX
let mouseY = e.clientY
this.mouseX = ((mouseX)/this.winWidth)*2-1
this.mouseY = -((mouseY)/this.winHeight)*2+1
}
onResize() {
this.winWidth = window.innerWidth
this.winHeight = window.innerHeight
this.winRatio = this.winWidth/this.winHeight
this.camera2.aspect = this.winRatio;
this.camera2.updateProjectionMatrix();
this.renderer.setSize(this.winWidth, this.winHeight)
}
animate() {
requestAnimationFrame(this.animate.bind(this))
this.camera2.lookAt(this.target);
this.time += 1
// this.grid.grid.material.uniforms.u_time.value = this.time
this.grid2.grid.material.uniforms.u_time.value = this.time
// this.mousePos = new THREE.Vector3(this.mouseX, this.mouseY,-1).unproject(this.camera)
// this.grid.grid.material.uniforms.u_mouse.value.x = this.mousePos.x
// this.grid.grid.material.uniforms.u_mouse.value.y = this.mousePos.y
//this.grid.grid.material.uniforms.u_time = this.time
this.renderer.render(this.scene, this.camera2)
}
}
new createApp()
This Pen doesn't use any external CSS resources.