-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path09_models_and_images.py
148 lines (106 loc) · 4.01 KB
/
09_models_and_images.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
140
141
142
143
144
145
146
147
148
import math
import os
import sys
import glm
import moderngl
import pygame
from objloader import Obj
from PIL import Image
os.environ['SDL_WINDOWS_DPI_AWARENESS'] = 'permonitorv2'
pygame.init()
pygame.display.set_mode((800, 800), flags=pygame.OPENGL | pygame.DOUBLEBUF, vsync=True)
class ImageTexture:
def __init__(self, path):
self.ctx = moderngl.get_context()
img = Image.open(path).convert('RGBA')
self.texture = self.ctx.texture(img.size, 4, img.tobytes())
self.sampler = self.ctx.sampler(texture=self.texture)
def use(self):
self.sampler.use()
class ModelGeometry:
def __init__(self, path):
self.ctx = moderngl.get_context()
obj = Obj.open(path)
self.vbo = self.ctx.buffer(obj.pack('vx vy vz nx ny nz tx ty'))
def vertex_array(self, program):
return self.ctx.vertex_array(program, [(self.vbo, '3f 12x 2f', 'in_vertex', 'in_uv')])
class Mesh:
def __init__(self, program, geometry, texture=None):
self.ctx = moderngl.get_context()
self.vao = geometry.vertex_array(program)
self.texture = texture
def render(self, position, color, scale):
self.vao.program['use_texture'] = False
if self.texture:
self.vao.program['use_texture'] = True
self.texture.use()
self.vao.program['position'] = position
self.vao.program['color'] = color
self.vao.program['scale'] = scale
self.vao.render()
class Scene:
def __init__(self):
self.ctx = moderngl.get_context()
self.program = self.ctx.program(
vertex_shader='''
#version 330 core
uniform mat4 camera;
uniform vec3 position;
uniform float scale;
layout (location = 0) in vec3 in_vertex;
layout (location = 1) in vec3 in_normal;
layout (location = 2) in vec2 in_uv;
out vec3 v_vertex;
out vec3 v_normal;
out vec2 v_uv;
void main() {
v_vertex = position + in_vertex * scale;
v_normal = in_normal;
v_uv = in_uv;
gl_Position = camera * vec4(v_vertex, 1.0);
}
''',
fragment_shader='''
#version 330 core
uniform sampler2D Texture;
uniform bool use_texture;
uniform vec3 color;
in vec3 v_vertex;
in vec3 v_normal;
in vec2 v_uv;
layout (location = 0) out vec4 out_color;
void main() {
out_color = vec4(color, 1.0);
if (use_texture) {
out_color *= texture(Texture, v_uv);
}
}
''',
)
self.logo_texture = ImageTexture('j.jpg')
self.car_geometry = ModelGeometry('Labrador-Retriever_02.OBJ')
self.car = Mesh(self.program, self.car_geometry)
self.crate_geometry = ModelGeometry('Labrador-Retriever_02.OBJ')
self.crate = Mesh(self.program, self.crate_geometry, self.logo_texture)
def camera_matrix(self):
now = pygame.time.get_ticks() / 1000.0
eye = (math.cos(now), math.sin(now), 0.5)
proj = glm.perspective(45.0, 1.0, 0.1, 1000.0)
look = glm.lookAt(eye, (0.0, 0.0, 0.0), (0.0, 0.0, 1.0))
return proj * look
def render(self):
camera = self.camera_matrix()
self.ctx.clear()
self.ctx.enable(self.ctx.DEPTH_TEST)
self.program['camera'].write(camera)
self.car.render((-0.4, 0.0, 0.0), (1.0, 0.0, 0.0), 0.2)
self.crate.render((0.0, 0.0, 0.0), (1.0, 1.0, 1.0), 0.2)
self.car.render((0.4, 0.0, 0.0), (0.0, 0.0, 1.0), 0.2)
scene = Scene()
while True:
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
scene.render()
pygame.display.flip()