使用Python制作一盏 3D 花灯喜迎元宵佳节


Posted in Python onFebruary 26, 2021

说起元宵节,各位有没有觉得这是咱们中国人最浪漫的节日呢?国人向来拘谨古板,一年到头都是小心谨慎地过日子,唯有元宵节这天可以纵情豪放一把。东风夜放花千树,宝马雕车香满路,火树银花霓虹闪烁,豪车遍地美女如云。细品,你甚至都能嗅到香奈儿的味道!月上柳梢头,人约黄昏后,这又是何等的浪漫!比起烛光晚宴、鲜花加持,这份浪漫更显纯真。晚至明清,民间元宵节的喜庆气氛,堪比西班牙的奔牛节、巴西的狂欢节、泰国的泼水节。

由于众所周知的原因,估计今年的趵突泉元宵节灯会又要黄了。去哪儿体验“花市灯如昼”的节日气氛呢?Don't worry,没有什么事能够难倒程序员——用3D技术也可以做出下图这样的走马灯,算是聊胜于无吧。

使用Python制作一盏 3D 花灯喜迎元宵佳节

1.原材料

花灯纸

如下所示,还可以加上自己喜欢的图案、文字等。

使用Python制作一盏 3D 花灯喜迎元宵佳节

Python环境和模块

一台安装了Python环境的电脑,Python环境需要安装以下模块。

  • numpy
  • pillow
  • wxgl

如果没有上述模块,请参考下面的命令安装。

pip install numpy
pip install pillow
pip install wxgl

NumPy和 pillow 是 Python 旗下最常用的科学计算库和图像处理库,属于常用模块。WxGL 是一个基于 PyOpenGL 的三维数据可视化库,以 wx 为显示后端,提供 Matplotlib 风格的交互式应用模式,同时,也可以和 wxPython 无缝结合,在wx的窗体上绘制三维模型。

2.制作工序

花灯制作工序非常简单,只需要三十行代码,可以直接在Python IDLE中以交互方式逐行执行。

导入模块

>>> import numpy as np
>>> from PIL import Image
>>> import wxgl.wxplot as plt

打开花灯纸图像

>>> fn = r'D:\temp\light0115\res\paper.png'
>>> im = np.array(Image.open(fn))/255
>>> im.shape
(400, 942, 3)

fn定义的是图像存储路径,请据实修改。Image.open(fn)打开文件,返回一个PIL对象,np.array()将PIL对象转成numpy.ndarray数组对象。除以255,将图像数据从0到255的值域范围变成0到1,适应WxGL的接口要求。查看数组的shape,显示图像分辨率为400像素高、942像素宽,每个像素有三种颜色(此处为RGB)。

根据花灯纸的大小制作龙骨

纸长942像素,卷成圆筒,半径就是149.9像素,如果把半径视为1个单位,则高度400像素相当于2.668个单位。

>>> rows, cols, deep = im.shape
>>> cols/(2*np.pi)
149.9239563925654
>>> r = 1
>>> h = 2*np.pi*rows/cols
>>> h
2.6680192387174464

接下来需要制作半径1个单位、高度2.668个单位的圆筒状龙骨了。

>>> theta = np.linspace(0, 2*np.pi, cols)
>>> x = r * np.cos(theta)
>>> y = r * np.sin(theta)
>>> z = np.linspace(0, h, rows)
>>> xs = np.tile(x, (rows,1))
>>> ys = np.tile(y, (rows,1))
>>> zs = z.repeat(cols).reshape((rows,cols))

这里的xs、ys、zs就是圆筒状龙骨上各个点的x坐标、y坐标、z坐标。下面的代码,每隔10个点抽取1个点,用mesh的方法画出龙骨形状。当然,也可以画出全部的点,那样顶点就会连成一片。

>>> plt.mesh(xs[::10,::10], ys[::10,::10], zs[::10,::10], mode='FLBL')
>>> plt.show()

用3D的方式画出来的龙骨,效果如下。

