2018-07-12 18:46:02 +02:00
|
|
|
#!/usr/bin/env python
|
2015-12-24 14:07:00 -06:00
|
|
|
# -*- coding: utf-8 -*-
|
2018-07-12 18:46:02 +02:00
|
|
|
from __future__ import absolute_import, division, unicode_literals
|
2017-12-09 14:35:08 +01:00
|
|
|
from logging import getLogger
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2018-10-24 17:55:04 +02:00
|
|
|
from ..utils import etree
|
|
|
|
from .. import utils, path_ops, variables as v, state
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-09-04 16:34:25 +02:00
|
|
|
###############################################################################
|
2016-08-29 18:44:27 +02:00
|
|
|
|
2018-06-21 19:24:37 +02:00
|
|
|
LOG = getLogger('PLEX.videonodes')
|
2016-08-29 18:44:27 +02:00
|
|
|
|
2016-09-04 16:34:25 +02:00
|
|
|
###############################################################################
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
|
|
|
|
class VideoNodes(object):
|
|
|
|
|
|
|
|
def commonRoot(self, order, label, tagname, roottype=1):
|
|
|
|
|
|
|
|
if roottype == 0:
|
|
|
|
# Index
|
2018-10-14 11:48:44 +02:00
|
|
|
root = etree.Element('node', {'order': "%s" % order})
|
2015-12-24 14:07:00 -06:00
|
|
|
elif roottype == 1:
|
|
|
|
# Filter
|
2018-06-21 19:24:37 +02:00
|
|
|
root = etree.Element('node',
|
2018-10-14 11:48:44 +02:00
|
|
|
{'order': "%s" % order, 'type': "filter"})
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(root, 'match').text = "all"
|
|
|
|
# Add tag rule
|
2018-06-21 19:24:37 +02:00
|
|
|
rule = etree.SubElement(root,
|
|
|
|
'rule',
|
2018-10-14 11:48:44 +02:00
|
|
|
{'field': "tag", 'operator': "is"})
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(rule, 'value').text = tagname
|
|
|
|
else:
|
|
|
|
# Folder
|
2018-06-21 19:24:37 +02:00
|
|
|
root = etree.Element('node',
|
2018-10-14 11:48:44 +02:00
|
|
|
{'order': "%s" % order, 'type': "folder"})
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
etree.SubElement(root, 'label').text = label
|
2016-03-02 18:56:42 +01:00
|
|
|
etree.SubElement(root, 'icon').text = "special://home/addons/plugin.video.plexkodiconnect/icon.png"
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
return root
|
|
|
|
|
2018-06-21 19:24:37 +02:00
|
|
|
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
|
2016-03-15 13:09:51 +01:00
|
|
|
# How many items do we get at most?
|
2018-02-13 07:24:39 +01:00
|
|
|
limit = state.FETCH_PMS_ITEM_NUMBER
|
2016-03-03 09:04:15 +01:00
|
|
|
mediatypes = {
|
|
|
|
'movie': 'movies',
|
|
|
|
'show': 'tvshows',
|
|
|
|
'photo': 'photos',
|
|
|
|
'homevideo': 'homevideos',
|
2018-07-24 21:04:31 +02:00
|
|
|
'musicvideos': 'musicvideos',
|
|
|
|
'artist': 'albums'
|
2016-03-03 09:04:15 +01:00
|
|
|
}
|
|
|
|
mediatype = mediatypes[mediatype]
|
|
|
|
|
2015-12-24 14:07:00 -06:00
|
|
|
if viewtype == "mixed":
|
2016-03-02 18:56:42 +01:00
|
|
|
dirname = "%s-%s" % (viewid, mediatype)
|
2015-12-24 14:07:00 -06:00
|
|
|
else:
|
2016-02-23 00:00:24 -06:00
|
|
|
dirname = viewid
|
2017-01-20 15:46:34 +01:00
|
|
|
|
2017-05-03 20:30:33 +02:00
|
|
|
# Returns strings
|
2018-06-23 18:25:18 +02:00
|
|
|
path = path_ops.translate_path('special://profile/library/video/')
|
|
|
|
nodepath = path_ops.translate_path(
|
|
|
|
'special://profile/library/video/Plex-%s/' % dirname)
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2017-01-20 15:46:34 +01:00
|
|
|
if delete:
|
2018-06-23 18:25:18 +02:00
|
|
|
if path_ops.exists(nodepath):
|
|
|
|
path_ops.rmtree(nodepath)
|
2018-06-21 19:24:37 +02:00
|
|
|
LOG.info("Sucessfully removed videonode: %s." % tagname)
|
2017-01-20 15:46:34 +01:00
|
|
|
return
|
|
|
|
|
2015-12-24 14:07:00 -06:00
|
|
|
# Verify the video directory
|
2018-06-23 18:25:18 +02:00
|
|
|
if not path_ops.exists(path):
|
|
|
|
path_ops.copy_tree(
|
|
|
|
src=path_ops.translate_path(
|
|
|
|
'special://xbmc/system/library/video'),
|
|
|
|
dst=path_ops.translate_path('special://profile/library/video'),
|
2018-06-09 12:56:21 +02:00
|
|
|
preserve_mode=0) # do not copy permission bits!
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-03-07 16:16:57 +01:00
|
|
|
# Create the node directory
|
2016-06-04 18:48:22 +02:00
|
|
|
if mediatype != "photos":
|
2018-06-23 18:25:18 +02:00
|
|
|
if not path_ops.exists(nodepath):
|
2016-03-07 16:16:57 +01:00
|
|
|
# folder does not exist yet
|
2018-06-21 19:24:37 +02:00
|
|
|
LOG.debug('Creating folder %s' % nodepath)
|
2018-06-23 18:25:18 +02:00
|
|
|
path_ops.makedirs(nodepath)
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
# Create index entry
|
|
|
|
nodeXML = "%sindex.xml" % nodepath
|
|
|
|
# Set windows property
|
2016-03-02 18:56:42 +01:00
|
|
|
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
|
2018-06-21 19:24:37 +02:00
|
|
|
if utils.window('Plex.nodes.%s.index' % i) == path:
|
2015-12-24 14:07:00 -06:00
|
|
|
return
|
|
|
|
|
2016-06-04 18:48:22 +02:00
|
|
|
if mediatype == "photos":
|
2017-03-08 17:53:43 +01:00
|
|
|
path = "plugin://plugin.video.plexkodiconnect?mode=browseplex&key=/library/sections/%s&id=%s" % (viewid, viewid)
|
2017-05-03 20:30:33 +02:00
|
|
|
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.window('Plex.nodes.%s.index' % indexnumber, value=path)
|
2017-05-03 20:30:33 +02:00
|
|
|
|
2015-12-24 14:07:00 -06:00
|
|
|
# Root
|
2016-06-04 18:48:22 +02:00
|
|
|
if not mediatype == "photos":
|
2016-02-22 17:56:08 -06:00
|
|
|
if viewtype == "mixed":
|
2016-03-02 18:56:42 +01:00
|
|
|
specialtag = "%s-%s" % (tagname, mediatype)
|
2017-05-03 20:30:33 +02:00
|
|
|
root = self.commonRoot(order=0,
|
|
|
|
label=specialtag,
|
|
|
|
tagname=tagname,
|
|
|
|
roottype=0)
|
2016-02-22 17:56:08 -06:00
|
|
|
else:
|
2017-05-03 20:30:33 +02:00
|
|
|
root = self.commonRoot(order=0,
|
|
|
|
label=tagname,
|
|
|
|
tagname=tagname,
|
|
|
|
roottype=0)
|
2016-01-11 22:20:34 +01:00
|
|
|
try:
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.indent(root)
|
2017-05-03 20:30:33 +02:00
|
|
|
except:
|
|
|
|
pass
|
2017-05-29 15:39:11 +02:00
|
|
|
etree.ElementTree(root).write(nodeXML, encoding="UTF-8")
|
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",
|
2017-03-08 16:41:49 +01:00
|
|
|
'12': "ondeck",
|
2018-07-17 13:48:09 +02:00
|
|
|
'13': 'browsefiles',
|
2015-12-24 14:07:00 -06:00
|
|
|
}
|
|
|
|
mediatypes = {
|
|
|
|
# label according to nodetype per mediatype
|
2017-03-08 16:41:49 +01:00
|
|
|
'movies':
|
2016-01-13 01:03:35 +01:00
|
|
|
{
|
2017-03-08 16:41:49 +01:00
|
|
|
'1': tagname,
|
|
|
|
'2': 30174,
|
|
|
|
# '4': 30177,
|
|
|
|
# '6': 30189,
|
|
|
|
'8': 39501,
|
|
|
|
'9': 135,
|
|
|
|
'10': 30227,
|
|
|
|
'11': 30230,
|
|
|
|
'12': 39500,
|
2018-07-17 13:48:09 +02:00
|
|
|
'13': 39702,
|
2016-01-13 01:03:35 +01:00
|
|
|
},
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2017-03-08 16:41:49 +01:00
|
|
|
'tvshows':
|
2016-01-13 01:03:35 +01:00
|
|
|
{
|
2017-03-08 16:41:49 +01:00
|
|
|
'1': tagname,
|
|
|
|
# '2': 30170,
|
|
|
|
'3': 30174,
|
|
|
|
# '4': 30171,
|
|
|
|
# '5': 30178,
|
|
|
|
# '7': 30179,
|
|
|
|
'9': 135,
|
|
|
|
'10': 30227,
|
|
|
|
# '11': 30230,
|
|
|
|
'12': 39500,
|
2018-07-17 13:48:09 +02:00
|
|
|
'13': 39702,
|
2016-01-13 01:03:35 +01:00
|
|
|
},
|
2017-03-08 16:41:49 +01:00
|
|
|
|
|
|
|
'homevideos':
|
2016-01-13 01:03:35 +01:00
|
|
|
{
|
2017-03-08 16:41:49 +01:00
|
|
|
'1': tagname,
|
|
|
|
'2': 30251,
|
|
|
|
'11': 30253,
|
2018-07-17 13:48:09 +02:00
|
|
|
'13': 39702,
|
|
|
|
'14': 136
|
2016-01-13 01:03:35 +01:00
|
|
|
},
|
2017-03-08 16:41:49 +01:00
|
|
|
|
|
|
|
'photos':
|
2016-01-13 01:03:35 +01:00
|
|
|
{
|
2017-03-08 16:41:49 +01:00
|
|
|
'1': tagname,
|
|
|
|
'2': 30252,
|
|
|
|
'8': 30255,
|
|
|
|
'11': 30254,
|
|
|
|
'13': 39702
|
2016-01-13 01:03:35 +01:00
|
|
|
},
|
2016-02-04 06:22:33 -06:00
|
|
|
|
2017-03-08 16:41:49 +01:00
|
|
|
'musicvideos':
|
2016-01-18 20:17:14 +01:00
|
|
|
{
|
2017-03-08 16:41:49 +01:00
|
|
|
'1': tagname,
|
|
|
|
'2': 30256,
|
|
|
|
'4': 30257,
|
|
|
|
'6': 30258,
|
|
|
|
'13': 39702
|
2018-07-24 21:04:31 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
'albums':
|
|
|
|
{
|
|
|
|
'1': tagname,
|
|
|
|
'2': 517, # Recently played albums
|
|
|
|
'2': 359, # Recently added albums
|
|
|
|
'13': 39702, # browse by folder
|
2016-02-04 06:22:33 -06:00
|
|
|
}
|
2015-12-24 14:07:00 -06:00
|
|
|
}
|
|
|
|
|
2016-03-15 13:09:51 +01:00
|
|
|
# Key: nodetypes, value: sort order in Kodi
|
|
|
|
sortorder = {
|
|
|
|
'1': '3', # "all",
|
|
|
|
'2': '2', # "recent",
|
|
|
|
'3': '2', # "recentepisodes",
|
|
|
|
# '4': # "inprogress",
|
|
|
|
# '5': # "inprogressepisodes",
|
|
|
|
# '6': # "unwatched",
|
|
|
|
# '7': # "nextepisodes",
|
|
|
|
'8': '7', # "sets",
|
|
|
|
'9': '6', # "genres",
|
|
|
|
'10': '8', # "random",
|
|
|
|
'11': '5', # "recommended",
|
|
|
|
'12': '1', # "ondeck"
|
2018-07-17 13:48:09 +02:00
|
|
|
'13': '9', # browse by folder
|
2016-03-15 13:09:51 +01:00
|
|
|
}
|
|
|
|
|
2015-12-24 14:07:00 -06:00
|
|
|
nodes = mediatypes[mediatype]
|
|
|
|
for node in nodes:
|
|
|
|
|
|
|
|
nodetype = nodetypes[node]
|
2016-02-23 00:00:24 -06:00
|
|
|
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":
|
2018-06-21 19:24:37 +02:00
|
|
|
label = utils.lang(stringid)
|
2015-12-24 14:07:00 -06:00
|
|
|
else:
|
|
|
|
label = stringid
|
|
|
|
|
|
|
|
# Set window properties
|
2018-06-21 19:24:37 +02:00
|
|
|
if ((mediatype == "homevideos" or mediatype == "photos") and
|
|
|
|
nodetype == "all"):
|
2016-01-11 22:20:34 +01:00
|
|
|
# Custom query
|
2016-06-04 18:48:22 +02:00
|
|
|
path = ("plugin://plugin.video.plexkodiconnect/?id=%s&mode=browseplex&type=%s"
|
|
|
|
% (viewid, mediatype))
|
|
|
|
elif (mediatype == "homevideos" or mediatype == "photos"):
|
2016-01-11 22:20:34 +01:00
|
|
|
# Custom query
|
2016-06-04 18:48:22 +02:00
|
|
|
path = ("plugin://plugin.video.plexkodiconnect/?id=%s&mode=browseplex&type=%s&folderid=%s"
|
|
|
|
% (viewid, mediatype, nodetype))
|
2016-01-11 22:20:34 +01:00
|
|
|
elif nodetype == "nextepisodes":
|
2015-12-24 14:07:00 -06:00
|
|
|
# Custom query
|
2016-03-15 13:09:51 +01:00
|
|
|
path = "plugin://plugin.video.plexkodiconnect/?id=%s&mode=nextup&limit=%s" % (tagname, limit)
|
2017-01-24 16:56:07 +01:00
|
|
|
# elif v.KODIVERSION == 14 and nodetype == "recentepisodes":
|
2016-03-15 14:19:56 +01:00
|
|
|
elif nodetype == "recentepisodes":
|
2015-12-24 14:07:00 -06:00
|
|
|
# Custom query
|
2016-03-15 14:19:56 +01:00
|
|
|
path = ("plugin://plugin.video.plexkodiconnect/?id=%s&mode=recentepisodes&type=%s&tagname=%s&limit=%s"
|
|
|
|
% (viewid, mediatype, tagname, limit))
|
2017-01-24 16:56:07 +01:00
|
|
|
elif v.KODIVERSION == 14 and nodetype == "inprogressepisodes":
|
2015-12-24 14:07:00 -06:00
|
|
|
# Custom query
|
2016-03-15 13:09:51 +01:00
|
|
|
path = "plugin://plugin.video.plexkodiconnect/?id=%s&mode=inprogressepisodes&limit=%s" % (tagname, limit)
|
2016-03-14 17:47:05 +01:00
|
|
|
elif nodetype == 'ondeck':
|
|
|
|
# PLEX custom query
|
2016-03-15 13:09:51 +01:00
|
|
|
if mediatype == "tvshows":
|
|
|
|
path = ("plugin://plugin.video.plexkodiconnect/?id=%s&mode=ondeck&type=%s&tagname=%s&limit=%s"
|
|
|
|
% (viewid, mediatype, tagname, limit))
|
|
|
|
elif mediatype =="movies":
|
|
|
|
# Reset nodetype; we got the label
|
|
|
|
nodetype = 'inprogress'
|
2017-03-08 16:41:49 +01:00
|
|
|
elif nodetype == 'browsefiles':
|
|
|
|
path = 'plugin://plugin.video.plexkodiconnect?mode=browseplex&key=/library/sections/%s/folder' % viewid
|
2015-12-24 14:07:00 -06:00
|
|
|
else:
|
2016-03-02 18:56:42 +01:00
|
|
|
path = "library://video/Plex-%s/%s_%s.xml" % (dirname, viewid, nodetype)
|
2017-05-03 20:30:33 +02:00
|
|
|
|
2016-06-04 18:48:22 +02:00
|
|
|
if mediatype == "photos":
|
2016-01-11 22:20:34 +01:00
|
|
|
windowpath = "ActivateWindow(Pictures,%s,return)" % path
|
|
|
|
else:
|
2017-01-24 16:56:07 +01:00
|
|
|
if v.KODIVERSION >= 17:
|
2016-08-12 18:35:03 +02:00
|
|
|
# Krypton
|
|
|
|
windowpath = "ActivateWindow(Videos,%s,return)" % path
|
|
|
|
else:
|
|
|
|
windowpath = "ActivateWindow(Video,%s,return)" % path
|
2017-05-03 20:30:33 +02:00
|
|
|
|
2015-12-24 14:07:00 -06:00
|
|
|
if nodetype == "all":
|
|
|
|
|
|
|
|
if viewtype == "mixed":
|
2016-03-02 18:56:42 +01:00
|
|
|
templabel = "%s-%s" % (tagname, mediatype)
|
2015-12-24 14:07:00 -06:00
|
|
|
else:
|
|
|
|
templabel = label
|
|
|
|
|
2016-05-31 08:06:42 +02:00
|
|
|
embynode = "Plex.nodes.%s" % indexnumber
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.window('%s.title' % embynode, value=templabel)
|
|
|
|
utils.window('%s.path' % embynode, value=windowpath)
|
|
|
|
utils.window('%s.content' % embynode, value=path)
|
|
|
|
utils.window('%s.type' % embynode, value=mediatype)
|
2015-12-24 14:07:00 -06:00
|
|
|
else:
|
2016-05-31 08:06:42 +02:00
|
|
|
embynode = "Plex.nodes.%s.%s" % (indexnumber, nodetype)
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.window('%s.title' % embynode, value=label)
|
|
|
|
utils.window('%s.path' % embynode, value=windowpath)
|
|
|
|
utils.window('%s.content' % embynode, value=path)
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2016-06-04 18:48:22 +02:00
|
|
|
if mediatype == "photos":
|
2018-06-21 19:24:37 +02: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
|
2016-01-11 22:20:34 +01:00
|
|
|
continue
|
2017-05-03 20:30:33 +02:00
|
|
|
|
2018-06-23 18:25:18 +02:00
|
|
|
if path_ops.exists(nodeXML):
|
2015-12-24 14:07:00 -06:00
|
|
|
# Don't recreate xml if already exists
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Create the root
|
2018-07-17 13:48:09 +02:00
|
|
|
if (nodetype in ("nextepisodes",
|
|
|
|
"ondeck",
|
|
|
|
'recentepisodes',
|
2018-07-30 13:20:40 +02:00
|
|
|
'browsefiles') or mediatype == "homevideos"):
|
2015-12-24 14:07:00 -06:00
|
|
|
# Folder type with plugin path
|
2018-06-21 19:24:37 +02:00
|
|
|
root = self.commonRoot(order=sortorder[node],
|
|
|
|
label=label,
|
|
|
|
tagname=tagname,
|
|
|
|
roottype=2)
|
2015-12-24 14:07:00 -06:00
|
|
|
else:
|
2018-06-21 19:24:37 +02:00
|
|
|
root = self.commonRoot(order=sortorder[node],
|
|
|
|
label=label,
|
|
|
|
tagname=tagname)
|
2018-07-24 21:04:31 +02:00
|
|
|
# Set the content type
|
2018-07-30 13:04:51 +02:00
|
|
|
if mediatype == 'tvshows' and nodetype != 'all':
|
2018-07-24 21:04:31 +02:00
|
|
|
etree.SubElement(root, 'content').text = 'episodes'
|
|
|
|
else:
|
|
|
|
etree.SubElement(root, 'content').text = mediatype
|
|
|
|
# Now fill the view
|
|
|
|
if (nodetype in ("nextepisodes",
|
|
|
|
"ondeck",
|
|
|
|
'recentepisodes',
|
2018-07-30 13:20:40 +02:00
|
|
|
'browsefiles') or mediatype == "homevideos"):
|
2018-07-24 21:04:31 +02:00
|
|
|
etree.SubElement(root, 'path').text = path
|
|
|
|
else:
|
2015-12-24 14:07:00 -06:00
|
|
|
# Elements per nodetype
|
|
|
|
if nodetype == "all":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "ascending"}).text = "sorttitle"
|
2015-12-24 14:07:00 -06:00
|
|
|
elif nodetype == "recent":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "descending"}).text = "dateadded"
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(root, 'limit').text = limit
|
2018-06-21 19:24:37 +02:00
|
|
|
if utils.settings('MovieShowWatched') == 'false':
|
2016-05-15 18:26:07 +02:00
|
|
|
rule = etree.SubElement(root,
|
|
|
|
'rule',
|
|
|
|
{'field': "playcount",
|
|
|
|
'operator': "is"})
|
|
|
|
etree.SubElement(rule, 'value').text = "0"
|
2015-12-24 14:07:00 -06:00
|
|
|
elif nodetype == "inprogress":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'rule',
|
|
|
|
{'field': "inprogress", 'operator': "true"})
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(root, 'limit').text = limit
|
2016-04-08 13:06:20 +02:00
|
|
|
etree.SubElement(
|
|
|
|
root,
|
|
|
|
'order',
|
|
|
|
{'direction': 'descending'}
|
|
|
|
).text = 'lastplayed'
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
elif nodetype == "genres":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "ascending"}).text = "sorttitle"
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(root, 'group').text = "genres"
|
|
|
|
elif nodetype == "unwatched":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "ascending"}).text = "sorttitle"
|
|
|
|
rule = etree.SubElement(root,
|
|
|
|
"rule",
|
|
|
|
{'field': "playcount", 'operator': "is"})
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(rule, 'value').text = "0"
|
|
|
|
elif nodetype == "sets":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "ascending"}).text = "sorttitle"
|
2016-04-11 09:49:02 +02:00
|
|
|
etree.SubElement(root, 'group').text = "tags"
|
2015-12-24 14:07:00 -06:00
|
|
|
elif nodetype == "random":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "ascending"}).text = "random"
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(root, 'limit').text = limit
|
|
|
|
elif nodetype == "recommended":
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "descending"}).text = "rating"
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(root, 'limit').text = limit
|
2018-06-21 19:24:37 +02:00
|
|
|
rule = etree.SubElement(root,
|
|
|
|
'rule',
|
|
|
|
{'field': "playcount", 'operator': "is"})
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(rule, 'value').text = "0"
|
2018-06-21 19:24:37 +02:00
|
|
|
rule2 = etree.SubElement(root,
|
|
|
|
'rule',
|
2018-10-14 11:48:44 +02:00
|
|
|
{'field': "rating", 'operator': "greaterthan"})
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(rule2, 'value').text = "7"
|
|
|
|
elif nodetype == "recentepisodes":
|
|
|
|
# Kodi Isengard, Jarvis
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "descending"}).text = "dateadded"
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(root, 'limit').text = limit
|
2018-06-21 19:24:37 +02:00
|
|
|
rule = etree.SubElement(root,
|
|
|
|
'rule',
|
|
|
|
{'field': "playcount", 'operator': "is"})
|
2015-12-24 14:07:00 -06:00
|
|
|
etree.SubElement(rule, 'value').text = "0"
|
|
|
|
elif nodetype == "inprogressepisodes":
|
|
|
|
# Kodi Isengard, Jarvis
|
2016-03-15 13:09:51 +01:00
|
|
|
etree.SubElement(root, 'limit').text = limit
|
2018-06-21 19:24:37 +02:00
|
|
|
rule = etree.SubElement(root,
|
|
|
|
'rule',
|
2018-10-14 11:48:44 +02:00
|
|
|
{'field': "inprogress", 'operator':"true"})
|
2015-12-24 14:07:00 -06:00
|
|
|
try:
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.indent(root)
|
2017-05-03 20:30:33 +02:00
|
|
|
except:
|
|
|
|
pass
|
2018-06-23 18:25:18 +02:00
|
|
|
etree.ElementTree(root).write(path_ops.encode_path(nodeXML),
|
|
|
|
encoding="UTF-8")
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
def singleNode(self, indexnumber, tagname, mediatype, itemtype):
|
2018-06-23 18:25:18 +02:00
|
|
|
cleantagname = utils.normalize_nodes(tagname)
|
|
|
|
nodepath = path_ops.translate_path('special://profile/library/video/')
|
2016-03-02 18:56:42 +01:00
|
|
|
nodeXML = "%splex_%s.xml" % (nodepath, cleantagname)
|
|
|
|
path = "library://video/plex_%s.xml" % cleantagname
|
2017-01-24 16:56:07 +01:00
|
|
|
if v.KODIVERSION >= 17:
|
2016-08-12 18:35:03 +02:00
|
|
|
# Krypton
|
|
|
|
windowpath = "ActivateWindow(Videos,%s,return)" % path
|
|
|
|
else:
|
|
|
|
windowpath = "ActivateWindow(Video,%s,return)" % path
|
2016-05-07 13:15:02 +02:00
|
|
|
|
2015-12-24 14:07:00 -06:00
|
|
|
# Create the video node directory
|
2018-06-23 18:25:18 +02:00
|
|
|
if not path_ops.exists(nodepath):
|
2015-12-24 14:07:00 -06:00
|
|
|
# We need to copy over the default items
|
2018-06-23 18:25:18 +02:00
|
|
|
path_ops.copy_tree(
|
|
|
|
src=path_ops.translate_path(
|
|
|
|
'special://xbmc/system/library/video'),
|
|
|
|
dst=path_ops.translate_path('special://profile/library/video'),
|
2018-06-09 12:56:21 +02:00
|
|
|
preserve_mode=0) # do not copy permission bits!
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
labels = {
|
|
|
|
'Favorite movies': 30180,
|
|
|
|
'Favorite tvshows': 30181,
|
|
|
|
'channels': 30173
|
|
|
|
}
|
2018-06-21 19:24:37 +02:00
|
|
|
label = utils.lang(labels[tagname])
|
2016-05-31 08:06:42 +02:00
|
|
|
embynode = "Plex.nodes.%s" % indexnumber
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.window('%s.title' % embynode, value=label)
|
|
|
|
utils.window('%s.path' % embynode, value=windowpath)
|
|
|
|
utils.window('%s.content' % embynode, value=path)
|
|
|
|
utils.window('%s.type' % embynode, value=itemtype)
|
2015-12-24 14:07:00 -06:00
|
|
|
|
2018-06-23 18:25:18 +02:00
|
|
|
if path_ops.exists(nodeXML):
|
2015-12-24 14:07:00 -06:00
|
|
|
# Don't recreate xml if already exists
|
|
|
|
return
|
|
|
|
|
|
|
|
if itemtype == "channels":
|
2017-05-03 20:30:33 +02:00
|
|
|
root = self.commonRoot(order=1,
|
|
|
|
label=label,
|
|
|
|
tagname=tagname,
|
|
|
|
roottype=2)
|
2018-06-21 19:24:37 +02:00
|
|
|
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)
|
2018-06-21 19:24:37 +02:00
|
|
|
etree.SubElement(root,
|
|
|
|
'order',
|
|
|
|
{'direction': "ascending"}).text = "sorttitle"
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
etree.SubElement(root, 'content').text = mediatype
|
|
|
|
|
|
|
|
try:
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.indent(root)
|
2017-05-03 20:30:33 +02:00
|
|
|
except:
|
|
|
|
pass
|
2017-05-29 15:39:11 +02:00
|
|
|
etree.ElementTree(root).write(nodeXML, encoding="UTF-8")
|
2015-12-24 14:07:00 -06:00
|
|
|
|
|
|
|
def clearProperties(self):
|
|
|
|
|
2018-06-21 19:24:37 +02:00
|
|
|
LOG.info("Clearing nodes properties.")
|
|
|
|
plexprops = utils.window('Plex.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"
|
|
|
|
]
|
|
|
|
|
2016-05-31 08:06:42 +02:00
|
|
|
if plexprops:
|
|
|
|
totalnodes = int(plexprops)
|
2015-12-24 14:07:00 -06:00
|
|
|
for i in range(totalnodes):
|
|
|
|
for prop in propnames:
|
2018-06-21 19:24:37 +02:00
|
|
|
utils.window('Plex.nodes.%s.%s' % (str(i), prop),
|
|
|
|
clear=True)
|