diff options
author | eracknaphobia <eracknaphobia@hotmail.com> | 2017-03-19 11:28:22 -0400 |
---|---|---|
committer | enen92 <enen92@users.noreply.github.com> | 2017-03-19 15:28:22 +0000 |
commit | 4802b7e3b5cea3b306712e756226ce37b3e743e9 (patch) | |
tree | 48fe4369fd6c7316d465f4a1beacb93234e2ef2c | |
parent | 07733ec5215ea46855eee4c648c582e1e9ae9d51 (diff) |
[plugin.video.mmlive] 2017.3.19 (#1063)
* [plugin.video.mmlive] 2017.3.18
* Fixes for Krypton Guidelines
-rw-r--r-- | plugin.video.mmlive/LICENSE.txt | 339 | ||||
-rw-r--r-- | plugin.video.mmlive/README.md | 5 | ||||
-rw-r--r-- | plugin.video.mmlive/addon.xml | 24 | ||||
-rw-r--r-- | plugin.video.mmlive/changelog.txt | 26 | ||||
-rw-r--r-- | plugin.video.mmlive/fanart.jpg | bin | 0 -> 682497 bytes | |||
-rw-r--r-- | plugin.video.mmlive/icon.png | bin | 0 -> 16145 bytes | |||
-rw-r--r-- | plugin.video.mmlive/main.py | 222 | ||||
-rw-r--r-- | plugin.video.mmlive/resources/__init__.py | 1 | ||||
-rw-r--r-- | plugin.video.mmlive/resources/adobepass.py | 224 | ||||
-rw-r--r-- | plugin.video.mmlive/resources/globals.py | 437 | ||||
-rw-r--r-- | plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po | 41 | ||||
-rw-r--r-- | plugin.video.mmlive/resources/settings.xml | 7 |
12 files changed, 1326 insertions, 0 deletions
diff --git a/plugin.video.mmlive/LICENSE.txt b/plugin.video.mmlive/LICENSE.txt new file mode 100644 index 0000000..d159169 --- /dev/null +++ b/plugin.video.mmlive/LICENSE.txt @@ -0,0 +1,339 @@ + GNU GENERAL PUBLIC LICENSE + Version 2, June 1991 + + Copyright (C) 1989, 1991 Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +License is intended to guarantee your freedom to share and change free +software--to make sure the software is free for all its users. This +General Public License applies to most of the Free Software +Foundation's software and to any other program whose authors commit to +using it. (Some other Free Software Foundation software is covered by +the GNU Lesser General Public License instead.) You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + + To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if you +distribute copies of the software, or if you modify it. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must give the recipients all the rights that +you have. You must make sure that they, too, receive or can get the +source code. And you must show them these terms so they know their +rights. + + We protect your rights with two steps: (1) copyright the software, and +(2) offer you this license which gives you legal permission to copy, +distribute and/or modify the software. + + Also, for each author's protection and ours, we want to make certain +that everyone understands that there is no warranty for this free +software. If the software is modified by someone else and passed on, we +want its recipients to know that what they have is not the original, so +that any problems introduced by others will not reflect on the original +authors' reputations. + + Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that redistributors of a free +program will individually obtain patent licenses, in effect making the +program proprietary. To prevent this, we have made it clear that any +patent must be licensed for everyone's free use or not licensed at all. + + The precise terms and conditions for copying, distribution and +modification follow. + + GNU GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License applies to any program or other work which contains +a notice placed by the copyright holder saying it may be distributed +under the terms of this General Public License. The "Program", below, +refers to any such program or work, and a "work based on the Program" +means either the Program or any derivative work under copyright law: +that is to say, a work containing the Program or a portion of it, +either verbatim or with modifications and/or translated into another +language. (Hereinafter, translation is included without limitation in +the term "modification".) Each licensee is addressed as "you". + +Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running the Program is not restricted, and the output from the Program +is covered only if its contents constitute a work based on the +Program (independent of having been made by running the Program). +Whether that is true depends on what the Program does. + + 1. You may copy and distribute verbatim copies of the Program's +source code as you receive it, in any medium, provided that you +conspicuously and appropriately publish on each copy an appropriate +copyright notice and disclaimer of warranty; keep intact all the +notices that refer to this License and to the absence of any warranty; +and give any other recipients of the Program a copy of this License +along with the Program. + +You may charge a fee for the physical act of transferring a copy, and +you may at your option offer warranty protection in exchange for a fee. + + 2. You may modify your copy or copies of the Program or any portion +of it, thus forming a work based on the Program, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) You must cause the modified files to carry prominent notices + stating that you changed the files and the date of any change. + + b) You must cause any work that you distribute or publish, that in + whole or in part contains or is derived from the Program or any + part thereof, to be licensed as a whole at no charge to all third + parties under the terms of this License. + + c) If the modified program normally reads commands interactively + when run, you must cause it, when started running for such + interactive use in the most ordinary way, to print or display an + announcement including an appropriate copyright notice and a + notice that there is no warranty (or else, saying that you provide + a warranty) and that users may redistribute the program under + these conditions, and telling the user how to view a copy of this + License. (Exception: if the Program itself is interactive but + does not normally print such an announcement, your work based on + the Program is not required to print an announcement.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Program, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Program. + +In addition, mere aggregation of another work not based on the Program +with the Program (or with a work based on the Program) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may copy and distribute the Program (or a work based on it, +under Section 2) in object code or executable form under the terms of +Sections 1 and 2 above provided that you also do one of the following: + + a) Accompany it with the complete corresponding machine-readable + source code, which must be distributed under the terms of Sections + 1 and 2 above on a medium customarily used for software interchange; or, + + b) Accompany it with a written offer, valid for at least three + years, to give any third party, for a charge no more than your + cost of physically performing source distribution, a complete + machine-readable copy of the corresponding source code, to be + distributed under the terms of Sections 1 and 2 above on a medium + customarily used for software interchange; or, + + c) Accompany it with the information you received as to the offer + to distribute corresponding source code. (This alternative is + allowed only for noncommercial distribution and only if you + received the program in object code or executable form with such + an offer, in accord with Subsection b above.) + +The source code for a work means the preferred form of the work for +making modifications to it. For an executable work, complete source +code means all the source code for all modules it contains, plus any +associated interface definition files, plus the scripts used to +control compilation and installation of the executable. However, as a +special exception, the source code distributed need not include +anything that is normally distributed (in either source or binary +form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component +itself accompanies the executable. + +If distribution of executable or object code is made by offering +access to copy from a designated place, then offering equivalent +access to copy the source code from the same place counts as +distribution of the source code, even though third parties are not +compelled to copy the source along with the object code. + + 4. You may not copy, modify, sublicense, or distribute the Program +except as expressly provided under this License. Any attempt +otherwise to copy, modify, sublicense or distribute the Program is +void, and will automatically terminate your rights under this License. +However, parties who have received copies, or rights, from you under +this License will not have their licenses terminated so long as such +parties remain in full compliance. + + 5. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Program or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Program (or any work based on the +Program), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Program or works based on it. + + 6. Each time you redistribute the Program (or any work based on the +Program), the recipient automatically receives a license from the +original licensor to copy, distribute or modify the Program subject to +these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + + 7. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Program. + +If any portion of this section is held invalid or unenforceable under +any particular circumstance, the balance of the section is intended to +apply and the section as a whole is intended to apply in other +circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system, which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 8. If the distribution and/or use of the Program is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Program under this License +may add an explicit geographical distribution limitation excluding +those countries, so that distribution is permitted only in or among +countries not thus excluded. In such case, this License incorporates +the limitation as if written in the body of this License. + + 9. The Free Software Foundation may publish revised and/or new versions +of the General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +Each version is given a distinguishing version number. If the Program +specifies a version number of this License which applies to it and "any +later version", you have the option of following the terms and conditions +either of that version or of any later version published by the Free +Software Foundation. If the Program does not specify a version number of +this License, you may choose any version ever published by the Free Software +Foundation. + + 10. If you wish to incorporate parts of the Program into other free +programs whose distribution conditions are different, write to the author +to ask for permission. For software which is copyrighted by the Free +Software Foundation, write to the Free Software Foundation; we sometimes +make exceptions for this. Our decision will be guided by the two goals +of preserving the free status of all derivatives of our free software and +of promoting the sharing and reuse of software generally. + + NO WARRANTY + + 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, +REPAIR OR CORRECTION. + + 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE +POSSIBILITY OF SUCH DAMAGES. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + <one line to give the program's name and a brief idea of what it does.> + Copyright (C) <year> <name of author> + + 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 2 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, write to the Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + +Also add information on how to contact you by electronic and paper mail. + +If the program is interactive, make it output a short notice like this +when it starts in an interactive mode: + + Gnomovision version 69, Copyright (C) year name of author + Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, the commands you use may +be called something other than `show w' and `show c'; they could even be +mouse-clicks or menu items--whatever suits your program. + +You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the program, if +necessary. Here is a sample; alter the names: + + Yoyodyne, Inc., hereby disclaims all copyright interest in the program + `Gnomovision' (which makes passes at compilers) written by James Hacker. + + <signature of Ty Coon>, 1 April 1989 + Ty Coon, President of Vice + +This General Public License does not permit incorporating your program into +proprietary programs. If your program is a subroutine library, you may +consider it more useful to permit linking proprietary applications with the +library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. diff --git a/plugin.video.mmlive/README.md b/plugin.video.mmlive/README.md new file mode 100644 index 0000000..78ac825 --- /dev/null +++ b/plugin.video.mmlive/README.md @@ -0,0 +1,5 @@ +plugin.video.mmlive +====================== + +KODI plugin March Madness Live + diff --git a/plugin.video.mmlive/addon.xml b/plugin.video.mmlive/addon.xml new file mode 100644 index 0000000..f82a157 --- /dev/null +++ b/plugin.video.mmlive/addon.xml @@ -0,0 +1,24 @@ +<?xml version="1.0" encoding="UTF-8" standalone="yes"?> +<addon id="plugin.video.mmlive" name="March Madness Live" version="2017.3.19" provider-name="eracknaphobia"> + <requires> + <import addon="xbmc.python" version="2.25.0"/> + </requires> + <extension point="xbmc.python.pluginsource" library="main.py"> + <provides>video</provides> + </extension> + + <extension point="xbmc.addon.metadata"> + <platform>all</platform> + <summary lang="en_GB"></summary> + <description lang="en_GB">The National Collegiate Athletic Association (NCAA) Men's Division I Basketball Tournament is a single-elimination tournament played each spring in the United States, currently featuring 68 college basketball teams, to determine the national championship of the major college basketball teams.</description> + <disclaimer lang="en_GB"></disclaimer> + <language>en</language> + <platform>all</platform> + <license>GNU GENERAL PUBLIC LICENSE. Version 2, June 1991</license> + <forum>http://forum.kodi.tv/showthread.php?tid=264888</forum> + <website></website> + <email></email> + <source>https://github.com/eracknaphobia/plugin.video.mmlive/tree/krypton</source> + </extension> +</addon> + diff --git a/plugin.video.mmlive/changelog.txt b/plugin.video.mmlive/changelog.txt new file mode 100644 index 0000000..46fd9a9 --- /dev/null +++ b/plugin.video.mmlive/changelog.txt @@ -0,0 +1,26 @@ +2017.3.19 +- Bumped python version for Krypton + +2017.3.18 +- Fixed completed games not showing final +- Fixed some stream parameter issues +- Code clean up + +2017.3.17 +- Added a deauthorize option +- Added seed number to team name +- Archive games are ~2min Recaps (for now) +- Removed httplib2 requirement + +2017.3.16 +- Added Classic Games Section +- Switched to registration code login + +2016.3.18.3 +- Fix Cox cable login + +2016.3.18.2 +- Fix settings bug + +2016.3.18 +- Beta released
\ No newline at end of file diff --git a/plugin.video.mmlive/fanart.jpg b/plugin.video.mmlive/fanart.jpg Binary files differnew file mode 100644 index 0000000..f5e3512 --- /dev/null +++ b/plugin.video.mmlive/fanart.jpg diff --git a/plugin.video.mmlive/icon.png b/plugin.video.mmlive/icon.png Binary files differnew file mode 100644 index 0000000..330d9a3 --- /dev/null +++ b/plugin.video.mmlive/icon.png diff --git a/plugin.video.mmlive/main.py b/plugin.video.mmlive/main.py new file mode 100644 index 0000000..7cbbdc8 --- /dev/null +++ b/plugin.video.mmlive/main.py @@ -0,0 +1,222 @@ +from resources.globals import * +from resources.adobepass import ADOBE + +#Add-on specific Adobepass variables +SERVICE_VARS = {'requestor_id':'MML', + 'public_key':'XfId78vskMBegCUx9fuiNQL3XvxP3SzN', + 'private_key':'60OKORsYmOkUMgDm', + 'activate_url':'ncaa.com/activate' + } + +def categories(): + addDir('Today\'s Games','/live',1,ICON,FANART) + addDir('Archive Games','/live',2,ICON,FANART) + addDir('Classic Games','/classic',3,ICON,FANART) + addDir('Deauthorize this Device','/deauth',4,ICON,FANART) + + +def todaysGames(archive=None): + now = datetime.now() + url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/bracket.json' + + req = urllib2.Request(url) + req.add_header('Connection', 'keep-alive') + req.add_header('Accept', '*/*') + req.add_header('User-Agent', UA_MMOD) + req.add_header('Accept-Language', 'en-us') + req.add_header('Accept-Encoding', 'deflate') + + response = urllib2.urlopen(req) + json_source = json.load(response) + response.close() + + tourn_day = json_source['bracket']['tournDay'] + teams = getTournamentInfo() + + + if not archive: + setTodaysStream(tourn_day, json_source, teams) + else: + setArchiveStreams(tourn_day, json_source, teams) + + +def setTodaysStream(tourn_day, json_source, teams): + tomorrow = str(int(tourn_day) + 86400) + try: + current_games = getCurrentInfo() + except: + pass + + #Sort By Start Time + json_source = sorted(json_source['bracket']['game'],key=lambda x:x['time']) + + for game in json_source: + if game['time'] >= tourn_day and game['time'] < tomorrow: + if game['tmH'] != '' and game['tmV'] != '': + game_id = game['id'] + hTeam = getTeamInfo(teams, game['tmH']) + vTeam = getTeamInfo(teams,game['tmV']) + game_time = time.strftime('%I:%M %p', time.localtime(int(game['time']))).lstrip('0') + state = game['state'] + archive_video = game['rcpV'] + + title = vTeam['school'] + ' vs ' + hTeam['school'] + + if NO_SPOILERS == '1' or NO_SPOILERS == '2': + name = title + else: + name = '#'+ vTeam['seed']+ ' ' + vTeam['school'] + ' ' + colorString(game['ptsV'], SCORE_COLOR) + ' vs #'+ hTeam['seed']+ ' ' + hTeam['school'] + ' ' + colorString(game['ptsH'], SCORE_COLOR) + + if state == "1": + name = colorString(game_time, UPCOMING) + ' ' + name + elif state == "4": + name = colorString("FINAL", FINAL) + ' ' + name + else: + clock = getGameClock(current_games, game_id) + if clock == '': + clock = 'LIVE' + name = colorString(clock,GAMETIME_COLOR) + ' ' + name + + + link_url = '' + addStream(name,link_url,title,game_id) + + +def classicGames(): + now = datetime.now() + url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/vod/classic_games.json' + req = urllib2.Request(url) + req.add_header('Connection', 'keep-alive') + req.add_header('Accept', '*/*') + req.add_header('User-Agent', UA_MMOD) + req.add_header('Accept-Language', 'en-us') + req.add_header('Accept-Encoding', 'deflate') + + response = urllib2.urlopen(req) + json_source = json.load(response) + response.close() + + for game in json_source['videos']: + title = game['title'] + url = game['connected'] + '|User-Agent='+UA_MMOD + icon = game['thumbnails']['large'] + fanart = game['thumbnails']['raw'] + addLink(title,url,icon,fanart) + + + +def setArchiveStreams(tourn_day, json_source, teams): + json_source = sorted(json_source['bracket']['game'],key=lambda x:x['time'], reverse=True) + for game in json_source: + if game['time'] < tourn_day: + if game['tmH'] != '' and game['tmV'] != '': + game_id = game['id'] + hTeam = getTeamInfo(teams, game['tmH']) + vTeam = getTeamInfo(teams,game['tmV']) + game_time = time.strftime('%I:%M %p', time.localtime(int(game['time']))).lstrip('0') + live_video = game['video'] + archive_video = game['rcpV'] + + title = vTeam['school'] + ' vs ' + hTeam['school'] + + if NO_SPOILERS == '1' or NO_SPOILERS == '3': + name = title + else: + name = '#'+ vTeam['seed']+ ' ' + vTeam['school'] + ' ' + colorString(game['ptsV'], SCORE_COLOR) + ' vs ' + '#'+ hTeam['seed']+ ' ' + hTeam['school'] + ' ' + colorString(game['ptsH'], SCORE_COLOR) + + name = colorString('FINAL ',FINAL) + name + link_url = 'archive' + addStream(name,link_url,title,game_id) + + +def startStream(game_id): + stream_url = fetchStream(game_id,addon_url) + + if addon_url == 'archive': + playable_stream = stream_url + '|User-Agent='+UA_MMOD + else: + adobe = ADOBE(SERVICE_VARS) + resource_id = 'truTV' + mvpd = adobe.authorizeDevice(resource_id) + media_token = adobe.mediaToken(resource_id) + playable_stream = tokenTurner(media_token,stream_url,mvpd) + + listitem = xbmcgui.ListItem(path=playable_stream) + xbmcplugin.setResolvedUrl(handle=addon_handle, succeeded=True, listitem=listitem) + + + +def get_params(): + param=[] + paramstring=sys.argv[2] + if len(paramstring)>=2: + params=sys.argv[2] + cleanedparams=params.replace('?','') + if (params[len(params)-1]=='/'): + params=params[0:len(params)-2] + pairsofparams=cleanedparams.split('&') + param={} + for i in range(len(pairsofparams)): + splitparams={} + splitparams=pairsofparams[i].split('=') + if (len(splitparams))==2: + param[splitparams[0]]=splitparams[1] + + return param + + +params=get_params() +addon_url=None +name=None +mode=None +game_id=None +icon_image = None + +try: + addon_url=urllib.unquote_plus(params["url"]) +except: + pass +try: + name=urllib.unquote_plus(params["name"]) +except: + pass +try: + mode=int(params["mode"]) +except: + pass +try: + game_id=urllib.unquote_plus(params["game_id"]) +except: + pass +try: + icon_image=urllib.unquote_plus(params["icon_image"]) +except: + pass + + +if mode==None: + categories() +elif mode==1: + todaysGames() +elif mode==2: + todaysGames(archive=True) +elif mode==3: + classicGames() +elif mode==4: + msg = 'Are you sure you wish to deauthorize this device?' + dialog = xbmcgui.Dialog() + answer = dialog.yesno('Deauthorize Devices', msg) + if answer: + adobe = ADOBE(SERVICE_VARS) + adobe.deauthorizeDevice() + sys.exit() + +elif mode==104: + startStream(game_id) + + +#Don't cache todays games +if mode==1: + xbmcplugin.endOfDirectory(addon_handle, cacheToDisc=False) +else: + xbmcplugin.endOfDirectory(addon_handle) diff --git a/plugin.video.mmlive/resources/__init__.py b/plugin.video.mmlive/resources/__init__.py new file mode 100644 index 0000000..2b620f6 --- /dev/null +++ b/plugin.video.mmlive/resources/__init__.py @@ -0,0 +1 @@ +# dummy file to init directory
\ No newline at end of file diff --git a/plugin.video.mmlive/resources/adobepass.py b/plugin.video.mmlive/resources/adobepass.py new file mode 100644 index 0000000..e08be36 --- /dev/null +++ b/plugin.video.mmlive/resources/adobepass.py @@ -0,0 +1,224 @@ +import os, sys +import uuid, hmac, hashlib, base64, time +import xbmc, xbmcgui, xbmcaddon +import cookielib, urllib, urllib2, json +from urllib2 import URLError, HTTPError + +class ADOBE(): + api_url = 'http://api.auth.adobe.com' + base_url = 'http://sp.auth.adobe.com' + activate_url = '' + requestor_id = '' + public_key = '' + private_key = '' + device_id = '' + regcode = '' + user_agent = 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/43.0.2357.81 Safari/537.36' + + + def __init__(self, service_vars): + self.requestor_id = service_vars['requestor_id'] + self.public_key = service_vars['public_key'] + self.private_key = service_vars['private_key'] + self.activate_url = service_vars['activate_url'] + self.device_id = self.getDeviceID() + + + def getDeviceID(self): + addon_profile_path = xbmc.translatePath(xbmcaddon.Addon().getAddonInfo('profile')) + fname = os.path.join(addon_profile_path, 'device.id') + #xbmc.log("FILE PATH == "+str(fname)) + if not os.path.isfile(fname): + if not os.path.exists(addon_profile_path): + os.makedirs(addon_profile_path) + new_device_id =str(uuid.uuid1()) + device_file = open(fname,'w') + device_file.write(new_device_id) + device_file.close() + + fname = os.path.join(addon_profile_path, 'device.id') + device_file = open(fname,'r') + device_id = device_file.readline() + device_file.close() + + return device_id + + + def createAuthorization(self, request_method, request_uri): + nonce = str(uuid.uuid4()) + epochtime = str(int(time.time() * 1000)) + authorization = request_method + " requestor_id="+self.requestor_id+", nonce="+nonce+", signature_method=HMAC-SHA1, request_time="+epochtime+", request_uri="+request_uri + signature = hmac.new(self.private_key , authorization, hashlib.sha1) + signature = base64.b64encode(signature.digest()) + authorization += ", public_key="+self.public_key+", signature="+signature + + return authorization + + + def registerDevice(self): + reggie_url = '/reggie/v1/'+self.requestor_id+'/regcode' + authorization = self.createAuthorization('POST',reggie_url) + url = self.api_url+reggie_url + headers = [ ("Accept", "*/*"), + ("Content-type", "application/x-www-form-urlencoded"), + ("Authorization", authorization), + ("Accept-Language", "en-US"), + ("Accept-Encoding", "gzip, deflate"), + ("User-Agent", self.user_agent), + ("Connection", "Keep-Alive"), + ("Pragma", "no-cache") + ] + + + body = 'registrationURL='+self.base_url+'/adobe-services' + body += '&ttl=2700' + body += '&deviceId='+self.device_id + body += '&format=json' + + json_source = self.requestJSON(url, headers, body) + + + msg = '1. Go to [B][COLOR yellow]'+self.activate_url+'[/COLOR][/B][CR]' + msg += '2. Select any platform, it does not matter[CR]' + msg += '3. Enter [B][COLOR yellow]'+json_source['code']+'[/COLOR][/B] as your activation code' + self.regcode = json_source['code'] + dialog = xbmcgui.Dialog() + ok = dialog.ok('Activate Device', msg) + + + + def authorizeDevice(self, resource_id): + auth_url = '/api/v1/authorize' + authorization = self.createAuthorization('GET',auth_url) + url = self.api_url+auth_url + url += '?deviceId='+self.device_id + url += '&requestor='+self.requestor_id + url += '&resource='+urllib.quote(resource_id) + + url += '&format=json' + #req = urllib2.Request(url) + + headers = [ ("Accept", "*/*"), + ("Content-type", "application/x-www-form-urlencoded"), + ("Authorization", authorization), + ("Accept-Language", "en-US"), + ("Accept-Encoding", "deflate"), + ("User-Agent", self.user_agent), + ("Connection", "Keep-Alive"), + ("Pragma", "no-cache") + ] + + json_source = self.requestJSON(url, headers) + mvpd = json_source['mvpd'] + + return mvpd + + + def authenticate(self): + auth_url = '/api/v1/authenticate/PL2MVJL' + authorization = self.createAuthorization('GET',auth_url) + url = self.api_url+auth_url + url += '?deviceId='+self.device_id + url += '&requestor='+self.requestor_id + url += '&deviceType=Win8Universal' + #req = urllib2.Request(url) + + headers = [ ("Accept", "*/*"), + ("Content-type", "application/x-www-form-urlencoded"), + ("Authorization", authorization), + ("Accept-Language", "en-US"), + ("Accept-Encoding", "deflate"), + ("User-Agent", self.user_agent), + ("Connection", "Keep-Alive"), + ("Pragma", "no-cache") + ] + + json_source = self.requestJSON(url, headers) + + def deauthorizeDevice(self): + auth_url = '/api/v1/logout' + authorization = self.createAuthorization('DELETE',auth_url) + url = self.api_url+auth_url + url += '?deviceId='+self.device_id + url += '&requestor='+self.requestor_id + url += '&format=json' + #req = urllib2.Request(url) + + headers = [ ("Accept", "*/*"), + ("Content-type", "application/x-www-form-urlencoded"), + ("Authorization", authorization), + ("Accept-Language", "en-US"), + ("Accept-Encoding", "deflate"), + ("User-Agent", self.user_agent), + ("Connection", "Keep-Alive"), + ("Pragma", "no-cache") + ] + + try: json_source = self.requestJSON(url, headers, None, 'DELETE') + except: pass + + + def mediaToken(self, resource_id): + url = 'http://api.auth.adobe.com/api/v1/tokens/media' + url += '?deviceId='+self.device_id + url += '&requestor='+self.requestor_id + url += '&resource='+urllib.quote(resource_id) + url += '&format=json' + authorization = self.createAuthorization('GET','/api/v1/tokens/media') + headers = [ ("Accept", "*/*"), + ("Content-type", "application/x-www-form-urlencoded"), + ("Authorization", authorization), + ("Accept-Language", "en-US"), + ("Accept-Encoding", "deflate"), + ("User-Agent", self.user_agent), + ("Connection", "Keep-Alive"), + ("Pragma", "no-cache") + ] + + json_source = self.requestJSON(url, headers) + + return json_source['serializedToken'] + + + + def requestJSON(self, url, headers, body=None, method=None): + addon_profile_path = xbmc.translatePath(xbmcaddon.Addon().getAddonInfo('profile')) + cj = cookielib.LWPCookieJar(os.path.join(addon_profile_path, 'cookies.lwp')) + try: cj.load(os.path.join(addon_profile_path, 'cookies.lwp'),ignore_discard=True) + except: pass + opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(cj)) + opener.addheaders = headers + json_source = '' + + try: + request = urllib2.Request(url, body) + if method == 'DELETE': request.get_method = lambda: method + response = opener.open(request) + json_source = json.load(response) + response.close() + self.saveCookie(cj) + except HTTPError as e: + if e.code == 403: + msg = 'Your device is not authorized to view the selected stream.\n Would you like to authorize this device now?' + dialog = xbmcgui.Dialog() + answer = dialog.yesno('Account Not Authorized', msg) + if answer: + self.registerDevice() + else: + sys.exit(0) + else: + sys.exit(0) + + return json_source + + + def saveCookie(self, cj): + # Cookielib patch for Year 2038 problem + # Possibly wrap this in if to check if device is using a 32bit OS + for cookie in cj: + # Jan, 1 2038 + if cookie.expires >= 2145916800: + #Jan, 1 2037 + cookie.expires = 2114380800 + + cj.save(ignore_discard=True)
\ No newline at end of file diff --git a/plugin.video.mmlive/resources/globals.py b/plugin.video.mmlive/resources/globals.py new file mode 100644 index 0000000..cc5fd2a --- /dev/null +++ b/plugin.video.mmlive/resources/globals.py @@ -0,0 +1,437 @@ +import uuid +import hmac +import hashlib +import string, random +from StringIO import StringIO +import gzip +from urllib2 import URLError, HTTPError +import sys +import xbmc,xbmcplugin, xbmcgui, xbmcaddon +import re, os, time +import urllib, urllib2 +import json +import HTMLParser +import calendar +from datetime import datetime, timedelta +import time +import cookielib +import base64 + + +addon_handle = int(sys.argv[1]) +SCORE_COLOR = 'FF00B7EB' +UPCOMING = 'FFD2D2D2' +CRITICAL ='FFD10D0D' +FINAL = 'FF666666' +FREE = 'FF43CD80' +GAMETIME_COLOR = 'FFFFFF66' +now = datetime.now() +BASE_PATH = 'https://data.ncaa.com/mml/'+str(now.year)+'/mobile' + +def colorString(string, color): + return '[COLOR='+color+']'+string+'[/COLOR]' + +def stringToDate(string, date_format): + try: + date = datetime.strptime(str(string), date_format) + except TypeError: + date = datetime(*(time.strptime(str(string), date_format)[0:6])) + + return date + +def FIND(source,start_str,end_str): + start = source.find(start_str) + end = source.find(end_str,start+len(start_str)) + + if start != -1: + return source[start+len(start_str):end] + else: + return '' + + +def SET_STREAM_QUALITY(url): + + stream_url = {} + stream_title = [] + + #Open master file a get cookie(s) + cj = cookielib.LWPCookieJar() + opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(cj)) + opener.addheaders = [ ("Accept", "*/*"), + ("Accept-Encoding", "deflate"), + ("Accept-Language", "en-us"), + ("User-Agent", UA_MMOD)] + + resp = opener.open(url) + master = resp.read() + resp.close() + + cookies = '' + for cookie in cj: + if cookies != '': + cookies = cookies + "; " + cookies = cookies + cookie.name + "=" + cookie.value + + line = re.compile("(.+?)\n").findall(master) + + for temp_url in line: + if '#EXT' not in temp_url: + temp_url = temp_url.rstrip() + start = 0 + if 'http' not in temp_url: + if 'master' in url: + start = url.find('master') + elif 'manifest' in url: + start = url.find('manifest') + + if url.find('?') != -1: + replace_url_chunk = url[start:url.find('?')] + else: + replace_url_chunk = url[start:] + + + temp_url = url.replace(replace_url_chunk,temp_url) + temp_url = temp_url.rstrip() + "|User-Agent=" + UA_MMOD + + #if cookies != '': + #temp_url = temp_url + "&Cookie=" + cookies + + stream_title.append(desc) + stream_url.update({desc:temp_url}) + else: + desc = '' + start = temp_url.find('BANDWIDTH=') + if start > 0: + start = start + len('BANDWIDTH=') + end = temp_url.find(',',start) + desc = temp_url[start:end] + try: + int(desc) + desc = str(int(desc)/1000) + ' kbps' + except: + pass + + + if len(stream_title) > 0: + ret =-1 + stream_title.sort(key=natural_sort_key) + print "PLAY BEST SETTING" + print PLAY_BEST + if str(PLAY_BEST) == 'true': + ret = len(stream_title)-1 + else: + dialog = xbmcgui.Dialog() + ret = dialog.select('Choose Stream Quality', stream_title) + print ret + if ret >=0: + url = stream_url.get(stream_title[ret]) + else: + sys.exit() + else: + msg = "No playable streams found." + dialog = xbmcgui.Dialog() + ok = dialog.ok('Streams Not Found', msg) + + + return url + + +def natural_sort_key(s): + _nsre = re.compile('([0-9]+)') + return [int(text) if text.isdigit() else text.lower() + for text in re.split(_nsre, s)] + + +def SAVE_COOKIE(cj): + # Cookielib patch for Year 2038 problem + # Possibly wrap this in if to check if device is using a 32bit OS + for cookie in cj: + # Jan, 1 2038 + if cookie.expires >= 2145916800: + #Jan, 1 2037 + cookie.expires = 2114380800 + + cj.save(ignore_discard=True); + + + + +def getTournamentInfo(): + now = datetime.now() + url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/tournament.json' + req = urllib2.Request(url) + #req.add_header('User-Agent', 'Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2228.0 Safari/537.36') + req.add_header('Connection', 'keep-alive') + req.add_header('Accept', '*/*') + req.add_header('User-Agent', UA_MMOD) + req.add_header('Accept-Language', 'en-us') + req.add_header('Accept-Encoding', 'deflate') + + response = urllib2.urlopen(req) + json_source = json.load(response) + response.close() + + teams = json.dumps(json_source['tournament']['teams']['team']) + + return teams + +def getTeamInfo(teams, team_id): + all_teams = json.loads(teams) + team_name = '' + link_name = '' + + for team in all_teams: + if str(team['id']) == str(team_id): + #team_name = team['school'] + #link_name = team['link'] + break + + return team + +def getCurrentInfo(): + now = datetime.now() + url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/current.json' + req = urllib2.Request(url) + #req.add_header('User-Agent', 'Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2228.0 Safari/537.36') + req.add_header('Connection', 'keep-alive') + req.add_header('Accept', '*/*') + req.add_header('User-Agent', UA_MMOD) + req.add_header('Accept-Language', 'en-us') + req.add_header('Accept-Encoding', 'deflate') + + response = urllib2.urlopen(req) + json_source = json.load(response) + response.close() + + current_games = '' + + try: + current_games = json.dumps(json_source['current']['game']) + except: + pass + + return current_games + + +def getGameClock(current_games, game_id): + games = json.loads(current_games) + clock = 'Final' + ordinal_indicator = '' + + for game in games: + if str(game['id']) == str(game_id): + print game + clock = str(game['clock']) + per = str(game['per']) + state = str(game['state']) + if state != '4' and per != '': + if per == '1' and clock == '00:00': + clock = 'HALF' + else: + if per == '1': + ordinal_indicator = "st" + elif per == '2': + ordinal_indicator = "nd" + clock = clock + ' ' + per + ordinal_indicator + break + + + return clock + + +def getAppConfig(): + #https://data.ncaa.com/mml/2017/mobile/appConfig_iPhone.json + now = datetime.now() + url = 'https://data.ncaa.com/mml/'+str(now.year)+'/mobile/appConfig_iPhone.json' + req = urllib2.Request(url) + req.add_header('Accept', '*/*') + req.add_header('User-Agent', UA_IPHONE) + req.add_header('Accept-Language', 'en-us') + req.add_header('Accept-Encoding', 'deflate') + + response = urllib2.urlopen(req) + json_source = json.load(response) + response.close() + + api = json_source['api'] + #BASE_PATH = api['base']['sche'] + + + +def tokenTurner(media_token, stream_url, mvpd): + #url = 'http://token.vgtf.net/token/turner' + url = 'https://token.vgtf.net/token/token_spe_mml?profile=mml' + + cj = cookielib.LWPCookieJar() + cj.load(os.path.join(ADDON_PATH_PROFILE, 'cookies.lwp'),ignore_discard=True) + opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(cj)) + opener.addheaders = [ ("Current-Type", "application/x-www-form-urlencoded"), + ("Pragma", "no-cache"), + ("Content-Type", "application/x-www-form-urlencoded"), + ("Accept-Encoding", "deflate"), + ("Connection", "keep-alive"), + ("User-Agent", UA_MMOD)] + + xbmc.log(str(base64.b64decode(media_token))) + payload = urllib.urlencode({'accessToken' : str(base64.b64decode(media_token)), + 'accessTokenType' : 'Adobe', + #'sessionId': '05a564d7c8622b5dd1c67f0ae737c5bb1515d66a~auth~win8~mml~100~1489621236122', + #'throttled' : 'no', + 'appData' : '{"clientTime":0}', + 'mvpd' : mvpd, + 'path' : FIND(stream_url,BASE_PATH,'master.m3u8')+'*' + }) + + resp = opener.open(url, payload) + response = resp.read() + resp.close() + hdnts = FIND(response,'<token>','</token>') + stream_url += '?hdnts='+hdnts + xbmc.log(stream_url) + stream_url = stream_url + '|User-Agent='+UA_MMOD + + + return stream_url + + +def fetchStream(game_id,archive=None): + now = datetime.now() + url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/video/'+game_id+'_bk.json' + if archive == 'archive': url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/game/game_'+game_id+'.json' + now = datetime.now() + req = urllib2.Request(url) + req.add_header('Accept', '*/*') + req.add_header('User-Agent', UA_MMOD) + req.add_header('Accept-Language', 'en-us') + req.add_header('Accept-Encoding', 'deflate') + + response = urllib2.urlopen(req) + json_source = json.load(response) + response.close() + + if archive == 'archive': + stream_url = json_source['game']['videos']['video'][0]['connected'] + else: + stream_url = json_source['connected1'] + + + return stream_url + + +def getAuthCookie(): + mediaAuth = '' + try: + cj = cookielib.LWPCookieJar(os.path.join(ADDON_PATH_PROFILE, 'cookies.lwp')) + cj.load(os.path.join(ADDON_PATH_PROFILE, 'cookies.lwp'),ignore_discard=True) + + #If authorization cookie is missing or stale, perform login + for cookie in cj: + if cookie.name == "mediaAuth" and not cookie.is_expired(): + mediaAuth = 'mediaAuth='+cookie.value + except: + pass + + return mediaAuth + + +def addStream(name,link_url,title,game_id,icon=None,fanart=None): + ok=True + u=sys.argv[0]+"?url="+urllib.quote_plus(link_url)+"&mode="+str(104)+"&name="+urllib.quote_plus(name)+"&game_id="+urllib.quote_plus(str(game_id)) + + liz=xbmcgui.ListItem(name) + liz.setArt({'icon': ICON, 'thumb': ICON}) + + if fanart != None: + liz.setArt({'fanart': fanart}) + else: + liz.setArt({'fanart': FANART}) + + liz.setProperty("IsPlayable", "true") + liz.setInfo( type="Video", infoLabels={ "Title": title } ) + + ok=xbmcplugin.addDirectoryItem(handle=int(sys.argv[1]),url=u,listitem=liz,isFolder=False) + xbmcplugin.setContent(addon_handle, 'episodes') + return ok + +def addDir(name,url,mode,icon,fanart=None): + ok=True + u=sys.argv[0]+"?url="+urllib.quote_plus(url)+"&mode="+str(mode)+"&name="+urllib.quote_plus(name) + + liz=xbmcgui.ListItem(name) + liz.setArt({'icon': icon, 'thumb': icon}) + liz.setInfo( type="Video", infoLabels={ "Title": name } ) + + if fanart != None: + liz.setArt({'fanart': fanart}) + else: + liz.setArt({'fanart': FANART}) + + ok=xbmcplugin.addDirectoryItem(handle=int(sys.argv[1]),url=u,listitem=liz,isFolder=True) + xbmcplugin.setContent(int(sys.argv[1]), 'episodes') + return ok + + +def addLink(name,url,icon,fanart=None): + ok=True + + liz=xbmcgui.ListItem(name) + liz.setArt({'icon': icon, 'thumb': icon}) + liz.setInfo( type="Video", infoLabels={ "Title": name } ) + liz.setProperty("IsPlayable", "true") + + if fanart != None: + liz.setArt({'fanart': fanart}) + else: + liz.setArt({'fanart': FANART}) + + + ok=xbmcplugin.addDirectoryItem(handle=int(sys.argv[1]),url=url,listitem=liz) + xbmcplugin.setContent(int(sys.argv[1]), 'episodes') + return ok + + +# KODI ADDON GLOBALS +ADDON_HANDLE = int(sys.argv[1]) +ADDON = xbmcaddon.Addon() +ROOTDIR = ADDON.getAddonInfo('path') +ADDON_ID = ADDON.getAddonInfo('id') +ADDON_VERSION = ADDON.getAddonInfo('version') +ADDON_PATH = xbmc.translatePath(ADDON.getAddonInfo('path')) +ADDON_PATH_PROFILE = xbmc.translatePath(ADDON.getAddonInfo('profile')) +KODI_VERSION = float(re.findall(r'\d{2}\.\d{1}', xbmc.getInfoLabel("System.BuildVersion"))[0]) +LOCAL_STRING = ADDON.getLocalizedString +FANART = ROOTDIR+"/fanart.jpg" +ICON = ROOTDIR+"/icon.png" + +#Main settings +NO_SPOILERS = str(ADDON.getSetting(id="no_spoilers")) + +#User Agents +UA_IPHONE = 'Mozilla/5.0 (iPhone; CPU iPhone OS 8_4 like Mac OS X) AppleWebKit/600.1.4 (KHTML, like Gecko) Mobile/12H143' +UA_PC = 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/43.0.2357.81 Safari/537.36' +UA_ADOBE_PASS = 'AdobePassNativeClient/1.9.4 (iPhone; U; CPU iPhone OS 9.2.1 like Mac OS X; en-us)' +UA_MMOD = 'MML iOS/73 CFNetwork/808.3 Darwin/16.3.0' + + + +#Create Random Device ID and save it to a file +fname = os.path.join(ADDON_PATH_PROFILE, 'device.id') +if not os.path.isfile(fname): + if not os.path.exists(ADDON_PATH_PROFILE): + os.makedirs(ADDON_PATH_PROFILE) + new_device_id = ''.join([random.choice('0123456789abcdef') for x in range(64)]) + device_file = open(fname,'w') + device_file.write(new_device_id) + device_file.close() + +fname = os.path.join(ADDON_PATH_PROFILE, 'device.id') +device_file = open(fname,'r') +DEVICE_ID = device_file.readline() +device_file.close() + + +#Event Colors +FREE = 'FF43CD80' +LIVE = 'FF00B7EB' +UPCOMING = 'FFFFB266' +FREE_UPCOMING = 'FFCC66FF'
\ No newline at end of file diff --git a/plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po b/plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po new file mode 100644 index 0000000..6b0b0dc --- /dev/null +++ b/plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po @@ -0,0 +1,41 @@ +# Kodi Media Center language file +# Addon Name: March Madness Live +# Addon id: plugin.video.mmlive +# Addon Provider: eracknaphobia +msgid "" +msgstr "" +"Project-Id-Version: Kodi Addons\n" +"Report-Msgid-Bugs-To: alanwww1@xbmc.org\n" +"POT-Creation-Date: YEAR-MO-DA HO:MI+ZONE\n" +"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" +"Last-Translator: Kodi Translation Team\n" +"Language-Team: English (http://www.transifex.com/projects/p/xbmc-addons/language/en/)\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: en\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgctxt "#30000" +msgid "Visual" +msgstr "" + +msgctxt "#30100" +msgid "Hide Scores" +msgstr "" + +msgctxt "#30101" +msgid "Off" +msgstr "" + +msgctxt "#30102" +msgid "On" +msgstr "" + +msgctxt "#30103" +msgid "Only Today's Games" +msgstr "" + +msgctxt "#30104" +msgid "Only Archive Games" +msgstr ""
\ No newline at end of file diff --git a/plugin.video.mmlive/resources/settings.xml b/plugin.video.mmlive/resources/settings.xml new file mode 100644 index 0000000..b173deb --- /dev/null +++ b/plugin.video.mmlive/resources/settings.xml @@ -0,0 +1,7 @@ +<?xml version="1.0" encoding="utf-8" standalone="yes"?> +<settings> + <!-- Visual --> + <category label="30000"> + <setting id="no_spoilers" type="enum" label="30100" lvalues="30101|30102|30103|30104" default="30101" /> + </category> +</settings> |