bevy_sprite_instancing
A plugin for Bevy to render lots of instanced sprites in a single draw call.
Instead of going through all similar entity sprites one-by-one,
the plugin provides SpriteInstancingGroup
s which can contain
Entity ID's of these sprites. The plugin then collect all these sprites'
transforms and submits them to the GPU as one large buffer, which is then
used to draw all of them at once.
Here's a simple example of how to draw lots of sprites:
```rust pub const ENTITY_COUNT: usize = 100000;
fn setup(mut commands: Commands, asset_server: Res
// Load a spritesheet
let spritesheet = InstancedSpritesheet {
image: asset_server.load("textures/my_spritesheet.png"),
// Size of this spritesheet in tiles
width_tiles: 32,
height_tiles: 32,
};
// Create an instancing group for the sprites
let mut instancing_group = SpriteInstancingGroup {
entities: HashSet::new(),
};
let group_id = commands.spawn_empty().id();
// Spawn the sprites
for _ in 0..ENTITY_COUNT {
let position = Vec3::new(..., 0.0); // some random position
let scale = Vec3::new(..., 1.0); // some random size
let transform = ...; // create a transform from these
let sprite = InstancedSprite {
group_id,
texture_index: 0
};
let entity = commands.spawn((transform, sprite)).id();
instancing_group.entities.insert(entity);
}
// Attach the instancing group and its spritesheet
commands.entity(group_id).insert((instancing_group, spritesheet));
}
fn main() { App::new() .addplugins(DefaultPlugins) .addplugin(InstancedSpriteRenderPlugin) .addstartupsystem(setup) .run() } ```
There're some high-level features I haven't yet implemented:
texture_index
for each sprite,
maybe it would be a nice idea to submit anim_start_index
and anim_len
through a separate instancing buffer... and some low-level ones:
SpriteInstancingGroup
as static (i.e. entities do not
move) to avoid re-submitting its instancing data each frameInstancedSpritesheet
only needs to be extracted once when
starting up, but when I tried this, I got QueryDoesNotMatch
, lol. ExtractedSpriteInstancingBuffer
each frame. Tried to
implement this, but it seems like wgpu
doesn't allow vertex buffers to be
mapped for writing, which's a bit weird (Vulkan allows this AFAIK).