1 /* QueryGenerator.cs $
\r
3 This file is part of the HandBrake source code.
\r
4 Homepage: <http://handbrake.fr/>.
\r
5 It may be used under the terms of the GNU General Public License. */
\r
8 using System.Windows.Forms;
\r
9 using System.Globalization;
\r
11 using System.Collections.Generic;
\r
12 using Handbrake.Model;
\r
14 namespace Handbrake.Functions
\r
16 class QueryGenerator
\r
19 /// Generates a full CLI query for either encoding or previe encoeds if duration and preview are defined.
\r
21 /// <param name="mainWindow">The Main Window</param>
\r
22 /// <param name="mode">What Mode. (Point to Point Encoding) Chapters, Seconds, Frames OR Preview Encode</param>
\r
23 /// <param name="duration">time in seconds for preview mode</param>
\r
24 /// <param name="preview"> --start-at-preview (int) </param>
\r
25 /// <returns>CLI Query </returns>
\r
26 public static string GenerateCLIQuery(frmMain mainWindow, int mode, int duration, string preview)
\r
30 if (!string.IsNullOrEmpty(mainWindow.sourcePath) && mainWindow.sourcePath.Trim() != "Select \"Source\" to continue")
\r
31 query = " -i " + '"' + mainWindow.sourcePath + '"';
\r
33 if (mainWindow.drp_dvdtitle.Text != "")
\r
35 string[] titleInfo = mainWindow.drp_dvdtitle.Text.Split(' ');
\r
36 query += " -t " + titleInfo[0];
\r
39 if (!Properties.Settings.Default.noDvdNav && mainWindow.drop_angle.Items.Count != 0)
\r
40 query += " --angle " + mainWindow.drop_angle.SelectedItem;
\r
42 // Decide what part of the video we want to encode.
\r
46 if (mainWindow.drop_chapterFinish.Text == mainWindow.drop_chapterStart.Text && mainWindow.drop_chapterStart.Text != "")
\r
47 query += string.Format(" -c {0}", mainWindow.drop_chapterStart.Text);
\r
48 else if (mainWindow.drop_chapterStart.Text != "" && mainWindow.drop_chapterFinish.Text != "")
\r
49 query += string.Format(" -c {0}-{1}", mainWindow.drop_chapterStart.Text, mainWindow.drop_chapterFinish.Text);
\r
53 int.TryParse(mainWindow.drop_chapterStart.Text, out start);
\r
54 int.TryParse(mainWindow.drop_chapterFinish.Text, out end);
\r
55 int calculatedDuration = end - start;
\r
57 query += string.Format(" --start-at duration:{0} --stop-at duration:{1}", mainWindow.drop_chapterStart.Text, calculatedDuration);
\r
60 int.TryParse(mainWindow.drop_chapterStart.Text, out start);
\r
61 int.TryParse(mainWindow.drop_chapterFinish.Text, out end);
\r
62 calculatedDuration = end - start;
\r
64 query += string.Format(" --start-at frame:{0} --stop-at frame:{1}", mainWindow.drop_chapterStart.Text, calculatedDuration);
\r
67 query += " --previews " + Properties.Settings.Default.previewScanCount + " ";
\r
68 query += " --start-at-preview " + preview;
\r
69 query += " --stop-at duration:" + duration + " ";
\r
71 if (mainWindow.text_destination.Text != "")
\r
72 query += string.Format(" -o \"{0}\" ", mainWindow.text_destination.Text.Replace(".m", "_sample.m"));
\r
78 query += string.Format(" -o \"{0}\" ", mainWindow.text_destination.Text);
\r
80 query += GenerateTabbedComponentsQuery(mainWindow);
\r
86 /// Generates part of the CLI query, for the tabbed components only.
\r
88 /// <param name="mainWindow"></param>
\r
89 /// <returns></returns>
\r
90 public static string GenerateTabbedComponentsQuery(frmMain mainWindow)
\r
94 #region Output Settings Box
\r
95 query += " -f " + mainWindow.drop_format.Text.ToLower().Replace(" file", "");
\r
97 // These are output settings features
\r
98 if (mainWindow.check_largeFile.Checked)
\r
101 if (mainWindow.check_iPodAtom.Checked)
\r
104 if (mainWindow.check_optimiseMP4.Checked)
\r
108 #region Picture Settings Tab
\r
110 // Use MaxWidth for built-in presets and width for user settings.
\r
111 if (mainWindow.PictureSettings.PresetMaximumResolution.Width == 0)
\r
113 if (mainWindow.PictureSettings.text_width.Value != 0)
\r
114 if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex != 1) // Prevent usage for strict anamorphic
\r
115 query += " -w " + mainWindow.PictureSettings.text_width.Text;
\r
119 if (mainWindow.PictureSettings.text_width.Value != 0)
\r
120 if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex != 1)
\r
121 query += " -X " + mainWindow.PictureSettings.text_width.Text;
\r
124 // Use MaxHeight for built-in presets and height for user settings.
\r
125 if (mainWindow.PictureSettings.PresetMaximumResolution.Height == 0)
\r
127 if (mainWindow.PictureSettings.text_height.Value != 0)
\r
128 if (mainWindow.PictureSettings.text_height.Text != "")
\r
129 if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 0 || mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 3) // Prevent usage for strict anamorphic
\r
130 query += " -l " + mainWindow.PictureSettings.text_height.Text;
\r
134 if (mainWindow.PictureSettings.text_height.Value != 0)
\r
135 if (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 0 || mainWindow.PictureSettings.drp_anamorphic.SelectedIndex == 3)
\r
136 query += " -Y " + mainWindow.PictureSettings.text_height.Text;
\r
139 string cropTop = mainWindow.PictureSettings.crop_top.Text;
\r
140 string cropBottom = mainWindow.PictureSettings.crop_bottom.Text;
\r
141 string cropLeft = mainWindow.PictureSettings.crop_left.Text;
\r
142 string cropRight = mainWindow.PictureSettings.crop_right.Text;
\r
144 if (mainWindow.PictureSettings.check_customCrop.Checked)
\r
146 if (mainWindow.PictureSettings.crop_top.Text == string.Empty)
\r
148 if (mainWindow.PictureSettings.crop_bottom.Text == string.Empty)
\r
150 if (mainWindow.PictureSettings.crop_left.Text == string.Empty)
\r
152 if (mainWindow.PictureSettings.crop_right.Text == string.Empty)
\r
155 query += " --crop " + cropTop + ":" + cropBottom + ":" + cropLeft + ":" + cropRight;
\r
158 switch (mainWindow.PictureSettings.drp_anamorphic.SelectedIndex)
\r
161 query += " --strict-anamorphic ";
\r
164 query += " --loose-anamorphic ";
\r
167 query += " --custom-anamorphic ";
\r
169 if (mainWindow.PictureSettings.drp_modulus.SelectedIndex != 0)
\r
170 query += " --modulus " + mainWindow.PictureSettings.drp_modulus.SelectedItem;
\r
172 if (mainWindow.PictureSettings.check_KeepAR.Checked)
\r
173 query += " --display-width " + mainWindow.PictureSettings.updownDisplayWidth.Text + " ";
\r
175 if (mainWindow.PictureSettings.check_KeepAR.Checked)
\r
176 query += " --keep-display-aspect ";
\r
178 if (!mainWindow.PictureSettings.check_KeepAR.Checked)
\r
179 if (mainWindow.PictureSettings.updownParWidth.Text != "" && mainWindow.PictureSettings.updownParHeight.Text != "")
\r
180 query += " --pixel-aspect " + mainWindow.PictureSettings.updownParWidth.Text + ":" + mainWindow.PictureSettings.updownParHeight.Text + " ";
\r
186 query += mainWindow.Filters.GetCLIQuery;
\r
189 #region Video Settings Tab
\r
191 switch (mainWindow.drp_videoEncoder.Text)
\r
193 case "MPEG-4 (FFmpeg)":
\r
194 query += " -e ffmpeg";
\r
196 case "H.264 (x264)":
\r
197 query += " -e x264";
\r
199 case "VP3 (Theora)":
\r
200 query += " -e theora";
\r
203 query += " -e x264";
\r
208 if (mainWindow.radio_avgBitrate.Checked)
\r
209 query += " -b " + mainWindow.text_bitrate.Text;
\r
211 if (mainWindow.radio_targetFilesize.Checked)
\r
212 query += " -S " + mainWindow.text_filesize.Text;
\r
214 // Video Quality Setting
\r
215 if (mainWindow.radio_cq.Checked)
\r
217 double cqStep = Properties.Settings.Default.x264cqstep;
\r
219 switch (mainWindow.drp_videoEncoder.Text)
\r
221 case "MPEG-4 (FFmpeg)":
\r
222 value = 31 - (mainWindow.slider_videoQuality.Value - 1);
\r
223 query += " -q " + value.ToString(new CultureInfo("en-US"));
\r
225 case "H.264 (x264)":
\r
226 CultureInfo culture = CultureInfo.CreateSpecificCulture("en-US");
\r
227 value = 51 - mainWindow.slider_videoQuality.Value * cqStep;
\r
228 value = Math.Round(value, 2);
\r
229 query += " -q " + value.ToString(culture);
\r
231 case "VP3 (Theora)":
\r
232 value = mainWindow.slider_videoQuality.Value;
\r
233 query += " -q " + value.ToString(new CultureInfo("en-US"));
\r
238 if (mainWindow.check_2PassEncode.Checked)
\r
241 if (mainWindow.check_turbo.Checked)
\r
244 if (mainWindow.drp_videoFramerate.Text != "Same as source")
\r
245 query += " -r " + mainWindow.drp_videoFramerate.Text;
\r
248 #region Audio Settings Tab
\r
250 ListView audioTracks = mainWindow.AudioSettings.GetAudioPanel();
\r
251 List<string> tracks = new List<string>();
\r
252 List<string> codecs = new List<string>();
\r
253 List<string> mixdowns = new List<string>();
\r
254 List<string> samplerates = new List<string>();
\r
255 List<string> bitrates = new List<string>();
\r
256 List<string> drcs = new List<string>();
\r
259 if (audioTracks.Items.Count == 0)
\r
260 query += " -a none ";
\r
262 // Gather information about each audio track and store them in the declared lists.
\r
263 foreach (ListViewItem row in audioTracks.Items)
\r
265 // Audio Track (-a)
\r
266 if (row.SubItems[1].Text == "Automatic")
\r
268 else if (row.Text != "None")
\r
270 string[] tempSub = row.SubItems[1].Text.Split(' ');
\r
271 tracks.Add(tempSub[0]);
\r
274 // Audio Codec (-E)
\r
275 if (row.SubItems[2].Text != String.Empty)
\r
276 codecs.Add(GetAudioEncoder(row.SubItems[2].Text));
\r
278 // Audio Mixdown (-6)
\r
279 if (row.SubItems[3].Text != String.Empty)
\r
280 mixdowns.Add(GetMixDown(row.SubItems[3].Text));
\r
282 // Sample Rate (-R)
\r
283 if (row.SubItems[4].Text != String.Empty)
\r
284 samplerates.Add(row.SubItems[4].Text);
\r
286 // Audio Bitrate (-B)
\r
287 if (row.SubItems[5].Text != String.Empty)
\r
288 bitrates.Add(row.SubItems[5].Text.Replace("Auto", "auto"));
\r
291 if (row.SubItems[6].Text != String.Empty)
\r
292 drcs.Add(row.SubItems[6].Text);
\r
295 // Audio Track (-a)
\r
296 String audioItems = "";
\r
297 Boolean firstLoop = true;
\r
299 foreach (String item in tracks)
\r
303 audioItems = item; firstLoop = false;
\r
306 audioItems += "," + item;
\r
308 if (audioItems.Trim() != String.Empty)
\r
309 query += " -a " + audioItems;
\r
310 firstLoop = true; audioItems = ""; // Reset for another pass.
\r
312 // Audio Codec (-E)
\r
313 foreach (String item in codecs)
\r
318 audioItems = item; firstLoop = false;
\r
321 audioItems += "," + item;
\r
323 if (audioItems.Trim() != String.Empty)
\r
324 query += " -E " + audioItems;
\r
325 firstLoop = true; audioItems = ""; // Reset for another pass.
\r
327 // Audio Mixdown (-6)
\r
328 foreach (String item in mixdowns)
\r
332 audioItems = item; firstLoop = false;
\r
335 audioItems += "," + item;
\r
337 if (audioItems.Trim() != String.Empty)
\r
338 query += " -6 " + audioItems;
\r
339 firstLoop = true; audioItems = ""; // Reset for another pass.
\r
341 // Sample Rate (-R)
\r
342 foreach (String item in samplerates)
\r
346 audioItems = item; firstLoop = false;
\r
349 audioItems += "," + item;
\r
351 if (audioItems.Trim() != String.Empty)
\r
352 query += " -R " + audioItems;
\r
353 firstLoop = true; audioItems = ""; // Reset for another pass.
\r
355 // Audio Bitrate (-B)
\r
356 foreach (String item in bitrates)
\r
360 audioItems = item; firstLoop = false;
\r
363 audioItems += "," + item;
\r
365 if (audioItems.Trim() != String.Empty)
\r
366 query += " -B " + audioItems;
\r
367 firstLoop = true; audioItems = ""; // Reset for another pass.
\r
370 foreach (var itm in drcs)
\r
372 string item = itm.ToString(new CultureInfo("en-US"));
\r
375 audioItems = item; firstLoop = false;
\r
378 audioItems += "," + item;
\r
380 if (audioItems.Trim() != String.Empty)
\r
381 query += " -D " + audioItems;
\r
385 #region Subtitles Tab
\r
386 if (mainWindow.Subtitles.lv_subList.Items.Count != 0) // If we have subtitle tracks
\r
388 IDictionary<string, string> langMap = Main.MapLanguages();
\r
391 string subtitleTracks = String.Empty;
\r
392 string subtitleForced = String.Empty;
\r
393 string subtitleBurn = String.Empty;
\r
394 string subtitleDefault = String.Empty;
\r
397 string srtFile = String.Empty;
\r
398 string srtCodeset = String.Empty;
\r
399 string srtOffset = String.Empty;
\r
400 string srtLang = String.Empty;
\r
401 string srtDefault = String.Empty;
\r
404 List<SubtitleInfo> SubList = mainWindow.Subtitles.GetSubtitleInfoList();
\r
406 foreach (var item in SubList)
\r
408 string itemToAdd, trackID;
\r
410 if (item.SrtPath != "-") // We have an SRT file
\r
412 srtCount++; // SRT track id.
\r
414 srtLang += srtLang == "" ? langMap[item.SrtLang] : "," + langMap[item.SrtLang];
\r
415 srtCodeset += srtCodeset == "" ? item.SrtCharCode : "," + item.SrtCharCode;
\r
417 if (item.Default == "Yes") // default
\r
418 srtDefault = srtCount.ToString();
\r
420 itemToAdd = item.SrtPath;
\r
421 srtFile += srtFile == "" ? itemToAdd : "," + itemToAdd;
\r
423 itemToAdd = item.SrtOffset.ToString();
\r
424 srtOffset += srtOffset == "" ? itemToAdd : "," + itemToAdd;
\r
426 else // We have Bitmap or CC
\r
430 // Find --subtitle <string>
\r
431 if (item.Track.Contains("Foreign Audio Search"))
\r
432 itemToAdd = "scan";
\r
435 tempSub = item.Track.Split(' ');
\r
436 itemToAdd = tempSub[0];
\r
439 subtitleTracks += subtitleTracks == "" ? itemToAdd : "," + itemToAdd;
\r
441 // Find --subtitle-forced
\r
443 tempSub = item.Track.Split(' ');
\r
444 trackID = tempSub[0];
\r
446 if (item.Forced == "Yes")
\r
447 itemToAdd = "scan";
\r
449 if (itemToAdd != "")
\r
450 subtitleForced += subtitleForced == "" ? itemToAdd : "," + itemToAdd;
\r
452 // Find --subtitle-burn and --subtitle-default
\r
453 trackID = tempSub[0];
\r
455 if (trackID.Trim() == "Foreign")
\r
458 if (item.Burned == "Yes") // burn
\r
459 subtitleBurn = trackID;
\r
461 if (item.Default == "Yes") // default
\r
462 subtitleDefault = trackID;
\r
466 // Build The CLI Subtitles Query
\r
467 if (subtitleTracks != "")
\r
469 query += " --subtitle " + subtitleTracks;
\r
471 if (subtitleForced != "")
\r
472 query += " --subtitle-forced=" + subtitleForced;
\r
473 if (subtitleBurn != "")
\r
474 query += " --subtitle-burn=" + subtitleBurn;
\r
475 if (subtitleDefault != "")
\r
476 query += " --subtitle-default=" + subtitleDefault;
\r
479 if (srtFile != "") // SRTs
\r
481 query += " --srt-file " + "\"" + srtFile + "\"";
\r
483 if (srtCodeset != "")
\r
484 query += " --srt-codeset " + srtCodeset;
\r
485 if (srtOffset != "")
\r
486 query += " --srt-offset " + srtOffset;
\r
488 query += " --srt-lang " + srtLang;
\r
489 if (srtDefault != "")
\r
490 query += " --srt-default=" + srtDefault;
\r
496 #region Chapter Markers
\r
498 // Attach Source name and dvd title to the start of the chapters.csv filename.
\r
499 // This is for the queue. It allows different chapter name files for each title.
\r
500 string[] destName = mainWindow.text_destination.Text.Split('\\');
\r
501 string dest_name = destName[destName.Length - 1];
\r
502 dest_name = dest_name.Replace("\"", "");
\r
503 dest_name = dest_name.Replace(".mp4", "").Replace(".m4v", "").Replace(".mkv", "");
\r
505 string source_title = mainWindow.drp_dvdtitle.Text;
\r
506 string[] titlesplit = source_title.Split(' ');
\r
507 source_title = titlesplit[0];
\r
509 if (mainWindow.Check_ChapterMarkers.Checked && mainWindow.Check_ChapterMarkers.Enabled)
\r
511 if (dest_name.Trim() != String.Empty)
\r
513 string path = source_title != "Automatic"
\r
514 ? Path.Combine(Path.GetTempPath(), dest_name + "-" + source_title + "-chapters.csv")
\r
515 : Path.Combine(Path.GetTempPath(), dest_name + "-chapters.csv");
\r
517 if (ChapterCSVSave(mainWindow, path) == false)
\r
520 query += " --markers=" + "\"" + path + "\"";
\r
528 if (mainWindow.x264Panel.X264Query != "")
\r
529 query += " -x " + mainWindow.x264Panel.X264Query;
\r
532 #region Processors / Other
\r
533 string processors = Properties.Settings.Default.Processors;
\r
534 if (processors != "Automatic")
\r
535 query += " -C " + processors + " ";
\r
537 query += " -v " + Properties.Settings.Default.verboseLevel;
\r
539 if (Properties.Settings.Default.noDvdNav)
\r
540 query += " --no-dvdnav";
\r
547 /// Return the CLI Mixdown name
\r
549 /// <param name="selectedAudio">GUI mixdown name</param>
\r
550 /// <returns>CLI mixdown name</returns>
\r
551 private static string GetMixDown(string selectedAudio)
\r
553 switch (selectedAudio)
\r
561 case "Dolby Surround":
\r
563 case "Dolby Pro Logic II":
\r
565 case "6 Channel Discrete":
\r
573 /// Get the CLI Audio Encoder name
\r
575 /// <param name="selectedEncoder">string The GUI Encode name</param>
\r
576 /// <returns>string CLI encoder name</returns>
\r
577 private static string GetAudioEncoder(string selectedEncoder)
\r
579 switch (selectedEncoder)
\r
585 case "Vorbis (vorbis)":
\r
587 case "AC3 Passthru":
\r
589 case "DTS Passthru":
\r
597 /// Create a CSV file with the data from the Main Window Chapters tab
\r
599 /// <param name="mainWindow">Main Window</param>
\r
600 /// <param name="filePathName">Path to save the csv file</param>
\r
601 /// <returns>True if successful </returns>
\r
602 private static Boolean ChapterCSVSave(frmMain mainWindow, string filePathName)
\r
608 foreach (DataGridViewRow row in mainWindow.data_chpt.Rows)
\r
610 csv += row.Cells[0].Value.ToString();
\r
612 csv += row.Cells[1].Value.ToString().Replace(",","\\,");
\r
613 csv += Environment.NewLine;
\r
615 StreamWriter file = new StreamWriter(filePathName);
\r
621 catch (Exception exc)
\r
623 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);
\r