wxWidgets/wxPython/wx/lib/ErrorDialogs.py
2003-11-12 21:34:20 +00:00

880 lines
35 KiB
Python

#----------------------------------------------------------------------------
# Name: ErrorDialogs.py
# Version: 1.0
# Created: September--October 2001
# Author: Chris Fama of Wholly Snakes Software,
# Chris.Fama@whollysnakes.com
#----------------------------------------------------------------------------
"""
ErrorDialogs.py: by Christopher J. Fama (trading under the name
Wholly Snakes Software {Australian Business Number: 28379514278}).
This code is released under the auspices of the modified version of
the GNU Library Public License (LGPL) that constitutes the license
under which the GUI package wxPython is released [see the file
LICENSE.TXT accompanying that distribution). I must also thank Graham
Boyd, of Boyd Mining, and CEO of the Minserve group of companies
(www.minserve.com.au), for kindly allowing the release of this
module under a "free" license, despite a certain part of it's
development having taken place while working for him...
Please note that this code, written for Python 2.1, is derives from
code written when Python 1.5.2 was current. Although since Python 2.0
the sys.excepthook variable has been available, for ease and potential
backwards compatibility (or to be more realistic
backwards-PARTIAL-compatibility!), the code catches errors by
assigning a custom object to sys.stderr and monitoring calls to it's
writ) method. Such calls which take place with a new value of
sys.last_traceback stand as evidence that at interpreter error has
just occurred; please note that this means that NO OTHER OUTPUT TO
sys.stderr WILL BE DISPLAYED. THIS INCLUDES "warnings" generated by
the interpreter. As far as I am aware, unless your code itself writes
to sys.stderr itself, these will be the only things you miss out
on--and many, if not most or all, of these will occur before your code
even gets the opportunity to set one of these file-like objects in
place. If this is a problem for you and you can't work around it,
please contact means or the wxPython-users mailing list.
DOCUMENTATION:
This is a module to display errors--either explicitly requested by a
script or arbitrary interpreter errors--these needs arose in the
course of a commercial (contract) project, but were not developed in
the course of work on such [well, very little, albeit concurrently]...
[NBNB.Does not currently support display of other than interpreter errors.]
Usage, after 'from wxPython.lib.ErrorDialogs import *' (all
identifiers defined in this module begin with "wxPy", and many of them
with "wxPyError_", so there should be no namespace conflicts...):
wxPyNewErrorDialog (<win> (['frame='] <frame (can be None)>,
<OPTIONS>))
...
wxPyDestroyErrorDialogIfPresent () # e.g. when top frame destroyed)
for unhandled errors, or
returnval = wxpyNonFatalError (<frame (can be None)>,
<HTML message>
[,<OPTIONS>])
or
returnval = wxPyFatalError (<HTML message>,
[,<OPTIONS, an extra one of which may be
'frame=' <frame (defaults to None)>>])
or
wxPybNonWindowingError (message)
for explicit errors.
<win> is one of
wxPyNonFatalErrorDialog
wxPyFatalErrorDialog
wxPyFatalErrorDialogWithTraceback
wxPyNonFatalErrorDialogWithTraceback
wxPyNonWindowingErrorHandler
and the OPTIONS (with defaults) are: (please note that the options for
wxPyNonWindowingErrorHandler / wxPyNonWindowingError are 'almost' a (small))
subset of these):
'modal' [default 1]: block until dismissed.
'programname' [default "Python program"]: appears inThe
caption of the dialog, amidst it's text and (by default) in mailings.
'whendismissed' option, if given, this should be a string, to be
executed in a restricted environment (see rexec module) after
dialog dismissal. Typically, this should be Python code to "clean
up" after what was presumably the partial execution of some
operation started by the user, after the unexpected interruption
by some error [which--at least the way I work, means an unexpected
error in the code, since exceptions that may be raised by some
foreseen (in a programmatic sense) should normally be handled by
try...except clauses].
NOTE THAT CURRENTLY THE rexec CODE IS NOT WORKING, SO THIS IS JUST DONE
BY exec...
'mailto': if None, give no "e-mail support" option, otherwise this
is meant to be an address for 'bug reports'; a command button will
be created for this, which pops up another dialog [Linux], or a window of
another program [Windows].
On Windows, this will launch your mailer (assuming your mailer
would start when a file with the '.eml'extension is double-clicked
in Windows Explorer--this will be the case for Microsoft Outlook
Express [tm and all those legal necessities] if installed, and the
association has not been taken over by some other program. On
Linux, it will open a file dialog to save the message, by default
as a '.html' file...{Please note that, on Windows and with current
(when I got the machine I'm writing this on--early 2000) versions
of Outlook Express, you may need to enter your e-mail address in
the box beside the "mail support" button.)
The template for the mail message is in the 'MessageTemplate'
attribute of the dialog-derived class in question (e.g.,
wxPyNonFatalErrorDialog). Search for this in the code below to
see the default (note that this template is in HTML format). This
attributes uses the '%{name}s' string conversion feature of Python
[see sec.2 of the Python Library Reference]; allowed Lance are:
programname version exceptionname exceptionvalue
extraexceptioninformation traceback
'version': str(this) will appear after 'Version' below "Error in
<programname>" at the top of the dialog.
EXAMPLES:
sys.stderr = wxPyNonFatalErrorWindowWithTraceback (
parentframe,
programname='sumthing',
mailto='me@sumwear',
whendismissed="from wxPython.wx import * ; wxBell()")
FOR INTERNATIONAL [NON-ENGLISH-SPEAKING] USE:
wxPyNonFatalErrorDialog and relatives have the method
SetText(Number NUMBER, STRING)
where STRING is to displayed in the wxStaticText control with ID
wxPyError_ID_TEXT<NUMBER>--see the automatically-generated code
for information about the meaning of these...
"""
_debug = 0
#_debug = 1 # uncomment to display some information (to stdout)
Version = 1.3
from wxPython.wx import *
import string, sys, traceback, time, rexec, operator, types, cStringIO, os
#from wxPython.lib.createandsendHTMLmail import *# now inline
#import MimeWriter, mimetools, tempfile, smtplib
import urllib, webbrowser
from ErrorDialogs_wdr import *
# You may see from the above line that I used the excellent RAD tool
# wxDesigner, by Robert Roebling, to accelerate development of this
# module... The above is left for the convenience of making future
# changes with wxDesigner; also so the wxDesigner-generated codedoes
# not need to precede the "hand-generated" code in this file; finally,
# as a personal endorsement: it is truly a brilliant time-saver!
# Please note that, at the time of writing, the wxDesigner-generated
# output requires manual removal of the PythonBitmaps function--an
# appropriate version of this function will be imported from a
# similarly- named module. Another manual change will have to be made
# to the automatically-generated source: "parent.sizerAroundText = "
# should be added [immediately] before the text similar to "item13 =
# wxStaticBoxSizer( item14, wxVERTICAL )", this sizer being the one
# containing the wxTextCtrl... [IMPORTANT NOTE: THIS LINE SHOULD BE
# THE ONE INSTANTIATING A wxStat2icBoxSizer, *NOT* THE wxStaticBox
# ITSELF...] As of version 1.2 [November 2001], this also needs to be
# done for the {sizers around the} wxPyClickableHtmlWindow's generated in
# populate_wxPyNonFatalError and populate_wxPyFatalError--note that
# for ease this sizer is still called "sizerAroundText"...
def wxPyDestroyErrorDialogIfPresent():
if isinstance(sys.stderr,wxPyNonFatalErrorDialog):
sys.stderr.Destroy()
sys.stderr = None
def wxPyNewErrorDialog(dlg):
wxPyDestroyErrorDialogIfPresent()
sys.stderr = dlg
class wxPyNonWindowingErrorHandler:
this_exception = 0
softspace = 0
def __init__(self,fatal=0,file=sys.__stderr__):
self.fatal = fatal
self.file = file
def write(self,s):
import sys
if s.find("Warning") <> 0\
and self.this_exception is not sys.last_traceback:
wxPyNonWindowingError("The Python interpreter encountered an error "
"not handled by any\nexception handler--this "
"may represent some programming error.",
fatal=self.fatal,
stderr=self.file,
last=1)
self.this_exception = sys.last_traceback
def wxPyNonWindowingError(msg,#output=None,errors=None,
stderr=sys.__stderr__,
fatal=1,
last=None):
if os.path.exists("wxPyNonWindowingErrors.txt"):
mode = 'a+'
else:
mode = 'w'
fl = open("wxPyNonWindowingErrors.txt",mode)
if stderr is not None:
l = [fl,stderr] # so that the error will be written to the file
# before any potential error in stderr.write()... (this is largely
# for my own sake in developing...)
else:
l = [fl]
for f in l:
f.write(time.ctime (time.time ()) + ": ")
f.write(msg)
#f.flush()
if sys.exc_info () [0] is not None:
if last:
f.write('Currently handled exception:\n')
f.flush()
traceback.print_exc(file=f)
if last:
f.write('\nPrevious (?) error:\n')
elif last or sys.last_traceback:
f.write("\n\n(For wizards only) ")
if last:
if type(last) <> types.ListType or len(last) < 3:
if (hasattr(sys,"last_traceback") and sys.last_traceback):
last = [sys.last_type ,sys.last_value,sys.last_traceback]
if type(last) == types.ListType:
traceback.print_exception(last[0],last[1],last[2],
None,f)
#f.flush()
if f is sys.__stderr__:
s = ' (see the file "wxPyNonWindowingErrors.txt")'
else:
s = ""
f.write("Please contact the author with a copy of this\n"
"message%s.\n" % s)
#f.flush()
fl.close()
if fatal and stderr is sys.__stderr__:
if sys.__stderr__ and sys.platform in ["windows",'nt',"win32"]:
sys.__stderr__.write(
"\nYou may have to manually close this window to exit.")
sys.exit()
class wxPythonRExec (rexec.RExec):
def __init__(self,securityhole=0,*args,**kwargs):
apply(rexec.RExec.__init__, (self,) + args, kwargs)
if securityhole:
self.ok_builtin_modules = self.ok_builtin_modules + \
('wxPython', 'wxPython.wxc','wxPython.wx','wxPython.misc',
'wxPython.misc2', 'wxPython.windows', 'wxPython.gdi',
'wxPython.clip_dnd', 'wxPython.events', 'wxPython.mdi',
'wxPython.frames', 'wxPython.stattool', 'wxPython.controls',
'wxPython.controls2', 'wxPython.windows2', 'wxPython.cmndlgs',
'wxPython.windows3', 'wxPython.image', 'wxPython.printfw',
'wxc','misc', 'misc2', 'windows', 'gdi', 'clip_dnd', 'events',
'mdi', 'frames', 'stattool', 'controls', 'controls2', 'windows2',
'cmndlgs', 'windows3', 'image', 'printfw', 'wx')
# possible security hole!
##def wxPyFatalError(msg,frame=None,**kwargs):
## kwargs.update({'fatal' : 1})
## apply(wxPyNonFatalError,
## (frame,msg),
## kwargs)
class wxPyNonFatalErrorDialogWithTraceback(wxDialog):
this_exception = 0
populate_function = populate_wxPyNonFatalErrorDialogWithTraceback
no_continue_button = False
fatal = False
modal = True
exitjustreturns = False # really only for testing!
def __init__(self, parent, id,
pos=wxDefaultPosition,
size=wxDefaultSize,
style=wxDEFAULT_DIALOG_STYLE,
programname="Python program",
version="?",
mailto=None,
whendismissed="",
extraversioninformation="",
caption="Python error!",
versionname=None,
errorname=None,
disable_exit_button=False):
if self.fatal:
whetherNF = ""
else:
whetherNF = "non-"
title = "A (%sfatal) error has occurred in %s!"\
% (whetherNF,programname)
self.programname = programname # save for later use
self.mailto = mailto # save for later use
self.parent = parent # save for later use
self.whendismissed = whendismissed # save for later use
self.dialogtitle = title # save for later use
wxDialog.__init__(self, parent, id, title, pos, size, style)
self.topsizer = self.populate_function( False,True )
self.SetProgramName(programname)
self.SetVersion(version)
if errorname:
self.FindWindowById(wxPyError_ID_TEXT1).SetLabel(str(errorname))
if versionname:
self.FindWindowById(wxPyError_ID_TEXT2).SetLabel(str(versionname))
self.FindWindowById(wxPyError_ID_VERSIONNUMBER).SetLabel(str(version))
self.FindWindowById(wxPyError_ID_EXTRA_VERSION_INFORMATION).SetLabel(str(
extraversioninformation))
if caption:
self.SetTitle(caption)
if not self.no_continue_button:
EVT_BUTTON(self, wxPyError_ID_CONTINUE, self.OnContinue)
if mailto:
disable_mail_button = 0
else:
disable_mail_button = 1
if not disable_mail_button:
EVT_BUTTON(self, wxPyError_ID_MAIL, self.OnMail)
else:
self.GetMailButton().Enable(False)
# disable the entry box for an e-mail address by default (NOT PROPERLY DOCUMENTED)
if not hasattr(self,"enable_mail_address_box"):
self.FindWindowById(wxPyError_ID_ADDRESS).Enable(False)
if not disable_exit_button:
EVT_BUTTON(self, wxPyError_ID_EXIT, self.OnExit)
def GetExtraInformation(self):
return self.extraexceptioninformation
def SetExtraInformation(self,value):
self.extraexceptioninformation = value
c = self.GetExtraInformationCtrl()
if c is not None:
c.SetLabel(str(value))
self.topsizer.Layout()
def GetExtraInformationCtrl(self):
return self.FindWindowById(wxPyError_ID_EXTRAINFORMATION)
def GetExceptionName(self):
return str(self.exceptiontype)
def SetExceptionName(self,value):
self.exceptiontype = str(value)
c = self.GetExceptionNameCtrl()
if c is not None:
c.SetLabel(str(value))
self.topsizer.Layout()
def GetExceptionNameCtrl(self):
return self.FindWindowById(wxPyError_ID_EXCEPTIONNAME)
def GetTraceback(self):
try:
return self.traceback
except AttributeError:
return None
def SetTraceback(self,value):
self.traceback = value
c = self.GetTracebackCtrl()
if c is not None:
s,cs = c.GetSize(), c.GetClientSize()
if value[-1] == '\n':
value = value[:-1]
if _debug:
print "%s.SetTraceback(): ...SetValue('%s' (^M=\\r; ^J=\\n))"\
% (self,value.replace('\n',"^J"))
c.SetValue(value)
# Despite using the wxADJUST_MINSIZE flag in the
# appropriate AddWindow method of the sizer, this doesn't
# size the control appropriately... evidently the control's
# GetBestSize method is not returning the "correct"
# value... So we perform a rather ugly "fix"... note that
# this also requires that we remove the wxADJUST_MINSIZE
# flag from the AddWindow method of the sizer containing
# the wxTextCtrl, which adds the wxTextCtrl... (this
# amounts, as of wxDesigner 2.6, to only a few mouse
# clicks...)
if _debug:
size = c.GetBestSize()
print "%s.SetTraceback(): %s.GetBestSize() = (%s,%s)"\
% (self,c,size.width,size.height)
w,h = 0,0
for v in value.split("\n"):
pw,ph,d,e = t = c.GetFullTextExtent(v)
if _debug:
print v, t
h = h + ph + e# + d
pw = pw + wxSystemSettings_GetSystemMetric(wxSYS_VSCROLL_X)
if pw > w:
w = pw
w = w + s.width - cs.width
h = h + s.height - cs.height
if _debug:
print "%s.SetTraceback(): calculated w,h =" % c,\
w,h,"and sys.platform = '%s'" % sys.platform
self.sizerAroundText.SetItemMinSize (c,w,h)
c.SetSize ((w,h))
c.SetSizeHints (w,h,w,h)
c.Refresh()#.SetAutoLayout(False)
#^ the reason we need the above seems to be to replace the
#faulty GetBestSize of wxTextCtrl...
#self.sizerAroundText.Layout()
self.topsizer.Layout()
def GetTracebackCtrl(self):
return self.FindWindowById(wxPyError_ID_TEXTCTRL)
def GetVersion(self):
return self.version
def SetVersion(self,value):
self.version = value
c = self.GetVersionNumberCtrl()
if c is not None:
c.SetLabel(value)
self.topsizer.Layout()
def GetVersionNumberCtrl(self):
return self.FindWindowById(wxPyError_ID_VERSIONNUMBER)
def GetProgramName(self):
return self.programname
def SetProgramName(self,value):
self.programname = value
c = self.GetProgramNameCtrl()
if c is not None:
c.SetLabel(value)
self.topsizer.Layout()
def GetProgramNameCtrl(self):
return self.FindWindowById(wxPyError_ID_PROGRAMNAME)
def GetContinueButton(self):
return self.FindWindowById(wxPyError_ID_CONTINUE)
def GetMailButton(self):
return self.FindWindowById(wxPyError_ID_MAIL)
def GetExitButton(self):
return self.FindWindowById(wxPyError_ID_EXIT)
# write handler (which is really the guts of the thing...
# [Note that this doesn't use sys.excepthook because I already had a
# working body of code...
def write(self,s):
if self.this_exception is not sys.last_traceback:
if not wxThread_IsMain():
# Aquire the GUI mutex before making GUI calls. Mutex is released
# when locker is deleted at the end of this function.
locker = wxMutexGuiLocker()
self.this_exception = sys.last_traceback
# this is meant to be done once per traceback's sys.stderr.write's
# - on the first in fact.....
try:
#from wxPython.wx import wxBell
wxBell()
if _debug:
if sys.stdout: sys.stdout.write(
'in %s.write(): ' % self)
self.exceptiontype = sys.last_type
self.extraexceptioninformation = sys.last_value
c = cStringIO.StringIO()
traceback.print_last(None,c)
self.traceback = c.getvalue()
if _debug:
#import traceback
traceback.print_last(None,sys.stdout)
self.SetExceptionName(str(self.exceptiontype))
self.SetExtraInformation(str(self.extraexceptioninformation))
self.SetTraceback(str(self.traceback))
self.topsizer.Fit(self)
self.topsizer.SetSizeHints(self)
self.CentreOnScreen()
if self.modal:
self.ShowModal()
else:
self.Show(True)
except:
if not locals().has_key("c"):
c = cStringIO.StringIO()
c.write("[Exception occurred before data from "
"sys.last_traceback available]")
wxPyNonWindowingError("Warning: "
"a %s error was encountered trying to "
"handle the exception\n%s\nThis was:"#%s\n"
% (sys.exc_type, c.getvalue()),#, c2.getvalue()),
stderr=sys.stdout,
last=0)
# button handlers:
def OnContinue(self, event):
try:
if self.whendismissed:
parent = self.parent # so whendismissed can refer to "parent"
if 1:
if _debug:
if sys.stdout: sys.stdout.write("exec '''%s''': "
% (self.whendismissed))
exec self.whendismissed
if _debug: print "\n",
else:
if _debug:
if sys.stdout: sys.stdout.write("wxPythonRExec(%s).r_exec('''%s'''): "
% (self.securityhole,
self.whendismissed))
wxPythonRExec(self.securityhole).r_exec(self.whendismissed)
if _debug: print "\n",
if self.modal:
self.EndModal(wxID_OK)
else:
self.Close ()
if _debug: print "reimporting ",
for m in sys.modules.values():
if m and m.__dict__["__name__"][0] in string.uppercase:#hack!
if _debug:
print m.__dict__["__name__"],
reload (m)
if _debug:
print ' ',
if _debug:
print '\nENDING %s.OnContinue()..\n\n' % (self,),
except:
wxPyNonWindowingError("Warning: the following exception information"
" may not be the full story.. (because "
"a %s(%s) error was encountered trying to "
"handle the exception)\n\n"
% tuple(sys.exc_info()[:2]),
stderr=sys.stdout,
last=0)
PlainMessageTemplate = \
"Hello,\n\n"\
'%(programname)s'\
" error.\n\n"\
"I encountered the following error when running your "\
'program %(programname)s,'\
"at %(date)s.\n\n"\
"(The following has been automatically generated...)\n"\
'%(traceback)s\n\n'\
"More information follows:\n\n"\
'[Insert more '\
"information about the error here, such as what you were "\
"trying to do at the time of the error. Please "\
"understand that failure to fill in this field will be "\
"interpreted as an invitation to consign this e-mail "\
"straight to the trash can!]\n\n"\
'Yours sincerely,\n'\
"[insert your name here]\n"
HTMLMessageTemplate = \
'<html>\n'\
'<body>'\
"<p>\n"\
"<i><b>Hello,</b></i>\n<p>\n"\
'<p><h2><font color="#CC6C00">%(programname)s</font>'\
" error.</h2>\n"\
"I encountered the following error when running your "\
'program <font color="#CC6C00">%(programname)s</font>,'\
"at %(date)s.\n<p>\n"\
"<p>"\
"<h2>Traceback (automatically generated):</h2>\n"\
'<p><font size="-1">\n<pre>%(traceback)s</pre>\n<p></font><p>'\
"\n<p>\n<h2>More information follows:</h2>\n<p>\n"\
'<font color="#CC6C00">'\
'<i>[Insert more '\
"information about the error here, such as what you were "\
"trying to do at the time of the error. Please "\
"understand that failure to fill in this field will be "\
"interpreted as an invitation to consign this e-mail "\
"straight to the trash can!]\n</i><p>\n"\
"</font><p>\n"\
'<i><b>Yours sincerely,</b></i>\n<p>'\
'<font color="#CC6C00">'\
"[insert your name here]\n"\
"</font>\n"\
"</body>\n"\
"</html>\n"
# text="#000000" bgcolor="#FFFFFF">\n'\
def OnMail(self,event):
try:
if _debug:
print 'Attempting to write mail message.\n',
gmtdate = time.asctime(time.gmtime(time.time())) + ' GMT'
tm = time.localtime(time.time())
date = time.asctime(tm) + ' ' +\
time.strftime("%Z",tm)
programname = self.programname
traceback = self.traceback
mailto = self.mailto
subject = "Un-caught exception when running %s." % programname
mailfrom = None#self.FindWindowById (wxPyError_ID_ADDRESS)
if mailfrom:
mailfrom = mailfrom.GetValue()
if _startmailerwithhtml(mailto,subject,
self.HTMLMessageTemplate % vars(),
text=self.PlainMessageTemplate % vars(),
mailfrom=mailfrom):
if not (hasattr(self,"fatal") and self.fatal):
self.OnContinue(event) # if ok, then act as if "Continue" selected
except:
wxPyNonWindowingError("Warning: the following exception information"
" may not be the full story... (because "
"a %s error was encountered trying to "
"handle the original exception)\n\n"#%s"
% (sys.exc_type,),#self.msg),
stderr=sys.stdout,
last=0)
def OnExit(self, event):
if self.IsModal():
self.EndModal(wxID_CANCEL)
if self.exitjustreturns:
return
wxGetApp().ExitMainLoop()
def SetText(self,number,string):
self.FindWindowById(eval("wxPyError_ID_TEXT%d"
% number)).SetLabel(string)
self.topsizer.Layout()
class wxPyFatalErrorDialogWithTraceback(wxPyNonFatalErrorDialogWithTraceback):
populate_function = populate_wxPyFatalErrorDialogWithTraceback
no_continue_button = True
fatal = True
class wxPyNonFatalErrorDialog(wxPyNonFatalErrorDialogWithTraceback):
populate_function = populate_wxPyNonFatalErrorDialog
class wxPyFatalErrorDialog(wxPyFatalErrorDialogWithTraceback):
populate_function = populate_wxPyFatalErrorDialog
def _startmailerwithhtml(mailto,subject,html,text=None,mailfrom=None):
if sys.hexversion >= 0x02000000:#\
# and sys.platform in ["windows",'nt',"w is in32"]:
s = 'mailto:%s?subject=%s&body=%s' % (mailto,
urllib.quote(subject),
urllib.quote(
text.replace('\n','\r\n'),
""))
# Note that RFC 2368 requires that line breaks in the body of
# a message contained in a mailto URL MUST be encoded with
# "%0D%0A"--even on Unix/Linux. Also note that there appears
# to be no way to specify that the body of a mailto tag be
# interpreted as HTML (mailto tags shouldn't stuff around with
# the MIME-Version/Content-Type headers, the RFC says, so I
# can't even be bothered trying, as the Python
# webbrowser/urllib modules quite likely won't allow
# this... anyway, a plain text message is [at least!] fine...).
if _debug:
t = urllib.quote(text)
if len(t) > 20 * 80:
t = t[0:20 * 80] + "..."
print "\nSummarizing (only shortened version of argument "\
"printed here), ",
print 'webbrowser.open("' \
'mailto:%s?subject=%s&body=%s"' % (mailto,
urllib.quote(subject),
t)
webbrowser.open(s)
return 1
else:
return _writehtmlmessage(mailto,subject,html,text,mailfrom=mailfrom)
def _writehtmlmessage(mailto,subject,html,text=None,parent=None,mailfrom=None):
dlg = wxFileDialog (parent,
"Please choose a a file to save the message to...",
".",
"bug-report",
"HTML files (*.htm,*.html)|*.htm,*.html|"
"All files (*)|*",
wxSAVE | wxHIDE_READONLY)
if dlg.ShowModal() <> wxID_CANCEL:
f = open(dlg.GetPath(),"w")
dlg.Destroy()
f.write(_createhtmlmail(html,text,subject,to=mailto,mailfrom=mailfrom))
f.close()
return 1
else:
return 0
# PLEASE NOTE THAT THE CODE BELOW FOR WRITING A GIVEN
#(HTML) MESSAGE IS BY ART GILLESPIE [with slight modifications by yours truly].
def _createhtmlmail (html, text, subject, to=None, mailfrom=None):
"""Create a mime-message that will render HTML in popular
MUAs, text in better ones (if indeed text is not unTrue (e.g. None)
"""
import MimeWriter, mimetools, cStringIO
out = cStringIO.StringIO() # output buffer for our message
htmlin = cStringIO.StringIO(html)
if text:
txtin = cStringIO.StringIO(text)
writer = MimeWriter.MimeWriter(out)
#
# set up some basic headers... we put subject here
# because smtplib.sendmail expects it to be in the
# message body
#
if mailfrom:
writer.addheader("From", mailfrom)
#writer.addheader("Reply-to", mailfrom)
writer.addheader("Subject", subject)
if to:
writer.addheader("To", to)
writer.addheader("MIME-Version", "1.0")
#
# start the multipart section of the message
# multipart/alternative seems to work better
# on some MUAs than multipart/mixed
#
writer.startmultipartbody("alternative")
writer.flushheaders()
#
# the plain text section
#
if text:
subpart = writer.nextpart()
subpart.addheader("Content-Transfer-Encoding", "quoted-printable")
pout = subpart.startbody("text/plain", [("charset", 'us-ascii')])
mimetools.encode(txtin, pout, 'quoted-printable')
txtin.close()
#
# start the html subpart of the message
#
subpart = writer.nextpart()
subpart.addheader("Content-Transfer-Encoding", "quoted-printable")
pout = subpart.startbody("text/html", [("charset", 'us-ascii')])
mimetools.encode(htmlin, pout, 'quoted-printable')
htmlin.close()
#
# Now that we're done, close our writer and
# return the message body
#
writer.lastpart()
msg = out.getvalue()
out.close()
return msg
def _sendmail(mailto,subject,html,text):# currently unused
"""For illustration only--this function is not actually used."""
import smtplib
message = _createhtmlmail(html, text, subject)
server = smtplib.SMTP("localhost")
server.sendmail(mailto, subject, message)
server.quit()
def wxPyFatalError(parent,msg,**kw):
return wxPyFatalOrNonFatalError(parent,msg,fatal=1,**kw)
def wxPyNonFatalError(parent,msg,**kw):
return wxPyFatalOrNonFatalError(parent,msg,fatal=0,**kw)
def wxPyResizeHTMLWindowToDispelScrollbar(window,
fraction,
sizer=None,
defaultfraction=0.7):
# Try to `grow' parent window (typically a dialog), only so far as
# no scrollbar is necessary, mantaining aspect ratio of display.
# Will go no further than specified fraction of display size.
w = 200
if type(fraction) == type(''):
fraction = int(fraction[:-1]) / 100.
ds = wxDisplaySize ()
c = window.GetInternalRepresentation ()
while w < ds[0] * fraction:
c.Layout(w)
if _debug:
print '(c.GetHeight() + 20, w * ds[1]/ds[0]):',\
(c.GetHeight() + 20, w * ds[1]/ds[0])
if c.GetHeight() + 20 < w * ds[1]/ds[0]:
size = (w,min(int ((w) * ds[1]/ds[0]),
c.GetHeight()))
break
w = w + 20
else:
if type(defaultfraction) == type(''):
defaultfraction = int(defaultfraction[:-1]) / 100.
defaultsize = (defaultfraction * ds[0], defaultfraction * ds[1])
if _debug:
print 'defaultsize =',defaultsize
size = defaultsize
window.SetSize(size)
if sizer is not None:
sizer.SetMinSize(size)
sizer.Fit(window)
#sizer.SetSizeHints(window)
def wxPyFatalOrNonFatalError(parent,
msg,
fatal=0,
extraversioninformation="",
caption=None,
versionname=None,
errorname=None,
version="?",
programname="Python program",
tback=None,# backwards compatibility, and for
#possible future inclusion of ability to display
#a traceback along with the given message
#"msg"... currently ignored though...
modal=0):
if not wxThread_IsMain():
# Aquire the GUI mutex before making GUI calls. Mutex is released
# when locker is deleted at the end of this function.
locker = wxMutexGuiLocker()
dlg = wxDialog(parent,-1,"Error!")
if fatal:
populate_function = populate_wxPyFatalError
else:
populate_function = populate_wxPyNonFatalError
sizer = populate_function(dlg,False,True)
window = dlg.FindWindowById(wxPyError_ID_HTML)
window.SetPage(msg)
wxPyResizeHTMLWindowToDispelScrollbar(window,
"85%",
sizer=dlg.sizerAroundText)
dlg.FindWindowById(wxPyError_ID_PROGRAMNAME).SetLabel(str(programname))
if errorname:
dlg.FindWindowById(wxPyError_ID_TEXT1).SetLabel(str(errorname))
if versionname:
dlg.FindWindowById(wxPyError_ID_TEXT2).SetLabel(str(versionname))
dlg.FindWindowById(wxPyError_ID_VERSIONNUMBER).SetLabel(str(version))
dlg.FindWindowById(wxPyError_ID_EXTRA_VERSION_INFORMATION).SetLabel(str(
extraversioninformation))
if caption:
dlg.SetTitle(caption)
sizer.Fit(dlg)
sizer.SetSizeHints(dlg)
dlg.CentreOnScreen()
if modal:
v = dlg.ShowModal()
dlg.Destroy()
return v
else:
dlg.Show(True)