Use Cases (Advanced)

Using Sub-Assets

There are many elements of a glTF file including Scenes, Nodes, Meshes, Materials, Skins, Nodes, AnimationClips, etc. With appropriate care, these elements can be used to build up specific hierarchies for niche use cases.

Bevy's Gltf struct

Bevy's Gltf struct contains the processed glTF data, with and without names.

We'll be using named access because its clearer for written instructional content, which means we can use these fields on Gltf:

named_scenes: HashMap<Box<str>, Handle<Scene>>
named_meshes: HashMap<Box<str>, Handle<GltfMesh>>
named_materials: HashMap<Box<str>, Handle<StandardMaterial>>
named_nodes: HashMap<Box<str>, Handle<GltfNode>>
named_skins: HashMap<Box<str>, Handle<GltfSkin>>
named_animations: HashMap<Box<str>, Handle<AnimationClip>>

Spawning a Mesh (Custom Material)

Spawning a mesh from a glTF by itself seems straightforward at first, but Bevy's Mesh and GltfMesh are not the same concept. A Bevy Mesh is closer to a GltfPrimitive than a GltfMesh.

let handle = gltf.named_meshes.get("Sphere").unwrap();
let gltf_mesh = gltf_meshes.get(handle).unwrap();

commands.spawn((
    Mesh3d(gltf_mesh.primitives[0].mesh.clone()),
    MeshMaterial3d(materials.force_field.clone()),
    gltf_mesh.primitives[0]
        .extras
        .clone()
        .unwrap_or(GltfExtras::default()),
));

Using a Material

You can use materials in two ways: with or without the Bevy Components you applied in Blender.

Without Components

To apply a material from a glTF file you can access the handle via named_meshes by material name and use the handle directly. This approach is simpler but will not apply the Components you inserted in Blender.

let gltf = gltf.get(&gltf_handle).ok_or("couldn't get gltf")?;
let material_handle = gltf
    .named_materials
    .get("Wooden Panel")
    .ok_or("Couldn't get material handle")?;

commands.spawn((
    Mesh3d(meshes.add(Plane3d::new(
        Vec3::Y,
        Vec2::new(1.5, 1.5),
    ))),
    MeshMaterial3d(material_handle.clone()),
    Transform::from_xyz(0., 0., 0.)
));

With Components

If there are Components you applied in Blender that you want to also apply, you must gain access to the GltfExtras component and insert it alongside the material handle. If you followed the instructions in Exporting Materials to Files then you will have a simple mesh like a Sphere or Cube, which will have a single primitive associated with the material.

Everything here is set up to exist, but we'll use ok_or()? anyway because Bevy supports returning Result from systems.

let gltf = gltf.get(&gltf_handle).ok_or("couldn't get gltf")?;
let primitive = &gltf_meshes
    .get(
        gltf.named_meshes
            .get("WallBricksMesh")
            .ok_or("couldn't get gltf_mesh")?,
    )
    .ok_or("couldn't get gltf_primitive")?
    .primitives[0];

commands.spawn((
    Mesh3d(meshes.add(Plane3d::new(
        Vec3::Y,
        Vec2::new(1.5, 1.5),
    ))),
    MeshMaterial3d(
        primitive
            .material
            .clone()
            .ok_or("Option was None")?,
    ),
    Transform::from_xyz(0., 0., 0.),
    primitive
        .material_extras
        .clone()
        .ok_or("Option was None")?,
));
Previous
Replace a Blender Material