#physics-engine #png #bevy #collider #physics-2d #2d #rapier

bevy_collider_gen

a library for generating colliders, for bevy apps, from images with transparency

6 releases (3 breaking)

new 0.4.0 Jan 8, 2025
0.3.0 Nov 19, 2024
0.2.2 Aug 14, 2024
0.2.1 May 12, 2024
0.1.0 Mar 2, 2024

#122 in Game dev

Download history 17/week @ 2024-09-23 17/week @ 2024-09-30 5/week @ 2024-10-07 24/week @ 2024-10-14 16/week @ 2024-11-04 144/week @ 2024-11-18 55/week @ 2024-12-02 80/week @ 2024-12-09 14/week @ 2024-12-16 28/week @ 2024-12-23 145/week @ 2025-01-06

196 downloads per month

MIT/Apache

42KB
252 lines

bevy_collider_gen

Crates.io Crates.io MIT/Apache 2.0

A library for generating 2d colliders, for bevy apps, from images with transparency

Specifying your dependency

By default, both bevy_rapier2d and avian2d (formerly bevy_xpbd_2d) are enabled. This is to help with the out of box experience, specifically, being able to run both examples and tinker.

But you'll probably only want to just use one of the physics engines supported so when you use it in your own crate fill in in the bevy_collider_gen dependencies with something like this for bevy_rapier2d

[dependencies.bevy_collider_gen]
# replace "*" with the most recent version of bevy_collider_gen
version = "*"

Or this for avian2d

[dependencies.bevy_collider_gen]
# replace "*" with the most recent version of bevy_collider_gen
version = "*"
features = ["avian2d", "parallel"]
default-features = false

Example

example with a car, terrain, and boulders

To see this in action you can run the example, with no arguments it generates a scene with various colliders using PNG's in the assets/sprite directory

bevy_rapier2d

cargo run --example rapier2d_colliders

avian2d

cargo run --example avian2d_colliders -F avian2d

You can also specify a path to an image yourself the example will attempt to generate one or more convex_polyline colliders for the objects it finds

About / why

I was looking for a way to iterate on some 2d scenes with colliders on things with more sophisticated shapes than simple geometry, I figured there should be enough info in an image with transparency to generate colliders, and... there is! So i packaged up my approach here in case anyone else could benefit.

How it works

😄 head on over to the edges crate to learn more https://github.com/shnewto/edges

Caveats

  • as mentioned here and there in these docs, this implementation requires images to have transparency in order to distinguish object from non-object :)
  • i imagine for generating things at a larger scale, i.e. colliders for sets of sprites bigger than pixel counts in the hundreds, this implementation won't be performant to do at runtime. I'll suggest serializing the colliders you like and deserializing in your app instead of doing all the number crunching on load when you need a performance boost

Examples of colliders generated for assets/sprite/car.png

(as in pictures of the sort of thing you can expect, not the runnable bevy app example. That's a couple headings up)

convex polyline (bevy_raiper2d only)

convex polyline collider on an upside down car sprite

polyline

polyline collider on an upside down car sprite

convex hull

convex hull collider on an upside down car sprite

heightfield

The current implementation does best if the image you're generating a heightfield from is either centered in the image or spans the entire width of the image...

heightfield collider on an upside down car sprite

convex decomposition

I didn't add support for convex decomposition directly because when sprites were small, and collisions were forceful, they were sort of unreliable (occasional panics because of bounds indexing in rapier's dependencies 💀). But if you wanted to use convex decomposition colliders you could construct them with the edge coordinates from your image with something like this

let sprite_image = image_assets.get(sprite_handle.unwrap()).unwrap();
let edges = Edges::try_from(sprite_image).unwrap();
let edge_coordinate_groups = edges.multi_translated();
for coords in edge_coordinate_groups {
    let indices: Vec<[u32; 2]> = (0..coords.len()).map(|i| [i as u32, i as u32]).collect();
    let collider = Collider::convex_decomposition(&coords, &indices);
    commands.spawn((
        collider,
        RigidBody::Fixed,
        SpriteBundle {
            texture: sprite_handle.unwrap().clone(),
            ..default()
        },
    ));
}

convex decomposition collider on a car sprite

License

All code in this repository is dual-licensed under either:

At your option.

Dependencies

~30–63MB
~1M SLoC