This is a rewrite of the original NanoVG library using the Zig programming language.
NanoVG is a small anti-aliased hardware-accelerated vector graphics library. It has a lean API modeled after the HTML5 canvas API. It is aimed to be a practical and fun toolset for building scalable user interfaces or any other real time visualizations.

There’s a WebAssembly example using WebGL which you can immediately try here: https://fabioarnold.github.io/nanovg-zig. The source for this example can be found in example_wasm.zig and can be built by running zig build -Dtarget=wasm32-freestanding.
A native cross-platform example using GLFW can be found in example_glfw.zig and can be built and run with zig build run. It requires GLFW to be installed. On Windows vcpkg is an additional requirement.
For an example on how to use nanovg-zig in your project’s build.zig you can take a look at https://github.com/fabioarnold/MiniPixel/blob/main/build.zig.
The NanoVG API is modeled loosely on the HTML5 canvas API. If you know canvas, you’re up to speed with NanoVG in no time.
The drawing context is created using a backend-specific initialization function. If you’re using the OpenGL backend the context is created as follows:
const nvg = @import("nanovg");
...
var vg = try nvg.gl.init(allocator, .{
	.debug = true,
});
defer vg.deinit();The second parameter defines options for creating the renderer.
antialias means that the renderer adjusts the geometry to include anti-aliasing. If you’re using MSAA, you can omit this option to be default initialized as false.stencil_strokes means that the render uses better quality rendering for (overlapping) strokes. The quality is mostly visible on wider strokes. If you want speed, you can omit this option to be default initialized as false.Currently, there is an OpenGL backend for NanoVG: nanovg_gl.zig for OpenGL 2.0 and WebGL. WebGL is automatically chosen when targeting WebAssembly. There’s an interface called Params defined in internal.zig, which can be implemented by additional backends.
NOTE: The render target you’re rendering to must have a stencil buffer.
Drawing a simple shape using NanoVG consists of four steps:
vg.beginPath();
vg.rect(100,100, 120,30);
vg.fillColor(nvg.rgba(255,192,0,255));
vg.fill();Calling beginPath() will clear any existing paths and start drawing from a blank slate. There are a number of functions to define the path to draw, such as rectangle, rounded rectangle and ellipse, or you can use the common moveTo, lineTo, bezierTo and arcTo API to compose a path step-by-step.
Because of the way the rendering backend is built in NanoVG, drawing a composite path - that is a path consisting of multiple paths defining holes and fills - is a bit more involved. NanoVG uses the even-odd filling rule and by default the paths are wound in counterclockwise order. Keep that in mind when drawing using the low-level drawing API. In order to wind one of the predefined shapes as a hole, you should call pathWinding(nvg.Winding.solidity(.hole)), or pathWinding(.cw) after defining the path.
vg.beginPath();
vg.rect(100,100, 120,30);
vg.circle(120,120, 5);
vg.pathWinding(.cw); // Mark circle as a hole.
vg.fillColor(nvg.rgba(255,192,0,255));
vg.fill();Here’s how to integrate nanovg-zig into your Zig project:
Add the library as a dependency:
Run the following command in your project directory to fetch and save nanovg-zig as a dependency:
zig fetch --save git+https://github.com/fabioarnold/nanovg-zig.gitImport the library in your build.zig file:
Add the following code snippet to your build.zig file. This creates a dependency on nanovg-zig and imports the nanovg module into your executable:
const nanovg_zig = b.dependency("nanovg-zig", .{
    .target = target,
    .optimize = optimize,
});
exe.root_module.addImport("nanovg", nanovg_zig.module("nanovg"));Replace exe with your target executable.
Link against OpenGL:
nanovg-zig relies on OpenGL for rendering.  You must manually link against OpenGL in your build.zig file.  The build.zig file in the repository demonstrates how to do this.
Use NanoVG in your code:
You can now import and use the nanovg module in your Zig source files:
const nvg = @import("nanovg");Refer to the “Creating a drawing context” and “Drawing shapes with NanoVG” sections above for examples of how to use the NanoVG API.
See nanovg.zig for an API reference.
The original library and this rewrite are licensed under the zlib license
Fonts used in the examples:
Uses stb_truetype for font rendering. Uses stb_image for image loading.