multimedia_paradise 1.1.314

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of multimedia_paradise might be problematic. Click here for more details.

Files changed (417) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +2340 -0
  3. data/bin/audio_player +12 -0
  4. data/bin/auto_title +7 -0
  5. data/bin/convert_audio_to_mp4video_with_image +7 -0
  6. data/bin/cut_audio +7 -0
  7. data/bin/cut_multimedia +7 -0
  8. data/bin/extract_audio +7 -0
  9. data/bin/ffmpeg_merge +7 -0
  10. data/bin/file_duration +7 -0
  11. data/bin/gtk_radio +7 -0
  12. data/bin/loop_this_video +7 -0
  13. data/bin/merge_avi_files +7 -0
  14. data/bin/merge_mp3 +7 -0
  15. data/bin/missing_video_files +7 -0
  16. data/bin/movie_searcher +7 -0
  17. data/bin/mp3_to_opus +7 -0
  18. data/bin/mplayer_wrapper +7 -0
  19. data/bin/multimedia_information +11 -0
  20. data/bin/multimedia_paradise +9 -0
  21. data/bin/multimedia_paradise_sinatra +7 -0
  22. data/bin/play_random_simpsons_video +7 -0
  23. data/bin/playlist +7 -0
  24. data/bin/random_video +7 -0
  25. data/bin/remove_audio +9 -0
  26. data/bin/remove_last_second +7 -0
  27. data/bin/remove_subtitles +7 -0
  28. data/bin/tag_mp3_files +7 -0
  29. data/bin/to_aiff +10 -0
  30. data/bin/to_mp4 +7 -0
  31. data/bin/verbose_analyse_this_mp3_file_for_id3_tags +9 -0
  32. data/bin/video_codec +7 -0
  33. data/bin/video_information +8 -0
  34. data/bin/video_player +11 -0
  35. data/bin/video_to_images +7 -0
  36. data/bin/vp9 +9 -0
  37. data/bin/waveform +73 -0
  38. data/doc/CHANGELOG.md +44 -0
  39. data/doc/LINKS.md +6 -0
  40. data/doc/MOTIVATION_FOR_THIS_PROJECT.md +22 -0
  41. data/doc/MergingVideoLectures.md +12 -0
  42. data/doc/README.gen +2300 -0
  43. data/doc/Readme_for_the_cut_audio_component.md +168 -0
  44. data/doc/todo/todo_for_the_GUIs.md +0 -0
  45. data/doc/todo/todo_for_the_multimedia_paradise_project.md +111 -0
  46. data/lib/multimedia_paradise/audio/audio_player/audio_player.rb +83 -0
  47. data/lib/multimedia_paradise/audio/audio_player/constants.rb +136 -0
  48. data/lib/multimedia_paradise/audio/audio_player/menu.rb +164 -0
  49. data/lib/multimedia_paradise/audio/audio_player/misc.rb +718 -0
  50. data/lib/multimedia_paradise/audio/audio_player/reset.rb +77 -0
  51. data/lib/multimedia_paradise/audio/audio_player/tab.rb +35 -0
  52. data/lib/multimedia_paradise/audio/audio_tag_reader/README.md +7 -0
  53. data/lib/multimedia_paradise/audio/audio_tag_reader/audio_tag_reader.rb +503 -0
  54. data/lib/multimedia_paradise/audio/base.rb +67 -0
  55. data/lib/multimedia_paradise/audio/compress.rb +97 -0
  56. data/lib/multimedia_paradise/audio/create_m3u_playlist.rb +245 -0
  57. data/lib/multimedia_paradise/audio/extract_audio/constants.rb +37 -0
  58. data/lib/multimedia_paradise/audio/extract_audio/extract_audio.rb +417 -0
  59. data/lib/multimedia_paradise/audio/file_duration/constants.rb +53 -0
  60. data/lib/multimedia_paradise/audio/file_duration/file_duration.rb +602 -0
  61. data/lib/multimedia_paradise/audio/genres/boogie.rb +35 -0
  62. data/lib/multimedia_paradise/audio/genres/concerts.rb +35 -0
  63. data/lib/multimedia_paradise/audio/genres/constants.rb +53 -0
  64. data/lib/multimedia_paradise/audio/genres/eurodance.rb +35 -0
  65. data/lib/multimedia_paradise/audio/genres/genre.rb +696 -0
  66. data/lib/multimedia_paradise/audio/genres/hip_hop.rb +35 -0
  67. data/lib/multimedia_paradise/audio/genres/italian_songs.rb +35 -0
  68. data/lib/multimedia_paradise/audio/genres/the_1980s.rb +35 -0
  69. data/lib/multimedia_paradise/audio/genres/trance.rb +35 -0
  70. data/lib/multimedia_paradise/audio/lyrics_fetcher.rb +238 -0
  71. data/lib/multimedia_paradise/audio/merge_audio_files.rb +136 -0
  72. data/lib/multimedia_paradise/audio/modify_year_of_audio_file.rb +151 -0
  73. data/lib/multimedia_paradise/audio/n_audio_songs.rb +216 -0
  74. data/lib/multimedia_paradise/audio/play_all_audio_files.rb +122 -0
  75. data/lib/multimedia_paradise/audio/playlist/playlist.rb +1559 -0
  76. data/lib/multimedia_paradise/audio/remove_audio.rb +98 -0
  77. data/lib/multimedia_paradise/audio/remove_last_second.rb +83 -0
  78. data/lib/multimedia_paradise/audio/report_missing_id.rb +139 -0
  79. data/lib/multimedia_paradise/audio/streamripper/constants.rb +51 -0
  80. data/lib/multimedia_paradise/audio/streamripper/streamripper_wrapper.rb +279 -0
  81. data/lib/multimedia_paradise/audio/to_mp3.rb +138 -0
  82. data/lib/multimedia_paradise/audio/to_ogg.rb +104 -0
  83. data/lib/multimedia_paradise/audio/wav_to_mp3.rb +87 -0
  84. data/lib/multimedia_paradise/audio/waveform/class.rb +341 -0
  85. data/lib/multimedia_paradise/audio/waveform/constants.rb +38 -0
  86. data/lib/multimedia_paradise/audio/waveform/log.rb +101 -0
  87. data/lib/multimedia_paradise/autoinclude.rb +3 -0
  88. data/lib/multimedia_paradise/autoinclude_remove_audio.rb +6 -0
  89. data/lib/multimedia_paradise/base/base.rb +34 -0
  90. data/lib/multimedia_paradise/base/colours.rb +354 -0
  91. data/lib/multimedia_paradise/base/commandline_arguments.rb +52 -0
  92. data/lib/multimedia_paradise/base/constants.rb +19 -0
  93. data/lib/multimedia_paradise/base/encoding.rb +31 -0
  94. data/lib/multimedia_paradise/base/misc.rb +618 -0
  95. data/lib/multimedia_paradise/base/namespace.rb +36 -0
  96. data/lib/multimedia_paradise/base/time.rb +25 -0
  97. data/lib/multimedia_paradise/colours/colours.rb +82 -0
  98. data/lib/multimedia_paradise/commandline/menu.rb +498 -0
  99. data/lib/multimedia_paradise/configuration/play_zoomed.yml +1 -0
  100. data/lib/multimedia_paradise/constants/constants.rb +23 -0
  101. data/lib/multimedia_paradise/constants/conversions.rb +62 -0
  102. data/lib/multimedia_paradise/constants/directory_constants.rb +133 -0
  103. data/lib/multimedia_paradise/constants/encodings.rb +26 -0
  104. data/lib/multimedia_paradise/constants/file_constants.rb +171 -0
  105. data/lib/multimedia_paradise/constants/misc.rb +80 -0
  106. data/lib/multimedia_paradise/constants/my_video_directory.rb +30 -0
  107. data/lib/multimedia_paradise/constants/namespace.rb +14 -0
  108. data/lib/multimedia_paradise/constants/newline.rb +14 -0
  109. data/lib/multimedia_paradise/constants/video_filetypes.rb +27 -0
  110. data/lib/multimedia_paradise/constants/web_constants.rb +150 -0
  111. data/lib/multimedia_paradise/conversions/README.md +2 -0
  112. data/lib/multimedia_paradise/conversions/conversions.rb +930 -0
  113. data/lib/multimedia_paradise/css/project.css +15 -0
  114. data/lib/multimedia_paradise/data/asoundrc +12 -0
  115. data/lib/multimedia_paradise/ffmpeg/README.md +2 -0
  116. data/lib/multimedia_paradise/ffmpeg/ffmpeg.rb +620 -0
  117. data/lib/multimedia_paradise/gui/fox/play_from_radio_station.rb +75 -0
  118. data/lib/multimedia_paradise/gui/gosu/video_player/video_player.rb +109 -0
  119. data/lib/multimedia_paradise/gui/gtk2/lyrics/lyrics.rb +33 -0
  120. data/lib/multimedia_paradise/gui/gtk2/multimedia_converter/multimedia_converter.rb +33 -0
  121. data/lib/multimedia_paradise/gui/gtk2/notebook.rb +144 -0
  122. data/lib/multimedia_paradise/gui/gtk2/play_video_from_my_collection/play_video_from_my_collection.rb +43 -0
  123. data/lib/multimedia_paradise/gui/gtk2/simple_play_widget/README.md +12 -0
  124. data/lib/multimedia_paradise/gui/gtk2/simple_play_widget/simple_play_widget.rb +40 -0
  125. data/lib/multimedia_paradise/gui/gtk2/video_collection/video_collection.rb +56 -0
  126. data/lib/multimedia_paradise/gui/gtk2/widget_increase_or_decrease_audio/widget_increase_or_decrease_audio.rb +42 -0
  127. data/lib/multimedia_paradise/gui/gtk2/youtube_downloader/youtube_downloader.rb +32 -0
  128. data/lib/multimedia_paradise/gui/gtk3/change_metadata_widget/README.md +3 -0
  129. data/lib/multimedia_paradise/gui/gtk3/change_metadata_widget/change_metadata_widget.rb +150 -0
  130. data/lib/multimedia_paradise/gui/gtk3/cut_multimedia/cut_multimedia.rb +321 -0
  131. data/lib/multimedia_paradise/gui/gtk3/information_about_a_mp3_file/information_about_a_mp3_file.rb +421 -0
  132. data/lib/multimedia_paradise/gui/gtk3/lyrics/lyrics.rb +33 -0
  133. data/lib/multimedia_paradise/gui/gtk3/multimedia_converter/multimedia_converter.rb +34 -0
  134. data/lib/multimedia_paradise/gui/gtk3/multimedia_notebook/multimedia_notebook.rb +34 -0
  135. data/lib/multimedia_paradise/gui/gtk3/play_video_from_my_collection/play_video_from_my_collection.rb +34 -0
  136. data/lib/multimedia_paradise/gui/gtk3/playlist/playlist.rb +34 -0
  137. data/lib/multimedia_paradise/gui/gtk3/radio/misc.rb +487 -0
  138. data/lib/multimedia_paradise/gui/gtk3/radio/radio.rb +698 -0
  139. data/lib/multimedia_paradise/gui/gtk3/simple_play_widget/simple_play_widget.rb +38 -0
  140. data/lib/multimedia_paradise/gui/gtk3/sound_effect_widget/sound_effect_widget.rb +34 -0
  141. data/lib/multimedia_paradise/gui/gtk3/tag_mp3_files/connect_skeleton.rb +61 -0
  142. data/lib/multimedia_paradise/gui/gtk3/tag_mp3_files/create.rb +405 -0
  143. data/lib/multimedia_paradise/gui/gtk3/tag_mp3_files/misc.rb +1820 -0
  144. data/lib/multimedia_paradise/gui/gtk3/tag_mp3_files/tag_mp3_files.config +6 -0
  145. data/lib/multimedia_paradise/gui/gtk3/tag_mp3_files/tag_mp3_files.rb +66 -0
  146. data/lib/multimedia_paradise/gui/gtk3/video_collection/video_collection.rb +43 -0
  147. data/lib/multimedia_paradise/gui/gtk3/video_editor/video_editor.rb +34 -0
  148. data/lib/multimedia_paradise/gui/gtk3/video_player/video_player.rb +34 -0
  149. data/lib/multimedia_paradise/gui/gtk3/webcam_widget/webcam_widget.rb +425 -0
  150. data/lib/multimedia_paradise/gui/gtk3/widget_increase_or_decrease_audio/widget_increase_or_decrease_audio.rb +36 -0
  151. data/lib/multimedia_paradise/gui/gtk3/youtube_channels/youtube_channels.rb +210 -0
  152. data/lib/multimedia_paradise/gui/gtk3/youtube_downloader/youtube_downloader.rb +32 -0
  153. data/lib/multimedia_paradise/gui/gui_base.rb +92 -0
  154. data/lib/multimedia_paradise/gui/libui/change_metadata_widget/change_metadata_widget.rb +115 -0
  155. data/lib/multimedia_paradise/gui/libui/cut_multimedia/cut_multimedia.rb +119 -0
  156. data/lib/multimedia_paradise/gui/libui/lyrics/lyrics.rb +67 -0
  157. data/lib/multimedia_paradise/gui/libui/simple_play_widget/simple_play_widget.rb +119 -0
  158. data/lib/multimedia_paradise/gui/libui/tag_mp3_files/tag_mp3_files.rb +268 -0
  159. data/lib/multimedia_paradise/gui/libui/video_player/video_player.rb +81 -0
  160. data/lib/multimedia_paradise/gui/libui/widget_increase_or_decrease_audio/widget_increase_or_decrease_audio.rb +126 -0
  161. data/lib/multimedia_paradise/gui/libui/youtube_channels/youtube_channels.rb +116 -0
  162. data/lib/multimedia_paradise/gui/libui/youtube_downloader/youtube_downloader.rb +107 -0
  163. data/lib/multimedia_paradise/gui/shared_code/change_metadata_widget/change_metadata_widget_module.rb +538 -0
  164. data/lib/multimedia_paradise/gui/shared_code/cut_multimedia/cut_multimedia_module.rb +268 -0
  165. data/lib/multimedia_paradise/gui/shared_code/information_about_a_mp3_file/information_about_a_mp3_file_module.rb +120 -0
  166. data/lib/multimedia_paradise/gui/shared_code/lyrics/lyrics_module.rb +141 -0
  167. data/lib/multimedia_paradise/gui/shared_code/multimedia_converter/multimedia_converter_module.rb +478 -0
  168. data/lib/multimedia_paradise/gui/shared_code/multimedia_notebook/multimedia_notebook_module.rb +171 -0
  169. data/lib/multimedia_paradise/gui/shared_code/play_video_from_my_collection/play_video_from_my_collection_module.rb +464 -0
  170. data/lib/multimedia_paradise/gui/shared_code/playlist/playlist_module.rb +151 -0
  171. data/lib/multimedia_paradise/gui/shared_code/simple_play_widget/simple_play_widget_module.rb +257 -0
  172. data/lib/multimedia_paradise/gui/shared_code/sound_effect_widget/sound_effect_widget_module.rb +173 -0
  173. data/lib/multimedia_paradise/gui/shared_code/tag_mp3_files/tag_mp3_files_module.rb +73 -0
  174. data/lib/multimedia_paradise/gui/shared_code/video_editor/video_editor_module.rb +287 -0
  175. data/lib/multimedia_paradise/gui/shared_code/video_player/video_player_module.rb +185 -0
  176. data/lib/multimedia_paradise/gui/shared_code/widget_increase_or_decrease_audio/widget_increase_or_decrease_audio_module.rb +310 -0
  177. data/lib/multimedia_paradise/gui/shared_code/youtube_downloader/youtube_downloader_module.rb +339 -0
  178. data/lib/multimedia_paradise/gui/tk/tk_multimedia_wrapper.rb +60 -0
  179. data/lib/multimedia_paradise/help/help.rb +75 -0
  180. data/lib/multimedia_paradise/images/MULTIMEDIA.jpg +0 -0
  181. data/lib/multimedia_paradise/images/MULTIMEDIA_PARADISE_ID_TAG_GUI_IN_RUBY_GTK3.png +0 -0
  182. data/lib/multimedia_paradise/images/MULTIMEDIA_PARADISE_LOGO.png +0 -0
  183. data/lib/multimedia_paradise/images/UK_flag.png +0 -0
  184. data/lib/multimedia_paradise/images/US_flag.png +0 -0
  185. data/lib/multimedia_paradise/images/austrian_flag.png +0 -0
  186. data/lib/multimedia_paradise/images/french_flag.png +0 -0
  187. data/lib/multimedia_paradise/images/german_flag.png +0 -0
  188. data/lib/multimedia_paradise/images/radio_image.png +0 -0
  189. data/lib/multimedia_paradise/images/trance.png +0 -0
  190. data/lib/multimedia_paradise/java/AudioPlayer.class +0 -0
  191. data/lib/multimedia_paradise/java/AudioPlayer.java +103 -0
  192. data/lib/multimedia_paradise/java/README.md +3 -0
  193. data/lib/multimedia_paradise/misc/long_format_to_milliseconds_converter.rb +182 -0
  194. data/lib/multimedia_paradise/misc/milliseconds_to_long_format_converter.rb +186 -0
  195. data/lib/multimedia_paradise/multimedia/analyse_multimedia_file.rb +180 -0
  196. data/lib/multimedia_paradise/multimedia/avisynth/README.md +4 -0
  197. data/lib/multimedia_paradise/multimedia/avisynth/avisynth_code.avs +471 -0
  198. data/lib/multimedia_paradise/multimedia/base.rb +60 -0
  199. data/lib/multimedia_paradise/multimedia/chord.rb +174 -0
  200. data/lib/multimedia_paradise/multimedia/convert_audio_to_video_with_image.rb +171 -0
  201. data/lib/multimedia_paradise/multimedia/cut_multimedia/10_minutes_chop.rb +29 -0
  202. data/lib/multimedia_paradise/multimedia/cut_multimedia/15_minutes_chop.rb +29 -0
  203. data/lib/multimedia_paradise/multimedia/cut_multimedia/30_minutes_chop.rb +29 -0
  204. data/lib/multimedia_paradise/multimedia/cut_multimedia/5_minutes_chop.rb +29 -0
  205. data/lib/multimedia_paradise/multimedia/cut_multimedia/cut_multimedia.rb +2017 -0
  206. data/lib/multimedia_paradise/multimedia/cut_multimedia/evaluate_from_this_file.rb +79 -0
  207. data/lib/multimedia_paradise/multimedia/cut_multimedia/interactive_menu.rb +514 -0
  208. data/lib/multimedia_paradise/multimedia/interactive_shell.rb +84 -0
  209. data/lib/multimedia_paradise/multimedia/merge_multimedia.rb +125 -0
  210. data/lib/multimedia_paradise/multimedia/note.rb +500 -0
  211. data/lib/multimedia_paradise/multimedia/play_from_this_list.rb +163 -0
  212. data/lib/multimedia_paradise/multimedia/read_meta_tags.rb +173 -0
  213. data/lib/multimedia_paradise/multimedia/simulate_youtube_playlist.rb +209 -0
  214. data/lib/multimedia_paradise/multimedia/start_length_duration.rb +205 -0
  215. data/lib/multimedia_paradise/multimedia/video_downloader/video_downloader.rb +143 -0
  216. data/lib/multimedia_paradise/project/project.rb +47 -0
  217. data/lib/multimedia_paradise/requires/require_audio_files.rb +74 -0
  218. data/lib/multimedia_paradise/requires/require_cut_multimedia.rb +7 -0
  219. data/lib/multimedia_paradise/requires/require_extract_audio.rb +7 -0
  220. data/lib/multimedia_paradise/requires/require_file_duration.rb +7 -0
  221. data/lib/multimedia_paradise/requires/require_streamripper.rb +7 -0
  222. data/lib/multimedia_paradise/requires/require_the_audio_player.rb +7 -0
  223. data/lib/multimedia_paradise/requires/require_the_multimedia_paradise_project.rb +38 -0
  224. data/lib/multimedia_paradise/requires/require_the_sinatra_components.rb +7 -0
  225. data/lib/multimedia_paradise/requires/require_toplevel_methods_files.rb +25 -0
  226. data/lib/multimedia_paradise/requires/require_video_files.rb +25 -0
  227. data/lib/multimedia_paradise/requires/require_video_player.rb +7 -0
  228. data/lib/multimedia_paradise/sinatra/app.rb +311 -0
  229. data/lib/multimedia_paradise/sinatra/embeddable_interface.rb +379 -0
  230. data/lib/multimedia_paradise/sound_effects/README.md +12 -0
  231. data/lib/multimedia_paradise/sound_effects/cat_meow.mp3 +0 -0
  232. data/lib/multimedia_paradise/sound_effects/channel_opened.mp3 +0 -0
  233. data/lib/multimedia_paradise/sound_effects/phone_ring.mp3 +0 -0
  234. data/lib/multimedia_paradise/statistics/README.md +5 -0
  235. data/lib/multimedia_paradise/statistics/video.rb +84 -0
  236. data/lib/multimedia_paradise/toplevel_cut_audio.rb +16 -0
  237. data/lib/multimedia_paradise/toplevel_methods/analyze_audio_stream.rb +31 -0
  238. data/lib/multimedia_paradise/toplevel_methods/chop_into_segments_of_n_seconds_size.rb +32 -0
  239. data/lib/multimedia_paradise/toplevel_methods/chop_off_first_five_minutes.rb +22 -0
  240. data/lib/multimedia_paradise/toplevel_methods/chop_off_first_n_seconds.rb +92 -0
  241. data/lib/multimedia_paradise/toplevel_methods/chop_off_first_ten_minutes.rb +22 -0
  242. data/lib/multimedia_paradise/toplevel_methods/chop_off_first_two_minutes.rb +24 -0
  243. data/lib/multimedia_paradise/toplevel_methods/codecs.rb +50 -0
  244. data/lib/multimedia_paradise/toplevel_methods/copy_and_merge_this_video_n_times.rb +51 -0
  245. data/lib/multimedia_paradise/toplevel_methods/copy_file.rb +18 -0
  246. data/lib/multimedia_paradise/toplevel_methods/create_video_from_this_audio.rb +65 -0
  247. data/lib/multimedia_paradise/toplevel_methods/cut_from_to.rb +156 -0
  248. data/lib/multimedia_paradise/toplevel_methods/delay_audio.rb +31 -0
  249. data/lib/multimedia_paradise/toplevel_methods/denoise.rb +90 -0
  250. data/lib/multimedia_paradise/toplevel_methods/deshake.rb +43 -0
  251. data/lib/multimedia_paradise/toplevel_methods/e.rb +16 -0
  252. data/lib/multimedia_paradise/toplevel_methods/encode_this_video.rb +49 -0
  253. data/lib/multimedia_paradise/toplevel_methods/ensure_that_the_output_directory_exists.rb +27 -0
  254. data/lib/multimedia_paradise/toplevel_methods/esystem.rb +43 -0
  255. data/lib/multimedia_paradise/toplevel_methods/files_and_directories.rb +90 -0
  256. data/lib/multimedia_paradise/toplevel_methods/flip_and_rotate.rb +58 -0
  257. data/lib/multimedia_paradise/toplevel_methods/increase_volume_of_this_audio_file.rb +61 -0
  258. data/lib/multimedia_paradise/toplevel_methods/is_a_multimedia_file.rb +27 -0
  259. data/lib/multimedia_paradise/toplevel_methods/is_audio_file.rb +27 -0
  260. data/lib/multimedia_paradise/toplevel_methods/is_image_file.rb +31 -0
  261. data/lib/multimedia_paradise/toplevel_methods/is_on_roebe.rb +17 -0
  262. data/lib/multimedia_paradise/toplevel_methods/is_video_file.rb +62 -0
  263. data/lib/multimedia_paradise/toplevel_methods/merge_multimedia_file.rb +327 -0
  264. data/lib/multimedia_paradise/toplevel_methods/merge_these_videos.rb +105 -0
  265. data/lib/multimedia_paradise/toplevel_methods/misc.rb +885 -0
  266. data/lib/multimedia_paradise/toplevel_methods/n_local_videos.rb +56 -0
  267. data/lib/multimedia_paradise/toplevel_methods/opn.rb +24 -0
  268. data/lib/multimedia_paradise/toplevel_methods/output_directory.rb +59 -0
  269. data/lib/multimedia_paradise/toplevel_methods/player_in_use.rb +80 -0
  270. data/lib/multimedia_paradise/toplevel_methods/query_the_audio_codec_of_this_file.rb +24 -0
  271. data/lib/multimedia_paradise/toplevel_methods/radio.rb +360 -0
  272. data/lib/multimedia_paradise/toplevel_methods/rds.rb +24 -0
  273. data/lib/multimedia_paradise/toplevel_methods/return_all_video_files.rb +64 -0
  274. data/lib/multimedia_paradise/toplevel_methods/return_data_from_video_collection_file_for_this_entry.rb +38 -0
  275. data/lib/multimedia_paradise/toplevel_methods/return_duration_of_this_multimedia_file.rb +27 -0
  276. data/lib/multimedia_paradise/toplevel_methods/return_full_name_for_video_at_this_position.rb +54 -0
  277. data/lib/multimedia_paradise/toplevel_methods/return_path_to_random_simpsons_video_file.rb +43 -0
  278. data/lib/multimedia_paradise/toplevel_methods/return_random_video_file_from_the_video_collection.rb +73 -0
  279. data/lib/multimedia_paradise/toplevel_methods/return_screen_resolution.rb +24 -0
  280. data/lib/multimedia_paradise/toplevel_methods/run_sys_command.rb +30 -0
  281. data/lib/multimedia_paradise/toplevel_methods/scale_video.rb +27 -0
  282. data/lib/multimedia_paradise/toplevel_methods/set_title_of.rb +65 -0
  283. data/lib/multimedia_paradise/toplevel_methods/slow_down_this_video_file.rb +33 -0
  284. data/lib/multimedia_paradise/toplevel_methods/start_screencast.rb +70 -0
  285. data/lib/multimedia_paradise/toplevel_methods/subtitles.rb +59 -0
  286. data/lib/multimedia_paradise/toplevel_methods/to_flac.rb +30 -0
  287. data/lib/multimedia_paradise/toplevel_methods/to_mp4.rb +24 -0
  288. data/lib/multimedia_paradise/toplevel_methods/total_duration.rb +35 -0
  289. data/lib/multimedia_paradise/toplevel_methods/use_lame_codec.rb +32 -0
  290. data/lib/multimedia_paradise/toplevel_methods/video_dataset.rb +76 -0
  291. data/lib/multimedia_paradise/version/version.rb +28 -0
  292. data/lib/multimedia_paradise/video/all_videos.rb +101 -0
  293. data/lib/multimedia_paradise/video/capture_screen.rb +285 -0
  294. data/lib/multimedia_paradise/video/check_numbers.rb +215 -0
  295. data/lib/multimedia_paradise/video/columbo/columbo.rb +139 -0
  296. data/lib/multimedia_paradise/video/copy_missing_video_files.rb +293 -0
  297. data/lib/multimedia_paradise/video/copy_random_video.rb +98 -0
  298. data/lib/multimedia_paradise/video/correct_video_numbers.rb +278 -0
  299. data/lib/multimedia_paradise/video/create_dvd.rb +50 -0
  300. data/lib/multimedia_paradise/video/encode_video.rb +81 -0
  301. data/lib/multimedia_paradise/video/ffmpeg_options.rb +156 -0
  302. data/lib/multimedia_paradise/video/find_video/constants.rb +23 -0
  303. data/lib/multimedia_paradise/video/find_video/find_video.rb +493 -0
  304. data/lib/multimedia_paradise/video/fix_married_with_children_videos.rb +70 -0
  305. data/lib/multimedia_paradise/video/guess_video_name.rb +189 -0
  306. data/lib/multimedia_paradise/video/imdb_rating/imdb_rating.rb +141 -0
  307. data/lib/multimedia_paradise/video/mike_hammer/mike_hammer.rb +74 -0
  308. data/lib/multimedia_paradise/video/missing_video_files/missing_video_files.rb +286 -0
  309. data/lib/multimedia_paradise/video/movie_searcher.rb +281 -0
  310. data/lib/multimedia_paradise/video/mplayer_wrapper.rb +504 -0
  311. data/lib/multimedia_paradise/video/play_random_file.rb +91 -0
  312. data/lib/multimedia_paradise/video/play_random_realvideo.rb +87 -0
  313. data/lib/multimedia_paradise/video/prepare_video_lecture.rb +272 -0
  314. data/lib/multimedia_paradise/video/random_video.rb +310 -0
  315. data/lib/multimedia_paradise/video/registered_video_file.rb +131 -0
  316. data/lib/multimedia_paradise/video/remove_metadata.rb +52 -0
  317. data/lib/multimedia_paradise/video/rename_video_file.rb +149 -0
  318. data/lib/multimedia_paradise/video/report_local_videos.rb +106 -0
  319. data/lib/multimedia_paradise/video/report_missing_videos_in_the_yaml_file.rb +107 -0
  320. data/lib/multimedia_paradise/video/resolution.rb +159 -0
  321. data/lib/multimedia_paradise/video/simpsons.rb +129 -0
  322. data/lib/multimedia_paradise/video/speed_up_video.rb +135 -0
  323. data/lib/multimedia_paradise/video/split_this_video.rb +126 -0
  324. data/lib/multimedia_paradise/video/srt_regex.rb +211 -0
  325. data/lib/multimedia_paradise/video/store_available_video_files.rb +138 -0
  326. data/lib/multimedia_paradise/video/the_simpsons/README.md +9 -0
  327. data/lib/multimedia_paradise/video/the_simpsons/good_the_simpsons_episodes.rb +107 -0
  328. data/lib/multimedia_paradise/video/the_simpsons/the_simpsons.rb +82 -0
  329. data/lib/multimedia_paradise/video/video.rb +14 -0
  330. data/lib/multimedia_paradise/video/video_encoding_settings.rb +57 -0
  331. data/lib/multimedia_paradise/video/video_genres.rb +337 -0
  332. data/lib/multimedia_paradise/video/video_information.rb +443 -0
  333. data/lib/multimedia_paradise/video/video_metadata.rb +132 -0
  334. data/lib/multimedia_paradise/video/video_player.rb +581 -0
  335. data/lib/multimedia_paradise/video/video_renamer.rb +160 -0
  336. data/lib/multimedia_paradise/video/watermark.rb +290 -0
  337. data/lib/multimedia_paradise/video/youtube/youtube.rb +17 -0
  338. data/lib/multimedia_paradise/video/youtube_embedder.rb +235 -0
  339. data/lib/multimedia_paradise/www/GIS/GIS.cgi +147 -0
  340. data/lib/multimedia_paradise/www/HDMI/HDMI.cgi +34 -0
  341. data/lib/multimedia_paradise/www/TALES_FROM_THE_CRYPT.html +71 -0
  342. data/lib/multimedia_paradise/www/alsa/alsa.cgi +7 -0
  343. data/lib/multimedia_paradise/www/alsa/alsa.rb +309 -0
  344. data/lib/multimedia_paradise/www/alsa/alsa.sinatra +56 -0
  345. data/lib/multimedia_paradise/www/audacity.cgi +30 -0
  346. data/lib/multimedia_paradise/www/audio/audio.cgi +9 -0
  347. data/lib/multimedia_paradise/www/audio/audio.rb +2739 -0
  348. data/lib/multimedia_paradise/www/audio/audio.sinatra +56 -0
  349. data/lib/multimedia_paradise/www/avisynth/avisynth.cgi +172 -0
  350. data/lib/multimedia_paradise/www/best_voices/best_voices.cgi +60 -0
  351. data/lib/multimedia_paradise/www/cassette/cassette.cgi +723 -0
  352. data/lib/multimedia_paradise/www/codecs/codecs.cgi +95 -0
  353. data/lib/multimedia_paradise/www/componists/vivaldi.cgi +218 -0
  354. data/lib/multimedia_paradise/www/decibel.cgi +79 -0
  355. data/lib/multimedia_paradise/www/ffmpeg/ffmpeg.cgi +439 -0
  356. data/lib/multimedia_paradise/www/lame/lame.cgi +7 -0
  357. data/lib/multimedia_paradise/www/lame/lame.rb +101 -0
  358. data/lib/multimedia_paradise/www/lame/lame.sinatra +56 -0
  359. data/lib/multimedia_paradise/www/lilypond.cgi +96 -0
  360. data/lib/multimedia_paradise/www/lyrics/lyrics.cgi +49 -0
  361. data/lib/multimedia_paradise/www/midi/al_adagi.mid +0 -0
  362. data/lib/multimedia_paradise/www/play_songs.cgi +102 -0
  363. data/lib/multimedia_paradise/www/play_this_video_file.cgi +100 -0
  364. data/lib/multimedia_paradise/www/pulseaudio/pulseaudio.cgi +7 -0
  365. data/lib/multimedia_paradise/www/pulseaudio/pulseaudio.rb +168 -0
  366. data/lib/multimedia_paradise/www/pulseaudio/pulseaudio.sinatra +56 -0
  367. data/lib/multimedia_paradise/www/rhythmbox/rhythmbox.cgi +120 -0
  368. data/lib/multimedia_paradise/www/sound_theory.cgi +163 -0
  369. data/lib/multimedia_paradise/www/tv/tv.cgi +182 -0
  370. data/lib/multimedia_paradise/www/video/video.cgi +7 -0
  371. data/lib/multimedia_paradise/www/video/video.rb +7709 -0
  372. data/lib/multimedia_paradise/www/video/video.sinatra +52 -0
  373. data/lib/multimedia_paradise/www/web_streams.cgi +58 -0
  374. data/lib/multimedia_paradise/yaml/audio_formats.yml +14 -0
  375. data/lib/multimedia_paradise/yaml/best_songs_in_an_era/eurodance/top20.yml +28 -0
  376. data/lib/multimedia_paradise/yaml/game_shows/ruck_zuck/ruck_zuck.yml +25 -0
  377. data/lib/multimedia_paradise/yaml/genres/1980s.yml +198 -0
  378. data/lib/multimedia_paradise/yaml/genres/README.md +5 -0
  379. data/lib/multimedia_paradise/yaml/genres/boogie.yml +68 -0
  380. data/lib/multimedia_paradise/yaml/genres/concerts.yml +29 -0
  381. data/lib/multimedia_paradise/yaml/genres/eurodance.yml +456 -0
  382. data/lib/multimedia_paradise/yaml/genres/hip_hop.yml +49 -0
  383. data/lib/multimedia_paradise/yaml/genres/italian_songs.yml +53 -0
  384. data/lib/multimedia_paradise/yaml/genres/trance.yml +167 -0
  385. data/lib/multimedia_paradise/yaml/good_horror_movies/good_horror_movies.yml +12 -0
  386. data/lib/multimedia_paradise/yaml/good_movie_actors/good_movie_actors.yml +6 -0
  387. data/lib/multimedia_paradise/yaml/image_formats.yml +8 -0
  388. data/lib/multimedia_paradise/yaml/internal/README.md +2 -0
  389. data/lib/multimedia_paradise/yaml/internal/installed_binaries.yml +31 -0
  390. data/lib/multimedia_paradise/yaml/lyrics.yml +6275 -0
  391. data/lib/multimedia_paradise/yaml/movie_reviews/the_hateful_eight_september_2016.md +305 -0
  392. data/lib/multimedia_paradise/yaml/music_genres.yml +51 -0
  393. data/lib/multimedia_paradise/yaml/playlist.yml +127 -0
  394. data/lib/multimedia_paradise/yaml/radio/README.md +5 -0
  395. data/lib/multimedia_paradise/yaml/radio/radio_stations.yml +359 -0
  396. data/lib/multimedia_paradise/yaml/song_tags.yml +815 -0
  397. data/lib/multimedia_paradise/yaml/tales_from_the_crypt/tales_from_the_crypt.yml +418 -0
  398. data/lib/multimedia_paradise/yaml/tom_and_jerry/README.md +2 -0
  399. data/lib/multimedia_paradise/yaml/tom_and_jerry/tom_and_jerry_episodes.yml +27 -0
  400. data/lib/multimedia_paradise/yaml/tv_channels/tv_channels.yml +46 -0
  401. data/lib/multimedia_paradise/yaml/use_this_video_player.yml +1 -0
  402. data/lib/multimedia_paradise/yaml/video/video.yml +26 -0
  403. data/lib/multimedia_paradise/yaml/video_collection/video_collection.yml +1933 -0
  404. data/lib/multimedia_paradise/yaml/video_encoding_settings.yml +62 -0
  405. data/lib/multimedia_paradise/yaml/video_filter_aliases.yml +221 -0
  406. data/lib/multimedia_paradise/yaml/youtube/alltagsgeschichte/alltagsgeschichte.yml +172 -0
  407. data/lib/multimedia_paradise/yaml/youtube/songs/README.md +3 -0
  408. data/lib/multimedia_paradise/yaml/youtube/songs/songs.yml +6 -0
  409. data/lib/multimedia_paradise/yaml/youtube_playlist/youtube_playlist.yml +6 -0
  410. data/lib/multimedia_paradise.rb +5 -0
  411. data/multimedia_paradise.gemspec +76 -0
  412. data/test/testing_audio_player.rb +15 -0
  413. data/test/testing_ffmpeg_options.rb +14 -0
  414. data/test/testing_file_duration.rb +17 -0
  415. data/test/testing_modify_year_of_audio_file.rb +14 -0
  416. data/test/testing_multimedia_paradise_project.rb +151 -0
  417. metadata +581 -0
