如何在动画中使用剪辑路径?

时间:2014-12-23 22:07:27

标签: css css3 animation svg clip

我已经阅读了很多帖子,并且一直在尝试动画剪辑路径。此时我只能在Firefox中显示剪辑路径,它总是将路径固定为0,0。

以下是示例代码 - > http://jsfiddle.net/justintense/4torLok9/1/

我使用简单的内联SVG作为路径:

    <clipPath id="clipping">

        <polygon fill="#FFFFFF" points="40,35 40,15 20,35   "/>
        <polygon fill="#FFFFFF" points="0,35 20,35 0,15     "/>

        <circle id="white_overlay_9_" fill="#FFFFFF" cx="20" cy="18.77" r="7.393"/>
    </clipPath>

我不确定我尝试做的事情是否可行,所以我只是寻找指针,这样我才能获得这一曲目。

修改

我正在尝试重新创建类似于此网站的行为--------&gt; http://uppymama.com/

enter image description here

1 个答案:

答案 0 :(得分:6)

定义clipPath

您的clipPath中不需要三个元素,您可以将其简化为单个path元素。

<path d="M0,0 L40,0 L20,15z M20,5 m-5,0 a5,5 0 1,0 10,0 a5,5 0 1,0 -10,0z" />

clipPath应用于div

要应用clip-path,如果您想要 maximum browser support ,则可以通过{{1}将div导入svg元素} svg元素。然后,将foreignObject应用于clip-path

如果你通过CSS应用foreignObject,那么这是 browser support


动画clip-path

现在,为了将指针对齐在菜单项的中间,您需要使用JavaScript进行一些计算并为clipPath的{​​{1}}属性设置动画。

  1. 当页面加载计算从左窗口角落到第一个菜单项目的距离并将clipPath移动到该坐标时。

  2. 当窗口调整大小时,再次计算距离并移动指针。

  3. 单击菜单项时,使用transform同步代码计算距离并为指针设置动画,直到它到达正确的位置。

  4.   

    同步代码表示一次执行一个操作,并且在一个操作结束之前,代码执行被阻止进入下一个操作。


    site的完全复制品:

    正如您在该网站上看到的那样,指针不居中。

    我的解决方案将始终根据菜单项的宽度居中指针。

    演示CodePen

    clip-path
    setTimeout()
    var items = document.getElementsByClassName('menu-item');
    var clipPath = document.getElementById('clip-path');
    var lastActive = items[0];
    for (i = 0; i < items.length; i++) {
      items[i].addEventListener('click', function() {
        var x = this.getBoundingClientRect().left + (this.offsetWidth / 2) - 20;
        animStart(x);
        lastActive = this;
      })
    }
    
    function animStart(end) {
      for (start = 0; start <= end; start += 0.1) {
        anim(start);
      }
    }
    
    function anim(start) {
      setTimeout(function() {
        clipPath.setAttribute('transform', 'translate(' + start + ', 0)');
      }, 1 * start);
    }
    
    function align() {
      var x = lastActive.getBoundingClientRect().left + (lastActive.offsetWidth / 2) - 20;
      animStart(x);
    }
    
    function resize() {
      var x = lastActive.getBoundingClientRect().left + (lastActive.offsetWidth / 2) - 20;
      clipPath.setAttribute('transform', 'translate(' + x + ', 0)');
    }
    window.onresize = resize;
    window.onload = align;


    更好:

    保留指针的最后位置。

    演示CodePen

    body {
      background: #222222;
      margin: 0;
    }
    .nav-container {
      top: 20px;
      position: relative;
      width: 100%;
      height: 50px;
      background: repeating-linear-gradient(90deg, palegoldenrod 5px, palegreen 10px, palegreen 15px, paleturquoise 15px, paleturquoise 20px, plum 20px, plum 25px);
    }
    .nav {
      width: 100%;
      height: 50px;
      margin: 0 auto;
      text-align: center;
    }
    .menu-item {
      display: inline-block;
      width: 70px;
      padding: 17px 20px;
      color: #222222;
      font-size: 14px;
    }
    .menu-item:hover {
      color: seagreen;
      cursor: pointer;
    }
    #bottom {
      width: 100%;
      height: 35px;
      background: repeating-linear-gradient(90deg, palegoldenrod 5px, palegreen 10px, palegreen 15px, paleturquoise 15px, paleturquoise 20px, plum 20px, plum 25px);
    }
    #clip-path {
      transition: 0.5s transform;
    }
    <body>
      <div class="nav-container">
        <div class="nav">
          <div id="menu-item-1" class="menu-item">Home</div>
          <div id="menu-item-2" class="menu-item">About</div>
          <div id="menu-item-3" class="menu-item">Services</div>
          <div id="menu-item-4" class="menu-item">Locations</div>
          <div id="menu-item-5" class="menu-item">Contact Us</div>
        </div>
        <svg style="position: relative; top: -1px;" class="svg-defs" height="35" width="100%">
          <defs>
            <clipPath id="clipping">
              <path id="clip-path" transform="translate(0,0)" d="M0,0 L40,0 L20,15z M20,6 m-5,0 a5,5 0 1,0 10,0 a5,5 0 1,0 -10,0z" />
            </clipPath>
          </defs>
          <foreignObject x="0" y="0" clip-path="url(#clipping)" height="35" width="100%">
            <div id="bottom"></div>
          </foreignObject>
        </svg>
      </div>
    </body>
    var items = document.getElementsByClassName('menu-item');
    var clipPath = document.getElementById('clip-path');
    var last = items[0];
    for (i = 0; i < items.length; i++) {
      items[i].addEventListener('click', function() {
        var x = this.getBoundingClientRect().left + (this.offsetWidth / 2) - 20;
        var currentX = clipPath.getAttribute('transform').replace('translate(', '');
        currentX = parseInt(currentX.replace(', 0)', ''), 10);
        animStart(currentX, x);
        last = this;
      })
    }
    
    function animStart(begin, end) {
      if (begin < end) {
        for (start = begin; start <= end; start += 0.1) {
          anim(start);
        }
      } else {
        var c = end;
        for (start = begin; end <= start; start -= 0.1) {
          animD(start, c);
          c += 0.1
        }
      }
    }
    
    function anim(start) {
      setTimeout(function() {
        clipPath.setAttribute('transform', 'translate(' + start + ', 0)');
      }, 1 * start);
    }
    
    function animD(start, c) {
      setTimeout(function() {
        clipPath.setAttribute('transform', 'translate(' + start + ', 0)');
      }, 1 * c);
    }
    
    function align() {
      var x = last.getBoundingClientRect().left + (last.offsetWidth / 2) - 20;
      animStart(0, x);
    }
    
    function resize() {
      var x = last.getBoundingClientRect().left + (last.offsetWidth / 2) - 20;
      clipPath.setAttribute('transform', 'translate(' + x + ', 0)');
    }
    window.onresize = resize;
    window.onload = align;