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.Collections.Generic;
\r
9 using System.Drawing;
\r
10 using System.Globalization;
\r
11 using System.Windows.Forms;
\r
13 using System.Diagnostics;
\r
14 using System.Threading;
\r
15 using Handbrake.EncodeQueue;
\r
16 using Handbrake.Functions;
\r
17 using Handbrake.Presets;
\r
18 using Handbrake.Parsing;
\r
22 public partial class frmMain : Form
\r
24 // Objects which may be used by one or more other objects *************
\r
25 EncodeAndQueueHandler encodeQueue = new EncodeAndQueueHandler();
\r
26 PresetsHandler presetHandler = new PresetsHandler();
\r
27 QueryGenerator queryGen = new QueryGenerator();
\r
29 // Globals: Mainly used for tracking. *********************************
\r
30 public Title selectedTitle;
\r
31 private DVD thisDVD;
\r
32 private frmQueue queueWindow;
\r
33 private frmPreview qtpreview;
\r
34 private frmActivityWindow ActivityWindow;
\r
35 private Form splash;
\r
36 public string sourcePath;
\r
37 private string lastAction;
\r
38 private SourceType selectedSourceType;
\r
39 private string dvdDrivePath;
\r
40 private string dvdDriveLabel;
\r
42 // Delegates **********************************************************
\r
43 private delegate void UpdateWindowHandler();
\r
45 // Applicaiton Startup ************************************************
\r
47 #region Application Startup
\r
50 // Load and setup the splash screen in this thread
\r
51 splash = new frmSplashScreen();
\r
53 Label lblStatus = new Label { Size = new Size(150, 20), Location = new Point(182, 102) };
\r
54 splash.Controls.Add(lblStatus);
\r
56 InitializeComponent();
\r
58 // Update the users config file with the CLI version data.
\r
59 lblStatus.Text = "Setting Version Data ...";
\r
60 Application.DoEvents();
\r
61 Main.setCliVersionData();
\r
63 // Show the form, but leave disabled until preloading is complete then show the main form
\r
64 this.Enabled = false;
\r
66 Application.DoEvents(); // Forces frmMain to draw
\r
68 // Check for new versions, if update checking is enabled
\r
69 if (Properties.Settings.Default.updateStatus)
\r
71 DateTime now = DateTime.Now;
\r
72 DateTime lastCheck = Properties.Settings.Default.lastUpdateCheckDate;
\r
73 TimeSpan elapsed = now.Subtract(lastCheck);
\r
74 if (elapsed.TotalDays > Properties.Settings.Default.daysBetweenUpdateCheck)
\r
76 lblStatus.Text = "Checking for updates ...";
\r
77 Application.DoEvents();
\r
79 Main.BeginCheckForUpdates(new AsyncCallback(UpdateCheckDone), false);
\r
83 // Setup the GUI components
\r
84 lblStatus.Text = "Setting up the GUI ...";
\r
85 Application.DoEvents();
\r
86 loadPresetPanel(); // Load the Preset Panel
\r
87 treeView_presets.ExpandAll();
\r
88 lbl_encode.Text = "";
\r
89 queueWindow = new frmQueue(encodeQueue); // Prepare the Queue
\r
90 if (!Properties.Settings.Default.QueryEditorTab)
\r
91 tabs_panel.TabPages.RemoveAt(7); // Remove the query editor tab if the user does not want it enabled.
\r
93 // Load the user's default settings or Normal Preset
\r
94 if (Properties.Settings.Default.defaultPreset != "")
\r
96 if (presetHandler.GetPreset(Properties.Settings.Default.defaultPreset) != null)
\r
98 string query = presetHandler.GetPreset(Properties.Settings.Default.defaultPreset).Query;
\r
99 Boolean loadPictureSettings = presetHandler.GetPreset(Properties.Settings.Default.defaultPreset).PictureSettings;
\r
103 //Ok, Reset all the H264 widgets before changing the preset
\r
104 x264Panel.reset2Defaults();
\r
106 // Send the query from the file to the Query Parser class, then load the preset
\r
107 QueryParser presetQuery = QueryParser.Parse(query);
\r
108 PresetLoader.presetLoader(this, presetQuery, Properties.Settings.Default.defaultPreset, loadPictureSettings);
\r
110 // The x264 widgets will need updated, so do this now:
\r
111 x264Panel.X264_StandardizeOptString();
\r
112 x264Panel.X264_SetCurrentSettingsInPanel();
\r
116 loadNormalPreset();
\r
119 loadNormalPreset();
\r
121 // Enabled GUI tooltip's if Required
\r
122 if (Properties.Settings.Default.tooltipEnable)
\r
123 ToolTip.Active = true;
\r
125 // Register with Growl (if not using Growl for the encoding completion action, this wont hurt anything)
\r
126 GrowlCommunicator.Register();
\r
129 lblStatus.Text = "Loading Complete!";
\r
130 Application.DoEvents();
\r
133 this.Enabled = true;
\r
135 // Event Handlers and Queue Recovery
\r
140 private void UpdateCheckDone(IAsyncResult result)
\r
142 if (InvokeRequired)
\r
144 Invoke(new MethodInvoker(() => UpdateCheckDone(result)));
\r
148 UpdateCheckInformation info;
\r
152 info = Main.EndCheckForUpdates(result);
\r
154 if (info.NewVersionAvailable)
\r
156 frmUpdater updateWindow = new frmUpdater(info.BuildInformation);
\r
157 updateWindow.ShowDialog();
\r
160 catch (Exception ex)
\r
162 if ((bool)result.AsyncState)
\r
163 MessageBox.Show("Unable to check for updates, Please try again later.\n\nDetailed Error Information:\n" + ex, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
167 // Startup Functions
\r
168 private void queueRecovery()
\r
170 if (Main.checkQueueRecovery())
\r
172 DialogResult result = MessageBox.Show("HandBrake has detected unfinished items on the queue from the last time the application was launched. Would you like to recover these?", "Queue Recovery Possible", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
174 if (result == DialogResult.Yes)
\r
175 encodeQueue.LoadQueueFromFile("hb_queue_recovery.xml"); // Start Recovery
\r
178 // Remove the Queue recovery file if the user doesn't want to recovery the last queue.
\r
179 string queuePath = Path.Combine(Path.GetTempPath(), "hb_queue_recovery.xml");
\r
180 if (File.Exists(queuePath))
\r
181 File.Delete(queuePath);
\r
188 public string SourceName
\r
192 if (this.selectedSourceType == SourceType.DvdDrive)
\r
194 return this.dvdDriveLabel;
\r
197 return Path.GetFileNameWithoutExtension(this.sourcePath);
\r
203 // Encoding Events for setting up the GUI
\r
204 private void events()
\r
206 // Handle Widget changes when preset is selected.
\r
207 RegisterPresetEventHandler();
\r
209 // Handle Window Resize
\r
210 if (Properties.Settings.Default.MainWindowMinimize)
\r
211 this.Resize += new EventHandler(frmMain_Resize);
\r
213 // Handle Encode Start / Finish / Pause
\r
214 encodeQueue.CurrentJobCompleted += new EventHandler(encodeEnded);
\r
215 encodeQueue.QueuePauseRequested += new EventHandler(encodePaused);
\r
216 encodeQueue.NewJobStarted += new EventHandler(encodeStarted);
\r
218 // Handle a file being draged onto the GUI.
\r
219 this.DragEnter += new DragEventHandler(frmMain_DragEnter);
\r
220 this.DragDrop += new DragEventHandler(frmMain_DragDrop);
\r
223 // Change the preset label to custom when a user changes a setting. Don't want to give the impression that users can change settings and still be using a preset
\r
224 private void RegisterPresetEventHandler()
\r
227 drop_format.SelectedIndexChanged += new EventHandler(changePresetLabel);
\r
228 check_largeFile.CheckedChanged += new EventHandler(changePresetLabel);
\r
229 check_iPodAtom.CheckedChanged += new EventHandler(changePresetLabel);
\r
230 check_optimiseMP4.CheckedChanged += new EventHandler(changePresetLabel);
\r
232 // Picture Settings
\r
233 //PictureSettings.PictureSettingsChanged += new EventHandler(changePresetLabel);
\r
236 Filters.FilterSettingsChanged += new EventHandler(changePresetLabel);
\r
239 drp_videoEncoder.SelectedIndexChanged += new EventHandler(changePresetLabel);
\r
240 check_2PassEncode.CheckedChanged += new EventHandler(changePresetLabel);
\r
241 check_turbo.CheckedChanged += new EventHandler(changePresetLabel);
\r
242 text_filesize.TextChanged += new EventHandler(changePresetLabel);
\r
243 text_bitrate.TextChanged += new EventHandler(changePresetLabel);
\r
244 slider_videoQuality.ValueChanged += new EventHandler(changePresetLabel);
\r
247 AudioSettings.AudioListChanged += new EventHandler(changePresetLabel);
\r
250 x264Panel.rtf_x264Query.TextChanged += new EventHandler(changePresetLabel);
\r
252 private void UnRegisterPresetEventHandler()
\r
254 // Output Settings
\r
255 drop_format.SelectedIndexChanged -= new EventHandler(changePresetLabel);
\r
256 check_largeFile.CheckedChanged -= new EventHandler(changePresetLabel);
\r
257 check_iPodAtom.CheckedChanged -= new EventHandler(changePresetLabel);
\r
258 check_optimiseMP4.CheckedChanged -= new EventHandler(changePresetLabel);
\r
260 // Picture Settings
\r
261 //PictureSettings.PictureSettingsChanged -= new EventHandler(changePresetLabel);
\r
264 Filters.FilterSettingsChanged -= new EventHandler(changePresetLabel);
\r
267 drp_videoEncoder.SelectedIndexChanged -= new EventHandler(changePresetLabel);
\r
268 check_2PassEncode.CheckedChanged -= new EventHandler(changePresetLabel);
\r
269 check_turbo.CheckedChanged -= new EventHandler(changePresetLabel);
\r
270 text_filesize.TextChanged -= new EventHandler(changePresetLabel);
\r
271 text_bitrate.TextChanged -= new EventHandler(changePresetLabel);
\r
272 slider_videoQuality.ValueChanged -= new EventHandler(changePresetLabel);
\r
275 AudioSettings.AudioListChanged -= new EventHandler(changePresetLabel);
\r
278 x264Panel.rtf_x264Query.TextChanged -= new EventHandler(changePresetLabel);
\r
280 private void changePresetLabel(object sender, EventArgs e)
\r
282 labelPreset.Text = "Output Settings (Preset: Custom)";
\r
285 private static void frmMain_DragEnter(object sender, DragEventArgs e)
\r
287 if (e.Data.GetDataPresent(DataFormats.FileDrop, false))
\r
288 e.Effect = DragDropEffects.All;
\r
290 private void frmMain_DragDrop(object sender, DragEventArgs e)
\r
292 string[] fileList = e.Data.GetData(DataFormats.FileDrop) as string[];
\r
293 sourcePath = string.Empty;
\r
295 if (fileList != null)
\r
297 if (fileList[0].StartsWith("\\"))
\r
300 "Sorry, HandBrake does not support UNC file paths. \nTry mounting the network share as a network drive in My Computer",
\r
301 "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
302 UpdateSourceLabel();
\r
306 if (fileList[0] != "")
\r
307 startScan(fileList[0]);
\r
309 UpdateSourceLabel();
\r
313 UpdateSourceLabel();
\r
315 private void encodeStarted(object sender, EventArgs e)
\r
317 lastAction = "encode";
\r
318 setEncodeStarted();
\r
320 // Experimental HBProc Process Monitoring.
\r
321 if (Properties.Settings.Default.enocdeStatusInGui)
\r
323 Thread encodeMon = new Thread(encodeMonitorThread);
\r
327 private void encodeEnded(object sender, EventArgs e)
\r
329 setEncodeFinished();
\r
331 private void encodePaused(object sender, EventArgs e)
\r
333 setEncodeFinished();
\r
337 // User Interface Menus / Tool Strips *********************************
\r
340 private void mnu_killCLI_Click(object sender, EventArgs e)
\r
344 private void mnu_exit_Click(object sender, EventArgs e)
\r
346 Application.Exit();
\r
351 private void mnu_encode_Click(object sender, EventArgs e)
\r
353 queueWindow.Show();
\r
355 private void mnu_encodeLog_Click(object sender, EventArgs e)
\r
357 String file = lastAction == "scan" ? "last_scan_log.txt" : "last_encode_log.txt";
\r
359 frmActivityWindow dvdInfoWindow = new frmActivityWindow(file, encodeQueue, this);
\r
360 dvdInfoWindow.Show();
\r
362 private void mnu_options_Click(object sender, EventArgs e)
\r
364 Form options = new frmOptions(this);
\r
365 options.ShowDialog();
\r
369 #region Presets Menu
\r
370 private void mnu_presetReset_Click(object sender, EventArgs e)
\r
372 presetHandler.UpdateBuiltInPresets();
\r
374 if (treeView_presets.Nodes.Count == 0)
\r
375 MessageBox.Show("Unable to load the presets.xml file. Please select \"Update Built-in Presets\" from the Presets Menu. \nMake sure you are running the program in Admin mode if running on Vista. See Windows FAQ for details!", "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
377 MessageBox.Show("Presets have been updated!", "Alert", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
379 treeView_presets.ExpandAll();
\r
381 private void mnu_delete_preset_Click(object sender, EventArgs e)
\r
383 presetHandler.RemoveBuiltInPresets();
\r
384 loadPresetPanel(); // Reload the preset panel
\r
386 private void mnu_SelectDefault_Click(object sender, EventArgs e)
\r
388 loadNormalPreset();
\r
390 private void mnu_importMacPreset_Click(object sender, EventArgs e)
\r
394 private void btn_new_preset_Click(object sender, EventArgs e)
\r
396 Form preset = new frmAddPreset(this, queryGen.GenerateCLIQuery(this, 0, null), presetHandler);
\r
397 preset.ShowDialog();
\r
402 private void mnu_handbrake_forums_Click(object sender, EventArgs e)
\r
404 Process.Start("http://forum.handbrake.fr/");
\r
406 private void mnu_user_guide_Click(object sender, EventArgs e)
\r
408 Process.Start("http://trac.handbrake.fr/wiki/HandBrakeGuide");
\r
410 private void mnu_handbrake_home_Click(object sender, EventArgs e)
\r
412 Process.Start("http://handbrake.fr");
\r
414 private void mnu_UpdateCheck_Click(object sender, EventArgs e)
\r
416 lbl_updateCheck.Visible = true;
\r
417 Main.BeginCheckForUpdates(new AsyncCallback(updateCheckDoneMenu), false);
\r
419 private void updateCheckDoneMenu(IAsyncResult result)
\r
421 // Make sure it's running on the calling thread
\r
422 if (InvokeRequired)
\r
424 Invoke(new MethodInvoker(() => updateCheckDoneMenu(result)));
\r
427 UpdateCheckInformation info;
\r
430 // Get the information about the new build, if any, and close the window
\r
431 info = Main.EndCheckForUpdates(result);
\r
433 if (info.NewVersionAvailable && info.BuildInformation != null)
\r
435 frmUpdater updateWindow = new frmUpdater(info.BuildInformation);
\r
436 updateWindow.ShowDialog();
\r
439 MessageBox.Show("There are no new updates at this time.", "Update Check", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
440 lbl_updateCheck.Visible = false;
\r
443 catch (Exception ex)
\r
445 if ((bool)result.AsyncState)
\r
446 MessageBox.Show("Unable to check for updates, Please try again later.\n\nDetailed Error Information:\n" + ex, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
449 private void mnu_about_Click(object sender, EventArgs e)
\r
451 using (frmAbout About = new frmAbout())
\r
453 About.ShowDialog();
\r
459 // Right Click Menu Code
\r
460 private void pmnu_expandAll_Click(object sender, EventArgs e)
\r
462 treeView_presets.ExpandAll();
\r
464 private void pmnu_collapse_Click(object sender, EventArgs e)
\r
466 treeView_presets.CollapseAll();
\r
468 private void pmnu_import_Click(object sender, EventArgs e)
\r
472 private void pmnu_saveChanges_Click(object sender, EventArgs e)
\r
474 DialogResult result = MessageBox.Show("Do you wish to include picture settings when updating the preset: " + treeView_presets.SelectedNode.Text, "Update Preset", MessageBoxButtons.YesNoCancel, MessageBoxIcon.Question);
\r
475 if (result == DialogResult.Yes)
\r
476 presetHandler.Update(treeView_presets.SelectedNode.Text, QueryGenerator.GenerateTabbedComponentsQuery(this), true);
\r
477 else if (result == DialogResult.No)
\r
478 presetHandler.Update(treeView_presets.SelectedNode.Text, QueryGenerator.GenerateTabbedComponentsQuery(this), false);
\r
480 private void pmnu_delete_click(object sender, EventArgs e)
\r
482 if (treeView_presets.SelectedNode != null)
\r
484 presetHandler.Remove(treeView_presets.SelectedNode.Text);
\r
485 treeView_presets.Nodes.Remove(treeView_presets.SelectedNode);
\r
487 treeView_presets.Select();
\r
489 private void presets_menu_Opening(object sender, System.ComponentModel.CancelEventArgs e)
\r
491 // Make sure that the save menu is always disabled by default
\r
492 pmnu_saveChanges.Enabled = false;
\r
494 // Now enable the save menu if the selected preset is a user preset
\r
495 if (treeView_presets.SelectedNode != null)
\r
496 if (presetHandler.CheckIfUserPresetExists(treeView_presets.SelectedNode.Text))
\r
497 pmnu_saveChanges.Enabled = true;
\r
499 treeView_presets.Select();
\r
502 // Presets Management
\r
503 private void btn_addPreset_Click(object sender, EventArgs e)
\r
505 Form preset = new frmAddPreset(this, QueryGenerator.GenerateTabbedComponentsQuery(this), presetHandler);
\r
506 preset.ShowDialog();
\r
508 private void btn_removePreset_Click(object sender, EventArgs e)
\r
510 DialogResult result = MessageBox.Show("Are you sure you wish to delete the selected preset?", "Preset", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
511 if (result == DialogResult.Yes)
\r
513 if (treeView_presets.SelectedNode != null)
\r
515 presetHandler.Remove(treeView_presets.SelectedNode.Text);
\r
516 treeView_presets.Nodes.Remove(treeView_presets.SelectedNode);
\r
519 treeView_presets.Select();
\r
521 private void btn_setDefault_Click(object sender, EventArgs e)
\r
523 if (treeView_presets.SelectedNode != null)
\r
525 DialogResult result = MessageBox.Show("Are you sure you wish to set this preset as the default?", "Preset", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
526 if (result == DialogResult.Yes)
\r
528 Properties.Settings.Default.defaultPreset = treeView_presets.SelectedNode.Text;
\r
529 Properties.Settings.Default.Save();
\r
530 MessageBox.Show("New default preset set.", "Alert", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
534 MessageBox.Show("Please select a preset first.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
536 private void treeview_presets_mouseUp(object sender, MouseEventArgs e)
\r
538 if (e.Button == MouseButtons.Right)
\r
539 treeView_presets.SelectedNode = treeView_presets.GetNodeAt(e.Location);
\r
540 else if (e.Button == MouseButtons.Left)
\r
542 if (treeView_presets.GetNodeAt(e.Location) != null)
\r
544 if (labelPreset.Text.Contains(treeView_presets.GetNodeAt(e.Location).Text))
\r
549 treeView_presets.Select();
\r
551 private void treeView_presets_AfterSelect(object sender, TreeViewEventArgs e)
\r
555 private void treeView_presets_deleteKey(object sender, KeyEventArgs e)
\r
557 if (e.KeyCode == Keys.Delete)
\r
559 DialogResult result = MessageBox.Show("Are you sure you wish to delete the selected preset?", "Preset", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
560 if (result == DialogResult.Yes)
\r
562 if (treeView_presets.SelectedNode != null)
\r
563 presetHandler.Remove(treeView_presets.SelectedNode.Text);
\r
565 // Remember each nodes expanded status so we can reload it
\r
566 List<Boolean> nodeStatus = new List<Boolean>();
\r
567 foreach (TreeNode node in treeView_presets.Nodes)
\r
568 nodeStatus.Add(node.IsExpanded);
\r
570 // Now reload the preset panel
\r
573 // And finally, re-expand any of the nodes if required
\r
575 foreach (TreeNode node in treeView_presets.Nodes)
\r
585 private void selectPreset()
\r
587 if (treeView_presets.SelectedNode != null)
\r
589 // Ok, so, we've selected a preset. Now we want to load it.
\r
590 string presetName = treeView_presets.SelectedNode.Text;
\r
591 if (presetHandler.GetPreset(presetName) != null)
\r
593 string query = presetHandler.GetPreset(presetName).Query;
\r
594 Boolean loadPictureSettings = presetHandler.GetPreset(presetName).PictureSettings;
\r
598 //Ok, Reset all the H264 widgets before changing the preset
\r
599 x264Panel.reset2Defaults();
\r
601 // Send the query from the file to the Query Parser class
\r
602 QueryParser presetQuery = QueryParser.Parse(query);
\r
604 // Now load the preset
\r
605 PresetLoader.presetLoader(this, presetQuery, presetName, loadPictureSettings);
\r
607 // The x264 widgets will need updated, so do this now:
\r
608 x264Panel.X264_StandardizeOptString();
\r
609 x264Panel.X264_SetCurrentSettingsInPanel();
\r
614 private void loadNormalPreset()
\r
616 foreach (TreeNode treenode in treeView_presets.Nodes)
\r
618 foreach (TreeNode node in treenode.Nodes)
\r
620 if (node.Text.Equals("Normal"))
\r
621 treeView_presets.SelectedNode = treeView_presets.Nodes[treenode.Index].Nodes[0];
\r
625 private void importPreset()
\r
627 Import imp = new Import();
\r
628 if (openPreset.ShowDialog() == DialogResult.OK)
\r
630 QueryParser parsed = imp.importMacPreset(openPreset.FileName);
\r
631 if (presetHandler.CheckIfUserPresetExists(parsed.PresetName + " (Imported)"))
\r
633 DialogResult result = MessageBox.Show("This preset appears to already exist. Would you like to overwrite it?", "Overwrite preset?",
\r
634 MessageBoxButtons.YesNo, MessageBoxIcon.Warning);
\r
635 if (result == DialogResult.Yes)
\r
637 PresetLoader.presetLoader(this, parsed, parsed.PresetName, parsed.UsesPictureSettings);
\r
638 presetHandler.Update(parsed.PresetName + " (Imported)", queryGen.GenerateCLIQuery(this, 0, null),
\r
639 parsed.UsesPictureSettings);
\r
644 PresetLoader.presetLoader(this, parsed, parsed.PresetName, parsed.UsesPictureSettings);
\r
645 presetHandler.Add(parsed.PresetName, queryGen.GenerateCLIQuery(this, 0, null), parsed.UsesPictureSettings);
\r
647 if (presetHandler.Add(parsed.PresetName + " (Imported)", queryGen.GenerateCLIQuery(this, 0, null), parsed.UsesPictureSettings))
\r
649 TreeNode preset_treeview = new TreeNode(parsed.PresetName + " (Imported)") { ForeColor = Color.Black };
\r
650 treeView_presets.Nodes.Add(preset_treeview);
\r
658 private void btn_source_Click(object sender, EventArgs e)
\r
660 mnu_dvd_drive.Visible = true;
\r
661 Thread driveInfoThread = new Thread(getDriveInfoThread);
\r
662 driveInfoThread.Start();
\r
664 private void btn_start_Click(object sender, EventArgs e)
\r
666 if (btn_start.Text == "Stop")
\r
668 DialogResult result = MessageBox.Show("Are you sure you wish to cancel the encode?", "Cancel Encode?", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
670 if (result == DialogResult.Yes)
\r
673 encodeQueue.RequestPause();
\r
675 // Allow the CLI to exit cleanly
\r
676 Win32.SetForegroundWindow((int)encodeQueue.processHandle);
\r
677 SendKeys.Send("^C");
\r
680 setEncodeFinished();
\r
685 if (encodeQueue.Count != 0 || (!string.IsNullOrEmpty(sourcePath) && !string.IsNullOrEmpty(text_destination.Text)))
\r
687 string generatedQuery = queryGen.GenerateCLIQuery(this, 0, null);
\r
688 string specifiedQuery = rtf_query.Text != "" ? rtf_query.Text : queryGen.GenerateCLIQuery(this, 0, null);
\r
689 string query = string.Empty;
\r
691 // Check to make sure the generated query matches the GUI settings
\r
692 if (Properties.Settings.Default.PromptOnUnmatchingQueries && !string.IsNullOrEmpty(specifiedQuery) && generatedQuery != specifiedQuery)
\r
694 DialogResult result = MessageBox.Show("The query under the \"Query Editor\" tab " +
\r
695 "does not match the current GUI settings.\n\nBecause the manual query takes " +
\r
696 "priority over the GUI, your recently updated settings will not be taken " +
\r
697 "into account when encoding this job." + Environment.NewLine + Environment.NewLine +
\r
698 "Do you want to replace the manual query with the updated GUI-generated query?",
\r
699 "Manual Query does not Match GUI",
\r
700 MessageBoxButtons.YesNoCancel, MessageBoxIcon.Asterisk,
\r
701 MessageBoxDefaultButton.Button3);
\r
705 case DialogResult.Yes:
\r
706 // Replace the manual query with the generated one
\r
707 query = generatedQuery;
\r
708 rtf_query.Text = generatedQuery;
\r
710 case DialogResult.No:
\r
711 // Use the manual query
\r
712 query = specifiedQuery;
\r
714 case DialogResult.Cancel:
\r
715 // Don't start the encode
\r
721 query = specifiedQuery;
\r
724 DialogResult overwrite = DialogResult.Yes;
\r
725 if (text_destination.Text != "")
\r
726 if (File.Exists(text_destination.Text))
\r
727 overwrite = MessageBox.Show("The destination file already exists. Are you sure you want to overwrite it?", "Overwrite File?", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
729 if (overwrite == DialogResult.Yes)
\r
731 if (encodeQueue.Count == 0)
\r
732 encodeQueue.AddJob(query, sourcePath, text_destination.Text, (rtf_query.Text != ""));
\r
734 queueWindow.setQueue();
\r
735 if (encodeQueue.Count > 1)
\r
736 queueWindow.Show(false);
\r
738 setEncodeStarted(); // Encode is running, so setup the GUI appropriately
\r
739 encodeQueue.StartEncodeQueue(); // Start The Queue Encoding Process
\r
740 lastAction = "encode"; // Set the last action to encode - Used for activity window.
\r
742 if (ActivityWindow != null)
\r
743 ActivityWindow.SetLogView(false);
\r
748 else if (string.IsNullOrEmpty(sourcePath) || string.IsNullOrEmpty(text_destination.Text))
\r
749 MessageBox.Show("No source or destination selected.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
752 private void btn_add2Queue_Click(object sender, EventArgs e)
\r
754 if (string.IsNullOrEmpty(sourcePath) || string.IsNullOrEmpty(text_destination.Text))
\r
755 MessageBox.Show("No source or destination selected.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
758 String query = queryGen.GenerateCLIQuery(this, 0, null);
\r
759 if (rtf_query.Text != "")
\r
760 query = rtf_query.Text;
\r
762 if (encodeQueue.CheckForDestinationDuplicate(text_destination.Text))
\r
764 DialogResult result = MessageBox.Show("There is already a queue item for this destination path. \n\n If you continue, the encode will be overwritten. Do you wish to continue?",
\r
765 "Warning", MessageBoxButtons.YesNo, MessageBoxIcon.Warning);
\r
766 if (result == DialogResult.Yes)
\r
767 encodeQueue.AddJob(query, sourcePath, text_destination.Text, (rtf_query.Text != ""));
\r
771 encodeQueue.AddJob(query, sourcePath, text_destination.Text, (rtf_query.Text != ""));
\r
773 lbl_encode.Text = encodeQueue.Count + " encode(s) pending in the queue";
\r
775 queueWindow.Show();
\r
778 private void btn_showQueue_Click(object sender, EventArgs e)
\r
780 queueWindow.Show();
\r
781 queueWindow.Activate();
\r
783 private void tb_preview_Click(object sender, EventArgs e)
\r
785 if (string.IsNullOrEmpty(sourcePath) || string.IsNullOrEmpty(text_destination.Text))
\r
786 MessageBox.Show("No source or destination selected.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
789 if (qtpreview == null)
\r
791 qtpreview = new frmPreview(this);
\r
794 else if (qtpreview.IsDisposed)
\r
796 qtpreview = new frmPreview(this);
\r
800 MessageBox.Show(qtpreview, "The preview window is already open!", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
803 private void btn_ActivityWindow_Click(object sender, EventArgs e)
\r
805 String file = lastAction == "scan" ? "last_scan_log.txt" : "last_encode_log.txt";
\r
806 if (ActivityWindow == null)
\r
807 ActivityWindow = new frmActivityWindow(file, encodeQueue, this);
\r
809 ActivityWindow.SetLogView(!encodeQueue.isEncoding);
\r
811 ActivityWindow.Show();
\r
815 #region System Tray Icon
\r
816 private void frmMain_Resize(object sender, EventArgs e)
\r
818 if (FormWindowState.Minimized == this.WindowState)
\r
820 notifyIcon.Visible = true;
\r
823 else if (FormWindowState.Normal == this.WindowState)
\r
824 notifyIcon.Visible = false;
\r
826 private void notifyIcon_MouseDoubleClick(object sender, MouseEventArgs e)
\r
828 this.Visible = true;
\r
830 this.WindowState = FormWindowState.Normal;
\r
831 notifyIcon.Visible = false;
\r
833 private void btn_restore_Click(object sender, EventArgs e)
\r
835 this.Visible = true;
\r
837 this.WindowState = FormWindowState.Normal;
\r
838 notifyIcon.Visible = false;
\r
842 #region Tab Control
\r
845 private void btn_dvd_source_Click(object sender, EventArgs e)
\r
847 if (DVD_Open.ShowDialog() == DialogResult.OK)
\r
849 this.selectedSourceType = SourceType.Folder;
\r
850 selectSource(DVD_Open.SelectedPath);
\r
853 UpdateSourceLabel();
\r
855 private void btn_file_source_Click(object sender, EventArgs e)
\r
857 if (ISO_Open.ShowDialog() == DialogResult.OK)
\r
859 this.selectedSourceType = SourceType.VideoFile;
\r
860 selectSource(ISO_Open.FileName);
\r
863 UpdateSourceLabel();
\r
865 private void mnu_dvd_drive_Click(object sender, EventArgs e)
\r
867 if (this.dvdDrivePath == null) return;
\r
868 this.selectedSourceType = SourceType.DvdDrive;
\r
869 selectSource(this.dvdDrivePath);
\r
871 private void selectSource(string file)
\r
873 Check_ChapterMarkers.Enabled = true;
\r
874 lastAction = "scan";
\r
875 sourcePath = string.Empty;
\r
877 if (file == string.Empty) // Must have a file or path
\r
879 UpdateSourceLabel();
\r
883 if (file.StartsWith("\\")) // NO UNC Paths
\r
886 "Sorry, HandBrake does not support UNC file paths. \nTry mounting the share as a network drive in My Computer",
\r
887 "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
888 UpdateSourceLabel();
\r
892 sourcePath = Path.GetFileName(file);
\r
895 private void drp_dvdtitle_Click(object sender, EventArgs e)
\r
897 if ((drp_dvdtitle.Items.Count == 1) && (drp_dvdtitle.Items[0].ToString() == "Automatic"))
\r
898 MessageBox.Show("There are no titles to select. Please load a source file by clicking the 'Source' button above before trying to select a title.", "Alert", MessageBoxButtons.OK, MessageBoxIcon.Asterisk);
\r
900 private void drp_dvdtitle_SelectedIndexChanged(object sender, EventArgs e)
\r
902 UnRegisterPresetEventHandler();
\r
904 PictureSettings.lbl_Aspect.Text = "Select a Title"; // Reset some values on the form
\r
905 drop_chapterStart.Items.Clear();
\r
906 drop_chapterFinish.Items.Clear();
\r
908 // If the dropdown is set to automatic nothing else needs to be done.
\r
909 // Otheriwse if its not, title data has to be loased from parsing.
\r
910 if (drp_dvdtitle.Text != "Automatic")
\r
912 selectedTitle = drp_dvdtitle.SelectedItem as Title;
\r
913 lbl_duration.Text = selectedTitle.Duration.ToString();
\r
914 PictureSettings.Source = selectedTitle; // Setup Picture Settings Tab Control
\r
916 // Populate the Angles dropdown
\r
917 drop_angle.Items.Clear();
\r
918 if (!Properties.Settings.Default.noDvdNav)
\r
920 drop_angle.Visible = true;
\r
921 lbl_angle.Visible = true;
\r
922 drop_angle.Items.AddRange(selectedTitle.Angles.ToArray());
\r
923 if (drop_angle.Items.Count != 0)
\r
924 drop_angle.SelectedIndex = 0;
\r
928 drop_angle.Visible = false;
\r
929 lbl_angle.Visible = false;
\r
932 // Populate the Start chapter Dropdown
\r
933 drop_chapterStart.Items.Clear();
\r
934 drop_chapterStart.Items.AddRange(selectedTitle.Chapters.ToArray());
\r
935 if (drop_chapterStart.Items.Count > 0)
\r
936 drop_chapterStart.Text = drop_chapterStart.Items[0].ToString();
\r
938 // Populate the Final Chapter Dropdown
\r
939 drop_chapterFinish.Items.Clear();
\r
940 drop_chapterFinish.Items.AddRange(selectedTitle.Chapters.ToArray());
\r
941 if (drop_chapterFinish.Items.Count > 0)
\r
942 drop_chapterFinish.Text = drop_chapterFinish.Items[drop_chapterFinish.Items.Count - 1].ToString();
\r
944 // Populate the Audio Channels Dropdown
\r
945 AudioSettings.SetTrackList(selectedTitle);
\r
947 // Populate the Subtitles dropdown
\r
948 Subtitles.drp_subtitleTracks.Items.Clear();
\r
949 Subtitles.drp_subtitleTracks.Items.Add("Foreign Audio Search (Bitmap)");
\r
950 Subtitles.drp_subtitleTracks.Items.AddRange(selectedTitle.Subtitles.ToArray());
\r
951 Subtitles.drp_subtitleTracks.SelectedIndex = 0;
\r
953 Subtitles.SetSubtitleTrackAuto();
\r
955 // Update the source label if we have multiple streams
\r
956 if (selectedTitle != null)
\r
957 if (!string.IsNullOrEmpty(selectedTitle.SourceName))
\r
958 labelSource.Text = labelSource.Text = Path.GetFileName(selectedTitle.SourceName);
\r
960 // Run the autoName & chapterNaming functions
\r
961 if (Properties.Settings.Default.autoNaming)
\r
963 string autoPath = Main.autoName(this);
\r
964 if (autoPath != null)
\r
965 text_destination.Text = autoPath;
\r
967 MessageBox.Show("You currently have \"Automatically name output files\" enabled for the destination file box, but you do not have a default directory set.\n\nYou should set a \"Default Path\" in HandBrakes preferences. (See 'Tools' menu -> 'Options' -> 'General' Tab -> 'Default Path')", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
970 data_chpt.Rows.Clear();
\r
971 if (selectedTitle.Chapters.Count != 1)
\r
973 DataGridView chapterGridView = Main.chapterNaming(data_chpt, drop_chapterFinish.Text);
\r
974 if (chapterGridView != null)
\r
975 data_chpt = chapterGridView;
\r
979 Check_ChapterMarkers.Checked = false;
\r
980 Check_ChapterMarkers.Enabled = false;
\r
983 // Hack to force the redraw of the scrollbars which don't resize properly when the control is disabled.
\r
984 data_chpt.Columns[0].Width = 166;
\r
985 data_chpt.Columns[0].Width = 165;
\r
987 RegisterPresetEventHandler();
\r
989 private void chapersChanged(object sender, EventArgs e)
\r
991 Control ctl = (Control)sender;
\r
992 int chapterStart, chapterEnd;
\r
993 int.TryParse(drop_chapterStart.Text, out chapterStart);
\r
994 int.TryParse(drop_chapterFinish.Text, out chapterEnd);
\r
998 case "drop_chapterStart":
\r
999 if (drop_chapterFinish.SelectedIndex == -1 && drop_chapterFinish.Items.Count != 0)
\r
1000 drop_chapterFinish.SelectedIndex = drop_chapterFinish.Items.Count - 1;
\r
1002 if (chapterEnd != 0)
\r
1003 if (chapterStart > chapterEnd)
\r
1004 drop_chapterFinish.Text = chapterStart.ToString();
\r
1006 case "drop_chapterFinish":
\r
1007 if (drop_chapterStart.Items.Count >= 1 && drop_chapterStart.SelectedIndex == -1)
\r
1008 drop_chapterStart.SelectedIndex = 0;
\r
1010 if (chapterStart != 0)
\r
1011 if (chapterEnd < chapterStart)
\r
1012 drop_chapterFinish.Text = chapterStart.ToString();
\r
1014 // Add more rows to the Chapter menu if needed.
\r
1015 if (Check_ChapterMarkers.Checked)
\r
1017 int i = data_chpt.Rows.Count, finish = 0;
\r
1018 int.TryParse(drop_chapterFinish.Text, out finish);
\r
1020 while (i < finish)
\r
1022 int n = data_chpt.Rows.Add();
\r
1023 data_chpt.Rows[n].Cells[0].Value = (i + 1);
\r
1024 data_chpt.Rows[n].Cells[1].Value = "Chapter " + (i + 1);
\r
1025 data_chpt.Rows[n].Cells[0].ValueType = typeof(int);
\r
1026 data_chpt.Rows[n].Cells[1].ValueType = typeof(string);
\r
1033 // Update the Duration
\r
1034 lbl_duration.Text = Main.calculateDuration(drop_chapterStart.SelectedIndex, drop_chapterFinish.SelectedIndex, selectedTitle).ToString();
\r
1036 // Run the Autonaming function
\r
1037 if (Properties.Settings.Default.autoNaming)
\r
1038 text_destination.Text = Main.autoName(this);
\r
1040 // Disable chapter markers if only 1 chapter is selected.
\r
1041 if (chapterStart == chapterEnd)
\r
1043 Check_ChapterMarkers.Enabled = false;
\r
1044 btn_importChapters.Enabled = false;
\r
1045 data_chpt.Enabled = false;
\r
1049 Check_ChapterMarkers.Enabled = true;
\r
1050 if (Check_ChapterMarkers.Checked)
\r
1052 btn_importChapters.Enabled = true;
\r
1053 data_chpt.Enabled = true;
\r
1059 private void btn_destBrowse_Click(object sender, EventArgs e)
\r
1061 // This removes the file extension from the filename box on the save file dialog.
\r
1062 // It's daft but some users don't realise that typing an extension overrides the dropdown extension selected.
\r
1063 DVD_Save.FileName = Path.GetFileNameWithoutExtension(text_destination.Text);
\r
1065 if (Path.IsPathRooted(text_destination.Text))
\r
1066 DVD_Save.InitialDirectory = Path.GetDirectoryName(text_destination.Text);
\r
1068 // Show the dialog and set the main form file path
\r
1069 if (drop_format.SelectedIndex.Equals(0))
\r
1070 DVD_Save.FilterIndex = 1;
\r
1071 else if (drop_format.SelectedIndex.Equals(1))
\r
1072 DVD_Save.FilterIndex = 2;
\r
1074 if (DVD_Save.ShowDialog() == DialogResult.OK)
\r
1076 if (DVD_Save.FileName.StartsWith("\\"))
\r
1077 MessageBox.Show("Sorry, HandBrake does not support UNC file paths. \nTry mounting the share as a network drive in My Computer", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
1080 // Add a file extension manually, as FileDialog.AddExtension has issues with dots in filenames
\r
1081 switch (DVD_Save.FilterIndex)
\r
1084 if (!Path.GetExtension(DVD_Save.FileName).Equals(".mp4", StringComparison.InvariantCultureIgnoreCase))
\r
1085 if (Properties.Settings.Default.useM4v)
\r
1086 DVD_Save.FileName = DVD_Save.FileName.Replace(".mp4", ".m4v").Replace(".mkv", ".m4v");
\r
1088 DVD_Save.FileName = DVD_Save.FileName.Replace(".m4v", ".mp4").Replace(".mkv", ".mp4");
\r
1091 if (!Path.GetExtension(DVD_Save.FileName).Equals(".mkv", StringComparison.InvariantCultureIgnoreCase))
\r
1092 DVD_Save.FileName = DVD_Save.FileName.Replace(".mp4", ".mkv").Replace(".m4v", ".mkv");
\r
1098 text_destination.Text = DVD_Save.FileName;
\r
1100 // Quicktime requires .m4v file for chapter markers to work. If checked, change the extension to .m4v (mp4 and m4v are the same thing)
\r
1101 if (Check_ChapterMarkers.Checked && DVD_Save.FilterIndex != 2)
\r
1102 SetExtension(".m4v");
\r
1106 private void text_destination_TextChanged(object sender, EventArgs e)
\r
1108 string path = text_destination.Text;
\r
1109 if (path.EndsWith(".mp4") || path.EndsWith(".m4v"))
\r
1110 drop_format.SelectedIndex = 0;
\r
1111 else if (path.EndsWith(".mkv"))
\r
1112 drop_format.SelectedIndex = 1;
\r
1115 // Output Settings
\r
1116 private void drop_format_SelectedIndexChanged(object sender, EventArgs e)
\r
1118 switch (drop_format.SelectedIndex)
\r
1121 if (Properties.Settings.Default.useM4v || Check_ChapterMarkers.Checked || AudioSettings.RequiresM4V() || Subtitles.RequiresM4V())
\r
1122 SetExtension(".m4v");
\r
1124 SetExtension(".mp4");
\r
1127 SetExtension(".mkv");
\r
1131 AudioSettings.SetContainer(drop_format.Text);
\r
1132 Subtitles.SetContainer(drop_format.SelectedIndex);
\r
1134 if (drop_format.Text.Contains("MP4"))
\r
1136 if (drp_videoEncoder.Items.Contains("VP3 (Theora)"))
\r
1138 drp_videoEncoder.Items.Remove("VP3 (Theora)");
\r
1139 drp_videoEncoder.SelectedIndex = 1;
\r
1142 else if (drop_format.Text.Contains("MKV"))
\r
1143 drp_videoEncoder.Items.Add("VP3 (Theora)");
\r
1145 public void SetExtension(string newExtension)
\r
1147 if (newExtension == ".mp4" || newExtension == ".m4v")
\r
1148 if (Properties.Settings.Default.useM4v || Check_ChapterMarkers.Checked || AudioSettings.RequiresM4V() || Subtitles.RequiresM4V())
\r
1149 newExtension = ".m4v";
\r
1151 newExtension = ".mp4";
\r
1153 if (Path.HasExtension(newExtension))
\r
1154 text_destination.Text = Path.ChangeExtension(text_destination.Text, newExtension);
\r
1158 private void drp_videoEncoder_SelectedIndexChanged(object sender, EventArgs e)
\r
1160 setContainerOpts();
\r
1162 //Turn off some options which are H.264 only when the user selects a non h.264 encoder
\r
1163 if (drp_videoEncoder.Text.Contains("H.264"))
\r
1165 if (check_2PassEncode.CheckState == CheckState.Checked)
\r
1166 check_turbo.Enabled = true;
\r
1168 tab_advanced.Enabled = true;
\r
1169 if ((drop_format.Text.Contains("MP4")) || (drop_format.Text.Contains("M4V")))
\r
1170 check_iPodAtom.Enabled = true;
\r
1172 check_iPodAtom.Enabled = false;
\r
1176 check_turbo.CheckState = CheckState.Unchecked;
\r
1177 check_turbo.Enabled = false;
\r
1178 tab_advanced.Enabled = false;
\r
1179 x264Panel.x264Query = "";
\r
1180 check_iPodAtom.Enabled = false;
\r
1181 check_iPodAtom.Checked = false;
\r
1184 // Setup the CQ Slider
\r
1185 switch (drp_videoEncoder.Text)
\r
1187 case "MPEG-4 (FFmpeg)":
\r
1188 if (slider_videoQuality.Value > 31)
\r
1189 slider_videoQuality.Value = 20; // Just reset to 70% QP 10 on encode change.
\r
1190 slider_videoQuality.Minimum = 1;
\r
1191 slider_videoQuality.Maximum = 31;
\r
1193 case "H.264 (x264)":
\r
1194 slider_videoQuality.Minimum = 0;
\r
1195 slider_videoQuality.TickFrequency = 1;
\r
1197 CultureInfo culture = CultureInfo.CreateSpecificCulture("en-US");
\r
1198 double cqStep = Properties.Settings.Default.x264cqstep;
\r
1199 double multiplier = 1.0 / cqStep;
\r
1200 double value = slider_videoQuality.Value * multiplier;
\r
1202 switch (Properties.Settings.Default.x264cqstep.ToString(culture))
\r
1205 slider_videoQuality.Maximum = 255;
\r
1208 slider_videoQuality.Maximum = 204;
\r
1211 slider_videoQuality.Maximum = 102;
\r
1214 slider_videoQuality.Maximum = 51;
\r
1217 slider_videoQuality.Maximum = 51;
\r
1220 if (value < slider_videoQuality.Maximum)
\r
1221 slider_videoQuality.Value = slider_videoQuality.Maximum - (int)value;
\r
1224 case "VP3 (Theora)":
\r
1225 if (slider_videoQuality.Value > 63)
\r
1226 slider_videoQuality.Value = 45; // Just reset to 70% QP 45 on encode change.
\r
1227 slider_videoQuality.Minimum = 0;
\r
1228 slider_videoQuality.Maximum = 63;
\r
1233 /// Set the container format options
\r
1235 public void setContainerOpts()
\r
1237 if ((drop_format.Text.Contains("MP4")) || (drop_format.Text.Contains("M4V")))
\r
1239 check_largeFile.Enabled = true;
\r
1240 check_optimiseMP4.Enabled = true;
\r
1241 check_iPodAtom.Enabled = true;
\r
1245 check_largeFile.Enabled = false;
\r
1246 check_optimiseMP4.Enabled = false;
\r
1247 check_iPodAtom.Enabled = false;
\r
1248 check_largeFile.Checked = false;
\r
1249 check_optimiseMP4.Checked = false;
\r
1250 check_iPodAtom.Checked = false;
\r
1253 private double _cachedCqStep = Properties.Settings.Default.x264cqstep;
\r
1255 /// Update the CQ slider for x264 for a new CQ step. This is set from option
\r
1257 public void setQualityFromSlider()
\r
1259 // Work out the current RF value.
\r
1260 double cqStep = _cachedCqStep;
\r
1261 double rfValue = 51.0 - slider_videoQuality.Value * cqStep;
\r
1263 // Change the maximum value for the slider
\r
1264 switch (Properties.Settings.Default.x264cqstep.ToString(new CultureInfo("en-US")))
\r
1267 slider_videoQuality.Maximum = 255;
\r
1270 slider_videoQuality.Maximum = 204;
\r
1273 slider_videoQuality.Maximum = 102;
\r
1276 slider_videoQuality.Maximum = 51;
\r
1279 slider_videoQuality.Maximum = 51;
\r
1283 // Reset the CQ slider to RF0
\r
1284 slider_videoQuality.Value = slider_videoQuality.Maximum;
\r
1286 // Reset the CQ slider back to the previous value as close as possible
\r
1287 double cqStepNew = Properties.Settings.Default.x264cqstep;
\r
1288 double rfValueCurrent = 51.0 - slider_videoQuality.Value * cqStepNew;
\r
1289 while (rfValueCurrent < rfValue)
\r
1291 slider_videoQuality.Value--;
\r
1292 rfValueCurrent = 51.0 - slider_videoQuality.Value * cqStepNew;
\r
1295 // Cache the CQ step for the next calculation
\r
1296 _cachedCqStep = Properties.Settings.Default.x264cqstep;
\r
1298 private void slider_videoQuality_Scroll(object sender, EventArgs e)
\r
1300 double cqStep = Properties.Settings.Default.x264cqstep;
\r
1301 switch (drp_videoEncoder.Text)
\r
1303 case "MPEG-4 (FFmpeg)":
\r
1304 double rfValue = 31 - (slider_videoQuality.Value - 1);
\r
1305 double max = slider_videoQuality.Maximum;
\r
1306 double min = slider_videoQuality.Minimum;
\r
1307 double val = ((max - min) - (rfValue - min)) / (max - min);
\r
1308 SliderValue.Text = Math.Round((val * 100), 2).ToString(new CultureInfo("en-US")) + "% QP:" + (32 - slider_videoQuality.Value);
\r
1310 case "H.264 (x264)":
\r
1311 rfValue = 51.0 - slider_videoQuality.Value * cqStep;
\r
1312 max = slider_videoQuality.Maximum * cqStep;
\r
1313 min = slider_videoQuality.Minimum;
\r
1314 val = ((max - min) - (rfValue - min)) / (max - min);
\r
1315 rfValue = Math.Round(rfValue, 2);
\r
1316 SliderValue.Text = Math.Round((val * 100), 2).ToString(new CultureInfo("en-US")) + "% RF:" + rfValue.ToString(new CultureInfo("en-US"));
\r
1318 case "VP3 (Theora)":
\r
1319 rfValue = slider_videoQuality.Value;
\r
1320 double value = rfValue / 63;
\r
1321 SliderValue.Text = Math.Round((value * 100), 2).ToString(new CultureInfo("en-US")) + "% QP:" + slider_videoQuality.Value;
\r
1325 private void radio_targetFilesize_CheckedChanged(object sender, EventArgs e)
\r
1327 text_bitrate.Enabled = false;
\r
1328 text_filesize.Enabled = true;
\r
1329 slider_videoQuality.Enabled = false;
\r
1331 check_2PassEncode.Enabled = true;
\r
1333 private void radio_avgBitrate_CheckedChanged(object sender, EventArgs e)
\r
1335 text_bitrate.Enabled = true;
\r
1336 text_filesize.Enabled = false;
\r
1337 slider_videoQuality.Enabled = false;
\r
1339 check_2PassEncode.Enabled = true;
\r
1341 private void radio_cq_CheckedChanged(object sender, EventArgs e)
\r
1343 text_bitrate.Enabled = false;
\r
1344 text_filesize.Enabled = false;
\r
1345 slider_videoQuality.Enabled = true;
\r
1347 check_2PassEncode.Enabled = false;
\r
1348 check_2PassEncode.CheckState = CheckState.Unchecked;
\r
1350 private void check_2PassEncode_CheckedChanged(object sender, EventArgs e)
\r
1352 if (check_2PassEncode.CheckState.ToString() == "Checked")
\r
1354 if (drp_videoEncoder.Text.Contains("H.264"))
\r
1355 check_turbo.Enabled = true;
\r
1359 check_turbo.Enabled = false;
\r
1360 check_turbo.CheckState = CheckState.Unchecked;
\r
1364 // Chapter Marker Tab
\r
1365 private void Check_ChapterMarkers_CheckedChanged(object sender, EventArgs e)
\r
1367 if (Check_ChapterMarkers.Checked)
\r
1369 if (drop_format.SelectedIndex != 1)
\r
1370 SetExtension(".m4v");
\r
1371 data_chpt.Enabled = true;
\r
1372 btn_importChapters.Enabled = true;
\r
1376 if (drop_format.SelectedIndex != 1 && !Properties.Settings.Default.useM4v)
\r
1377 SetExtension(".mp4");
\r
1378 data_chpt.Enabled = false;
\r
1379 btn_importChapters.Enabled = false;
\r
1382 private void btn_importChapters_Click(object sender, EventArgs e)
\r
1384 if (File_ChapterImport.ShowDialog() == DialogResult.OK)
\r
1386 String filename = File_ChapterImport.FileName;
\r
1387 DataGridView imported = Main.importChapterNames(data_chpt, filename);
\r
1388 if (imported != null)
\r
1389 data_chpt = imported;
\r
1392 private void mnu_resetChapters_Click(object sender, EventArgs e)
\r
1394 data_chpt.Rows.Clear();
\r
1395 DataGridView chapterGridView = Main.chapterNaming(data_chpt, drop_chapterFinish.Text);
\r
1396 if (chapterGridView != null)
\r
1398 data_chpt = chapterGridView;
\r
1402 // Query Editor Tab
\r
1403 private void btn_generate_Query_Click(object sender, EventArgs e)
\r
1405 rtf_query.Text = queryGen.GenerateCLIQuery(this, 0, null);
\r
1407 private void btn_clear_Click(object sender, EventArgs e)
\r
1409 rtf_query.Clear();
\r
1413 // MainWindow Components, Actions and Functions ***********************
\r
1415 #region Source Scan
\r
1416 public Boolean isScanning { get; set; }
\r
1417 private static int scanProcessID { get; set; }
\r
1418 private void startScan(String filename)
\r
1420 // Setup the GUI components for the scan.
\r
1421 sourcePath = filename;
\r
1422 foreach (Control ctrl in Controls)
\r
1423 if (!(ctrl is StatusStrip || ctrl is MenuStrip || ctrl is ToolStrip))
\r
1424 ctrl.Enabled = false;
\r
1426 lbl_encode.Visible = true;
\r
1427 lbl_encode.Text = "Scanning ...";
\r
1428 btn_source.Enabled = false;
\r
1429 btn_start.Enabled = false;
\r
1430 btn_showQueue.Enabled = false;
\r
1431 btn_add2Queue.Enabled = false;
\r
1432 tb_preview.Enabled = false;
\r
1433 mnu_killCLI.Visible = true;
\r
1435 // Start hte Scan Thread
\r
1438 if (ActivityWindow != null)
\r
1439 ActivityWindow.SetLogView(true);
\r
1440 isScanning = true;
\r
1441 ThreadPool.QueueUserWorkItem(scanProcess);
\r
1443 catch (Exception exc)
\r
1445 MessageBox.Show("frmMain.cs - startScan " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1448 private void scanProcess(object state)
\r
1452 string handbrakeCLIPath = Path.Combine(Application.StartupPath, "HandBrakeCLI.exe");
\r
1453 string logDir = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData) + "\\HandBrake\\logs";
\r
1454 string dvdInfoPath = Path.Combine(logDir, "last_scan_log.txt");
\r
1456 // Make we don't pick up a stale last_encode_log.txt (and that we have rights to the file)
\r
1457 if (File.Exists(dvdInfoPath))
\r
1458 File.Delete(dvdInfoPath);
\r
1460 String dvdnav = string.Empty;
\r
1461 if (Properties.Settings.Default.noDvdNav)
\r
1462 dvdnav = " --no-dvdnav";
\r
1463 string strCmdLine = String.Format(@"cmd /c """"{0}"" -i ""{1}"" -t0 {2} -v >""{3}"" 2>&1""", handbrakeCLIPath, sourcePath, dvdnav, dvdInfoPath);
\r
1465 ProcessStartInfo hbParseDvd = new ProcessStartInfo("CMD.exe", strCmdLine) { WindowStyle = ProcessWindowStyle.Hidden };
\r
1467 Boolean cleanExit = true;
\r
1468 using (hbproc = Process.Start(hbParseDvd))
\r
1470 Process[] before = Process.GetProcesses(); // Get a list of running processes before starting.
\r
1471 scanProcessID = Main.getCliProcess(before);
\r
1472 hbproc.WaitForExit();
\r
1473 if (hbproc.ExitCode != 0)
\r
1474 cleanExit = false;
\r
1477 if (cleanExit) // If 0 exit code, CLI exited cleanly.
\r
1479 if (!File.Exists(dvdInfoPath))
\r
1480 throw new Exception("Unable to retrieve the DVD Info. last_scan_log.txt is missing. \nExpected location of last_scan_log.txt: \n"
\r
1483 using (StreamReader sr = new StreamReader(dvdInfoPath))
\r
1485 thisDVD = DVD.Parse(sr);
\r
1490 updateUIafterScan();
\r
1493 catch (Exception exc)
\r
1495 MessageBox.Show("frmMain.cs - scanProcess() " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1499 private void updateUIafterScan()
\r
1503 if (InvokeRequired)
\r
1505 BeginInvoke(new UpdateWindowHandler(updateUIafterScan));
\r
1509 // Setup some GUI components
\r
1510 drp_dvdtitle.Items.Clear();
\r
1511 if (thisDVD.Titles.Count != 0)
\r
1512 drp_dvdtitle.Items.AddRange(thisDVD.Titles.ToArray());
\r
1514 // Now select the longest title
\r
1515 if (thisDVD.Titles.Count != 0)
\r
1516 drp_dvdtitle.SelectedItem = Main.selectLongestTitle(thisDVD);
\r
1518 // Enable the creation of chapter markers if the file is an image of a dvd.
\r
1519 if (sourcePath.ToLower().Contains(".iso") || sourcePath.Contains("VIDEO_TS"))
\r
1520 Check_ChapterMarkers.Enabled = true;
\r
1521 else if (Directory.Exists(Path.Combine(sourcePath, "VIDEO_TS")))
\r
1522 Check_ChapterMarkers.Enabled = true;
\r
1525 Check_ChapterMarkers.Enabled = false;
\r
1526 Check_ChapterMarkers.Checked = false;
\r
1527 data_chpt.Rows.Clear();
\r
1530 // If no titles were found, Display an error message
\r
1531 if (drp_dvdtitle.Items.Count == 0)
\r
1534 "No Title(s) found. \n\nYour Source may be copy protected, badly mastered or in a format which HandBrake does not support. \nPlease refer to the Documentation and FAQ (see Help Menu).",
\r
1535 "Error", MessageBoxButtons.OK, MessageBoxIcon.Hand);
\r
1536 sourcePath = string.Empty;
\r
1538 UpdateSourceLabel();
\r
1540 // Enable the GUI components and enable any disabled components
\r
1543 catch (Exception exc)
\r
1545 MessageBox.Show("frmMain.cs - updateUIafterScan " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1549 private void enableGUI()
\r
1553 if (InvokeRequired)
\r
1554 BeginInvoke(new UpdateWindowHandler(enableGUI));
\r
1555 lbl_encode.Text = "Scan Completed";
\r
1556 foreach (Control ctrl in Controls)
\r
1557 ctrl.Enabled = true;
\r
1558 btn_start.Enabled = true;
\r
1559 btn_showQueue.Enabled = true;
\r
1560 btn_add2Queue.Enabled = true;
\r
1561 tb_preview.Enabled = true;
\r
1562 btn_source.Enabled = true;
\r
1563 mnu_killCLI.Visible = false;
\r
1565 catch (Exception exc)
\r
1567 MessageBox.Show("frmMain.cs - enableGUI() " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1570 private void killScan()
\r
1577 Process[] prs = Process.GetProcesses();
\r
1578 foreach (Process process in prs)
\r
1580 if (process.Id == scanProcessID)
\r
1582 process.Refresh();
\r
1583 if (!process.HasExited)
\r
1588 catch (Exception ex)
\r
1590 MessageBox.Show("Unable to kill HandBrakeCLI.exe \nYou may need to manually kill HandBrakeCLI.exe using the Windows Task Manager if it does not close automatically within the next few minutes. \n\nError Information: \n" + ex.Message, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1593 private void resetGUI()
\r
1595 drp_dvdtitle.Items.Clear();
\r
1596 drop_chapterStart.Items.Clear();
\r
1597 drop_chapterFinish.Items.Clear();
\r
1598 lbl_duration.Text = "Select a Title";
\r
1599 PictureSettings.lbl_src_res.Text = "Select a Title";
\r
1600 PictureSettings.lbl_Aspect.Text = "Select a Title";
\r
1601 sourcePath = String.Empty;
\r
1602 text_destination.Text = String.Empty;
\r
1604 selectedTitle = null;
\r
1605 isScanning = false;
\r
1607 private void UpdateSourceLabel()
\r
1609 labelSource.Text = string.IsNullOrEmpty(sourcePath) ? "Select \"Source\" to continue." : this.SourceName;
\r
1611 if (selectedTitle != null)
\r
1612 if (!string.IsNullOrEmpty(selectedTitle.SourceName)) // If it's one of multiple source files, make sure we don't use the folder name
\r
1613 labelSource.Text = Path.GetFileName(selectedTitle.SourceName);
\r
1619 /// Set the GUI to it's finished encoding state.
\r
1621 private void setEncodeFinished()
\r
1625 if (InvokeRequired)
\r
1627 BeginInvoke(new UpdateWindowHandler(setEncodeFinished));
\r
1631 lbl_encode.Text = "Encoding Finished";
\r
1632 btn_start.Text = "Start";
\r
1633 btn_start.ToolTipText = "Start the encoding process";
\r
1634 btn_start.Image = Properties.Resources.Play;
\r
1636 // If the window is minimized, display the notification in a popup.
\r
1637 if (Properties.Settings.Default.trayIconAlerts)
\r
1638 if (FormWindowState.Minimized == this.WindowState)
\r
1640 notifyIcon.BalloonTipText = lbl_encode.Text;
\r
1641 notifyIcon.ShowBalloonTip(500);
\r
1644 catch (Exception exc)
\r
1646 MessageBox.Show(exc.ToString());
\r
1651 /// Set the GUI to it's started encoding state.
\r
1653 private void setEncodeStarted()
\r
1657 if (InvokeRequired)
\r
1659 BeginInvoke(new UpdateWindowHandler(setEncodeStarted));
\r
1663 lbl_encode.Visible = true;
\r
1664 lbl_encode.Text = "Encoding with " + encodeQueue.Count + " encode(s) pending";
\r
1665 btn_start.Text = "Stop";
\r
1666 btn_start.ToolTipText = "Stop the encoding process.";
\r
1667 btn_start.Image = Properties.Resources.stop;
\r
1669 catch (Exception exc)
\r
1671 MessageBox.Show(exc.ToString());
\r
1676 #region DVD Drive Detection
\r
1677 private void getDriveInfoThread()
\r
1681 if (InvokeRequired)
\r
1683 BeginInvoke(new UpdateWindowHandler(getDriveInfoThread));
\r
1687 Boolean foundDrive = false;
\r
1688 DriveInfo[] theCollectionOfDrives = DriveInfo.GetDrives();
\r
1689 foreach (DriveInfo curDrive in theCollectionOfDrives)
\r
1691 if (curDrive.DriveType == DriveType.CDRom && curDrive.IsReady)
\r
1693 if (File.Exists(curDrive.RootDirectory + "VIDEO_TS\\VIDEO_TS.IFO"))
\r
1695 this.dvdDrivePath = curDrive.RootDirectory + "VIDEO_TS";
\r
1696 this.dvdDriveLabel = curDrive.VolumeLabel;
\r
1697 mnu_dvd_drive.Text = this.dvdDrivePath + " (" + this.dvdDriveLabel + ")";
\r
1698 foundDrive = true;
\r
1704 if (foundDrive == false)
\r
1705 mnu_dvd_drive.Text = "[No DVD Drive Ready]";
\r
1709 mnu_dvd_drive.Text = "[No DVD Drive Ready / Found]";
\r
1714 #region Public Methods
\r
1716 /// Access the preset Handler and setup the preset panel.
\r
1718 public void loadPresetPanel()
\r
1720 if (presetHandler.CheckIfPresetsAreOutOfDate())
\r
1721 if (!Properties.Settings.Default.presetNotification)
\r
1722 MessageBox.Show(splash,
\r
1723 "HandBrake has determined your built-in presets are out of date... These presets will now be updated.",
\r
1724 "Preset Update", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
1726 presetHandler.GetPresetPanel(ref treeView_presets);
\r
1727 treeView_presets.Update();
\r
1733 /// If the queue is being processed, prompt the user to confirm application close.
\r
1735 /// <param name="e"></param>
\r
1736 protected override void OnFormClosing(FormClosingEventArgs e)
\r
1738 // If currently encoding, the queue isn't paused, and there are queue items to process, prompt to confirm close.
\r
1739 if ((encodeQueue.isEncoding) && (!encodeQueue.PauseRequested) && (encodeQueue.Count > 0))
\r
1741 DialogResult result = MessageBox.Show("HandBrake has queue items to process. Closing HandBrake will not stop the current encoding, but will stop processing the queue.\n\nDo you want to close HandBrake?",
\r
1742 "Close HandBrake?", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
1743 if (result == DialogResult.No)
\r
1746 base.OnFormClosing(e);
\r
1750 #region In-GUI Encode Status (Experimental)
\r
1751 private void encodeMonitorThread()
\r
1755 Parser encode = new Parser(encodeQueue.hbProcess.StandardOutput.BaseStream);
\r
1756 encode.OnEncodeProgress += encodeOnEncodeProgress;
\r
1757 while (!encode.EndOfStream)
\r
1758 encode.readEncodeStatus();
\r
1760 catch (Exception exc)
\r
1762 MessageBox.Show(exc.ToString(), "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1765 private void encodeOnEncodeProgress(object Sender, int CurrentTask, int TaskCount, float PercentComplete, float CurrentFps, float AverageFps, TimeSpan TimeRemaining)
\r
1767 if (this.InvokeRequired)
\r
1769 this.BeginInvoke(new EncodeProgressEventHandler(encodeOnEncodeProgress),
\r
1770 new object[] { Sender, CurrentTask, TaskCount, PercentComplete, CurrentFps, AverageFps, TimeRemaining });
\r
1773 lbl_encode.Text = string.Format("Encode Progress: {0}%, FPS: {1}, Avg FPS: {2}, Time Remaining: {3} ", PercentComplete, CurrentFps, AverageFps, TimeRemaining);
\r
1778 private enum SourceType
\r
1787 // This is the END of the road ****************************************
\r