Files
FABLE/generators/montage.py

183 lines
8.0 KiB
Python
Raw Normal View History

2023-05-15 10:11:04 +02:00
import json
import os
import requests
import pysrt
import random
from generators.speak import VoiceGenerator, voices
2023-05-15 10:11:04 +02:00
from moviepy.video.VideoClip import ImageClip
from moviepy.editor import concatenate_videoclips, CompositeAudioClip, concatenate_audioclips
2023-05-15 10:11:04 +02:00
from moviepy.audio.io.AudioFileClip import AudioFileClip
from moviepy.audio.fx.all import volumex, audio_fadein, audio_fadeout # type: ignore
from utils.misc import getenv
2023-07-02 11:17:10 +02:00
from utils.wiki_downloader import download_image as wiki_download_image
unsplash_access = getenv("unsplash_access_key")
if not unsplash_access:
raise Exception("UNSPLASH_ACCESS_KEY is not set in .env file")
unsplash_url = "https://source.unsplash.com/random/?"
2023-05-15 10:11:04 +02:00
marp_image = """
<style>
section {
display: flex;
justify-content: center;
align-items: center;
}
.image-container {
width: 90%;
max-height: 90%;
display: flex;
justify-content: center;
align-items: center;
}
.image-container img {
object-fit: contain;
width: 100%;
height: 100%;
}
</style>
<div class="image-container">
<img src="[imagesrc]"/>
</div>
"""
async def prepare(path):
with open(os.path.join(path, "script.json"), 'r', encoding='utf-8') as f:
2023-05-15 10:11:04 +02:00
script = json.load(f)
f.close()
if not os.path.exists(path + "/slides"): os.mkdir(path + "/slides")
2023-07-02 11:17:10 +02:00
if not os.path.exists(path + "/audio"): os.mkdir(path + "/audio")
choosen_voice = random.choice(voices)
with open(os.path.join(os.getcwd(), "prompts", "marp.md"), 'r', encoding='utf-8') as f:
2023-05-15 10:11:04 +02:00
marp = f.read()
2023-07-02 11:17:10 +02:00
f.close()
for i in range(len(script)):
audio_path = os.path.join(path, "audio", "audio" + str(i) + ".wav")
generator = None
if not os.path.exists(audio_path):
if not generator:
generator = VoiceGenerator(speaker=choosen_voice)
print("Generating audio for slide " + str(i))
generator.generate_voice(audio_path, script[i]['spoken'])
if os.path.exists(os.path.join(path, "slides", "slide" + str(i) + ".md")):
#skip this slide
#continue
# TODO: Do not skip for now, add support for also checking for assets
pass
2023-07-02 11:17:10 +02:00
if "image" in script[i]:
if not os.path.exists(path + "/slides/assets"):
os.mkdir(path + "/slides/assets")
slide_asset_path = os.path.abspath(os.path.join(path, "slides", "assets", "slide" + str(i) + ".jpg"))
w = 0
while (not os.path.exists(slide_asset_path) or w < 5) and not os.path.exists(path + "/slides/slide" + str(i) + ".md"):
url= unsplash_url + script[i]['image'].replace("+", ",")
real_url = url
with open(slide_asset_path, 'wb') as f:
f.write(requests.get(real_url, allow_redirects=True).content)
f.close()
content = marp
content += "\n\n" + marp_image
content = content.replace("[imagesrc]", "assets/slide" + str(i) + ".jpg")
with open(path + "/slides/slide" + str(i) + ".md", 'w', encoding='utf-8') as f:
f.write(content)
w += 1
2023-07-02 11:17:10 +02:00
elif "wikimage" in script[i]:
if not os.path.exists(path + "/slides/assets"):
os.mkdir(path + "/slides/assets")
w = 0
slide_asset_path = os.path.abspath(os.path.join(path, "slides", "assets", "slide" + str(i) + ".jpg"))
while not (os.path.exists(os.path.join(path, "slides", "assets", "slide" + str(i) + ".jpg")) and os.path.exists(os.path.abspath(os.path.join(path, "slides", "slide" + str(i) + ".md")))):
print("Trying to download image for slide " + str(i))
wiki_download_image(script[i]['wikimage'], slide_asset_path)
content = marp
content += "\n\n" + marp_image
content = content.replace("[imagesrc]", "assets/slide" + str(i) + ".jpg")
with open(path + "/slides/slide" + str(i) + ".md", 'w', encoding='utf-8') as f:
f.write(content)
w += 1
2023-07-02 11:17:10 +02:00
elif "markdown" in script[i]:
while not os.path.exists(path + "/slides/slide" + str(i) + ".md"):
with open(path + "/slides/slide" + str(i) + ".md", 'w', encoding='utf-8') as f:
f.write(marp + "\n\n" + script[i]['markdown'])
2023-07-02 11:17:10 +02:00
elif "huge" in script[i]:
while not os.path.exists(path + "/slides/slide" + str(i) + ".md"):
with open(path + "/slides/slide" + str(i) + ".md", 'w', encoding='utf-8') as f:
f.write(marp + "\n\n# <!-- fit --> " + script[i]['huge'])
2023-07-02 11:17:10 +02:00
else:
while not os.path.exists(path + "/slides/slide" + str(i) + ".md"):
with open(path + "/slides/slide" + str(i) + ".md", 'w', encoding='utf-8') as f:
f.write(marp + "\n\n") # blank slide
2023-05-15 10:11:04 +02:00
for i in range(len(script)):
markdown_path = os.path.join(path, f"slides/slide{i}.md")
markdown_path = os.path.abspath(markdown_path)
image_path = os.path.join(path, f"slides/slide{i}.png")
image_path = os.path.abspath(image_path)
2023-07-02 11:17:10 +02:00
if not os.path.exists(image_path):
command = f'marp.exe --html "{markdown_path}" -o "{image_path}" --allow-local-files'
2023-07-02 11:17:10 +02:00
os.system(command)
2023-05-15 10:11:04 +02:00
return script
def convert_seconds_to_time_string(seconds):
milliseconds = int((seconds - int(seconds)) * 1000)
seconds = int(seconds)
minutes, seconds = divmod(seconds, 60)
hours, minutes = divmod(minutes, 60)
return f"{hours:02}:{minutes:02}:{seconds:02},{milliseconds:03}"
def subs(length, total, text, srt, index):
#first format the start and end in xx:xx:xx,xxx from float seconds like xx.xxxxxx
start = convert_seconds_to_time_string(total - length)
stop = convert_seconds_to_time_string(total)
sub = pysrt.SubRipItem(index=index, start=start, end=stop, text=text)
srt.append(sub)
return srt
async def mount(path, script):
if not os.path.exists(path + "/montage.mp4"):
num_slides = len(os.listdir(path + "/audio"))
clips = []
srt = pysrt.SubRipFile()
total_length = 0
for i in range(num_slides):
audio = AudioFileClip(path + "/audio/audio" + str(i) + ".wav")
complete_audio = CompositeAudioClip([
AudioFileClip("silence.mp3").set_duration(1),
audio,
AudioFileClip("silence.mp3").set_duration(1)
])
length = complete_audio.duration
total_length += length
srt = subs(length, total_length, script[i]['spoken'], srt, i)
slide = ImageClip(path + "/slides/slide" + str(i) + ".png").set_duration(length)
slide = slide.set_audio(complete_audio)
clips.append(slide)
randmusic = random.choice(os.listdir("musics"))
while randmusic.endswith(".txt"): randmusic = random.choice(os.listdir("musics"))
randpath = "musics/" + randmusic
2023-07-02 11:17:10 +02:00
music = AudioFileClip(randpath)
music = audio_fadein(music, 20)
music = audio_fadeout(music, 20)
music = volumex(music, 0.2)
musics = []
if music.duration < total_length:
for i in range(int(total_length / music.duration)):
musics.append(music)
music = concatenate_audioclips(musics)
2023-07-02 11:17:10 +02:00
music = music.set_duration(total_length)
final_clip = concatenate_videoclips(clips, method="compose")
existing_audio = final_clip.audio
final_audio = CompositeAudioClip([existing_audio, music])
final_clip = final_clip.set_audio(final_audio)
final_clip.write_videofile(path + "/montage.mp4", fps=60, codec="nvenc")
srt.save(path + "/montage.srt")
with open (randpath.split(".")[0] + ".txt", 'r', encoding='utf-8') as f:
music_credit = f.read()
f.close()
return music_credit or ""
else:
2023-07-02 11:17:10 +02:00
return ""