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);
  15. if (preferDefaultTrack)
  16. {
  17. var defaultStream = streams.FirstOrDefault(i => i.IsDefault);
  18. if (defaultStream != 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. .ToList();
  42. MediaStream? stream = null;
  43. if (mode == SubtitlePlaybackMode.Default)
  44. {
  45. // Prefer embedded metadata over smart logic
  46. stream = sortedStreams.FirstOrDefault(s => s.IsExternal || s.IsForced || s.IsDefault);
  47. // if the audio language is not understood by the user, load their preferred subs, if there are any
  48. if (stream == null && !preferredLanguages.Contains(audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  49. {
  50. stream = sortedStreams.FirstOrDefault(s => !s.IsForced && preferredLanguages.Contains(s.Language, StringComparison.OrdinalIgnoreCase));
  51. }
  52. }
  53. else if (mode == SubtitlePlaybackMode.Smart)
  54. {
  55. // if the audio language is not understood by the user, load their preferred subs, if there are any
  56. if (!preferredLanguages.Contains(audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  57. {
  58. stream = streams.FirstOrDefault(s => !s.IsForced && preferredLanguages.Contains(s.Language, StringComparison.OrdinalIgnoreCase)) ??
  59. streams.FirstOrDefault(s => preferredLanguages.Contains(s.Language, StringComparison.OrdinalIgnoreCase));
  60. }
  61. }
  62. else if (mode == SubtitlePlaybackMode.Always)
  63. {
  64. // always load the most suitable full subtitles
  65. stream = sortedStreams.FirstOrDefault(s => !s.IsForced);
  66. }
  67. else if (mode == SubtitlePlaybackMode.OnlyForced)
  68. {
  69. // always load the most suitable full subtitles
  70. stream = sortedStreams.FirstOrDefault(x => x.IsForced);
  71. }
  72. // load forced subs if we have found no suitable full subtitles
  73. stream ??= sortedStreams.FirstOrDefault(s => s.IsForced && string.Equals(s.Language, audioTrackLanguage, StringComparison.OrdinalIgnoreCase));
  74. return stream?.Index;
  75. }
  76. private static IEnumerable<MediaStream> GetSortedStreams(IEnumerable<MediaStream> streams, MediaStreamType type, IReadOnlyList<string> languagePreferences)
  77. {
  78. // Give some preference to external text subs for better performance
  79. return streams
  80. .Where(i => i.Type == type)
  81. .OrderBy(i =>
  82. {
  83. var index = languagePreferences.FindIndex(x => string.Equals(x, i.Language, StringComparison.OrdinalIgnoreCase));
  84. return index == -1 ? 100 : index;
  85. })
  86. .ThenBy(i => GetBooleanOrderBy(i.IsDefault))
  87. .ThenBy(i => GetBooleanOrderBy(i.SupportsExternalStream))
  88. .ThenBy(i => GetBooleanOrderBy(i.IsTextSubtitleStream))
  89. .ThenBy(i => GetBooleanOrderBy(i.IsExternal))
  90. .ThenBy(i => i.Index);
  91. }
  92. public static void SetSubtitleStreamScores(
  93. IReadOnlyList<MediaStream> streams,
  94. IReadOnlyList<string> preferredLanguages,
  95. SubtitlePlaybackMode mode,
  96. string audioTrackLanguage)
  97. {
  98. if (mode == SubtitlePlaybackMode.None)
  99. {
  100. return;
  101. }
  102. var sortedStreams = GetSortedStreams(streams, MediaStreamType.Subtitle, preferredLanguages);
  103. var filteredStreams = new List<MediaStream>();
  104. if (mode == SubtitlePlaybackMode.Default)
  105. {
  106. // Prefer embedded metadata over smart logic
  107. filteredStreams = sortedStreams.Where(s => s.IsForced || s.IsDefault)
  108. .ToList();
  109. }
  110. else if (mode == SubtitlePlaybackMode.Smart)
  111. {
  112. // Prefer smart logic over embedded metadata
  113. if (!preferredLanguages.Contains(audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  114. {
  115. filteredStreams = sortedStreams.Where(s => !s.IsForced && preferredLanguages.Contains(s.Language, StringComparison.OrdinalIgnoreCase))
  116. .ToList();
  117. }
  118. }
  119. else if (mode == SubtitlePlaybackMode.Always)
  120. {
  121. // always load the most suitable full subtitles
  122. filteredStreams = sortedStreams.Where(s => !s.IsForced).ToList();
  123. }
  124. else if (mode == SubtitlePlaybackMode.OnlyForced)
  125. {
  126. // always load the most suitable full subtitles
  127. filteredStreams = sortedStreams.Where(s => s.IsForced).ToList();
  128. }
  129. // load forced subs if we have found no suitable full subtitles
  130. var iterStreams = filteredStreams.Count == 0
  131. ? sortedStreams.Where(s => s.IsForced && string.Equals(s.Language, audioTrackLanguage, StringComparison.OrdinalIgnoreCase))
  132. : filteredStreams;
  133. foreach (var stream in iterStreams)
  134. {
  135. stream.Score = GetSubtitleScore(stream, preferredLanguages);
  136. }
  137. }
  138. private static int GetSubtitleScore(MediaStream stream, IReadOnlyList<string> languagePreferences)
  139. {
  140. var values = new List<int>();
  141. var index = languagePreferences.FindIndex(x => string.Equals(x, stream.Language, StringComparison.OrdinalIgnoreCase));
  142. values.Add(index == -1 ? 0 : 100 - index);
  143. values.Add(stream.IsForced ? 1 : 0);
  144. values.Add(stream.IsDefault ? 1 : 0);
  145. values.Add(stream.SupportsExternalStream ? 1 : 0);
  146. values.Add(stream.IsTextSubtitleStream ? 1 : 0);
  147. values.Add(stream.IsExternal ? 1 : 0);
  148. values.Reverse();
  149. var scale = 1;
  150. var score = 0;
  151. foreach (var value in values)
  152. {
  153. score += scale * (value + 1);
  154. scale *= 10;
  155. }
  156. return score;
  157. }
  158. private static int GetBooleanOrderBy(bool value)
  159. {
  160. return value ? 0 : 1;
  161. }
  162. }
  163. }