render to second window in multiple_windows example
This commit is contained in:
parent
8a8d01aa88
commit
ca4726ea7d
@ -26,3 +26,22 @@ impl From<&Texture> for TextureDescriptor {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
impl Default for TextureDescriptor {
|
||||||
|
fn default() -> Self {
|
||||||
|
TextureDescriptor {
|
||||||
|
size: Extent3d {
|
||||||
|
width: 1,
|
||||||
|
height: 1,
|
||||||
|
depth: 1,
|
||||||
|
},
|
||||||
|
mip_level_count: 1,
|
||||||
|
sample_count: 1,
|
||||||
|
dimension: TextureDimension::D2,
|
||||||
|
format: TextureFormat::Rgba8UnormSrgb,
|
||||||
|
usage: TextureUsage::SAMPLED | TextureUsage::COPY_DST,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@ -13,6 +13,7 @@ pub struct WindowResized {
|
|||||||
/// An event that indicates that a new window should be created.
|
/// An event that indicates that a new window should be created.
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct CreateWindow {
|
pub struct CreateWindow {
|
||||||
|
pub id: WindowId,
|
||||||
pub descriptor: WindowDescriptor,
|
pub descriptor: WindowDescriptor,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -38,6 +38,7 @@ impl AppPlugin for WindowPlugin {
|
|||||||
let mut create_window_event =
|
let mut create_window_event =
|
||||||
app.resources().get_mut::<Events<CreateWindow>>().unwrap();
|
app.resources().get_mut::<Events<CreateWindow>>().unwrap();
|
||||||
create_window_event.send(CreateWindow {
|
create_window_event.send(CreateWindow {
|
||||||
|
id: WindowId::new(),
|
||||||
descriptor: primary_window_descriptor.clone(),
|
descriptor: primary_window_descriptor.clone(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@ -9,6 +9,10 @@ pub enum WindowReference {
|
|||||||
pub struct WindowId(Uuid);
|
pub struct WindowId(Uuid);
|
||||||
|
|
||||||
impl WindowId {
|
impl WindowId {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
WindowId(Uuid::new_v4())
|
||||||
|
}
|
||||||
|
|
||||||
pub fn to_string(&self) -> String {
|
pub fn to_string(&self) -> String {
|
||||||
self.0.to_simple().to_string()
|
self.0.to_simple().to_string()
|
||||||
}
|
}
|
||||||
@ -24,9 +28,9 @@ pub struct Window {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Window {
|
impl Window {
|
||||||
pub fn new(window_descriptor: &WindowDescriptor) -> Self {
|
pub fn new(id: WindowId, window_descriptor: &WindowDescriptor) -> Self {
|
||||||
Window {
|
Window {
|
||||||
id: WindowId(Uuid::new_v4()),
|
id,
|
||||||
height: window_descriptor.height,
|
height: window_descriptor.height,
|
||||||
width: window_descriptor.width,
|
width: window_descriptor.width,
|
||||||
title: window_descriptor.title.clone(),
|
title: window_descriptor.title.clone(),
|
||||||
|
|||||||
@ -155,7 +155,7 @@ fn handle_create_window_events(
|
|||||||
let create_window_events = resources.get::<Events<CreateWindow>>().unwrap();
|
let create_window_events = resources.get::<Events<CreateWindow>>().unwrap();
|
||||||
let mut window_created_events = resources.get_mut::<Events<WindowCreated>>().unwrap();
|
let mut window_created_events = resources.get_mut::<Events<WindowCreated>>().unwrap();
|
||||||
for create_window_event in create_window_event_reader.iter(&create_window_events) {
|
for create_window_event in create_window_event_reader.iter(&create_window_events) {
|
||||||
let window = Window::new(&create_window_event.descriptor);
|
let window = Window::new(create_window_event.id, &create_window_event.descriptor);
|
||||||
winit_windows.create_window(event_loop, &window);
|
winit_windows.create_window(event_loop, &window);
|
||||||
let window_id = window.id;
|
let window_id = window.id;
|
||||||
windows.add(window);
|
windows.add(window);
|
||||||
|
|||||||
@ -1,20 +1,152 @@
|
|||||||
use bevy::{prelude::*, window::CreateWindow};
|
use bevy::{prelude::*, window::CreateWindow};
|
||||||
|
use bevy_render::{pass::{StoreOp, LoadOp, TextureAttachment, RenderPassColorAttachmentDescriptor, PassDescriptor, RenderPassDepthStencilAttachmentDescriptor}, texture::{TextureDescriptor, TextureFormat, TextureUsage}};
|
||||||
|
use bevy_window::{WindowId, WindowReference};
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
App::build()
|
App::build()
|
||||||
.add_default_plugins()
|
.add_default_plugins()
|
||||||
.add_startup_system(setup.system())
|
.add_startup_system(create_second_window_system.system())
|
||||||
|
.add_startup_system(setup_scene.system())
|
||||||
.run();
|
.run();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn setup(mut create_window_events: ResMut<Events<CreateWindow>>) {
|
fn create_second_window_system(
|
||||||
|
mut create_window_events: ResMut<Events<CreateWindow>>,
|
||||||
|
mut render_graph: ResMut<RenderGraph>,
|
||||||
|
) {
|
||||||
|
let window_id = WindowId::new();
|
||||||
|
|
||||||
// sends out a "CreateWindow" event, which will be received by the windowing backend
|
// sends out a "CreateWindow" event, which will be received by the windowing backend
|
||||||
create_window_events.send(CreateWindow {
|
create_window_events.send(CreateWindow {
|
||||||
|
id: window_id,
|
||||||
descriptor: WindowDescriptor {
|
descriptor: WindowDescriptor {
|
||||||
width: 800,
|
width: 800,
|
||||||
height: 600,
|
height: 600,
|
||||||
vsync: false,
|
vsync: false,
|
||||||
title: "another window".to_string(),
|
title: "second window".to_string(),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// here we setup our render graph to draw our second camera to the new window's swap chain
|
||||||
|
|
||||||
|
// add a swapchain node for our new window
|
||||||
|
render_graph.add_node(
|
||||||
|
"second_window_swap_chain",
|
||||||
|
WindowSwapChainNode::new(WindowReference::Id(window_id)),
|
||||||
|
);
|
||||||
|
|
||||||
|
// add a new depth texture node for our new window
|
||||||
|
render_graph.add_node(
|
||||||
|
"second_window_depth_texture",
|
||||||
|
WindowTextureNode::new(
|
||||||
|
WindowReference::Id(window_id),
|
||||||
|
TextureDescriptor {
|
||||||
|
format: TextureFormat::Depth32Float,
|
||||||
|
usage: TextureUsage::OUTPUT_ATTACHMENT,
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut second_window_pass = PassNode::new(PassDescriptor {
|
||||||
|
color_attachments: vec![RenderPassColorAttachmentDescriptor {
|
||||||
|
attachment: TextureAttachment::Input("color".to_string()),
|
||||||
|
resolve_target: None,
|
||||||
|
load_op: LoadOp::Clear,
|
||||||
|
store_op: StoreOp::Store,
|
||||||
|
clear_color: Color::rgb(0.1, 0.1, 0.1),
|
||||||
|
}],
|
||||||
|
depth_stencil_attachment: Some(RenderPassDepthStencilAttachmentDescriptor {
|
||||||
|
attachment: TextureAttachment::Input("depth".to_string()),
|
||||||
|
depth_load_op: LoadOp::Clear,
|
||||||
|
depth_store_op: StoreOp::Store,
|
||||||
|
stencil_load_op: LoadOp::Clear,
|
||||||
|
stencil_store_op: StoreOp::Store,
|
||||||
|
stencil_read_only: false,
|
||||||
|
depth_read_only: false,
|
||||||
|
clear_depth: 1.0,
|
||||||
|
clear_stencil: 0,
|
||||||
|
}),
|
||||||
|
sample_count: 1,
|
||||||
|
});
|
||||||
|
|
||||||
|
// TODO: use different camera here
|
||||||
|
second_window_pass.add_camera(bevy::render::base_render_graph::camera::CAMERA);
|
||||||
|
|
||||||
|
render_graph.add_node(
|
||||||
|
"second_window_pass",
|
||||||
|
second_window_pass,
|
||||||
|
);
|
||||||
|
|
||||||
|
render_graph
|
||||||
|
.add_slot_edge(
|
||||||
|
"second_window_swap_chain",
|
||||||
|
WindowSwapChainNode::OUT_TEXTURE,
|
||||||
|
"second_window_pass",
|
||||||
|
"color",
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
render_graph
|
||||||
|
.add_slot_edge(
|
||||||
|
"second_window_depth_texture",
|
||||||
|
WindowTextureNode::OUT_TEXTURE,
|
||||||
|
"second_window_pass",
|
||||||
|
"depth",
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn setup_scene(
|
||||||
|
command_buffer: &mut CommandBuffer,
|
||||||
|
asset_server: Res<AssetServer>,
|
||||||
|
mut materials: ResMut<Assets<StandardMaterial>>,
|
||||||
|
) {
|
||||||
|
// load the mesh
|
||||||
|
let mesh_handle = asset_server
|
||||||
|
.load("assets/models/monkey/Monkey.gltf")
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
// create a material for the mesh
|
||||||
|
let material_handle = materials.add(StandardMaterial {
|
||||||
|
albedo: Color::rgb(0.5, 0.4, 0.3),
|
||||||
|
..Default::default()
|
||||||
|
});
|
||||||
|
|
||||||
|
// add entities to the world
|
||||||
|
command_buffer
|
||||||
|
.build()
|
||||||
|
// mesh
|
||||||
|
.entity_with(MeshComponents {
|
||||||
|
mesh: mesh_handle,
|
||||||
|
material: material_handle,
|
||||||
|
..Default::default()
|
||||||
|
})
|
||||||
|
// light
|
||||||
|
.entity_with(LightComponents {
|
||||||
|
translation: Translation::new(4.0, 5.0, 4.0),
|
||||||
|
..Default::default()
|
||||||
|
})
|
||||||
|
// main camera
|
||||||
|
.entity_with(PerspectiveCameraComponents {
|
||||||
|
transform: Transform::new_sync_disabled(Mat4::face_toward(
|
||||||
|
Vec3::new(0.0, 0.0, 6.0),
|
||||||
|
Vec3::new(0.0, 0.0, 0.0),
|
||||||
|
Vec3::new(0.0, 1.0, 0.0),
|
||||||
|
)),
|
||||||
|
..Default::default()
|
||||||
|
});
|
||||||
|
// // second window camera
|
||||||
|
// .entity_with(PerspectiveCameraComponents {
|
||||||
|
// camera: Camera {
|
||||||
|
// name: Some("Secondary".to_string()),
|
||||||
|
// ..Default::default()
|
||||||
|
// },
|
||||||
|
// transform: Transform::new_sync_disabled(Mat4::face_toward(
|
||||||
|
// Vec3::new(0.0, 0.0, 6.0),
|
||||||
|
// Vec3::new(0.0, 0.0, 0.0),
|
||||||
|
// Vec3::new(0.0, 1.0, 0.0),
|
||||||
|
// )),
|
||||||
|
// ..Default::default()
|
||||||
|
// });
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user