jagua_rs/entities/problems/
strip_packing.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
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
use std::{iter, slice};

use crate::collision_detection::hazard::HazardEntity;
use crate::entities::bin::Bin;
use crate::entities::instances::instance_generic::InstanceGeneric;
use crate::entities::instances::strip_packing::SPInstance;
use crate::entities::layout::Layout;
use crate::entities::placed_item::PItemKey;
use crate::entities::placing_option::PlacingOption;
use crate::entities::problems::problem_generic::ProblemGeneric;
use crate::entities::problems::problem_generic::private::ProblemGenericPrivate;
use crate::entities::problems::problem_generic::{LayoutIndex, STRIP_LAYOUT_IDX};
use crate::entities::solution::Solution;
use crate::fsize;
use crate::geometry::geo_traits::{Shape, Transformable};
use crate::geometry::primitives::aa_rectangle::AARectangle;
use crate::util::assertions;
use crate::util::config::CDEConfig;
use itertools::Itertools;
use log::error;

/// Strip Packing Problem
#[derive(Clone)]
pub struct SPProblem {
    pub instance: SPInstance,
    pub layout: Layout,
    missing_item_qtys: Vec<isize>,
    layout_id_counter: usize,
    solution_id_counter: usize,
}

impl SPProblem {
    pub fn new(instance: SPInstance, strip_width: fsize, cde_config: CDEConfig) -> Self {
        let strip_height = instance.strip_height;
        let missing_item_qtys = instance
            .items
            .iter()
            .map(|(_, qty)| *qty as isize)
            .collect_vec();
        let strip_rect = AARectangle::new(0.0, 0.0, strip_width, strip_height);
        let strip_bin = Bin::from_strip(strip_rect, cde_config);
        let layout_id_counter = 0;
        let layout = Layout::new(layout_id_counter, strip_bin);

        Self {
            instance,
            layout,
            missing_item_qtys,
            layout_id_counter,
            solution_id_counter: 0,
        }
    }

    /// Adds or removes width in the back of the strip.
    pub fn modify_strip_in_back(&mut self, new_width: fsize) {
        let bbox = self.layout.bin.outer.bbox();
        let new_strip_shape =
            AARectangle::new(bbox.x_min, bbox.y_min, bbox.x_min + new_width, bbox.y_max);
        self.modify_strip(new_strip_shape);
    }

    /// Adds or removes width at the front of the strip.
    pub fn modify_strip_at_front(&mut self, new_width: fsize) {
        let bbox = self.layout.bin.outer.bbox();
        let new_strip_shape =
            AARectangle::new(bbox.x_max - new_width, bbox.y_min, bbox.x_max, bbox.y_max);
        self.modify_strip(new_strip_shape);
    }

    /// Adds or removes width, dividing it equally at the front and back of the current items.
    pub fn modify_strip_centered(&mut self, new_width: fsize) {
        let current_range = self.occupied_range().unwrap_or((0.0, 0.0));
        let current_width = self.occupied_width();

        //divide the added or removed width to the left and right of the strip
        let added_width = new_width - current_width;
        let new_x_min = current_range.0 - added_width / 2.0;
        let new_x_max = current_range.1 + added_width / 2.0;

        let new_strip_shape = AARectangle::new(
            new_x_min,
            self.layout.bin.outer.bbox().y_min,
            new_x_max,
            self.layout.bin.outer.bbox().y_max,
        );

        self.modify_strip(new_strip_shape);
    }

    /// Modifies the shape of the strip to a new rectangle.
    /// All items that fit in the new strip are kept, the rest are removed.
    pub fn modify_strip(&mut self, rect: AARectangle) {
        let placed_items = self
            .layout
            .placed_items()
            .iter()
            .map(|(_, pi)| (pi.item_id, pi.d_transf))
            .collect_vec();

        //reset the missing item quantities
        self.missing_item_qtys
            .iter_mut()
            .enumerate()
            .for_each(|(i, qty)| *qty = self.instance.item_qty(i) as isize);

        //Modifying the width causes the bin to change, so the layout must be replaced
        self.layout = Layout::new(
            self.next_layout_id(),
            Bin::from_strip(rect, self.layout.bin.base_cde.config()),
        );

        //place the items back in the new layout
        for (item_id, d_transf) in placed_items {
            let item = self.instance.item(item_id);
            let entities_to_ignore = self
                .layout
                .cde()
                .all_hazards()
                .filter(|h| h.entity != HazardEntity::BinExterior)
                .map(|h| h.entity)
                .collect_vec();
            let shape = &item.shape;
            let transform = d_transf.compose();
            let transformed_shape = shape.transform_clone(&transform);
            let cde = self.layout.cde();
            if !cde.poly_collides(&transformed_shape, entities_to_ignore.as_ref()) {
                let insert_opt = PlacingOption {
                    layout_idx: STRIP_LAYOUT_IDX,
                    item_id,
                    d_transf,
                };
                self.place_item(insert_opt);
            } else {
                let collisions =
                    cde.collect_poly_collisions(&transformed_shape, entities_to_ignore.as_ref());
                error!(
                    "Item {} could not be placed back in the strip after resizing. Collisions: {:?}",
                    item_id, collisions
                );
            }
        }
    }

