影片中加入文字
這篇文章會介紹使用 Python 的 moviepy 第三方函式庫讀取影片,搭配 Pillow 函式庫在影片中加入中文與英文字,最後還會將加入文字的影片轉換成 gif 動畫,做出有趣的梗圖效果 ( 因為 moviepy 內建方法 TextClip 還要額外安裝 ImageMagick,所以使用 Pillow 加入文字 )。
快速導覽:
本篇使用的 Python 版本為 3.7.12,所有範例可使用 Google Colab 實作,不用安裝任何軟體 ( 參考:使用 Google Colab )
安裝 moviepy 和 Pillow
輸入下列指令安裝 moviepy,根據個人環境使用 pip 或 pip3。
!pip install moviepy
輸入下列指令安裝 Pillow,根據個人環境使用 pip 或 pip3,如果使用 Colab 或 Anaconda Jupyter,已經內建 Pillow 函式庫。
!pip install Pillow
由於影片轉檔會使用 ffmpeg,因此也要安裝 ffmpeg ( 影片存檔常見錯誤「TypeError: must be real number, not NoneType」往往都是 ffmpeg 沒有安裝導致 ),根據個人環境使用 pip 或 pip3,Anaconda Jupyter 可以使用 conda install。
!pip install ffmpeg
使用 Pillow 產生文字字卡
載入 PIL 的 Image、ImageFont 和 ImageDraw 模組,按照下列步驟建立背景透明的文字字卡:
- 使用 Image.new 建立色彩模式為 RGBA,尺寸 360x180 的空白圖片。
- 使用 ImageFont.truetype 設定文字的字型和尺寸 ( 字型使用 Google Font 的 NotoSansTC-Regular )。
- 使用 ImageDraw.Draw 建立繪圖物件。
- 使用 text 方法在圖片中寫入文字 ( 參數說明:ImageDraw.text )。
import os
os.chdir('/content/drive/MyDrive/Colab Notebooks') # 使用 Colab 要換路徑使用
from PIL import Image, ImageFont, ImageDraw
img = Image.new('RGBA', (360, 180)) # 建立色彩模式為 RGBA,尺寸 360x180 的空白圖片
font = ImageFont.truetype('NotoSansTC-Regular.otf', 40) # 設定字型與尺寸
draw = ImageDraw.Draw(img) # 準備在圖片上繪圖
# 將文字畫入圖片
draw.text((10,120),'OXXO.STUDIO',fill=(255,255,255),font=font,stroke_width=2,stroke_fill='red')
draw.text(xy=(50,0), text='大家好\n哈哈', align='center', fill=(255,255,255), font=font, stroke_width=2, stroke_fill='blue')
img.save('ok.png') # 儲存為 png
合成影片與文字
使用 moviepy 讀取影片後,按照下列步驟將文字加入影片中:
- 使用 resize 將影片尺寸調整為字卡的大小 ( 也可以在產生字卡圖片時,做成和影片同樣大小,就不需這個步驟 )。
- 使用 subclip 剪輯出兩秒的長度 ( 看個人需求,範例使用兩秒 )。
- 使用 ImageClip 和 set_duration 將靜態圖片建立為長度兩秒的影片物件 ( transparent=True 設定背景透明 )。
- 使用 CompositeVideoClip 將兩段影片混合。
import os
os.chdir('/content/drive/MyDrive/Colab Notebooks') # 使用 Colab 要換路徑使用
from moviepy.editor import *
from PIL import Image, ImageFont, ImageDraw
img = Image.new('RGBA', (360, 180))
font = ImageFont.truetype('NotoSansTC-Regular.otf', 40)
draw = ImageDraw.Draw(img)
draw.text((10,120), 'OXXO.STUDIO', fill=(255,255,255), font=font, stroke_width=2, stroke_fill='red')
draw.text(xy=(50,0), text='大家好\n哈哈', align='center', fill=(255,255,255), font=font, stroke_width=2, stroke_fill='blue')
img.save('ok.png')
video = VideoFileClip("baby_shark.mp4") # 讀取影片
clip = video.resize((360,180)).subclip(10,12) # 縮小影片尺寸,剪輯出 10~12 秒的片段
text_clip = ImageClip("ok.png", transparent=True).set_duration(2) # 讀取圖片,將圖片變成長度兩秒的影片
output = CompositeVideoClip([clip, text_clip]) # 混合影片
output.write_videofile("output.mp4",temp_audiofile="temp-audio.m4a", remove_temp=True, codec="libx264", audio_codec="aac")
print('ok')
影片中加入文字,轉換成 gif 動畫
了解影片加入文字的原理後,參考「影片轉換為 git 動畫」、「剪輯影片」文章,就能將一段影片分割成多段影片,分別加入對應的文字,製做出有趣的梗圖 gif 動畫 ( 詳細說明寫在程式碼內,影片來源:無間道梁朝偉超神演技 )。
import os
os.chdir('/content/drive/MyDrive/Colab Notebooks') # 使用 Colab 要換路徑使用
from moviepy.editor import *
from PIL import Image, ImageFont, ImageDraw
img_empty = Image.new('RGBA', (360, 180)) # 產生 RGBA 空圖片
font = ImageFont.truetype('NotoSansTC-Regular.otf', 24) # 設定文字字體和大小
video = VideoFileClip("oxxostudio.mp4").resize((360,180)) # 讀取影片,改變尺寸
output_list = [] # 記錄最後要組合的影片片段
# 建立文字字卡函式
def text_clip(xy, text, name):
img = img_empty.copy() # 複製空圖片
draw = ImageDraw.Draw(img) # 建立繪圖物件,並寫入文字
draw.text(xy, text, fill=(255,255,255), font=font, stroke_width=2, stroke_fill='black')
img.save(name) # 儲存
# 建立影片和文字合併的函式
def text_in_video(t, text_img):
clip = video.subclip(t[0],t[1]) # 剪輯影片到指定長度
text = ImageClip(text_img, transparent=True).set_duration(t[1]-t[0]) # 讀取字卡,調整為影片長度
combine_clip = CompositeVideoClip([clip, text]) # 合併影片和文字
output_list.append(combine_clip) # 添加到影片片段裡
# 文字串列,包含座標和內容
text_list = [
[(100,140),'你到底要怎樣?'],
[(90,140),'給我 CDPRO2 呀!'],
[(60,140),'但是 CDPRO2 過時啦!']
]
# 影片串列,包含要切取的時間片段
video_list = [
[13,16],
[21,24],
[38,41]
]
# 使用 for 迴圈,產生文字字卡
for i in range(len(text_list)):
text_clip(text_list[i][0], text_list[i][1], f'text_{i}.png')
# 使用 for 迴圈,合併字卡和影片
for i in range(len(video_list)):
text_in_video(video_list[i], f'text_{i}.png')
output = concatenate_videoclips(output_list) # 合併所有影片片段
output.write_gif("output.gif", fps=6, colors=32) # 轉換成 gif 動畫
print('ok')
意見回饋
如果有任何建議或問題,可傳送「意見表單」給我,謝謝~