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
#![crate_name = "obj"]
#![crate_type = "lib"]
#[cfg(feature = "usegenmesh")]
extern crate genmesh;
use std::fs::File;
use std::path::Path;
use std::io::{self, BufReader};
use std::collections::HashMap;
use std::rc::Rc;
use std::iter::Filter;
use std::str::Split;
pub use obj::{Obj, Object, Group, IndexTuple, GenPolygon, SimplePolygon};
pub use mtl::{Mtl, Material};
mod obj;
mod mtl;
pub fn load<P: GenPolygon>(path: &Path) -> io::Result<Obj<Rc<Material>,P>> {
File::open(path).map(|f| {
let mut f = BufReader::new(f);
let obj = Obj::load(&mut f);
let mut materials = HashMap::new();
for m in obj.materials().iter() {
let mut p = path.to_path_buf();
p.pop();
p.push(m);
let file = File::open(&p).ok().expect("failed to open material");
let mut f = BufReader::new(file);
let m = Mtl::load(&mut f);
for m in m.materials.into_iter() {
materials.insert(m.name.clone(), Rc::new(m));
}
}
obj.map(|g| {
let Group {
name,
index,
material,
indices
} = g;
let material: Option<Rc<Material>> = match material {
Some(m) => materials.get(&m).map(|m| m.clone()),
None => None
};
Group {
name: name,
index: index,
material: material,
indices: indices
}
})
})
}
type Words<'a> = Filter<Split<'a, fn(char) -> bool>, fn(&&str) -> bool>;
fn words<'a>(s: &'a str) -> Words<'a> {
fn is_not_empty(s: &&str) -> bool { !s.is_empty() }
let is_not_empty: fn(&&str) -> bool = is_not_empty;
fn is_whitespace(c: char) -> bool { c.is_whitespace() }
let is_whitespace: fn(char) -> bool = is_whitespace;
s.split(is_whitespace).filter(is_not_empty)
}