mirror of
				https://github.com/amiaopensource/ffmprovisr.git
				synced 2025-10-25 15:10:52 +02:00 
			
		
		
		
	Compare commits
	
		
			42 Commits
		
	
	
		
			v2020-11-1
			...
			v2025-04-1
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | b8eeeb88b8 | ||
|  | bb4cc16b8e | ||
|  | 4c8f63b05f | ||
|  | f57e2d443a | ||
|  | fc2cd7f23e | ||
|  | e46bdc189a | ||
|  | 8ea516d0a0 | ||
|  | afe768d1df | ||
|  | f16406504d | ||
|  | 7629b26417 | ||
|  | f3d98bc712 | ||
|  | 8290b25fe5 | ||
|  | 53334033d0 | ||
|  | 86a04859a2 | ||
|  | dada53dff4 | ||
|  | f100b46233 | ||
|  | d1083e012c | ||
|  | 08a27055c3 | ||
|  | 63033942f1 | ||
|  | aba38e0a08 | ||
|  | 393a8cc22f | ||
|  | 20cbf26144 | ||
|  | e6552c5494 | ||
|  | 0d27ffa5d1 | ||
|  | bf93a20da8 | ||
|  | 3c393a688e | ||
|  | 961f401c34 | ||
|  | f3cc3b518f | ||
|  | 3555a9b61e | ||
|  | 9a36ef66ba | ||
|  | e7d8dd58b9 | ||
|  | a25c20a855 | ||
|  | c0d3761ce5 | ||
|  | 869ffda8c4 | ||
|  | 789329b415 | ||
|  | 23e0097b54 | ||
|  | ca86a0eca4 | ||
|  | bc0edae268 | ||
|  | 87314cef76 | ||
|  | 77a7a2b3dc | ||
|  | f0aab69f0f | ||
|  | 5c0b98bded | 
							
								
								
									
										
											BIN
										
									
								
								img/deinterlaced_video_frames.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/deinterlaced_video_frames.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 741 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/interlaced_video_fields.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/interlaced_video_fields.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 422 KiB | 
							
								
								
									
										128
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										128
									
								
								index.html
									
									
									
									
									
								
							| @@ -47,23 +47,24 @@ | |||||||
