using MediaBrowser.Common.Extensions;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
using MediaBrowser.Model.Logging;
using MediaBrowser.Model.Querying;
using MediaBrowser.Model.Search;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
namespace MediaBrowser.Server.Implementations.Library
{
    /// 
    /// Class LuceneSearchEngine
    /// http://www.codeproject.com/Articles/320219/Lucene-Net-ultra-fast-search-for-MVC-or-WebForms
    /// 
    public class SearchEngine : ISearchEngine
    {
        private readonly ILibraryManager _libraryManager;
        private readonly IUserManager _userManager;
        private readonly ILogger _logger;
        public SearchEngine(ILogManager logManager, ILibraryManager libraryManager, IUserManager userManager)
        {
            _libraryManager = libraryManager;
            _userManager = userManager;
            _logger = logManager.GetLogger("Lucene");
        }
        public async Task> GetSearchHints(SearchQuery query)
        {
            User user = null;
            if (string.IsNullOrWhiteSpace(query.UserId))
            {
            }
            else
            {
                user = _userManager.GetUserById(query.UserId);
            }
            var results = await GetSearchHints(query, user).ConfigureAwait(false);
            var searchResultArray = results.ToArray();
            results = searchResultArray;
            var count = searchResultArray.Length;
            if (query.StartIndex.HasValue)
            {
                results = results.Skip(query.StartIndex.Value);
            }
            if (query.Limit.HasValue)
            {
                results = results.Take(query.Limit.Value);
            }
            return new QueryResult
            {
                TotalRecordCount = count,
                Items = results.ToArray()
            };
        }
        private void AddIfMissing(List list, string value)
        {
            if (!list.Contains(value, StringComparer.OrdinalIgnoreCase))
            {
                list.Add(value);
            }
        }
        /// 
        /// Gets the search hints.
        /// 
        /// The query.
        /// The user.
        /// IEnumerable{SearchHintResult}.
        /// searchTerm
        private Task> GetSearchHints(SearchQuery query, User user)
        {
            var searchTerm = query.SearchTerm;
            if (string.IsNullOrWhiteSpace(searchTerm))
            {
                throw new ArgumentNullException("searchTerm");
            }
            searchTerm = searchTerm.RemoveDiacritics();
            var terms = GetWords(searchTerm);
            var hints = new List>();
            var excludeItemTypes = new List();
            var includeItemTypes = (query.IncludeItemTypes ?? new string[] { }).ToList();
            excludeItemTypes.Add(typeof(Year).Name);
            if (query.IncludeGenres && (includeItemTypes.Count == 0 || includeItemTypes.Contains("Genre", StringComparer.OrdinalIgnoreCase)))
            {
                if (!query.IncludeMedia)
                {
                    AddIfMissing(includeItemTypes, typeof(Genre).Name);
                    AddIfMissing(includeItemTypes, typeof(GameGenre).Name);
                    AddIfMissing(includeItemTypes, typeof(MusicGenre).Name);
                }
            }
            else
            {
                AddIfMissing(excludeItemTypes, typeof(Genre).Name);
                AddIfMissing(excludeItemTypes, typeof(GameGenre).Name);
                AddIfMissing(excludeItemTypes, typeof(MusicGenre).Name);
            }
            if (query.IncludePeople && (includeItemTypes.Count == 0 || includeItemTypes.Contains("People", StringComparer.OrdinalIgnoreCase)))
            {
                if (!query.IncludeMedia)
                {
                    AddIfMissing(includeItemTypes, typeof(Person).Name);
                }
            }
            else
            {
                AddIfMissing(excludeItemTypes, typeof(Person).Name);
            }
            if (query.IncludeStudios && (includeItemTypes.Count == 0 || includeItemTypes.Contains("Studio", StringComparer.OrdinalIgnoreCase)))
            {
                if (!query.IncludeMedia)
                {
                    AddIfMissing(includeItemTypes, typeof(Studio).Name);
                }
            }
            else
            {
                AddIfMissing(excludeItemTypes, typeof(Studio).Name);
            }
            if (query.IncludeArtists && (includeItemTypes.Count == 0 || includeItemTypes.Contains("MusicArtist", StringComparer.OrdinalIgnoreCase)))
            {
                if (!query.IncludeMedia)
                {
                    AddIfMissing(includeItemTypes, typeof(MusicArtist).Name);
                }
            }
            else
            {
                AddIfMissing(excludeItemTypes, typeof(MusicArtist).Name);
            }
            var mediaItems = _libraryManager.GetItems(new InternalItemsQuery
            {
                NameContains = searchTerm,
                ExcludeItemTypes = excludeItemTypes.ToArray(),
                IncludeItemTypes = includeItemTypes.ToArray(),
                MaxParentalRating = user == null ? null : user.Policy.MaxParentalRating,
                Limit = query.Limit.HasValue ? query.Limit * 3 : null
            }).Items;
            // Add search hints based on item name
            hints.AddRange(mediaItems.Where(i => IncludeInSearch(i) && IsVisible(i, user) && !(i is CollectionFolder)).Select(item =>
            {
                var index = GetIndex(item.Name, searchTerm, terms);
                return new Tuple(item, index.Item1, index.Item2);
            }));
            var returnValue = hints.Where(i => i.Item3 >= 0).OrderBy(i => i.Item3).Select(i => new SearchHintInfo
            {
                Item = i.Item1,
                MatchedTerm = i.Item2
            });
            return Task.FromResult(returnValue);
        }
        private bool IsVisible(BaseItem item, User user)
        {
            if (user == null)
            {
                return true;
            }
            if (item is IItemByName)
            {
                var dual = item as IHasDualAccess;
                if (dual == null || dual.IsAccessedByName)
                {
                    return true;
                }
            }
            return item.IsVisibleStandalone(user);
        }
        private bool IncludeInSearch(BaseItem item)
        {
            var episode = item as Episode;
            if (episode != null)
            {
                if (episode.IsMissingEpisode)
                {
                    return false;
                }
            }
            return true;
        }
        /// 
        /// Gets the index.
        /// 
        /// The input.
        /// The search input.
        /// The search input.
        /// System.Int32.
        private Tuple GetIndex(string input, string searchInput, List searchWords)
        {
            if (string.IsNullOrWhiteSpace(input))
            {
                throw new ArgumentNullException("input");
            }
            input = input.RemoveDiacritics();
            if (string.Equals(input, searchInput, StringComparison.OrdinalIgnoreCase))
            {
                return new Tuple(searchInput, 0);
            }
            var index = input.IndexOf(searchInput, StringComparison.OrdinalIgnoreCase);
            if (index == 0)
            {
                return new Tuple(searchInput, 1);
            }
            if (index > 0)
            {
                return new Tuple(searchInput, 2);
            }
            var items = GetWords(input);
            for (var i = 0; i < searchWords.Count; i++)
            {
                var searchTerm = searchWords[i];
                for (var j = 0; j < items.Count; j++)
                {
                    var item = items[j];
                    if (string.Equals(item, searchTerm, StringComparison.OrdinalIgnoreCase))
                    {
                        return new Tuple(searchTerm, 3 + (i + 1) * (j + 1));
                    }
                    index = item.IndexOf(searchTerm, StringComparison.OrdinalIgnoreCase);
                    if (index == 0)
                    {
                        return new Tuple(searchTerm, 4 + (i + 1) * (j + 1));
                    }
                    if (index > 0)
                    {
                        return new Tuple(searchTerm, 5 + (i + 1) * (j + 1));
                    }
                }
            }
            return new Tuple(null, -1);
        }
        /// 
        /// Gets the words.
        /// 
        /// The term.
        /// System.String[][].
        private List GetWords(string term)
        {
            var stoplist = GetStopList().ToList();
            return term.Split()
                .Where(i => !string.IsNullOrWhiteSpace(i) && !stoplist.Contains(i, StringComparer.OrdinalIgnoreCase))
                .ToList();
        }
        private IEnumerable GetStopList()
        {
            return new[]
            {
                "the",
                "a",
                "of",
                "an"
            };
        }
    }
}