用Python制作音樂(lè)海報(bào)
前段時(shí)間在一個(gè)朋友那么得到的靈感,想到可以用音樂(lè)播放頁(yè)面作為一張海報(bào)圖片。其實(shí)接下來(lái)要講的和海報(bào)還是有差距的,而具體實(shí)現(xiàn)也只是簡(jiǎn)單的圖片粘貼,但是在效果上還是不錯(cuò)的。效果圖如下,希望大家喜歡:
左邊是原圖,右邊是需要添加到中間的圖,也是圖的主角。其實(shí)如果直接用ps實(shí)現(xiàn)上面的圖是非常簡(jiǎn)單的,反倒是用代碼實(shí)現(xiàn)有點(diǎn)曲折,不過(guò)實(shí)現(xiàn)過(guò)程還是非常有趣的,希望這篇博客可以可以讓你學(xué)到知識(shí)。
用Pillow創(chuàng)建圓形圖在上面的圖片中,中間是一個(gè)圓形圖片,而Pillow本身是沒(méi)有提供生成圓形圖片的方法(也可能是我沒(méi)找到),所以就需要自己實(shí)現(xiàn)。在實(shí)現(xiàn)之前,我們先安裝Pillow模塊:
pip install pillow
要?jiǎng)?chuàng)建圓形圖,我們先根據(jù)原圖的大小,創(chuàng)建一個(gè)RGBA模式的透明圖:
# 該方法傳入三個(gè)參數(shù),第一個(gè)為模式,第二個(gè)為大小的元組,第三個(gè)為顏色im = Image.new(’RGBA’, (300, 300), (255, 255, 255, 0))
上述代碼是創(chuàng)建了一個(gè)完全透明的300*300的圖片,我們?cè)谠搱D片上繪制一個(gè)最大的圓:
# 獲取繪制者drawer = ImageDraw.Draw(im)# 繪制一個(gè)黃色的圓,ellipse方法傳入三個(gè)參數(shù),第一個(gè)為包含該圓的最小正方形的區(qū)域,第二個(gè)為顏色,第三個(gè)為邊寬drawer.ellipse((0, 0, 300, 300), fill=(255, 255, 0), width=0)
生成圖片如下:
我們準(zhǔn)備一張300*300的正方形圖片,然后遍歷圖片的每個(gè)像素,如果像素值的A==0(即像素不透明)那我們就將圖片該區(qū)域的像素值設(shè)置為透明。
代碼如下:
# 打開要轉(zhuǎn)換成圓形的圖片,我們事先把圖片裁剪好pic = Image.open(img_path).convert(’RGBA’)# 遍歷圖片的每個(gè)像素for i in range(300): for j in range(300): # 獲取該像素點(diǎn)的像素 r, g, b, a = im.getpixel((i, j)) # 當(dāng)rgb值不是黃色時(shí),即像素值為透明時(shí) if (r, g, b) != (255, 255, 0): # 將原圖的像素值設(shè)置為透明 pic.putpixel((i, j), (255, 255, 255, 0))
我們的pic就是圓形圖片了,完整代碼如下:
# 背景圖中圓的直徑radius = 533# 圖片的大小circle_size = (radius, radius)def generate_circle_image(img_path): # 創(chuàng)建一個(gè)透明的正方形 im = Image.new(’RGBA’, circle_size, (255, 255, 255, 0)) # 獲取繪畫者 drawer = ImageDraw.Draw(im) # 在透明的正方形上畫一個(gè)黃色的圓 drawer.ellipse((0, 0, circle_size[0], circle_size[1]), fill=(255, 255, 0), width=0) # 打開要轉(zhuǎn)換成圓形的圖片,我們事先把圖片裁剪好 pic = Image.open(img_path).convert(’RGBA’) # 修改圖片大小,讓圖片和圓大小一樣 re_pic = pic.resize(circle_size, Image.ANTIALIAS) # 遍歷圖片的每個(gè)像素 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
在上面的方法中,我們改進(jìn)了一些代碼,之前我們需要使用指定大小的圖片作為素材,現(xiàn)在只需要是正方形圖片就可以了。
生成海報(bào)原本我以為直接將圓形圖片粘貼到背景圖片上就可以了,但是試過(guò)之后發(fā)現(xiàn)效果和我想的不太一樣,反正就是沒(méi)成功,效果如下:
是粘貼上去了沒(méi)錯(cuò),而且圖片也是透明效果,但是從這個(gè)效果來(lái)看粘貼只是像素替換,而不是圖片疊加。于是我又想到了遍歷像素的辦法。
我通過(guò)特殊手段獲取了上面正方形離左邊的像素,和離上邊的像素(其實(shí)就是用ps看了一下)。我嘗試過(guò)圖形檢測(cè)和像素判斷的辦法,想自動(dòng)識(shí)別中間圓的位置,但是效果不佳(反正就是失敗了),所以只能無(wú)恥的用ps查看像素。
回到正題,我們用遍歷像素的辦法不需要遍歷整個(gè)圖片,只需要從(left, top)像素開始,遍歷到(left+radius, top+radius)區(qū)域即可,也就是遍歷正方形區(qū)域(left、top和radius都是我通過(guò)ps獲取的)。
我們先將背景圖片拷貝一份,然后在副本上進(jìn)行粘貼。然后遍歷粘貼后的圖片,如果像素值為透明,我們就將原圖該位置的像素替換至副本處,實(shí)現(xiàn)原理和上面一樣,代碼如下:
def generate_music_post(circle_im, bg_im): ''' 傳入圓形圖片和背景圖片生成音樂(lè)海報(bào) 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, ImageDrawleft = 273 # 圓離左邊的距離top = 573 # 圓離上邊的距離radius = 533 # 圓的直徑circle_size = (radius, radius) # 圓的外接正方形的大小def generate_circle_image(img_path): # 創(chuàng)建一個(gè)透明的正方形 im = Image.new(’RGBA’, circle_size, (255, 255, 255, 0)) # 獲取繪畫者 drawer = ImageDraw.Draw(im) # 在透明的正方形上畫一個(gè)黃色的圓 drawer.ellipse((0, 0, circle_size[0], circle_size[1]), fill=(255, 255, 0), width=0) # 打開要轉(zhuǎn)換成圓形的圖片,我們事先把圖片裁剪好 pic = Image.open(img_path).convert(’RGBA’) re_pic = pic.resize(circle_size, Image.ANTIALIAS) # 遍歷圖片的每個(gè)像素 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_picdef generate_music_post(circle_im, bg_im): ''' 傳入圓形圖片和背景圖片生成音樂(lè)海報(bào) 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’)# 生成音樂(lè)海報(bào)music_post = generate_music_post(pic, bg_im)music_post.show()
另外,這個(gè)例子還可以更加智能。我們可以使用OpenCV識(shí)別主體圖片的人臉,然后根據(jù)人臉區(qū)域計(jì)算一個(gè)比較適合的正方形區(qū)域,這樣我們就不必傳入正方形(不過(guò)還要考慮人臉識(shí)別的精確度等問(wèn)題)。
代碼已上傳GitHub地址如下: https://github.com/IronSpiderMan/MusicPost
素材圖片也在里面。
以上就是用Python制作音樂(lè)海報(bào)的詳細(xì)內(nèi)容,更多關(guān)于python 制作音樂(lè)海報(bào)的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. PHP循環(huán)與分支知識(shí)點(diǎn)梳理2. 無(wú)線標(biāo)記語(yǔ)言(WML)基礎(chǔ)之WMLScript 基礎(chǔ)第1/2頁(yè)3. 前端從瀏覽器的渲染到性能優(yōu)化4. ASP基礎(chǔ)入門第三篇(ASP腳本基礎(chǔ))5. 利用CSS3新特性創(chuàng)建透明邊框三角6. css代碼優(yōu)化的12個(gè)技巧7. 解析原生JS getComputedStyle8. 讀大數(shù)據(jù)量的XML文件的讀取問(wèn)題9. ASP實(shí)現(xiàn)加法驗(yàn)證碼10. ASP刪除img標(biāo)簽的style屬性只保留src的正則函數(shù)
