//! Controls morph targets in a loaded scene. //! //! Illustrates: //! //! - How to access and modify individual morph target weights. //! See the `update_weights` system for details. //! - How to read morph target names in `name_morphs`. //! - How to play morph target animations in `setup_animations`. use bevy::prelude::*; use std::f32::consts::PI; fn main() { App::new() .add_plugins(DefaultPlugins.set(WindowPlugin { primary_window: Some(Window { title: "morph targets".to_string(), ..default() }), ..default() })) .insert_resource(AmbientLight { brightness: 150.0, ..default() }) .add_systems(Startup, setup) .add_systems(Update, (name_morphs, setup_animations)) .run(); } #[derive(Resource)] struct MorphData { the_wave: Handle, mesh: Handle, } fn setup(asset_server: Res, mut commands: Commands) { commands.insert_resource(MorphData { the_wave: asset_server .load(GltfAssetLabel::Animation(2).from_asset("models/animated/MorphStressTest.gltf")), mesh: asset_server.load( GltfAssetLabel::Primitive { mesh: 0, primitive: 0, } .from_asset("models/animated/MorphStressTest.gltf"), ), }); commands.spawn(SceneRoot(asset_server.load( GltfAssetLabel::Scene(0).from_asset("models/animated/MorphStressTest.gltf"), ))); commands.spawn(( DirectionalLight::default(), Transform::from_rotation(Quat::from_rotation_z(PI / 2.0)), )); commands.spawn(( Camera3d::default(), Transform::from_xyz(3.0, 2.1, 10.2).looking_at(Vec3::ZERO, Vec3::Y), )); } /// Plays an [`AnimationClip`] from the loaded [`Gltf`] on the [`AnimationPlayer`] created by the spawned scene. fn setup_animations( mut has_setup: Local, mut commands: Commands, mut players: Query<(Entity, &Name, &mut AnimationPlayer)>, morph_data: Res, mut graphs: ResMut>, ) { if *has_setup { return; } for (entity, name, mut player) in &mut players { // The name of the entity in the GLTF scene containing the AnimationPlayer for our morph targets is "Main" if name.as_str() != "Main" { continue; } let (graph, animation) = AnimationGraph::from_clip(morph_data.the_wave.clone()); commands .entity(entity) .insert(AnimationGraphHandle(graphs.add(graph))); player.play(animation).repeat(); *has_setup = true; } } /// You can get the target names in their corresponding [`Mesh`]. /// They are in the order of the weights. fn name_morphs( mut has_printed: Local, morph_data: Res, meshes: Res>, ) { if *has_printed { return; } let Some(mesh) = meshes.get(&morph_data.mesh) else { return; }; let Some(names) = mesh.morph_target_names() else { return; }; info!("Target names:"); for name in names { info!(" {name}"); } *has_printed = true; }