update plugin is now very user friendly :)
[enigma2.git] / lib / python / Screens / ChannelSelection.py
1 from Screen import Screen
2 from Components.Button import Button
3 from Components.ServiceList import ServiceList
4 from Components.ActionMap import NumberActionMap
5 from EpgSelection import EPGSelection
6 from enigma import eServiceReference, eEPGCache, eEPGCachePtr, eServiceCenter, eServiceCenterPtr, iMutableServiceListPtr, iStaticServiceInformationPtr, eTimer
7 from Components.config import config, configElement, ConfigSubsection, configText
8 from Screens.FixedMenu import FixedMenu
9 from Tools.NumericalTextInput import NumericalTextInput
10 from Components.NimManager import nimmanager
11
12 import xml.dom.minidom
13
14 class BouquetSelector(FixedMenu):
15         def __init__(self, session, bouquets, selectedFunc):
16                 self.selectedFunc=selectedFunc
17                 entrys = [ ]
18                 for x in bouquets:
19                         entrys.append((x[0], self.bouquetSelected, x[1]))
20                 FixedMenu.__init__(self, session, "Bouquetlist", entrys)
21                 self.skinName = "Menu"
22
23         def bouquetSelected(self):
24                 self.selectedFunc(self["menu"].getCurrent()[2])
25
26 class ChannelContextMenu(FixedMenu):
27         def __init__(self, session, csel):
28                 self.csel = csel
29
30                 menu = [ ]
31
32                 inBouquetRootList = csel.servicelist.getRoot().getPath().find('FROM BOUQUET "bouquets.') != -1 #FIXME HACK
33                 inBouquet = csel.getMutableList() is not None
34                 haveBouquets = csel.bouquet_root.getPath().find('FROM BOUQUET "bouquets.') != -1
35
36                 if not csel.bouquet_mark_edit and not csel.movemode and not inBouquetRootList:
37                         if (csel.getCurrentSelection().flags & eServiceReference.flagDirectory) != eServiceReference.flagDirectory:
38                                 if haveBouquets:
39                                         menu.append((_("add service to bouquet"), self.addServiceToBouquetSelected))
40                                 else:
41                                         menu.append((_("add service to favourites"), self.addServiceToBouquetSelected))
42                         if inBouquet:
43                                 menu.append((_("remove service"), self.removeCurrentService))
44
45                 if inBouquet: # current list is editable?
46                         if not csel.bouquet_mark_edit:
47                                 if not csel.movemode:
48                                         menu.append((_("enable move mode"), self.toggleMoveMode))
49                                         if not inBouquetRootList:
50                                                 if haveBouquets:
51                                                         menu.append((_("enable bouquet edit"), self.bouquetMarkStart))
52                                                 else:
53                                                         menu.append((_("enable favourite edit"), self.bouquetMarkStart))
54                                 else:
55                                         menu.append((_("disable move mode"), self.toggleMoveMode))
56                         elif not inBouquetRootList:
57                                 if haveBouquets:
58                                         menu.append((_("end bouquet edit"), self.bouquetMarkEnd))
59                                         menu.append((_("abort bouquet edit"), self.bouquetMarkAbort))
60                                 else:
61                                         menu.append((_("end favourites edit"), self.bouquetMarkEnd))
62                                         menu.append((_("abort favourites edit"), self.bouquetMarkAbort))
63
64                 menu.append((_("back"), self.close))
65
66                 FixedMenu.__init__(self, session, _("Channel Selection"), menu)
67                 self.skinName = "Menu"
68
69         def addServiceToBouquetSelected(self):
70                 bouquets = self.csel.getBouquetList()
71                 if bouquets is None:
72                         cnt = 0
73                 else:
74                         cnt = len(bouquets)
75                 if cnt > 1: # show bouquet list
76                         self.session.open(BouquetSelector, bouquets, self.addCurrentServiceToBouquet)
77                 elif cnt == 1: # add to only one existing bouquet
78                         self.addCurrentServiceToBouquet(bouquets[0][1])
79                 else: #no bouquets in root.. so assume only one favourite list is used
80                         self.addCurrentServiceToBouquet(self.csel.bouquet_root)
81
82         def addCurrentServiceToBouquet(self, dest):
83                 self.csel.addCurrentServiceToBouquet(dest)
84                 self.close()
85
86         def removeCurrentService(self):
87                 self.csel.removeCurrentService()
88                 self.close()
89
90         def toggleMoveMode(self):
91                 self.csel.toggleMoveMode()
92                 self.close()
93
94         def bouquetMarkStart(self):
95                 self.csel.startMarkedEdit()
96                 self.close()
97
98         def bouquetMarkEnd(self):
99                 self.csel.endMarkedEdit(abort=False)
100                 self.close()
101
102         def bouquetMarkAbort(self):
103                 self.csel.endMarkedEdit(abort=True)
104                 self.close()
105
106 class ChannelSelectionEdit:
107         def __init__(self):
108                 self.entry_marked = False
109                 self.movemode = False
110                 self.bouquet_mark_edit = False
111                 self.mutableList = None
112                 self.__marked = [ ]
113                 self.saved_title = None
114                 self.saved_root = None
115
116         def getMutableList(self, root=eServiceReference()):
117                 if not self.mutableList is None:
118                         return self.mutableList
119                 serviceHandler = eServiceCenter.getInstance()
120                 if not root.valid():
121                         root=self.servicelist.getRoot()
122                 list = serviceHandler.list(root)
123                 if list is not None:
124                         return list.startEdit()
125                 return None
126
127 #  multiple marked entry stuff ( edit mode, later multiepg selection )
128         def startMarkedEdit(self):
129                 self.mutableList = self.getMutableList()
130                 # add all services from the current list to internal marked set in listboxservicecontent
131                 self.bouquetRoot = self.servicelist.getRoot()
132                 self.clearMarks() # this clears the internal marked set in the listboxservicecontent
133                 self.saved_title = self.instance.getTitle()
134                 new_title = self.saved_title
135                 if self.bouquet_root.getPath().find('FROM BOUQUET "bouquets.') != -1:
136                         new_title += ' ' + _("[bouquet edit]")
137                 else:
138                         new_title += ' ' + _("[favourite edit]")
139                 self.instance.setTitle(new_title)
140                 self.bouquet_mark_edit = True
141                 self.__marked = self.servicelist.getRootServices()
142                 for x in self.__marked:
143                         self.servicelist.addMarked(eServiceReference(x))
144                 self.saved_root = self.getRoot()
145                 self.showAllServices()
146
147         def endMarkedEdit(self, abort):
148                 if not abort and self.mutableList is not None:
149                         new_marked = set(self.servicelist.getMarked())
150                         old_marked = set(self.__marked)
151                         removed = old_marked - new_marked
152                         added = new_marked - old_marked
153                         changed = False
154                         for x in removed:
155                                 changed = True
156                                 self.mutableList.removeService(eServiceReference(x))
157                         for x in added:
158                                 changed = True
159                                 self.mutableList.addService(eServiceReference(x))
160                         if changed:
161                                 self.mutableList.flushChanges()
162                 self.__marked = []
163                 self.clearMarks()
164                 self.bouquet_mark_edit = False
165                 self.bouquetRoot = None
166                 self.mutableList = None
167                 self.instance.setTitle(self.saved_title)
168                 self.saved_title = None
169                 self.setRoot(self.saved_root)
170
171         def clearMarks(self):
172                 self.servicelist.clearMarks()
173
174         def doMark(self):
175                 ref = self.servicelist.getCurrent()
176                 if self.servicelist.isMarked(ref):
177                         self.servicelist.removeMarked(ref)
178                 else:
179                         self.servicelist.addMarked(ref)
180
181         def removeCurrentService(self):
182                 ref = self.servicelist.getCurrent()
183                 mutableList = self.getMutableList()
184                 if ref.valid() and mutableList is not None:
185                         if not mutableList.removeService(ref):
186                                 currentIndex = self.servicelist.getCurrentIndex()
187                                 self.servicelist.moveDown()
188                                 if self.servicelist.getCurrentIndex() == currentIndex:
189                                         currentIndex -= 1
190                                 mutableList.flushChanges() #FIXME dont flush on each single removed service
191                                 self.setRoot(self.servicelist.getRoot())
192                                 self.servicelist.moveToIndex(currentIndex)
193
194         def addCurrentServiceToBouquet(self, dest):
195                 mutableList = self.getMutableList(dest)
196                 if not mutableList is None:
197                         if not mutableList.addService(self.servicelist.getCurrent()):
198                                 mutableList.flushChanges()
199                 self.close()
200
201         def toggleMoveMode(self):
202                 if self.movemode:
203                         if self.entry_marked:
204                                 self.toggleMoveMarked() # unmark current entry
205                         self.movemode = False
206                         self.mutableList.flushChanges() # FIXME add check if changes was made
207                         self.mutableList = None
208                         self.instance.setTitle(self.saved_title)
209                         self.saved_title = None
210                 else:
211                         self.mutableList = self.getMutableList()
212                         self.movemode = True
213                         self.saved_title = self.instance.getTitle()
214                         new_title = self.saved_title
215                         new_title += ' ' + _("[move mode]");
216                         self.instance.setTitle(new_title);
217
218         def handleEditCancel(self):
219                 if self.movemode: #movemode active?
220                         self.channelSelected() # unmark
221                         self.toggleMoveMode() # disable move mode
222                 elif self.bouquet_mark_edit:
223                         self.endMarkedEdit(True) # abort edit mode
224
225         def toggleMoveMarked(self):
226                 if self.entry_marked:
227                         self.servicelist.setCurrentMarked(False)
228                         self.entry_marked = False
229                 else:
230                         self.servicelist.setCurrentMarked(True)
231                         self.entry_marked = True
232
233         def doContext(self):
234                 self.session.open(ChannelContextMenu, self)
235
236 USE_MULTIBOUQUETS = False
237
238 class ChannelSelectionBase(Screen):
239         def __init__(self, session):
240                 Screen.__init__(self, session)
241
242                 # this makes it much simple to implement a selectable radio or tv mode :)
243                 self.service_types_tv = '1:7:1:0:0:0:0:0:0:0:(type == 1) || (type == 17)'
244 #               self.service_types_radio = '1:7:1:0:0:0:0:0:0:0:(type == 2)'
245
246                 self.service_types = self.service_types_tv
247                 if USE_MULTIBOUQUETS:
248                         self.bouquet_root = eServiceReference('1:7:1:0:0:0:0:0:0:0:(type == 1) FROM BOUQUET "bouquets.tv" ORDER BY bouquet')
249                 else:
250                         self.bouquet_root = eServiceReference('%s FROM BOUQUET "userbouquet.favourites.tv" ORDER BY bouquet'%(self.service_types))
251
252                 self["key_red"] = Button(_("All"))
253                 self["key_green"] = Button(_("Satellites"))
254                 self["key_yellow"] = Button(_("Provider"))
255                 self["key_blue"] = Button(_("Favourites"))
256
257                 self["list"] = ServiceList()
258                 self.servicelist = self["list"]
259
260                 #self["okbutton"] = Button("ok", [self.channelSelected])
261
262                 self.numericalTextInput = NumericalTextInput()
263
264         def getBouquetNumOffset(self, bouquet):
265                 if self.bouquet_root.getPath().find('FROM BOUQUET "bouquets.') == -1: #FIXME HACK
266                         return 0
267                 offsetCount = 0
268                 serviceHandler = eServiceCenter.getInstance()
269                 bouquetlist = serviceHandler.list(self.bouquet_root)
270                 if not bouquetlist is None:
271                         while True:
272                                 bouquetIterator = bouquetlist.getNext()
273                                 if not bouquetIterator.valid() or bouquetIterator == bouquet: #end of list or bouquet found
274                                         break
275                                 if ((bouquetIterator.flags & eServiceReference.flagDirectory) != eServiceReference.flagDirectory):
276                                         continue
277                                 servicelist = serviceHandler.list(bouquetIterator)
278                                 if not servicelist is None:
279                                         while True:
280                                                 serviceIterator = servicelist.getNext()
281                                                 if not serviceIterator.valid(): #check if end of list
282                                                         break
283                                                 if serviceIterator.flags: #playable services have no flags
284                                                         continue
285                                                 offsetCount += 1
286                 return offsetCount
287
288         def setRootBase(self, root, justSet=False):
289                 path = root.getPath()
290                 inBouquetRootList = path.find('FROM BOUQUET "bouquets.') != -1 #FIXME HACK
291                 pos = path.find(' FROM BOUQUET')
292                 isBouquet = pos != -1
293                 if not inBouquetRootList and isBouquet:
294                         self.servicelist.setMode(ServiceList.MODE_FAVOURITES)
295                         self.servicelist.setNumberOffset(self.getBouquetNumOffset(root))
296                         refstr = self.service_types + path[pos:]
297                         root = eServiceReference(refstr)
298                 else:
299                         self.servicelist.setMode(ServiceList.MODE_NORMAL)
300                 self.servicelist.setRoot(root, justSet)
301
302         def moveUp(self):
303                 self.servicelist.moveUp()
304
305         def moveDown(self):
306                 self.servicelist.moveDown()
307
308         def showAllServices(self):
309                 ref = eServiceReference('%s ORDER BY name'%(self.service_types))
310                 self.setRoot(ref)
311
312         def showSatellites(self):
313                 ref = eServiceReference('%s FROM SATELLITES ORDER BY satellitePosition'%(self.service_types))
314                 self.setRoot(ref, True)
315                 serviceHandler = eServiceCenter.getInstance()
316                 servicelist = serviceHandler.list(ref)
317                 if not servicelist is None:
318                         while True:
319                                 service = servicelist.getNext()
320                                 if not service.valid(): #check if end of list
321                                         break
322                                 orbpos = service.getData(4) >> 16
323                                 if service.getPath().find("FROM PROVIDER") != -1:
324                                         service_name = _("Providers")
325                                 else:
326                                         service_name = _("Services")
327                                 try:
328                                         service_name += str(' - %s'%(nimmanager.getSatDescription(orbpos)))
329                                         service.setName(service_name) # why we need this cast?
330                                 except:
331                                         if orbpos > 1800: # west
332                                                 service.setName("%s (%3.1f" + _("W") + ")" %(str, (0 - (orbpos - 3600)) / 10.0))
333                                         else:
334                                                 service.setName("%s (%3.1f" + _("E") + ")" % (str, orbpos / 10.0))
335                                 self.servicelist.addService(service)
336                 self.servicelist.finishFill()
337
338         def showProviders(self):
339                 ref = eServiceReference('%s FROM PROVIDERS ORDER BY name'%(self.service_types))
340                 self.setRoot(ref)
341
342         def showFavourites(self):
343                 self.setRoot(self.bouquet_root)
344
345         def keyNumberGlobal(self, number):
346                 char = self.numericalTextInput.getKey(number)
347                 print "You pressed number " + str(number)
348                 print "You would go to character " + str(char)
349                 self.servicelist.moveToChar(char)
350
351         def enterBouquet(self, action):
352                 if action[:7] == "bouquet":
353                         if action.find("FROM BOUQUET") != -1:
354                                 self.setRoot(eServiceReference("1:7:1:0:0:0:0:0:0:0:" + action[8:]))
355                         else:
356                                 self.setRoot(eServiceReference("1:0:1:0:0:0:0:0:0:0:" + action[8:]))
357                         return True
358                 return False
359
360         def getRoot(self):
361                 return self.servicelist.getRoot()
362
363         def getCurrentSelection(self):
364                 return self.servicelist.getCurrent()
365
366         def setCurrentSelection(self, service):
367                 self.servicelist.setCurrent(service)
368
369         def cancel(self):
370                 self.close(None)
371
372         def getBouquetList(self):
373                 serviceCount=0
374                 bouquets = [ ]
375                 serviceHandler = eServiceCenter.getInstance()
376                 list = serviceHandler.list(self.bouquet_root)
377                 if not list is None:
378                         while True:
379                                 s = list.getNext()
380                                 if not s.valid():
381                                         break
382                                 if ((s.flags & eServiceReference.flagDirectory) == eServiceReference.flagDirectory):
383                                         info = serviceHandler.info(s)
384                                         if not info is None:
385                                                 bouquets.append((info.getName(s), s))
386                                 else:
387                                         serviceCount += 1
388                         if len(bouquets) == 0 and serviceCount > 0:
389                                 info = serviceHandler.info(self.bouquet_root)
390                                 if not info is None:
391                                         bouquets.append((info.getName(self.bouquet_root), self.bouquet_root))
392                         return bouquets
393                 return None
394
395 class ChannelSelection(ChannelSelectionBase, ChannelSelectionEdit):
396         def __init__(self, session):
397                 ChannelSelectionBase.__init__(self,session)
398                 ChannelSelectionEdit.__init__(self)
399
400                 #config for lastservice
401                 config.tv = ConfigSubsection();
402                 config.tv.lastservice = configElement("config.tv.lastservice", configText, "", 0);
403                 config.tv.lastroot = configElement("config.tv.lastroot", configText, "", 0);
404
405                 class ChannelActionMap(NumberActionMap):
406                         def action(self, contexts, action):
407                                 if not self.csel.enterBouquet(action):
408                                         if action == "cancel":
409                                                 self.csel.handleEditCancel()
410                                         NumberActionMap.action(self, contexts, action)
411                 self["actions"] = ChannelActionMap(["ChannelSelectActions", "OkCancelActions", "ContextMenuActions"],
412                         {
413                                 "cancel": self.cancel,
414                                 "ok": self.channelSelected,
415                                 "mark": self.doMark,
416                                 "contextMenu": self.doContext,
417                                 "showFavourites": self.showFavourites,
418                                 "showAllServices": self.showAllServices,
419                                 "showProviders": self.showProviders,
420                                 "showSatellites": self.showSatellites,
421                                 "showEPGList": self.showEPGList,
422                                 "1": self.keyNumberGlobal,
423                                 "2": self.keyNumberGlobal,
424                                 "3": self.keyNumberGlobal,
425                                 "4": self.keyNumberGlobal,
426                                 "5": self.keyNumberGlobal,
427                                 "6": self.keyNumberGlobal,
428                                 "7": self.keyNumberGlobal,
429                                 "8": self.keyNumberGlobal,
430                                 "9": self.keyNumberGlobal,
431                                 "0": self.keyNumberGlobal
432                         })
433                 self["actions"].csel = self
434                 self.onShown.append(self.onShow)
435
436 #               self.onLayoutFinish.append(self.onCreate)
437                 self.lastChannelRootTimer = eTimer()
438                 self.lastChannelRootTimer.timeout.get().append(self.onCreate)
439                 self.lastChannelRootTimer.start(100,True)
440
441         def onCreate(self):
442                 lastroot=eServiceReference(config.tv.lastroot.value)
443                 if lastroot.valid():
444                         self.setRoot(lastroot)
445                 else:
446                         self.showFavourites()
447                         self.saveRoot(self.getRoot())
448                 lastservice=eServiceReference(config.tv.lastservice.value)
449                 if lastservice.valid():
450                         self.session.nav.playService(lastservice)
451                         self.servicelist.setCurrent(lastservice)
452
453         def onShow(self):
454                 ref = self.session.nav.getCurrentlyPlayingServiceReference()
455                 if ref is not None and ref.valid() and ref.getPath() == "":
456                         self.servicelist.setPlayableIgnoreService(ref)
457                 else:
458                         self.servicelist.setPlayableIgnoreService(eServiceReference())
459
460         def showEPGList(self):
461                 ref=self.getCurrentSelection()
462                 ptr=eEPGCache.getInstance()
463                 if ptr.startTimeQuery(ref) != -1:
464                         self.session.open(EPGSelection, ref)
465                 else:
466                         print 'no epg for service', ref.toString()
467
468         def channelSelected(self):
469                 ref = self.getCurrentSelection()
470                 if self.movemode:
471                         self.toggleMoveMarked()
472                 elif (ref.flags & 7) == 7:
473                         self.setRoot(ref)
474                 elif self.bouquet_mark_edit:
475                         self.doMark()
476                 else:
477                         self.zap()
478                         self.close(ref)
479
480         def setRoot(self, root, justSet=False):
481                 if not self.movemode:
482                         self.setRootBase(root, justSet)
483
484         #called from infoBar and channelSelected
485         def zap(self):
486                 self.session.nav.playService(self.getCurrentSelection())
487                 self.saveRoot(self.getRoot())
488                 self.saveChannel()
489
490         def saveRoot(self, root):
491                 if root is not None:
492                         config.tv.lastroot.value = root.toString()
493                         config.tv.lastroot.save()
494
495         def saveChannel(self):
496                 ref = self.session.nav.getCurrentlyPlayingServiceReference()
497                 if ref is not None:
498                         refstr = ref.toString()
499                 else:
500                         refstr = ""
501                 config.tv.lastservice.value = refstr
502                 config.tv.lastservice.save()
503
504         def cancel(self):
505                 self.close(None)
506                 lastroot=eServiceReference(config.tv.lastroot.value)
507                 lastservice=eServiceReference(config.tv.lastservice.value)
508                 if lastroot.valid() and self.getRoot() != lastroot:
509                         self.setRoot(lastroot)
510                 if lastservice.valid() and self.getCurrentSelection() != lastservice:
511                         self.servicelist.setCurrent(lastservice)
512
513 class SimpleChannelSelection(ChannelSelectionBase):
514         def __init__(self, session, title):
515                 ChannelSelectionBase.__init__(self, session)
516                 self.title = title
517                 self.onShown.append(self.onExecCallback)
518
519                 class ChannelActionMap(NumberActionMap):
520                         def action(self, contexts, action):
521                                 if not self.csel.enterBouquet(action):
522                                         NumberActionMap.action(self, contexts, action)
523                 self["actions"] = ChannelActionMap(["ChannelSelectActions", "OkCancelActions", "ContextMenuActions"],
524                         {
525                                 "cancel": self.cancel,
526                                 "ok": self.channelSelected,
527                                 "showFavourites": self.showFavourites,
528                                 "showAllServices": self.showAllServices,
529                                 "showProviders": self.showProviders,
530                                 "showSatellites": self.showSatellites,
531                                 "1": self.keyNumberGlobal,
532                                 "2": self.keyNumberGlobal,
533                                 "3": self.keyNumberGlobal,
534                                 "4": self.keyNumberGlobal,
535                                 "5": self.keyNumberGlobal,
536                                 "6": self.keyNumberGlobal,
537                                 "7": self.keyNumberGlobal,
538                                 "8": self.keyNumberGlobal,
539                                 "9": self.keyNumberGlobal,
540                                 "0": self.keyNumberGlobal
541                         })
542                 self["actions"].csel = self
543
544         def onExecCallback(self):
545                 print "onExecCallback"
546                 self.showFavourites()
547                 self.session.currentDialog.instance.setTitle(self.title)
548
549         def channelSelected(self): # just return selected service
550                 ref = self.getCurrentSelection()
551                 self.close(ref)
552
553         def setRoot(self, root):
554                 self.setRootBase(root)
555