1 /* EncodeAndQueueHandler.cs $
\r
3 This file is part of the HandBrake source code.
\r
4 Homepage: <http://handbrake.fr/>.
\r
5 It may be used under the terms of the GNU General Public License. */
\r
8 using System.Collections.Generic;
\r
9 using System.Collections.ObjectModel;
\r
10 using System.Diagnostics;
\r
12 using System.Threading;
\r
13 using System.Windows.Forms;
\r
14 using System.Xml.Serialization;
\r
15 using Handbrake.Functions;
\r
17 namespace Handbrake.EncodeQueue
\r
19 public class EncodeAndQueueHandler
\r
21 private static XmlSerializer serializer = new XmlSerializer(typeof(List<Job>));
\r
22 private List<Job> queue = new List<Job>();
\r
23 private int nextJobId;
\r
25 #region Event Handlers
\r
27 /// Fires when an encode job has been started.
\r
29 public event EventHandler NewJobStarted;
\r
32 /// Fires when a pause to the encode queue has been requested.
\r
34 public event EventHandler QueuePauseRequested;
\r
37 /// Fires when an encode job has been completed.
\r
39 public event EventHandler CurrentJobCompleted;
\r
42 /// Fires when the entire encode queue has completed.
\r
44 public event EventHandler QueueCompleted;
\r
49 /// Gets and removes the next job in the queue.
\r
51 /// <returns>The job that was removed from the queue.</returns>
\r
52 private Job GetNextJob()
\r
56 RemoveJob(0); // Remove the item which we are about to pass out.
\r
58 WriteQueueStateToFile("hb_queue_recovery.xml");
\r
64 /// Gets the current state of the encode queue.
\r
66 public ReadOnlyCollection<Job> CurrentQueue
\r
68 get { return queue.AsReadOnly(); }
\r
72 /// Gets the number of items in the queue.
\r
76 get { return queue.Count; }
\r
80 /// Adds an item to the queue.
\r
82 /// <param name="query">The query that will be passed to the HandBrake CLI.</param>
\r
83 /// <param name="source">The location of the source video.</param>
\r
84 /// <param name="destination">The location where the encoded video will be.</param>
\r
85 public void AddJob(string query, string source, string destination)
\r
87 Job newJob = new Job { Id = nextJobId++, Query = query, Source = source, Destination = destination };
\r
90 WriteQueueStateToFile("hb_queue_recovery.xml");
\r
94 /// Removes an item from the queue.
\r
96 /// <param name="index">The zero-based location of the job in the queue.</param>
\r
97 public void RemoveJob(int index)
\r
99 queue.RemoveAt(index);
\r
100 WriteQueueStateToFile("hb_queue_recovery.xml");
\r
104 /// Moves an item up one position in the queue.
\r
106 /// <param name="index">The zero-based location of the job in the queue.</param>
\r
107 public void MoveUp(int index)
\r
111 Job item = queue[index];
\r
113 queue.RemoveAt(index);
\r
114 queue.Insert((index - 1), item);
\r
117 WriteQueueStateToFile("hb_queue_recovery.xml"); // Update the queue recovery file
\r
121 /// Moves an item down one position in the queue.
\r
123 /// <param name="index">The zero-based location of the job in the queue.</param>
\r
124 public void MoveDown(int index)
\r
126 if (index < queue.Count - 1)
\r
128 Job item = queue[index];
\r
130 queue.RemoveAt(index);
\r
131 queue.Insert((index + 1), item);
\r
134 WriteQueueStateToFile("hb_queue_recovery.xml"); // Update the queue recovery file
\r
138 /// Writes the current state of the queue to a file.
\r
140 /// <param name="file">The location of the file to write the queue to.</param>
\r
141 public void WriteQueueStateToFile(string file)
\r
143 string tempPath = file == "hb_queue_recovery.xml" ? Path.Combine(Path.GetTempPath(), "hb_queue_recovery.xml") : file;
\r
147 using (FileStream strm = new FileStream(tempPath, FileMode.Create, FileAccess.Write))
\r
149 serializer.Serialize(strm, queue);
\r
156 // Any Errors will be out of diskspace/permissions problems.
\r
157 // Don't report them as they'll annoy the user.
\r
162 /// Writes the current state of the queue in the form of a batch (.bat) file.
\r
164 /// <param name="file">The location of the file to write the batch file to.</param>
\r
165 public void WriteBatchScriptToFile(string file)
\r
167 string queries = "";
\r
168 foreach (Job queue_item in queue)
\r
170 string q_item = queue_item.Query;
\r
171 string fullQuery = '"' + Application.StartupPath + "\\HandBrakeCLI.exe" + '"' + q_item;
\r
173 if (queries == string.Empty)
\r
174 queries = queries + fullQuery;
\r
176 queries = queries + " && " + fullQuery;
\r
178 string strCmdLine = queries;
\r
184 // Create a StreamWriter and open the file, Write the batch file query to the file and
\r
185 // Close the stream
\r
186 using (StreamWriter line = new StreamWriter(file))
\r
188 line.WriteLine(strCmdLine);
\r
191 MessageBox.Show("Your batch script has been sucessfully saved.", "Status", MessageBoxButtons.OK, MessageBoxIcon.Asterisk);
\r
195 MessageBox.Show("Unable to write to the file. Please make sure that the location has the correct permissions for file writing.", "Error", MessageBoxButtons.OK, MessageBoxIcon.Hand);
\r
202 /// Reads a serialized XML file that represents a queue of encoding jobs.
\r
204 /// <param name="file">The location of the file to read the queue from.</param>
\r
205 public void LoadQueueFromFile(string file)
\r
207 string tempPath = file == "hb_queue_recovery.xml" ? Path.Combine(Path.GetTempPath(), "hb_queue_recovery.xml") : file;
\r
209 if (File.Exists(tempPath))
\r
211 using (FileStream strm = new FileStream(tempPath, FileMode.Open, FileAccess.Read))
\r
213 if (strm.Length != 0)
\r
215 List<Job> list = serializer.Deserialize(strm) as List<Job>;
\r
218 foreach (Job item in list)
\r
221 if (file != "hb_queue_recovery.xml")
\r
222 WriteQueueStateToFile("hb_queue_recovery.xml");
\r
229 /// Checks the current queue for an existing instance of the specified destination.
\r
231 /// <param name="destination">The destination of the encode.</param>
\r
232 /// <returns>Whether or not the supplied destination is already in the queue.</returns>
\r
233 public bool CheckForDestinationDuplicate(string destination)
\r
235 foreach (Job checkItem in queue)
\r
237 if (checkItem.Destination.Contains(destination.Replace("\\\\", "\\")))
\r
249 /// Gets the last encode that was processed.
\r
251 /// <returns></returns>
\r
252 public Job LastEncode { get; set; }
\r
257 public Boolean PauseRequested { get; private set; }
\r
260 /// Starts encoding the first job in the queue and continues encoding until all jobs
\r
261 /// have been encoded.
\r
263 public void StartEncodeQueue()
\r
265 if (this.Count != 0)
\r
267 if (PauseRequested)
\r
268 PauseRequested = false;
\r
271 PauseRequested = false;
\r
274 Thread theQueue = new Thread(startProcess) { IsBackground = true };
\r
277 catch (Exception exc)
\r
279 MessageBox.Show(exc.ToString());
\r
286 /// Requests a pause of the encode queue.
\r
288 public void RequestPause()
\r
290 PauseRequested = true;
\r
292 if (QueuePauseRequested != null)
\r
293 QueuePauseRequested(this, new EventArgs());
\r
297 /// Stops the current job.
\r
299 public void EndEncodeJob()
\r
304 private void startProcess(object state)
\r
306 // Run through each item on the queue
\r
307 while (this.Count != 0)
\r
309 string query = GetNextJob().Query;
\r
310 WriteQueueStateToFile("hb_queue_recovery.xml"); // Update the queue recovery file
\r
314 if (NewJobStarted != null)
\r
315 NewJobStarted(this, new EventArgs());
\r
317 hbProcess.WaitForExit();
\r
319 AddCLIQueryToLog(query);
\r
320 CopyLog(LastEncode.Destination);
\r
323 hbProcess.Dispose();
\r
325 isEncoding = false;
\r
328 if (Properties.Settings.Default.growlEncode)
\r
329 GrowlCommunicator.Notify("Encode Completed", "Put down that cocktail...\nyour Handbrake encode is done.");
\r
331 if (CurrentJobCompleted != null)
\r
332 CurrentJobCompleted(this, new EventArgs());
\r
334 while (PauseRequested) // Need to find a better way of doing this.
\r
336 Thread.Sleep(5000);
\r
340 if (QueueCompleted != null)
\r
341 QueueCompleted(this, new EventArgs());
\r
343 // After the encode is done, we may want to shutdown, suspend etc.
\r
344 AfterEncodeAction();
\r
349 #region CLI and Log Handling
\r
350 public Process hbProcess { get; set; }
\r
351 public int processID { get; set; }
\r
352 public IntPtr processHandle { get; set; }
\r
353 public String currentQuery { get; set; }
\r
354 public Boolean isEncoding { get; set; }
\r
357 /// Execute a HandBrakeCLI process.
\r
359 /// <param name="query">The CLI Query</param>
\r
360 public void RunCli(string query)
\r
364 string handbrakeCLIPath = Path.Combine(Application.StartupPath, "HandBrakeCLI.exe");
\r
365 string logPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData) + "\\HandBrake\\logs", "last_encode_log.txt");
\r
366 string strCmdLine = String.Format(@" /C """"{0}"" {1} 2>""{2}"" """, handbrakeCLIPath, query, logPath);
\r
367 ProcessStartInfo cliStart = new ProcessStartInfo("CMD.exe", strCmdLine);
\r
369 if (Properties.Settings.Default.enocdeStatusInGui)
\r
371 cliStart.RedirectStandardOutput = true;
\r
372 cliStart.UseShellExecute = false;
\r
374 if (Properties.Settings.Default.cli_minimized)
\r
375 cliStart.WindowStyle = ProcessWindowStyle.Minimized;
\r
377 Process[] before = Process.GetProcesses(); // Get a list of running processes before starting.
\r
378 hbProcess = Process.Start(cliStart);
\r
379 processID = Main.getCliProcess(before);
\r
381 currentQuery = query;
\r
382 if (hbProcess != null)
\r
383 processHandle = hbProcess.MainWindowHandle; // Set the process Handle
\r
385 // Set the process Priority
\r
386 Process hbCliProcess = null;
\r
387 if (processID != -1)
\r
388 hbCliProcess = Process.GetProcessById(processID);
\r
390 if (hbCliProcess != null)
\r
391 switch (Properties.Settings.Default.processPriority)
\r
394 hbCliProcess.PriorityClass = ProcessPriorityClass.RealTime;
\r
397 hbCliProcess.PriorityClass = ProcessPriorityClass.High;
\r
399 case "Above Normal":
\r
400 hbCliProcess.PriorityClass = ProcessPriorityClass.AboveNormal;
\r
403 hbCliProcess.PriorityClass = ProcessPriorityClass.Normal;
\r
406 hbCliProcess.PriorityClass = ProcessPriorityClass.Idle;
\r
409 hbCliProcess.PriorityClass = ProcessPriorityClass.BelowNormal;
\r
413 catch (Exception exc)
\r
415 MessageBox.Show("It would appear that HandBrakeCLI has not started correctly. You should take a look at the Activity log as it may indicate the reason why.\n\n Detailed Error Information: error occured in runCli()\n\n" + exc, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
\r
421 /// Kill the CLI process
\r
423 private void CloseCLI()
\r
426 isEncoding = false;
\r
430 /// Perform an action after an encode. e.g a shutdown, standby, restart etc.
\r
432 private void AfterEncodeAction()
\r
434 isEncoding = false;
\r
435 currentQuery = String.Empty;
\r
438 if (Properties.Settings.Default.growlQueue)
\r
439 GrowlCommunicator.Notify("Queue Completed", "Put down that cocktail...\nyour Handbrake queue is done.");
\r
441 // Do something whent he encode ends.
\r
442 switch (Properties.Settings.Default.CompletionOption)
\r
445 Process.Start("Shutdown", "-s -t 60");
\r
448 Win32.ExitWindowsEx(0, 0);
\r
451 Application.SetSuspendState(PowerState.Suspend, true, true);
\r
454 Application.SetSuspendState(PowerState.Hibernate, true, true);
\r
456 case "Lock System":
\r
457 Win32.LockWorkStation();
\r
459 case "Quit HandBrake":
\r
460 Application.Exit();
\r
468 /// Append the CLI query to the start of the log file.
\r
470 /// <param name="query"></param>
\r
471 private static void AddCLIQueryToLog(string query)
\r
473 string logDir = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData) + "\\HandBrake\\logs";
\r
474 string logPath = Path.Combine(logDir, "last_encode_log.txt");
\r
476 StreamReader reader = new StreamReader(File.Open(logPath, FileMode.Open, FileAccess.Read, FileShare.Read));
\r
477 String log = reader.ReadToEnd();
\r
480 StreamWriter writer = new StreamWriter(File.Create(logPath));
\r
482 writer.Write("### CLI Query: " + query + "\n\n");
\r
483 writer.Write("#########################################\n\n");
\r
484 writer.WriteLine(log);
\r
490 /// Save a copy of the log to the users desired location or a default location
\r
491 /// if this feature is enabled in options.
\r
493 /// <param name="destination"></param>
\r
494 private static void CopyLog(string destination)
\r
498 string logDir = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData) + "\\HandBrake\\logs";
\r
499 string tempLogFile = Path.Combine(logDir, "last_encode_log.txt");
\r
501 string encodeDestinationPath = Path.GetDirectoryName(destination);
\r
502 String destinationFile = Path.GetFileName(destination);
\r
503 string encodeLogFile = destinationFile + " " + DateTime.Now.ToString().Replace("/", "-").Replace(":", "-") + ".txt";
\r
505 // Make sure the log directory exists.
\r
506 if (!Directory.Exists(logDir))
\r
507 Directory.CreateDirectory(logDir);
\r
509 // Copy the Log to HandBrakes log folder in the users applciation data folder.
\r
510 File.Copy(tempLogFile, Path.Combine(logDir, encodeLogFile));
\r
512 // Save a copy of the log file in the same location as the enocde.
\r
513 if (Properties.Settings.Default.saveLogWithVideo)
\r
514 File.Copy(tempLogFile, Path.Combine(encodeDestinationPath, encodeLogFile));
\r
516 // Save a copy of the log file to a user specified location
\r
517 if (Directory.Exists(Properties.Settings.Default.saveLogPath))
\r
518 if (Properties.Settings.Default.saveLogPath != String.Empty && Properties.Settings.Default.saveLogToSpecifiedPath)
\r
519 File.Copy(tempLogFile, Path.Combine(Properties.Settings.Default.saveLogPath, encodeLogFile));
\r
521 catch (Exception exc)
\r
523 MessageBox.Show("Something went a bit wrong trying to copy your log file.\nError Information:\n\n" + exc, "Error",
\r
524 MessageBoxButtons.OK, MessageBoxIcon.Error);
\r