mirror of
https://github.com/Kareadita/Kavita.git
synced 2025-05-24 00:52:23 -04:00
176 lines
6.2 KiB
C#
176 lines
6.2 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using API.Entities.Enums;
|
|
using API.Entities.Interfaces;
|
|
using API.Entities.Metadata;
|
|
|
|
namespace API.Entities;
|
|
|
|
public class Series : IEntityDate, IHasReadTimeEstimate, IHasCoverImage
|
|
{
|
|
public int Id { get; set; }
|
|
/// <summary>
|
|
/// The UI visible Name of the Series. This may or may not be the same as the OriginalName
|
|
/// </summary>
|
|
public required string Name { get; set; }
|
|
/// <summary>
|
|
/// Used internally for name matching. <see cref="Services.Tasks.Scanner.Parser.Parser.Normalize"/>
|
|
/// </summary>
|
|
public required string NormalizedName { get; set; }
|
|
/// <summary>
|
|
/// Used internally for localized name matching. <see cref="Services.Tasks.Scanner.Parser.Parser.Normalize"/>
|
|
/// </summary>
|
|
public required string NormalizedLocalizedName { get; set; }
|
|
/// <summary>
|
|
/// The name used to sort the Series. By default, will be the same as Name.
|
|
/// </summary>
|
|
public required string SortName { get; set; }
|
|
/// <summary>
|
|
/// Name in original language (Japanese for Manga). By default, will be same as Name.
|
|
/// </summary>
|
|
public required string LocalizedName { get; set; }
|
|
/// <summary>
|
|
/// Original Name on disk. Not exposed to UI.
|
|
/// </summary>
|
|
public required string OriginalName { get; set; }
|
|
/// <summary>
|
|
/// Time of creation
|
|
/// </summary>
|
|
public DateTime Created { get; set; }
|
|
/// <summary>
|
|
/// Whenever a modification occurs. ex: New volumes, removed volumes, title update, etc
|
|
/// </summary>
|
|
public DateTime LastModified { get; set; }
|
|
|
|
public DateTime CreatedUtc { get; set; }
|
|
public DateTime LastModifiedUtc { get; set; }
|
|
|
|
/// <summary>
|
|
/// Absolute path to the (managed) image file
|
|
/// </summary>
|
|
/// <remarks>The file is managed internally to Kavita's APPDIR</remarks>
|
|
public string? CoverImage { get; set; }
|
|
/// <summary>
|
|
/// Denotes if the CoverImage has been overridden by the user. If so, it will not be updated during normal scan operations.
|
|
/// </summary>
|
|
public bool CoverImageLocked { get; set; }
|
|
/// <summary>
|
|
/// Sum of all Volume page counts
|
|
/// </summary>
|
|
public int Pages { get; set; }
|
|
/// <summary>
|
|
/// Highest path (that is under library root) that contains the series.
|
|
/// </summary>
|
|
/// <remarks><see cref="Services.Tasks.Scanner.Parser.Parser.NormalizePath"/> must be used before setting</remarks>
|
|
public string? FolderPath { get; set; }
|
|
/// <summary>
|
|
/// Lowest path (that is under library root) that contains all files for the series.
|
|
/// </summary>
|
|
/// <remarks><see cref="Services.Tasks.Scanner.Parser.Parser.NormalizePath"/> must be used before setting</remarks>
|
|
public string? LowestFolderPath { get; set; }
|
|
/// <summary>
|
|
/// Last time the folder was scanned
|
|
/// </summary>
|
|
public DateTime LastFolderScanned { get; set; }
|
|
/// <summary>
|
|
/// Last time the folder was scanned in Utc
|
|
/// </summary>
|
|
public DateTime LastFolderScannedUtc { get; set; }
|
|
/// <summary>
|
|
/// The type of all the files attached to this series
|
|
/// </summary>
|
|
public MangaFormat Format { get; set; } = MangaFormat.Unknown;
|
|
|
|
public string PrimaryColor { get; set; } = string.Empty;
|
|
public string SecondaryColor { get; set; } = string.Empty;
|
|
|
|
public bool SortNameLocked { get; set; }
|
|
public bool LocalizedNameLocked { get; set; }
|
|
|
|
/// <summary>
|
|
/// When a Chapter was last added onto the Series
|
|
/// </summary>
|
|
public DateTime LastChapterAdded { get; set; }
|
|
public DateTime LastChapterAddedUtc { get; set; }
|
|
|
|
/// <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; }
|
|
|
|
public int MinHoursToRead { get; set; }
|
|
public int MaxHoursToRead { get; set; }
|
|
public float AvgHoursToRead { get; set; }
|
|
|
|
#region KavitaPlus
|
|
/// <summary>
|
|
/// Do not match the series with any external Metadata service. This will automatically opt it out of scrobbling.
|
|
/// </summary>
|
|
public bool DontMatch { get; set; }
|
|
/// <summary>
|
|
/// If the series was unable to match, it will be blacklisted until a manual metadata match overrides it
|
|
/// </summary>
|
|
public bool IsBlacklisted { get; set; }
|
|
#endregion
|
|
|
|
public SeriesMetadata Metadata { get; set; } = null!;
|
|
public ExternalSeriesMetadata ExternalSeriesMetadata { get; set; } = null!;
|
|
|
|
public ICollection<AppUserRating> Ratings { get; set; } = null!;
|
|
public ICollection<AppUserProgress> Progress { get; set; } = null!;
|
|
public ICollection<AppUserCollection> Collections { get; set; } = null!;
|
|
|
|
/// <summary>
|
|
/// Relations to other Series, like Sequels, Prequels, etc
|
|
/// </summary>
|
|
/// <remarks>1 to Many relationship</remarks>
|
|
public ICollection<SeriesRelation> Relations { get; set; } = null!;
|
|
public ICollection<SeriesRelation> RelationOf { get; set; } = null!;
|
|
|
|
|
|
|
|
|
|
// Relationships
|
|
public List<Volume> Volumes { get; set; } = null!;
|
|
public Library Library { get; set; } = null!;
|
|
public int LibraryId { get; set; }
|
|
|
|
|
|
public void UpdateLastFolderScanned()
|
|
{
|
|
LastFolderScanned = DateTime.Now;
|
|
LastFolderScannedUtc = DateTime.UtcNow;
|
|
}
|
|
|
|
public void UpdateLastChapterAdded()
|
|
{
|
|
LastChapterAdded = DateTime.Now;
|
|
LastChapterAddedUtc = DateTime.UtcNow;
|
|
}
|
|
|
|
public bool MatchesSeriesByName(string nameNormalized, string localizedNameNormalized)
|
|
{
|
|
return NormalizedName == nameNormalized ||
|
|
NormalizedLocalizedName == nameNormalized ||
|
|
NormalizedName == localizedNameNormalized ||
|
|
NormalizedLocalizedName == localizedNameNormalized;
|
|
}
|
|
|
|
public void ResetColorScape()
|
|
{
|
|
PrimaryColor = string.Empty;
|
|
SecondaryColor = string.Empty;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Is this Series capable of Scrobbling
|
|
/// </summary>
|
|
/// <remarks>This includes if there is no Match/Manual Match needed, the series is blacklisted, or has a NoMatch</remarks>
|
|
/// <returns></returns>
|
|
public bool WillScrobble()
|
|
{
|
|
return !IsBlacklisted && !DontMatch;
|
|
}
|
|
}
|