DIY 雪花特效(一)

对任意一张图片,我们可以用代码给它加点雪花特效,做成动图。

原图:

加特效后:

首先我们需要雪花的图片,比如下面这张(其实这张有点不太真实,不过无妨,我只是拿它来做演示)。

做图像分割后我们就得到了16张雪花的图片(其实一张也足够):

用pickle模块将它们保存到本地,以供后续调用。

万事俱备,下面我们来给原图加上雪花。主要代码如下:

"""
Created on Sat Dec 15 10:15:18 2019
@author: wangsp
"""
import numpy as np
from matplotlib import pyplot as plt
import cv2
import pickle
import imageio
from random import randint,normalvariate

def create_gif(frames, gif_name,duration=0.3,reverse=False):
    if reverse: frames.sort(reverse=True)
    imageio.mimsave(gif_name,frames,"GIF",duration=duration)
    return frames
    
BG = plt.imread("plum.jpg")#加载背景图
H,W,C = BG.shape
frame  = BG.astype(np.uint16)
frames = []

class Snow:
    def __init__(self,type_,x0,y0,speed0,scale):
        self.scale = scale
        #type_ : from 1 to 16
        self.data = self.get_snow(type_)
        self.x = x0
        self.y = y0
        self.speed = speed0
        
    def get_snow(self, type_):
        with open("snow%d.data"%type_,"rb") as f:
            data = pickle.load(f)
            #缩小一点
            self.h,self.w = data.shape
            data = cv2.resize(data,(int(self.h*self.scale),int(self.w*self.scale)))
            
            self.h,self.w = data.shape
            snow = np.zeros((self.h,self.w,3),dtype = np.uint8)
            snow[data==1] = np.array([255,255,255])
        return snow
        
    def update(self): #动画效果
        global frames, frame
        
        c = self.w%2 #compensation, 偶数0, 奇数1
        #restore
        if self.y < self.h:
            frame[0:self.y, self.x-int(self.w/2):self.x+int(self.w/2)+c] = BG[0:self.y,self.x-int(self.w/2):self.x+int(self.w/2)+c]
        elif self.y < H:
            frame[self.y-self.h:self.y, self.x-int(self.w/2):self.x+int(self.w/2)+c]= BG[self.y-self.h:self.y,self.x-int(self.w/2):self.x+int(self.w/2)+c]
        elif self.y < H + self.h:
            frame[self.y-self.h:, self.x-int(self.w/2):self.x+int(self.w/2)+c] = BG[self.y-self.h:,self.x-int(self.w/2):self.x+int(self.w/2)+c]
        else:
            pass
        
        self.x += 0
        self.y += self.speed
        if self.y < self.h:
            frame[0:self.y, self.x-int(self.w/2):self.x+int(self.w/2)+c] += self.data[self.h-self.y:,:]
        elif self.y < H:
            frame[self.y-self.h:self.y, self.x-int(self.w/2):self.x+int(self.w/2)+c] += self.data
        elif self.y < H + self.h:
            frame[self.y-self.h:, self.x-int(self.w/2):self.x+int(self.w/2)+c] += self.data[:H-(self.y-self.h),:]
        else:
            self.y = 0
        frame[frame>255] = 255
       

n_snows = randint(100,120) # include both ends
snows  = []
for i in range(n_snows):
    type_ = randint(1,16) #include both ends
    x0 = randint(10,W-10)
    y0 = randint(0,H)
    speed = int(normalvariate(3,0.5))
    scale = normalvariate(0.02,0.005)
    snow = Snow(type_,x0, y0, speed,scale)
    snows.append(snow)
              
n_frames = 30 #因为公众号只能穿5M以下的图,所以帧数弄少一点
for n in range(n_frames):
    for i in range(n_snows):
        snows[i].update()
    frames.append(frame.astype(np.uint8))
                
create_gif(frames,"snow_plum_5M.gif",0.01) #创建GIF

至此 ,我们可以给小倩下场雪:

效果略显粗糙,有时间的话会再做优化。

本文分享自微信公众号 - Python编程 pyqt matplotlib(wsplovePython),作者:wsp001

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2020-03-09

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • PyQt5 对话框 数据验证

    本篇介绍PyQt5对话框的数据合法性的验证。有两种验证方式:预防式验证(preventative)和 提交后验证 (post-mortem)。预防式验证适合于单...

    用户6021899
  • PyQt5 非模态对话框(apply 型)

    如果希望用户可以重复更改对话框中的参数并能马上看到修改结果,那么就要使用非模态对话框,这样用户就可以按照他们喜欢的方式来持续不断地修改数据并验证修改的结果了。

    用户6021899
  • PyQt5 多文档窗口界面

    本篇介绍多文档窗口界面的写法。代码中并未实现关闭窗口前文档未保存的提醒对话框,因为之前已有介绍。

    用户6021899
  • tornado学习笔记

    tornado是默认自动开启转义的,大家可以根据需求来选是否转义,但是要知道转义的本意是来防止浏览器意外执行恶意代码的,所以去掉转义的时候需要谨慎选择

    py3study
  • Python魔术方法-Magic Method

    目录[-] 介绍 在Python中,所有以“__”双下划线包起来的方法,都统称为“Magic Method”,例如类的初始化方法 __init__ ,Pyt...

    jhao104
  • Python魔法方法指南

    什么是魔法方法呢?它们在面向对象的Python的处处皆是。它们是一些可以让你对类添加“魔法”的特殊方法。 它们经常是两个下划线包围来命名的(比如 __init_...

    py3study
  • python 长连接 mysql数据库

    python链接mysql中没有长链接的概念,但我们可以利用mysql的ping机制,来实现长链接功能

    py3study
  • Seleninum&PhamtomJS爬取煎蛋网妹子图

    mylog.py  日志模块,记录一些爬取过程中的信息,在大量爬取的时候,没有log帮助定位,很难找到错误点

    py3study
  • 高效处理流量加解密——Burpy

    先来地址:Github: https://github.com/mr-m0nst3r/Burpy

    用户2202688
  • Tiknter例子3

    ============================================

    py3study

扫码关注云+社区

领取腾讯云代金券