Plex-Meta-Manager/modules/overlays.py

693 lines
46 KiB
Python
Raw Normal View History

import os, re
2022-05-05 13:21:17 +00:00
from datetime import datetime
from modules import plex, util, overlay
2022-04-18 18:16:39 +00:00
from modules.builder import CollectionBuilder
2024-03-28 20:45:55 +00:00
from modules.util import Failed, FilterFailed, NotScheduled, LimitReached
2022-07-26 18:30:40 +00:00
from num2words import num2words
2022-04-18 18:16:39 +00:00
from plexapi.exceptions import BadRequest
2023-04-28 03:43:26 +00:00
from plexapi.video import Season, Episode
2022-05-23 18:42:27 +00:00
from PIL import Image, ImageFilter
2022-04-18 18:16:39 +00:00
logger = util.logger
class Overlays:
def __init__(self, config, library):
self.config = config
self.library = library
self.overlays = []
def run_overlays(self):
2022-05-05 13:21:17 +00:00
overlay_start = datetime.now()
2022-04-18 18:16:39 +00:00
logger.info("")
logger.separator(f"{self.library.name} Library Overlays")
logger.info("")
os.makedirs(self.library.overlay_backup, exist_ok=True)
2022-10-26 06:56:12 +00:00
key_to_overlays = {}
2023-03-10 20:32:37 +00:00
properties = {}
2022-10-26 06:56:12 +00:00
if not self.library.remove_overlays:
2022-10-28 23:32:48 +00:00
key_to_overlays, properties = self.compile_overlays()
2022-10-26 06:56:12 +00:00
ignore_list = [rk for rk in key_to_overlays]
2022-04-26 19:56:46 +00:00
old_overlays = [la for la in self.library.Plex.listFilterChoices("label") if str(la.title).lower().endswith(" overlay")]
if old_overlays:
logger.separator(f"Removing Old Overlays for the {self.library.name} Library")
2022-04-19 22:21:05 +00:00
logger.info("")
for old_overlay in old_overlays:
label_items = self.get_overlay_items(label=old_overlay)
if label_items:
logger.info("")
logger.separator(f"Removing {old_overlay.title}")
logger.info("")
for i, item in enumerate(label_items, 1):
2022-06-09 14:20:43 +00:00
item_title = self.library.get_item_sort_title(item, atr="title")
2022-04-26 15:28:04 +00:00
logger.ghost(f"Restoring {old_overlay.title}: {i}/{len(label_items)} {item_title}")
self.remove_overlay(item, item_title, old_overlay.title, [
os.path.join(self.library.overlay_folder, old_overlay.title[:-8], f"{item.ratingKey}.png")
])
2022-10-26 06:56:12 +00:00
logger.info("")
2022-04-18 18:16:39 +00:00
remove_overlays = self.get_overlay_items(ignore=ignore_list)
if self.library.is_show:
remove_overlays.extend(self.get_overlay_items(libtype="episode", ignore=ignore_list))
remove_overlays.extend(self.get_overlay_items(libtype="season", ignore=ignore_list))
elif self.library.is_music:
remove_overlays.extend(self.get_overlay_items(libtype="album", ignore=ignore_list))
if remove_overlays:
2022-10-26 06:56:12 +00:00
logger.separator(f"Removing {'All ' if self.library.remove_overlays else ''}Overlays for the {self.library.name} Library")
for i, item in enumerate(remove_overlays, 1):
2022-06-09 14:20:43 +00:00
item_title = self.library.get_item_sort_title(item, atr="title")
logger.ghost(f"Restoring: {i}/{len(remove_overlays)} {item_title}")
self.remove_overlay(item, item_title, "Overlay", [
os.path.join(self.library.overlay_backup, f"{item.ratingKey}.png"),
2024-05-01 16:06:19 +00:00
os.path.join(self.library.overlay_backup, f"{item.ratingKey}.jpg"),
os.path.join(self.library.overlay_backup, f"{item.ratingKey}.webp")
])
logger.exorcise()
else:
logger.separator(f"No Overlays to Remove for the {self.library.name} Library")
logger.info("")
if not self.library.remove_overlays:
2022-06-26 19:08:17 +00:00
logger.separator(f"{'Re-' if self.library.reapply_overlays else ''}Applying Overlays for the {self.library.name} Library")
logger.info("")
_trakt_ratings = None
def trakt_ratings():
nonlocal _trakt_ratings
if _trakt_ratings is None:
_trakt_ratings = self.config.Trakt.user_ratings(self.library.is_movie)
if not _trakt_ratings:
raise Failed
return _trakt_ratings
reverse_anidb = {}
for k, v in self.library.anidb_map.items():
reverse_anidb[v] = k
reverse_mal = {}
for k, v in self.library.mal_map.items():
reverse_mal[v] = k
2022-06-09 14:20:43 +00:00
for i, (over_key, (item, over_names)) in enumerate(sorted(key_to_overlays.items(), key=lambda io: self.library.get_item_sort_title(io[1][0])), 1):
2022-10-06 18:58:57 +00:00
item_title = self.library.get_item_sort_title(item, atr="title")
2022-04-20 04:06:42 +00:00
try:
2022-04-26 15:28:04 +00:00
logger.ghost(f"Overlaying: {i}/{len(key_to_overlays)} {item_title}")
2022-04-20 04:06:42 +00:00
image_compare = None
overlay_compare = None
2022-04-25 22:55:59 +00:00
poster = None
2022-04-20 04:06:42 +00:00
if self.config.Cache:
2022-04-25 20:18:04 +00:00
image, image_compare, overlay_compare = self.config.Cache.query_image_map(item.ratingKey, f"{self.library.image_table_name}_overlays")
2023-02-06 20:34:40 +00:00
self.library.reload(item, force=True)
2022-04-22 02:57:26 +00:00
2022-04-26 19:56:46 +00:00
overlay_compare = [] if overlay_compare is None else util.get_list(overlay_compare, split="|")
2022-06-07 20:42:17 +00:00
has_overlay = any([item_tag.tag.lower() == "overlay" for item_tag in self.library.item_labels(item)])
2022-04-20 04:06:42 +00:00
2022-05-15 03:06:32 +00:00
compare_names = {properties[ov].get_overlay_compare(): ov for ov in over_names}
blur_num = 0
2022-06-03 15:46:25 +00:00
applied_names = []
queue_overlays = {}
2022-05-15 03:06:32 +00:00
for over_name in over_names:
current_overlay = properties[over_name]
if current_overlay.name.startswith("blur"):
2022-06-30 13:48:19 +00:00
logger.info(over_name)
blur_test = int(re.search("\\(([^)]+)\\)", current_overlay.name).group(1))
2022-05-15 03:06:32 +00:00
if blur_test > blur_num:
blur_num = blur_test
2022-10-28 23:32:48 +00:00
elif current_overlay.queue_name:
if current_overlay.queue not in queue_overlays:
queue_overlays[current_overlay.queue] = {}
if current_overlay.weight in queue_overlays[current_overlay.queue]:
2022-06-30 13:48:19 +00:00
raise Failed("Overlay Error: Overlays in a queue cannot have the same weight")
queue_overlays[current_overlay.queue][current_overlay.weight] = over_name
2022-05-15 03:06:32 +00:00
else:
2022-06-30 13:48:19 +00:00
applied_names.append(over_name)
2022-04-27 15:11:10 +00:00
2023-03-10 15:07:25 +00:00
overlay_change = "" if has_overlay else "No Overlay Label"
2022-04-20 04:06:42 +00:00
if not overlay_change:
for oc in overlay_compare:
2022-04-27 15:11:10 +00:00
if oc not in compare_names:
2023-03-10 15:07:25 +00:00
overlay_change = f"{oc} not in {compare_names}"
2022-05-15 03:06:32 +00:00
2022-04-20 04:06:42 +00:00
if not overlay_change:
2022-05-15 03:06:32 +00:00
for compare_name, original_name in compare_names.items():
if compare_name not in overlay_compare or properties[original_name].updated:
2023-03-10 15:07:25 +00:00
overlay_change = f"{compare_name} not in {overlay_compare} or {properties[original_name].updated}"
2022-05-15 03:06:32 +00:00
2022-06-03 15:46:25 +00:00
if self.config.Cache:
for over_name in over_names:
2023-03-10 15:07:25 +00:00
if properties[over_name].name.startswith("text"):
2022-07-28 04:51:01 +00:00
for cache_key, cache_value in self.config.Cache.query_overlay_special_text(item.ratingKey).items():
actual = plex.attribute_translation[cache_key] if cache_key in plex.attribute_translation else cache_key
2023-03-10 15:07:25 +00:00
if not hasattr(item, actual):
continue
2023-04-28 03:43:26 +00:00
real_value = getattr(item, actual)
if cache_value is None or real_value is None:
2022-07-28 04:51:01 +00:00
continue
if cache_key in overlay.float_vars:
cache_value = float(cache_value)
if cache_key in overlay.int_vars:
cache_value = int(cache_value)
if cache_key in overlay.date_vars:
2023-04-28 03:43:26 +00:00
real_value = real_value.strftime("%Y-%m-%d")
if real_value != cache_value:
overlay_change = f"Special Text Changed from {cache_value} to {real_value}"
2022-04-25 22:55:59 +00:00
try:
2022-06-26 16:31:22 +00:00
poster, background, item_dir, name = self.library.find_item_assets(item)
2022-06-30 15:17:48 +00:00
if not poster and self.library.assets_for_all:
if (isinstance(item, Episode) and self.library.show_missing_episode_assets) or \
(isinstance(item, Season) and self.library.show_missing_season_assets) or \
(not isinstance(item, (Episode, Season)) and self.library.show_missing_assets):
if self.library.asset_folders:
logger.warning(f"Asset Warning: No poster found for '{item_title}' in the assets folder '{item_dir}'")
else:
logger.warning(f"Asset Warning: No poster '{name}' found in the assets folders")
2022-06-26 16:31:22 +00:00
if background:
self.library.upload_images(item, background=background)
2022-04-25 22:55:59 +00:00
except Failed as e:
if self.library.assets_for_all and self.library.show_missing_assets:
logger.warning(e)
2022-04-20 04:06:42 +00:00
has_original = None
new_backup = None
2023-04-28 15:18:45 +00:00
changed_image = False
2022-04-20 04:06:42 +00:00
if poster:
if image_compare and str(poster.compare) != str(image_compare):
changed_image = True
if os.path.exists(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.png")):
os.remove(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.png"))
if os.path.exists(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.jpg")):
os.remove(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.jpg"))
2024-05-01 16:06:19 +00:00
if os.path.exists(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.webp")):
os.remove(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.webp"))
2022-04-20 04:06:42 +00:00
elif has_overlay:
2022-10-26 06:56:12 +00:00
if os.path.exists(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.png")):
2022-04-20 04:06:42 +00:00
has_original = os.path.join(self.library.overlay_backup, f"{item.ratingKey}.png")
elif os.path.exists(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.jpg")):
has_original = os.path.join(self.library.overlay_backup, f"{item.ratingKey}.jpg")
2024-05-01 16:06:19 +00:00
elif os.path.exists(os.path.join(self.library.overlay_backup, f"{item.ratingKey}.webp")):
has_original = os.path.join(self.library.overlay_backup, f"{item.ratingKey}.webp")
2023-01-05 21:33:21 +00:00
if self.library.reset_overlays:
reset_list = self.library.reset_overlays
elif has_original is None and not self.library.reset_overlays:
reset_list = ["plex", "tmdb"]
else:
reset_list = []
try:
2023-02-05 19:10:08 +00:00
new_backup = self.library.item_posters(item, providers=reset_list)
except Failed as e:
if any(r in reset_list for r in ["plex", "tmdb"]):
logger.error(e)
2022-04-20 04:06:42 +00:00
else:
new_backup = item.posterUrl
2023-03-01 17:08:01 +00:00
logger.info(f"\n{item_title}")
2022-04-20 04:06:42 +00:00
if new_backup:
try:
has_original = self.library.check_image_for_overlay(new_backup, os.path.join(self.library.overlay_backup, f"{item.ratingKey}"))
except Failed as e:
2023-03-01 22:10:46 +00:00
raise Failed(f" Overlay Error: {e}")
2022-04-20 04:06:42 +00:00
poster_compare = None
if poster is None and has_original is None:
2023-03-01 22:10:46 +00:00
logger.error(f" Overlay Error: No poster found")
2023-04-28 15:18:45 +00:00
elif self.library.reapply_overlays or new_backup or overlay_change or changed_image:
2022-04-20 04:06:42 +00:00
try:
2023-03-10 15:07:25 +00:00
if not self.library.reapply_overlays and new_backup:
logger.trace(" Overlay Reason: New image detected")
2023-03-01 17:08:01 +00:00
elif not self.library.reapply_overlays and overlay_change:
2023-03-10 15:07:25 +00:00
logger.trace(f" Overlay Reason: Overlay changed {overlay_change}")
2022-10-19 20:13:38 +00:00
canvas_width, canvas_height = overlay.get_canvas_size(item)
2023-02-28 17:06:45 +00:00
with Image.open(poster.location if poster else has_original) as new_poster:
exif_tags = new_poster.getexif()
exif_tags[0x04bc] = "overlay"
new_poster = new_poster.convert("RGB").resize((canvas_width, canvas_height), Image.LANCZOS)
2022-05-15 20:29:54 +00:00
2023-02-28 17:06:45 +00:00
if blur_num > 0:
new_poster = new_poster.filter(ImageFilter.GaussianBlur(blur_num))
2022-06-03 15:46:25 +00:00
2023-02-28 17:06:45 +00:00
def get_text(text_overlay):
full_text = text_overlay.name[5:-1]
for format_var in overlay.vars_by_type[text_overlay.level]:
if f"<<{format_var}" in full_text and format_var == "originally_available[":
mod = re.search("<<originally_available\\[(.+)]>>", full_text).group(1)
format_var = "originally_available"
elif f"<<{format_var}>>" in full_text and format_var.endswith(tuple(m for m in overlay.double_mods)):
mod = format_var[-2:]
format_var = format_var[:-2]
elif f"<<{format_var}>>" in full_text and format_var.endswith(tuple(m for m in overlay.single_mods)):
mod = format_var[-1]
format_var = format_var[:-1]
elif f"<<{format_var}>>" in full_text:
mod = ""
2022-07-28 04:51:01 +00:00
else:
2022-06-03 15:46:25 +00:00
continue
2023-02-28 17:06:45 +00:00
if format_var == "show_title":
actual_attr = "parentTitle" if text_overlay.level == "season" else "grandparentTitle"
elif format_var in plex.attribute_translation:
actual_attr = plex.attribute_translation[format_var]
else:
actual_attr = format_var
if format_var == "bitrate":
actual_value = None
for media in item.media:
current = int(media.bitrate)
if actual_value is None:
actual_value = current
if mod == "":
break
elif mod == "H" and current > actual_value:
actual_value = current
elif mod == "L" and current < actual_value:
actual_value = current
elif format_var in overlay.rating_sources:
found_rating = None
try:
2024-03-28 20:45:55 +00:00
item_to_id = item.show() if isinstance(item, (Season, Episode)) else item
tmdb_id, tvdb_id, imdb_id = self.library.get_ids(item_to_id)
if format_var == "tmdb_rating":
2024-03-28 20:45:55 +00:00
_item = self.config.TMDb.get_item(item_to_id, tmdb_id, tvdb_id, imdb_id, is_movie=self.library.is_movie)
if _item:
2024-03-28 20:45:55 +00:00
if isinstance(item, Episode):
found_rating = self.config.TMDb.get_episode(_item.tmdb_id, item.seasonNumber, item.episodeNumber).vote_average
elif isinstance(item, Season):
for season in _item.seasons:
if item.seasonNumber == season.season_number:
found_rating = season.average
break
else:
found_rating = _item.vote_average
else:
raise Failed(f"No TMDb ID for Guid: {item.guid}")
elif format_var == "imdb_rating":
2024-03-28 20:45:55 +00:00
if isinstance(item, Episode):
found_rating = self.config.IMDb.get_episode_rating(imdb_id, item.seasonNumber, item.episodeNumber)
else:
found_rating = self.config.IMDb.get_rating(imdb_id)
elif format_var == "omdb_rating":
if self.config.OMDb.limit is not False:
raise Failed("Daily OMDb Limit Reached")
elif not imdb_id:
raise Failed(f"No IMDb ID for Guid: {item.guid}")
else:
try:
found_rating = self.config.OMDb.get_omdb(imdb_id).imdb_rating
except Exception:
logger.error(f"IMDb ID: {imdb_id}")
raise
elif format_var == "trakt_user_rating":
_ratings = trakt_ratings()
_id = tmdb_id if self.library.is_movie else tvdb_id
if _id in _ratings:
found_rating = _ratings[_id]
else:
raise Failed("No Trakt User Rating Found")
elif str(format_var).startswith("mdb"):
mdb_item = None
if self.config.Mdblist.limit is False:
if self.library.is_show and tvdb_id:
try:
mdb_item = self.config.Mdblist.get_series(tvdb_id)
except LimitReached as err:
logger.debug(err)
except Failed as err:
logger.error(str(err))
except Exception:
logger.trace(f"TVDb ID: {tvdb_id}")
raise
if self.library.is_movie and tmdb_id:
try:
mdb_item = self.config.Mdblist.get_movie(tmdb_id)
except LimitReached as err:
logger.debug(err)
except Failed as err:
logger.error(str(err))
except Exception:
logger.trace(f"TMDb ID: {tmdb_id}")
raise
if imdb_id and not mdb_item:
try:
mdb_item = self.config.Mdblist.get_imdb(imdb_id)
except LimitReached as err:
logger.debug(err)
except Failed as err:
logger.error(str(err))
except Exception:
logger.trace(f"IMDb ID: {imdb_id}")
raise
if not mdb_item:
raise Failed(f"No MdbItem for {item.title} (Guid: {item.guid})")
if format_var == "mdb_average_rating":
found_rating = mdb_item.average / 10 if mdb_item.average else None
elif format_var == "mdb_imdb_rating":
found_rating = mdb_item.imdb_rating if mdb_item.imdb_rating else None
elif format_var == "mdb_metacritic_rating":
found_rating = mdb_item.metacritic_rating / 10 if mdb_item.metacritic_rating else None
elif format_var == "mdb_metacriticuser_rating":
found_rating = mdb_item.metacriticuser_rating if mdb_item.metacriticuser_rating else None
elif format_var == "mdb_trakt_rating":
found_rating = mdb_item.trakt_rating / 10 if mdb_item.trakt_rating else None
elif format_var == "mdb_tomatoes_rating":
found_rating = mdb_item.tomatoes_rating / 10 if mdb_item.tomatoes_rating else None
elif format_var == "mdb_tomatoesaudience_rating":
found_rating = mdb_item.tomatoesaudience_rating / 10 if mdb_item.tomatoesaudience_rating else None
elif format_var == "mdb_tmdb_rating":
found_rating = mdb_item.tmdb_rating / 10 if mdb_item.tmdb_rating else None
elif format_var == "mdb_letterboxd_rating":
found_rating = mdb_item.letterboxd_rating * 2 if mdb_item.letterboxd_rating else None
elif format_var == "mdb_myanimelist_rating":
found_rating = mdb_item.myanimelist_rating if mdb_item.myanimelist_rating else None
else:
found_rating = mdb_item.score / 10 if mdb_item.score else None
elif str(format_var).startswith(("anidb", "mal")):
anidb_id = None
if item.ratingKey in reverse_anidb:
anidb_id = reverse_anidb[item.ratingKey]
elif tvdb_id in self.config.Convert._tvdb_to_anidb:
anidb_id = self.config.Convert._tvdb_to_anidb[tvdb_id]
elif imdb_id in self.config.Convert._imdb_to_anidb:
anidb_id = self.config.Convert._imdb_to_anidb[imdb_id]
if str(format_var).startswith("anidb"):
if anidb_id:
anidb_obj = self.config.AniDB.get_anime(anidb_id)
if format_var == "anidb_rating_rating":
found_rating = anidb_obj.rating
elif format_var == "anidb_average_rating":
found_rating = anidb_obj.average
elif format_var == "anidb_score_rating":
found_rating = anidb_obj.score
else:
raise Failed(f"No AniDB ID for Guid: {item.guid}")
else:
if item.ratingKey in reverse_mal:
mal_id = reverse_mal[item.ratingKey]
elif not anidb_id:
raise Failed(f"Convert Warning: No AniDB ID to Convert to MyAnimeList ID for Guid: {item.guid}")
elif anidb_id not in self.config.Convert._anidb_to_mal:
raise Failed(f"Convert Warning: No MyAnimeList Found for AniDB ID: {anidb_id} of Guid: {item.guid}")
else:
mal_id = self.config.Convert._anidb_to_mal[anidb_id]
if mal_id:
found_rating = self.config.MyAnimeList.get_anime(mal_id).score
except Failed as err:
logger.error(err)
if found_rating:
actual_value = found_rating
else:
raise Failed(f"No Rating Found for {item_title}")
elif format_var == "runtime" and text_overlay.level in ["show", "season", "artist", "album"]:
if hasattr(item, "duration") and item.duration:
actual_value = item.duration
else:
sub_items = item.episodes() if text_overlay.level in ["show", "season"] else item.tracks()
sub_items = [ep.duration for ep in sub_items if hasattr(ep, "duration") and ep.duration]
actual_value = sum(sub_items) / len(sub_items)
2023-02-28 17:06:45 +00:00
else:
if not hasattr(item, actual_attr) or getattr(item, actual_attr) is None:
raise Failed(f"Overlay Warning: No {full_text} found")
actual_value = getattr(item, actual_attr)
if format_var == "versions":
actual_value = len(actual_value)
if self.config.Cache:
cache_store = actual_value.strftime("%Y-%m-%d") if format_var in overlay.date_vars else actual_value
self.config.Cache.update_overlay_special_text(item.ratingKey, format_var, cache_store)
sub_value = None
if format_var == "originally_available":
if mod:
sub_value = "<<originally_available\\[(.+)]>>"
final_value = actual_value.strftime(mod)
else:
final_value = actual_value.strftime("%Y-%m-%d")
elif format_var == "runtime":
if mod == "H":
final_value = int((actual_value / 60000) // 60)
elif mod == "M":
final_value = int((actual_value / 60000) % 60)
else:
final_value = int(actual_value / 60000)
elif mod == "%":
2024-03-28 20:45:55 +00:00
final_value = int(float(actual_value) * 10)
2023-02-28 17:06:45 +00:00
elif mod == "#":
2024-03-28 20:45:55 +00:00
actual_value = f"{float(actual_value):.1f}"
final_value = actual_value[:-2] if actual_value.endswith(".0") else actual_value
2023-08-08 18:15:03 +00:00
elif mod == "/":
final_value = f"{float(actual_value) / 2:.1f}"
2023-02-28 17:06:45 +00:00
elif mod == "W":
final_value = num2words(int(actual_value))
elif mod == "WU":
final_value = num2words(int(actual_value)).upper()
elif mod == "WL":
final_value = num2words(int(actual_value)).lower()
2023-02-28 17:06:45 +00:00
elif mod == "0":
final_value = f"{int(actual_value):02}"
elif mod == "00":
final_value = f"{int(actual_value):03}"
elif mod == "U":
final_value = str(actual_value).upper()
elif mod == "L":
final_value = str(actual_value).lower()
elif mod == "P":
final_value = str(actual_value).title()
elif format_var in overlay.rating_sources:
2024-03-28 20:45:55 +00:00
final_value = f"{float(actual_value):.1f}"
2023-02-28 17:06:45 +00:00
else:
final_value = actual_value
if sub_value:
full_text = re.sub(sub_value, str(final_value), full_text)
else:
full_text = full_text.replace(f"<<{format_var}{mod}>>", str(final_value))
return str(full_text)
2022-06-03 15:46:25 +00:00
2023-02-28 17:06:45 +00:00
for over_name in applied_names:
current_overlay = properties[over_name]
if current_overlay.name.startswith("text"):
2023-02-28 17:06:45 +00:00
if "<<" in current_overlay.name:
image_box = current_overlay.image.size if current_overlay.image else None
try:
overlay_image, addon_box = current_overlay.get_backdrop((canvas_width, canvas_height), box=image_box, text=get_text(current_overlay))
except Failed as e:
2023-03-01 22:10:46 +00:00
logger.warning(f" {e}")
2023-02-28 17:06:45 +00:00
continue
new_poster.paste(overlay_image, (0, 0), overlay_image)
else:
overlay_image, addon_box = current_overlay.get_canvas(item)
new_poster.paste(overlay_image, (0, 0), overlay_image)
if current_overlay.image:
new_poster.paste(current_overlay.image, addon_box, current_overlay.image)
2023-02-28 17:06:45 +00:00
elif current_overlay.name == "backdrop":
overlay_image, _ = current_overlay.get_canvas(item)
new_poster.paste(overlay_image, (0, 0), overlay_image)
2022-06-03 15:46:25 +00:00
else:
2023-02-28 17:06:45 +00:00
if current_overlay.has_coordinates():
overlay_image, overlay_box = current_overlay.get_canvas(item)
2023-03-16 20:45:14 +00:00
if overlay_image is not None:
2023-02-28 17:06:45 +00:00
new_poster.paste(overlay_image, (0, 0), overlay_image)
new_poster.paste(current_overlay.image, overlay_box, current_overlay.image)
else:
new_poster = new_poster.resize(current_overlay.image.size, Image.LANCZOS)
new_poster.paste(current_overlay.image, (0, 0), current_overlay.image)
new_poster = new_poster.resize((canvas_width, canvas_height), Image.LANCZOS)
for queue, weights in queue_overlays.items():
cords = self.library.queues[queue]
sorted_weights = sorted(weights.items(), reverse=True)
for o, cord in enumerate(cords):
if len(sorted_weights) <= o:
break
over_name = sorted_weights[o][1]
current_overlay = properties[over_name]
if current_overlay.name.startswith("text"):
image_box = current_overlay.image.size if current_overlay.image else None
try:
overlay_image, addon_box = current_overlay.get_backdrop((canvas_width, canvas_height), box=image_box, text=get_text(current_overlay), new_cords=cord)
except Failed as e:
2023-03-01 22:10:46 +00:00
logger.warning(f" {e}")
2023-02-28 17:06:45 +00:00
continue
2022-06-03 15:46:25 +00:00
new_poster.paste(overlay_image, (0, 0), overlay_image)
2023-02-28 17:06:45 +00:00
if current_overlay.image:
new_poster.paste(current_overlay.image, addon_box, current_overlay.image)
2022-06-03 15:46:25 +00:00
else:
2023-02-28 17:06:45 +00:00
if current_overlay.has_back:
overlay_image, overlay_box = current_overlay.get_backdrop((canvas_width, canvas_height), box=current_overlay.image.size, new_cords=cord)
new_poster.paste(overlay_image, (0, 0), overlay_image)
else:
overlay_box = current_overlay.get_coordinates((canvas_width, canvas_height), box=current_overlay.image.size, new_cords=cord)
new_poster.paste(current_overlay.image, overlay_box, current_overlay.image)
temp = os.path.join(self.library.overlay_folder, f"temp.{self.library.overlay_filetype}")
2024-05-01 16:06:19 +00:00
if self.library.overlay_quality and self.library.overlay_filetype in ["jpg", "webp"]:
new_poster.save(temp, exif=exif_tags, quality=self.library.overlay_quality)
else:
new_poster.save(temp, exif=exif_tags)
2023-02-28 17:06:45 +00:00
self.library.upload_poster(item, temp)
self.library.edit_tags("label", item, add_tags=["Overlay"], do_print=False)
poster_compare = poster.compare if poster else item.thumb
2023-03-01 22:10:46 +00:00
logger.info(f" Overlays Applied: {', '.join(over_names)}")
2022-07-05 18:12:31 +00:00
except (OSError, BadRequest, SyntaxError) as e:
2022-04-20 04:06:42 +00:00
logger.stacktrace()
2023-03-01 22:10:46 +00:00
raise Failed(f" Overlay Error: {e}")
2023-03-01 17:08:01 +00:00
else:
2023-03-01 22:10:46 +00:00
logger.info(" Overlay Update Not Needed")
2022-04-20 04:06:42 +00:00
if self.config.Cache and poster_compare:
2022-07-26 18:30:40 +00:00
self.config.Cache.update_image_map(item.ratingKey, f"{self.library.image_table_name}_overlays", item.thumb, poster_compare, overlay='|'.join(compare_names))
2022-04-20 04:06:42 +00:00
except Failed as e:
2023-03-01 22:10:46 +00:00
logger.error(f" {e}\n Overlays Attempted on {item_title}: {', '.join(over_names)}")
2022-11-08 03:02:07 +00:00
except Exception as e:
logger.info(e)
logger.info(type(e))
2023-03-16 20:21:09 +00:00
logger.stacktrace()
2024-01-04 20:24:53 +00:00
logger.info("")
2022-11-08 03:02:07 +00:00
logger.error(f"Overlays Attempted on {item_title}: {', '.join(over_names)}")
2022-04-18 18:16:39 +00:00
logger.exorcise()
2023-02-28 17:06:45 +00:00
for _, over in properties.items():
if over.image:
over.image.close()
2022-05-05 13:21:17 +00:00
overlay_run_time = str(datetime.now() - overlay_start).split('.')[0]
logger.info("")
logger.separator(f"Finished {self.library.name} Library Overlays\nOverlays Run Time: {overlay_run_time}")
return overlay_run_time
def compile_overlays(self):
key_to_item = {}
properties = {}
overlay_groups = {}
key_to_overlays = {}
for overlay_file in self.library.overlay_files:
for k, v in overlay_file.overlays.items():
try:
builder = CollectionBuilder(self.config, overlay_file, k, v, library=self.library, overlay=True)
logger.info("")
logger.separator(f"Gathering Items for {k} Overlay", space=False, border=False)
2022-12-20 21:29:00 +00:00
prop_name = builder.overlay.mapping_name
properties[prop_name] = builder.overlay
2022-10-03 19:32:50 +00:00
builder.display_filters()
2022-07-20 14:27:52 +00:00
for method, value in builder.builders:
logger.debug("")
logger.debug(f"Builder: {method}: {value}")
logger.info("")
2022-07-19 13:25:48 +00:00
try:
builder.filter_and_save_items(builder.gather_ids(method, value))
2023-07-24 16:55:41 +00:00
except Failed as e:
2022-07-19 13:25:48 +00:00
if builder.ignore_blank_results:
2024-01-05 21:06:39 +00:00
logger.info("")
2022-07-19 13:25:48 +00:00
logger.warning(e)
else:
raise Failed(e)
added_titles = []
2022-11-07 17:05:21 +00:00
if builder.found_items:
for item in builder.found_items:
2023-08-17 17:28:29 +00:00
if builder.limit and len(added_titles) >= builder.limit:
break
key_to_item[item.ratingKey] = item
2022-04-26 15:28:04 +00:00
added_titles.append(item)
2022-12-20 21:29:00 +00:00
if item.ratingKey not in properties[prop_name].keys:
properties[prop_name].keys.append(item.ratingKey)
if added_titles:
2022-12-20 21:29:00 +00:00
logger.info(f"{len(added_titles)} Items found for {prop_name}")
2022-10-18 20:54:28 +00:00
logger.trace(f"Titles Found: {[self.library.get_item_sort_title(a, atr='title') for a in added_titles]}")
2022-07-19 14:54:13 +00:00
else:
2022-12-20 21:29:00 +00:00
logger.warning(f"No Items found for {prop_name}")
2022-07-20 14:27:52 +00:00
logger.info("")
2022-05-12 15:04:52 +00:00
except NotScheduled as e:
logger.info(e)
2022-11-12 16:42:53 +00:00
except FilterFailed:
pass
except Failed as e:
2022-05-15 03:06:32 +00:00
logger.stacktrace()
logger.error(e)
2022-07-18 18:14:32 +00:00
logger.info("")
2022-11-12 16:42:53 +00:00
except Exception as e:
logger.stacktrace()
logger.error(f"Unknown Error: {e}")
logger.info("")
2022-10-26 06:56:12 +00:00
logger.separator(f"Overlay Operation for the {self.library.name} Library")
logger.debug("")
logger.debug(f"Remove Overlays: {self.library.remove_overlays}")
logger.debug(f"Reapply Overlays: {self.library.reapply_overlays}")
logger.debug(f"Reset Overlays: {self.library.reset_overlays}")
logger.debug("")
logger.separator(f"Number of Items Per Overlay", space=False, border=False)
logger.debug("")
longest = 7
for overlay_name in properties:
if len(overlay_name) > longest:
longest = len(overlay_name)
logger.debug(f"{'Overlay':^{longest}} | Number |")
logger.debug(f"{logger.separating_character * longest} | {logger.separating_character * 6} |")
for overlay_name, over_obj in properties.items():
logger.debug(f"{overlay_name:<{longest}} | {len(over_obj.keys):^6} |")
logger.debug("")
2022-05-15 03:06:32 +00:00
for overlay_name, over_obj in properties.items():
if over_obj.group:
if over_obj.group not in overlay_groups:
overlay_groups[over_obj.group] = {}
overlay_groups[over_obj.group][overlay_name] = over_obj.weight
2022-05-15 03:06:32 +00:00
for overlay_name, over_obj in properties.items():
for over_key in over_obj.keys:
if over_key not in key_to_overlays:
key_to_overlays[over_key] = (key_to_item[over_key], [])
key_to_overlays[over_key][1].append(overlay_name)
for over_key, (item, over_names) in key_to_overlays.items():
group_status = {}
for over_name in over_names:
for suppress_name in properties[over_name].suppress:
if suppress_name in over_names:
key_to_overlays[over_key][1].remove(suppress_name)
for over_name in over_names:
for overlay_group, group_names in overlay_groups.items():
if over_name in group_names:
if overlay_group not in group_status:
group_status[overlay_group] = []
group_status[overlay_group].append(over_name)
for gk, gv in group_status.items():
if len(gv) > 1:
final = None
for v in gv:
if final is None or overlay_groups[gk][v] > overlay_groups[gk][final]:
final = v
for v in gv:
if final != v:
key_to_overlays[over_key][1].remove(v)
2022-10-28 23:32:48 +00:00
return key_to_overlays, properties
def get_overlay_items(self, label="Overlay", libtype=None, ignore=None):
items = self.library.search(label=label, libtype=libtype)
return items if not ignore else [o for o in items if o.ratingKey not in ignore]
2022-04-26 15:28:04 +00:00
def remove_overlay(self, item, item_title, label, locations):
2022-04-25 22:55:59 +00:00
try:
poster, _, _, _ = self.library.find_item_assets(item)
except Failed:
poster = None
is_url = False
2022-10-26 06:56:12 +00:00
poster_location = None
if poster:
poster_location = poster.location
elif any([os.path.exists(loc) for loc in locations]):
poster_location = next((loc for loc in locations if os.path.exists(loc)))
2022-10-26 06:56:12 +00:00
if not poster_location:
is_url = True
2022-10-26 06:56:12 +00:00
try:
poster_location = self.library.item_posters(item)
except Failed:
pass
if poster_location:
self.library.upload_poster(item, poster_location, url=is_url)
self.library.edit_tags("label", item, remove_tags=[label], do_print=False)
for loc in locations:
if os.path.exists(loc):
os.remove(loc)
else:
2022-05-05 13:21:17 +00:00
logger.error(f"No Poster found to restore for {item_title}")