forked from bevyengine/bevy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patharray_texture.rs
152 lines (131 loc) · 4.68 KB
/
array_texture.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
use bevy::{
prelude::*,
reflect::TypeUuid,
render::{
mesh::shape,
pipeline::{PipelineDescriptor, RenderPipeline},
render_graph::{base, AssetRenderResourcesNode, RenderGraph},
renderer::RenderResources,
shader::{ShaderStage, ShaderStages},
},
};
/// This example illustrates how to create a texture for use with a texture2DArray shader uniform variable.
fn main() {
App::build()
.add_plugins(DefaultPlugins)
.add_asset::<MyArrayTexture>()
.add_startup_system(setup.system())
.add_system(create_array_texture.system())
.run();
}
#[derive(RenderResources, Default, TypeUuid)]
#[uuid = "93fb26fc-6c05-489b-9029-601edf703b6b"]
struct MyArrayTexture {
pub texture: Handle<Texture>,
}
const VERTEX_SHADER: &str = r#"
#version 450
layout(location = 0) in vec3 Vertex_Position;
layout(location = 0) out vec4 v_Position;
layout(set = 0, binding = 0) uniform Camera {
mat4 ViewProj;
};
layout(set = 1, binding = 0) uniform Transform {
mat4 Model;
};
void main() {
v_Position = ViewProj * Model * vec4(Vertex_Position, 1.0);
gl_Position = v_Position;
}
"#;
const FRAGMENT_SHADER: &str = r#"
#version 450
layout(location = 0) in vec4 v_Position;
layout(location = 0) out vec4 o_Target;
layout(set = 2, binding = 0) uniform texture2DArray MyArrayTexture_texture;
layout(set = 2, binding = 1) uniform sampler MyArrayTexture_texture_sampler;
void main() {
// Screen-space coordinates determine which layer of the array texture we sample.
vec2 ss = v_Position.xy / v_Position.w;
float layer = 0.0;
if (ss.x > 0.0 && ss.y > 0.0) {
layer = 0.0;
} else if (ss.x < 0.0 && ss.y > 0.0) {
layer = 1.0;
} else if (ss.x > 0.0 && ss.y < 0.0) {
layer = 2.0;
} else {
layer = 3.0;
}
// Convert to texture coordinates.
vec2 uv = (ss + vec2(1.0)) / 2.0;
o_Target = texture(sampler2DArray(MyArrayTexture_texture, MyArrayTexture_texture_sampler), vec3(uv, layer));
}
"#;
struct LoadingTexture(Option<Handle<Texture>>);
struct MyPipeline(Handle<PipelineDescriptor>);
fn setup(
commands: &mut Commands,
asset_server: Res<AssetServer>,
mut pipelines: ResMut<Assets<PipelineDescriptor>>,
mut shaders: ResMut<Assets<Shader>>,
mut render_graph: ResMut<RenderGraph>,
) {
// Start loading the texture.
commands.insert_resource(LoadingTexture(Some(
asset_server.load("textures/array_texture.png"),
)));
// Create a new shader pipeline.
let pipeline_handle = pipelines.add(PipelineDescriptor::default_config(ShaderStages {
vertex: shaders.add(Shader::from_glsl(ShaderStage::Vertex, VERTEX_SHADER)),
fragment: Some(shaders.add(Shader::from_glsl(ShaderStage::Fragment, FRAGMENT_SHADER))),
}));
commands.insert_resource(MyPipeline(pipeline_handle));
// Add an AssetRenderResourcesNode to our Render Graph. This will bind MyArrayTexture resources to our shader.
render_graph.add_system_node(
"my_array_texture",
AssetRenderResourcesNode::<MyArrayTexture>::new(true),
);
// Add a Render Graph edge connecting our new "my_array_texture" node to the main pass node. This ensures "my_array_texture"
// runs before the main pass.
render_graph
.add_node_edge("my_array_texture", base::node::MAIN_PASS)
.unwrap();
commands.spawn(PerspectiveCameraBundle {
transform: Transform::from_xyz(2.0, 2.0, 2.0).looking_at(Vec3::default(), Vec3::unit_y()),
..Default::default()
});
}
fn create_array_texture(
commands: &mut Commands,
my_pipeline: Res<MyPipeline>,
mut loading_texture: ResMut<LoadingTexture>,
mut textures: ResMut<Assets<Texture>>,
mut meshes: ResMut<Assets<Mesh>>,
mut array_textures: ResMut<Assets<MyArrayTexture>>,
) {
let (handle, texture) = match loading_texture.0.as_ref() {
Some(handle) => {
if let Some(texture) = textures.get_mut(handle) {
(loading_texture.0.take().unwrap(), texture)
} else {
return;
}
}
None => return,
};
// Create a new array texture asset from the loaded texture.
let array_layers = 4;
texture.reinterpret_stacked_2d_as_array(array_layers);
let array_texture = array_textures.add(MyArrayTexture { texture: handle });
// Spawn a cube that's shaded using the array texture.
commands
.spawn(MeshBundle {
mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })),
render_pipelines: RenderPipelines::from_pipelines(vec![RenderPipeline::new(
my_pipeline.0.clone(),
)]),
..Default::default()
})
.with(array_texture);
}