3D开发
概述
学习目标:
- 理解 Bevy 3D 渲染的基本概念
- 掌握如何创建简单的 3D 场景
- 学会使用 PBR 材质和光照系统
- 了解如何加载 GLTF 模型
- 掌握后处理效果的使用
- 理解透明度和多相机渲染
前置知识要求:
- Bevy 快速入门
- ECS 基础
- 资源管理基础
- 2D 基础(可选)
核心概念
什么是 3D 渲染?
3D 渲染是在三维空间中显示模型和场景。Bevy 提供了强大的 3D 渲染系统,支持 PBR 材质、多种光照类型、阴影、后处理等功能。
为什么需要 3D 渲染?
- 游戏开发:大多数 3D 游戏需要 3D 渲染
- 视觉效果:3D 渲染可以提供更真实的视觉效果
- 沉浸感:3D 渲染可以增强游戏的沉浸感
- 灵活性:3D 渲染提供了更多的创作空间
3D 渲染的核心组件
Bevy 3D 渲染系统包含以下核心组件:
- Mesh3d:3D 网格,用于定义 3D 模型的形状
- MeshMaterial3d:3D 材质,用于定义 3D 模型的外观
- Camera3d:3D 相机,用于控制 3D 视图
- Transform:变换,用于控制位置、旋转、缩放
- Light:光照,用于照亮场景
基础用法
创建简单的 3D 场景
创建并显示一个简单的 3D 场景。
源代码文件:bevy/examples/3d/3d_scene.rs
代码示例:
use bevy::prelude::*;
fn main() {
App::new()
.add_plugins(DefaultPlugins)
.add_systems(Startup, setup)
.run();
}
/// 设置简单的 3D 场景
fn setup(
mut commands: Commands,
mut meshes: ResMut<Assets<Mesh>>,
mut materials: ResMut<Assets<StandardMaterial>>,
) {
// 圆形底座
commands.spawn((
Mesh3d(meshes.add(Circle::new(4.0))),
MeshMaterial3d(materials.add(Color::WHITE)),
Transform::from_rotation(Quat::from_rotation_x(-std::f32::consts::FRAC_PI_2)),
));
// 立方体
commands.spawn((
Mesh3d(meshes.add(Cuboid::new(1.0, 1.0, 1.0))),
MeshMaterial3d(materials.add(Color::srgb_u8(124, 144, 255))),
Transform::from_xyz(0.0, 0.5, 0.0),
));
// 光照
commands.spawn((
PointLight {
shadows_enabled: true,
..default()
},
Transform::from_xyz(4.0, 8.0, 4.0),
));
// 相机
commands.spawn((
Camera3d::default(),
Transform::from_xyz(-2.5, 4.5, 9.0).looking_at(Vec3::ZERO, Vec3::Y),
));
}关键要点:
- 使用
Mesh3d创建 3D 网格 - 使用
MeshMaterial3d应用材质 - 使用
Transform控制位置、旋转、缩放 - 使用
PointLight添加点光源 - 使用
Camera3d创建 3D 相机
说明: 这是最简单的 3D 场景示例。它展示了如何创建基本的 3D 对象、添加光照和相机。
PBR 材质
使用 PBR(Physically Based Rendering)材质创建真实的材质效果。
源代码文件:bevy/examples/3d/pbr.rs
代码示例:
use bevy::prelude::*;
fn main() {
App::new()
.add_plugins(DefaultPlugins)
.add_systems(Startup, setup)
.run();
}
fn setup(
mut commands: Commands,
mut meshes: ResMut<Assets<Mesh>>,
mut materials: ResMut<Assets<StandardMaterial>>,
) {
let sphere_mesh = meshes.add(Sphere::new(0.45));
// 创建网格的球体,每个球体有不同的 PBR 参数
for y in -2..=2 {
for x in -5..=5 {
let x01 = (x + 5) as f32 / 10.0;
let y01 = (y + 2) as f32 / 4.0;
// 球体
commands.spawn((
Mesh3d(sphere_mesh.clone()),
MeshMaterial3d(materials.add(StandardMaterial {
base_color: Srgba::hex("#ffd891").unwrap().into(),
// 在网格的球体上变化关键的 PBR 参数以显示效果
metallic: y01,
perceptual_roughness: x01,
..default()
})),
Transform::from_xyz(x as f32, y as f32 + 0.5, 0.0),
));
}
}
// 无光照球体
commands.spawn((
Mesh3d(sphere_mesh),
MeshMaterial3d(materials.add(StandardMaterial {
base_color: Srgba::hex("#ffd891").unwrap().into(),
unlit: true,
..default()
})),
Transform::from_xyz(-5.0, -2.5, 0.0),
));
// 方向光
commands.spawn((
DirectionalLight {
illuminance: 1_500.,
..default()
},
Transform::from_xyz(50.0, 50.0, 50.0).looking_at(Vec3::ZERO, Vec3::Y),
));
}关键要点:
- 使用
StandardMaterial创建 PBR 材质 metallic控制金属度perceptual_roughness控制粗糙度base_color控制基础颜色unlit可以禁用光照
说明: PBR 材质使用物理属性来模拟真实材质的外观。通过调整金属度和粗糙度,可以创建各种材质效果。
光照系统
使用不同类型的光照照亮场景。
源代码文件:bevy/examples/3d/lighting.rs
代码示例:
use bevy::prelude::*;
fn main() {
App::new()
.add_plugins(DefaultPlugins)
.add_systems(Startup, setup)
.add_systems(Update, (update_exposure, toggle_ambient_light, movement, animate_light_direction))
.run();
}
fn setup(
mut commands: Commands,
mut meshes: ResMut<Assets<Mesh>>,
mut materials: ResMut<Assets<StandardMaterial>>,
) {
// 地面平面
commands.spawn((
Mesh3d(meshes.add(Plane3d::default().mesh().size(10.0, 10.0))),
MeshMaterial3d(materials.add(StandardMaterial {
base_color: Color::WHITE,
perceptual_roughness: 1.0,
..default()
})),
));
// 立方体
commands.spawn((
Mesh3d(meshes.add(Cuboid::new(1.0, 1.0, 1.0))),
MeshMaterial3d(materials.add(StandardMaterial {
base_color: Color::srgb_u8(124, 144, 255),
..default()
})),
Transform::from_xyz(0.0, 0.5, 0.0),
));
// 环境光
commands.insert_resource(AmbientLight {
color: Color::WHITE,
brightness: 150.0,
});
// 方向光
commands.spawn((
DirectionalLight {
illuminance: 1_500.,
shadows_enabled: true,
..default()
},
Transform::from_xyz(4.0, 8.0, 4.0).looking_at(Vec3::ZERO, Vec3::Y),
));
// 点光源
commands.spawn((
PointLight {
intensity: 1_500_000.0,
shadows_enabled: true,
..default()
},
Transform::from_xyz(4.0, 8.0, 4.0),
));
// 聚光灯
commands.spawn((
SpotLight {
intensity: 1_500_000.0,
shadows_enabled: true,
..default()
},
Transform::from_xyz(4.0, 8.0, 4.0).looking_at(Vec3::ZERO, Vec3::Y),
));
// 相机
commands.spawn((
Camera3d::default(),
Transform::from_xyz(-2.5, 4.5, 9.0).looking_at(Vec3::ZERO, Vec3::Y),
));
}关键要点:
AmbientLight提供环境光DirectionalLight提供方向光(如太阳光)PointLight提供点光源(如灯泡)SpotLight提供聚光灯- 可以启用阴影来增强真实感
说明: 光照系统是 3D 渲染的重要组成部分。不同类型的光照可以创建不同的视觉效果。Bevy 支持环境光、方向光、点光源和聚光灯。
加载 GLTF 模型
加载和使用 GLTF 模型。
源代码文件:bevy/examples/3d/load_gltf.rs
代码示例:
use bevy::prelude::*;
fn main() {
App::new()
.insert_resource(DirectionalLightShadowMap { size: 4096 })
.add_plugins(DefaultPlugins)
.add_systems(Startup, setup)
.add_systems(Update, animate_light_direction)
.run();
}
fn setup(mut commands: Commands, asset_server: Res<AssetServer>) {
// 相机
commands.spawn((
Camera3d::default(),
Transform::from_xyz(0.7, 0.7, 1.0).looking_at(Vec3::new(0.0, 0.3, 0.0), Vec3::Y),
EnvironmentMapLight {
diffuse_map: asset_server.load("environment_maps/pisa_diffuse_rgb9e5_zstd.ktx2"),
specular_map: asset_server.load("environment_maps/pisa_specular_rgb9e5_zstd.ktx2"),
intensity: 250.0,
..default()
},
));
// 方向光
commands.spawn((
DirectionalLight {
shadows_enabled: true,
..default()
},
CascadeShadowConfigBuilder {
num_cascades: 1,
maximum_distance: 1.6,
..default()
}
.build(),
));
// 加载 GLTF 场景
commands.spawn(SceneRoot(asset_server.load(
GltfAssetLabel::Scene(0).from_asset("models/FlightHelmet/FlightHelmet.gltf"),
)));
}关键要点:
- 使用
SceneRoot加载 GLTF 场景 - 使用
GltfAssetLabel::Scene(0)指定场景索引 - GLTF 文件可以包含多个场景、网格、材质等
- 可以使用环境贴图增强渲染效果
说明: GLTF 是一种标准的 3D 模型格式。Bevy 支持加载 GLTF 文件,包括场景、网格、材质、动画等。
进阶用法
后处理效果
使用后处理效果增强渲染效果。
源代码文件:bevy/examples/3d/post_processing.rs
代码示例:
use bevy::prelude::*;
fn main() {
App::new()
.add_plugins(DefaultPlugins)
.add_systems(Startup, setup)
.add_systems(Update, handle_keyboard_input)
.run();
}
fn setup(mut commands: Commands, asset_server: Res<AssetServer>) {
// 相机,包含色差效果
commands.spawn((
Camera3d::default(),
Hdr,
Transform::from_xyz(0.7, 0.7, 1.0).looking_at(Vec3::new(0.0, 0.3, 0.0), Vec3::Y),
DistanceFog {
color: Color::srgb_u8(43, 44, 47),
falloff: FogFalloff::Linear {
start: 1.0,
end: 8.0,
},
..default()
},
EnvironmentMapLight {
diffuse_map: asset_server.load("environment_maps/pisa_diffuse_rgb9e5_zstd.ktx2"),
specular_map: asset_server.load("environment_maps/pisa_specular_rgb9e5_zstd.ktx2"),
intensity: 2000.0,
..default()
},
// 包含色差效果组件
ChromaticAberration::default(),
));
}关键要点:
- 使用
ChromaticAberration添加色差效果 - 使用
Hdr启用 HDR 渲染 - 使用
DistanceFog添加距离雾效果 - 后处理效果可以组合使用
说明: 后处理效果可以在渲染完成后对图像进行处理。Bevy 支持多种后处理效果,如色差、色调映射、泛光等。
透明度
处理透明对象。
源代码文件:bevy/examples/3d/transparency_3d.rs
关键信息:
- 使用
AlphaMode控制透明度模式 AlphaMode::Blend用于半透明对象AlphaMode::Mask用于带透明度的纹理- 透明对象需要正确的渲染顺序
说明: 透明度是 3D 渲染中的重要特性。Bevy 支持多种透明度模式,可以创建玻璃、水、烟雾等效果。
多相机渲染
使用多个相机渲染不同的视图。
源代码文件:bevy/examples/3d/multiple_windows.rs
关键信息:
- 可以创建多个相机
- 每个相机可以渲染到不同的窗口
- 可以使用
RenderTarget指定渲染目标 - 适合创建分屏或多显示器应用
说明: 多相机渲染允许同时显示多个视图。这对于创建分屏游戏、多显示器应用或调试视图非常有用。
实际应用
在游戏开发中的应用场景
3D 渲染在游戏开发中有广泛的应用:
- 3D 游戏:创建 3D 游戏场景和角色
- 可视化:创建数据可视化应用
- 模拟:创建物理模拟和训练应用
- 艺术创作:创建 3D 艺术作品和动画
常见问题
问题 1:如何优化 3D 渲染性能?
解决方案:可以使用 LOD(细节层次)、遮挡剔除、视锥剔除等技术来优化 3D 渲染性能。
问题 2:如何处理透明对象?
解决方案:透明对象需要正确的渲染顺序。Bevy 会自动处理透明对象的排序。
问题 3:如何创建自定义材质?
解决方案:可以实现 Material trait 来创建自定义材质。可以参考 StandardMaterial 的实现。
性能考虑
- 网格优化:使用优化的网格减少顶点数量
- 材质优化:使用简单的材质减少渲染开销
- 光照优化:合理使用光照类型和阴影
- 剔除优化:使用视锥剔除和遮挡剔除减少渲染对象
相关资源
相关源代码文件:
bevy/examples/3d/3d_scene.rs- 简单 3D 场景示例bevy/examples/3d/pbr.rs- PBR 材质示例bevy/examples/3d/lighting.rs- 光照系统示例bevy/examples/3d/load_gltf.rs- GLTF 加载示例bevy/examples/3d/post_processing.rs- 后处理效果示例bevy/examples/3d/transparency_3d.rs- 透明度示例bevy/examples/3d/multiple_windows.rs- 多相机渲染示例
官方文档链接:
进一步学习建议:
- 学习相机系统,了解如何控制 3D 视图
- 学习动画系统,了解如何为 3D 模型添加动画
- 学习自定义渲染,了解如何创建自定义渲染效果
索引:返回上级目录