using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Linq; using API.Data; using API.DTOs.Metadata; using API.Entities; using API.Extensions; namespace API.Helpers; public static class TagHelper { /// /// /// /// /// /// Callback for every item. Will give said item back and a bool if item was added public static void UpdateTag(ICollection allTags, IEnumerable names, Action action) { foreach (var name in names) { if (string.IsNullOrEmpty(name.Trim())) continue; var added = false; var normalizedName = name.ToNormalized(); var genre = allTags.FirstOrDefault(p => p.NormalizedTitle.Equals(normalizedName)); if (genre == null) { added = true; genre = DbFactory.Tag(name); allTags.Add(genre); } action(genre, added); } } public static void KeepOnlySameTagBetweenLists(ICollection existingTags, ICollection removeAllExcept, Action? action = null) { var existing = existingTags.ToList(); foreach (var genre in existing) { var existingPerson = removeAllExcept.FirstOrDefault(g => genre.NormalizedTitle.Equals(g.NormalizedTitle)); if (existingPerson != null) continue; existingTags.Remove(genre); action?.Invoke(genre); } } /// /// Adds the tag to the list if it's not already in there. This will ignore the ExternalTag. /// /// /// public static void AddTagIfNotExists(ICollection metadataTags, Tag tag) { var existingGenre = metadataTags.FirstOrDefault(p => p.NormalizedTitle == tag.Title.ToNormalized()); if (existingGenre == null) { metadataTags.Add(tag); } } public static void AddTagIfNotExists(BlockingCollection metadataTags, Tag tag) { var existingGenre = metadataTags.FirstOrDefault(p => p.NormalizedTitle == tag.Title.ToNormalized()); if (existingGenre == null) { metadataTags.Add(tag); } } /// /// Remove tags on a list /// /// Used to remove before we update/add new tags /// Existing tags on Entity /// Tags from metadata /// Callback which will be executed for each tag removed public static void RemoveTags(ICollection existingTags, IEnumerable tags, Action? action = null) { var normalizedTags = tags.Select(Services.Tasks.Scanner.Parser.Parser.Normalize).ToList(); foreach (var person in normalizedTags) { var existingTag = existingTags.FirstOrDefault(p => person.Equals(p.NormalizedTitle)); if (existingTag == null) continue; existingTags.Remove(existingTag); action?.Invoke(existingTag); } } public static void UpdateTagList(ICollection? tags, Series series, IReadOnlyCollection allTags, Action handleAdd, Action onModified) { if (tags == null) return; var isModified = false; // I want a union of these 2 lists. Return only elements that are in both lists, but the list types are different var existingTags = series.Metadata.Tags.ToList(); foreach (var existing in existingTags.Where(existing => tags.SingleOrDefault(t => t.Id == existing.Id) == null)) { // Remove tag series.Metadata.Tags.Remove(existing); isModified = true; } // At this point, all tags that aren't in dto have been removed. foreach (var tagTitle in tags.Select(t => t.Title)) { var normalizedTitle = tagTitle.ToNormalized(); var existingTag = allTags.SingleOrDefault(t => t.NormalizedTitle.Equals(normalizedTitle)); if (existingTag != null) { if (series.Metadata.Tags.All(t => t.NormalizedTitle != normalizedTitle)) { handleAdd(existingTag); isModified = true; } } else { // Add new tag handleAdd(DbFactory.Tag(tagTitle)); isModified = true; } } if (isModified) { onModified(); } } }