diff --git a/frontend/src/AddMovie/AddNewMovie/AddNewMovieSearchResult.js b/frontend/src/AddMovie/AddNewMovie/AddNewMovieSearchResult.js index 45f4e529b6..84ead94346 100644 --- a/frontend/src/AddMovie/AddNewMovie/AddNewMovieSearchResult.js +++ b/frontend/src/AddMovie/AddNewMovie/AddNewMovieSearchResult.js @@ -183,7 +183,7 @@ class AddNewMovieSearchResult extends Component {
diff --git a/frontend/src/Components/HeartRating.css b/frontend/src/Components/HeartRating.css index 705adfcaed..0dd8f64ec4 100644 --- a/frontend/src/Components/HeartRating.css +++ b/frontend/src/Components/HeartRating.css @@ -1,4 +1,5 @@ -.heart { +.image { + align-content: center; margin-right: 5px; - color: $themeRed; + vertical-align: -0.125em; } diff --git a/frontend/src/Components/HeartRating.js b/frontend/src/Components/HeartRating.js index eba46cb1fb..07dbc0930a 100644 --- a/frontend/src/Components/HeartRating.js +++ b/frontend/src/Components/HeartRating.js @@ -1,30 +1,65 @@ import PropTypes from 'prop-types'; -import React from 'react'; -import Icon from 'Components/Icon'; -import { icons } from 'Helpers/Props'; +import React, { PureComponent } from 'react'; import styles from './HeartRating.css'; -function HeartRating({ rating, iconSize, hideHeart }) { - return ( - - { - !hideHeart && - - } +class HeartRating extends PureComponent { - {rating * 10}% - - ); + // + // Render + + render() { + + const { + ratings, + hideIcon, + iconSize + } = this.props; + + const imdbImage = ''; + const tmdbImage = ''; + + let rating = ratings.imdb; + + if (!rating) { + rating = ratings.tmdb; + } + + let ratingString = '0%'; + + if (rating) { + ratingString = ratings.imdb ? `${rating.value}` : `${rating.value * 10}%`; + } + + return ( + + { + !hideIcon && !ratings.imdb ? + : + + } + + {ratingString} + + ); + } } HeartRating.propTypes = { - rating: PropTypes.number.isRequired, + ratings: PropTypes.object.isRequired, iconSize: PropTypes.number.isRequired, - hideHeart: PropTypes.bool + hideIcon: PropTypes.bool }; HeartRating.defaultProps = { diff --git a/frontend/src/Components/ImdbRating.css b/frontend/src/Components/ImdbRating.css new file mode 100644 index 0000000000..78d489fd05 --- /dev/null +++ b/frontend/src/Components/ImdbRating.css @@ -0,0 +1,5 @@ +.imdb { + align-content: center; + margin-right: 5px; + vertical-align: -0.125em; +} diff --git a/frontend/src/Components/ImdbRating.js b/frontend/src/Components/ImdbRating.js new file mode 100644 index 0000000000..054233a578 --- /dev/null +++ b/frontend/src/Components/ImdbRating.js @@ -0,0 +1,57 @@ +import PropTypes from 'prop-types'; +import React, { PureComponent } from 'react'; +import styles from './ImdbRating.css'; + +class ImdbRating extends PureComponent { + + // + // Render + + render() { + + const { + ratings, + hideIcon, + iconSize + } = this.props; + + const imdbImage = ''; + + const rating = ratings.imdb; + + let ratingString = '0.0'; + + if (rating) { + ratingString = `${rating.value}`; + } + + return ( + + { + !hideIcon && + + } + + {ratingString} + + ); + } +} + +ImdbRating.propTypes = { + ratings: PropTypes.object.isRequired, + iconSize: PropTypes.number.isRequired, + hideIcon: PropTypes.bool +}; + +ImdbRating.defaultProps = { + iconSize: 14 +}; + +export default ImdbRating; diff --git a/frontend/src/Components/RottenTomatoRating.css b/frontend/src/Components/RottenTomatoRating.css new file mode 100644 index 0000000000..0dd8f64ec4 --- /dev/null +++ b/frontend/src/Components/RottenTomatoRating.css @@ -0,0 +1,5 @@ +.image { + align-content: center; + margin-right: 5px; + vertical-align: -0.125em; +} diff --git a/frontend/src/Components/RottenTomatoRating.js b/frontend/src/Components/RottenTomatoRating.js new file mode 100644 index 0000000000..fe4ed402e5 --- /dev/null +++ b/frontend/src/Components/RottenTomatoRating.js @@ -0,0 +1,58 @@ +import PropTypes from 'prop-types'; +import React, { PureComponent } from 'react'; +import styles from './RottenTomatoRating.css'; + +class RottenTomatoRating extends PureComponent { + + // + // Render + + render() { + + const { + ratings, + hideIcon, + iconSize + } = this.props; + + const rtRotten = ''; + const rtFresh = ''; + + const rating = ratings.rottenTomatoes; + + let ratingString = '0%'; + + if (rating) { + ratingString = `${rating.value}%`; + } + + return ( + + { + !hideIcon && + 50 ? rtFresh : rtRotten} + style={{ + height: `${iconSize}px` + }} + /> + } + + {ratingString} + + ); + } +} + +RottenTomatoRating.propTypes = { + ratings: PropTypes.object.isRequired, + iconSize: PropTypes.number.isRequired, + hideIcon: PropTypes.bool +}; + +RottenTomatoRating.defaultProps = { + iconSize: 14 +}; + +export default RottenTomatoRating; diff --git a/frontend/src/Components/TmdbRating.css b/frontend/src/Components/TmdbRating.css new file mode 100644 index 0000000000..1ac395144b --- /dev/null +++ b/frontend/src/Components/TmdbRating.css @@ -0,0 +1,5 @@ +.tmdb { + align-content: center; + margin-right: 5px; + vertical-align: -0.125em; +} diff --git a/frontend/src/Components/TmdbRating.js b/frontend/src/Components/TmdbRating.js new file mode 100644 index 0000000000..cc53bde456 --- /dev/null +++ b/frontend/src/Components/TmdbRating.js @@ -0,0 +1,57 @@ +import PropTypes from 'prop-types'; +import React, { PureComponent } from 'react'; +import styles from './TmdbRating.css'; + +class TmdbRating extends PureComponent { + + // + // Render + + render() { + + const { + ratings, + hideIcon, + iconSize + } = this.props; + + const tmdbImage = ''; + + const rating = ratings.tmdb; + + let ratingString = '0%'; + + if (rating) { + ratingString = `${rating.value * 10}%`; + } + + return ( + + { + !hideIcon && + + } + + {ratingString} + + ); + } +} + +TmdbRating.propTypes = { + ratings: PropTypes.object.isRequired, + iconSize: PropTypes.number.isRequired, + hideIcon: PropTypes.bool +}; + +TmdbRating.defaultProps = { + iconSize: 14 +}; + +export default TmdbRating; diff --git a/frontend/src/DiscoverMovie/Overview/DiscoverMovieOverviewInfo.js b/frontend/src/DiscoverMovie/Overview/DiscoverMovieOverviewInfo.js index 5352dd464a..f399272a63 100644 --- a/frontend/src/DiscoverMovie/Overview/DiscoverMovieOverviewInfo.js +++ b/frontend/src/DiscoverMovie/Overview/DiscoverMovieOverviewInfo.js @@ -73,7 +73,7 @@ function getInfoRowProps(row, props) { return { title: translate('Ratings'), iconName: icons.HEART, - label: `${props.ratings.value * 10}%` + label: `${props.ratings.tmdb.value * 10}%` }; } diff --git a/frontend/src/DiscoverMovie/Posters/DiscoverMoviePosterInfo.js b/frontend/src/DiscoverMovie/Posters/DiscoverMoviePosterInfo.js index 334aa71eaa..a1b2b2b87e 100644 --- a/frontend/src/DiscoverMovie/Posters/DiscoverMoviePosterInfo.js +++ b/frontend/src/DiscoverMovie/Posters/DiscoverMoviePosterInfo.js @@ -112,7 +112,7 @@ function DiscoverMoviePosterInfo(props) { return (
); @@ -129,7 +129,7 @@ DiscoverMoviePosterInfo.propTypes = { digitalRelease: PropTypes.string, physicalRelease: PropTypes.string, runtime: PropTypes.number, - ratings: PropTypes.object, + ratings: PropTypes.arrayOf(PropTypes.object).isRequired, sortKey: PropTypes.string.isRequired, showRelativeDates: PropTypes.bool.isRequired, shortDateFormat: PropTypes.string.isRequired, diff --git a/frontend/src/DiscoverMovie/Table/DiscoverMovieRow.js b/frontend/src/DiscoverMovie/Table/DiscoverMovieRow.js index f2d35d597b..c39f09eaad 100644 --- a/frontend/src/DiscoverMovie/Table/DiscoverMovieRow.js +++ b/frontend/src/DiscoverMovie/Table/DiscoverMovieRow.js @@ -246,7 +246,7 @@ class DiscoverMovieRow extends Component { className={styles[name]} > ); @@ -373,7 +373,7 @@ DiscoverMovieRow.propTypes = { digitalRelease: PropTypes.string, runtime: PropTypes.number, genres: PropTypes.arrayOf(PropTypes.string).isRequired, - ratings: PropTypes.object.isRequired, + ratings: PropTypes.arrayOf(PropTypes.object).isRequired, certification: PropTypes.string, collection: PropTypes.object, columns: PropTypes.arrayOf(PropTypes.object).isRequired, diff --git a/frontend/src/Movie/Details/MovieDetails.css b/frontend/src/Movie/Details/MovieDetails.css index e8ae6553ca..4b1ed1441d 100644 --- a/frontend/src/Movie/Details/MovieDetails.css +++ b/frontend/src/Movie/Details/MovieDetails.css @@ -5,7 +5,7 @@ .header { position: relative; width: 100%; - height: 350px; + height: 375px; } .errorMessage { @@ -41,8 +41,8 @@ .poster { flex-shrink: 0; margin-right: 35px; - width: 200px; - height: 294px; + width: 217px; + height: 319px; } .info { diff --git a/frontend/src/Movie/Details/MovieDetails.js b/frontend/src/Movie/Details/MovieDetails.js index 1cbd8f0d96..698db8e25b 100644 --- a/frontend/src/Movie/Details/MovieDetails.js +++ b/frontend/src/Movie/Details/MovieDetails.js @@ -3,8 +3,8 @@ import PropTypes from 'prop-types'; import React, { Component } from 'react'; import { Tab, TabList, TabPanel, Tabs } from 'react-tabs'; import TextTruncate from 'react-text-truncate'; -import HeartRating from 'Components/HeartRating'; import Icon from 'Components/Icon'; +import ImdbRating from 'Components/ImdbRating'; import InfoLabel from 'Components/InfoLabel'; import IconButton from 'Components/Link/IconButton'; import Marquee from 'Components/Marquee'; @@ -16,6 +16,8 @@ import PageToolbar from 'Components/Page/Toolbar/PageToolbar'; import PageToolbarButton from 'Components/Page/Toolbar/PageToolbarButton'; import PageToolbarSection from 'Components/Page/Toolbar/PageToolbarSection'; import PageToolbarSeparator from 'Components/Page/Toolbar/PageToolbarSeparator'; +import RottenTomatoRating from 'Components/RottenTomatoRating'; +import TmdbRating from 'Components/TmdbRating'; import Popover from 'Components/Tooltip/Popover'; import Tooltip from 'Components/Tooltip/Tooltip'; import { icons, kinds, sizes, tooltipPositions } from 'Helpers/Props'; @@ -449,17 +451,6 @@ class MovieDetails extends Component { } - { - !!ratings && - - - - } - {
+
+ { + !!ratings.tmdb && + + + + } + { + !!ratings.imdb && + + + + } + { + !!ratings.rottenTomatoes && + + + + } +
+
); diff --git a/frontend/src/Store/Actions/discoverMovieActions.js b/frontend/src/Store/Actions/discoverMovieActions.js index 62f3534da3..3b2f85b9e8 100644 --- a/frontend/src/Store/Actions/discoverMovieActions.js +++ b/frontend/src/Store/Actions/discoverMovieActions.js @@ -200,7 +200,7 @@ export const defaultState = { ratings: function(item) { const { ratings = {} } = item; - return ratings.value; + return ratings.tmdb? ratings.tmdb.value : 0; } }, @@ -330,8 +330,23 @@ export const defaultState = { valueType: filterBuilderValueTypes.MINIMUM_AVAILABILITY }, { - name: 'ratings', - label: 'Rating', + name: 'tmdbRating', + label: translate('TmdbRating'), + type: filterBuilderTypes.NUMBER + }, + { + name: 'tmdbVotes', + label: translate('TmdbVotes'), + type: filterBuilderTypes.NUMBER + }, + { + name: 'imdbRating', + label: translate('ImdbRating'), + type: filterBuilderTypes.NUMBER + }, + { + name: 'imdbVotes', + label: translate('ImdbVotes'), type: filterBuilderTypes.NUMBER }, { diff --git a/frontend/src/Store/Actions/movieActions.js b/frontend/src/Store/Actions/movieActions.js index 4aa7d55fc4..0208420435 100644 --- a/frontend/src/Store/Actions/movieActions.js +++ b/frontend/src/Store/Actions/movieActions.js @@ -131,10 +131,38 @@ export const filterPredicates = { return dateFilterPredicate(item.digitalRelease, filterValue, type); }, - ratings: function(item, filterValue, type) { + tmdbRating: function(item, filterValue, type) { const predicate = filterTypePredicates[type]; - return predicate(item.ratings.value * 10, filterValue); + const rating = item.ratings.tmdb ? item.ratings.tmdb.value : 0; + + return predicate(rating * 10, filterValue); + }, + + tmdbVotes: function(item, filterValue, type) { + const predicate = filterTypePredicates[type]; + + const rating = item.ratings.tmdb ? item.ratings.tmdb.votes : 0; + + return predicate(rating, filterValue); + }, + + imdbRating: function(item, filterValue, type) { + const predicate = filterTypePredicates[type]; + + console.log(item.ratings); + + const rating = item.ratings.imdb ? item.ratings.imdb.value : 0; + + return predicate(rating, filterValue); + }, + + imdbVotes: function(item, filterValue, type) { + const predicate = filterTypePredicates[type]; + + const rating = item.ratings.imdb ? item.ratings.imdb.votes : 0; + + return predicate(rating, filterValue); }, qualityCutoffNotMet: function(item) { diff --git a/frontend/src/Store/Actions/movieIndexActions.js b/frontend/src/Store/Actions/movieIndexActions.js index 7a9171366a..a904a82551 100644 --- a/frontend/src/Store/Actions/movieIndexActions.js +++ b/frontend/src/Store/Actions/movieIndexActions.js @@ -209,7 +209,7 @@ export const defaultState = { ratings: function(item) { const { ratings = {} } = item; - return ratings.value; + return ratings.tmdb? ratings.tmdb.value : 0; } }, @@ -357,8 +357,23 @@ export const defaultState = { } }, { - name: 'ratings', - label: translate('Ratings'), + name: 'tmdbRating', + label: translate('TmdbRating'), + type: filterBuilderTypes.NUMBER + }, + { + name: 'tmdbVotes', + label: translate('TmdbVotes'), + type: filterBuilderTypes.NUMBER + }, + { + name: 'imdbRating', + label: translate('ImdbRating'), + type: filterBuilderTypes.NUMBER + }, + { + name: 'imdbVotes', + label: translate('ImdbVotes'), type: filterBuilderTypes.NUMBER }, { diff --git a/src/NzbDrone.Core/Datastore/Migration/206_multiple_ratings_support.cs b/src/NzbDrone.Core/Datastore/Migration/206_multiple_ratings_support.cs new file mode 100644 index 0000000000..717aa37228 --- /dev/null +++ b/src/NzbDrone.Core/Datastore/Migration/206_multiple_ratings_support.cs @@ -0,0 +1,107 @@ +using System.Collections.Generic; +using System.Data; +using System.Text.Json; +using System.Text.Json.Serialization; +using Dapper; +using FluentMigrator; +using NzbDrone.Core.Datastore.Migration.Framework; + +namespace NzbDrone.Core.Datastore.Migration +{ + [Migration(206)] + public class multiple_ratings_support : NzbDroneMigrationBase + { + private readonly JsonSerializerOptions _serializerSettings; + + public multiple_ratings_support() + { + _serializerSettings = new JsonSerializerOptions + { + AllowTrailingCommas = true, + DefaultIgnoreCondition = JsonIgnoreCondition.Never, + PropertyNameCaseInsensitive = true, + DictionaryKeyPolicy = JsonNamingPolicy.CamelCase, + PropertyNamingPolicy = JsonNamingPolicy.CamelCase, + WriteIndented = true + }; + } + + protected override void MainDbUpgrade() + { + Execute.Sql("UPDATE CustomFilters SET Filters = Replace(Filters, 'ratings', 'tmdbRating') WHERE Type = 'discoverMovie';"); + Execute.Sql("UPDATE CustomFilters SET Filters = Replace(Filters, 'ratings', 'tmdbRating') WHERE Type = 'movieIndex';"); + + Execute.WithConnection((conn, tran) => FixRatings(conn, tran, "Movies")); + Execute.WithConnection((conn, tran) => FixRatings(conn, tran, "ImportListMovies")); + } + + private void FixRatings(IDbConnection conn, IDbTransaction tran, string table) + { + var rows = conn.Query($"SELECT Id, Ratings FROM {table}"); + + var corrected = new List(); + + foreach (var row in rows) + { + var oldRatings = JsonSerializer.Deserialize(row.Ratings, _serializerSettings); + + var newRatings = new Ratings206 + { + Tmdb = new RatingChild206 + { + Votes = oldRatings.Votes, + Value = oldRatings.Value, + Type = RatingType206.User + } + }; + + corrected.Add(new Movie206 + { + Id = row.Id, + Ratings = JsonSerializer.Serialize(newRatings, _serializerSettings) + }); + } + + var updateSql = $"UPDATE {table} SET Ratings = @Ratings WHERE Id = @Id"; + conn.Execute(updateSql, corrected, transaction: tran); + } + + private class Movie205 + { + public int Id { get; set; } + public string Ratings { get; set; } + } + + private class Ratings205 + { + public int Votes { get; set; } + public decimal Value { get; set; } + } + + private class Movie206 + { + public int Id { get; set; } + public string Ratings { get; set; } + } + + private class Ratings206 + { + public RatingChild206 Tmdb { get; set; } + public RatingChild206 Imdb { get; set; } + public RatingChild206 Metacritic { get; set; } + public RatingChild206 RottenTomatoes { get; set; } + } + + private class RatingChild206 + { + public int Votes { get; set; } + public decimal Value { get; set; } + public RatingType206 Type { get; set; } + } + + private enum RatingType206 + { + User + } + } +} diff --git a/src/NzbDrone.Core/Datastore/TableMapping.cs b/src/NzbDrone.Core/Datastore/TableMapping.cs index b5589bba5c..16e391533d 100644 --- a/src/NzbDrone.Core/Datastore/TableMapping.cs +++ b/src/NzbDrone.Core/Datastore/TableMapping.cs @@ -179,6 +179,7 @@ private static void RegisterMappers() SqlMapper.AddTypeHandler(new EmbeddedDocumentConverter>()); SqlMapper.AddTypeHandler(new EmbeddedDocumentConverter(new QualityIntConverter(), new LanguageIntConverter())); SqlMapper.AddTypeHandler(new EmbeddedDocumentConverter()); + SqlMapper.AddTypeHandler(new EmbeddedDocumentConverter()); SqlMapper.AddTypeHandler(new EmbeddedDocumentConverter>()); SqlMapper.AddTypeHandler(new EmbeddedDocumentConverter>()); SqlMapper.AddTypeHandler(new OsPathConverter()); diff --git a/src/NzbDrone.Core/Extras/Metadata/Consumers/MediaBrowser/MediaBrowserMetadata.cs b/src/NzbDrone.Core/Extras/Metadata/Consumers/MediaBrowser/MediaBrowserMetadata.cs index 89bb2b0717..617c4da7e4 100644 --- a/src/NzbDrone.Core/Extras/Metadata/Consumers/MediaBrowser/MediaBrowserMetadata.cs +++ b/src/NzbDrone.Core/Extras/Metadata/Consumers/MediaBrowser/MediaBrowserMetadata.cs @@ -75,7 +75,7 @@ public override MetadataFileResult MovieMetadata(Movie movie, MovieFile movieFil movieElement.Add(new XElement("Overview", movie.Overview)); movieElement.Add(new XElement("LocalTitle", movie.Title)); - movieElement.Add(new XElement("Rating", movie.Ratings.Value)); + movieElement.Add(new XElement("Rating", movie.Ratings.Tmdb?.Value ?? 0)); movieElement.Add(new XElement("ProductionYear", movie.Year)); movieElement.Add(new XElement("RunningTime", movie.Runtime)); movieElement.Add(new XElement("IMDB", movie.ImdbId)); diff --git a/src/NzbDrone.Core/Extras/Metadata/Consumers/Xbmc/XbmcMetadata.cs b/src/NzbDrone.Core/Extras/Metadata/Consumers/Xbmc/XbmcMetadata.cs index 9f9d6e98d4..22980ed69b 100644 --- a/src/NzbDrone.Core/Extras/Metadata/Consumers/Xbmc/XbmcMetadata.cs +++ b/src/NzbDrone.Core/Extras/Metadata/Consumers/Xbmc/XbmcMetadata.cs @@ -154,19 +154,32 @@ public override MetadataFileResult MovieMetadata(Movie movie, MovieFile movieFil details.Add(new XElement("sorttitle", movie.SortTitle)); - if (movie.Ratings != null && movie.Ratings.Votes > 0) + if (movie.Ratings.Tmdb?.Votes > 0 || movie.Ratings.Imdb?.Votes > 0) { var setRating = new XElement("ratings"); - var setRatethemoviedb = new XElement("rating", new XAttribute("name", "themoviedb"), new XAttribute("max", "10"), new XAttribute("default", "true")); - setRatethemoviedb.Add(new XElement("value", movie.Ratings.Value)); - setRatethemoviedb.Add(new XElement("votes", movie.Ratings.Votes)); - setRating.Add(setRatethemoviedb); + + if (movie.Ratings.Tmdb?.Votes > 0) + { + var setRatethemoviedb = new XElement("rating", new XAttribute("name", "themoviedb"), new XAttribute("max", "10"), new XAttribute("default", "true")); + setRatethemoviedb.Add(new XElement("value", movie.Ratings.Tmdb.Value)); + setRatethemoviedb.Add(new XElement("votes", movie.Ratings.Tmdb.Votes)); + setRating.Add(setRatethemoviedb); + } + + if (movie.Ratings.Imdb?.Votes > 0) + { + var setRateImdb = new XElement("rating", new XAttribute("name", "imdb"), new XAttribute("max", "10")); + setRateImdb.Add(new XElement("value", movie.Ratings.Imdb.Value)); + setRateImdb.Add(new XElement("votes", movie.Ratings.Imdb.Votes)); + setRating.Add(setRateImdb); + } + details.Add(setRating); } - if (movie.Ratings != null && movie.Ratings.Votes > 0) + if (movie.Ratings?.Tmdb?.Votes > 0) { - details.Add(new XElement("rating", movie.Ratings.Value)); + details.Add(new XElement("rating", movie.Ratings.Tmdb.Value)); } details.Add(new XElement("userrating")); diff --git a/src/NzbDrone.Core/ImportLists/ImportListMovies/ImportListMovie.cs b/src/NzbDrone.Core/ImportLists/ImportListMovies/ImportListMovie.cs index bd6fce9552..288f6ba40e 100644 --- a/src/NzbDrone.Core/ImportLists/ImportListMovies/ImportListMovie.cs +++ b/src/NzbDrone.Core/ImportLists/ImportListMovies/ImportListMovie.cs @@ -13,6 +13,7 @@ public ImportListMovie() Images = new List(); Genres = new List(); Translations = new List(); + Ratings = new Ratings(); } public int TmdbId { get; set; } diff --git a/src/NzbDrone.Core/Localization/Core/en.json b/src/NzbDrone.Core/Localization/Core/en.json index 3754da9e17..9ff22d4b2d 100644 --- a/src/NzbDrone.Core/Localization/Core/en.json +++ b/src/NzbDrone.Core/Localization/Core/en.json @@ -226,6 +226,10 @@ "DetailedProgressBar": "Detailed Progress Bar", "DetailedProgressBarHelpText": "Show text on progress bar", "Details": "Details", + "TmdbRating": "TMDb Rating", + "TmdbVotes": "TMDb Votes", + "ImdbRating": "IMDb Rating", + "ImdbVotes": "IMDb Votes", "DigitalRelease": "Digital Release", "Disabled": "Disabled", "Discord": "Discord", diff --git a/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/MovieResource.cs b/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/MovieResource.cs index aefe90f72f..15bc72ef40 100644 --- a/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/MovieResource.cs +++ b/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/MovieResource.cs @@ -11,7 +11,10 @@ public class MovieResource public string Title { get; set; } public string OriginalTitle { get; set; } public string TitleSlug { get; set; } - public List Ratings { get; set; } + + //Depricated but left in place until cache fills new object (MovieRatings) + public List Ratings { get; set; } + public RatingResource MovieRatings { get; set; } public int? Runtime { get; set; } public List Images { get; set; } public List Genres { get; set; } diff --git a/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/RatingResource.cs b/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/RatingResource.cs index 18e9830b0a..20bd8a296d 100644 --- a/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/RatingResource.cs +++ b/src/NzbDrone.Core/MetadataSource/SkyHook/Resource/RatingResource.cs @@ -1,6 +1,14 @@ namespace NzbDrone.Core.MetadataSource.SkyHook.Resource { public class RatingResource + { + public RatingItem Tmdb { get; set; } + public RatingItem Imdb { get; set; } + public RatingItem Metacritic { get; set; } + public RatingItem RottenTomatoes { get; set; } + } + + public class RatingItem { public int Count { get; set; } public decimal Value { get; set; } diff --git a/src/NzbDrone.Core/MetadataSource/SkyHook/SkyHookProxy.cs b/src/NzbDrone.Core/MetadataSource/SkyHook/SkyHookProxy.cs index 7bf60603ca..996e3635ba 100644 --- a/src/NzbDrone.Core/MetadataSource/SkyHook/SkyHookProxy.cs +++ b/src/NzbDrone.Core/MetadataSource/SkyHook/SkyHookProxy.cs @@ -202,10 +202,21 @@ public Movie MapMovie(MovieResource resource) var certificationCountry = _configService.CertificationCountry.ToString(); movie.Certification = resource.Certifications.FirstOrDefault(m => m.Country == certificationCountry)?.Certification; - movie.Ratings = resource.Ratings.Select(MapRatings).FirstOrDefault() ?? new Ratings(); + movie.Ratings = MapRatings(resource.MovieRatings) ?? new Ratings(); movie.Genres = resource.Genres; movie.Recommendations = resource.Recommendations?.Select(r => r.TmdbId).ToList() ?? new List(); + //Workaround due to metadata change until cache cleans up + if (movie.Ratings.Tmdb == null) + { + var tmdbRating = resource.Ratings.FirstOrDefault(); + movie.Ratings.Tmdb = new RatingChild + { + Votes = tmdbRating.Count, + Value = tmdbRating.Value + }; + } + var now = DateTime.Now; movie.Status = MovieStatusType.Announced; @@ -512,18 +523,56 @@ private static MovieTranslation MapTranslation(TranslationResource arg) return newAlternativeTitle; } - private static Ratings MapRatings(RatingResource rating) + private static Ratings MapRatings(RatingResource ratings) { - if (rating == null) + if (ratings == null) { return new Ratings(); } - return new Ratings + var mappedRatings = new Ratings(); + + if (ratings.Tmdb != null) { - Votes = rating.Count, - Value = rating.Value - }; + mappedRatings.Tmdb = new RatingChild + { + Type = (RatingType)Enum.Parse(typeof(RatingType), ratings.Tmdb.Type), + Value = ratings.Tmdb.Value, + Votes = ratings.Tmdb.Count + }; + } + + if (ratings.Imdb != null) + { + mappedRatings.Imdb = new RatingChild + { + Type = (RatingType)Enum.Parse(typeof(RatingType), ratings.Imdb.Type), + Value = ratings.Imdb.Value, + Votes = ratings.Imdb.Count + }; + } + + if (ratings.Metacritic != null) + { + mappedRatings.Metacritic = new RatingChild + { + Type = (RatingType)Enum.Parse(typeof(RatingType), ratings.Metacritic.Type), + Value = ratings.Metacritic.Value, + Votes = ratings.Metacritic.Count + }; + } + + if (ratings.RottenTomatoes != null) + { + mappedRatings.RottenTomatoes = new RatingChild + { + Type = (RatingType)Enum.Parse(typeof(RatingType), ratings.RottenTomatoes.Type), + Value = ratings.RottenTomatoes.Value, + Votes = ratings.RottenTomatoes.Count + }; + } + + return mappedRatings; } private static MediaCover.MediaCover MapImage(ImageResource arg) diff --git a/src/NzbDrone.Core/Movies/Ratings.cs b/src/NzbDrone.Core/Movies/Ratings.cs index 568485dffc..3cf61fdeaa 100644 --- a/src/NzbDrone.Core/Movies/Ratings.cs +++ b/src/NzbDrone.Core/Movies/Ratings.cs @@ -1,10 +1,25 @@ -using NzbDrone.Core.Datastore; +using NzbDrone.Core.Datastore; namespace NzbDrone.Core.Movies { public class Ratings : IEmbeddedDocument + { + public RatingChild Imdb { get; set; } + public RatingChild Tmdb { get; set; } + public RatingChild Metacritic { get; set; } + public RatingChild RottenTomatoes { get; set; } + } + + public class RatingChild { public int Votes { get; set; } public decimal Value { get; set; } + public RatingType Type { get; set; } + } + + public enum RatingType + { + User, + Critic } } diff --git a/src/NzbDrone.Core/Notifications/Discord/Discord.cs b/src/NzbDrone.Core/Notifications/Discord/Discord.cs index 8cd441ad23..c7a0934e7f 100644 --- a/src/NzbDrone.Core/Notifications/Discord/Discord.cs +++ b/src/NzbDrone.Core/Notifications/Discord/Discord.cs @@ -70,7 +70,7 @@ public override void OnGrab(GrabMessage message) break; case DiscordGrabFieldType.Rating: discordField.Name = "Rating"; - discordField.Value = message.Movie.Ratings.Value.ToString(); + discordField.Value = message.Movie.Ratings.Tmdb?.Value.ToString() ?? string.Empty; break; case DiscordGrabFieldType.Genres: discordField.Name = "Genres"; @@ -157,7 +157,7 @@ public override void OnDownload(DownloadMessage message) break; case DiscordImportFieldType.Rating: discordField.Name = "Rating"; - discordField.Value = message.Movie.Ratings.Value.ToString(); + discordField.Value = message.Movie.Ratings.Tmdb?.Value.ToString() ?? string.Empty; break; case DiscordImportFieldType.Genres: discordField.Name = "Genres";