|       <h2 class="heading" id="about">About ffmprovisr</h2> |       <h2 class="heading" id="about">About ffmprovisr</h2> | ||||||
|       <h3>Making FFmpeg Easier</h3> |       <h3>Making FFmpeg Easier</h3> | ||||||
|       <p>FFmpeg is a powerful tool for manipulating audiovisual files. Unfortunately, it also has a steep learning curve, especially for users unfamiliar with a command line interface. This app helps users through the command generation process so that more people can reap the benefits of FFmpeg.</p> |       <p>FFmpeg is a powerful tool for manipulating audiovisual files. Unfortunately, it also has a steep learning curve, especially for users unfamiliar with a command line interface. This app helps users through the command generation process so that more people can reap the benefits of FFmpeg.</p> | ||||||
|       <p>Each button displays helpful information about how to perform a wide variety of tasks using FFmpeg. To use this site, click on the task you would like to perform. A new window will open up with a sample command and a description of how that command works. You can copy this command and understand how the command works with a breakdown of each of the flags.</p> |       <p>Each button displays helpful information about how to perform a wide variety of tasks using FFmpeg. To use this site, click on the task you would like to perform. You will jump to a single command or a list of related commands. Click on a command description, and the site will display a sample command as well as an explanation of how that command works with a breakdown of each of its flags (or options).</p> | ||||||
|       <p>This page does not have search functionality, but you can open all recipes (second option in the sidebar) and use your browser's search tool (often ctrl+f or cmd+f) to perform a keyword search through all recipes.</p> |       <p>This page does not have search functionality, but you can open all recipes (second option in the sidebar) and use your browser's search tool (often ctrl+f or cmd+f) to perform a keyword search through all recipes.</p> | ||||||
|       <h3>Tutorials</h3> |       <h3>Tutorials</h3> | ||||||
|       <p>For FFmpeg basics, check out the program’s <a href="https://ffmpeg.org/" target="_blank">official website</a>.</p> |       <p>For FFmpeg basics, check out the program’s <a href="https://ffmpeg.org/" target="_blank">official website</a>.</p> | ||||||
|       <p>For instructions on how to install FFmpeg on Mac, Linux, and Windows, refer to Reto Kromer’s <a href="https://avpres.net/FFmpeg/#ch1" target="_blank">installation instructions</a>.</p> |       <p>For instructions on how to install FFmpeg on macOS, Linux, and Windows, refer to Reto Kromer’s <a href="https://avpres.net/FFmpeg/#ch1" target="_blank">installation instructions</a>.</p> | ||||||
|       <p>For Bash and command line basics, try the <a href="https://learnpythonthehardway.org/book/appendixa.html" target="_blank">Command Line Crash Course</a>. For a little more context presented in an ffmprovisr style, try <a href="https://explainshell.com/" target="_blank">explainshell.com</a>!</p> |       <p>For Bash and command line basics, try the <a href="https://learnpythonthehardway.org/book/appendixa.html" target="_blank">Command Line Crash Course</a>. For a little more context presented in an ffmprovisr style, try <a href="https://explainshell.com/" target="_blank">explainshell.com</a>!</p> | ||||||
|       <h3>License</h3> |       <h3>License</h3> | ||||||
|       <p class="license"> |       <p class="license"> | ||||||
|         <a href="https://creativecommons.org/licenses/by/4.0/" target="_blank"><img alt="Creative Commons License" src="img/cc.png"></a><br> |         <a href="https://creativecommons.org/licenses/by/4.0/" target="_blank"><img alt="Creative Commons License" src="img/cc.png"></a><br> | ||||||
|         This work is licensed under a <a href="https://creativecommons.org/licenses/by/4.0/" target="_blank">Creative Commons Attribution 4.0 International License</a>. |         This work is licensed under a <a href="https://creativecommons.org/licenses/by/4.0/" target="_blank">Creative Commons Attribution 4.0 International License</a>. | ||||||
|       </p> |       </p> | ||||||
|       <h3>Sister projects</h3> |       <h3>Sibling projects</h3> | ||||||
|       <p><a href="https://dd388.github.io/crals/" target="_blank">Script Ahoy</a>: Community Resource for Archivists and Librarians Scripting</p> |       <p><a href="https://dd388.github.io/crals/" target="_blank">Script Ahoy</a>: Community Resource for Archivists and Librarians Scripting</p> | ||||||
|       <p><a href="https://datapraxis.github.io/sourcecaster/" target="_blank">The Sourcecaster</a>: an app that helps you use the command line to work through common challenges that come up when working with digital primary sources.</p> |       <p><a href="https://datapraxis.github.io/sourcecaster/" target="_blank">The Sourcecaster</a>: an app that helps you use the command line to work through common challenges that come up when working with digital primary sources.</p> | ||||||
|       <p><a href="https://pugetsoundandvision.github.io/micropops/" target="_blank">Micropops</a>: One liners and automation tools from Moving Image Preservation of Puget Sound</p> |       <p><a href="https://pugetsoundandvision.github.io/micropops/" target="_blank">Micropops</a>: One liners and automation tools from Moving Image Preservation of Puget Sound</p> | ||||||
|       <p><a href="https://amiaopensource.github.io/cable-bible/" target="_blank">Cable Bible</a>: A Guide to Cables and Connectors Used for Audiovisual Tech</p> |       <p><a href="https://amiaopensource.github.io/cable-bible/" target="_blank">Cable Bible</a>: A Guide to Cables and Connectors Used for Audiovisual Tech</p> | ||||||
|       <p><a href="https://eaasi.gitlab.io/qemu-qed/" target="_blank">QEMU QED</a>: instructions for using QEMU (Quick EMUlator), a command line application for computer emulation and virtualization</p> |       <p><a href="https://eaasi.gitlab.io/program_docs/qemu-qed/" target="_blank">QEMU QED</a>: instructions for using QEMU (Quick EMUlator), a command line application for computer emulation and virtualization</p> | ||||||
|  |       <p><a href="https://amiaopensource.github.io/ffmpeg-artschool/" target="_blank">ffmpeg-artschool</a>: An AMIA workshop featuring scripts, exercises, and activities to make art using FFmpeg</p> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|     <div class="well"> |     <div class="well"> | ||||||
| @@ -107,6 +108,18 @@ | |||||||
|     <div class="well"> |     <div class="well"> | ||||||
|  |  | ||||||
|     <h2 id="concepts">Learn about more advanced FFmpeg concepts</h2> |     <h2 id="concepts">Learn about more advanced FFmpeg concepts</h2> | ||||||
|  |     <!-- Loop usage explanation --> | ||||||
|  |     <label class="recipe" for="batch-loop">Batch and Loop script usage</label> | ||||||
|  |     <input type="checkbox" id="batch-loop"> | ||||||
|  |     <div class="hiding"> | ||||||
|  |       <h5>Batch and Loop script usage</h5> | ||||||
|  |       <p><code>ffmpeg -nostdin -i <em>input_file</em> ...</code></p> | ||||||
|  |       <p>One of the frequent uses of FFmpeg is to run batch commands within loops to, for example, generate access files for an entire collection at once.</p> | ||||||
|  |       <p>When running an FFmpeg command within a loop it is often necessary to use the <code>-nostdin</code> flag prior to the input in order to ensure successful execution of the commands. This is needed to override FFmpeg's default behavior of enabling interaction on standard input which can result in errors as loop inputs are fed to the ongoing command.</p> | ||||||
|  |       <p class="link"></p> | ||||||
|  |     </div> | ||||||
|  |     <!-- End loop usage explanation --> | ||||||
|  |  | ||||||
|     <!-- Codec Defaults explanation --> |     <!-- Codec Defaults explanation --> | ||||||
|     <label class="recipe" for="codec-defaults">Codec defaults</label> |     <label class="recipe" for="codec-defaults">Codec defaults</label> | ||||||
|     <input type="checkbox" id="codec-defaults"> |     <input type="checkbox" id="codec-defaults"> | ||||||
| @@ -212,7 +225,7 @@ | |||||||
|         The new container you are rewrapping to is defined by the filename extension used here, e.g. .mkv, .mp4, .mov.</dd> |         The new container you are rewrapping to is defined by the filename extension used here, e.g. .mkv, .mp4, .mov.</dd> | ||||||
|       </dl> |       </dl> | ||||||
|       <h4>Important caveat</h4> |       <h4>Important caveat</h4> | ||||||
|       <p>It may not be possible to rewrap a file's contents to a new container without re-encoding one or more of the streams within (that is, the video, audio, and subtitle tracks). Some containers can only contain streams of a certain encoding type: for example, the .mp4 container does not support uncompressed audio tracks. (In practice .mp4 goes hand-in-hand with a H.264-encoded video stream and an AAC-encoded video stream, although other types of video and audio streams are possible). Another example is that the Matroska container does not allow data tracks; see the <a href="#mkv-to-mp4">MKV to MP4 recipe</a>.</p> |       <p>It may not be possible to rewrap a file's contents to a new container without re-encoding one or more of the streams within (that is, the video, audio, and subtitle tracks). Some containers can only contain streams of a certain encoding type: for example, the .mp4 container does not support uncompressed audio tracks. (In practice .mp4 goes hand-in-hand with a H.264-encoded video stream and an AAC-encoded video stream, although other types of video and audio streams are possible). Another example is that the Matroska container does not allow data tracks.</p> | ||||||
|       <p>In such cases, FFmpeg will throw an error. If you encounter errors of this kind, you may wish to consult the <a href="#transcode">list of transcoding recipes</a>.</p> |       <p>In such cases, FFmpeg will throw an error. If you encounter errors of this kind, you may wish to consult the <a href="#transcode">list of transcoding recipes</a>.</p> | ||||||
|       <p class="link"></p> |       <p class="link"></p> | ||||||
|     </div> |     </div> | ||||||
| @@ -337,6 +350,38 @@ | |||||||
|     </div> |     </div> | ||||||
|     <!-- ends Transcode to H.264 --> |     <!-- ends Transcode to H.264 --> | ||||||
|  |  | ||||||
|  |     <!-- Transcode to H.264 or H.265 using the GPU --> | ||||||
|  |     <label class="recipe" for="transcode_gpu">Transcode to H.264/H.265 using the GPU</label> | ||||||
|  |     <input type="checkbox" id="transcode_gpu"> | ||||||
|  |     <div class="hiding"> | ||||||
|  |       <h5>Transcode to H.264/H.265 using the GPU</h5> | ||||||
|  |       <p><code>ffmpeg -i <em>input_file</em> -c:v h264_nvenc -preset llhq -rc:v vbr_hq -cq:v 19 -b:v 8000k -maxrate:v 12000k -profile:v high -c:a copy <em>output_file</em></code></p> | ||||||
|  |       <p>This command takes an input file and transcodes it to H.264 using the encoding functionality of an Nvidia GPU (without transcoding the audio). If you're using H.264 with AAC or AC3 audio, you can output to an .mp4 file; if you're using HEVC and/or more exotic audio, you should output to .mkv. While Nvidia's fixed-function hardware can be 10x as performant as encoding on the CPU, it requires a few more parameters in order to optimize quality at lower bitrates.</p> | ||||||
|  |       <dl> | ||||||
|  |         <dt>ffmpeg</dt><dd>starts the command</dd> | ||||||
|  |         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|  |         <dt>-c:v <em>h264_nvenc</em></dt><dd>tells FFmpeg to encode the video stream as H.264 using Nvidia's encoder.</dd> | ||||||
|  |         <dt>-preset <em>llhq</em></dt><dd>uses the "low latency, high quality" encoding preset, a good default when working with nvenc.</dd> | ||||||
|  |         <dt>-rc:v <em>vbr_hq</em></dt><dd>means "variable bitrate, high quality," allowing you to set a minimum and maximum bitrate for the encode.</dd> | ||||||
|  |         <dt>-cq:v <em>19</em></dt><dd>is the same as the CRF quality level specified using x264 or other CPU-based encoders, where 0 is lossless, 51 is the worst possible quality, and values from 18-23 are typical.</dd> | ||||||
|  |         <dt>-b:v <em>8000k -maxrate:v 12000k</em></dt><dd>corresponds to a minimum bitrate of 8 megabits (8000k) per second, and a maximum of 12 megabits per second. nvenc is not as good at estimating bitrates as CPU-based encoders, and without this data, will occasionally choose a visibly lower bitrate. The 8-12 mbit range is generally a good one for high-quality 1080p h264.</dd> | ||||||
|  |         <dt>-profile:v <em>high</em></dt><dd>uses the "high quality" profile of h264, something that's been baked in to the spec for a long time so that older players can declare compatibility; almost all h264 video now uses high.</dd> | ||||||
|  |         <dt>-c:a <em>copy</em></dt><dd>will skip reencoding the audio stream, and copy the audio from the source file.</dd> | ||||||
|  |         <dt><em>output_file</em></dt><dd>path, name and extension of the output file</dd> | ||||||
|  |       </dl> | ||||||
|  |       <p>In order to encode to HEVC instead, and optionally transcode the audio, you can try changing the command like this:</p> | ||||||
|  |       <p><code>ffmpeg -i <em>input_file</em> -c:v hevc_nvenc -preset llhq -rc:v vbr_hq -cq:v 19 -b:v 5000k -maxrate:v 8000k -profile:v main10 -c:a aac <em>output_file</em></code></p> | ||||||
|  |       <dl> | ||||||
|  |         <dt>-c:v <em>hevc_nvenc</em></dt><dd>encodes to HEVC (also called H.265), a more efficient codec supported on GPUs from approximately 2015 and newer.</dd> | ||||||
|  |         <dt>-b:v <em>5000k -maxrate:v 8000k</em></dt><dd>specifies a slightly lower bitrate than when using h264, per HEVC's greater efficiency.</dd> | ||||||
|  |         <dt>-profile:v <em>main10</em></dt><dd>declares the "main10" profile for working with HEVC; one of the primary advantages of this codec is better support for 10-bit video, enabling consumer HDR.</dd> | ||||||
|  |         <dt>-c:a <em>aac</em></dt><dd>reencodes the audio to AAC with default parameters, a very common and widely supported format for access copies.</dd> | ||||||
|  |       </dl> | ||||||
|  |       <p>Much of the information in this entry was taken from <a href="https://superuser.com/a/1236387" target="_blank">this superuser.com post</a> provided by an Nvidia developer, one of the best sources of information on the ffmpeg Nvidia encoders.</p> | ||||||
|  |       <p class="link"></p> | ||||||
|  |     </div> | ||||||
|  |     <!-- ends Transcode to H.264 or H.265 using the GPU --> | ||||||
|  |  | ||||||
|     <!-- H.264 from DCP --> |     <!-- H.264 from DCP --> | ||||||
|     <label class="recipe" for="dcp_to_h264">Transcode from DCP to an H.264 access file</label> |     <label class="recipe" for="dcp_to_h264">Transcode from DCP to an H.264 access file</label> | ||||||
|     <input type="checkbox" id="dcp_to_h264"> |     <input type="checkbox" id="dcp_to_h264"> | ||||||
| @@ -459,7 +504,6 @@ | |||||||
|       <h5>Transcode to Ogg/Theora</h5> |       <h5>Transcode to Ogg/Theora</h5> | ||||||
|       <p><code>ffmpeg -i <em>input_file</em> -acodec libvorbis -b:v 690k <em>output_file</em></code></p> |       <p><code>ffmpeg -i <em>input_file</em> -acodec libvorbis -b:v 690k <em>output_file</em></code></p> | ||||||
|       <p>This command takes an input file and transcodes it to Ogg/Theora in an .ogv wrapper with 690k video bitrate.</p> |       <p>This command takes an input file and transcodes it to Ogg/Theora in an .ogv wrapper with 690k video bitrate.</p> | ||||||
|       <p><strong>Note:</strong> FFmpeg must be installed with support for Ogg Theora. If you are using Homebrew, you can check with <code>brew info ffmpeg</code> and then update it with <code>brew upgrade ffmpeg --with-theora --with-libvorbis</code> if necessary.</p> |  | ||||||
|       <dl> |       <dl> | ||||||
|         <dt>ffmpeg</dt><dd>starts the command</dd> |         <dt>ffmpeg</dt><dd>starts the command</dd> | ||||||
|         <dt>-i <em>input file</em></dt><dd>path, name and extension of the input file</dd> |         <dt>-i <em>input file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
| @@ -636,22 +680,23 @@ | |||||||
|     </div> |     </div> | ||||||
|     <!-- ends SD to HD --> |     <!-- ends SD to HD --> | ||||||
|  |  | ||||||
|     <!-- Change display aspect ratio without re-encoding video--> |     <!-- Change display aspect ratio without re-encoding --> | ||||||
|     <label class="recipe" for="change_DAR">Change display aspect ratio without re-encoding</label> |     <label class="recipe" for="change_DAR">Change display aspect ratio without re-encoding</label> | ||||||
|     <input type="checkbox" id="change_DAR"> |     <input type="checkbox" id="change_DAR"> | ||||||
|     <div class="hiding"> |     <div class="hiding"> | ||||||
|       <h5>Change Display Aspect Ratio without re-encoding video</h5> |       <h5>Change Display Aspect Ratio without re-encoding</h5> | ||||||
|       <p><code>ffmpeg -i <em>input_file</em> -c:v copy -aspect 4:3 <em>output_file</em></code></p> |       <p><code>ffmpeg -i <em>input_file</em> -c:a copy -c:v copy -aspect 4:3 <em>output_file</em></code></p> | ||||||
|       <dl> |       <dl> | ||||||
|         <dt>ffmpeg</dt><dd>starts the command</dd> |         <dt>ffmpeg</dt><dd>starts the command</dd> | ||||||
|         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> |         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|  |         <dt>-c:a copy</dt><dd>Copy all mapped audio streams.</dd> | ||||||
|         <dt>-c:v copy</dt><dd>Copy all mapped video streams.</dd> |         <dt>-c:v copy</dt><dd>Copy all mapped video streams.</dd> | ||||||
|         <dt>-aspect 4:3</dt><dd>Change Display Aspect Ratio to <code>4:3</code>. Experiment with other aspect ratios such as <code>16:9</code>. If used together with <code>-c:v copy</code>, it will affect the aspect ratio stored at container level, but not the aspect ratio stored in encoded frames, if it exists.</dd> |         <dt>-aspect 4:3</dt><dd>Change Display Aspect Ratio to <code>4:3</code>. Experiment with other aspect ratios such as <code>16:9</code>. If used together with <code>-c:v copy</code>, it will affect the aspect ratio stored at container level, but not the aspect ratio stored in encoded frames, if it exists.</dd> | ||||||
|         <dt><em>output_file</em></dt><dd>path, name and extension of the output file</dd> |         <dt><em>output_file</em></dt><dd>path, name and extension of the output file</dd> | ||||||
|       </dl> |       </dl> | ||||||
|       <p class="link"></p> |       <p class="link"></p> | ||||||
|     </div> |     </div> | ||||||
|     <!-- ends Change display aspect ratio without re-encoding video --> |     <!-- ends Change display aspect ratio without re-encoding --> | ||||||
|  |  | ||||||
|     <!-- Convert colorspace --> |     <!-- Convert colorspace --> | ||||||
|     <label class="recipe" for="convert-colorspace">Convert colorspace of video</label> |     <label class="recipe" for="convert-colorspace">Convert colorspace of video</label> | ||||||
| @@ -723,7 +768,7 @@ | |||||||
|           <li>In the sound filter <code>atempo</code> the numerator <code>output_fps</code> sets the output speed and the denominator <code>input_fps</code> sets the input speed; both values are given in frames per second.</li> |           <li>In the sound filter <code>atempo</code> the numerator <code>output_fps</code> sets the output speed and the denominator <code>input_fps</code> sets the input speed; both values are given in frames per second.</li> | ||||||
|         </ul> |         </ul> | ||||||
|         The different filters in a complex filter can be divided either by comma or semicolon. The quotation marks allow to insert a space between the filters for readability.</dd> |         The different filters in a complex filter can be divided either by comma or semicolon. The quotation marks allow to insert a space between the filters for readability.</dd> | ||||||
|         <dt>-map "[v]"</dt><dd>maps the video stream and:</dd> |         <dt>-map "[v]"</dt><dd>maps the video stream and</dd> | ||||||
|         <dt>-map "[a]"</dt><dd>maps the audio stream together into:</dd> |         <dt>-map "[a]"</dt><dd>maps the audio stream together into:</dd> | ||||||
|         <dt><em>output_file</em></dt><dd>path, name and extension of the output file</dd> |         <dt><em>output_file</em></dt><dd>path, name and extension of the output file</dd> | ||||||
|       </dl> |       </dl> | ||||||
| @@ -1060,16 +1105,18 @@ | |||||||
|     <div class="hiding"> |     <div class="hiding"> | ||||||
|       <h5>Join files together</h5> |       <h5>Join files together</h5> | ||||||
|       <p><code>ffmpeg -f concat -i mylist.txt -c copy <em>output_file</em></code></p> |       <p><code>ffmpeg -f concat -i mylist.txt -c copy <em>output_file</em></code></p> | ||||||
|       <p>This command takes two or more files of the same file type and joins them together to make a single file. All that the program needs is a text file with a list specifying the files that should be joined. However, it only works properly if the files to be combined have the exact same codec and technical specifications. Be careful, FFmpeg may appear to have successfully joined two video files with different codecs, but may only bring over the audio from the second file or have other weird behaviors. Don’t use this command for joining files with different codecs and technical specs and always preview your resulting video file!</p> |       <p>This command takes two or more files of the same file type and joins them together to make a single file. All that the program needs is a text file with a list specifying the files that should be joined. If possible, run the command from the same directory where the files and the text file reside. Otherwise you'll have to use <code>-safe 0</code>, see below for more information. However, it only works properly if the files to be combined have the exact same codec and technical specifications. Be careful, FFmpeg may appear to have successfully joined two video files with different codecs, but may only bring over the audio from the second file or have other weird behaviors. Don’t use this command for joining files with different codecs and technical specs and always preview your resulting video file!</p> | ||||||
|       <dl> |       <dl> | ||||||
|         <dt>ffmpeg</dt><dd>starts the command</dd> |         <dt>ffmpeg</dt><dd>starts the command</dd> | ||||||
|         <dt>-f concat</dt><dd>forces ffmpeg to concatenate the files and to keep the same file format</dd> |         <dt>-f concat</dt><dd>forces ffmpeg to concatenate the files and to keep the same file format</dd> | ||||||
|         <dt>-i <em>mylist.txt</em></dt><dd>path, name and extension of the input file. Per the <a href="https://ffmpeg.org/ffmpeg-formats.html#Options" target="_blank">FFmpeg documentation</a>, it is preferable to specify relative rather than absolute file paths, as allowing absolute file paths may pose a security risk.<br> |         <dt>-i <em>mylist.txt</em></dt><dd>path, name and extension of the input file. Per the <a href="https://ffmpeg.org/ffmpeg-formats.html#Options" target="_blank">FFmpeg documentation</a>, it is preferable to specify relative rather than absolute file paths, as allowing absolute file paths may pose a security risk.<br> | ||||||
|         This text file contains the list of files to be concatenated and should be formatted as follows: |         This text file contains the list of files (without their absolute path) to be concatenated and should be formatted as follows: | ||||||
|         <pre>file '<em>./first_file.ext</em>' | <pre> | ||||||
|   file '<em>./second_file.ext</em>' |   file '<em>first_file.ext</em>' | ||||||
|  |   file '<em>second_file.ext</em>' | ||||||
|   . . . |   . . . | ||||||
|   file '<em>./last_file.ext</em>'</pre> |   file '<em>last_file.ext</em>' | ||||||
|  | </pre> | ||||||
|   In the above, <strong>file</strong> is simply the word "file". Straight apostrophes ('like this') rather than curved quotation marks (‘like this’) must be used to enclose the file paths.<br> |   In the above, <strong>file</strong> is simply the word "file". Straight apostrophes ('like this') rather than curved quotation marks (‘like this’) must be used to enclose the file paths.<br> | ||||||
|   <strong>Note:</strong> If specifying absolute file paths in the .txt file, add <code>-safe 0</code> before the input file.<br> |   <strong>Note:</strong> If specifying absolute file paths in the .txt file, add <code>-safe 0</code> before the input file.<br> | ||||||
|   e.g.: <code>ffmpeg -f concat -safe 0 -i mylist.txt -c copy <em>output_file</em></code></dd> |   e.g.: <code>ffmpeg -f concat -safe 0 -i mylist.txt -c copy <em>output_file</em></code></dd> | ||||||
| @@ -1363,6 +1410,43 @@ | |||||||
|     </div> |     </div> | ||||||
|     <!-- ends Deinterlace video --> |     <!-- ends Deinterlace video --> | ||||||
|  |  | ||||||
|  |     <!-- Deinterlace video fields --> | ||||||
|  |     <label class="recipe" for="deinterlace_fields">Deinterlace video fields to frames</label> | ||||||
|  |     <input type="checkbox" id="deinterlace_fields"> | ||||||
|  |     <div class="hiding"> | ||||||
|  |       <h5>Deinterlace video fields to frames</h5> | ||||||
|  |       <p><code>ffmpeg -i <em>input_file</em> -c:v libx264 -vf "idet,bwdif,format=yuv420p" <em>output_file</em></code></p> | ||||||
|  |       <p>This command takes an interlaced input file and outputs a deinterlaced H.264 MP4, with each field separated into its own frame.  This is preferred for interlaced video that contains a lot of motion, as the double-rate output preserves the visual cadence of the source material.</p> | ||||||
|  |       <dl> | ||||||
|  |         <dt>ffmpeg</dt><dd>starts the command</dd> | ||||||
|  |         <dt>-i <em>input file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|  |         <dt>-c:v libx264</dt><dd>tells FFmpeg to encode the video stream as H.264</dd> | ||||||
|  |         <dt>-vf</dt><dd>video filtering will be used (<code>-vf</code> is an alias of <code>-filter:v</code>)</dd> | ||||||
|  |         <dt>"</dt><dd>start of filtergraph (see below)</dd> | ||||||
|  |         <dt>idet</dt><dd>detect interlaced video field order<br> | ||||||
|  |          <a href="https://ffmpeg.org/ffmpeg-filters.html#idet" target="_blank">idet</a> will try to detect if the video is interlaced, and if so, what the order of the fields are (top-field-first, or bottom-field-first).  This is done to ensure the output of the deinterlacing filter is correct.</dd> | ||||||
|  |         <dt>bwdif</dt><dd>deinterlacing filter (‘Bob Weaver Deinterlacing Filter’)<br> | ||||||
|  |          By default, <a href="https://ffmpeg.org/ffmpeg-filters.html#bwdif-1" target="_blank">bwdif</a> will output one frame for each field, matching the visual cadence of interlaced video. </dd> | ||||||
|  |         <dt>,</dt><dd>separates filters</dd> | ||||||
|  |         <dt>format=yuv420p</dt><dd>chroma subsampling set to 4:2:0<br> | ||||||
|  |         By default, <code>libx264</code> will use a chroma subsampling scheme that is the closest match to that of the input. This can result in Y′C<sub>B</sub>C<sub>R</sub> 4:2:0, 4:2:2, or 4:4:4 chroma subsampling. QuickTime and most other non-FFmpeg based players can’t decode H.264 files that are not 4:2:0, therefore it’s advisable to specify 4:2:0 chroma subsampling.</dd> | ||||||
|  |         <dt>"</dt><dd>end of filtergraph</dd> | ||||||
|  |         <dt><em>output file</em></dt><dd>path, name and extension of the output file</dd> | ||||||
|  |       </dl> | ||||||
|  |       <p><code>"idet,bwdif,format=yuv420p"</code> is an FFmpeg <a href="https://trac.ffmpeg.org/wiki/FilteringGuide#FiltergraphChainFilterrelationship" target="_blank">filtergraph</a>. Here the filtergraph is made up of one filter chain, which is itself made up of the three filters (separated by the comma).<br> | ||||||
|  |       The enclosing quote marks are necessary when you use spaces within the filtergraph, e.g. <code>-vf "idet, bwdif, format=yuv420p"</code>, and are included above as an example of good practice.</p> | ||||||
|  |       <p><strong>Note:</strong> bwdif also supports the older method of outputting one frame for each frame (thereby halving the number of output frames per second) with the syntax <code>bwdif=mode=send_frame</code>.  This can be used when the presentation device is not capable of reproducing 50 (PAL) or 60 (NTSC) frames per second.</p> | ||||||
|  |       <p>For more H.264 encoding options, see the latter section of the <a href="#transcode_h264">encode H.264 command</a>.</p> | ||||||
|  |       <div class="sample-image"> | ||||||
|  |         <h2>Example</h2> | ||||||
|  |         <p>Before and after deinterlacing with bwdif:</p> | ||||||
|  |         <img src="img/interlaced_video_fields.png" alt="VLC screenshot of original interlaced video"> | ||||||
|  |         <img src="img/deinterlaced_video_frames.png" alt="VLC screenshot of deinterlaced video"> | ||||||
|  |       </div> | ||||||
|  |       <p class="link"></p> | ||||||
|  |     </div> | ||||||
|  |     <!-- ends Deinterlace video fields --> | ||||||
|  |  | ||||||
|     <!-- Inverse telecine --> |     <!-- Inverse telecine --> | ||||||
|     <label class="recipe" for="inverse-telecine">Inverse telecine</label> |     <label class="recipe" for="inverse-telecine">Inverse telecine</label> | ||||||
|     <input type="checkbox" id="inverse-telecine"> |     <input type="checkbox" id="inverse-telecine"> | ||||||
| @@ -1422,7 +1506,7 @@ | |||||||
|         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> |         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|         <dt>-filter:v idet</dt><dd>This calls the <a href="https://ffmpeg.org/ffmpeg-filters.html#idet" target="_blank">idet (detect video interlacing type) filter</a>.</dd> |         <dt>-filter:v idet</dt><dd>This calls the <a href="https://ffmpeg.org/ffmpeg-filters.html#idet" target="_blank">idet (detect video interlacing type) filter</a>.</dd> | ||||||
|         <dt>-f null</dt><dd>Video is decoded with the <code>null</code> muxer. This allows video decoding without creating an output file.</dd> |         <dt>-f null</dt><dd>Video is decoded with the <code>null</code> muxer. This allows video decoding without creating an output file.</dd> | ||||||
|         <dt>-</dt><dd>FFmpeg syntax requires a specified output, and <code>-</code> is just a place holder. No file is actually created.</dd> |         <dt>-</dt><dd>The FFmpeg syntax requires a specified output, and <code>-</code> prints the output to the screen (STDOUT), rather than creating a file.</dd> | ||||||
|       </dl> |       </dl> | ||||||
|       <p class="link"></p> |       <p class="link"></p> | ||||||
|     </div> |     </div> | ||||||
| @@ -1938,7 +2022,7 @@ | |||||||
|         <dt>ffmpeg</dt><dd>starts the command</dd> |         <dt>ffmpeg</dt><dd>starts the command</dd> | ||||||
|         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> |         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|         <dt>-f null</dt><dd>Video is decoded with the <code>null</code> muxer. This allows video decoding without creating an output file.</dd> |         <dt>-f null</dt><dd>Video is decoded with the <code>null</code> muxer. This allows video decoding without creating an output file.</dd> | ||||||
|         <dt>-</dt><dd>FFmpeg syntax requires a specified output, and <code>-</code> is just a place holder. No file is actually created.</dd> |         <dt>-</dt><dd>The FFmpeg syntax requires a specified output, and <code>-</code> prints the output to the screen (STDOUT), rather than creating a file.</dd> | ||||||
|       </dl> |       </dl> | ||||||
|       <p class="link"></p> |       <p class="link"></p> | ||||||
|     </div> |     </div> | ||||||
| @@ -1957,7 +2041,7 @@ | |||||||
|         <dt>-report</dt><dd>Dump full command line and console output to a file named <em>ffmpeg-YYYYMMDD-HHMMSS.log</em> in the current directory. It also implies <code>-loglevel verbose</code>.</dd> |         <dt>-report</dt><dd>Dump full command line and console output to a file named <em>ffmpeg-YYYYMMDD-HHMMSS.log</em> in the current directory. It also implies <code>-loglevel verbose</code>.</dd> | ||||||
|         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> |         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|         <dt>-f null</dt><dd>Video is decoded with the <code>null</code> muxer. This allows video decoding without creating an output file.</dd> |         <dt>-f null</dt><dd>Video is decoded with the <code>null</code> muxer. This allows video decoding without creating an output file.</dd> | ||||||
|         <dt>-</dt><dd>FFmpeg syntax requires a specified output, and <code>-</code> is just a place holder. No file is actually created.</dd> |         <dt>-</dt><dd>The FFmpeg syntax requires a specified output, and <code>-</code> prints the output to the screen (STDOUT), rather than creating a file.</dd> | ||||||
|       </dl> |       </dl> | ||||||
|       <p class="link"></p> |       <p class="link"></p> | ||||||
|     </div> |     </div> | ||||||
| @@ -2044,7 +2128,7 @@ | |||||||
|         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> |         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|         <dt>-map 0:v:0</dt><dd>designated the first video stream as the stream on which to perform this hash generation operation. <code>-map 0</code> can be used to run the operation on all streams.</dd> |         <dt>-map 0:v:0</dt><dd>designated the first video stream as the stream on which to perform this hash generation operation. <code>-map 0</code> can be used to run the operation on all streams.</dd> | ||||||
|         <dt>-f hash -hash md5</dt><dd>produce a checksum hash, and set the hash algorithm to md5. See the official <a href="https://ffmpeg.org/ffmpeg-formats.html#hash" target="_blank">documentation on hash</a> for other algorithms.</dd> |         <dt>-f hash -hash md5</dt><dd>produce a checksum hash, and set the hash algorithm to md5. See the official <a href="https://ffmpeg.org/ffmpeg-formats.html#hash" target="_blank">documentation on hash</a> for other algorithms.</dd> | ||||||
|         <dt>-</dt><dd>FFmpeg syntax requires a specified output, and <code>-</code> is just a place holder. No file is actually created.</dd> |         <dt>-</dt><dd>The FFmpeg syntax requires a specified output, and <code>-</code> prints the output to the screen (STDOUT), rather than creating a file.</dd> | ||||||
|       </dl> |       </dl> | ||||||
|       <p class="link"></p> |       <p class="link"></p> | ||||||
|     </div> |     </div> | ||||||
| @@ -2064,7 +2148,7 @@ | |||||||
|         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> |         <dt>-i <em>input_file</em></dt><dd>path, name and extension of the input file</dd> | ||||||
|         <dt>-map 0</dt><dd>map ALL streams from input file to output. If you omit this, ffmpeg chooses only the first "best" (*) stream: 1 for audio, 1 for video (not all streams).</dd> |         <dt>-map 0</dt><dd>map ALL streams from input file to output. If you omit this, ffmpeg chooses only the first "best" (*) stream: 1 for audio, 1 for video (not all streams).</dd> | ||||||
|         <dt>-f streamhash -hash md5</dt><dd>produce a checksum hash per-stream, and set the hash algorithm to md5. See the official <a href="https://www.ffmpeg.org/ffmpeg-formats.html#streamhash-1" target="_blank">documentation on streamhash</a> for other algorithms and more details.</dd> |         <dt>-f streamhash -hash md5</dt><dd>produce a checksum hash per-stream, and set the hash algorithm to md5. See the official <a href="https://www.ffmpeg.org/ffmpeg-formats.html#streamhash-1" target="_blank">documentation on streamhash</a> for other algorithms and more details.</dd> | ||||||
|         <dt>-</dt><dd>FFmpeg syntax requires a specified output, and <code>-</code> is just a place holder. No file is actually created. Choose an output filename to write the hashcode lines into a textfile.</dd> |         <dt>-</dt><dd>The FFmpeg syntax requires a specified output, and <code>-</code> prints the output to the screen (STDOUT), rather than creating a file. Choose an output filename to write the hashcode lines into a textfile.</dd> | ||||||
|         <dt>-v quiet</dt><dd>(Optional) Disables FFmpeg's processing output. With this option it's easier to see the text output of the hashes.</dd> |         <dt>-v quiet</dt><dd>(Optional) Disables FFmpeg's processing output. With this option it's easier to see the text output of the hashes.</dd> | ||||||
|       </dl> |       </dl> | ||||||
|       <p>The output looks like this, for example (1 video, 2 audio streams): |       <p>The output looks like this, for example (1 video, 2 audio streams): | ||||||
| @@ -2335,7 +2419,6 @@ | |||||||
|     <input type="checkbox" id="ocr_on_top"> |     <input type="checkbox" id="ocr_on_top"> | ||||||
|     <div class="hiding"> |     <div class="hiding"> | ||||||
|       <h5>Plays video with OCR on top</h5> |       <h5>Plays video with OCR on top</h5> | ||||||
|       <p>Note: ffmpeg must be compiled with the tesseract library for this script to work (<code>--with-tesseract</code> if using the <code>brew install ffmpeg</code> method).</p> |  | ||||||
|       <p><code>ffplay input_file -vf "ocr,drawtext=fontfile=/Library/Fonts/Andale Mono.ttf:text=%{metadata\\\:lavfi.ocr.text}:fontcolor=white"</code></p> |       <p><code>ffplay input_file -vf "ocr,drawtext=fontfile=/Library/Fonts/Andale Mono.ttf:text=%{metadata\\\:lavfi.ocr.text}:fontcolor=white"</code></p> | ||||||
|       <dl> |       <dl> | ||||||
|         <dt>ffplay</dt><dd>starts the command</dd> |         <dt>ffplay</dt><dd>starts the command</dd> | ||||||
| @@ -2359,7 +2442,6 @@ | |||||||
|     <input type="checkbox" id="ffprobe_ocr"> |     <input type="checkbox" id="ffprobe_ocr"> | ||||||
|     <div class="hiding"> |     <div class="hiding"> | ||||||
|       <h5>Exports OCR data to screen</h5> |       <h5>Exports OCR data to screen</h5> | ||||||
|       <p>Note: FFmpeg must be compiled with the tesseract library for this script to work (<code>--with-tesseract</code> if using the <code>brew install ffmpeg</code> method)</p> |  | ||||||
|       <p><code>ffprobe -show_entries frame_tags=lavfi.ocr.text -f lavfi -i "movie=<em>input_file</em>,ocr"</code></p> |       <p><code>ffprobe -show_entries frame_tags=lavfi.ocr.text -f lavfi -i "movie=<em>input_file</em>,ocr"</code></p> | ||||||
|       <dl> |       <dl> | ||||||
|         <dt>ffprobe</dt><dd>starts the command</dd> |         <dt>ffprobe</dt><dd>starts the command</dd> | ||||||
|   | |||||||
| @@ -173,7 +173,7 @@ Last updated: 2019-12-11 | |||||||
|  |  | ||||||
| [The Cable Bible](https://amiaopensource.github.io/cable-bible/): A Guide to Cables and Connectors Used for Audiovisual Tech   | [The Cable Bible](https://amiaopensource.github.io/cable-bible/): A Guide to Cables and Connectors Used for Audiovisual Tech   | ||||||
| [FFCommand_Engine](https://github.com/ColorlabMD/FFCommand_Engine): a tool for easier use of FFmpeg binaries   | [FFCommand_Engine](https://github.com/ColorlabMD/FFCommand_Engine): a tool for easier use of FFmpeg binaries   | ||||||
| [QEMU QED](https://eaasi.gitlab.io/qemu-qed): instructions for using QEMU (Quick EMUlator), a command line application for computer emulation and virtualization   | [QEMU QED](https://eaasi.gitlab.io/program_docs/qemu-qed/): instructions for using QEMU (Quick EMUlator), a command line application for computer emulation and virtualization   | ||||||
| [Script Ahoy](http://dd388.github.io/crals/): Community Resource for Archivists and Librarians Scripting   | [Script Ahoy](http://dd388.github.io/crals/): Community Resource for Archivists and Librarians Scripting   | ||||||
| [sourcecaster](https://datapraxis.github.io/sourcecaster/): helps you use the command line to work through common challenges that come up when working with digital primary sources. | [sourcecaster](https://datapraxis.github.io/sourcecaster/): helps you use the command line to work through common challenges that come up when working with digital primary sources. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -16,6 +16,10 @@ ffmpeg -i input_file -map 0 -dn -c:v ffv1 -level 3 -g 1 -slicecrc 1 -slices 16 - | |||||||
| ffmpeg -i concat:input_file_1\|input_file_2\|input_file_3 -c:v libx264 -c:a aac output_file.mp4 | ffmpeg -i concat:input_file_1\|input_file_2\|input_file_3 -c:v libx264 -c:a aac output_file.mp4 | ||||||
| # Transcode to an H.265/HEVC MP4 | # Transcode to an H.265/HEVC MP4 | ||||||
| ffmpeg -i input_file -c:v libx265 -pix_fmt yuv420p -c:a copy output_file | ffmpeg -i input_file -c:v libx265 -pix_fmt yuv420p -c:a copy output_file | ||||||
|  | # Transcode to H.264 using the GPU | ||||||
|  | ffmpeg -i input_file -c:v h264_nvenc -preset llhq -rc:v vbr_hq -cq:v 19 -b:v 8000k -maxrate:v 12000k -profile:v high -c:a copy output_file | ||||||
|  | # Transcode to H.265 using the GPU | ||||||
|  | ffmpeg -i input_file -c:v hevc_nvenc -preset llhq -rc:v vbr_hq -cq:v 19 -b:v 5000k -maxrate:v 8000k -profile:v main10 -c:a copy output_file | ||||||
| # Transcode to an Ogg Theora | # Transcode to an Ogg Theora | ||||||
| ffmpeg -i input_file -acodec libvorbis -b:v 690k output_file | ffmpeg -i input_file -acodec libvorbis -b:v 690k output_file | ||||||
| # Convert WAV to MP3 | # Convert WAV to MP3 | ||||||
| @@ -33,7 +37,7 @@ ffmpeg -i input_file -filter:v "hflip,vflip" -c:a copy output_file | |||||||
| # Transform SD to HD with pillarbox | # Transform SD to HD with pillarbox | ||||||
| ffmpeg -i input_file -filter:v "colormatrix=bt601:bt709, scale=1440:1080:flags=lanczos, pad=1920:1080:240:0" -c:a copy output_file | ffmpeg -i input_file -filter:v "colormatrix=bt601:bt709, scale=1440:1080:flags=lanczos, pad=1920:1080:240:0" -c:a copy output_file | ||||||
| # Change display aspect ratio without re-encoding | # Change display aspect ratio without re-encoding | ||||||
| ffmpeg -i input_file -c:v copy -aspect 4:3 output_file | ffmpeg -i input_file -c:a copy -c:v copy -aspect 4:3 output_file | ||||||
| # Convert colorspace of video | # Convert colorspace of video | ||||||
| ffmpeg -i input_file -c:v libx264 -vf colormatrix=src:dst output_file | ffmpeg -i input_file -c:v libx264 -vf colormatrix=src:dst output_file | ||||||
| # Modify image and sound speed | # Modify image and sound speed | ||||||
| @@ -88,6 +92,8 @@ ffmpeg -i input_file -af areverse,silenceremove=start_threshold=-57dB:start_dura | |||||||
| ffmpeg -i input_file -c:v libx264 -filter:v "yadif, scale=1440:1080:flags=lanczos, pad=1920:1080:(ow-iw)/2:(oh-ih)/2, format=yuv420p" output_file | ffmpeg -i input_file -c:v libx264 -filter:v "yadif, scale=1440:1080:flags=lanczos, pad=1920:1080:(ow-iw)/2:(oh-ih)/2, format=yuv420p" output_file | ||||||
| # Deinterlace video | # Deinterlace video | ||||||
| ffmpeg -i input_file -c:v libx264 -vf "yadif,format=yuv420p" output_file | ffmpeg -i input_file -c:v libx264 -vf "yadif,format=yuv420p" output_file | ||||||
|  | # Deinterlace video fields to frames | ||||||
|  | ffmpeg -i input_file -c:v libx264 -vf "idet,bwdif,format=yuv420p" output_file | ||||||
| # Inverse telecine | # Inverse telecine | ||||||
| ffmpeg -i input_file -c:v libx264 -vf "fieldmatch,yadif,decimate" output_file | ffmpeg -i input_file -c:v libx264 -vf "fieldmatch,yadif,decimate" output_file | ||||||
| # Set field order for interlaced video | # Set field order for interlaced video | ||||||
|   | |||||||
| @@ -4,17 +4,14 @@ | |||||||
| # connected to the Web, and the local version otherwise. | # connected to the Web, and the local version otherwise. | ||||||
|  |  | ||||||
| if [[ "$(uname -s)" = "Darwin" ]] ; then | if [[ "$(uname -s)" = "Darwin" ]] ; then | ||||||
|     default_browser=$(plutil -convert json ~/Library/Preferences/com.apple.LaunchServices/com.apple.launchservices.secure.plist -r -o - | grep https -b1 | tail -n1 | cut -d'"' -f4) |  | ||||||
|     if ping -c 1 amiaopensource.github.io >/dev/null 2>&1 ; then |     if ping -c 1 amiaopensource.github.io >/dev/null 2>&1 ; then | ||||||
|         ffmprovisr_path='https://amiaopensource.github.io/ffmprovisr/' |         ffmprovisr_path='https://amiaopensource.github.io/ffmprovisr/' | ||||||
|  |     elif [[ -d /usr/local/Cellar/ffmprovisr ]] ; then | ||||||
|  |         ffmprovisr_path=$(find /usr/local/Cellar/ffmprovisr -name 'index.html' | sort -M | tail -n1) | ||||||
|     else |     else | ||||||
|         ffmprovisr_path=$(find /usr/local/Cellar/ffmprovisr -iname 'index.html' | sort -M | tail -n1) |         ffmprovisr_path=$(find /opt/homebrew/Cellar/ffmprovisr -name 'index.html' | sort -M | tail -n1) | ||||||
|     fi |  | ||||||
|     if [[ -n "${default_browser}" ]] ; then |  | ||||||
|         open -b "${default_browser}" "${ffmprovisr_path}" |  | ||||||
|     else |  | ||||||
|         open "${ffmprovisr_path}" |  | ||||||
|     fi |     fi | ||||||
|  |     open "${ffmprovisr_path}" | ||||||
| elif [[ "$(uname -s)" = "Linux" ]] ; then | elif [[ "$(uname -s)" = "Linux" ]] ; then | ||||||
|     if ping -c 1 amiaopensource.github.io >/dev/null 2>&1 ; then |     if ping -c 1 amiaopensource.github.io >/dev/null 2>&1 ; then | ||||||
|         ffmprovisr_path='https://amiaopensource.github.io/ffmprovisr/' |         ffmprovisr_path='https://amiaopensource.github.io/ffmprovisr/' | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user