MediaStreamSelector.cs 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  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. .OrderBy(i =>
  83. {
  84. var index = languagePreferences.FindIndex(x => string.Equals(x, i.Language, StringComparison.OrdinalIgnoreCase));
  85. return index == -1 ? 100 : index;
  86. })
  87. .ThenBy(i => GetBooleanOrderBy(i.IsDefault))
  88. .ThenBy(i => GetBooleanOrderBy(i.SupportsExternalStream))
  89. .ThenBy(i => GetBooleanOrderBy(i.IsTextSubtitleStream))
  90. .ThenBy(i => GetBooleanOrderBy(i.IsExternal))
  91. .ThenBy(i => i.Index);
  92. }
  93. public static void SetSubtitleStreamScores(
  94. IReadOnlyList<MediaStream> streams,
  95. IReadOnlyList<string> preferredLanguages,
  96. SubtitlePlaybackMode mode,
  97. string audioTrackLanguage)
  98. {
  99. if (mode == SubtitlePlaybackMode.None)
  100. {
  101. return;
  102. }
  103. var sortedStreams = GetSortedStreams(streams, MediaStreamType.Subtitle, preferredLanguages);
  104. var filteredStreams = new List<MediaStream>();
  105. if (mode == SubtitlePlaybackMode.Default)
  106. {
  107. // Prefer embedded metadata over smart logic
  108. filteredStreams = sortedStreams.Where(s => s.IsForced || s.IsDefault)
  109. .ToList();
  110. }
  111. else if (mode == SubtitlePlaybackMode.Smart)
  112. {
  113. // Prefer smart logic over embedded metadata
  114. if (!preferredLanguages.Contains(audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  115. {
  116. filteredStreams = sortedStreams.Where(s => !s.IsForced && preferredLanguages.Contains(s.Language, StringComparison.OrdinalIgnoreCase))
  117. .ToList();
  118. }
  119. }
  120. else if (mode == SubtitlePlaybackMode.Always)
  121. {
  122. // always load the most suitable full subtitles
  123. filteredStreams = sortedStreams.Where(s => !s.IsForced).ToList();
  124. }
  125. else if (mode == SubtitlePlaybackMode.OnlyForced)
  126. {
  127. // always load the most suitable full subtitles
  128. filteredStreams = sortedStreams.Where(s => s.IsForced).ToList();
  129. }
  130. // load forced subs if we have found no suitable full subtitles
  131. var iterStreams = filteredStreams.Count == 0
  132. ? sortedStreams.Where(s => s.IsForced && string.Equals(s.Language, audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  133. : filteredStreams;
  134. foreach (var stream in iterStreams)
  135. {
  136. stream.Score = GetSubtitleScore(stream, preferredLanguages);
  137. }
  138. }
  139. private static int GetSubtitleScore(MediaStream stream, IReadOnlyList<string> languagePreferences)
  140. {
  141. var values = new List<int>();
  142. var index = languagePreferences.FindIndex(x => string.Equals(x, stream.Language, StringComparison.OrdinalIgnoreCase));
  143. values.Add(index == -1 ? 0 : 100 - index);
  144. values.Add(stream.IsForced ? 1 : 0);
  145. values.Add(stream.IsDefault ? 1 : 0);
  146. values.Add(stream.SupportsExternalStream ? 1 : 0);
  147. values.Add(stream.IsTextSubtitleStream ? 1 : 0);
  148. values.Add(stream.IsExternal ? 1 : 0);
  149. values.Reverse();
  150. var scale = 1;
  151. var score = 0;
  152. foreach (var value in values)
  153. {
  154. score += scale * (value + 1);
  155. scale *= 10;
  156. }
  157. return score;
  158. }
  159. private static int GetBooleanOrderBy(bool value)
  160. {
  161. return value ? 0 : 1;
  162. }
  163. }
  164. }