| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  | #!/usr/bin/env python3 | 
					
						
							|  |  |  | # call pyvideothumbnailer for every video file found | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # calculates rows based on the duration of the video | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import math | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import subprocess | 
					
						
							| 
									
										
										
										
											2023-08-14 00:14:21 +02:00
										 |  |  | import click | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | import mimetypes | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | from addict import Dict | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  | from pymediainfo import MediaInfo | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | def is_video_file(video_file_path): | 
					
						
							|  |  |  |     type_ = mimetypes.guess_type(video_file_path)[0] | 
					
						
							|  |  |  |     return type_ is not None and type_.startswith("video/") | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | def find_video_files(directory): | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  |     # Recursively walk through directory | 
					
						
							|  |  |  |     for dirpath, dirs, files in os.walk(directory): | 
					
						
							|  |  |  |         for filename in files: | 
					
						
							|  |  |  |             filename_path = os.path.join(dirpath, filename) | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |             if is_video_file(filename_path): | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  |                 yield os.path.abspath(filename_path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_video_duration(video_file_path): | 
					
						
							|  |  |  |     # Get media information | 
					
						
							|  |  |  |     media_info = MediaInfo.parse(video_file_path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Iterate through tracks in the media file | 
					
						
							|  |  |  |     for track in media_info.tracks: | 
					
						
							|  |  |  |         # If the track type is video | 
					
						
							|  |  |  |         if track.track_type == "Video": | 
					
						
							|  |  |  |             # Get duration and return it | 
					
						
							|  |  |  |             duration_str = track.duration | 
					
						
							|  |  |  |             duration_sec = float(duration_str) / 1000  # Convert to seconds | 
					
						
							|  |  |  |             return duration_sec | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Return None if no video track found | 
					
						
							|  |  |  |     return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | def call_thumbnailer(video_file, width, columns, rows, *, skip_seconds=10): | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  |     # Call external thumbnailer program | 
					
						
							|  |  |  |     subprocess.run( | 
					
						
							|  |  |  |         [ | 
					
						
							|  |  |  |             "pyvideothumbnailer", | 
					
						
							|  |  |  |             "--width", | 
					
						
							|  |  |  |             str(width), | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |             "--columns", | 
					
						
							|  |  |  |             str(columns), | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  |             "--rows", | 
					
						
							|  |  |  |             str(rows), | 
					
						
							|  |  |  |             "--jpeg-quality", | 
					
						
							|  |  |  |             str(90), | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |             "--header-font", | 
					
						
							|  |  |  |             config.font, | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  |             "--skip-seconds", | 
					
						
							|  |  |  |             str(skip_seconds), | 
					
						
							|  |  |  |             video_file, | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-14 00:14:21 +02:00
										 |  |  | def process(searchdir): | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |     video_files = find_video_files(searchdir) | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for video_file in video_files: | 
					
						
							|  |  |  |         # Skip if .jpg file already exists | 
					
						
							|  |  |  |         thumbnails_file = video_file + ".jpg" | 
					
						
							| 
									
										
										
										
											2024-02-06 22:40:19 +01:00
										 |  |  |         if os.path.lexists(thumbnails_file): | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |             if config.verbose: | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  |                 print(f"Skipping: {video_file} (thumbnails already exist)") | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             duration = get_video_duration(video_file) | 
					
						
							|  |  |  |             if duration is not None: | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |                 rows = math.ceil(duration / config.every / config.columns) | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 skip_seconds = 10 | 
					
						
							|  |  |  |                 if duration < 60: | 
					
						
							|  |  |  |                     skip_seconds = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Call the thumbnailer | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |                 call_thumbnailer( | 
					
						
							|  |  |  |                     video_file, | 
					
						
							|  |  |  |                     config.width, | 
					
						
							|  |  |  |                     config.columns, | 
					
						
							|  |  |  |                     rows, | 
					
						
							|  |  |  |                     skip_seconds=skip_seconds, | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  |         except (TypeError, UnicodeEncodeError, FileNotFoundError) as e: | 
					
						
							|  |  |  |             print(f"Error for {video_file}: {e}") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | config = Dict() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-14 00:14:21 +02:00
										 |  |  | @click.command() | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | @click.argument("searchdirs", nargs=-1) | 
					
						
							|  |  |  | @click.option("-v", "--verbose", is_flag=True, default=False) | 
					
						
							|  |  |  | def main(searchdirs, verbose): | 
					
						
							| 
									
										
										
										
											2023-08-14 00:14:21 +02:00
										 |  |  |     """Run pyvideothumbnailer for every video file found""" | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     config.verbose = verbose | 
					
						
							|  |  |  |     config.width = 1600 | 
					
						
							|  |  |  |     config.columns = 4  # pyvideothumbnailer default | 
					
						
							|  |  |  |     config.every = 5 * 60  # seconds | 
					
						
							|  |  |  |     config.font = "DejaVuSansMono.ttf" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-14 00:14:21 +02:00
										 |  |  |     if not searchdirs: | 
					
						
							| 
									
										
										
										
											2023-08-20 19:12:24 +02:00
										 |  |  |         searchdirs = ["."] | 
					
						
							| 
									
										
										
										
											2023-08-14 00:14:21 +02:00
										 |  |  |     for searchdir in searchdirs: | 
					
						
							|  |  |  |         process(searchdir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-13 23:02:22 +02:00
										 |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     main() |