1. 場景
如果你經常刷抖音和微信朋友圈,一定發現了最近九宮格短視頻很火!
從朋友圈九宮格圖片,到九宮格視頻,相比傳統的圖片視頻,前者似乎更有個性和逼格
除了傳統的剪輯軟件可以實現,是否有其他更加快捷方便的方式?比如:一鍵生成,批量生成?
廢話不多說,本篇文章將大家使用 Python 一鍵生成九宮格短視頻,優雅地幫你在朋友圈裝一次逼!
2.准備
在開始實戰之前,使用 pip 安裝 2 個依賴,分別是:
1、視頻處理依賴 moviepy
2、圖片處理依賴 PIL
# 安裝兩個依賴
# 視頻處理
pip3 install moviepy
# 圖片處理依賴
pip3 install Pillow
3.實戰一下
在實戰之前,先准備一段原始視頻素材
下面通過 6 個步驟,將原始視頻轉換為九宮格視頻
1、新建處理文件夾
新建一個臨時文件夾和一個視頻輸出文件夾
def mkdir_folder(file_path):
"""
創建一個文件夾,如果不存在就創建;否則不做處理
:param file_path:
:return:
"""
if os.path.exists(file_path):
return
os.mkdir(file_path)
# 新建臨時文件夾和輸出文件夾
mkdir_folder(self.path_temp)
mkdir_folder(self.path_output)
2、獲取視頻的音頻文件及視頻基礎信息
首先,根據原始視頻,使用 moviepy 構建一個 VideoFileClip 對象,從而獲取到視頻的寬、高、幀率、時長等信息
self.video_raw_clip = VideoFileClip(file_path)
# 寬、高
self.video_width, self.video_height = self.video_raw_clip.w, self.video_raw_clip.h
# 幀率
self.fps = self.video_raw_clip.fps
# 視頻時長
self.during = self.video_raw_clip.duration
接着,從視頻中提取 BGM 音頻對象,並寫入到文件中
def get_audio_from_video(video_raw_clip, output_path):
"""
從視頻中提取音頻
:param video_raw_clip: 視頻Clip對象
:param output_path: 輸出音頻文件完整路徑
:return:
"""
audio = video_raw_clip.audio
audio.write_audiofile(output_path)
return output_path
3、處理視頻幀
我們使用原始視頻 Clip 對象的 iter_frames() 方法,循環獲取所有的視頻幀圖片
需要指出的是,為了保證后面視頻合成的便捷性,這里對視頻幀的文件名按順序進行命令
i = 1
for frame in self.video_raw_clip.iter_frames():
image = Image.fromarray(frame)
# 視頻幀圖片保存的臨時路徑(完整路徑)
frame_file_complete_path = self.path_temp + "%04d.jpg" % i
i += 1
視頻每一幀都被裁剪成 9 張圖片,我們可以顯式指定圖片之間的距離,然后計算出新畫布的寬和高,最后繪制一個白底背景的圖片
# 1、剪成9張圖片,計算每張圖片的寬、高
item_width = int(self.video_width / 3)
item_height = int(self.video_height / 3)
# 2、新的寬、高
item_width_new = self.video_width + self.item_space * 2
item_height_new = self.video_height + self.item_space * 2
# 3、重新建一個畫布背景
new_image = Image.new(image.mode, (item_width_new, item_height_new),
color='white')
接着,獲取每一塊區域的坐標值,針對橫向、縱向第 2、3 個圖片區域加上間隔偏移,粘貼到上面新建的圖片上即可
# 4、裁剪圖片,然后粘貼到新的畫布中去
# i:橫向、j:縱向
for i in range(0, 3):
for j in range(0, 3):
# 裁剪區域
box = (j * item_width, i * item_height, (j + 1) * item_width, (i + 1) * item_height)
# 根據區域,裁剪圖片
crop_image = image.crop(box)
# 橫向、縱向第2塊和第3塊,要加上偏移距離
x = 0 if j == 0 else (item_width + self.item_space) * j
y = 0 if i == 0 else (item_height + self.item_space) * i
# 將9張圖片,按照上面計算的坐標值,粘貼到背景中去
new_image.paste(crop_image, (int(x), int(y)))
# 保存圖片到本地
new_image.save(frame_file_complete_path)
4、一籃子圖片重新合成視頻
把上一步生成的幀圖片,按照原視頻的幀率轉為視頻
需要注意的是,為了保證生成的視頻不會錯亂,最好對幀圖片按照名稱進行一次排序
def pics_to_video(pics_path, output_path, fps):
"""
圖片轉為視頻
pics_to_video('./../gif_temp/', './../video_temp/temp1.mp4', 20)
:param pics_path:
:param output_path:
:return:
"""
image_paths = list(map(lambda x: pics_path + x, os.listdir(pics_path)))
# 注意:這里必須進行一次排序,保證所有幀的順序是一致
image_paths = sort_strings_with_emb_numbers(image_paths)
# 過濾掉非圖片
image_paths = list(filter(lambda image_path: image_path.endswith('.jpg'), image_paths))
# 圖片剪輯類
clip = ImageSequenceClip(image_paths,
fps=fps)
clip.write_videofile(output_path)
5、加入 BGM 背景音樂
將原始視頻的音頻文件設置到上一步生成的視頻文件,然后寫入一個新的文件中去
def video_with_audio(path_video_raw, path_bgm_raw, output):
"""
視頻合成音頻
:return:
"""
videoclip = VideoFileClip(path_video_raw)
audioclip = AudioFileClip(path_bgm_raw)
# 設置視頻音頻,並寫入到文件中去
videoclip.set_audio(audioclip).write_videofile(output,
codec='libx264',
audio_codec='aac',
temp_audiofile='temp-audio.m4a',
remove_temp=True
)
6、刪除臨時文件
利用 shutil將上面視頻處理的臨時文件,包含幀圖片、臨時視頻文件刪除掉
def remove_folder(file_path):
"""
刪除文件夾
:param file_path:
:return:
"""
shutil.rmtree(file_path)
# 刪除臨時文件
remove_folder(self.path_temp)
4.最后
通過上面的一系列操作,我們實現了將一段視頻處理成九宮格視頻
我已經將文中全部源碼,包含:生成九宮格視頻和圖片兩套代碼 上傳到后台,關注公眾號「AirPython 」后回復「九宮格 」即可獲得全部源碼
如果你覺得文章還不錯,請大家 點贊、分享、留言下,因為這將是我持續輸出更多優質文章的最強動力!
推薦閱讀