MediaStreamSelector.cs 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #pragma warning disable CS1591
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using Jellyfin.Data.Enums;
  6. using Jellyfin.Extensions;
  7. using MediaBrowser.Model.Entities;
  8. namespace Emby.Server.Implementations.Library
  9. {
  10. public static class MediaStreamSelector
  11. {
  12. public static int? GetDefaultAudioStreamIndex(IReadOnlyList<MediaStream> streams, IReadOnlyList<string> preferredLanguages, bool preferDefaultTrack)
  13. {
  14. var sortedStreams = GetSortedStreams(streams, MediaStreamType.Audio, preferredLanguages).ToList();
  15. if (preferDefaultTrack)
  16. {
  17. var defaultStream = sortedStreams.FirstOrDefault(i => i.IsDefault);
  18. if (defaultStream is not null)
  19. {
  20. return defaultStream.Index;
  21. }
  22. }
  23. return sortedStreams.FirstOrDefault()?.Index;
  24. }
  25. public static int? GetDefaultSubtitleStreamIndex(
  26. IEnumerable<MediaStream> streams,
  27. IReadOnlyList<string> preferredLanguages,
  28. SubtitlePlaybackMode mode,
  29. string audioTrackLanguage)
  30. {
  31. if (mode == SubtitlePlaybackMode.None)
  32. {
  33. return null;
  34. }
  35. var sortedStreams = streams
  36. .Where(i => i.Type == MediaStreamType.Subtitle)
  37. .OrderByDescending(x => x.IsExternal)
  38. .ThenByDescending(x => x.IsForced && string.Equals(x.Language, audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  39. .ThenByDescending(x => x.IsForced)
  40. .ThenByDescending(x => x.IsDefault)
  41. .ThenByDescending(x => preferredLanguages.Contains(x.Language, StringComparison.OrdinalIgnoreCase))
  42. .ToList();
  43. MediaStream? stream = null;
  44. if (mode == SubtitlePlaybackMode.Default)
  45. {
  46. // Load subtitles according to external, forced and default flags.
  47. stream = sortedStreams.FirstOrDefault(x => x.IsExternal || x.IsForced || x.IsDefault);
  48. }
  49. else if (mode == SubtitlePlaybackMode.Smart)
  50. {
  51. // Only attempt to load subtitles if the audio language is not one of the user's preferred subtitle languages.
  52. // If no subtitles of preferred language available, use default behaviour.
  53. if (!preferredLanguages.Contains(audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  54. {
  55. stream = sortedStreams.FirstOrDefault(x => preferredLanguages.Contains(x.Language, StringComparison.OrdinalIgnoreCase)) ??
  56. sortedStreams.FirstOrDefault(x => x.IsExternal || x.IsForced || x.IsDefault);
  57. }
  58. else
  59. {
  60. // Respect forced flag.
  61. stream = sortedStreams.FirstOrDefault(x => x.IsForced);
  62. }
  63. }
  64. else if (mode == SubtitlePlaybackMode.Always)
  65. {
  66. // Always load (full/non-forced) subtitles of the user's preferred subtitle language if possible, otherwise default behaviour.
  67. stream = sortedStreams.FirstOrDefault(x => !x.IsForced && preferredLanguages.Contains(x.Language, StringComparison.OrdinalIgnoreCase)) ??
  68. sortedStreams.FirstOrDefault(x => x.IsExternal || x.IsForced || x.IsDefault);
  69. }
  70. else if (mode == SubtitlePlaybackMode.OnlyForced)
  71. {
  72. // Only load subtitles that are flagged forced.
  73. stream = sortedStreams.FirstOrDefault(x => x.IsForced);
  74. }
  75. return stream?.Index;
  76. }
  77. private static IEnumerable<MediaStream> GetSortedStreams(IEnumerable<MediaStream> streams, MediaStreamType type, IReadOnlyList<string> languagePreferences)
  78. {
  79. // Give some preference to external text subs for better performance
  80. return streams
  81. .Where(i => i.Type == type)
  82. .OrderByDescending(i => GetStreamScore(i, languagePreferences));
  83. }
  84. public static void SetSubtitleStreamScores(
  85. IReadOnlyList<MediaStream> streams,
  86. IReadOnlyList<string> preferredLanguages,
  87. SubtitlePlaybackMode mode,
  88. string audioTrackLanguage)
  89. {
  90. if (mode == SubtitlePlaybackMode.None)
  91. {
  92. return;
  93. }
  94. var sortedStreams = GetSortedStreams(streams, MediaStreamType.Subtitle, preferredLanguages).ToList();
  95. List<MediaStream>? filteredStreams = null;
  96. if (mode == SubtitlePlaybackMode.Default)
  97. {
  98. // Prefer embedded metadata over smart logic
  99. filteredStreams = sortedStreams.Where(s => s.IsForced || s.IsDefault)
  100. .ToList();
  101. }
  102. else if (mode == SubtitlePlaybackMode.Smart)
  103. {
  104. // Prefer smart logic over embedded metadata
  105. if (!preferredLanguages.Contains(audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  106. {
  107. filteredStreams = sortedStreams.Where(s => !s.IsForced && preferredLanguages.Contains(s.Language, StringComparison.OrdinalIgnoreCase))
  108. .ToList();
  109. }
  110. }
  111. else if (mode == SubtitlePlaybackMode.Always)
  112. {
  113. // Always load the most suitable full subtitles
  114. filteredStreams = sortedStreams.Where(s => !s.IsForced).ToList();
  115. }
  116. else if (mode == SubtitlePlaybackMode.OnlyForced)
  117. {
  118. // Always load the most suitable full subtitles
  119. filteredStreams = sortedStreams.Where(s => s.IsForced).ToList();
  120. }
  121. // Load forced subs if we have found no suitable full subtitles
  122. var iterStreams = filteredStreams is null || filteredStreams.Count == 0
  123. ? sortedStreams.Where(s => s.IsForced && string.Equals(s.Language, audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  124. : filteredStreams;
  125. foreach (var stream in iterStreams)
  126. {
  127. stream.Score = GetStreamScore(stream, preferredLanguages);
  128. }
  129. }
  130. internal static int GetStreamScore(MediaStream stream, IReadOnlyList<string> languagePreferences)
  131. {
  132. var index = languagePreferences.FindIndex(x => string.Equals(x, stream.Language, StringComparison.OrdinalIgnoreCase));
  133. var score = index == -1 ? 1 : 101 - index;
  134. score = (score * 10) + (stream.IsForced ? 2 : 1);
  135. score = (score * 10) + (stream.IsDefault ? 2 : 1);
  136. score = (score * 10) + (stream.SupportsExternalStream ? 2 : 1);
  137. score = (score * 10) + (stream.IsTextSubtitleStream ? 2 : 1);
  138. score = (score * 10) + (stream.IsExternal ? 2 : 1);
  139. return score;
  140. }
  141. }
  142. }