jagua_rs/entities/
solution.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
use std::time::Instant;

use itertools::Itertools;

use crate::entities::instances::instance::Instance;
use crate::entities::instances::instance_generic::InstanceGeneric;
use crate::entities::layout::LayoutSnapshot;
use crate::fsize;
use crate::geometry::geo_traits::Shape;

/// Represents a snapshot of a `Problem` at a specific moment.
/// Solutions can be used to restore the state of a `Problem` to a previous state.
#[derive(Debug, Clone)]
pub struct Solution {
    /// Unique identifier for the solution
    pub id: usize,
    /// Snapshots of all `Layout`s in the `Problem` at the moment the solution was created
    pub layout_snapshots: Vec<LayoutSnapshot>,
    /// Average usage of bins in the solution
    pub usage: fsize,
    /// Quantity of placed items for each `Item` in the solution
    pub placed_item_qtys: Vec<usize>,
    /// Target quantity of each `Item` in the solution
    pub target_item_qtys: Vec<usize>,
    /// Quantity of bins used for each type of bin
    pub bin_qtys: Vec<usize>,
    /// Instant the solution was created
    pub time_stamp: Instant,
}

impl Solution {
    pub fn new(
        id: usize,
        layout_snapshots: Vec<LayoutSnapshot>,
        usage: fsize,
        placed_item_qtys: Vec<usize>,
        target_item_qtys: Vec<usize>,
        bin_qtys: Vec<usize>,
    ) -> Self {
        Solution {
            id,
            layout_snapshots,
            usage,
            placed_item_qtys,
            target_item_qtys,
            bin_qtys,
            time_stamp: Instant::now(),
        }
    }

    /// Whether all items demanded in the `instance` are placed
    pub fn is_complete(&self, instance: &dyn InstanceGeneric) -> bool {
        self.placed_item_qtys
            .iter()
            .enumerate()
            .all(|(i, &qty)| qty >= instance.item_qty(i))
    }

    /// Ratio of included item area vs total demanded item area in the instance
    pub fn completeness(&self, instance: &Instance) -> fsize {
        let total_item_area = instance.item_area();
        let included_item_area = self
            .placed_item_qtys
            .iter()
            .enumerate()
            .map(|(i, qty)| instance.item(i).shape.area() * *qty as fsize)
            .sum::<fsize>();
        included_item_area / total_item_area
    }

    /// Returns the quantities of the items that still need to be placed to reach a complete solution.
    pub fn missing_item_qtys(&self, instance: &Instance) -> Vec<isize> {
        debug_assert!(instance.items().len() == self.placed_item_qtys.len());
        self.placed_item_qtys
            .iter()
            .enumerate()
            .map(|(i, &qty)| instance.item_qty(i) as isize - qty as isize)
            .collect_vec()
    }

    pub fn n_items_placed(&self) -> usize {
        self.placed_item_qtys.iter().sum()
    }
}