Muscular Hydrostats با جاوا اسکریپت و CSS

Muscular Hydrostats

HTML

<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://github.com/soulwire/Muscular-Hydrostats/zipball/master" target="_blank">Download</a>
    <a href="https://github.com/soulwire/Muscular-Hydrostats" target="_blank">View on Github</a>
  </nav></header>
<div id="container"></div>

CSS

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 {
  ۰% { -webkit-transform: rotateY(120deg); }
  ۱۰۰% { -webkit-transform: rotateY(0deg); }
}

@-moz-keyframes show-info {
  ۰% { -moz-transform: rotateY(120deg); }
  ۱۰۰% { -moz-transform: rotateY(0deg); }
}

@-ms-keyframes show-info {
  ۰% { -ms-transform: rotateY(120deg); }
  ۱۰۰% { -ms-transform: rotateY(0deg); }
}

@-o-keyframes show-info {
  ۰% { -o-transform: rotateY(120deg); }
  ۱۰۰% { -o-transform: rotateY(0deg); }
}

@keyframes show-info {
  ۰% { transform: rotateY(120deg); }
  ۱۰۰% { 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;
}

Java 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 );
});

 

دموی آنلاین

تا بارگذاری کامل کمی صبر کنید

See the Pen Muscular Hydrostats by Justin Windle (@soulwire) on CodePen

سوالی دارید؟

بدون گرفتن نتیجه اینجا رو ترک نکنید! هر سوالی دارید میتونید در عرض ۱۵ ثانیه ثبت نام و خیلی زود جواب بگیرید.

پاسخ دهید

نشانی ایمیل شما منتشر نخواهد شد. بخش‌های موردنیاز علامت‌گذاری شده‌اند *

به سوال زیر پاسخ بدهید *