< Summary - Jellyfin

Information
Class: MediaBrowser.Model.Entities.MediaStream
Assembly: MediaBrowser.Model
File(s): /srv/git/jellyfin/MediaBrowser.Model/Entities/MediaStream.cs
Line coverage
75%
Covered lines: 170
Uncovered lines: 56
Coverable lines: 226
Total lines: 834
Line coverage: 75.2%
Branch coverage
66%
Covered branches: 193
Total branches: 291
Branch coverage: 66.3%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity Line coverage
.cctor()100%11100%
get_VideoRange()100%11100%
get_VideoRangeType()100%11100%
get_VideoDoViTitle()8%852554.16%
get_AudioSpatialFormat()87.5%88100%
get_DisplayTitle()57.31%4688261.42%
get_ReferenceFrameRate()100%22100%
get_IsTextSubtitleStream()100%66100%
get_IsPgsSubtitleStream()100%66100%
get_IsExtractableSubtitleStream()100%22100%
GetResolutionText()93.24%747496.66%
IsTextFormat(...)50%1212100%
IsPgsFormat(...)50%44100%
SupportsSubtitleConversionTo(...)50%171058.33%
GetVideoColorRange()55%2626061.7%

File(s)

/srv/git/jellyfin/MediaBrowser.Model/Entities/MediaStream.cs

