2dc19bfdbb7dbfa6a05af24434bb792aa437462a
[enigma2.git] / lib / python / Components / config.py
1 from time import *
2 from Tools.NumericalTextInput import NumericalTextInput
3 from Tools.Directories import *
4
5 class configFile:
6         def __init__(self):
7                 self.changed = 0
8                 self.configElements = { }
9                 try:
10                         self.file = open(resolveFilename(SCOPE_CONFIG, "config"))
11                 except IOError:
12                         print "cannot open config file"
13                         return 
14                 
15                 while 1:
16                         line = self.file.readline()
17                         if line == "":
18                                 break
19                         
20                         if line.startswith("#"):                #skip comments
21                                 continue        
22                                 
23                         self.addElement(line)
24                 self.file.close()
25
26         def addElement(self, line):
27                 x = line.find("=")
28                 if x > -1:
29                         self.configElements[line[:x]] = line[x + 1:-1]
30
31         def getKey(self, key):
32                 return self.configElements[key]
33
34         def setKey(self, key, value, isDefaultKey=False):
35                 self.changed = 1
36                 if isDefaultKey and self.configElements.has_key(key):
37                         del self.configElements[key]
38                 else:
39                         self.configElements[key] = value
40
41         def getResolvedKey(self, key):
42                 str = self.configElements[key]
43                 if len(str):
44                         pos = str.find('*')
45                         if pos != -1:
46                                 str = str[pos+1:]
47                                 pos = str.find('*')
48                                 if pos != -1:
49                                         return str[:pos]
50                         return str
51                 return None
52
53         def save(self):
54                 if self.changed == 0:           #no changes, so no write to disk needed
55                         return
56                         
57                 fileHandle = open(resolveFilename(SCOPE_CONFIG, "config"), "w")
58                 
59                 keys = self.configElements.keys()
60                 keys.sort()
61                 for x in keys:
62                         wstr = x + "=" + self.configElements[x] + "\n"
63
64                         fileHandle.write(wstr)
65
66                 fileHandle.close()
67                 
68 def currentConfigSelectionElement(element):
69         return element.vals[element.value][0]
70
71 def getConfigSelectionElement(element, value):
72         count = 0
73         for x in element.vals:
74                 if x[0] == value:
75                         return count
76                 count += 1
77         return -1
78
79 class configSelection:
80         def __init__(self, parent):
81                 self.parent = parent
82                 
83         def checkValues(self):
84                 if self.parent.value < 0:
85                         self.parent.value = len(self.parent.vals) - 1   
86                 elif(self.parent.value > (len(self.parent.vals) - 1)):
87                         self.parent.value = 0
88
89         def cancel(self):
90                 self.parent.reload()
91
92         def save(self):
93                 self.parent.save()
94
95         def handleKey(self, key):
96                 if key == config.key["prevElement"]:
97                         self.parent.value = self.parent.value - 1
98                 if key == config.key["nextElement"]:
99                         self.parent.value = self.parent.value + 1
100                 
101                 self.checkValues()                      
102
103                 self.parent.change()
104
105         def __call__(self, selected):                   #needed by configlist
106                 self.checkValues()
107
108                 returnValue = _(self.parent.vals[self.parent.value])
109                 if not isinstance(returnValue, str):
110                         returnValue = returnValue[1]
111
112                 # FIXME: it's not really nice to translate this here.
113                 # however, configSelections are persistent.
114                 
115                 # WORKAROUND: don't translate ""
116                 if returnValue:
117                         returnValue = _(returnValue)
118                 
119                 return ("text", returnValue)
120                 
121 class configDateTime:
122         def __init__(self, parent):
123                 self.parent = parent
124                 
125         def checkValues(self):
126                 pass
127 #               if self.parent.value < 0:
128                         #self.parent.value = 0  
129
130                 #if(self.parent.value >= (len(self.parent.vals) - 1)):
131                         #self.parent.value = len(self.parent.vals) - 1
132
133         def cancel(self):
134                 self.parent.reload()
135
136         def save(self):
137                 self.parent.save()
138
139         def handleKey(self, key):
140                 if key == config.key["prevElement"]:
141                         self.parent.value = self.parent.value - self.parent.vals[1]
142                 if key == config.key["nextElement"]:
143                         self.parent.value = self.parent.value + self.parent.vals[1]
144                 
145                 self.checkValues()
146
147                 self.parent.change()    
148
149         def __call__(self, selected):                   #needed by configlist
150                 self.checkValues()
151                 return ("text", strftime(self.parent.vals[0], localtime(self.parent.value)))
152         
153 class configSatlist:
154         def __init__(self, parent):
155                 self.parent = parent
156
157         def checkValues(self):
158                 if self.parent.value < 0:
159                         self.parent.value = 0   
160
161                 if(self.parent.value >= (len(self.parent.vals) - 1)):
162                         self.parent.value = len(self.parent.vals) - 1
163                         
164         def cancel(self):
165                 self.parent.reload()
166
167         def save(self):
168                 self.parent.save()
169
170         def handleKey(self, key):
171                 if key == config.key["prevElement"]:
172                         self.parent.value = self.parent.value - 1
173                 if key == config.key["nextElement"]:
174                         self.parent.value = self.parent.value + 1
175                 
176                 self.checkValues()                      
177
178                 self.parent.change()    
179
180         def __call__(self, selected):                   #needed by configlist
181                 self.checkValues()
182                 #fixme
183                 return ("text", str(self.parent.vals[self.parent.value][0]))
184
185 class configSequenceArg:
186         def get(self, type, args = ()):
187                 # configsequencearg.get ("IP")
188                 if (type == "IP"):
189                         return (("."), [(0,255),(0,255),(0,255),(0,255)], "")
190                 # configsequencearg.get ("MAC")
191                 if (type == "POSITION"):
192                         return ((","), [(0,args[0]),(0,args[1]),(0,args[2]),(0,args[3])], "")
193                 if (type == "MAC"):
194                         return ((":"), [(1,255),(1,255),(1,255),(1,255),(1,255),(1,255)], "")
195                 # configsequencearg.get ("CLOCK")
196                 if (type == "CLOCK"):
197                         return ((":"), [(0,23),(0,59)], "")
198                 # configsequencearg.get("INTEGER", (min, max)) => x with min <= x <= max
199                 if (type == "INTEGER"):
200                         return ((":"), [args], "")
201                 # configsequencearg.get("PINCODE", (number, "*")) => pin with number = length of pincode and "*" as numbers shown as stars
202                 # configsequencearg.get("PINCODE", (number, "")) => pin with number = length of pincode and numbers shown
203                 if (type == "PINCODE"):
204                         return ((":"), [(0, (10**args[0])-1)], args[1])
205                 # configsequencearg.get("FLOAT", [(min,max),(min1,max1)]) => x.y with min <= x <= max and min1 <= y <= max1
206                 if (type == "FLOAT"):
207                         return (("."), args, "")
208                 
209         def getFloat(self, element):
210                 return float(("%d.%0" + str(len(str(element.vals[1][1][1]))) + "d") % (element.value[0], element.value[1]))
211
212 configsequencearg = configSequenceArg()
213                 
214 class configSequence:
215         def __init__(self, parent):
216                 self.parent = parent
217                 self.markedPos = 0
218                 self.seperator = self.parent.vals[0]
219                 self.valueBounds = self.parent.vals[1]
220                 self.censorChar = self.parent.vals[2]
221
222         def checkValues(self):
223                 maxPos = 0
224                 num = 0
225                 for i in self.parent.value:
226                         maxPos += len(str(self.valueBounds[num][1]))
227                         while (self.valueBounds[num][0] > self.parent.value[num]):
228                                 self.parent.value[num] += 1
229
230                         while (self.valueBounds[num][1] < self.parent.value[num]):
231                                 self.parent.value[num] -= 1
232                                 
233 #                       if (self.valueBounds[num][0] <= i <= self.valueBounds[num][1]):
234                                 #pass
235                         #else:
236                                 #self.parent.value[num] = self.valueBounds[num][0]
237                         num += 1
238                 
239                 if self.markedPos >= maxPos:
240                         self.markedPos = maxPos - 1
241                 if self.markedPos < 0:
242                         self.markedPos = 0
243                         
244         def cancel(self):
245                 self.parent.reload()
246
247         def save(self):
248                 self.parent.save()
249
250         def handleKey(self, key):
251                 #this will no change anything on the value itself
252                 #so we can handle it here in gui element
253                 if key == config.key["prevElement"]:
254                         self.markedPos -= 1
255                 if key == config.key["nextElement"]:
256                         self.markedPos += 1
257                 
258                 if key >= config.key["0"] and key <= config.key["9"]:
259                         self.blockLen = []
260                         for x in self.valueBounds:
261                                 self.blockLen.append(len(str(x[1])))
262                                 
263                         pos = 0
264                         blocknumber = 0
265                         self.blockLenTotal = [0,]
266                         for x in self.blockLen:
267                                 pos += self.blockLen[blocknumber]
268                                 self.blockLenTotal.append(pos)
269                                 if (pos - 1 >= self.markedPos):
270                                         pass
271                                 else:
272                                         blocknumber += 1
273                                         
274                         number = 9 - config.key["9"] + key
275                         # length of numberblock
276                         numberLen = len(str(self.valueBounds[blocknumber][1]))
277                         # position in the block
278                         posinblock = self.markedPos - self.blockLenTotal[blocknumber]
279                         
280                         oldvalue = self.parent.value[blocknumber]
281                         olddec = oldvalue % 10 ** (numberLen - posinblock) - (oldvalue % 10 ** (numberLen - posinblock - 1))
282                         newvalue = oldvalue - olddec + (10 ** (numberLen - posinblock - 1) * number)
283                         
284                         self.parent.value[blocknumber] = newvalue
285                         self.markedPos += 1
286                 
287                 self.checkValues()
288
289                 #FIXME: dont call when press left/right
290                 self.parent.change()    
291
292         def __call__(self, selected):                   #needed by configlist
293                 value = ""
294                 mPos = self.markedPos
295                 num = 0;
296                 for i in self.parent.value:
297                         if len(value):  #fixme no heading separator possible
298                                 value += self.seperator
299                                 if mPos >= len(value) - 1:
300                                         mPos += 1
301                                 
302                         #diff =         self.valueBounds - len(str(i))
303                         #if diff > 0:
304                                 ## if this helps?!
305                                 #value += " " * diff
306                         if (self.censorChar == ""):
307                                 value += ("%0" + str(len(str(self.valueBounds[num][1]))) + "d") % i
308                         else:
309                                 value += (self.censorChar * len(str(self.valueBounds[num][1])))
310                         num += 1
311                         # only mark cursor when we are selected
312                         # (this code is heavily ink optimized!)
313                 if (self.parent.enabled == True):
314                         return ("mtext"[1-selected:], value, [mPos])
315                 else:
316                         return ("text", value)
317
318 class configNothing:
319         def __init__(self, parent):
320                 self.parent = parent
321                 self.markedPos = 0
322
323         def cancel(self):
324                 self.parent.reload()
325
326         def save(self):
327                 self.parent.save()
328                 
329         def nextEntry(self):
330                 self.parent.vals[1](self.parent.getConfigPath())
331
332         def handleKey(self, key):
333                 pass
334
335         def __call__(self, selected):                   #needed by configlist
336                 return ("text", "")
337
338 class configText(NumericalTextInput):
339         # used as first parameter
340         # is the text of a fixed size or is the user able to extend the length of the text
341         extendableSize = 1
342         fixedSize = 2
343
344         def __init__(self, parent):
345                 NumericalTextInput.__init__(self, self.nextEntry)
346                 self.parent = parent
347                 self.markedPos = 0
348                 self.mode = self.parent.vals[0]
349                 try:
350                         self.Text = self.parent.value.decode("utf-8")
351                 except UnicodeDecodeError:
352                         print "utf8 kaputt!"
353
354         def checkValues(self):
355                 if (self.markedPos < 0):
356                         self.markedPos = 0
357                 if (self.markedPos >= len(self.Text)):
358                         self.markedPos = len(self.Text) - 1
359
360         def cancel(self):
361                 self.parent.reload()
362
363         def save(self):
364                 self.parent.save()
365
366         def nextEntry(self):
367                 self.parent.vals[1](self.parent.getConfigPath())
368
369         def handleKey(self, key):
370                 #this will no change anything on the value itself
371                 #so we can handle it here in gui element
372                 if key == config.key["delete"]:
373                         self.Text = self.Text[0:self.markedPos] + self.Text[self.markedPos + 1:]
374                         self.parent.value = self.Text.encode("utf-8")
375                 elif key == config.key["prevElement"]:
376                         self.nextKey()
377                         self.markedPos -= 1
378                 elif key == config.key["nextElement"]:
379                         self.nextKey()
380                         self.markedPos += 1
381                         if (self.mode == self.extendableSize):
382                                 if (self.markedPos >= len(self.Text)):
383                                         self.Text = self.Text.ljust(len(self.Text) + 1)
384                                         self.parent.value = self.Text.encode("utf-8")
385                 elif key >= config.key["0"] and key <= config.key["9"]:
386                         number = 9 - config.key["9"] + key
387                         self.Text = self.Text[0:self.markedPos] + self.getKey(number) + self.Text[self.markedPos + 1:]
388                         self.parent.value = self.Text.encode("utf-8")
389                 self.checkValues()
390                 self.parent.change()
391
392         def __call__(self, selected):                   #needed by configlist
393                 return ("mtext"[1-selected:], self.parent.value, [self.markedPos])
394
395 class configValue:
396         def __init__(self, obj):
397                 self.obj = obj
398                 
399         def __str__(self):
400                 return self.obj
401
402 class Config:
403         def __init__(self):
404                 self.key = { "choseElement": 0,
405                                          "prevElement": 1,
406                                          "nextElement": 2,
407                                          "delete": 3,
408                                          "0": 10,
409                                          "1": 11,
410                                          "2": 12,
411                                          "3": 13,
412                                          "4": 14,
413                                          "5": 15,
414                                          "6": 16,
415                                          "7": 17,
416                                          "8": 18,
417                                          "9": 19 }
418                 
419 config = Config();
420
421 configfile = configFile()
422
423 class configSlider:
424         def __init__(self, parent):
425                 self.parent = parent
426
427         def cancel(self):
428                 self.parent.reload()
429
430         def save(self):
431                 self.parent.save()
432
433         def checkValues(self):
434                 if self.parent.value < 0:
435                         self.parent.value = 0   
436
437                 if self.parent.value > self.parent.vals[1]:
438                         self.parent.value = self.parent.vals[1]
439
440         def handleKey(self, key):
441                 if key == config.key["prevElement"]:
442                         self.parent.value = self.parent.value - self.parent.vals[0]
443                 if key == config.key["nextElement"]:
444                         self.parent.value = self.parent.value + self.parent.vals[0]
445                                         
446                 self.checkValues()      
447                 self.parent.change()    
448
449         def __call__(self, selected):                   #needed by configlist
450                 self.checkValues()
451                 return ("slider", self.parent.value, self.parent.vals[1])
452
453 class ConfigSubsection:
454         def __init__(self):
455                 pass
456
457 class configElement:
458
459         def getIndexbyEntry(self, data):
460                 cnt = 0;
461                 tcnt = -1; #for defaultval
462                 for x in self.vals:
463                         if int(x[1]) == int(data):
464                                         return cnt
465                         if int(x[1]) == int(self.defaultValue):
466                                         tcnt = cnt
467                         cnt += 1
468                 if tcnt != -1:
469                         return tcnt
470                 return 0        #prevent bigger then array
471
472         def datafromFile(self, control, data):
473                 if control == configSlider:
474                         return int(data)
475                 elif control == configSelection:
476                         try:
477                                 return int(data)
478                         except:
479                                 for x in data.split(":"):
480                                         if x[0] == "*":
481                                                 count = 0
482                                                 for y in self.vals:
483                                                         if y[0] == x[1:-1]:
484                                                                 return count
485                                                         count += 1
486                                 return self.defaultValue
487                 elif control == configDateTime:
488                         return int(data)
489                 elif control == configText:
490                         return str(data)
491                 elif control == configSequence:
492                         list = [ ]
493                         part = data.split(self.vals[0])
494                         for x in part:
495                                 list.append(int(x))
496                         return list
497                 elif control == configSatlist:
498                         return self.getIndexbyEntry(data)
499                 else: 
500                         return ""       
501
502         def datatoFile(self, control, data):
503                 if control == configSlider:
504                         return str(data)
505                 elif control == configSelection:
506                         if len(self.vals) < data + 1:
507                                 return "0"
508                         if isinstance(self.vals[data], str):
509                                 return str(data)
510                         else:
511                                 confList = []
512                                 count = 0
513                                 for x in self.vals:
514                                         if count == data:
515                                                 confList.append("*" + str(x[0] + "*"))
516                                         else:
517                                                 confList.append(x[0])
518                                         count += 1
519                                 return ":".join(confList)
520                         return str(data)
521                 elif control == configDateTime:
522                         return str(data)
523                 elif control == configText:
524                         return str(data.strip())
525                 elif control == configSequence:
526 #                       print self.vals
527 #                       print self.value
528                         try:
529                                 value = ""
530                                 count = 0
531                                 for i in data:
532                                         if value !="":
533                                                 value += self.vals[0]
534                                         value += (("%0" + str(len(str(self.vals[1][count][1]))) + "d") % i)
535                                         count += 1
536                                         #value = ((len(data) * ("%d" + self.vals[0]))[0:-1]) % tuple(data)
537                         except: 
538                                 value = str(data)       
539                         return value
540                 elif control == configSatlist:
541                         return str(self.vals[self.value][1]);
542                 else: 
543                         return ""       
544
545         def loadData(self):
546                 #print "load:" + self.configPath
547                 try:
548                         value = self.datafromFile(self.controlType, configfile.getKey(self.configPath))
549                 except:         
550                         value = ""
551
552                 if value == "":
553                         #print "value not found - using default"
554                         if self.controlType == configSatlist:
555                                 self.value = self.getIndexbyEntry(self.defaultValue)
556                         elif self.controlType == configSequence:
557                                 self.value = self.defaultValue[:]
558                         else:
559                                 self.value = self.defaultValue
560
561                         self.save()             #add missing value to dict
562                 else:
563                         #print "set val:" + str(value)
564                         self.value = value
565
566                 #is this right? activate settings after load/cancel and use default     
567                 self.change()
568
569         def __init__(self, configPath, control, defaultValue, vals, saveDefaults = True):
570                 self.configPath = configPath
571                 self.defaultValue = defaultValue
572                 self.controlType = control
573                 self.vals = vals
574                 self.notifierList = [ ]
575                 self.enabled = True
576                 self.saveDefaults = saveDefaults
577                 self.loadData()         
578                 
579         def getConfigPath(self):
580                 return self.configPath
581         
582         def addNotifier(self, notifier):
583                 self.notifierList.append(notifier);
584                 notifier(self);
585
586         def change(self):
587                 for notifier in self.notifierList:
588                         notifier(self)
589
590         def reload(self):
591                 self.loadData()
592
593         def save(self):
594                 if self.controlType == configSatlist:
595                         defaultValue = self.getIndexbyEntry(self.defaultValue)
596                 else:
597                         defaultValue = self.defaultValue
598                 if self.value != defaultValue or self.saveDefaults:
599                         configfile.setKey(self.configPath, self.datatoFile(self.controlType, self.value))
600                 else:
601                         try:
602                                 oldValue = configfile.getKey(self.configPath)
603                         except:
604                                 oldValue = None
605                         if oldValue is not None and oldValue != defaultValue:
606                                 configfile.setKey(self.configPath, self.datatoFile(self.controlType, self.value), True)
607
608 class configElement_nonSave(configElement):
609         def __init__(self, configPath, control, defaultValue, vals):
610                 configElement.__init__(self, configPath, control, defaultValue, vals)
611
612         def save(self):
613                 pass
614
615 def getConfigListEntry(description, element):
616         b = element
617         item = b.controlType(b)
618         return ((description, item))
619
620 def configElementBoolean(name, default, texts=(_("Disable"), _("Enable"))):
621         return configElement(name, configSelection, default, texts)
622
623 config.misc = ConfigSubsection()