    /// Shrinks the strip to the minimum width that fits all items.
    pub fn fit_strip(&mut self) {
        let n_items_in_old_strip = self.layout.placed_items().len();

        let fitted_width = self.occupied_width() * 1.00001; //add some tolerance to avoid rounding errors or false collision positives
        self.modify_strip_centered(fitted_width);

        assert_eq!(
            n_items_in_old_strip,
            self.layout.placed_items().len(),
            "fitting the strip should not remove any items"
        );
    }

    /// Returns the horizontal range occupied by the placed items. If no items are placed, returns None.
    pub fn occupied_range(&self) -> Option<(fsize, fsize)> {
        occupied_range(&self.layout)
    }

    /// Returns the width occupied by the placed items.
    pub fn occupied_width(&self) -> fsize {
        occupied_width(&self.layout)
    }

    pub fn strip_width(&self) -> fsize {
        self.layout.bin.outer.bbox().width()
    }

    pub fn strip_height(&self) -> fsize {
        self.layout.bin.outer.bbox().height()
    }
}

impl ProblemGeneric for SPProblem {
    fn place_item(&mut self, p_opt: PlacingOption) -> (LayoutIndex, PItemKey) {
        assert_eq!(
            p_opt.layout_idx, STRIP_LAYOUT_IDX,
            "Strip packing problems only have a single layout"
        );
        let item_id = p_opt.item_id;
        let item = self.instance.item(item_id);
        let placed_item_key = self.layout.place_item(item, p_opt.d_transf);

        self.register_included_item(item_id);
        (STRIP_LAYOUT_IDX, placed_item_key)
    }

    fn remove_item(
        &mut self,
        layout_index: LayoutIndex,
        pik: PItemKey,
        commit_instantly: bool,
    ) -> PlacingOption {
        assert_eq!(
            layout_index, STRIP_LAYOUT_IDX,
            "strip packing problems only have a single layout"
        );
        let pi = self.layout.remove_item(pik, commit_instantly);
        self.deregister_included_item(pi.item_id);

        PlacingOption::from_placed_item(layout_index, &pi)
    }

    fn create_solution(&mut self, _old_solution: Option<&Solution>) -> Solution {
        let id = self.next_solution_id();
        let included_item_qtys = self.placed_item_qtys().collect_vec();
        let bin_qtys = self.bin_qtys().to_vec();
        let layout_snapshots = vec![self.layout.create_snapshot()];
        let target_item_qtys = self
            .instance
            .items
            .iter()
            .map(|(_, qty)| *qty)
            .collect_vec();

        let solution = Solution::new(
            id,
            layout_snapshots,
            self.usage(),
            included_item_qtys,
            target_item_qtys,
            bin_qtys,
        );

        debug_assert!(assertions::problem_matches_solution(self, &solution));

        solution
    }

    fn restore_to_solution(&mut self, solution: &Solution) {
        debug_assert!(solution.layout_snapshots.len() == 1);

        //restore the layout
        let layout_snapshot = &solution.layout_snapshots[0];
        match self.layout.id() == layout_snapshot.id {
            true => self.layout.restore(layout_snapshot),
            false => self.layout = Layout::from_snapshot(layout_snapshot),
        }

        //restore the missing item quantities
        self.missing_item_qtys
            .iter_mut()
            .enumerate()
            .for_each(|(i, qty)| {
                *qty = (self.instance.item_qty(i) - solution.placed_item_qtys[i]) as isize
            });

        debug_assert!(assertions::problem_matches_solution(self, solution));
    }

    fn layouts(&self) -> &[Layout] {
        slice::from_ref(&self.layout)
    }

    fn layouts_mut(&mut self) -> &mut [Layout] {
        slice::from_mut(&mut self.layout)
    }

    fn template_layouts(&self) -> &[Layout] {
        &[]
    }

    fn missing_item_qtys(&self) -> &[isize] {
        &self.missing_item_qtys
    }

    fn template_layout_indices_with_stock(&self) -> impl Iterator<Item = LayoutIndex> {
        iter::empty::<LayoutIndex>()
    }

    fn bin_qtys(&self) -> &[usize] {
        &[0]
    }

    fn instance(&self) -> &dyn InstanceGeneric {
        &self.instance
    }
}

impl ProblemGenericPrivate for SPProblem {
    fn next_solution_id(&mut self) -> usize {
        self.solution_id_counter += 1;
        self.solution_id_counter
    }

    fn next_layout_id(&mut self) -> usize {
        self.layout_id_counter += 1;
        self.layout_id_counter
    }

    fn missing_item_qtys_mut(&mut self) -> &mut [isize] {
        &mut self.missing_item_qtys
    }
}

/// Returns the horizontal range occupied by the placed items. If no items are placed, returns None.
pub fn occupied_range(layout: &Layout) -> Option<(fsize, fsize)> {
    if layout.placed_items().is_empty() {
        return None;
    }

    let mut min_x = fsize::MAX;
    let mut max_x = fsize::MIN;

    for pi in layout.placed_items().values() {
        let bbox = pi.shape.bbox();
        min_x = min_x.min(bbox.x_min);
        max_x = max_x.max(bbox.x_max);
    }

    Some((min_x, max_x))
}

/// Returns the total width occupied by the placed items.
pub fn occupied_width(layout: &Layout) -> fsize {
    let range = occupied_range(layout);
    match range {
        Some((min_x, max_x)) => max_x - min_x,
        None => 0.0,
    }
}

/// Returns the width of the strip in the solution.
pub fn strip_width(solution: &Solution) -> fsize {
    solution.layout_snapshots[0].bin.outer.bbox().width()
}