main1.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543
  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. import rarfile
  21. def cKey(r,g,b,fuzz):
  22. col=openshot.Color()
  23. col.red=openshot.Keyframe(r)
  24. col.green=openshot.Keyframe(g)
  25. col.blue=openshot.Keyframe(b)
  26. return openshot.ChromaKey(col, openshot.Keyframe(fuzz))
  27. def video_writer_init(path):
  28. w = openshot.FFmpegWriter(path)
  29. w.SetAudioOptions(True, "aac", 44100, 2, openshot.LAYOUT_STEREO, 3000000)
  30. w.SetVideoOptions(True, "libx264", openshot.Fraction(30000, 1000), 1280, 720,
  31. openshot.Fraction(1, 1), False, False, 3000000)
  32. return w
  33. def video_photo_clip(video=None,layer=None, position=None, end=None
  34. ,scale_x=1,scale_y=1,location_x=0,location_y=0,ck=None,audio=True):
  35. clip = openshot.Clip(video)
  36. clip.Layer(layer)
  37. clip.Position(position)
  38. clip.End(end)
  39. clip.scale_x=openshot.Keyframe(scale_x)
  40. clip.scale_y=openshot.Keyframe(scale_y)
  41. clip.location_x=openshot.Keyframe(location_x)
  42. clip.location_y=openshot.Keyframe(location_y)
  43. if ck!=None:
  44. clip.AddEffect(ck)
  45. if audio==True:
  46. clip.has_audio=openshot.Keyframe(1)
  47. else:
  48. clip.has_audio=openshot.Keyframe(0)
  49. return clip
  50. def trim_punctuation(s):
  51. pat_block = u'[^\u4e00-\u9fff0-9a-zA-Z]+'
  52. pattern = u'([0-9]+{0}[0-9]+)|{0}'.format(pat_block)
  53. res = re.sub(pattern, lambda x: x.group(1) if x.group(1) else u" " ,s)
  54. return res
  55. def randomString(stringLength=10):
  56. letters = string.ascii_lowercase
  57. return ''.join(random.choice(letters) for i in range(stringLength))
  58. def mp3_to_anchor(fname):
  59. conn = rpyc.classic.connect("192.168.192.221",18812)
  60. fr=open(fname,'rb')
  61. ropen = conn.builtins.open
  62. randname=randomString(10)
  63. finalname=randomString(10)
  64. fw=ropen('/tmp/'+randname+'.mp4','wb')
  65. fw.write(fr.read())
  66. fw.close()
  67. ros = conn.modules.os
  68. ros.system('/root/to_video/p9.sh '+randname+".mp4 "+finalname+".mp4")
  69. return 'http://192.168.192.221/video/'+finalname+'.mp4'
  70. # conn.execute('import os')
  71. def download_mp4(url):
  72. headers = {'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36'}
  73. with open('input_self/AI_girl/ai_spokesgirl.mp4','wb') as f:
  74. r = requests.get(url, headers=headers, stream=True)
  75. if r.status_code == 404:
  76. return False
  77. for chunk in r.iter_content(chunk_size=1024):
  78. if chunk:
  79. f.write(chunk)
  80. return True
  81. def text_to_short_vedio_create(read_csv_use="導盲犬協會.csv",pwd_use="導盲犬協會影片素材2/",op="input_self/LOGO_OP_4.mp4",ed="input_self/LOGO_ED.mp4",bg="input_self/xchoozmo_long.mp4",input_zip="input/input_data.zip"):
  82. files=os.listdir("input/")
  83. print(files)
  84. if len(files)==0:
  85. print("input資料夾沒有檔案,請放入")
  86. return
  87. elif len(files)>1:
  88. print("檔案只能有一個,且為壓縮檔")
  89. return
  90. if files[0][-4:]==".zip":
  91. input_zip="input/"+files[0]
  92. with zipfile.ZipFile(input_zip, 'r') as zf:
  93. for fn in zf.namelist():
  94. right_fn = fn.encode('cp437').decode('big5') # 將檔名正確編碼
  95. check_p=right_fn.split("/")
  96. if right_fn[-1]=="/" :
  97. os.mkdir("input/"+right_fn[:-1])
  98. pwd_use=right_fn
  99. break
  100. if len(check_p)==2:
  101. os.mkdir("input/"+check_p[0])
  102. pwd_use=check_p[0]+"/"
  103. break
  104. read_csv_use = 0
  105. with zipfile.ZipFile(input_zip, 'r') as zf:
  106. for fn in zf.namelist():
  107. right_fn = fn.encode('cp437').decode('big5') # 將檔名正確編碼
  108. check_p=right_fn.split("/")
  109. if right_fn[-1]=="/" :
  110. continue
  111. if right_fn[-4:]==".csv" or right_fn[-5:]==".xlsx":
  112. read_csv_use = right_fn
  113. with open("input/"+right_fn, 'wb') as output_file: # 建立並開啟新檔案
  114. with zf.open(fn, 'r') as origin_file: # 開啟原檔案
  115. shutil.copyfileobj(origin_file, output_file) # 將原檔案內容複製到新檔案
  116. elif files[0][-4:]==".rar":
  117. rar_ = rarfile.RarFile("input/"+files[0])
  118. names = rar_.namelist()
  119. for name in names:
  120. if name[-4:]==".csv" or name[-5:]==".xlsx":
  121. read_csv_use = name
  122. pwd_use=name.split("/")[0]+"/"
  123. rar_.extract(name,"input/")
  124. rar_.close()
  125. else:
  126. print("目前不支援此壓縮格式或不為壓縮檔")
  127. return
  128. # 去背參數
  129. # ck=cKey(0,254,0,270)
  130. ck=cKey(0,255,0,320)
  131. ck_anchor=None
  132. #時間
  133. time_= 0
  134. try:
  135. if read_csv_use[-4:]==".csv":
  136. csv_use=pd.read_csv("input/"+read_csv_use)
  137. elif read_csv_use[-5:]==".xlsx":
  138. csv_use = pd.read_excel("input/"+read_csv_use)
  139. except:
  140. print("excel不存在或excel格式不支援或損毀")
  141. return
  142. csv_use=csv_use.dropna(how='all')
  143. csv_use.reset_index(inplace=True)
  144. # print(csv_use)
  145. t1 = openshot.Timeline(1280, 720, openshot.Fraction(30000, 1000), 44100, 2, openshot.LAYOUT_STEREO)
  146. t1.Open()
  147. frames = 0
  148. for i in range(len(csv_use)):
  149. # for i in range(3):
  150. pwd_=str(csv_use.loc[i,['音檔']].values[0])
  151. locals()['anchor_music_t'+str(i)] = openshot.FFmpegReader("input/"+pwd_use+pwd_)
  152. locals()['anchor_music_t'+str(i)].Open()
  153. locals()['anchor_music_t'+str(i)+'clip'] = video_photo_clip(video=locals()['anchor_music_t'+str(i)],layer=3,scale_x=0,scale_y=0,
  154. location_x=0,location_y=0,position=time_, end=locals()['anchor_music_t'+str(i)].info.duration,ck=ck_anchor,audio=True)
  155. t1.AddClip(locals()['anchor_music_t'+str(i)+'clip'])
  156. locals()['anchor_music_t'+str(i)].Close()
  157. time_+=locals()['anchor_music_t'+str(i)].info.duration
  158. w = video_writer_init("input_self/AI_girl/%s.mp4"%('anchor_music_t'))
  159. w.Open()
  160. frames = int(t1.info.fps)*int(time_)
  161. for n in range(frames):
  162. f=t1.GetFrame(n)
  163. w.WriteFrame(f)
  164. w.Close()
  165. t1.Close()
  166. fname=mp3_to_anchor("input_self/AI_girl/%s.mp4"%('anchor_music_t'))
  167. # print(fname)
  168. time.sleep(300)
  169. while True:
  170. result = download_mp4(fname)
  171. if result:
  172. break
  173. print('等待...')
  174. time.sleep(60)
  175. t = openshot.Timeline(1280, 720, openshot.Fraction(30000, 1000), 44100, 2, openshot.LAYOUT_STEREO)
  176. t.Open()
  177. ck=cKey(0,255,0,320)
  178. ck_anchor=None
  179. #時間
  180. time_= 0
  181. try:
  182. if read_csv_use[-4:]==".csv":
  183. csv_use=pd.read_csv("input/"+read_csv_use)
  184. elif read_csv_use[-5:]==".xlsx":
  185. csv_use = pd.read_excel("input/"+read_csv_use)
  186. except:
  187. print("excel不存在或excel格式不支援或損毀")
  188. return
  189. csv_use=csv_use.dropna(how='all')
  190. csv_use.reset_index(inplace=True)
  191. anchor_op = openshot.FFmpegReader(op)
  192. anchor_op.Open()
  193. anchor_clip_op = video_photo_clip(video=anchor_op,layer=2,scale_x=1,scale_y=1,
  194. location_x=0,location_y=0,position=time_, end=anchor_op.info.duration,ck=ck_anchor,audio=True)
  195. t.AddClip(anchor_clip_op)
  196. anchor_op.Close()
  197. time_+=anchor_op.info.duration
  198. start_time = time_
  199. for i in range(len(csv_use)):
  200. # for i in range(3):
  201. pwd_=str(csv_use.loc[i,['音檔']].values[0])
  202. layer_choose = 4
  203. location_x_choose=-0.04
  204. location_y_choose=-0.04
  205. locals()['anchor_music'+str(i)] = openshot.FFmpegReader("input/"+pwd_use+pwd_)
  206. locals()['anchor_music'+str(i)].Open()
  207. locals()['anchor_music'+str(i)+'clip'] = video_photo_clip(video=locals()['anchor_music'+str(i)],layer=3,scale_x=0,scale_y=0,
  208. location_x=0,location_y=0,position=time_, end=locals()['anchor_music'+str(i)].info.duration,ck=ck_anchor,audio=True)
  209. t.AddClip(locals()['anchor_music'+str(i)+'clip'])
  210. locals()['anchor_music'+str(i)].Close()
  211. if str(csv_use.loc[i,['是否要場景']].values[0])=="是":
  212. scale_x_use = 0.59
  213. scale_y_use = 0.59
  214. else:
  215. scale_x_use = 1
  216. scale_y_use = 1
  217. layer_choose = 8
  218. location_x_choose=0
  219. location_y_choose=0
  220. locals()['anchor_black'+str(i)] = openshot.QtImageReader("input_self/black.jpg")
  221. locals()['anchor_black'+str(i)].Open()
  222. locals()['anchor_black'+str(i)+'clip'] = video_photo_clip(video=locals()['anchor_black'+str(i)],layer=7,scale_x=scale_x_use,scale_y=scale_y_use,
  223. location_x=0,location_y=0,position=time_, end=locals()['anchor_music'+str(i)].info.duration,ck=ck_anchor,audio=False)
  224. t.AddClip(locals()['anchor_black'+str(i)+'clip'])
  225. locals()['anchor_black'+str(i)].Close()
  226. choose=str(csv_use.loc[i,['素材']].values[0]).split(".")[-1]
  227. pwd_p1=str(csv_use.loc[i,['素材']].values[0])
  228. if choose == 'mp4':
  229. locals()['anchor'+str(i)] = openshot.FFmpegReader("input/"+pwd_use+pwd_p1)
  230. locals()['anchor'+str(i)].Open()
  231. locals()['anchor'+str(i)+'clip'] = video_photo_clip(video=locals()['anchor'+str(i)],layer=layer_choose,scale_x=scale_x_use,scale_y=scale_y_use,
  232. location_x=location_x_choose,location_y=location_y_choose,position=time_, end=locals()['anchor_music'+str(i)].info.duration,ck=ck_anchor,audio=False)
  233. t.AddClip(locals()['anchor'+str(i)+'clip'])
  234. locals()['anchor'+str(i)].Close()
  235. elif choose == 'jpg' or choose == 'png':
  236. locals()['anchor'+str(i)] = openshot.QtImageReader("input/"+pwd_use+pwd_p1)
  237. locals()['anchor'+str(i)].Open()
  238. locals()['anchor'+str(i)+'clip'] = video_photo_clip(video=locals()['anchor'+str(i)],layer=layer_choose,scale_x=scale_x_use,scale_y=scale_y_use,
  239. location_x=location_x_choose,location_y=location_y_choose,position=time_, end=locals()['anchor_music'+str(i)].info.duration,ck=ck_anchor,audio=False)
  240. t.AddClip(locals()['anchor'+str(i)+'clip'])
  241. locals()['anchor'+str(i)].Close()
  242. time_+=locals()['anchor_music'+str(i)].info.duration
  243. locals()['anchor_bg'] = openshot.FFmpegReader(bg)
  244. locals()['anchor_bg'].Open()
  245. locals()['anchor_clip_bg'] = video_photo_clip(video=locals()['anchor_bg'],layer=5,scale_x=1,scale_y=1,
  246. location_x=0,location_y=0,position=start_time, end=time_-start_time,ck=ck,audio=False)
  247. t.AddClip(locals()['anchor_clip_bg'])
  248. locals()['anchor_bg'].Close()
  249. locals()['anchor_ad'] = openshot.FFmpegReader('input_self/AI_girl/ai_spokesgirl.mp4')
  250. locals()['anchor_ad'].Open()
  251. locals()['anchor_clip_ad'] = video_photo_clip(video=locals()['anchor_ad'],layer=6,scale_x=0.8,scale_y=0.8,
  252. location_x=0.38,location_y=0.35,position=start_time , end=time_-start_time,ck=ck,audio=False)
  253. t.AddClip(locals()['anchor_clip_ad'])
  254. locals()['anchor_ad'].Close()
  255. anchor_ed = openshot.FFmpegReader(ed)
  256. anchor_ed.Open()
  257. anchor_clip_ed = video_photo_clip(video=anchor_ed,layer=2,scale_x=1,scale_y=1,
  258. location_x=0,location_y=0,position=time_, end=anchor_ed.info.duration,ck=ck_anchor,audio=True)
  259. time_+=anchor_ed.info.duration
  260. t.AddClip(anchor_clip_ed)
  261. anchor_ed.Close()
  262. w = video_writer_init("output/no_captions/test.mp4")
  263. w.Open()
  264. frames = int(t.info.fps)*int(time_)
  265. for n in range(frames):
  266. f=t.GetFrame(n)
  267. w.WriteFrame(f)
  268. t.Close()
  269. w.Close()
  270. shutil.rmtree('input')
  271. os.mkdir('input')
  272. shutil.rmtree('input_self/AI_girl')
  273. os.mkdir('input_self/AI_girl')
  274. return True
  275. #文字轉圖片
  276. def txt2image(content, save_target,lang='zh',size=26,fon="input_self/font/DFT_B7.ttc"):
  277. unicode_text = trim_punctuation(content)
  278. font = ''
  279. if lang=='zh':
  280. font = ImageFont.truetype(font=fon, size=size)
  281. else :
  282. font = ImageFont.truetype(font="input_self/font/arial.ttf", size=size)
  283. W, H = (1280,500)
  284. canvas = Image.new('RGB', (W, H), "#00FF00")
  285. draw = ImageDraw.Draw(canvas)
  286. text= content
  287. if "\n" in text:
  288. w, h = draw.textsize(text.split("\n")[0],font = font)
  289. #draw.text(((W-w)/2,0), text[0:18],'black', font)
  290. text_border(draw,(W-w)/2,0,text.split("\n")[0],font,'black','white')
  291. w, h = draw.textsize(text.split("\n")[1],font = font)
  292. #draw.text(((W-w)/2,h+2), text[18:],'black', font)
  293. text_border(draw,(W-w)/2,h+2,text.split("\n")[1],font,'black','white')
  294. else:
  295. w, h = draw.textsize(content,font = font)
  296. #draw.text(((W-w)/2,0), text,'black', font)
  297. text_border(draw,(W-w)/2,0,text,font,'black','white')
  298. canvas.save(save_target, "PNG")
  299. def text_border(draw,x,y,text,font,shadowcolor,fillcolor):
  300. draw.text((x-1, y), text, font=font, fill=shadowcolor)
  301. draw.text((x+1, y), text, font=font, fill=shadowcolor)
  302. draw.text((x, y-1), text, font=font, fill=shadowcolor)
  303. draw.text((x, y+1), text, font=font, fill=shadowcolor)
  304. draw.text((x-1, y+1), text, font=font, fill=shadowcolor)
  305. draw.text((x+1, y-1), text, font=font, fill=shadowcolor)
  306. draw.text((x-1, y-1), text, font=font, fill=shadowcolor)
  307. draw.text((x+1, y+1), text, font=font, fill=shadowcolor)
  308. # thicker border
  309. draw.text((x-2, y-2), text, font=font, fill=shadowcolor)
  310. draw.text((x+2, y-2), text, font=font, fill=shadowcolor)
  311. draw.text((x-2, y+2), text, font=font, fill=shadowcolor)
  312. draw.text((x+2, y+2), text, font=font, fill=shadowcolor)
  313. # now draw the text over it
  314. draw.text((x, y), text, font=font, fill=fillcolor)
  315. def srt_to_csv(srt_file):
  316. subs = pysrt.open(srt_file)
  317. # csv_file = srt_file.split('.')[0] + ".csv"
  318. with open("output/csv_produce/test.csv", 'w', newline='',encoding="big5") as csvfile:
  319. # 建立 CSV 檔寫入器
  320. writer = csv.writer(csvfile)
  321. writer.writerow(["序號","開始時間","結束時間","字幕內容"])
  322. for context in subs:
  323. writer.writerow([context.index, context.start,context.end, context.text])
  324. return
  325. def csv_to_text(csv_file,text_font):
  326. text_form = []
  327. with open(csv_file, newline='',encoding="big5") as csvfile:
  328. # 讀取 CSV 檔案內容
  329. rows = csv.reader(csvfile)
  330. # 以迴圈輸出每一列
  331. for row in rows:
  332. start = datetime.strptime(row[1], "%H:%M:%S,%f")
  333. end = datetime.strptime(row[2], "%H:%M:%S,%f") - datetime.strptime(row[1], "%H:%M:%S,%f")
  334. end_timeStamp=end.seconds+0.000001*end.microseconds
  335. start_timeStamp=start.minute*60+start.second+ 0.000001*start.microsecond
  336. text_form.append({'text':row[3],'start':start_timeStamp,'end':end_timeStamp,'size':36,'font':text_font})
  337. return text_form
  338. def text_to_short_vedio(mp4_file ,sound_file,output_filename,text_font):
  339. t = openshot.Timeline(1280, 720, openshot.Fraction(30000, 1000), 44100, 2, openshot.LAYOUT_STEREO)
  340. t.Open()
  341. # 去背參數
  342. ck = cKey(0, 254, 0, 270)
  343. ck_anchor = cKey(0, 255, 0, 320)
  344. anchor = openshot.FFmpegReader(mp4_file)
  345. anchor.Open()
  346. anchor_clip = video_photo_clip(video=anchor,layer=2,scale_x=1,scale_y=1,
  347. location_x=0,location_y=0,position=0, end=anchor.info.duration,audio=True)
  348. t.AddClip(anchor_clip)
  349. anchor.Close()
  350. number = 0
  351. sound_srt_file = ""
  352. #音檔自動產生srt(逐字稿)
  353. if ".srt" in sound_file:
  354. sound_srt_file = sound_file
  355. elif not sound_file is None:
  356. cmd = "autosub -S zh-TW -D zh-TW " + sound_file
  357. os.system(cmd)
  358. sound_srt_file = sound_file.split('.')[0] + ".srt"
  359. csv_file = srt_to_csv(sound_srt_file)
  360. text_form = csv_to_text(csv_file,text_font)
  361. print(sound_srt_file)
  362. #開啟srt檔
  363. try:
  364. number = 0
  365. for text_tmp in text_form:
  366. file_name = "input_self/tmp/save_target_" + str(number) + ".png"
  367. txt2image(text_tmp['text'], file_name,lang='zh',size = text_tmp['size'],fon = text_tmp['font'])
  368. exec('text_anchor_{} = openshot.QtImageReader("input_self/tmp/save_target_{}.png")'.format(number,number))
  369. exec('text_anchor_{}.Open()'.format(number))
  370. exec('text_anchor_{}.Open()'.format(number))
  371. exec('text_anchor_clip_{} = video_photo_clip(video=text_anchor_{},layer=4,scale_x=1,scale_y=1,\
  372. location_x=0,location_y=0.78,position=text_tmp["start"], end=text_tmp["end"],ck=ck_anchor,audio=True)'.format(number,number))
  373. exec('t.AddClip(text_anchor_clip_{})'.format(number))
  374. exec('text_anchor_{}.Close()'.format(number))
  375. number = number+1
  376. except:
  377. print("無法開啟srt檔案(字幕產生失敗)")
  378. w = video_writer_init(output_filename)
  379. w.Open()
  380. frames = int(t.info.fps)*int(anchor.info.duration)
  381. for n in range(frames):
  382. f=t.GetFrame(n)
  383. w.WriteFrame(f)
  384. t.Close()
  385. w.Close()
  386. # 刪除暫存檔案
  387. # shutil.rmtree('input_self/tmp')
  388. # os.mkdir('input_self/tmp')
  389. # shutil.rmtree('input_self/tmp1')
  390. # os.mkdir('input_self/tmp1')
  391. def vedio_to_csv(sound_file):
  392. cmd = "autosub -S zh-TW -D zh-TW " + sound_file
  393. os.system(cmd)
  394. sound_srt_file = sound_file.split('.')[0] + ".srt"
  395. srt_to_csv(sound_srt_file)
  396. if __name__ == '__main__':
  397. # down=text_to_short_vedio_create()
  398. # if down:
  399. # vedio_to_csv("output/no_captions/test.mp4")
  400. vedio_to_csv("output/no_captions/test.mp4")