2020-04-06 08:57:00 +00:00
|
|
|
use legion::prelude::*;
|
|
|
|
use std::{cmp::Ordering, collections::HashMap};
|
|
|
|
|
2020-04-16 02:40:24 +00:00
|
|
|
pub enum System {
|
2020-04-06 08:57:00 +00:00
|
|
|
Schedulable(Box<dyn Schedulable>),
|
|
|
|
ThreadLocal(Box<dyn Runnable>),
|
|
|
|
ThreadLocalFn(Box<dyn FnMut(&mut World, &mut Resources)>),
|
|
|
|
}
|
|
|
|
|
2020-04-16 02:40:24 +00:00
|
|
|
impl From<Box<dyn Schedulable>> for System {
|
|
|
|
fn from(system: Box<dyn Schedulable>) -> Self {
|
|
|
|
System::Schedulable(system)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Box<dyn Runnable>> for System {
|
|
|
|
fn from(system: Box<dyn Runnable>) -> Self {
|
|
|
|
System::ThreadLocal(system)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> From<T> for System
|
|
|
|
where
|
|
|
|
T: FnMut(&mut World, &mut Resources) + 'static,
|
|
|
|
{
|
|
|
|
fn from(system: T) -> Self {
|
|
|
|
System::ThreadLocalFn(Box::new(system))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-06 08:57:00 +00:00
|
|
|
#[derive(Default)]
|
|
|
|
pub struct SchedulePlan {
|
|
|
|
stages: HashMap<String, Vec<System>>,
|
|
|
|
stage_order: Vec<String>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SchedulePlan {
|
2020-04-16 19:13:05 +00:00
|
|
|
pub fn build(&mut self) -> Schedule {
|
2020-04-06 08:57:00 +00:00
|
|
|
let mut schedule_builder = Schedule::builder();
|
|
|
|
|
|
|
|
for stage in self.stage_order.drain(..) {
|
|
|
|
if let Some((_, mut systems)) = self.stages.remove_entry(&stage) {
|
|
|
|
let system_count = systems.len();
|
|
|
|
for system in systems.drain(..) {
|
|
|
|
match system {
|
|
|
|
System::Schedulable(schedulable) => {
|
|
|
|
schedule_builder = schedule_builder.add_system(schedulable);
|
|
|
|
}
|
|
|
|
System::ThreadLocal(runnable) => {
|
|
|
|
schedule_builder = schedule_builder.add_thread_local(runnable);
|
|
|
|
}
|
|
|
|
System::ThreadLocalFn(thread_local) => {
|
|
|
|
schedule_builder = schedule_builder.add_thread_local_fn(thread_local);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if system_count > 0 {
|
|
|
|
schedule_builder = schedule_builder.flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
schedule_builder.build()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_stage(&mut self, stage: &str) {
|
|
|
|
if let Some(_) = self.stages.get(stage) {
|
|
|
|
panic!("Stage already exists: {}", stage);
|
|
|
|
} else {
|
|
|
|
self.stages.insert(stage.to_string(), Vec::new());
|
|
|
|
self.stage_order.push(stage.to_string());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_stage_after(&mut self, target: &str, stage: &str) {
|
|
|
|
if let Some(_) = self.stages.get(stage) {
|
|
|
|
panic!("Stage already exists: {}", stage);
|
|
|
|
}
|
|
|
|
|
|
|
|
let target_index = self
|
|
|
|
.stage_order
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.find(|(_i, stage)| stage.as_str() == target)
|
|
|
|
.map(|(i, _)| i)
|
|
|
|
.unwrap_or_else(|| panic!("Target stage does not exist: {}", target));
|
|
|
|
|
|
|
|
self.stages.insert(stage.to_string(), Vec::new());
|
|
|
|
self.stage_order.insert(target_index + 1, stage.to_string());
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_stage_before(&mut self, target: &str, stage: &str) {
|
|
|
|
if let Some(_) = self.stages.get(stage) {
|
|
|
|
panic!("Stage already exists: {}", stage);
|
|
|
|
}
|
|
|
|
|
|
|
|
let target_index = self
|
|
|
|
.stage_order
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.find(|(_i, stage)| stage.as_str() == target)
|
|
|
|
.map(|(i, _)| i)
|
|
|
|
.unwrap_or_else(|| panic!("Target stage does not exist: {}", target));
|
|
|
|
|
|
|
|
self.stages.insert(stage.to_string(), Vec::new());
|
|
|
|
self.stage_order.insert(target_index, stage.to_string());
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_system_to_stage(
|
|
|
|
&mut self,
|
|
|
|
stage_name: &str,
|
2020-04-16 02:40:24 +00:00
|
|
|
system: impl Into<System>,
|
2020-04-06 08:57:00 +00:00
|
|
|
) -> &mut Self {
|
|
|
|
let systems = self
|
|
|
|
.stages
|
|
|
|
.get_mut(stage_name)
|
|
|
|
.unwrap_or_else(|| panic!("Stage does not exist: {}", stage_name));
|
2020-04-16 02:40:24 +00:00
|
|
|
systems.push(system.into());
|
2020-04-06 08:57:00 +00:00
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// working around the famous "rust float ordering" problem
|
|
|
|
#[derive(PartialOrd)]
|
|
|
|
struct FloatOrd(f32);
|
|
|
|
|
|
|
|
impl Ord for FloatOrd {
|
|
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
|
|
|
self.0.partial_cmp(&other.0).unwrap_or_else(|| {
|
|
|
|
if self.0.is_nan() && !other.0.is_nan() {
|
|
|
|
Ordering::Less
|
|
|
|
} else if !self.0.is_nan() && other.0.is_nan() {
|
|
|
|
Ordering::Greater
|
|
|
|
} else {
|
|
|
|
Ordering::Equal
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq for FloatOrd {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
if self.0.is_nan() && other.0.is_nan() {
|
|
|
|
true
|
|
|
|
} else {
|
|
|
|
self.0 == other.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for FloatOrd {}
|