1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     Helper functions/classes for the wxPython extension module 
   9 // Copyright:   (c) 1998 by Total Control Software 
  10 // Licence:     wxWindows license 
  11 ///////////////////////////////////////////////////////////////////////////// 
  13 #include <stdio.h>  // get the correct definition of NULL 
  20 #include <wx/msw/private.h> 
  23 #undef LoadAccelerators 
  30 #include <gdk/gdkprivate.h> 
  31 #include <wx/gtk/win_gtk.h> 
  32 //#include <gdk/gdk.h> 
  33 //#include <gdk/gdkx.h> 
  34 //#include <gtk/gtkwindow.h> 
  36 //extern GtkWidget *wxRootWindow; 
  41 //--------------------------------------------------------------------------- 
  43 //wxHashTable*  wxPyWindows = NULL; 
  46 wxPoint     wxPyDefaultPosition
;        //wxDefaultPosition); 
  47 wxSize      wxPyDefaultSize
;            //wxDefaultSize); 
  48 wxString    
wxPyEmptyStr(""); 
  52 #ifdef __WXMSW__             // If building for win32... 
  53 //---------------------------------------------------------------------- 
  54 // This gets run when the DLL is loaded.  We just need to save a handle. 
  55 //---------------------------------------------------------------------- 
  58     HINSTANCE   hinstDLL
,    // handle to DLL module 
  59     DWORD       fdwReason
,   // reason for calling function 
  60     LPVOID      lpvReserved  
// reserved 
  63     wxSetInstance(hinstDLL
); 
  68 //---------------------------------------------------------------------- 
  69 // Class for implementing the wxp main application shell. 
  70 //---------------------------------------------------------------------- 
  72 wxPyApp 
