-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathpy3d.py
executable file
·140 lines (120 loc) · 3.99 KB
/
py3d.py
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
130
131
132
133
134
135
136
137
138
139
#!/usr/bin/python
import sys
from PIL import Image
from math import *
FN = "teapot_small.obj"
xs, ys = 500,500 # image size
M = 60 # number of sampled points per face,
# increase if there are black dots
lx,ly,lz=1,-1,-1 # light source direction
AMB = 70 # ambient light intensity
LIGHT = 130 # direct light intensity
# Note that AMB + LIGHT must be less than 256
PFAC = 200 # perspective factor,
# a higher value e.g. 1000 will produce less distortion
im = Image.new("RGB", (xs, ys))
zbuf = xs * ys * [1e6] # Z buffer array
alpha = int(sys.argv[1])*pi/180 # rotation
beta = int(sys.argv[2])*pi/180 # angles
s = 2 # object scaling factor
xoff = 0 # X and Y
yoff = 0 # offsets
# vertices/faces/normals lists
V, F, N = [], [], []
# read input file
for l in open(FN):
if l[0] == "v" and l[1] != "n":
d, x, y, z = l.split()
V.append([float(x), float(y), float(z)])
if l[0] == "v" and l[1] == "n":
d, x, y, z = l.split()
N.append([float(x), float(y), float(z)])
if l[0] == "f":
d, x, y, z = l.split()
x = x.split("/")[0]
y = y.split("/")[0]
z = z.split("/")[0]
F.append([int(x), int(y), int(z)])
print(f"{len(V)} vertices, {len(N)} normals, {len(F)} faces")
w1 = sin(alpha)
w2 = cos(alpha)
w3 = sin(beta)
w4 = cos(beta)
# transformed vertices/vertex normals and average Z distance for each face
TV, TVN, FD = [], [], []
# apply rotation to vertices
for x,y,z in V:
px = x*w1 + y*w2
py = x*w2*w4 - y*w1*w4 + z*w3
pz = x*w3*w2 - y*w3*w1 - z*w4
TV.append([px,py,pz])
# plot each vertex:
#im.putpixel((int(s*px+xs/2), int(s*py+ys/2)), (255,255,255))
# apply rotation to vertex normals
for x,y,z in N:
px = x*w1 + y*w2
py = x*w2*w4 - y*w1*w4 + z*w3
pz = x*w3*w2 - y*w3*w1 - z*w4
TVN.append([px,py,pz])
# sort faces by depth
for a,b,c in F:
z1 = TV[a-1][2]
z2 = TV[b-1][2]
z3 = TV[c-1][2]
za = (z1+z2+z3) / 3
FD.append(za)
FL = list(zip(FD, F))
FL.sort(key = lambda x: -x[0])
FLIST = FL
FLEN = len(FLIST)
ind = 0
def vlen(x,y,z):
return sqrt(x*x + y*y + z*z)
# get light intensity based on surface normal and light direction
def getcol(n1, n2, n3):
ang = (lx*n1 + ly*n2 + lz*n3) / (vlen(lx,ly,lz) * vlen(n1,n2,n3))
c = max(AMB, AMB + LIGHT * ang)
return int(c)
# loop over all faces
for dd, ff in FLIST:
ind += 1
x1,y1,z1 = TV[ff[0]-1][0], TV[ff[0]-1][1], TV[ff[0]-1][2]
x2,y2,z2 = TV[ff[1]-1][0], TV[ff[1]-1][1], TV[ff[1]-1][2]
x3,y3,z3 = TV[ff[2]-1][0], TV[ff[2]-1][1], TV[ff[2]-1][2]
dx1,dy1,dz1 = x2-x1,y2-y1,z2-z1
dx2,dy2,dz2 = x3-x1,y3-y1,z3-z1
n1, n2, n3 = TVN[ff[0]-1][0], TVN[ff[0]-1][1], TVN[ff[0]-1][2]
c1 = getcol(n1, n2, n3)
n1, n2, n3 = TVN[ff[1]-1][0], TVN[ff[1]-1][1], TVN[ff[1]-1][2]
c2 = getcol(n1, n2, n3)
n1, n2, n3 = TVN[ff[2]-1][0], TVN[ff[2]-1][1], TVN[ff[2]-1][2]
c3 = getcol(n1, n2, n3)
cd1 = c2 - c1
cd2 = c3 - c1
for l in range(M+1):
xxa = x1 + dx1*l/M
yya = y1 + dy1*l/M
zza = z1 + dz1*l/M
xxb = x1 + dx2*l/M
yyb = y1 + dy2*l/M
zzb = z1 + dz2*l/M
ca = c1 + cd1*l/M
cb = c1 + cd2*l/M
for n in range(M+1):
f1 = n/M
f2 = 1 - f1
xxx = f1 * xxa + f2 * xxb
yyy = f1 * yya + f2 * yyb
zzz = f1 * zza + f2 * zzb
d = (PFAC - zzz) / PFAC # apply some linear perspective
c = int(f1*ca + f2*cb)
xpos, ypos = int(xoff+s*d*xxx+xs/2), int(yoff+s*d*yyy+ys/2)
if xpos < 0 or ypos < 0 or xpos > xs - 1 or ypos > ys - 1:
continue
zind = ypos * xs + xpos
# use Z buffer to determine if a point is visible
if zzz < zbuf[zind]:
zbuf[zind] = zzz
im.putpixel((xpos, ypos), (c,c,c))
print()
im.save("img%04u.png" % int(sys.argv[3]))