Use 2d sprites in a 3d scene. This was my go-to workflow back when I was using Unity. This crate replicates it in bevy.
Useful for:
- 2d games using bevy's lighting (orthographic camera, 3d sprites)
- 2d games with easier parallax and scale (perspective camera, 3d sprites)
- 2d games in a 3d world (perspective camera, both 3d sprites and meshes)
- 3d games with billboard sprites (a la
Delver)
Both meshes and materials are internally cached, so you can use this for things like tilemaps without issue.
Example using bevy_sprite3d
:
Some more examples. These don't use bevy, but demonstrate the effect style:
One small complication to bevy_sprite3d
is that your image assets need to be
loaded prior to spawning, as the crate uses some properties of the image
(such as size and aspect ratio) in constructing the 3d mesh.
The following examples will use
bevy_asset_loader
for
simplicity. Even still, there's a fair amount of boilerplate due to this
loading-before-spawning requirement. If anyone knows a simpler way to write the
examples, please update it!
```rust use bevy::prelude::; use bevy_sprite3d::; use bevyassetloader::{AssetLoader, AssetCollection};
enum GameState { Loading, Loaded, }
fn main() { let mut app = App::new();
AssetLoader::new(GameState::Loading)
.continue_to_state(GameState::Loaded)
.with_collection::<ImageAssets>()
.build(&mut app);
app.add_state(GameState::Loading)
.add_plugins(DefaultPlugins)
.add_plugin(Sprite3dPlugin)
.add_system_set( SystemSet::on_enter(GameState::Loaded).with_system(setup) );
app.run();
}
struct ImageAssets {
#[asset(path = "branding/icon.png")]
icon: Handle
fn setup(
mut commands: Commands,
images: Res
commands.spawn_bundle(OrthographicCameraBundle::new_3d())
.insert(Transform::from_xyz(0., 0., 18.5));
commands.spawn_bundle(Sprite3d {
image: images.icon.clone(),
pixels_per_metre: 400.,
partial_alpha: true,
unlit: true,
// transform: Transform::from_xyz(0., 0., 0.),
// pivot: Some(Vec2::new(0.5, 0.5)),
..default()
}.bundle(&mut sprite_params));
} ```
```rust use bevy::prelude::; use bevy_sprite3d::; use bevyassetloader::{AssetLoader, AssetCollection};
enum GameState { Loading, Loaded, }
fn main() { let mut app = App::new();
AssetLoader::new(GameState::Loading)
.continue_to_state(GameState::Loaded)
.with_collection::<ImageAssets>()
.build(&mut app);
app.add_state(GameState::Loading)
.add_plugins(DefaultPlugins)
.add_plugin(Sprite3dPlugin)
.add_system_set( SystemSet::on_enter(GameState::Loaded).with_system(setup) );
app.run();
}
struct ImageAssets {
#[asset(textureatlas(tilesizex = 24., tilesizey = 24.))]
#[asset(textureatlas(columns = 7, rows = 1))]
#[asset(path = "textures/rpg/chars/gabe/gabe-idle-run.png")]
run: Handle
fn setup(
mut commands: Commands,
images: Res
commands.spawn_bundle(OrthographicCameraBundle::new_3d())
.insert(Transform::from_xyz(0., 0., 18.5));
commands.spawn_bundle(AtlasSprite3d {
atlas: images.run.clone(),
pixels_per_metre: 32.,
partial_alpha: true,
unlit: true,
index: 3,
// transform: Transform::from_xyz(0., 0., 0.),
// pivot: Some(Vec2::new(0.5, 0.5)),
..default()
}.bundle(&mut sprite_params));
} ```