From f8d8e60719b7f37aea7d0feb0b8469acc5871079 Mon Sep 17 00:00:00 2001 From: William Hart Date: Sat, 10 Aug 2024 17:40:30 +1000 Subject: [PATCH 01/11] Exit splash screen on escape --- src/screens/splash.rs | 49 ++++++++++++------------------------------- 1 file changed, 13 insertions(+), 36 deletions(-) diff --git a/src/screens/splash.rs b/src/screens/splash.rs index fd40a6de..5bd7b06a 100644 --- a/src/screens/splash.rs +++ b/src/screens/splash.rs @@ -1,6 +1,7 @@ //! A splash screen that plays briefly at startup. use bevy::{ + input::common_conditions::input_just_pressed, prelude::*, render::texture::{ImageLoaderSettings, ImageSampler}, }; @@ -35,48 +36,24 @@ pub(super) fn plugin(app: &mut App) { ) .run_if(in_state(Screen::Splash)), ); + + // exit the splash screen early if the player hits escape + app.add_systems( + Update, + exit_splash_screen + .run_if(input_just_pressed(KeyCode::Escape).and_then(in_state(Screen::Splash))), + ); } const SPLASH_BACKGROUND_COLOR: Color = Color::srgb(0.157, 0.157, 0.157); const SPLASH_DURATION_SECS: f32 = 1.8; const SPLASH_FADE_DURATION_SECS: f32 = 0.6; -fn spawn_splash(mut commands: Commands, asset_server: Res) { - commands - .ui_root() - .insert(( - Name::new("Splash screen"), - BackgroundColor(SPLASH_BACKGROUND_COLOR), - StateScoped(Screen::Splash), - )) - .with_children(|children| { - children.spawn(( - Name::new("Splash image"), - ImageBundle { - style: Style { - margin: UiRect::all(Val::Auto), - width: Val::Percent(70.0), - ..default() - }, - image: UiImage::new(asset_server.load_with_settings( - // This should be an embedded asset for instant loading, but that is - // currently [broken on Windows Wasm builds](https://github.com/bevyengine/bevy/issues/14246). - "images/splash.png", - |settings: &mut ImageLoaderSettings| { - // Make an exception for the splash image in case - // `ImagePlugin::default_nearest()` is used for pixel art. - settings.sampler = ImageSampler::linear(); - }, - )), - ..default() - }, - UiImageFadeInOut { - total_duration: SPLASH_DURATION_SECS, - fade_duration: SPLASH_FADE_DURATION_SECS, - t: 0.0, - }, - )); - }); + +fn exit_splash_screen(mut next_screen: ResMut>) { + next_screen.set(Screen::Loading); +} + } #[derive(Component, Reflect)] From 5d9b42474be6279a7a38306d944fc2572ae99b8f Mon Sep 17 00:00:00 2001 From: William Hart Date: Sat, 10 Aug 2024 17:40:46 +1000 Subject: [PATCH 02/11] Use a vec of splash screen images --- src/screens/splash.rs | 95 ++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 93 insertions(+), 2 deletions(-) diff --git a/src/screens/splash.rs b/src/screens/splash.rs index 5bd7b06a..8a6da3cf 100644 --- a/src/screens/splash.rs +++ b/src/screens/splash.rs @@ -1,5 +1,7 @@ //! A splash screen that plays briefly at startup. +use std::collections::VecDeque; + use bevy::{ input::common_conditions::input_just_pressed, prelude::*, @@ -12,6 +14,8 @@ use crate::{theme::prelude::*, AppSet}; pub(super) fn plugin(app: &mut App) { // Spawn splash screen. app.insert_resource(ClearColor(SPLASH_BACKGROUND_COLOR)); + app.init_resource::(); + app.add_systems(OnEnter(Screen::Splash), spawn_splash); // Animate splash screen. @@ -49,11 +53,71 @@ const SPLASH_BACKGROUND_COLOR: Color = Color::srgb(0.157, 0.157, 0.157); const SPLASH_DURATION_SECS: f32 = 1.8; const SPLASH_FADE_DURATION_SECS: f32 = 0.6; +#[derive(Component)] +struct SplashScreenContainer; + +#[derive(Resource)] +struct SplashScreenImageList(VecDeque<&'static str>); + +impl Default for SplashScreenImageList { + fn default() -> Self { + // Use paths here rather than an ImageHandle as the loading screen hasn't + // run yet. To show your own splash images, replace or add here + Self(VecDeque::from_iter(["images/splash.png"].into_iter())) + } +} fn exit_splash_screen(mut next_screen: ResMut>) { next_screen.set(Screen::Loading); } +fn splash_image_bundle(asset_server: &AssetServer, path: &'static str) -> impl Bundle { + ( + Name::new("Splash image"), + ImageBundle { + style: Style { + margin: UiRect::all(Val::Auto), + width: Val::Percent(70.0), + ..default() + }, + image: UiImage::new(asset_server.load_with_settings( + // This should be an embedded asset for instant loading, but that is + // currently [broken on Windows Wasm builds](https://github.com/bevyengine/bevy/issues/14246). + path, + |settings: &mut ImageLoaderSettings| { + // Make an exception for the splash image in case + // `ImagePlugin::default_nearest()` is used for pixel art. + settings.sampler = ImageSampler::linear(); + }, + )), + background_color: BackgroundColor(Color::srgba(0., 0., 0., 0.)), + ..default() + }, + UiImageFadeInOut { + total_duration: SPLASH_DURATION_SECS, + fade_duration: SPLASH_FADE_DURATION_SECS, + t: 0.0, + }, + ) +} + +fn spawn_splash( + mut commands: Commands, + splash_images: Res, + mut next_screen: ResMut>, +) { + if splash_images.0.is_empty() { + // if there are no splash images, go directly to the loading screen + next_screen.set(Screen::Loading); + } else { + // spawn the UI root but wait for the first timer tick to show an image + commands.ui_root().insert(( + Name::new("Splash screen container"), + BackgroundColor(SPLASH_BACKGROUND_COLOR), + StateScoped(Screen::Splash), + SplashScreenContainer, + )); + } } #[derive(Component, Reflect)] @@ -112,8 +176,35 @@ fn tick_splash_timer(time: Res