动画缩放旋转 svg 元素

Animate scale rotate svg element

假设我有一个用 inkscape 制作的 svg。 在这个带有视图框的 SVG 集中,我想为 SVG 中的每个元素设置动画。 平移或不透明度没有问题...但是当我需要旋转或缩放单个元素时,它表现得很奇怪。

我试图正确理解视图框的概念,但我需要一些帮助。

我知道当我只有一个viewbox时我只有一个原点我应该设置多个viewbox吗?

<?xml version="1.0" encoding="UTF-8"?>
<svg id="SVGRoot" version="1.1" viewBox="0 0 700 500" xmlns="http://www.w3.org/2000/svg">
 
  // rotate or scale acting weird
  <ellipse id="path9238" cx="332.91" cy="143.85" rx="64.941" ry="67.676" style="fill-rule:evenodd;fill:#f00;stroke:#000"/>
  // rotate or scale acting weird
  <rect id="rect9240" x="400.59" y="270.31" width="173.63" height="177.73" style="fill-rule:evenodd;fill:#f00;paint-order:normal"/>
  // rotate or scale acting weird
  <path id="path9242" d="m233.79 453.52-153.64-138.25 196.55-63.937z" style="fill-rule:evenodd;fill:#f00;paint-order:normal"/>
 
</svg>

我正在使用 anime.js 3.0 或 CSS 或者我可以尝试其他任何东西

我猜你指的是 transform-origin CSS 属性。它与 svg 文件的中心点有关。然后,您需要计算与文档中心点相关的元素的中心点以进行动画处理。
对于动画,您可以使用 CSS animations.

viewBox的基础是4个数字是“x y width height”。一般来说x/y坐标是0,这样你的原点就在左上角。人们经常做的事情就是把原点在中间。

为此,您将 viewBox 左上角移动宽度的一半和高度的一半。在你的情况下“-350 -250 700 500”。

在Svg中,任何图形的坐标总是有一个从SVG左上角算起的绝对值canvas。
因此,应用scale(2)命令时,图形的中心坐标也会增加一倍,图形会向右下方偏移。

<svg id="SVGRoot" version="1.1"  width="500" height="500" viewBox="0 0 500 500" xmlns="http://www.w3.org/2000/svg" style="border:1px solid grey;">
     <rect  id="rect9240" transform="scale(2)" x="100" y="100" width="100" height="100"   style="fill-rule:evenodd;fill:#f00;> stroke:#000; paint-order:normal">
 
 <animateTransform
     xlink:href="#rect9240"
  attributeName="transform"
  type="scale"
  values="1;2;2;1;1"
  dur="8s"
  fill="freeze"
  repeatcount="indefinite"
  />
 </rect>  
      <circle cx="150" cy="150" r="3" fill="black" />
  <circle cx="300" cy="300" r="3" fill="dodgerblue" />
  <text x="150" y="140" font-size="16px" text-anchor="middle" > Center (150,150) </text>
   <text x="300" y="290" font-size="16px" text-anchor="middle" > Center (300,300) </text>
</svg>

把return放大图放回原来的位置,必须用命令translate (X, Y)

@Paul LeBeau 的精彩 post here 对此进行了详细解释。

CSS解决方案

为了不计算图形中心的位置,可以使用CSS规则transform-box: fill-box

选择 fill-box 属性的值时

The object bounding box is used as the reference box.

下面是增加和减少数字大小的例子:

svg {
width:50%;
}
.ellipse1, .rect1, .path1 {
transform-box: fill-box;
    animation: scale 3s linear infinite alternate;
 animation-direction: alternate;
transform-origin:50% 50%;
}


@keyframes scale {
    0% { transform: scale(0.5); }
    100% {  transform: scale(1); }
}
<svg id="SVGRoot"  version="1.1" viewBox="0 0 700 500" xmlns="http://www.w3.org/2000/svg">
   <ellipse class="ellipse1" id="path9238" cx="332.91" cy="143.85" rx="64.941" ry="67.676" style="fill-rule:evenodd;fill:#f00;stroke:#000"/>
  
  <rect class="rect1" id="rect9240" x="400.59" y="270.31" width="173.63" height="177.73" style="fill-rule:evenodd;fill:#f00; stroke:#000; paint-order:normal"/>
  
  <path  class="path1" id="path9242" d="m233.79 453.52-153.64-138.25 196.55-63.937z" style="fill-rule:evenodd;fill:#f00;stroke:#000; paint-order:normal"/>
 </svg>

  • 旋转示例

svg {
width:50%;
}
.ellipse1, .rect1, .path1 {
transform-box: fill-box;
    animation: spin 4s linear infinite alternate;
transform-origin:50% 50%;
}


@keyframes spin {
    0% { transform: rotate(0deg); }
    100% {  transform: rotate(359deg); }
}




  <path  class="path1" id="path9242" d="m233.79 453.52-153.64-138.25 196.55-63.937z" style="fill-rule:evenodd;fill:#f00;stroke:#000; paint-order:normal"/>
 
</svg>
<svg id="SVGRoot" version="1.1" viewBox="0 0 700 500" xmlns="http://www.w3.org/2000/svg">

  <ellipse class="ellipse1" id="path9238" cx="332.91" cy="143.85" rx="64.941" ry="67.676" style="fill-rule:evenodd;fill:#f00;stroke:#000"/>

  <rect class="rect1" id="rect9240" x="400.59" y="270.31" width="173.63" height="177.73" style="fill-rule:evenodd;fill:#f00; stroke:#000; paint-order:normal"/>

  <path  class="path1" id="path9242" d="m233.79 453.52-153.64-138.25 196.55-63.937z" style="fill-rule:evenodd;fill:#f00;stroke:#000; paint-order:normal"/>
 
</svg>

  • 增加和旋转

svg {
width:50%;
}
.ellipse1, .rect1, .path1 {
transform-box: fill-box;
    animation: scale1 4s linear, spin 4s linear 4s infinite alternate;
 transform-origin:50% 50%;
}


@keyframes spin {
    0% { transform: rotate(0deg); }
    100% { transform: rotate(360deg); }
}

@keyframes scale1 {
    0% { transform: scale(0.5);}
    100% {  transform: scale(1);}
}
<svg id="SVGRoot" version="1.1" viewBox="0 0 700 500" xmlns="http://www.w3.org/2000/svg">
   
  <ellipse class="ellipse1" id="path9238" cx="332.91" cy="143.85" rx="64.941" ry="67.676" style="fill-rule:evenodd;fill:#f00;stroke:#000"/>
  
  <rect class="rect1" id="rect9240" x="400.59" y="270.31" width="173.63" height="177.73" style="fill-rule:evenodd;fill:#f00; stroke:#000; paint-order:normal"/>
 
  <path  class="path1" id="path9242" d="m233.79 453.52-153.64-138.25 196.55-63.937z" style="fill-rule:evenodd;fill:#f00;stroke:#000; paint-order:normal"/>
 
</svg>