AssParser.cs 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  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 (reader.ReadLine() != "[Events]")
  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. var sections = line.Substring(10).Split(',');
  42. subEvent.StartPositionTicks = GetTicks(sections[headers["Start"]]);
  43. subEvent.EndPositionTicks = GetTicks(sections[headers["End"]]);
  44. subEvent.Text = string.Join(",", sections.Skip(headers["Text"]));
  45. RemoteNativeFormatting(subEvent);
  46. subEvent.Text = subEvent.Text.Replace("\\n", ParserValues.NewLine, StringComparison.OrdinalIgnoreCase);
  47. subEvent.Text = Regex.Replace(subEvent.Text, @"\{(\\[\w]+\(?([\w\d]+,?)+\)?)+\}", string.Empty, RegexOptions.IgnoreCase);
  48. trackEvents.Add(subEvent);
  49. }
  50. }
  51. trackInfo.TrackEvents = trackEvents.ToArray();
  52. return trackInfo;
  53. }
  54. private long GetTicks(ReadOnlySpan<char> time)
  55. {
  56. return TimeSpan.TryParseExact(time, @"h\:mm\:ss\.ff", _usCulture, out var span)
  57. ? span.Ticks : 0;
  58. }
  59. private Dictionary<string, int> ParseFieldHeaders(string line)
  60. {
  61. var fields = line.Substring(8).Split(',').Select(x => x.Trim()).ToList();
  62. return new Dictionary<string, int>
  63. {
  64. { "Start", fields.IndexOf("Start") },
  65. { "End", fields.IndexOf("End") },
  66. { "Text", fields.IndexOf("Text") }
  67. };
  68. }
  69. private void RemoteNativeFormatting(SubtitleTrackEvent p)
  70. {
  71. int indexOfBegin = p.Text.IndexOf('{');
  72. string pre = string.Empty;
  73. while (indexOfBegin >= 0 && p.Text.IndexOf('}') > indexOfBegin)
  74. {
  75. string s = p.Text.Substring(indexOfBegin);
  76. if (s.StartsWith("{\\an1}", StringComparison.Ordinal) ||
  77. s.StartsWith("{\\an2}", StringComparison.Ordinal) ||
  78. s.StartsWith("{\\an3}", StringComparison.Ordinal) ||
  79. s.StartsWith("{\\an4}", StringComparison.Ordinal) ||
  80. s.StartsWith("{\\an5}", StringComparison.Ordinal) ||
  81. s.StartsWith("{\\an6}", StringComparison.Ordinal) ||
  82. s.StartsWith("{\\an7}", StringComparison.Ordinal) ||
  83. s.StartsWith("{\\an8}", StringComparison.Ordinal) ||
  84. s.StartsWith("{\\an9}", StringComparison.Ordinal))
  85. {
  86. pre = s.Substring(0, 6);
  87. }
  88. else if (s.StartsWith("{\\an1\\", StringComparison.Ordinal) ||
  89. s.StartsWith("{\\an2\\", StringComparison.Ordinal) ||
  90. s.StartsWith("{\\an3\\", StringComparison.Ordinal) ||
  91. s.StartsWith("{\\an4\\", StringComparison.Ordinal) ||
  92. s.StartsWith("{\\an5\\", StringComparison.Ordinal) ||
  93. s.StartsWith("{\\an6\\", StringComparison.Ordinal) ||
  94. s.StartsWith("{\\an7\\", StringComparison.Ordinal) ||
  95. s.StartsWith("{\\an8\\", StringComparison.Ordinal) ||
  96. s.StartsWith("{\\an9\\", StringComparison.Ordinal))
  97. {
  98. pre = s.Substring(0, 5) + "}";
  99. }
  100. int indexOfEnd = p.Text.IndexOf('}');
  101. p.Text = p.Text.Remove(indexOfBegin, (indexOfEnd - indexOfBegin) + 1);
  102. indexOfBegin = p.Text.IndexOf('{');
  103. }
  104. p.Text = pre + p.Text;
  105. }
  106. }
  107. }