*wxPythonApp 
= NULL
;            // Global instance of application object 
  76 //    printf("**** ctor\n"); 
  80 //    printf("**** dtor\n"); 
  84 // This one isn't acutally called...  See __wxStart() 
  85 bool wxPyApp::OnInit(void) { 
  89 int  wxPyApp::MainLoop(void) { 
  92     DeletePendingObjects(); 
  94     m_initialized 
= wxTopLevelWindows
.GetCount() != 0; 
  98         retval 
= wxApp::MainLoop(); 
  99         wxPythonApp
->OnExit(); 
 105 //--------------------------------------------------------------------- 
 106 //---------------------------------------------------------------------- 
 108 int  WXDLLEXPORT 
wxEntryStart( int argc
, char** argv 
); 
 109 int  WXDLLEXPORT 
wxEntryInitGui(); 
 110 void WXDLLEXPORT 
wxEntryCleanup(); 
 113 #ifdef WXP_WITH_THREAD 
 114 PyThreadState
*  wxPyEventThreadState 
= NULL
; 
 116 static char* __nullArgv
[1] = { 0 }; 
 119 // This is where we pick up the first part of the wxEntry functionality... 
 120 // The rest is in __wxStart and  __wxCleanup.  This function is called when 
 121 // wxcmodule is imported.  (Before there is a wxApp object.) 
 124 #ifdef WXP_WITH_THREAD 
 125     PyEval_InitThreads(); 
 126     wxPyEventThreadState 
= PyThreadState_Get(); 
 129     // Bail out if there is already windows created.  This means that the 
 130     // toolkit has already been initialized, as in embedding wxPython in 
 131     // a C++ wxWindows app. 
 132     if (wxTopLevelWindows
.Number() > 0) 
 136     PyObject
* sysargv 
= PySys_GetObject("argv"); 
 137     int argc 
= PyList_Size(sysargv
); 
 138     char** argv 
= new char*[argc
+1]; 
 140     for(x
=0; x
<argc
; x
++) 
 141         argv
[x
] = PyString_AsString(PyList_GetItem(sysargv
, x
)); 
 144     wxEntryStart(argc
, argv
); 
 150 // Start the user application, user App's OnInit method is a parameter here 
 151 PyObject
* __wxStart(PyObject
* /* self */, PyObject
* args
) 
 153     PyObject
*   onInitFunc 
= NULL
; 
 158     if (!PyArg_ParseTuple(args
, "O", &onInitFunc
)) 
 161     if (wxTopLevelWindows
.Number() > 0) { 
 162         PyErr_SetString(PyExc_TypeError
, "Only 1 wxApp per process!"); 
 167     // This is the next part of the wxEntry functionality... 
 168     PyObject
* sysargv 
= PySys_GetObject("argv"); 
 169     int argc 
= PyList_Size(sysargv
); 
 170     char** argv 
= new char*[argc
+1]; 
 172     for(x
=0; x
<argc
; x
++) 
 173         argv
[x
] = PyString_AsString(PyList_GetItem(sysargv
, x
)); 
 176     wxPythonApp
->argc 
= argc
; 
 177     wxPythonApp
->argv 
= argv
; 
 181     // Call the Python App's OnInit function 
 182     arglist 
= PyTuple_New(0); 
 183     result 
= PyEval_CallObject(onInitFunc
, arglist
); 
 184     if (!result
) {      // an exception was raised. 
 188     if (! PyInt_Check(result
)) { 
 189         PyErr_SetString(PyExc_TypeError
, "OnInit should return a boolean value"); 
 192     bResult 
= PyInt_AS_LONG(result
); 
 194         PyErr_SetString(PyExc_SystemExit
, "OnInit returned FALSE, exiting..."); 
 199     wxTheApp
->m_initialized 
= (wxTopLevelWindows
.GetCount() > 0); 
 212 PyObject
* wxPython_dict
; 
 213 PyObject
* __wxSetDictionary(PyObject
* /* self */, PyObject
* args
) 
 216     if (!PyArg_ParseTuple(args
, "O", &wxPython_dict
)) 
 219     if (!PyDict_Check(wxPython_dict
)) { 
 220         PyErr_SetString(PyExc_TypeError
, "_wxSetDictionary must have dictionary object!"); 
 224 #define wxPlatform "__WXMOTIF__" 
 227 #define wxPlatform "__WXQT__" 
 230 #define wxPlatform "__WXGTK__" 
 232 #if defined(__WIN32__) || defined(__WXMSW__) 
 233 #define wxPlatform "__WXMSW__" 
 236 #define wxPlatform "__WXMAC__" 
 239     PyDict_SetItemString(wxPython_dict
, "wxPlatform", PyString_FromString(wxPlatform
)); 
 246 //--------------------------------------------------------------------------- 
 248 PyObject
* wxPyConstructObject(void* ptr
, const char* className
) { 
 249     char    buff
[64];               // should always be big enough... 
 252     sprintf(buff
, "_%s_p", className
); 
 253     SWIG_MakePtr(swigptr
, ptr
, buff
); 
 255     sprintf(buff
, "%sPtr", className
); 
 256     PyObject
* classobj 
= PyDict_GetItemString(wxPython_dict
, buff
); 
 262     PyObject
* arg 
= Py_BuildValue("(s)", swigptr
); 
 263     PyObject
* obj 
= PyInstance_New(classobj
, arg
, NULL
); 
 269 //--------------------------------------------------------------------------- 
 271 static unsigned int _wxPyNestCount 
= 0; 
 273 static PyThreadState
* myPyThreadState_Get() { 
 274     PyThreadState
* current
; 
 275     current 
= PyThreadState_Swap(NULL
); 
 276     PyThreadState_Swap(current
); 
 281 HELPEREXPORT 
bool wxPyRestoreThread() { 
 282     // NOTE: The Python API docs state that if a thread already has the 
 283     // interpreter lock and calls PyEval_RestoreThread again a deadlock 
 284     // occurs, so I put in this code as a guard condition since there are 
 285     // many possibilites for nested events and callbacks in wxPython.  If 
 286     // The current thread is our thread, then we can assume that we 
 287     // already have the lock.  (I hope!) 
 289 #ifdef WXP_WITH_THREAD 
 291     if (wxPyEventThreadState 
!= myPyThreadState_Get()) { 
 292         PyEval_RestoreThread(wxPyEventThreadState
); 
 301 HELPEREXPORT 
void wxPySaveThread(bool doSave
) { 
 302 #ifdef WXP_WITH_THREAD 
 304         wxPyEventThreadState 
= PyEval_SaveThread(); 
 310 //--------------------------------------------------------------------------- 
 313 IMPLEMENT_ABSTRACT_CLASS(wxPyCallback
, wxObject
); 
 315 wxPyCallback::wxPyCallback(PyObject
* func
) { 
 320 wxPyCallback::wxPyCallback(const wxPyCallback
& other
) { 
 321     m_func 
= other
.m_func
; 
 325 wxPyCallback::~wxPyCallback() { 
 326     bool doSave 
= wxPyRestoreThread(); 
 328     wxPySaveThread(doSave
); 
 333 // This function is used for all events destined for Python event handlers. 
 334 void wxPyCallback::EventThunker(wxEvent
& event
) { 
 335     wxPyCallback
*   cb 
= (wxPyCallback
*)event
.m_callbackUserData
; 
 336     PyObject
*       func 
= cb
->m_func
; 
 342     bool doSave 
= wxPyRestoreThread(); 
 343     wxString className 
= event
.GetClassInfo()->GetClassName(); 
 345     if (className 
== "wxPyEvent") 
 346         arg 
= ((wxPyEvent
*)&event
)->GetSelf(); 
 347     else if (className 
== "wxPyCommandEvent") 
 348         arg 
= ((wxPyCommandEvent
*)&event
)->GetSelf(); 
 350         arg 
= wxPyConstructObject((void*)&event
, className
); 
 352     tuple 
= PyTuple_New(1); 
 353     PyTuple_SET_ITEM(tuple
, 0, arg
); 
 354     result 
= PyEval_CallObject(func
, tuple
); 
 362     wxPySaveThread(doSave
); 
 366 //---------------------------------------------------------------------- 
 368 wxPyCallbackHelper::wxPyCallbackHelper() { 
 376 wxPyCallbackHelper::~wxPyCallbackHelper() { 
 377     bool doSave 
= wxPyRestoreThread(); 
 382     wxPySaveThread(doSave
); 
 385 wxPyCallbackHelper::wxPyCallbackHelper(const wxPyCallbackHelper
& other
) { 
 387       m_self 
= other
.m_self
; 
 388       m_class 
= other
.m_class
; 
 396 void wxPyCallbackHelper::setSelf(PyObject
* self
, PyObject
* _class
, int incref
) { 
 407 // If the object (m_self) has an attibute of the given name, and if that 
 408 // attribute is a method, and if that method's class is not from a base class, 
 409 // then we'll save a pointer to the method so callCallback can call it. 
 410 bool wxPyCallbackHelper::findCallback(const wxString
& name
) const { 
 411     wxPyCallbackHelper
* self 
= (wxPyCallbackHelper
*)this; // cast away const 
 412     self
->m_lastFound 
= NULL
; 
 413     if (m_self 
&& PyObject_HasAttrString(m_self
, (char*)name
.c_str())) { 
 415         method 
= PyObject_GetAttrString(m_self
, (char*)name
.c_str()); 
 417         if (PyMethod_Check(method
) && 
 418             ((PyMethod_GET_CLASS(method
) == m_class
) || 
 419              PyClass_IsSubclass(PyMethod_GET_CLASS(method
), m_class
))) { 
 421             self
->m_lastFound 
= method
; 
 424     return m_lastFound 
!= NULL
; 
 428 int wxPyCallbackHelper::callCallback(PyObject
* argTuple
) const { 
 432     result 
= callCallbackObj(argTuple
); 
 433     if (result
) {                       // Assumes an integer return type... 
 434         retval 
= PyInt_AsLong(result
); 
 436         PyErr_Clear();                  // forget about it if it's not... 
 441 // Invoke the Python callable object, returning the raw PyObject return 
 442 // value.  Caller should DECREF the return value and also call PyEval_SaveThread. 
 443 PyObject
* wxPyCallbackHelper::callCallbackObj(PyObject
* argTuple
) const { 
 446     result 
= PyEval_CallObject(m_lastFound
, argTuple
); 
 456 //--------------------------------------------------------------------------- 
 457 //--------------------------------------------------------------------------- 
 458 // These classes can be derived from in Python and passed through the event 
 459 // system without loosing anything.  They do this by keeping a reference to 
 460 // themselves and some special case handling in wxPyCallback::EventThunker. 
 463 wxPyEvtSelfRef::wxPyEvtSelfRef() { 
 464     //m_self = Py_None;         // **** We don't do normal ref counting to prevent 
 465     //Py_INCREF(m_self);        //      circular loops... 
 469 wxPyEvtSelfRef::~wxPyEvtSelfRef() { 
 470     bool doSave 
= wxPyRestoreThread(); 
 473     wxPySaveThread(doSave
); 
 476 void wxPyEvtSelfRef::SetSelf(PyObject
* self
, bool clone
) { 
 477     bool doSave 
= wxPyRestoreThread(); 
 485     wxPySaveThread(doSave
); 
 488 PyObject
* wxPyEvtSelfRef::GetSelf() const { 
 494 wxPyEvent::wxPyEvent(int id
) 
 498 wxPyEvent::~wxPyEvent() { 
 501 // This one is so the event object can be Cloned... 
 502 void wxPyEvent::CopyObject(wxObject
& dest
) const { 
 503     wxEvent::CopyObject(dest
); 
 504     ((wxPyEvent
*)&dest
)->SetSelf(m_self
, TRUE
); 
 508 IMPLEMENT_DYNAMIC_CLASS(wxPyEvent
, wxEvent
); 
 511 wxPyCommandEvent::wxPyCommandEvent(wxEventType commandType
, int id
) 
 512     : wxCommandEvent(commandType
, id
) { 
 515 wxPyCommandEvent::~wxPyCommandEvent() { 
 518 void wxPyCommandEvent::CopyObject(wxObject
& dest
) const { 
 519     wxCommandEvent::CopyObject(dest
); 
 520     ((wxPyCommandEvent
*)&dest
)->SetSelf(m_self
, TRUE
); 
 524 IMPLEMENT_DYNAMIC_CLASS(wxPyCommandEvent
, wxCommandEvent
); 
 528 //--------------------------------------------------------------------------- 
 529 //--------------------------------------------------------------------------- 
 532 wxPyTimer::wxPyTimer(PyObject
* callback
) { 
 537 wxPyTimer::~wxPyTimer() { 
 538     bool doSave 
= wxPyRestoreThread(); 
 540     wxPySaveThread(doSave
); 
 543 void wxPyTimer::Notify() { 
 544     bool doSave 
= wxPyRestoreThread(); 
 547     PyObject
*   args 
= Py_BuildValue("()"); 
 549     result 
= PyEval_CallObject(func
, args
); 
 557     wxPySaveThread(doSave
); 
 562 //--------------------------------------------------------------------------- 
 563 //--------------------------------------------------------------------------- 
 564 // Convert a wxList to a Python List 
 566 PyObject
* wxPy_ConvertList(wxListBase
* list
, const char* className
) { 
 570     wxNode
*     node 
= list
->First(); 
 572     bool doSave 
= wxPyRestoreThread(); 
 573     pyList 
= PyList_New(0); 
 575         wxObj 
= node
->Data(); 
 576         pyObj 
= wxPyConstructObject(wxObj
, className
); 
 577         PyList_Append(pyList
, pyObj
); 
 580     wxPySaveThread(doSave
); 
 584 //---------------------------------------------------------------------- 
 586 long wxPyGetWinHandle(wxWindow
* win
) { 
 588     return (long)win
->GetHandle(); 
 591     // Find and return the actual X-Window. 
 593     if (win
->m_wxwindow
) { 
 594         GdkWindowPrivate
* bwin 
= (GdkWindowPrivate
*)GTK_PIZZA(win
->m_wxwindow
)->bin_window
; 
 596             return (long)bwin
->xwindow
; 
 603 //---------------------------------------------------------------------- 
 604 // Some helper functions for typemaps in my_typemaps.i, so they won't be 
 605 // included in every file... 
 608 byte
* byte_LIST_helper(PyObject
* source
) { 
 609     if (!PyList_Check(source
)) { 
 610         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 613     int count 
= PyList_Size(source
); 
 614     byte
* temp 
= new byte
[count
]; 
 616         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 619     for (int x
=0; x
<count
; x
++) { 
 620         PyObject
* o 
= PyList_GetItem(source
, x
); 
 621         if (! PyInt_Check(o
)) { 
 622             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
 625         temp
[x
] = (byte
)PyInt_AsLong(o
); 
 631 int* int_LIST_helper(PyObject
* source
) { 
 632     if (!PyList_Check(source
)) { 
 633         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 636     int count 
= PyList_Size(source
); 
 637     int* temp 
= new int[count
]; 
 639         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 642     for (int x
=0; x
<count
; x
++) { 
 643         PyObject
* o 
= PyList_GetItem(source
, x
); 
 644         if (! PyInt_Check(o
)) { 
 645             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
 648         temp
[x
] = PyInt_AsLong(o
); 
 654 long* long_LIST_helper(PyObject
* source
) { 
 655     if (!PyList_Check(source
)) { 
 656         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 659     int count 
= PyList_Size(source
); 
 660     long* temp 
= new long[count
]; 
 662         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 665     for (int x
=0; x
<count
; x
++) { 
 666         PyObject
* o 
= PyList_GetItem(source
, x
); 
 667         if (! PyInt_Check(o
)) { 
 668             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
 671         temp
[x
] = PyInt_AsLong(o
); 
 677 char** string_LIST_helper(PyObject
* source
) { 
 678     if (!PyList_Check(source
)) { 
 679         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 682     int count 
= PyList_Size(source
); 
 683     char** temp 
= new char*[count
]; 
 685         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 688     for (int x
=0; x
<count
; x
++) { 
 689         PyObject
* o 
= PyList_GetItem(source
, x
); 
 690         if (! PyString_Check(o
)) { 
 691             PyErr_SetString(PyExc_TypeError
, "Expected a list of strings."); 
 694         temp
[x
] = PyString_AsString(o
); 
 701 wxPoint
* wxPoint_LIST_helper(PyObject
* source
) { 
 702     if (!PyList_Check(source
)) { 
 703         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 706     int count 
= PyList_Size(source
); 
 707     wxPoint
* temp 
= new wxPoint
[count
]; 
 709         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 712     for (int x
=0; x
<count
; x
++) { 
 713         PyObject
* o 
= PyList_GetItem(source
, x
); 
 714         if (PyTuple_Check(o
)) { 
 715             PyObject
* o1 
= PyTuple_GetItem(o
, 0); 
 716             PyObject
* o2 
= PyTuple_GetItem(o
, 1); 
 718             temp
[x
].x 
= PyInt_AsLong(o1
); 
 719             temp
[x
].y 
= PyInt_AsLong(o2
); 
 721         else if (PyInstance_Check(o
)) { 
 723             if (SWIG_GetPtrObj(o
,(void **) &pt
,"_wxPoint_p")) { 
 724                 PyErr_SetString(PyExc_TypeError
,"Expected _wxPoint_p."); 
 730             PyErr_SetString(PyExc_TypeError
, "Expected a list of 2-tuples or wxPoints."); 
 738 wxBitmap
** wxBitmap_LIST_helper(PyObject
* source
) { 
 739     if (!PyList_Check(source
)) { 
 740         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 743     int count 
= PyList_Size(source
); 
 744     wxBitmap
** temp 
= new wxBitmap
*[count
]; 
 746         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 749     for (int x
=0; x
<count
; x
++) { 
 750         PyObject
* o 
= PyList_GetItem(source
, x
); 
 751         if (PyInstance_Check(o
)) { 
 753             if (SWIG_GetPtrObj(o
, (void **) &pt
,"_wxBitmap_p")) { 
 754                 PyErr_SetString(PyExc_TypeError
,"Expected _wxBitmap_p."); 
 760             PyErr_SetString(PyExc_TypeError
, "Expected a list of wxBitmaps."); 
 769 wxString
* wxString_LIST_helper(PyObject
* source
) { 
 770     if (!PyList_Check(source
)) { 
 771         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 774     int count 
= PyList_Size(source
); 
 775     wxString
* temp 
= new wxString
[count
]; 
 777         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 780     for (int x
=0; x
<count
; x
++) { 
 781         PyObject
* o 
= PyList_GetItem(source
, x
); 
 782         if (! PyString_Check(o
)) { 
 783             PyErr_SetString(PyExc_TypeError
, "Expected a list of strings."); 
 786         temp
[x
] = PyString_AsString(o
); 
 792 wxAcceleratorEntry
* wxAcceleratorEntry_LIST_helper(PyObject
* source
) { 
 793     if (!PyList_Check(source
)) { 
 794         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 797     int count                
= PyList_Size(source
); 
 798     wxAcceleratorEntry
* temp 
= new wxAcceleratorEntry
[count
]; 
 800         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 803     for (int x
=0; x
<count
; x
++) { 
 804         PyObject
* o 
= PyList_GetItem(source
, x
); 
 805         if (PyInstance_Check(o
)) { 
 806             wxAcceleratorEntry
* ae
; 
 807             if (SWIG_GetPtrObj(o
, (void **) &ae
,"_wxAcceleratorEntry_p")) { 
 808                 PyErr_SetString(PyExc_TypeError
,"Expected _wxAcceleratorEntry_p."); 
 813         else if (PyTuple_Check(o
)) { 
 814             PyObject
* o1 
= PyTuple_GetItem(o
, 0); 
 815             PyObject
* o2 
= PyTuple_GetItem(o
, 1); 
 816             PyObject
* o3 
= PyTuple_GetItem(o
, 2); 
 818             temp
[x
].m_flags   
= PyInt_AsLong(o1
); 
 819             temp
[x
].m_keyCode 
= PyInt_AsLong(o2
); 
 820             temp
[x
].m_command 
= PyInt_AsLong(o3
); 
 823             PyErr_SetString(PyExc_TypeError
, "Expected a list of 3-tuples or wxAcceleratorEntry objects."); 
 832 //---------------------------------------------------------------------- 
 834 bool wxSize_helper(PyObject
* source
, wxSize
** obj
) { 
 836     // If source is an object instance then it may already be the right type 
 837     if (PyInstance_Check(source
)) { 
 839         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxSize_p")) 
 844     // otherwise a 2-tuple of integers is expected 
 845     else if (PySequence_Check(source
) && PyObject_Length(source
) == 2) { 
 846         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
 847         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
 848         **obj 
= wxSize(PyInt_AsLong(o1
), PyInt_AsLong(o2
)); 
 853     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of integers or a wxSize object."); 
 857 bool wxPoint_helper(PyObject
* source
, wxPoint
** obj
) { 
 859     // If source is an object instance then it may already be the right type 
 860     if (PyInstance_Check(source
)) { 
 862         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxPoint_p")) 
 867     // otherwise a 2-tuple of integers is expected 
 868     else if (PySequence_Check(source
) && PyObject_Length(source
) == 2) { 
 869         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
 870         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
 871         **obj 
= wxPoint(PyInt_AsLong(o1
), PyInt_AsLong(o2
)); 
 876     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of integers or a wxPoint object."); 
 882 bool wxRealPoint_helper(PyObject
* source
, wxRealPoint
** obj
) { 
 884     // If source is an object instance then it may already be the right type 
 885     if (PyInstance_Check(source
)) { 
 887         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxRealPoint_p")) 
 892     // otherwise a 2-tuple of floats is expected 
 893     else if (PySequence_Check(source
) && PyObject_Length(source
) == 2) { 
 894         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
 895         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
 896         **obj 
= wxRealPoint(PyFloat_AsDouble(o1
), PyFloat_AsDouble(o2
)); 
 901     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of floats or a wxRealPoint object."); 
 908 bool wxRect_helper(PyObject
* source
, wxRect
** obj
) { 
 910     // If source is an object instance then it may already be the right type 
 911     if (PyInstance_Check(source
)) { 
 913         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxRect_p")) 
 918     // otherwise a 4-tuple of integers is expected 
 919     else if (PySequence_Check(source
) && PyObject_Length(source
) == 4) { 
 920         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
 921         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
 922         PyObject
* o3 
= PySequence_GetItem(source
, 2); 
 923         PyObject
* o4 
= PySequence_GetItem(source
, 3); 
 924         **obj 
= wxRect(PyInt_AsLong(o1
), PyInt_AsLong(o2
), 
 925                      PyInt_AsLong(o3
), PyInt_AsLong(o4
)); 
 930     PyErr_SetString(PyExc_TypeError
, "Expected a 4-tuple of integers or a wxRect object."); 
 936 bool wxColour_helper(PyObject
* source
, wxColour
** obj
) { 
 938     // If source is an object instance then it may already be the right type 
 939     if (PyInstance_Check(source
)) { 
 941         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxColour_p")) 
 946     // otherwise a string is expected 
 947     else if (PyString_Check(source
)) { 
 948         wxString spec 
= PyString_AS_STRING(source
); 
 949         if (spec
[0] == '#' && spec
.Length() == 7) {  // It's  #RRGGBB 
 951             int red   
= strtol(spec
.Mid(1,2), &junk
, 16); 
 952             int green 
= strtol(spec
.Mid(3,2), &junk
, 16); 
 953             int blue  
= strtol(spec
.Mid(5,2), &junk
, 16); 
 954             **obj 
= wxColour(red
, green
, blue
); 
 957         else {                                       // it's a colour name 
 958             **obj 
= wxColour(spec
); 
 964     PyErr_SetString(PyExc_TypeError
, "Expected a wxColour object or a string containing a colour name or #RRGGBB."); 
 969 //---------------------------------------------------------------------- 
 970 //---------------------------------------------------------------------- 
 971 //----------------------------------------------------------------------