@@ -0,0 +1,885 @@
1
+ #!/usr/bin/ruby -w
2
+ # Encoding: UTF-8
3
+ # frozen_string_literal: true
4
+ # =========================================================================== #
5
+ # require 'multimedia_paradise/toplevel_methods/misc.rb'
6
+ # MultimediaParadise.bitrate_of_this_song
7
+ # MultimediaParadise.verbose_analyse_this_mp3_file_for_id3_tags
8
+ # MultimediaParadise.return_video_codec_of_this_file
9
+ # =========================================================================== #
10
+ module MultimediaParadise
11
+
12
+ require 'multimedia_paradise/constants/misc.rb'
13
+ require 'multimedia_paradise/toplevel_methods/esystem.rb'
14
+ require 'multimedia_paradise/ffmpeg/ffmpeg.rb'
15
+
16
+ # ========================================================================= #
17
+ # === MultimediaParadise.loop_this_video
18
+ #
19
+ # This method can be used to loop a video, via ffmpeg.
20
+ #
21
+ # A commandline variant for this would look as follows:
22
+ #
23
+ # ffmpeg -stream_loop 3 -i Tata.webm -c copy output.webm
24
+ #
25
+ # ========================================================================= #
26
+ def self.loop_this_video(this_video = 'foobar.webm', n_times = '5x')
27
+ result = []
28
+ [this_video].flatten.compact.each {|work_on_this_video_file|
29
+ n_times = n_times.to_s.delete('x').sub(/ times/,'')
30
+ output_file = "output_#{work_on_this_video_file}"
31
+ result << output_file
32
+ cmd = "ffmpeg -stream_loop #{n_times} -i #{work_on_this_video_file} -c copy #{output_file}"
33
+ esystem cmd
34
+ }
35
+ return result
36
+ end
37
+
38
+ # ========================================================================= #
39
+ # === MultimediaParadise.bin_dir?
40
+ # ========================================================================= #
41
+ def self.bin_dir?
42
+ RbConfig::CONFIG['bindir'].to_s+'/' # Ensure trailing '/'.
43
+ end; self.instance_eval { alias bindir? bin_dir? } # === MultimediaParadise.bindir?
44
+
45
+ # ========================================================================= #
46
+ # === MultimediaParadise.copy_bin_directory
47
+ #
48
+ # This method can be used to copy the content of the bin/ directory
49
+ # onto the current working directory.
50
+ #
51
+ # Usage example from within ruby code:
52
+ #
53
+ # require 'multimedia_paradise'; MultimediaParadise.copy_bin_directory
54
+ #
55
+ # ========================================================================= #
56
+ def self.copy_bin_directory
57
+ file_installed_binaries = "#{MultimediaParadise.project_base_directory?}"\
58
+ "yaml/internal/installed_binaries.yml"
59
+ mkdir('bin/')
60
+ cd('bin/')
61
+ if File.exist? file_installed_binaries
62
+ e 'Making use of the file '+sfile(file_installed_binaries)+'.'
63
+ install_these_binaries = YAML.load_file(file_installed_binaries)
64
+ installation_directory = `gem env`.split("\n").select {|line| # Check for INSTALLATION DIRECTORY: /root/.gem
65
+ line.include? 'INSTALLATION DIRECTORY:'
66
+ }.first.
67
+ scan(/INSTALLATION DIRECTORY: (.+)$/).flatten.first
68
+ install_these_binaries.each {|entry|
69
+ target = installation_directory+'/bin/'+entry
70
+ if File.exist? target
71
+ e "Copying the file #{target} into the current working "\
72
+ "directory next."
73
+ copy_file(target, Dir.pwd)
74
+ else
75
+ # ================================================================= #
76
+ # Since as of January 2023 first try the bin_dir, before giving up.
77
+ # ================================================================= #
78
+ target = MultimediaParadise.bin_dir?+File.basename(entry)
79
+ if File.exist? target
80
+ e "Copying the file #{target} into the current working "\
81
+ "directory next."
82
+ copy_file(target, Dir.pwd)
83
+ else
84
+ no_file_exists_at(target)
85
+ end
86
+ end
87
+ }
88
+ end
89
+ end
90
+
91
+ # ========================================================================= #
92
+ # === MultimediaParadise.return_video_codec_of_this_file
93
+ #
94
+ # This method will use ffmpeg's ffprobe and return the video codec of the
95
+ # given file at hand.
96
+ # ========================================================================= #
97
+ def self.return_video_codec_of_this_file(i = ARGV)
98
+ if i
99
+ if i.is_a? Array
100
+ i = i.first
101
+ end
102
+ cmd = "ffprobe -v error -select_streams v:0 -show_entries stream=codec_name -of default=noprint_wrappers=1:nokey=1 #{i}"
103
+ result = `#{cmd}`
104
+ return result
105
+ end
106
+ end; self.instance_eval { alias video_codec return_video_codec_of_this_file } # === MultimediaParadise.video_codec
107
+ self.instance_eval { alias video_codec? return_video_codec_of_this_file } # === MultimediaParadise.video_codec?
108
+
109
+ # ========================================================================= #
110
+ # === MultimediaParadise.to_gif
111
+ #
112
+ # This method can be used to quickly turn an .avi file or a .mp4
113
+ # file into an animated .gif.
114
+ #
115
+ # The quality is pretty low, so this is not ideal, but if you
116
+ # have a use case for animated .gif files then this is one
117
+ # way to do so.
118
+ #
119
+ # Usage example:
120
+ #
121
+ # MultimediaParadise.to_gif('foobar.mp4')
122
+ #
123
+ # ========================================================================= #
124
+ def self.to_gif(i)
125
+ [i].flatten.compact.each {|this_video_file|
126
+ _ = File.basename(this_video_file).
127
+ delete_suffix(
128
+ File.extname(this_video_file)
129
+ )
130
+ name_of_the_output_gif_file = "output_#{_}.gif"
131
+ cmd = "ffmpeg -i #{this_video_file} -pix_fmt rgb8 #{name_of_the_output_gif_file}" # Or use 0rgb.
132
+ esystem(cmd)
133
+ return name_of_the_output_gif_file
134
+ }
135
+ end
136
+
137
+ # ========================================================================= #
138
+ # === MultimediaParadise.create_noise_profile
139
+ #
140
+ # Note that this method depends on the external program called "sox".
141
+ #
142
+ # Usage example:
143
+ #
144
+ # MultimediaParadise.create_noise_profile('foobar.mp3')
145
+ #
146
+ # ========================================================================= #
147
+ def self.create_noise_profile(
148
+ of_this_mp3_file
149
+ )
150
+ array_results = []
151
+ [of_this_mp3_file].flatten.compact.each {|work_on_this_mp3_file|
152
+ work_on_this_mp3_file = File.absolute_path(work_on_this_mp3_file)
153
+ filename_without_extension = work_on_this_mp3_file.delete_suffix('.mp3')
154
+ this_profile_file_is_to_be_created = "noise_profile_based_on_#{File.basename(filename_without_extension)}"
155
+ if File.exist?(this_profile_file_is_to_be_created) # Always remove old profiles.
156
+ File.delete(this_profile_file_is_to_be_created)
157
+ end
158
+ cmd_to_run = "sox #{work_on_this_mp3_file} -n noiseprof #{this_profile_file_is_to_be_created}.profile"
159
+ e cmd_to_run
160
+ `#{cmd_to_run}`
161
+ array_results << this_profile_file_is_to_be_created
162
+ }
163
+ return array_results
164
+ end
165
+
166
+ # ========================================================================= #
167
+ # === MultimediaParadise.shrink_quality_of_these_mp3_files
168
+ #
169
+ # This method can be used to batch-shrink several .mp3 files in one go.
170
+ #
171
+ # Usage example:
172
+ #
173
+ # MultimediaParadise.shrink_quality_of_these_mp3_files(Dir['*.mp3'])
174
+ #
175
+ # ========================================================================= #
176
+ def self.shrink_quality_of_these_mp3_files(*array)
177
+ array.flatten.each {|this_mp3_file|
178
+ esystem "lame --mp3input -b 64 #{this_mp3_file}"
179
+ }
180
+ end
181
+
182
+ # ========================================================================= #
183
+ # === MultimediaParadise.report_how_many_real_videos_are_stored
184
+ #
185
+ # This method will report how many "real" videos are stored in the
186
+ # file video_collection.yml, which is distributed as part of
187
+ # this gem. You can, however had, also specify your own .yml
188
+ # file, as first argument to this method.
189
+ #
190
+ # To tap into this method, do issue:
191
+ #
192
+ # mpa --n-videos?
193
+ #
194
+ # ========================================================================= #
195
+ def self.report_how_many_real_videos_are_stored(
196
+ in_this_yaml_file = file_video_collection?
197
+ )
198
+ if File.exist? in_this_yaml_file
199
+ dataset = YAML.load_file(in_this_yaml_file)
200
+ e rev+
201
+ 'There are '+sfancy(dataset.keys.size)+' registered videos '+
202
+ 'in the file'
203
+ e '`'+sfile(in_this_yaml_file)+'`.'
204
+ else
205
+ e 'Please supply an existing yaml file (.yml) to this method.'
206
+ end
207
+ end
208
+
209
+ # ========================================================================= #
210
+ # === MultimediaParadise.return_the_mpeg_layer_from_this_mp3_file
211
+ # ========================================================================= #
212
+ def self.return_the_mpeg_layer_from_this_mp3_file(i)
213
+ cmd = "mpg123 -t #{i} 2>&1"
214
+ result = `#{cmd}`
215
+ splitted = result.split("\n")
216
+ selected = splitted.select {|line|
217
+ line.start_with? 'MPEG ' # Obtain the MPEG layer here.
218
+ }.uniq
219
+ result = selected.first # Pick the first result here.
220
+ # ======================================================================= #
221
+ # We may have to clean up that result a little bit still.
222
+ # ======================================================================= #
223
+ if result and result.include?(' ') and result.include?('III')
224
+ result = result[0 .. (result.index('III')+2)]
225
+ end
226
+ return result
227
+ end
228
+
229
+ # ========================================================================= #
230
+ # === MultimediaParadise.overlay_two_audio_files
231
+ #
232
+ # This method can merge in two audio files, usually .mp3 files.
233
+ #
234
+ # The functionality depends on "sox".
235
+ # ========================================================================= #
236
+ def self.overlay_two_audio_files(
237
+ first,
238
+ second
239
+ )
240
+ output_file = 'output_file.mp3'
241
+ esystem 'sox --combine mix '+
242
+ first.to_s+
243
+ ' '+
244
+ second.to_s+
245
+ ' '+
246
+ output_file
247
+ end
248
+
249
+ # ========================================================================= #
250
+ # === MultimediaParadise.samplerate_of_this_song
251
+ #
252
+ # Usage example:
253
+ #
254
+ # MultimediaParadise.samplerate_of_this_song('foobar.mp3') # => 44100
255
+ #
256
+ # ========================================================================= #
257
+ def self.samplerate_of_this_song(this_song)
258
+ if this_song.is_a? Array
259
+ this_song = this_song.first
260
+ end
261
+ result = `ffprobe -select_streams a -show_streams #{this_song} 2>&1`
262
+ if result
263
+ result = result.split("\n").select {|entry|
264
+ entry.start_with? 'sample_rate'
265
+ }.first.to_s
266
+ end
267
+ if result and result.include?('=')
268
+ result = result.split('=').last
269
+ end
270
+ return result
271
+ end
272
+
273
+ # ========================================================================= #
274
+ # === MultimediaParadise.bitrate_of_this_song
275
+ #
276
+ # This method will make use of ffmpeg's ffprobe to query the bitrate of
277
+ # e. g. a .mp3 song. The return value is a String such as "192 kb/s".
278
+ #
279
+ # The commandline variant would be the following:
280
+ #
281
+ # ffprobe foobar.mp3 2>&1 | grep Stream
282
+ #
283
+ # Result:
284
+ #
285
+ # Stream #0:0: Audio: mp3, 11025 Hz, mono, s16p, 32 kb/s
286
+ #
287
+ # Usage example:
288
+ #
289
+ # MultimediaParadise.bitrate_of_this_song('foobar.mp3') # => "192 kb"
290
+ #
291
+ # ========================================================================= #
292
+ def self.bitrate_of_this_song(this_song)
293
+ use_this_regex = /(\d{1,3} ?kb)\/s/ # See: https://rubular.com/r/CfmHoK5w0goYpU
294
+ if this_song.is_a? Array
295
+ this_song = this_song.first
296
+ end
297
+ result = `ffprobe #{this_song} 2>&1 | grep Stream`.strip
298
+ result = result.scan(use_this_regex).flatten.first.to_s
299
+ return result
300
+ end
301
+
302
+ # ========================================================================= #
303
+ # === MultimediaParadise.reverse_audio
304
+ #
305
+ # Reverse an audio file.
306
+ # ========================================================================= #
307
+ def self.reverse_audio(
308
+ this_file = 'output.wav'
309
+ )
310
+ output_file = 'reversed_'+this_file
311
+ cmd = 'sox -n '+this_file+' '+output_file+' reverse'
312
+ run_sys_command cmd
313
+ return output_file
314
+ end
315
+
316
+ # ========================================================================= #
317
+ # === MultimediaParadise.generate_sine_tone
318
+ #
319
+ # Generate a n-second sine tone.
320
+ # ========================================================================= #
321
+ def self.generate_sine_tone(
322
+ n_seconds = 5
323
+ )
324
+ output_file = 'output.wav'
325
+ esystem 'sox -n '+output_file+' synth '+n_seconds.to_s+' sine 440'
326
+ return output_file
327
+ end
328
+
329
+ # ========================================================================= #
330
+ # === MultimediaParadise.generate_sine_tone_sweep
331
+ #
332
+ # Generate a n-second sine tone sweep.
333
+ # ========================================================================= #
334
+ def self.generate_sine_tone_sweep(
335
+ n_seconds = 5
336
+ )
337
+ output_file = 'output.wav'
338
+ esystem 'sox -n '+output_file+' synth '+n_seconds.to_s+' sine 0:800'
339
+ return output_file
340
+ end
341
+
342
+ # ========================================================================= #
343
+ # === MultimediaParadise.repeat_this_song_n_times
344
+ #
345
+ # This will make use of sox and repeat a given song n times.
346
+ # ========================================================================= #
347
+ def self.repeat_this_song_n_times(
348
+ this_song = ARGV,
349
+ n_times = 3
350
+ )
351
+ [this_song].flatten.compact.each {|a_song|
352
+ if File.exist? a_song
353
+ esystem 'sox '+a_song+' repeated_song_'+a_song+' repeat '+n_times.to_s
354
+ else
355
+ no_file_exists_at(a_song)
356
+ end
357
+ }
358
+ end
359
+
360
+ # ========================================================================= #
361
+ # === MultimediaParadise.append_silence_to_this_song
362
+ #
363
+ # Note that this method currently depends on sox. Perhaps someone knows
364
+ # how to do so via ffmpeg; I don't right now.
365
+ #
366
+ # The first argument to this method is the delay, in n seconds. You can
367
+ # use a fraction, such as 0.25 here as well.
368
+ #
369
+ # The second argument is the song, such as "bla.mp3". This filename should
370
+ # exist.
371
+ #
372
+ # The method will then simply add "silence" to the song at hand - it
373
+ # will append "silence".
374
+ #
375
+ # Usage examples:
376
+ #
377
+ # MultimediaParadise.append_silence_to_this_song(5, 'foobar.mp3')
378
+ # MultimediaParadise.append_silence_to_this_song(5, 'ack.mp3')
379
+ # MultimediaParadise.append_silence_to_this_song(.027, 'foobar.mp3')
380
+ #
381
+ # ========================================================================= #
382
+ def self.append_silence_to_this_song(
383
+ add_this_delay = 5,
384
+ work_on_these_songs = ['foobar.mp3']
385
+ )
386
+ [work_on_these_songs].flatten.compact.each {|this_song|
387
+ if File.exist? this_song
388
+ cmd = "sox #{this_song} padded_#{this_song} pad 0 #{add_this_delay.to_f}"
389
+ esystem cmd
390
+ else
391
+ e 'No file exists at '+this_song.to_s+'.'
392
+ end
393
+ }
394
+ end
395
+
396
+ # ========================================================================= #
397
+ # === MultimediaParadise.overlay_this_image_onto_that_video
398
+ #
399
+ # This method can be used to overlay an image over an existing local
400
+ # video file.
401
+ #
402
+ # The options in use are as follows:
403
+ #
404
+ # overlay=25:25: The image will be positioned 25px to the right and
405
+ # 25px down, originating from the top left corner (0:0).
406
+ #
407
+ # enable='between(t,0,20)': The overlay image will be shown from
408
+ # 00:00:00 to 00:00:20
409
+ #
410
+ # ========================================================================= #
411
+ def self.overlay_this_image_onto_that_video(
412
+ this_image,
413
+ that_video,
414
+ duration_in_n_seconds = 3
415
+ )
416
+ _ = 'ffmpeg -y -i '+that_video+' '\
417
+ '-i '+this_image+' -filter_complex "[0:v][1:v] '\
418
+ 'overlay=80:80:enable=\'between(t,0,'+
419
+ duration_in_n_seconds.to_s+')\'" '\
420
+ '-pix_fmt yuv420p -c:a copy '\
421
+ 'output.mp4'
422
+ e
423
+ esystem _
424
+ e
425
+ end
426
+
427
+ # ========================================================================= #
428
+ # === MultimediaParadise.split_into_chunks_of_n_minutes
429
+ # ========================================================================= #
430
+ def self.split_into_chunks_of_n_minutes(
431
+ this_video = '*.mp4',
432
+ n_minutes = 5
433
+ )
434
+ this_video = this_video.first if this_video.is_a? Array
435
+ _ = 'ffmpeg -i '+this_video+' -c copy -map 0 -segment_time '\
436
+ '00:0'+n_minutes.to_s+':00 -f segment output%03d.mp4'
437
+ e
438
+ esystem _
439
+ e
440
+ end
441
+
442
+ # ========================================================================= #
443
+ # === MultimediaParadise.split_into_chunks_of_5_minutes
444
+ # ========================================================================= #
445
+ def self.split_into_chunks_of_5_minutes(
446
+ i = '*.mp4'
447
+ )
448
+ MultimediaParadise.split_into_chunks_of_n_minutes(i, 5)
449
+ end
450
+
451
+ # ========================================================================= #
452
+ # === MultimediaParadise.return_ss_mm_hh
453
+ #
454
+ # This may return a string such as "16:35:07".
455
+ # ========================================================================= #
456
+ def self.return_ss_mm_hh
457
+ ::Time.now.strftime('%H:%M:%S')
458
+ end
459
+
460
+ # ========================================================================= #
461
+ # === MultimediaParadise.to_ts
462
+ #
463
+ # This will batch-convert all .mp4 files into .ts files.
464
+ # ========================================================================= #
465
+ def self.to_ts
466
+ files = Dir['*.mp4']
467
+ files.each {|this_file|
468
+ cmd = 'ffmpeg -i '+this_file+' -c copy '\
469
+ '-bsf:v h264_mp4toannexb -f mpegts '+
470
+ this_file.sub(/#{File.extname(this_file)}$/,'')+
471
+ '.ts'
472
+
473
+ esystem cmd
474
+ }
475
+ end
476
+
477
+ # ========================================================================= #
478
+ # === MultimediaParadise.set_the_aspect_ratio_of_this_video
479
+ #
480
+ # This method will set the aspect-ratio of a given video file, via
481
+ # ffmpeg. The commandline-flag for this is called -aspect.
482
+ #
483
+ # Original example:
484
+ #
485
+ # ffmpeg -i foo.mp4 -aspect 16:9 bar.mp4
486
+ #
487
+ # Example for the MultimediaParadise namespace:
488
+ #
489
+ # MultimediaParadise.set_the_aspect_ratio_of_this_video('foo.mpg')
490
+ #
491
+ # ========================================================================= #
492
+ def self.set_the_aspect_ratio_of_this_video(
493
+ this_video, use_this_aspect_ratio = '16:9'
494
+ )
495
+ extension_to_use = File.extname(this_video)
496
+ cmd_to_use = 'ffmpeg -i '+this_video+' -aspect '+
497
+ use_this_aspect_ratio+' output_'+
498
+ this_video.sub(/#{extension_to_use}$/,'')+extension_to_use
499
+ e
500
+ e cmd_to_use
501
+ e
502
+ system cmd_to_use
503
+ end
504
+
505
+ # ========================================================================= #
506
+ # === MultimediaParadise.extract_images_from_this_video_file
507
+ #
508
+ # This variant uses the default rate, which is 25 images per second.
509
+ #
510
+ # Note that the method is named incorrectly; in the long run I may
511
+ # rename it (written in January 2020).
512
+ #
513
+ # Usage example:
514
+ #
515
+ # MultimediaParadise.extract_images_from_this_video_file('foobar.mp4')
516
+ #
517
+ # ========================================================================= #
518
+ def self.extract_images_from_this_video_file(*i)
519
+ i.flatten.each {|entry|
520
+ _extension_to_use = File.extname(entry) # No longer in use.
521
+ _ = 'ffmpeg '\
522
+ '-i '+entry+
523
+ ' %08d.png' #OUTPUT_'+entry.sub(/#{extension_to_use}$/,'')+
524
+ # extension_to_use+
525
+ e
526
+ esystem _
527
+ e
528
+ }
529
+ end
530
+
531
+ # ========================================================================= #
532
+ # === MultimediaParadise.extract_one_image_per_second
533
+ #
534
+ # This method will extract one-image-per-second from a given
535
+ # input video file.
536
+ #
537
+ # The first argument to this method should be the video file you
538
+ # are working with, such as 'foobar.mp4'.
539
+ # ========================================================================= #
540
+ def self.extract_one_image_per_second(
541
+ from_this_video_file = 'my_video.mpeg',
542
+ frame_size_to_use = '75x75'
543
+ )
544
+ extension_to_use = File.extname(from_this_video_file)
545
+ _ = 'ffmpeg -i '+from_this_video_file+
546
+ ' -r 1 '+ # ← Specify the video-frame here, aka "1 image per second".
547
+ '-s '+frame_size_to_use+
548
+ ' OUTPUT_'+from_this_video_file.sub(/#{extension_to_use}$/,'')+
549
+ extension_to_use
550
+ e
551
+ esystem _
552
+ e
553
+ end
554
+
555
+ # ========================================================================= #
556
+ # === MultimediaParadise.convert_these_images_into_a_video
557
+ #
558
+ # This method can be used to "assemble" images into a video.
559
+ # ========================================================================= #
560
+ def self.convert_these_images_into_a_video(*images)
561
+ _ = 'ffmpeg -f image2 -i '+images.join(' ').strip+' video.mpg'
562
+ esystem _
563
+ end
564
+
565
+ # ========================================================================= #
566
+ # === MultimediaParadise.high_quality_transcoding_of_this_video_file
567
+ #
568
+ # This method can be used for high-quality transcoding of a given
569
+ # input video file.
570
+ #
571
+ # - C:V is the video codec of choice
572
+ # - preset is the compression preset (in this case slow for higher
573
+ # quality compression)
574
+ # - CRF is the Constant Rate Factor, which preserves an overall
575
+ # level of quality throughout the file by adjusting each frame's
576
+ # bitrate based on the given quality level. The higher the CRF,
577
+ # the lower the overall quality level.
578
+ # - The video filter flag (-vf) is used to call FFMPEG's pre-bundled
579
+ # video filters.
580
+ # - yadif (Yet Another Deinterlacing Filter) deinterlaces an interlaced
581
+ # input, as progressive video is not only easier to compress and most
582
+ # current computer monitors and televisions are progressive scan.
583
+ # - When encoding with H.264/MPEG-4 AVC, the audio format used is AAC
584
+ # (Advanced Audio Coding); in order to enable FFMPEG's experimental,
585
+ # native AAC encoder.
586
+ # - -strict -2 needs to be added to the command.
587
+ # - An external library such as libfaac can also be used, and -strict
588
+ # -2 can be omitted.
589
+ #
590
+ # ========================================================================= #
591
+ def self.high_quality_transcoding_of_this_video_file(this_video_file)
592
+ output_file = 'OUTPUT_'+this_video_file.
593
+ sub(/#{File.extname(this_video_file)}$/,'')+'.mp4'
594
+ _ = 'ffmpeg -i '+this_video_file+' -c:v libx264 -preset slow '\
595
+ '-crf 18 -vf yadif -strict -2 '+
596
+ output_file
597
+ e
598
+ esystem _
599
+ e
600
+ end; self.instance_eval { alias high_quality_transcoding high_quality_transcoding_of_this_video_file } # === MultimediaParadise.high_quality_transcoding
601
+
602
+ # ========================================================================= #
603
+ # === MultimediaParadise.identify_video
604
+ #
605
+ # Use this to identify a given video or audio file.
606
+ #
607
+ # Another way would be this:
608
+ #
609
+ # mplayer -vo null -ao null -frames 0 -identify POP_America_TheLastUnicorn.mp3
610
+ #
611
+ # You will have to parse this lateron.
612
+ # ========================================================================= #
613
+ def self.identify_video(this_file)
614
+ cmd = "ffmpeg -i #{this_file}"
615
+ e
616
+ e cmd
617
+ e
618
+ result = `{cmd}`
619
+ return result
620
+ end
621
+
622
+ # ========================================================================= #
623
+ # === MultimediaParadise.show_available_audio_formats
624
+ #
625
+ # Use this method if you want to show the available (registered)
626
+ # audio formats.
627
+ # ========================================================================= #
628
+ def self.show_available_audio_formats
629
+ e; AUDIO_FORMATS.each {|entry|
630
+ e " - #{entry}" # It is ok to prepend the '-' since this is for display purposes.
631
+ }; e
632
+ end
633
+
634
+ # ========================================================================= #
635
+ # === MultimediaParadise.show_artist
636
+ #
637
+ # Use this for .mp3 files, via ffmpeg. It will show the artist.
638
+ #
639
+ # Input should be a .mp3 file.
640
+ #
641
+ # Usage example from within ruby code:
642
+ #
643
+ # MultimediaParadise.show_artist('/Depot/Audio/YoutubeMix_VivaldisFourSeasons.mp3')
644
+ # MultimediaParadise.show_artist('/Depot/Audio/Vivaldi_Concerto8.mp3')
645
+ #
646
+ # ========================================================================= #
647
+ def self.show_artist(
648
+ i, report_classname = true
649
+ )
650
+ result = MultimediaParadise.return_artist(i)
651
+ if result.empty?
652
+ opnn if report_classname
653
+ e 'This file does not have any artist-tag.'
654
+ else
655
+ opnn if report_classname
656
+ e "The artist is: #{simp(result)}"
657
+ end
658
+ end
659
+
660
+ # ========================================================================= #
661
+ # === MultimediaParadise.return_artist
662
+ # ========================================================================= #
663
+ def self.return_artist(
664
+ i,
665
+ report_classname = true
666
+ )
667
+ i = i.to_s
668
+ cmd =
669
+ 'ffprobe -loglevel error -show_entries format_tags=artist '\
670
+ '-of default=noprint_wrappers=1:nokey=1 '+i
671
+ result = `#{cmd}`.chomp
672
+ return result
673
+ end
674
+
675
+ # ========================================================================= #
676
+ # === MultimediaParadise.load_yaml
677
+ # ========================================================================= #
678
+ def self.load_yaml(i)
679
+ YAML.load_file(i)
680
+ end
681
+
682
+ # ========================================================================= #
683
+ # === MultimediaParadise.open_yaml_file
684
+ #
685
+ # This will open my video collection.
686
+ # ========================================================================= #
687
+ def self.open_yaml_file(
688
+ i = FILE_VIDEO_COLLECTION
689
+ )
690
+ esystem "bluefish #{i}"
691
+ end
692
+
693
+ # ========================================================================= #
694
+ # === MultimediaParadise.flip_video_to_the_left
695
+ #
696
+ # This toplevel-method can be used to flip a video to the left,
697
+ # that is, to rotate it by -90 degrees.
698
+ #
699
+ # It will tap into ffmpeg for this task.
700
+ # ========================================================================= #
701
+ def self.flip_video_to_the_left(i)
702
+ ffmpeg_flip_video_to_the_left(i)
703
+ end
704
+
705
+ # ========================================================================= #
706
+ # === MultimediaParadise.flip_video_to_the_right
707
+ #
708
+ # This toplevel-method can be used to flip a video to the right,
709
+ # that is, to rotate it by +90 degrees.
710
+ # ========================================================================= #
711
+ def self.flip_video_to_the_right(i)
712
+ ffmpeg_flip_video_to_the_right(i)
713
+ end
714
+
715
+ # ========================================================================= #
716
+ # === glob
717
+ # ========================================================================= #
718
+ def glob(i)
719
+ Dir[i]
720
+ end
721
+
722
+ # ========================================================================= #
723
+ # === video_collection?
724
+ # ========================================================================= #
725
+ def video_collection?
726
+ ::MultimediaParadise.file_video_collection
727
+ end
728
+
729
+ # ========================================================================= #
730
+ # === MultimediaParadise::PLAY_ZOOMED
731
+ #
732
+ # If this constant is set to true, then we will play in a zoomed way.
733
+ # ========================================================================= #
734
+ PLAY_ZOOMED = YAML.load_file(
735
+ "#{PROJECT_BASE_DIRECTORY}configuration/play_zoomed.yml"
736
+ )
737
+
738
+ # ========================================================================= #
739
+ # === MultimediaParadise.verbose_analyse_this_mp3_file_for_id3_tags
740
+ # ========================================================================= #
741
+ def self.verbose_analyse_this_mp3_file_for_id3_tags(this_mp3_file)
742
+ require 'id3lib'
743
+ if this_mp3_file.is_a? Array
744
+ this_mp3_file.each {|entry|
745
+ verbose_analyse_this_mp3_file_for_id3_tags(entry)
746
+ }
747
+ else
748
+ dataset = ID3Lib::Tag.new(this_mp3_file)
749
+ e "#{::Colours.rev}"\
750
+ "The .mp3 file is called: #{sfancy(this_mp3_file)}"
751
+ if dataset.empty?
752
+ e 'Has no id3-tags!'
753
+ else
754
+ # e 'Has some id3-tags.' # <- Dont report this in this case, though.
755
+ # ===================================================================== #
756
+ # Iterate through these entries next.
757
+ # ===================================================================== #
758
+ dataset.each_entry {|inner_hash|
759
+ value = inner_hash[:text].to_s
760
+ case inner_hash[:id]
761
+ # =================================================================== #
762
+ # === :TIT2
763
+ # =================================================================== #
764
+ when :TIT2
765
+ e 'The title is: '+
766
+ Colours.steelblue(value) unless value.empty?
767
+ # =================================================================== #
768
+ # === :TCON
769
+ #
770
+ # The genre.
771
+ # =================================================================== #
772
+ when :TCON
773
+ e 'The genre is: '+
774
+ Colours.steelblue(value) unless value.empty?
775
+ # =================================================================== #
776
+ # === :TYER
777
+ #
778
+ # The year.
779
+ # =================================================================== #
780
+ when :TYER
781
+ e 'The year is: '+
782
+ Colours.steelblue(value) unless value.empty?
783
+ # =================================================================== #
784
+ # === :TPE1
785
+ #
786
+ # The artist.
787
+ # =================================================================== #
788
+ when :TPE1
789
+ e 'The artist is: '+
790
+ Colours.steelblue(value) unless value.empty?
791
+ end
792
+ }
793
+ # pp dataset # <- for debugging.
794
+ end
795
+ end
796
+ end
797
+
798
+ # ========================================================================= #
799
+ # === MultimediaParadise.analyze_this_directory_for_id3_tags
800
+ # ========================================================================= #
801
+ def self.analyze_this_directory_for_id3_tags(i = Dir.pwd)
802
+ require 'id3lib'
803
+ all_files = Dir["#{i}/*.mp3"]
804
+ all_files.each {|this_mp3_file|
805
+ verbose_analyse_this_mp3_file_for_id3_tags(this_mp3_file)
806
+ }
807
+ end
808
+
809
+ # ========================================================================= #
810
+ # === MultimediaParadise.compress_this_ogg_file
811
+ #
812
+ # This method will use a sampling rate of 22050 Hz. This should be
813
+ # perfectly adequate for speech-related audio, but I also found it
814
+ # to work well when compressing a tetris.ogg file (for the game
815
+ # tetris).
816
+ # ========================================================================= #
817
+ def self.compress_this_ogg_file(i = 'foobar.ogg')
818
+ _ = 'ffmpeg -i '+entry+' '\
819
+ '-c:a libvorbis -ab 32k '\
820
+ '-ar 22050'\
821
+ ' OUTPUT_'+i
822
+ e
823
+ esystem _
824
+ e
825
+ end
826
+
827
+ # ========================================================================= #
828
+ # === MultimediaParadise.try_to_rename_kde_konsole_tab
829
+ # ========================================================================= #
830
+ def self.try_to_rename_kde_konsole_tab(
831
+ new_title = '_',
832
+ try_to_rename_the_kde_konsole_tab = TRY_TO_RENAME_THE_KDE_KONSOLE_TAB
833
+ )
834
+ if try_to_rename_the_kde_konsole_tab
835
+ begin
836
+ require 'roebe/requires/require_kde_konsole.rb'
837
+ rescue LoadError
838
+ e 'roebe/requires/require_kde_konsole.rb is not available'
839
+ end
840
+ if Object.const_defined?(:Roebe) and
841
+ Roebe.respond_to?(:rename_konsole)
842
+ Roebe.rename_konsole(new_title)
843
+ end
844
+ end
845
+ end; self.instance_eval { alias rename_kde_konsole try_to_rename_kde_konsole_tab } # === MultimediaParadise.rename_kde_konsole
846
+ self.instance_eval { alias do_rename_konsole try_to_rename_kde_konsole_tab } # === MultimediaParadise.do_rename_konsole
847
+ self.instance_eval { alias rename_kde_konsole_tab try_to_rename_kde_konsole_tab } # === MultimediaParadise.rename_kde_konsole_tab
848
+
849
+ # ========================================================================= #
850
+ # === MultimediaParadise.crop_this_video
851
+ #
852
+ # This method can be used to crop a particular video file, via
853
+ # ffmpeg.
854
+ #
855
+ # The raw ffmpeg-command would go like something like this:
856
+ #
857
+ # ffmpeg -i input.mp4 -filter:v "crop=out_w:out_h:x:y" output.mp4
858
+ #
859
+ # These variables are:
860
+ #
861
+ # → out_w is the width of the output rectangle
862
+ # → out_h is the height of the output rectangle
863
+ # → x and y specify the top left corner of the output rectangle
864
+ # → output.mp4 is the output file
865
+ #
866
+ # ========================================================================= #
867
+ def self.crop_this_video(
868
+ i = 'input.mp4',
869
+ output_file = 'output.mp4'
870
+ )
871
+ [i].flatten.compact.each {|this_video_file|
872
+ esystem 'ffmpeg -i '+
873
+ this_video_file.to_s+
874
+ ' -filter:v "crop=w:h:x:y" '+
875
+ output_file.to_s
876
+ }
877
+ end
878
+
879
+ end
880
+
881
+ if __FILE__ == $PROGRAM_NAME
882
+ include MultimediaParadise
883
+ map_number_to_videofile
884
+ puts 'The version of the MultimediaParadise project is: '+version?
885
+ end # rb shared.rb