Muscular Hydrostats
html, body {
background: #f2f2f2;
background: -moz-radial-gradient(center, ellipse cover, #ffffff 0%, #ffffff 26%, #f5f5f5 59%, #f5f5f5 77%, #cecece 100%);
background: -webkit-gradient(radial, center center, 0, center center, 100%, color-stop(0%,#ffffff), color-stop(26%,#ffffff), color-stop(59%,#f5f5f5), color-stop(77%,#f5f5f5), color-stop(100%,#cecece));
background: -webkit-radial-gradient(center, ellipse cover, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
background: -o-radial-gradient(center, ellipse cover, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
background: -ms-radial-gradient(center, ellipse cover, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
background: radial-gradient(ellipse at center, #ffffff 0%,#ffffff 26%,#f5f5f5 59%,#f5f5f5 77%,#cecece 100%);
overflow: hidden;
padding: 0;
margin: 0;
height: 100%;
}
body:before {
background-image: url();
pointer-events: none;
position: absolute;
content: '';
height: 100%;
width: 100%;
left: 0;
top: 0;
}
html.dark, body.dark {
background: #283442;
background: -moz-radial-gradient(center, ellipse cover, #283442 0%, #122335 40%, #000d19 100%);
background: -webkit-gradient(radial, center center, 0px, center center, 100%, color-stop(0%,#283442), color-stop(40%,#122335), color-stop(100%,#000d19));
background: -webkit-radial-gradient(center, ellipse cover, #283442 0%,#122335 40%,#000d19 100%);
background: -o-radial-gradient(center, ellipse cover, #283442 0%,#122335 40%,#000d19 100%);
background: -ms-radial-gradient(center, ellipse cover, #283442 0%,#122335 40%,#000d19 100%);
background: radial-gradient(ellipse at center, #283442 0%,#122335 40%,#000d19 100%);
filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#283442', endColorstr='#000d19',GradientType=1 );
}
#container {
cursor: pointer;
}
/* Info */
@-webkit-keyframes show-info {
0% { -webkit-transform: rotateY(120deg); }
100% { -webkit-transform: rotateY(0deg); }
}
@-moz-keyframes show-info {
0% { -moz-transform: rotateY(120deg); }
100% { -moz-transform: rotateY(0deg); }
}
@-ms-keyframes show-info {
0% { -ms-transform: rotateY(120deg); }
100% { -ms-transform: rotateY(0deg); }
}
@-o-keyframes show-info {
0% { -o-transform: rotateY(120deg); }
100% { -o-transform: rotateY(0deg); }
}
@keyframes show-info {
0% { transform: rotateY(120deg); }
100% { transform: rotateY(0deg); }
}
.info {
-webkit-transition: all 180ms ease-out;
-moz-transition: all 180ms ease-out;
-ms-transition: all 180ms ease-out;
-o-transition: all 180ms ease-out;
transition: all 180ms ease-out;
-webkit-transform-style: preserve-3d;
-moz-transform-style: preserve-3d;
-ms-transform-style: preserve-3d;
-o-transform-style: preserve-3d;
transform-style: preserve-3d;
-webkit-transform: perspective(800);
-moz-transform: perspective(800);
-ms-transform: perspective(800);
-o-transform: perspective(800);
transform: perspective(800);
font-family: 'Quantico', sans-serif;
position: absolute;
font-size: 12px;
opacity: 0.85;
color: #fff;
width: 240px;
left: 0px;
top: 20px;
}
.info:hover {
box-shadow: 0 0 0 4px rgba(0,0,0,0.1);
opacity: 1.0;
}
.info h1,
.info h2,
.info h3 {
line-height: 1;
margin: 5px 0;
}
.info .about, .info .more {
-webkit-transform-origin: 0% 50%;
-moz-transform-origin: 0% 50%;
-ms-transform-origin: 0% 50%;
-o-transform-origin: 0% 50%;
transform-origin: 0% 50%;
-webkit-transform: rotateY(120deg);
-moz-transform: rotateY(120deg);
-ms-transform: rotateY(120deg);
-o-transform: rotateY(120deg);
transform: rotateY(120deg);
margin-bottom: 1px;
background: rgba(0,0,0,0.95);
padding: 12px 15px 12px 20px;
}
.info .about {
-webkit-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
-moz-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
-ms-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
-o-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 600ms 1 normal forwards;
padding-bottom: 15px;
}
.info .about h1 {
letter-spacing: -1px;
font-weight: 300;
font-size: 19px;
opacity: 0.95;
}
.info .about h2 {
font-weight: 300;
font-size: 13px;
opacity: 0.8;
}
.info .about h3 {
text-transform: uppercase;
margin-top: 10px;
font-size: 11px;
}
.info .about h3:after {
margin-left: 4px;
font-size: 14px;
content: '\203A';
}
.info .more {
-webkit-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
-moz-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
-ms-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
-o-animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
animation: show-info 500ms cubic-bezier(0.230, 1.000, 0.320, 1.000) 500ms 1 normal forwards;
padding: 5px 15px 10px 20px;
}
.info .more a {
-webkit-transition: all 200ms ease-out;
-moz-transition: all 200ms ease-out;
-ms-transition: all 200ms ease-out;
-o-transition: all 200ms ease-out;
transition: all 200ms ease-out;
border-bottom: 1px dotted rgba(255,255,255,0.4);
text-transform: uppercase;
text-decoration: none;
margin-right: 10px;
font-size: 10px;
opacity: 0.6;
color: #fff;
}
.info .more a:hover {
opacity: 0.99;
}
<script src="//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.3/jquery.min.js"></script>
<script src="https://rawgithub.com/soulwire/sketch.js/v1.0/js/sketch.min.js"></script>
<script src="https://dat-gui.googlecode.com/git/build/dat.gui.min.js"></script>
<script src="https://dat-gui.googlecode.com/git/build/dat.color.min.js"></script>
/**
* Copyright (C) 2012 by Justin Windle
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
$(function() {
var settings = {
interactive: false,
darkTheme: true,
headRadius: 60,
thickness: 18,
tentacles: 40,
friction: 0.02,
gravity: 0.5,
colour: { h:0, s:0, v:0.1 },
length: 70,
pulse: true,
wind: -0.5
};
var utils = {
curveThroughPoints: function( points, ctx ) {
var i, n, a, b, x, y;
for ( i = 1, n = points.length - 2; i < n; i++ ) {
a = points[i];
b = points[i + 1];
x = ( a.x + b.x ) * 0.5;
y = ( a.y + b.y ) * 0.5;
ctx.quadraticCurveTo( a.x, a.y, x, y );
}
a = points[i];
b = points[i + 1];
ctx.quadraticCurveTo( a.x, a.y, b.x, b.y );
}
};
var Node = function( x, y ) {
this.x = this.ox = x || 0.0;
this.y = this.oy = y || 0.0;
this.vx = 0.0;
this.vy = 0.0;
};
var Tentacle = function( options ) {
this.length = options.length || 10;
this.radius = options.radius || 10;
this.spacing = options.spacing || 20;
this.friction = options.friction || 0.8;
this.shade = random( 0.85, 1.1 );
this.nodes = [];
this.outer = [];
this.inner = [];
this.theta = [];
for ( var i = 0; i < this.length; i++ ) {
this.nodes.push( new Node() );
}
};
Tentacle.prototype = {
move: function( x, y, instant ) {
this.nodes[0].x = x;
this.nodes[0].y = y;
if ( instant ) {
var i, node;
for ( i = 1; i < this.length; i++ ) {
node = this.nodes[i];
node.x = x;
node.y = y;
}
}
},
update: function() {
var i, n, s, c, dx, dy, da, px, py, node, prev = this.nodes[0];
var radius = this.radius * settings.thickness;
var step = radius / this.length;
for ( i = 1, j = 0; i < this.length; i++, j++ ) {
node = this.nodes[i];
node.x += node.vx;
node.y += node.vy;
dx = prev.x - node.x;
dy = prev.y - node.y;
da = Math.atan2( dy, dx );
px = node.x + cos( da ) * this.spacing * settings.length;
py = node.y + sin( da ) * this.spacing * settings.length;
node.x = prev.x - ( px - node.x );
node.y = prev.y - ( py - node.y );
node.vx = node.x - node.ox;
node.vy = node.y - node.oy;
node.vx *= this.friction * (1 - settings.friction);
node.vy *= this.friction * (1 - settings.friction);
node.vx += settings.wind;
node.vy += settings.gravity;
node.ox = node.x;
node.oy = node.y;
s = sin( da + HALF_PI );
c = cos( da + HALF_PI );
this.outer[j] = {
x: prev.x + c * radius,
y: prev.y + s * radius
};
this.inner[j] = {
x: prev.x - c * radius,
y: prev.y - s * radius
};
this.theta[j] = da;
radius -= step;
prev = node;
}
},
draw: function( ctx ) {
var h, s, v, e;
s = this.outer[0];
e = this.inner[0];
ctx.beginPath();
ctx.moveTo( s.x, s.y );
utils.curveThroughPoints( this.outer, ctx );
utils.curveThroughPoints( this.inner.reverse(), ctx );
ctx.lineTo( e.x, e.y );
ctx.closePath();
h = settings.colour.h * this.shade;
s = settings.colour.s * 100 * this.shade;
v = settings.colour.v * 100 * this.shade;
ctx.fillStyle = 'hsl(' + h + ',' + s + '%,' + v + '%)';
ctx.fill();
if ( settings.thickness > 2 ) {
v += settings.darkTheme ? -10 : 10;
ctx.strokeStyle = 'hsl(' + h + ',' + s + '%,' + v + '%)';
ctx.lineWidth = 1;
ctx.stroke();
}
}
};
var demo = true;
var ease = 0.1;
var modified = false;
var radius = settings.headRadius;
var tentacles = [];
var center = { x:0, y:0 };
var sketch = Sketch.create({
container: document.getElementById( 'container' ),
setup: function() {
center.x = this.width / 2;
center.y = this.height / 2;
var tentacle;
for ( var i = 0; i < 100; i++ ) {
tentacle = new Tentacle({
length: random( 10, 20 ),
radius: random( 0.05, 1.0 ),
spacing: random( 0.2, 1.0 ),
friction: random( 0.7, 0.88 )
});
tentacle.move( center.x, center.y, true );
tentacles.push( tentacle );
}
},
update: function() {
var t, cx, cy, pulse, touch;
t = this.millis * 0.001;
if ( settings.pulse ) {
pulse = pow( sin( t * PI ), 18 );
radius = settings.headRadius * 0.5 + settings.headRadius * 0.5 * pulse;
}
if ( settings.interactive ) {
ease += ( 0.7 - ease ) * 0.05;
touch = this.touches[0] || this.mouse;
center.x += ( touch.x - center.x ) * ease;
center.y += ( touch.y - center.y ) * ease;
} else {
t = this.millis;
cx = this.width * 0.5;
cy = this.height * 0.5;
center.x = cx + sin( t * 0.002 ) * cos( t * 0.00005 ) * cx * 0.5;
center.y = cy + sin( t * 0.003 ) * tan( sin( t * 0.0003 ) * 1.15 ) * cy * 0.4;
}
var px, py, theta, tentacle;
var step = TWO_PI / settings.tentacles;
for ( var i = 0, n = settings.tentacles; i < n; i++ ) {
tentacle = tentacles[i];
theta = i * step;
px = cos( theta ) * radius;
py = sin( theta ) * radius;
tentacle.move( center.x + px, center.y + py );
tentacle.update();
}
},
draw: function() {
var h = settings.colour.h * 0.95;
var s = settings.colour.s * 100 * 0.95;
var v = settings.colour.v * 100 * 0.95;
var w = v + ( settings.darkTheme ? -10 : 10 );
this.beginPath();
this.arc( center.x, center.y, radius + settings.thickness, 0, TWO_PI );
this.lineWidth = settings.headRadius * 0.3;
this.globalAlpha = 0.2;
this.strokeStyle = 'hsl(' + h + ',' + s + '%,' + w + '%)';
this.stroke();
this.globalAlpha = 1.0;
for ( var i = 0, n = settings.tentacles; i < n; i++ ) {
tentacles[i].draw( this );
}
this.beginPath();
this.arc( center.x, center.y, radius + settings.thickness, 0, TWO_PI );
this.fillStyle = 'hsl(' + h + ',' + s + '%,' + v + '%)';
this.fill();
},
mousedown: function() {
if ( demo ) {
demo = false;
settings.interactive = true;
interactiveGUI.updateDisplay();
if ( !modified ) {
settings.length = 60;
settings.gravity = 0.1;
settings.wind = 0.0;
}
}
},
save: function() {
window.open( this.canvas.toDataURL(), 'tentacles', "top=20,left=20,width=" + this.width + ",height=" + this.height );
}
});
function onSettingsChanged() {
modified = true;
}
function onThemeChanged( dark ) {
settings.colour.h = 0;
settings.colour.s = 0;
settings.colour.v = dark ? 0.8 : 0.1;
document.body.className = dark ? 'dark' : '';
colourGUI.updateDisplay();
}
var gui = new dat.GUI();
gui.add( settings, 'headRadius' ).min( 0.0 ).max( 100.0 ).onChange( onSettingsChanged );
gui.add( settings, 'tentacles' ).min( 1 ).max( 100 ).onChange( onSettingsChanged );
gui.add( settings, 'thickness' ).min( 1.0 ).max( 40.0 ).onChange( onSettingsChanged );
gui.add( settings, 'length' ).min( 10.0 ).max( 100.0 ).onChange( onSettingsChanged );
gui.add( settings, 'gravity' ).min( -3.0 ).max( 3.0 ).onChange( onSettingsChanged );
gui.add( settings, 'wind' ).min( -3.0 ).max( 3.0 ).onChange( onSettingsChanged );
gui.add( settings, 'friction' ).min( 0.0 ).max( 1.0 ).onChange( onSettingsChanged );
var colourGUI = gui.addColor( settings, 'colour' );
gui.add( settings, 'darkTheme' ).onChange( onThemeChanged );
gui.add( settings, 'pulse' );
var interactiveGUI = gui.add( settings, 'interactive' );
gui.add( sketch, 'autoclear' );
gui.add( sketch, 'save' );
gui.close();
onThemeChanged( true );
});
Tentacle simulation using inverse kinematics
A Pen by HARUN PEHLİVAN on CodePen.
<header class="info">
<hgroup class="about">
<h1>Muscular Hydrostats</h1>
<h2>Tentacle simulation using inverse kinematics</h2>
<h3>Click creature to control</h3>
</hgroup>
<nav class="more">
<a href="https://gist.github.com/harunpehlivan/41bd915e63e8f70220c04d331f9646b3" target="_blank">Download</a>
<a href="https://github.com/harunpehlivan/Muscular-Hydrostats" target="_blank">View on Github</a>
</nav></header>
<div id="container"></div>