/* Title.cs $ This file is part of the HandBrake source code. Homepage: . It may be used under the terms of the GNU General Public License. */ namespace Handbrake.Parsing { using System; using System.Collections.Generic; using System.Drawing; using System.Globalization; using System.IO; using System.Text.RegularExpressions; /// /// An object that represents a single Title of a DVD /// public class Title { /// /// The Culture Info /// private static readonly CultureInfo Culture = new CultureInfo("en-US", false); /// /// Initializes a new instance of the class. /// The constructor for this object /// public Title() { Angles = new List(); AudioTracks = new List(); Chapters = new List(); Subtitles = new List(); } /// /// Gets a Collection of chapters in this Title /// public List Chapters { get; private set; } /// /// Gets a Collection of audio tracks associated with this Title /// public List AudioTracks { get; private set; } /// /// Gets aCollection of subtitles associated with this Title /// public List Subtitles { get; private set; } /// /// Gets The track number of this Title /// public int TitleNumber { get; private set; } /// /// Gets a value indicating whether this is a MainTitle. /// public bool MainTitle { get; private set; } /// /// Gets the Source Name /// public string SourceName { get; private set; } /// /// Gets the length in time of this Title /// public TimeSpan Duration { get; private set; } /// /// Gets the resolution (width/height) of this Title /// public Size Resolution { get; private set; } /// /// Gets the aspect ratio of this Title /// public float AspectRatio { get; private set; } /// /// Gets Par Value /// public Size ParVal { get; private set; } /// /// Gets the automatically detected crop region for this Title. /// This is an int array with 4 items in it as so: /// 0: T /// 1: B /// 2: L /// 3: R /// public int[] AutoCropDimensions { get; private set; } /// /// Gets a Collection of Angles in this Title /// public List Angles { get; private set; } /// /// Gets the FPS of the source. /// public float Fps { get; private set; } /// /// Parse the Title Information /// /// A stingreader of output data /// A Title public static Title Parse(StringReader output) { var thisTitle = new Title(); Match m = Regex.Match(output.ReadLine(), @"^\+ title ([0-9]*):"); // Match track number for this title if (m.Success) thisTitle.TitleNumber = int.Parse(m.Groups[1].Value.Trim()); // If we are scanning a groupd of files, we'll want to get the source name. string path = output.ReadLine(); m = Regex.Match(path, @" \+ Main Feature"); if (m.Success) { thisTitle.MainTitle = true; path = output.ReadLine(); } m = Regex.Match(path, @"^ \+ stream:"); if (m.Success) thisTitle.SourceName = path.Replace("+ stream:", string.Empty).Trim(); if (!Properties.Settings.Default.noDvdNav) { // Get the Angles for the title. m = Regex.Match(output.ReadLine(), @" \+ angle\(s\) ([0-9])"); if (m.Success) { string angleList = m.Value.Replace("+ angle(s) ", string.Empty).Trim(); int angleCount; int.TryParse(angleList, out angleCount); for (int i = 1; i <= angleCount; i++) thisTitle.Angles.Add(i.ToString()); } } // Get duration for this title m = Regex.Match(output.ReadLine(), @"^ \+ duration: ([0-9]{2}:[0-9]{2}:[0-9]{2})"); if (m.Success) thisTitle.Duration = TimeSpan.Parse(m.Groups[1].Value); // Get resolution, aspect ratio and FPS for this title m = Regex.Match(output.ReadLine(), @"^ \+ size: ([0-9]*)x([0-9]*), pixel aspect: ([0-9]*)/([0-9]*), display aspect: ([0-9]*\.[0-9]*), ([0-9]*\.[0-9]*) fps"); if (m.Success) { thisTitle.Resolution = new Size(int.Parse(m.Groups[1].Value), int.Parse(m.Groups[2].Value)); thisTitle.ParVal = new Size(int.Parse(m.Groups[3].Value), int.Parse(m.Groups[4].Value)); thisTitle.AspectRatio = float.Parse(m.Groups[5].Value, Culture); thisTitle.Fps = float.Parse(m.Groups[6].Value, Culture); } // Get autocrop region for this title m = Regex.Match(output.ReadLine(), @"^ \+ autocrop: ([0-9]*)/([0-9]*)/([0-9]*)/([0-9]*)"); if (m.Success) thisTitle.AutoCropDimensions = new[] { int.Parse(m.Groups[1].Value), int.Parse(m.Groups[2].Value), int.Parse(m.Groups[3].Value), int.Parse(m.Groups[4].Value) }; thisTitle.Chapters.AddRange(Chapter.ParseList(output)); thisTitle.AudioTracks.AddRange(AudioTrack.ParseList(output)); thisTitle.Subtitles.AddRange(Subtitle.ParseList(output)); return thisTitle; } /// /// Return a list of parsed titles /// /// The Output /// A List of titles public static Title[] ParseList(string output) { var titles = new List(); var sr = new StringReader(output); while (sr.Peek() == '+' || sr.Peek() == ' ') { // If the the character is a space, then chances are the line if (sr.Peek() == ' ') // If the character is a space, then chances are it's the combing detected line. sr.ReadLine(); // Skip over it else titles.Add(Parse(sr)); } return titles.ToArray(); } /// <summary> /// Override of the ToString method to provide an easy way to use this object in the UI /// </summary> /// <returns>A string representing this track in the format: {title #} (00:00:00)</returns> public override string ToString() { return string.Format("{0} ({1:00}:{2:00}:{3:00})", TitleNumber, Duration.Hours, Duration.Minutes, Duration.Seconds); } } }