plugin.video.viervijfzes/resources/lib/modules/catalog.py

317 lines
12 KiB
Python
Raw Normal View History

2020-03-19 16:45:31 +01:00
# -*- coding: utf-8 -*-
""" Catalog module """
from __future__ import absolute_import, division, unicode_literals
import logging
from datetime import datetime
import dateutil.tz
2020-03-19 16:45:31 +01:00
from resources.lib import kodiutils
from resources.lib.kodiutils import TitleItem
from resources.lib.modules.menu import Menu
from resources.lib.viervijfzes.auth import AuthApi
from resources.lib.viervijfzes.content import CACHE_PREVENT, ContentApi, UnavailableException
2020-03-19 16:45:31 +01:00
_LOGGER = logging.getLogger(__name__)
2020-03-19 16:45:31 +01:00
class Catalog:
""" Menu code related to the catalog """
def __init__(self):
""" Initialise object """
self._auth = AuthApi(kodiutils.get_setting('username'), kodiutils.get_setting('password'), kodiutils.get_tokens_path())
self._api = ContentApi(self._auth, cache_path=kodiutils.get_cache_path())
2020-03-19 16:45:31 +01:00
def show_catalog(self):
""" Show all the programs of all channels """
try:
items = self._api.get_programs()
2020-03-19 16:45:31 +01:00
except Exception as ex:
kodiutils.notification(message=str(ex))
raise
2020-04-20 08:59:10 +02:00
listing = [Menu.generate_titleitem(item) for item in items]
2020-03-19 16:45:31 +01:00
# Sort items by title
2020-03-19 16:45:31 +01:00
# Used for A-Z listing or when movies and episodes are mixed.
kodiutils.show_listing(listing, 30003, content='tvshows', sort='title')
2020-03-19 16:45:31 +01:00
def show_catalog_channel(self, channel):
""" Show the programs of a specific channel
:type channel: str
"""
try:
items = self._api.get_programs(channel)
except Exception as ex:
kodiutils.notification(message=str(ex))
raise
listing = []
for item in items:
2020-04-20 08:59:10 +02:00
listing.append(Menu.generate_titleitem(item))
2020-03-19 16:45:31 +01:00
# Sort items by title
2020-03-19 16:45:31 +01:00
# Used for A-Z listing or when movies and episodes are mixed.
kodiutils.show_listing(listing, 30003, content='tvshows', sort='title')
2020-03-19 16:45:31 +01:00
def show_program(self, program_id):
2020-03-19 16:45:31 +01:00
""" Show a program from the catalog
:type program_id: str
"""
try:
program = self._api.get_program(program_id, extract_clips=True, cache=CACHE_PREVENT) # Use CACHE_PREVENT since we want fresh data
2020-03-19 16:45:31 +01:00
except UnavailableException:
kodiutils.ok_dialog(message=kodiutils.localize(30717)) # This program is not available in the catalogue.
2020-03-19 16:45:31 +01:00
kodiutils.end_of_directory()
return
2020-04-20 08:59:10 +02:00
if not program.episodes and not program.clips:
kodiutils.ok_dialog(message=kodiutils.localize(30717)) # This program is not available in the catalogue.
2020-03-19 16:45:31 +01:00
kodiutils.end_of_directory()
return
2020-04-20 08:59:10 +02:00
# Go directly to the season when we have only one season and no clips
if not program.clips and len(program.seasons) == 1:
self.show_program_season(program_id, list(program.seasons.values())[0].uuid)
2020-03-19 16:45:31 +01:00
return
listing = []
# Add an '* All seasons' entry when configured in Kodi
2020-04-20 08:59:10 +02:00
if program.seasons and kodiutils.get_global_setting('videolibrary.showallitems') is True:
2020-03-19 16:45:31 +01:00
listing.append(
2020-03-20 13:53:21 +01:00
TitleItem(
title='* %s' % kodiutils.localize(30204), # * All seasons
path=kodiutils.url_for('show_catalog_program_season', program=program_id, season='-1'),
2020-03-20 13:53:21 +01:00
art_dict={
'fanart': program.fanart,
'poster': program.poster,
'landscape': program.thumb,
2020-03-20 13:53:21 +01:00
},
info_dict={
'tvshowtitle': program.title,
'title': kodiutils.localize(30204), # All seasons
'plot': program.description,
'set': program.title,
}
)
2020-03-19 16:45:31 +01:00
)
# Add the seasons
for season in list(program.seasons.values()):
2020-03-19 16:45:31 +01:00
listing.append(
2020-03-20 13:53:21 +01:00
TitleItem(
title=season.title, # kodiutils.localize(30205, season=season.number), # Season {season}
path=kodiutils.url_for('show_catalog_program_season', program=program_id, season=season.uuid),
2020-03-20 13:53:21 +01:00
art_dict={
'fanart': program.fanart,
'poster': program.poster,
'landscape': program.thumb,
2020-03-20 13:53:21 +01:00
},
info_dict={
'tvshowtitle': program.title,
'title': kodiutils.localize(30205, season=season.number), # Season {season}
'plot': season.description or program.description,
2020-03-20 13:53:21 +01:00
'set': program.title,
}
)
2020-03-19 16:45:31 +01:00
)
2020-04-20 08:59:10 +02:00
# Add Clips
if program.clips:
listing.append(
TitleItem(
title=kodiutils.localize(30059, program=program.title), # Clips for {program}
path=kodiutils.url_for('show_catalog_program_clips', program=program_id),
2020-04-20 08:59:10 +02:00
art_dict={
'fanart': program.fanart,
'poster': program.poster,
'landscape': program.thumb,
2020-04-20 08:59:10 +02:00
},
info_dict={
'tvshowtitle': program.title,
'title': kodiutils.localize(30059, program=program.title), # Clips for {program}
'plot': kodiutils.localize(30060, program=program.title), # Watch short clips of {program}
'set': program.title,
}
)
)
2020-03-19 16:45:31 +01:00
# Sort by label. Some programs return seasons unordered.
kodiutils.show_listing(listing, 30003, content='tvshows')
2020-03-19 16:45:31 +01:00
def show_program_season(self, program_id, season_uuid):
2020-03-19 16:45:31 +01:00
""" Show the episodes of a program from the catalog
:type program_id: str
:type season_uuid: str
2020-03-19 16:45:31 +01:00
"""
try:
program = self._api.get_program(program_id)
2020-03-19 16:45:31 +01:00
except UnavailableException:
kodiutils.ok_dialog(message=kodiutils.localize(30717)) # This program is not available in the catalogue.
2020-03-19 16:45:31 +01:00
kodiutils.end_of_directory()
return
if season_uuid == "-1":
2020-03-19 16:45:31 +01:00
# Show all episodes
episodes = program.episodes
else:
# Show the episodes of the season that was selected
episodes = [e for e in program.episodes if e.season_uuid == season_uuid]
2020-03-19 16:45:31 +01:00
2020-04-20 08:59:10 +02:00
listing = [Menu.generate_titleitem(episode) for episode in episodes]
2020-03-19 16:45:31 +01:00
# Sort by episode number by default. Takes seasons into account.
kodiutils.show_listing(listing, 30003, content='episodes', sort=['episode', 'duration'])
2020-04-20 08:59:10 +02:00
def show_program_clips(self, program_id):
2020-04-20 08:59:10 +02:00
""" Show the clips of a program from the catalog
:type program_id: str
"""
try:
# We need to query the backend, since we don't cache clips.
program = self._api.get_program(program_id, extract_clips=True, cache=CACHE_PREVENT)
2020-04-20 08:59:10 +02:00
except UnavailableException:
kodiutils.ok_dialog(message=kodiutils.localize(30717)) # This program is not available in the catalogue.
kodiutils.end_of_directory()
return
listing = [Menu.generate_titleitem(episode) for episode in program.clips]
# Sort like we get our results back.
kodiutils.show_listing(listing, 30003, content='episodes')
def show_categories(self):
""" Shows the categories """
categories = self._api.get_categories()
listing = []
for category in categories:
listing.append(TitleItem(title=category.title,
path=kodiutils.url_for('show_category', category=category.uuid),
info_dict={
'title': category.title,
}))
kodiutils.show_listing(listing, 30003, sort=['title'])
def show_category(self, uuid):
""" Shows a category """
programs = self._api.get_category_content(int(uuid))
listing = [
Menu.generate_titleitem(program) for program in programs
]
kodiutils.show_listing(listing, 30003, content='tvshows')
def show_recommendations(self):
""" Shows the recommendations """
# "Meest bekeken" has a specific API endpoint, the other categories are scraped from the website.
listing = [
TitleItem(title='Meest bekeken',
path=kodiutils.url_for('show_recommendations_category', category='meest-bekeken'),
info_dict={
'title': 'Meest bekeken',
})
]
recommendations = self._api.get_recommendation_categories()
for category in recommendations:
listing.append(TitleItem(title=category.title,
path=kodiutils.url_for('show_recommendations_category', category=category.uuid),
info_dict={
'title': category.title,
}))
kodiutils.show_listing(listing, 30005, content='tvshows')
def show_recommendations_category(self, uuid):
""" Shows the a category of the recommendations """
if uuid == 'meest-bekeken':
programs = self._api.get_popular_programs()
episodes = []
else:
recommendations = self._api.get_recommendation_categories()
category = next(category for category in recommendations if category.uuid == uuid)
programs = category.programs
episodes = category.episodes
listing = []
for episode in episodes:
title_item = Menu.generate_titleitem(episode)
title_item.info_dict['title'] = episode.program_title + ' - ' + title_item.title
listing.append(title_item)
for program in programs:
listing.append(Menu.generate_titleitem(program))
kodiutils.show_listing(listing, 30005, content='tvshows')
def show_mylist(self):
""" Show all the programs of all channels """
try:
mylist, _ = self._auth.get_dataset('myList', 'myList')
except Exception as ex:
kodiutils.notification(message=str(ex))
raise
items = []
if mylist:
for item in mylist:
program = self._api.get_program_by_uuid(item.get('id'))
if program:
program.my_list = True
items.append(program)
listing = [Menu.generate_titleitem(item) for item in items]
# Sort items by title
# Used for A-Z listing or when movies and episodes are mixed.
kodiutils.show_listing(listing, 30011, content='tvshows', sort='title')
def mylist_add(self, uuid):
""" Add a program to My List """
if not uuid:
kodiutils.end_of_directory()
return
mylist, sync_info = self._auth.get_dataset('myList', 'myList')
if not mylist:
mylist = []
if uuid not in [item.get('id') for item in mylist]:
# Python 2.7 doesn't support .timestamp(), and windows doesn't do '%s', so we need to calculate it ourself
epoch = datetime(1970, 1, 1, tzinfo=dateutil.tz.gettz('UTC'))
now = datetime.now(tz=dateutil.tz.gettz('UTC'))
timestamp = int((now - epoch).total_seconds()) * 1000
mylist.append({
'id': uuid,
'timestamp': timestamp,
})
self._auth.put_dataset('myList', 'myList', mylist, sync_info)
kodiutils.end_of_directory()
def mylist_del(self, uuid):
""" Remove a program from My List """
if not uuid:
kodiutils.end_of_directory()
return
mylist, sync_info = self._auth.get_dataset('myList', 'myList')
if not mylist:
mylist = []
new_mylist = [item for item in mylist if item.get('id') != uuid]
self._auth.put_dataset('myList', 'myList', new_mylist, sync_info)
kodiutils.end_of_directory()