AssParser.cs 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. #pragma warning disable CS1591
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Globalization;
  5. using System.IO;
  6. using System.Linq;
  7. using System.Text.RegularExpressions;
  8. using System.Threading;
  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. /// <inheritdoc />
  16. public SubtitleTrackInfo Parse(Stream stream, CancellationToken cancellationToken)
  17. {
  18. var trackInfo = new SubtitleTrackInfo();
  19. var trackEvents = new List<SubtitleTrackEvent>();
  20. var eventIndex = 1;
  21. using (var reader = new StreamReader(stream))
  22. {
  23. string line;
  24. while (!string.Equals(reader.ReadLine(), "[Events]", StringComparison.Ordinal))
  25. {
  26. }
  27. var headers = ParseFieldHeaders(reader.ReadLine());
  28. while ((line = reader.ReadLine()) != null)
  29. {
  30. cancellationToken.ThrowIfCancellationRequested();
  31. if (string.IsNullOrWhiteSpace(line))
  32. {
  33. continue;
  34. }
  35. if (line[0] == '[')
  36. {
  37. break;
  38. }
  39. var subEvent = new SubtitleTrackEvent { Id = eventIndex.ToString(_usCulture) };
  40. eventIndex++;
  41. const string Dialogue = "Dialogue: ";
  42. var sections = line.Substring(Dialogue.Length).Split(',');
  43. subEvent.StartPositionTicks = GetTicks(sections[headers["Start"]]);
  44. subEvent.EndPositionTicks = GetTicks(sections[headers["End"]]);
  45. subEvent.Text = string.Join(',', sections[headers["Text"]..]);
  46. RemoteNativeFormatting(subEvent);
  47. subEvent.Text = subEvent.Text.Replace("\\n", ParserValues.NewLine, StringComparison.OrdinalIgnoreCase);
  48. subEvent.Text = Regex.Replace(subEvent.Text, @"\{(\\[\w]+\(?([\w0-9]+,?)+\)?)+\}", string.Empty, RegexOptions.IgnoreCase);
  49. trackEvents.Add(subEvent);
  50. }
  51. }
  52. trackInfo.TrackEvents = trackEvents;
  53. return trackInfo;
  54. }
  55. private long GetTicks(ReadOnlySpan<char> time)
  56. {
  57. return TimeSpan.TryParseExact(time, @"h\:mm\:ss\.ff", _usCulture, out var span)
  58. ? span.Ticks : 0;
  59. }
  60. internal static Dictionary<string, int> ParseFieldHeaders(string line)
  61. {
  62. const string Format = "Format: ";
  63. var fields = line.Substring(Format.Length).Split(',').Select(x => x.Trim()).ToList();
  64. return new Dictionary<string, int>
  65. {
  66. { "Start", fields.IndexOf("Start") },
  67. { "End", fields.IndexOf("End") },
  68. { "Text", fields.IndexOf("Text") }
  69. };
  70. }
  71. private void RemoteNativeFormatting(SubtitleTrackEvent p)
  72. {
  73. int indexOfBegin = p.Text.IndexOf('{', StringComparison.Ordinal);
  74. string pre = string.Empty;
  75. while (indexOfBegin >= 0 && p.Text.IndexOf('}', StringComparison.Ordinal) > indexOfBegin)
  76. {
  77. string s = p.Text.Substring(indexOfBegin);
  78. if (s.StartsWith("{\\an1}", StringComparison.Ordinal) ||
  79. s.StartsWith("{\\an2}", StringComparison.Ordinal) ||
  80. s.StartsWith("{\\an3}", StringComparison.Ordinal) ||
  81. s.StartsWith("{\\an4}", StringComparison.Ordinal) ||
  82. s.StartsWith("{\\an5}", StringComparison.Ordinal) ||
  83. s.StartsWith("{\\an6}", StringComparison.Ordinal) ||
  84. s.StartsWith("{\\an7}", StringComparison.Ordinal) ||
  85. s.StartsWith("{\\an8}", StringComparison.Ordinal) ||
  86. s.StartsWith("{\\an9}", StringComparison.Ordinal))
  87. {
  88. pre = s.Substring(0, 6);
  89. }
  90. else if (s.StartsWith("{\\an1\\", StringComparison.Ordinal) ||
  91. s.StartsWith("{\\an2\\", StringComparison.Ordinal) ||
  92. s.StartsWith("{\\an3\\", StringComparison.Ordinal) ||
  93. s.StartsWith("{\\an4\\", StringComparison.Ordinal) ||
  94. s.StartsWith("{\\an5\\", StringComparison.Ordinal) ||
  95. s.StartsWith("{\\an6\\", StringComparison.Ordinal) ||
  96. s.StartsWith("{\\an7\\", StringComparison.Ordinal) ||
  97. s.StartsWith("{\\an8\\", StringComparison.Ordinal) ||
  98. s.StartsWith("{\\an9\\", StringComparison.Ordinal))
  99. {
  100. pre = s.Substring(0, 5) + "}";
  101. }
  102. int indexOfEnd = p.Text.IndexOf('}', StringComparison.Ordinal);
  103. p.Text = p.Text.Remove(indexOfBegin, (indexOfEnd - indexOfBegin) + 1);
  104. indexOfBegin = p.Text.IndexOf('{', StringComparison.Ordinal);
  105. }
  106. p.Text = pre + p.Text;
  107. }
  108. }
  109. }