2024-09-30 16:51:52 +00:00
|
|
|
//! This example demonstrates how to query a [`StandardMaterial`] within a glTF scene.
|
|
|
|
//! It is particularly useful for glTF scenes with a mesh that consists of multiple primitives.
|
|
|
|
|
|
|
|
use std::f32::consts::PI;
|
|
|
|
|
2024-10-27 19:06:19 +00:00
|
|
|
use bevy::{gltf::GltfMaterialName, prelude::*, render::mesh::VertexAttributeValues};
|
2024-09-30 16:51:52 +00:00
|
|
|
|
|
|
|
fn main() {
|
|
|
|
App::new()
|
|
|
|
.add_plugins(DefaultPlugins)
|
|
|
|
.add_systems(Startup, setup)
|
|
|
|
.add_systems(Update, find_top_material_and_mesh)
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn find_top_material_and_mesh(
|
|
|
|
mut materials: ResMut<Assets<StandardMaterial>>,
|
|
|
|
mut meshes: ResMut<Assets<Mesh>>,
|
|
|
|
time: Res<Time>,
|
2024-10-07 23:03:16 +00:00
|
|
|
mat_query: Query<(
|
|
|
|
&MeshMaterial3d<StandardMaterial>,
|
|
|
|
&Mesh3d,
|
|
|
|
&GltfMaterialName,
|
|
|
|
)>,
|
2024-09-30 16:51:52 +00:00
|
|
|
) {
|
|
|
|
for (mat_handle, mesh_handle, name) in mat_query.iter() {
|
|
|
|
// locate a material by material name
|
|
|
|
if name.0 == "Top" {
|
|
|
|
if let Some(material) = materials.get_mut(mat_handle) {
|
|
|
|
if let Color::Hsla(ref mut hsla) = material.base_color {
|
2024-10-16 21:09:32 +00:00
|
|
|
*hsla = hsla.rotate_hue(time.delta_secs() * 100.0);
|
2024-09-30 16:51:52 +00:00
|
|
|
} else {
|
2024-10-27 19:06:19 +00:00
|
|
|
material.base_color = Color::from(Hsla::hsl(0.0, 0.9, 0.7));
|
2024-09-30 16:51:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(mesh) = meshes.get_mut(mesh_handle) {
|
|
|
|
if let Some(VertexAttributeValues::Float32x3(positions)) =
|
|
|
|
mesh.attribute_mut(Mesh::ATTRIBUTE_POSITION)
|
|
|
|
{
|
2024-10-27 19:06:19 +00:00
|
|
|
for position in positions {
|
|
|
|
*position = (
|
|
|
|
position[0],
|
|
|
|
1.5 + 0.5 * ops::sin(time.elapsed_secs() / 2.0),
|
|
|
|
position[2],
|
|
|
|
)
|
|
|
|
.into();
|
|
|
|
}
|
2024-09-30 16:51:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn setup(mut commands: Commands, asset_server: Res<AssetServer>) {
|
|
|
|
commands.spawn((
|
2024-10-05 01:59:52 +00:00
|
|
|
Camera3d::default(),
|
2024-10-27 19:06:19 +00:00
|
|
|
Transform::from_xyz(4.0, 4.0, 12.0).looking_at(Vec3::new(0.0, 0.0, 0.5), Vec3::Y),
|
2024-09-30 16:51:52 +00:00
|
|
|
));
|
|
|
|
|
2024-10-01 03:20:43 +00:00
|
|
|
commands.spawn((
|
2024-10-27 19:06:19 +00:00
|
|
|
Transform::from_rotation(Quat::from_euler(EulerRot::ZYX, 0.0, 1.0, -PI / 4.)),
|
|
|
|
DirectionalLight::default(),
|
Migrate scenes to required components (#15579)
# Objective
A step in the migration to required components: scenes!
## Solution
As per the [selected
proposal](https://hackmd.io/@bevy/required_components/%2FPJtNGVMMQhyM0zIvCJSkbA):
- Deprecate `SceneBundle` and `DynamicSceneBundle`.
- Add `SceneRoot` and `DynamicSceneRoot` components, which wrap a
`Handle<Scene>` and `Handle<DynamicScene>` respectively.
## Migration Guide
Asset handles for scenes and dynamic scenes must now be wrapped in the
`SceneRoot` and `DynamicSceneRoot` components. Raw handles as components
no longer spawn scenes.
Additionally, `SceneBundle` and `DynamicSceneBundle` have been
deprecated. Instead, use the scene components directly.
Previously:
```rust
let model_scene = asset_server.load(GltfAssetLabel::Scene(0).from_asset("model.gltf"));
commands.spawn(SceneBundle {
scene: model_scene,
transform: Transform::from_xyz(-4.0, 0.0, -3.0),
..default()
});
```
Now:
```rust
let model_scene = asset_server.load(GltfAssetLabel::Scene(0).from_asset("model.gltf"));
commands.spawn((
SceneRoot(model_scene),
Transform::from_xyz(-4.0, 0.0, -3.0),
));
```
2024-10-01 22:42:11 +00:00
|
|
|
));
|
2024-10-27 19:06:19 +00:00
|
|
|
|
|
|
|
commands.spawn(SceneRoot(asset_server.load(
|
|
|
|
GltfAssetLabel::Scene(0).from_asset("models/GltfPrimitives/gltf_primitives.glb"),
|
|
|
|
)));
|
2024-09-30 16:51:52 +00:00
|
|
|
}
|