From 31fb73a15d12559b15f5506622c3902476d3ea0f Mon Sep 17 00:00:00 2001 From: Stefan Pluecken Date: Wed, 22 Feb 2006 22:14:19 +0000 Subject: [PATCH] cleanup the plugins to fit the new namespace --- configure.ac | 15 +- lib/python/Plugins/DemoPlugins/Makefile.am | 1 + .../DemoPlugins/TestPlugin/Makefile.am | 6 + .../DemoPlugins/TestPlugin/__init__.py | 0 .../Plugins/DemoPlugins/TestPlugin/plugin.py | 69 ++++++ lib/python/Plugins/Extensions/Makefile.am | 1 + .../Extensions/TuxboxPlugins/Makefile.am | 8 + .../Extensions/TuxboxPlugins/__init__.py | 0 .../Extensions/TuxboxPlugins/plugin.py | 46 ++++ .../Extensions/TuxboxPlugins/tuxbox.png | Bin 0 -> 1805 bytes .../Extensions/WebInterface/Makefile.am | 7 + .../Extensions/WebInterface/__init__.py | 0 .../Plugins/Extensions/WebInterface/plugin.py | 19 ++ .../FrontprocessorUpgrade/Makefile.am | 6 + .../FrontprocessorUpgrade/__init__.py | 0 .../FrontprocessorUpgrade/plugin.py | 54 +++++ lib/python/Plugins/SystemPlugins/Makefile.am | 1 + .../SystemPlugins/SoftwareUpdate/Makefile.am | 7 + .../SystemPlugins/SoftwareUpdate/__init__.py | 0 .../SystemPlugins/SoftwareUpdate/plugin.py | 222 ++++++++++++++++++ .../SystemPlugins/SoftwareUpdate/update.png | Bin 0 -> 1090 bytes 21 files changed, 456 insertions(+), 6 deletions(-) create mode 100644 lib/python/Plugins/DemoPlugins/Makefile.am create mode 100644 lib/python/Plugins/DemoPlugins/TestPlugin/Makefile.am create mode 100644 lib/python/Plugins/DemoPlugins/TestPlugin/__init__.py create mode 100644 lib/python/Plugins/DemoPlugins/TestPlugin/plugin.py create mode 100644 lib/python/Plugins/Extensions/Makefile.am create mode 100644 lib/python/Plugins/Extensions/TuxboxPlugins/Makefile.am create mode 100644 lib/python/Plugins/Extensions/TuxboxPlugins/__init__.py create mode 100644 lib/python/Plugins/Extensions/TuxboxPlugins/plugin.py create mode 100644 lib/python/Plugins/Extensions/TuxboxPlugins/tuxbox.png create mode 100644 lib/python/Plugins/Extensions/WebInterface/Makefile.am create mode 100644 lib/python/Plugins/Extensions/WebInterface/__init__.py create mode 100644 lib/python/Plugins/Extensions/WebInterface/plugin.py create mode 100644 lib/python/Plugins/SystemPlugins/FrontprocessorUpgrade/Makefile.am create mode 100644 lib/python/Plugins/SystemPlugins/FrontprocessorUpgrade/__init__.py create mode 100644 lib/python/Plugins/SystemPlugins/FrontprocessorUpgrade/plugin.py create mode 100644 lib/python/Plugins/SystemPlugins/Makefile.am create mode 100644 lib/python/Plugins/SystemPlugins/SoftwareUpdate/Makefile.am create mode 100644 lib/python/Plugins/SystemPlugins/SoftwareUpdate/__init__.py create mode 100644 lib/python/Plugins/SystemPlugins/SoftwareUpdate/plugin.py create mode 100644 lib/python/Plugins/SystemPlugins/SoftwareUpdate/update.png diff --git a/configure.ac b/configure.ac index b5a19aad..c77d3ceb 100644 --- a/configure.ac +++ b/configure.ac @@ -60,12 +60,15 @@ lib/python/Makefile lib/python/Components/Makefile lib/python/Screens/Makefile lib/python/Plugins/Makefile -lib/python/Plugins/update/Makefile -lib/python/Plugins/FileManager/Makefile -lib/python/Plugins/fpupgrade/Makefile -lib/python/Plugins/test/Makefile -lib/python/Plugins/tuxboxplugins/Makefile -lib/python/Plugins/web/Makefile +lib/python/Plugins/SystemPlugins/Makefile +lib/python/Plugins/SystemPlugins/SoftwareUpdate/Makefile +lib/python/Plugins/SystemPlugins/FrontprocessorUpgrade/Makefile +lib/python/Plugins/DemoPlugins/Makefile +lib/python/Plugins/DemoPlugins/TestPlugin/Makefile +lib/python/Plugins/Extensions/Makefile +lib/python/Plugins/Extensions/TuxboxPlugins/Makefile +lib/python/Plugins/Extensions/WebInterface/Makefile +lib/python/Plugins/Extensions/FileManager/Makefile lib/python/Tools/Makefile lib/service/Makefile lib/components/Makefile diff --git a/lib/python/Plugins/DemoPlugins/Makefile.am b/lib/python/Plugins/DemoPlugins/Makefile.am new file mode 100644 index 00000000..86e98ec2 --- /dev/null +++ b/lib/python/Plugins/DemoPlugins/Makefile.am @@ -0,0 +1 @@ +SUBDIRS = TestPlugin diff --git a/lib/python/Plugins/DemoPlugins/TestPlugin/Makefile.am b/lib/python/Plugins/DemoPlugins/TestPlugin/Makefile.am new file mode 100644 index 00000000..dd6980ac --- /dev/null +++ b/lib/python/Plugins/DemoPlugins/TestPlugin/Makefile.am @@ -0,0 +1,6 @@ +installdir = $(LIBDIR)/enigma2/python/Plugins/TestPlugin + +install_PYTHON = \ + __init__.py \ + plugin.py + diff --git a/lib/python/Plugins/DemoPlugins/TestPlugin/__init__.py b/lib/python/Plugins/DemoPlugins/TestPlugin/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/lib/python/Plugins/DemoPlugins/TestPlugin/plugin.py b/lib/python/Plugins/DemoPlugins/TestPlugin/plugin.py new file mode 100644 index 00000000..5991f594 --- /dev/null +++ b/lib/python/Plugins/DemoPlugins/TestPlugin/plugin.py @@ -0,0 +1,69 @@ +from enigma import * +from Screens.Screen import Screen +from Screens.MessageBox import MessageBox +from Components.ActionMap import NumberActionMap +from Components.Label import Label +from Components.Input import Input +from Components.GUIComponent import * +from Components.Pixmap import Pixmap +from Components.FileList import FileEntryComponent, FileList +from Plugins.Plugin import PluginDescriptor + +import os + +class Test(Screen): + skin = """ + + + + + """ + def __init__(self, session, args = None): + self.skin = Test.skin + Screen.__init__(self, session) + + self["list"] = FileList("/", matchingPattern = "^.*\.(png|avi|mp3|mpeg|ts)") + self["pixmap"] = Pixmap() + + #self["text"] = Input("1234", maxSize=True, type=Input.NUMBER) + + self["actions"] = NumberActionMap(["WizardActions", "InputActions"], + { + "ok": self.ok, + "back": self.close, +# "left": self.keyLeft, +# "right": self.keyRight, + "1": self.keyNumberGlobal, + "2": self.keyNumberGlobal, + "3": self.keyNumberGlobal, + "4": self.keyNumberGlobal, + "5": self.keyNumberGlobal, + "6": self.keyNumberGlobal, + "7": self.keyNumberGlobal, + "8": self.keyNumberGlobal, + "9": self.keyNumberGlobal, + "0": self.keyNumberGlobal + }, -1) + + def keyLeft(self): + self["text"].left() + + def keyRight(self): + self["text"].right() + + def ok(self): + selection = self["list"].getSelection() + if selection[1] == True: # isDir + self["list"].changeDir(selection[0]) + else: + self["pixmap"].instance.setPixmapFromFile(selection[0]) + + def keyNumberGlobal(self, number): + print "pressed", number + self["text"].number(number) + +def main(session): + session.open(Test) + +def Plugins(): + return PluginDescriptor(name="Test", description="plugin to test some capabilities", where = PluginDescriptor.WHERE_PLUGINMENU, fnc=main) diff --git a/lib/python/Plugins/Extensions/Makefile.am b/lib/python/Plugins/Extensions/Makefile.am new file mode 100644 index 00000000..a8998c13 --- /dev/null +++ b/lib/python/Plugins/Extensions/Makefile.am @@ -0,0 +1 @@ +SUBDIRS = TuxboxPlugins WebInterface FileManager diff --git a/lib/python/Plugins/Extensions/TuxboxPlugins/Makefile.am b/lib/python/Plugins/Extensions/TuxboxPlugins/Makefile.am new file mode 100644 index 00000000..529d9888 --- /dev/null +++ b/lib/python/Plugins/Extensions/TuxboxPlugins/Makefile.am @@ -0,0 +1,8 @@ +installdir = $(LIBDIR)/enigma2/python/Plugins/TuxboxPlugins + +install_PYTHON = \ + __init__.py \ + plugin.py \ + tuxbox.png + + diff --git a/lib/python/Plugins/Extensions/TuxboxPlugins/__init__.py b/lib/python/Plugins/Extensions/TuxboxPlugins/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/lib/python/Plugins/Extensions/TuxboxPlugins/plugin.py b/lib/python/Plugins/Extensions/TuxboxPlugins/plugin.py new file mode 100644 index 00000000..7ab02da7 --- /dev/null +++ b/lib/python/Plugins/Extensions/TuxboxPlugins/plugin.py @@ -0,0 +1,46 @@ +# must be fixed for the new plugin interface +from enigma import * +from Screens.Screen import Screen +from Screens.MessageBox import MessageBox +from Components.ActionMap import ActionMap +from Components.Label import Label +from Tools.BoundFunction import boundFunction +from Tools.Directories import pathExists +from Plugins.Plugin import PluginDescriptor + +import os + +TUXBOX_PLUGINS_PATH = "/usr/lib/tuxbox/plugins/" + +def getPlugins(): + pluginlist = [] + + if pathExists(TUXBOX_PLUGINS_PATH): + dir = os.listdir(TUXBOX_PLUGINS_PATH) + + for x in dir: + if x[-3:] == "cfg": + params = getPluginParams(x) + pluginlist.append(PluginDescriptor(name=params["name"], description=params["desc"], where = PluginDescriptor.WHERE_PLUGINMENU, icon="tuxbox.png", fnc=boundFunction(main, plugin=x))) + + return pluginlist + +def getPluginParams(file): + params = {} + try: + file = open(TUXBOX_PLUGINS_PATH + file, "r") + for x in file.readlines(): + split = x.split("=") + params[split[0]] = split[1] + file.close() + except IOError: + print "no tuxbox plugins found" + + return params + +def main(session, plugin): + print "Running plugin " + plugin[:-4] + ".so with config file", plugin + print getPluginParams(plugin) + +def Plugins(): + return getPlugins() \ No newline at end of file diff --git a/lib/python/Plugins/Extensions/TuxboxPlugins/tuxbox.png b/lib/python/Plugins/Extensions/TuxboxPlugins/tuxbox.png new file mode 100644 index 0000000000000000000000000000000000000000..0d28db01de15041e3746e71e5dea6eb39f2ffeb4 GIT binary patch literal 1805 zcmV+o2lDudP)pVOm{cWL#umQ*T67XL37u zUtVTyXk%i9VjGfS09|NwSZj4=XJun(YFuk^VrynWeqUf~cWP*Cgl8>#X+vaeb7O9A zU2l47Y-wn0Z$N@)a&KZ_bAEeqPGxj@p=}0Hg=cJZd}?)pa&&QJc!O+rcxrinr*Z&+ zbY5_Ge|&aib9Z`be2AWQA)0&-dU$+peuH^?c#nKNYk`JxeTsN}e{O+{a)F0nWl`JqR*HC)0Pg(p8%@prW7Dq8g^5pxdGk%cn5ZtN@>&6_rz{0=5zQe%8#lXMr&J^>}0K>+@%3}4@AkWFj&CJ8i%g@fx&DGD&&C=D;)6m@4)z;hH+T7XT;@{xq==SKr z>Fen2@9XXI_3-lV^Y!xi^z!!j_4@hm`TY3%{rdU;|Nj5{ql>$i00001bW%=J06^y0 zW&i*H32;bRa{vGf6951U69E94oEQKA00(qQO+^RR0SgZ>Fc1;CK>z>)lSxEDR5;6h z(|=4GWf%u=xGNxg>2T@@OcX|fTbf0e4F~Enf)R((1~w-{=YB}R#=340a~o{akqwp3 zt`w#rii$_(979*c%(3gn!ZOEhM|!nfENe*^rK6!xaJY1PO|P#n{YQ!%&wKyfC*SAE zd(TBhMU?;LWJUJ9sKv@2M?@23!m-7S91sK{PnC5`6Hhb=03dPFA_l}p16lkM06dar zB(>*J@@&6~0A12%Z+4x$=Wz=O4oDhbLE@qPU&Jy1R7%g2_RGNF*kB_`Nau{^Tpk=7 zyYLYp3#84m`U~H;ocoH%l1rLS#f@+4&wFMv;>m0YGLzt(cDbeyx(!GmVUjaK!TW&_ zL@*VA+mPZT4-QdTz}vp#5Fa+FO4CH1@1^iz5!7%ghKe4sUzPy z{+;`OCSi#W!SF#is;P^24%z3C$wvyr5DEsd5zzydVMjb(`@}pH*Nrm*5JM;qcQKBk zalhLeA$rI-4+^1xh#*Ft7S=K3ofzdK%-dk;eHaWv5sC=;8OH7!ar-9zj4(fLNF2qp zLJ?mNaxe_<81eciBh6Lea+-g(*KAUsh28Jlo7{NA*;q@;zL6#{4bLqvb|cdpATZ(@1N zd5-J-J8a%7E7^4#k8Be-?qZHSYrUKx9?MgmsJq1VepqY`8;x46-nH-Omsc-xH&5rp zf|%$S8S%uv#-DyYT3ndBxi*}$TUVi_wC}w6+MYcd)6x~s7nW6QeeU%mN4BXpzm&Un zjcPV0zt&(d&_=rb#QPs;jU99wO&eQZeCCx`x9r%GxpMgmY`SyXJFZ!+)@!&rI@(*? z=r)r{pZ?TmCr>ssJi9#UNxUD^MJAf2oAu2&%4js`^|c2nDmN*!q2cH2*I!6UT0Mz_ z%5?fqt1FKm)9Y)h4_4`_s!8w#pr&gDi v?xrY1`@BuA(&$ait>)i;x8ELK=;! + + + + + + """ + def __init__(self, session): + self.skin = FPUpgrade.skin + Screen.__init__(self, session) + + from Tools.DreamboxHardware import getFPVersion + version = str(getFPVersion() or "N/A") + newversion = str(getUpgradeVersion() or "N/A") + + self["text"] = Label(_("Your frontprocessor firmware must be upgraded.\nPress OK to start upgrade.")) + self["oldversion_label"] = Label(_("Current version:")) + self["newversion_label"] = Label(_("New version:")) + + self["oldversion"] = Label(version) + self["newversion"] = Label(newversion) + + self["actions"] = ActionMap(["OkCancelActions"], + { + "ok": self.ok, + "cancel": self.close, + }) + + def ok(self): + self.close(4) + +def Plugins(): + from Tools.DreamboxHardware import getFPVersion + version = getFPVersion() + newversion = getUpgradeVersion() or 0 + if version is not None and version < newversion: + return PluginDescriptor(name="FP Upgrade", where = PluginDescriptor.WHERE_WIZARD, fnc=FPUpgrade) + else: + return [ ] diff --git a/lib/python/Plugins/SystemPlugins/Makefile.am b/lib/python/Plugins/SystemPlugins/Makefile.am new file mode 100644 index 00000000..2b71624c --- /dev/null +++ b/lib/python/Plugins/SystemPlugins/Makefile.am @@ -0,0 +1 @@ +SUBDIRS = SoftwareUpdate FrontprocessorUpgrade diff --git a/lib/python/Plugins/SystemPlugins/SoftwareUpdate/Makefile.am b/lib/python/Plugins/SystemPlugins/SoftwareUpdate/Makefile.am new file mode 100644 index 00000000..9f7bd4c4 --- /dev/null +++ b/lib/python/Plugins/SystemPlugins/SoftwareUpdate/Makefile.am @@ -0,0 +1,7 @@ +installdir = $(LIBDIR)/enigma2/python/Plugins/SoftwareUpdate + +install_PYTHON = \ + __init__.py \ + plugin.py \ + update.png + diff --git a/lib/python/Plugins/SystemPlugins/SoftwareUpdate/__init__.py b/lib/python/Plugins/SystemPlugins/SoftwareUpdate/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/lib/python/Plugins/SystemPlugins/SoftwareUpdate/plugin.py b/lib/python/Plugins/SystemPlugins/SoftwareUpdate/plugin.py new file mode 100644 index 00000000..c065c815 --- /dev/null +++ b/lib/python/Plugins/SystemPlugins/SoftwareUpdate/plugin.py @@ -0,0 +1,222 @@ +from enigma import * +from Screens.Screen import Screen +from Screens.MessageBox import MessageBox +from Components.ActionMap import ActionMap, NumberActionMap +from Components.ScrollLabel import ScrollLabel +from Components.GUIComponent import * +from Components.MenuList import MenuList +from Components.Input import Input +from Screens.Console import Console +from Plugins.Plugin import PluginDescriptor + +import os + +class UpdatePluginMenu(Screen): + skin = """ + + + """ + + def __init__(self, session, args = 0): + self.skin = UpdatePluginMenu.skin + Screen.__init__(self, session) + + self.menu = args + + list = [] + if self.menu == 0: + list.append((_("Upgrade"), "upgrade")) + list.append((_("Advanced"), "advanced")) + elif self.menu == 1: + list.append((_("Choose source"), "source")) + list.append((_("Packet management"), "ipkg")) + list.append((_("Settings"), "setup")) + + self["menu"] = MenuList(list) + + self["actions"] = ActionMap(["WizardActions", "DirectionActions"], + { + "ok": self.go, + "back": self.close, + }, -1) + + def go(self): + if self.menu == 0: + if (self["menu"].l.getCurrentSelection()[1] == "upgrade"): + self.session.openWithCallback(self.runUpgrade, MessageBox, _("Do you want to update your Dreambox?\nAfter pressing OK, please wait!")) + if (self["menu"].l.getCurrentSelection()[1] == "advanced"): + self.session.open(UpdatePluginMenu, 1) + if self.menu == 1: + if (self["menu"].l.getCurrentSelection()[1] == "source"): + self.session.open(IPKGSource) + elif (self["menu"].l.getCurrentSelection()[1] == "ipkg"): + self.session.open(Ipkg) + elif (self["menu"].l.getCurrentSelection()[1] == "setup"): + self.session.open(MessageBox, _("Function not yet implemented"), MessageBox.TYPE_ERROR) + def runUpgrade(self, result): + if result: + self.session.open(Console, ["ipkg update", "ipkg upgrade -force-defaults -force-overwrite"]) + +class IPKGSource(Screen): + skin = """ + + + """ + + def __init__(self, session, args = None): + self.skin = IPKGSource.skin + Screen.__init__(self, session) + + fp = file('/etc/ipkg/official-feed.conf', 'r') + sources = fp.readlines() + fp.close() + + self["text"] = Input(sources[0], maxSize=False, type=Input.TEXT) + + self["actions"] = NumberActionMap(["WizardActions", "InputActions"], + { + "ok": self.go, + "back": self.close, + "left": self.keyLeft, + "right": self.keyRight, + "1": self.keyNumberGlobal, + "2": self.keyNumberGlobal, + "3": self.keyNumberGlobal, + "4": self.keyNumberGlobal, + "5": self.keyNumberGlobal, + "6": self.keyNumberGlobal, + "7": self.keyNumberGlobal, + "8": self.keyNumberGlobal, + "9": self.keyNumberGlobal, + "0": self.keyNumberGlobal + }, -1) + + def go(self): + fp = file('/etc/ipkg/official-feed.conf', 'w') + fp.write(self["text"].getText()) + fp.close() + self.close() + + def keyLeft(self): + self["text"].left() + + def keyRight(self): + self["text"].right() + + def keyNumberGlobal(self, number): + print "pressed", number + self["text"].number(number) + +RT_HALIGN_LEFT = 0 +RT_HALIGN_RIGHT = 1 +RT_HALIGN_CENTER = 2 +RT_HALIGN_BLOCK = 4 + +RT_VALIGN_TOP = 0 +RT_VALIGN_CENTER = 8 +RT_VALIGN_BOTTOM = 16 + +def PacketEntryComponent(packet): + res = [ packet ] + + res.append((eListboxPythonMultiContent.TYPE_TEXT, 0, 0,250, 30, 0, RT_HALIGN_LEFT|RT_VALIGN_CENTER, packet[0])) + res.append((eListboxPythonMultiContent.TYPE_TEXT, 250, 0, 200, 30, 1, RT_HALIGN_LEFT|RT_VALIGN_CENTER, packet[1])) + res.append((eListboxPythonMultiContent.TYPE_TEXT, 450, 0, 100, 30, 1, RT_HALIGN_LEFT|RT_VALIGN_CENTER, packet[2])) + return res + +class PacketList(GUIComponent): + def __init__(self, list): + GUIComponent.__init__(self) + self.l = eListboxPythonMultiContent() + self.l.setList(list) + self.l.setFont(0, gFont("Regular", 20)) + self.l.setFont(1, gFont("Regular", 18)) + + def getCurrent(self): + return self.l.getCurrentSelection() + + def GUIcreate(self, parent): + self.instance = eListbox(parent) + self.instance.setContent(self.l) + self.instance.setItemHeight(30) + + def GUIdelete(self): + self.instance.setContent(None) + self.instance = None + + def invalidate(self): + self.l.invalidate() + +class Ipkg(Screen): + skin = """ + + + """ + + def __init__(self, session, args = None): + self.skin = Ipkg.skin + Screen.__init__(self, session) + + list = [] + self.list = list + self.fillPacketList() + + self["list"] = PacketList(self.list) + + self["actions"] = ActionMap(["WizardActions"], + { + "ok": self.close, + "back": self.close + }, -1) + + + def fillPacketList(self): + lines = os.popen("ipkg list", "r").readlines() + packetlist = [] + for x in lines: + split = x.split(' - ') + packetlist.append([split[0].strip(), split[1].strip()]) + + lines = os.popen("ipkg list_installed", "r").readlines() + + installedlist = {} + for x in lines: + split = x.split(' - ') + installedlist[split[0].strip()] = split[1].strip() + + for x in packetlist: + status = "" + if installedlist.has_key(x[0]): + if installedlist[x[0]] == x[1]: + status = "installed" + else: + status = "upgradable" + self.list.append(PacketEntryComponent([x[0], x[1], status])) + + def go(self): + if self.update: + self.session.openWithCallback(self.doUpdate, MessageBox, _("Do you want to update your Dreambox?\nAfter pressing OK, please wait!")) + else: + self.close() + + def doUpdateDelay(self): + lines = os.popen("ipkg update && ipkg upgrade", "r").readlines() + string = "" + for x in lines: + string += x + self["text"].setText(_("Updating finished. Here is the result:") + "\n\n" + string) + self.update = False + + + def doUpdate(self, val = False): + if val == True: + self["text"].setText(_("Updating... Please wait... This can take some minutes...")) + self.delayTimer.start(0, 1) + else: + self.close() + +def UpgradeMain(session): + session.open(UpdatePluginMenu) + +def Plugins(): + return PluginDescriptor(name="Softwareupdate", description="Updates your receiver's software", icon="update.png", where = PluginDescriptor.WHERE_PLUGINMENU, fnc=UpgradeMain) diff --git a/lib/python/Plugins/SystemPlugins/SoftwareUpdate/update.png b/lib/python/Plugins/SystemPlugins/SoftwareUpdate/update.png new file mode 100644 index 0000000000000000000000000000000000000000..6bc17b1e5eea771fb9ba1ed5f80f61e20780ddd3 GIT binary patch literal 1090 zcmV-I1ikx-P)(*7IVvYDRZm<_NM1}vTX1WAh=h}(pQ=bhS1Tq%Lp)b`_ zWj!`vIy782F;^oWHys)-XJLdsIAuOKV{2xJH7;rl1zrjTLJbE`3kF3J3T_DnHyRRk z6b@Pt3rGzKISmOn34^|QlOA!o34hlI72rd#0Ne>G?3kWR_3q2wlfIT*kNkW!OM3+iL zkWfjKVOyk>kiodM)&~PT2?R_J2YL<%au5i41_Lk=33w3+brA`25D9M*3V0Fpm49?CAs;UWf zbq4?c55>g{Wn~8c{}A~25by5~?CcMoo(g?^2& z62rp}sHhE`oD6k!33PM`_VyI-?-T6o6Y1#_=H?RF*%8dl5Q>Tme0&OIWC-Qt9^Ty{ z*w-V<$|0ww8)|70R#gty)g;l+Bge%dhJ_aY|33ZwLH_ibYfxxpP-|vVR#HM-SVu=hH!&|JiG_8IiFsyW zPGVh4J2)+BWm02bRn{kzUH||932;bRa{vGf6951U69E94oEQKA00(qQO+^RR0Rjyo zId8S;(f|MfAW1|)RCwBBC@p463{xZ-#3VF$AaP8CYnjW|+x%ZANv) z3?x@T-A|lmBzH5xokP6MGf-{D5nwZL_!%A+FjbgQg*UX}wxWhMdLluKPy8t?JS3>X zC6r9f`g-|p?u=xn?NQC6nh9#QFwf4{%_bR`8`?RVG8p7oi_@% literal 0 HcmV?d00001 -- 2.30.2