Japanese kana SVGs with animatable strokes.
The characters stroke SVGs provided here are based on Klee One (licensed under the SIL Open Font License) which closely resembles handwriting. This makes them ideal for teaching how to write.
Many existing animated stroke diagrams are based on simple line drawings which look unpolished. Other diagrams are based on fonts which don't resemble handwriting. While these projects didn't fit my needs, they do provide diagrams for a much larger set of characters.
The following character shapes are modified from Klee One for clarity.
- ク
- グ
- ム
- ル
- レ
This folder contains optimized svg files with animatable strokes for the given character.
This is where the raw unoptimized svg files are stored and directly edited using Inkscape/whatever.
These files should NOT be used as is.
This Inkscape extention partially automates SVG creation.
Here is some CSS to have the stroke animation automatically start on page load. This is based on the CSS from AnimCJK. However, by itself, this only runs on page load.
svg[data-strokesvg] > g[data-strokesvg="strokes"] > * {
--time: 0.7s; /* time to draw a "max length stroke" */
--gap: 0.7s; /* time between stroke starts */
--delay: 0.5s; /* time before first stroke */
animation: s var(--time) linear forwards calc(
var(--i) * var(--gap) + var(--delay)
);
stroke-dasharray: 3333;
stroke-dashoffset: 3333;
}
@keyframes s {
to {
stroke-dashoffset: 0;
}
}
You can add javascript to restart the CSS animation programmatically, but you can also just control the stroke-dasharray
and stroke-dashoffset
properties without using the CSS animation.
See strokeAnimator
in index.js
for an overengineered example. Feel free to use it.
The shadows and strokes have a default color which you can override using CSS variables.
/* default colors */
svg[data-strokesvg] {
--shadow: #ccc;
--stroke: #000;
}
For more specific customization, you can just change the fill
, stroke
, and anything else directly.
svg[data-strokesvg] > g[data-strokesvg="shadows"] > :nth-child(2) {
fill: blue;
}
svg[data-strokesvg] > g[data-strokesvg="strokes"] > :nth-child(2) {
stroke: red;
}
For clarity with using custom css, see the format section below or look at the file directly.
<svg data-strokesvg="あ">
<g data-strokesvg="shadows" style="fill:var(--shadow,#ccc)">
<path .../>
... more paths or <g> with <path> children
</g>
<g
data-strokesvg="strokes"
style="stroke:var(--stroke,#000);fill:none;stroke-width:128;stroke-linecap:round"
>
<path style="--i:0" clip-path="url(#...)" />
... more paths or <g> with <path> children
</g>
</svg>
The first <g>
element contains elements representing shadows for each stroke/stroke section. These are visible while the strokes are animated. These are also used as clip masks for the strokes.
The second <g>
element contains elements representing each stroke. These can be animated using the stroke-dasharray
and stroke-dashoffset
css properties. The stroke order is represented by the element order and duplicated via the --i
css variable (useful for the CSS snippet).
Some strokes self intersect, like in loops, so they are created from multiple stroke "shadows" and corresponding strokes so that the clipping works correctly. In these cases, parts of the same stroke are grouped together. See /dist/hiragana/あ.svg
for an example.
Run the demo site locally.
pnpm run dev
Generate optimized files in /dist
after changing files in the /src
folder.
pnpm run min
Steps 1, 2, and 4 can be automated using the Inkscape extension at src/org.strokesvg.helper
.
- Create an SVG document with the desired size (1024 x 1024 px).
- Make a text element with the desired font-family (Klee One) and font-size (1024px). Ensure the character is placed correctly on the page. In Inkscape, this is done by creating a flow text element with the top left corner snapped to the page's top left corner.
- Convert the text element to an object (path) and then separate the shape into stroke shapes.
- Place strokes in stroke order, then give each stroke shape a distinct color (for clarity).
-
Stroke width is 128px, line-cap is round.
-
Draw lines that cover each stroke shape. The start line-cap should be placed entirely before the start of the stroke, and the end line-cap should just barely cover the end of the stroke.
-
If a stroke unavoidably self-intersects, do the following.
- Split the stroke shape into 2 (or more if necessary).
- Duplicate the drawn stroke. Adjust the path of the second so that it doesn't self intersect the first stroke shape.
- For each stroke, clone the shape and use it as a clip path for the drawn stroke.
- Make sure the strokes are above the shapes at the end.
pnpm run min
- Make sure the animation looks good via the demo site.
stroke-dasharray
andstroke-dashoffset
act weird- There are precision issues with stroke lengths, especially after reducing number precision, which I addressed by adding an epsilon whenever dealing with stroke paths in my
strokeAnimator
. Otherwise strokes can flicker in and out of view whenstroke-dashoffset
is set to stroke length. - Relatedly, offseting
stroke-linecap: round
strokes withstroke-dashoffset
acts inconsistently (popping into view instead of sliding as offset changes). I found that if I offset the nodes by 1/2 path width before a stroke starts in the SVG itself, then the round linecap slides into view as expected.
- There are precision issues with stroke lengths, especially after reducing number precision, which I addressed by adding an epsilon whenever dealing with stroke paths in my