使用Python制作一盏 3D 花灯喜迎元宵佳节

给龙骨贴上花灯纸

有了龙骨,接下来就可以把花灯纸贴在龙骨上了。继续操作之前,记得先把刚才弹出的3D龙骨窗口关闭。

>>> plt.mesh(xs, ys, zs, im)
>>> plt.show()

不过,你会立刻发现,花灯纸上下方向贴反了。没关系,我们可以像下面这样反转方向。

>>> plt.mesh(xs, ys, zs, im[::-1])
>>> plt.show()

怎么样,是不是有一点走马灯的雏形了呢?

使用Python制作一盏 3D 花灯喜迎元宵佳节

制作旋转叶轮

走马灯之所以能够转动,是因为里面有蜡烛加热形成上升气流,推动顶部的叶轮旋转,从而带动花灯旋转。当然,这里的叶轮仅仅是个样子,花灯旋转依赖另外的机制实现。

>>> theta = np.linspace(0, 2*np.pi, 18, endpoint=False)
>>> x = r * np.cos(theta)
>>> y = r * np.sin(theta)
>>> x[2::3] = x[1::3]
>>> x[1::3] = 0
>>> y[2::3] = y[1::3]
>>> y[1::3] = 0
>>> z = np.ones(18) * h * 0.9
>>> vs = np.stack((x,y,z), axis=1)
>>> plt.mesh(xs, ys, zs, im[::-1])
>>> plt.surface(vs, color='#C03000', method='T', mode='FCBC', alpha=0.8)
>>> plt.show()

叶轮设计有6片,用三角形模拟,颜色深红,透明度0.8,整体效果略显粗糙了一点。

使用Python制作一盏 3D 花灯喜迎元宵佳节

加上照明灯和提系

照明灯用一个白色的圆球表示,提系则是红色的一条直线,兼做照明灯的电源线。

>>> plt.mesh(xs, ys, zs, im[::-1])
>>> plt.surface(vs, color='#C03000', method='T', mode='FCBC', alpha=0.8)
>>> plt.sphere((0,0,h*0.4), 0.4, '#FFFFFF', slices=60, mode='FCBC')
>>> plt.plot((0,0), (0,0), (0.4*h, 1.5*h), width=3.0, style='solid', cmap='hsv', caxis='z')

让花灯转起来

花灯旋转的实现非常简单,只需要给show方法一个rotation参数就可以。

plt.show(rotation='h-')

最终的花灯效果如下。

使用Python制作一盏 3D 花灯喜迎元宵佳节

3.完整代码

# -*- coding: utf-8 -*-

import numpy as np
from PIL import Image
import wxgl.wxplot as plt

im = np.array(Image.open('res/paper.png'))/255
rows, cols, deep = im.shape

r, h = 1, 2*np.pi*rows/cols
theta = np.linspace(0, 2*np.pi, cols)
x = r*np.cos(theta)
y = r*np.sin(theta)
z = np.linspace(0, h, rows)
xs = np.tile(x, (rows,1))
ys = np.tile(y, (rows,1))
zs = z.repeat(cols).reshape((rows,cols))

theta = np.linspace(0, 2*np.pi, 18, endpoint=False)
x = r*np.cos(theta)
y = r*np.sin(theta)
x[2::3] = x[1::3]
x[1::3] = 0
y[2::3] = y[1::3]
y[1::3] = 0
z = np.ones(18) * h * 0.9
vs = np.stack((x,y,z), axis=1)

plt.mesh(xs, ys, zs, im[::-1])
plt.surface(vs, color='#C03000', method='T', mode='FCBC', alpha=0.8)
plt.sphere((0,0,h*0.4), 0.4, '#FFFFFF', slices=60, mode='FCBC')
plt.plot((0,0), (0,0), (0.4*h, 1.5*h), width=3.0, style='solid', cmap='hsv', caxis='z')
plt.show(rotation='h-')

