AssParser.cs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Text.RegularExpressions;
  7. using System.Threading;
  8. using MediaBrowser.Model.Extensions;
  9. using MediaBrowser.Model.MediaInfo;
  10. namespace MediaBrowser.MediaEncoding.Subtitles
  11. {
  12. public class AssParser : ISubtitleParser
  13. {
  14. private readonly CultureInfo _usCulture = new CultureInfo("en-US");
  15. public SubtitleTrackInfo Parse(Stream stream, CancellationToken cancellationToken)
  16. {
  17. var trackInfo = new SubtitleTrackInfo();
  18. var trackEvents = new List<SubtitleTrackEvent>();
  19. var eventIndex = 1;
  20. using (var reader = new StreamReader(stream))
  21. {
  22. string line;
  23. while (reader.ReadLine() != "[Events]")
  24. { }
  25. var headers = ParseFieldHeaders(reader.ReadLine());
  26. while ((line = reader.ReadLine()) != null)
  27. {
  28. cancellationToken.ThrowIfCancellationRequested();
  29. if (string.IsNullOrWhiteSpace(line))
  30. {
  31. continue;
  32. }
  33. if (line.StartsWith("["))
  34. break;
  35. if (string.IsNullOrEmpty(line))
  36. continue;
  37. var subEvent = new SubtitleTrackEvent { Id = eventIndex.ToString(_usCulture) };
  38. eventIndex++;
  39. var sections = line.Substring(10).Split(',');
  40. subEvent.StartPositionTicks = GetTicks(sections[headers["Start"]]);
  41. subEvent.EndPositionTicks = GetTicks(sections[headers["End"]]);
  42. subEvent.Text = string.Join(",", sections.Skip(headers["Text"]));
  43. RemoteNativeFormatting(subEvent);
  44. subEvent.Text = subEvent.Text.Replace("\\n", ParserValues.NewLine, StringComparison.OrdinalIgnoreCase);
  45. subEvent.Text = Regex.Replace(subEvent.Text, @"\{(\\[\w]+\(?([\w\d]+,?)+\)?)+\}", string.Empty, RegexOptions.IgnoreCase);
  46. trackEvents.Add(subEvent);
  47. }
  48. }
  49. trackInfo.TrackEvents = trackEvents.ToArray();
  50. return trackInfo;
  51. }
  52. long GetTicks(string time)
  53. {
  54. return TimeSpan.TryParseExact(time, @"h\:mm\:ss\.ff", _usCulture, out var span)
  55. ? span.Ticks : 0;
  56. }
  57. private Dictionary<string, int> ParseFieldHeaders(string line)
  58. {
  59. var fields = line.Substring(8).Split(',').Select(x => x.Trim()).ToList();
  60. var result = new Dictionary<string, int> {
  61. {"Start", fields.IndexOf("Start")},
  62. {"End", fields.IndexOf("End")},
  63. {"Text", fields.IndexOf("Text")}
  64. };
  65. return result;
  66. }
  67. /// <summary>
  68. /// Credit: https://github.com/SubtitleEdit/subtitleedit/blob/master/src/Logic/SubtitleFormats/AdvancedSubStationAlpha.cs
  69. /// </summary>
  70. private void RemoteNativeFormatting(SubtitleTrackEvent p)
  71. {
  72. int indexOfBegin = p.Text.IndexOf('{');
  73. string pre = string.Empty;
  74. while (indexOfBegin >= 0 && p.Text.IndexOf('}') > indexOfBegin)
  75. {
  76. string s = p.Text.Substring(indexOfBegin);
  77. if (s.StartsWith("{\\an1}", StringComparison.Ordinal) ||
  78. s.StartsWith("{\\an2}", StringComparison.Ordinal) ||
  79. s.StartsWith("{\\an3}", StringComparison.Ordinal) ||
  80. s.StartsWith("{\\an4}", StringComparison.Ordinal) ||
  81. s.StartsWith("{\\an5}", StringComparison.Ordinal) ||
  82. s.StartsWith("{\\an6}", StringComparison.Ordinal) ||
  83. s.StartsWith("{\\an7}", StringComparison.Ordinal) ||
  84. s.StartsWith("{\\an8}", StringComparison.Ordinal) ||
  85. s.StartsWith("{\\an9}", StringComparison.Ordinal))
  86. {
  87. pre = s.Substring(0, 6);
  88. }
  89. else if (s.StartsWith("{\\an1\\", StringComparison.Ordinal) ||
  90. s.StartsWith("{\\an2\\", StringComparison.Ordinal) ||
  91. s.StartsWith("{\\an3\\", StringComparison.Ordinal) ||
  92. s.StartsWith("{\\an4\\", StringComparison.Ordinal) ||
  93. s.StartsWith("{\\an5\\", StringComparison.Ordinal) ||
  94. s.StartsWith("{\\an6\\", StringComparison.Ordinal) ||
  95. s.StartsWith("{\\an7\\", StringComparison.Ordinal) ||
  96. s.StartsWith("{\\an8\\", StringComparison.Ordinal) ||
  97. s.StartsWith("{\\an9\\", StringComparison.Ordinal))
  98. {
  99. pre = s.Substring(0, 5) + "}";
  100. }
  101. int indexOfEnd = p.Text.IndexOf('}');
  102. p.Text = p.Text.Remove(indexOfBegin, (indexOfEnd - indexOfBegin) + 1);
  103. indexOfBegin = p.Text.IndexOf('{');
  104. }
  105. p.Text = pre + p.Text;
  106. }
  107. }
  108. }