Python生成截圖選餐GIF動(dòng)畫
之前群里有小伙伴問今天中午該吃什么,然后另一位小伙伴發(fā)了一張下面的動(dòng)圖:

我個(gè)人覺得還挺有意思的,截圖還真像抽獎(jiǎng)一樣隨機(jī)選一個(gè)菜名。考慮到這張動(dòng)圖中的菜名候選并不見得都是我們能夠吃的菜。我們可以用python根據(jù)菜名列表生成這樣的動(dòng)圖玩玩。
之前還看到什么截圖選頭像之類的動(dòng)圖,那類通過圖片生成的動(dòng)圖都比較簡(jiǎn)單,通過文中提到的Imagine的動(dòng)畫作坊工具就可以做。所以本文只演示如何生成文字動(dòng)圖。
python生成文字動(dòng)圖
下面我們一步步來(lái)完成這個(gè)操作:
下載表情圖片到本地
為了分析這種表情圖片,第一步需要先下載下來(lái),但是對(duì)于微信的表情動(dòng)圖,經(jīng)過測(cè)試還真沒法直接下載下來(lái)。
雖然通過文件監(jiān)控工具分析出,gif表情動(dòng)圖存儲(chǔ)位置在C:\Users\ASUS\Documents\WeChat Files\你的微信ID\FileStorage\CustomEmotion\xx\xxxx位置,但是卻無(wú)法用圖片工具查看。用winhex分析二進(jìn)制得到了V1MMWX這樣的文件頭,說(shuō)明微信對(duì)表情都進(jìn)行了一定程度的加密。雖然可以解密,但這樣大動(dòng)干戈未免過于麻煩。
后面終于想到了一個(gè)簡(jiǎn)單的方案,那就是把向你有權(quán)限登錄后臺(tái)的公眾號(hào)發(fā)送這個(gè)表情,再去公眾號(hào)后臺(tái)下載:

微信發(fā)送的動(dòng)圖都是存儲(chǔ)為自己特有V1MMWX加密格式,可能是為了使用自己獨(dú)創(chuàng)的壓縮算法有更大的壓縮比吧。那說(shuō)明我們想直接看本地微信存儲(chǔ)的gif動(dòng)圖,只能自行開發(fā)專門針對(duì)這種微信格式的解碼器了。
分析動(dòng)圖
下面我使用小工具Imagine,并使用動(dòng)畫作坊打開:

可以看到這張動(dòng)圖由22張文字圖片組成,幀切換時(shí)間為20毫秒。
生成單張圖片
分析完成我們考慮用PIL庫(kù)來(lái)生成單張圖片,如果還沒有安裝該庫(kù)的童鞋,使用以下命令安裝該庫(kù):
pip install pillow
下面選擇了用藍(lán)底做背景。我們先來(lái)繪制中間的菜名文字:
from PIL import Image, ImageFont, ImageDraw
text = "珍珠土豆?fàn)F牛腩"
size = 320
fontsize = (size-20)//len(text)
im = Image.new(mode='RGB', size=(size, size), color="lightblue")
draw = ImageDraw.Draw(im=im)
draw.text(xy=(10, (size-fontsize*1.5)/2),
text=text, fill=0,
font=ImageFont.truetype('msyh.ttc', size=fontsize))
im

由于菜品的名字文字個(gè)數(shù)不一致,為了都能填滿整圖,作了自動(dòng)文字大小調(diào)整處理。
字體我選擇了微軟雅黑,當(dāng)然微軟雅黑也有三種子字體,可以通過系統(tǒng)字體安裝目錄查看字體文件的屬性從而知道字體對(duì)應(yīng)的文件名:

