use crate::models::common::{Distance, Duration, Location, Schedule, TimeWindow};
use crate::models::problem::{Actor, Job, Multi, Single};
use crate::models::solution::Tour;
use crate::utils::{compare_shared, short_type_name};
use rosomaxa::prelude::compare_floats;
use std::cmp::Ordering;
use std::fmt::{Debug, Formatter};
use std::sync::Arc;
#[derive(Clone, Debug, Default)]
pub struct Commute {
pub forward: CommuteInfo,
pub backward: CommuteInfo,
}
#[derive(Clone, Debug)]
pub struct CommuteInfo {
pub location: Location,
pub distance: Distance,
pub duration: Duration,
}
#[derive(Clone, Debug)]
pub struct Place {
pub location: Location,
pub duration: Duration,
pub time: TimeWindow,
}
#[derive(Debug)]
pub struct Activity {
pub place: Place,
pub schedule: Schedule,
pub job: Option<Arc<Single>>,
pub commute: Option<Commute>,
}
pub struct Route {
pub actor: Arc<Actor>,
pub tour: Tour,
}
impl Route {
pub fn deep_copy(&self) -> Self {
Self { actor: self.actor.clone(), tour: self.tour.deep_copy() }
}
}
impl Debug for Route {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
f.debug_struct(short_type_name::<Self>()).field("actor", self.actor.as_ref()).field("tour", &self.tour).finish()
}
}
impl Activity {
pub fn new_with_job(job: Arc<Single>) -> Self {
Activity {
place: Place { location: 0, duration: 0.0, time: TimeWindow { start: 0.0, end: f64::MAX } },
schedule: Schedule { arrival: 0.0, departure: 0.0 },
job: Some(job),
commute: None,
}
}
pub fn deep_copy(&self) -> Self {
Self {
place: Place {
location: self.place.location,
duration: self.place.duration,
time: self.place.time.clone(),
},
schedule: self.schedule.clone(),
job: self.job.clone(),
commute: self.commute.clone(),
}
}
pub fn has_same_job(&self, job: &Job) -> bool {
match self.retrieve_job() {
Some(j) => match (&j, job) {
(Job::Multi(lhs), Job::Multi(rhs)) => compare_shared(lhs, rhs),
(Job::Single(lhs), Job::Single(rhs)) => compare_shared(lhs, rhs),
_ => false,
},
_ => false,
}
}
pub fn retrieve_job(&self) -> Option<Job> {
match self.job.as_ref() {
Some(single) => Multi::roots(single).map(Job::Multi).or_else(|| Some(Job::Single(single.clone()))),
_ => None,
}
}
}
impl Commute {
pub fn is_zero_distance(&self) -> bool {
self.forward.is_zero_distance() & self.backward.is_zero_distance()
}
pub fn duration(&self) -> Duration {
self.forward.duration + self.backward.duration
}
}
impl Default for CommuteInfo {
fn default() -> Self {
Self { location: 0, distance: 0., duration: 0. }
}
}
impl CommuteInfo {
pub fn is_zero_distance(&self) -> bool {
let is_zero_distance = compare_floats(self.distance, 0.) == Ordering::Equal;
if is_zero_distance && compare_floats(self.duration, 0.) != Ordering::Equal {
unreachable!("expected to have duration to be zero, got: {}", self.duration);
}
is_zero_distance
}
}