1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     Helper functions/classes for the wxPython extension module 
   7 // Created:     1-July-1997 
   9 // Copyright:   (c) 1998 by Total Control Software 
  10 // Licence:     wxWindows license 
  11 ///////////////////////////////////////////////////////////////////////////// 
  16 #include "wx/wxPython/wxPython_int.h" 
  17 #include "wx/wxPython/pyistream.h" 
  20 #include <wx/msw/private.h> 
  21 #include <wx/msw/winundef.h> 
  22 #include <wx/msw/msvcrt.h> 
  27 #include <gdk/gdkprivate.h> 
  28 #include <wx/gtk/win_gtk.h> 
  32 #include <wx/mac/private.h> 
  35 #include <wx/clipbrd.h> 
  36 #include <wx/mimetype.h> 
  39 //---------------------------------------------------------------------- 
  41 #if PYTHON_API_VERSION <= 1007 && wxUSE_UNICODE 
  42 #error Python must support Unicode to use wxWindows Unicode 
  45 //---------------------------------------------------------------------- 
  47 wxPyApp
* wxPythonApp 
= NULL
;  // Global instance of application object 
  48 bool wxPyDoCleanup 
= False
; 
  49 bool wxPyDoingCleanup 
= False
; 
  52 #ifdef WXP_WITH_THREAD 
  53 struct wxPyThreadState 
{ 
  55     PyThreadState
* tstate
; 
  57     wxPyThreadState(unsigned long _tid
=0, PyThreadState
* _tstate
=NULL
) 
  58         : tid(_tid
), tstate(_tstate
) {} 
  61 #include <wx/dynarray.h> 
  62 WX_DECLARE_OBJARRAY(wxPyThreadState
, wxPyThreadStateArray
); 
  63 #include <wx/arrimpl.cpp> 
  64 WX_DEFINE_OBJARRAY(wxPyThreadStateArray
); 
  66 wxPyThreadStateArray
* wxPyTStates 
= NULL
; 
  67 wxMutex
*              wxPyTMutex 
= NULL
; 
  71 static PyObject
* wxPython_dict 
= NULL
; 
  72 static PyObject
* wxPyAssertionError 
= NULL
; 
  74 PyObject
* wxPyPtrTypeMap 
= NULL
; 
  77 #ifdef __WXMSW__             // If building for win32... 
  78 //---------------------------------------------------------------------- 
  79 // This gets run when the DLL is loaded.  We just need to save a handle. 
  80 //---------------------------------------------------------------------- 
  83     HINSTANCE   hinstDLL
,    // handle to DLL module 
  84     DWORD       fdwReason
,   // reason for calling function 
  85     LPVOID      lpvReserved  
// reserved 
  88     // If wxPython is embedded in another wxWindows app then 
  89     // the inatance has already been set. 
  90     if (! wxGetInstance()) 
  91         wxSetInstance(hinstDLL
); 
  96 //---------------------------------------------------------------------- 
  97 // Classes for implementing the wxp main application shell. 
  98 //---------------------------------------------------------------------- 
 100 IMPLEMENT_ABSTRACT_CLASS(wxPyApp
, wxApp
); 
 104     m_assertMode 
= wxPYAPP_ASSERT_EXCEPTION
; 
 105     m_startupComplete 
= False
; 
 109 wxPyApp::~wxPyApp() { 
 113 // This one isn't acutally called...  We fake it with _BootstrapApp 
 114 bool wxPyApp::OnInit() { 
 119 int  wxPyApp::MainLoop() { 
 122     DeletePendingObjects(); 
 123     bool initialized 
= wxTopLevelWindows
.GetCount() != 0; 
 125         if ( m_exitOnFrameDelete 
== Later 
) { 
 126             m_exitOnFrameDelete 
= Yes
; 
 129         retval 
= wxApp::MainLoop(); 
 136 bool wxPyApp::OnInitGui() { 
 138     wxApp::OnInitGui();  // in this case always call the base class version 
 139     bool blocked 
= wxPyBeginBlockThreads(); 
 140     if (wxPyCBH_findCallback(m_myInst
, "OnInitGui")) 
 141         rval 
= wxPyCBH_callCallback(m_myInst
, Py_BuildValue("()")); 
 142     wxPyEndBlockThreads(blocked
); 
 147 int wxPyApp::OnExit() { 
 149     bool blocked 
= wxPyBeginBlockThreads(); 
 150     if (wxPyCBH_findCallback(m_myInst
, "OnExit")) 
 151         rval 
= wxPyCBH_callCallback(m_myInst
, Py_BuildValue("()")); 
 152     wxPyEndBlockThreads(blocked
); 
 153     wxApp::OnExit();  // in this case always call the base class version 
 159 void wxPyApp::OnAssert(const wxChar 
*file
, 
 164     // if we're not fully initialized then just log the error 
 165     if (! m_startupComplete
) { 
 168         buf
.Printf(wxT("%s(%d): assert \"%s\" failed"), 
 178     // If the OnAssert is overloaded in the Python class then call it... 
 180     bool blocked 
= wxPyBeginBlockThreads(); 
 181     if ((found 
= wxPyCBH_findCallback(m_myInst
, "OnAssert"))) { 
 182         PyObject
* fso 
= wx2PyString(file
); 
 183         PyObject
* cso 
= wx2PyString(file
); 
 186             mso 
= wx2PyString(file
); 
 188             mso 
= Py_None
; Py_INCREF(Py_None
); 
 190         wxPyCBH_callCallback(m_myInst
, Py_BuildValue("(OiOO)", fso
, line
, cso
, mso
)); 
 195     wxPyEndBlockThreads(blocked
); 
 197     // ...otherwise do our own thing with it 
 200         if (m_assertMode 
& wxPYAPP_ASSERT_SUPPRESS
) 
 203         // turn it into a Python exception? 
 204         if (m_assertMode 
& wxPYAPP_ASSERT_EXCEPTION
) { 
 207             buf
.Printf(wxT("C++ assertion \"%s\" failed in %s(%d)"), cond
, file
, line
); 
 214             bool blocked 
= wxPyBeginBlockThreads(); 
 215             PyObject
* s 
= wx2PyString(buf
); 
 216             PyErr_SetObject(wxPyAssertionError
, s
); 
 218             wxPyEndBlockThreads(blocked
); 
 220             // Now when control returns to whatever API wrapper was called from 
 221             // Python it should detect that an exception is set and will return 
 222             // NULL, signalling the exception to Python. 
 225         // Send it to the normal log destination, but only if 
 226         // not _DIALOG because it will call this too 
 227         if ( (m_assertMode 
& wxPYAPP_ASSERT_LOG
) && !(m_assertMode 
& wxPYAPP_ASSERT_DIALOG
)) { 
 230             buf
.Printf(wxT("%s(%d): assert \"%s\" failed"), 
 239         // do the normal wx assert dialog? 
 240         if (m_assertMode 
& wxPYAPP_ASSERT_DIALOG
) 
 241             wxApp::OnAssert(file
, line
, cond
, msg
); 
 246     // For catching Apple Events 
 247 void wxPyApp::MacOpenFile(const wxString 
&fileName
) 
 249     bool blocked 
= wxPyBeginBlockThreads(); 
 250     if (wxPyCBH_findCallback(m_myInst
, "MacOpenFile")) { 
 251         PyObject
* s 
= wx2PyString(fileName
); 
 252         wxPyCBH_callCallback(m_myInst
, Py_BuildValue("(O)", s
)); 
 255     wxPyEndBlockThreads(blocked
); 
 258 void wxPyApp::MacPrintFile(const wxString 
&fileName
) 
 260     bool blocked 
= wxPyBeginBlockThreads(); 
 261     if (wxPyCBH_findCallback(m_myInst
, "MacPrintFile")) { 
 262         PyObject
* s 
= wx2PyString(fileName
); 
 263         wxPyCBH_callCallback(m_myInst
, Py_BuildValue("(O)", s
)); 
 266     wxPyEndBlockThreads(blocked
); 
 269 void wxPyApp::MacNewFile() 
 271     bool blocked 
= wxPyBeginBlockThreads(); 
 272     if (wxPyCBH_findCallback(m_myInst
, "MacNewFile")) 
 273         wxPyCBH_callCallback(m_myInst
, Py_BuildValue("()")); 
 274     wxPyEndBlockThreads(blocked
); 
 277 void wxPyApp::MacReopenApp() 
 279     bool blocked 
= wxPyBeginBlockThreads(); 
 280     if (wxPyCBH_findCallback(m_myInst
, "MacReopenApp")) 
 281         wxPyCBH_callCallback(m_myInst
, Py_BuildValue("()")); 
 282     wxPyEndBlockThreads(blocked
); 
 287 bool wxPyApp::GetMacSupportPCMenuShortcuts() { 
 289     return s_macSupportPCMenuShortcuts
; 
 296 long wxPyApp::GetMacAboutMenuItemId() { 
 298     return s_macAboutMenuItemId
; 
 305 long wxPyApp::GetMacPreferencesMenuItemId() { 
 307     return s_macPreferencesMenuItemId
; 
 314 long wxPyApp::GetMacExitMenuItemId() { 
 316     return s_macExitMenuItemId
; 
 323 wxString 
wxPyApp::GetMacHelpMenuTitleName() { 
 325     return s_macHelpMenuTitleName
; 
 327     return wxEmptyString
; 
 332 void wxPyApp::SetMacSupportPCMenuShortcuts(bool val
) { 
 334     s_macSupportPCMenuShortcuts 
= val
; 
 339 void wxPyApp::SetMacAboutMenuItemId(long val
) { 
 341     s_macAboutMenuItemId 
= val
; 
 346 void wxPyApp::SetMacPreferencesMenuItemId(long val
) { 
 348     s_macPreferencesMenuItemId 
= val
; 
 353 void wxPyApp::SetMacExitMenuItemId(long val
) { 
 355     s_macExitMenuItemId 
= val
; 
 360 void wxPyApp::SetMacHelpMenuTitleName(const wxString
& val
) { 
 362     s_macHelpMenuTitleName 
= val
; 
 367 // This finishes the initialization of wxWindows and then calls the OnInit 
 368 // that should be present in the derived (Python) class. 
 369 void wxPyApp::_BootstrapApp() 
 372     PyObject
*   retval 
= NULL
; 
 373     PyObject
*   pyint  
= NULL
; 
 376     // Get any command-line args passed to this program from the sys module 
 379     bool blocked 
= wxPyBeginBlockThreads(); 
 380     PyObject
* sysargv 
= PySys_GetObject("argv"); 
 381     if (sysargv 
!= NULL
) { 
 382         argc 
= PyList_Size(sysargv
); 
 383         argv 
= new char*[argc
+1]; 
 385         for(x
=0; x
<argc
; x
++) { 
 386             PyObject 
*pyArg 
= PyList_GetItem(sysargv
, x
); 
 387             argv
[x
] = PyString_AsString(pyArg
); 
 391     wxPyEndBlockThreads(blocked
); 
 393     result 
= wxEntryStart(argc
, argv
); 
 396     blocked 
= wxPyBeginBlockThreads(); 
 398         PyErr_SetString(PyExc_SystemError
, "wxEntryStart failed!"); 
 402     // The stock objects were all NULL when they were loaded into 
 403     // SWIG generated proxies, so re-init those now... 
 404     wxPy_ReinitStockObjects(3); 
 406     // It's now ok to generate exceptions for assertion errors. 
 407     wxPythonApp
->SetStartupComplete(True
); 
 409     // Call the Python wxApp's OnInit function 
 410     if (wxPyCBH_findCallback(m_myInst
, "OnInit")) { 
 412         PyObject
* method 
= m_myInst
.GetLastFound(); 
 413         PyObject
* argTuple 
= PyTuple_New(0); 
 414         retval 
= PyEval_CallObject(method
, argTuple
); 
 420         pyint 
= PyNumber_Int(retval
); 
 422             PyErr_SetString(PyExc_TypeError
, "OnInit should return a boolean value"); 
 425         result 
= PyInt_AS_LONG(pyint
); 
 428         // Is it okay if there is no OnInit?  Probably so... 
 434         PyErr_SetString(PyExc_SystemExit
, "OnInit returned False, exiting..."); 
 441     wxPyEndBlockThreads(blocked
); 
 444 //--------------------------------------------------------------------- 
 445 //---------------------------------------------------------------------- 
 449 static char* wxPyCopyCString(const wxChar
* src
) 
 451     wxWX2MBbuf buff 
= (wxWX2MBbuf
)wxConvCurrent
->cWX2MB(src
); 
 452     size_t len 
= strlen(buff
); 
 453     char*  dest 
= new char[len
+1]; 
 459 static char* wxPyCopyCString(const char* src
)   // we need a char version too 
 461     size_t len 
= strlen(src
); 
 462     char*  dest 
= new char[len
+1]; 
 468 static wxChar
* wxPyCopyWString(const char *src
) 
 470     //wxMB2WXbuf buff = wxConvCurrent->cMB2WX(src); 
 471     wxString 
str(src
, *wxConvCurrent
); 
 472     return copystring(str
); 
 476 static wxChar
* wxPyCopyWString(const wxChar 
*src
) 
 478     return copystring(src
); 
 484 inline const char* dropwx(const char* name
) { 
 485     if (name
[0] == 'w' && name
[1] == 'x') 
 491 //---------------------------------------------------------------------- 
 493 // This function is called when the wxc module is imported to do some initial 
 494 // setup.  (Before there is a wxApp object.)  The rest happens in 
 495 // wxPyApp::_BootstrapApp 
 496 void __wxPyPreStart(PyObject
* moduleDict
) 
 500 //    wxCrtSetDbgFlag(_CRTDBG_LEAK_CHECK_DF); 
 503 #ifdef WXP_WITH_THREAD 
 504     PyEval_InitThreads(); 
 505     wxPyTStates 
= new wxPyThreadStateArray
; 
 506     wxPyTMutex 
= new wxMutex
; 
 508     // Save the current (main) thread state in our array 
 509     PyThreadState
* tstate 
= wxPyBeginAllowThreads(); 
 510     wxPyEndAllowThreads(tstate
); 
 513     // Ensure that the build options in the DLL (or whatever) match this build 
 514     wxApp::CheckBuildOptions(WX_BUILD_OPTIONS_SIGNATURE
, "wxPython"); 
 516     // Init the stock objects to a non-NULL value so SWIG doesn't create them as None 
 517     wxPy_ReinitStockObjects(1); 
 522 void __wxPyCleanup() { 
 523     wxPyDoingCleanup 
= True
; 
 525         wxPyDoCleanup 
= False
; 
 528 #ifdef WXP_WITH_THREAD 
 531     wxPyTStates
->Empty(); 
 538 // Save a reference to the dictionary of the wx.core module, and inject 
 539 // a few more things into it. 
 540 PyObject
* __wxPySetDictionary(PyObject
* /* self */, PyObject
* args
) 
 543     if (!PyArg_ParseTuple(args
, "O", &wxPython_dict
)) 
 546     if (!PyDict_Check(wxPython_dict
)) { 
 547         PyErr_SetString(PyExc_TypeError
, "_wxPySetDictionary must have dictionary object!"); 
 551     if (! wxPyPtrTypeMap
) 
 552         wxPyPtrTypeMap 
= PyDict_New(); 
 553     PyDict_SetItemString(wxPython_dict
, "__wxPyPtrTypeMap", wxPyPtrTypeMap
); 
 555     // Create an exception object to use for wxASSERTions 
 556     wxPyAssertionError 
= PyErr_NewException("wx.core.PyAssertionError", 
 557                                             PyExc_AssertionError
, NULL
); 
 558     PyDict_SetItemString(wxPython_dict
, "PyAssertionError", wxPyAssertionError
); 
 562 #define wxPlatform "__WXMOTIF__" 
 563 #define wxPlatName "wxMotif" 
 566 #define wxPlatform "__WXX11__" 
 567 #define wxPlatName "wxX11" 
 570 #define wxPlatform "__WXGTK__" 
 571 #define wxPlatName "wxGTK" 
 574 #define wxPlatform "__WXMSW__" 
 575 #define wxPlatName "wxMSW" 
 578 #define wxPlatform "__WXMAC__" 
 579 #define wxPlatName "wxMac" 
 588     // These should be deprecated in favor of the PlatformInfo tuple built below... 
 589     PyDict_SetItemString(wxPython_dict
, "Platform", PyString_FromString(wxPlatform
)); 
 590     PyDict_SetItemString(wxPython_dict
, "USE_UNICODE", PyInt_FromLong(wxUSE_UNICODE
)); 
 591     PyDict_SetItemString(wxPython_dict
, "__WXDEBUG__", PyInt_FromLong(wxdebug
)); 
 594     PyObject
* PlatInfo 
= PyList_New(0); 
 597 #define _AddInfoString(st) \ 
 598     obj = PyString_FromString(st); \ 
 599     PyList_Append(PlatInfo, obj); \ 
 602     _AddInfoString(wxPlatform
); 
 603     _AddInfoString(wxPlatName
); 
 605     _AddInfoString("unicode"); 
 607     _AddInfoString("ascii"); 
 611     _AddInfoString("gtk2"); 
 613     _AddInfoString("gtk1"); 
 617 #undef _AddInfoString 
 619     PyObject
* PlatInfoTuple 
= PyList_AsTuple(PlatInfo
); 
 621     PyDict_SetItemString(wxPython_dict
, "PlatformInfo", PlatInfoTuple
); 
 627 //--------------------------------------------------------------------------- 
 629 // Python's PyInstance_Check does not return True for instances of new-style 
 630 // classes.  This should get close enough for both new and old classes but I 
 631 // should re-evaluate the need for doing instance checks... 
 632 bool wxPyInstance_Check(PyObject
* obj
) { 
 633     return PyObject_HasAttrString(obj
, "__class__") != 0; 
 637 // This one checks if the object is an instance of a SWIG proxy class (it has 
 638 // a .this attribute) 
 639 bool wxPySwigInstance_Check(PyObject
* obj
) { 
 640     return PyObject_HasAttrString(obj
, "this") != 0; 
 643 //--------------------------------------------------------------------------- 
 645 // The stock objects are no longer created when the wxc module is imported, 
 646 // but only after the app object has been created.  The 
 647 // wxPy_ReinitStockObjects function will be called 3 times to pass the stock 
 648 // objects though various stages of evolution: 
 650 //   pass 1: Set all the pointers to a non-NULL value so the Python proxy 
 651 //           object will be created (otherwise it will just use None.) 
 653 //   pass 2: After the module has been imported and the python proxys have 
 654 //           been created, then set the __class__ to be _wxPyUnbornObject so 
 655 //           it will catch any access to the object and will raise an exception. 
 657 //   pass 3: Finally, from OnInit patch things up so the stock objects can 
 661 PyObject
* __wxPyFixStockObjects(PyObject
* /* self */, PyObject
* args
) 
 663     wxPy_ReinitStockObjects(2); 
 668 static void rsoPass2(const char* name
) 
 670     static PyObject
* unbornObjectClass 
= NULL
; 
 673     if (unbornObjectClass 
== NULL
) { 
 674         unbornObjectClass 
= PyDict_GetItemString(wxPython_dict
, "_wxPyUnbornObject"); 
 675         Py_INCREF(unbornObjectClass
); 
 678     // Find the object instance 
 679     obj 
= PyDict_GetItemString(wxPython_dict
, (char*)dropwx(name
)); 
 680     wxCHECK_RET(obj 
!= NULL
, wxT("Unable to find stock object")); 
 681     wxCHECK_RET(wxPySwigInstance_Check(obj
), wxT("Not a swig instance")); 
 684     PyObject_SetAttrString(obj
, "__class__",  unbornObjectClass
); 
 688 static void rsoPass3(const char* name
, const char* classname
, void* ptr
) 
 694     // Find the object instance 
 695     obj 
= PyDict_GetItemString(wxPython_dict
, (char*)dropwx(name
)); 
 696     wxCHECK_RET(obj 
!= NULL
, wxT("Unable to find stock object")); 
 697     wxCHECK_RET(wxPySwigInstance_Check(obj
), wxT("Not a swig instance")); 
 699     // Find the class object and put it back in the instance 
 700     classobj 
= PyDict_GetItemString(wxPython_dict
, (char*)dropwx(classname
)); 
 701     wxCHECK_RET(classobj 
!= NULL
, wxT("Unable to find stock class object")); 
 702     PyObject_SetAttrString(obj
, "__class__",  classobj
); 
 704     // Rebuild the .this swigified pointer with the new value of the C++ pointer 
 705     ptrobj 
= wxPyMakeSwigPtr(ptr
, wxString(classname
, *wxConvCurrent
)); 
 706     PyObject_SetAttrString(obj
, "this", ptrobj
); 
 712 void wxPy_ReinitStockObjects(int pass
) 
 715 #define REINITOBJ(name, classname) \ 
 716     if (pass == 1) { name = (classname*)0xC0C0C0C0; } \ 
 717     else if (pass == 2) { rsoPass2(#name); } \ 
 718     else if (pass == 3) { rsoPass3(#name, #classname, (void*)name); } 
 721 #define REINITOBJ2(name, classname) \ 
 723     else if (pass == 2) { rsoPass2(#name); } \ 
 724     else if (pass == 3) { rsoPass3(#name, #classname, (void*)&name); } 
 726     // If there is already an App object then wxPython is probably embedded in 
 727     // a wx C++ application, so there is no need to do all this. 
 728     static bool embedded 
= false; 
 729     if ((pass 
== 1 || pass 
== 2) && wxTheApp
) { 
 733     if (pass 
== 3 && embedded
) 
 737     REINITOBJ(wxNORMAL_FONT
, wxFont
); 
 738     REINITOBJ(wxSMALL_FONT
, wxFont
); 
 739     REINITOBJ(wxITALIC_FONT
, wxFont
); 
 740     REINITOBJ(wxSWISS_FONT
, wxFont
); 
 742     REINITOBJ(wxRED_PEN
, wxPen
); 
 743     REINITOBJ(wxCYAN_PEN
, wxPen
); 
 744     REINITOBJ(wxGREEN_PEN
, wxPen
); 
 745     REINITOBJ(wxBLACK_PEN
, wxPen
); 
 746     REINITOBJ(wxWHITE_PEN
, wxPen
); 
 747     REINITOBJ(wxTRANSPARENT_PEN
, wxPen
); 
 748     REINITOBJ(wxBLACK_DASHED_PEN
, wxPen
); 
 749     REINITOBJ(wxGREY_PEN
, wxPen
); 
 750     REINITOBJ(wxMEDIUM_GREY_PEN
, wxPen
); 
 751     REINITOBJ(wxLIGHT_GREY_PEN
, wxPen
); 
 753     REINITOBJ(wxBLUE_BRUSH
, wxBrush
); 
 754     REINITOBJ(wxGREEN_BRUSH
, wxBrush
); 
 755     REINITOBJ(wxWHITE_BRUSH
, wxBrush
); 
 756     REINITOBJ(wxBLACK_BRUSH
, wxBrush
); 
 757     REINITOBJ(wxTRANSPARENT_BRUSH
, wxBrush
); 
 758     REINITOBJ(wxCYAN_BRUSH
, wxBrush
); 
 759     REINITOBJ(wxRED_BRUSH
, wxBrush
); 
 760     REINITOBJ(wxGREY_BRUSH
, wxBrush
); 
 761     REINITOBJ(wxMEDIUM_GREY_BRUSH
, wxBrush
); 
 762     REINITOBJ(wxLIGHT_GREY_BRUSH
, wxBrush
); 
 764     REINITOBJ(wxBLACK
, wxColour
); 
 765     REINITOBJ(wxWHITE
, wxColour
); 
 766     REINITOBJ(wxRED
, wxColour
); 
 767     REINITOBJ(wxBLUE
, wxColour
); 
 768     REINITOBJ(wxGREEN
, wxColour
); 
 769     REINITOBJ(wxCYAN
, wxColour
); 
 770     REINITOBJ(wxLIGHT_GREY
, wxColour
); 
 772     REINITOBJ(wxSTANDARD_CURSOR
, wxCursor
); 
 773     REINITOBJ(wxHOURGLASS_CURSOR
, wxCursor
); 
 774     REINITOBJ(wxCROSS_CURSOR
, wxCursor
); 
 776     REINITOBJ2(wxNullBitmap
, wxBitmap
); 
 777     REINITOBJ2(wxNullIcon
, wxIcon
); 
 778     REINITOBJ2(wxNullCursor
, wxCursor
); 
 779     REINITOBJ2(wxNullPen
, wxPen
); 
 780     REINITOBJ2(wxNullBrush
, wxBrush
); 
 781     REINITOBJ2(wxNullPalette
, wxPalette
); 
 782     REINITOBJ2(wxNullFont
, wxFont
); 
 783     REINITOBJ2(wxNullColour
, wxColour
); 
 785     REINITOBJ(wxTheFontList
, wxFontList
); 
 786     REINITOBJ(wxThePenList
, wxPenList
); 
 787     REINITOBJ(wxTheBrushList
, wxBrushList
); 
 788     REINITOBJ(wxTheColourDatabase
, wxColourDatabase
); 
 791     REINITOBJ(wxTheClipboard
, wxClipboard
); 
 792     REINITOBJ2(wxDefaultValidator
, wxValidator
); 
 793     REINITOBJ2(wxNullImage
, wxImage
); 
 794     REINITOBJ2(wxNullAcceleratorTable
, wxAcceleratorTable
); 
 800 //--------------------------------------------------------------------------- 
 802 void wxPyClientData_dtor(wxPyClientData
* self
) { 
 803     if (! wxPyDoingCleanup
) {           // Don't do it during cleanup as Python 
 804                                         // may have already garbage collected the object... 
 805         bool blocked 
= wxPyBeginBlockThreads(); 
 806         Py_DECREF(self
->m_obj
); 
 808         wxPyEndBlockThreads(blocked
); 
 812 void wxPyUserData_dtor(wxPyUserData
* self
) { 
 813     if (! wxPyDoingCleanup
) { 
 814         bool blocked 
= wxPyBeginBlockThreads(); 
 815         Py_DECREF(self
->m_obj
); 
 817         wxPyEndBlockThreads(blocked
); 
 822 // This is called when an OOR controled object is being destroyed.  Although 
 823 // the C++ object is going away there is no way to force the Python object 
 824 // (and all references to it) to die too.  This causes problems (crashes) in 
 825 // wxPython when a python shadow object attempts to call a C++ method using 
 826 // the now bogus pointer... So to try and prevent this we'll do a little black 
 827 // magic and change the class of the python instance to a class that will 
 828 // raise an exception for any attempt to call methods with it.  See 
 829 // _wxPyDeadObject in _core_ex.py for the implementation of this class. 
 830 void wxPyOORClientData_dtor(wxPyOORClientData
* self
) { 
 832     static PyObject
* deadObjectClass 
= NULL
; 
 834     bool blocked 
= wxPyBeginBlockThreads(); 
 835     if (deadObjectClass 
== NULL
) { 
 836         deadObjectClass 
= PyDict_GetItemString(wxPython_dict
, "_wxPyDeadObject"); 
 837         // TODO:  Can not wxASSERT here because inside a wxPyBeginBlock Threads, 
 838         // will lead to a deadlock when it tries to aquire the GIL again. 
 839         //wxASSERT_MSG(deadObjectClass != NULL, wxT("Can't get _wxPyDeadObject class!")); 
 840         Py_INCREF(deadObjectClass
); 
 844     // Only if there is more than one reference to the object 
 845     if ( !wxPyDoingCleanup 
&& self
->m_obj
->ob_refcnt 
> 1 ) { 
 846         // bool isInstance = wxPyInstance_Check(self->m_obj); 
 848         //wxASSERT_MSG(isInstance, wxT("m_obj not an instance!?!?!")); 
 850         // Call __del__, if there is one. 
 851         PyObject
* func 
= PyObject_GetAttrString(self
->m_obj
, "__del__"); 
 853             PyObject
* rv 
= PyObject_CallMethod(self
->m_obj
, "__del__", NULL
); 
 857         if (PyErr_Occurred()) 
 858             PyErr_Clear();      // just ignore it for now 
 861         PyObject
* dict 
= PyObject_GetAttrString(self
->m_obj
, "__dict__"); 
 863             // Clear the instance's dictionary 
 866             // put the name of the old class into the instance, and then reset the 
 867             // class to be the dead class. 
 868             PyObject
* klass 
= PyObject_GetAttrString(self
->m_obj
, "__class__"); 
 869             PyObject
* name 
=  PyObject_GetAttrString(klass
, "__name__"); 
 870             PyDict_SetItemString(dict
, "_name", name
); 
 871             PyObject_SetAttrString(self
->m_obj
, "__class__",  deadObjectClass
); 
 872             //Py_INCREF(deadObjectClass); 
 878     // m_obj is DECREF'd in the base class dtor... 
 879     wxPyEndBlockThreads(blocked
); 
 883 //--------------------------------------------------------------------------- 
 884 // Stuff used by OOR to find the right wxPython class type to return and to 
 888 // The pointer type map is used when the "pointer" type name generated by SWIG 
 889 // is not the same as the shadow class name, for example wxPyTreeCtrl 
 890 // vs. wxTreeCtrl.  It needs to be referenced in Python as well as from C++, 
 891 // so we'll just make it a Python dictionary in the wx module's namespace. 
 892 // (See __wxSetDictionary) 
 893 void wxPyPtrTypeMap_Add(const char* commonName
, const char* ptrName
) { 
 894     if (! wxPyPtrTypeMap
) 
 895         wxPyPtrTypeMap 
= PyDict_New(); 
 896     PyDict_SetItemString(wxPyPtrTypeMap
, 
 898                          PyString_FromString((char*)ptrName
)); 
 904 PyObject
*  wxPyMake_wxObject(wxObject
* source
, bool checkEvtHandler
) { 
 905     PyObject
* target 
= NULL
; 
 906     bool      isEvtHandler 
= False
; 
 909         // If it's derived from wxEvtHandler then there may 
 910         // already be a pointer to a Python object that we can use 
 912         if (checkEvtHandler 
&& wxIsKindOf(source
, wxEvtHandler
)) { 
 914             wxEvtHandler
* eh 
= (wxEvtHandler
*)source
; 
 915             wxPyOORClientData
* data 
= (wxPyOORClientData
*)eh
->GetClientObject(); 
 917                 target 
= data
->m_obj
; 
 924             // Otherwise make it the old fashioned way by making a new shadow 
 925             // object and putting this pointer in it.  Look up the class 
 926             // heirarchy until we find a class name that is located in the 
 928             const wxClassInfo
* info   
= source
->GetClassInfo(); 
 929             wxString           name   
= info
->GetClassName(); 
 930             bool               exists 
= wxPyCheckSwigType(name
); 
 931             while (info 
&& !exists
) { 
 932                 info 
= info
->GetBaseClass1(); 
 933                 name 
= info
->GetClassName(); 
 934                 exists 
= wxPyCheckSwigType(name
); 
 937                 target 
= wxPyConstructObject((void*)source
, name
, False
); 
 938                 if (target 
&& isEvtHandler
) 
 939                     ((wxEvtHandler
*)source
)->SetClientObject(new wxPyOORClientData(target
)); 
 941                 wxString 
msg(wxT("wxPython class not found for ")); 
 942                 msg 
+= source
->GetClassInfo()->GetClassName(); 
 943                 PyErr_SetString(PyExc_NameError
, msg
.mbc_str()); 
 947     } else {  // source was NULL so return None. 
 948         Py_INCREF(Py_None
); target 
= Py_None
; 
 954 PyObject
*  wxPyMake_wxSizer(wxSizer
* source
) { 
 955     PyObject
* target 
= NULL
; 
 957     if (source 
&& wxIsKindOf(source
, wxSizer
)) { 
 958         // If it's derived from wxSizer then there may already be a pointer to 
 959         // a Python object that we can use in the OOR data. 
 960         wxSizer
* sz 
= (wxSizer
*)source
; 
 961         wxPyOORClientData
* data 
= (wxPyOORClientData
*)sz
->GetClientObject(); 
 963             target 
= data
->m_obj
; 
 969         target 
= wxPyMake_wxObject(source
, False
); 
 970         if (target 
!= Py_None
) 
 971             ((wxSizer
*)source
)->SetClientObject(new wxPyOORClientData(target
)); 
 977 //--------------------------------------------------------------------------- 
 980 #ifdef WXP_WITH_THREAD 
 982 unsigned long wxPyGetCurrentThreadId() { 
 983     return wxThread::GetCurrentId(); 
 986 static wxPyThreadState gs_shutdownTState
; 
 989 wxPyThreadState
* wxPyGetThreadState() { 
 990     if (wxPyTMutex 
== NULL
) // Python is shutting down... 
 991         return &gs_shutdownTState
; 
 993     unsigned long ctid 
= wxPyGetCurrentThreadId(); 
 994     wxPyThreadState
* tstate 
= NULL
; 
 997     for(size_t i
=0; i 
< wxPyTStates
->GetCount(); i
++) { 
 998         wxPyThreadState
& info 
= wxPyTStates
->Item(i
); 
 999         if (info
.tid 
== ctid
) { 
1004     wxPyTMutex
->Unlock(); 
1005     wxASSERT_MSG(tstate
, wxT("PyThreadState should not be NULL!")); 
1011 void wxPySaveThreadState(PyThreadState
* tstate
) { 
1012     if (wxPyTMutex 
== NULL
) { // Python is shutting down, assume a single thread... 
1013         gs_shutdownTState
.tstate 
= tstate
; 
1016     unsigned long ctid 
= wxPyGetCurrentThreadId(); 
1018     for(size_t i
=0; i 
< wxPyTStates
->GetCount(); i
++) { 
1019         wxPyThreadState
& info 
= wxPyTStates
->Item(i
); 
1020         if (info
.tid 
== ctid
) { 
1022             if (info
.tstate 
!= tstate
) 
1023                 wxLogMessage("*** tstate mismatch!???"); 
1025             // info.tstate = tstate;    *** DO NOT update existing ones??? 
1026             // Normally it will never change, but apparently COM callbacks 
1027             // (i.e. ActiveX controls) will (incorrectly IMHO) use a transient 
1028             // tstate which will then be garbage the next time we try to use 
1030             wxPyTMutex
->Unlock(); 
1034     // not found, so add it... 
1035     wxPyTStates
->Add(new wxPyThreadState(ctid
, tstate
)); 
1036     wxPyTMutex
->Unlock(); 
1043 // Calls from Python to wxWindows code are wrapped in calls to these 
1046 PyThreadState
* wxPyBeginAllowThreads() { 
1047 #ifdef WXP_WITH_THREAD 
1048     PyThreadState
* saved 
= PyEval_SaveThread();  // Py_BEGIN_ALLOW_THREADS; 
1049     wxPySaveThreadState(saved
); 
1056 void wxPyEndAllowThreads(PyThreadState
* saved
) { 
1057 #ifdef WXP_WITH_THREAD 
1058     PyEval_RestoreThread(saved
);   // Py_END_ALLOW_THREADS; 
1064 // Calls from wxWindows back to Python code, or even any PyObject 
1065 // manipulations, PyDECREF's and etc. are wrapped in calls to these functions: 
1067 bool wxPyBeginBlockThreads() { 
1068 #ifdef WXP_WITH_THREAD 
1069     // This works in for 2.3, maybe a good alternative to find the needed tstate? 
1070     // PyThreadState *check = PyGILState_GetThisThreadState();   
1072     PyThreadState 
*current 
= _PyThreadState_Current
; 
1074     // Only block if there wasn't already a tstate, or if the current one is 
1075     // not the one we are wanting to change to.  This should prevent deadlock 
1076     // if there are nested calls to wxPyBeginBlockThreads 
1077     bool blocked 
= false; 
1078     wxPyThreadState
* tstate 
= wxPyGetThreadState(); 
1079     if (current 
!= tstate
->tstate
) { 
1080         PyEval_RestoreThread(tstate
->tstate
); 
1088 void wxPyEndBlockThreads(bool blocked
) { 
1089 #ifdef WXP_WITH_THREAD 
1090     // Only unblock if we blocked in the last call to wxPyBeginBlockThreads. 
1091     // The value of blocked passed in needs to be the same as that returned 
1092     // from wxPyBeginBlockThreads at the same nesting level. 
1094         PyEval_SaveThread(); 
1100 //--------------------------------------------------------------------------- 
1101 // wxPyInputStream and wxPyCBInputStream methods 
1104 void wxPyInputStream::close() { 
1105     /* do nothing for now */ 
1108 void wxPyInputStream::flush() { 
1109     /* do nothing for now */ 
1112 bool wxPyInputStream::eof() { 
1114         return m_wxis
->Eof(); 
1119 wxPyInputStream::~wxPyInputStream() { 
1126 PyObject
* wxPyInputStream::read(int size
) { 
1127     PyObject
* obj 
= NULL
; 
1129     const int BUFSIZE 
= 1024; 
1131     // check if we have a real wxInputStream to work with 
1133         bool blocked 
= wxPyBeginBlockThreads(); 
1134         PyErr_SetString(PyExc_IOError
, "no valid C-wxInputStream"); 
1135         wxPyEndBlockThreads(blocked
); 
1140         // read while bytes are available on the stream 
1141         while ( m_wxis
->CanRead() ) { 
1142             m_wxis
->Read(buf
.GetAppendBuf(BUFSIZE
), BUFSIZE
); 
1143             buf
.UngetAppendBuf(m_wxis
->LastRead()); 
1146     } else {  // Read only size number of characters 
1147         m_wxis
->Read(buf
.GetWriteBuf(size
), size
); 
1148         buf
.UngetWriteBuf(m_wxis
->LastRead()); 
1152     bool blocked 
= wxPyBeginBlockThreads(); 
1153     wxStreamError err 
= m_wxis
->GetLastError(); 
1154     if (err 
!= wxSTREAM_NO_ERROR 
&& err 
!= wxSTREAM_EOF
) { 
1155         PyErr_SetString(PyExc_IOError
,"IOError in wxInputStream"); 
1158         // We use only strings for the streams, not unicode 
1159         obj 
= PyString_FromStringAndSize(buf
, buf
.GetDataLen()); 
1161     wxPyEndBlockThreads(blocked
); 
1166 PyObject
* wxPyInputStream::readline(int size
) { 
1167     PyObject
* obj 
= NULL
; 
1172     // check if we have a real wxInputStream to work with 
1174         bool blocked 
= wxPyBeginBlockThreads(); 
1175         PyErr_SetString(PyExc_IOError
,"no valid C-wxInputStream"); 
1176         wxPyEndBlockThreads(blocked
); 
1180     // read until \n or byte limit reached 
1181     for (i
=ch
=0; (ch 
!= '\n') && (m_wxis
->CanRead()) && ((size 
< 0) || (i 
< size
)); i
++) { 
1182         ch 
= m_wxis
->GetC(); 
1187     bool blocked 
= wxPyBeginBlockThreads(); 
1188     wxStreamError err 
= m_wxis
->GetLastError(); 
1189     if (err 
!= wxSTREAM_NO_ERROR 
&& err 
!= wxSTREAM_EOF
) { 
1190         PyErr_SetString(PyExc_IOError
,"IOError in wxInputStream"); 
1193         // We use only strings for the streams, not unicode 
1194         obj 
= PyString_FromStringAndSize((char*)buf
.GetData(), buf
.GetDataLen()); 
1196     wxPyEndBlockThreads(blocked
); 
1201 PyObject
* wxPyInputStream::readlines(int sizehint
) { 
1204     // check if we have a real wxInputStream to work with 
1206         bool blocked 
= wxPyBeginBlockThreads(); 
1207         PyErr_SetString(PyExc_IOError
,"no valid C-wxInputStream"); 
1208         wxPyEndBlockThreads(blocked
); 
1213     bool blocked 
= wxPyBeginBlockThreads(); 
1214     pylist 
= PyList_New(0); 
1215     wxPyEndBlockThreads(blocked
); 
1218         bool blocked 
= wxPyBeginBlockThreads(); 
1220         wxPyEndBlockThreads(blocked
); 
1224     // read sizehint bytes or until EOF 
1226     for (i
=0; (m_wxis
->CanRead()) && ((sizehint 
< 0) || (i 
< sizehint
));) { 
1227         PyObject
* s 
= this->readline(); 
1229             bool blocked 
= wxPyBeginBlockThreads(); 
1231             wxPyEndBlockThreads(blocked
); 
1234         bool blocked 
= wxPyBeginBlockThreads(); 
1235         PyList_Append(pylist
, s
); 
1236         i 
+= PyString_Size(s
); 
1237         wxPyEndBlockThreads(blocked
); 
1241     wxStreamError err 
= m_wxis
->GetLastError(); 
1242     if (err 
!= wxSTREAM_NO_ERROR 
&& err 
!= wxSTREAM_EOF
) { 
1243         bool blocked 
= wxPyBeginBlockThreads(); 
1245         PyErr_SetString(PyExc_IOError
,"IOError in wxInputStream"); 
1246         wxPyEndBlockThreads(blocked
); 
1254 void wxPyInputStream::seek(int offset
, int whence
) { 
1256         m_wxis
->SeekI(offset
, wxSeekMode(whence
)); 
1259 int wxPyInputStream::tell(){ 
1261         return m_wxis
->TellI(); 
1268 wxPyCBInputStream::wxPyCBInputStream(PyObject 
*r
, PyObject 
*s
, PyObject 
*t
, bool block
) 
1269     : wxInputStream(), m_read(r
), m_seek(s
), m_tell(t
), m_block(block
) 
1273 wxPyCBInputStream::~wxPyCBInputStream() { 
1275     if (m_block
) blocked 
= wxPyBeginBlockThreads(); 
1279     if (m_block
) wxPyEndBlockThreads(blocked
); 
1283 wxPyCBInputStream
* wxPyCBInputStream::create(PyObject 
*py
, bool block
) { 
1285     if (block
) blocked 
= wxPyBeginBlockThreads(); 
1287     PyObject
* read 
= getMethod(py
, "read"); 
1288     PyObject
* seek 
= getMethod(py
, "seek"); 
1289     PyObject
* tell 
= getMethod(py
, "tell"); 
1292         PyErr_SetString(PyExc_TypeError
, "Not a file-like object"); 
1296         if (block
) wxPyEndBlockThreads(blocked
); 
1300     if (block
) wxPyEndBlockThreads(blocked
); 
1301     return new wxPyCBInputStream(read
, seek
, tell
, block
); 
1305 wxPyCBInputStream
* wxPyCBInputStream_create(PyObject 
*py
, bool block
) { 
1306     return wxPyCBInputStream::create(py
, block
); 
1309 PyObject
* wxPyCBInputStream::getMethod(PyObject
* py
, char* name
) { 
1310     if (!PyObject_HasAttrString(py
, name
)) 
1312     PyObject
* o 
= PyObject_GetAttrString(py
, name
); 
1313     if (!PyMethod_Check(o
) && !PyCFunction_Check(o
)) { 
1321 size_t wxPyCBInputStream::GetSize() const { 
1322     wxPyCBInputStream
* self 
= (wxPyCBInputStream
*)this; // cast off const 
1323     if (m_seek 
&& m_tell
) { 
1324         off_t temp 
= self
->OnSysTell(); 
1325         off_t ret 
= self
->OnSysSeek(0, wxFromEnd
); 
1326         self
->OnSysSeek(temp
, wxFromStart
); 
1334 size_t wxPyCBInputStream::OnSysRead(void *buffer
, size_t bufsize
) { 
1338     bool blocked 
= wxPyBeginBlockThreads(); 
1339     PyObject
* arglist 
= Py_BuildValue("(i)", bufsize
); 
1340     PyObject
* result 
= PyEval_CallObject(m_read
, arglist
); 
1344     if ((result 
!= NULL
) && PyString_Check(result
)) { 
1345         o 
= PyString_Size(result
); 
1347             m_lasterror 
= wxSTREAM_EOF
; 
1350         memcpy((char*)buffer
, PyString_AsString(result
), o
);  // strings only, not unicode... 
1355         m_lasterror 
= wxSTREAM_READ_ERROR
; 
1356     wxPyEndBlockThreads(blocked
); 
1360 size_t wxPyCBInputStream::OnSysWrite(const void *buffer
, size_t bufsize
) { 
1361     m_lasterror 
= wxSTREAM_WRITE_ERROR
; 
1365 off_t 
wxPyCBInputStream::OnSysSeek(off_t off
, wxSeekMode mode
) { 
1366     bool blocked 
= wxPyBeginBlockThreads(); 
1368     // off_t is a 64-bit value... 
1369     PyObject
* arglist 
= Py_BuildValue("(Li)", off
, mode
); 
1371     PyObject
* arglist 
= Py_BuildValue("(ii)", off
, mode
); 
1373     PyObject
* result 
= PyEval_CallObject(m_seek
, arglist
); 
1376     wxPyEndBlockThreads(blocked
); 
1381 off_t 
wxPyCBInputStream::OnSysTell() const { 
1382     bool blocked 
= wxPyBeginBlockThreads(); 
1383     PyObject
* arglist 
= Py_BuildValue("()"); 
1384     PyObject
* result 
= PyEval_CallObject(m_tell
, arglist
); 
1387     if (result 
!= NULL
) { 
1389         if (PyLong_Check(result
)) 
1390             o 
= PyLong_AsLongLong(result
); 
1393             o 
= PyInt_AsLong(result
); 
1396     wxPyEndBlockThreads(blocked
); 
1400 //---------------------------------------------------------------------- 
1402 IMPLEMENT_ABSTRACT_CLASS(wxPyCallback
, wxObject
); 
1404 wxPyCallback::wxPyCallback(PyObject
* func
) { 
1409 wxPyCallback::wxPyCallback(const wxPyCallback
& other
) { 
1410     m_func 
= other
.m_func
; 
1414 wxPyCallback::~wxPyCallback() { 
1415     bool blocked 
= wxPyBeginBlockThreads(); 
1417     wxPyEndBlockThreads(blocked
); 
1421 #define wxPy_PRECALLINIT     "_preCallInit" 
1422 #define wxPy_POSTCALLCLEANUP "_postCallCleanup" 
1424 // This function is used for all events destined for Python event handlers. 
1425 void wxPyCallback::EventThunker(wxEvent
& event
) { 
1426     wxPyCallback
*   cb 
= (wxPyCallback
*)event
.m_callbackUserData
; 
1427     PyObject
*       func 
= cb
->m_func
; 
1431     bool            checkSkip 
= False
; 
1433     bool blocked 
= wxPyBeginBlockThreads(); 
1434     wxString className 
= event
.GetClassInfo()->GetClassName(); 
1436     // If the event is one of these types then pass the original 
1437     // event object instead of the one passed to us. 
1438     if ( className 
== wxT("wxPyEvent") ) { 
1439         arg 
=       ((wxPyEvent
*)&event
)->GetSelf(); 
1440         checkSkip 
= ((wxPyEvent
*)&event
)->GetCloned(); 
1442     else if ( className 
== wxT("wxPyCommandEvent") ) { 
1443         arg 
=       ((wxPyCommandEvent
*)&event
)->GetSelf(); 
1444         checkSkip 
= ((wxPyCommandEvent
*)&event
)->GetCloned(); 
1447         arg 
= wxPyConstructObject((void*)&event
, className
); 
1453         // "intern" the pre/post method names to speed up the HasAttr 
1454         static PyObject
* s_preName  
= NULL
; 
1455         static PyObject
* s_postName 
= NULL
; 
1456         if (s_preName 
== NULL
) { 
1457             s_preName  
= PyString_FromString(wxPy_PRECALLINIT
); 
1458             s_postName 
= PyString_FromString(wxPy_POSTCALLCLEANUP
); 
1461         // Check if the event object needs some preinitialization 
1462         if (PyObject_HasAttr(arg
, s_preName
)) { 
1463             result 
= PyObject_CallMethodObjArgs(arg
, s_preName
, arg
, NULL
); 
1465                 Py_DECREF(result
);   // result is ignored, but we still need to decref it 
1466                 PyErr_Clear();       // Just in case... 
1472         // Call the event handler, passing the event object 
1473         tuple 
= PyTuple_New(1); 
1474         PyTuple_SET_ITEM(tuple
, 0, arg
);  // steals ref to arg 
1475         result 
= PyEval_CallObject(func
, tuple
); 
1477             Py_DECREF(result
);   // result is ignored, but we still need to decref it 
1478             PyErr_Clear();       // Just in case... 
1483         // Check if the event object needs some post cleanup 
1484         if (PyObject_HasAttr(arg
, s_postName
)) { 
1485             result 
= PyObject_CallMethodObjArgs(arg
, s_postName
, arg
, NULL
); 
1487                 Py_DECREF(result
);   // result is ignored, but we still need to decref it 
1488                 PyErr_Clear();       // Just in case... 
1495             // if the event object was one of our special types and 
1496             // it had been cloned, then we need to extract the Skipped 
1497             // value from the original and set it in the clone. 
1498             result 
= PyObject_CallMethod(arg
, "GetSkipped", ""); 
1500                 event
.Skip(PyInt_AsLong(result
)); 
1508     wxPyEndBlockThreads(blocked
); 
1512 //---------------------------------------------------------------------- 
1514 wxPyCallbackHelper::wxPyCallbackHelper(const wxPyCallbackHelper
& other
) { 
1516       m_self 
= other
.m_self
; 
1517       m_class 
= other
.m_class
; 
1525 void wxPyCallbackHelper::setSelf(PyObject
* self
, PyObject
* klass
, int incref
) { 
1536 #if PYTHON_API_VERSION >= 1011 
1538 // Prior to Python 2.2 PyMethod_GetClass returned the class object 
1539 // in which the method was defined.  Starting with 2.2 it returns 
1540 // "class that asked for the method" which seems totally bogus to me 
1541 // but apprently it fixes some obscure problem waiting to happen in 
1542 // Python.  Since the API was not documented Guido and the gang felt 
1543 // safe in changing it.  Needless to say that totally screwed up the 
1544 // logic below in wxPyCallbackHelper::findCallback, hence this icky 
1545 // code to find the class where the method is actually defined... 
1548 PyObject
* PyFindClassWithAttr(PyObject 
*klass
, PyObject 
*name
) 
1552     if (PyType_Check(klass
)) {      // new style classes 
1553         // This code is borrowed/adapted from _PyType_Lookup in typeobject.c 
1554         PyTypeObject
* type 
= (PyTypeObject
*)klass
; 
1555         PyObject 
*mro
, *res
, *base
, *dict
; 
1556         /* Look in tp_dict of types in MRO */ 
1558         assert(PyTuple_Check(mro
)); 
1559         n 
= PyTuple_GET_SIZE(mro
); 
1560         for (i 
= 0; i 
< n
; i
++) { 
1561             base 
= PyTuple_GET_ITEM(mro
, i
); 
1562             if (PyClass_Check(base
)) 
1563                 dict 
= ((PyClassObject 
*)base
)->cl_dict
; 
1565                 assert(PyType_Check(base
)); 
1566                 dict 
= ((PyTypeObject 
*)base
)->tp_dict
; 
1568             assert(dict 
&& PyDict_Check(dict
)); 
1569             res 
= PyDict_GetItem(dict
, name
); 
1576     else if (PyClass_Check(klass
)) { // old style classes 
1577         // This code is borrowed/adapted from class_lookup in classobject.c 
1578         PyClassObject
* cp 
= (PyClassObject
*)klass
; 
1579         PyObject 
*value 
= PyDict_GetItem(cp
->cl_dict
, name
); 
1580         if (value 
!= NULL
) { 
1581             return (PyObject
*)cp
; 
1583         n 
= PyTuple_Size(cp
->cl_bases
); 
1584         for (i 
= 0; i 
< n
; i
++) { 
1585             PyObject
* base 
= PyTuple_GetItem(cp
->cl_bases
, i
); 
1586             PyObject 
*v 
= PyFindClassWithAttr(base
, name
); 
1598 PyObject
* PyMethod_GetDefiningClass(PyObject
* method
, const char* name
) 
1600     PyObject
* mgc 
= PyMethod_GET_CLASS(method
); 
1602 #if PYTHON_API_VERSION <= 1010    // prior to Python 2.2, the easy way 
1604 #else                             // 2.2 and after, the hard way... 
1606     PyObject
* nameo 
= PyString_FromString(name
); 
1607     PyObject
* klass 
= PyFindClassWithAttr(mgc
, nameo
); 
1615 bool wxPyCallbackHelper::findCallback(const char* name
) const { 
1616     wxPyCallbackHelper
* self 
= (wxPyCallbackHelper
*)this; // cast away const 
1617     self
->m_lastFound 
= NULL
; 
1619     // If the object (m_self) has an attibute of the given name... 
1620     if (m_self 
&& PyObject_HasAttrString(m_self
, (char*)name
)) { 
1621         PyObject 
*method
, *klass
; 
1622         method 
= PyObject_GetAttrString(m_self
, (char*)name
); 
1624         // ...and if that attribute is a method, and if that method's class is 
1625         // not from a base class... 
1626         if (PyMethod_Check(method
) && 
1627             (klass 
= PyMethod_GetDefiningClass(method
, (char*)name
)) != NULL 
&& 
1628             ((klass 
== m_class
) || PyObject_IsSubclass(klass
, m_class
))) { 
1630             // ...then we'll save a pointer to the method so callCallback can call it. 
1631             self
->m_lastFound 
= method
; 
1637     return m_lastFound 
!= NULL
; 
1641 int wxPyCallbackHelper::callCallback(PyObject
* argTuple
) const { 
1645     result 
= callCallbackObj(argTuple
); 
1646     if (result
) {                       // Assumes an integer return type... 
1647         retval 
= PyInt_AsLong(result
); 
1649         PyErr_Clear();                  // forget about it if it's not... 
1654 // Invoke the Python callable object, returning the raw PyObject return 
1655 // value.  Caller should DECREF the return value and also manage the GIL. 
1656 PyObject
* wxPyCallbackHelper::callCallbackObj(PyObject
* argTuple
) const { 
1659     // Save a copy of the pointer in case the callback generates another 
1660     // callback.  In that case m_lastFound will have a different value when 
1661     // it gets back here... 
1662     PyObject
* method 
= m_lastFound
; 
1664     result 
= PyEval_CallObject(method
, argTuple
); 
1665     Py_DECREF(argTuple
); 
1674 void wxPyCBH_setCallbackInfo(wxPyCallbackHelper
& cbh
, PyObject
* self
, PyObject
* klass
, int incref
) { 
1675     cbh
.setSelf(self
, klass
, incref
); 
1678 bool wxPyCBH_findCallback(const wxPyCallbackHelper
& cbh
, const char* name
) { 
1679     return cbh
.findCallback(name
); 
1682 int  wxPyCBH_callCallback(const wxPyCallbackHelper
& cbh
, PyObject
* argTuple
) { 
1683     return cbh
.callCallback(argTuple
); 
1686 PyObject
* wxPyCBH_callCallbackObj(const wxPyCallbackHelper
& cbh
, PyObject
* argTuple
) { 
1687     return cbh
.callCallbackObj(argTuple
); 
1691 void wxPyCBH_delete(wxPyCallbackHelper
* cbh
) { 
1692     if (cbh
->m_incRef
) { 
1693         bool blocked 
= wxPyBeginBlockThreads(); 
1694         Py_XDECREF(cbh
->m_self
); 
1695         Py_XDECREF(cbh
->m_class
); 
1696         wxPyEndBlockThreads(blocked
); 
1700 //--------------------------------------------------------------------------- 
1701 //--------------------------------------------------------------------------- 
1702 // These event classes can be derived from in Python and passed through the event 
1703 // system without losing anything.  They do this by keeping a reference to 
1704 // themselves and some special case handling in wxPyCallback::EventThunker. 
1707 wxPyEvtSelfRef::wxPyEvtSelfRef() { 
1708     //m_self = Py_None;         // **** We don't do normal ref counting to prevent 
1709     //Py_INCREF(m_self);        //      circular loops... 
1713 wxPyEvtSelfRef::~wxPyEvtSelfRef() { 
1714     bool blocked 
= wxPyBeginBlockThreads(); 
1717     wxPyEndBlockThreads(blocked
); 
1720 void wxPyEvtSelfRef::SetSelf(PyObject
* self
, bool clone
) { 
1721     bool blocked 
= wxPyBeginBlockThreads(); 
1729     wxPyEndBlockThreads(blocked
); 
1732 PyObject
* wxPyEvtSelfRef::GetSelf() const { 
1738 IMPLEMENT_ABSTRACT_CLASS(wxPyEvent
, wxEvent
); 
1739 IMPLEMENT_ABSTRACT_CLASS(wxPyCommandEvent
, wxCommandEvent
); 
1742 wxPyEvent::wxPyEvent(int winid
, wxEventType commandType
) 
1743     : wxEvent(winid
, commandType
) { 
1747 wxPyEvent::wxPyEvent(const wxPyEvent
& evt
) 
1750     SetSelf(evt
.m_self
, True
); 
1754 wxPyEvent::~wxPyEvent() { 
1758 wxPyCommandEvent::wxPyCommandEvent(wxEventType commandType
, int id
) 
1759     : wxCommandEvent(commandType
, id
) { 
1763 wxPyCommandEvent::wxPyCommandEvent(const wxPyCommandEvent
& evt
) 
1764     : wxCommandEvent(evt
) 
1766     SetSelf(evt
.m_self
, True
); 
1770 wxPyCommandEvent::~wxPyCommandEvent() { 
1777 //--------------------------------------------------------------------------- 
1778 //--------------------------------------------------------------------------- 
1779 // Convert a wxList to a Python List, only works for lists of wxObjects 
1781 PyObject
* wxPy_ConvertList(wxListBase
* listbase
) { 
1782     wxList
*     list 
= (wxList
*)listbase
;  // this is probably bad... 
1786     wxNode
*     node 
= list
->GetFirst(); 
1788     bool blocked 
= wxPyBeginBlockThreads(); 
1789     pyList 
= PyList_New(0); 
1791         wxObj 
= node
->GetData(); 
1792         pyObj 
= wxPyMake_wxObject(wxObj
); 
1793         PyList_Append(pyList
, pyObj
); 
1794         node 
= node
->GetNext(); 
1796     wxPyEndBlockThreads(blocked
); 
1800 //---------------------------------------------------------------------- 
1802 long wxPyGetWinHandle(wxWindow
* win
) { 
1805     return (long)win
->GetHandle(); 
1808     // Find and return the actual X-Window. 
1810     if (win
->m_wxwindow
) { 
1812         return (long) GDK_WINDOW_XWINDOW(GTK_PIZZA(win
->m_wxwindow
)->bin_window
); 
1814         GdkWindowPrivate
* bwin 
= (GdkWindowPrivate
*)GTK_PIZZA(win
->m_wxwindow
)->bin_window
; 
1816             return (long)bwin
->xwindow
; 
1823     return (long)MAC_WXHWND(win
->MacGetRootWindow()); 
1829 //---------------------------------------------------------------------- 
1830 // Some helper functions for typemaps in my_typemaps.i, so they won't be 
1831 // included in every file over and over again... 
1833 #if PYTHON_API_VERSION >= 1009 
1834     static char* wxStringErrorMsg 
= "String or Unicode type required"; 
1836     static char* wxStringErrorMsg 
= "String type required"; 
1840 wxString
* wxString_in_helper(PyObject
* source
) { 
1842 #if PYTHON_API_VERSION >= 1009  // Have Python unicode API 
1843     if (!PyString_Check(source
) && !PyUnicode_Check(source
)) { 
1844         PyErr_SetString(PyExc_TypeError
, wxStringErrorMsg
); 
1848     if (PyUnicode_Check(source
)) { 
1849         target 
= new wxString(); 
1850         size_t len 
= PyUnicode_GET_SIZE(source
); 
1852             PyUnicode_AsWideChar((PyUnicodeObject
*)source
, target
->GetWriteBuf(len
), len
); 
1853             target
->UngetWriteBuf(); 
1856         // It is a string, get pointers to it and transform to unicode 
1857         char* tmpPtr
; int tmpSize
; 
1858         PyString_AsStringAndSize(source
, &tmpPtr
, &tmpSize
); 
1859         target 
= new wxString(tmpPtr
, *wxConvCurrent
, tmpSize
); 
1862     char* tmpPtr
; int tmpSize
; 
1863     if (PyString_AsStringAndSize(source
, &tmpPtr
, &tmpSize
) == -1) { 
1864         PyErr_SetString(PyExc_TypeError
, "Unable to convert string"); 
1867     target 
= new wxString(tmpPtr
, tmpSize
); 
1868 #endif // wxUSE_UNICODE 
1870 #else  // No Python unicode API (1.5.2) 
1871     if (!PyString_Check(source
)) { 
1872         PyErr_SetString(PyExc_TypeError
, wxStringErrorMsg
); 
1875     target 
= new wxString(PyString_AS_STRING(source
), PyString_GET_SIZE(source
)); 
1881 // Similar to above except doesn't use "new" and doesn't set an exception 
1882 wxString 
Py2wxString(PyObject
* source
) 
1885     bool     doDecRef 
= False
; 
1887 #if PYTHON_API_VERSION >= 1009  // Have Python unicode API 
1888     if (!PyString_Check(source
) && !PyUnicode_Check(source
)) { 
1889         // Convert to String if not one already...  (TODO: Unicode too?) 
1890         source 
= PyObject_Str(source
); 
1895     if (PyUnicode_Check(source
)) { 
1896         size_t len 
= PyUnicode_GET_SIZE(source
); 
1898             PyUnicode_AsWideChar((PyUnicodeObject
*)source
, target
.GetWriteBuf(len
), len
); 
1899             target
.UngetWriteBuf(); 
1902         // It is a string, get pointers to it and transform to unicode 
1903         char* tmpPtr
; int tmpSize
; 
1904         PyString_AsStringAndSize(source
, &tmpPtr
, &tmpSize
); 
1905         target 
= wxString(tmpPtr
, *wxConvCurrent
, tmpSize
); 
1908     char* tmpPtr
; int tmpSize
; 
1909     PyString_AsStringAndSize(source
, &tmpPtr
, &tmpSize
); 
1910     target 
= wxString(tmpPtr
, tmpSize
); 
1911 #endif // wxUSE_UNICODE 
1913 #else  // No Python unicode API (1.5.2) 
1914     if (!PyString_Check(source
)) { 
1915         // Convert to String if not one already... 
1916         source 
= PyObject_Str(source
); 
1919     target 
= wxString(PyString_AS_STRING(source
), PyString_GET_SIZE(source
)); 
1928 // Make either a Python String or Unicode object, depending on build mode 
1929 PyObject
* wx2PyString(const wxString
& src
) 
1933     str 
= PyUnicode_FromWideChar(src
.c_str(), src
.Len()); 
1935     str 
= PyString_FromStringAndSize(src
.c_str(), src
.Len()); 
1941 //---------------------------------------------------------------------- 
1944 byte
* byte_LIST_helper(PyObject
* source
) { 
1945     if (!PyList_Check(source
)) { 
1946         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1949     int count 
= PyList_Size(source
); 
1950     byte
* temp 
= new byte
[count
]; 
1952         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1955     for (int x
=0; x
<count
; x
++) { 
1956         PyObject
* o 
= PyList_GetItem(source
, x
); 
1957         if (! PyInt_Check(o
)) { 
1958             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
1961         temp
[x
] = (byte
)PyInt_AsLong(o
); 
1967 int* int_LIST_helper(PyObject
* source
) { 
1968     if (!PyList_Check(source
)) { 
1969         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1972     int count 
= PyList_Size(source
); 
1973     int* temp 
= new int[count
]; 
1975         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1978     for (int x
=0; x
<count
; x
++) { 
1979         PyObject
* o 
= PyList_GetItem(source
, x
); 
1980         if (! PyInt_Check(o
)) { 
1981             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
1984         temp
[x
] = PyInt_AsLong(o
); 
1990 long* long_LIST_helper(PyObject
* source
) { 
1991     if (!PyList_Check(source
)) { 
1992         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1995     int count 
= PyList_Size(source
); 
1996     long* temp 
= new long[count
]; 
1998         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
2001     for (int x
=0; x
<count
; x
++) { 
2002         PyObject
* o 
= PyList_GetItem(source
, x
); 
2003         if (! PyInt_Check(o
)) { 
2004             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
2007         temp
[x
] = PyInt_AsLong(o
); 
2013 char** string_LIST_helper(PyObject
* source
) { 
2014     if (!PyList_Check(source
)) { 
2015         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
2018     int count 
= PyList_Size(source
); 
2019     char** temp 
= new char*[count
]; 
2021         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
2024     for (int x
=0; x
<count
; x
++) { 
2025         PyObject
* o 
= PyList_GetItem(source
, x
); 
2026         if (! PyString_Check(o
)) { 
2027             PyErr_SetString(PyExc_TypeError
, "Expected a list of strings."); 
2030         temp
[x
] = PyString_AsString(o
); 
2035 //-------------------------------- 
2036 // Part of patch from Tim Hochberg 
2037 static inline bool wxPointFromObjects(PyObject
* o1
, PyObject
* o2
, wxPoint
* point
) { 
2038     if (PyInt_Check(o1
) && PyInt_Check(o2
)) { 
2039         point
->x 
= PyInt_AS_LONG(o1
); 
2040         point
->y 
= PyInt_AS_LONG(o2
); 
2043     if (PyFloat_Check(o1
) && PyFloat_Check(o2
)) { 
2044         point
->x 
= (int)PyFloat_AS_DOUBLE(o1
); 
2045         point
->y 
= (int)PyFloat_AS_DOUBLE(o2
); 
2048     if (wxPySwigInstance_Check(o1
) || wxPySwigInstance_Check(o2
)) {  // TODO: Why??? 
2049         // Disallow instances because they can cause havok 
2052     if (PyNumber_Check(o1
) && PyNumber_Check(o2
)) { 
2053         // I believe this excludes instances, so this should be safe without INCREFFing o1 and o2 
2054         point
->x 
= PyInt_AsLong(o1
); 
2055         point
->y 
= PyInt_AsLong(o2
); 
2062 wxPoint
* wxPoint_LIST_helper(PyObject
* source
, int *count
) { 
2063     // Putting all of the declarations here allows 
2064     // us to put the error handling all in one place. 
2067     PyObject 
*o
, *o1
, *o2
; 
2068     bool isFast 
= PyList_Check(source
) || PyTuple_Check(source
); 
2070     if (!PySequence_Check(source
)) { 
2074     // The length of the sequence is returned in count. 
2075     *count 
= PySequence_Length(source
); 
2080     temp 
= new wxPoint
[*count
]; 
2082         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
2085     for (x
=0; x
<*count
; x
++) { 
2086         // Get an item: try fast way first. 
2088             o 
= PySequence_Fast_GET_ITEM(source
, x
); 
2091             o 
= PySequence_GetItem(source
, x
); 
2097         // Convert o to wxPoint. 
2098         if ((PyTuple_Check(o
) && PyTuple_GET_SIZE(o
) == 2) || 
2099             (PyList_Check(o
) && PyList_GET_SIZE(o
) == 2)) { 
2100             o1 
= PySequence_Fast_GET_ITEM(o
, 0); 
2101             o2 
= PySequence_Fast_GET_ITEM(o
, 1); 
2102             if (!wxPointFromObjects(o1
, o2
, &temp
[x
])) { 
2106         else if (wxPySwigInstance_Check(o
)) { 
2108             if (! wxPyConvertSwigPtr(o
, (void **)&pt
, wxT("wxPoint"))) { 
2113         else if (PySequence_Check(o
) && PySequence_Length(o
) == 2) { 
2114             o1 
= PySequence_GetItem(o
, 0); 
2115             o2 
= PySequence_GetItem(o
, 1); 
2116             if (!wxPointFromObjects(o1
, o2
, &temp
[x
])) { 
2140     PyErr_SetString(PyExc_TypeError
, "Expected a sequence of length-2 sequences or wxPoints."); 
2144 //------------------------------ 
2147 wxBitmap
** wxBitmap_LIST_helper(PyObject
* source
) { 
2148     if (!PyList_Check(source
)) { 
2149         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
2152     int count 
= PyList_Size(source
); 
2153     wxBitmap
** temp 
= new wxBitmap
*[count
]; 
2155         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
2158     for (int x
=0; x
<count
; x
++) { 
2159         PyObject
* o 
= PyList_GetItem(source
, x
); 
2160         if (wxPySwigInstance_Check(o
)) { 
2162             if (! wxPyConvertSwigPtr(o
, (void **) &pt
, wxT("wxBitmap"))) { 
2163                 PyErr_SetString(PyExc_TypeError
,"Expected wxBitmap."); 
2169             PyErr_SetString(PyExc_TypeError
, "Expected a list of wxBitmaps."); 
2178 wxString
* wxString_LIST_helper(PyObject
* source
) { 
2179     if (!PyList_Check(source
)) { 
2180         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
2183     int count 
= PyList_Size(source
); 
2184     wxString
* temp 
= new wxString
[count
]; 
2186         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
2189     for (int x
=0; x
<count
; x
++) { 
2190         PyObject
* o 
= PyList_GetItem(source
, x
); 
2191 #if PYTHON_API_VERSION >= 1009 
2192         if (! PyString_Check(o
) && ! PyUnicode_Check(o
)) { 
2193             PyErr_SetString(PyExc_TypeError
, "Expected a list of string or unicode objects."); 
2197         if (! PyString_Check(o
)) { 
2198             PyErr_SetString(PyExc_TypeError
, "Expected a list of strings."); 
2203         wxString
* pStr 
= wxString_in_helper(o
); 
2211 wxAcceleratorEntry
* wxAcceleratorEntry_LIST_helper(PyObject
* source
) { 
2212     if (!PyList_Check(source
)) { 
2213         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
2216     int count                
= PyList_Size(source
); 
2217     wxAcceleratorEntry
* temp 
= new wxAcceleratorEntry
[count
]; 
2219         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
2222     for (int x
=0; x
<count
; x
++) { 
2223         PyObject
* o 
= PyList_GetItem(source
, x
); 
2224         if (wxPySwigInstance_Check(o
)) { 
2225             wxAcceleratorEntry
* ae
; 
2226             if (! wxPyConvertSwigPtr(o
, (void **) &ae
, wxT("wxAcceleratorEntry"))) { 
2227                 PyErr_SetString(PyExc_TypeError
,"Expected wxAcceleratorEntry."); 
2232         else if (PyTuple_Check(o
)) { 
2233             PyObject
* o1 
= PyTuple_GetItem(o
, 0); 
2234             PyObject
* o2 
= PyTuple_GetItem(o
, 1); 
2235             PyObject
* o3 
= PyTuple_GetItem(o
, 2); 
2236             temp
[x
].Set(PyInt_AsLong(o1
), PyInt_AsLong(o2
), PyInt_AsLong(o3
)); 
2239             PyErr_SetString(PyExc_TypeError
, "Expected a list of 3-tuples or wxAcceleratorEntry objects."); 
2247 wxPen
** wxPen_LIST_helper(PyObject
* source
) { 
2248     if (!PyList_Check(source
)) { 
2249         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
2252     int count 
= PyList_Size(source
); 
2253     wxPen
** temp 
= new wxPen
*[count
]; 
2255         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
2258     for (int x
=0; x
<count
; x
++) { 
2259         PyObject
* o 
= PyList_GetItem(source
, x
); 
2260         if (wxPySwigInstance_Check(o
)) { 
2262             if (! wxPyConvertSwigPtr(o
, (void **)&pt
, wxT("wxPen"))) { 
2264                 PyErr_SetString(PyExc_TypeError
,"Expected wxPen."); 
2271             PyErr_SetString(PyExc_TypeError
, "Expected a list of wxPens."); 
2279 bool wxPy2int_seq_helper(PyObject
* source
, int* i1
, int* i2
) { 
2280     bool isFast 
= PyList_Check(source
) || PyTuple_Check(source
); 
2283     if (!PySequence_Check(source
) || PySequence_Length(source
) != 2) 
2287         o1 
= PySequence_Fast_GET_ITEM(source
, 0); 
2288         o2 
= PySequence_Fast_GET_ITEM(source
, 1); 
2291         o1 
= PySequence_GetItem(source
, 0); 
2292         o2 
= PySequence_GetItem(source
, 1); 
2295     *i1 
= PyInt_AsLong(o1
); 
2296     *i2 
= PyInt_AsLong(o2
); 
2306 bool wxPy4int_seq_helper(PyObject
* source
, int* i1
, int* i2
, int* i3
, int* i4
) { 
2307     bool isFast 
= PyList_Check(source
) || PyTuple_Check(source
); 
2308     PyObject 
*o1
, *o2
, *o3
, *o4
; 
2310     if (!PySequence_Check(source
) || PySequence_Length(source
) != 4) 
2314         o1 
= PySequence_Fast_GET_ITEM(source
, 0); 
2315         o2 
= PySequence_Fast_GET_ITEM(source
, 1); 
2316         o3 
= PySequence_Fast_GET_ITEM(source
, 2); 
2317         o4 
= PySequence_Fast_GET_ITEM(source
, 3); 
2320         o1 
= PySequence_GetItem(source
, 0); 
2321         o2 
= PySequence_GetItem(source
, 1); 
2322         o3 
= PySequence_GetItem(source
, 2); 
2323         o4 
= PySequence_GetItem(source
, 3); 
2326     *i1 
= PyInt_AsLong(o1
); 
2327     *i2 
= PyInt_AsLong(o2
); 
2328     *i3 
= PyInt_AsLong(o3
); 
2329     *i4 
= PyInt_AsLong(o4
); 
2341 //---------------------------------------------------------------------- 
2343 bool wxPySimple_typecheck(PyObject
* source
, const wxChar
* classname
, int seqLen
) 
2347     if (wxPySwigInstance_Check(source
) && 
2348         wxPyConvertSwigPtr(source
, (void **)&ptr
, classname
)) 
2352     if (PySequence_Check(source
) && PySequence_Length(source
) == seqLen
) 
2358 bool wxSize_helper(PyObject
* source
, wxSize
** obj
) 
2360     if (source 
== Py_None
) { 
2361         **obj 
= wxSize(-1,-1); 
2364     return wxPyTwoIntItem_helper(source
, obj
, wxT("wxSize")); 
2368 bool wxPoint_helper(PyObject
* source
, wxPoint
** obj
) 
2370     if (source 
== Py_None
) { 
2371         **obj 
= wxPoint(-1,-1); 
2374     return wxPyTwoIntItem_helper(source
, obj
, wxT("wxPoint")); 
2379 bool wxRealPoint_helper(PyObject
* source
, wxRealPoint
** obj
) { 
2381     if (source 
== Py_None
) { 
2382         **obj 
= wxRealPoint(-1,-1); 
2386     // If source is an object instance then it may already be the right type 
2387     if (wxPySwigInstance_Check(source
)) { 
2389         if (! wxPyConvertSwigPtr(source
, (void **)&ptr
, wxT("wxRealPoint"))) 
2394     // otherwise a 2-tuple of floats is expected 
2395     else if (PySequence_Check(source
) && PyObject_Length(source
) == 2) { 
2396         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
2397         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
2398         if (!PyNumber_Check(o1
) || !PyNumber_Check(o2
)) { 
2403         **obj 
= wxRealPoint(PyFloat_AsDouble(o1
), PyFloat_AsDouble(o2
)); 
2410     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of floats or a wxRealPoint object."); 
2416 bool wxRect_helper(PyObject
* source
, wxRect
** obj
) { 
2418     if (source 
== Py_None
) { 
2419         **obj 
= wxRect(-1,-1,-1,-1); 
2423     // If source is an object instance then it may already be the right type 
2424     if (wxPySwigInstance_Check(source
)) { 
2426         if (! wxPyConvertSwigPtr(source
, (void **)&ptr
, wxT("wxRect"))) 
2431     // otherwise a 4-tuple of integers is expected 
2432     else if (PySequence_Check(source
) && PyObject_Length(source
) == 4) { 
2433         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
2434         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
2435         PyObject
* o3 
= PySequence_GetItem(source
, 2); 
2436         PyObject
* o4 
= PySequence_GetItem(source
, 3); 
2437         if (!PyNumber_Check(o1
) || !PyNumber_Check(o2
) || 
2438             !PyNumber_Check(o3
) || !PyNumber_Check(o4
)) { 
2445         **obj 
= wxRect(PyInt_AsLong(o1
), PyInt_AsLong(o2
), 
2446                        PyInt_AsLong(o3
), PyInt_AsLong(o4
)); 
2455     PyErr_SetString(PyExc_TypeError
, "Expected a 4-tuple of integers or a wxRect object."); 
2461 bool wxColour_helper(PyObject
* source
, wxColour
** obj
) { 
2463     if (source 
== Py_None
) { 
2464         **obj 
= wxNullColour
; 
2468     // If source is an object instance then it may already be the right type 
2469     if (wxPySwigInstance_Check(source
)) { 
2471         if (! wxPyConvertSwigPtr(source
, (void **)&ptr
, wxT("wxColour"))) 
2476     // otherwise check for a string 
2477     else if (PyString_Check(source
) || PyUnicode_Check(source
)) { 
2478         wxString spec 
= Py2wxString(source
); 
2479         if (spec
.GetChar(0) == '#' && spec
.Length() == 7) {  // It's  #RRGGBB 
2480             long red
, green
, blue
; 
2481             red 
= green 
= blue 
= 0; 
2482             spec
.Mid(1,2).ToLong(&red
,   16); 
2483             spec
.Mid(3,2).ToLong(&green
, 16); 
2484             spec
.Mid(5,2).ToLong(&blue
,  16); 
2486             **obj 
= wxColour(red
, green
, blue
); 
2489         else {                                       // it's a colour name 
2490             **obj 
= wxColour(spec
); 
2494     // last chance: 3-tuple of integers is expected 
2495     else if (PySequence_Check(source
) && PyObject_Length(source
) == 3) { 
2496         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
2497         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
2498         PyObject
* o3 
= PySequence_GetItem(source
, 2); 
2499         if (!PyNumber_Check(o1
) || !PyNumber_Check(o2
) || !PyNumber_Check(o3
)) { 
2505         **obj 
= wxColour(PyInt_AsLong(o1
), PyInt_AsLong(o2
), PyInt_AsLong(o3
)); 
2513     PyErr_SetString(PyExc_TypeError
, 
2514                     "Expected a wxColour object or a string containing a colour name or '#RRGGBB'."); 
2519 bool wxColour_typecheck(PyObject
* source
) { 
2521     if (wxPySimple_typecheck(source
, wxT("wxColour"), 3)) 
2524     if (PyString_Check(source
) || PyUnicode_Check(source
)) 
2532 bool wxPoint2D_helper(PyObject
* source
, wxPoint2D
** obj
) { 
2534     if (source 
== Py_None
) { 
2535         **obj 
= wxPoint2D(-1,-1); 
2539     // If source is an object instance then it may already be the right type 
2540     if (wxPySwigInstance_Check(source
)) { 
2542         if (! wxPyConvertSwigPtr(source
, (void **)&ptr
, wxT("wxPoint2D"))) 
2547     // otherwise a length-2 sequence of floats is expected 
2548     if (PySequence_Check(source
) && PySequence_Length(source
) == 2) { 
2549         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
2550         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
2551         // This should really check for floats, not numbers -- but that would break code. 
2552         if (!PyNumber_Check(o1
) || !PyNumber_Check(o2
)) { 
2557         **obj 
= wxPoint2D(PyFloat_AsDouble(o1
), PyFloat_AsDouble(o2
)); 
2563     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of floats or a wxPoint2D object."); 
2568 //---------------------------------------------------------------------- 
2570 PyObject
* wxArrayString2PyList_helper(const wxArrayString
& arr
) { 
2572     PyObject
* list 
= PyList_New(0); 
2573     for (size_t i
=0; i 
< arr
.GetCount(); i
++) { 
2575         PyObject
* str 
= PyUnicode_FromWideChar(arr
[i
].c_str(), arr
[i
].Len()); 
2577         PyObject
* str 
= PyString_FromStringAndSize(arr
[i
].c_str(), arr
[i
].Len()); 
2579         PyList_Append(list
, str
); 
2586 PyObject
* wxArrayInt2PyList_helper(const wxArrayInt
& arr
) { 
2588     PyObject
* list 
= PyList_New(0); 
2589     for (size_t i
=0; i 
< arr
.GetCount(); i
++) { 
2590         PyObject
* number 
= PyInt_FromLong(arr
[i
]); 
2591         PyList_Append(list
, number
); 
2598 //---------------------------------------------------------------------- 
2599 //----------------------------------------------------------------------