Demo available on http://maxwellito.github.io/vivus
Vivus is a lightweight JavaScript class (with no dependencies) that allows you to animate SVGs, giving them the appearence of being drawn. There are a variety of different animations available, as well as the option to create a custom script to draw your SVG in whatever way you like.
Available via:
- Bower:
bower install vivus
- NPM:
npm install vivus
- SPM:
spm install vivus
- jsDelivr CDN:
//cdn.jsdelivr.net/vivus/0.2.2/vivus.min.js
- WebJars
Join the conversation on Gitter
On the following images, the pink color represents the duration
value, and the blue one is for delay
value.
Every path element is drawn at the same time with a small delay at the start. This is currently the default animation.
Each line is drawn asynchronously. They all start and finish at the same time, hence the name async
.
Each path element is drawn one after the other. This animation gives the best impression of live drawing. The duration for each line depends on their length to make a constant drawing speed.
To get this effect, the script uses the CSS property strokeDashoffset
. This property manages the stroke offset on every line of the SVG. Now, all we have to do is add some JavaScript to update this value progressively and the magic begins.
However, there's a problem with this. The strokeDashoffset
property is only available on the path elements. This is an issue because in an SVG there are a lot of elements such as circle
, rect
, line
and polyline
which will break the animation. So to fix this, there is another class available in the repo called pathformer
. It's made for transforming all objects of your SVG into path
elements to be able to use strokeDashoffset
and animate your SVGs.
The animation always draws elements in the same order as they are defined in the SVG tag.
There are few conditions that your SVG must meet:
-
The SVG must be inline in your HTML, otherwise JavaScript cannot manipulate it.
-
All elements must have a stroke property and cannot be filled. This is because the animation only looks to progressively draw strokes and will not check for filled colours. For example: fill: "none"; stroke: "#FFF";
-
You shoud avoid creating any hidden path elements in your SVG. Vivus considers them all eligible to be animated, so it is advised to remove them before playing with it. If they are not removed the animation might not achieve the desired effect, with blank areas and gaps appearing.
-
text
elements aren't allowed, they cannot be transformed intopath
elements. See #22 for more details.
The code is inspired from other repositories. The drawer is inspired from the excellent Codrops about the post SVG Drawing Animation (if you don't know this website, get ready to have your mind blown). Then for the pathformer, there is a lot of work from SVGPathConverter by Waest.
As I said, no dependencies here. All you need to do is include the scripts.
// Inline SVG
new Vivus('my-svg-id', {type: 'delayed', duration: 200}, myCallback);
// Dynamic load
new Vivus('my-div-id', {type: 'delayed', duration: 200, file: 'link/to/my.svg'}, myCallback);
The Vivus constructor asks for 3 parameters:
- ID (or object) of DOM element to interact with.
It can be an inline SVG or a wrapper element to append an object tag from the optionfile
- Option object (described in the following)
- Callback to call at the end of the animation (optional)
The option object must respect the following structure :
type
(string) defines what kind of animation will be used:delayed
,async
,oneByOne
,script
,scenario
orscenario-sync
duration
(integer) animation duration, in framesstart
(string) defines how to trigger the animationinViewport
once the SVG is in the viewportmanual
gives you the freedom to call draw method to startautostart
makes it start right now
delay
(integer) time between the drawing of first and last path, in frames (only for delayed animations)file
(string) link to the SVG to animate. If set, Vivus will create an object tag and append it to the DOM element given to the constructor. Be careful, use theonReady
callback before playing with the Vivus instance.onReady
(function) function called when the instance is ready to playdashGap
(integer) whitespace extra margin between dashes. The default value is2
. Increase it in case of glitches at the initial state of the animationpathTimingFunction
(function) timing animation function for each path element of the SVG. The function must accept a value between 0 and 1, then return a number between 0 and 1.animTimingFunction
(function) timing animation function for the complete SVG. The function must accept a value between 0 and 1, then return a number between 0 and 1.forceRender
(boolean) force the browser to re-render all updated path items. By default, the value istrue
on IE only. (check the 'troubleshoot' section for more details)selfDestroy
(boolean) removes all extra styling on the SVG, and leaves it as original
The Vivus object has 3 control methods:
play(speed)
Plays the animation with the speed given in parameter. This value can be negative to go backward, between 0 and 1 to go slowly, or superior to 1 to go fast. By default the value is 1.stop()
Stops the animation.reset()
Reinitialises the SVG to the original state: undrawn.finish()
Set the SVG to the final state: drawn.setFrameProgress(progress)
Set the progress of the animation. Progress must be a number between 0 and 1.destroy()
Reset the SVG but make the instance out of order.
These control methods return the object so you can chain the actions.
var myVivus = new Vivus('my-svg-id');
myVivus
.stop()
.reset()
.play(2)
To give more freedom, it's possible to override the animation of each path and/or the entire SVG. It works a bit like the CSS animation timing function. But instead of using a cubic-bezier function, it use a simple JavaScript function. It must accept a number as parameter (between 0 to 1), then return a number (also between 0 and 1). It's a hook.
If you don't want to create your own, timing methods are available via the constructor object: EASE
, EASE_IN
, EASE_OUT
and EASE_OUT_BOUNCE
. Then set it in the option object to enjoy them.
// Here, the ease animation will be use for the global drawing.
new Vivus('my-svg-id', {
type: 'delayed',
duration: 200,
animTimingFunction: Vivus.EASE
}, myCallback);
WARNING: animTimingFunction
is called at every frame of the animation, and pathTimingFunction
is also called at every frame for each path of your SVG. So be careful about them. Keep it simple, or it can affect the performances.
This feature allows you to script the animation of your SVG. For this, the custom values will be set directly in the DOM of the SVG.
This type is easier to understand, but longer to implement. You just have to define the start and duration of each element with data-start
and data-duration
attributes. If it is missing, it will use the default value given to the constructor.
The best part of this type is the flexibility it provides. You don't have to respect the order/stack of the SVG and you can start with the last element, then continue with the first to finish with all the rest at the same time.
You will then have to define custom rules for each element in your SVG via extra attributes in your SVG DOM :
data-start
(integer) time when the animation must start, in framesdata-duration
(integer) animation duration of this path, in frames
<svg>
<path data-start="0" data-duration="10" .../>
<path data-start="20" data-duration="10" .../>
<path data-start="20" data-duration="20" .../>
<path data-start="0" data-duration="30" .../>
</svg>
It's not the sexiest code ever, but it's quite flexible. In addition to this, the behaviour is fairly different.
By using this animation type, the default behaviour is the same as oneByOne
. However, you can define some properties on a specific path item such as the duration, the delay to start (from the end of the previous path) and if it should be played asynchronously.
data-delay
(integer) time between the end of the animation of the previous path and the start of the current path, in framesdata-duration
(integer) duration of this path animation, in framesdata-async
(no value required) make the drawing of this path asynchronous. It means the next path will start at the same time. If a path does not have an attribute for duration or delay then the default values, set in the options, will be used.
Example: here is a simple SVG containing 5 elements. With the following options {duration: 20, delay: 0}
, we should get this timeline
This looks like 'oneByOne' animation, synchronous mode. But to make it a bit custom, here is what I can do:
<svg>
<path data-duration="10" .../>
<path data-delay="10" data-async .../>
<path data-delay="15" .../>
<path data-duration="10" data-delay="45" data-async .../>
<path data-duration="50" data-delay="5" .../>
</svg>
This scenario should give us
I'm sorry if it does not look very sexy, and it's not really easy to use. I'm happy to make any changes, as long as the idea sounds interesting. Post an issue and I'll be very happy to talk about it!
To make it simpler a gulp file is set up to automise minifying, JShint and tests.
If you have never used Gulp before this is a good opportunity. To use it, you need to install NodeJS first then run sudo npm install -g gulp
.
To start, you will need to install the repo dependencies:
$ npm install
Then you can run Gulp with one of the following tasks:
distrib
make the build (generatedist/vivus.js
anddist/vivus.min.js
)lint
run JShint on the source filestest
run Karmadevelop
keep watching your files, if any change is applied, Gulp will run the task(s) related to it.
Some SVG weren't working at all. The only solution found was to clone and replace each updated path element. Of course this solution requires more resources and a lot of DOM manipulation, but it will give a smooth animation like other browsers. This fallback is only applied on Internet Explorer (all versions), and can be disabled via the option forceRender
.
Replacing each updated path by a clone was the only way to force IE to re-render the SVG. On some SVGs this trick is not necessary, but IE can be a bit tricky with this. If you're worried about performances, I would recommend you to check if your SVG works correctly by disabling the forceRender
option. If it works correctly on IE, then keep it like this.
By default, forceRender
is true
on Internet Explorer only.
For Firefox users, you might encounter some glitches depending on your SVG and browser version. On versions before 36, there is a problem retrieving path length via getTotalLength
method. Returning 174321516544 instead of 209 (I'm not exaggerating, this comes from a real case), messing up the entire animation treatment. Unfortunately, there's nothing that this library can do, this is due to Firefox. I hope to find a workaround, but at the moment I can only recommend that you test your animation on previous versions of Firefox.
For an easier debug have a look to the attribute map
of your Vivus object. This contains the mapping of your animation. If you're using Google Chrome, I recommend you use console.table
to get a nice output of the array which will make your debug easier.
var logo = new Vivus('myLogo', {type: 'scenario-sync'});
// The property 'map' contain all the SVG mapping
console.table(logo.map);