/* QueryGenerator.cs $ This file is part of the HandBrake source code. Homepage: . It may be used under the terms of the GNU General Public License. */ namespace Handbrake.Functions { using System; using System.Collections.Generic; using System.Globalization; using System.IO; using System.Linq; using System.Windows.Forms; using HandBrake.ApplicationServices; using HandBrake.ApplicationServices.Functions; using HandBrake.ApplicationServices.Model.Encoding; using HandBrake.ApplicationServices.Services; using HandBrake.ApplicationServices.Services.Interfaces; using Handbrake.Model; using UserSettingConstants = Handbrake.UserSettingConstants; /// /// Generate a CLI Query for HandBrakeCLI /// public class QueryGenerator { private static readonly IUserSettingService UserSettingService = ServiceManager.UserSettingService; /// /// The Culture /// private static readonly CultureInfo Culture = new CultureInfo("en-US", false); public static string GenerateQueryForPreset(frmMain mainWindow, QueryPictureSettingsMode mode, bool filters, int width, int height) { string query = string.Empty; query += GenerateTabbedComponentsQuery(mainWindow, filters, mode, width, height); return query; } public static string GeneratePreviewQuery(frmMain mainWindow, int duration, string preview) { string query = string.Empty; query += SourceQuery(mainWindow, 3, duration, preview); query += DestinationQuery(mainWindow, QueryEncodeMode.Preview); query += GenerateTabbedComponentsQuery(mainWindow, true, QueryPictureSettingsMode.UserInterfaceSettings, 0, 0); return query; } public static string GenerateFullQuery(frmMain mainWindow) { string query = string.Empty; query += SourceQuery(mainWindow, mainWindow.drop_mode.SelectedIndex, 0, null); query += DestinationQuery(mainWindow, QueryEncodeMode.Standard); query += GenerateTabbedComponentsQuery(mainWindow, true, QueryPictureSettingsMode.UserInterfaceSettings, 0, 0); return query; } #region Individual Query Sections private static string GenerateTabbedComponentsQuery(frmMain mainWindow, bool filters, QueryPictureSettingsMode mode, int width, int height) { string query = string.Empty; // Output Settings query += OutputSettingsQuery(mainWindow); // Filters Panel if (filters) query += FiltersQuery(mainWindow); // Picture Settings query += PictureSettingsQuery(mainWindow, mode, width, height); // Video Settings query += VideoSettingsQuery(mainWindow); // Audio Settings query += AudioSettingsQuery(mainWindow); // Subtitles Panel query += mainWindow.Subtitles.GetCliQuery; // Chapter Markers query += ChapterMarkersQuery(mainWindow); // X264 Panel query += X264Query(mainWindow); // Extra Settings query += ExtraSettings(); return query; } private static string SourceQuery(frmMain mainWindow, int mode, int duration, string preview) { string query = string.Empty; string sourcePath = string.Empty; sourcePath = mainWindow.selectedTitle != null && File.Exists(mainWindow.selectedTitle.SourceName) ? mainWindow.selectedTitle.SourceName.Trim() : mainWindow.sourcePath; if (!string.IsNullOrEmpty(mainWindow.sourcePath) && mainWindow.sourcePath.Trim() != "Select \"Source\" to continue") { if (mainWindow.sourcePath.EndsWith("\\")) { query = " -i " + sourcePath; } else { query = " -i " + '"' + sourcePath + '"'; } } if (mainWindow.drp_dvdtitle.Text != string.Empty) { string[] titleInfo = mainWindow.drp_dvdtitle.Text.Split(' '); query += " -t " + titleInfo[0]; } if (!UserSettingService.GetUserSetting(ASUserSettingConstants.DisableLibDvdNav) && mainWindow.drop_angle.Items.Count != 0) query += " --angle " + mainWindow.drop_angle.SelectedItem; // Decide what part of the video we want to encode. switch (mode) { case 0: // Chapters if (mainWindow.drop_chapterFinish.Text == mainWindow.drop_chapterStart.Text && mainWindow.drop_chapterStart.Text != string.Empty) query += string.Format(" -c {0}", mainWindow.drop_chapterStart.Text); else if (mainWindow.drop_chapterStart.Text != string.Empty && mainWindow.drop_chapterFinish.Text != string.Empty) query += string.Format(" -c {0}-{1}", mainWindow.drop_chapterStart.Text, mainWindow.drop_chapterFinish.Text); break; case 1: // Seconds int start, end; int.TryParse(mainWindow.drop_chapterStart.Text, out start); int.TryParse(mainWindow.drop_chapterFinish.Text, out end); int calculatedDuration = end - start; query += string.Format(" --start-at duration:{0} --stop-at duration:{1}", mainWindow.drop_chapterStart.Text, calculatedDuration); break; case 2: // Frames int.TryParse(mainWindow.drop_chapterStart.Text, out start); int.TryParse(mainWindow.drop_chapterFinish.Text, out end); calculatedDuration = end - start; query += string.Format(" --start-at frame:{0} --stop-at frame:{1}", mainWindow.drop_chapterStart.Text, calculatedDuration); break; case 3: // Preview query += " --previews " + UserSettingService.GetUserSetting(UserSettingConstants.PreviewScanCount) + " "; query += " --start-at-preview " + preview; query += " --stop-at duration:" + duration + " "; break; default: break; } return query; } private static string DestinationQuery(frmMain mainWindow, QueryEncodeMode mode) { string query = string.Empty; if (string.IsNullOrEmpty(mainWindow.text_destination.Text)) { return string.Empty; } if (mode != QueryEncodeMode.Preview) query += string.Format(" -o \"{0}\" ", mainWindow.text_destination.Text); else { if (mainWindow.text_destination.Text != string.Empty) query += string.Format(" -o \"{0}\" ", mainWindow.text_destination.Text.Replace(".m", "_sample.m")); } return query; } private static string OutputSettingsQuery(frmMain mainWindow) { string query = string.Empty; query += " -f " + mainWindow.drop_format.Text.ToLower().Replace(" file", string.Empty); // These are output settings features if (mainWindow.check_largeFile.Checked) query += " -4 "; if (mainWindow.check_iPodAtom.Checked) query += " -I "; if (mainWindow.check_optimiseMP4.Checked) query += " -O "; return query; } private static string PictureSettingsQuery(frmMain mainWindow, QueryPictureSettingsMode mode, int width, int height) { string query = string.Empty; if (mode == QueryPictureSettingsMode.UserInterfaceSettings) { if (mainWindow.PictureSettings.text_width.Value != 0) if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex != 1) // Prevent usage for strict anamorphic query += " -w " + mainWindow.PictureSettings.text_width.Text; if (mainWindow.PictureSettings.text_height.Value != 0 && mainWindow.PictureSettings.text_height.Text != string.Empty) if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 0 || mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 3) // Prevent usage for strict anamorphic query += " -l " + mainWindow.PictureSettings.text_height.Text; } else if (mode == QueryPictureSettingsMode.Custom) // For Add Preset Only. { query += " -X " + width; query += " -Y " + height; } else if (mode == QueryPictureSettingsMode.SourceMaximum) // For Add Preset Only. { if (mainWindow.PictureSettings.text_width.Value != 0) if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex != 1) // Prevent usage for strict anamorphic query += " -X " + mainWindow.PictureSettings.text_width.Text; if (mainWindow.PictureSettings.text_height.Value != 0 && mainWindow.PictureSettings.text_height.Text != string.Empty) if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 0 || mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 3) // Prevent usage for strict anamorphic query += " -Y " + mainWindow.PictureSettings.text_height.Text; } string cropTop = mainWindow.PictureSettings.crop_top.Text; string cropBottom = mainWindow.PictureSettings.crop_bottom.Text; string cropLeft = mainWindow.PictureSettings.crop_left.Text; string cropRight = mainWindow.PictureSettings.crop_right.Text; if (mainWindow.PictureSettings.check_customCrop.Checked && mode != QueryPictureSettingsMode.None) { if (mainWindow.PictureSettings.crop_top.Text == string.Empty) cropTop = "0"; if (mainWindow.PictureSettings.crop_bottom.Text == string.Empty) cropBottom = "0"; if (mainWindow.PictureSettings.crop_left.Text == string.Empty) cropLeft = "0"; if (mainWindow.PictureSettings.crop_right.Text == string.Empty) cropRight = "0"; query += " --crop " + cropTop + ":" + cropBottom + ":" + cropLeft + ":" + cropRight; } switch (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex) { case 0: if (mainWindow.PictureSettings.drp_modulus.SelectedIndex != 0) query += " --modulus " + mainWindow.PictureSettings.drp_modulus.SelectedItem; break; case 1: query += " --strict-anamorphic "; break; case 2: query += " --loose-anamorphic "; if (mainWindow.PictureSettings.drp_modulus.SelectedIndex != 0) query += " --modulus " + mainWindow.PictureSettings.drp_modulus.SelectedItem; break; case 3: query += " --custom-anamorphic "; if (mainWindow.PictureSettings.drp_modulus.SelectedIndex != 0) query += " --modulus " + mainWindow.PictureSettings.drp_modulus.SelectedItem; if (mainWindow.PictureSettings.check_KeepAR.Checked) query += " --display-width " + mainWindow.PictureSettings.updownDisplayWidth.Text + " "; if (mainWindow.PictureSettings.check_KeepAR.Checked) query += " --keep-display-aspect "; if (!mainWindow.PictureSettings.check_KeepAR.Checked) if (mainWindow.PictureSettings.updownParWidth.Text != string.Empty && mainWindow.PictureSettings.updownParHeight.Text != string.Empty) query += " --pixel-aspect " + mainWindow.PictureSettings.updownParWidth.Text + ":" + mainWindow.PictureSettings.updownParHeight.Text + " "; break; } return query; } private static string FiltersQuery(frmMain mainWindow) { return mainWindow.Filters.GetCliQuery; } private static string VideoSettingsQuery(frmMain mainWindow) { string query = string.Empty; switch (mainWindow.drp_videoEncoder.Text) { case "MPEG-4 (FFmpeg)": query += " -e ffmpeg"; break; case "MPEG-2 (FFmpeg)": query += " -e ffmpeg2"; break; case "H.264 (x264)": query += " -e x264"; break; case "VP3 (Theora)": query += " -e theora"; break; default: query += " -e x264"; break; } // Video Settings if (mainWindow.radio_avgBitrate.Checked) query += " -b " + mainWindow.text_bitrate.Text; // Video Quality Setting if (mainWindow.radio_cq.Checked) { double cqStep = UserSettingService.GetUserSetting(ASUserSettingConstants.X264Step); double value; switch (mainWindow.drp_videoEncoder.Text) { case "MPEG-4 (FFmpeg)": case "MPEG-2 (FFmpeg)": value = 31 - (mainWindow.slider_videoQuality.Value - 1); query += " -q " + value.ToString(new CultureInfo("en-US")); break; case "H.264 (x264)": CultureInfo culture = CultureInfo.CreateSpecificCulture("en-US"); value = 51 - (mainWindow.slider_videoQuality.Value * cqStep); value = Math.Round(value, 2); query += " -q " + value.ToString(culture); break; case "VP3 (Theora)": value = mainWindow.slider_videoQuality.Value; query += " -q " + value.ToString(new CultureInfo("en-US")); break; } } if (mainWindow.check_2PassEncode.Checked) query += " -2 "; if (mainWindow.check_turbo.Checked) query += " -T "; if (mainWindow.drp_videoFramerate.Text != "Same as source") query += " -r " + mainWindow.drp_videoFramerate.Text; if (mainWindow.drp_videoFramerate.SelectedIndex == 0) { // If we use Same as Source, we can either output CFR or VFR query += mainWindow.radio_constantFramerate.Checked ? " --cfr " : " --vfr "; } else { // We have a hard framerate set, so we can either be Constant or peak (VFR) framerate query += mainWindow.radio_constantFramerate.Checked ? " --cfr " : " --pfr "; } return query; } private static string AudioSettingsQuery(frmMain mainWindow) { // Queries for each option string tracks = string.Empty; string encoders = string.Empty; string mixdowns = string.Empty; string samplerates = string.Empty; string bitrates = string.Empty; string drvValues = string.Empty; string gainValues = string.Empty; string trackNames = string.Empty; // If we have no audio tracks, set the query to none if (mainWindow.AudioSettings.AudioTracks.ToList().Count == 0) { return " -a none"; } // Generate the sub queries foreach (AudioTrack audioTrack in mainWindow.AudioSettings.AudioTracks) { // Audio Track (-a) string track = audioTrack.Track.HasValue ? audioTrack.Track.ToString() : "1"; // Default to "1" tracks += string.IsNullOrEmpty(tracks) ? track : string.Format(",{0}", track); // Audio Encoder (-E) encoders += string.IsNullOrEmpty(encoders) ? Converters.GetCliAudioEncoder(audioTrack.Encoder) : string.Format(",{0}", Converters.GetCliAudioEncoder(audioTrack.Encoder)); // Audio Mixdowns (-6) mixdowns += string.IsNullOrEmpty(mixdowns) ? Converters.GetCliMixDown(audioTrack.MixDown) : string.Format(",{0}", Converters.GetCliMixDown(audioTrack.MixDown)); // Audio Samplerates (-R) string rate = audioTrack.SampleRate == 0 ? "Auto" : audioTrack.SampleRate.ToString(); // Default to "Auto" samplerates += string.IsNullOrEmpty(samplerates) ? rate : string.Format(",{0}", rate); // Audio Bitrates (-B) bitrates += string.IsNullOrEmpty(bitrates) ? audioTrack.Bitrate.ToString(Culture) : string.Format(",{0}", audioTrack.Bitrate); // Audio DRC Values drvValues += string.IsNullOrEmpty(drvValues) ? audioTrack.DRC.ToString(Culture) : string.Format(",{0}", audioTrack.DRC.ToString(Culture)); // Audio Gain Control gainValues += string.IsNullOrEmpty(gainValues) ? audioTrack.Gain.ToString(Culture) : string.Format(",{0}", audioTrack.Gain.ToString(Culture)); trackNames += string.IsNullOrEmpty(trackNames) ? string.IsNullOrEmpty(audioTrack.TrackName) ? "\"\"" : string.Format("\"{0}\"", audioTrack.TrackName.Trim()) : string.IsNullOrEmpty(audioTrack.TrackName) ? ",\"\"" : string.Format(",\"{0}\"", audioTrack.TrackName.Trim()); } string audioQuery = string.Format( " -a {0} -E {1} -B {2} -6 {3} -R {4} -D {5} --gain={6}", tracks, encoders, bitrates, mixdowns, samplerates, drvValues, gainValues); if (!string.IsNullOrEmpty(trackNames.Trim()) && !string.IsNullOrEmpty(trackNames.Replace("\"", string.Empty).Replace(",", string.Empty).Trim())) { audioQuery += string.Format(" --aname={0}", trackNames); } return audioQuery; } private static string ChapterMarkersQuery(frmMain mainWindow) { string query = string.Empty; // Attach Source name and dvd title to the start of the chapters.csv filename. // This is for the queue. It allows different chapter name files for each title. string[] destNameSplit = mainWindow.text_destination.Text.Split('\\'); string destName = destNameSplit[destNameSplit.Length - 1]; destName = destName.Replace("\"", string.Empty); destName = destName.Replace(".mp4", string.Empty).Replace(".m4v", string.Empty).Replace(".mkv", string.Empty); string sourceTitle = mainWindow.drp_dvdtitle.Text; string[] titlesplit = sourceTitle.Split(' '); sourceTitle = titlesplit[0]; if (mainWindow.Check_ChapterMarkers.Checked && mainWindow.Check_ChapterMarkers.Enabled) { if (destName.Trim() != String.Empty) { string path = sourceTitle != "Automatic" ? Path.Combine(Path.GetTempPath(), destName + "-" + sourceTitle + "-chapters.csv") : Path.Combine(Path.GetTempPath(), destName + "-chapters.csv"); if (ChapterCsvSave(mainWindow, path) == false) query += " -m "; else query += " --markers=" + "\"" + path + "\""; } else query += " -m"; } return query; } private static string X264Query(frmMain mainWindow) { string advancedOptions = string.Empty; if (mainWindow.drp_videoEncoder.SelectedItem.ToString().Contains("FFmpeg")) { advancedOptions = string.IsNullOrEmpty(mainWindow.advancedEncoderOpts.AdavancedQuery.Trim()) ? string.Empty : mainWindow.advancedEncoderOpts.AdavancedQuery; } else if (mainWindow.drp_videoEncoder.SelectedItem.ToString().Contains("x264")) { advancedOptions = string.IsNullOrEmpty(mainWindow.x264Panel.X264Query.Trim()) ? string.Empty : mainWindow.x264Panel.X264Query; } return !string.IsNullOrEmpty(advancedOptions) ? " -x " + advancedOptions : string.Empty; } private static string ExtraSettings() { string query = string.Empty; // Verbosity Level query += " --verbose=" + UserSettingService.GetUserSetting(ASUserSettingConstants.Verbosity); // LibDVDNav if (UserSettingService.GetUserSetting(ASUserSettingConstants.DisableLibDvdNav)) query += " --no-dvdnav"; return query; } #endregion #region Helpers /// /// Create a CSV file with the data from the Main Window Chapters tab /// /// Main Window /// Path to save the csv file /// True if successful private static bool ChapterCsvSave(frmMain mainWindow, string filePathName) { try { string csv = string.Empty; foreach (DataGridViewRow row in mainWindow.data_chpt.Rows) { csv += row.Cells[0].Value.ToString(); csv += ","; csv += row.Cells[1].Value.ToString().Replace(",", "\\,"); csv += Environment.NewLine; } StreamWriter file = new StreamWriter(filePathName); file.Write(csv); file.Close(); file.Dispose(); return true; } catch (Exception exc) { MessageBox.Show("Unable to save Chapter Makrers file! \nChapter marker names will NOT be saved in your encode \n\n" + exc, "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning); return false; } } #endregion } }