Compare commits

...

4 Commits

5 changed files with 70 additions and 23 deletions

4
.gitignore vendored
View File

@@ -19,3 +19,7 @@ __pycache__/
# Plugin runtime caches # Plugin runtime caches
/addon/plugins/*_cache.json /addon/plugins/*_cache.json
# Projektdokumentation (lokal)
/PROJECT_INDEX.md
/FUNCTION_MAP.md

View File

@@ -1,3 +1,19 @@
## 0.1.78.0-dev - 2026-03-11
- dev: Trakt-Scrobbling fuer alle Wiedergabe-Pfade
## 0.1.77.5-dev - 2026-03-10
- dev: Max. Eintraege pro Seite Setting pro Plugin
## 0.1.77.0-dev - 2026-03-10
- dev: Changelog-Dialog nur anzeigen wenn Eintrag vorhanden
## 0.1.76.5-dev - 2026-03-10
- dev: Versionsfilter fuer 4-teilige Versionsnummern korrigiert
## 0.1.76.0-dev - 2026-03-10 ## 0.1.76.0-dev - 2026-03-10
- dev: bump to 0.1.76.0-dev aeltere Versionen im Update-Dialog, Release-Branch-Zuordnung, README ueberarbeitet - dev: bump to 0.1.76.0-dev aeltere Versionen im Update-Dialog, Release-Branch-Zuordnung, README ueberarbeitet

View File

@@ -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.76.5-dev" provider-name="ViewIt"> <addon id="plugin.video.viewit" name="ViewIt" version="0.1.78.5-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" />

View File

@@ -1418,7 +1418,7 @@ def _extract_changelog_section(changelog_text: str, version: str) -> str:
start = idx start = idx
break break
if start < 0: if start < 0:
return f"Kein Changelog-Abschnitt fuer Version {wanted} gefunden." return ""
end = len(lines) end = len(lines)
for idx in range(start + 1, len(lines)): for idx in range(start + 1, len(lines)):
@@ -1441,7 +1441,7 @@ def _fetch_changelog_for_channel(channel: int, version: str) -> str:
try: try:
text = _read_text_url(url) text = _read_text_url(url)
except Exception: except Exception:
return "Changelog konnte nicht geladen werden." return ""
return _extract_changelog_section(text, version) return _extract_changelog_section(text, version)
@@ -3176,8 +3176,9 @@ def _show_popular(plugin_name: str | None = None, page: int = 1) -> None:
def _show_new_titles(plugin_name: str, page: int = 1, *, action_name: str = "new_titles") -> None: def _show_new_titles(plugin_name: str, page: int = 1, *, action_name: str = "new_titles") -> None:
handle = _get_handle() handle = _get_handle()
page_size = LIST_PAGE_SIZE
page = max(1, int(page or 1)) page = max(1, int(page or 1))
max_items_key = f"{(plugin_name or '').strip().casefold()}_max_page_items"
page_size = _get_setting_int(max_items_key, default=15) or LIST_PAGE_SIZE
plugin_name = (plugin_name or "").strip() plugin_name = (plugin_name or "").strip()
plugin = _discover_plugins().get(plugin_name) plugin = _discover_plugins().get(plugin_name)
@@ -3218,6 +3219,8 @@ def _show_new_titles(plugin_name: str, page: int = 1, *, action_name: str = "new
xbmcplugin.endOfDirectory(handle) xbmcplugin.endOfDirectory(handle)
return return
page_items = [str(t).strip() for t in page_items if t and str(t).strip()] page_items = [str(t).strip() for t in page_items if t and str(t).strip()]
if page_size > 0 and len(page_items) > page_size:
page_items = page_items[:page_size]
page_items.sort(key=lambda value: value.casefold()) page_items.sort(key=lambda value: value.casefold())
else: else:
try: try:
@@ -3284,15 +3287,16 @@ def _show_new_titles(plugin_name: str, page: int = 1, *, action_name: str = "new
meta = plugin_meta.get(title) meta = plugin_meta.get(title)
info_labels, art, cast = _merge_metadata(title, tmdb_info, tmdb_art, tmdb_cast, meta) info_labels, art, cast = _merge_metadata(title, tmdb_info, tmdb_art, tmdb_cast, meta)
info_labels = dict(info_labels or {}) info_labels = dict(info_labels or {})
info_labels.setdefault("mediatype", "movie") is_direct_play = bool(
plugin_name.casefold() == "einschalten"
and _get_setting_bool("einschalten_enable_playback", default=False)
)
info_labels.setdefault("mediatype", "movie" if is_direct_play else "tvshow")
playstate = _title_playstate(plugin_name, title) playstate = _title_playstate(plugin_name, title)
info_labels = _apply_playstate_to_info(dict(info_labels), playstate) info_labels = _apply_playstate_to_info(dict(info_labels), playstate)
display_label = _label_with_duration(title, info_labels) display_label = _label_with_duration(title, info_labels)
display_label = _label_with_playstate(display_label, playstate) display_label = _label_with_playstate(display_label, playstate)
direct_play = bool( direct_play = is_direct_play
plugin_name.casefold() == "einschalten"
and _get_setting_bool("einschalten_enable_playback", default=False)
)
_add_directory_item( _add_directory_item(
handle, handle,
display_label, display_label,
@@ -3378,8 +3382,6 @@ def _show_latest_episodes(plugin_name: str, page: int = 1) -> None:
playstate = _get_playstate(key) playstate = _get_playstate(key)
label = f"{title} - S{season_number:02d}E{episode_number:02d}" label = f"{title} - S{season_number:02d}E{episode_number:02d}"
if airdate:
label = f"{label} ({airdate})"
label = _label_with_playstate(label, playstate) label = _label_with_playstate(label, playstate)
info_labels: dict[str, object] = { info_labels: dict[str, object] = {
@@ -3732,12 +3734,13 @@ def _show_version_selector() -> None:
version = versions[selected] version = versions[selected]
changelog = _fetch_changelog_for_channel(channel, version) changelog = _fetch_changelog_for_channel(channel, version)
viewer = getattr(xbmcgui.Dialog(), "textviewer", None) if changelog:
if callable(viewer): viewer = getattr(xbmcgui.Dialog(), "textviewer", None)
try: if callable(viewer):
viewer(f"Changelog {version}", changelog) try:
except Exception: viewer(f"Changelog {version}", changelog)
pass except Exception:
pass
action = xbmcgui.Dialog().select( action = xbmcgui.Dialog().select(
f"Version {version} installieren?", f"Version {version} installieren?",
@@ -4118,15 +4121,14 @@ def _play_episode(
title_key = (title or "").strip().casefold() title_key = (title or "").strip().casefold()
_tmdb_id = _tmdb_cache_get(_TMDB_ID_CACHE, title_key, 0) _tmdb_id = _tmdb_cache_get(_TMDB_ID_CACHE, title_key, 0)
_imdb_id = "" _imdb_id = ""
trakt_media: dict[str, object] | None = None _kind = _tmdb_cache_get(_MEDIA_TYPE_CACHE, title_key, "tv") if _tmdb_id else "tv"
if _tmdb_id: if _tmdb_id:
_kind = _tmdb_cache_get(_MEDIA_TYPE_CACHE, title_key, "tv")
_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)
trakt_media = { 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,
} }
_play_final_link( _play_final_link(
final_link, final_link,
@@ -4230,6 +4232,19 @@ def _play_episode_url(
if episode_number > 0: if episode_number > 0:
info_labels["episode"] = str(episode_number) info_labels["episode"] = str(episode_number)
display_title = _label_with_duration(display_title, info_labels) display_title = _label_with_duration(display_title, info_labels)
title_key = (title or "").strip().casefold()
_tmdb_id = _tmdb_cache_get(_TMDB_ID_CACHE, title_key, 0)
_imdb_id = ""
_kind = _tmdb_cache_get(_MEDIA_TYPE_CACHE, title_key, "tv") if _tmdb_id else "tv"
if _tmdb_id:
_imdb_id = _fetch_and_cache_imdb_id(title_key, _tmdb_id, _kind)
_set_trakt_ids_property(title, _tmdb_id, _imdb_id)
trakt_media: dict[str, object] = {
"title": title, "tmdb_id": _tmdb_id, "imdb_id": _imdb_id, "kind": _kind,
"season": season_number or 0, "episode": episode_number or 0,
}
_play_final_link( _play_final_link(
final_link, final_link,
display_title=display_title, display_title=display_title,
@@ -4237,6 +4252,7 @@ def _play_episode_url(
art=art, art=art,
cast=cast, cast=cast,
resolve_handle=resolve_handle, resolve_handle=resolve_handle,
trakt_media=trakt_media,
) )
_track_playback_and_update_state_async( _track_playback_and_update_state_async(
_playstate_key(plugin_name=plugin_name, title=title, season=season_label, episode=episode_label) _playstate_key(plugin_name=plugin_name, title=title, season=season_label, episode=episode_label)

View File

@@ -38,6 +38,17 @@
<setting id="tmdb_log_responses" type="bool" label="TMDB API-Antworten loggen" default="false" /> <setting id="tmdb_log_responses" type="bool" label="TMDB API-Antworten loggen" default="false" />
</category> </category>
<category label="Anzeige">
<setting id="filmpalast_max_page_items" type="number" label="Filmpalast: Max. Eintraege pro Seite" default="15" />
<setting id="topstreamfilm_max_page_items" type="number" label="TopStream: Max. Eintraege pro Seite" default="15" />
<setting id="aniworld_max_page_items" type="number" label="AniWorld: Max. Eintraege pro Seite" default="15" />
<setting id="netzkkino_max_page_items" type="number" label="Netzkino: Max. Eintraege pro Seite" default="15" />
<setting id="kkiste_max_page_items" type="number" label="KKiste: Max. Eintraege pro Seite" default="15" />
<setting id="hdfilme_max_page_items" type="number" label="HDFilme: Max. Eintraege pro Seite" default="15" />
<setting id="moflix_max_page_items" type="number" label="Moflix: Max. Eintraege pro Seite" default="15" />
<setting id="einschalten_max_page_items" type="number" label="Einschalten: Max. Eintraege pro Seite" default="15" />
</category>
<category label="Wiedergabe"> <category label="Wiedergabe">
<setting id="autoplay_enabled" type="bool" label="Autoplay (bevorzugten Hoster automatisch waehlen)" default="false" /> <setting id="autoplay_enabled" type="bool" label="Autoplay (bevorzugten Hoster automatisch waehlen)" default="false" />
<setting id="preferred_hoster" type="text" label="Bevorzugter Hoster" default="voe" /> <setting id="preferred_hoster" type="text" label="Bevorzugter Hoster" default="voe" />