aboutsummaryrefslogtreecommitdiff
path: root/MediaBrowser.Controller/Providers/SortNameProvider.cs
blob: 933b0c78c3cf4b5d6b79b69e530fec0c2abf2129 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Entities.TV;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;

namespace MediaBrowser.Controller.Providers
{
    /// <summary>
    /// Class SortNameProvider
    /// </summary>
    public class SortNameProvider : BaseMetadataProvider
    {
        /// <summary>
        /// Supportses the specified item.
        /// </summary>
        /// <param name="item">The item.</param>
        /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns>
        public override bool Supports(BaseItem item)
        {
            return true;
        }

        /// <summary>
        /// Gets the priority.
        /// </summary>
        /// <value>The priority.</value>
        public override MetadataProviderPriority Priority
        {
            get { return MetadataProviderPriority.Last; }
        }

        /// <summary>
        /// Needses the refresh internal.
        /// </summary>
        /// <param name="item">The item.</param>
        /// <param name="providerInfo">The provider info.</param>
        /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns>
        protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
        {
            return !string.IsNullOrEmpty(item.Name) && string.IsNullOrEmpty(item.SortName);
        }

        /// <summary>
        /// Fetches metadata and returns true or false indicating if any work that requires persistence was done
        /// </summary>
        /// <param name="item">The item.</param>
        /// <param name="force">if set to <c>true</c> [force].</param>
        /// <param name="cancellationToken">The cancellation token.</param>
        /// <returns>Task{System.Boolean}.</returns>
        protected override Task<bool> FetchAsyncInternal(BaseItem item, bool force, CancellationToken cancellationToken)
        {
            return SetSortName(item, cancellationToken) ? TrueTaskResult : FalseTaskResult;
        }

        /// <summary>
        /// Sets the name of the sort.
        /// </summary>
        /// <param name="item">The item.</param>
        /// <param name="cancellationToken">The cancellation token.</param>
        /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns>
        protected bool SetSortName(BaseItem item, CancellationToken cancellationToken)
        {
            if (!string.IsNullOrWhiteSpace(item.SortName)) return false; //let the earlier provider win

            cancellationToken.ThrowIfCancellationRequested();
            
            if (item is Episode)
            {
                //special handling for TV episodes season and episode number
                item.SortName = (item.ParentIndexNumber != null ? item.ParentIndexNumber.Value.ToString("000-") : "") 
                    + (item.IndexNumber != null ? item.IndexNumber.Value.ToString("0000 - ") : "") + item.Name;
                
            }
            else if (item is Season)
            {
                //sort seasons by season number - numerically
                item.SortName = item.IndexNumber != null ? item.IndexNumber.Value.ToString("0000") : item.Name;
            }
            else if (item is Audio)
            {
                //sort tracks by production year and index no so they will sort in order if in a multi-album list
                item.SortName = (item.ProductionYear != null ? item.ProductionYear.Value.ToString("000-") : "") 
                    + (item.IndexNumber != null ? item.IndexNumber.Value.ToString("0000 - ") : "") + item.Name;
            }
            else if (item is MusicAlbum)
            {
                //sort albums by year
                item.SortName = item.ProductionYear != null ? item.ProductionYear.Value.ToString("0000") : item.Name;
            }
            else
            {
                if (item.Name == null) return false; //some items may not have name filled in properly

                var sortable = item.Name.Trim().ToLower();
                sortable = Kernel.Instance.Configuration.SortRemoveCharacters.Aggregate(sortable, (current, search) => current.Replace(search.ToLower(), string.Empty));

                sortable = Kernel.Instance.Configuration.SortReplaceCharacters.Aggregate(sortable, (current, search) => current.Replace(search.ToLower(), " "));

                foreach (var search in Kernel.Instance.Configuration.SortRemoveWords)
                {
                    cancellationToken.ThrowIfCancellationRequested();

                    var searchLower = search.ToLower();
                    // Remove from beginning if a space follows
                    if (sortable.StartsWith(searchLower + " "))
                    {
                        sortable = sortable.Remove(0, searchLower.Length + 1);
                    }
                    // Remove from middle if surrounded by spaces
                    sortable = sortable.Replace(" " + searchLower + " ", " ");

                    // Remove from end if followed by a space
                    if (sortable.EndsWith(" " + searchLower))
                    {
                        sortable = sortable.Remove(sortable.Length - (searchLower.Length + 1));
                    }
                }
                item.SortName = sortable;
            }

            return true;
        }

    }
}