use bevy::{
prelude::*,
tasks::{futures_lite::future, IoTaskPool, Task},
};
use de_core::fs::conf_dir;
use de_core::state::AppState;
use de_gui::ToastEvent;
use iyes_progress::prelude::*;
use tracing::error;
use crate::macros::ConfigLoadError;
use crate::Configuration;
pub(super) struct ConfPlugin;
impl Plugin for ConfPlugin {
fn build(&self, app: &mut App) {
app.add_systems(OnEnter(AppState::AppLoading), start_loading)
.add_systems(OnExit(AppState::AppLoading), cleanup)
.add_systems(
Update,
poll_conf
.track_progress()
.run_if(in_state(AppState::AppLoading)),
);
}
}
#[derive(Resource)]
struct LoadingTask(Task<Result<Configuration, ConfigLoadError>>);
fn cleanup(mut commands: Commands) {
commands.remove_resource::<LoadingTask>();
}
fn start_loading(mut commands: Commands) {
let task = IoTaskPool::get().spawn(async {
let path = conf_dir().map_err(ConfigLoadError::from)?.join("conf.yaml");
Configuration::load(path.as_path()).await
});
commands.insert_resource(LoadingTask(task));
}
fn poll_conf(
mut commands: Commands,
task: Option<ResMut<LoadingTask>>,
conf: Option<Res<Configuration>>,
mut toasts: EventWriter<ToastEvent>,
) -> Progress {
if conf.is_some() {
return true.into();
}
match task {
Some(mut task) => match future::block_on(future::poll_once(&mut task.0)) {
Some(result) => match result {
Ok(configuration) => {
commands.insert_resource(configuration);
true.into()
}
Err(err) => {
error!("{err}");
toasts.send(ToastEvent::new("Configuration loading failed."));
commands.init_resource::<Configuration>();
true.into()
}
},
None => false.into(),
},
None => false.into(),
}
}