mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-11-04 03:27:05 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			273 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			273 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
using System;
 | 
						|
using System.Collections.Generic;
 | 
						|
using System.Globalization;
 | 
						|
using API.Entities.Enums;
 | 
						|
using API.Entities.Interfaces;
 | 
						|
using API.Entities.Metadata;
 | 
						|
using API.Entities.MetadataMatching;
 | 
						|
using API.Entities.Person;
 | 
						|
using API.Extensions;
 | 
						|
using API.Services.Tasks.Scanner.Parser;
 | 
						|
 | 
						|
namespace API.Entities;
 | 
						|
 | 
						|
public class Chapter : IEntityDate, IHasReadTimeEstimate, IHasCoverImage, IHasKPlusMetadata
 | 
						|
{
 | 
						|
    public int Id { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Range of numbers. Chapter 2-4 -> "2-4". Chapter 2 -> "2". If the chapter is a special, will return the Special Name
 | 
						|
    /// </summary>
 | 
						|
    public required string Range { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Smallest number of the Range. Can be a partial like Chapter 4.5
 | 
						|
    /// </summary>
 | 
						|
    [Obsolete("Use MinNumber and MaxNumber instead")]
 | 
						|
    public required string Number { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Minimum Chapter Number.
 | 
						|
    /// </summary>
 | 
						|
    public float MinNumber { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Maximum Chapter Number
 | 
						|
    /// </summary>
 | 
						|
    public float MaxNumber { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// The sorting order of the Chapter. Inherits from MinNumber, but can be overridden.
 | 
						|
    /// </summary>
 | 
						|
    public float SortOrder { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Can the sort order be updated on scan or is it locked from UI
 | 
						|
    /// </summary>
 | 
						|
    public bool SortOrderLocked { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// The files that represent this Chapter
 | 
						|
    /// </summary>
 | 
						|
    public ICollection<MangaFile> Files { get; set; } = null!;
 | 
						|
    public DateTime Created { get; set; }
 | 
						|
    public DateTime LastModified { get; set; }
 | 
						|
    public DateTime CreatedUtc { get; set; }
 | 
						|
    public DateTime LastModifiedUtc { get; set; }
 | 
						|
 | 
						|
    public string? CoverImage { get; set; }
 | 
						|
    public string PrimaryColor { get; set; }
 | 
						|
    public string SecondaryColor { get; set; }
 | 
						|
    public bool CoverImageLocked { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Total number of pages in all MangaFiles
 | 
						|
    /// </summary>
 | 
						|
    public int Pages { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// If this Chapter contains files that could only be identified as Series or has Special Identifier from filename
 | 
						|
    /// </summary>
 | 
						|
    public bool IsSpecial { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Used for books/specials to display custom title. For non-specials/books, will be set to <see cref="Range"/>
 | 
						|
    /// </summary>
 | 
						|
    public string? Title { get; set; }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Age Rating for the issue/chapter
 | 
						|
    /// </summary>
 | 
						|
    public AgeRating AgeRating { get; set; }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Chapter title
 | 
						|
    /// </summary>
 | 
						|
    /// <remarks>This should not be confused with Title which is used for special filenames.</remarks>
 | 
						|
    public string TitleName { get; set; } = string.Empty;
 | 
						|
    /// <summary>
 | 
						|
    /// Date which chapter was released
 | 
						|
    /// </summary>
 | 
						|
    public DateTime ReleaseDate { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Summary for the Chapter/Issue
 | 
						|
    /// </summary>
 | 
						|
    public string? Summary { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Language for the Chapter/Issue
 | 
						|
    /// </summary>
 | 
						|
    public string? Language { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Total number of issues or volumes in the series. This is straight from ComicInfo
 | 
						|
    /// </summary>
 | 
						|
    public int TotalCount { get; set; } = 0;
 | 
						|
    /// <summary>
 | 
						|
    /// Number of the Total Count (progress the Series is complete)
 | 
						|
    /// </summary>
 | 
						|
    /// <remarks>This is either the highest of ComicInfo Count field and (nonparsed volume/chapter number)</remarks>
 | 
						|
    public int Count { get; set; } = 0;
 | 
						|
    /// <summary>
 | 
						|
    /// SeriesGroup tag in ComicInfo
 | 
						|
    /// </summary>
 | 
						|
    public string SeriesGroup { get; set; } = string.Empty;
 | 
						|
    public string StoryArc { get; set; } = string.Empty;
 | 
						|
    public string StoryArcNumber { get; set; } = string.Empty;
 | 
						|
    public string AlternateNumber { get; set; } = string.Empty;
 | 
						|
    public string AlternateSeries { get; set; } = string.Empty;
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Not currently used in Kavita
 | 
						|
    /// </summary>
 | 
						|
    public int AlternateCount { get; set; } = 0;
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Total Word count of all chapters in this chapter.
 | 
						|
    /// </summary>
 | 
						|
    /// <remarks>Word Count is only available from EPUB files</remarks>
 | 
						|
    public long WordCount { get; set; }
 | 
						|
    /// <inheritdoc cref="IHasReadTimeEstimate"/>
 | 
						|
    public int MinHoursToRead { get; set; }
 | 
						|
    /// <inheritdoc cref="IHasReadTimeEstimate"/>
 | 
						|
    public int MaxHoursToRead { get; set; }
 | 
						|
    /// <inheritdoc cref="IHasReadTimeEstimate"/>
 | 
						|
    public float AvgHoursToRead { get; set; }
 | 
						|
    /// <summary>
 | 
						|
    /// Comma-separated link of urls to external services that have some relation to the Chapter
 | 
						|
    /// </summary>
 | 
						|
    public string WebLinks { get; set; } = string.Empty;
 | 
						|
    public string ISBN { get; set; } = string.Empty;
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Tracks which metadata has been set by K+
 | 
						|
    /// </summary>
 | 
						|
    public IList<MetadataSettingField> KPlusOverrides { get; set; } = [];
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// (Kavita+) Average rating from Kavita+ metadata
 | 
						|
    /// </summary>
 | 
						|
    public float AverageExternalRating { get; set; } = 0f;
 | 
						|
 | 
						|
    #region Locks
 | 
						|
 | 
						|
    public bool AgeRatingLocked { get; set; }
 | 
						|
    public bool TitleNameLocked { get; set; }
 | 
						|
    public bool GenresLocked { get; set; }
 | 
						|
    public bool TagsLocked { get; set; }
 | 
						|
    public bool WriterLocked { get; set; }
 | 
						|
    public bool CharacterLocked { get; set; }
 | 
						|
    public bool ColoristLocked { get; set; }
 | 
						|
    public bool EditorLocked { get; set; }
 | 
						|
    public bool InkerLocked { get; set; }
 | 
						|
    public bool ImprintLocked { get; set; }
 | 
						|
    public bool LettererLocked { get; set; }
 | 
						|
    public bool PencillerLocked { get; set; }
 | 
						|
    public bool PublisherLocked { get; set; }
 | 
						|
    public bool TranslatorLocked { get; set; }
 | 
						|
    public bool TeamLocked { get; set; }
 | 
						|
    public bool LocationLocked { get; set; }
 | 
						|
    public bool CoverArtistLocked { get; set; }
 | 
						|
    public bool LanguageLocked { get; set; }
 | 
						|
    public bool SummaryLocked { get; set; }
 | 
						|
    public bool ISBNLocked { get; set; }
 | 
						|
    public bool ReleaseDateLocked { get; set; }
 | 
						|
 | 
						|
    #endregion
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// All people attached at a Chapter level. Usually Comics will have different people per issue.
 | 
						|
    /// </summary>
 | 
						|
    public ICollection<ChapterPeople> People { get; set; } = new List<ChapterPeople>();
 | 
						|
    /// <summary>
 | 
						|
    /// Genres for the Chapter
 | 
						|
    /// </summary>
 | 
						|
    public ICollection<Genre> Genres { get; set; } = new List<Genre>();
 | 
						|
    public ICollection<Tag> Tags { get; set; } = new List<Tag>();
 | 
						|
    public ICollection<AppUserChapterRating> Ratings { get; set; } = [];
 | 
						|
 | 
						|
    public ICollection<AppUserProgress> UserProgress { get; set; }
 | 
						|
 | 
						|
 | 
						|
    // Relationships
 | 
						|
    public Volume Volume { get; set; } = null!;
 | 
						|
    public int VolumeId { get; set; }
 | 
						|
 | 
						|
    public ICollection<ExternalReview> ExternalReviews { get; set; } = [];
 | 
						|
    public ICollection<ExternalRating> ExternalRatings { get; set; } = null!;
 | 
						|
 | 
						|
    public void UpdateFrom(ParserInfo info)
 | 
						|
    {
 | 
						|
        Files ??= new List<MangaFile>();
 | 
						|
        IsSpecial = info.IsSpecialInfo();
 | 
						|
        if (IsSpecial)
 | 
						|
        {
 | 
						|
            Number = Parser.DefaultChapter;
 | 
						|
            MinNumber = Parser.DefaultChapterNumber;
 | 
						|
            MaxNumber = Parser.DefaultChapterNumber;
 | 
						|
        }
 | 
						|
        Title = (IsSpecial && info.Format is MangaFormat.Epub or MangaFormat.Pdf)
 | 
						|
            ? info.Title
 | 
						|
            : Parser.RemoveExtensionIfSupported(Range);
 | 
						|
 | 
						|
        var specialTreatment = info.IsSpecialInfo();
 | 
						|
        Range = specialTreatment ? info.Filename : info.Chapters;
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Returns the Chapter Number. If the chapter is a range, returns that, formatted.
 | 
						|
    /// </summary>
 | 
						|
    /// <returns></returns>
 | 
						|
    public string GetNumberTitle()
 | 
						|
    {
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (MinNumber.Is(MaxNumber))
 | 
						|
            {
 | 
						|
                if (MinNumber.Is(Parser.DefaultChapterNumber) && IsSpecial)
 | 
						|
                {
 | 
						|
                    return Parser.RemoveExtensionIfSupported(Title);
 | 
						|
                }
 | 
						|
 | 
						|
                if (MinNumber.Is(0f) && !float.TryParse(Range, CultureInfo.InvariantCulture, out _))
 | 
						|
                {
 | 
						|
                    return $"{Range.ToString(CultureInfo.InvariantCulture)}";
 | 
						|
                }
 | 
						|
 | 
						|
                return $"{MinNumber.ToString(CultureInfo.InvariantCulture)}";
 | 
						|
 | 
						|
            }
 | 
						|
 | 
						|
            return $"{MinNumber.ToString(CultureInfo.InvariantCulture)}-{MaxNumber.ToString(CultureInfo.InvariantCulture)}";
 | 
						|
        }
 | 
						|
        catch (Exception)
 | 
						|
        {
 | 
						|
            return MinNumber.ToString(CultureInfo.InvariantCulture);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Is the Chapter representing a single Volume (volume 1.cbz). If so, Min/Max will be Default and will not be special
 | 
						|
    /// </summary>
 | 
						|
    /// <returns></returns>
 | 
						|
    public bool IsSingleVolumeChapter()
 | 
						|
    {
 | 
						|
        return MinNumber.Is(Parser.DefaultChapterNumber) && !IsSpecial;
 | 
						|
    }
 | 
						|
 | 
						|
    public void ResetColorScape()
 | 
						|
    {
 | 
						|
        PrimaryColor = string.Empty;
 | 
						|
        SecondaryColor = string.Empty;
 | 
						|
    }
 | 
						|
 | 
						|
    public bool IsPersonRoleLocked(PersonRole role)
 | 
						|
    {
 | 
						|
        return role switch
 | 
						|
        {
 | 
						|
            PersonRole.Character => CharacterLocked,
 | 
						|
            PersonRole.Writer => WriterLocked,
 | 
						|
            PersonRole.Penciller => PencillerLocked,
 | 
						|
            PersonRole.Inker => InkerLocked,
 | 
						|
            PersonRole.Colorist => ColoristLocked,
 | 
						|
            PersonRole.Letterer => LettererLocked,
 | 
						|
            PersonRole.CoverArtist => CoverArtistLocked,
 | 
						|
            PersonRole.Editor => EditorLocked,
 | 
						|
            PersonRole.Publisher => PublisherLocked,
 | 
						|
            PersonRole.Translator => TranslatorLocked,
 | 
						|
            PersonRole.Imprint => ImprintLocked,
 | 
						|
            PersonRole.Team => TeamLocked,
 | 
						|
            PersonRole.Location => LocationLocked,
 | 
						|
            _ => throw new ArgumentOutOfRangeException(nameof(role), role, null)
 | 
						|
        };
 | 
						|
    }
 | 
						|
}
 |