OSDN Git Service

WinGui:
[handbrake-jp/handbrake-jp-git.git] / win / C# / Functions / Main.cs
index af0ce7f..f78d38a 100644 (file)
-/*  Common.cs $\r
-       \r
-          This file is part of the HandBrake source code.\r
-          Homepage: <http://handbrake.fr>.\r
-          It may be used under the terms of the GNU General Public License. */\r
-\r
-using System;\r
-using System.Collections;\r
-using System.Windows.Forms;\r
-using System.IO;\r
-using System.Diagnostics;\r
-using System.Text.RegularExpressions;\r
-using System.Collections.Generic;\r
-using System.Xml.Serialization;\r
+/*  Main.cs $\r
+    This file is part of the HandBrake source code.\r
+    Homepage: <http://handbrake.fr>.\r
+    It may be used under the terms of the GNU General Public License. */\r
 \r
 namespace Handbrake.Functions\r
 {\r
-    class Main\r
+    using System;\r
+    using System.Collections.Generic;\r
+    using System.Diagnostics;\r
+    using System.IO;\r
+    using System.Linq;\r
+    using System.Security.Cryptography;\r
+    using System.Text;\r
+    using System.Text.RegularExpressions;\r
+    using System.Windows.Forms;\r
+    using System.Xml.Serialization;\r
+\r
+    using HandBrake.Framework.Services;\r
+    using HandBrake.Framework.Services.Interfaces;\r
+    using HandBrake.ApplicationServices.Model;\r
+    using HandBrake.ApplicationServices.Parsing;\r
+    using HandBrake.ApplicationServices.Services.Interfaces;\r
+    using Model;\r
+\r
+    /// <summary>\r
+    /// Useful functions which various screens can use.\r
+    /// </summary>\r
+    public static class Main\r
     {\r
-        // Private Variables\r
-        private static XmlSerializer ser = new XmlSerializer(typeof(List<Queue.QueueItem>));\r
+        /// <summary>\r
+        /// The Error Service\r
+        /// </summary>\r
+        private static readonly IErrorService errorService = new ErrorService();\r
+\r
+        /// <summary>\r
+        /// The XML Serializer\r
+        /// </summary>\r
+        private static readonly XmlSerializer Ser = new XmlSerializer(typeof(List<QueueTask>));\r
 \r
         /// <summary>\r
         /// Calculate the duration of the selected title and chapters\r
         /// </summary>\r
-        public TimeSpan calculateDuration(string chapter_start, string chapter_end, Parsing.Title selectedTitle)\r
+        /// <param name="chapterStart">\r
+        /// The chapter Start.\r
+        /// </param>\r
+        /// <param name="chapterEnd">\r
+        /// The chapter End.\r
+        /// </param>\r
+        /// <param name="selectedTitle">\r
+        /// The selected Title.\r
+        /// </param>\r
+        /// <returns>\r
+        /// The calculated duration.\r
+        /// </returns>\r
+        public static TimeSpan CalculateDuration(int chapterStart, int chapterEnd, Title selectedTitle)\r
         {\r
-            TimeSpan Duration = TimeSpan.FromSeconds(0.0);\r
-\r
-            // Get the durations between the 2 chapter points and add them together.\r
-            if (chapter_start != "Auto" && chapter_end != "Auto")\r
+            TimeSpan duration = TimeSpan.FromSeconds(0.0);\r
+            chapterStart++;\r
+            chapterEnd++;\r
+            if (chapterStart != 0 && chapterEnd != 0 && chapterEnd <= selectedTitle.Chapters.Count)\r
             {\r
-                int start_chapter, end_chapter = 0;\r
-                int.TryParse(chapter_start, out start_chapter);\r
-                int.TryParse(chapter_end, out end_chapter);\r
-\r
-                int position = start_chapter - 1;\r
-\r
-                if (start_chapter <= end_chapter)\r
-                {\r
-                    if (end_chapter > selectedTitle.Chapters.Count)\r
-                        end_chapter = selectedTitle.Chapters.Count;\r
-\r
-                    while (position != end_chapter)\r
-                    {\r
-                        TimeSpan dur = selectedTitle.Chapters[position].Duration;\r
-                        Duration = Duration + dur;\r
-                        position++;\r
-                    }\r
-                }\r
+                for (int i = chapterStart; i <= chapterEnd; i++)\r
+                    duration += selectedTitle.Chapters[i - 1].Duration;\r
             }\r
-            return Duration;\r
+\r
+            return duration;\r
         }\r
 \r
         /// <summary>\r
-        /// Calculate the non-anamorphic resoltuion of the source\r
+        /// Set's up the DataGridView on the Chapters tab (frmMain)\r
         /// </summary>\r
-        /// <param name="width"></param>\r
-        /// <returns></returns>\r
-        public int cacluateNonAnamorphicHeight(int width, decimal top, decimal bottom, decimal left, decimal right, Parsing.Title selectedTitle)\r
+        /// <param name="title">\r
+        /// The currently selected title object.\r
+        /// This will be used to get chapter names if they exist.\r
+        /// </param>\r
+        /// <param name="dataChpt">\r
+        /// The DataGridView Control\r
+        /// </param>\r
+        /// <param name="chapterEnd">\r
+        /// The chapter End.\r
+        /// </param>\r
+        /// <returns>\r
+        /// The chapter naming.\r
+        /// </returns>\r
+        public static DataGridView ChapterNaming(Title title, DataGridView dataChpt, string chapterEnd)\r
         {\r
-            float aspect = selectedTitle.AspectRatio;\r
-            int aw = 0;\r
-            int ah = 0;\r
-            if (aspect.ToString() == "1.78")\r
-            {\r
-                aw = 16;\r
-                ah = 9;\r
-            }\r
-            else if (aspect.ToString() == "1.33")\r
-            {\r
-                aw = 4;\r
-                ah = 3;\r
-            }\r
-\r
-            if (aw != 0)\r
-            {\r
-                double a = width * selectedTitle.Resolution.Width * ah * (selectedTitle.Resolution.Height - (double)top - (double)bottom);\r
-                double b = selectedTitle.Resolution.Height * aw * (selectedTitle.Resolution.Width - (double)left - (double)right);\r
+            int i = 0, finish = 0;\r
 \r
-                double y = a / b;\r
+            if (chapterEnd != "Auto")\r
+                int.TryParse(chapterEnd, out finish);\r
 \r
-                // If it's not Mod 16, make it mod 16\r
-                if ((y % 16) != 0)\r
+            while (i < finish)\r
+            {\r
+                string chapterName = string.Empty;\r
+                if (title != null)\r
                 {\r
-                    double mod16 = y % 16;\r
-                    if (mod16 >= 8)\r
+                    if (title.Chapters.Count <= i && title.Chapters[i] != null)\r
                     {\r
-                        mod16 = 16 - mod16;\r
-                        y = y + mod16;\r
-                    }\r
-                    else\r
-                    {\r
-                        y = y - mod16;\r
+                        chapterName = title.Chapters[i].ChapterName;\r
                     }\r
                 }\r
 \r
-                //16 * (421 / 16)\r
-                //double z = ( 16 * (( y + 8 ) / 16 ) );\r
-                int x = int.Parse(y.ToString());\r
-                return x;\r
+                int n = dataChpt.Rows.Add();\r
+                dataChpt.Rows[n].Cells[0].Value = i + 1;\r
+                dataChpt.Rows[n].Cells[1].Value = string.IsNullOrEmpty(chapterName) ? "Chapter " + (i + 1) : chapterName;\r
+                dataChpt.Rows[n].Cells[0].ValueType = typeof(int);\r
+                dataChpt.Rows[n].Cells[1].ValueType = typeof(string);\r
+                i++;\r
             }\r
-            return 0;\r
+\r
+            return dataChpt;\r
         }\r
 \r
         /// <summary>\r
-        /// Select the longest title in the DVD title dropdown menu on frmMain\r
+        /// Import a CSV file which contains Chapter Names\r
         /// </summary>\r
-        public Parsing.Title selectLongestTitle(ComboBox drp_dvdtitle)\r
+        /// <param name="dataChpt">\r
+        /// The DataGridView Control\r
+        /// </param>\r
+        /// <param name="filename">\r
+        /// The filepath and name\r
+        /// </param>\r
+        /// <returns>A Populated DataGridView</returns>\r
+        public static DataGridView ImportChapterNames(DataGridView dataChpt, string filename)\r
         {\r
-            int current_largest = 0;\r
-            Handbrake.Parsing.Title title2Select;\r
-\r
-            // Check if there are titles in the DVD title dropdown menu and make sure, it's not just "Automatic"\r
-            if (drp_dvdtitle.Items[0].ToString() != "Automatic")\r
-                title2Select = (Handbrake.Parsing.Title)drp_dvdtitle.Items[0];\r
-            else\r
-                title2Select = null;\r
-\r
-            // So, If there are titles in the DVD Title dropdown menu, lets select the longest.\r
-            if (title2Select != null)\r
+            IDictionary<int, string> chapterMap = new Dictionary<int, string>();\r
+            try\r
             {\r
-                foreach (Handbrake.Parsing.Title x in drp_dvdtitle.Items)\r
+                StreamReader sr = new StreamReader(filename);\r
+                string csv = sr.ReadLine();\r
+                while (csv != null)\r
                 {\r
-                    string title = x.ToString();\r
-                    if (title != "Automatic")\r
+                    if (csv.Trim() != string.Empty)\r
                     {\r
-                        string[] y = title.Split(' ');\r
-                        string time = y[1].Replace("(", "").Replace(")", "");\r
-                        string[] z = time.Split(':');\r
-\r
-                        int hours = int.Parse(z[0]) * 60 * 60;\r
-                        int minutes = int.Parse(z[1]) * 60;\r
-                        int seconds = int.Parse(z[2]);\r
-                        int total_sec = hours + minutes + seconds;\r
-\r
-                        if (current_largest == 0)\r
-                        {\r
-                            current_largest = hours + minutes + seconds;\r
-                            title2Select = x;\r
-                        }\r
-                        else\r
-                        {\r
-                            if (total_sec > current_largest)\r
-                            {\r
-                                current_largest = total_sec;\r
-                                title2Select = x;\r
-                            }\r
-                        }\r
+                        csv = csv.Replace("\\,", "<!comma!>");\r
+                        string[] contents = csv.Split(',');\r
+                        int chapter;\r
+                        int.TryParse(contents[0], out chapter);\r
+                        chapterMap.Add(chapter, contents[1].Replace("<!comma!>", ","));\r
                     }\r
+                    csv = sr.ReadLine();\r
                 }\r
             }\r
-            return title2Select;\r
+            catch (Exception)\r
+            {\r
+                return null;\r
+            }\r
+\r
+            foreach (DataGridViewRow item in dataChpt.Rows)\r
+            {\r
+                string name;\r
+                chapterMap.TryGetValue((int)item.Cells[0].Value, out name);\r
+                item.Cells[1].Value = name ?? "Chapter " + item.Cells[0].Value;\r
+            }\r
+\r
+            return dataChpt;\r
         }\r
 \r
         /// <summary>\r
-        /// Set's up the DataGridView on the Chapters tab (frmMain)\r
+        /// Create a CSV file with the data from the Main Window Chapters tab\r
         /// </summary>\r
-        /// <param name="mainWindow"></param>\r
-        public DataGridView chapterNaming(DataGridView data_chpt, string chapter_end)\r
+        /// <param name="mainWindow">Main Window</param>\r
+        /// <param name="filePathName">Path to save the csv file</param>\r
+        /// <returns>True if successful </returns>\r
+        public static bool SaveChapterMarkersToCsv(frmMain mainWindow, string filePathName)\r
         {\r
-            int i = 0, finish = 0;\r
-\r
-            if (chapter_end != "Auto")\r
-                int.TryParse(chapter_end, out finish);\r
+            try\r
+            {\r
+                string csv = string.Empty;\r
 \r
-            while (i < finish)\r
+                foreach (DataGridViewRow row in mainWindow.data_chpt.Rows)\r
+                {\r
+                    csv += row.Cells[0].Value.ToString();\r
+                    csv += ",";\r
+                    csv += row.Cells[1].Value.ToString().Replace(",", "\\,");\r
+                    csv += Environment.NewLine;\r
+                }\r
+                StreamWriter file = new StreamWriter(filePathName);\r
+                file.Write(csv);\r
+                file.Close();\r
+                file.Dispose();\r
+                return true;\r
+            }\r
+            catch (Exception exc)\r
             {\r
-                int n = data_chpt.Rows.Add();\r
-                data_chpt.Rows[n].Cells[0].Value = (i + 1);\r
-                data_chpt.Rows[n].Cells[1].Value = "Chapter " + (i + 1);\r
-                data_chpt.Rows[n].Cells[0].ValueType = typeof(int);\r
-                data_chpt.Rows[n].Cells[1].ValueType = typeof(string);\r
-                i++;\r
+                ShowExceptiowWindow("Unable to save Chapter Makrers file! \nChapter marker names will NOT be saved in your encode", exc.ToString());\r
+                return false;\r
             }\r
-\r
-            return data_chpt;\r
         }\r
 \r
         /// <summary>\r
         /// Function which generates the filename and path automatically based on \r
         /// the Source Name, DVD title and DVD Chapters\r
         /// </summary>\r
-        /// <param name="mainWindow"></param>\r
-        public string autoName(ComboBox drp_dvdtitle, string chapter_start, string chatper_end, string source, string dest, int format)\r
+        /// <param name="mainWindow">\r
+        /// The main Window.\r
+        /// </param>\r
+        /// <returns>\r
+        /// The Generated FileName\r
+        /// </returns>\r
+        public static string AutoName(frmMain mainWindow)\r
         {\r
-            string AutoNamePath = string.Empty;\r
-            if (drp_dvdtitle.Text != "Automatic")\r
+            string autoNamePath = string.Empty;\r
+            if (mainWindow.drp_dvdtitle.Text != "Automatic")\r
             {\r
-                // Get the Source Name - THIS NEEDS FIXED\r
-                string[] sourceName = source.Split('\\');\r
-                source = sourceName[sourceName.Length - 1].Replace(".iso", "").Replace(".mpg", "").Replace(".ts", "").Replace(".ps", "");\r
-                source.Replace(".wmv", "").Replace(".mp4", "").Replace(".m4v", "").Replace(".avi", "").Replace(".ogm", "").Replace(".tivo", "").Replace(".img", "");\r
-                source.Replace(".mov", "").Replace(".rm", "");\r
+                // Get the Source Name and remove any invalid characters\r
+\r
+                string sourceName = Path.GetInvalidFileNameChars().Aggregate(Path.GetFileNameWithoutExtension(mainWindow.SourceName), (current, character) => current.Replace(character.ToString(), string.Empty));\r
+\r
+                if (Properties.Settings.Default.AutoNameRemoveUnderscore)\r
+                    sourceName = sourceName.Replace("_", " ");\r
+\r
+                if (Properties.Settings.Default.AutoNameTitleCase)\r
+                    sourceName = TitleCase(sourceName);\r
 \r
                 // Get the Selected Title Number\r
-                string[] titlesplit = drp_dvdtitle.Text.Split(' ');\r
-                string dvdTitle = titlesplit[0].Replace("Automatic", "");\r
+                string[] titlesplit = mainWindow.drp_dvdtitle.Text.Split(' ');\r
+                string dvdTitle = titlesplit[0].Replace("Automatic", string.Empty);\r
 \r
                 // Get the Chapter Start and Chapter End Numbers\r
-                string chapterStart = chapter_start.Replace("Auto", "");\r
-                string chapterFinish = chatper_end.Replace("Auto", "");\r
+                string chapterStart = mainWindow.drop_chapterStart.Text.Replace("Auto", string.Empty);\r
+                string chapterFinish = mainWindow.drop_chapterFinish.Text.Replace("Auto", string.Empty);\r
                 string combinedChapterTag = chapterStart;\r
-                if (chapterFinish != chapterStart && chapterFinish != "")\r
+                if (chapterFinish != chapterStart && chapterFinish != string.Empty)\r
                     combinedChapterTag = chapterStart + "-" + chapterFinish;\r
 \r
                 // Get the destination filename.\r
-                string destination_filename = "";\r
-                if (Properties.Settings.Default.autoNameFormat != "")\r
+                string destinationFilename;\r
+                if (Properties.Settings.Default.autoNameFormat != string.Empty)\r
                 {\r
-                    destination_filename = Properties.Settings.Default.autoNameFormat;\r
-                    destination_filename = destination_filename.Replace("{source}", source).Replace("{title}", dvdTitle).Replace("{chapters}", combinedChapterTag);\r
+                    destinationFilename = Properties.Settings.Default.autoNameFormat;\r
+                    destinationFilename = destinationFilename.Replace("{source}", sourceName)\r
+                                                             .Replace("{title}", dvdTitle)\r
+                                                             .Replace("{chapters}", combinedChapterTag);\r
                 }\r
                 else\r
-                    destination_filename = source + "_T" + dvdTitle + "_C" + combinedChapterTag;\r
+                    destinationFilename = sourceName + "_T" + dvdTitle + "_C" + combinedChapterTag;\r
+\r
+                // Add the appropriate file extension\r
+                if (mainWindow.drop_format.SelectedIndex == 0)\r
+                {\r
+                    destinationFilename += Properties.Settings.Default.useM4v == 0 || Properties.Settings.Default.useM4v == 2 || mainWindow.Check_ChapterMarkers.Checked ||\r
+                                           mainWindow.AudioSettings.RequiresM4V() || mainWindow.Subtitles.RequiresM4V()\r
+                                               ? ".m4v"\r
+                                               : ".mp4";\r
+                }\r
+                else if (mainWindow.drop_format.SelectedIndex == 1)\r
+                    destinationFilename += ".mkv";\r
 \r
                 // Now work out the path where the file will be stored.\r
                 // First case: If the destination box doesn't already contain a path, make one.\r
-                if (!dest.Contains("\\"))\r
+                if (!mainWindow.text_destination.Text.Contains(Path.DirectorySeparatorChar.ToString()))\r
                 {\r
-                    string filePath = "";\r
-                    if (Properties.Settings.Default.autoNamePath.Trim() != "")\r
+                    // If there is an auto name path, use it...\r
+                    if (Properties.Settings.Default.autoNamePath.Trim() == "{source_path}" && !string.IsNullOrEmpty(mainWindow.sourcePath))\r
                     {\r
-                        if (Properties.Settings.Default.autoNamePath.Trim() != "Click 'Browse' to set the default location")\r
-                            filePath = Properties.Settings.Default.autoNamePath + "\\";\r
+                        autoNamePath = Path.Combine(Path.GetDirectoryName(mainWindow.sourcePath), destinationFilename);\r
+                        if (autoNamePath == mainWindow.sourcePath)\r
+                        {\r
+                            // Append out_ to files that already exist or is the source file\r
+                            autoNamePath = Path.Combine(Path.GetDirectoryName(mainWindow.sourcePath), "output_" + destinationFilename);\r
+                        }\r
                     }\r
-\r
-                    if (format == 0)\r
-                        AutoNamePath = filePath + destination_filename + ".mp4";\r
-                    else if (format == 1)\r
-                        AutoNamePath = filePath + destination_filename + ".m4v";\r
-                    else if (format == 2)\r
-                        AutoNamePath = filePath + destination_filename + ".mkv";\r
-                    else if (format == 3)\r
-                        AutoNamePath = filePath + destination_filename + ".avi";\r
-                    else if (format == 4)\r
-                        AutoNamePath = filePath + destination_filename + ".ogm";\r
+                    else if (Properties.Settings.Default.autoNamePath.Trim() != string.Empty && Properties.Settings.Default.autoNamePath.Trim() != "Click 'Browse' to set the default location")\r
+                    {\r
+                        autoNamePath = Path.Combine(Properties.Settings.Default.autoNamePath, destinationFilename);\r
+                    }\r
+                    else // ...otherwise, output to the source directory\r
+                        autoNamePath = null;\r
                 }\r
                 else // Otherwise, use the path that is already there.\r
                 {\r
-                    string destination = AutoNamePath;\r
-                    string[] destName = dest.Split('\\');\r
-                    string[] extension = dest.Split('.');\r
-                    string ext = extension[extension.Length - 1];\r
-\r
-                    destName[destName.Length - 1] = destination_filename + "." + ext;\r
+                    // Use the path and change the file extension to match the previous destination\r
+                    autoNamePath = Path.Combine(Path.GetDirectoryName(mainWindow.text_destination.Text), destinationFilename);\r
 \r
-                    string fullDest = "";\r
-                    foreach (string part in destName)\r
-                    {\r
-                        if (fullDest != "")\r
-                            fullDest = fullDest + "\\" + part;\r
-                        else\r
-                            fullDest = fullDest + part;\r
-                    }\r
-                    return fullDest;\r
+                    if (Path.HasExtension(mainWindow.text_destination.Text))\r
+                        autoNamePath = Path.ChangeExtension(autoNamePath,\r
+                                                            Path.GetExtension(mainWindow.text_destination.Text));\r
                 }\r
             }\r
 \r
-            return AutoNamePath;\r
+            return autoNamePath;\r
         }\r
 \r
         /// <summary>\r
-        /// Checks for updates and returns true if an update is available.\r
+        /// Get's HandBrakes version data from the CLI.\r
         /// </summary>\r
-        /// <param name="debug">Turns on debug mode. Don't use on program startup</param>\r
-        /// <returns>Boolean True = Update available</returns>\r
-        public Boolean updateCheck(Boolean debug)\r
+        public static void SetCliVersionData()\r
         {\r
-            try\r
-            {\r
-                Functions.AppcastReader rssRead = new Functions.AppcastReader();\r
-                rssRead.getInfo(); // Initializes the class.\r
-                string build = rssRead.build();\r
+            string line;\r
 \r
-                int latest = int.Parse(build);\r
-                int current = Properties.Settings.Default.hb_build;\r
-                int skip = Properties.Settings.Default.skipversion;\r
+            // 0 = SVN Build / Version\r
+            // 1 = Build Date\r
 \r
-                if (latest == skip)\r
-                    return false;\r
-                else\r
-                {\r
-                    Boolean update = (latest > current);\r
-                    return update;\r
-                }\r
+            // Get the SHA1 Hash of HandBrakeCLI\r
+            byte[] hash;\r
+            using (Stream stream = File.OpenRead(Path.Combine(Application.StartupPath, "HandBrakeCLI.exe")))\r
+            {\r
+                hash = SHA1.Create().ComputeHash(stream);\r
             }\r
-            catch (Exception exc)\r
+            string base64Hash = Convert.ToBase64String(hash);\r
+         \r
+            // Compare the hash with the last known hash. If it's the same, return.\r
+            if (Properties.Settings.Default.CliExeHash == base64Hash)\r
             {\r
-                if (debug == true)\r
-                    MessageBox.Show("Unable to check for updates, Please try again later. \n" + exc.ToString(), "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);\r
-                return false;\r
+                return;\r
             }\r
-        }\r
 \r
-        /// <summary>\r
-        /// Get's HandBrakes version data from the CLI.\r
-        /// </summary>\r
-        /// <returns>Arraylist of Version Data. 0 = hb_version 1 = hb_build</returns>\r
-        public ArrayList getCliVersionData()\r
-        {\r
-            ArrayList cliVersionData = new ArrayList();\r
-            String line;\r
-\r
-            // 0 = SVN Build / Version\r
-            // 1 = Build Date\r
+            // It's not the same, so start the CLI to get it's version data.\r
             Process cliProcess = new Process();\r
-            ProcessStartInfo handBrakeCLI = new ProcessStartInfo("HandBrakeCLI.exe", " -u");\r
-            handBrakeCLI.UseShellExecute = false;\r
-            handBrakeCLI.RedirectStandardError = true;\r
-            handBrakeCLI.RedirectStandardOutput = true;\r
-            handBrakeCLI.CreateNoWindow = true;\r
-            cliProcess.StartInfo = handBrakeCLI;\r
+            ProcessStartInfo handBrakeCli = new ProcessStartInfo("HandBrakeCLI.exe", " -u -v0")\r
+                                                {\r
+                                                    UseShellExecute = false,\r
+                                                    RedirectStandardError = true,\r
+                                                    RedirectStandardOutput = true,\r
+                                                    CreateNoWindow = true\r
+                                                };\r
+            cliProcess.StartInfo = handBrakeCli;\r
 \r
             try\r
             {\r
                 cliProcess.Start();\r
+\r
                 // Retrieve standard output and report back to parent thread until the process is complete\r
                 TextReader stdOutput = cliProcess.StandardError;\r
 \r
                 while (!cliProcess.HasExited)\r
                 {\r
-                    line = stdOutput.ReadLine();\r
-                    if (line == null) line = "";\r
-                    Match m = Regex.Match(line, @"HandBrake ([0-9\.]*)*(svn[0-9]*[M]*)* \([0-9]*\)");\r
+                    line = stdOutput.ReadLine() ?? string.Empty;\r
+                    Match m = Regex.Match(line, @"HandBrake ([svnM0-9.]*) \(([0-9]*)\)");\r
+                    Match platform = Regex.Match(line, @"- ([A-Za-z0-9\s ]*) -");\r
 \r
                     if (m.Success)\r
                     {\r
-                        string data = line.Replace("(", "").Replace(")", "").Replace("HandBrake ", "");\r
-                        string[] arr = data.Split(' ');\r
-                        cliVersionData.Add(arr[0]);\r
-                        cliVersionData.Add(arr[1]);\r
-                        return cliVersionData;\r
+                        string version = m.Groups[1].Success ? m.Groups[1].Value : string.Empty;\r
+                        string build = m.Groups[2].Success ? m.Groups[2].Value : string.Empty;\r
+\r
+                        int buildValue;\r
+                        int.TryParse(build, out buildValue);\r
+\r
+                        Properties.Settings.Default.hb_build = buildValue;\r
+                        Properties.Settings.Default.hb_version = version;\r
+                    }\r
+\r
+                    if (platform.Success)\r
+                    {\r
+                        Properties.Settings.Default.hb_platform = platform.Value.Replace("-", string.Empty).Trim();\r
                     }\r
-                    if (cliProcess.TotalProcessorTime.Seconds > 10) // Don't wait longer than 10 seconds.\r
-                        killCLI();\r
 \r
+                    if (cliProcess.TotalProcessorTime.Seconds > 10) // Don't wait longer than 10 seconds.\r
+                    {\r
+                        Process cli = Process.GetProcessById(cliProcess.Id);\r
+                        if (!cli.HasExited)\r
+                        {\r
+                            cli.Kill();\r
+                        }\r
+                    }\r
                 }\r
+\r
+                Properties.Settings.Default.CliExeHash = base64Hash;\r
+\r
+                Properties.Settings.Default.Save();\r
             }\r
             catch (Exception e)\r
             {\r
-                MessageBox.Show("Unable to retrieve version information from the CLI. \nError:\n" + e);\r
-            }\r
-\r
-            cliVersionData.Add(0);\r
-            cliVersionData.Add("0");\r
-            return cliVersionData;\r
-        }\r
-        private void killCLI()\r
-        {\r
-            string AppName = "HandBrakeCLI";\r
-            AppName = AppName.ToUpper();\r
+                Properties.Settings.Default.hb_build = 0;\r
+                Properties.Settings.Default.CliExeHash = null;\r
+                Properties.Settings.Default.Save();\r
 \r
-            System.Diagnostics.Process[] prs = Process.GetProcesses();\r
-            foreach (System.Diagnostics.Process proces in prs)\r
-            {\r
-                if (proces.ProcessName.ToUpper() == AppName)\r
-                {\r
-                    proces.Refresh();\r
-                    if (!proces.HasExited)\r
-                        proces.Kill();\r
-                }\r
+                ShowExceptiowWindow("Unable to retrieve version information from the CLI.", e.ToString());\r
             }\r
         }\r
 \r
@@ -368,27 +377,376 @@ namespace Handbrake.Functions
         /// If it does, it means the last queue did not complete before HandBrake closed.\r
         /// So, return a boolean if true. \r
         /// </summary>\r
-        public Boolean check_queue_recovery()\r
+        /// <returns>\r
+        /// True if there is a queue to recover.\r
+        /// </returns>\r
+        public static List<string> CheckQueueRecovery()\r
         {\r
             try\r
             {\r
-                string tempPath = Path.Combine(Path.GetTempPath(), "hb_queue_recovery.xml");\r
-                if (File.Exists(tempPath))\r
+                string tempPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), @"HandBrake\");\r
+                List<string> queueFiles = new List<string>();\r
+\r
+                DirectoryInfo info = new DirectoryInfo(tempPath);\r
+                FileInfo[] logFiles = info.GetFiles("*.xml");\r
+                foreach (FileInfo file in logFiles)\r
                 {\r
-                    using (FileStream strm = new FileStream(tempPath, FileMode.Open, FileAccess.Read))\r
+                    if (!file.Name.Contains("hb_queue_recovery"))\r
+                        continue;\r
+\r
+                    using (FileStream strm = new FileStream(Path.Combine(file.DirectoryName, file.Name), FileMode.Open, FileAccess.Read))\r
                     {\r
-                        List<Queue.QueueItem> list = ser.Deserialize(strm) as List<Queue.QueueItem>;\r
-                        if (list.Count != 0)\r
-                            return true;\r
+                        List<QueueTask> list = Ser.Deserialize(strm) as List<QueueTask>;\r
+                        if (list != null)\r
+                        {\r
+                            if (list.Count != 0)\r
+                            {\r
+                                queueFiles.Add(file.Name);\r
+                            }\r
+                        }\r
                     }\r
                 }\r
-                return false;\r
+\r
+                return queueFiles;\r
             }\r
             catch (Exception)\r
             {\r
-                return false; // Keep quiet about the error.\r
+                return new List<string>(); // Keep quiet about the error.\r
+            }\r
+        }\r
+\r
+        /// <summary>\r
+        /// Recover a queue from file.\r
+        /// </summary>\r
+        /// <param name="encodeQueue">\r
+        /// The encode Queue.\r
+        /// </param>\r
+        public static void RecoverQueue(IQueueProcessor encodeQueue)\r
+        {\r
+            string appDataPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), @"HandBrake\");\r
+\r
+            DialogResult result = DialogResult.None;\r
+            List<string> queueFiles = CheckQueueRecovery();\r
+            if (queueFiles.Count == 1)\r
+            {\r
+                result = MessageBox.Show(\r
+                        "HandBrake has detected unfinished items on the queue from the last time the application was launched. Would you like to recover these?",\r
+                        "Queue Recovery Possible", MessageBoxButtons.YesNo, MessageBoxIcon.Question);\r
+            }\r
+            else if (queueFiles.Count > 1)\r
+            {\r
+                result = MessageBox.Show(\r
+                        "HandBrake has detected multiple unfinished queue files. These will be from multiple instances of HandBrake running. Would you like to recover all unfinished jobs?",\r
+                        "Queue Recovery Possible", MessageBoxButtons.YesNo, MessageBoxIcon.Question);\r
+            }\r
+\r
+            if (result == DialogResult.Yes)\r
+            {\r
+                foreach (string file in queueFiles)\r
+                {\r
+                    encodeQueue.QueueManager.RestoreQueue(appDataPath + file); // Start Recovery\r
+                }\r
+            }\r
+            else\r
+            {\r
+                if (IsMultiInstance) return; // Don't tamper with the files if we are multi instance\r
+\r
+                foreach (string file in queueFiles)\r
+                {\r
+                    if (File.Exists(Path.Combine(appDataPath, file)))\r
+                        File.Delete(Path.Combine(appDataPath, file));\r
+                }\r
+            }\r
+        }\r
+\r
+        /// <summary>\r
+        /// Gets a value indicating whether HandBrake is running in multi instance mode\r
+        /// </summary>\r
+        /// <returns>True if the UI has another instance running</returns>\r
+        public static bool IsMultiInstance\r
+        {\r
+            get\r
+            {\r
+                return Process.GetProcessesByName("HandBrake").Length > 0 ? true : false;\r
+            }\r
+        }\r
+\r
+        /// <summary>\r
+        /// Map languages and their iso639_2 value into a IDictionary\r
+        /// </summary>\r
+        /// <returns>A Dictionary containing the language and iso code</returns>\r
+        public static IDictionary<string, string> MapLanguages()\r
+        {\r
+            IDictionary<string, string> languageMap = new Dictionary<string, string>\r
+                                                          {\r
+                                                              {"Any", "und"}, \r
+                                                              {"Afar", "aar"}, \r
+                                                              {"Abkhazian", "abk"}, \r
+                                                              {"Afrikaans", "afr"}, \r
+                                                              {"Akan", "aka"}, \r
+                                                              {"Albanian", "sqi"}, \r
+                                                              {"Amharic", "amh"}, \r
+                                                              {"Arabic", "ara"}, \r
+                                                              {"Aragonese", "arg"}, \r
+                                                              {"Armenian", "hye"}, \r
+                                                              {"Assamese", "asm"}, \r
+                                                              {"Avaric", "ava"}, \r
+                                                              {"Avestan", "ave"}, \r
+                                                              {"Aymara", "aym"}, \r
+                                                              {"Azerbaijani", "aze"}, \r
+                                                              {"Bashkir", "bak"}, \r
+                                                              {"Bambara", "bam"}, \r
+                                                              {"Basque", "eus"}, \r
+                                                              {"Belarusian", "bel"}, \r
+                                                              {"Bengali", "ben"}, \r
+                                                              {"Bihari", "bih"}, \r
+                                                              {"Bislama", "bis"}, \r
+                                                              {"Bosnian", "bos"}, \r
+                                                              {"Breton", "bre"}, \r
+                                                              {"Bulgarian", "bul"}, \r
+                                                              {"Burmese", "mya"}, \r
+                                                              {"Catalan", "cat"}, \r
+                                                              {"Chamorro", "cha"}, \r
+                                                              {"Chechen", "che"}, \r
+                                                              {"Chinese", "zho"}, \r
+                                                              {"Church Slavic", "chu"}, \r
+                                                              {"Chuvash", "chv"}, \r
+                                                              {"Cornish", "cor"}, \r
+                                                              {"Corsican", "cos"}, \r
+                                                              {"Cree", "cre"}, \r
+                                                              {"Czech", "ces"}, \r
+                                                              {"Dansk", "dan"}, \r
+                                                              {"Divehi", "div"}, \r
+                                                              {"Nederlands", "nld"}, \r
+                                                              {"Dzongkha", "dzo"}, \r
+                                                              {"English", "eng"}, \r
+                                                              {"Esperanto", "epo"}, \r
+                                                              {"Estonian", "est"}, \r
+                                                              {"Ewe", "ewe"}, \r
+                                                              {"Faroese", "fao"}, \r
+                                                              {"Fijian", "fij"}, \r
+                                                              {"Suomi", "fin"}, \r
+                                                              {"Francais", "fra"}, \r
+                                                              {"Western Frisian", "fry"}, \r
+                                                              {"Fulah", "ful"}, \r
+                                                              {"Georgian", "kat"}, \r
+                                                              {"Deutsch", "deu"}, \r
+                                                              {"Gaelic (Scots)", "gla"}, \r
+                                                              {"Irish", "gle"}, \r
+                                                              {"Galician", "glg"}, \r
+                                                              {"Manx", "glv"}, \r
+                                                              {"Greek Modern", "ell"}, \r
+                                                              {"Guarani", "grn"}, \r
+                                                              {"Gujarati", "guj"}, \r
+                                                              {"Haitian", "hat"}, \r
+                                                              {"Hausa", "hau"}, \r
+                                                              {"Hebrew", "heb"}, \r
+                                                              {"Herero", "her"}, \r
+                                                              {"Hindi", "hin"}, \r
+                                                              {"Hiri Motu", "hmo"}, \r
+                                                              {"Magyar", "hun"}, \r
+                                                              {"Igbo", "ibo"}, \r
+                                                              {"Islenska", "isl"}, \r
+                                                              {"Ido", "ido"}, \r
+                                                              {"Sichuan Yi", "iii"}, \r
+                                                              {"Inuktitut", "iku"}, \r
+                                                              {"Interlingue", "ile"}, \r
+                                                              {"Interlingua", "ina"}, \r
+                                                              {"Indonesian", "ind"}, \r
+                                                              {"Inupiaq", "ipk"}, \r
+                                                              {"Italiano", "ita"}, \r
+                                                              {"Javanese", "jav"}, \r
+                                                              {"Japanese", "jpn"}, \r
+                                                              {"Kalaallisut", "kal"}, \r
+                                                              {"Kannada", "kan"}, \r
+                                                              {"Kashmiri", "kas"}, \r
+                                                              {"Kanuri", "kau"}, \r
+                                                              {"Kazakh", "kaz"}, \r
+                                                              {"Central Khmer", "khm"}, \r
+                                                              {"Kikuyu", "kik"}, \r
+                                                              {"Kinyarwanda", "kin"}, \r
+                                                              {"Kirghiz", "kir"}, \r
+                                                              {"Komi", "kom"}, \r
+                                                              {"Kongo", "kon"}, \r
+                                                              {"Korean", "kor"}, \r
+                                                              {"Kuanyama", "kua"}, \r
+                                                              {"Kurdish", "kur"}, \r
+                                                              {"Lao", "lao"}, \r
+                                                              {"Latin", "lat"}, \r
+                                                              {"Latvian", "lav"}, \r
+                                                              {"Limburgan", "lim"}, \r
+                                                              {"Lingala", "lin"}, \r
+                                                              {"Lithuanian", "lit"}, \r
+                                                              {"Luxembourgish", "ltz"}, \r
+                                                              {"Luba-Katanga", "lub"}, \r
+                                                              {"Ganda", "lug"}, \r
+                                                              {"Macedonian", "mkd"}, \r
+                                                              {"Marshallese", "mah"}, \r
+                                                              {"Malayalam", "mal"}, \r
+                                                              {"Maori", "mri"}, \r
+                                                              {"Marathi", "mar"}, \r
+                                                              {"Malay", "msa"}, \r
+                                                              {"Malagasy", "mlg"}, \r
+                                                              {"Maltese", "mlt"}, \r
+                                                              {"Moldavian", "mol"}, \r
+                                                              {"Mongolian", "mon"}, \r
+                                                              {"Nauru", "nau"}, \r
+                                                              {"Navajo", "nav"}, \r
+                                                              {"Ndebele, South", "nbl"}, \r
+                                                              {"Ndebele, North", "nde"}, \r
+                                                              {"Ndonga", "ndo"}, \r
+                                                              {"Nepali", "nep"}, \r
+                                                              {"Norwegian Nynorsk", "nno"}, \r
+                                                              {"Norwegian Bokmål", "nob"}, \r
+                                                              {"Norsk", "nor"}, \r
+                                                              {"Chichewa; Nyanja", "nya"}, \r
+                                                              {"Occitan", "oci"}, \r
+                                                              {"Ojibwa", "oji"}, \r
+                                                              {"Oriya", "ori"}, \r
+                                                              {"Oromo", "orm"}, \r
+                                                              {"Ossetian", "oss"}, \r
+                                                              {"Panjabi", "pan"}, \r
+                                                              {"Persian", "fas"}, \r
+                                                              {"Pali", "pli"}, \r
+                                                              {"Polish", "pol"}, \r
+                                                              {"Portugues", "por"}, \r
+                                                              {"Pushto", "pus"}, \r
+                                                              {"Quechua", "que"}, \r
+                                                              {"Romansh", "roh"}, \r
+                                                              {"Romanian", "ron"}, \r
+                                                              {"Rundi", "run"}, \r
+                                                              {"Russian", "rus"}, \r
+                                                              {"Sango", "sag"}, \r
+                                                              {"Sanskrit", "san"}, \r
+                                                              {"Serbian", "srp"}, \r
+                                                              {"Hrvatski", "hrv"}, \r
+                                                              {"Sinhala", "sin"}, \r
+                                                              {"Slovak", "slk"}, \r
+                                                              {"Slovenian", "slv"}, \r
+                                                              {"Northern Sami", "sme"}, \r
+                                                              {"Samoan", "smo"}, \r
+                                                              {"Shona", "sna"}, \r
+                                                              {"Sindhi", "snd"}, \r
+                                                              {"Somali", "som"}, \r
+                                                              {"Sotho Southern", "sot"}, \r
+                                                              {"Espanol", "spa"}, \r
+                                                              {"Sardinian", "srd"}, \r
+                                                              {"Swati", "ssw"}, \r
+                                                              {"Sundanese", "sun"}, \r
+                                                              {"Swahili", "swa"}, \r
+                                                              {"Svenska", "swe"}, \r
+                                                              {"Tahitian", "tah"}, \r
+                                                              {"Tamil", "tam"}, \r
+                                                              {"Tatar", "tat"}, \r
+                                                              {"Telugu", "tel"}, \r
+                                                              {"Tajik", "tgk"}, \r
+                                                              {"Tagalog", "tgl"}, \r
+                                                              {"Thai", "tha"}, \r
+                                                              {"Tibetan", "bod"}, \r
+                                                              {"Tigrinya", "tir"}, \r
+                                                              {"Tonga", "ton"}, \r
+                                                              {"Tswana", "tsn"}, \r
+                                                              {"Tsonga", "tso"}, \r
+                                                              {"Turkmen", "tuk"}, \r
+                                                              {"Turkish", "tur"}, \r
+                                                              {"Twi", "twi"}, \r
+                                                              {"Uighur", "uig"}, \r
+                                                              {"Ukrainian", "ukr"}, \r
+                                                              {"Urdu", "urd"}, \r
+                                                              {"Uzbek", "uzb"}, \r
+                                                              {"Venda", "ven"}, \r
+                                                              {"Vietnamese", "vie"}, \r
+                                                              {"Volapük", "vol"}, \r
+                                                              {"Welsh", "cym"}, \r
+                                                              {"Walloon", "wln"}, \r
+                                                              {"Wolof", "wol"}, \r
+                                                              {"Xhosa", "xho"}, \r
+                                                              {"Yiddish", "yid"}, \r
+                                                              {"Yoruba", "yor"}, \r
+                                                              {"Zhuang", "zha"}, \r
+                                                              {"Zulu", "zul"}\r
+                                                          };\r
+            return languageMap;\r
+        }\r
+\r
+        /// <summary>\r
+        /// Change a string to Title Case/\r
+        /// </summary>\r
+        /// <param name="input">\r
+        /// The input.\r
+        /// </param>\r
+        /// <returns>\r
+        /// A string in title case.\r
+        /// </returns>\r
+        public static string TitleCase(string input)\r
+        {\r
+            string[] tokens = input.Split(' ');\r
+            StringBuilder sb = new StringBuilder(input.Length);\r
+            foreach (string s in tokens)\r
+            {\r
+                if (!string.IsNullOrEmpty(s))\r
+                {\r
+                    sb.Append(s[0].ToString().ToUpper());\r
+                    sb.Append(s.Substring(1).ToLower());\r
+                    sb.Append(" ");\r
+                }\r
             }\r
+\r
+            return sb.ToString().Trim();\r
         }\r
 \r
+        /// <summary>\r
+        /// Show the Exception Window\r
+        /// </summary>\r
+        /// <param name="shortError">\r
+        /// The short error.\r
+        /// </param>\r
+        /// <param name="longError">\r
+        /// The long error.\r
+        /// </param>\r
+        public static void ShowExceptiowWindow(string shortError, string longError)\r
+        {\r
+            errorService.ShowError(shortError, longError);\r
+        }\r
+\r
+        /// <summary>\r
+        /// Get The Source from the CLI Query\r
+        /// </summary>\r
+        /// <param name="query">Full CLI Query</param>\r
+        /// <returns>The Source Path</returns>\r
+        public static string GetSourceFromQuery(string query)\r
+        {\r
+            int startIndex = query.IndexOf("-i \"");\r
+            if (startIndex != -1)\r
+            {\r
+                string input = query.Substring(startIndex).Replace("-i \"", string.Empty).Trim();\r
+\r
+                int closeIndex = input.IndexOf('"');\r
+\r
+                return closeIndex == -1 ? "Unknown" : input.Substring(0, closeIndex);\r
+            }\r
+\r
+            return "Unknown";\r
+        }\r
+\r
+        /// <summary>\r
+        /// Get the Destination from the CLI Query\r
+        /// </summary>\r
+        /// <param name="query">Full CLI Query</param>\r
+        /// <returns>The Destination path</returns>\r
+        public static string GetDestinationFromQuery(string query)\r
+        {\r
+            int startIndex = query.IndexOf("-o \"");\r
+            if (startIndex != -1)\r
+            {\r
+                string output = query.Substring(startIndex).Replace("-o \"", string.Empty).Trim();\r
+\r
+                int closeIndex = output.IndexOf('"');\r
+\r
+                return closeIndex == -1 ? "Unknown" : output.Substring(0, closeIndex);\r
+            }\r
+\r
+            return "Unknown";\r
+        }\r
     }\r
 }
\ No newline at end of file