forked from fogleman/fauxgl
-
Notifications
You must be signed in to change notification settings - Fork 1
/
obj.go
79 lines (76 loc) · 1.81 KB
/
obj.go
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
package fauxgl
import (
"bufio"
"os"
"strconv"
"strings"
)
func parseIndex(value string, length int) int {
parsed, _ := strconv.ParseInt(value, 0, 0)
n := int(parsed)
if n < 0 {
n += length
}
return n
}
func LoadOBJ(path string) (*Mesh, error) {
file, err := os.Open(path)
if err != nil {
return nil, err
}
defer file.Close()
vs := make([]Vector, 1, 1024) // 1-based indexing
vts := make([]Vector, 1, 1024) // 1-based indexing
vns := make([]Vector, 1, 1024) // 1-based indexing
var triangles []*Triangle
scanner := bufio.NewScanner(file)
for scanner.Scan() {
line := scanner.Text()
fields := strings.Fields(line)
if len(fields) == 0 {
continue
}
keyword := fields[0]
args := fields[1:]
switch keyword {
case "v":
f := ParseFloats(args)
v := Vector{f[0], f[1], f[2]}
vs = append(vs, v)
case "vt":
f := ParseFloats(args)
v := Vector{f[0], f[1], 0}
vts = append(vts, v)
case "vn":
f := ParseFloats(args)
v := Vector{f[0], f[1], f[2]}
vns = append(vns, v)
case "f":
fvs := make([]int, len(args))
fvts := make([]int, len(args))
fvns := make([]int, len(args))
for i, arg := range args {
vertex := strings.Split(arg+"//", "/")
fvs[i] = parseIndex(vertex[0], len(vs))
fvts[i] = parseIndex(vertex[1], len(vts))
fvns[i] = parseIndex(vertex[2], len(vns))
}
for i := 1; i < len(fvs)-1; i++ {
i1, i2, i3 := 0, i, i+1
t := Triangle{}
t.V1.Position = vs[fvs[i1]]
t.V2.Position = vs[fvs[i2]]
t.V3.Position = vs[fvs[i3]]
t.V1.Normal = vns[fvns[i1]]
t.V2.Normal = vns[fvns[i2]]
t.V3.Normal = vns[fvns[i3]]
t.V1.Texture = vts[fvts[i1]]
t.V2.Texture = vts[fvts[i2]]
t.V3.Texture = vts[fvts[i3]]
t.FixNormals()
triangles = append(triangles, &t)
}
}
}
return NewTriangleMesh(triangles), scanner.Err()
}