下方帶陰影的的文字生成起來(lái)會(huì)麻煩一些,我的思路是先繪制純黑的文字,在繪制帶黑色邊緣白色填充的文字向上偏移幾個(gè)單位:
def text_border(text, x, y, font, shadowcolor, fillcolor):
draw.text((x - 1, y), text, font=font, fill=shadowcolor)
draw.text((x + 1, y), text, font=font, fill=shadowcolor)
draw.text((x, y - 1), text, font=font, fill=shadowcolor)
draw.text((x, y + 1), text, font=font, fill=shadowcolor)
draw.text((x - 1, y - 1), text, font=font, fill=shadowcolor)
draw.text((x + 1, y - 1), text, font=font, fill=shadowcolor)
draw.text((x - 1, y + 1), text, font=font, fill=shadowcolor)
draw.text((x + 1, y + 1), text, font=font, fill=shadowcolor)
draw.text((x, y), text, font=font, fill=fillcolor)
bottomtext = "不知道吃什么?截圖吃飯"
bottom_fontsize = 27
bottom_font = ImageFont.truetype('STHUPO.TTF', size=bottom_fontsize)
x, y = (size-bottom_fontsize*len(bottomtext))/2, size-bottom_fontsize*1.2
draw.text(xy=(x, y), text=bottomtext,
fill=0, font=bottom_font)
text_border(bottomtext, x, y-4,
bottom_font, 0, (255, 255, 255))
im

上述代碼選擇了華文琥珀作為字體,個(gè)人用來(lái)繪制文字邊框的方法比較簡(jiǎn)單粗暴,如果有更好的辦法,歡迎留言交流。
考慮到后續(xù)圖片發(fā)送到微信上顯示都很小,干脆現(xiàn)在就壓縮一下像素大?。?/p>
im.thumbnail((128, 128)) im

下面我們封裝一下生成代碼,方便后續(xù)調(diào)用:
from PIL import Image, ImageFont, ImageDraw
def text_img(text, bgcolor="lightblue", bottomtext="不知道吃什么?截圖吃飯", size=360, result_size=(128, 128)):
def text_border(text, x, y, font, shadowcolor, fillcolor):
draw.text((x - 1, y), text, font=font, fill=shadowcolor)
draw.text((x + 1, y), text, font=font, fill=shadowcolor)
draw.text((x, y - 1), text, font=font, fill=shadowcolor)
draw.text((x, y + 1), text, font=font, fill=shadowcolor)
draw.text((x - 1, y - 1), text, font=font, fill=shadowcolor)
draw.text((x + 1, y - 1), text, font=font, fill=shadowcolor)
draw.text((x - 1, y + 1), text, font=font, fill=shadowcolor)
draw.text((x + 1, y + 1), text, font=font, fill=shadowcolor)
draw.text((x, y), text, font=font, fill=fillcolor)
im = Image.new(mode='RGB', size=(size, size), color=bgcolor)
draw = ImageDraw.Draw(im=im)
fontsize = (size-20)//len(text)
draw.text(xy=(10, (size-fontsize*1.5)/2),
text=text, fill=0,
font=ImageFont.truetype('msyh.ttc', size=fontsize))
bottom_fontsize = (size-20)//len(bottomtext)
bottom_font = ImageFont.truetype('STHUPO.TTF', size=bottom_fontsize)
x, y = (size-bottom_fontsize*len(bottomtext))/2, size-bottom_fontsize*1.2
draw.text(xy=(x, y), text=bottomtext,
fill=0, font=bottom_font)
text_border(bottomtext, x, y-4,
bottom_font, 0, (255, 255, 255))
im.thumbnail(result_size)
return im
測(cè)試一下:
text_img("魚香茄子")

ok,現(xiàn)在我們就已經(jīng)能夠給任何菜品生成圖片了。但是菜品的名字哪里來(lái)呢?我找到了一個(gè)網(wǎng)站,下面考慮爬一下它:
爬取菜品數(shù)據(jù)
網(wǎng)址是:https://m.meishij.net/caipu/
這個(gè)網(wǎng)站結(jié)果非常簡(jiǎn)單,一個(gè)簡(jiǎn)單的xpath即可獲取到所有的菜品名稱:

