用Python制作音樂海報

前言

前段時間在一個朋友那麼得到的靈感,想到可以用音樂播放頁面作為一張海報圖片。其實接下來要講的和海報還是有差距的,而具體實現也隻是簡單的圖片粘貼,但是在效果上還是不錯的。效果圖如下,希望大傢喜歡:

左邊是原圖,右邊是需要添加到中間的圖,也是圖的主角。其實如果直接用ps實現上面的圖是非常簡單的,反倒是用代碼實現有點曲折,不過實現過程還是非常有趣的,希望這篇博客可以可以讓你學到知識。

用Pillow創建圓形圖

在上面的圖片中,中間是一個圓形圖片,而Pillow本身是沒有提供生成圓形圖片的方法(也可能是我沒找到),所以就需要自己實現。在實現之前,我們先安裝Pillow模塊:

pip install pillow

要創建圓形圖,我們先根據原圖的大小,創建一個RGBA模式的透明圖:

# 該方法傳入三個參數,第一個為模式,第二個為大小的元組,第三個為顏色
im = Image.new('RGBA', (300, 300), (255, 255, 255, 0))

上述代碼是創建瞭一個完全透明的300*300的圖片,我們在該圖片上繪制一個最大的圓:

# 獲取繪制者
drawer = ImageDraw.Draw(im)
# 繪制一個黃色的圓,ellipse方法傳入三個參數,第一個為包含該圓的最小正方形的區域,第二個為顏色,第三個為邊寬
drawer.ellipse((0, 0, 300, 300), fill=(255, 255, 0), width=0)

生成圖片如下:

我們準備一張300*300的正方形圖片,然後遍歷圖片的每個像素,如果像素值的A==0(即像素不透明)那我們就將圖片該區域的像素值設置為透明。

代碼如下:

# 打開要轉換成圓形的圖片,我們事先把圖片裁剪好
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):
  # 創建一個透明的正方形
  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

在上面的方法中,我們改進瞭一些代碼,之前我們需要使用指定大小的圖片作為素材,現在隻需要是正方形圖片就可以瞭。

生成海報

原本我以為直接將圓形圖片粘貼到背景圖片上就可以瞭,但是試過之後發現效果和我想的不太一樣,反正就是沒成功,效果如下:

是粘貼上去瞭沒錯,而且圖片也是透明效果,但是從這個效果來看粘貼隻是像素替換,而不是圖片疊加。於是我又想到瞭遍歷像素的辦法。

我通過特殊手段獲取瞭上面正方形離左邊的像素,和離上邊的像素(其實就是用ps看瞭一下)。我嘗試過圖形檢測和像素判斷的辦法,想自動識別中間圓的位置,但是效果不佳(反正就是失敗瞭),所以隻能無恥的用ps查看像素。

回到正題,我們用遍歷像素的辦法不需要遍歷整個圖片,隻需要從(left, top)像素開始,遍歷到(left+radius, top+radius)區域即可,也就是遍歷正方形區域(left、top和radius都是我通過ps獲取的)。

我們先將背景圖片拷貝一份,然後在副本上進行粘貼。然後遍歷粘貼後的圖片,如果像素值為透明,我們就將原圖該位置的像素替換至副本處,實現原理和上面一樣,代碼如下:

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))
  # 遍歷像素正方形區域
  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):
  # 創建一個透明的正方形
  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))
  # 遍歷像素正方形區域
  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識別主體圖片的人臉,然後根據人臉區域計算一個比較適合的正方形區域,這樣我們就不必傳入正方形(不過還要考慮人臉識別的精確度等問題)。

代碼已上傳GitHub地址如下: https://github.com/IronSpiderMan/MusicPost

素材圖片也在裡面。

以上就是用Python制作音樂海報的詳細內容,更多關於python 制作音樂海報的資料請關註WalkonNet其它相關文章!

推薦閱讀: