# -*- coding: utf-8 -*- ''' Genesis Add-on Copyright (C) 2015 lambda -Mofidied by The Crew -Copyright (C) 2019 The Crew This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . ''' import os import sys import six from six.moves import urllib_parse from kodi_six import xbmc, xbmcaddon, xbmcgui, xbmcplugin, xbmcvfs def six_encode(txt, char='utf-8', errors='replace'): if six.PY2 and isinstance(txt, six.text_type): txt = txt.encode(char, errors=errors) return txt def six_decode(txt, char='utf-8', errors='replace'): if six.PY3 and isinstance(txt, six.binary_type): txt = txt.decode(char, errors=errors) return txt def getKodiVersion(): return int(xbmc.getInfoLabel("System.BuildVersion").split(".")[0]) integer = 1000 lang = xbmcaddon.Addon().getLocalizedString lang2 = xbmc.getLocalizedString setting = xbmcaddon.Addon().getSetting setSetting = xbmcaddon.Addon().setSetting addon = xbmcaddon.Addon addItem = xbmcplugin.addDirectoryItem addItems = xbmcplugin.addDirectoryItems item = xbmcgui.ListItem directory = xbmcplugin.endOfDirectory content = xbmcplugin.setContent sortMethod = xbmcplugin.addSortMethod property = xbmcplugin.setProperty addonInfo = xbmcaddon.Addon().getAddonInfo infoLabel = xbmc.getInfoLabel condVisibility = xbmc.getCondVisibility jsonrpc = xbmc.executeJSONRPC window = xbmcgui.Window(10000) dialog = xbmcgui.Dialog() progressDialog = xbmcgui.DialogProgress() progressDialogBG = xbmcgui.DialogProgressBG() windowDialog = xbmcgui.WindowDialog() button = xbmcgui.ControlButton image = xbmcgui.ControlImage getCurrentDialogId = xbmcgui.getCurrentWindowDialogId() keyboard = xbmc.Keyboard monitor = xbmc.Monitor() # Modified `sleep` command that honors a user exit request def sleep(time): while time > 0 and not monitor.abortRequested(): xbmc.sleep(min(100, time)) time = time - 100 execute = xbmc.executebuiltin skin = xbmc.getSkinDir() player = xbmc.Player() playlist = xbmc.PlayList(xbmc.PLAYLIST_VIDEO) resolve = xbmcplugin.setResolvedUrl legalFilename = xbmc.makeLegalFilename if int(getKodiVersion()) < 19 else xbmcvfs.makeLegalFilename openFile = xbmcvfs.File makeFile = xbmcvfs.mkdir deleteFile = xbmcvfs.delete deleteDir = xbmcvfs.rmdir listDir = xbmcvfs.listdir transPath = xbmc.translatePath if getKodiVersion() < 19 else xbmcvfs.translatePath skinPath = transPath('special://skin/') addonPath = transPath(addonInfo('path')) dataPath = transPath(addonInfo('profile')) settingsFile = os.path.join(dataPath, 'settings.xml') viewsFile = os.path.join(dataPath, 'views.db') bookmarksFile = os.path.join(dataPath, 'bookmarks.db') providercacheFile = os.path.join(dataPath, 'providers.13.db') metacacheFile = os.path.join(dataPath, 'meta.5.db') searchFile = os.path.join(dataPath, 'search.1.db') libcacheFile = os.path.join(dataPath, 'library.db') cacheFile = os.path.join(dataPath, 'cache.db') dbFile = os.path.join(dataPath, 'debridcache.db') key = "RgUkXp2s5v8x/A?D(G+KbPeShVmYq3t6" iv = "p2s5v8y/B?E(H+Mb" def metadataClean(metadata): # Filter out non-existing/custom keys. Otherise there are tons of errors in Kodi 18 log. if metadata == None: return metadata allowed = ['genre', 'country', 'year', 'episode', 'season', 'sortepisode', 'sortseason', 'episodeguide', 'showlink', 'top250', 'setid', 'tracknumber', 'rating', 'userrating', 'watched', 'playcount', 'overlay', 'cast', 'castandrole', 'director', 'mpaa', 'plot', 'plotoutline', 'title', 'originaltitle', 'sorttitle', 'duration', 'studio', 'tagline', 'writer', 'tvshowtitle', 'premiered', 'status', 'set', 'setoverview', 'tag', 'imdbnumber', 'code', 'aired', 'credits', 'lastplayed', 'album', 'artist', 'votes', 'path', 'trailer', 'dateadded', 'mediatype', 'dbid'] return {k: v for k, v in six.iteritems(metadata) if k in allowed} def addonIcon(): theme = appearance() art = artPath() if not (art is None and theme in ['-', '']): return os.path.join(art, 'icon.png') return addonInfo('icon') def addonThumb(): theme = appearance() art = artPath() if not (art is None and theme in ['-', '']): return os.path.join(art, 'poster.png') elif theme == '-': return 'DefaultFolder.png' return addonInfo('icon') def addonPoster(): theme = appearance() art = artPath() if not (art is None and theme in ['-', '']): return os.path.join(art, 'poster.png') return 'DefaultVideo.png' def addonBanner(): theme = appearance() art = artPath() if not (art is None and theme in ['-', '']): return os.path.join(art, 'banner.png') return 'DefaultVideo.png' def addonFanart(): theme = appearance() art = artPath() if not (art is None and theme in ['-', '']): return os.path.join(art, 'fanart.jpg') return addonInfo('fanart') def addonNext(): theme = appearance() art = artPath() if not (art is None and theme in ['-', '']): return os.path.join(art, 'next.png') return 'DefaultVideo.png' def addonId(): return addonInfo('id') def addonName(): return addonInfo('name') def get_plugin_url(queries): try: query = urllib_parse.urlencode(queries) except UnicodeEncodeError: for k in queries: if isinstance(queries[k], six.text_type): queries[k] = six_encode(queries[k]) query = urllib_parse.urlencode(queries) addon_id = sys.argv[0] if not addon_id: addon_id = addonId() return addon_id + '?' + query def artPath(): theme = appearance() if theme in ['-', '']: return elif condVisibility('System.HasAddon(script.thecrew.artwork)'): return os.path.join( xbmcaddon.Addon('script.thecrew.artwork').getAddonInfo('path'), 'resources', 'media', theme) def appearance(): appearance = setting('appearance.1').lower() if condVisibility( 'System.HasAddon(script.thecrew.artwork)') else setting('appearance.alt').lower() return appearance def artwork(): execute('RunPlugin(plugin://script.thecrew.artwork)') def infoDialog(message, heading=addonInfo('name'), icon='', time=3000, sound=False): if icon == '': icon = addonIcon() elif icon == 'INFO': icon = xbmcgui.NOTIFICATION_INFO elif icon == 'WARNING': icon = xbmcgui.NOTIFICATION_WARNING elif icon == 'ERROR': icon = xbmcgui.NOTIFICATION_ERROR dialog.notification(heading, message, icon, time, sound=sound) def yesnoDialog(message, heading=addonInfo('name'), nolabel='', yeslabel=''): if int(getKodiVersion()) < 19: return dialog.yesno(heading, message, '', '', nolabel, yeslabel) else: return dialog.yesno(heading, message, nolabel, yeslabel) #TC 2/01/19 started def selectDialog(list, heading=addonInfo('name')): return dialog.select(heading, list) def metaFile(): if condVisibility('System.HasAddon(script.thecrew.metadata)'): return os.path.join( xbmcaddon.Addon('script.thecrew.metadata').getAddonInfo('path'), 'resources', 'data', 'meta.db') def apiLanguage(ret_name=None): langDict = { 'Bulgarian': 'bg', 'Chinese': 'zh', 'Croatian': 'hr', 'Czech': 'cs', 'Danish': 'da', 'Dutch': 'nl', 'English': 'en', 'Finnish': 'fi', 'French': 'fr', 'German': 'de', 'Greek': 'el', 'Hebrew': 'he', 'Hungarian': 'hu', 'Italian': 'it', 'Japanese': 'ja', 'Korean': 'ko', 'Norwegian': 'no', 'Polish': 'pl', 'Portuguese': 'pt', 'Romanian': 'ro', 'Russian': 'ru', 'Serbian': 'sr', 'Slovak': 'sk', 'Slovenian': 'sl', 'Spanish': 'es', 'Swedish': 'sv', 'Thai': 'th', 'Turkish': 'tr', 'Ukrainian': 'uk'} trakt = ['bg', 'cs', 'da', 'de', 'el', 'en', 'es', 'fi', 'fr', 'he', 'hr', 'hu', 'it', 'ja', 'ko', 'nl', 'no', 'pl', 'pt', 'ro', 'ru', 'sk', 'sl', 'sr', 'sv', 'th', 'tr', 'uk', 'zh'] tvdb = ['en', 'sv', 'no', 'da', 'fi', 'nl', 'de', 'it', 'es', 'fr', 'pl', 'hu', 'el', 'tr', 'ru', 'he', 'ja', 'pt', 'zh', 'cs', 'sl', 'hr', 'ko'] youtube = ['gv', 'gu', 'gd', 'ga', 'gn', 'gl', 'ty', 'tw', 'tt', 'tr', 'ts', 'tn', 'to', 'tl', 'tk', 'th', 'ti', 'tg', 'te', 'ta', 'de', 'da', 'dz', 'dv', 'qu', 'zh', 'za', 'zu', 'wa', 'wo', 'jv', 'ja', 'ch', 'co', 'ca', 'ce', 'cy', 'cs', 'cr', 'cv', 'cu', 'ps', 'pt', 'pa', 'pi', 'pl', 'mg', 'ml', 'mn', 'mi', 'mh', 'mk', 'mt', 'ms', 'mr', 'my', 've', 'vi', 'is', 'iu', 'it', 'vo', 'ii', 'ik', 'io', 'ia', 'ie', 'id', 'ig', 'fr', 'fy', 'fa', 'ff', 'fi', 'fj', 'fo', 'ss', 'sr', 'sq', 'sw', 'sv', 'su', 'st', 'sk', 'si', 'so', 'sn', 'sm', 'sl', 'sc', 'sa', 'sg', 'se', 'sd', 'lg', 'lb', 'la', 'ln', 'lo', 'li', 'lv', 'lt', 'lu', 'yi', 'yo', 'el', 'eo', 'en', 'ee', 'eu', 'et', 'es', 'ru', 'rw', 'rm', 'rn', 'ro', 'be', 'bg', 'ba', 'bm', 'bn', 'bo', 'bh', 'bi', 'br', 'bs', 'om', 'oj', 'oc', 'os', 'or', 'xh', 'hz', 'hy', 'hr', 'ht', 'hu', 'hi', 'ho', 'ha', 'he', 'uz', 'ur', 'uk', 'ug', 'aa', 'ab', 'ae', 'af', 'ak', 'am', 'an', 'as', 'ar', 'av', 'ay', 'az', 'nl', 'nn', 'no', 'na', 'nb', 'nd', 'ne', 'ng', 'ny', 'nr', 'nv', 'ka', 'kg', 'kk', 'kj', 'ki', 'ko', 'kn', 'km', 'kl', 'ks', 'kr', 'kw', 'kv', 'ku', 'ky'] name = None name = setting('api.language') if not name: name = 'AUTO' if name[-1].isupper(): try: name = xbmc.getLanguage(xbmc.ENGLISH_NAME).split(' ')[0] except Exception: pass try: name = langDict[name] except Exception: name = 'en' lang = {'trakt': name} if name in trakt else {'trakt': 'en'} lang['tvdb'] = name if name in tvdb else 'en' lang['youtube'] = name if name in youtube else 'en' if ret_name: lang['trakt'] = [i[0] for i in six.iteritems(langDict)if i[1] == lang['trakt']][0] lang['tvdb'] = [i[0] for i in six.iteritems(langDict) if i[1] == lang['tvdb']][0] lang['youtube'] = [i[0] for i in six.iteritems(langDict) if i[1] == lang['youtube']][0] return lang def version(): num = '' try: version = addon('xbmc.addon').getAddonInfo('version') except Exception: version = '999' for i in version: if i.isdigit(): num += i else: break return int(num) def cdnImport(uri, name): import imp from resources.lib.modules import client path = os.path.join(dataPath, 'py' + name) path = six_decode(path) deleteDir(os.path.join(path, ''), force=True) makeFile(dataPath) makeFile(path) r = client.request(uri) p = os.path.join(path, name + '.py') f = openFile(p, 'w') f.write(r) f.close() m = imp.load_source(name, p) deleteDir(os.path.join(path, ''), force=True) return m def openSettings(query=None, id=addonInfo('id')): try: idle() execute('Addon.OpenSettings(%s)' % id) if query is None: raise Exception() c, f = query.split('.') if int(getKodiVersion()) >= 18: execute('SetFocus(%i)' % (int(c) - 100)) execute('SetFocus(%i)' % (int(f) - 80)) else: execute('SetFocus(%i)' % (int(c) + 100)) execute('SetFocus(%i)' % (int(f) + 200)) except Exception: return def getCurrentViewId(): win = xbmcgui.Window(xbmcgui.getCurrentWindowId()) return str(win.getFocusId()) def refresh(): return execute('Container.Refresh') def busy(): if int(getKodiVersion()) >= 18: return execute('ActivateWindow(busydialognocancel)') else: return execute('ActivateWindow(busydialog)') def idle(): if int(getKodiVersion()) >= 18: return execute('Dialog.Close(busydialognocancel)') else: return execute('Dialog.Close(busydialog)') def queueItem(): return execute('Action(Queue)') def installAddon(addon_id): addon_path = os.path.join(transPath('special://home/addons'), addon_id) if not os.path.exists(addon_path) == True: xbmc.executebuiltin('InstallAddon(%s)' % (addon_id)) else: infoDialog('{0} is already installed'.format(addon_id), sound=True)