using MediaBrowser.Controller.Dto;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Persistence;
using MediaBrowser.Model.Dto;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Querying;
using ServiceStack;
using System;
using System.Collections.Generic;
using System.Linq;
namespace MediaBrowser.Api.UserLibrary
{
    /// 
    /// Class BaseItemsByNameService
    /// 
    /// The type of the T item type.
    public abstract class BaseItemsByNameService : BaseApiService
        where TItemType : BaseItem, IItemByName
    {
        /// 
        /// The _user manager
        /// 
        protected readonly IUserManager UserManager;
        /// 
        /// The library manager
        /// 
        protected readonly ILibraryManager LibraryManager;
        protected readonly IUserDataManager UserDataRepository;
        protected readonly IItemRepository ItemRepository;
        protected IDtoService DtoService { get; private set; }
        /// 
        /// Initializes a new instance of the  class.
        /// 
        /// The user manager.
        /// The library manager.
        /// The user data repository.
        /// The item repository.
        /// The dto service.
        protected BaseItemsByNameService(IUserManager userManager, ILibraryManager libraryManager, IUserDataManager userDataRepository, IItemRepository itemRepository, IDtoService dtoService)
        {
            UserManager = userManager;
            LibraryManager = libraryManager;
            UserDataRepository = userDataRepository;
            ItemRepository = itemRepository;
            DtoService = dtoService;
        }
        /// 
        /// Gets the specified request.
        /// 
        /// The request.
        /// Task{ItemsResult}.
        protected ItemsResult GetResult(GetItemsByName request)
        {
            User user = null;
            BaseItem parentItem;
            List libraryItems;
            if (request.UserId.HasValue)
            {
                user = UserManager.GetUserById(request.UserId.Value);
                parentItem = string.IsNullOrEmpty(request.ParentId) ? user.RootFolder : LibraryManager.GetItemById(request.ParentId);
                libraryItems = user.RootFolder.GetRecursiveChildren(user).ToList();
            }
            else
            {
                parentItem = string.IsNullOrEmpty(request.ParentId) ? LibraryManager.RootFolder : LibraryManager.GetItemById(request.ParentId);
                libraryItems = LibraryManager.RootFolder.GetRecursiveChildren().ToList();
            }
            IEnumerable items;
            var excludeItemTypes = request.GetExcludeItemTypes();
            var includeItemTypes = request.GetIncludeItemTypes();
            var mediaTypes = request.GetMediaTypes();
            Func filter = i => FilterItem(request, i, excludeItemTypes, includeItemTypes, mediaTypes);
            if (parentItem.IsFolder)
            {
                var folder = (Folder)parentItem;
                if (request.UserId.HasValue)
                {
                    items = request.Recursive ?
                        folder.GetRecursiveChildren(user, filter) :
                        folder.GetChildren(user, true).Where(filter);
                }
                else
                {
                    items = request.Recursive ?
                        folder.GetRecursiveChildren(filter) :
                        folder.Children.Where(filter);
                }
            }
            else
            {
                items = new[] { parentItem }.Where(filter);
            }
            var extractedItems = GetAllItems(request, items);
            var filteredItems = FilterItems(request, extractedItems, user);
            filteredItems = FilterByLibraryItems(request, filteredItems, user, libraryItems);
            filteredItems = LibraryManager.Sort(filteredItems, user, request.GetOrderBy(), request.SortOrder ?? SortOrder.Ascending).Cast();
            var ibnItemsArray = filteredItems.ToList();
            IEnumerable ibnItems = ibnItemsArray;
            var result = new ItemsResult
            {
                TotalRecordCount = ibnItemsArray.Count
            };
            if (request.StartIndex.HasValue || request.Limit.HasValue)
            {
                if (request.StartIndex.HasValue)
                {
                    ibnItems = ibnItems.Skip(request.StartIndex.Value);
                }
                if (request.Limit.HasValue)
                {
                    ibnItems = ibnItems.Take(request.Limit.Value);
                }
            }
            var tuples = ibnItems.Select(i => new Tuple>(i, i.GetTaggedItems(libraryItems).ToList()));
            var dtoOptions = GetDtoOptions(request);
            var dtos = tuples.Select(i => GetDto(i.Item1, user, dtoOptions, i.Item2));
            result.Items = dtos.Where(i => i != null).ToArray();
            return result;
        }
        private IEnumerable FilterByLibraryItems(GetItemsByName request, IEnumerable items, User user, IEnumerable libraryItems)
        {
            var filters = request.GetFilters().ToList();
            if (filters.Contains(ItemFilter.IsPlayed))
            {
                items = items.Where(i => i.GetTaggedItems(libraryItems).All(l => l.IsPlayed(user)));
            }
            if (filters.Contains(ItemFilter.IsUnplayed))
            {
                items = items.Where(i => i.GetTaggedItems(libraryItems).All(l => l.IsUnplayed(user)));
            }
            if (request.IsPlayed.HasValue)
            {
                var val = request.IsPlayed.Value;
                items = items.Where(i => i.GetTaggedItems(libraryItems).All(l => l.IsPlayed(user)) == val);
            }
            return items;
        }
        /// 
        /// Filters the items.
        /// 
        /// The request.
        /// The items.
        /// The user.
        /// IEnumerable{`0}.
        private IEnumerable FilterItems(GetItemsByName request, IEnumerable items, User user)
        {
            if (!string.IsNullOrEmpty(request.NameStartsWithOrGreater))
            {
                items = items.Where(i => string.Compare(request.NameStartsWithOrGreater, i.SortName, StringComparison.CurrentCultureIgnoreCase) < 1);
            }
            if (!string.IsNullOrEmpty(request.NameStartsWith))
            {
                items = items.Where(i => string.Compare(request.NameStartsWith, i.SortName.Substring(0, 1), StringComparison.CurrentCultureIgnoreCase) == 0);
            }
            if (!string.IsNullOrEmpty(request.NameLessThan))
            {
                items = items.Where(i => string.Compare(request.NameLessThan, i.SortName, StringComparison.CurrentCultureIgnoreCase) == 1);
            }
            var imageTypes = request.GetImageTypes().ToList();
            if (imageTypes.Count > 0)
            {
                items = items.Where(item => imageTypes.Any(item.HasImage));
            }
            var filters = request.GetFilters().ToList();
            if (filters.Contains(ItemFilter.Dislikes))
            {
                items = items.Where(i =>
                    {
                        var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
                        return userdata != null && userdata.Likes.HasValue && !userdata.Likes.Value;
                    });
            }
            if (filters.Contains(ItemFilter.Likes))
            {
                items = items.Where(i =>
                {
                    var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
                    return userdata != null && userdata.Likes.HasValue && userdata.Likes.Value;
                });
            }
            if (filters.Contains(ItemFilter.IsFavoriteOrLikes))
            {
                items = items.Where(i =>
                {
                    var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
                    var likes = userdata.Likes ?? false;
                    var favorite = userdata.IsFavorite;
                    return likes || favorite;
                });
            }
            if (filters.Contains(ItemFilter.IsFavorite))
            {
                items = items.Where(i =>
                {
                    var userdata = UserDataRepository.GetUserData(user.Id, i.GetUserDataKey());
                    return userdata != null && userdata.IsFavorite;
                });
            }
            // Avoid implicitly captured closure
            var currentRequest = request;
            return items.Where(i => ApplyAdditionalFilters(currentRequest, i, user, false));
        }
        private bool ApplyAdditionalFilters(BaseItemsRequest request, BaseItem i, User user, bool isPreFiltered)
        {
            if (!isPreFiltered)
            {
                // Apply tag filter
                var tags = request.GetTags();
                if (tags.Length > 0)
                {
                    var hasTags = i as IHasTags;
                    if (hasTags == null)
                    {
                        return false;
                    }
                    if (!(tags.Any(v => hasTags.Tags.Contains(v, StringComparer.OrdinalIgnoreCase))))
                    {
                        return false;
                    }
                }
                // Apply official rating filter
                var officialRatings = request.GetOfficialRatings();
                if (officialRatings.Length > 0 && !officialRatings.Contains(i.OfficialRating ?? string.Empty))
                {
                    return false;
                }
                // Apply genre filter
                var genres = request.GetGenres();
                if (genres.Length > 0 && !(genres.Any(v => i.Genres.Contains(v, StringComparer.OrdinalIgnoreCase))))
                {
                    return false;
                }
                // Apply year filter
                var years = request.GetYears();
                if (years.Length > 0 && !(i.ProductionYear.HasValue && years.Contains(i.ProductionYear.Value)))
                {
                    return false;
                }
            }
            return true;
        }
        /// 
        /// Filters the items.
        /// 
        /// The request.
        /// The f.
        /// The exclude item types.
        /// The include item types.
        /// The media types.
        /// IEnumerable{BaseItem}.
        protected bool FilterItem(GetItemsByName request, BaseItem f, string[] excludeItemTypes, string[] includeItemTypes, string[] mediaTypes)
        {
            // Exclude item types
            if (excludeItemTypes.Length > 0)
            {
                if (excludeItemTypes.Contains(f.GetType().Name, StringComparer.OrdinalIgnoreCase))
                {
                    return false;
                }
            }
            // Include item types
            if (includeItemTypes.Length > 0)
            {
                if (!includeItemTypes.Contains(f.GetType().Name, StringComparer.OrdinalIgnoreCase))
                {
                    return false;
                }
            }
            // Include MediaTypes
            if (mediaTypes.Length > 0)
            {
                if (!mediaTypes.Contains(f.MediaType ?? string.Empty, StringComparer.OrdinalIgnoreCase))
                {
                    return false;
                }
            }
            return true;
        }
        /// 
        /// Gets all items.
        /// 
        /// The request.
        /// The items.
        /// IEnumerable{Task{`0}}.
        protected abstract IEnumerable GetAllItems(GetItemsByName request, IEnumerable items);
        /// 
        /// Gets the dto.
        /// 
        /// The item.
        /// The user.
        /// The options.
        /// The library items.
        /// Task{DtoBaseItem}.
        private BaseItemDto GetDto(TItemType item, User user, DtoOptions options, List libraryItems)
        {
            var dto = DtoService.GetItemByNameDto(item, options, libraryItems, user);
            return dto;
        }
    }
    /// 
    /// Class GetItemsByName
    /// 
    public class GetItemsByName : BaseItemsRequest, IReturn
    {
        /// 
        /// Gets or sets the user id.
        /// 
        /// The user id.
        [ApiMember(Name = "UserId", Description = "Optional. Filter by user id, and attach user data", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
        public Guid? UserId { get; set; }
        [ApiMember(Name = "NameStartsWithOrGreater", Description = "Optional filter by items whose name is sorted equally or greater than a given input string.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
        public string NameStartsWithOrGreater { get; set; }
        [ApiMember(Name = "NameStartsWith", Description = "Optional filter by items whose name is sorted equally than a given input string.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
        public string NameStartsWith { get; set; }
        [ApiMember(Name = "NameLessThan", Description = "Optional filter by items whose name is sorted less than a given input string.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
        public string NameLessThan { get; set; }
        public GetItemsByName()
        {
            Recursive = true;
        }
    }
}