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

wut*_*ang 2 css animation svg css3 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>
Run Code Online (Sandbox Code Playgroud)

我不确定我尝试做的事情是否可能,所以我只是寻找指针,所以我可以得到这一个轨道.

编辑:

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

在此输入图像描述

Wea*_*.py 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" />
Run Code Online (Sandbox Code Playgroud)

适用clipPathdiv:

对于应用clip-path,如果你想maximum browser support,你可以导入div到一个svg通过元素svgforeignObject元素.然后,应用clip-pathforeignObject.

这里的browser support,如果您将clip-path通过CSS.


动画clipPath:

现在,对准菜单项中间的指针向右,你需要的JavaScript做一些计算和动画clipPathtransform属性.

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

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

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

同步代码意味着一次执行一个操作,并且直到一个操作结束,代码执行被阻止进入下一个操作.


网站的精确复制品:

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

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

Demo on CodePen

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;
Run Code Online (Sandbox Code Playgroud)
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;
}
Run Code Online (Sandbox Code Playgroud)
<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>
Run Code Online (Sandbox Code Playgroud)


更好:

保留指针的最后位置.

Demo on CodePen

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;
Run Code Online (Sandbox Code Playgroud)
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: 600px;
  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);
}
Run Code Online (Sandbox Code Playgroud)
<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>
Run Code Online (Sandbox Code Playgroud)