using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Sorting;
using MediaBrowser.Model.Querying;
using System;
namespace MediaBrowser.Server.Implementations.Sorting
{
    class AiredEpisodeOrderComparer : IBaseItemComparer
    {
        /// 
        /// Compares the specified x.
        /// 
        /// The x.
        /// The y.
        /// System.Int32.
        public int Compare(BaseItem x, BaseItem y)
        {
            if (x.PremiereDate.HasValue && y.PremiereDate.HasValue)
            {
                var val = DateTime.Compare(x.PremiereDate.Value, y.PremiereDate.Value);
                if (val != 0)
                {
                    //return val;
                }
            }
            var episode1 = x as Episode;
            var episode2 = y as Episode;
            if (episode1 == null)
            {
                if (episode2 == null)
                {
                    return 0;
                }
                return 1;
            }
            if (episode2 == null)
            {
                return -1;
            }
            return Compare(episode1, episode2);
        }
        private int Compare(Episode x, Episode y)
        {
            var isXSpecial = (x.PhysicalSeasonNumber ?? -1) == 0;
            var isYSpecial = (y.PhysicalSeasonNumber ?? -1) == 0;
            if (isXSpecial && isYSpecial)
            {
                return CompareSpecials(x, y);
            }
            if (!isXSpecial && !isYSpecial)
            {
                return CompareEpisodes(x, y);
            }
            if (!isXSpecial && isYSpecial)
            {
                return CompareEpisodeToSpecial(x, y);
            }
            return CompareEpisodeToSpecial(y, x) * -1;
        }
        private int CompareEpisodeToSpecial(Episode x, Episode y)
        {
            var xSeason = x.PhysicalSeasonNumber ?? -1;
            var ySeason = y.AirsAfterSeasonNumber ?? y.AirsBeforeSeasonNumber ?? -1;
            if (xSeason != ySeason)
            {
                return xSeason.CompareTo(ySeason);
            }
            // Now we know they have the same season
            // Compare episode number
            // Add 1 to to non-specials to account for AirsBeforeEpisodeNumber
            var xEpisode = x.IndexNumber ?? -1;
            xEpisode++;
            var yEpisode = y.AirsBeforeEpisodeNumber ?? 10000;
            return xEpisode.CompareTo(yEpisode);
        }
        private int CompareSpecials(Episode x, Episode y)
        {
            return GetSpecialCompareValue(x).CompareTo(GetSpecialCompareValue(y));
        }
        private int GetSpecialCompareValue(Episode item)
        {
            // First sort by season number
            // Since there are three sort orders, pad with 9 digits (3 for each, figure 1000 episode buffer should be enough)
            var val = (item.AirsAfterSeasonNumber ?? item.AirsBeforeSeasonNumber ?? 0) * 1000000000;
            // Second sort order is if it airs after the season
            if (item.AirsAfterSeasonNumber.HasValue)
            {
                val += 1000000;
            }
            // Third level is the episode number
            val += (item.AirsBeforeEpisodeNumber ?? 0) * 1000;
            // Finally, if that's still the same, last resort is the special number itself
            val += item.IndexNumber ?? 0;
            return val;
        }
        private int CompareEpisodes(Episode x, Episode y)
        {
            var xValue = ((x.PhysicalSeasonNumber ?? -1) * 1000) + (x.IndexNumber ?? -1);
            var yValue = ((y.PhysicalSeasonNumber ?? -1) * 1000) + (y.IndexNumber ?? -1);
            return xValue.CompareTo(yValue);
        }
        /// 
        /// Gets the name.
        /// 
        /// The name.
        public string Name
        {
            get { return ItemSortBy.AiredEpisodeOrder; }
        }
    }
}