PlexKodiConnect/resources/lib/videonodes.py

397 lines
16 KiB
Python
Raw Normal View History

2015-12-24 14:07:00 -06:00
# -*- coding: utf-8 -*-
###############################################################################
2015-12-24 14:07:00 -06:00
import shutil
import xml.etree.ElementTree as etree
import xbmc
import xbmcvfs
import utils
###############################################################################
2015-12-24 14:07:00 -06:00
@utils.logging
2015-12-24 14:07:00 -06:00
class VideoNodes(object):
def __init__(self):
2016-02-17 02:13:37 -06:00
self.kodiversion = int(xbmc.getInfoLabel('System.BuildVersion')[:2])
2015-12-24 14:07:00 -06:00
def commonRoot(self, order, label, tagname, roottype=1):
if roottype == 0:
# Index
root = etree.Element('node', attrib={'order': "%s" % order})
elif roottype == 1:
# Filter
root = etree.Element('node', attrib={'order': "%s" % order, 'type': "filter"})
etree.SubElement(root, 'match').text = "all"
# Add tag rule
rule = etree.SubElement(root, 'rule', attrib={'field': "tag", 'operator': "is"})
etree.SubElement(rule, 'value').text = tagname
else:
# Folder
root = etree.Element('node', attrib={'order': "%s" % order, 'type': "folder"})
etree.SubElement(root, 'label').text = label
etree.SubElement(root, 'icon').text = "special://home/addons/plugin.video.plexkodiconnect/icon.png"
2015-12-24 14:07:00 -06:00
return root
def viewNode(self, indexnumber, tagname, mediatype, viewtype, viewid, delete=False):
2016-03-03 09:04:15 +01:00
# Plex: reassign mediatype due to Kodi inner workings
mediatypes = {
'movie': 'movies',
'show': 'tvshows',
'photo': 'photos',
'homevideo': 'homevideos',
'musicvideos': 'musicvideos'
}
mediatype = mediatypes[mediatype]
2016-02-04 06:22:33 -06:00
window = utils.window
2015-12-24 14:07:00 -06:00
kodiversion = self.kodiversion
if viewtype == "mixed":
dirname = "%s-%s" % (viewid, mediatype)
2015-12-24 14:07:00 -06:00
else:
dirname = viewid
2015-12-24 14:07:00 -06:00
path = xbmc.translatePath("special://profile/library/video/").decode('utf-8')
nodepath = xbmc.translatePath(
"special://profile/library/video/Plex-%s/" % dirname).decode('utf-8')
2015-12-24 14:07:00 -06:00
# Verify the video directory
# KODI BUG
# Kodi caches the result of exists for directories
# so try creating a file
if utils.IfExists(path) is False:
2015-12-24 14:07:00 -06:00
shutil.copytree(
src=xbmc.translatePath("special://xbmc/system/library/video").decode('utf-8'),
dst=xbmc.translatePath("special://profile/library/video").decode('utf-8'))
2015-12-24 14:07:00 -06:00
# Create the node directory
if mediatype != "photo":
if utils.IfExists(nodepath) is False:
# folder does not exist yet
self.logMsg('Creating folder %s' % nodepath, 1)
xbmcvfs.mkdirs(nodepath.encode('utf-8'))
if delete:
dirs, files = xbmcvfs.listdir(nodepath.encode('utf-8'))
for file in files:
xbmcvfs.delete(
(nodepath + file.decode('utf-8')).encode('utf-8'))
self.logMsg("Sucessfully removed videonode: %s."
% tagname, 1)
return
2015-12-24 14:07:00 -06:00
# Create index entry
nodeXML = "%sindex.xml" % nodepath
# Set windows property
path = "library://video/Plex-%s/" % dirname
2015-12-24 14:07:00 -06:00
for i in range(1, indexnumber):
# Verify to make sure we don't create duplicates
2016-02-04 06:22:33 -06:00
if window('Emby.nodes.%s.index' % i) == path:
2015-12-24 14:07:00 -06:00
return
if mediatype == "photo":
path = "plugin://plugin.video.plexkodiconnect/?id=%s&mode=getsubfolders" % indexnumber
2016-02-04 06:22:33 -06:00
window('Emby.nodes.%s.index' % indexnumber, value=path)
2015-12-24 14:07:00 -06:00
# Root
if not mediatype == "photo":
2016-02-22 17:56:08 -06:00
if viewtype == "mixed":
specialtag = "%s-%s" % (tagname, mediatype)
2016-02-22 20:42:23 -06:00
root = self.commonRoot(order=0, label=specialtag, tagname=tagname, roottype=0)
2016-02-22 17:56:08 -06:00
else:
root = self.commonRoot(order=0, label=tagname, tagname=tagname, roottype=0)
try:
utils.indent(root)
except: pass
etree.ElementTree(root).write(nodeXML)
2015-12-24 14:07:00 -06:00
nodetypes = {
'1': "all",
'2': "recent",
'3': "recentepisodes",
'4': "inprogress",
'5': "inprogressepisodes",
'6': "unwatched",
2016-01-06 08:59:41 +01:00
'7': "nextepisodes",
2015-12-24 14:07:00 -06:00
'8': "sets",
'9': "genres",
'10': "random",
2016-01-13 01:03:35 +01:00
'11': "recommended",
2015-12-24 14:07:00 -06:00
}
mediatypes = {
# label according to nodetype per mediatype
2016-03-03 09:04:15 +01:00
'movies':
2016-01-13 01:03:35 +01:00
{
2015-12-24 14:07:00 -06:00
'1': tagname,
'2': 30174,
'4': 30177,
'6': 30189,
'8': 20434,
'9': 135,
'10': 30229,
2016-01-13 01:03:35 +01:00
'11': 30230
},
2015-12-24 14:07:00 -06:00
2016-03-03 09:04:15 +01:00
'tvshows':
2016-01-13 01:03:35 +01:00
{
2015-12-24 14:07:00 -06:00
'1': tagname,
'2': 30170,
'3': 30175,
'4': 30171,
'5': 30178,
'7': 30179,
'9': 135,
'10': 30229,
2016-01-13 01:03:35 +01:00
'11': 30230
},
2016-03-03 09:04:15 +01:00
'homevideoss':
2016-01-13 01:03:35 +01:00
{
'1': tagname,
2016-01-13 01:03:35 +01:00
'2': 30251,
'11': 30253
},
2016-03-03 09:04:15 +01:00
'photos':
2016-01-13 01:03:35 +01:00
{
'1': tagname,
2016-01-13 01:03:35 +01:00
'2': 30252,
'8': 30255,
'11': 30254
},
2016-02-04 06:22:33 -06:00
2016-01-18 20:17:14 +01:00
'musicvideos':
{
'1': tagname,
'2': 30256,
'4': 30257,
'6': 30258
2016-02-04 06:22:33 -06:00
}
2015-12-24 14:07:00 -06:00
}
nodes = mediatypes[mediatype]
for node in nodes:
nodetype = nodetypes[node]
nodeXML = "%s%s_%s.xml" % (nodepath, viewid, nodetype)
2015-12-24 14:07:00 -06:00
# Get label
stringid = nodes[node]
2016-02-17 02:13:37 -06:00
if node != "1":
2015-12-24 14:07:00 -06:00
label = utils.language(stringid)
if not label:
label = xbmc.getLocalizedString(stringid)
else:
label = stringid
# Set window properties
if (mediatype == "homevideos" or mediatype == "photo") and nodetype == "all":
# Custom query
path = ("plugin://plugin.video.plexkodiconnect/?id=%s&mode=browsecontent&type=%s"
2016-03-02 17:29:20 +01:00
% (tagname, mediatype))
elif (mediatype == "homevideos" or mediatype == "photo"):
# Custom query
path = ("plugin://plugin.video.plexkodiconnect/?id=%s&mode=browsecontent&type=%s&folderid=%s"
2016-03-02 17:29:20 +01:00
% (tagname, mediatype, nodetype))
elif nodetype == "nextepisodes":
2015-12-24 14:07:00 -06:00
# Custom query
path = "plugin://plugin.video.plexkodiconnect/?id=%s&mode=nextup&limit=50" % tagname
2015-12-24 14:07:00 -06:00
elif kodiversion == 14 and nodetype == "recentepisodes":
# Custom query
path = "plugin://plugin.video.plexkodiconnect/?id=%s&mode=recentepisodes&limit=50" % tagname
2015-12-24 14:07:00 -06:00
elif kodiversion == 14 and nodetype == "inprogressepisodes":
# Custom query
path = "plugin://plugin.video.plexkodiconnect/?id=%s&mode=inprogressepisodes&limit=50"% tagname
2015-12-24 14:07:00 -06:00
else:
path = "library://video/Plex-%s/%s_%s.xml" % (dirname, viewid, nodetype)
if mediatype == "photo":
windowpath = "ActivateWindow(Pictures,%s,return)" % path
else:
windowpath = "ActivateWindow(Video,%s,return)" % path
2015-12-24 14:07:00 -06:00
if nodetype == "all":
if viewtype == "mixed":
templabel = "%s-%s" % (tagname, mediatype)
2015-12-24 14:07:00 -06:00
else:
templabel = label
embynode = "Emby.nodes.%s" % indexnumber
2016-02-04 06:22:33 -06:00
window('%s.title' % embynode, value=templabel)
window('%s.path' % embynode, value=windowpath)
window('%s.content' % embynode, value=path)
window('%s.type' % embynode, value=mediatype)
2015-12-24 14:07:00 -06:00
else:
embynode = "Emby.nodes.%s.%s" % (indexnumber, nodetype)
2016-02-04 06:22:33 -06:00
window('%s.title' % embynode, value=label)
window('%s.path' % embynode, value=windowpath)
window('%s.content' % embynode, value=path)
2015-12-24 14:07:00 -06:00
if mediatype == "photo":
2016-02-04 06:22:33 -06:00
# For photos, we do not create a node in videos but we do want the window props
# to be created.
# To do: add our photos nodes to kodi picture sources somehow
continue
2016-03-07 15:31:07 +01:00
if xbmcvfs.exists(nodeXML.encode('utf-8')):
2015-12-24 14:07:00 -06:00
# Don't recreate xml if already exists
continue
# Create the root
2016-02-04 06:22:33 -06:00
if (nodetype == "nextepisodes" or mediatype == "homevideos" or
(kodiversion == 14 and nodetype in ('recentepisodes', 'inprogressepisodes'))):
2015-12-24 14:07:00 -06:00
# Folder type with plugin path
root = self.commonRoot(order=node, label=label, tagname=tagname, roottype=2)
etree.SubElement(root, 'path').text = path
etree.SubElement(root, 'content').text = "episodes"
else:
root = self.commonRoot(order=node, label=label, tagname=tagname)
if nodetype in ('recentepisodes', 'inprogressepisodes'):
etree.SubElement(root, 'content').text = "episodes"
else:
etree.SubElement(root, 'content').text = mediatype
limit = "50"
2015-12-24 14:07:00 -06:00
# Elements per nodetype
if nodetype == "all":
etree.SubElement(root, 'order', {'direction': "ascending"}).text = "sorttitle"
elif nodetype == "recent":
etree.SubElement(root, 'order', {'direction': "descending"}).text = "dateadded"
etree.SubElement(root, 'limit').text = limit
rule = etree.SubElement(root, 'rule', {'field': "playcount", 'operator': "is"})
etree.SubElement(rule, 'value').text = "0"
elif nodetype == "inprogress":
etree.SubElement(root, 'rule', {'field': "inprogress", 'operator': "true"})
etree.SubElement(root, 'limit').text = limit
elif nodetype == "genres":
etree.SubElement(root, 'order', {'direction': "ascending"}).text = "sorttitle"
etree.SubElement(root, 'group').text = "genres"
elif nodetype == "unwatched":
etree.SubElement(root, 'order', {'direction': "ascending"}).text = "sorttitle"
rule = etree.SubElement(root, "rule", {'field': "playcount", 'operator': "is"})
etree.SubElement(rule, 'value').text = "0"
elif nodetype == "sets":
etree.SubElement(root, 'order', {'direction': "ascending"}).text = "sorttitle"
etree.SubElement(root, 'group').text = "sets"
elif nodetype == "random":
etree.SubElement(root, 'order', {'direction': "ascending"}).text = "random"
etree.SubElement(root, 'limit').text = limit
elif nodetype == "recommended":
etree.SubElement(root, 'order', {'direction': "descending"}).text = "rating"
etree.SubElement(root, 'limit').text = limit
rule = etree.SubElement(root, 'rule', {'field': "playcount", 'operator': "is"})
etree.SubElement(rule, 'value').text = "0"
rule2 = etree.SubElement(root, 'rule',
attrib={'field': "rating", 'operator': "greaterthan"})
etree.SubElement(rule2, 'value').text = "7"
elif nodetype == "recentepisodes":
# Kodi Isengard, Jarvis
etree.SubElement(root, 'order', {'direction': "descending"}).text = "dateadded"
etree.SubElement(root, 'limit').text = limit
rule = etree.SubElement(root, 'rule', {'field': "playcount", 'operator': "is"})
etree.SubElement(rule, 'value').text = "0"
elif nodetype == "inprogressepisodes":
# Kodi Isengard, Jarvis
etree.SubElement(root, 'limit').text = "50"
2015-12-24 14:07:00 -06:00
rule = etree.SubElement(root, 'rule',
attrib={'field': "inprogress", 'operator':"true"})
try:
utils.indent(root)
except: pass
etree.ElementTree(root).write(nodeXML)
def singleNode(self, indexnumber, tagname, mediatype, itemtype):
2016-02-04 06:22:33 -06:00
window = utils.window
2015-12-24 14:07:00 -06:00
tagname = tagname.encode('utf-8')
cleantagname = utils.normalize_nodes(tagname)
nodepath = xbmc.translatePath("special://profile/library/video/").decode('utf-8')
nodeXML = "%splex_%s.xml" % (nodepath, cleantagname)
path = "library://video/plex_%s.xml" % cleantagname
2016-01-06 20:08:16 +11:00
windowpath = "ActivateWindow(Video,%s,return)" % path
2015-12-24 14:07:00 -06:00
# Create the video node directory
if not xbmcvfs.exists(nodepath):
# We need to copy over the default items
shutil.copytree(
src=xbmc.translatePath("special://xbmc/system/library/video").decode('utf-8'),
dst=xbmc.translatePath("special://profile/library/video").decode('utf-8'))
xbmcvfs.exists(path)
labels = {
'Favorite movies': 30180,
'Favorite tvshows': 30181,
'channels': 30173
}
label = utils.language(labels[tagname])
embynode = "Emby.nodes.%s" % indexnumber
2016-02-04 06:22:33 -06:00
window('%s.title' % embynode, value=label)
window('%s.path' % embynode, value=windowpath)
window('%s.content' % embynode, value=path)
window('%s.type' % embynode, value=itemtype)
2015-12-24 14:07:00 -06:00
if xbmcvfs.exists(nodeXML):
# Don't recreate xml if already exists
return
if itemtype == "channels":
root = self.commonRoot(order=1, label=label, tagname=tagname, roottype=2)
etree.SubElement(root, 'path').text = "plugin://plugin.video.plexkodiconnect/?id=0&mode=channels"
2015-12-24 14:07:00 -06:00
else:
root = self.commonRoot(order=1, label=label, tagname=tagname)
etree.SubElement(root, 'order', {'direction': "ascending"}).text = "sorttitle"
etree.SubElement(root, 'content').text = mediatype
try:
utils.indent(root)
except: pass
etree.ElementTree(root).write(nodeXML)
def clearProperties(self):
2016-02-04 06:22:33 -06:00
window = utils.window
2015-12-24 14:07:00 -06:00
self.logMsg("Clearing nodes properties.", 1)
2016-02-04 06:22:33 -06:00
embyprops = window('Emby.nodes.total')
2015-12-24 14:07:00 -06:00
propnames = [
"index","path","title","content",
"inprogress.content","inprogress.title",
"inprogress.content","inprogress.path",
"nextepisodes.title","nextepisodes.content",
"nextepisodes.path","unwatched.title",
"unwatched.content","unwatched.path",
"recent.title","recent.content","recent.path",
"recentepisodes.title","recentepisodes.content",
"recentepisodes.path","inprogressepisodes.title",
"inprogressepisodes.content","inprogressepisodes.path"
]
if embyprops:
totalnodes = int(embyprops)
for i in range(totalnodes):
for prop in propnames:
2016-02-04 06:22:33 -06:00
window('Emby.nodes.%s.%s' % (str(i), prop), clear=True)