music-kraken-core/music_kraken/objects/song.py

755 lines
24 KiB
Python
Raw Permalink Normal View History

2023-12-19 12:58:39 +00:00
from __future__ import annotations
2024-06-05 10:05:38 +00:00
import copy
2023-04-16 12:36:33 +00:00
import random
2023-04-16 15:39:53 +00:00
from collections import defaultdict
2024-06-05 10:05:38 +00:00
from typing import Dict, List, Optional, Tuple, Type, Union
2023-04-04 20:10:35 +00:00
2023-01-13 13:37:15 +00:00
import pycountry
2022-12-06 13:45:18 +00:00
2024-06-05 10:05:38 +00:00
from ..utils.config import main_settings
from ..utils.enums.album import AlbumStatus, AlbumType
from ..utils.enums.colors import BColors
from ..utils.shared import DEBUG_PRINT_ID
from ..utils.string_processing import unify
from .artwork import ArtworkCollection
from .collection import AppendHookArguments, Collection
from .contact import Contact
from .country import Country, Language
2023-04-04 20:10:35 +00:00
from .formatted_text import FormattedText
from .lyrics import Lyrics
2024-06-05 10:05:38 +00:00
from .metadata import ID3Timestamp
from .metadata import Mapping as id3Mapping
from .metadata import Metadata
2023-03-10 20:28:13 +00:00
from .option import Options
2024-06-05 10:05:38 +00:00
from .parents import OuterProxy
from .parents import OuterProxy as Base
from .parents import P
2023-04-04 20:10:35 +00:00
from .source import Source, SourceCollection
from .target import Target
2022-12-06 13:45:18 +00:00
2022-12-07 14:51:38 +00:00
"""
All Objects dependent
"""
2023-02-22 16:56:52 +00:00
CountryTyping = type(list(pycountry.countries)[0])
2024-04-25 21:20:07 +00:00
OPTION_BACKGROUND = BColors.GREY
OPTION_FOREGROUND = BColors.OKBLUE
def get_collection_string(
collection: Collection[Base],
template: str,
ignore_titles: Set[str] = None,
2024-04-25 21:20:07 +00:00
background: BColors = OPTION_BACKGROUND,
2024-04-29 16:18:57 +00:00
foreground: BColors = OPTION_FOREGROUND,
2024-05-03 12:52:12 +00:00
add_id: bool = DEBUG_PRINT_ID,
2024-04-25 21:20:07 +00:00
) -> str:
if collection.empty:
return ""
foreground = foreground.value
background = background.value
ignore_titles = ignore_titles or set()
2024-04-25 21:20:07 +00:00
r = background
2024-04-29 16:18:57 +00:00
def get_element_str(element) -> str:
nonlocal add_id
r = element.title_string.strip()
2024-05-06 12:33:03 +00:00
if add_id and False:
2024-04-29 16:18:57 +00:00
r += " " + str(element.id)
return r
2024-04-25 21:20:07 +00:00
element: Base
2024-04-29 16:18:57 +00:00
titel_list: List[str] = [get_element_str(element) for element in collection if element.title_string not in ignore_titles]
for i, titel in enumerate(titel_list):
2024-04-25 21:20:07 +00:00
delimiter = ", "
if i == len(collection) - 1:
delimiter = ""
elif i == len(collection) - 2:
delimiter = " and "
r += foreground + titel + BColors.ENDC.value + background + delimiter + BColors.ENDC.value
2024-04-25 21:20:07 +00:00
r += BColors.ENDC.value
return template.format(r)
2022-12-08 08:28:28 +00:00
2023-12-19 12:58:39 +00:00
class Song(Base):
title: str
unified_title: str
isrc: str
length: int
genre: str
note: FormattedText
2024-02-28 13:27:35 +00:00
tracksort: int
2024-06-05 10:05:38 +00:00
artwork: ArtworkCollection
2023-12-19 12:58:39 +00:00
source_collection: SourceCollection
target_collection: Collection[Target]
lyrics_collection: Collection[Lyrics]
2024-04-17 12:15:56 +00:00
artist_collection: Collection[Artist]
2023-12-19 12:58:39 +00:00
feature_artist_collection: Collection[Artist]
album_collection: Collection[Album]
_default_factories = {
"note": FormattedText,
"length": lambda: 0,
"source_collection": SourceCollection,
"target_collection": Collection,
"lyrics_collection": Collection,
2024-06-05 10:05:38 +00:00
"artwork": ArtworkCollection,
2023-12-19 12:58:39 +00:00
"album_collection": Collection,
"artist_collection": Collection,
2023-12-19 21:11:46 +00:00
"feature_artist_collection": Collection,
2023-12-20 08:55:09 +00:00
2024-05-08 19:06:40 +00:00
"title": lambda: None,
2023-12-20 08:55:09 +00:00
"unified_title": lambda: None,
"isrc": lambda: None,
"genre": lambda: None,
2024-02-28 13:27:35 +00:00
"tracksort": lambda: 0,
2023-12-19 12:58:39 +00:00
}
def __init__(
self,
title: str = None,
isrc: str = None,
length: int = None,
genre: str = None,
note: FormattedText = None,
source_list: List[Source] = None,
target_list: List[Target] = None,
lyrics_list: List[Lyrics] = None,
2024-05-16 12:29:50 +00:00
artist_list: List[Artist] = None,
feature_artist_list: List[Artist] = None,
album_list: List[Album] = None,
tracksort: int = 0,
2024-06-05 10:05:38 +00:00
artwork: Optional[ArtworkCollection] = None,
**kwargs
) -> None:
real_kwargs = copy.copy(locals())
real_kwargs.update(real_kwargs.pop("kwargs", {}))
Base.__init__(**real_kwargs)
2023-12-29 20:16:09 +00:00
UPWARDS_COLLECTION_STRING_ATTRIBUTES = ("artist_collection", "feature_artist_collection", "album_collection")
2024-04-10 08:25:05 +00:00
TITEL = "title"
2023-12-29 20:16:09 +00:00
@staticmethod
def register_artwork_parent(append_hook_arguments: AppendHookArguments):
album: Album = append_hook_arguments.new_object
song: Song
for song in append_hook_arguments.collection_root_objects:
song.artwork.parent_artworks.add(album.artwork)
2023-12-19 12:58:39 +00:00
def __init_collections__(self) -> None:
2024-05-16 12:29:50 +00:00
self.feature_artist_collection.push_to = [self.artist_collection]
self.artist_collection.pull_from = [self.feature_artist_collection]
self.album_collection.sync_on_append = {
"artist_collection": self.artist_collection,
}
2023-12-19 12:58:39 +00:00
self.album_collection.append_object_to_attribute = {
"song_collection": self,
}
self.artist_collection.extend_object_to_attribute = {
"album_collection": self.album_collection
}
2024-05-16 15:14:18 +00:00
self.feature_artist_collection.extend_object_to_attribute = {
"album_collection": self.album_collection
}
self.album_collection.append_callbacks = set((Song.register_artwork_parent, ))
2024-01-15 11:48:36 +00:00
def _add_other_db_objects(self, object_type: Type[OuterProxy], object_list: List[OuterProxy]):
if object_type is Song:
return
if isinstance(object_list, Lyrics):
self.lyrics_collection.extend(object_list)
return
if isinstance(object_list, Artist):
self.feature_artist_collection.extend(object_list)
2024-01-15 11:48:36 +00:00
return
if isinstance(object_list, Album):
self.album_collection.extend(object_list)
return
2024-06-06 15:53:17 +00:00
def _compile(self):
self.artwork.compile()
2024-06-06 15:53:17 +00:00
2024-04-29 16:18:57 +00:00
INDEX_DEPENDS_ON = ("title", "isrc", "source_collection")
2023-03-10 08:09:35 +00:00
@property
def indexing_values(self) -> List[Tuple[str, object]]:
return [
('title', unify(self.title)),
2023-03-10 17:38:32 +00:00
('isrc', self.isrc),
2024-04-29 16:18:57 +00:00
*self.source_collection.indexing_values(),
2023-03-10 08:09:35 +00:00
]
2023-03-10 09:54:15 +00:00
2023-03-10 08:09:35 +00:00
@property
def metadata(self) -> Metadata:
metadata = Metadata({
id3Mapping.TITLE: [self.title],
id3Mapping.ISRC: [self.isrc],
id3Mapping.LENGTH: [self.length],
id3Mapping.GENRE: [self.genre],
2024-04-10 07:28:28 +00:00
id3Mapping.TRACKNUMBER: [self.tracksort_str],
id3Mapping.COMMENT: [self.note.markdown],
2024-04-29 15:06:31 +00:00
id3Mapping.FILE_WEBPAGE_URL: self.source_collection.url_list,
id3Mapping.SOURCE_WEBPAGE_URL: self.source_collection.homepage_list,
2023-03-10 08:09:35 +00:00
})
# metadata.merge_many([s.get_song_metadata() for s in self.source_collection]) album sources have no relevant metadata for id3
2023-03-10 08:09:35 +00:00
metadata.merge_many([a.metadata for a in self.album_collection])
metadata.merge_many([a.metadata for a in self.artist_collection])
2023-03-10 08:09:35 +00:00
metadata.merge_many([a.metadata for a in self.feature_artist_collection])
metadata.merge_many([lyrics.metadata for lyrics in self.lyrics_collection])
return metadata
2022-12-13 10:16:19 +00:00
def get_artist_credits(self) -> str:
main_artists = ", ".join([artist.name for artist in self.artist_collection])
2023-02-10 12:52:18 +00:00
feature_artists = ", ".join([artist.name for artist in self.feature_artist_collection])
2023-02-22 16:56:52 +00:00
2023-02-10 12:52:18 +00:00
if len(feature_artists) == 0:
return main_artists
return f"{main_artists} feat. {feature_artists}"
2022-12-13 10:16:19 +00:00
2023-03-10 09:54:15 +00:00
@property
def option_string(self) -> str:
r = "song "
r += OPTION_FOREGROUND.value + self.title_string + BColors.ENDC.value + OPTION_BACKGROUND.value
r += get_collection_string(self.album_collection, " from {}", ignore_titles={self.title})
r += get_collection_string(self.artist_collection, " by {}")
2024-05-16 15:09:36 +00:00
r += get_collection_string(self.feature_artist_collection, " feat. {}" if len(self.artist_collection) > 0 else " by {}")
2024-04-11 11:58:06 +00:00
return r
2023-03-10 09:54:15 +00:00
2023-03-10 20:28:13 +00:00
@property
def tracksort_str(self) -> str:
"""
2023-03-10 20:33:26 +00:00
if the album tracklist is empty, it sets it length to 1, this song has to be on the Album
2023-03-10 20:28:13 +00:00
:returns id3_tracksort: {song_position}/{album.length_of_tracklist}
"""
2023-03-24 09:30:40 +00:00
if len(self.album_collection) == 0:
return f"{self.tracksort}"
2023-04-12 10:00:29 +00:00
2023-03-31 08:46:56 +00:00
return f"{self.tracksort}/{len(self.album_collection[0].song_collection) or 1}"
2023-02-09 08:40:57 +00:00
2022-12-06 13:45:18 +00:00
2023-12-19 12:58:39 +00:00
class Album(Base):
title: str
unified_title: str
2023-12-19 15:10:02 +00:00
album_status: AlbumStatus
2023-12-19 12:58:39 +00:00
album_type: AlbumType
2023-12-19 15:10:02 +00:00
language: Language
date: ID3Timestamp
barcode: str
albumsort: int
notes: FormattedText
2024-06-05 10:05:38 +00:00
artwork: ArtworkCollection
2023-12-19 15:10:02 +00:00
source_collection: SourceCollection
2024-04-19 15:45:49 +00:00
2023-12-19 15:10:02 +00:00
song_collection: Collection[Song]
artist_collection: Collection[Artist]
feature_artist_collection: Collection[Artist]
2023-12-19 15:10:02 +00:00
label_collection: Collection[Label]
2023-12-19 12:58:39 +00:00
_default_factories = {
2023-12-29 20:50:40 +00:00
"title": lambda: None,
2023-12-29 20:16:09 +00:00
"unified_title": lambda: None,
"album_status": lambda: None,
"barcode": lambda: None,
"albumsort": lambda: None,
2023-12-19 15:10:02 +00:00
"album_type": lambda: AlbumType.OTHER,
"language": lambda: Language.by_alpha_2("en"),
"date": ID3Timestamp,
"notes": FormattedText,
2023-12-19 21:11:46 +00:00
2024-06-05 10:05:38 +00:00
"artwork": lambda: ArtworkCollection(crop_images=False),
2023-12-19 21:11:46 +00:00
"source_collection": SourceCollection,
2023-12-19 21:11:46 +00:00
"song_collection": Collection,
"artist_collection": Collection,
"feature_artist_collection": Collection,
2023-12-19 21:11:46 +00:00
"label_collection": Collection,
2023-12-19 12:58:39 +00:00
}
2024-04-10 08:25:05 +00:00
TITEL = "title"
2023-12-29 20:16:09 +00:00
# This is automatically generated
def __init__(
self,
title: str = None,
unified_title: str = None,
album_status: AlbumStatus = None,
album_type: AlbumType = None,
language: Language = None,
date: ID3Timestamp = None,
barcode: str = None,
albumsort: int = None,
notes: FormattedText = None,
2024-06-05 10:05:38 +00:00
artwork: ArtworkCollection = None,
source_list: List[Source] = None,
artist_list: List[Artist] = None,
song_list: List[Song] = None,
label_list: List[Label] = None,
**kwargs
) -> None:
real_kwargs = copy.copy(locals())
real_kwargs.update(real_kwargs.pop("kwargs", {}))
Base.__init__(**real_kwargs)
2023-12-29 20:16:09 +00:00
2024-01-15 09:56:59 +00:00
DOWNWARDS_COLLECTION_STRING_ATTRIBUTES = ("song_collection",)
UPWARDS_COLLECTION_STRING_ATTRIBUTES = ("label_collection", "artist_collection")
2023-09-14 21:35:37 +00:00
@staticmethod
def register_artwork_parent(append_hook_arguments: AppendHookArguments):
song: Song = append_hook_arguments.new_object
for root_object in append_hook_arguments.collection_root_objects:
song.artwork.parent_artworks.add(root_object.artwork)
2023-12-19 15:10:02 +00:00
def __init_collections__(self):
2024-05-16 12:29:50 +00:00
self.feature_artist_collection.push_to = [self.artist_collection]
self.artist_collection.pull_from = [self.feature_artist_collection]
self.song_collection.append_object_to_attribute = {
"album_collection": self
}
self.song_collection.sync_on_append = {
"artist_collection": self.artist_collection
2024-04-18 13:43:01 +00:00
}
2023-02-23 22:52:41 +00:00
2024-04-17 12:15:56 +00:00
self.artist_collection.append_object_to_attribute = {
"album_collection": self
2024-04-17 12:15:56 +00:00
}
self.artist_collection.extend_object_to_attribute = {
2024-04-17 12:15:56 +00:00
"label_collection": self.label_collection
}
self.song_collection.append_callbacks = set((Album.register_artwork_parent, ))
2024-01-15 11:48:36 +00:00
def _add_other_db_objects(self, object_type: Type[OuterProxy], object_list: List[OuterProxy]):
if object_type is Song:
self.song_collection.extend(object_list)
return
if object_type is Artist:
self.artist_collection.extend(object_list)
return
if object_type is Album:
return
if object_type is Label:
self.label_collection.extend(object_list)
return
2024-04-29 16:18:57 +00:00
INDEX_DEPENDS_ON = ("title", "barcode", "source_collection")
@property
def indexing_values(self) -> List[Tuple[str, object]]:
return [
2024-04-12 12:14:10 +00:00
('title', unify(self.title)),
('barcode', self.barcode),
2024-04-29 16:18:57 +00:00
*self.source_collection.indexing_values(),
]
2023-03-10 09:54:15 +00:00
2023-03-10 08:09:35 +00:00
@property
def metadata(self) -> Metadata:
2023-12-19 15:10:02 +00:00
"""
TODO
- barcode
:return:
"""
2023-03-10 08:09:35 +00:00
return Metadata({
id3Mapping.ALBUM: [self.title],
id3Mapping.COPYRIGHT: [self.copyright],
2023-03-31 08:46:56 +00:00
id3Mapping.LANGUAGE: [self.iso_639_2_lang],
2023-03-10 08:09:35 +00:00
id3Mapping.ALBUM_ARTIST: [a.name for a in self.artist_collection],
2023-06-20 17:30:48 +00:00
id3Mapping.DATE: [self.date.strftime("%d%m")] if self.date.has_year and self.date.has_month else [],
id3Mapping.TIME: [self.date.strftime(("%H%M"))] if self.date.has_hour and self.date.has_minute else [],
id3Mapping.YEAR: [str(self.date.year).zfill(4)] if self.date.has_year else [],
id3Mapping.RELEASE_DATE: [self.date.timestamp],
id3Mapping.ORIGINAL_RELEASE_DATE: [self.date.timestamp],
2023-04-16 12:36:33 +00:00
id3Mapping.ALBUMSORTORDER: [str(self.albumsort)] if self.albumsort is not None else []
2023-03-10 08:09:35 +00:00
})
2023-03-10 09:54:15 +00:00
@property
def option_string(self) -> str:
r = "album "
r += OPTION_FOREGROUND.value + self.title_string + BColors.ENDC.value + OPTION_BACKGROUND.value
2024-04-25 21:20:07 +00:00
r += get_collection_string(self.artist_collection, " by {}")
if len(self.artist_collection) <= 0:
r += get_collection_string(self.feature_artist_collection, " by {}")
2024-04-25 21:20:07 +00:00
r += get_collection_string(self.label_collection, " under {}")
2023-03-10 09:54:15 +00:00
2024-04-25 21:20:07 +00:00
if len(self.song_collection) > 0:
r += f" with {len(self.song_collection)} songs"
return r
2023-03-10 20:28:13 +00:00
def _compile(self):
self.analyze_implied_album_type()
self.update_tracksort()
self.fix_artist_collection()
def analyze_implied_album_type(self):
# if the song collection has only one song, it is reasonable to assume that it is a single
if len(self.song_collection) == 1:
self.album_type = AlbumType.SINGLE
return
# if the album already has an album type, we don't need to do anything
if self.album_type is not AlbumType.OTHER:
return
# for information on EP's I looked at https://www.reddit.com/r/WeAreTheMusicMakers/comments/a354ql/whats_the_cutoff_length_between_ep_and_album/
if len(self.song_collection) < 9:
self.album_type = AlbumType.EP
return
2023-02-23 22:52:41 +00:00
def update_tracksort(self):
"""
This updates the tracksort attributes, of the songs in
`self.song_collection`, and sorts the songs, if possible.
2022-12-09 17:24:58 +00:00
2023-02-23 22:52:41 +00:00
It is advised to only call this function, once all the tracks are
added to the songs.
2022-12-07 14:51:38 +00:00
2023-02-23 22:52:41 +00:00
:return:
"""
2023-04-03 16:09:05 +00:00
if self.song_collection.empty:
return
tracksort_map: Dict[int, Song] = {
2024-04-11 13:44:59 +00:00
song.tracksort: song for song in self.song_collection if song.tracksort != 0
}
2023-02-23 22:52:41 +00:00
2024-04-11 13:44:59 +00:00
existing_list = self.song_collection.shallow_list
for i in range(1, len(existing_list) + 1):
if i not in tracksort_map:
tracksort_map[i] = existing_list.pop(0)
tracksort_map[i].tracksort = i
def fix_artist_collection(self):
"""
I add artists, that could only be feature artists to the feature artist collection.
They get automatically moved to main artist collection, if a matching artist exists in the main artist collection or is appended to it later on.
If I am not sure for any artist, I try to analyze the most common artist in the song collection of one album.
"""
# move all artists that are in all feature_artist_collections, of every song, to the artist_collection
pass
2023-03-10 08:09:35 +00:00
@property
def copyright(self) -> str:
2023-01-30 22:54:21 +00:00
if self.date is None:
2023-02-22 16:56:52 +00:00
return ""
2023-02-23 22:52:41 +00:00
if self.date.has_year or len(self.label_collection) == 0:
2023-02-22 16:56:52 +00:00
return ""
2023-01-13 13:37:15 +00:00
2023-02-23 22:52:41 +00:00
return f"{self.date.year} {self.label_collection[0].name}"
2023-01-13 13:37:15 +00:00
2023-02-27 15:51:55 +00:00
@property
def iso_639_2_lang(self) -> Optional[str]:
2023-01-13 13:37:15 +00:00
if self.language is None:
return None
return self.language.alpha_3
2023-02-28 00:13:53 +00:00
@property
def is_split(self) -> bool:
"""
A split Album is an Album from more than one Artists
usually half the songs are made by one Artist, the other half by the other one.
In this case split means either that or one artist featured by all songs.
:return:
"""
return len(self.artist_collection) > 1
2024-01-15 09:56:59 +00:00
2023-06-15 09:28:35 +00:00
@property
def album_type_string(self) -> str:
return self.album_type.value
2023-02-28 00:13:53 +00:00
2023-03-10 09:54:15 +00:00
2023-12-19 12:58:39 +00:00
class Artist(Base):
2023-12-19 21:11:46 +00:00
name: str
country: Country
formed_in: ID3Timestamp
notes: FormattedText
lyrical_themes: List[str]
2023-09-14 21:35:37 +00:00
2023-12-19 21:11:46 +00:00
general_genre: str
2024-04-29 11:49:16 +00:00
unformatted_location: str
2023-05-24 23:27:05 +00:00
2024-06-05 10:05:38 +00:00
artwork: ArtworkCollection
2024-06-04 05:58:18 +00:00
2023-12-19 21:11:46 +00:00
source_collection: SourceCollection
contact_collection: Collection[Contact]
2023-02-25 21:16:32 +00:00
album_collection: Collection[Album]
2023-12-19 21:11:46 +00:00
label_collection: Collection[Label]
2022-12-12 18:30:18 +00:00
2023-12-19 21:11:46 +00:00
_default_factories = {
"name": lambda: None,
2023-12-29 20:16:09 +00:00
"country": lambda: None,
2024-04-29 11:49:16 +00:00
"unformatted_location": lambda: None,
2023-12-29 20:16:09 +00:00
2023-12-19 21:11:46 +00:00
"formed_in": ID3Timestamp,
"notes": FormattedText,
"lyrical_themes": list,
"general_genre": lambda: "",
2023-12-20 08:55:09 +00:00
2024-06-05 10:05:38 +00:00
"artwork": ArtworkCollection,
2024-06-04 05:58:18 +00:00
2023-12-19 21:11:46 +00:00
"source_collection": SourceCollection,
"album_collection": Collection,
2023-12-19 21:11:46 +00:00
"contact_collection": Collection,
"label_collection": Collection,
}
2023-08-30 19:14:03 +00:00
2024-04-10 08:25:05 +00:00
TITEL = "name"
2023-12-29 20:16:09 +00:00
# This is automatically generated
def __init__(
self,
name: str = None,
unified_name: str = None,
country: Country = None,
formed_in: ID3Timestamp = None,
notes: FormattedText = None,
lyrical_themes: List[str] = None,
general_genre: str = None,
2024-06-05 10:05:38 +00:00
artwork: ArtworkCollection = None,
unformatted_location: str = None,
source_list: List[Source] = None,
contact_list: List[Contact] = None,
feature_song_list: List[Song] = None,
2024-05-16 12:29:50 +00:00
album_list: List[Album] = None,
label_list: List[Label] = None,
**kwargs
) -> None:
real_kwargs = copy.copy(locals())
real_kwargs.update(real_kwargs.pop("kwargs", {}))
Base.__init__(**real_kwargs)
2023-12-29 20:16:09 +00:00
DOWNWARDS_COLLECTION_STRING_ATTRIBUTES = ("album_collection",)
2024-01-15 09:56:59 +00:00
UPWARDS_COLLECTION_STRING_ATTRIBUTES = ("label_collection",)
2023-01-24 08:40:01 +00:00
2023-12-19 21:11:46 +00:00
def __init_collections__(self):
2024-05-16 15:09:36 +00:00
self.album_collection.append_object_to_attribute = {
"feature_artist_collection": self
}
2023-12-19 21:11:46 +00:00
self.label_collection.append_object_to_attribute = {
"current_artist_collection": self
}
2023-04-16 12:36:33 +00:00
2024-01-15 11:48:36 +00:00
def _add_other_db_objects(self, object_type: Type[OuterProxy], object_list: List[OuterProxy]):
if object_type is Song:
# this doesn't really make sense
return
if object_type is Artist:
return
if object_type is Album:
self.album_collection.extend(object_list)
2024-01-15 11:48:36 +00:00
return
if object_type is Label:
self.label_collection.extend(object_list)
return
def _compile(self):
self.update_albumsort()
2023-04-16 12:36:33 +00:00
def update_albumsort(self):
"""
This updates the albumsort attributes, of the albums in
`self.album_collection`, and sorts the albums, if possible.
2023-04-16 12:36:33 +00:00
It is advised to only call this function, once all the albums are
added to the artist.
:return:
"""
2023-12-19 21:11:46 +00:00
type_section: Dict[AlbumType, int] = defaultdict(lambda: 2, {
2023-04-16 15:39:53 +00:00
AlbumType.OTHER: 0, # if I don't know it, I add it to the first section
AlbumType.STUDIO_ALBUM: 0,
AlbumType.EP: 0,
AlbumType.SINGLE: 1
2023-09-10 14:27:09 +00:00
}) if main_settings["sort_album_by_type"] else defaultdict(lambda: 0)
2023-04-16 15:39:53 +00:00
sections = defaultdict(list)
# order albums in the previously defined section
album: Album
for album in self.album_collection:
2023-04-16 15:39:53 +00:00
sections[type_section[album.album_type]].append(album)
def sort_section(_section: List[Album], last_albumsort: int) -> int:
# album is just a value used in loops
nonlocal album
2023-09-10 14:27:09 +00:00
if main_settings["sort_by_date"]:
2023-04-16 15:39:53 +00:00
_section.sort(key=lambda _album: _album.date, reverse=True)
new_last_albumsort = last_albumsort
for album_index, album in enumerate(_section):
if album.albumsort is None:
album.albumsort = new_last_albumsort = album_index + 1 + last_albumsort
_section.sort(key=lambda _album: _album.albumsort)
return new_last_albumsort
# sort the sections individually
_last_albumsort = 1
for section_index in sorted(sections):
_last_albumsort = sort_section(sections[section_index], _last_albumsort)
# merge all sections again
album_list = []
for section_index in sorted(sections):
album_list.extend(sections[section_index])
# replace the old collection with the new one
self.album_collection._data = album_list
2023-04-16 12:36:33 +00:00
2024-04-29 16:18:57 +00:00
INDEX_DEPENDS_ON = ("name", "source_collection", "contact_collection")
@property
def indexing_values(self) -> List[Tuple[str, object]]:
return [
2024-04-12 12:14:10 +00:00
('name', unify(self.name)),
2024-04-29 16:18:57 +00:00
*[('contact', contact.value) for contact in self.contact_collection],
*self.source_collection.indexing_values(),
]
2023-03-10 09:54:15 +00:00
2023-03-10 08:09:35 +00:00
@property
def metadata(self) -> Metadata:
metadata = Metadata({
2024-04-29 15:06:31 +00:00
id3Mapping.ARTIST: [self.name],
id3Mapping.ARTIST_WEBPAGE_URL: self.source_collection.url_list,
2023-03-10 08:09:35 +00:00
})
return metadata
2023-03-10 09:54:15 +00:00
@property
def option_string(self) -> str:
r = "artist "
r += OPTION_FOREGROUND.value + self.title_string + BColors.ENDC.value + OPTION_BACKGROUND.value
2024-04-25 21:20:07 +00:00
r += get_collection_string(self.label_collection, " under {}")
r += OPTION_BACKGROUND.value
if len(self.album_collection) > 0:
r += f" with {len(self.album_collection)} albums"
2024-04-25 21:20:07 +00:00
r += BColors.ENDC.value
return r
2023-03-10 09:54:15 +00:00
2023-02-09 08:40:57 +00:00
2023-12-19 12:58:39 +00:00
class Label(Base):
2023-09-14 21:35:37 +00:00
COLLECTION_STRING_ATTRIBUTES = ("album_collection", "current_artist_collection")
2023-12-19 21:11:46 +00:00
2023-09-14 21:35:37 +00:00
DOWNWARDS_COLLECTION_STRING_ATTRIBUTES = COLLECTION_STRING_ATTRIBUTES
2023-12-19 21:11:46 +00:00
name: str
unified_name: str
notes: FormattedText
source_collection: SourceCollection
contact_collection: Collection[Contact]
2023-09-14 21:35:37 +00:00
2023-12-19 21:11:46 +00:00
album_collection: Collection[Album]
current_artist_collection: Collection[Artist]
_default_factories = {
"notes": FormattedText,
"album_collection": Collection,
"current_artist_collection": Collection,
"source_collection": SourceCollection,
2023-12-20 08:55:09 +00:00
"contact_collection": Collection,
2023-12-29 20:50:40 +00:00
"name": lambda: None,
2023-12-20 08:55:09 +00:00
"unified_name": lambda: None,
2023-12-19 21:11:46 +00:00
}
2023-05-24 23:27:05 +00:00
2024-04-10 08:25:05 +00:00
TITEL = "name"
def __init__(
self,
name: str = None,
unified_name: str = None,
notes: FormattedText = None,
source_list: List[Source] = None,
contact_list: List[Contact] = None,
album_list: List[Album] = None,
current_artist_list: List[Artist] = None,
**kwargs
) -> None:
real_kwargs = copy.copy(locals())
real_kwargs.update(real_kwargs.pop("kwargs", {}))
Base.__init__(**real_kwargs)
2023-02-23 22:52:41 +00:00
2024-04-17 12:15:56 +00:00
def __init_collections__(self):
self.album_collection.append_object_to_attribute = {
"label_collection": self
}
self.current_artist_collection.append_object_to_attribute = {
"label_collection": self
}
@property
def indexing_values(self) -> List[Tuple[str, object]]:
return [
2024-04-12 12:14:10 +00:00
('name', unify(self.name)),
2023-03-10 09:21:57 +00:00
*[('url', source.url) for source in self.source_collection]
]
2023-03-10 20:28:13 +00:00
2024-01-15 11:48:36 +00:00
def _add_other_db_objects(self, object_type: Type[OuterProxy], object_list: List[OuterProxy]):
if object_type is Song:
return
if object_type is Artist:
self.current_artist_collection.extend(object_list)
return
if object_type is Album:
self.album_collection.extend(object_list)
return
2023-03-10 20:28:13 +00:00
@property
2023-12-19 21:11:46 +00:00
def options(self) -> List[P]:
2023-03-10 20:28:13 +00:00
options = [self]
options.extend(self.current_artist_collection.shallow_list)
2023-03-20 22:11:55 +00:00
options.extend(self.album_collection.shallow_list)
2024-01-15 09:56:59 +00:00
2023-06-12 12:56:14 +00:00
return options
2024-04-10 11:47:35 +00:00
@property
def option_string(self):
return "label " + OPTION_FOREGROUND.value + self.name + BColors.ENDC.value