下面開始下載:
from lxml import etree
import requests
req = requests.get("https://m.meishij.net/caipu/")
html = etree.HTML(req.text)
menu = html.xpath("http://dl[@class='recipe_list']//a/text()")
menu = list(set([_.strip(".") for _ in menu]))
print(len(menu), menu[:10], menu[-10:])
3744 ['排骨藕湯', '芋圓', '海鮮湯', '涼拌杏鮑菇', '三汁燜鍋', '奶香玉米汁', '炒豆角', '茄子醬', '芒果糯米糍', '饅頭'] ['清蒸茄子', '西蘭花炒雞', '老式蛋糕', '排骨年糕', '清炒絲瓜', '芋頭蒸排骨', '木耳炒肉', '蠔油油麥菜', '麻辣雞塊', '荷葉餅']
有了這些菜名,我們已經(jīng)可以用來(lái)生成動(dòng)圖了。不過為了以后還能夠?qū)W做菜,我們可以將菜名保存起來(lái),要學(xué)做菜的時(shí)候呢打開網(wǎng)頁(yè):https://so.meishi.cc/?q=菜名,進(jìn)行搜索。
保存菜名:
with open("meau.csv", "w", encoding="u8") as f:
f.write("菜名\n")
for row in menu:
f.write(row)
f.write("\n")
下面我們開始生成菜名動(dòng)圖:
生成菜名動(dòng)圖
3767多個(gè)菜名畢竟是太多,我們可以隨意取30個(gè)菜名來(lái)生成動(dòng)圖:
import random gif_list = random.choices(menu, k=30) print(gif_list)
['蒸水蛋', '肉桂卷', '涼瓜炒蛋', '芝士焗紅薯', '香蕉酥', '酸奶慕斯', '雞蛋腸粉', '紅油肚絲', '玉米雞蛋餅', '酸辣豆腐湯', '蘿卜燉牛腩', '苦瓜排骨湯', '腐竹拌芹菜', '西紅柿炒土', '蒜蓉蒸茄子', '豆沙面包', '蘑菇炒肉', '清炒蓮藕', '黑椒牛肉粒', '南瓜煎餅', '炒黃瓜', '雜糧饅頭', '桃山皮月餅', '蔥爆肉', '小炒牛肉', '豆瓣鯽魚', '蝦仁燴豆腐', '素餡餃子', '涼拌黃瓜', '砂鍋魚頭']
PS:還是自己選好菜名,寫死列表更好😅
import imageio
frames = [text_img(text) for text in gif_list]
imageio.mimsave("meau.gif", frames, 'GIF', duration=0.02)
生成結(jié)果:

根據(jù)菜名列表生成動(dòng)圖的完整代碼
import imageio
from PIL import Image, ImageFont, ImageDraw
def text_img(text, bgcolor="lightblue", bottomtext="不知道吃什么?截圖吃飯", size=360, result_size=(128, 128)):
def text_border(text, x, y, font, shadowcolor, fillcolor):
draw.text((x - 1, y), text, font=font, fill=shadowcolor)
draw.text((x + 1, y), text, font=font, fill=shadowcolor)
draw.text((x, y - 1), text, font=font, fill=shadowcolor)
draw.text((x, y + 1), text, font=font, fill=shadowcolor)
draw.text((x - 1, y - 1), text, font=font, fill=shadowcolor)
draw.text((x + 1, y - 1), text, font=font, fill=shadowcolor)
draw.text((x - 1, y + 1), text, font=font, fill=shadowcolor)
draw.text((x + 1, y + 1), text, font=font, fill=shadowcolor)
draw.text((x, y), text, font=font, fill=fillcolor)
im = Image.new(mode='RGB', size=(size, size), color=bgcolor)
draw = ImageDraw.Draw(im=im)
fontsize = (size-20)//len(text)
draw.text(xy=(10, (size-fontsize*1.5)/2),
text=text, fill=0,
font=ImageFont.truetype('msyh.ttc', size=fontsize))
bottom_fontsize = (size-20)//len(bottomtext)
bottom_font = ImageFont.truetype('STHUPO.TTF', size=bottom_fontsize)
x, y = (size-bottom_fontsize*len(bottomtext))/2, size-bottom_fontsize*1.2
draw.text(xy=(x, y), text=bottomtext,
fill=0, font=bottom_font)
text_border(bottomtext, x, y-4,
bottom_font, 0, (255, 255, 255))
im.thumbnail(result_size)
return im
def save_meau_gif(savename, meau):
frames = [text_img(text) for text in meau]
imageio.mimsave(savename, frames, 'GIF', duration=0.02)
使用示例:
meau = [
"荷葉糯米雞", "烤羊肉", "黑椒牛排", "家常大盤雞", "蒜泥豆角",
"洋蔥炒牛肉", "絲瓜炒雞蛋", "平菇炒雞蛋", "雞刨豆腐", "芙蓉鮮蔬湯",
"炒西葫蘆", "茄子豆角", "滑蛋牛肉", "香菇青菜", "地三鮮",
"醬燒杏鮑菇", "腐乳雞翅", "醋溜藕片", "椰子燉雞", "香菇燒豆腐",
"咖喱雞腿飯", "雞汁土豆泥", "茄子燉土豆", "炒烏冬面", "咖喱土豆雞",
"上湯娃娃菜", "蒜蓉蒸茄子", "芝士焗紅薯", "栗子黃燜雞", "絲瓜豆腐湯",
]
save_meau_gif("meau.gif", meau)
生成結(jié)果:

自從我們的動(dòng)圖就生成完畢啦!不知道吃啥的時(shí)候都可以拿出來(lái)截圖玩玩~🐶
😆祝大家選餐愉快~
PIL操作gif的其他操作
其實(shí)用專門動(dòng)圖處理軟件就可以操作,下面還是補(bǔ)充一下,python的操作API記錄一下:
Gif拆分
比如我們拆分一下這張圖:

from PIL import Image, ImageSequence
img = Image.open('功夫熊.gif')
for i, f in enumerate(ImageSequence.Iterator(img), 1):
f.save(f'拆分/功夫熊-{i}.png')
拆分結(jié)果:

GIF倒放
下面我們?cè)賹⑸厦孢@張動(dòng)圖倒放一下:
from PIL import Image, ImageSequence
import imageio
im = Image.open('功夫熊.gif')
sequence = [f.copy() for f in ImageSequence.Iterator(im)]
sequence.reverse() # 將列表中的幀通過reverse()函數(shù)進(jìn)行倒序
sequence[0].save('倒放功夫熊.gif', save_all=True, append_images=sequence[1:])

到此這篇關(guān)于Python生成截圖選餐GIF動(dòng)畫的文章就介紹到這了,更多相關(guān)Python生成截圖GIF動(dòng)畫內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
學(xué)習(xí)Django知識(shí)點(diǎn)分享
在本篇文章里小編給大家整理的是關(guān)于學(xué)習(xí)Django的一些心得知識(shí)點(diǎn),對(duì)此有興趣的朋友們可以參考下。2019-09-09
Python-re中search()函數(shù)的用法詳解(查找ip)
這篇文章主要介紹了Python-re中search()函數(shù)的用法-----查找ip,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03
Python趣味入門教程之循環(huán)語(yǔ)句while
這篇文章主要給大家介紹了關(guān)于Python趣味入門教程之循環(huán)語(yǔ)句while的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
pycharm 關(guān)閉search everywhere的解決操作
這篇文章主要介紹了pycharm 關(guān)閉search everywhere的解決操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2021-01-01
Python?Flask實(shí)現(xiàn)后臺(tái)任務(wù)輕松構(gòu)建高效API應(yīng)用
本文介紹如何使用Python?Flask框架實(shí)現(xiàn)后臺(tái)任務(wù),以快速構(gòu)建高效的API應(yīng)用。通過實(shí)例演示,讀者將學(xué)會(huì)如何利用Flask框架搭建后臺(tái)任務(wù),實(shí)現(xiàn)異步處理和多線程操作等高級(jí)功能,提升應(yīng)用性能和用戶體驗(yàn)2023-04-04
python實(shí)現(xiàn)修改xml文件內(nèi)容
這篇文章主要介紹了python實(shí)現(xiàn)修改xml文件內(nèi)容,XML 指可擴(kuò)展標(biāo)記語(yǔ)言,是一種標(biāo)記語(yǔ)言,是從標(biāo)準(zhǔn)通用標(biāo)記語(yǔ)言(SGML)中簡(jiǎn)化修改出來(lái)的2022-07-07
Django 實(shí)現(xiàn)購(gòu)物車功能的示例代碼
這篇文章主要介紹了Django 實(shí)現(xiàn)購(gòu)物車功能的示例代碼,實(shí)現(xiàn)了刪除產(chǎn)品和顯示購(gòu)物車的一系列購(gòu)物車的實(shí)現(xiàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2018-10-10
Python切片列表字符串如何實(shí)現(xiàn)切換
這篇文章主要介紹了Python切片列表字符串如何實(shí)現(xiàn)切換,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08