到此这篇关于使用Python制作一盏 3D 花灯喜迎元宵佳节的文章就介绍到这了,更多相关Python制作 3D 花灯内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
Python中文件操作简明介绍
Apr 13 Python
python利用datetime模块计算时间差
Aug 04 Python
Python利用带权重随机数解决抽奖和游戏爆装备问题
Jun 16 Python
python 实现得到当前时间偏移day天后的日期方法
Dec 31 Python
利用Python对文件夹下图片数据进行批量改名的代码实例
Feb 21 Python
Python3实现的简单三级菜单功能示例
Mar 12 Python
python实现随机漫步方法和原理
Jun 10 Python
Python实现快速排序的方法详解
Oct 25 Python
wxpython自定义下拉列表框过程图解
Feb 14 Python
python 给图像添加透明度(alpha通道)
Apr 09 Python
详解用 python-docx 创建浮动图片
Jan 24 Python
No module named ‘win32gui‘ 的解决方法(踩坑之旅)
Feb 18 Python
Python plt 利用subplot 实现在一张画布同时画多张图
Feb 26 #Python
Python xlwings插入Excel图片的实现方法
Feb 26 #Python
基于tensorflow __init__、build 和call的使用小结
Feb 26 #Python
python实现MySQL指定表增量同步数据到clickhouse的脚本
Feb 26 #Python
详解python的xlwings库读写excel操作总结
Feb 26 #Python
pytorch 中forward 的用法与解释说明
Feb 26 #Python
浅谈Python xlwings 读取Excel文件的正确姿势
Feb 26 #Python
You might like
PHP array_multisort()函数的使用札记
2011/07/03 PHP
PHP之正则表达式捕获组与非捕获组(详解)
2015/07/29 PHP
PHP请求Socket接口测试实例
2016/08/12 PHP
PHP微信开发之微信录音临时转永久存储
2018/01/26 PHP
实例讲解PHP中使用命名空间
2019/01/27 PHP
PHP使用gearman进行异步的邮件或短信发送操作详解
2020/02/27 PHP
浅谈利用JavaScript进行的DDoS攻击原理与防御
2015/06/04 Javascript
js HTML5上传示例代码完整版
2016/10/10 Javascript
Vue实现双向绑定的方法
2016/12/22 Javascript
详解支持Angular 2的表格控件
2017/01/19 Javascript
vue父组件中获取子组件中的数据(实例讲解)
2017/09/27 Javascript
Vue中util的工具函数实例详解
2019/07/08 Javascript
基于Echarts图表在div动态切换时不显示的解决方式
2020/07/20 Javascript
Element Collapse 折叠面板的使用方法
2020/07/26 Javascript
Django应用程序中如何发送电子邮件详解
2017/02/04 Python
Python爬虫实例爬取网站搞笑段子
2017/11/08 Python
Python Flask-web表单使用详解
2017/11/18 Python
Python语言描述KNN算法与Kd树
2017/12/13 Python
对Django中内置的User模型实例详解
2019/08/16 Python
Python 仅获取响应头, 不获取实体的实例
2019/08/21 Python
python os.path.isfile 的使用误区详解
2019/11/29 Python
python mysql 字段与关键字冲突的解决方式
2020/03/02 Python
Python+Appium实现自动化测试的使用步骤
2020/03/24 Python
python如何操作mysql
2020/08/17 Python
Python趣味入门教程之循环语句while
2020/08/26 Python
巧用CSS3的calc()宽度计算做响应模式布局的方法
2018/03/22 HTML / CSS
工程地质勘察专业大学生求职信
2013/10/13 职场文书
法律专业应届生自荐信范文
2014/01/06 职场文书
三月雷锋月活动总结
2014/07/03 职场文书
2014年小学校长工作总结
2014/12/08 职场文书
初中作文评语集锦
2014/12/25 职场文书
雷锋的观后感
2015/06/10 职场文书
退货证明模板
2015/06/23 职场文书
运动会主持词大全
2015/07/02 职场文书
用Python将库打包发布到pypi
2021/04/13 Python
解决MySQL报“too many connections“错误
2022/04/19 MySQL