use crate::BlenderMesh;
use failure::Fail;
use std::collections::hash_map::Entry;
use std::collections::HashMap;
pub type MeshesByFilename = HashMap<String, MeshesByMeshName>;
pub type MeshesByMeshName = HashMap<String, BlenderMesh>;
pub fn parse_meshes_from_blender_stdout(blender_stdout: &str) -> MeshesByFilename {
let mut filenames_to_meshes = HashMap::new();
let mut index = 0;
while let Some((filename_to_mesh, next_start_index)) =
find_first_mesh_after_index(blender_stdout, index)
{
for (filename, meshes) in filename_to_mesh.into_iter() {
match filenames_to_meshes.entry(filename) {
Entry::Vacant(v) => {
v.insert(meshes);
}
Entry::Occupied(ref mut o) => {
o.get_mut().extend(meshes);
}
}
}
index = next_start_index;
}
filenames_to_meshes
}
pub type FlattenedExportedMeshes = HashMap<String, BlenderMesh>;
pub fn flatten_exported_meshes_owned(
meshes_by_filename: MeshesByFilename,
) -> Result<FlattenedExportedMeshes, FlattenMeshError> {
let mut flattened_meshes = HashMap::new();
let mut duplicate_meshes: HashMap<String, Vec<String>> = HashMap::new();
for (source_filename, meshes) in meshes_by_filename.into_iter() {
for (mesh_name, mesh) in meshes.into_iter() {
match duplicate_meshes.entry(mesh_name.to_string()) {
Entry::Occupied(mut duplicates) => {
duplicates.get_mut().push(source_filename.to_string());
}
Entry::Vacant(filenames) => {
filenames.insert(vec![source_filename.to_string()]);
}
};
flattened_meshes.insert(mesh_name, mesh);
}
}
validate_no_duplicates(duplicate_meshes)?;
Ok(flattened_meshes)
}
pub fn flatten_exported_meshes(
meshes_by_filename: &MeshesByFilename,
) -> Result<HashMap<&str, &BlenderMesh>, FlattenMeshError> {
let mut flattened_meshes = HashMap::new();
let mut duplicate_meshes: HashMap<String, Vec<String>> = HashMap::new();
for (source_filename, meshes) in meshes_by_filename.iter() {
for (mesh_name, mesh) in meshes.iter() {
flattened_meshes.insert(mesh_name.as_str(), mesh);
match duplicate_meshes.entry(mesh_name.to_string()) {
Entry::Occupied(mut duplicates) => {
duplicates.get_mut().push(source_filename.to_string());
}
Entry::Vacant(filenames) => {
filenames.insert(vec![source_filename.to_string()]);
}
};
}
}
validate_no_duplicates(duplicate_meshes)?;
Ok(flattened_meshes)
}
fn validate_no_duplicates(
duplicate_meshes: HashMap<String, Vec<String>>,
) -> Result<(), FlattenMeshError> {
let duplicate_meshes: HashMap<String, Vec<String>> = duplicate_meshes
.into_iter()
.filter(|(_mesh_name, filenames)| filenames.len() > 1)
.collect();
if duplicate_meshes.len() > 0 {
return Err(FlattenMeshError::DuplicateMeshNamesAcrossFiles {
duplicates: duplicate_meshes,
});
}
Ok(())
}
#[derive(Debug, Fail)]
pub enum FlattenMeshError {
#[fail(display = "Duplicate meshes found: {:#?}", duplicates)]
DuplicateMeshNamesAcrossFiles {
duplicates: HashMap<String, Vec<String>>,
},
}
fn find_first_mesh_after_index(
blender_stdout: &str,
index: usize,
) -> Option<(MeshesByFilename, usize)> {
let blender_stdout = &blender_stdout[index as usize..];
if let Some(mesh_start_index) = blender_stdout.find("START_MESH_JSON") {
let mut filenames_to_meshes = HashMap::new();
let mut mesh_name_to_data = HashMap::new();
let mesh_end_index = blender_stdout.find("END_MESH_JSON").unwrap();
let mesh_data = &blender_stdout[mesh_start_index..mesh_end_index];
let mut lines = mesh_data.lines();
let first_line = lines.next().unwrap();
let mesh_filename: Vec<&str> = first_line.split(" ").collect();
let mesh_filename = mesh_filename[1].to_string();
let mesh_name = first_line.split(" ").last().unwrap().to_string();
let mesh_data: String = lines.collect();
let mesh_data: BlenderMesh = serde_json::from_str(&mesh_data).unwrap();
mesh_name_to_data.insert(mesh_name, mesh_data);
filenames_to_meshes.insert(mesh_filename, mesh_name_to_data);
return Some((filenames_to_meshes, index + mesh_end_index + 1));
}
return None;
}