jagua_rs/entities/
bin.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
use std::sync::Arc;

use itertools::Itertools;

use crate::collision_detection::cd_engine::CDEngine;
use crate::collision_detection::hazard::Hazard;
use crate::collision_detection::hazard::HazardEntity;
use crate::entities::quality_zone::InferiorQualityZone;
use crate::entities::quality_zone::N_QUALITIES;
use crate::fsize;
use crate::geometry::geo_traits::Shape;
use crate::geometry::primitives::aa_rectangle::AARectangle;
use crate::geometry::primitives::simple_polygon::SimplePolygon;
use crate::geometry::transformation::Transformation;
use crate::util::config::CDEConfig;

/// A container in which items can be placed.
#[derive(Clone, Debug)]
pub struct Bin {
    pub id: usize,
    /// The contour of the bin
    pub outer: Arc<SimplePolygon>,
    /// The cost of using the bin
    pub value: u64,
    /// Transformation applied to the shape with respect to the original shape in the input file (for example to center it).
    pub pretransform: Transformation,
    /// Shapes of holes/defects in the bins, if any
    pub holes: Vec<Arc<SimplePolygon>>,
    /// Zones of different qualities in the bin, stored per quality.
    pub quality_zones: [Option<InferiorQualityZone>; N_QUALITIES],
    /// The starting state of the `CDEngine` for this bin.
    pub base_cde: Arc<CDEngine>,
    pub area: fsize,
}

impl Bin {
    pub fn new(
        id: usize,
        outer: SimplePolygon,
        value: u64,
        pretransform: Transformation,
        holes: Vec<SimplePolygon>,
        quality_zones: Vec<InferiorQualityZone>,
        cde_config: CDEConfig,
    ) -> Self {
        let outer = Arc::new(outer);
        let holes = holes.into_iter().map(Arc::new).collect_vec();
        assert_eq!(
            quality_zones.len(),
            quality_zones.iter().map(|qz| qz.quality).unique().count(),
            "Quality zones must have unique qualities"
        );
        assert!(
            quality_zones
                .iter()
                .map(|qz| qz.quality)
                .all(|q| q < N_QUALITIES),
            "All quality zones must be below N_QUALITIES"
        );
        let quality_zones = {
            let mut qz = <[_; N_QUALITIES]>::default();
            for q in quality_zones {
                let quality = q.quality;
                qz[quality] = Some(q);
            }
            qz
        };

        let bin_hazards = generate_bin_hazards(&outer, &holes, &quality_zones);

        let base_cde = CDEngine::new(outer.bbox().inflate_to_square(), bin_hazards, cde_config);
        let base_cde = Arc::new(base_cde);
        let area = outer.area() - holes.iter().map(|h| h.area()).sum::<fsize>();

        Self {
            id,
            outer,
            value,
            pretransform,
            holes,
            quality_zones,
            base_cde,
            area,
        }
    }

    /// Create a new `Bin` for a strip-packing problem. Instead of a shape, the bin is always rectangular.
    pub fn from_strip(rect: AARectangle, cde_config: CDEConfig) -> Self {
        let id = 0;
        //The "original" x_min and y_min of the strip should always be at (0, 0)
        let pretransform = Transformation::from_translation((rect.x_min, rect.y_min));

        let poly = SimplePolygon::from(rect);
        let value = poly.area() as u64;

        Bin::new(id, poly, value, pretransform, vec![], vec![], cde_config)
    }

    pub fn bbox(&self) -> AARectangle {
        self.outer.bbox()
    }
}

fn generate_bin_hazards(
    outer: &Arc<SimplePolygon>,
    holes: &[Arc<SimplePolygon>],
    quality_zones: &[Option<InferiorQualityZone>],
) -> Vec<Hazard> {
    //Hazard induced by the outside of the bin
    let mut hazards = vec![Hazard::new(HazardEntity::BinExterior, outer.clone())];

    //Hazard induced by any holes in the bin
    hazards.extend(holes.iter().enumerate().map(|(i, shape)| {
        let haz_entity = HazardEntity::BinHole { id: i };
        Hazard::new(haz_entity, shape.clone())
    }));

    //Hazards induced by quality zones
    for q_zone in quality_zones.iter().flatten() {
        for (id, shape) in q_zone.zones.iter().enumerate() {
            let haz_entity = HazardEntity::InferiorQualityZone {
                quality: q_zone.quality,
                id,
            };
            hazards.push(Hazard::new(haz_entity, shape.clone()));
        }
    }
    hazards
}