<div class="animation-wrapper">
<div class="layered-animations">
<svg class="large shape" viewBox="0 0 96 96">
<defs>
<linearGradient id="circleGradient" x1="0%" x2="100%" y1="20%" y2="80%">
<stop stop-color="#373734" offset="0%"/>
<stop stop-color="#242423" offset="50%"/>
<stop stop-color="#0D0D0C" offset="100%"/>
</linearGradient>
</defs>
<circle cx="48" cy="48" r="28" fill-rule="evenodd" stroke-linecap="square" fill="url(#circleGradient)"/>
</svg>
<svg class="small shape color-red" viewBox="0 0 96 96">
<polygon fill-rule="evenodd" points="48 17.28 86.4 80.11584 9.6 80.11584" stroke-linecap="square"/>
</svg>
<svg class="large shape" viewBox="0 0 96 96">
<defs>
<linearGradient id="triangleGradient" x1="0%" x2="100%" y1="20%" y2="80%">
<stop stop-color="#373734" offset="0%"/>
<stop stop-color="#242423" offset="50%"/>
<stop stop-color="#0D0D0C" offset="100%"/>
</linearGradient>
</defs>
<polygon fill-rule="evenodd" points="48 17.28 86.4 80.11584 9.6 80.11584" stroke-linecap="square" fill="url(#triangleGradient)"/>
</svg>
<svg class="x-small shape" viewBox="0 0 96 96">
<polygon fill-rule="evenodd" points="48 17.28 86.4 80.11584 9.6 80.11584" stroke-linecap="square"/>
</svg>
<svg class="x-small shape" viewBox="0 0 96 96">
<rect width="48" height="48" x="24" y="24" fill-rule="evenodd" stroke-linecap="square"/>
</svg>
<svg class="small shape color-red" viewBox="0 0 96 96">
<rect width="48" height="48" x="24" y="24" fill-rule="evenodd" stroke-linecap="square"/>
</svg>
<svg class="large shape" viewBox="0 0 96 96">
<defs>
<linearGradient id="rectGradient" x1="0%" x2="100%" y1="20%" y2="80%">
<stop stop-color="#373734" offset="0%"/>
<stop stop-color="#242423" offset="50%"/>
<stop stop-color="#0D0D0C" offset="100%"/>
</linearGradient>
</defs>
<rect width="48" height="48" x="24" y="24" fill-rule="evenodd" stroke-linecap="square" fill="url(#rectGradient)"/>
</svg>
<svg class="small shape color-red" viewBox="0 0 96 96">
<circle cx="48" cy="48" r="32" fill-rule="evenodd" stroke-linecap="square"/>
</svg>
<svg class="x-small shape" viewBox="0 0 96 96">
<circle cx="48" cy="48" r="32" fill-rule="evenodd" stroke-linecap="square"/>
</svg>
</div>
</div>
body {
display: flex;
justify-content: center;
align-items: center;
position: absolute;
width: 100%;
height: 100vh;
}
.animation-wrapper {
width: 80%;
padding-bottom: 40%;
}
.layered-animations {
position: absolute;
top: 50%;
left: 50%;
display: flex;
align-items: center;
justify-content: center;
width: 1100px;
height: 550px;
margin: -275px 0 0 -550px;
}
.layered-animations .shape {
position: absolute;
top: 50%;
overflow: visible;
width: 280px;
height: 280px;
margin-top: -140px;
stroke: transparent;
stroke-width: 1px;
fill: url(#shapesGradient);
}
@media (min-width: 740px) {
.layered-animations .shape {
stroke-width: .5px;
}
}
.layered-animations .small.shape {
width: 64px;
height: 64px;
margin-top: -32px;
stroke: currentColor;
fill: currentColor;
}
.layered-animations .x-small.shape {
width: 32px;
height: 32px;
margin-top: -16px;
stroke: currentColor;
fill: currentColor;
}
function fitElementToParent(el, padding) {
var timeout = null;
function resize() {
if (timeout) clearTimeout(timeout);
anime.set(el, {scale: 1});
var pad = padding || 0;
var parentEl = el.parentNode;
var elOffsetWidth = el.offsetWidth - pad;
var parentOffsetWidth = parentEl.offsetWidth;
var ratio = parentOffsetWidth / elOffsetWidth;
timeout = setTimeout(anime.set(el, {scale: ratio}), 10);
}
resize();
window.addEventListener('resize', resize);
}
var layeredAnimation = (function() {
var transformEls = document.querySelectorAll('.transform-progress');
var layeredAnimationEl = document.querySelector('.layered-animations');
var shapeEls = layeredAnimationEl.querySelectorAll('.shape');
var triangleEl = layeredAnimationEl.querySelector('polygon');
var trianglePoints = triangleEl.getAttribute('points').split(' ');
var easings = ['easeInOutQuad', 'easeInOutCirc', 'easeInOutSine', 'spring'];
fitElementToParent(layeredAnimationEl);
function createKeyframes(value) {
var keyframes = [];
for (var i = 0; i < 30; i++) keyframes.push({ value: value });
return keyframes;
}
function animateShape(el) {
var circleEl = el.querySelector('circle');
var rectEl = el.querySelector('rect');
var polyEl = el.querySelector('polygon');
var animation = anime.timeline({
targets: el,
duration: function() { return anime.random(600, 2200); },
easing: function() { return easings[anime.random(0, easings.length - 1)]; },
complete: function(anim) { animateShape(anim.animatables[0].target); },
})
.add({
translateX: createKeyframes(function(el) {
return el.classList.contains('large') ? anime.random(-300, 300) : anime.random(-520, 520);
}),
translateY: createKeyframes(function(el) {
return el.classList.contains('large') ? anime.random(-110, 110) : anime.random(-280, 280);
}),
rotate: createKeyframes(function() { return anime.random(-180, 180); }),
}, 0);
if (circleEl) {
animation.add({
targets: circleEl,
r: createKeyframes(function() { return anime.random(32, 72); }),
}, 0);
}
if (rectEl) {
animation.add({
targets: rectEl,
width: createKeyframes(function() { return anime.random(64, 120); }),
height: createKeyframes(function() { return anime.random(64, 120); }),
}, 0);
}
if (polyEl) {
animation.add({
targets: polyEl,
points: createKeyframes(function() {
var scale = anime.random(72, 180) / 100;
return trianglePoints.map(function(p) { return p * scale; }).join(' ');
}),
}, 0);
}
}
for (var i = 0; i < shapeEls.length; i++) {
animateShape(shapeEls[i]);
}
})();