jagua_rs/entities/problems/
problem_generic.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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
use std::borrow::Borrow;

use crate::entities::instances::instance_generic::InstanceGeneric;
use crate::entities::layout::Layout;
use crate::entities::placed_item::PItemKey;
use crate::entities::placing_option::PlacingOption;
use crate::entities::problems::problem_generic::private::ProblemGenericPrivate;
use crate::entities::solution::Solution;
use crate::fsize;

/// Trait for public shared functionality of all problem variants.
pub trait ProblemGeneric: ProblemGenericPrivate {
    /// Places an item into the problem instance according to the given `PlacingOption`.
    /// Returns the index of the layout where the item was placed.
    fn place_item(&mut self, p_opt: PlacingOption) -> (LayoutIndex, PItemKey);

    /// Removes a placed item (with its unique key) from a specific `Layout`.
    /// Returns a `PlacingOption` that can be used to place the item back in the same configuration.
    /// For more information about `commit_instantly`, see [`crate::collision_detection::cd_engine::CDEngine::deregister_hazard`].
    fn remove_item(
        &mut self,
        layout_index: LayoutIndex,
        pik: PItemKey,
        commit_instantly: bool,
    ) -> PlacingOption;

    /// Saves the current state of the problem as a `Solution`.
    fn create_solution(&mut self, old_solution: Option<&Solution>) -> Solution;

    /// Restores the state of the problem to a previous `Solution`.
    fn restore_to_solution(&mut self, solution: &Solution);

    fn layouts(&self) -> &[Layout];

    fn layouts_mut(&mut self) -> &mut [Layout];

    /// Template layouts are empty and immutable.
    /// For every unique bin in the problem instance, there is a template layout.
    /// When an item is placed in a template layout, it is cloned into a real layout.
    fn template_layouts(&self) -> &[Layout];

    /// The quantity of each item that is requested but currently missing in the problem instance, indexed by item id.
    fn missing_item_qtys(&self) -> &[isize];

    /// The quantity of each item that is currently placed in the problem instance, indexed by item id.
    fn placed_item_qtys(&self) -> impl Iterator<Item = usize> {
        self.missing_item_qtys()
            .iter()
            .enumerate()
            .map(|(i, missing_qty)| (self.instance().item_qty(i) as isize - missing_qty) as usize)
    }

    fn usage(&mut self) -> fsize {
        let (total_bin_area, total_used_area) =
            self.layouts_mut().iter_mut().fold((0.0, 0.0), |acc, l| {
                let bin_area = l.bin.area;
                let used_area = bin_area * l.usage();
                (acc.0 + bin_area, acc.1 + used_area)
            });
        total_used_area / total_bin_area
    }

    fn used_bin_cost(&self) -> u64 {
        self.layouts().iter().map(|l| l.bin.value).sum()
    }

    /// Returns the `LayoutIndex` of all layouts.
    fn layout_indices(&self) -> impl Iterator<Item = LayoutIndex> {
        (0..self.layouts().len()).map(LayoutIndex::Real)
    }

    /// Returns the `LayoutIndex` of all template layouts that have remaining stock.
    fn template_layout_indices_with_stock(&self) -> impl Iterator<Item = LayoutIndex> {
        self.template_layouts()
            .iter()
            .enumerate()
            .filter_map(|(i, l)| match self.bin_qtys()[l.bin.id] {
                0 => None,
                _ => Some(LayoutIndex::Template(i)),
            })
    }

    fn get_layout(&self, index: impl Borrow<LayoutIndex>) -> &Layout {
        match index.borrow() {
            LayoutIndex::Real(i) => &self.layouts()[*i],
            LayoutIndex::Template(i) => &self.template_layouts()[*i],
        }
    }

    fn bin_qtys(&self) -> &[usize];

    /// Makes sure that the all collision detection engines are completely updated with the changes made to the layouts.
    fn flush_changes(&mut self) {
        self.layouts_mut()
            .iter_mut()
            .for_each(|l| l.flush_changes());
    }

    fn instance(&self) -> &dyn InstanceGeneric;
}

pub(super) mod private {
    /// Trait for shared functionality of all problem variants, but not exposed to the public.
    pub trait ProblemGenericPrivate: Clone {
        fn next_solution_id(&mut self) -> usize;

        fn next_layout_id(&mut self) -> usize;

        fn missing_item_qtys_mut(&mut self) -> &mut [isize];

        fn register_included_item(&mut self, item_id: usize) {
            self.missing_item_qtys_mut()[item_id] -= 1;
        }

        fn deregister_included_item(&mut self, item_id: usize) {
            self.missing_item_qtys_mut()[item_id] += 1;
        }
    }
}

pub const STRIP_LAYOUT_IDX: LayoutIndex = LayoutIndex::Real(0);

#[derive(Debug, Clone, Copy, Eq, PartialEq, Hash)]
/// Unique index for a `Layout` in a problem instance.
pub enum LayoutIndex {
    Real(usize),
    Template(usize),
}

impl Into<usize> for LayoutIndex {
    fn into(self) -> usize {
        match self {
            LayoutIndex::Real(i) | LayoutIndex::Template(i) => i,
        }
    }
}