#LineLine coverage
 1#nullable disable
 2#pragma warning disable CS1591
 3
 4using System;
 5using System.Collections.Generic;
 6using System.ComponentModel;
 7using System.Globalization;
 8using System.Linq;
 9using System.Text;
 10using System.Text.Json.Serialization;
 11using Jellyfin.Data.Enums;
 12using Jellyfin.Extensions;
 13using MediaBrowser.Model.Dlna;
 14using MediaBrowser.Model.Extensions;
 15using MediaBrowser.Model.MediaInfo;
 16
 17namespace MediaBrowser.Model.Entities
 18{
 19    /// <summary>
 20    /// Class MediaStream.
 21    /// </summary>
 22    public class MediaStream
 23    {
 124        private static readonly string[] _specialCodes =
 125        {
 126            // Uncoded languages.
 127            "mis",
 128            // Multiple languages.
 129            "mul",
 130            // Undetermined.
 131            "und",
 132            // No linguistic content; not applicable.
 133            "zxx"
 134        };
 35
 36        /// <summary>
 37        /// Gets or sets the codec.
 38        /// </summary>
 39        /// <value>The codec.</value>
 40        public string Codec { get; set; }
 41
 42        /// <summary>
 43        /// Gets or sets the codec tag.
 44        /// </summary>
 45        /// <value>The codec tag.</value>
 46        public string CodecTag { get; set; }
 47
 48        /// <summary>
 49        /// Gets or sets the language.
 50        /// </summary>
 51        /// <value>The language.</value>
 52        public string Language { get; set; }
 53
 54        /// <summary>
 55        /// Gets or sets the color range.
 56        /// </summary>
 57        /// <value>The color range.</value>
 58        public string ColorRange { get; set; }
 59
 60        /// <summary>
 61        /// Gets or sets the color space.
 62        /// </summary>
 63        /// <value>The color space.</value>
 64        public string ColorSpace { get; set; }
 65
 66        /// <summary>
 67        /// Gets or sets the color transfer.
 68        /// </summary>
 69        /// <value>The color transfer.</value>
 70        public string ColorTransfer { get; set; }
 71
 72        /// <summary>
 73        /// Gets or sets the color primaries.
 74        /// </summary>
 75        /// <value>The color primaries.</value>
 76        public string ColorPrimaries { get; set; }
 77
 78        /// <summary>
 79        /// Gets or sets the Dolby Vision version major.
 80        /// </summary>
 81        /// <value>The Dolby Vision version major.</value>
 82        public int? DvVersionMajor { get; set; }
 83
 84        /// <summary>
 85        /// Gets or sets the Dolby Vision version minor.
 86        /// </summary>
 87        /// <value>The Dolby Vision version minor.</value>
 88        public int? DvVersionMinor { get; set; }
 89
 90        /// <summary>
 91        /// Gets or sets the Dolby Vision profile.
 92        /// </summary>
 93        /// <value>The Dolby Vision profile.</value>
 94        public int? DvProfile { get; set; }
 95
 96        /// <summary>
 97        /// Gets or sets the Dolby Vision level.
 98        /// </summary>
 99        /// <value>The Dolby Vision level.</value>
 100        public int? DvLevel { get; set; }
 101
 102        /// <summary>
 103        /// Gets or sets the Dolby Vision rpu present flag.
 104        /// </summary>
 105        /// <value>The Dolby Vision rpu present flag.</value>
 106        public int? RpuPresentFlag { get; set; }
 107
 108        /// <summary>
 109        /// Gets or sets the Dolby Vision el present flag.
 110        /// </summary>
 111        /// <value>The Dolby Vision el present flag.</value>
 112        public int? ElPresentFlag { get; set; }
 113
 114        /// <summary>
 115        /// Gets or sets the Dolby Vision bl present flag.
 116        /// </summary>
 117        /// <value>The Dolby Vision bl present flag.</value>
 118        public int? BlPresentFlag { get; set; }
 119
 120        /// <summary>
 121        /// Gets or sets the Dolby Vision bl signal compatibility id.
 122        /// </summary>
 123        /// <value>The Dolby Vision bl signal compatibility id.</value>
 124        public int? DvBlSignalCompatibilityId { get; set; }
 125
 126        /// <summary>
 127        /// Gets or sets the Rotation in degrees.
 128        /// </summary>
 129        /// <value>The video rotation.</value>
 130        public int? Rotation { get; set; }
 131
 132        /// <summary>
 133        /// Gets or sets the comment.
 134        /// </summary>
 135        /// <value>The comment.</value>
 136        public string Comment { get; set; }
 137
 138        /// <summary>
 139        /// Gets or sets the time base.
 140        /// </summary>
 141        /// <value>The time base.</value>
 142        public string TimeBase { get; set; }
 143
 144        /// <summary>
 145        /// Gets or sets the codec time base.
 146        /// </summary>
 147        /// <value>The codec time base.</value>
 148        public string CodecTimeBase { get; set; }
 149
 150        /// <summary>
 151        /// Gets or sets the title.
 152        /// </summary>
 153        /// <value>The title.</value>
 154        public string Title { get; set; }
 155
 156        /// <summary>
 157        /// Gets the video range.
 158        /// </summary>
 159        /// <value>The video range.</value>
 160        [DefaultValue(VideoRange.Unknown)]
 161        public VideoRange VideoRange
 162        {
 163            get
 164            {
 32165                var (videoRange, _) = GetVideoColorRange();
 166
 32167                return videoRange;
 168            }
 169        }
 170
 171        /// <summary>
 172        /// Gets the video range type.
 173        /// </summary>
 174        /// <value>The video range type.</value>
 175        public VideoRangeType VideoRangeType
 176        {
 177            get
 178            {
 906179                var (_, videoRangeType) = GetVideoColorRange();
 180
 906181                return videoRangeType;
 182            }
 183        }
 184
 185        /// <summary>
 186        /// Gets the video dovi title.
 187        /// </summary>
 188        /// <value>The video dovi title.</value>
 189        public string VideoDoViTitle
 190        {
 191            get
 192            {
 32193                var dvProfile = DvProfile;
 32194                var rpuPresentFlag = RpuPresentFlag == 1;
 32195                var blPresentFlag = BlPresentFlag == 1;
 32196                var dvBlCompatId = DvBlSignalCompatibilityId;
 197
 32198                if (rpuPresentFlag
 32199                    && blPresentFlag
 32200                    && (dvProfile == 4
 32201                        || dvProfile == 5
 32202                        || dvProfile == 7
 32203                        || dvProfile == 8
 32204                        || dvProfile == 9
 32205                        || dvProfile == 10))
 206                {
 0207                    var title = "Dolby Vision Profile " + dvProfile;
 208
 0209                    if (dvBlCompatId > 0)
 210                    {
 0211                        title += "." + dvBlCompatId;
 212                    }
 213
 0214                    return dvBlCompatId switch
 0215                    {
 0216                        1 => title + " (HDR10)",
 0217                        2 => title + " (SDR)",
 0218                        4 => title + " (HLG)",
 0219                        6 => title + " (HDR10)", // Technically means Blu-ray, but practically always HDR10
 0220                        _ => title
 0221                    };
 222                }
 223
 32224                return null;
 225            }
 226        }
 227
 228        /// <summary>
 229        /// Gets the audio spatial format.
 230        /// </summary>
 231        /// <value>The audio spatial format.</value>
 232        [DefaultValue(AudioSpatialFormat.None)]
 233        public AudioSpatialFormat AudioSpatialFormat
 234        {
 235            get
 236            {
 34237                if (Type != MediaStreamType.Audio || string.IsNullOrEmpty(Profile))
 238                {
 32239                    return AudioSpatialFormat.None;
 240                }
 241
 2242                return
 2243                    Profile.Contains("Dolby Atmos", StringComparison.OrdinalIgnoreCase) ? AudioSpatialFormat.DolbyAtmos 
 2244                    Profile.Contains("DTS:X", StringComparison.OrdinalIgnoreCase) ? AudioSpatialFormat.DTSX :
 2245                    AudioSpatialFormat.None;
 246            }
 247        }
 248
 249        public string LocalizedUndefined { get; set; }
 250
 251        public string LocalizedDefault { get; set; }
 252
 253        public string LocalizedForced { get; set; }
 254
 255        public string LocalizedExternal { get; set; }
 256
 257        public string LocalizedHearingImpaired { get; set; }
 258
 259        public string DisplayTitle
 260        {
 261            get
 262            {
 40263                switch (Type)
 264                {
 265                    case MediaStreamType.Audio:
 266                    {
 14267                        var attributes = new List<string>();
 268
 269                        // Do not display the language code in display titles if unset or set to a special code. Show it
 14270                        if (!string.IsNullOrEmpty(Language) && !_specialCodes.Contains(Language, StringComparison.Ordina
 271                        {
 272                            // Get full language string i.e. eng -> English. Will not work for some languages which use 
 3273                            string fullLanguage = CultureInfo
 3274                                .GetCultures(CultureTypes.NeutralCultures)
 3275                                .FirstOrDefault(r => r.ThreeLetterISOLanguageName.Equals(Language, StringComparison.Ordi
 3276                                ?.DisplayName;
 3277                            attributes.Add(StringHelper.FirstToUpper(fullLanguage ?? Language));
 278                        }
 279
 14280                        if (!string.IsNullOrEmpty(Profile) && !string.Equals(Profile, "lc", StringComparison.OrdinalIgno
 281                        {
 0282                            attributes.Add(Profile);
 283                        }
 14284                        else if (!string.IsNullOrEmpty(Codec))
 285                        {
 2286                            attributes.Add(AudioCodec.GetFriendlyName(Codec));
 287                        }
 288
 14289                        if (!string.IsNullOrEmpty(ChannelLayout))
 290                        {
 0291                            attributes.Add(StringHelper.FirstToUpper(ChannelLayout));
 292                        }
 14293                        else if (Channels.HasValue)
 294                        {
 0295                            attributes.Add(Channels.Value.ToString(CultureInfo.InvariantCulture) + " ch");
 296                        }
 297
 14298                        if (IsDefault)
 299                        {
 4300                            attributes.Add(string.IsNullOrEmpty(LocalizedDefault) ? "Default" : LocalizedDefault);
 301                        }
 302
 14303                        if (IsExternal)
 304                        {
 8305                            attributes.Add(string.IsNullOrEmpty(LocalizedExternal) ? "External" : LocalizedExternal);
 306                        }
 307
 14308                        if (!string.IsNullOrEmpty(Title))
 309                        {
 2310                            var result = new StringBuilder(Title);
 16311                            foreach (var tag in attributes)
 312                            {
 313                                // Keep Tags that are not already in Title.
 6314                                if (!Title.Contains(tag, StringComparison.OrdinalIgnoreCase))
 315                                {
 6316                                    result.Append(" - ").Append(tag);
 317                                }
 318                            }
 319
 2320                            return result.ToString();
 321                        }
 322
 12323                        return string.Join(" - ", attributes);
 324                    }
 325
 326                    case MediaStreamType.Video:
 327                    {
 0328                        var attributes = new List<string>();
 329
 0330                        var resolutionText = GetResolutionText();
 331
 0332                        if (!string.IsNullOrEmpty(resolutionText))
 333                        {
 0334                            attributes.Add(resolutionText);
 335                        }
 336
 0337                        if (!string.IsNullOrEmpty(Codec))
 338                        {
 0339                            attributes.Add(Codec.ToUpperInvariant());
 340                        }
 341
 0342                        if (VideoDoViTitle is not null)
 343                        {
 0344                            attributes.Add(VideoDoViTitle);
 345                        }
 0346                        else if (VideoRange != VideoRange.Unknown)
 347                        {
 0348                            attributes.Add(VideoRange.ToString());
 349                        }
 350
 0351                        if (!string.IsNullOrEmpty(Title))
 352                        {
 0353                            var result = new StringBuilder(Title);
 0354                            foreach (var tag in attributes)
 355                            {
 356                                // Keep Tags that are not already in Title.
 0357                                if (!Title.Contains(tag, StringComparison.OrdinalIgnoreCase))
 358                                {
 0359                                    result.Append(" - ").Append(tag);
 360                                }
 361                            }
 362
 0363                            return result.ToString();
 364                        }
 365
 0366                        return string.Join(' ', attributes);
 367                    }
 368
 369                    case MediaStreamType.Subtitle:
 370                    {
 26371                        var attributes = new List<string>();
 372
 26373                        if (!string.IsNullOrEmpty(Language))
 374                        {
 375                            // Get full language string i.e. eng -> English. Will not work for some languages which use 
 16376                            string fullLanguage = CultureInfo
 16377                                .GetCultures(CultureTypes.NeutralCultures)
 16378                                .FirstOrDefault(r => r.ThreeLetterISOLanguageName.Equals(Language, StringComparison.Ordi
 16379                                ?.DisplayName;
 16380                            attributes.Add(StringHelper.FirstToUpper(fullLanguage ?? Language));
 381                        }
 382                        else
 383                        {
 10384                            attributes.Add(string.IsNullOrEmpty(LocalizedUndefined) ? "Und" : LocalizedUndefined);
 385                        }
 386
 26387                        if (IsHearingImpaired == true)
 388                        {
 6389                            attributes.Add(string.IsNullOrEmpty(LocalizedHearingImpaired) ? "Hearing Impaired" : Localiz
 390                        }
 391
 26392                        if (IsDefault)
 393                        {
 11394                            attributes.Add(string.IsNullOrEmpty(LocalizedDefault) ? "Default" : LocalizedDefault);
 395                        }
 396
 26397                        if (IsForced)
 398                        {
 12399                            attributes.Add(string.IsNullOrEmpty(LocalizedForced) ? "Forced" : LocalizedForced);
 400                        }
 401
 26402                        if (!string.IsNullOrEmpty(Codec))
 403                        {
 8404                            attributes.Add(Codec.ToUpperInvariant());
 405                        }
 406
 26407                        if (IsExternal)
 408                        {
 2409                            attributes.Add(string.IsNullOrEmpty(LocalizedExternal) ? "External" : LocalizedExternal);
 410                        }
 411
 26412                        if (!string.IsNullOrEmpty(Title))
 413                        {
 20414                            var result = new StringBuilder(Title);
 154415                            foreach (var tag in attributes)
 416                            {
 417                                // Keep Tags that are not already in Title.
 57418                                if (!Title.Contains(tag, StringComparison.OrdinalIgnoreCase))
 419                                {
 53420                                    result.Append(" - ").Append(tag);
 421                                }
 422                            }
 423
 20424                            return result.ToString();
 425                        }
 426
 6427                        return string.Join(" - ", attributes);
 428                    }
 429
 430                    default:
 0431                        return null;
 432                }
 433            }
 434        }
 435
 436        public string NalLengthSize { get; set; }
 437
 438        /// <summary>
 439        /// Gets or sets a value indicating whether this instance is interlaced.
 440        /// </summary>
 441        /// <value><c>true</c> if this instance is interlaced; otherwise, <c>false</c>.</value>
 442        public bool IsInterlaced { get; set; }
 443
 444        public bool? IsAVC { get; set; }
 445
 446        /// <summary>
 447        /// Gets or sets the channel layout.
 448        /// </summary>
 449        /// <value>The channel layout.</value>
 450        public string ChannelLayout { get; set; }
 451
 452        /// <summary>
 453        /// Gets or sets the bit rate.
 454        /// </summary>
 455        /// <value>The bit rate.</value>
 456        public int? BitRate { get; set; }
 457
 458        /// <summary>
 459        /// Gets or sets the bit depth.
 460        /// </summary>
 461        /// <value>The bit depth.</value>
 462        public int? BitDepth { get; set; }
 463
 464        /// <summary>
 465        /// Gets or sets the reference frames.
 466        /// </summary>
 467        /// <value>The reference frames.</value>
 468        public int? RefFrames { get; set; }
 469
 470        /// <summary>
 471        /// Gets or sets the length of the packet.
 472        /// </summary>
 473        /// <value>The length of the packet.</value>
 474        public int? PacketLength { get; set; }
 475
 476        /// <summary>
 477        /// Gets or sets the channels.
 478        /// </summary>
 479        /// <value>The channels.</value>
 480        public int? Channels { get; set; }
 481
 482        /// <summary>
 483        /// Gets or sets the sample rate.
 484        /// </summary>
 485        /// <value>The sample rate.</value>
 486        public int? SampleRate { get; set; }
 487
 488        /// <summary>
 489        /// Gets or sets a value indicating whether this instance is default.
 490        /// </summary>
 491        /// <value><c>true</c> if this instance is default; otherwise, <c>false</c>.</value>
 492        public bool IsDefault { get; set; }
 493
 494        /// <summary>
 495        /// Gets or sets a value indicating whether this instance is forced.
 496        /// </summary>
 497        /// <value><c>true</c> if this instance is forced; otherwise, <c>false</c>.</value>
 498        public bool IsForced { get; set; }
 499
 500        /// <summary>
 501        /// Gets or sets a value indicating whether this instance is for the hearing impaired.
 502        /// </summary>
 503        /// <value><c>true</c> if this instance is for the hearing impaired; otherwise, <c>false</c>.</value>
 504        public bool IsHearingImpaired { get; set; }
 505
 506        /// <summary>
 507        /// Gets or sets the height.
 508        /// </summary>
 509        /// <value>The height.</value>
 510        public int? Height { get; set; }
 511
 512        /// <summary>
 513        /// Gets or sets the width.
 514        /// </summary>
 515        /// <value>The width.</value>
 516        public int? Width { get; set; }
 517
 518        /// <summary>
 519        /// Gets or sets the average frame rate.
 520        /// </summary>
 521        /// <value>The average frame rate.</value>
 522        public float? AverageFrameRate { get; set; }
 523
 524        /// <summary>
 525        /// Gets or sets the real frame rate.
 526        /// </summary>
 527        /// <value>The real frame rate.</value>
 528        public float? RealFrameRate { get; set; }
 529
 530        /// <summary>
 531        /// Gets the framerate used as reference.
 532        /// Prefer AverageFrameRate, if that is null or an unrealistic value
 533        /// then fallback to RealFrameRate.
 534        /// </summary>
 535        /// <value>The reference frame rate.</value>
 536        public float? ReferenceFrameRate
 537        {
 538            get
 539            {
 540                // In some cases AverageFrameRate for videos will be read as 1000fps even if it is not.
 541                // This is probably due to a library compatibility issue.
 542                // See https://github.com/jellyfin/jellyfin/pull/12603#discussion_r1748044018 for more info.
 1051543                return AverageFrameRate < 1000 ? AverageFrameRate : RealFrameRate;
 544            }
 545        }
 546
 547        /// <summary>
 548        /// Gets or sets the profile.
 549        /// </summary>
 550        /// <value>The profile.</value>
 551        public string Profile { get; set; }
 552
 553        /// <summary>
 554        /// Gets or sets the type.
 555        /// </summary>
 556        /// <value>The type.</value>
 557        public MediaStreamType Type { get; set; }
 558
 559        /// <summary>
 560        /// Gets or sets the aspect ratio.
 561        /// </summary>
 562        /// <value>The aspect ratio.</value>
 563        public string AspectRatio { get; set; }
 564
 565        /// <summary>
 566        /// Gets or sets the index.
 567        /// </summary>
 568        /// <value>The index.</value>
 569        public int Index { get; set; }
 570
 571        /// <summary>
 572        /// Gets or sets the score.
 573        /// </summary>
 574        /// <value>The score.</value>
 575        public int? Score { get; set; }
 576
 577        /// <summary>
 578        /// Gets or sets a value indicating whether this instance is external.
 579        /// </summary>
 580        /// <value><c>true</c> if this instance is external; otherwise, <c>false</c>.</value>
 581        public bool IsExternal { get; set; }
 582
 583        /// <summary>
 584        /// Gets or sets the method.
 585        /// </summary>
 586        /// <value>The method.</value>
 587        public SubtitleDeliveryMethod? DeliveryMethod { get; set; }
 588
 589        /// <summary>
 590        /// Gets or sets the delivery URL.
 591        /// </summary>
 592        /// <value>The delivery URL.</value>
 593        public string DeliveryUrl { get; set; }
 594
 595        /// <summary>
 596        /// Gets or sets a value indicating whether this instance is external URL.
 597        /// </summary>
 598        /// <value><c>null</c> if [is external URL] contains no value, <c>true</c> if [is external URL]; otherwise, <c>f
 599        public bool? IsExternalUrl { get; set; }
 600
 601        public bool IsTextSubtitleStream
 602        {
 603            get
 604            {
 2041605                if (Type != MediaStreamType.Subtitle)
 606                {
 49607                    return false;
 608                }
 609
 1992610                if (string.IsNullOrEmpty(Codec) && !IsExternal)
 611                {
 30612                    return false;
 613                }
 614
 1962615                return IsTextFormat(Codec);
 616            }
 617        }
 618
 619        [JsonIgnore]
 620        public bool IsPgsSubtitleStream
 621        {
 622            get
 623            {
 60624                if (Type != MediaStreamType.Subtitle)
 625                {
 26626                    return false;
 627                }
 628
 34629                if (string.IsNullOrEmpty(Codec) && !IsExternal)
 630                {
 30631                    return false;
 632                }
 633
 4634                return IsPgsFormat(Codec);
 635            }
 636        }
 637
 638        /// <summary>
 639        /// Gets a value indicating whether this is a subtitle steam that is extractable by ffmpeg.
 640        /// All text-based and pgs subtitles can be extracted.
 641        /// </summary>
 642        /// <value><c>true</c> if this is a extractable subtitle steam otherwise, <c>false</c>.</value>
 643        [JsonIgnore]
 32644        public bool IsExtractableSubtitleStream => IsTextSubtitleStream || IsPgsSubtitleStream;
 645
 646        /// <summary>
 647        /// Gets or sets a value indicating whether [supports external stream].
 648        /// </summary>
 649        /// <value><c>true</c> if [supports external stream]; otherwise, <c>false</c>.</value>
 650        public bool SupportsExternalStream { get; set; }
 651
 652        /// <summary>
 653        /// Gets or sets the filename.
 654        /// </summary>
 655        /// <value>The filename.</value>
 656        public string Path { get; set; }
 657
 658        /// <summary>
 659        /// Gets or sets the pixel format.
 660        /// </summary>
 661        /// <value>The pixel format.</value>
 662        public string PixelFormat { get; set; }
 663
 664        /// <summary>
 665        /// Gets or sets the level.
 666        /// </summary>
 667        /// <value>The level.</value>
 668        public double? Level { get; set; }
 669
 670        /// <summary>
 671        /// Gets or sets whether this instance is anamorphic.
 672        /// </summary>
 673        /// <value><c>true</c> if this instance is anamorphic; otherwise, <c>false</c>.</value>
 674        public bool? IsAnamorphic { get; set; }
 675
 676        internal string GetResolutionText()
 677        {
 73678            if (!Width.HasValue || !Height.HasValue)
 679            {
 3680                return null;
 681            }
 682
 70683            return Width switch
 70684            {
 70685                // 256x144 (16:9 square pixel format)
 4686                <= 256 when Height <= 144 => IsInterlaced ? "144i" : "144p",
 70687                // 426x240 (16:9 square pixel format)
 4688                <= 426 when Height <= 240 => IsInterlaced ? "240i" : "240p",
 70689                // 640x360 (16:9 square pixel format)
 14690                <= 640 when Height <= 360 => IsInterlaced ? "360i" : "360p",
 70691                // 682x384 (16:9 square pixel format)
 3692                <= 682 when Height <= 384 => IsInterlaced ? "384i" : "384p",
 70693                // 720x404 (16:9 square pixel format)
 7694                <= 720 when Height <= 404 => IsInterlaced ? "404i" : "404p",
 70695                // 854x480 (16:9 square pixel format)
 10696                <= 854 when Height <= 480 => IsInterlaced ? "480i" : "480p",
 70697                // 960x544 (16:9 square pixel format)
 11698                <= 960 when Height <= 544 => IsInterlaced ? "540i" : "540p",
 70699                // 1024x576 (16:9 square pixel format)
 9700                <= 1024 when Height <= 576 => IsInterlaced ? "576i" : "576p",
 70701                // 1280x720
 26702                <= 1280 when Height <= 962 => IsInterlaced ? "720i" : "720p",
 70703                // 2560x1080 (FHD ultra wide 21:9) using 1440px width to accommodate WQHD
 46704                <= 2560 when Height <= 1440 => IsInterlaced ? "1080i" : "1080p",
 70705                // 4K
 12706                <= 4096 when Height <= 3072 => "4K",
 70707                // 8K
 6708                <= 8192 when Height <= 6144 => "8K",
 0709                _ => null
 70710            };
 711        }
 712
 713        public static bool IsTextFormat(string format)
 714        {
 3344715            string codec = format ?? string.Empty;
 716
 717            // microdvd and dvdsub/vobsub share the ".sub" file extension, but it's text-based.
 718
 3344719            return codec.Contains("microdvd", StringComparison.OrdinalIgnoreCase)
 3344720                   || (!codec.Contains("pgs", StringComparison.OrdinalIgnoreCase)
 3344721                       && !codec.Contains("dvdsub", StringComparison.OrdinalIgnoreCase)
 3344722                       && !codec.Contains("dvbsub", StringComparison.OrdinalIgnoreCase)
 3344723                       && !string.Equals(codec, "sup", StringComparison.OrdinalIgnoreCase)
 3344724                       && !string.Equals(codec, "sub", StringComparison.OrdinalIgnoreCase));
 725        }
 726
 727        public static bool IsPgsFormat(string format)
 728        {
 8729            string codec = format ?? string.Empty;
 730
 8731            return codec.Contains("pgs", StringComparison.OrdinalIgnoreCase)
 8732                   || string.Equals(codec, "sup", StringComparison.OrdinalIgnoreCase);
 733        }
 734
 735        public bool SupportsSubtitleConversionTo(string toCodec)
 736        {
 286737            if (!IsTextSubtitleStream)
 738            {
 0739                return false;
 740            }
 741
 286742            var fromCodec = Codec;
 743
 744            // Can't convert from this
 286745            if (string.Equals(fromCodec, "ass", StringComparison.OrdinalIgnoreCase))
 746            {
 0747                return false;
 748            }
 749
 286750            if (string.Equals(fromCodec, "ssa", StringComparison.OrdinalIgnoreCase))
 751            {
 0752                return false;
 753            }
 754
 755            // Can't convert to this
 286756            if (string.Equals(toCodec, "ass", StringComparison.OrdinalIgnoreCase))
 757            {
 0758                return false;
 759            }
 760
 286761            if (string.Equals(toCodec, "ssa", StringComparison.OrdinalIgnoreCase))
 762            {
 0763                return false;
 764            }
 765
 286766            return true;
 767        }
 768
 769        public (VideoRange VideoRange, VideoRangeType VideoRangeType) GetVideoColorRange()
 770        {
 938771            if (Type != MediaStreamType.Video)
 772            {
 64773                return (VideoRange.Unknown, VideoRangeType.Unknown);
 774            }
 775
 874776            var codecTag = CodecTag;
 874777            var dvProfile = DvProfile;
 874778            var rpuPresentFlag = RpuPresentFlag == 1;
 874779            var blPresentFlag = BlPresentFlag == 1;
 874780            var dvBlCompatId = DvBlSignalCompatibilityId;
 781
 874782            var isDoViProfile = dvProfile == 5 || dvProfile == 7 || dvProfile == 8 || dvProfile == 10;
 874783            var isDoViFlag = rpuPresentFlag && blPresentFlag && (dvBlCompatId == 0 || dvBlCompatId == 1 || dvBlCompatId 
 784
 874785            if ((isDoViProfile && isDoViFlag)
 874786                || string.Equals(codecTag, "dovi", StringComparison.OrdinalIgnoreCase)
 874787                || string.Equals(codecTag, "dvh1", StringComparison.OrdinalIgnoreCase)
 874788                || string.Equals(codecTag, "dvhe", StringComparison.OrdinalIgnoreCase)
 874789                || string.Equals(codecTag, "dav1", StringComparison.OrdinalIgnoreCase))
 790            {
 115791                return dvProfile switch
 115792                {
 64793                    5 => (VideoRange.HDR, VideoRangeType.DOVI),
 51794                    8 => dvBlCompatId switch
 51795                    {
 51796                        1 => (VideoRange.HDR, VideoRangeType.DOVIWithHDR10),
 0797                        4 => (VideoRange.HDR, VideoRangeType.DOVIWithHLG),
 0798                        2 => (VideoRange.SDR, VideoRangeType.DOVIWithSDR),
 51799                        // While not in Dolby Spec, Profile 8 CCid 6 media are possible to create, and since CCid 6 stem
 0800                        6 => (VideoRange.HDR, VideoRangeType.DOVIWithHDR10),
 51801                        // There is no other case to handle here as per Dolby Spec. Default case included for completene
 0802                        _ => (VideoRange.SDR, VideoRangeType.SDR)
 51803                    },
 0804                    7 => (VideoRange.HDR, VideoRangeType.HDR10),
 0805                    10 => dvBlCompatId switch
 0806                    {
 0807                        0 => (VideoRange.HDR, VideoRangeType.DOVI),
 0808                        1 => (VideoRange.HDR, VideoRangeType.DOVIWithHDR10),
 0809                        2 => (VideoRange.SDR, VideoRangeType.DOVIWithSDR),
 0810                        4 => (VideoRange.HDR, VideoRangeType.DOVIWithHLG),
 0811                        // While not in Dolby Spec, Profile 8 CCid 6 media are possible to create, and since CCid 6 stem
 0812                        6 => (VideoRange.HDR, VideoRangeType.DOVIWithHDR10),
 0813                        // There is no other case to handle here as per Dolby Spec. Default case included for completene
 0814                        _ => (VideoRange.SDR, VideoRangeType.SDR)
 0815                    },
 0816                    _ => (VideoRange.SDR, VideoRangeType.SDR)
 115817                };
 818            }
 819
 759820            var colorTransfer = ColorTransfer;
 821
 759822            if (string.Equals(colorTransfer, "smpte2084", StringComparison.OrdinalIgnoreCase))
 823            {
 198824                return (VideoRange.HDR, VideoRangeType.HDR10);
 825            }
 561826            else if (string.Equals(colorTransfer, "arib-std-b67", StringComparison.OrdinalIgnoreCase))
 827            {
 0828                return (VideoRange.HDR, VideoRangeType.HLG);
 829            }
 830
 561831            return (VideoRange.SDR, VideoRangeType.SDR);
 832        }
 833    }
 834}