1 from Screens.Screen import Screen
2 from Screens.ChannelSelection import *
3 from Components.ActionMap import HelpableActionMap, ActionMap, NumberActionMap
4 from Components.Sources.List import List
5 from Components.Sources.StaticText import StaticText
6 from Components.config import ConfigNothing
7 from Components.ConfigList import ConfigList
8 from Components.Label import Label
9 from Components.SelectionList import SelectionList
10 from Components.MenuList import MenuList
11 from ServiceReference import ServiceReference
12 from Plugins.Plugin import PluginDescriptor
13 from xml.etree.cElementTree import parse as ci_parse
14 from Tools.XMLTools import elementsWithTag, mergeText, stringToXML
15 from enigma import eDVBCI_UI, eDVBCIInterfaces
17 from os import system, path as os_path
19 class CIselectMainMenu(Screen):
21 <screen position="205,150" size="310,270" title="CI Assignment" >
22 <widget name="CiList" position="10,10" size="290,200" scrollbarMode="showOnDemand" />
23 <ePixmap position="10,210" size="290,2" pixmap="skin_default/div-h.png" transparent="1" alphatest="on" />
24 <ePixmap pixmap="skin_default/buttons/red.png" position="10,220" size="140,40" alphatest="on" />
25 <ePixmap pixmap="skin_default/buttons/green.png" position="160,220" size="140,40" alphatest="on" />
26 <widget source="key_red" render="Label" position="10,220" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#9f1313" transparent="1" />
27 <widget source="key_green" render="Label" position="160,220" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
30 def __init__(self, session, args = 0):
31 self.skin = CIselectMainMenu.skin
32 Screen.__init__(self, session)
34 self["key_red"] = StaticText(_("Cancel"))
35 self["key_green"] = StaticText(_("Config"))
37 self["actions"] = ActionMap(["ColorActions","SetupActions"],
39 "green": self.greenPressed,
40 "red": self.redPressed,
41 "yellow": self.yellowPressed,
42 "ok": self.greenPressed,
46 NUM_CI=eDVBCIInterfaces.getInstance().getNumOfSlots()
48 print "[CI_Wizzard] FOUND %d CI Slots " % NUM_CI
54 for slot in range(NUM_CI):
55 state = eDVBCI_UI.getInstance().getState(slot)
57 appname = _("Slot %d") %(slot+1) + " - " + _("no module")
59 appname = _("Slot %d") %(slot+1) + " - " + _("init modules")
61 appname = _("Slot %d") %(slot+1) + " - " + eDVBCI_UI.getInstance().getAppName(slot)
62 self.list.append( (appname, ConfigNothing(), 0, slot) )
64 self.list.append( (_("no CI slots found") , ConfigNothing(), 1, -1) )
66 menuList = ConfigList(self.list)
67 menuList.list = self.list
68 menuList.l.setList(self.list)
69 self["CiList"] = menuList
71 def greenPressed(self):
72 cur = self["CiList"].getCurrent()
73 if cur and len(cur) > 2:
77 print "[CI_Wizzard] there is no CI Slot in your receiver"
79 print "[CI_Wizzard] selected CI Slot : %d" % slot
80 if config.usage.setup_level.index > 1: # advanced
81 self.session.open(CIconfigMenu, slot)
83 self.session.open(easyCIconfigMenu, slot)
85 def yellowPressed(self):
86 NUM_CI=eDVBCIInterfaces.getInstance().getNumOfSlots()
87 print "[CI_Check] FOUND %d CI Slots " % NUM_CI
89 for ci in range(NUM_CI):
90 print eDVBCIInterfaces.getInstance().getDescrambleRules(ci)
95 print "[CI_Config] RED BUTTON not implemented yet - only use self.cancel()"
101 class CIconfigMenu(Screen):
103 <screen position="60,80" size="595,436" title="CI Assignment" >
104 <widget name="CAidList.desc" position="10,10" size="575,22" font="Regular;20" />
105 <widget name="CAidList" position="10,40" size="575,45" font="Regular;20" />
106 <widget name="ServiceList.desc" position="10,90" size="575,22" font="Regular;20" />
107 <widget name="ServiceList" position="10,120" size="575,250" scrollbarMode="showOnDemand" />
108 <ePixmap position="10,380" size="575,2" pixmap="skin_default/div-h.png" transparent="1" alphatest="on" />
109 <ePixmap pixmap="skin_default/buttons/red.png" position="10,390" size="140,40" alphatest="on" />
110 <ePixmap pixmap="skin_default/buttons/green.png" position="155,390" size="140,40" alphatest="on" />
111 <ePixmap pixmap="skin_default/buttons/yellow.png" position="300,390" size="140,40" alphatest="on" />
112 <ePixmap pixmap="skin_default/buttons/blue.png" position="445,390" size="140,40" alphatest="on" />
113 <widget source="key_red" render="Label" position="10,390" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#9f1313" transparent="1" />
114 <widget source="key_green" render="Label" position="155,390" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
115 <widget source="key_yellow" render="Label" position="300,390" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
116 <widget source="key_blue" render="Label" position="445,390" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
119 def __init__(self, session, ci_slot="9"):
120 self.skin = CIconfigMenu.skin
121 Screen.__init__(self, session)
123 self.filename="/etc/enigma2/ci"+str(self.ci_slot)+".xml"
125 self["key_red"] = StaticText(_("delete"))
126 self["key_green"] = StaticText(_("add Service"))
127 self["key_yellow"] = StaticText(_("add Provider"))
128 self["key_blue"] = StaticText(_("select CAId"))
129 self["CAidList.desc"] = Label(_("assigned CAIds"))
130 self["ServiceList.desc"] = Label(_("assigned Services/Provider"))
132 self["actions"] = ActionMap(["ColorActions","SetupActions"],
134 "green": self.greenPressed,
135 "red": self.redPressed,
136 "yellow": self.yellowPressed,
137 "blue": self.bluePressed,
138 "ok": self.okPressed,
139 "cancel": self.cancel
142 print "[CI_Wizzard_Config] Configuring CI Slots : %d " % self.ci_slot
146 print eDVBCIInterfaces.getInstance().readCICaIds(self.ci_slot)
147 for caid in eDVBCIInterfaces.getInstance().readCICaIds(self.ci_slot):
149 self.caidlist.append((str(hex(int(caid))),str(caid),i))
151 print "[CI_Wizzard_Config_CI%d] read following CAIds from CI: %s" %(self.ci_slot, self.caidlist)
152 self.selectedcaid =[]
153 self.servicelist = []
154 self.caids=_("no CAId selected")
155 self["CAidList"] = Label(self.caids)
157 serviceList = ConfigList(self.servicelist)
158 serviceList.list = self.servicelist
159 serviceList.l.setList(self.servicelist)
160 self["ServiceList"] = serviceList
163 # if config mode !=advanced autoselect any caid
164 if config.usage.setup_level.index <= 1: # advanced
165 self.selectedcaid=self.caidlist
167 def redPressed(self):
170 def greenPressed(self):
171 self.session.openWithCallback( self.finishedChannelSelection, myChannelSelection, None)
173 def yellowPressed(self):
174 self.session.openWithCallback( self.finishedProviderSelection, myProviderSelection, None)
176 def bluePressed(self):
177 self.session.openWithCallback(self.finishedCAidSelection, CAidSelect, self.caidlist, self.selectedcaid)
180 print "[CI_Config_CI%d] OK BUTTON not implemented yet" %self.ci_slot
188 cur = self["ServiceList"].getCurrent()
189 if cur and len(cur) > 2:
190 self.servicelist.remove(cur)
191 self["ServiceList"].l.setList(self.servicelist)
193 def finishedChannelSelection(self, *args):
196 service_ref = ServiceReference(ref)
197 service_name = service_ref.getServiceName()
198 if find_in_list(self.servicelist, service_name, 0)==False:
199 split_ref=service_ref.ref.toString().split(":")
200 if split_ref[0] == "1": #== dvb service und nicht muell von None
201 self.servicelist.append( (service_name , ConfigNothing(), 0, service_ref.ref.toString()) )
202 self["ServiceList"].l.setList(self.servicelist)
204 def finishedProviderSelection(self, *args):
205 if len(args)>1: # bei nix selected kommt nur 1 arg zurueck (==None)
208 if find_in_list(self.servicelist, name, 0)==False:
209 self.servicelist.append( (name , ConfigNothing(), 1, dvbnamespace) )
210 self["ServiceList"].l.setList(self.servicelist)
212 def finishedCAidSelection(self, *args):
214 self.selectedcaid=args[0]
216 for item in self.selectedcaid:
218 self.caids+= ", " + item[0]
223 self.caids=_("no CAId selected")
224 self["CAidList"].setText(self.caids)
228 fp = file(self.filename, 'w')
229 fp.write("<?xml version=\"1.0\" encoding=\"utf-8\" ?>\n")
231 fp.write("\t<slot>\n")
232 fp.write("\t\t<id>%s</id>\n" % self.ci_slot)
233 for item in self.selectedcaid:
234 if len(self.selectedcaid):
235 fp.write("\t\t<caid id=\"%s\" />\n" % item[0])
236 for item in self.servicelist:
237 if len(self.servicelist):
239 fp.write("\t\t<provider name=\"%s\" dvbnamespace=\"%s\" />\n" % (item[0], item[3]))
241 fp.write("\t\t<service name=\"%s\" ref=\"%s\" />\n" % (item[0], item[3]))
242 fp.write("\t</slot>\n")
246 print "[CI_Config_CI%d] xml not written" %self.ci_slot
247 os.unlink(self.filename)
250 if not os_path.exists(self.filename):
253 def getValue(definitions, default):
255 Len = len(definitions)
256 return Len > 0 and definitions[Len-1].text or default
259 tree = ci_parse(self.filename).getroot()
260 self.read_services=[]
261 self.read_providers=[]
264 # for ci in tree.findall("ci"):
265 for slot in tree.findall("slot"):
266 read_slot = getValue(slot.findall("id"), False).encode("UTF-8")
267 print "ci " + read_slot
270 for caid in slot.findall("caid"):
271 read_caid = caid.get("id").encode("UTF-8")
272 self.selectedcaid.append((str(read_caid),str(read_caid),i))
273 self.usingcaid.append(long(read_caid,16))
276 for service in slot.findall("service"):
277 read_service_name = service.get("name").encode("UTF-8")
278 read_service_ref = service.get("ref").encode("UTF-8")
279 self.read_services.append (read_service_ref)
281 for provider in slot.findall("provider"):
282 read_provider_name = provider.get("name").encode("UTF-8")
283 read_provider_dvbname = provider.get("dvbnamespace").encode("UTF-8")
284 self.read_providers.append((read_provider_name,read_provider_dvbname))
286 self.ci_config.append((int(read_slot), (self.read_services, self.read_providers, self.usingcaid)))
288 print "[CI_Config_CI%d] error parsing xml..." %self.ci_slot
290 for item in self.read_services:
292 self.finishedChannelSelection(item)
294 for item in self.read_providers:
296 self.finishedProviderSelection(item[0],item[1])
299 self.finishedCAidSelection(self.selectedcaid)
300 self["ServiceList"].l.setList(self.servicelist)
302 class easyCIconfigMenu(CIconfigMenu):
304 <screen position="80,80" size="470,420" title="CI Assignment" >
305 <widget name="ServiceList.desc" position="10,10" size="420,22" font="Regular;20" />
306 <widget name="ServiceList" position="10,40" size="450,340" scrollbarMode="showOnDemand" />
307 <ePixmap position="10,360" size="450,2" pixmap="skin_default/div-h.png" transparent="1" alphatest="on" />
308 <ePixmap pixmap="skin_default/buttons/red.png" position="10,370" size="140,40" alphatest="on" />
309 <ePixmap pixmap="skin_default/buttons/green.png" position="160,370" size="140,40" alphatest="on" />
310 <ePixmap pixmap="skin_default/buttons/yellow.png" position="310,370" size="140,40" alphatest="on" />
311 <widget source="key_red" render="Label" position="10,370" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#9f1313" transparent="1" />
312 <widget source="key_green" render="Label" position="160,370" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
313 <widget source="key_yellow" render="Label" position="310,370" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
316 def __init__(self, session, ci_slot="9"):
318 CIconfigMenu.__init__(self, session, ci_slot)
319 self.skin = easyCIconfigMenu.skin
321 self["actions"] = ActionMap(["ColorActions","SetupActions"],
323 "green": self.greenPressed,
324 "red": self.redPressed,
325 "yellow": self.yellowPressed,
326 "blue": self.bluePressed,
327 "ok": self.okPressed,
328 "cancel": self.cancel
331 def bluePressed(self):
334 class CAidSelect(Screen):
336 <screen position="210,140" size="310,290" title="select CAId's" >
337 <widget name="list" position="10,10" size="290,210" scrollbarMode="showOnDemand" />
338 <ePixmap position="10,230" size="290,2" pixmap="skin_default/div-h.png" transparent="1" alphatest="on" />
339 <ePixmap pixmap="skin_default/buttons/red.png" position="10,240" size="140,40" alphatest="on" />
340 <ePixmap pixmap="skin_default/buttons/green.png" position="160,240" size="140,40" alphatest="on" />
341 <widget source="key_red" render="Label" position="10,240" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#9f1313" transparent="1" />
342 <widget source="key_green" render="Label" position="160,240" zPosition="1" size="140,40" font="Regular;19" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
345 def __init__(self, session, list, selected_caids):
346 self.skin = CAidSelect.skin
347 Screen.__init__(self, session)
349 self.list = SelectionList()
350 self["list"] = self.list
352 for listindex in range(len(list)):
353 if find_in_list(selected_caids,list[listindex][0],0):
354 self.list.addSelection(list[listindex][0], list[listindex][1], listindex, True)
356 self.list.addSelection(list[listindex][0], list[listindex][1], listindex, False)
358 self["key_red"] = StaticText(_("Cancel"))
359 self["key_green"] = StaticText(_("Save"))
361 self["actions"] = ActionMap(["ColorActions","SetupActions"],
363 "ok": self.list.toggleSelection,
364 "cancel": self.cancel,
365 "green": self.greenPressed,
369 def greenPressed(self):
370 list = self.list.getSelectionsList()
377 class myProviderSelection(ChannelSelectionBase):
379 <screen position="80,80" size="560,430" title="Select provider to add...">
380 <ePixmap pixmap="skin_default/buttons/red.png" position="0,0" size="140,40" alphatest="on" />
381 <ePixmap pixmap="skin_default/buttons/green.png" position="140,0" size="140,40" alphatest="on" />
382 <ePixmap pixmap="skin_default/buttons/yellow.png" position="280,0" size="140,40" alphatest="on" />
383 <ePixmap pixmap="skin_default/buttons/blue.png" position="420,0" size="140,40" alphatest="on" />
384 <widget name="key_red" position="0,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#9f1313" transparent="1" />
385 <widget name="key_green" position="140,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
386 <widget name="key_yellow" position="280,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#a08500" transparent="1" />
387 <widget name="key_blue" position="420,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#18188b" transparent="1" />
388 <widget name="list" position="00,45" size="560,364" scrollbarMode="showOnDemand" />
391 def __init__(self, session, title):
392 ChannelSelectionBase.__init__(self, session)
393 self.onShown.append(self.__onExecCallback)
395 self["actions"] = ActionMap(["OkCancelActions", "ChannelSelectBaseActions"],
397 "showFavourites": self.doNothing,
398 "showAllServices": self.doNothing,
399 "showProviders": self.doNothing,
400 "showSatellites": self.doNothing,
401 "cancel": self.cancel,
402 "ok": self.channelSelected,
404 self["key_red"] = StaticText(_(""))
405 self["key_green"] = StaticText(_(""))
406 self["key_yellow"] = StaticText(_(""))
407 self["key_blue"] = StaticText(_(""))
410 print "nothing to do..."
412 def __onExecCallback(self):
413 self.showSatellites()
415 def channelSelected(self): # just return selected service
416 ref = self.getCurrentSelection()
417 splited_ref=ref.toString().split(":")
418 if ref.flags == 7 and splited_ref[6] != "0":
419 self.dvbnamespace=splited_ref[6]
422 self.close(ref.getName(), self.dvbnamespace)
424 def showSatellites(self):
425 if not self.pathChangeDisabled:
426 refstr = '%s FROM SATELLITES ORDER BY satellitePosition'%(self.service_types)
427 if not self.preEnterPath(refstr):
428 ref = eServiceReference(refstr)
432 if self.isBasePathEqual(ref):
433 if self.isPrevPathEqual(ref):
435 prev = self.pathUp(justSet)
437 currentRoot = self.getRoot()
438 if currentRoot is None or currentRoot != ref:
441 self.enterPath(ref, True)
443 serviceHandler = eServiceCenter.getInstance()
444 servicelist = serviceHandler.list(ref)
445 if not servicelist is None:
447 service = servicelist.getNext()
448 if not service.valid(): #check if end of list
450 unsigned_orbpos = service.getUnsignedData(4) >> 16
451 orbpos = service.getData(4) >> 16
454 if service.getPath().find("FROM PROVIDER") != -1:
455 service_type = _("Providers")
457 # why we need this cast?
458 service_name = str(nimmanager.getSatDescription(orbpos))
460 if unsigned_orbpos == 0xFFFF: #Cable
461 service_name = _("Cable")
462 elif unsigned_orbpos == 0xEEEE: #Terrestrial
463 service_name = _("Terrestrial")
465 if orbpos > 1800: # west
466 orbpos = 3600 - orbpos
470 service_name = ("%d.%d" + h) % (orbpos / 10, orbpos % 10)
471 service.setName("%s - %s" % (service_name, service_type))
472 self.servicelist.addService(service)
473 self.servicelist.finishFill()
475 self.setCurrentSelection(prev)
480 class myChannelSelection(ChannelSelectionBase):
482 <screen position="80,80" size="560,430" title="Select service to add...">
483 <ePixmap pixmap="skin_default/buttons/red.png" position="0,0" size="140,40" alphatest="on" />
484 <ePixmap pixmap="skin_default/buttons/green.png" position="140,0" size="140,40" alphatest="on" />
485 <ePixmap pixmap="skin_default/buttons/yellow.png" position="280,0" size="140,40" alphatest="on" />
486 <ePixmap pixmap="skin_default/buttons/blue.png" position="420,0" size="140,40" alphatest="on" />
487 <widget name="key_red" position="0,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#9f1313" transparent="1" />
488 <widget name="key_green" position="140,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#1f771f" transparent="1" />
489 <widget name="key_yellow" position="280,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#a08500" transparent="1" />
490 <widget name="key_blue" position="420,0" zPosition="1" size="140,40" font="Regular;20" halign="center" valign="center" backgroundColor="#18188b" transparent="1" />
491 <widget name="list" position="00,45" size="560,364" scrollbarMode="showOnDemand" />
494 def __init__(self, session, title):
495 ChannelSelectionBase.__init__(self, session)
496 self.onShown.append(self.__onExecCallback)
497 service = self.session.nav.getCurrentService()
499 info = service.info()
501 refstr = info.getInfoString(iServiceInformation.sServiceref)
502 self.servicelist.setPlayableIgnoreService(eServiceReference(refstr))
504 self["actions"] = ActionMap(["OkCancelActions", "TvRadioActions", "ChannelSelectBaseActions"],
506 "showProviders": self.doNothing,
507 "showSatellites": self.doNothing,
508 "cancel": self.cancel,
509 "ok": self.channelSelected,
510 "keyRadio": self.setModeRadio,
511 "keyTV": self.setModeTv
514 self["key_green"] = StaticText(_(""))
515 self["key_yellow"] = StaticText(_(""))
517 def __onExecCallback(self):
521 print "nothing to do..."
523 def channelSelected(self): # just return selected service
524 ref = self.getCurrentSelection()
525 if (ref.flags & 7) == 7:
527 elif not (ref.flags & eServiceReference.isMarker):
528 ref = self.getCurrentSelection()
533 self.showFavourites()
535 def setModeRadio(self):
537 self.showFavourites()
542 def activate_all(session):
543 NUM_CI=eDVBCIInterfaces.getInstance().getNumOfSlots()
544 print "[CI_Activate] FOUND %d CI Slots " % NUM_CI
547 def getValue(definitions, default):
550 # How many definitions are present
551 Len = len(definitions)
552 return Len > 0 and definitions[Len-1].text or default
554 for ci in range(NUM_CI):
555 filename="/etc/enigma2/ci"+str(ci)+".xml"
557 if not os_path.exists(filename):
558 print "[CI_Activate_Config_CI%d] no config file found" %ci
561 tree = ci_parse(filename).getroot()
565 for slot in tree.findall("slot"):
566 read_slot = getValue(slot.findall("id"), False).encode("UTF-8")
568 for caid in slot.findall("caid"):
569 read_caid = caid.get("id").encode("UTF-8")
570 usingcaid.append(long(read_caid,16))
572 for service in slot.findall("service"):
573 read_service_ref = service.get("ref").encode("UTF-8")
574 read_services.append (read_service_ref)
576 for provider in slot.findall("provider"):
577 read_provider_name = provider.get("name").encode("UTF-8")
578 read_provider_dvbname = provider.get("dvbnamespace").encode("UTF-8")
579 read_providers.append((read_provider_name,long(read_provider_dvbname,16)))
581 ci_config.append((int(read_slot), (read_services, read_providers, usingcaid)))
583 print "[CI_Activate_Config_CI%d] error parsing xml..." %ci
585 for item in ci_config:
586 print "[CI_Activate] activate CI%d with following settings:" %item[0]
590 eDVBCIInterfaces.getInstance().setDescrambleRules(item[0],item[1])
592 print "[CI_Activate_Config_CI%d] error setting DescrambleRules..." %item[0]
594 def find_in_list(list, search, listpos=0):
596 if item[listpos]==search:
600 global_session = None
602 def sessionstart(reason, session):
603 global global_session
604 global_session = session
606 def autostart(reason, **kwargs):
607 global global_session
609 print "[CI_Assignment] activating ci configs:"
610 activate_all(global_session)
612 global_session = None
614 def main(session, **kwargs):
615 session.open(CIselectMainMenu)
617 def menu(menuid, **kwargs):
618 if menuid == "setup" and eDVBCIInterfaces.getInstance().getNumOfSlots():
619 return [(_("Common Interface Assignment"), main, "ci_assign", 11)]
622 def Plugins(**kwargs):
623 if config.usage.setup_level.index > 1:
624 return [PluginDescriptor( where = PluginDescriptor.WHERE_SESSIONSTART, fnc = sessionstart ),
625 PluginDescriptor( where = PluginDescriptor.WHERE_AUTOSTART, fnc = autostart ),
626 PluginDescriptor( name = "CommonInterfaceAssignment", description = _("a gui to assign services/providers/caids to common interface modules"), where = PluginDescriptor.WHERE_MENU, fnc = menu )]
628 return [PluginDescriptor( where = PluginDescriptor.WHERE_SESSIONSTART, fnc = sessionstart ),
629 PluginDescriptor( where = PluginDescriptor.WHERE_AUTOSTART, fnc = autostart ),
630 PluginDescriptor( name = "CommonInterfaceAssignment", description = _("a gui to assign services/providers to common interface modules"), where = PluginDescriptor.WHERE_MENU, fnc = menu )]