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
39 // Delegates **********************************************************
\r
40 private delegate void UpdateWindowHandler();
\r
42 // Applicaiton Startup ************************************************
\r
44 #region Application Startup
\r
47 // Load and setup the splash screen in this thread
\r
48 splash = new frmSplashScreen();
\r
50 Label lblStatus = new Label { Size = new Size(150, 20), Location = new Point(182, 102) };
\r
51 splash.Controls.Add(lblStatus);
\r
53 InitializeComponent();
\r
55 // Update the users config file with the CLI version data.
\r
56 lblStatus.Text = "Setting Version Data ...";
\r
57 Application.DoEvents();
\r
58 Main.setCliVersionData();
\r
60 // Show the form, but leave disabled until preloading is complete then show the main form
\r
61 this.Enabled = false;
\r
63 Application.DoEvents(); // Forces frmMain to draw
\r
65 // Check for new versions, if update checking is enabled
\r
66 if (Properties.Settings.Default.updateStatus)
\r
68 DateTime now = DateTime.Now;
\r
69 DateTime lastCheck = Properties.Settings.Default.lastUpdateCheckDate;
\r
70 TimeSpan elapsed = now.Subtract(lastCheck);
\r
71 if (elapsed.TotalDays > Properties.Settings.Default.daysBetweenUpdateCheck)
\r
73 lblStatus.Text = "Checking for updates ...";
\r
74 Application.DoEvents();
\r
76 Main.BeginCheckForUpdates(new AsyncCallback(UpdateCheckDone), false);
\r
80 // Setup the GUI components
\r
81 lblStatus.Text = "Setting up the GUI ...";
\r
82 Application.DoEvents();
\r
83 loadPresetPanel(); // Load the Preset Panel
\r
84 treeView_presets.ExpandAll();
\r
85 lbl_encode.Text = "";
\r
86 queueWindow = new frmQueue(encodeQueue); // Prepare the Queue
\r
87 if (!Properties.Settings.Default.QueryEditorTab)
\r
88 tabs_panel.TabPages.RemoveAt(7); // Remove the query editor tab if the user does not want it enabled.
\r
90 // Load the user's default settings or Normal Preset
\r
91 if (Properties.Settings.Default.defaultSettings && Properties.Settings.Default.defaultPreset != "")
\r
93 if (presetHandler.getPreset(Properties.Settings.Default.defaultPreset) != null)
\r
95 string query = presetHandler.getPreset(Properties.Settings.Default.defaultPreset).Query;
\r
96 Boolean loadPictureSettings = presetHandler.getPreset(Properties.Settings.Default.defaultPreset).PictureSettings;
\r
100 //Ok, Reset all the H264 widgets before changing the preset
\r
101 x264Panel.reset2Defaults();
\r
103 // Send the query from the file to the Query Parser class, then load the preset
\r
104 QueryParser presetQuery = QueryParser.Parse(query);
\r
105 PresetLoader.presetLoader(this, presetQuery, Properties.Settings.Default.defaultPreset, loadPictureSettings);
\r
107 // The x264 widgets will need updated, so do this now:
\r
108 x264Panel.X264_StandardizeOptString();
\r
109 x264Panel.X264_SetCurrentSettingsInPanel();
\r
113 loadNormalPreset();
\r
116 loadNormalPreset();
\r
118 // Enabled GUI tooltip's if Required
\r
119 if (Properties.Settings.Default.tooltipEnable)
\r
120 ToolTip.Active = true;
\r
122 // Register with Growl (if not using Growl for the encoding completion action, this wont hurt anything)
\r
123 GrowlCommunicator.Register();
\r
126 lblStatus.Text = "Loading Complete!";
\r
127 Application.DoEvents();
\r
130 this.Enabled = true;
\r
132 // Event Handlers and Queue Recovery
\r
137 private void UpdateCheckDone(IAsyncResult result)
\r
139 if (InvokeRequired)
\r
141 Invoke(new MethodInvoker(() => UpdateCheckDone(result)));
\r
145 UpdateCheckInformation info;
\r
149 info = Main.EndCheckForUpdates(result);
\r
151 if (info.NewVersionAvailable)
\r
153 frmUpdater updateWindow = new frmUpdater(info.BuildInformation);
\r
154 updateWindow.ShowDialog();
\r
157 catch (Exception ex)
\r
159 if ((bool)result.AsyncState)
\r
160 MessageBox.Show("Unable to check for updates, Please try again later.\n\nDetailed Error Information:\n" + ex, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
164 // Startup Functions
\r
165 private void queueRecovery()
\r
167 if (Main.checkQueueRecovery())
\r
169 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
171 if (result == DialogResult.Yes)
\r
172 encodeQueue.LoadQueueFromFile("hb_queue_recovery.xml"); // Start Recovery
\r
175 // Remove the Queue recovery file if the user doesn't want to recovery the last queue.
\r
176 string queuePath = Path.Combine(Path.GetTempPath(), "hb_queue_recovery.xml");
\r
177 if (File.Exists(queuePath))
\r
178 File.Delete(queuePath);
\r
185 // Encoding Events for setting up the GUI
\r
186 private void events()
\r
188 // Handle Widget changes when preset is selected.
\r
189 RegisterPresetEventHandler();
\r
191 // Handle Window Resize
\r
192 if (Properties.Settings.Default.MainWindowMinimize)
\r
193 this.Resize += new EventHandler(frmMain_Resize);
\r
195 // Handle Encode Start / Finish / Pause
\r
196 encodeQueue.CurrentJobCompleted += new EventHandler(encodeEnded);
\r
197 encodeQueue.QueuePauseRequested += new EventHandler(encodePaused);
\r
198 encodeQueue.NewJobStarted += new EventHandler(encodeStarted);
\r
200 // Handle a file being draged onto the GUI.
\r
201 this.DragEnter += new DragEventHandler(frmMain_DragEnter);
\r
202 this.DragDrop += new DragEventHandler(frmMain_DragDrop);
\r
205 // 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
206 private void RegisterPresetEventHandler()
\r
209 drop_format.SelectedIndexChanged += new EventHandler(changePresetLabel);
\r
210 check_largeFile.CheckedChanged += new EventHandler(changePresetLabel);
\r
211 check_iPodAtom.CheckedChanged += new EventHandler(changePresetLabel);
\r
212 check_optimiseMP4.CheckedChanged += new EventHandler(changePresetLabel);
\r
214 // Picture Settings
\r
215 //PictureSettings.PictureSettingsChanged += new EventHandler(changePresetLabel);
\r
218 Filters.FilterSettingsChanged += new EventHandler(changePresetLabel);
\r
221 drp_videoEncoder.SelectedIndexChanged += new EventHandler(changePresetLabel);
\r
222 check_2PassEncode.CheckedChanged += new EventHandler(changePresetLabel);
\r
223 check_turbo.CheckedChanged += new EventHandler(changePresetLabel);
\r
224 text_filesize.TextChanged += new EventHandler(changePresetLabel);
\r
225 text_bitrate.TextChanged += new EventHandler(changePresetLabel);
\r
226 slider_videoQuality.ValueChanged += new EventHandler(changePresetLabel);
\r
229 AudioSettings.AudioListChanged += new EventHandler(changePresetLabel);
\r
232 x264Panel.rtf_x264Query.TextChanged += new EventHandler(changePresetLabel);
\r
234 private void UnRegisterPresetEventHandler()
\r
236 // Output Settings
\r
237 drop_format.SelectedIndexChanged -= new EventHandler(changePresetLabel);
\r
238 check_largeFile.CheckedChanged -= new EventHandler(changePresetLabel);
\r
239 check_iPodAtom.CheckedChanged -= new EventHandler(changePresetLabel);
\r
240 check_optimiseMP4.CheckedChanged -= new EventHandler(changePresetLabel);
\r
242 // Picture Settings
\r
243 //PictureSettings.PictureSettingsChanged -= new EventHandler(changePresetLabel);
\r
246 Filters.FilterSettingsChanged -= new EventHandler(changePresetLabel);
\r
249 drp_videoEncoder.SelectedIndexChanged -= new EventHandler(changePresetLabel);
\r
250 check_2PassEncode.CheckedChanged -= new EventHandler(changePresetLabel);
\r
251 check_turbo.CheckedChanged -= new EventHandler(changePresetLabel);
\r
252 text_filesize.TextChanged -= new EventHandler(changePresetLabel);
\r
253 text_bitrate.TextChanged -= new EventHandler(changePresetLabel);
\r
254 slider_videoQuality.ValueChanged -= new EventHandler(changePresetLabel);
\r
257 AudioSettings.AudioListChanged -= new EventHandler(changePresetLabel);
\r
260 x264Panel.rtf_x264Query.TextChanged -= new EventHandler(changePresetLabel);
\r
262 private void changePresetLabel(object sender, EventArgs e)
\r
264 labelPreset.Text = "Output Settings (Preset: Custom)";
\r
267 private static void frmMain_DragEnter(object sender, DragEventArgs e)
\r
269 if (e.Data.GetDataPresent(DataFormats.FileDrop, false))
\r
270 e.Effect = DragDropEffects.All;
\r
272 private void frmMain_DragDrop(object sender, DragEventArgs e)
\r
274 string[] fileList = e.Data.GetData(DataFormats.FileDrop) as string[];
\r
275 sourcePath = string.Empty;
\r
277 if (fileList != null)
\r
279 if (fileList[0].StartsWith("\\"))
\r
282 "Sorry, HandBrake does not support UNC file paths. \nTry mounting the network share as a network drive in My Computer",
\r
283 "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
284 UpdateSourceLabel();
\r
288 if (fileList[0] != "")
\r
289 startScan(fileList[0]);
\r
291 UpdateSourceLabel();
\r
295 UpdateSourceLabel();
\r
297 private void encodeStarted(object sender, EventArgs e)
\r
299 lastAction = "encode";
\r
300 setEncodeStarted();
\r
302 // Experimental HBProc Process Monitoring.
\r
303 if (Properties.Settings.Default.enocdeStatusInGui)
\r
305 Thread encodeMon = new Thread(encodeMonitorThread);
\r
309 private void encodeEnded(object sender, EventArgs e)
\r
311 setEncodeFinished();
\r
313 private void encodePaused(object sender, EventArgs e)
\r
315 setEncodeFinished();
\r
319 // User Interface Menus / Tool Strips *********************************
\r
322 private void mnu_killCLI_Click(object sender, EventArgs e)
\r
326 private void mnu_exit_Click(object sender, EventArgs e)
\r
328 Application.Exit();
\r
333 private void mnu_encode_Click(object sender, EventArgs e)
\r
335 queueWindow.Show();
\r
337 private void mnu_encodeLog_Click(object sender, EventArgs e)
\r
339 String file = lastAction == "scan" ? "last_scan_log.txt" : "last_encode_log.txt";
\r
341 frmActivityWindow dvdInfoWindow = new frmActivityWindow(file, encodeQueue, this);
\r
342 dvdInfoWindow.Show();
\r
344 private void mnu_options_Click(object sender, EventArgs e)
\r
346 Form options = new frmOptions(this);
\r
347 options.ShowDialog();
\r
351 #region Presets Menu
\r
352 private void mnu_presetReset_Click(object sender, EventArgs e)
\r
354 presetHandler.updateBuiltInPresets();
\r
356 if (treeView_presets.Nodes.Count == 0)
\r
357 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
359 MessageBox.Show("Presets have been updated!", "Alert", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
361 treeView_presets.ExpandAll();
\r
363 private void mnu_delete_preset_Click(object sender, EventArgs e)
\r
365 presetHandler.removeBuiltInPresets();
\r
366 loadPresetPanel(); // Reload the preset panel
\r
368 private void mnu_SelectDefault_Click(object sender, EventArgs e)
\r
370 loadNormalPreset();
\r
372 private void mnu_importMacPreset_Click(object sender, EventArgs e)
\r
376 private void btn_new_preset_Click(object sender, EventArgs e)
\r
378 Form preset = new frmAddPreset(this, queryGen.generateCLIQuery(this, 0, null), presetHandler);
\r
379 preset.ShowDialog();
\r
384 private void mnu_handbrake_forums_Click(object sender, EventArgs e)
\r
386 Process.Start("http://forum.handbrake.fr/");
\r
388 private void mnu_user_guide_Click(object sender, EventArgs e)
\r
390 Process.Start("http://trac.handbrake.fr/wiki/HandBrakeGuide");
\r
392 private void mnu_handbrake_home_Click(object sender, EventArgs e)
\r
394 Process.Start("http://handbrake.fr");
\r
396 private void mnu_UpdateCheck_Click(object sender, EventArgs e)
\r
398 lbl_updateCheck.Visible = true;
\r
399 Main.BeginCheckForUpdates(new AsyncCallback(updateCheckDoneMenu), false);
\r
401 private void updateCheckDoneMenu(IAsyncResult result)
\r
403 // Make sure it's running on the calling thread
\r
404 if (InvokeRequired)
\r
406 Invoke(new MethodInvoker(() => updateCheckDoneMenu(result)));
\r
409 UpdateCheckInformation info;
\r
412 // Get the information about the new build, if any, and close the window
\r
413 info = Main.EndCheckForUpdates(result);
\r
415 if (info.NewVersionAvailable && info.BuildInformation != null)
\r
417 frmUpdater updateWindow = new frmUpdater(info.BuildInformation);
\r
418 updateWindow.ShowDialog();
\r
421 MessageBox.Show("There are no new updates at this time.", "Update Check", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
422 lbl_updateCheck.Visible = false;
\r
425 catch (Exception ex)
\r
427 if ((bool)result.AsyncState)
\r
428 MessageBox.Show("Unable to check for updates, Please try again later.\n\nDetailed Error Information:\n" + ex, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
431 private void mnu_about_Click(object sender, EventArgs e)
\r
433 using (frmAbout About = new frmAbout())
\r
435 About.ShowDialog();
\r
441 // Right Click Menu Code
\r
442 private void pmnu_expandAll_Click(object sender, EventArgs e)
\r
444 treeView_presets.ExpandAll();
\r
446 private void pmnu_collapse_Click(object sender, EventArgs e)
\r
448 treeView_presets.CollapseAll();
\r
450 private void pmnu_import_Click(object sender, EventArgs e)
\r
454 private void pmnu_saveChanges_Click(object sender, EventArgs e)
\r
456 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
457 if (result == DialogResult.Yes)
\r
458 presetHandler.updatePreset(treeView_presets.SelectedNode.Text, QueryGenerator.generateTabbedComponentsQuery(this), true);
\r
459 else if (result == DialogResult.No)
\r
460 presetHandler.updatePreset(treeView_presets.SelectedNode.Text, QueryGenerator.generateTabbedComponentsQuery(this), false);
\r
462 private void pmnu_delete_click(object sender, EventArgs e)
\r
464 if (treeView_presets.SelectedNode != null)
\r
466 presetHandler.remove(treeView_presets.SelectedNode.Text);
\r
467 treeView_presets.Nodes.Remove(treeView_presets.SelectedNode);
\r
469 treeView_presets.Select();
\r
471 private void presets_menu_Opening(object sender, System.ComponentModel.CancelEventArgs e)
\r
473 // Make sure that the save menu is always disabled by default
\r
474 pmnu_saveChanges.Enabled = false;
\r
476 // Now enable the save menu if the selected preset is a user preset
\r
477 if (treeView_presets.SelectedNode != null)
\r
478 if (presetHandler.checkIfUserPresetExists(treeView_presets.SelectedNode.Text))
\r
479 pmnu_saveChanges.Enabled = true;
\r
481 treeView_presets.Select();
\r
484 // Presets Management
\r
485 private void btn_addPreset_Click(object sender, EventArgs e)
\r
487 Form preset = new frmAddPreset(this, QueryGenerator.generateTabbedComponentsQuery(this), presetHandler);
\r
488 preset.ShowDialog();
\r
490 private void btn_removePreset_Click(object sender, EventArgs e)
\r
492 DialogResult result = MessageBox.Show("Are you sure you wish to delete the selected preset?", "Preset", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
493 if (result == DialogResult.Yes)
\r
495 if (treeView_presets.SelectedNode != null)
\r
497 presetHandler.remove(treeView_presets.SelectedNode.Text);
\r
498 treeView_presets.Nodes.Remove(treeView_presets.SelectedNode);
\r
501 treeView_presets.Select();
\r
503 private void btn_setDefault_Click(object sender, EventArgs e)
\r
505 if (treeView_presets.SelectedNode != null)
\r
507 DialogResult result = MessageBox.Show("Are you sure you wish to set this preset as the default?", "Preset", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
508 if (result == DialogResult.Yes)
\r
510 Properties.Settings.Default.defaultPreset = treeView_presets.SelectedNode.Text;
\r
511 Properties.Settings.Default.Save();
\r
512 MessageBox.Show("New default preset set.", "Alert", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
516 MessageBox.Show("Please select a preset first.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
518 private void treeview_presets_mouseUp(object sender, MouseEventArgs e)
\r
520 if (e.Button == MouseButtons.Right)
\r
521 treeView_presets.SelectedNode = treeView_presets.GetNodeAt(e.Location);
\r
522 else if (e.Button == MouseButtons.Left)
\r
524 if (treeView_presets.GetNodeAt(e.Location) != null)
\r
526 if (labelPreset.Text.Contains(treeView_presets.GetNodeAt(e.Location).Text))
\r
531 treeView_presets.Select();
\r
533 private void treeView_presets_AfterSelect(object sender, TreeViewEventArgs e)
\r
537 private void treeView_presets_deleteKey(object sender, KeyEventArgs e)
\r
539 if (e.KeyCode == Keys.Delete)
\r
541 DialogResult result = MessageBox.Show("Are you sure you wish to delete the selected preset?", "Preset", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
542 if (result == DialogResult.Yes)
\r
544 if (treeView_presets.SelectedNode != null)
\r
545 presetHandler.remove(treeView_presets.SelectedNode.Text);
\r
547 // Remember each nodes expanded status so we can reload it
\r
548 List<Boolean> nodeStatus = new List<Boolean>();
\r
549 foreach (TreeNode node in treeView_presets.Nodes)
\r
550 nodeStatus.Add(node.IsExpanded);
\r
552 // Now reload the preset panel
\r
555 // And finally, re-expand any of the nodes if required
\r
557 foreach (TreeNode node in treeView_presets.Nodes)
\r
567 private void selectPreset()
\r
569 if (treeView_presets.SelectedNode != null)
\r
571 // Ok, so, we've selected a preset. Now we want to load it.
\r
572 string presetName = treeView_presets.SelectedNode.Text;
\r
573 if (presetHandler.getPreset(presetName) != null)
\r
575 string query = presetHandler.getPreset(presetName).Query;
\r
576 Boolean loadPictureSettings = presetHandler.getPreset(presetName).PictureSettings;
\r
580 //Ok, Reset all the H264 widgets before changing the preset
\r
581 x264Panel.reset2Defaults();
\r
583 // Send the query from the file to the Query Parser class
\r
584 QueryParser presetQuery = QueryParser.Parse(query);
\r
586 // Now load the preset
\r
587 PresetLoader.presetLoader(this, presetQuery, presetName, loadPictureSettings);
\r
589 // The x264 widgets will need updated, so do this now:
\r
590 x264Panel.X264_StandardizeOptString();
\r
591 x264Panel.X264_SetCurrentSettingsInPanel();
\r
596 private void loadNormalPreset()
\r
598 foreach (TreeNode treenode in treeView_presets.Nodes)
\r
600 foreach (TreeNode node in treenode.Nodes)
\r
602 if (node.Text.Equals("Normal"))
\r
603 treeView_presets.SelectedNode = treeView_presets.Nodes[treenode.Index].Nodes[0];
\r
607 private void importPreset()
\r
609 Import imp = new Import();
\r
610 if (openPreset.ShowDialog() == DialogResult.OK)
\r
612 QueryParser parsed = imp.importMacPreset(openPreset.FileName);
\r
613 if (presetHandler.checkIfUserPresetExists(parsed.PresetName + " (Imported)"))
\r
615 DialogResult result = MessageBox.Show("This preset appears to already exist. Would you like to overwrite it?", "Overwrite preset?",
\r
616 MessageBoxButtons.YesNo, MessageBoxIcon.Warning);
\r
617 if (result == DialogResult.Yes)
\r
619 PresetLoader.presetLoader(this, parsed, parsed.PresetName, parsed.UsesPictureSettings);
\r
620 presetHandler.updatePreset(parsed.PresetName + " (Imported)", queryGen.generateCLIQuery(this, 0, null),
\r
621 parsed.UsesPictureSettings);
\r
626 PresetLoader.presetLoader(this, parsed, parsed.PresetName, parsed.UsesPictureSettings);
\r
627 presetHandler.addPreset(parsed.PresetName, queryGen.generateCLIQuery(this, 0, null), parsed.UsesPictureSettings);
\r
629 if (presetHandler.addPreset(parsed.PresetName + " (Imported)", queryGen.generateCLIQuery(this, 0, null), parsed.UsesPictureSettings))
\r
631 TreeNode preset_treeview = new TreeNode(parsed.PresetName + " (Imported)") { ForeColor = Color.Black };
\r
632 treeView_presets.Nodes.Add(preset_treeview);
\r
640 private void btn_source_Click(object sender, EventArgs e)
\r
642 if (Properties.Settings.Default.drive_detection)
\r
644 mnu_dvd_drive.Visible = true;
\r
645 Thread driveInfoThread = new Thread(getDriveInfoThread);
\r
646 driveInfoThread.Start();
\r
649 mnu_dvd_drive.Visible = false;
\r
651 private void btn_start_Click(object sender, EventArgs e)
\r
653 if (btn_start.Text == "Stop")
\r
655 DialogResult result = MessageBox.Show("Are you sure you wish to cancel the encode?", "Cancel Encode?", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
657 if (result == DialogResult.Yes)
\r
660 encodeQueue.RequestPause();
\r
662 // Allow the CLI to exit cleanly
\r
663 Win32.SetForegroundWindow((int)encodeQueue.processHandle);
\r
664 SendKeys.Send("^C");
\r
667 setEncodeFinished();
\r
672 if (encodeQueue.Count != 0 || (!string.IsNullOrEmpty(sourcePath) && !string.IsNullOrEmpty(text_destination.Text)))
\r
674 string generatedQuery = queryGen.generateCLIQuery(this, 0, null);
\r
675 string specifiedQuery = rtf_query.Text != "" ? rtf_query.Text : queryGen.generateCLIQuery(this, 0, null);
\r
676 string query = string.Empty;
\r
678 // Check to make sure the generated query matches the GUI settings
\r
679 if (Properties.Settings.Default.PromptOnUnmatchingQueries && !string.IsNullOrEmpty(specifiedQuery) && generatedQuery != specifiedQuery)
\r
681 DialogResult result = MessageBox.Show("The query under the \"Query Editor\" tab " +
\r
682 "does not match the current GUI settings. Because the manual query takes " +
\r
683 "priority over the GUI, your recently updated settings will not be taken " +
\r
684 "into account when encoding this job." + Environment.NewLine + Environment.NewLine +
\r
685 "Do you want to replace the manual query with the GUI-generated query?",
\r
686 "Manual Query does not Match GUI",
\r
687 MessageBoxButtons.YesNoCancel, MessageBoxIcon.Asterisk,
\r
688 MessageBoxDefaultButton.Button3);
\r
692 case DialogResult.Yes:
\r
693 // Replace the manual query with the generated one
\r
694 query = generatedQuery;
\r
695 rtf_query.Text = generatedQuery;
\r
697 case DialogResult.No:
\r
698 // Use the manual query
\r
699 query = specifiedQuery;
\r
701 case DialogResult.Cancel:
\r
702 // Don't start the encode
\r
708 query = generatedQuery;
\r
711 DialogResult overwrite = DialogResult.Yes;
\r
712 if (text_destination.Text != "")
\r
713 if (File.Exists(text_destination.Text))
\r
714 overwrite = MessageBox.Show("The destination file already exists. Are you sure you want to overwrite it?", "Overwrite File?", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
716 if (overwrite == DialogResult.Yes)
\r
718 if (encodeQueue.Count == 0)
\r
719 encodeQueue.AddJob(query, sourcePath, text_destination.Text);
\r
721 queueWindow.setQueue();
\r
722 if (encodeQueue.Count > 1)
\r
723 queueWindow.Show(false);
\r
725 setEncodeStarted(); // Encode is running, so setup the GUI appropriately
\r
726 if (ActivityWindow != null)
\r
727 ActivityWindow.setLogView(false);
\r
728 encodeQueue.StartEncodeQueue(); // Start The Queue Encoding Process
\r
729 lastAction = "encode"; // Set the last action to encode - Used for activity window.
\r
733 else if (string.IsNullOrEmpty(sourcePath) || string.IsNullOrEmpty(text_destination.Text))
\r
734 MessageBox.Show("No source or destination selected.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
737 private void btn_add2Queue_Click(object sender, EventArgs e)
\r
739 if (string.IsNullOrEmpty(sourcePath) || string.IsNullOrEmpty(text_destination.Text))
\r
740 MessageBox.Show("No source or destination selected.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
743 String query = queryGen.generateCLIQuery(this, 0, null);
\r
744 if (rtf_query.Text != "")
\r
745 query = rtf_query.Text;
\r
747 if (encodeQueue.CheckForDestinationDuplicate(text_destination.Text))
\r
749 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
750 "Warning", MessageBoxButtons.YesNo, MessageBoxIcon.Warning);
\r
751 if (result == DialogResult.Yes)
\r
752 encodeQueue.AddJob(query, sourcePath, text_destination.Text);
\r
756 encodeQueue.AddJob(query, sourcePath, text_destination.Text);
\r
758 queueWindow.Show();
\r
761 private void btn_showQueue_Click(object sender, EventArgs e)
\r
763 queueWindow.Show();
\r
765 private void tb_preview_Click(object sender, EventArgs e)
\r
767 if (string.IsNullOrEmpty(sourcePath) || string.IsNullOrEmpty(text_destination.Text))
\r
768 MessageBox.Show("No source or destination selected.", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
771 if (qtpreview == null)
\r
773 qtpreview = new frmPreview(this);
\r
776 else if (qtpreview.IsDisposed)
\r
778 qtpreview = new frmPreview(this);
\r
782 MessageBox.Show(qtpreview, "The preview window is already open!", "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
785 private void btn_ActivityWindow_Click(object sender, EventArgs e)
\r
787 String file = lastAction == "scan" ? "last_scan_log.txt" : "last_encode_log.txt";
\r
788 if (ActivityWindow == null)
\r
789 ActivityWindow = new frmActivityWindow(file, encodeQueue, this);
\r
791 ActivityWindow.Show();
\r
795 #region System Tray Icon
\r
796 private void frmMain_Resize(object sender, EventArgs e)
\r
798 if (FormWindowState.Minimized == this.WindowState)
\r
800 notifyIcon.Visible = true;
\r
803 else if (FormWindowState.Normal == this.WindowState)
\r
804 notifyIcon.Visible = false;
\r
806 private void notifyIcon_MouseDoubleClick(object sender, MouseEventArgs e)
\r
808 this.Visible = true;
\r
810 this.WindowState = FormWindowState.Normal;
\r
811 notifyIcon.Visible = false;
\r
813 private void btn_restore_Click(object sender, EventArgs e)
\r
815 this.Visible = true;
\r
817 this.WindowState = FormWindowState.Normal;
\r
818 notifyIcon.Visible = false;
\r
822 #region Tab Control
\r
825 private void btn_dvd_source_Click(object sender, EventArgs e)
\r
827 if (DVD_Open.ShowDialog() == DialogResult.OK)
\r
828 selectSource(DVD_Open.SelectedPath, 1);
\r
830 UpdateSourceLabel();
\r
832 private void btn_file_source_Click(object sender, EventArgs e)
\r
834 if (ISO_Open.ShowDialog() == DialogResult.OK)
\r
835 selectSource(ISO_Open.FileName, 2);
\r
837 UpdateSourceLabel();
\r
839 private void mnu_dvd_drive_Click(object sender, EventArgs e)
\r
841 if (!mnu_dvd_drive.Text.Contains("VIDEO_TS")) return;
\r
842 string[] path = mnu_dvd_drive.Text.Split(' ');
\r
843 selectSource(path[0], 3);
\r
845 private void selectSource(string file, int type)
\r
847 Check_ChapterMarkers.Enabled = true;
\r
848 lastAction = "scan";
\r
849 sourcePath = string.Empty;
\r
851 if (file == string.Empty) // Must have a file or path
\r
853 UpdateSourceLabel();
\r
857 if (file.StartsWith("\\")) // NO UNC Paths
\r
860 "Sorry, HandBrake does not support UNC file paths. \nTry mounting the share as a network drive in My Computer",
\r
861 "Warning", MessageBoxButtons.OK, MessageBoxIcon.Warning);
\r
862 UpdateSourceLabel();
\r
866 switch (type) // Start the scan
\r
868 case 1: // btn_dvd_source_Click()
\r
869 case 3: // mnu_dvd_drive_Click()
\r
870 sourcePath = Path.GetFullPath(file);
\r
873 case 2: // btn_file_source_Click()
\r
874 sourcePath = Path.GetFileName(file);
\r
879 private void drp_dvdtitle_Click(object sender, EventArgs e)
\r
881 if ((drp_dvdtitle.Items.Count == 1) && (drp_dvdtitle.Items[0].ToString() == "Automatic"))
\r
882 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
884 private void drp_dvdtitle_SelectedIndexChanged(object sender, EventArgs e)
\r
886 UnRegisterPresetEventHandler();
\r
888 PictureSettings.lbl_Aspect.Text = "Select a Title"; // Reset some values on the form
\r
889 drop_chapterStart.Items.Clear();
\r
890 drop_chapterFinish.Items.Clear();
\r
892 // If the dropdown is set to automatic nothing else needs to be done.
\r
893 // Otheriwse if its not, title data has to be loased from parsing.
\r
894 if (drp_dvdtitle.Text != "Automatic")
\r
896 selectedTitle = drp_dvdtitle.SelectedItem as Title;
\r
897 lbl_duration.Text = selectedTitle.Duration.ToString();
\r
898 PictureSettings.Source = selectedTitle; // Setup Picture Settings Tab Control
\r
900 // Populate the Angles dropdown
\r
901 drop_angle.Items.Clear();
\r
902 if (Properties.Settings.Default.dvdnav)
\r
904 drop_angle.Visible = true;
\r
905 lbl_angle.Visible = true;
\r
906 drop_angle.Items.AddRange(selectedTitle.Angles.ToArray());
\r
907 if (drop_angle.Items.Count != 0)
\r
908 drop_angle.SelectedIndex = 0;
\r
912 drop_angle.Visible = false;
\r
913 lbl_angle.Visible = false;
\r
916 // Populate the Start chapter Dropdown
\r
917 drop_chapterStart.Items.Clear();
\r
918 drop_chapterStart.Items.AddRange(selectedTitle.Chapters.ToArray());
\r
919 if (drop_chapterStart.Items.Count > 0)
\r
920 drop_chapterStart.Text = drop_chapterStart.Items[0].ToString();
\r
922 // Populate the Final Chapter Dropdown
\r
923 drop_chapterFinish.Items.Clear();
\r
924 drop_chapterFinish.Items.AddRange(selectedTitle.Chapters.ToArray());
\r
925 if (drop_chapterFinish.Items.Count > 0)
\r
926 drop_chapterFinish.Text = drop_chapterFinish.Items[drop_chapterFinish.Items.Count - 1].ToString();
\r
928 // Populate the Audio Channels Dropdown
\r
929 AudioSettings.SetTrackList(selectedTitle);
\r
931 // Populate the Subtitles dropdown
\r
932 Subtitles.drp_subtitleTracks.Items.Clear();
\r
933 Subtitles.drp_subtitleTracks.Items.Add("Foreign Audio Search (Bitmap)");
\r
934 Subtitles.drp_subtitleTracks.Items.AddRange(selectedTitle.Subtitles.ToArray());
\r
935 Subtitles.drp_subtitleTracks.SelectedIndex = 0;
\r
936 Subtitles.setSubtitleTrackAuto();
\r
939 // Run the autoName & chapterNaming functions
\r
940 if (Properties.Settings.Default.autoNaming)
\r
942 string autoPath = Main.autoName(this);
\r
943 if (autoPath != null)
\r
944 text_destination.Text = autoPath;
\r
946 MessageBox.Show("You currently have \"Automatically name output files\" enabled for the destination file box, but you do not have a default direcotry 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
949 data_chpt.Rows.Clear();
\r
950 if (selectedTitle.Chapters.Count != 1)
\r
952 DataGridView chapterGridView = Main.chapterNaming(data_chpt, drop_chapterFinish.Text);
\r
953 if (chapterGridView != null)
\r
954 data_chpt = chapterGridView;
\r
958 Check_ChapterMarkers.Checked = false;
\r
959 Check_ChapterMarkers.Enabled = false;
\r
962 // Hack to force the redraw of the scrollbars which don't resize properly when the control is disabled.
\r
963 data_chpt.Columns[0].Width = 166;
\r
964 data_chpt.Columns[0].Width = 165;
\r
966 RegisterPresetEventHandler();
\r
968 private void chapersChanged(object sender, EventArgs e)
\r
970 Control ctl = (Control) sender;
\r
971 int chapterStart, chapterEnd;
\r
972 int.TryParse(drop_chapterStart.Text, out chapterStart);
\r
973 int.TryParse(drop_chapterFinish.Text, out chapterEnd);
\r
977 case "drop_chapterStart":
\r
978 if (drop_chapterFinish.SelectedIndex == -1 && drop_chapterFinish.Items.Count != 0)
\r
979 drop_chapterFinish.SelectedIndex = drop_chapterFinish.Items.Count - 1;
\r
981 if (chapterEnd != 0)
\r
982 if (chapterStart > chapterEnd)
\r
983 drop_chapterFinish.Text = chapterStart.ToString();
\r
985 case "drop_chapterFinish":
\r
986 if (drop_chapterStart.Items.Count >= 1 && drop_chapterStart.SelectedIndex == -1)
\r
987 drop_chapterStart.SelectedIndex = 0;
\r
989 if (chapterStart != 0)
\r
990 if (chapterEnd < chapterStart)
\r
991 drop_chapterFinish.Text = chapterStart.ToString();
\r
993 // Add more rows to the Chapter menu if needed.
\r
994 if (Check_ChapterMarkers.Checked)
\r
996 int i = data_chpt.Rows.Count, finish = 0;
\r
997 int.TryParse(drop_chapterFinish.Text, out finish);
\r
1001 int n = data_chpt.Rows.Add();
\r
1002 data_chpt.Rows[n].Cells[0].Value = (i + 1);
\r
1003 data_chpt.Rows[n].Cells[1].Value = "Chapter " + (i + 1);
\r
1004 data_chpt.Rows[n].Cells[0].ValueType = typeof(int);
\r
1005 data_chpt.Rows[n].Cells[1].ValueType = typeof(string);
\r
1012 // Update the Duration
\r
1013 lbl_duration.Text = Main.calculateDuration(drop_chapterStart.SelectedIndex, drop_chapterFinish.SelectedIndex, selectedTitle).ToString();
\r
1015 // Run the Autonaming function
\r
1016 if (Properties.Settings.Default.autoNaming)
\r
1017 text_destination.Text = Main.autoName(this);
\r
1019 // Disable chapter markers if only 1 chapter is selected.
\r
1020 if (chapterStart == chapterEnd)
\r
1022 Check_ChapterMarkers.Enabled = false;
\r
1023 btn_importChapters.Enabled = false;
\r
1024 data_chpt.Enabled = false;
\r
1028 Check_ChapterMarkers.Enabled = true;
\r
1029 if (Check_ChapterMarkers.Checked)
\r
1031 btn_importChapters.Enabled = true;
\r
1032 data_chpt.Enabled = true;
\r
1038 private void btn_destBrowse_Click(object sender, EventArgs e)
\r
1040 // This removes the file extension from the filename box on the save file dialog.
\r
1041 // It's daft but some users don't realise that typing an extension overrides the dropdown extension selected.
\r
1042 DVD_Save.FileName = Path.GetFileNameWithoutExtension(text_destination.Text);
\r
1044 if (Path.IsPathRooted(text_destination.Text))
\r
1045 DVD_Save.InitialDirectory = Path.GetDirectoryName(text_destination.Text);
\r
1047 // Show the dialog and set the main form file path
\r
1048 if (drop_format.SelectedIndex.Equals(0))
\r
1049 DVD_Save.FilterIndex = 1;
\r
1050 else if (drop_format.SelectedIndex.Equals(1))
\r
1051 DVD_Save.FilterIndex = 2;
\r
1053 if (DVD_Save.ShowDialog() == DialogResult.OK)
\r
1055 if (DVD_Save.FileName.StartsWith("\\"))
\r
1056 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
1059 // Add a file extension manually, as FileDialog.AddExtension has issues with dots in filenames
\r
1060 switch (DVD_Save.FilterIndex)
\r
1063 if (!Path.GetExtension(DVD_Save.FileName).Equals(".mp4", StringComparison.InvariantCultureIgnoreCase))
\r
1064 if (Properties.Settings.Default.useM4v)
\r
1065 DVD_Save.FileName = DVD_Save.FileName.Replace(".mp4", ".m4v").Replace(".mkv",".m4v");
\r
1067 DVD_Save.FileName = DVD_Save.FileName.Replace(".m4v", ".mp4").Replace(".mkv", ".mp4");
\r
1070 if (!Path.GetExtension(DVD_Save.FileName).Equals(".mkv", StringComparison.InvariantCultureIgnoreCase))
\r
1071 DVD_Save.FileName = DVD_Save.FileName .Replace(".mp4", ".mkv").Replace(".m4v",".mkv");
\r
1077 text_destination.Text = DVD_Save.FileName;
\r
1079 // Quicktime requires .m4v file for chapter markers to work. If checked, change the extension to .m4v (mp4 and m4v are the same thing)
\r
1080 if (Check_ChapterMarkers.Checked && DVD_Save.FilterIndex != 2)
\r
1081 setExtension(".m4v");
\r
1085 private void text_destination_TextChanged(object sender, EventArgs e)
\r
1087 string path = text_destination.Text;
\r
1088 if (path.EndsWith(".mp4") || path.EndsWith(".m4v"))
\r
1089 drop_format.SelectedIndex = 0;
\r
1090 else if (path.EndsWith(".mkv"))
\r
1091 drop_format.SelectedIndex = 1;
\r
1094 // Output Settings
\r
1095 private void drop_format_SelectedIndexChanged(object sender, EventArgs e)
\r
1097 switch (drop_format.SelectedIndex)
\r
1100 if (Properties.Settings.Default.useM4v || Check_ChapterMarkers.Checked || AudioSettings.RequiresM4V() || Subtitles.RequiresM4V())
\r
1101 setExtension(".m4v");
\r
1103 setExtension(".mp4");
\r
1106 setExtension(".mkv");
\r
1110 AudioSettings.SetContainer(drop_format.Text);
\r
1111 Subtitles.setContainer(drop_format.SelectedIndex);
\r
1113 if ((drop_format.Text.Contains("MP4")) || (drop_format.Text.Contains("M4V")))
\r
1115 if (drp_videoEncoder.Items.Contains("VP3 (Theora)"))
\r
1117 drp_videoEncoder.Items.Remove("VP3 (Theora)");
\r
1118 drp_videoEncoder.SelectedIndex = 1;
\r
1121 else if (drop_format.Text.Contains("MKV"))
\r
1122 drp_videoEncoder.Items.Add("VP3 (Theora)");
\r
1124 private void setExtension(string newExtension)
\r
1126 text_destination.Text = text_destination.Text.Replace(".mp4", newExtension);
\r
1127 text_destination.Text = text_destination.Text.Replace(".m4v", newExtension);
\r
1128 text_destination.Text = text_destination.Text.Replace(".mkv", newExtension);
\r
1132 private void drp_videoEncoder_SelectedIndexChanged(object sender, EventArgs e)
\r
1134 setContainerOpts();
\r
1136 //Turn off some options which are H.264 only when the user selects a non h.264 encoder
\r
1137 if (drp_videoEncoder.Text.Contains("H.264"))
\r
1139 if (check_2PassEncode.CheckState == CheckState.Checked)
\r
1140 check_turbo.Enabled = true;
\r
1142 tab_advanced.Enabled = true;
\r
1143 if ((drop_format.Text.Contains("MP4")) || (drop_format.Text.Contains("M4V")))
\r
1144 check_iPodAtom.Enabled = true;
\r
1146 check_iPodAtom.Enabled = false;
\r
1150 check_turbo.CheckState = CheckState.Unchecked;
\r
1151 check_turbo.Enabled = false;
\r
1152 tab_advanced.Enabled = false;
\r
1153 x264Panel.x264Query = "";
\r
1154 check_iPodAtom.Enabled = false;
\r
1155 check_iPodAtom.Checked = false;
\r
1158 // Setup the CQ Slider
\r
1159 switch (drp_videoEncoder.Text)
\r
1161 case "MPEG-4 (FFmpeg)":
\r
1162 if (slider_videoQuality.Value > 31)
\r
1163 slider_videoQuality.Value = 20; // Just reset to 70% QP 10 on encode change.
\r
1164 slider_videoQuality.Minimum = 1;
\r
1165 slider_videoQuality.Maximum = 31;
\r
1167 case "H.264 (x264)":
\r
1168 slider_videoQuality.Minimum = 0;
\r
1169 slider_videoQuality.TickFrequency = 1;
\r
1171 CultureInfo culture = CultureInfo.CreateSpecificCulture("en-US");
\r
1172 double cqStep = Properties.Settings.Default.x264cqstep;
\r
1173 double multiplier = 1.0 / cqStep;
\r
1174 double value = slider_videoQuality.Value * multiplier;
\r
1176 switch (Properties.Settings.Default.x264cqstep.ToString(culture))
\r
1179 slider_videoQuality.Maximum = 255;
\r
1182 slider_videoQuality.Maximum = 204;
\r
1185 slider_videoQuality.Maximum = 102;
\r
1188 slider_videoQuality.Maximum = 51;
\r
1191 slider_videoQuality.Maximum = 51;
\r
1194 if (value < slider_videoQuality.Maximum)
\r
1195 slider_videoQuality.Value = slider_videoQuality.Maximum - (int)value;
\r
1198 case "VP3 (Theora)":
\r
1199 if (slider_videoQuality.Value > 63)
\r
1200 slider_videoQuality.Value = 45; // Just reset to 70% QP 45 on encode change.
\r
1201 slider_videoQuality.Minimum = 0;
\r
1202 slider_videoQuality.Maximum = 63;
\r
1207 /// Set the container format options
\r
1209 public void setContainerOpts()
\r
1211 if ((drop_format.Text.Contains("MP4")) || (drop_format.Text.Contains("M4V")))
\r
1213 check_largeFile.Enabled = true;
\r
1214 check_optimiseMP4.Enabled = true;
\r
1215 check_iPodAtom.Enabled = true;
\r
1219 check_largeFile.Enabled = false;
\r
1220 check_optimiseMP4.Enabled = false;
\r
1221 check_iPodAtom.Enabled = false;
\r
1222 check_largeFile.Checked = false;
\r
1223 check_optimiseMP4.Checked = false;
\r
1224 check_iPodAtom.Checked = false;
\r
1227 private double _cachedCqStep = Properties.Settings.Default.x264cqstep;
\r
1229 /// Update the CQ slider for x264 for a new CQ step. This is set from option
\r
1231 public void setQualityFromSlider()
\r
1233 // Work out the current RF value.
\r
1234 double cqStep = _cachedCqStep;
\r
1235 double rfValue = 51.0 - slider_videoQuality.Value * cqStep;
\r
1237 // Change the maximum value for the slider
\r
1238 switch (Properties.Settings.Default.x264cqstep.ToString(new CultureInfo("en-US")))
\r
1241 slider_videoQuality.Maximum = 255;
\r
1244 slider_videoQuality.Maximum = 204;
\r
1247 slider_videoQuality.Maximum = 102;
\r
1250 slider_videoQuality.Maximum = 51;
\r
1253 slider_videoQuality.Maximum = 51;
\r
1257 // Reset the CQ slider to RF0
\r
1258 slider_videoQuality.Value = slider_videoQuality.Maximum;
\r
1260 // Reset the CQ slider back to the previous value as close as possible
\r
1261 double cqStepNew = Properties.Settings.Default.x264cqstep;
\r
1262 double rfValueCurrent = 51.0 - slider_videoQuality.Value * cqStepNew;
\r
1263 while (rfValueCurrent < rfValue)
\r
1265 slider_videoQuality.Value--;
\r
1266 rfValueCurrent = 51.0 - slider_videoQuality.Value * cqStepNew;
\r
1269 // Cache the CQ step for the next calculation
\r
1270 _cachedCqStep = Properties.Settings.Default.x264cqstep;
\r
1272 private void slider_videoQuality_Scroll(object sender, EventArgs e)
\r
1274 double cqStep = Properties.Settings.Default.x264cqstep;
\r
1275 switch (drp_videoEncoder.Text)
\r
1277 case "MPEG-4 (FFmpeg)":
\r
1278 double rfValue = 31 - (slider_videoQuality.Value - 1);
\r
1279 double max = slider_videoQuality.Maximum;
\r
1280 double min = slider_videoQuality.Minimum;
\r
1281 double val = ((max - min) - (rfValue - min)) / (max - min);
\r
1282 SliderValue.Text = Math.Round((val * 100), 2).ToString(new CultureInfo("en-US")) + "% QP:" + (32 - slider_videoQuality.Value);
\r
1284 case "H.264 (x264)":
\r
1285 rfValue = 51.0 - slider_videoQuality.Value * cqStep;
\r
1286 max = slider_videoQuality.Maximum * cqStep;
\r
1287 min = slider_videoQuality.Minimum;
\r
1288 val = ((max - min) - (rfValue - min)) / (max - min);
\r
1289 rfValue = Math.Round(rfValue, 2);
\r
1290 SliderValue.Text = Math.Round((val * 100), 2).ToString(new CultureInfo("en-US")) + "% RF:" + rfValue.ToString(new CultureInfo("en-US"));
\r
1292 case "VP3 (Theora)":
\r
1293 rfValue = slider_videoQuality.Value;
\r
1294 double value = rfValue / 63;
\r
1295 SliderValue.Text = Math.Round((value * 100), 2).ToString(new CultureInfo("en-US")) + "% QP:" + slider_videoQuality.Value;
\r
1299 private void radio_targetFilesize_CheckedChanged(object sender, EventArgs e)
\r
1301 text_bitrate.Enabled = false;
\r
1302 text_filesize.Enabled = true;
\r
1303 slider_videoQuality.Enabled = false;
\r
1305 check_2PassEncode.Enabled = true;
\r
1307 private void radio_avgBitrate_CheckedChanged(object sender, EventArgs e)
\r
1309 text_bitrate.Enabled = true;
\r
1310 text_filesize.Enabled = false;
\r
1311 slider_videoQuality.Enabled = false;
\r
1313 check_2PassEncode.Enabled = true;
\r
1315 private void radio_cq_CheckedChanged(object sender, EventArgs e)
\r
1317 text_bitrate.Enabled = false;
\r
1318 text_filesize.Enabled = false;
\r
1319 slider_videoQuality.Enabled = true;
\r
1321 check_2PassEncode.Enabled = false;
\r
1322 check_2PassEncode.CheckState = CheckState.Unchecked;
\r
1324 private void check_2PassEncode_CheckedChanged(object sender, EventArgs e)
\r
1326 if (check_2PassEncode.CheckState.ToString() == "Checked")
\r
1328 if (drp_videoEncoder.Text.Contains("H.264"))
\r
1329 check_turbo.Enabled = true;
\r
1333 check_turbo.Enabled = false;
\r
1334 check_turbo.CheckState = CheckState.Unchecked;
\r
1338 // Chapter Marker Tab
\r
1339 private void Check_ChapterMarkers_CheckedChanged(object sender, EventArgs e)
\r
1341 if (Check_ChapterMarkers.Checked)
\r
1343 if (drop_format.SelectedIndex != 1)
\r
1344 setExtension(".m4v");
\r
1345 data_chpt.Enabled = true;
\r
1346 btn_importChapters.Enabled = true;
\r
1350 if (drop_format.SelectedIndex != 1 && !Properties.Settings.Default.useM4v)
\r
1351 setExtension(".mp4");
\r
1352 data_chpt.Enabled = false;
\r
1353 btn_importChapters.Enabled = false;
\r
1356 private void btn_importChapters_Click(object sender, EventArgs e)
\r
1358 if (File_ChapterImport.ShowDialog() == DialogResult.OK)
\r
1360 String filename = File_ChapterImport.FileName;
\r
1361 DataGridView imported = Main.importChapterNames(data_chpt, filename);
\r
1362 if (imported != null)
\r
1363 data_chpt = imported;
\r
1366 private void mnu_resetChapters_Click(object sender, EventArgs e)
\r
1368 data_chpt.Rows.Clear();
\r
1369 DataGridView chapterGridView = Main.chapterNaming(data_chpt, drop_chapterFinish.Text);
\r
1370 if (chapterGridView != null)
\r
1372 data_chpt = chapterGridView;
\r
1376 // Query Editor Tab
\r
1377 private void btn_generate_Query_Click(object sender, EventArgs e)
\r
1379 rtf_query.Text = queryGen.generateCLIQuery(this, 0, null);
\r
1381 private void btn_clear_Click(object sender, EventArgs e)
\r
1383 rtf_query.Clear();
\r
1387 // MainWindow Components, Actions and Functions ***********************
\r
1389 #region Source Scan
\r
1390 public Boolean isScanning { get; set; }
\r
1391 private static int scanProcessID { get; set; }
\r
1392 private void startScan(String filename)
\r
1394 // Setup the GUI components for the scan.
\r
1395 sourcePath = filename;
\r
1396 foreach (Control ctrl in Controls)
\r
1398 if (!(ctrl is StatusStrip || ctrl is MenuStrip || ctrl is ToolStrip))
\r
1399 ctrl.Enabled = false;
\r
1401 lbl_encode.Visible = true;
\r
1402 lbl_encode.Text = "Scanning ...";
\r
1403 btn_source.Enabled = false;
\r
1404 btn_start.Enabled = false;
\r
1405 btn_showQueue.Enabled = false;
\r
1406 btn_add2Queue.Enabled = false;
\r
1407 tb_preview.Enabled = false;
\r
1408 mnu_killCLI.Visible = true;
\r
1410 // Start hte Scan Thread
\r
1413 if (ActivityWindow != null)
\r
1414 ActivityWindow.setLogView(true);
\r
1415 isScanning = true;
\r
1416 ThreadPool.QueueUserWorkItem(scanProcess);
\r
1418 catch (Exception exc)
\r
1420 MessageBox.Show("frmMain.cs - startScan " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1423 private void scanProcess(object state)
\r
1427 string handbrakeCLIPath = Path.Combine(Application.StartupPath, "HandBrakeCLI.exe");
\r
1428 string logDir = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData) + "\\HandBrake\\logs";
\r
1429 string dvdInfoPath = Path.Combine(logDir, "last_scan_log.txt");
\r
1431 // Make we don't pick up a stale last_encode_log.txt (and that we have rights to the file)
\r
1432 if (File.Exists(dvdInfoPath))
\r
1433 File.Delete(dvdInfoPath);
\r
1435 String dvdnav = string.Empty;
\r
1436 if (Properties.Settings.Default.dvdnav)
\r
1437 dvdnav = " --dvdnav";
\r
1438 string strCmdLine = String.Format(@"cmd /c """"{0}"" -i ""{1}"" -t0 {2} -v >""{3}"" 2>&1""", handbrakeCLIPath, sourcePath, dvdnav, dvdInfoPath);
\r
1440 ProcessStartInfo hbParseDvd = new ProcessStartInfo("CMD.exe", strCmdLine) { WindowStyle = ProcessWindowStyle.Hidden };
\r
1442 Boolean cleanExit = true;
\r
1443 using (hbproc = Process.Start(hbParseDvd))
\r
1445 Process[] before = Process.GetProcesses(); // Get a list of running processes before starting.
\r
1446 scanProcessID = Main.getCliProcess(before);
\r
1447 hbproc.WaitForExit();
\r
1448 if (hbproc.ExitCode != 0)
\r
1449 cleanExit = false;
\r
1452 if (cleanExit) // If 0 exit code, CLI exited cleanly.
\r
1454 if (!File.Exists(dvdInfoPath))
\r
1455 throw new Exception("Unable to retrieve the DVD Info. last_scan_log.txt is missing. \nExpected location of last_scan_log.txt: \n"
\r
1458 using (StreamReader sr = new StreamReader(dvdInfoPath))
\r
1460 thisDVD = DVD.Parse(sr);
\r
1465 updateUIafterScan();
\r
1468 catch (Exception exc)
\r
1470 MessageBox.Show("frmMain.cs - scanProcess() " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1474 private void updateUIafterScan()
\r
1478 if (InvokeRequired)
\r
1480 BeginInvoke(new UpdateWindowHandler(updateUIafterScan));
\r
1484 // Setup some GUI components
\r
1485 drp_dvdtitle.Items.Clear();
\r
1486 if (thisDVD.Titles.Count != 0)
\r
1487 drp_dvdtitle.Items.AddRange(thisDVD.Titles.ToArray());
\r
1489 // Now select the longest title
\r
1490 if (thisDVD.Titles.Count != 0)
\r
1491 drp_dvdtitle.SelectedItem = Main.selectLongestTitle(thisDVD);
\r
1493 // Enable the creation of chapter markers if the file is an image of a dvd.
\r
1494 if (sourcePath.ToLower().Contains(".iso") || sourcePath.Contains("VIDEO_TS"))
\r
1495 Check_ChapterMarkers.Enabled = true;
\r
1498 Check_ChapterMarkers.Enabled = false;
\r
1499 Check_ChapterMarkers.Checked = false;
\r
1500 data_chpt.Rows.Clear();
\r
1503 // If no titles were found, Display an error message
\r
1504 if (drp_dvdtitle.Items.Count == 0)
\r
1507 "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
1508 "Error", MessageBoxButtons.OK, MessageBoxIcon.Hand);
\r
1509 sourcePath = string.Empty;
\r
1511 UpdateSourceLabel();
\r
1513 // Enable the GUI components and enable any disabled components
\r
1516 catch (Exception exc)
\r
1518 MessageBox.Show("frmMain.cs - updateUIafterScan " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1522 private void enableGUI()
\r
1526 if (InvokeRequired)
\r
1527 BeginInvoke(new UpdateWindowHandler(enableGUI));
\r
1528 lbl_encode.Text = "Scan Completed";
\r
1529 foreach (Control ctrl in Controls)
\r
1530 ctrl.Enabled = true;
\r
1531 btn_start.Enabled = true;
\r
1532 btn_showQueue.Enabled = true;
\r
1533 btn_add2Queue.Enabled = true;
\r
1534 tb_preview.Enabled = true;
\r
1535 btn_source.Enabled = true;
\r
1536 mnu_killCLI.Visible = false;
\r
1538 catch (Exception exc)
\r
1540 MessageBox.Show("frmMain.cs - enableGUI() " + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1543 private void killScan()
\r
1550 Process[] prs = Process.GetProcesses();
\r
1551 foreach (Process process in prs)
\r
1553 if (process.Id == scanProcessID)
\r
1555 process.Refresh();
\r
1556 if (!process.HasExited)
\r
1561 catch (Exception ex)
\r
1563 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
1566 private void resetGUI()
\r
1568 drp_dvdtitle.Items.Clear();
\r
1569 drop_chapterStart.Items.Clear();
\r
1570 drop_chapterFinish.Items.Clear();
\r
1571 lbl_duration.Text = "Select a Title";
\r
1572 PictureSettings.lbl_src_res.Text = "Select a Title";
\r
1573 PictureSettings.lbl_Aspect.Text = "Select a Title";
\r
1574 sourcePath = String.Empty;
\r
1575 text_destination.Text = String.Empty;
\r
1577 selectedTitle = null;
\r
1578 isScanning = false;
\r
1580 private void UpdateSourceLabel()
\r
1582 labelSource.Text = string.IsNullOrEmpty(sourcePath) ? "Select \"Source\" to continue." : Path.GetFileName(sourcePath);
\r
1588 /// Set the GUI to it's finished encoding state.
\r
1590 private void setEncodeFinished()
\r
1594 if (InvokeRequired)
\r
1596 BeginInvoke(new UpdateWindowHandler(setEncodeFinished));
\r
1600 lbl_encode.Text = "Encoding Finished";
\r
1601 btn_start.Text = "Start";
\r
1602 btn_start.ToolTipText = "Start the encoding process";
\r
1603 btn_start.Image = Properties.Resources.Play;
\r
1605 // If the window is minimized, display the notification in a popup.
\r
1606 if (Properties.Settings.Default.trayIconAlerts)
\r
1607 if (FormWindowState.Minimized == this.WindowState)
\r
1609 notifyIcon.BalloonTipText = lbl_encode.Text;
\r
1610 notifyIcon.ShowBalloonTip(500);
\r
1613 catch (Exception exc)
\r
1615 MessageBox.Show(exc.ToString());
\r
1620 /// Set the GUI to it's started encoding state.
\r
1622 private void setEncodeStarted()
\r
1626 if (InvokeRequired)
\r
1628 BeginInvoke(new UpdateWindowHandler(setEncodeStarted));
\r
1632 lbl_encode.Visible = true;
\r
1633 lbl_encode.Text = "Encoding in Progress";
\r
1634 btn_start.Text = "Stop";
\r
1635 btn_start.ToolTipText = "Stop the encoding process.";
\r
1636 btn_start.Image = Properties.Resources.stop;
\r
1638 catch (Exception exc)
\r
1640 MessageBox.Show(exc.ToString());
\r
1645 #region DVD Drive Detection
\r
1646 private void getDriveInfoThread()
\r
1650 if (InvokeRequired)
\r
1652 BeginInvoke(new UpdateWindowHandler(getDriveInfoThread));
\r
1656 Boolean foundDrive = false;
\r
1657 DriveInfo[] theCollectionOfDrives = DriveInfo.GetDrives();
\r
1658 foreach (DriveInfo curDrive in theCollectionOfDrives)
\r
1660 if (curDrive.DriveType == DriveType.CDRom && curDrive.IsReady)
\r
1662 if (File.Exists(curDrive.RootDirectory + "VIDEO_TS\\VIDEO_TS.IFO"))
\r
1664 mnu_dvd_drive.Text = curDrive.RootDirectory + "VIDEO_TS (" + curDrive.VolumeLabel + ")";
\r
1665 foundDrive = true;
\r
1671 if (foundDrive == false)
\r
1672 mnu_dvd_drive.Text = "[No DVD Drive Ready]";
\r
1676 mnu_dvd_drive.Text = "[No DVD Drive Ready / Found]";
\r
1681 #region Public Methods
\r
1683 /// Access the preset Handler and setup the preset panel.
\r
1685 public void loadPresetPanel()
\r
1687 if (presetHandler.checkIfPresetsAreOutOfDate())
\r
1688 if (!Properties.Settings.Default.presetNotification)
\r
1689 MessageBox.Show(splash,
\r
1690 "HandBrake has determined your built-in presets are out of date... These presets will now be updated.",
\r
1691 "Preset Update", MessageBoxButtons.OK, MessageBoxIcon.Information);
\r
1693 presetHandler.getPresetPanel(ref treeView_presets);
\r
1694 treeView_presets.Update();
\r
1700 /// If the queue is being processed, prompt the user to confirm application close.
\r
1702 /// <param name="e"></param>
\r
1703 protected override void OnFormClosing(FormClosingEventArgs e)
\r
1705 // If currently encoding, the queue isn't paused, and there are queue items to process, prompt to confirm close.
\r
1706 if ((encodeQueue.isEncoding) && (!encodeQueue.PauseRequested) && (encodeQueue.Count > 0))
\r
1708 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
1709 "Close HandBrake?", MessageBoxButtons.YesNo, MessageBoxIcon.Question);
\r
1710 if (result == DialogResult.No)
\r
1713 base.OnFormClosing(e);
\r
1717 #region In-GUI Encode Status (Experimental)
\r
1718 private void encodeMonitorThread()
\r
1722 Parser encode = new Parser(encodeQueue.hbProcess.StandardOutput.BaseStream);
\r
1723 encode.OnEncodeProgress += encodeOnEncodeProgress;
\r
1724 while (!encode.EndOfStream)
\r
1725 encode.readEncodeStatus();
\r
1727 catch (Exception exc)
\r
1729 MessageBox.Show(exc.ToString(), "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
1732 private void encodeOnEncodeProgress(object Sender, int CurrentTask, int TaskCount, float PercentComplete, float CurrentFps, float AverageFps, TimeSpan TimeRemaining)
\r
1734 if (this.InvokeRequired)
\r
1736 this.BeginInvoke(new EncodeProgressEventHandler(encodeOnEncodeProgress),
\r
1737 new object[] { Sender, CurrentTask, TaskCount, PercentComplete, CurrentFps, AverageFps, TimeRemaining });
\r
1740 lbl_encode.Text = string.Format("Encode Progress: {0}%, FPS: {1}, Avg FPS: {2}, Time Remaining: {3} ", PercentComplete, CurrentFps, AverageFps, TimeRemaining);
\r
1744 // This is the END of the road ****************************************
\r