diff --git a/helper.py b/helper.py index 7152113..bc7137c 100644 --- a/helper.py +++ b/helper.py @@ -42,6 +42,7 @@ from resources.lib.league_tables import XBMCLeagueTable from resources.lib.live_scores_detail import XBMCLiveScoresDetail from resources.lib.results import XBMCResults +from resources.lib.fixtures import XBMCFixtures from resources.lib.utils import closeAddonSettings from resources.lib.menu import FootballHelperMenu from resources.lib.ticker import TickerOverlay @@ -153,12 +154,22 @@ def ToggleTicker(): # Close addon setting window (if open) closeAddonSettings() - # Create an instance of the XBMC League Table + # Create an instance of the XBMC Results xr = XBMCResults() # and display it! xr.start() +elif params.get("mode") == "fixtures": + # Close addon setting window (if open) + closeAddonSettings() + + # Create an instance of the XBMC Fixtures + xf = XBMCFixtures() + + # and display it! + xf.start() + elif params.get("mode") == "toggleticker": ToggleTicker() diff --git a/resources/language/English/strings.po b/resources/language/English/strings.po index 1f7719e..16f49a6 100644 --- a/resources/language/English/strings.po +++ b/resources/language/English/strings.po @@ -159,3 +159,11 @@ msgstr "" msgctxt "#32113" msgid "Getting results data" msgstr "" + +msgctxt "#32114" +msgid "Fixtures" +msgstr "" + +msgctxt "#32115" +msgid "Getting fixtures data" +msgstr "" diff --git a/resources/lib/fixtures.py b/resources/lib/fixtures.py new file mode 100644 index 0000000..d2a3547 --- /dev/null +++ b/resources/lib/fixtures.py @@ -0,0 +1,319 @@ +''' + 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 . +''' + +''' This script is part of the BBC Football Scores service by elParaguayo + + It allows users to select which leagues they wish to receive updates + for. + + It is called via the script configuration screen or by passing + parameters to trigger specific functions. + + The script accepts the following parameters: + toggle: Turns score notifications on and off + reset: Resets watched league data + + NB only one parameter should be passed at a time. +''' +import sys + +if sys.version_info >= (2, 7): + import json as json + from collections import OrderedDict +else: + import simplejson as json + from resources.lib.ordereddict import OrderedDict + +import xbmc +import xbmcgui +import xbmcaddon + +from resources.lib.footballscores import Fixtures +from resources.lib.utils import closeAddonSettings + +# Import PyXBMCt module. +from pyxbmct.addonwindow import * + +_A_ = xbmcaddon.Addon("service.bbclivefootballscores") +_S_ = _A_.getSetting + +def localise(id): + '''Gets localised string. + + Shamelessly copied from service.xbmc.versioncheck + ''' + string = _A_.getLocalizedString(id).encode( 'utf-8', 'ignore' ) + return string + +class XBMCFixtures(object): + + def __init__(self): + + # It may take a bit of time to display menus/fixtures so let's + # make sure the user knows what's going on + self.prog = xbmcgui.DialogProgressBG() + self.prog.create(localise(32114), localise(32107)) + + # variables for league fixtures display + self.redraw = False + self.offset = 0 + self.menu = True + self.leagueid = 0 + + # variables for root menu + self.showall = True + self.active = True + + # Get our favourite leagues + self.watchedleagues = json.loads(str(_S_("watchedleagues"))) + + # Create a Fixtures instance + self.fixtures = Fixtures() + + self.prog.update(25, localise(32108)) + + allcomps = self.fixtures.getCompetitions() + + allcomps = [x for x in allcomps if x["id"][:11] == "competition"] + + # Get all of the available leagues, store it in an Ordered Dict + # key=League name + # value=League ID + self.allleagues = OrderedDict((x["name"], + x["id"][-9:]) + for x in allcomps) + + self.prog.update(75, localise(32109)) + + # Create a similar Ordered Dict for just those leagues that we're + # currently followin + lgid = x["id"][-9:] + self.watchedleagues = OrderedDict((x["name"], x["id"][-9:]) + for x in allcomps + if (unicode(lgid).isnumeric() and + int(lgid) in self.watchedleagues)) + + self.prog.close() + + + def showMenu(self, all_leagues=False): + + # Setting this to False means that the menu won't display if + # we hit escape + self.active = False + + # Set the title and menu size + window = AddonDialogWindow(localise(32100)) + window.setGeometry(450,300,5,4) + + # Create a List object + self.leaguelist = List() + + # Get the appropriate list of leagues depending on what mode we're in + displaylist = self.allleagues if self.showall else self.watchedleagues + + #self.prog.update(92) + + # Add the List to the menu + window.placeControl(self.leaguelist, 0, 0, rowspan=4, columnspan=4) + self.leaguelist.addItems(displaylist.keys()) + + # Bind the list action + p = self.leaguelist.getSelectedPosition + window.connect(self.leaguelist, + lambda w = window: + self.setID(self.leaguelist.getListItem(p()).getLabel(), + w)) + + # Don't think these are needed, but what the hell... + window.connect(ACTION_PREVIOUS_MENU, lambda w=window: self.finish(w)) + window.connect(ACTION_NAV_BACK, lambda w=window: self.finish(w)) + + #self.prog.update(94) + + # Create the button to toggle mode + leaguetext = localise(32101) if self.showall else localise(32102) + self.leaguebutton = Button(leaguetext) + window.placeControl(self.leaguebutton, 4, 0, columnspan=2) + + # Bind the button + window.connect(self.leaguebutton, lambda w=window: self.toggleMode(w)) + + #self.prog.update(96) + + # Add the close button + self.closebutton = Button(localise(32103)) + window.placeControl(self.closebutton, 4, 2, columnspan=2) + window.setFocus(self.leaguelist) + # Connect the button to a function. + window.connect(self.closebutton, lambda w=window:self.finish(w)) + + #self.prog.update(98) + + # Handle navigation to make user experience better + self.leaguelist.controlLeft(self.leaguebutton) + self.leaguelist.controlRight(self.closebutton) + self.closebutton.controlUp(self.leaguelist) + self.closebutton.controlLeft(self.leaguebutton) + self.leaguebutton.controlRight(self.closebutton) + self.leaguebutton.controlUp(self.leaguelist) + + # Ready to go... + window.doModal() + + def showFixtures(self): + + # Basic variables + self.redraw = False + + # If there are multiple fixture dates for a competition + # Let's just get the required one + fixtures = self.rawdata[self.offset] + + #self.prog.update(92) + + # How many fixtures are there on this date? + # We'll need this to set the size of the display + n = len(fixtures["fixtures"]) + + # Create a window instance and size it + window = AddonDialogWindow(fixtures["date"]) + window.setGeometry(450, (n + 4) * 30, n + 3, 4) + + #self.prog.update(94) + + # Add the teams + for i,f in enumerate(fixtures["fixtures"]): + + fixture = Label(u"{hometeam} v {awayteam}".format(**f), + alignment=2) + + window.placeControl(fixture, i+1, 0, columnspan=4) + + + #self.prog.update(94) + + # Add the close button + closebutton = Button(localise(32103)) + window.placeControl(closebutton, n+2, 1, columnspan=2) + window.setFocus(closebutton) + # Connect the button to a function. + window.connect(closebutton, lambda w=window: self.finish(w)) + + # Not sure we need these... + window.connect(ACTION_PREVIOUS_MENU, lambda w=window: self.finish(w)) + window.connect(ACTION_NAV_BACK, lambda w=window: self.finish(w)) + + #self.prog.update(96) + + # We may need some extra buttons + nextbutton = Button(localise(32104)) + prevbutton = Button(localise(32105)) + + # There are more fixtures after the ones we're showing + if self.offset < (len(self.rawdata) - 1): + window.placeControl(nextbutton, n+2,3) + window.connect(nextbutton, lambda w=window: self.next(w)) + nextbutton.controlLeft(closebutton) + closebutton.controlRight(nextbutton) + + # There are more fixtures before the ones we're showing + if self.offset > 0: + window.placeControl(prevbutton, n+2,0) + window.connect(prevbutton, lambda w=window: self.previous(w)) + prevbutton.controlRight(closebutton) + closebutton.controlLeft(prevbutton) + + #self.prog.close() + + # Ready to go... + window.doModal() + + def getFixturesData(self, ID): + + self.prog.create(localise(32114), localise(32115)) + try: + raw = self.fixtures.getFixtures("competition-%s" + % (self.leagueid)) + + except: + print "ERROR" + raw = None + + self.prog.close() + + return raw + + def setID(self, ID, w): + # Gets the ID of the selected league + ID = self.allleagues[ID] + self.setleague(ID,w) + + def next(self,w): + # Display the next fixture day in the competion + self.offset += 1 + self.redraw = True + w.close() + + def previous(self,w): + # Display the previous fixture day the competition + self.offset -= 1 + self.redraw = True + w.close() + + def finish(self,w): + # We're done. Gracefully close down menu. + self.redraw = False + self.menu = False + self.active = False + w.close() + + def setleague(self,lg, w): + # Set up the variables to display the league fixtures + self.leagueid = lg + self.offset = 0 + self.redraw = True + w.close() + self.rawdata = self.getFixturesData(self.leagueid) + self.prog.update(90) + + def toggleMode(self,w): + # Toggle between showing all competitions and just our favourites + self.showall = not self.showall + self.active = True + w.close() + + def start(self): + + # Let's begin + while self.active: + # Show the main menu + self.showMenu() + + while self.redraw: + + # Show fixtures + self.showFixtures() + +if __name__ == "__main__": + + # Close addon setting window (if open) + closeAddonSettings() + + # Create an instance of the XBMC Fixtures + xf = XBMCFixtures() + + # and display it! + xf.start() diff --git a/resources/lib/menu.py b/resources/lib/menu.py index 0264a51..04da85c 100644 --- a/resources/lib/menu.py +++ b/resources/lib/menu.py @@ -41,9 +41,12 @@ def __init__(self): def show(self): + self.control_list = [] + # Set the title and menu size - self.window = AddonDialogWindow("Menu") + self.window = AddonDialogWindow("BBC Football Scores") self.window.setGeometry(450,300,5,2) + self.control_list.append(self.window) # ITEM 1 - LEAGUE TABLES self.ltbutton = Button("Show League Tables") @@ -57,11 +60,19 @@ def show(self): self.resbutton = Button("Show Results") self.window.placeControl(self.resbutton, 2, 0, columnspan = 2) + # ITEM 4 - MATCH DETAIL + self.fixbutton = Button("Show Fixtures") + self.window.placeControl(self.fixbutton, 3, 0, columnspan = 2) + # CLOSE BUTTON self.clbutton = Button("Close") self.window.placeControl(self.clbutton, 4, 0, columnspan = 2) + # add buttons to control_list + self.control_list += [self.ltbutton, self.mdbutton, + self.resbutton, self.fixbutton] + # Bind actions self.window.connect(ACTION_PREVIOUS_MENU, lambda: self.window.close()) self.window.connect(ACTION_NAV_BACK, lambda: self.window.close()) @@ -69,6 +80,7 @@ def show(self): self.window.connect(self.ltbutton, lambda: self.open("leaguetable")) self.window.connect(self.mdbutton, lambda: self.open("matchdetail")) self.window.connect(self.resbutton, lambda: self.open("results")) + self.window.connect(self.fixbutton, lambda: self.open("fixtures")) self.window.setFocus(self.ltbutton) @@ -76,19 +88,20 @@ def show(self): self.ltbutton.controlDown(self.mdbutton) self.mdbutton.controlUp(self.ltbutton) self.mdbutton.controlDown(self.resbutton) - self.resbutton.controlDown(self.clbutton) + self.resbutton.controlDown(self.fixbutton) self.resbutton.controlUp(self.mdbutton) - self.clbutton.controlUp(self.resbutton) - # self.leaguelist.controlLeft(self.leaguebutton) - # self.leaguelist.controlRight(self.closebutton) - # self.closebutton.controlUp(self.leaguelist) - # self.closebutton.controlLeft(self.leaguebutton) - # self.leaguebutton.controlRight(self.closebutton) - # self.leaguebutton.controlUp(self.leaguelist) + self.fixbutton.controlDown(self.clbutton) + self.fixbutton.controlUp(self.resbutton) + self.clbutton.controlUp(self.fixbutton) + # Ready to go... self.window.doModal() + # clean up + for control in self.control_list: + del control + def open(self, mode): self.window.close()