show "in timer" icon also in multiepg
[enigma2.git] / lib / python / Components / EpgList.py
1 from HTMLComponent import *
2 from GUIComponent import *
3
4 from enigma import *
5 from re import *
6 from time import localtime, time
7 from ServiceReference import ServiceReference
8 from Tools.Directories import resolveFilename, SCOPE_SKIN_IMAGE
9
10 EPG_TYPE_SINGLE = 0
11 EPG_TYPE_MULTI = 1
12
13 RT_HALIGN_LEFT = 0
14 RT_HALIGN_RIGHT = 1
15 RT_HALIGN_CENTER = 2
16 RT_HALIGN_BLOCK = 4
17
18 RT_VALIGN_TOP = 0
19 RT_VALIGN_CENTER = 8
20 RT_VALIGN_BOTTOM = 16
21
22 RT_WRAP = 32
23
24 SINGLE_CPP = 0
25
26 class Rect:
27         def __init__(self, x, y, width, height):
28                 self.__left = x
29                 self.__top = y
30                 self.__width = width
31                 self.__height = height
32
33         def left(self):
34                 return self.__left
35
36         def top(self):
37                 return self.__top
38
39         def height(self):
40                 return self.__height
41
42         def width(self):
43                 return self.__width
44
45 class EPGList(HTMLComponent, GUIComponent):
46         def __init__(self, type=EPG_TYPE_SINGLE, selChangedCB=None, timer = None):
47                 self.timer = timer
48                 self.onSelChanged = [ ]
49                 if selChangedCB is not None:
50                         self.onSelChanged.append(selChangedCB)
51                 GUIComponent.__init__(self)
52                 self.type=type
53                 if type == EPG_TYPE_SINGLE and SINGLE_CPP > 0:
54                         self.l = eListboxEPGContent()
55                 else:
56                         self.l = eListboxPythonMultiContent()
57                         if type == EPG_TYPE_SINGLE:
58                                 self.l.setBuildFunc(self.buildSingleEntry)
59                         else:
60                                 self.l.setBuildFunc(self.buildMultiEntry)
61                 self.epgcache = eEPGCache.getInstance()
62
63         def getEventFromId(self, service, eventid):
64                 event = None
65                 if self.epgcache is not None and eventid is not None:
66                         event = self.epgcache.lookupEventId(service.ref, eventid)
67                 return event
68
69         def getCurrentChangeCount(self):
70                 if self.type == EPG_TYPE_SINGLE:
71                         return 0
72                 return self.l.getCurrentSelection()[0]
73
74         def getCurrent(self):
75                 if self.type == EPG_TYPE_SINGLE:
76                         if SINGLE_CPP > 0:
77                                 evt = self.l.getCurrent()
78                         else:
79                                 cur = self.l.getCurrentSelection()
80                                 if cur is None:
81                                         return cur
82                                 else:
83                                         evt = self.getEventFromId(self.service, cur[0])
84                 else:
85                         tmp = self.l.getCurrentSelection()
86                         if tmp is None:
87                                 return ( None, None )
88                         eventid = tmp[2]
89                         service = ServiceReference(tmp[1])
90                         event = self.getEventFromId(service, eventid)
91                         evt = ( event, service )
92                 return evt
93
94         def moveUp(self):
95                 self.instance.moveSelection(self.instance.moveUp)
96
97         def moveDown(self):
98                 self.instance.moveSelection(self.instance.moveDown)
99
100         def connectSelectionChanged(func):
101                 if not self.onSelChanged.count(func):
102                         self.onSelChanged.append(func)
103
104         def disconnectSelectionChanged(func):
105                 self.onSelChanged.remove(func)
106
107         def selectionChanged(self):
108                 for x in self.onSelChanged:
109                         if x is not None:
110                                 try:
111                                         x()
112                                 except:
113                                         pass
114
115         def GUIcreate(self, parent):
116                 self.instance = eListbox(parent)
117                 self.instance.setWrapAround(True)
118                 self.instance.selectionChanged.get().append(self.selectionChanged)
119                 self.instance.setContent(self.l)
120                 if SINGLE_CPP > 0:
121                         self.instance.setItemHeight(25)
122
123         def GUIdelete(self):
124                 self.instance = None
125
126         def recalcEntrySize(self):
127                 if SINGLE_CPP == 0:
128                         esize = self.l.getItemSize()
129                         self.l.setFont(0, gFont("Regular", 22))
130                         self.l.setFont(1, gFont("Regular", 16))
131                         width = esize.width()
132                         height = esize.height()
133                         if self.type == EPG_TYPE_SINGLE:
134                                 w = width/20*5
135                                 self.datetime_rect = Rect(0,0, w, height)
136                                 self.descr_rect = Rect(w, 0, width/20*15, height)
137                         else:
138                                 xpos = 0;
139                                 w = width/10*3;
140                                 self.service_rect = Rect(xpos, 0, w-10, height)
141                                 xpos += w;
142                                 w = width/10*2;
143                                 self.start_end_rect = Rect(xpos, 0, w-10, height)
144                                 self.progress_rect = Rect(xpos, 4, w-10, height-8)
145                                 xpos += w
146                                 w = width/10*5;
147                                 self.descr_rect = Rect(xpos, 0, width, height)
148
149         def buildSingleEntry(self, eventId, beginTime, duration, EventName):
150                 rec=(self.timer.isInTimer(eventId, beginTime, duration, self.service) > ((duration/10)*8)) 
151                 r1=self.datetime_rect
152                 r2=self.descr_rect
153                 res = [ None ]  # no private data needed
154                 t = localtime(beginTime)
155                 res.append((eListboxPythonMultiContent.TYPE_TEXT, r1.left(), r1.top(), r1.width(), r1.height(), 0, RT_HALIGN_LEFT, "%02d.%02d, %02d:%02d"%(t[2],t[1],t[3],t[4])))
156                 if rec:
157                         res.append((eListboxPythonMultiContent.TYPE_PIXMAP_ALPHATEST, r2.left(), r2.top(), 21, 21, loadPNG(resolveFilename(SCOPE_SKIN_IMAGE, 'epgclock-fs8.png'))))
158                         res.append((eListboxPythonMultiContent.TYPE_TEXT, r2.left() + 25, r2.top(), r2.width(), r2.height(), 0, RT_HALIGN_LEFT, EventName))
159                 else:
160                         res.append((eListboxPythonMultiContent.TYPE_TEXT, r2.left(), r2.top(), r2.width(), r2.height(), 0, RT_HALIGN_LEFT, EventName))
161                 return res
162
163         def buildMultiEntry(self, changecount, service, eventId, begTime, duration, EventName, nowTime, service_name):
164                 rec=begTime and (self.timer.isInTimer(eventId, begTime, duration, service) > ((duration/10)*8))
165                 sname = service_name
166                 r1=self.service_rect
167                 r2=self.progress_rect
168                 r3=self.descr_rect
169                 r4=self.start_end_rect
170                 res = [ None ] # no private data needed
171                 re = compile('\xc2\x86.*?\xc2\x87')
172                 list = re.findall(sname)
173                 if len(list):
174                         sname=''
175                         for substr in list:
176                                 sname+=substr[2:len(substr)-2]
177                         if len(sname) == 0:
178                                 sname = service_name;
179                 if rec:
180                         res.append((eListboxPythonMultiContent.TYPE_TEXT, r1.left(), r1.top(), r1.width()-21, r1.height(), 0, RT_HALIGN_LEFT, sname))
181                         res.append((eListboxPythonMultiContent.TYPE_PIXMAP_ALPHATEST, r1.left()+r1.width()-16, r1.top(), 21, 21, loadPNG(resolveFilename(SCOPE_SKIN_IMAGE, 'epgclock-fs8.png'))))
182                 else:
183                         res.append((eListboxPythonMultiContent.TYPE_TEXT, r1.left(), r1.top(), r1.width(), r1.height(), 0, RT_HALIGN_LEFT, sname))
184                 if begTime is not None:
185                         if nowTime < begTime:
186                                 begin = localtime(begTime)
187                                 end = localtime(begTime+duration)
188 #                               print "begin", begin
189 #                               print "end", end
190                                 res.append((eListboxPythonMultiContent.TYPE_TEXT, r4.left(), r4.top(), r4.width(), r4.height(), 1, RT_HALIGN_CENTER|RT_VALIGN_CENTER, "%02d.%02d - %02d.%02d"%(begin[3],begin[4],end[3],end[4])));
191                                 res.append((eListboxPythonMultiContent.TYPE_TEXT, r3.left(), r3.top(), r3.width(), r3.height(), 0, RT_HALIGN_LEFT, EventName))
192                         else:
193                                 percent = (nowTime - begTime) * 100 / duration
194                                 res.append((eListboxPythonMultiContent.TYPE_PROGRESS, r2.left(), r2.top(), r2.width(), r2.height(), percent));
195                                 res.append((eListboxPythonMultiContent.TYPE_TEXT, r3.left(), r3.top(), r3.width(), r3.height(), 0, RT_HALIGN_LEFT, EventName))
196                 return res
197
198         def queryEPG(self, list, buildFunc=None):
199                 if self.epgcache is not None:
200                         if buildFunc is not None:
201                                 return self.epgcache.lookupEvent(list, buildFunc)
202                         else:
203                                 return self.epgcache.lookupEvent(list)
204                 return [ ]
205
206         def fillMultiEPG(self, services):
207                 t = time()
208                 test = [ '0RIBDTCN' ]
209                 for service in services:
210                         tuple = ( service.ref.toString(), 0 )
211                         test.append( tuple )
212                 self.list = self.queryEPG(test)
213                 self.l.setList(self.list)
214                 print time() - t
215                 self.selectionChanged()
216
217         def updateMultiEPG(self, direction):
218                 t = time()
219                 test = [ 'RIBDTCN' ]
220                 for x in self.list:
221                         service = x[1]
222                         begTime = x[3]
223                         duration = x[4]
224                         if begTime is None:
225                                 begTime = 0
226                         test.append((service, direction, begTime))
227                 tmp = self.queryEPG(test)
228                 cnt=0
229                 for x in tmp:
230                         changecount = self.list[cnt][0] + direction
231                         if changecount >= 0:
232                                 if x[2] is not None:
233                                         self.list[cnt]=(changecount, x[0], x[1], x[2], x[3], x[4], x[5], x[6])
234                         cnt+=1
235                 self.l.setList(self.list)
236                 print time() - t
237                 self.selectionChanged()
238
239         def fillSingleEPG(self, service):
240                 t = time()
241                 if SINGLE_CPP > 0:
242                         self.l.setRoot(service.ref)
243                 else:
244                         self.service = service
245                         test = [ 'IBDT', (service.ref.toString(), 0, -1, -1) ]
246                         self.l.setList(self.queryEPG(test))
247                 print time() - t
248                 self.selectionChanged()