2011-09-23 13 views
5

Recentemente ho aggiornato alla versione di sviluppo di wxPython (wxPython 2.9.2.4) poiché avevo bisogno della funzionalità di wx.NotificationMessage all'interno della mia applicazione. Ho cercato invano di creare bolle di notifica su determinati eventi utente a causa di qualcosa che potrebbe essere un possibile errore. Prima di inviare questo bug, volevo andare avanti e chiedere alla gente della mailing list cosa ritengono possa essere il problema e speriamo di trovare una soluzione all'interno del mio codice.Impossibile utilizzare wx.NotificationMessage correttamente con wxPython

Ecco il codice che ho usato: "Ciao Mondo"

import wx, sys 

app = wx.PySimpleApp() 

class TestTaskBarIcon(wx.TaskBarIcon): 

    def __init__(self): 
     wx.TaskBarIcon.__init__(self) 
     # create a test icon 
     bmp = wx.EmptyBitmap(16, 16) 
     dc = wx.MemoryDC(bmp) 
     dc.SetBrush(wx.RED_BRUSH) 
     dc.Clear() 
     dc.SelectObject(wx.NullBitmap) 

     testicon = wx.EmptyIcon() 
     testicon.CopyFromBitmap(bmp) 

     self.SetIcon(testicon) 
     self.Bind(wx.EVT_TASKBAR_LEFT_UP, lambda e: (self.RemoveIcon(),sys.exit())) 

     wx.NotificationMessage("", "Hello world!").Show() 

icon = TestTaskBarIcon() 
app.MainLoop() 

Sul mio computer con Windows 7, il codice crea un piccolo bianco barra delle applicazioni l'icona e crea un popup con la frase. Il problema? Il messaggio non è sulla mia icona. Un'altra icona viene creata e il messaggio viene posizionato lì. Vedere questa immagine: http://www.pasteall.org/pic/18068" >

Quello che ho pensato è stata che questo è probabilmente dovuto al fatto che ho passato senza parametro padre sulla linea 22:

wx.NotificationMessage("", "Hello world!").Show() 

Ecco quello che ho cambiato in:.

wx.NotificationMessage("", "Hello world!", self).Show() 

Dove 'sé' si riferisce al l'icona barra delle applicazioni Quando lo faccio, ottengo un errore:

Traceback (most recent call last): 
    File "C:\Python27\testnotificationmessage.py", line 24, in <module> 
    icon = TestTaskBarIcon() 
    File "C:\Python27\testnotificationmessage.py", line 22, in __init__ 
    wx.NotificationMessage("", "Hello world!", self).Show() 
    File "C:\Python27\lib\site-packages\wx-2.9.2-msw\wx\_misc.py", line 1213, in __init__ 
    _misc_.NotificationMessage_swiginit(self,_misc_.new_NotificationMessage(*args)) 
TypeError: in method 'new_NotificationMessage', expected argument 3 of type 'wxWindow *' 

Cosa sta succedendo? Se rimuovo quell'argomento, non ottengo il mio risultato, se aggiungo l'argomento, ottengo un errore! Come dovrei usare wx.NotificationMessage con un wx.TaskBarIcon!

Si prega di aiuto! Spero di aver fornito abbastanza dettagli. Si prega di commentare se avete bisogno di più!

+0

Hai trovato documentazione per 2.9.2.4? Sono cieco o sfortunato ... – Fenikso

risposta

9

Non mi consiglia di utilizzare 2.9 ancora. Ho incontrato alcuni bug strani quando lo provavo.

È possibile avere la stessa funzionalità in 2.8. Sto usando un codice un po 'modificato che ho trovato qualche tempo fa.

import wx, sys 

try: 
    import win32gui #, win32con 
    WIN32 = True 
except: 
    WIN32 = False 

