英文:
How to animate an element along an SVG path using only CSS and Scroll-driven Animations?
问题
我正在尝试使用Chrome Canary,并有以下代码,其中定义了一个SVG路径和一个图像元素。我想要在滚动时通过SVG路径来使图像动画化。如何仅使用CSS和滚动驱动的动画(不使用JavaScript)来实现这一目标?
<img id="car" width="256" height="256" src="
<details>
<summary>英文:</summary>
I am experimenting with Chrome Canary, and I have the following code which defines an SVG path and an image element. I would like to animate the image as I scroll, along the SVG path. How do I achieve this using only CSS and scroll-driven animations (without JavaScript)?
<!-- begin snippet: js hide: false console: true babel: false -->
<!-- language: lang-css -->
#car {
width: 25px;
height: 25px;
}
<!-- language: lang-html -->
<img id="car" width="256" height="256" src="">
<svg width="175" height="1300" preserveAspectRatio="none" viewBox="0 0 175 1300" fill="none" xmlns="http://www.w3.org/2000/svg">
<path id="my_path"
d="M 28 0 C -72 200 188 300 88 500 C -32 700 88 1100 148 1000 C 208 900 148 600 68 1000 C 28 1200 128 1300 128 1300"
stroke-width="4" stroke="#000" />
</svg>
<!-- end snippet -->
</details>
# 答案1
**得分**: -2
以下是您提供的文本的翻译:
**注意:这仅在启用了`experimental-web-platform-features`的Chrome Canary 115+中有效。**
总体而言,策略是使用`offset-path`样式属性和`url()` CSS函数引用元素应该遵循的路径。在这种情况下,您将其设置为`url(#my_path)`。然后在您的`@keyframes <name>`中,从`offset-distance: 0%;`过渡到`offset-distance: 100%;`。
然而,将滚动驱动的时间轴附加到您的动画的方法在很大程度上取决于您的页面结构以及路径在页面上的确切位置。具体来说,有两种情况。
**情况1:路径跨越整个滚动容器**
这是简单的情况。我们可以使用`animation-timeline: scroll();`将滚动时间轴与动画关联起来,该时间轴引用了最近祖先滚动容器的块轴。然后我们使用`animation: <name> linear;`将动画设置在元素上。
确保还通过`position: absolute`和`inset: 0`将元素的初始位置设置为偏移父元素的`(0,0)`。所有这些,代码应该看起来像这样:
```css
#car {
width: 25px;
height: 25px;
position: absolute;
inset: 0;
offset-path: url(#my_path);
animation: offsetDistance linear;
animation-timeline: scroll();
}
@keyframes offsetDistance {
from { offset-distance: 0%; }
to { offset-distance: 100%; }
}
情况2:路径是滚动容器的一部分
在这种情况下,我们必须使用View Progress Timeline,这需要更多的设置才能正确设置,因此更容易逐步解释。
- 在元素和路径的最近共同祖先上声明一个范围。这使得可以将视图时间轴的附加推迟到后代,后代将是路径元素,并由其他后代引用,后者将是图像元素。
#container {
position: relative;
timeline-scope: --container;
}
- 使用相同的名称将路径元素附加到视图时间轴,如下所示:
#my_path {
view-timeline-name: --container;
}
- 将命名的时间轴与元素的动画关联。
#car {
animation-timeline: --container; /* ... */
}
- 确定您的动画的时间轴范围。这也很大程度上取决于您页面的结构,可能需要一些手动测试。但是,这个工具在了解各种范围的行为方面非常有帮助。在我的示例中,我选择了以下范围:
#car {
animation-timeline: --container;
animation-range: exit-crossing -5% entry-crossing 105%;
/* ... */
}
- 最后,为了防止动画结束时出现视觉故障,请将
animation-fill-mode
更改为forwards
。
#car {
animation: <name> linear forwards;
animation-timeline: --container;
animation-range: exit-crossing -5% entry-crossing 105%;
/* ... */
}
最后,将所有内容放在一起,我们得到以下动画:
#container {
position: relative;
timeline-scope: --container;
}
#my_path {
view-timeline-name: --container;
}
#car {
width: 25px;
height: 25px;
position: absolute;
inset: 0;
offset-path: url(#my_path);
animation: offsetDistance linear forwards;
animation-timeline: --container;
animation-range: exit-crossing -5% entry-crossing 105%;
}
@keyframes offsetDistance {
from { offset-distance: 0%; }
to { offset-distance: 100%; }
}
希望这有助于您理解文本的内容和意义。
英文:
Note: This currently only works in Chrome Canary 115+ with experimental-web-platform-features
enabled.
In general, the strategy is to use the offset-path
style property with the url()
CSS function to reference the path the element should follow. In this case specifically, you would set it to url(#my_path)
. And then in your @keyframes <name>
, simply transition from offset-distance: 0%;
to offset-distance: 100%;
.
However, the method of attaching a scroll-driven timeline to your animation depends largely on the structure of your page and precisely where the path is located on the page. Namely, there are two cases.
Case 1: The path spans the entire scroll container
This is the simple case. We can associate a scroll timeline with the animation using animation-timeline: scroll();
, which references the block axis of the nearest ancestor scroll container. And then we set the animation on the element using animation: <name> linear;
.
Make sure to also set the initial position of the element to (0,0)
of the offset parent via position: absolute
and inset: 0
. All together, the code should look something like this:
<!-- begin snippet: js hide: false console: true babel: false -->
<!-- language: lang-css -->
#car {
width: 25px;
height: 25px;
position: absolute;
inset: 0;
offset-path: url(#my_path);
animation: offsetDistance linear;
animation-timeline: scroll();
}
@keyframes offsetDistance {
from { offset-distance: 0%; }
to { offset-distance: 100%; }
}
<!-- language: lang-html -->
<img id="car" width="256" height="256" src="">
<svg width="175" height="1300" viewBox="0 0 175 1300" fill="none" xmlns="http://www.w3.org/2000/svg">
<path id="my_path" d="M 28 0 C -72 200 188 300 88 500 C -32 700 88 1100 148 1000 C 208 900 148 600 68 1000 C 28 1200 128 1300 128 1300"
stroke-width="4" stroke="#000" />
</svg>
<!-- end snippet -->
Case 2: The path is a portion of the scroll container
In this case, we must use a View Progress Timeline which requires a little more effort to setup correctly, and so it's easier to explain step-by-step.
- Declare a scope on the nearest common ancestor of the element and the path. This enables the attachment of a view timeline to be deferred to a descendent, which will be the path element, and referenced by other decendents, which will be the image element.
#container {
timeline-scope: --container;
}
- Attach the path element to the view timeline with the same name, like so:
#my_path {
view-timeline-name: --container;
}
- Associate the named timeline with the element's animation.
#car {
animation-timeline: --container; /* ... */
}
- Determine your animation's timeline range. This also depends greatly on your page's structure and likely requires some manual testing. However, this tool is extremely helpful in understanding the behavior of various ranges. In my example, I choose the following:
#car {
animation-timeline: --container;
animation-range: exit-crossing -5% entry-crossing 105%;
/* ... */
}
- Lastly, to prevent visual glitches at the end of the animation, change the
animation-fill-mode
toforwards
.
#car {
animation: <name> linear forwards;
animation-timeline: --container;
animation-range: exit-crossing -5% entry-crossing 105%;
/* ... */
}
Finally, putting everything together, we get the following animation:
<!-- begin snippet: js hide: false console: true babel: false -->
<!-- language: lang-css -->
#container {
position: relative;
timeline-scope: --container;
}
#my_path {
view-timeline-name: --container;
}
#car {
width: 25px;
height: 25px;
position: absolute;
inset: 0;
offset-path: url(#my_path);
animation: offsetDistance linear forwards;
animation-timeline: --container;
animation-range: exit-crossing -5% entry-crossing 105%;
}
@keyframes offsetDistance {
from { offset-distance: 0%; }
to { offset-distance: 100%; }
}
<!-- language: lang-html -->
<div style="height: 150px;"></div>
<div id="container">
<img id="car" width="256" height="256"
src="">
<svg width="175" height="1300" preserveAspectRatio="none" viewBox="0 0 175 1300" fill="none"
xmlns="http://www.w3.org/2000/svg">
<path id="my_path" d="M 28 0 C -72 200 188 300 88 500 C -32 700 88 1100 148 1000 C 208 900 148 600 68 1000 C 28 1200 128 1300 128 1300"
stroke-width="4" stroke="#000" />
</svg>
</div>
<div style="height: 150px;"></div>
<!-- end snippet -->
通过集体智慧和协作来改善编程学习和解决问题的方式。致力于成为全球开发者共同参与的知识库,让每个人都能够通过互相帮助和分享经验来进步。
评论