Toggle navigation
Sign Up
Log In
Explore
Works
Folders
Tools
Collections
Artists
Groups
Groups
Topics
Tasks
Tasks
Jobs
Teams
Jobs
Recommendation
More Effects...
JS
(function() { var W = window, D = document, R = D.documentElement, B = D.body, watchScroll = false, WSCROLL = { x: 0, y: 0 }, noop = function(){}; // Check for canvas support, exit out if no supprt if ( !W.CanvasRenderingContext2D ) { return function(){ return false; }; } // Use Init function to keep users from having to use `new Canvallax`. Also keeps Canvallax prototype clean for util functions. function Canvallax(options){ return new Init(options); } // Only one scroll tracker that works for every Canvallax instance function onScroll(){ WSCROLL.x = R.scrollLeft || B.scrollLeft; WSCROLL.y = R.scrollTop || B.scrollTop; } function Init(options) { var C = this; if ( !watchScroll ) { watchScroll = true; onScroll(); W.addEventListener('scroll', onScroll); } if ( options ) { Canvallax.extend(this,options); } C.canvas = D.createElement('canvas'); C.canvas.className = 'canvallax ' + C.className; C.parent.insertBefore(C.canvas, C.parent.firstChild); if ( C.fullscreen ) { C.resizeFullscreen(); W.addEventListener('resize', C.resizeFullscreen.bind(C)); } else { C.canvas.width = C.width; C.canvas.height = C.height; } C.ctx = C.canvas.getContext('2d'); C.x = ( options.x !== undefined ? options.x : WSCROLL.x ); C.y = ( options.y !== undefined ? options.y : WSCROLL.y ); C.addElements(C.elements); C.damping = ( !C.damping || C.damping < 1 ? 1 : C.damping ); if ( C.animating ) { C.render(); } return C; } //////////////////////////////////////// function zIndexSort(a,b){ return (a.zIndex === b.zIndex ? 0 : a.zIndex < b.zIndex ? -1 : 1 ); } function stop(){ this.animating = false; return this; } Init.prototype = { scroll: true, parent: B, animating: true, elements: [], damping: 1, _x: 0, _y: 0, fullscreen: true, render: function() { var C = this, i = 0, len = C.elements.length, el, distance; C.now = Date.now(); if ( C.animating !== false ) { requestAnimationFrame(C.render.bind(C)); } if ( C.scroll ) { C.x = ( C.scroll === 'invert' || C.scroll === 'invertx' ? -WSCROLL.x : WSCROLL.x ); C.y = ( C.scroll === 'invert' || C.scroll === 'inverty' ? -WSCROLL.y : WSCROLL.y ); } C._x += ( -C.x - C._x ) / C.damping; C._y += ( -C.y - C._y ) / C.damping; C.ctx.clearRect(0, 0, C.canvas.width, C.canvas.height); for (; i < len; i++){ C.ctx.save(); C.elements[i]._render(C.ctx,C); C.ctx.restore(); } return this; }, resizeFullscreen: function() { this.canvas.width = W.innerWidth; this.canvas.height = W.innerHeight; return this; }, addElement: function(element){ this.elements.push(element); this.elements.sort(zIndexSort); return this; }, addElements: function(elements){ var i = 0, len = elements.length; for (; i < len; i++) { this.addElement(elements[i]); } return this; }, play: function(){ this.animation = true; this.render(); return this; }, stop: stop, pause: stop }; //////////////////////////////////////// Canvallax.extend = function(target) { target = target || {}; var length = arguments.length, i = 1; if ( arguments.length === 1) { target = this; i = 0; } for (; i < length; i++) { if (!arguments[i]) { continue; } for (var key in arguments[i]) { if ( arguments[i].hasOwnProperty(key) ) { target[key] = arguments[i][key]; } } } return target; }; var elementPrototype = { x: 0, y: 0, opacity: 1, distance: 1, scale: true, render: noop, init: noop, _render: function(ctx,C) { var el = this, distance = el.distance || 1; C.ctx.globalAlpha = el.opacity || 1; if ( el.scale ) { C.ctx.scale(distance,distance); C.ctx.translate(C._x, C._y); } else { C.ctx.translate(C._x * distance, C._y * distance); } if ( el.render ) { el.render.call(el,ctx,C); } } }; Canvallax.createElement = function(defaults){ function elInit(options){ Canvallax.extend(this,options); this.init.apply(this,arguments); } function el(options) { return new elInit(options); } elInit.prototype = Canvallax.extend({},elementPrototype,defaults); elInit.prototype.constructor = el; return el; } //////////////////////////////////////// Canvallax.Element = Canvallax.createElement(); //////////////////////////////////////// Canvallax.Image = Canvallax.createElement({ init: function(options){ if ( options.nodeType === 1 ) { this.image = options; } else if ( (!this.image && options.src) || (typeof options === typeof "") ) { this.image = new Image(); this.image.src = options.src || options; } }, render: function(ctx){ if ( this.image ) { var drawArgs = [this.image]; if ( this.crop ) { drawArgs.push(this.crop.x,this.crop.y,this.crop.width,this.crop.height); this.width = ( this.width ? this.width : this.crop.width ); this.height = ( this.height ? this.height : this.crop.height ); } drawArgs.push(this.x,this.y); if ( this.width && this.height ) { drawArgs.push(this.width, this.height); } ctx.drawImage.apply(ctx,drawArgs); } } }); //////////////////////////////////////// return W.Canvallax = Canvallax; })(); var can = Canvallax({ className: 'bg-canvas', scroll: true, // (boolean|'invert'|'invertx'|'inverty') If true, the X and Y of the scene are tied to document's scroll for a typical parallax experience. Set to false if you want to control the scene's X and Y manually. x: 0, // Starting x. If tied to scroll, this will be overridden on render. y: 0, // Starting y. If tied to scroll, this will be overridden on render. damping: 40 // the 'easing' of the x & y position when updated. 1 = none, higher is longer. If you're syncing parallax items to regular items in the scroll, then you'll probably want a low damping. }); /* // The canvallax instance can be animated with GSAP, not scroll, if you want! TweenMax.to(can,4,{ //x: 1000, y: 1000, yoyo: true, ease: Cubic.easeInOut, repeat: -1, }); */ // Clouds (function(){ //////////////////////////////////////// var origWidth = width = document.body.clientWidth, origHeight = height = document.body.clientHeight; var gradient = Canvallax.Element({ width: width * 1.5, height: height * 1.1, zIndex: 1, pattern: (function(){ var canvas = document.createElement('canvas'), ctx = canvas.getContext('2d'), gradient = ctx.createLinearGradient(0,0,0,height); gradient.addColorStop(0,'#07588A'); gradient.addColorStop(1,'#E1F6F4'); return gradient; })(), render: function(ctx) { ctx.fillStyle = this.pattern; ctx.fillRect(this.x, this.y, this.width, this.height); } }); can.addElement(gradient); window.addEventListener('resize', function(){ height = document.body.clientHeight; var i = can.elements.length, max = document.body.clientWidth, heightScale = height / origHeight; console.log(height,origHeight,heightScale); while (i--){ can.elements[i].maxX = max; //document.body.clientWidth; can.elements[i].origY = can.elements[i].origY || can.elements[i].y; can.elements[i].y = can.elements[i].origY * heightScale; } }); //////////////////////////////////////// // Adapted from http://bost.ocks.org/mike/algorithms/ function bestCandidateSampler(width, height, numCandidates) { var samples = []; function findDistance(a, b) { var dx = a[0] - b[0], dy = a[1] - b[1]; return dx * dx + dy * dy; } function findClosest(c){ var i = samples.length, sample, closest, distance, closestDistance; while(i--){ sample = samples[i]; distance = findDistance(sample,c); if ( !closestDistance || distance < closestDistance ) { closest = sample; closestDistance = distance; } } return closest; } function getSample() { var bestCandidate, bestDistance = 0, i = 0, c, d; c = [Math.random() * width, Math.random() * height]; if ( samples.length < 1 ) { bestCandidate = c; } else { for (; i < numCandidates; i++) { c = [Math.random() * width, Math.random() * height]; d = findDistance(findClosest(c), c); if (d > bestDistance) { bestDistance = d; bestCandidate = c; } } } samples.push(bestCandidate); //console.log('bestCandidate',bestCandidate); return bestCandidate; } getSample.all = function(){ return samples; }; getSample.samples = samples; return getSample; } var getCandidate = bestCandidateSampler(width,height,10); var CLOUD_COUNT = 40, CLOUD_WIDTH = 510, CLOUD_HEIGHT = 260; CLOUD_COUNT = Math.floor(( width * height ) / (CLOUD_WIDTH * CLOUD_HEIGHT)); var Cloud = Canvallax.createElement({ init: function(options) { this.tex = { x: 0, y: options.texOffset || 0, w: this.image.width, h: this.image.height }; }, render: function(ctx){ //this.x = (this.x * this.distance) < this.maxX ? this.x + this.speed : -this.tex.w; this.x = (this.x * this.distance) > -this.tex.w ? this.x - this.speed : offsetDistance(this.maxX,this.distance); // / (this.distance || 1); ctx.drawImage(this.image, this.tex.x, this.tex.y, this.tex.w, this.tex.h, this.x, this.y, this.tex.w, this.tex.h); } }); Cloud.prototype.maxX = width; Cloud.prototype.speed = 0.25; function offsetDistance(width,distance){ var d = ( width / distance ) / ( 2 - distance); return d; } function randomRange(min, max) { return Math.random() * (max - min) + min; } function randomizedCloud(image){ var canvas = document.createElement('canvas'), ctx = canvas.getContext('2d'), width = canvas.width = randomRange(400,700), //CLOUD_WIDTH, height = canvas.height = randomRange(200,260),//CLOUD_HEIGHT, w = image.width, h = image.height, halfw = w / 2, halfh = h / 2, i = Math.ceil(randomRange(20,90)), //60 flip, randScale, randTex, maxScale = 2.5, fullPi = Math.PI / 2; while (i--){ randScale = randomRange(0.4,maxScale); ctx.globalAlpha = Math.random() - 0.2; ctx.translate(randomRange(halfw, width - ( w * maxScale * 1.3)),randomRange(halfh,height - (h* maxScale))); ctx.scale(randScale,randomRange(randScale - 0.3,randScale)); ctx.translate(halfw,halfh); ctx.rotate(randomRange(0,fullPi)); ctx.drawImage(image, -halfw, -halfh);//, w, h, 0, 0, w, h); ctx.setTransform(1, 0, 0, 1, 0, 0); } //document.body.appendChild(canvas); //var img = document.createElement('img'); //img.src = canvas.toDataURL(); return canvas; } var cloudImg = new Image(); cloudImg.addEventListener('load',function(){ var i = CLOUD_COUNT, //Math.ceil(CLOUD_COUNT * 0.5), el, rand, pos, tex; while(i--) { rand = Math.round(randomRange(0.5, 1.1) * 10) / 10; pos = getCandidate(); tex = randomizedCloud(cloudImg); cloud = new Cloud({ image: tex, maxX: width, zIndex: rand * 13, x: pos[0], y: pos[1], opacity: (rand < 0.8 ? 0.8 : rand ), distance: rand, speed: rand * randomRange(0.1,0.3) }); can.addElement(cloud); } can.render(); }); cloudImg.src = ''; //////////////////////////////////////// // Full cloud texture image /* var texture = new Image(), CLOUD_SPRITE_DISTANCE = 300; texture.addEventListener('load', function() { //return false; var i = Math.ceil(CLOUD_COUNT * 0.5), cloud, rand, pos, tex; while(i--) { rand = Math.round(randomRange(0.3, 0.9) * 10) / 10; pos = getCandidate(); tex = Math.floor(rand < 0.6 ? randomRange(4,6) : randomRange(0,3)) * CLOUD_SPRITE_DISTANCE; cloud = new Cloud({ image: texture, tex: { x: 0, y: tex || 0, w: 510, h: 260 }, maxX: width, opacity: (rand < 0.8 ? 0.8 : rand ), x: pos[0] - randomRange(0,width * 0.25), //randomRange(-300,window.innerWidth), y: pos[1], //Math.ceil((1 - rand) * (CLOUD_SPRITE_COUNT + 1)) * CLOUD_SPRITE_DISTANCE, // Randomize the texture based on distance. Further away clouds are blurry. distance: rand, //Math.random() speed: rand * randomRange(0.025,0.2), }); can.addElement(cloud); } }); texture.src = 'http://i.imgur.com/48yRQIF.png'; /**/ })();
CSS
canvas { display: block; margin: 2em; } .bg-canvas { position: fixed; top: 0; left: 0; z-index: -1; width: 100%; height: 100%; margin: 0; } body { background-image: -webkit-linear-gradient(bottom, #d7dde6 40%, #97a8c0 80%); background-image: linear-gradient(to top, #d7dde6 40%, #97a8c0 80%); background-size: cover; width: 400vw; height: 400vh; }
HTML
Join Effecthub.com
Working with Global Gaming Artists and Developers!
Login
Sign Up
Or Login with Your Email Address:
Email
Password
Remember
Or Sign Up with Your Email Address:
Your Email
This field must contain a valid email
Set Password
Password should be at least 1 character
Stay informed via email