]>
git.saurik.com Git - wxWidgets.git/blob - wxPython/src/helpers.cpp
   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> 
  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 //---------------------------------------------------------------------- 
  35 int  WXDLLEXPORT 
wxEntryStart( int& argc
, char** argv 
); 
  37 int  WXDLLEXPORT 
wxEntryStart( int argc
, char** argv 
); 
  39 int  WXDLLEXPORT 
wxEntryInitGui(); 
  40 void WXDLLEXPORT 
wxEntryCleanup(); 
  42 wxPyApp
* wxPythonApp 
= NULL
;  // Global instance of application object 
  45 #ifdef WXP_WITH_THREAD 
  46 struct wxPyThreadState 
{ 
  48     PyThreadState
* tstate
; 
  50     wxPyThreadState(unsigned long _tid
=0, PyThreadState
* _tstate
=NULL
) 
  51         : tid(_tid
), tstate(_tstate
) {} 
  54 #include <wx/dynarray.h> 
  55 WX_DECLARE_OBJARRAY(wxPyThreadState
, wxPyThreadStateArray
); 
  56 #include <wx/arrimpl.cpp> 
  57 WX_DEFINE_OBJARRAY(wxPyThreadStateArray
); 
  59 wxPyThreadStateArray
* wxPyTStates 
= NULL
; 
  60 wxMutex
*              wxPyTMutex 
= NULL
; 
  64 #ifdef __WXMSW__             // If building for win32... 
  65 //---------------------------------------------------------------------- 
  66 // This gets run when the DLL is loaded.  We just need to save a handle. 
  67 //---------------------------------------------------------------------- 
  70     HINSTANCE   hinstDLL
,    // handle to DLL module 
  71     DWORD       fdwReason
,   // reason for calling function 
  72     LPVOID      lpvReserved  
// reserved 
  75     wxSetInstance(hinstDLL
); 
  80 //---------------------------------------------------------------------- 
  81 // Classes for implementing the wxp main application shell. 
  82 //---------------------------------------------------------------------- 
  86 //    printf("**** ctor\n"); 
  90 //    printf("**** dtor\n"); 
  94 // This one isn't acutally called...  See __wxStart() 
  95 bool wxPyApp::OnInit() { 
 100 int  wxPyApp::MainLoop() { 
 103     DeletePendingObjects(); 
 104     bool initialized 
= wxTopLevelWindows
.GetCount() != 0; 
 106     m_initialized 
= initialized
; 
 110         retval 
= wxApp::MainLoop(); 
 118 //--------------------------------------------------------------------- 
 119 //---------------------------------------------------------------------- 
 122 // This is where we pick up the first part of the wxEntry functionality... 
 123 // The rest is in __wxStart and  __wxCleanup.  This function is called when 
 124 // wxcmodule is imported.  (Before there is a wxApp object.) 
 129 //    wxCrtSetDbgFlag(_CRTDBG_LEAK_CHECK_DF); 
 132 #ifdef WXP_WITH_THREAD 
 133     PyEval_InitThreads(); 
 134     wxPyTStates 
= new wxPyThreadStateArray
; 
 135     wxPyTMutex 
= new wxMutex
; 
 138     // Bail out if there is already windows created.  This means that the 
 139     // toolkit has already been initialized, as in embedding wxPython in 
 140     // a C++ wxWindows app. 
 141     if (wxTopLevelWindows
.Number() > 0) 
 147     PyObject
* sysargv 
= PySys_GetObject("argv"); 
 148     if (sysargv 
!= NULL
) { 
 149         argc 
= PyList_Size(sysargv
); 
 150         argv 
= new char*[argc
+1]; 
 152         for(x
=0; x
<argc
; x
++) 
 153             argv
[x
] = copystring(PyString_AsString(PyList_GetItem(sysargv
, x
))); 
 157     wxEntryStart(argc
, argv
); 
 163 // Start the user application, user App's OnInit method is a parameter here 
 164 PyObject
* __wxStart(PyObject
* /* self */, PyObject
* args
) 
 166     PyObject
*   onInitFunc 
= NULL
; 
 171     if (!PyArg_ParseTuple(args
, "O", &onInitFunc
)) 
 174 #if 0  // Try it out without this check, see how it does... 
 175     if (wxTopLevelWindows
.Number() > 0) { 
 176         PyErr_SetString(PyExc_TypeError
, "Only 1 wxApp per process!"); 
 181     // This is the next part of the wxEntry functionality... 
 184     PyObject
* sysargv 
= PySys_GetObject("argv"); 
 185     if (sysargv 
!= NULL
) { 
 186         argc 
= PyList_Size(sysargv
); 
 187         argv 
= new char*[argc
+1]; 
 189         for(x
=0; x
<argc
; x
++) 
 190             argv
[x
] = copystring(PyString_AsString(PyList_GetItem(sysargv
, x
))); 
 193     wxPythonApp
->argc 
= argc
; 
 194     wxPythonApp
->argv 
= argv
; 
 198     // Call the Python App's OnInit function 
 199     arglist 
= PyTuple_New(0); 
 200     result 
= PyEval_CallObject(onInitFunc
, arglist
); 
 201     if (!result
) {      // an exception was raised. 
 205     if (! PyInt_Check(result
)) { 
 206         PyErr_SetString(PyExc_TypeError
, "OnInit should return a boolean value"); 
 209     bResult 
= PyInt_AS_LONG(result
); 
 211         PyErr_SetString(PyExc_SystemExit
, "OnInit returned FALSE, exiting..."); 
 216     wxTheApp
->m_initialized 
= (wxTopLevelWindows
.GetCount() > 0); 
 226 #ifdef WXP_WITH_THREAD 
 229     wxPyTStates
->Empty(); 
 237 static PyObject
* wxPython_dict 
= NULL
; 
 238 static PyObject
* wxPyPtrTypeMap 
= NULL
; 
 240 PyObject
* __wxSetDictionary(PyObject
* /* self */, PyObject
* args
) 
 243     if (!PyArg_ParseTuple(args
, "O", &wxPython_dict
)) 
 246     if (!PyDict_Check(wxPython_dict
)) { 
 247         PyErr_SetString(PyExc_TypeError
, "_wxSetDictionary must have dictionary object!"); 
 251     if (! wxPyPtrTypeMap
) 
 252         wxPyPtrTypeMap 
= PyDict_New(); 
 253     PyDict_SetItemString(wxPython_dict
, "__wxPyPtrTypeMap", wxPyPtrTypeMap
); 
 257 #define wxPlatform "__WXMOTIF__" 
 260 #define wxPlatform "__WXQT__" 
 263 #define wxPlatform "__WXGTK__" 
 265 #if defined(__WIN32__) || defined(__WXMSW__) 
 266 #define wxPlatform "__WXMSW__" 
 269 #define wxPlatform "__WXMAC__" 
 272     PyDict_SetItemString(wxPython_dict
, "wxPlatform", PyString_FromString(wxPlatform
)); 
 279 //--------------------------------------------------------------------------- 
 280 // Stuff used by OOR to find the right wxPython class type to return and to 
 284 // The pointer type map is used when the "pointer" type name generated by SWIG 
 285 // is not the same as the shadow class name, for example wxPyTreeCtrl 
 286 // vs. wxTreeCtrl.  It needs to be referenced in Python as well as from C++, 
 287 // so we'll just make it a Python dictionary in the wx module's namespace. 
 288 void wxPyPtrTypeMap_Add(const char* commonName
, const char* ptrName
) { 
 289     if (! wxPyPtrTypeMap
) 
 290         wxPyPtrTypeMap 
= PyDict_New(); 
 291     PyDict_SetItemString(wxPyPtrTypeMap
, 
 293                          PyString_FromString((char*)ptrName
)); 
 298 PyObject
* wxPyClassExists(const char* className
) { 
 303     char    buff
[64];               // should always be big enough... 
 305     sprintf(buff
, "%sPtr", className
); 
 306     PyObject
* classobj 
= PyDict_GetItemString(wxPython_dict
, buff
); 
 308     return classobj
;  // returns NULL if not found 
 312 PyObject
*  wxPyMake_wxObject(wxObject
* source
, bool checkEvtHandler
) { 
 313     PyObject
* target 
= NULL
; 
 314     bool      isEvtHandler 
= FALSE
; 
 317         // If it's derived from wxEvtHandler then there may 
 318         // already be a pointer to a Python object that we can use 
 320         if (checkEvtHandler 
&& wxIsKindOf(source
, wxEvtHandler
)) { 
 322             wxEvtHandler
* eh 
= (wxEvtHandler
*)source
; 
 323             wxPyClientData
* data 
= (wxPyClientData
*)eh
->GetClientObject(); 
 325                 target 
= data
->m_obj
; 
 331             // Otherwise make it the old fashioned way by making a 
 332             // new shadow object and putting this pointer in it. 
 333             wxClassInfo
* info 
= source
->GetClassInfo(); 
 334             wxChar
*      name 
= (wxChar
*)info
->GetClassName(); 
 335             PyObject
*    klass 
= wxPyClassExists(name
); 
 336             while (info 
&& !klass
) { 
 337                 name 
= (wxChar
*)info
->GetBaseClassName1(); 
 338                 info 
= wxClassInfo::FindClass(name
); 
 339                 klass 
= wxPyClassExists(name
); 
 342                 target 
= wxPyConstructObject(source
, name
, klass
, FALSE
); 
 343                 if (target 
&& isEvtHandler
) 
 344                     ((wxEvtHandler
*)source
)->SetClientObject(new wxPyClientData(target
)); 
 346                 wxString 
msg("wxPython class not found for "); 
 347                 msg 
+= source
->GetClassInfo()->GetClassName(); 
 348                 PyErr_SetString(PyExc_NameError
, msg
.c_str()); 
 352     } else {  // source was NULL so return None. 
 353         Py_INCREF(Py_None
); target 
= Py_None
; 
 359 PyObject
*  wxPyMake_wxSizer(wxSizer
* source
) { 
 360     PyObject
* target 
= NULL
; 
 362     if (source 
&& wxIsKindOf(source
, wxSizer
)) { 
 363         // If it's derived from wxSizer then there may 
 364         // already be a pointer to a Python object that we can use 
 366         wxSizer
* sz 
= (wxSizer
*)source
; 
 367         wxPyClientData
* data 
= (wxPyClientData
*)sz
->GetClientObject(); 
 369             target 
= data
->m_obj
; 
 374         target 
= wxPyMake_wxObject(source
, FALSE
); 
 375         if (target 
!= Py_None
) 
 376             ((wxSizer
*)source
)->SetClientObject(new wxPyClientData(target
)); 
 383 //--------------------------------------------------------------------------- 
 385 PyObject
* wxPyConstructObject(void* ptr
, 
 386                               const char* className
, 
 393     char      swigptr
[64];      // should always be big enough... 
 396     if ((item 
= PyDict_GetItemString(wxPyPtrTypeMap
, (char*)className
)) != NULL
) { 
 397         className 
= PyString_AsString(item
); 
 399     sprintf(buff
, "_%s_p", className
); 
 400     SWIG_MakePtr(swigptr
, ptr
, buff
); 
 402     arg 
= Py_BuildValue("(s)", swigptr
); 
 403     obj 
= PyInstance_New(klass
, arg
, NULL
); 
 407         PyObject
* one 
= PyInt_FromLong(1); 
 408         PyObject_SetAttrString(obj
, "thisown", one
); 
 416 PyObject
* wxPyConstructObject(void* ptr
, 
 417                               const char* className
, 
 426     char    buff
[64];               // should always be big enough... 
 427     sprintf(buff
, "%sPtr", className
); 
 429     wxASSERT_MSG(wxPython_dict
, "wxPython_dict is not set yet!!"); 
 431     PyObject
* classobj 
= PyDict_GetItemString(wxPython_dict
, buff
); 
 435                 "*** Unknown class name %s, tell Robin about it please ***", 
 437         obj 
= PyString_FromString(temp
); 
 441     return wxPyConstructObject(ptr
, className
, classobj
, setThisOwn
); 
 444 //--------------------------------------------------------------------------- 
 447 #ifdef WXP_WITH_THREAD 
 449 unsigned long wxPyGetCurrentThreadId() { 
 450     return wxThread::GetCurrentId(); 
 455 PyThreadState
* wxPyGetThreadState() { 
 456     unsigned long ctid 
= wxPyGetCurrentThreadId(); 
 457     PyThreadState
* tstate 
= NULL
; 
 460     for(size_t i
=0; i 
< wxPyTStates
->GetCount(); i
++) { 
 461         wxPyThreadState
& info 
= wxPyTStates
->Item(i
); 
 462         if (info
.tid 
== ctid
) { 
 463             tstate 
= info
.tstate
; 
 467     wxPyTMutex
->Unlock(); 
 468     wxASSERT_MSG(tstate
, "PyThreadState should not be NULL!"); 
 473 void wxPySaveThreadState(PyThreadState
* tstate
) { 
 474     unsigned long ctid 
= wxPyGetCurrentThreadId(); 
 476     for(size_t i
=0; i 
< wxPyTStates
->GetCount(); i
++) { 
 477         wxPyThreadState
& info 
= wxPyTStates
->Item(i
); 
 478         if (info
.tid 
== ctid
) { 
 479             info
.tstate 
= tstate
; 
 480             wxPyTMutex
->Unlock(); 
 484     // not found, so add it... 
 485     wxPyTStates
->Add(new wxPyThreadState(ctid
, tstate
)); 
 486     wxPyTMutex
->Unlock(); 
 492 // Calls from Python to wxWindows code are wrapped in calls to these 
 495 PyThreadState
* wxPyBeginAllowThreads() { 
 496 #ifdef WXP_WITH_THREAD 
 497     PyThreadState
* saved 
= PyEval_SaveThread();  // Py_BEGIN_ALLOW_THREADS; 
 498     wxPySaveThreadState(saved
); 
 505 void wxPyEndAllowThreads(PyThreadState
* saved
) { 
 506 #ifdef WXP_WITH_THREAD 
 507     PyEval_RestoreThread(saved
);   // Py_END_ALLOW_THREADS; 
 513 // Calls from wxWindows back to Python code, or even any PyObject 
 514 // manipulations, PyDECREF's and etc. are wrapped in calls to these functions: 
 516 void wxPyBeginBlockThreads() { 
 517 #ifdef WXP_WITH_THREAD 
 518     PyThreadState
* tstate 
= wxPyGetThreadState(); 
 519     PyEval_RestoreThread(tstate
); 
 524 void wxPyEndBlockThreads() { 
 525 #ifdef WXP_WITH_THREAD 
 526     PyThreadState
* tstate 
= PyEval_SaveThread(); 
 527     // Is there any need to save it again? 
 532 //--------------------------------------------------------------------------- 
 534 IMPLEMENT_ABSTRACT_CLASS(wxPyCallback
, wxObject
); 
 536 wxPyCallback::wxPyCallback(PyObject
* func
) { 
 541 wxPyCallback::wxPyCallback(const wxPyCallback
& other
) { 
 542     m_func 
= other
.m_func
; 
 546 wxPyCallback::~wxPyCallback() { 
 547     wxPyBeginBlockThreads(); 
 549     wxPyEndBlockThreads(); 
 554 // This function is used for all events destined for Python event handlers. 
 555 void wxPyCallback::EventThunker(wxEvent
& event
) { 
 556     wxPyCallback
*   cb 
= (wxPyCallback
*)event
.m_callbackUserData
; 
 557     PyObject
*       func 
= cb
->m_func
; 
 563     wxPyBeginBlockThreads(); 
 564     wxString className 
= event
.GetClassInfo()->GetClassName(); 
 566     if (className 
== "wxPyEvent") 
 567         arg 
= ((wxPyEvent
*)&event
)->GetSelf(); 
 568     else if (className 
== "wxPyCommandEvent") 
 569         arg 
= ((wxPyCommandEvent
*)&event
)->GetSelf(); 
 571         arg 
= wxPyConstructObject((void*)&event
, className
); 
 573     tuple 
= PyTuple_New(1); 
 574     PyTuple_SET_ITEM(tuple
, 0, arg
); 
 575     result 
= PyEval_CallObject(func
, tuple
); 
 579         PyErr_Clear();       // Just in case... 
 583     wxPyEndBlockThreads(); 
 587 //---------------------------------------------------------------------- 
 589 wxPyCallbackHelper::wxPyCallbackHelper(const wxPyCallbackHelper
& other
) { 
 591       m_self 
= other
.m_self
; 
 592       m_class 
= other
.m_class
; 
 600 void wxPyCallbackHelper::setSelf(PyObject
* self
, PyObject
* klass
, int incref
) { 
 611 #if PYTHON_API_VERSION >= 1011 
 613 // Prior to Python 2.2 PyMethod_GetClass returned the class object 
 614 // in which the method was defined.  Starting with 2.2 it returns 
 615 // "class that asked for the method" which seems totally bogus to me 
 616 // but apprently if fixes some obscure problem waiting to happen in 
 617 // Python.  Since the API was not documented Guido and the gang felt 
 618 // safe in changing it.  Needless to say that totally screwed up the 
 619 // logic below in wxPyCallbackHelper::findCallback, hence this icky 
 620 // code to find the class where the method is actuallt defined... 
 623 PyObject
* PyFindClassWithAttr(PyObject 
*klass
, PyObject 
*name
) 
 627     if (PyType_Check(klass
)) {      // new style classes 
 628         // This code is borrowed/adapted from _PyType_Lookup in typeobject.c 
 629         // (TODO: This part is not tested yet, so I'm not sure it is correct...) 
 630         PyTypeObject
* type 
= (PyTypeObject
*)klass
; 
 631         PyObject 
*mro
, *res
, *base
, *dict
; 
 632         /* Look in tp_dict of types in MRO */ 
 634         assert(PyTuple_Check(mro
)); 
 635         n 
= PyTuple_GET_SIZE(mro
); 
 636         for (i 
= 0; i 
< n
; i
++) { 
 637             base 
= PyTuple_GET_ITEM(mro
, i
); 
 638             if (PyClass_Check(base
)) 
 639                 dict 
= ((PyClassObject 
*)base
)->cl_dict
; 
 641                 assert(PyType_Check(base
)); 
 642                 dict 
= ((PyTypeObject 
*)base
)->tp_dict
; 
 644             assert(dict 
&& PyDict_Check(dict
)); 
 645             res 
= PyDict_GetItem(dict
, name
); 
 652     else if (PyClass_Check(klass
)) { // old style classes 
 653         // This code is borrowed/adapted from class_lookup in classobject.c 
 654         PyClassObject
* cp 
= (PyClassObject
*)klass
; 
 655         PyObject 
*value 
= PyDict_GetItem(cp
->cl_dict
, name
); 
 657             return (PyObject
*)cp
; 
 659         n 
= PyTuple_Size(cp
->cl_bases
); 
 660         for (i 
= 0; i 
< n
; i
++) { 
 661             PyObject
* base 
= PyTuple_GetItem(cp
->cl_bases
, i
); 
 662             PyObject 
*v 
= PyFindClassWithAttr(base
, name
); 
 673 PyObject
* PyMethod_GetDefiningClass(PyObject
* method
, const char* name
) 
 675     PyObject
* mgc 
= PyMethod_GET_CLASS(method
); 
 677 #if PYTHON_API_VERSION <= 1010    // prior to Python 2.2, the easy way 
 679 #else                             // 2.2 and after, the hard way... 
 681     PyObject
* nameo 
= PyString_FromString(name
); 
 682     PyObject
* klass 
= PyFindClassWithAttr(mgc
, nameo
); 
 690 bool wxPyCallbackHelper::findCallback(const char* name
) const { 
 691     wxPyCallbackHelper
* self 
= (wxPyCallbackHelper
*)this; // cast away const 
 692     self
->m_lastFound 
= NULL
; 
 694     // If the object (m_self) has an attibute of the given name... 
 695     if (m_self 
&& PyObject_HasAttrString(m_self
, (char*)name
)) { 
 696         PyObject 
*method
, *klass
; 
 697         method 
= PyObject_GetAttrString(m_self
, (char*)name
); 
 699         // ...and if that attribute is a method, and if that method's class is 
 700         // not from a base class... 
 701         if (PyMethod_Check(method
) && 
 702             (klass 
= PyMethod_GetDefiningClass(method
, (char*)name
)) != NULL 
&& 
 703             ((klass 
== m_class
) || PyClass_IsSubclass(klass
, m_class
))) { 
 705             // ...then we'll save a pointer to the method so callCallback can call it. 
 706             self
->m_lastFound 
= method
; 
 712     return m_lastFound 
!= NULL
; 
 716 int wxPyCallbackHelper::callCallback(PyObject
* argTuple
) const { 
 720     result 
= callCallbackObj(argTuple
); 
 721     if (result
) {                       // Assumes an integer return type... 
 722         retval 
= PyInt_AsLong(result
); 
 724         PyErr_Clear();                  // forget about it if it's not... 
 729 // Invoke the Python callable object, returning the raw PyObject return 
 730 // value.  Caller should DECREF the return value and also call PyEval_SaveThread. 
 731 PyObject
* wxPyCallbackHelper::callCallbackObj(PyObject
* argTuple
) const { 
 734     // Save a copy of the pointer in case the callback generates another 
 735     // callback.  In that case m_lastFound will have a different value when 
 736     // it gets back here... 
 737     PyObject
* method 
= m_lastFound
; 
 739     result 
= PyEval_CallObject(method
, argTuple
); 
 749 void wxPyCBH_setCallbackInfo(wxPyCallbackHelper
& cbh
, PyObject
* self
, PyObject
* klass
, int incref
) { 
 750     cbh
.setSelf(self
, klass
, incref
); 
 753 bool wxPyCBH_findCallback(const wxPyCallbackHelper
& cbh
, const char* name
) { 
 754     return cbh
.findCallback(name
); 
 757 int  wxPyCBH_callCallback(const wxPyCallbackHelper
& cbh
, PyObject
* argTuple
) { 
 758     return cbh
.callCallback(argTuple
); 
 761 PyObject
* wxPyCBH_callCallbackObj(const wxPyCallbackHelper
& cbh
, PyObject
* argTuple
) { 
 762     return cbh
.callCallbackObj(argTuple
); 
 766 void wxPyCBH_delete(wxPyCallbackHelper
* cbh
) { 
 768         wxPyBeginBlockThreads(); 
 769         Py_XDECREF(cbh
->m_self
); 
 770         Py_XDECREF(cbh
->m_class
); 
 771         wxPyEndBlockThreads(); 
 775 //--------------------------------------------------------------------------- 
 776 //--------------------------------------------------------------------------- 
 777 // These event classes can be derived from in Python and passed through the event 
 778 // system without losing anything.  They do this by keeping a reference to 
 779 // themselves and some special case handling in wxPyCallback::EventThunker. 
 782 wxPyEvtSelfRef::wxPyEvtSelfRef() { 
 783     //m_self = Py_None;         // **** We don't do normal ref counting to prevent 
 784     //Py_INCREF(m_self);        //      circular loops... 
 788 wxPyEvtSelfRef::~wxPyEvtSelfRef() { 
 789     wxPyBeginBlockThreads(); 
 792     wxPyEndBlockThreads(); 
 795 void wxPyEvtSelfRef::SetSelf(PyObject
* self
, bool clone
) { 
 796     wxPyBeginBlockThreads(); 
 804     wxPyEndBlockThreads(); 
 807 PyObject
* wxPyEvtSelfRef::GetSelf() const { 
 813 IMPLEMENT_ABSTRACT_CLASS(wxPyEvent
, wxEvent
); 
 814 IMPLEMENT_ABSTRACT_CLASS(wxPyCommandEvent
, wxCommandEvent
); 
 817 wxPyEvent::wxPyEvent(int id
) 
 822 wxPyEvent::wxPyEvent(const wxPyEvent
& evt
) 
 825     SetSelf(evt
.m_self
, TRUE
); 
 829 wxPyEvent::~wxPyEvent() { 
 833 wxPyCommandEvent::wxPyCommandEvent(wxEventType commandType
, int id
) 
 834     : wxCommandEvent(commandType
, id
) { 
 838 wxPyCommandEvent::wxPyCommandEvent(const wxPyCommandEvent
& evt
) 
 839     : wxCommandEvent(evt
) 
 841     SetSelf(evt
.m_self
, TRUE
); 
 845 wxPyCommandEvent::~wxPyCommandEvent() { 
 851 //--------------------------------------------------------------------------- 
 852 //--------------------------------------------------------------------------- 
 855 wxPyTimer::wxPyTimer(PyObject
* callback
) { 
 860 wxPyTimer::~wxPyTimer() { 
 861     wxPyBeginBlockThreads(); 
 863     wxPyEndBlockThreads(); 
 866 void wxPyTimer::Notify() { 
 867     if (!func 
|| func 
== Py_None
) { 
 871         wxPyBeginBlockThreads(); 
 874         PyObject
*   args 
= Py_BuildValue("()"); 
 876         result 
= PyEval_CallObject(func
, args
); 
 885         wxPyEndBlockThreads(); 
 891 //--------------------------------------------------------------------------- 
 892 //--------------------------------------------------------------------------- 
 893 // Convert a wxList to a Python List 
 895 PyObject
* wxPy_ConvertList(wxListBase
* list
, const char* className
) { 
 899     wxNode
*     node 
= list
->First(); 
 901     wxPyBeginBlockThreads(); 
 902     pyList 
= PyList_New(0); 
 904         wxObj 
= node
->Data(); 
 905         pyObj 
= wxPyMake_wxObject(wxObj
); //wxPyConstructObject(wxObj, className); 
 906         PyList_Append(pyList
, pyObj
); 
 909     wxPyEndBlockThreads(); 
 913 //---------------------------------------------------------------------- 
 915 long wxPyGetWinHandle(wxWindow
* win
) { 
 917     return (long)win
->GetHandle(); 
 920     // Find and return the actual X-Window. 
 922     if (win
->m_wxwindow
) { 
 923         GdkWindowPrivate
* bwin 
= (GdkWindowPrivate
*)GTK_PIZZA(win
->m_wxwindow
)->bin_window
; 
 925             return (long)bwin
->xwindow
; 
 932 //---------------------------------------------------------------------- 
 933 // Some helper functions for typemaps in my_typemaps.i, so they won't be 
 934 // included in every file... 
 937 byte
* byte_LIST_helper(PyObject
* source
) { 
 938     if (!PyList_Check(source
)) { 
 939         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 942     int count 
= PyList_Size(source
); 
 943     byte
* temp 
= new byte
[count
]; 
 945         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 948     for (int x
=0; x
<count
; x
++) { 
 949         PyObject
* o 
= PyList_GetItem(source
, x
); 
 950         if (! PyInt_Check(o
)) { 
 951             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
 954         temp
[x
] = (byte
)PyInt_AsLong(o
); 
 960 int* int_LIST_helper(PyObject
* source
) { 
 961     if (!PyList_Check(source
)) { 
 962         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 965     int count 
= PyList_Size(source
); 
 966     int* temp 
= new int[count
]; 
 968         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 971     for (int x
=0; x
<count
; x
++) { 
 972         PyObject
* o 
= PyList_GetItem(source
, x
); 
 973         if (! PyInt_Check(o
)) { 
 974             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
 977         temp
[x
] = PyInt_AsLong(o
); 
 983 long* long_LIST_helper(PyObject
* source
) { 
 984     if (!PyList_Check(source
)) { 
 985         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
 988     int count 
= PyList_Size(source
); 
 989     long* temp 
= new long[count
]; 
 991         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
 994     for (int x
=0; x
<count
; x
++) { 
 995         PyObject
* o 
= PyList_GetItem(source
, x
); 
 996         if (! PyInt_Check(o
)) { 
 997             PyErr_SetString(PyExc_TypeError
, "Expected a list of integers."); 
1000         temp
[x
] = PyInt_AsLong(o
); 
1006 char** string_LIST_helper(PyObject
* source
) { 
1007     if (!PyList_Check(source
)) { 
1008         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1011     int count 
= PyList_Size(source
); 
1012     char** temp 
= new char*[count
]; 
1014         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1017     for (int x
=0; x
<count
; x
++) { 
1018         PyObject
* o 
= PyList_GetItem(source
, x
); 
1019         if (! PyString_Check(o
)) { 
1020             PyErr_SetString(PyExc_TypeError
, "Expected a list of strings."); 
1023         temp
[x
] = PyString_AsString(o
); 
1028 //-------------------------------- 
1029 // Part of patch from Tim Hochberg 
1030 static inline bool wxPointFromObjects(PyObject
* o1
, PyObject
* o2
, wxPoint
* point
) { 
1031     if (PyInt_Check(o1
) && PyInt_Check(o2
)) { 
1032         point
->x 
= PyInt_AS_LONG(o1
); 
1033         point
->y 
= PyInt_AS_LONG(o2
); 
1036     if (PyFloat_Check(o1
) && PyFloat_Check(o2
)) { 
1037         point
->x 
= (int)PyFloat_AS_DOUBLE(o1
); 
1038         point
->y 
= (int)PyFloat_AS_DOUBLE(o2
); 
1041     if (PyInstance_Check(o1
) || PyInstance_Check(o2
)) { 
1042         // Disallow instances because they can cause havok 
1045     if (PyNumber_Check(o1
) && PyNumber_Check(o2
)) { 
1046         // I believe this excludes instances, so this should be safe without INCREFFing o1 and o2 
1047         point
->x 
= PyInt_AsLong(o1
); 
1048         point
->y 
= PyInt_AsLong(o2
); 
1055 wxPoint
* wxPoint_LIST_helper(PyObject
* source
, int *count
) { 
1056     // Putting all of the declarations here allows 
1057     // us to put the error handling all in one place. 
1060     PyObject 
*o
, *o1
, *o2
; 
1061     bool isFast 
= PyList_Check(source
) || PyTuple_Check(source
); 
1063     if (!PySequence_Check(source
)) { 
1067     // The length of the sequence is returned in count. 
1068     *count 
= PySequence_Length(source
); 
1073     temp 
= new wxPoint
[*count
]; 
1075         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1078     for (x
=0; x
<*count
; x
++) { 
1079         // Get an item: try fast way first. 
1081             o 
= PySequence_Fast_GET_ITEM(source
, x
); 
1084             o 
= PySequence_GetItem(source
, x
); 
1090         // Convert o to wxPoint. 
1091         if ((PyTuple_Check(o
) && PyTuple_GET_SIZE(o
) == 2) || 
1092             (PyList_Check(o
) && PyList_GET_SIZE(o
) == 2)) { 
1093             o1 
= PySequence_Fast_GET_ITEM(o
, 0); 
1094             o2 
= PySequence_Fast_GET_ITEM(o
, 1); 
1095             if (!wxPointFromObjects(o1
, o2
, &temp
[x
])) { 
1099         else if (PyInstance_Check(o
)) { 
1101             if (SWIG_GetPtrObj(o
, (void **)&pt
, "_wxPoint_p")) { 
1106         else if (PySequence_Check(o
) && PySequence_Length(o
) == 2) { 
1107             o1 
= PySequence_GetItem(o
, 0); 
1108             o2 
= PySequence_GetItem(o
, 1); 
1109             if (!wxPointFromObjects(o1
, o2
, &temp
[x
])) { 
1133     PyErr_SetString(PyExc_TypeError
, "Expected a sequence of length-2 sequences or wxPoints."); 
1137 //------------------------------ 
1140 wxBitmap
** wxBitmap_LIST_helper(PyObject
* source
) { 
1141     if (!PyList_Check(source
)) { 
1142         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1145     int count 
= PyList_Size(source
); 
1146     wxBitmap
** temp 
= new wxBitmap
*[count
]; 
1148         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1151     for (int x
=0; x
<count
; x
++) { 
1152         PyObject
* o 
= PyList_GetItem(source
, x
); 
1153         if (PyInstance_Check(o
)) { 
1155             if (SWIG_GetPtrObj(o
, (void **) &pt
,"_wxBitmap_p")) { 
1156                 PyErr_SetString(PyExc_TypeError
,"Expected _wxBitmap_p."); 
1162             PyErr_SetString(PyExc_TypeError
, "Expected a list of wxBitmaps."); 
1171 wxString
* wxString_LIST_helper(PyObject
* source
) { 
1172     if (!PyList_Check(source
)) { 
1173         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1176     int count 
= PyList_Size(source
); 
1177     wxString
* temp 
= new wxString
[count
]; 
1179         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1182     for (int x
=0; x
<count
; x
++) { 
1183         PyObject
* o 
= PyList_GetItem(source
, x
); 
1184 #if PYTHON_API_VERSION >= 1009 
1185         if (! PyString_Check(o
) && ! PyUnicode_Check(o
)) { 
1186             PyErr_SetString(PyExc_TypeError
, "Expected a list of string or unicode objects."); 
1192         if (PyString_AsStringAndSize(o
, &buff
, &length
) == -1) 
1194         temp
[x
] = wxString(buff
, length
); 
1196         if (! PyString_Check(o
)) { 
1197             PyErr_SetString(PyExc_TypeError
, "Expected a list of strings."); 
1200         temp
[x
] = PyString_AsString(o
); 
1207 wxAcceleratorEntry
* wxAcceleratorEntry_LIST_helper(PyObject
* source
) { 
1208     if (!PyList_Check(source
)) { 
1209         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1212     int count                
= PyList_Size(source
); 
1213     wxAcceleratorEntry
* temp 
= new wxAcceleratorEntry
[count
]; 
1215         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1218     for (int x
=0; x
<count
; x
++) { 
1219         PyObject
* o 
= PyList_GetItem(source
, x
); 
1220         if (PyInstance_Check(o
)) { 
1221             wxAcceleratorEntry
* ae
; 
1222             if (SWIG_GetPtrObj(o
, (void **) &ae
,"_wxAcceleratorEntry_p")) { 
1223                 PyErr_SetString(PyExc_TypeError
,"Expected _wxAcceleratorEntry_p."); 
1228         else if (PyTuple_Check(o
)) { 
1229             PyObject
* o1 
= PyTuple_GetItem(o
, 0); 
1230             PyObject
* o2 
= PyTuple_GetItem(o
, 1); 
1231             PyObject
* o3 
= PyTuple_GetItem(o
, 2); 
1232             temp
[x
].Set(PyInt_AsLong(o1
), PyInt_AsLong(o2
), PyInt_AsLong(o3
)); 
1235             PyErr_SetString(PyExc_TypeError
, "Expected a list of 3-tuples or wxAcceleratorEntry objects."); 
1243 wxPen
** wxPen_LIST_helper(PyObject
* source
) { 
1244     if (!PyList_Check(source
)) { 
1245         PyErr_SetString(PyExc_TypeError
, "Expected a list object."); 
1248     int count 
= PyList_Size(source
); 
1249     wxPen
** temp 
= new wxPen
*[count
]; 
1251         PyErr_SetString(PyExc_MemoryError
, "Unable to allocate temporary array"); 
1254     for (int x
=0; x
<count
; x
++) { 
1255         PyObject
* o 
= PyList_GetItem(source
, x
); 
1256         if (PyInstance_Check(o
)) { 
1258             if (SWIG_GetPtrObj(o
, (void **) &pt
,"_wxPen_p")) { 
1260                 PyErr_SetString(PyExc_TypeError
,"Expected _wxPen_p."); 
1267             PyErr_SetString(PyExc_TypeError
, "Expected a list of wxPens."); 
1275 bool _2int_seq_helper(PyObject
* source
, int* i1
, int* i2
) { 
1276     bool isFast 
= PyList_Check(source
) || PyTuple_Check(source
); 
1279     if (!PySequence_Check(source
) || PySequence_Length(source
) != 2) 
1283         o1 
= PySequence_Fast_GET_ITEM(source
, 0); 
1284         o2 
= PySequence_Fast_GET_ITEM(source
, 1); 
1287         o1 
= PySequence_GetItem(source
, 0); 
1288         o2 
= PySequence_GetItem(source
, 1); 
1291     *i1 
= PyInt_AsLong(o1
); 
1292     *i2 
= PyInt_AsLong(o2
); 
1302 bool _4int_seq_helper(PyObject
* source
, int* i1
, int* i2
, int* i3
, int* i4
) { 
1303     bool isFast 
= PyList_Check(source
) || PyTuple_Check(source
); 
1304     PyObject 
*o1
, *o2
, *o3
, *o4
; 
1306     if (!PySequence_Check(source
) || PySequence_Length(source
) != 4) 
1310         o1 
= PySequence_Fast_GET_ITEM(source
, 0); 
1311         o2 
= PySequence_Fast_GET_ITEM(source
, 1); 
1312         o3 
= PySequence_Fast_GET_ITEM(source
, 2); 
1313         o4 
= PySequence_Fast_GET_ITEM(source
, 3); 
1316         o1 
= PySequence_GetItem(source
, 0); 
1317         o2 
= PySequence_GetItem(source
, 1); 
1318         o3 
= PySequence_GetItem(source
, 2); 
1319         o4 
= PySequence_GetItem(source
, 3); 
1322     *i1 
= PyInt_AsLong(o1
); 
1323     *i2 
= PyInt_AsLong(o2
); 
1324     *i3 
= PyInt_AsLong(o3
); 
1325     *i4 
= PyInt_AsLong(o4
); 
1337 //---------------------------------------------------------------------- 
1339 bool wxSize_helper(PyObject
* source
, wxSize
** obj
) { 
1341     // If source is an object instance then it may already be the right type 
1342     if (PyInstance_Check(source
)) { 
1344         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxSize_p")) 
1349     // otherwise a 2-tuple of integers is expected 
1350     else if (PySequence_Check(source
) && PyObject_Length(source
) == 2) { 
1351         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
1352         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
1353         **obj 
= wxSize(PyInt_AsLong(o1
), PyInt_AsLong(o2
)); 
1358     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of integers or a wxSize object."); 
1362 bool wxPoint_helper(PyObject
* source
, wxPoint
** obj
) { 
1364     // If source is an object instance then it may already be the right type 
1365     if (PyInstance_Check(source
)) { 
1367         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxPoint_p")) 
1372     // otherwise a length-2 sequence of integers is expected 
1373     if (PySequence_Check(source
) && PySequence_Length(source
) == 2) { 
1374         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
1375         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
1376                 // This should really check for integers, not numbers -- but that would break code. 
1377                 if (!PyNumber_Check(o1
) || !PyNumber_Check(o2
)) { 
1382                 **obj 
= wxPoint(PyInt_AsLong(o1
), PyInt_AsLong(o2
)); 
1388     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of integers or a wxPoint object."); 
1394 bool wxRealPoint_helper(PyObject
* source
, wxRealPoint
** obj
) { 
1396     // If source is an object instance then it may already be the right type 
1397     if (PyInstance_Check(source
)) { 
1399         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxRealPoint_p")) 
1404     // otherwise a 2-tuple of floats is expected 
1405     else if (PySequence_Check(source
) && PyObject_Length(source
) == 2) { 
1406         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
1407         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
1408         **obj 
= wxRealPoint(PyFloat_AsDouble(o1
), PyFloat_AsDouble(o2
)); 
1413     PyErr_SetString(PyExc_TypeError
, "Expected a 2-tuple of floats or a wxRealPoint object."); 
1420 bool wxRect_helper(PyObject
* source
, wxRect
** obj
) { 
1422     // If source is an object instance then it may already be the right type 
1423     if (PyInstance_Check(source
)) { 
1425         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxRect_p")) 
1430     // otherwise a 4-tuple of integers is expected 
1431     else if (PySequence_Check(source
) && PyObject_Length(source
) == 4) { 
1432         PyObject
* o1 
= PySequence_GetItem(source
, 0); 
1433         PyObject
* o2 
= PySequence_GetItem(source
, 1); 
1434         PyObject
* o3 
= PySequence_GetItem(source
, 2); 
1435         PyObject
* o4 
= PySequence_GetItem(source
, 3); 
1436         **obj 
= wxRect(PyInt_AsLong(o1
), PyInt_AsLong(o2
), 
1437                      PyInt_AsLong(o3
), PyInt_AsLong(o4
)); 
1442     PyErr_SetString(PyExc_TypeError
, "Expected a 4-tuple of integers or a wxRect object."); 
1448 bool wxColour_helper(PyObject
* source
, wxColour
** obj
) { 
1450     // If source is an object instance then it may already be the right type 
1451     if (PyInstance_Check(source
)) { 
1453         if (SWIG_GetPtrObj(source
, (void **)&ptr
, "_wxColour_p")) 
1458     // otherwise a string is expected 
1459     else if (PyString_Check(source
)) { 
1460         wxString spec 
= PyString_AS_STRING(source
); 
1461         if (spec
[0U] == '#' && spec
.Length() == 7) {  // It's  #RRGGBB 
1463             int red   
= strtol(spec
.Mid(1,2), &junk
, 16); 
1464             int green 
= strtol(spec
.Mid(3,2), &junk
, 16); 
1465             int blue  
= strtol(spec
.Mid(5,2), &junk
, 16); 
1466             **obj 
= wxColour(red
, green
, blue
); 
1469         else {                                       // it's a colour name 
1470             **obj 
= wxColour(spec
); 
1476     PyErr_SetString(PyExc_TypeError
, "Expected a wxColour object or a string containing a colour name or '#RRGGBB'."); 
1481 //---------------------------------------------------------------------- 
1483 PyObject
* wxArrayString2PyList_helper(const wxArrayString
& arr
) { 
1485     PyObject
* list 
= PyList_New(0); 
1486     for (size_t i
=0; i 
< arr
.GetCount(); i
++) { 
1487         PyObject
* str 
= PyString_FromString(arr
[i
].c_str()); 
1488         PyList_Append(list
, str
); 
1495 PyObject
* wxArrayInt2PyList_helper(const wxArrayInt
& arr
) { 
1497     PyObject
* list 
= PyList_New(0); 
1498     for (size_t i
=0; i 
< arr
.GetCount(); i
++) { 
1499         PyObject
* number 
= PyInt_FromLong(arr
[i
]); 
1500         PyList_Append(list
, number
); 
1507 //---------------------------------------------------------------------- 
1508 //----------------------------------------------------------------------