main2.py 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. import openshot
  2. import re
  3. from PIL import Image,ImageDraw,ImageFont
  4. import pandas as pd
  5. import os
  6. import cv2
  7. import numpy as np
  8. # import moviepy.editor as mp
  9. import time
  10. import pysrt
  11. import shutil
  12. import rpyc
  13. import random
  14. import string
  15. import requests
  16. from bs4 import BeautifulSoup
  17. import zipfile
  18. import csv
  19. from datetime import datetime
  20. def cKey(r,g,b,fuzz):
  21. col=openshot.Color()
  22. col.red=openshot.Keyframe(r)
  23. col.green=openshot.Keyframe(g)
  24. col.blue=openshot.Keyframe(b)
  25. return openshot.ChromaKey(col, openshot.Keyframe(fuzz))
  26. def video_writer_init(path):
  27. w = openshot.FFmpegWriter(path)
  28. w.SetAudioOptions(True, "aac", 44100, 2, openshot.LAYOUT_STEREO, 3000000)
  29. w.SetVideoOptions(True, "libx264", openshot.Fraction(30000, 1000), 1280, 720,
  30. openshot.Fraction(1, 1), False, False, 3000000)
  31. return w
  32. def video_photo_clip(video=None,layer=None, position=None, end=None
  33. ,scale_x=1,scale_y=1,location_x=0,location_y=0,ck=None,audio=True):
  34. clip = openshot.Clip(video)
  35. clip.Layer(layer)
  36. clip.Position(position)
  37. clip.End(end)
  38. clip.scale_x=openshot.Keyframe(scale_x)
  39. clip.scale_y=openshot.Keyframe(scale_y)
  40. clip.location_x=openshot.Keyframe(location_x)
  41. clip.location_y=openshot.Keyframe(location_y)
  42. if ck!=None:
  43. clip.AddEffect(ck)
  44. if audio==True:
  45. clip.has_audio=openshot.Keyframe(1)
  46. else:
  47. clip.has_audio=openshot.Keyframe(0)
  48. return clip
  49. def trim_punctuation(s):
  50. pat_block = u'[^\u4e00-\u9fff0-9a-zA-Z]+'
  51. pattern = u'([0-9]+{0}[0-9]+)|{0}'.format(pat_block)
  52. res = re.sub(pattern, lambda x: x.group(1) if x.group(1) else u" " ,s)
  53. return res
  54. def randomString(stringLength=10):
  55. letters = string.ascii_lowercase
  56. return ''.join(random.choice(letters) for i in range(stringLength))
  57. #文字轉圖片
  58. def txt2image(content, save_target,lang='zh',size=26,fon="input_self/font/DFT_B7.ttc"):
  59. unicode_text = trim_punctuation(content)
  60. font = ''
  61. if lang=='zh':
  62. font = ImageFont.truetype(font=fon, size=size)
  63. else :
  64. font = ImageFont.truetype(font="input_self/font/arial.ttf", size=size)
  65. W, H = (1280,500)
  66. canvas = Image.new('RGB', (W, H), "#00FF00")
  67. draw = ImageDraw.Draw(canvas)
  68. text= content
  69. if "\n" in text:
  70. w, h = draw.textsize(text.split("\n")[0],font = font)
  71. #draw.text(((W-w)/2,0), text[0:18],'black', font)
  72. text_border(draw,(W-w)/2,0,text.split("\n")[0],font,'black','white')
  73. w, h = draw.textsize(text.split("\n")[1],font = font)
  74. #draw.text(((W-w)/2,h+2), text[18:],'black', font)
  75. text_border(draw,(W-w)/2,h+2,text.split("\n")[1],font,'black','white')
  76. else:
  77. w, h = draw.textsize(content,font = font)
  78. #draw.text(((W-w)/2,0), text,'black', font)
  79. text_border(draw,(W-w)/2,0,text,font,'black','white')
  80. canvas.save(save_target, "PNG")
  81. def text_border(draw,x,y,text,font,shadowcolor,fillcolor):
  82. draw.text((x-1, y), text, font=font, fill=shadowcolor)
  83. draw.text((x+1, y), text, font=font, fill=shadowcolor)
  84. draw.text((x, y-1), text, font=font, fill=shadowcolor)
  85. draw.text((x, y+1), text, font=font, fill=shadowcolor)
  86. draw.text((x-1, y+1), text, font=font, fill=shadowcolor)
  87. draw.text((x+1, y-1), text, font=font, fill=shadowcolor)
  88. draw.text((x-1, y-1), text, font=font, fill=shadowcolor)
  89. draw.text((x+1, y+1), text, font=font, fill=shadowcolor)
  90. # thicker border
  91. draw.text((x-2, y-2), text, font=font, fill=shadowcolor)
  92. draw.text((x+2, y-2), text, font=font, fill=shadowcolor)
  93. draw.text((x-2, y+2), text, font=font, fill=shadowcolor)
  94. draw.text((x+2, y+2), text, font=font, fill=shadowcolor)
  95. # now draw the text over it
  96. draw.text((x, y), text, font=font, fill=fillcolor)
  97. def srt_to_csv(srt_file):
  98. subs = pysrt.open(srt_file)
  99. csv_file = srt_file.split('.')[0] + ".csv"
  100. with open(csv_file, 'w', newline='',encoding="big5") as csvfile:
  101. # 建立 CSV 檔寫入器
  102. writer = csv.writer(csvfile)
  103. j=0
  104. for context in subs:
  105. writer.writerow([context.index, context.start,context.end, context.text])
  106. j+=1
  107. return csv_file
  108. def csv_to_text(csv_file,text_font):
  109. text_form = []
  110. with open(csv_file, newline='',encoding="big5") as csvfile:
  111. # 讀取 CSV 檔案內容
  112. rows = csv.reader(csvfile)
  113. j=0
  114. # 以迴圈輸出每一列
  115. for row in rows:
  116. if j==0:
  117. j = 1
  118. continue
  119. start = datetime.strptime(row[1], "%H:%M:%S,%f")
  120. end = datetime.strptime(row[2], "%H:%M:%S,%f") - datetime.strptime(row[1], "%H:%M:%S,%f")
  121. end_timeStamp=end.seconds+0.000001*end.microseconds
  122. start_timeStamp=start.minute*60+start.second+ 0.000001*start.microsecond
  123. text_form.append({'text':row[3],'start':start_timeStamp,'end':end_timeStamp,'size':36,'font':text_font})
  124. return text_form
  125. def text_to_short_vedio(mp4_file ,sound_file,output_filename,text_font):
  126. t = openshot.Timeline(1280, 720, openshot.Fraction(30000, 1000), 44100, 2, openshot.LAYOUT_STEREO)
  127. t.Open()
  128. # 去背參數
  129. ck = cKey(0, 254, 0, 270)
  130. ck_anchor = cKey(0, 255, 0, 320)
  131. anchor = openshot.FFmpegReader(mp4_file)
  132. anchor.Open()
  133. anchor_clip = video_photo_clip(video=anchor,layer=2,scale_x=1,scale_y=1,
  134. location_x=0,location_y=0,position=0, end=anchor.info.duration,audio=True)
  135. t.AddClip(anchor_clip)
  136. anchor.Close()
  137. number = 0
  138. # sound_srt_file = ""
  139. # #音檔自動產生srt(逐字稿)
  140. # if ".srt" in sound_file:
  141. # sound_srt_file = sound_file
  142. # elif not sound_file is None:
  143. # cmd = "autosub -S zh-TW -D zh-TW " + sound_file
  144. # os.system(cmd)
  145. # sound_srt_file = sound_file.split('.')[0] + ".srt"
  146. # csv_file = srt_to_csv(sound_srt_file)
  147. csv_file = "output/csv_produce/test.csv"
  148. text_form = csv_to_text(csv_file,text_font)
  149. # print(sound_srt_file)
  150. #開啟srt檔
  151. try:
  152. number = 0
  153. for text_tmp in text_form:
  154. file_name = "input_self/tmp/save_target_" + str(number) + ".png"
  155. txt2image(text_tmp['text'], file_name,lang='zh',size = text_tmp['size'],fon = text_tmp['font'])
  156. exec('text_anchor_{} = openshot.QtImageReader("input_self/tmp/save_target_{}.png")'.format(number,number))
  157. exec('text_anchor_{}.Open()'.format(number))
  158. exec('text_anchor_{}.Open()'.format(number))
  159. exec('text_anchor_clip_{} = video_photo_clip(video=text_anchor_{},layer=4,scale_x=1,scale_y=1,\
  160. location_x=0,location_y=0.78,position=text_tmp["start"], end=text_tmp["end"],ck=ck_anchor,audio=True)'.format(number,number))
  161. exec('t.AddClip(text_anchor_clip_{})'.format(number))
  162. exec('text_anchor_{}.Close()'.format(number))
  163. number = number+1
  164. except:
  165. print("無法開啟srt檔案(字幕產生失敗)")
  166. w = video_writer_init(output_filename)
  167. w.Open()
  168. frames = int(t.info.fps)*int(anchor.info.duration)
  169. for n in range(frames):
  170. f=t.GetFrame(n)
  171. w.WriteFrame(f)
  172. t.Close()
  173. w.Close()
  174. # 刪除暫存檔案
  175. shutil.rmtree('input_self/tmp')
  176. os.mkdir('input_self/tmp')
  177. # shutil.rmtree('input_self/tmp1')
  178. # os.mkdir('input_self/tmp1')
  179. if __name__ == '__main__':
  180. # text_to_short_vedio_create()
  181. text_to_short_vedio(mp4_file = "output/no_captions/test.mp4",sound_file ='output/no_captions/test.mp4',
  182. output_filename="output/finally_output/demo.mp4",text_font ="input_self/font/DFT_R7.ttc")