 01649f13e2
			
		
	
	
		01649f13e2
		
			
		
	
	
	
	
		
			
			# Objective This is a necessary precursor to #9122 (this was split from that PR to reduce the amount of code to review all at once). Moving `!Send` resource ownership to `App` will make it unambiguously `!Send`. `SubApp` must be `Send`, so it can't wrap `App`. ## Solution Refactor `App` and `SubApp` to not have a recursive relationship. Since `SubApp` no longer wraps `App`, once `!Send` resources are moved out of `World` and into `App`, `SubApp` will become unambiguously `Send`. There could be less code duplication between `App` and `SubApp`, but that would break `App` method chaining. ## Changelog - `SubApp` no longer wraps `App`. - `App` fields are no longer publicly accessible. - `App` can no longer be converted into a `SubApp`. - Various methods now return references to a `SubApp` instead of an `App`. ## Migration Guide - To construct a sub-app, use `SubApp::new()`. `App` can no longer convert into `SubApp`. - If you implemented a trait for `App`, you may want to implement it for `SubApp` as well. - If you're accessing `app.world` directly, you now have to use `app.world()` and `app.world_mut()`. - `App::sub_app` now returns `&SubApp`. - `App::sub_app_mut` now returns `&mut SubApp`. - `App::get_sub_app` now returns `Option<&SubApp>.` - `App::get_sub_app_mut` now returns `Option<&mut SubApp>.`
		
			
				
	
	
		
			85 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			85 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| use crate::{
 | |
|     extract_resource::ExtractResource,
 | |
|     prelude::Shader,
 | |
|     render_resource::{ShaderType, UniformBuffer},
 | |
|     renderer::{RenderDevice, RenderQueue},
 | |
|     Extract, ExtractSchedule, Render, RenderApp, RenderSet,
 | |
| };
 | |
| use bevy_app::{App, Plugin};
 | |
| use bevy_asset::{load_internal_asset, Handle};
 | |
| use bevy_core::FrameCount;
 | |
| use bevy_ecs::prelude::*;
 | |
| use bevy_reflect::prelude::*;
 | |
| use bevy_time::Time;
 | |
| 
 | |
| pub const GLOBALS_TYPE_HANDLE: Handle<Shader> = Handle::weak_from_u128(17924628719070609599);
 | |
| 
 | |
| pub struct GlobalsPlugin;
 | |
| 
 | |
| impl Plugin for GlobalsPlugin {
 | |
|     fn build(&self, app: &mut App) {
 | |
|         load_internal_asset!(app, GLOBALS_TYPE_HANDLE, "globals.wgsl", Shader::from_wgsl);
 | |
|         app.register_type::<GlobalsUniform>();
 | |
| 
 | |
|         if let Some(render_app) = app.get_sub_app_mut(RenderApp) {
 | |
|             render_app
 | |
|                 .init_resource::<GlobalsBuffer>()
 | |
|                 .init_resource::<Time>()
 | |
|                 .add_systems(ExtractSchedule, (extract_frame_count, extract_time))
 | |
|                 .add_systems(
 | |
|                     Render,
 | |
|                     prepare_globals_buffer.in_set(RenderSet::PrepareResources),
 | |
|                 );
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| fn extract_frame_count(mut commands: Commands, frame_count: Extract<Res<FrameCount>>) {
 | |
|     commands.insert_resource(**frame_count);
 | |
| }
 | |
| 
 | |
| fn extract_time(mut commands: Commands, time: Extract<Res<Time>>) {
 | |
|     commands.insert_resource(**time);
 | |
| }
 | |
| 
 | |
| /// Contains global values useful when writing shaders.
 | |
| /// Currently only contains values related to time.
 | |
| #[derive(Default, Clone, Resource, ExtractResource, Reflect, ShaderType)]
 | |
| #[reflect(Resource, Default)]
 | |
| pub struct GlobalsUniform {
 | |
|     /// The time since startup in seconds.
 | |
|     /// Wraps to 0 after 1 hour.
 | |
|     time: f32,
 | |
|     /// The delta time since the previous frame in seconds
 | |
|     delta_time: f32,
 | |
|     /// Frame count since the start of the app.
 | |
|     /// It wraps to zero when it reaches the maximum value of a u32.
 | |
|     frame_count: u32,
 | |
|     /// WebGL2 structs must be 16 byte aligned.
 | |
|     #[cfg(all(feature = "webgl", target_arch = "wasm32", not(feature = "webgpu")))]
 | |
|     _wasm_padding: f32,
 | |
| }
 | |
| 
 | |
| /// The buffer containing the [`GlobalsUniform`]
 | |
| #[derive(Resource, Default)]
 | |
| pub struct GlobalsBuffer {
 | |
|     pub buffer: UniformBuffer<GlobalsUniform>,
 | |
| }
 | |
| 
 | |
| fn prepare_globals_buffer(
 | |
|     render_device: Res<RenderDevice>,
 | |
|     render_queue: Res<RenderQueue>,
 | |
|     mut globals_buffer: ResMut<GlobalsBuffer>,
 | |
|     time: Res<Time>,
 | |
|     frame_count: Res<FrameCount>,
 | |
| ) {
 | |
|     let buffer = globals_buffer.buffer.get_mut();
 | |
|     buffer.time = time.elapsed_seconds_wrapped();
 | |
|     buffer.delta_time = time.delta_seconds();
 | |
|     buffer.frame_count = frame_count.0;
 | |
| 
 | |
|     globals_buffer
 | |
|         .buffer
 | |
|         .write_buffer(&render_device, &render_queue);
 | |
| }
 |