ifc_lite_processing/style/
indexed_colour.rs1use super::Rgba;
21use ifc_lite_core::{DecodedEntity, EntityDecoder};
22use ifc_lite_geometry::Mesh;
23
24#[derive(Debug, Clone)]
28pub struct FullIndexedColourMap {
29 pub geometry_id: u32,
31 pub colours: Vec<Rgba>,
33 pub triangle_palette: Vec<usize>,
35}
36
37impl FullIndexedColourMap {
38 pub(crate) fn distinct_used(&self) -> usize {
40 let mut seen = self.triangle_palette.clone();
41 seen.sort_unstable();
42 seen.dedup();
43 seen.len()
44 }
45
46 pub fn dominant(&self) -> Rgba {
49 let mut counts: rustc_hash::FxHashMap<usize, u32> = rustc_hash::FxHashMap::default();
50 for &p in &self.triangle_palette {
51 *counts.entry(p).or_insert(0) += 1;
52 }
53 let idx = counts
54 .iter()
55 .max_by_key(|(_, c)| *c)
56 .map(|(&i, _)| i)
57 .unwrap_or(0);
58 self.colours.get(idx).copied().unwrap_or(Rgba::new(0.8, 0.8, 0.8, 1.0))
59 }
60}
61
62pub fn resolve_indexed_colour_map_full(
70 entity: &DecodedEntity,
71 decoder: &mut EntityDecoder,
72) -> Option<FullIndexedColourMap> {
73 let geometry_id = entity.get_ref(0)?;
74 let opacity = entity
75 .get(1)
76 .and_then(|a| a.as_float())
77 .map(|v| v as f32)
78 .unwrap_or(1.0)
79 .clamp(0.0, 1.0);
80 let colours_id = entity.get_ref(2)?;
81 let index_attr = entity.get(3)?;
82 let index_list = index_attr.as_list()?;
83 if index_list.is_empty() {
84 return None;
85 }
86
87 let colours_entity = decoder.decode_by_id(colours_id).ok()?;
88 let colour_list = colours_entity.get(0)?.as_list()?;
89 let colours: Vec<Rgba> = colour_list
90 .iter()
91 .filter_map(|c| {
92 let rgb = c.as_list()?;
93 let r = rgb.first().and_then(|v| v.as_float())? as f32;
94 let g = rgb.get(1).and_then(|v| v.as_float())? as f32;
95 let b = rgb.get(2).and_then(|v| v.as_float())? as f32;
96 Some(Rgba::new(r, g, b, opacity))
97 })
98 .collect();
99 if colours.is_empty() {
100 return None;
101 }
102
103 let max_idx = colours.len() - 1;
104 let triangle_palette: Vec<usize> = index_list
105 .iter()
106 .map(|v| {
107 let one_based = v.as_int().unwrap_or(1).max(1) as usize;
108 (one_based - 1).min(max_idx)
109 })
110 .collect();
111
112 Some(FullIndexedColourMap {
113 geometry_id,
114 colours,
115 triangle_palette,
116 })
117}
118
119pub fn split_mesh_by_indexed_colour(
127 mesh: &Mesh,
128 map: &FullIndexedColourMap,
129) -> Option<Vec<(Rgba, Mesh)>> {
130 let tri_count = mesh.indices.len() / 3;
131 if tri_count == 0 || tri_count != map.triangle_palette.len() {
132 return None;
133 }
134 if map.distinct_used() < 2 {
135 return None; }
137
138 let has_normals = mesh.normals.len() == mesh.positions.len();
139 let rtc_applied = mesh.rtc_applied;
140
141 #[derive(Default)]
143 struct Group {
144 positions: Vec<f32>,
145 normals: Vec<f32>,
146 indices: Vec<u32>,
147 }
148 let mut groups: Vec<Option<Group>> = (0..map.colours.len()).map(|_| None).collect();
149
150 for (tri, &palette) in map.triangle_palette.iter().enumerate() {
151 let tri_in_range = (0..3).all(|k| {
155 let vi = mesh.indices[tri * 3 + k] as usize;
156 vi * 3 + 2 < mesh.positions.len()
157 });
158 if !tri_in_range {
159 continue;
160 }
161
162 let group = groups[palette].get_or_insert_with(Group::default);
163 for k in 0..3 {
164 let vi = mesh.indices[tri * 3 + k] as usize;
165 let base = vi * 3;
166 let new_index = (group.positions.len() / 3) as u32;
167 group.positions.push(mesh.positions[base]);
168 group.positions.push(mesh.positions[base + 1]);
169 group.positions.push(mesh.positions[base + 2]);
170 if has_normals {
171 group.normals.push(mesh.normals[base]);
172 group.normals.push(mesh.normals[base + 1]);
173 group.normals.push(mesh.normals[base + 2]);
174 }
175 group.indices.push(new_index);
176 }
177 }
178
179 let out: Vec<(Rgba, Mesh)> = groups
180 .into_iter()
181 .enumerate()
182 .filter_map(|(palette, group)| {
183 let group = group?;
184 if group.indices.is_empty() {
185 return None;
186 }
187 let mesh = Mesh {
188 positions: group.positions,
189 normals: group.normals,
190 indices: group.indices,
191 rtc_applied,
192 };
193 Some((map.colours[palette], mesh))
194 })
195 .collect();
196
197 (out.len() >= 2).then_some(out)
198}
199
200#[cfg(test)]
201mod tests {
202 use super::{split_mesh_by_indexed_colour, FullIndexedColourMap};
203 use crate::style::Rgba;
204 use ifc_lite_geometry::Mesh;
205
206 #[test]
207 fn split_drops_out_of_range_triangle_without_partial_geometry() {
208 let positions: Vec<f32> = (0..6).flat_map(|i| [i as f32, 0.0, 0.0]).collect();
212 let mesh = Mesh {
213 positions,
214 normals: Vec::new(),
215 indices: vec![0, 1, 2, 3, 4, 5, 0, 1, 99],
216 rtc_applied: false,
217 };
218 let map = FullIndexedColourMap {
219 geometry_id: 1,
220 colours: vec![Rgba::new(1.0, 0.0, 0.0, 1.0), Rgba::new(0.0, 1.0, 0.0, 1.0)],
221 triangle_palette: vec![0, 1, 0],
223 };
224
225 let parts = split_mesh_by_indexed_colour(&mesh, &map)
226 .expect("two valid palette groups survive after dropping the OOB triangle");
227
228 let total_tris: usize = parts
229 .iter()
230 .map(|(_, m)| {
231 assert_eq!(m.indices.len() % 3, 0, "index buffer must be whole triangles");
232 assert_eq!(m.positions.len() % 3, 0, "positions must be whole vertices");
233 m.indices.len() / 3
234 })
235 .sum();
236 assert_eq!(
237 total_tris, 2,
238 "the out-of-range triangle must be dropped, not partially emitted"
239 );
240 }
241}