ReportBuilder.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. using MediaBrowser.Controller.Entities;
  2. using MediaBrowser.Controller.Entities.Audio;
  3. using MediaBrowser.Controller.Entities.Movies;
  4. using MediaBrowser.Controller.Entities.TV;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Controller.Localization;
  7. using MediaBrowser.Model.Channels;
  8. using MediaBrowser.Model.Dto;
  9. using MediaBrowser.Model.Entities;
  10. using MediaBrowser.Model.Querying;
  11. using System;
  12. using System.Collections.Generic;
  13. using System.Linq;
  14. using System.Linq.Expressions;
  15. using System.Text;
  16. using System.Threading.Tasks;
  17. namespace MediaBrowser.Api.Reports
  18. {
  19. /// <summary> A report builder. </summary>
  20. /// <seealso cref="T:MediaBrowser.Api.Reports.ReportBuilderBase"/>
  21. public class ReportBuilder : ReportBuilderBase
  22. {
  23. #region [Constructors]
  24. /// <summary>
  25. /// Initializes a new instance of the MediaBrowser.Api.Reports.ReportBuilder class. </summary>
  26. /// <param name="libraryManager"> Manager for library. </param>
  27. public ReportBuilder(ILibraryManager libraryManager)
  28. : base(libraryManager)
  29. {
  30. }
  31. #endregion
  32. #region [Public Methods]
  33. /// <summary> Gets report result. </summary>
  34. /// <param name="items"> The items. </param>
  35. /// <param name="request"> The request. </param>
  36. /// <returns> The report result. </returns>
  37. public ReportResult GetResult(BaseItem[] items, IReportsQuery request)
  38. {
  39. ReportIncludeItemTypes reportRowType = ReportHelper.GetRowType(request.IncludeItemTypes);
  40. List<ReportOptions<BaseItem>> options = this.GetReportOptions<BaseItem>(request,
  41. () => this.GetDefaultHeaderMetadata(reportRowType),
  42. (hm) => this.GetOption(hm)).Where(x => x.Header.Visible == true).ToList();
  43. var headers = GetHeaders<BaseItem>(options);
  44. var rows = GetReportRows(items, options);
  45. ReportResult result = new ReportResult { Headers = headers };
  46. HeaderMetadata groupBy = ReportHelper.GetHeaderMetadataType(request.GroupBy);
  47. int i = headers.FindIndex(x => x.FieldName == groupBy);
  48. if (groupBy != HeaderMetadata.None && i >= 0)
  49. {
  50. var rowsGroup = rows.SelectMany(x => x.Columns[i].Name.Split(';'), (x, g) => new { Group = g.Trim(), Rows = x })
  51. .GroupBy(x => x.Group)
  52. .OrderBy(x => x.Key)
  53. .Select(x => new ReportGroup { Name = x.Key, Rows = x.Select(r => r.Rows).ToList() });
  54. result.Groups = rowsGroup.ToList();
  55. result.IsGrouped = true;
  56. }
  57. else
  58. {
  59. result.Rows = rows;
  60. result.IsGrouped = false;
  61. }
  62. return result;
  63. }
  64. #endregion
  65. #region [Protected Internal Methods]
  66. /// <summary> Gets the headers. </summary>
  67. /// <typeparam name="H"> Type of the header. </typeparam>
  68. /// <param name="request"> The request. </param>
  69. /// <returns> The headers. </returns>
  70. /// <seealso cref="M:MediaBrowser.Api.Reports.ReportBuilderBase.GetHeaders{H}(H)"/>
  71. protected internal override List<ReportHeader> GetHeaders<H>(H request)
  72. {
  73. ReportIncludeItemTypes reportRowType = ReportHelper.GetRowType(request.IncludeItemTypes);
  74. return this.GetHeaders<BaseItem>(request, () => this.GetDefaultHeaderMetadata(reportRowType), (hm) => this.GetOption(hm));
  75. }
  76. #endregion
  77. #region [Private Methods]
  78. /// <summary> Gets default report header metadata. </summary>
  79. /// <param name="reportIncludeItemTypes"> Type of the report row. </param>
  80. /// <returns> The default report header metadata. </returns>
  81. private List<HeaderMetadata> GetDefaultHeaderMetadata(ReportIncludeItemTypes reportIncludeItemTypes)
  82. {
  83. switch (reportIncludeItemTypes)
  84. {
  85. case ReportIncludeItemTypes.Season:
  86. return new List<HeaderMetadata>
  87. {
  88. HeaderMetadata.StatusImage,
  89. HeaderMetadata.Series,
  90. HeaderMetadata.Season,
  91. HeaderMetadata.SeasonNumber,
  92. HeaderMetadata.DateAdded,
  93. HeaderMetadata.Year,
  94. HeaderMetadata.Genres
  95. };
  96. case ReportIncludeItemTypes.Series:
  97. return new List<HeaderMetadata>
  98. {
  99. HeaderMetadata.StatusImage,
  100. HeaderMetadata.Name,
  101. HeaderMetadata.Network,
  102. HeaderMetadata.DateAdded,
  103. HeaderMetadata.Year,
  104. HeaderMetadata.Genres,
  105. HeaderMetadata.ParentalRating,
  106. HeaderMetadata.CommunityRating,
  107. HeaderMetadata.Runtime,
  108. HeaderMetadata.Trailers,
  109. HeaderMetadata.Specials
  110. };
  111. case ReportIncludeItemTypes.MusicAlbum:
  112. return new List<HeaderMetadata>
  113. {
  114. HeaderMetadata.StatusImage,
  115. HeaderMetadata.Name,
  116. HeaderMetadata.AlbumArtist,
  117. HeaderMetadata.DateAdded,
  118. HeaderMetadata.ReleaseDate,
  119. HeaderMetadata.Tracks,
  120. HeaderMetadata.Year,
  121. HeaderMetadata.Genres
  122. };
  123. case ReportIncludeItemTypes.MusicArtist:
  124. return new List<HeaderMetadata>
  125. {
  126. HeaderMetadata.StatusImage,
  127. HeaderMetadata.MusicArtist,
  128. HeaderMetadata.Countries,
  129. HeaderMetadata.DateAdded,
  130. HeaderMetadata.Year,
  131. HeaderMetadata.Genres
  132. };
  133. case ReportIncludeItemTypes.Game:
  134. return new List<HeaderMetadata>
  135. {
  136. HeaderMetadata.StatusImage,
  137. HeaderMetadata.Name,
  138. HeaderMetadata.GameSystem,
  139. HeaderMetadata.DateAdded,
  140. HeaderMetadata.ReleaseDate,
  141. HeaderMetadata.ParentalRating,
  142. HeaderMetadata.CommunityRating,
  143. HeaderMetadata.Players,
  144. HeaderMetadata.Year,
  145. HeaderMetadata.Genres,
  146. HeaderMetadata.Trailers
  147. };
  148. case ReportIncludeItemTypes.Movie:
  149. return new List<HeaderMetadata>
  150. {
  151. HeaderMetadata.StatusImage,
  152. HeaderMetadata.Name,
  153. HeaderMetadata.DateAdded,
  154. HeaderMetadata.ReleaseDate,
  155. HeaderMetadata.Year,
  156. HeaderMetadata.Genres,
  157. HeaderMetadata.ParentalRating,
  158. HeaderMetadata.CommunityRating,
  159. HeaderMetadata.Runtime,
  160. HeaderMetadata.Video,
  161. HeaderMetadata.Resolution,
  162. HeaderMetadata.Audio,
  163. HeaderMetadata.Subtitles,
  164. HeaderMetadata.Trailers,
  165. HeaderMetadata.Specials
  166. };
  167. case ReportIncludeItemTypes.Book:
  168. return new List<HeaderMetadata>
  169. {
  170. HeaderMetadata.StatusImage,
  171. HeaderMetadata.Name,
  172. HeaderMetadata.DateAdded,
  173. HeaderMetadata.ReleaseDate,
  174. HeaderMetadata.Year,
  175. HeaderMetadata.Genres,
  176. HeaderMetadata.ParentalRating,
  177. HeaderMetadata.CommunityRating
  178. };
  179. case ReportIncludeItemTypes.BoxSet:
  180. return new List<HeaderMetadata>
  181. {
  182. HeaderMetadata.StatusImage,
  183. HeaderMetadata.Name,
  184. HeaderMetadata.DateAdded,
  185. HeaderMetadata.ReleaseDate,
  186. HeaderMetadata.Year,
  187. HeaderMetadata.Genres,
  188. HeaderMetadata.ParentalRating,
  189. HeaderMetadata.CommunityRating,
  190. HeaderMetadata.Trailers
  191. };
  192. case ReportIncludeItemTypes.Audio:
  193. return new List<HeaderMetadata>
  194. {
  195. HeaderMetadata.StatusImage,
  196. HeaderMetadata.Name,
  197. HeaderMetadata.AudioAlbumArtist,
  198. HeaderMetadata.AudioAlbum,
  199. HeaderMetadata.Disc,
  200. HeaderMetadata.Track,
  201. HeaderMetadata.DateAdded,
  202. HeaderMetadata.ReleaseDate,
  203. HeaderMetadata.Year,
  204. HeaderMetadata.Genres,
  205. HeaderMetadata.ParentalRating,
  206. HeaderMetadata.CommunityRating,
  207. HeaderMetadata.Runtime,
  208. HeaderMetadata.Audio
  209. };
  210. case ReportIncludeItemTypes.Episode:
  211. return new List<HeaderMetadata>
  212. {
  213. HeaderMetadata.StatusImage,
  214. HeaderMetadata.Name,
  215. HeaderMetadata.EpisodeSeries,
  216. HeaderMetadata.Season,
  217. HeaderMetadata.DateAdded,
  218. HeaderMetadata.ReleaseDate,
  219. HeaderMetadata.Year,
  220. HeaderMetadata.Genres,
  221. HeaderMetadata.ParentalRating,
  222. HeaderMetadata.CommunityRating,
  223. HeaderMetadata.Runtime,
  224. HeaderMetadata.Video,
  225. HeaderMetadata.Resolution,
  226. HeaderMetadata.Audio,
  227. HeaderMetadata.Subtitles,
  228. HeaderMetadata.Trailers,
  229. HeaderMetadata.Specials
  230. };
  231. case ReportIncludeItemTypes.Video:
  232. case ReportIncludeItemTypes.MusicVideo:
  233. case ReportIncludeItemTypes.Trailer:
  234. case ReportIncludeItemTypes.BaseItem:
  235. default:
  236. return new List<HeaderMetadata>
  237. {
  238. HeaderMetadata.StatusImage,
  239. HeaderMetadata.Name,
  240. HeaderMetadata.DateAdded,
  241. HeaderMetadata.ReleaseDate,
  242. HeaderMetadata.Year,
  243. HeaderMetadata.Genres,
  244. HeaderMetadata.ParentalRating,
  245. HeaderMetadata.CommunityRating,
  246. HeaderMetadata.Runtime,
  247. HeaderMetadata.Video,
  248. HeaderMetadata.Resolution,
  249. HeaderMetadata.Audio,
  250. HeaderMetadata.Subtitles,
  251. HeaderMetadata.Trailers,
  252. HeaderMetadata.Specials
  253. };
  254. }
  255. }
  256. /// <summary> Gets report option. </summary>
  257. /// <param name="header"> The header. </param>
  258. /// <param name="sortField"> The sort field. </param>
  259. /// <returns> The report option. </returns>
  260. private ReportOptions<BaseItem> GetOption(HeaderMetadata header, string sortField = "")
  261. {
  262. HeaderMetadata internalHeader = header;
  263. ReportOptions<BaseItem> option = new ReportOptions<BaseItem>()
  264. {
  265. Header = new ReportHeader
  266. {
  267. HeaderFieldType = ReportFieldType.String,
  268. SortField = sortField,
  269. Type = "",
  270. ItemViewType = ItemViewType.None
  271. }
  272. };
  273. switch (header)
  274. {
  275. case HeaderMetadata.StatusImage:
  276. option.Header.ItemViewType = ItemViewType.StatusImage;
  277. internalHeader = HeaderMetadata.Status;
  278. option.Header.CanGroup = false;
  279. break;
  280. case HeaderMetadata.Name:
  281. option.Column = (i, r) => i.Name;
  282. option.Header.ItemViewType = ItemViewType.Detail;
  283. option.Header.SortField = "SortName";
  284. break;
  285. case HeaderMetadata.DateAdded:
  286. option.Column = (i, r) => i.DateCreated;
  287. option.Header.SortField = "DateCreated,SortName";
  288. option.Header.HeaderFieldType = ReportFieldType.DateTime;
  289. option.Header.Type = "";
  290. break;
  291. case HeaderMetadata.PremiereDate:
  292. case HeaderMetadata.ReleaseDate:
  293. option.Column = (i, r) => i.PremiereDate;
  294. option.Header.HeaderFieldType = ReportFieldType.DateTime;
  295. option.Header.SortField = "ProductionYear,PremiereDate,SortName";
  296. break;
  297. case HeaderMetadata.Runtime:
  298. option.Column = (i, r) => this.GetRuntimeDateTime(i.RunTimeTicks);
  299. option.Header.HeaderFieldType = ReportFieldType.Minutes;
  300. option.Header.SortField = "Runtime,SortName";
  301. break;
  302. case HeaderMetadata.PlayCount:
  303. option.Header.HeaderFieldType = ReportFieldType.Int;
  304. break;
  305. case HeaderMetadata.Season:
  306. option.Column = (i, r) => this.GetEpisode(i);
  307. option.Header.ItemViewType = ItemViewType.Detail;
  308. option.Header.SortField = "SortName";
  309. break;
  310. case HeaderMetadata.SeasonNumber:
  311. option.Column = (i, r) => this.GetObject<Season, string>(i, (x) => x.IndexNumber == null ? "" : x.IndexNumber.ToString());
  312. option.Header.SortField = "IndexNumber";
  313. option.Header.HeaderFieldType = ReportFieldType.Int;
  314. break;
  315. case HeaderMetadata.Series:
  316. option.Column = (i, r) => this.GetObject<IHasSeries, string>(i, (x) => x.SeriesName);
  317. option.Header.ItemViewType = ItemViewType.Detail;
  318. option.Header.SortField = "SeriesSortName,SortName";
  319. break;
  320. case HeaderMetadata.EpisodeSeries:
  321. option.Column = (i, r) => this.GetObject<IHasSeries, string>(i, (x) => x.SeriesName);
  322. option.Header.ItemViewType = ItemViewType.Detail;
  323. option.ItemID = (i) =>
  324. {
  325. Series series = this.GetObject<Episode, Series>(i, (x) => x.Series);
  326. if (series == null)
  327. return string.Empty;
  328. return series.Id;
  329. };
  330. option.Header.SortField = "SeriesSortName,SortName";
  331. internalHeader = HeaderMetadata.Series;
  332. break;
  333. case HeaderMetadata.EpisodeSeason:
  334. option.Column = (i, r) => this.GetObject<IHasSeries, string>(i, (x) => x.SeriesName);
  335. option.Header.ItemViewType = ItemViewType.Detail;
  336. option.ItemID = (i) =>
  337. {
  338. Season season = this.GetObject<Episode, Season>(i, (x) => x.Season);
  339. if (season == null)
  340. return string.Empty;
  341. return season.Id;
  342. };
  343. option.Header.SortField = "SortName";
  344. internalHeader = HeaderMetadata.Season;
  345. break;
  346. case HeaderMetadata.Network:
  347. option.Column = (i, r) => this.GetListAsString(i.Studios);
  348. option.ItemID = (i) => this.GetStudioID(i.Studios.FirstOrDefault());
  349. option.Header.ItemViewType = ItemViewType.ItemByNameDetails;
  350. option.Header.SortField = "Studio,SortName";
  351. break;
  352. case HeaderMetadata.Year:
  353. option.Column = (i, r) => this.GetSeriesProductionYear(i);
  354. option.Header.SortField = "ProductionYear,PremiereDate,SortName";
  355. break;
  356. case HeaderMetadata.ParentalRating:
  357. option.Column = (i, r) => i.OfficialRating;
  358. option.Header.SortField = "OfficialRating,SortName";
  359. break;
  360. case HeaderMetadata.CommunityRating:
  361. option.Column = (i, r) => i.CommunityRating;
  362. option.Header.SortField = "CommunityRating,SortName";
  363. break;
  364. case HeaderMetadata.Trailers:
  365. option.Column = (i, r) => this.GetBoolString(r.HasLocalTrailer);
  366. option.Header.ItemViewType = ItemViewType.TrailersImage;
  367. break;
  368. case HeaderMetadata.Specials:
  369. option.Column = (i, r) => this.GetBoolString(r.HasSpecials);
  370. option.Header.ItemViewType = ItemViewType.SpecialsImage;
  371. break;
  372. case HeaderMetadata.GameSystem:
  373. option.Column = (i, r) => this.GetObject<Game, string>(i, (x) => x.GameSystem);
  374. option.Header.SortField = "GameSystem,SortName";
  375. break;
  376. case HeaderMetadata.Players:
  377. option.Column = (i, r) => this.GetObject<Game, int?>(i, (x) => x.PlayersSupported);
  378. option.Header.SortField = "Players,GameSystem,SortName";
  379. break;
  380. case HeaderMetadata.AlbumArtist:
  381. option.Column = (i, r) => this.GetObject<MusicAlbum, string>(i, (x) => x.AlbumArtist);
  382. option.ItemID = (i) => this.GetPersonID(this.GetObject<MusicAlbum, string>(i, (x) => x.AlbumArtist));
  383. option.Header.ItemViewType = ItemViewType.Detail;
  384. option.Header.SortField = "AlbumArtist,Album,SortName";
  385. break;
  386. case HeaderMetadata.MusicArtist:
  387. option.Column = (i, r) => this.GetObject<MusicArtist, string>(i, (x) => x.GetLookupInfo().Name);
  388. option.Header.ItemViewType = ItemViewType.Detail;
  389. option.Header.SortField = "AlbumArtist,Album,SortName";
  390. internalHeader = HeaderMetadata.AlbumArtist;
  391. break;
  392. case HeaderMetadata.AudioAlbumArtist:
  393. option.Column = (i, r) => this.GetListAsString(this.GetObject<Audio, List<string>>(i, (x) => x.AlbumArtists));
  394. option.Header.SortField = "AlbumArtist,Album,SortName";
  395. internalHeader = HeaderMetadata.AlbumArtist;
  396. break;
  397. case HeaderMetadata.AudioAlbum:
  398. option.Column = (i, r) => this.GetObject<Audio, string>(i, (x) => x.Album);
  399. option.Header.SortField = "Album,SortName";
  400. internalHeader = HeaderMetadata.Album;
  401. break;
  402. case HeaderMetadata.Countries:
  403. option.Column = (i, r) => this.GetListAsString(this.GetObject<IHasProductionLocations, List<string>>(i, (x) => x.ProductionLocations));
  404. break;
  405. case HeaderMetadata.Disc:
  406. option.Column = (i, r) => i.ParentIndexNumber;
  407. break;
  408. case HeaderMetadata.Track:
  409. option.Column = (i, r) => i.IndexNumber;
  410. break;
  411. case HeaderMetadata.Tracks:
  412. option.Column = (i, r) => this.GetObject<MusicAlbum, List<Audio>>(i, (x) => x.Tracks.ToList(), new List<Audio>()).Count();
  413. break;
  414. case HeaderMetadata.Audio:
  415. option.Column = (i, r) => this.GetAudioStream(i);
  416. break;
  417. case HeaderMetadata.EmbeddedImage:
  418. break;
  419. case HeaderMetadata.Video:
  420. option.Column = (i, r) => this.GetVideoStream(i);
  421. break;
  422. case HeaderMetadata.Resolution:
  423. option.Column = (i, r) => this.GetVideoResolution(i);
  424. break;
  425. case HeaderMetadata.Subtitles:
  426. option.Column = (i, r) => this.GetBoolString(r.HasSubtitles);
  427. option.Header.ItemViewType = ItemViewType.SubtitleImage;
  428. break;
  429. case HeaderMetadata.Genres:
  430. option.Column = (i, r) => this.GetListAsString(i.Genres);
  431. break;
  432. }
  433. option.Header.Name = GetLocalizedHeader(internalHeader);
  434. option.Header.FieldName = header;
  435. return option;
  436. }
  437. /// <summary> Gets report rows. </summary>
  438. /// <param name="items"> The items. </param>
  439. /// <param name="options"> Options for controlling the operation. </param>
  440. /// <returns> The report rows. </returns>
  441. private List<ReportRow> GetReportRows(IEnumerable<BaseItem> items, List<ReportOptions<BaseItem>> options)
  442. {
  443. var rows = new List<ReportRow>();
  444. foreach (BaseItem item in items)
  445. {
  446. ReportRow rRow = GetRow(item);
  447. foreach (ReportOptions<BaseItem> option in options)
  448. {
  449. object itemColumn = option.Column != null ? option.Column(item, rRow) : "";
  450. object itemId = option.ItemID != null ? option.ItemID(item) : "";
  451. ReportItem rItem = new ReportItem
  452. {
  453. Name = ReportHelper.ConvertToString(itemColumn, option.Header.HeaderFieldType),
  454. Id = ReportHelper.ConvertToString(itemId, ReportFieldType.Object)
  455. };
  456. rRow.Columns.Add(rItem);
  457. }
  458. rows.Add(rRow);
  459. }
  460. return rows;
  461. }
  462. /// <summary> Gets a row. </summary>
  463. /// <param name="item"> The item. </param>
  464. /// <returns> The row. </returns>
  465. private ReportRow GetRow(BaseItem item)
  466. {
  467. var hasTrailers = item as IHasTrailers;
  468. var hasSpecialFeatures = item as IHasSpecialFeatures;
  469. var video = item as Video;
  470. ReportRow rRow = new ReportRow
  471. {
  472. Id = item.Id.ToString("N"),
  473. HasLockData = item.IsLocked,
  474. IsUnidentified = item.IsUnidentified,
  475. HasLocalTrailer = hasTrailers != null ? hasTrailers.GetTrailerIds().Count() > 0 : false,
  476. HasImageTagsPrimary = (item.ImageInfos != null && item.ImageInfos.Count(n => n.Type == ImageType.Primary) > 0),
  477. HasImageTagsBackdrop = (item.ImageInfos != null && item.ImageInfos.Count(n => n.Type == ImageType.Backdrop) > 0),
  478. HasImageTagsLogo = (item.ImageInfos != null && item.ImageInfos.Count(n => n.Type == ImageType.Logo) > 0),
  479. HasSpecials = hasSpecialFeatures != null ? hasSpecialFeatures.SpecialFeatureIds.Count > 0 : false,
  480. HasSubtitles = video != null ? video.HasSubtitles : false,
  481. RowType = ReportHelper.GetRowType(item.GetClientTypeName())
  482. };
  483. return rRow;
  484. }
  485. #endregion
  486. }
  487. }