Python作为目前较广泛的编程语言, 用于制作3D游戏可谓得心应手。本文讲解使用Python pyglet库自制简易3D引擎的方法技巧。
目录
- 导入pyglet及初始化
- 相机控制
- 3D图形绘制
- 用计时器实现动画效果
- 主程序实现
先放效果图:
导入pyglet及初始化
pyglet.window
部分用于实现窗口操作, pyglet.gl
以及pyglet.gl.glu
模块包含了OpenGL的绘图函数, 也是3D引擎中的关键部分。pyglet
模块可通过pip安装: pip install pyglet
。
import pyglet
from pyglet.gl import *
from pyglet.gl.glu import *
from pyglet.window import key
import math
from random import random, randint
定义常量和数据:
WIDTH=400;HEIGHT=400
angle_xy = math.pi / 2 # X-Y平面内的相机角度, 弧度制(0-360°变为0-2π)
angle_z = 0 # 相机绕Z轴旋转的角度
distance = 20
centerx,centery,centerz = 38,9, -11 # 中心点位置, 相机绕中心点旋转
data = [(3.2045, 50.7902), (1.5227, 49.5507), (0.2268, 47.9115), (-0.6292, 46.0584), (-0.9964, 44.1779), (-0.8378, 42.4295), (-0.5548, 41.6411), (-0.1348, 40.9221),
(0.4219, 40.2782), (1.9425, 39.2276), (2.9045, 38.8233), (4.0, 38.5), (2.9045, 38.1766), (1.9424, 37.7722), (0.4218, 36.7216), (-0.1349, 36.0778), (-0.5548, 35.3589), (-0.8378, 34.5706), (-0.9964, 32.8226),
(-0.6291, 30.9425), (0.227, 29.09), (1.5229, 27.4513), (3.2048, 26.2122), (5.2203, 25.5298), (7.5231, 25.5104), (10.0749, 26.2121), (12.8454, 27.6578), (15.812, 29.848),
(18.9593, 32.7712), (22.2779, 36.4131), (24.0, 38.5), (22.2778, 40.5874), (18.9591, 44.2302), (15.8118, 47.1541), (12.8451, 49.3448), (10.0746, 50.7907),
(7.5228, 51.4924), (5.22, 51.4729)]# 心形的矢量图数据
z1=8; z2=10 # 心形两个面的z坐标
初始化pyglet
的Window
对象, 用于绘制图形、接收事件。
window = pyglet.window.Window(height=HEIGHT, width=WIDTH)
相机控制
类似于一些3D游戏,我们这里的相机需要实现360°无死角的旋转。
convert_pos()
用于转换相机的坐标。其中:
相机绕着一个中心点, 2个方向旋转。
想象相机和中心点距离不变, 中心点是球心, 相机就相当于在一个球体表面自由移动。程序中angle_xy
类似于球体的“经度”, angle_z
类似“纬度”。通过“经度”和“纬度”, 就能控制相机移动。
另外, 一些3D游戏除了使用以上的“经度”和“纬度”, 还使用了第三根轴控制相机的旋转。
像下面这样:
def convert_pos():
# 将相机角度转换为相机的X,Y,Z坐标
if math.pi/2 < angle_z < math.pi * 1.5:
flag = -1 # 相机朝下
else:
flag = 1 # 相机朝上
cam_x=math.cos(angle_xy)*distance *math.cos(angle_z) + centerx
cam_y=math.sin(angle_xy)*distance *math.cos(angle_z) + centery
cam_z=math.sin(angle_z)*distance + centerz
return cam_x,cam_y,cam_z,flag
这是绑定pyglet窗口的事件。
按↑,↓,←,→键或拖动鼠标切换查看角度,按Page Up / Page Down键或滚动鼠标,调整远近。
@window.event
def on_key_press(k,_):
global angle_xy,angle_z,distance
if k==key.DOWN: # 下
angle_z -= math.pi * 1/18 # 10°
elif k==key.UP:# 上
angle_z += math.pi * 1/18
elif k==key.LEFT: # 左
angle_xy -= math.pi * 1/18
elif k==key.RIGHT: # 右
angle_xy += math.pi * 1/18
elif k==key.PAGEUP: # page up
distance/=1.15
elif k==key.PAGEDOWN: # page down
distance*=1.15
angle_z %= math.pi*2
on_draw()
@window.event
def on_mouse_drag(x,y,dx,dy,btn,_): # 拖动鼠标, dx和dy为鼠标位置变化的多少
global angle_xy, angle_z
angle_xy -= dx / 100
angle_z -= dy / 100
angle_z %= math.pi * 2
on_draw()
@window.event
def on_mouse_scroll(x,y, _, d): # 滚动鼠标, d为滚动的多少
global distance
distance /= 1.1**d
on_draw()
3D图形绘制
这是window对象的on_draw
事件,调用openGL绘图的代码都应该放在这个函数里面。
如果代码看不懂,可以看上篇中关于openGL 3D绘图的介绍。
@window.event # 表示绑定 window 对象的事件
def on_draw(): # 注意函数名, 必须是on_draw才能绑定绘制的事件
glMatrixMode(GL_PROJECTION) # 设置当前矩阵为投影矩阵
glLoadIdentity()
# 透视投影, 前4个参数类似游戏中的FOV(视角大小),
# 后2个参数分别是物体与相机的最近、最远距离
glFrustum(-5, 5, -5, 5, 2, 1000) # 透视投影
glMatrixMode(GL_MODELVIEW) # 模型视图矩阵
glLoadIdentity()
glViewport(0, 0, WIDTH, HEIGHT)
window.clear() # 或 glClear(GL_COLOR_BUFFER_BIT)
glClear(GL_DEPTH_BUFFER_BIT) # 清除深度(z排序)缓冲区
# 改变相机位置和角度
cam_x,cam_y,cam_z,flag = convert_pos()
gluLookAt(cam_x,cam_y,cam_z,centerx,centery,centerz,0,0,flag) # 0,0,flag为相机朝上方向
for dx,dy,dz in lst_pos: # 绘制列表中的各个心形
draw_heart(dx,dy,dz)
glFlush() # 刷新绘图缓冲区
下面的draw_heart()
函数用于在不同位置绘制3D心形, 根据二维的心形矢量图数据, 绘制正面和侧面, 形成三维的形状。glVertex3f
函数定义心形每个顶点的坐标。
def draw_heart(dx,dy,dz): # 绘制心形
# dx, dy, dz为心形从中心点向X, Y, Z轴正方向平移多少
# 绘制顶、底面
glBegin(GL_POLYGON)
glColor3f(random()*0.5+0.5, 0, random()*0.5+0.5) # 随机生成颜色
for x, y in data:
glVertex3f(y+dx, z1+dy, -x+dz) # 使用y,z和-x, 旋转心形, 使心形更易于查看
glEnd()
glBegin(GL_POLYGON)
for x, y in data:
glVertex3f(y+dx, z2+dy, -x+dz)
glEnd()
# 绘制侧面
glColor3f(0.5, 0, 0.5)
for i in range(len(data)):
if i + 1 == len(data): # 到达列表末尾
next_point = data[0]
else:
next_point = data[i + 1]
point = data[i]
glBegin(GL_POLYGON)
glVertex3f(point[1]+dx, z1+dy, -point[0]+dz)
glVertex3f(next_point[1]+dx, z1+dy, -next_point[0]+dz)
glVertex3f(next_point[1]+dx, z2+dy, -next_point[0]+dz)
glVertex3f(point[1]+dx, z2+dy, -point[0]+dz)
glEnd()
用计时器实现动画效果
pyglet库自带了计时器功能。首先,导入pyglet.clock
模块 (有点像pygame)。
from pyglet import clock
通过clock
模块中的schedule_interval
函数,可以设定计时器间隔。这里程序每隔0.02秒,调用一次animate函数。在animate函数中再加入实现动画的代码。
def animate(event):
global centerx
centerx+=1
xpos=lst_pos[0][0]
lst_pos[0]=(xpos+1,0,0)
on_draw() # 重新绘制
# 每隔0.02秒,调用一次animate函数
clock.schedule_interval(animate, 0.02)
主程序实现
主程序随机生成多个心形, 类似游戏中使用的过程生成技术。
# 随机生成多个心形
lst_pos = [(0,0,0)] # 中心的心形
for i in range(20):
lst_pos.append((randint(-200,200),randint(-200,200),randint(-200,200)))
glClearColor(0.8, 1, 1, 1)
glEnable(GL_DEPTH_TEST) # 开启深度(z排序), 使程序支持近的物体遮挡远的物体
pyglet.app.run()
学习了这么多, 应该可以编写自己的简易3D游戏了吧! 欢迎点赞、收藏。
作者的其他3D程序作品, 见作者的gitcode:gitcode.net/qfcy_/python/-/tree/master/opengl。
下面是自己的星空程序(space.py)的截图: