-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathgen_animation.py
121 lines (97 loc) · 4.01 KB
/
gen_animation.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#!/usr/bin/env python3
# -*-coding:utf-8 -*-
# =============================================================================
"""
@Author : Yujie He
@File : gen_animation.py
@Date created : 2021/11/28
@Maintainer : Yujie He
@Email : [email protected]
"""
# =============================================================================
"""
The module provides the workflow to generate videos and gifs from image folder
with ffmpeg.
"""
# =============================================================================
import os
import fnmatch
import argparse
import moviepy.editor as mpy
from qolo.core.crowdbot_data import CrowdBotDatabase
# need ffmpeg
def img_dir2video_ffmpeg(img_dir, video_path):
"""Generate video using ffmpeg via os.system()"""
# cmd: ffmpeg -y -r 15 -pattern_type glob -i 'tmp/*.png' -c:v libx264 -vf fps=30 -pix_fmt yuv420p 'tmp/frames.mp4'
header_cmd = "ffmpeg -y -r 15 -pattern_type glob -i "
in_images = "'{}/*.png'".format(img_dir)
middle_cmd = " -c:v libx264 -vf fps=30 -pix_fmt yuv420p "
out_video = "'{}'".format(video_path)
cmd = header_cmd + in_images + middle_cmd + out_video
os.system(cmd)
# need ffmpeg
def video2gif_ffmpeg(video_path, gif_path, w=640, h=480):
"""Generate gif from video using ffmpeg via os.system()"""
# cmd: ffmpeg -i lapse.mp4 -r 12 -s 640x360 output.gif
header_cmd = "ffmpeg -i "
in_images = "'{}'".format(video_seq_filepath)
middle_cmd = " -r 12 -s {}x{} ".format(w, h)
out_gif = "{}".format(gif_path)
cmd = header_cmd + in_images + middle_cmd + out_gif
os.system(cmd)
if __name__ == "__main__":
parser = argparse.ArgumentParser(description="convert data from rosbag")
parser.add_argument(
"-f",
"--folder",
default="nocam_rosbags",
type=str,
help="different subfolder in rosbag/ dir",
)
parser.add_argument(
"--overwrite",
dest="overwrite",
action="store_true",
help="Whether to overwrite existing rosbags (default: false)",
)
parser.set_defaults(overwrite=False)
args = parser.parse_args()
cb_data = CrowdBotDatabase(args.folder)
for seq_idx in range(cb_data.nr_seqs()):
seq = cb_data.seqs[seq_idx]
print(
"({}/{}): {} with {} frames".format(
seq_idx + 1, cb_data.nr_seqs(), seq, cb_data.nr_frames(seq_idx)
)
)
# seq src: data/xxxx_processed/media/img_o3d/seq
img3d_dir = os.path.join(cb_data.media_dir, "img_o3d")
img_seq_dir = os.path.join(img3d_dir, seq)
if not os.path.exists(img_seq_dir):
print("please generate images with gen_viz_img.py")
# seq dest: data/xxxx_processed/media/videos/seq.mp4
video_dir = os.path.join(cb_data.media_dir, "videos")
os.makedirs(video_dir, exist_ok=True)
video_seq_filepath = os.path.join(video_dir, seq + ".mp4")
if not os.path.exists(video_seq_filepath) or args.overwrite:
print("Images will be converted into {}".format(video_seq_filepath))
# img_dir2video_ffmpeg(img_seq_dir, video_seq_filepath)
# video2gif_ffmpeg(
# video_seq_filepath, video_seq_filepath.replace("mp4", "gif")
# )
img_files = sorted(fnmatch.filter(os.listdir(img_seq_dir), "*.png"))
if img_files:
print("{} frames detected".format(len(img_files)))
img_seq = [os.path.join(img_seq_dir, img) for img in img_files]
clip = mpy.ImageSequenceClip(img_seq, fps=30)
# .mp4 video
clip.write_videofile(video_seq_filepath, fps=15)
# .gif
clip.resize((720, 480)).write_gif(
video_seq_filepath.replace("mp4", "gif"), fps=15
)
else:
print("Please generate images for visualization using `gen_viz_img.py`")
else:
print("{} already generated!!!".format(video_seq_filepath))
continue