class BalloonTaskBarIcon(wx.TaskBarIcon): 
    """ 
    Base Taskbar Icon Class 
    """ 
    def __init__(self): 
     wx.TaskBarIcon.__init__(self) 
     self.icon = None 
     self.tooltip = "" 

    def ShowBalloon(self, title, text, msec = 0, flags = 0): 
     """ 
     Show Balloon tooltip 
     @param title - Title for balloon tooltip 
     @param msg - Balloon tooltip text 
     @param msec - Timeout for balloon tooltip, in milliseconds 
     @param flags - one of wx.ICON_INFORMATION, wx.ICON_WARNING, wx.ICON_ERROR 
     """ 
     if WIN32 and self.IsIconInstalled(): 
      try: 
       self.__SetBalloonTip(self.icon.GetHandle(), title, text, msec, flags) 
      except Exception: 
       pass # print(e) Silent error 

    def __SetBalloonTip(self, hicon, title, msg, msec, flags): 

     # translate flags 
     infoFlags = 0 

     if flags & wx.ICON_INFORMATION: 
      infoFlags |= win32gui.NIIF_INFO 
     elif flags & wx.ICON_WARNING: 
      infoFlags |= win32gui.NIIF_WARNING 
     elif flags & wx.ICON_ERROR: 
      infoFlags |= win32gui.NIIF_ERROR 

     # Show balloon 
     lpdata = (self.__GetIconHandle(), # hWnd 
        99,      # ID 
        win32gui.NIF_MESSAGE|win32gui.NIF_INFO|win32gui.NIF_ICON, # flags: Combination of NIF_* flags 
        0,      # CallbackMessage: Message id to be pass to hWnd when processing messages 
        hicon,     # hIcon: Handle to the icon to be displayed 
        '',      # Tip: Tooltip text 
        msg,      # Info: Balloon tooltip text 
        msec,      # Timeout: Timeout for balloon tooltip, in milliseconds 
        title,     # InfoTitle: Title for balloon tooltip 
        infoFlags     # InfoFlags: Combination of NIIF_* flags 
       ) 
     win32gui.Shell_NotifyIcon(win32gui.NIM_MODIFY, lpdata) 

     self.SetIcon(self.icon, self.tooltip) # Hack: because we have no access to the real CallbackMessage value 

    def __GetIconHandle(self): 
     """ 
     Find the icon window. 
     This is ugly but for now there is no way to find this window directly from wx 
     """ 
     if not hasattr(self, "_chwnd"): 
      try: 
       for handle in wx.GetTopLevelWindows(): 
        if handle.GetWindowStyle(): 
         continue 
        handle = handle.GetHandle() 
        if len(win32gui.GetWindowText(handle)) == 0: 
         self._chwnd = handle 
         break 
       if not hasattr(self, "_chwnd"): 
        raise Exception 
      except: 
       raise Exception, "Icon window not found" 
     return self._chwnd 

    def SetIcon(self, icon, tooltip = ""): 
     self.icon = icon 
     self.tooltip = tooltip 
     wx.TaskBarIcon.SetIcon(self, icon, tooltip) 

    def RemoveIcon(self): 
     self.icon = None 
     self.tooltip = "" 
     wx.TaskBarIcon.RemoveIcon(self) 

# =================================================================== 
app = wx.PySimpleApp() 

class TestTaskBarIcon(BalloonTaskBarIcon): 

    def __init__(self): 
     wx.TaskBarIcon.__init__(self) 
     # create a test icon 
     bmp = wx.EmptyBitmap(16, 16) 
     dc = wx.MemoryDC(bmp) 
     dc.SetBrush(wx.RED_BRUSH) 
     dc.Clear() 
     dc.SelectObject(wx.NullBitmap) 

     testicon = wx.EmptyIcon() 
     testicon.CopyFromBitmap(bmp) 

     self.SetIcon(testicon) 
     self.Bind(wx.EVT_TASKBAR_LEFT_UP, lambda e: (self.RemoveIcon(),sys.exit())) 

     self.ShowBalloon("", "Hello world!") 

icon = TestTaskBarIcon() 
app.MainLoop() 
6

esiste un metodo non documentato nascosto in TaskBarIcon chiamato ShowBalloon che viene realizzato solo per Windows.

Da the source:

def ShowBalloon(*args, **kwargs): 
    """ 
    ShowBalloon(self, String title, String text, unsigned int msec=0, int flags=0) -> bool 

    Show a balloon notification (the icon must have been already 
    initialized using SetIcon). Only implemented for Windows. 

    title and text are limited to 63 and 255 characters respectively, msec 
    is the timeout, in milliseconds, before the balloon disappears (will 
    be clamped down to the allowed 10-30s range by Windows if it's outside 
    it) and flags can include wxICON_ERROR/INFO/WARNING to show a 
    corresponding icon 

    Returns True if balloon was shown, False on error (incorrect parameters 
    or function unsupported by OS) 

    """ 
    return _windows_.TaskBarIcon_ShowBalloon(*args, **kwargs) 

ho testato su Windows con wxPython 2.9.4.0 e funziona bene.