Video.cs 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823
  1. using MediaBrowser.Controller.Library;
  2. using MediaBrowser.Controller.Persistence;
  3. using MediaBrowser.Controller.Providers;
  4. using MediaBrowser.Model.Dto;
  5. using MediaBrowser.Model.Entities;
  6. using MediaBrowser.Model.MediaInfo;
  7. using System;
  8. using System.Collections.Generic;
  9. using System.Globalization;
  10. using System.Linq;
  11. using System.Threading;
  12. using System.Threading.Tasks;
  13. using MediaBrowser.Common.Extensions;
  14. using MediaBrowser.Controller.Channels;
  15. using MediaBrowser.Controller.IO;
  16. using MediaBrowser.Model.IO;
  17. using MediaBrowser.Model.Serialization;
  18. using MediaBrowser.Model.Extensions;
  19. namespace MediaBrowser.Controller.Entities
  20. {
  21. /// <summary>
  22. /// Class Video
  23. /// </summary>
  24. public class Video : BaseItem,
  25. IHasAspectRatio,
  26. ISupportsPlaceHolders,
  27. IHasMediaSources
  28. {
  29. [IgnoreDataMember]
  30. public string PrimaryVersionId { get; set; }
  31. public string[] AdditionalParts { get; set; }
  32. public string[] LocalAlternateVersions { get; set; }
  33. public LinkedChild[] LinkedAlternateVersions { get; set; }
  34. [IgnoreDataMember]
  35. public override bool SupportsPlayedStatus
  36. {
  37. get
  38. {
  39. return true;
  40. }
  41. }
  42. [IgnoreDataMember]
  43. public override bool SupportsInheritedParentImages
  44. {
  45. get
  46. {
  47. return true;
  48. }
  49. }
  50. [IgnoreDataMember]
  51. public override bool SupportsPositionTicksResume
  52. {
  53. get
  54. {
  55. var extraType = ExtraType;
  56. if (extraType.HasValue)
  57. {
  58. if (extraType.Value == Model.Entities.ExtraType.Sample)
  59. {
  60. return false;
  61. }
  62. if (extraType.Value == Model.Entities.ExtraType.ThemeVideo)
  63. {
  64. return false;
  65. }
  66. if (extraType.Value == Model.Entities.ExtraType.Trailer)
  67. {
  68. return false;
  69. }
  70. }
  71. return true;
  72. }
  73. }
  74. public override string CreatePresentationUniqueKey()
  75. {
  76. if (!string.IsNullOrWhiteSpace(PrimaryVersionId))
  77. {
  78. return PrimaryVersionId;
  79. }
  80. return base.CreatePresentationUniqueKey();
  81. }
  82. [IgnoreDataMember]
  83. public override bool EnableRefreshOnDateModifiedChange
  84. {
  85. get
  86. {
  87. return VideoType == VideoType.VideoFile || VideoType == VideoType.Iso;
  88. }
  89. }
  90. [IgnoreDataMember]
  91. public override bool SupportsThemeMedia
  92. {
  93. get { return true; }
  94. }
  95. /// <summary>
  96. /// Gets or sets the timestamp.
  97. /// </summary>
  98. /// <value>The timestamp.</value>
  99. public TransportStreamTimestamp? Timestamp { get; set; }
  100. /// <summary>
  101. /// Gets or sets the subtitle paths.
  102. /// </summary>
  103. /// <value>The subtitle paths.</value>
  104. public string[] SubtitleFiles { get; set; }
  105. /// <summary>
  106. /// Gets or sets a value indicating whether this instance has subtitles.
  107. /// </summary>
  108. /// <value><c>true</c> if this instance has subtitles; otherwise, <c>false</c>.</value>
  109. public bool HasSubtitles { get; set; }
  110. public bool IsPlaceHolder { get; set; }
  111. public bool IsShortcut { get; set; }
  112. public string ShortcutPath { get; set; }
  113. /// <summary>
  114. /// Gets or sets the default index of the video stream.
  115. /// </summary>
  116. /// <value>The default index of the video stream.</value>
  117. public int? DefaultVideoStreamIndex { get; set; }
  118. /// <summary>
  119. /// Gets or sets the type of the video.
  120. /// </summary>
  121. /// <value>The type of the video.</value>
  122. public VideoType VideoType { get; set; }
  123. /// <summary>
  124. /// Gets or sets the type of the iso.
  125. /// </summary>
  126. /// <value>The type of the iso.</value>
  127. public IsoType? IsoType { get; set; }
  128. /// <summary>
  129. /// Gets or sets the video3 D format.
  130. /// </summary>
  131. /// <value>The video3 D format.</value>
  132. public Video3DFormat? Video3DFormat { get; set; }
  133. public string[] GetPlayableStreamFileNames()
  134. {
  135. var videoType = VideoType;
  136. if (videoType == VideoType.Iso && IsoType == Model.Entities.IsoType.BluRay)
  137. {
  138. videoType = VideoType.BluRay;
  139. }
  140. else if (videoType == VideoType.Iso && IsoType == Model.Entities.IsoType.Dvd)
  141. {
  142. videoType = VideoType.Dvd;
  143. }
  144. else
  145. {
  146. return new string[] { };
  147. }
  148. return MediaEncoder.GetPlayableStreamFileNames(Path, videoType);
  149. }
  150. /// <summary>
  151. /// Gets or sets the aspect ratio.
  152. /// </summary>
  153. /// <value>The aspect ratio.</value>
  154. public string AspectRatio { get; set; }
  155. public Video()
  156. {
  157. AdditionalParts = EmptyStringArray;
  158. LocalAlternateVersions = EmptyStringArray;
  159. SubtitleFiles = EmptyStringArray;
  160. LinkedAlternateVersions = EmptyLinkedChildArray;
  161. }
  162. public override bool CanDownload()
  163. {
  164. if (VideoType == VideoType.Dvd || VideoType == VideoType.BluRay)
  165. {
  166. return false;
  167. }
  168. var locationType = LocationType;
  169. return locationType != LocationType.Remote &&
  170. locationType != LocationType.Virtual;
  171. }
  172. [IgnoreDataMember]
  173. public override bool SupportsAddingToPlaylist
  174. {
  175. get { return true; }
  176. }
  177. [IgnoreDataMember]
  178. public int MediaSourceCount
  179. {
  180. get
  181. {
  182. if (!string.IsNullOrWhiteSpace(PrimaryVersionId))
  183. {
  184. var item = LibraryManager.GetItemById(PrimaryVersionId) as Video;
  185. if (item != null)
  186. {
  187. return item.MediaSourceCount;
  188. }
  189. }
  190. return LinkedAlternateVersions.Length + LocalAlternateVersions.Length + 1;
  191. }
  192. }
  193. [IgnoreDataMember]
  194. public bool IsStacked
  195. {
  196. get { return AdditionalParts.Length > 0; }
  197. }
  198. [IgnoreDataMember]
  199. public bool HasLocalAlternateVersions
  200. {
  201. get { return LocalAlternateVersions.Length > 0; }
  202. }
  203. public IEnumerable<Guid> GetAdditionalPartIds()
  204. {
  205. return AdditionalParts.Select(i => LibraryManager.GetNewItemId(i, typeof(Video)));
  206. }
  207. public IEnumerable<Guid> GetLocalAlternateVersionIds()
  208. {
  209. return LocalAlternateVersions.Select(i => LibraryManager.GetNewItemId(i, typeof(Video)));
  210. }
  211. [IgnoreDataMember]
  212. public override SourceType SourceType
  213. {
  214. get
  215. {
  216. if (IsActiveRecording())
  217. {
  218. return SourceType.LiveTV;
  219. }
  220. return base.SourceType;
  221. }
  222. }
  223. protected bool IsActiveRecording()
  224. {
  225. return LiveTvManager.GetActiveRecordingInfo(Path) != null;
  226. }
  227. public override bool CanDelete()
  228. {
  229. if (IsActiveRecording())
  230. {
  231. return false;
  232. }
  233. return base.CanDelete();
  234. }
  235. [IgnoreDataMember]
  236. public bool IsCompleteMedia
  237. {
  238. get { return !IsActiveRecording(); }
  239. }
  240. [IgnoreDataMember]
  241. protected virtual bool EnableDefaultVideoUserDataKeys
  242. {
  243. get
  244. {
  245. return true;
  246. }
  247. }
  248. public override List<string> GetUserDataKeys()
  249. {
  250. var list = base.GetUserDataKeys();
  251. if (EnableDefaultVideoUserDataKeys)
  252. {
  253. if (ExtraType.HasValue)
  254. {
  255. var key = this.GetProviderId(MetadataProviders.Tmdb);
  256. if (!string.IsNullOrWhiteSpace(key))
  257. {
  258. list.Insert(0, GetUserDataKey(key));
  259. }
  260. key = this.GetProviderId(MetadataProviders.Imdb);
  261. if (!string.IsNullOrWhiteSpace(key))
  262. {
  263. list.Insert(0, GetUserDataKey(key));
  264. }
  265. }
  266. else
  267. {
  268. var key = this.GetProviderId(MetadataProviders.Imdb);
  269. if (!string.IsNullOrWhiteSpace(key))
  270. {
  271. list.Insert(0, key);
  272. }
  273. key = this.GetProviderId(MetadataProviders.Tmdb);
  274. if (!string.IsNullOrWhiteSpace(key))
  275. {
  276. list.Insert(0, key);
  277. }
  278. }
  279. }
  280. return list;
  281. }
  282. private string GetUserDataKey(string providerId)
  283. {
  284. var key = providerId + "-" + ExtraType.ToString().ToLower();
  285. // Make sure different trailers have their own data.
  286. if (RunTimeTicks.HasValue)
  287. {
  288. key += "-" + RunTimeTicks.Value.ToString(CultureInfo.InvariantCulture);
  289. }
  290. return key;
  291. }
  292. public IEnumerable<Video> GetLinkedAlternateVersions()
  293. {
  294. return LinkedAlternateVersions
  295. .Select(GetLinkedChild)
  296. .Where(i => i != null)
  297. .OfType<Video>()
  298. .OrderBy(i => i.SortName);
  299. }
  300. /// <summary>
  301. /// Gets the additional parts.
  302. /// </summary>
  303. /// <returns>IEnumerable{Video}.</returns>
  304. public IEnumerable<Video> GetAdditionalParts()
  305. {
  306. return GetAdditionalPartIds()
  307. .Select(i => LibraryManager.GetItemById(i))
  308. .Where(i => i != null)
  309. .OfType<Video>()
  310. .OrderBy(i => i.SortName);
  311. }
  312. [IgnoreDataMember]
  313. public override string ContainingFolderPath
  314. {
  315. get
  316. {
  317. if (IsStacked)
  318. {
  319. return FileSystem.GetDirectoryName(Path);
  320. }
  321. if (!IsPlaceHolder)
  322. {
  323. if (VideoType == VideoType.BluRay || VideoType == VideoType.Dvd)
  324. {
  325. return Path;
  326. }
  327. }
  328. return base.ContainingFolderPath;
  329. }
  330. }
  331. [IgnoreDataMember]
  332. public override string FileNameWithoutExtension
  333. {
  334. get
  335. {
  336. if (LocationType == LocationType.FileSystem)
  337. {
  338. if (VideoType == VideoType.BluRay || VideoType == VideoType.Dvd)
  339. {
  340. return System.IO.Path.GetFileName(Path);
  341. }
  342. return System.IO.Path.GetFileNameWithoutExtension(Path);
  343. }
  344. return null;
  345. }
  346. }
  347. internal override bool IsValidFromResolver(BaseItem newItem)
  348. {
  349. var current = this;
  350. var newAsVideo = newItem as Video;
  351. if (newAsVideo != null)
  352. {
  353. if (!current.AdditionalParts.SequenceEqual(newAsVideo.AdditionalParts, StringComparer.OrdinalIgnoreCase))
  354. {
  355. return false;
  356. }
  357. if (!current.LocalAlternateVersions.SequenceEqual(newAsVideo.LocalAlternateVersions, StringComparer.OrdinalIgnoreCase))
  358. {
  359. return false;
  360. }
  361. if (newAsVideo.VideoType != VideoType)
  362. {
  363. return false;
  364. }
  365. }
  366. return base.IsValidFromResolver(newItem);
  367. }
  368. public static string[] QueryPlayableStreamFiles(string rootPath, VideoType videoType)
  369. {
  370. if (videoType == VideoType.Dvd)
  371. {
  372. return FileSystem.GetFiles(rootPath, new[] { ".vob" }, false, true)
  373. .OrderByDescending(i => i.Length)
  374. .ThenBy(i => i.FullName)
  375. .Take(1)
  376. .Select(i => i.FullName)
  377. .ToArray();
  378. }
  379. if (videoType == VideoType.BluRay)
  380. {
  381. return FileSystem.GetFiles(rootPath, new[] { ".m2ts" }, false, true)
  382. .OrderByDescending(i => i.Length)
  383. .ThenBy(i => i.FullName)
  384. .Take(1)
  385. .Select(i => i.FullName)
  386. .ToArray();
  387. }
  388. return new string[] { };
  389. }
  390. /// <summary>
  391. /// Gets a value indicating whether [is3 D].
  392. /// </summary>
  393. /// <value><c>true</c> if [is3 D]; otherwise, <c>false</c>.</value>
  394. [IgnoreDataMember]
  395. public bool Is3D
  396. {
  397. get { return Video3DFormat.HasValue; }
  398. }
  399. /// <summary>
  400. /// Gets the type of the media.
  401. /// </summary>
  402. /// <value>The type of the media.</value>
  403. [IgnoreDataMember]
  404. public override string MediaType
  405. {
  406. get
  407. {
  408. return Model.Entities.MediaType.Video;
  409. }
  410. }
  411. protected override async Task<bool> RefreshedOwnedItems(MetadataRefreshOptions options, List<FileSystemMetadata> fileSystemChildren, CancellationToken cancellationToken)
  412. {
  413. var hasChanges = await base.RefreshedOwnedItems(options, fileSystemChildren, cancellationToken).ConfigureAwait(false);
  414. if (IsStacked)
  415. {
  416. var tasks = AdditionalParts
  417. .Select(i => RefreshMetadataForOwnedVideo(options, true, i, cancellationToken));
  418. await Task.WhenAll(tasks).ConfigureAwait(false);
  419. }
  420. // Must have a parent to have additional parts or alternate versions
  421. // In other words, it must be part of the Parent/Child tree
  422. // The additional parts won't have additional parts themselves
  423. if (LocationType == LocationType.FileSystem && GetParent() != null)
  424. {
  425. if (!IsStacked)
  426. {
  427. RefreshLinkedAlternateVersions();
  428. var tasks = LocalAlternateVersions
  429. .Select(i => RefreshMetadataForOwnedVideo(options, false, i, cancellationToken));
  430. await Task.WhenAll(tasks).ConfigureAwait(false);
  431. }
  432. }
  433. return hasChanges;
  434. }
  435. private void RefreshLinkedAlternateVersions()
  436. {
  437. foreach (var child in LinkedAlternateVersions)
  438. {
  439. // Reset the cached value
  440. if (child.ItemId.HasValue && child.ItemId.Value == Guid.Empty)
  441. {
  442. child.ItemId = null;
  443. }
  444. }
  445. }
  446. public override async Task UpdateToRepository(ItemUpdateType updateReason, CancellationToken cancellationToken)
  447. {
  448. await base.UpdateToRepository(updateReason, cancellationToken).ConfigureAwait(false);
  449. var localAlternates = GetLocalAlternateVersionIds()
  450. .Select(i => LibraryManager.GetItemById(i))
  451. .Where(i => i != null);
  452. foreach (var item in localAlternates)
  453. {
  454. item.ImageInfos = ImageInfos;
  455. item.Overview = Overview;
  456. item.ProductionYear = ProductionYear;
  457. item.PremiereDate = PremiereDate;
  458. item.CommunityRating = CommunityRating;
  459. item.OfficialRating = OfficialRating;
  460. item.Genres = Genres;
  461. item.ProviderIds = ProviderIds;
  462. await item.UpdateToRepository(ItemUpdateType.MetadataDownload, cancellationToken).ConfigureAwait(false);
  463. }
  464. }
  465. public override IEnumerable<FileSystemMetadata> GetDeletePaths()
  466. {
  467. if (!IsInMixedFolder)
  468. {
  469. return new[] {
  470. new FileSystemMetadata
  471. {
  472. FullName = ContainingFolderPath,
  473. IsDirectory = true
  474. }
  475. };
  476. }
  477. return base.GetDeletePaths();
  478. }
  479. public List<MediaStream> GetMediaStreams()
  480. {
  481. return MediaSourceManager.GetMediaStreams(new MediaStreamQuery
  482. {
  483. ItemId = Id
  484. });
  485. }
  486. public virtual MediaStream GetDefaultVideoStream()
  487. {
  488. if (!DefaultVideoStreamIndex.HasValue)
  489. {
  490. return null;
  491. }
  492. return MediaSourceManager.GetMediaStreams(new MediaStreamQuery
  493. {
  494. ItemId = Id,
  495. Index = DefaultVideoStreamIndex.Value
  496. }).FirstOrDefault();
  497. }
  498. private List<Tuple<Video, MediaSourceType>> GetAllVideosForMediaSources()
  499. {
  500. var list = new List<Tuple<Video, MediaSourceType>>();
  501. list.Add(new Tuple<Video, MediaSourceType>(this, MediaSourceType.Default));
  502. list.AddRange(GetLinkedAlternateVersions().Select(i => new Tuple<Video, MediaSourceType>(i, MediaSourceType.Grouping)));
  503. if (!string.IsNullOrWhiteSpace(PrimaryVersionId))
  504. {
  505. var primary = LibraryManager.GetItemById(PrimaryVersionId) as Video;
  506. if (primary != null)
  507. {
  508. var existingIds = list.Select(i => i.Item1.Id).ToList();
  509. list.Add(new Tuple<Video, MediaSourceType>(primary, MediaSourceType.Grouping));
  510. list.AddRange(primary.GetLinkedAlternateVersions().Where(i => !existingIds.Contains(i.Id)).Select(i => new Tuple<Video, MediaSourceType>(i, MediaSourceType.Grouping)));
  511. }
  512. }
  513. var localAlternates = list
  514. .SelectMany(i => i.Item1.GetLocalAlternateVersionIds())
  515. .Select(LibraryManager.GetItemById)
  516. .Where(i => i != null)
  517. .OfType<Video>()
  518. .ToList();
  519. list.AddRange(localAlternates.Select(i => new Tuple<Video, MediaSourceType>(i, MediaSourceType.Default)));
  520. return list;
  521. }
  522. public virtual List<MediaSourceInfo> GetMediaSources(bool enablePathSubstitution)
  523. {
  524. if (SourceType == SourceType.Channel)
  525. {
  526. var sources = ChannelManager.GetStaticMediaSources(this, CancellationToken.None)
  527. .ToList();
  528. if (sources.Count > 0)
  529. {
  530. return sources;
  531. }
  532. return new List<MediaSourceInfo>
  533. {
  534. GetVersionInfo(enablePathSubstitution, this, MediaSourceType.Placeholder)
  535. };
  536. }
  537. var list = GetAllVideosForMediaSources();
  538. var result = list.Select(i => GetVersionInfo(enablePathSubstitution, i.Item1, i.Item2)).ToList();
  539. if (IsActiveRecording())
  540. {
  541. foreach (var mediaSource in result)
  542. {
  543. mediaSource.Type = MediaSourceType.Placeholder;
  544. }
  545. }
  546. return result.OrderBy(i =>
  547. {
  548. if (i.VideoType == VideoType.VideoFile)
  549. {
  550. return 0;
  551. }
  552. return 1;
  553. }).ThenBy(i => i.Video3DFormat.HasValue ? 1 : 0)
  554. .ThenByDescending(i =>
  555. {
  556. var stream = i.VideoStream;
  557. return stream == null || stream.Width == null ? 0 : stream.Width.Value;
  558. })
  559. .ToList();
  560. }
  561. private static MediaSourceInfo GetVersionInfo(bool enablePathSubstitution, Video media, MediaSourceType type)
  562. {
  563. if (media == null)
  564. {
  565. throw new ArgumentNullException("media");
  566. }
  567. var mediaStreams = MediaSourceManager.GetMediaStreams(media.Id);
  568. var locationType = media.LocationType;
  569. var info = new MediaSourceInfo
  570. {
  571. Id = media.Id.ToString("N"),
  572. IsoType = media.IsoType,
  573. Protocol = locationType == LocationType.Remote ? MediaProtocol.Http : MediaProtocol.File,
  574. MediaStreams = mediaStreams,
  575. Name = GetMediaSourceName(media, mediaStreams),
  576. Path = enablePathSubstitution ? GetMappedPath(media, media.Path, locationType) : media.Path,
  577. RunTimeTicks = media.RunTimeTicks,
  578. Video3DFormat = media.Video3DFormat,
  579. VideoType = media.VideoType,
  580. Container = media.Container,
  581. Size = media.Size,
  582. Timestamp = media.Timestamp,
  583. Type = type,
  584. SupportsDirectStream = media.VideoType == VideoType.VideoFile,
  585. IsRemote = media.IsShortcut
  586. };
  587. if (info.Protocol == MediaProtocol.File)
  588. {
  589. info.ETag = media.DateModified.Ticks.ToString(CultureInfo.InvariantCulture).GetMD5().ToString("N");
  590. }
  591. if (media.IsShortcut)
  592. {
  593. info.Path = media.ShortcutPath;
  594. if (!string.IsNullOrWhiteSpace(info.Path))
  595. {
  596. if (info.Path.StartsWith("Http", StringComparison.OrdinalIgnoreCase))
  597. {
  598. info.Protocol = MediaProtocol.Http;
  599. info.SupportsDirectStream = false;
  600. }
  601. else if (info.Path.StartsWith("Rtmp", StringComparison.OrdinalIgnoreCase))
  602. {
  603. info.Protocol = MediaProtocol.Rtmp;
  604. info.SupportsDirectStream = false;
  605. }
  606. else if (info.Path.StartsWith("Rtsp", StringComparison.OrdinalIgnoreCase))
  607. {
  608. info.Protocol = MediaProtocol.Rtsp;
  609. info.SupportsDirectStream = false;
  610. }
  611. else
  612. {
  613. info.Protocol = MediaProtocol.File;
  614. }
  615. }
  616. }
  617. if (string.IsNullOrEmpty(info.Container))
  618. {
  619. if (media.VideoType == VideoType.VideoFile || media.VideoType == VideoType.Iso)
  620. {
  621. if (!string.IsNullOrWhiteSpace(media.Path) && locationType != LocationType.Remote && locationType != LocationType.Virtual)
  622. {
  623. info.Container = System.IO.Path.GetExtension(media.Path).TrimStart('.');
  624. }
  625. }
  626. }
  627. info.Bitrate = media.TotalBitrate;
  628. info.InferTotalBitrate();
  629. return info;
  630. }
  631. private static string GetMediaSourceName(Video video, List<MediaStream> mediaStreams)
  632. {
  633. var terms = new List<string>();
  634. var videoStream = mediaStreams.FirstOrDefault(i => i.Type == MediaStreamType.Video);
  635. var audioStream = mediaStreams.FirstOrDefault(i => i.Type == MediaStreamType.Audio);
  636. if (video.Video3DFormat.HasValue)
  637. {
  638. terms.Add("3D");
  639. }
  640. if (video.VideoType == VideoType.BluRay)
  641. {
  642. terms.Add("Bluray");
  643. }
  644. else if (video.VideoType == VideoType.Dvd)
  645. {
  646. terms.Add("DVD");
  647. }
  648. else if (video.VideoType == VideoType.Iso)
  649. {
  650. if (video.IsoType.HasValue)
  651. {
  652. if (video.IsoType.Value == Model.Entities.IsoType.BluRay)
  653. {
  654. terms.Add("Bluray");
  655. }
  656. else if (video.IsoType.Value == Model.Entities.IsoType.Dvd)
  657. {
  658. terms.Add("DVD");
  659. }
  660. }
  661. else
  662. {
  663. terms.Add("ISO");
  664. }
  665. }
  666. if (videoStream != null)
  667. {
  668. if (videoStream.Width.HasValue)
  669. {
  670. if (videoStream.Width.Value >= 3800)
  671. {
  672. terms.Add("4K");
  673. }
  674. else if (videoStream.Width.Value >= 1900)
  675. {
  676. terms.Add("1080P");
  677. }
  678. else if (videoStream.Width.Value >= 1270)
  679. {
  680. terms.Add("720P");
  681. }
  682. else if (videoStream.Width.Value >= 700)
  683. {
  684. terms.Add("480P");
  685. }
  686. else
  687. {
  688. terms.Add("SD");
  689. }
  690. }
  691. }
  692. if (videoStream != null && !string.IsNullOrWhiteSpace(videoStream.Codec))
  693. {
  694. terms.Add(videoStream.Codec.ToUpper());
  695. }
  696. if (audioStream != null)
  697. {
  698. var audioCodec = string.Equals(audioStream.Codec, "dca", StringComparison.OrdinalIgnoreCase)
  699. ? audioStream.Profile
  700. : audioStream.Codec;
  701. if (!string.IsNullOrEmpty(audioCodec))
  702. {
  703. terms.Add(audioCodec.ToUpper());
  704. }
  705. }
  706. return string.Join("/", terms.ToArray(terms.Count));
  707. }
  708. }
  709. }