using System;
using System.Linq;
using API.Entities;
using API.Entities.Enums;
using API.Services;
using Kavita.Common.Extensions;
using Nager.ArticleNumber;
namespace API.Data.Metadata;
/// 
/// A representation of a ComicInfo.xml file
/// 
/// See reference of the loose spec here: https://anansi-project.github.io/docs/comicinfo/documentation
public class ComicInfo
{
    public string Summary { get; set; } = string.Empty;
    public string Title { get; set; } = string.Empty;
    public string Series { get; set; } = string.Empty;
    /// 
    /// Localized Series name. Not standard.
    /// 
    public string LocalizedSeries { get; set; } = string.Empty;
    public string SeriesSort { get; set; } = string.Empty;
    public string Number { get; set; } = string.Empty;
    /// 
    /// The total number of items in the series.
    /// 
    [System.ComponentModel.DefaultValueAttribute(0)]
    public int Count { get; set; } = 0;
    public string Volume { get; set; } = string.Empty;
    public string Notes { get; set; } = string.Empty;
    public string Genre { get; set; } = string.Empty;
    public int PageCount { get; set; }
    // ReSharper disable once InconsistentNaming
    /// 
    /// IETF BCP 47 Code to represent the language of the content
    /// 
    public string LanguageISO { get; set; } = string.Empty;
    // ReSharper disable once InconsistentNaming
    /// 
    /// ISBN for the underlying document
    /// 
    /// ComicInfo.xml will actually output a GTIN (Global Trade Item Number) and it is the responsibility of the Parser to extract the ISBN. EPub will return ISBN.
    public string Isbn { get; set; } = string.Empty;
    /// 
    /// This is only for deserialization and used within . Use  for the actual value.
    /// 
    public string GTIN { get; set; } = string.Empty;
    /// 
    /// This is the link to where the data was scraped from
    /// 
    /// This can be comma-separated
    public string Web { get; set; } = string.Empty;
    [System.ComponentModel.DefaultValueAttribute(0)]
    public int Day { get; set; } = 0;
    [System.ComponentModel.DefaultValueAttribute(0)]
    public int Month { get; set; } = 0;
    [System.ComponentModel.DefaultValueAttribute(0)]
    public int Year { get; set; } = 0;
    /// 
    /// Rating based on the content. Think PG-13, R for movies. See  for valid types
    /// 
    public string AgeRating { get; set; } = string.Empty;
    /// 
    /// User's rating of the content
    /// 
    public float UserRating { get; set; }
    /// 
    /// Can contain multiple comma separated strings, each create a 
    /// 
    public string SeriesGroup { get; set; } = string.Empty;
    /// 
    /// Can contain multiple comma separated numbers that match with StoryArcNumber
    /// 
    public string StoryArc { get; set; } = string.Empty;
    /// 
    /// Can contain multiple comma separated numbers that match with StoryArc
    /// 
    public string StoryArcNumber { get; set; } = string.Empty;
    public string AlternateNumber { get; set; } = string.Empty;
    public string AlternateSeries { get; set; } = string.Empty;
    /// 
    /// Not used
    /// 
    [System.ComponentModel.DefaultValueAttribute(0)]
    public int AlternateCount { get; set; } = 0;
    /// 
    /// This is Epub only: calibre:title_sort
    /// Represents the sort order for the title
    /// 
    public string TitleSort { get; set; } = string.Empty;
    /// 
    /// This comes from ComicInfo and is free form text. We use this to validate against a set of tags and mark a file as
    /// special.
    /// 
    public string Format { get; set; } = string.Empty;
    /// 
    /// The translator, can be comma separated. This is part of ComicInfo.xml draft v2.1
    /// 
    /// See https://github.com/anansi-project/comicinfo/issues/2 for information about this tag
    public string Translator { get; set; } = string.Empty;
    /// 
    /// Misc tags. This is part of ComicInfo.xml draft v2.1
    /// 
    /// See https://github.com/anansi-project/comicinfo/issues/1 for information about this tag
    public string Tags { get; set; } = string.Empty;
    /// 
    /// This is the Author. For Books, we map creator tag in OPF to this field. Comma separated if multiple.
    /// 
    public string Writer { get; set; } = string.Empty;
    public string Penciller { get; set; } = string.Empty;
    public string Inker { get; set; } = string.Empty;
    public string Colorist { get; set; } = string.Empty;
    public string Letterer { get; set; } = string.Empty;
    public string CoverArtist { get; set; } = string.Empty;
    public string Editor { get; set; } = string.Empty;
    public string Publisher { get; set; } = string.Empty;
    public string Characters { get; set; } = string.Empty;
    public static AgeRating ConvertAgeRatingToEnum(string value)
    {
        if (string.IsNullOrEmpty(value)) return Entities.Enums.AgeRating.Unknown;
        return Enum.GetValues()
            .SingleOrDefault(t => t.ToDescription().ToUpperInvariant().Equals(value.ToUpperInvariant()), Entities.Enums.AgeRating.Unknown);
    }
    public static void CleanComicInfo(ComicInfo? info)
    {
        if (info == null) return;
        info.Series = info.Series.Trim();
        info.SeriesSort = info.SeriesSort.Trim();
        info.LocalizedSeries = info.LocalizedSeries.Trim();
        info.Writer = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Writer);
        info.Colorist = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Colorist);
        info.Editor = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Editor);
        info.Inker = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Inker);
        info.Letterer = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Letterer);
        info.Penciller = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Penciller);
        info.Publisher = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Publisher);
        info.Characters = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Characters);
        info.Translator = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.Translator);
        info.CoverArtist = Services.Tasks.Scanner.Parser.Parser.CleanAuthor(info.CoverArtist);
        // We need to convert GTIN to ISBN
        if (!string.IsNullOrEmpty(info.GTIN))
        {
            // This is likely a valid ISBN
            if (info.GTIN[0] == '0')
            {
                var potentialISBN = info.GTIN.Substring(1, info.GTIN.Length - 1);
                if (ArticleNumberHelper.IsValidIsbn13(potentialISBN))
                {
                    info.Isbn = potentialISBN;
                }
            } else if (ArticleNumberHelper.IsValidIsbn10(info.GTIN) || ArticleNumberHelper.IsValidIsbn13(info.GTIN))
            {
                info.Isbn = info.GTIN;
            }
        }
    }
    /// 
    /// Uses both Volume and Number to make an educated guess as to what count refers to and it's highest number.
    /// 
    /// 
    public int CalculatedCount()
    {
        if (!string.IsNullOrEmpty(Number) && float.Parse(Number) > 0)
        {
            return (int) Math.Floor(float.Parse(Number));
        }
        if (!string.IsNullOrEmpty(Volume) && float.Parse(Volume) > 0)
        {
            return Math.Max(Count, (int) Math.Floor(float.Parse(Volume)));
        }
        return 0;
    }
}