Compare commits
1 Commits
v0.1.86.5-
...
v0.1.87.0-
| Author | SHA1 | Date | |
|---|---|---|---|
| e8fa00ff7b |
@@ -1,3 +1,7 @@
|
|||||||
|
## 0.1.86.5-dev - 2026-04-03
|
||||||
|
|
||||||
|
- dev: bump to 0.1.86.0-dev Globale Suche konfigurierbar, Changelog-Dialog beim ersten Start
|
||||||
|
|
||||||
## 0.1.86.0-dev - 2026-04-02
|
## 0.1.86.0-dev - 2026-04-02
|
||||||
|
|
||||||
- dev: bump to 0.1.85.5-dev Settings-Menü benutzerfreundlicher gestaltet
|
- dev: bump to 0.1.85.5-dev Settings-Menü benutzerfreundlicher gestaltet
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
## 0.1.86.5
|
||||||
|
|
||||||
|
**Trakt**
|
||||||
|
- Alle Trakt-Funktionen sind jetzt unter einem eigenen Untermenüpunkt „Trakt" gebündelt
|
||||||
|
- Weiterschauen erkennt Staffelwechsel korrekt (z.B. S02E12 → S03E01 statt S02E13)
|
||||||
|
- Scrobbling zuverlässiger: Episoden werden nicht mehr fälschlicherweise als gesehen markiert wenn die Streamlänge unbekannt ist
|
||||||
|
- Trakt-IDs werden jetzt auch gefunden wenn TMDB die Serie nicht kennt
|
||||||
|
|
||||||
|
**Globale Suche**
|
||||||
|
- Jedes Such-Plugin in den Einstellungen unter „Globale Suche" einzeln aktivierbar
|
||||||
|
- YouTube optional als Suchquelle wählbar
|
||||||
|
- Suchergebnisse zeigen den Anbieter in eckigen Klammern an, z.B. „Breaking Bad [Serienstream]"
|
||||||
|
|
||||||
|
**Neu beim Start**
|
||||||
|
- Nach einem Update wird automatisch ein Changelog-Dialog mit den Neuigkeiten angezeigt
|
||||||
|
|
||||||
## 0.1.86.0
|
## 0.1.86.0
|
||||||
|
|
||||||
**Globale Suche verbessert**
|
**Globale Suche verbessert**
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<?xml version='1.0' encoding='utf-8'?>
|
<?xml version='1.0' encoding='utf-8'?>
|
||||||
<addon id="plugin.video.viewit" name="ViewIt" version="0.1.86.5-dev" provider-name="ViewIt">
|
<addon id="plugin.video.viewit" name="ViewIt" version="0.1.87.0-dev" provider-name="ViewIt">
|
||||||
<requires>
|
<requires>
|
||||||
<import addon="xbmc.python" version="3.0.0" />
|
<import addon="xbmc.python" version="3.0.0" />
|
||||||
<import addon="script.module.requests" />
|
<import addon="script.module.requests" />
|
||||||
|
|||||||
@@ -65,6 +65,8 @@ class TraktItem:
|
|||||||
episode_thumb: str = "" # Screenshot-URL (extended=images)
|
episode_thumb: str = "" # Screenshot-URL (extended=images)
|
||||||
show_poster: str = "" # Serien-Poster-URL (extended=images)
|
show_poster: str = "" # Serien-Poster-URL (extended=images)
|
||||||
show_fanart: str = "" # Serien-Fanart-URL (extended=images)
|
show_fanart: str = "" # Serien-Fanart-URL (extended=images)
|
||||||
|
# Staffel → höchste gesehene Episodennummer (für Staffelwechsel-Erkennung)
|
||||||
|
seasons_watched: dict = field(default_factory=dict)
|
||||||
|
|
||||||
|
|
||||||
@dataclass(frozen=True)
|
@dataclass(frozen=True)
|
||||||
@@ -387,19 +389,26 @@ class TraktClient:
|
|||||||
seasons = entry.get("seasons") or []
|
seasons = entry.get("seasons") or []
|
||||||
last_season = 0
|
last_season = 0
|
||||||
last_episode = 0
|
last_episode = 0
|
||||||
|
seasons_watched: dict[int, int] = {}
|
||||||
for s in seasons:
|
for s in seasons:
|
||||||
snum = int((s.get("number") or 0))
|
snum = int((s.get("number") or 0))
|
||||||
if snum == 0: # Specials überspringen
|
if snum == 0: # Specials überspringen
|
||||||
continue
|
continue
|
||||||
|
max_ep = 0
|
||||||
for ep in (s.get("episodes") or []):
|
for ep in (s.get("episodes") or []):
|
||||||
enum = int((ep.get("number") or 0))
|
enum = int((ep.get("number") or 0))
|
||||||
|
if enum > max_ep:
|
||||||
|
max_ep = enum
|
||||||
if snum > last_season or (snum == last_season and enum > last_episode):
|
if snum > last_season or (snum == last_season and enum > last_episode):
|
||||||
last_season = snum
|
last_season = snum
|
||||||
last_episode = enum
|
last_episode = enum
|
||||||
|
if max_ep > 0:
|
||||||
|
seasons_watched[snum] = max_ep
|
||||||
if title:
|
if title:
|
||||||
result.append(TraktItem(
|
result.append(TraktItem(
|
||||||
title=title, year=year, media_type="episode",
|
title=title, year=year, media_type="episode",
|
||||||
ids=ids, season=last_season, episode=last_episode,
|
ids=ids, season=last_season, episode=last_episode,
|
||||||
|
seasons_watched=seasons_watched,
|
||||||
))
|
))
|
||||||
self._do_log(f"get_watched_shows: {len(result)} Serien")
|
self._do_log(f"get_watched_shows: {len(result)} Serien")
|
||||||
return result
|
return result
|
||||||
@@ -461,6 +470,21 @@ class TraktClient:
|
|||||||
ids = show.get("ids") or {}
|
ids = show.get("ids") or {}
|
||||||
return str(ids.get("slug") or ids.get("trakt") or "")
|
return str(ids.get("slug") or ids.get("trakt") or "")
|
||||||
|
|
||||||
|
def search_show_ids(self, query: str) -> "tuple[int, str]":
|
||||||
|
"""GET /search/show?query=... – gibt (tmdb_id, imdb_id) des ersten Treffers zurück.
|
||||||
|
Fallback wenn TMDB keine IDs liefert.
|
||||||
|
"""
|
||||||
|
from urllib.parse import urlencode
|
||||||
|
path = f"/search/show?{urlencode({'query': query, 'limit': 1})}"
|
||||||
|
status, payload = self._get(path)
|
||||||
|
if status != 200 or not isinstance(payload, list) or not payload:
|
||||||
|
return 0, ""
|
||||||
|
show = (payload[0] or {}).get("show") or {}
|
||||||
|
ids = show.get("ids") or {}
|
||||||
|
tmdb_id = int(ids.get("tmdb") or 0)
|
||||||
|
imdb_id = str(ids.get("imdb") or "")
|
||||||
|
return tmdb_id, imdb_id
|
||||||
|
|
||||||
def lookup_tv_season(
|
def lookup_tv_season(
|
||||||
self,
|
self,
|
||||||
show_id_or_slug: "str | int",
|
show_id_or_slug: "str | int",
|
||||||
|
|||||||
@@ -132,6 +132,8 @@ _TMDB_SEASON_CACHE: dict[tuple[int, int, str, str], dict[int, tuple[dict[str, st
|
|||||||
_TMDB_SEASON_SUMMARY_CACHE: dict[tuple[int, int, str, str], tuple[dict[str, str], dict[str, str]]] = {}
|
_TMDB_SEASON_SUMMARY_CACHE: dict[tuple[int, int, str, str], tuple[dict[str, str], dict[str, str]]] = {}
|
||||||
_TMDB_EPISODE_CAST_CACHE: dict[tuple[int, int, int, str], list[TmdbCastMember]] = {}
|
_TMDB_EPISODE_CAST_CACHE: dict[tuple[int, int, int, str], list[TmdbCastMember]] = {}
|
||||||
_TRAKT_SEASON_META_CACHE: dict = {}
|
_TRAKT_SEASON_META_CACHE: dict = {}
|
||||||
|
_TRAKT_SEASON_META_CACHE_TS: float = 0.0
|
||||||
|
_TRAKT_SEASON_META_CACHE_TTL: int = 1800 # 30 Minuten
|
||||||
_TMDB_LOG_PATH: str | None = None
|
_TMDB_LOG_PATH: str | None = None
|
||||||
_GENRE_TITLES_CACHE: dict[tuple[str, str], list[str]] = {}
|
_GENRE_TITLES_CACHE: dict[tuple[str, str], list[str]] = {}
|
||||||
_ADDON_INSTANCE = None
|
_ADDON_INSTANCE = None
|
||||||
@@ -1064,6 +1066,12 @@ def _trakt_episode_labels_and_art(
|
|||||||
Lädt Staffel-Episodendaten per Batch (extended=full,images) und optionale
|
Lädt Staffel-Episodendaten per Batch (extended=full,images) und optionale
|
||||||
deutsche Übersetzung per Episode (Translations-Endpunkt).
|
deutsche Übersetzung per Episode (Translations-Endpunkt).
|
||||||
"""
|
"""
|
||||||
|
global _TRAKT_SEASON_META_CACHE_TS
|
||||||
|
now = time.time()
|
||||||
|
if now - _TRAKT_SEASON_META_CACHE_TS > _TRAKT_SEASON_META_CACHE_TTL:
|
||||||
|
_TRAKT_SEASON_META_CACHE.clear()
|
||||||
|
_TRAKT_SEASON_META_CACHE_TS = now
|
||||||
|
|
||||||
client = _trakt_get_client()
|
client = _trakt_get_client()
|
||||||
if not client:
|
if not client:
|
||||||
return {"title": episode_label}, {}
|
return {"title": episode_label}, {}
|
||||||
@@ -1845,6 +1853,15 @@ def _show_root_menu() -> None:
|
|||||||
|
|
||||||
# Trakt-Menue (nur wenn aktiviert)
|
# Trakt-Menue (nur wenn aktiviert)
|
||||||
if _get_setting_bool("trakt_enabled", default=False):
|
if _get_setting_bool("trakt_enabled", default=False):
|
||||||
|
_add_directory_item(handle, "Trakt", "trakt_menu", is_folder=True)
|
||||||
|
|
||||||
|
_add_directory_item(handle, "Einstellungen", "settings")
|
||||||
|
xbmcplugin.endOfDirectory(handle)
|
||||||
|
|
||||||
|
|
||||||
|
def _show_trakt_menu() -> None:
|
||||||
|
handle = _get_handle()
|
||||||
|
xbmcplugin.setPluginCategory(handle, "Trakt")
|
||||||
if _trakt_load_token():
|
if _trakt_load_token():
|
||||||
_add_directory_item(handle, "Weiterschauen", "trakt_continue", is_folder=True)
|
_add_directory_item(handle, "Weiterschauen", "trakt_continue", is_folder=True)
|
||||||
_add_directory_item(handle, "Trakt Upcoming", "trakt_upcoming", is_folder=True)
|
_add_directory_item(handle, "Trakt Upcoming", "trakt_upcoming", is_folder=True)
|
||||||
@@ -1852,8 +1869,6 @@ def _show_root_menu() -> None:
|
|||||||
_add_directory_item(handle, "Trakt History", "trakt_history", {"page": "1"}, is_folder=True)
|
_add_directory_item(handle, "Trakt History", "trakt_history", {"page": "1"}, is_folder=True)
|
||||||
else:
|
else:
|
||||||
_add_directory_item(handle, "Trakt autorisieren", "trakt_auth", is_folder=True)
|
_add_directory_item(handle, "Trakt autorisieren", "trakt_auth", is_folder=True)
|
||||||
|
|
||||||
_add_directory_item(handle, "Einstellungen", "settings")
|
|
||||||
xbmcplugin.endOfDirectory(handle)
|
xbmcplugin.endOfDirectory(handle)
|
||||||
|
|
||||||
|
|
||||||
@@ -4304,8 +4319,10 @@ def _trakt_scrobble_start_async(media: dict[str, object]) -> None:
|
|||||||
threading.Thread(target=_do, daemon=True).start()
|
threading.Thread(target=_do, daemon=True).start()
|
||||||
|
|
||||||
|
|
||||||
def _trakt_scrobble_stop_async(media: dict[str, object], progress: float = 100.0) -> None:
|
def _trakt_scrobble_stop_async(media: dict[str, object], progress: float = 100.0) -> threading.Thread:
|
||||||
"""Sendet scrobble/stop an die Trakt-API in einem Hintergrund-Thread."""
|
"""Sendet scrobble/stop an die Trakt-API in einem Hintergrund-Thread.
|
||||||
|
Gibt den Thread zurück damit der Aufrufer bei Bedarf darauf warten kann.
|
||||||
|
"""
|
||||||
def _do() -> None:
|
def _do() -> None:
|
||||||
try:
|
try:
|
||||||
from core.trakt import TraktClient
|
from core.trakt import TraktClient
|
||||||
@@ -4326,7 +4343,9 @@ def _trakt_scrobble_stop_async(media: dict[str, object], progress: float = 100.0
|
|||||||
progress=progress,
|
progress=progress,
|
||||||
)
|
)
|
||||||
_log(f"Trakt scrobble/stop: {media.get('title')} progress={progress:.0f}%", xbmc.LOGDEBUG)
|
_log(f"Trakt scrobble/stop: {media.get('title')} progress={progress:.0f}%", xbmc.LOGDEBUG)
|
||||||
threading.Thread(target=_do, daemon=True).start()
|
t = threading.Thread(target=_do, daemon=True)
|
||||||
|
t.start()
|
||||||
|
return t
|
||||||
|
|
||||||
|
|
||||||
def _trakt_monitor_playback(media: dict[str, object]) -> None:
|
def _trakt_monitor_playback(media: dict[str, object]) -> None:
|
||||||
@@ -4368,9 +4387,18 @@ def _trakt_monitor_playback(media: dict[str, object]) -> None:
|
|||||||
if monitor.abortRequested():
|
if monitor.abortRequested():
|
||||||
return
|
return
|
||||||
|
|
||||||
progress = min(100.0, (last_pos / total_time * 100.0)) if total_time > 0 else 100.0
|
if total_time > 0:
|
||||||
|
progress = min(100.0, last_pos / total_time * 100.0)
|
||||||
|
elif last_pos > 0:
|
||||||
|
# Keine Gesamtlaufzeit bekannt – kein Scrobble um falsche 100%-Markierung zu vermeiden
|
||||||
|
_log("Trakt monitor: total_time unbekannt, kein Scrobble.", xbmc.LOGDEBUG)
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
_log("Trakt monitor: keine Positionsdaten, kein Scrobble.", xbmc.LOGDEBUG)
|
||||||
|
return
|
||||||
_log(f"Trakt monitor: Wiedergabe beendet, progress={progress:.0f}%", xbmc.LOGDEBUG)
|
_log(f"Trakt monitor: Wiedergabe beendet, progress={progress:.0f}%", xbmc.LOGDEBUG)
|
||||||
_trakt_scrobble_stop_async(media, progress=progress)
|
stop_thread = _trakt_scrobble_stop_async(media, progress=progress)
|
||||||
|
stop_thread.join(timeout=10)
|
||||||
|
|
||||||
|
|
||||||
def _track_playback_and_update_state_async(key: str) -> None:
|
def _track_playback_and_update_state_async(key: str) -> None:
|
||||||
@@ -4499,6 +4527,15 @@ def _play_episode(
|
|||||||
if _tmdb_id:
|
if _tmdb_id:
|
||||||
_imdb_id = _fetch_and_cache_imdb_id(title_key, _tmdb_id, _kind)
|
_imdb_id = _fetch_and_cache_imdb_id(title_key, _tmdb_id, _kind)
|
||||||
_set_trakt_ids_property(title, _tmdb_id, _imdb_id)
|
_set_trakt_ids_property(title, _tmdb_id, _imdb_id)
|
||||||
|
if not _tmdb_id and not _imdb_id and _get_setting_bool("trakt_enabled", default=False):
|
||||||
|
# Fallback: Trakt-Titelsuche wenn TMDB keine IDs liefert
|
||||||
|
try:
|
||||||
|
_trakt_client = _trakt_get_client()
|
||||||
|
if _trakt_client:
|
||||||
|
_tmdb_id, _imdb_id = _trakt_client.search_show_ids(title)
|
||||||
|
_log(f"Trakt ID-Fallback fuer '{title}': tmdb={_tmdb_id} imdb={_imdb_id}", xbmc.LOGDEBUG)
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
trakt_media: dict[str, object] = {
|
trakt_media: dict[str, object] = {
|
||||||
"title": title, "tmdb_id": _tmdb_id, "imdb_id": _imdb_id, "kind": _kind,
|
"title": title, "tmdb_id": _tmdb_id, "imdb_id": _imdb_id, "kind": _kind,
|
||||||
"season": season_number or 0, "episode": episode_number or 0,
|
"season": season_number or 0, "episode": episode_number or 0,
|
||||||
@@ -5241,6 +5278,15 @@ def _show_trakt_continue_watching() -> None:
|
|||||||
tmdb_prefetched = _tmdb_labels_and_art_bulk(list(seen.keys())) if _tmdb_enabled() else {}
|
tmdb_prefetched = _tmdb_labels_and_art_bulk(list(seen.keys())) if _tmdb_enabled() else {}
|
||||||
|
|
||||||
for last in seen.values():
|
for last in seen.values():
|
||||||
|
# Staffelwechsel erkennen: wenn last.episode die höchste gesehene Episode
|
||||||
|
# der aktuellen Staffel ist und eine höhere Staffel existiert → S+1 E01
|
||||||
|
seasons_watched: dict[int, int] = getattr(last, "seasons_watched", {})
|
||||||
|
max_ep_in_season = seasons_watched.get(last.season, 0)
|
||||||
|
next_seasons = sorted(s for s in seasons_watched if s > last.season)
|
||||||
|
if max_ep_in_season > 0 and last.episode >= max_ep_in_season and next_seasons:
|
||||||
|
next_season = next_seasons[0]
|
||||||
|
next_ep = 1
|
||||||
|
else:
|
||||||
next_season = last.season
|
next_season = last.season
|
||||||
next_ep = last.episode + 1
|
next_ep = last.episode + 1
|
||||||
|
|
||||||
@@ -5510,6 +5556,11 @@ def _route_random_title(params: dict[str, str]) -> None:
|
|||||||
_play_random_title(params.get("plugin", ""))
|
_play_random_title(params.get("plugin", ""))
|
||||||
|
|
||||||
|
|
||||||
|
@_router.route("trakt_menu")
|
||||||
|
def _route_trakt_menu(params: dict[str, str]) -> None:
|
||||||
|
_show_trakt_menu()
|
||||||
|
|
||||||
|
|
||||||
@_router.route("trakt_auth")
|
@_router.route("trakt_auth")
|
||||||
def _route_trakt_auth(params: dict[str, str]) -> None:
|
def _route_trakt_auth(params: dict[str, str]) -> None:
|
||||||
_trakt_authorize()
|
_trakt_authorize()
|
||||||
|
|||||||
@@ -44,7 +44,9 @@ Einstellungen
|
|||||||
|
|
||||||
Menues im Hauptmenue
|
Menues im Hauptmenue
|
||||||
|
|
||||||
Wenn Trakt aktiviert und autorisiert ist, erscheinen im ViewIT-Hauptmenue folgende Eintraege:
|
Wenn Trakt aktiviert ist, erscheint im ViewIT-Hauptmenue ein Untermenüpunkt "Trakt" (nach allen Quellen-Plugins).
|
||||||
|
|
||||||
|
Ein Klick darauf oeffnet das Trakt-Untermenue mit folgenden Eintraegen (nur wenn bereits autorisiert):
|
||||||
|
|
||||||
|
|
||||||
Weiterschauen
|
Weiterschauen
|
||||||
|
|||||||
Reference in New Issue
Block a user