To trigger an animation when the SVG scrolls into view, there are two steps you need to resolve:
- detect when the SVG is visible
- have the animation paused initially, then trigger it when you want
Detecting when an element is visible
You need to add an event handler for scroll events. In that event handler, you compare the page scroll location (window.scrollY
) against the SVG's location on the page. You can get that using getBoundingClientRect()
.
See the demos below for example code implementing this technique.
Starting an animation running
The second step is actually triggering the animation at that time. How you keep an animation paused and then start it running, depends on the method of animation you are using.
SVG SMIL animation
In this first example, we are using the SMIL animation elements that are part of the SVG standard.
To have an animation wait for you to start it running, set the begin
attribute to "indefinite"
. The you can start it running by calling beginElement()
on the <animate>
element.
// Get the position on the page of the SVG
var svgLocation = document.getElementById("mysvg").getBoundingClientRect();
// Scroll offset that triggers animation start.
// In this case it is the bottom of the SVG.
var offsetToTriggerAnimation = svgLocation.y + svgLocation.height;
// Function to handle the scroll event.
// Add an event handler to the document for the "onscroll" event
function scrollAnimTriggerCheck(evt)
{
var viewBottom = window.scrollY + window.innerHeight;
if (viewBottom > offsetToTriggerAnimation) {
// Start the SMIL animation
document.getElementById("anim").beginElement();
// Remove the event handler so it doesn't trigger again
document.removeEventListener("scroll", scrollAnimTriggerCheck);
}
}
// Add an event handler to the document for the "onscroll" event
document.addEventListener("scroll", scrollAnimTriggerCheck);
svg {
border: solid 1px grey;
}
p {
margin-bottom: 1000px;
}
<p>Scroll down</p>
<svg id="mysvg">
<rect x="0" width="150" height="150" fill="rebeccapurple">
<animate id="anim" attributeName="x" from="0" to="150" dur="2s"
fill="freeze" begin="indefinite"/>
</rect>
</svg>
<br>
<br>
<br>
<br>
与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…