前言
前段時間在一個朋友那么得到的靈感,想到可以用音樂播放頁面作為一張海報圖片。其實接下來要講的和海報還是有差距的,而具體實現(xiàn)也只是簡單的圖片粘貼,但是在效果上還是不錯的。效果圖如下,希望大家喜歡:
![](http://img.jbzj.com/file_images/article/202101/2021126152955563.jpeg?202102615303)
左邊是原圖,右邊是需要添加到中間的圖,也是圖的主角。其實如果直接用ps實現(xiàn)上面的圖是非常簡單的,反倒是用代碼實現(xiàn)有點曲折,不過實現(xiàn)過程還是非常有趣的,希望這篇博客可以可以讓你學到知識。
用Pillow創(chuàng)建圓形圖
在上面的圖片中,中間是一個圓形圖片,而Pillow本身是沒有提供生成圓形圖片的方法(也可能是我沒找到),所以就需要自己實現(xiàn)。在實現(xiàn)之前,我們先安裝Pillow模塊:
要創(chuàng)建圓形圖,我們先根據(jù)原圖的大小,創(chuàng)建一個RGBA模式的透明圖:
# 該方法傳入三個參數(shù),第一個為模式,第二個為大小的元組,第三個為顏色
im = Image.new('RGBA', (300, 300), (255, 255, 255, 0))
上述代碼是創(chuàng)建了一個完全透明的300*300的圖片,我們在該圖片上繪制一個最大的圓:
# 獲取繪制者
drawer = ImageDraw.Draw(im)
# 繪制一個黃色的圓,ellipse方法傳入三個參數(shù),第一個為包含該圓的最小正方形的區(qū)域,第二個為顏色,第三個為邊寬
drawer.ellipse((0, 0, 300, 300), fill=(255, 255, 0), width=0)
生成圖片如下:
![](http://img.jbzj.com/file_images/article/202101/2021126153106080.png?2021026153116)
我們準備一張300*300的正方形圖片,然后遍歷圖片的每個像素,如果像素值的A==0(即像素不透明)那我們就將圖片該區(qū)域的像素值設置為透明。
![](http://img.jbzj.com/file_images/article/202101/2021126153133044.jpeg?2021026153142)
代碼如下:
# 打開要轉換成圓形的圖片,我們事先把圖片裁剪好
pic = Image.open(img_path).convert('RGBA')
# 遍歷圖片的每個像素
for i in range(300):
for j in range(300):
# 獲取該像素點的像素
r, g, b, a = im.getpixel((i, j))
# 當rgb值不是黃色時,即像素值為透明時
if (r, g, b) != (255, 255, 0):
# 將原圖的像素值設置為透明
pic.putpixel((i, j), (255, 255, 255, 0))
我們的pic就是圓形圖片了,完整代碼如下:
# 背景圖中圓的直徑
radius = 533
# 圖片的大小
circle_size = (radius, radius)
def generate_circle_image(img_path):
# 創(chuàng)建一個透明的正方形
im = Image.new('RGBA', circle_size, (255, 255, 255, 0))
# 獲取繪畫者
drawer = ImageDraw.Draw(im)
# 在透明的正方形上畫一個黃色的圓
drawer.ellipse((0, 0, circle_size[0], circle_size[1]), fill=(255, 255, 0), width=0)
# 打開要轉換成圓形的圖片,我們事先把圖片裁剪好
pic = Image.open(img_path).convert('RGBA')
# 修改圖片大小,讓圖片和圓大小一樣
re_pic = pic.resize(circle_size, Image.ANTIALIAS)
# 遍歷圖片的每個像素
for i in range(circle_size[0]):
for j in range(circle_size[1]):
r, g, b, a = im.getpixel((i, j))
if (r, g, b) != (255, 255, 0):
re_pic.putpixel((i, j), (255, 255, 255, 0))
return re_pic
在上面的方法中,我們改進了一些代碼,之前我們需要使用指定大小的圖片作為素材,現(xiàn)在只需要是正方形圖片就可以了。
生成海報
原本我以為直接將圓形圖片粘貼到背景圖片上就可以了,但是試過之后發(fā)現(xiàn)效果和我想的不太一樣,反正就是沒成功,效果如下:
![](http://img.jbzj.com/file_images/article/202101/2021126153237350.jpeg?2021026153247)
是粘貼上去了沒錯,而且圖片也是透明效果,但是從這個效果來看粘貼只是像素替換,而不是圖片疊加。于是我又想到了遍歷像素的辦法。
我通過特殊手段獲取了上面正方形離左邊的像素,和離上邊的像素(其實就是用ps看了一下)。我嘗試過圖形檢測和像素判斷的辦法,想自動識別中間圓的位置,但是效果不佳(反正就是失敗了),所以只能無恥的用ps查看像素。
回到正題,我們用遍歷像素的辦法不需要遍歷整個圖片,只需要從(left, top)像素開始,遍歷到(left+radius, top+radius)區(qū)域即可,也就是遍歷正方形區(qū)域(left、top和radius都是我通過ps獲取的)。
我們先將背景圖片拷貝一份,然后在副本上進行粘貼。然后遍歷粘貼后的圖片,如果像素值為透明,我們就將原圖該位置的像素替換至副本處,實現(xiàn)原理和上面一樣,代碼如下:
def generate_music_post(circle_im, bg_im):
"""
傳入圓形圖片和背景圖片生成音樂海報
circle_im:圓形圖片
bg_im:背景圖片
return:生成的圖片
"""
# 拷貝副本
bg_copy = bg_im.copy()
# 將圓形圖片粘貼到副本上
bg_copy.paste(circle_im, (left, top))
# 遍歷像素正方形區(qū)域
for i in range(left, left+radius):
for j in range(top, top+radius):
# 獲取像素值
color = bg_copy.getpixel((i, j))
# 如果像素透明。color的值為(r,g,b,a),color[3]為a的值,即透明值
if color[3] == 0:
# 將原圖像素替換至副本透明處
bg_copy.putpixel((i, j), bg_im.getpixel((i, j)))
# 返回合成后的圖片
return bg_copy
這樣就完成了。完整代碼如下:
from PIL import Image, ImageDraw
left = 273 # 圓離左邊的距離
top = 573 # 圓離上邊的距離
radius = 533 # 圓的直徑
circle_size = (radius, radius) # 圓的外接正方形的大小
def generate_circle_image(img_path):
# 創(chuàng)建一個透明的正方形
im = Image.new('RGBA', circle_size, (255, 255, 255, 0))
# 獲取繪畫者
drawer = ImageDraw.Draw(im)
# 在透明的正方形上畫一個黃色的圓
drawer.ellipse((0, 0, circle_size[0], circle_size[1]), fill=(255, 255, 0), width=0)
# 打開要轉換成圓形的圖片,我們事先把圖片裁剪好
pic = Image.open(img_path).convert('RGBA')
re_pic = pic.resize(circle_size, Image.ANTIALIAS)
# 遍歷圖片的每個像素
for i in range(circle_size[0]):
for j in range(circle_size[1]):
r, g, b, a = im.getpixel((i, j))
if (r, g, b) != (255, 255, 0):
re_pic.putpixel((i, j), (255, 255, 255, 0))
return re_pic
def generate_music_post(circle_im, bg_im):
"""
傳入圓形圖片和背景圖片生成音樂海報
circle_im:圓形圖片
bg_im:背景圖片
return:生成的圖片
"""
# 拷貝副本
bg_copy = bg_im.copy()
# 將圓形圖片粘貼到副本上
bg_copy.paste(circle_im, (left, top))
# 遍歷像素正方形區(qū)域
for i in range(left, left+radius):
for j in range(top, top+radius):
# 獲取像素值
color = bg_copy.getpixel((i, j))
# 如果像素透明。color的值為(r,g,b,a),color[3]為a的值,即透明值
if color[3] == 0:
# 將原圖像素替換至副本透明處
bg_copy.putpixel((i, j), bg_im.getpixel((i, j)))
# 返回合成后的圖片
return bg_copy
# 生成圓形圖片
pic = generate_circle_image('girl.jpeg')
# 以RGBA模式讀取背景圖片
bg_im = Image.open('music.jpg').convert('RGBA')
# 生成音樂海報
music_post = generate_music_post(pic, bg_im)
music_post.show()
另外,這個例子還可以更加智能。我們可以使用OpenCV識別主體圖片的人臉,然后根據(jù)人臉區(qū)域計算一個比較適合的正方形區(qū)域,這樣我們就不必傳入正方形(不過還要考慮人臉識別的精確度等問題)。
代碼已上傳GitHub地址如下: https://github.com/IronSpiderMan/MusicPost
素材圖片也在里面。
以上就是用Python制作音樂海報的詳細內容,更多關于python 制作音樂海報的資料請關注腳本之家其它相關文章!
您可能感興趣的文章:- python使用PIL給圖片添加文字生成海報示例
- 如何使用Python進行PDF圖片識別OCR
- python爬蟲爬取圖片的簡單代碼
- python制作微博圖片爬取工具
- python郵件中附加文字、html、圖片、附件實現(xiàn)方法
- python 使用xlsxwriter循環(huán)向excel中插入數(shù)據(jù)和圖片的操作
- Python基于mediainfo批量重命名圖片文件
- python 調整圖片亮度的示例
- python 實現(xiàn)圖片修復(可用于去水印)
- python實現(xiàn)圖片,視頻人臉識別(dlib版)
- python實現(xiàn)圖片,視頻人臉識別(opencv版)