]> git.saurik.com Git - wxWidgets.git/blame_incremental - wxPython/samples/embedded/embedded.cpp
Updated version number
[wxWidgets.git] / wxPython / samples / embedded / embedded.cpp
... / ...
CommitLineData
1//----------------------------------------------------------------------
2// Name: embedded.cpp
3// Purpose: To serve as an example of how to use wxPython from
4// within a C++ wxWindows program.
5//
6// Author: Robin Dunn
7//
8// Created: 1-May-2002
9// RCS-ID: $Id$
10// Copyright: (c) 2002 by Total Control Software
11// Licence: wxWindows license
12//----------------------------------------------------------------------
13
14// For compilers that support precompilation, includes "wx/wx.h".
15#include <wx/wxprec.h>
16
17#ifdef __BORLANDC__
18 #pragma hdrstop
19#endif
20
21#ifndef WX_PRECOMP
22 #include <wx/wx.h>
23#endif
24
25#include <wx/splitter.h>
26
27#if defined(__WXGTK__) || defined(__WXMOTIF__) || defined(__WXMAC__) || defined(__WXMGL__)
28 #include "mondrian.xpm"
29#endif
30
31// Import Python and wxPython headers
32#include <Python.h>
33#include <wxPython.h>
34
35//----------------------------------------------------------------------
36// Class definitions
37
38class MyApp : public wxApp
39{
40public:
41 virtual bool OnInit();
42 virtual ~MyApp();
43 void Init_wxPython();
44};
45
46
47class MyFrame : public wxFrame
48{
49public:
50 MyFrame(const wxString& title, const wxPoint& pos, const wxSize& size);
51 void RedirectStdio();
52 wxWindow* DoPythonStuff(wxWindow* parent);
53 void OnExit(wxCommandEvent& event);
54 void OnPyFrame(wxCommandEvent& event);
55
56private:
57 DECLARE_EVENT_TABLE()
58};
59
60//----------------------------------------------------------------------
61// MyApp methods
62
63
64bool MyApp::OnInit()
65{
66 Init_wxPython();
67 MyFrame *frame = new MyFrame(_T("Embedded wxPython Test"),
68 wxPoint(50, 50), wxSize(700, 600));
69 frame->Show(TRUE);
70 return TRUE;
71}
72
73
74void MyApp::Init_wxPython()
75{
76 // Initialize Python
77 Py_Initialize();
78 PyEval_InitThreads();
79
80 // Load the wxPython core API. Imports the wxPython.wxc
81 // module and sets a pointer to a function table located there.
82 wxPyCoreAPI_IMPORT();
83
84 // Ensure that the new classes defined in the wxPython wrappers are
85 // recognised by the wx RTTI system. (If you don't use wxWindows in
86 // your C++ app you won't need to do this.)
87 wxClassInfo::CleanUpClasses();
88 wxClassInfo::InitializeClasses();
89
90 // Save the current Python thread state and release the
91 // Global Interpreter Lock.
92 wxPyBeginAllowThreads();
93}
94
95
96MyApp::~MyApp()
97{
98 // Restore the thread state and tell Python to cleanup after itself.
99 wxPyEndAllowThreads(true);
100 Py_Finalize();
101}
102
103IMPLEMENT_APP(MyApp)
104
105//----------------------------------------------------------------------
106
107enum
108{
109 ID_EXIT=1001,
110 ID_PYFRAME
111};
112
113
114BEGIN_EVENT_TABLE(MyFrame, wxFrame)
115 EVT_MENU(ID_EXIT, MyFrame::OnExit)
116 EVT_MENU(ID_PYFRAME, MyFrame::OnPyFrame)
117END_EVENT_TABLE()
118
119
120
121MyFrame::MyFrame(const wxString& title, const wxPoint& pos, const wxSize& size)
122 : wxFrame(NULL, -1, title, pos, size,
123 wxDEFAULT_FRAME_STYLE|wxNO_FULL_REPAINT_ON_RESIZE)
124{
125 SetIcon(wxICON(mondrian));
126
127 wxMenuBar* mbar = new wxMenuBar;
128 wxMenu* menu = new wxMenu;
129 menu->Append(ID_PYFRAME, "Make wx&Python frame");
130 menu->AppendSeparator();
131 menu->Append(ID_EXIT, "&Close Frame\tAlt-X");
132 mbar->Append(menu, "&File");
133 SetMenuBar(mbar);
134
135 CreateStatusBar();
136 RedirectStdio();
137
138 // Make some child windows from C++
139 wxSplitterWindow* sp = new wxSplitterWindow(this, -1);
140 wxPanel* p1 = new wxPanel(sp, -1);
141 p1->SetFont(wxFont(12, wxSWISS, wxNORMAL, wxBOLD));
142 new wxStaticText(p1, -1,
143 wxT("The frame, menu, splitter, this panel and this text were created in C++..."),
144 wxPoint(10,10));
145
146 // And get a panel from Python
147 wxWindow* p2 = DoPythonStuff(sp);
148
149 sp->SplitHorizontally(p1, p2, GetClientSize().y/4);
150}
151
152void MyFrame::OnExit(wxCommandEvent& event)
153{
154 Close();
155}
156
157
158//----------------------------------------------------------------------
159// This is were the fun begins...
160
161
162char* python_code1 = "\
163from wxPython.wx import wxFrame\n\
164f = wxFrame(None, -1, 'Hello from wxPython!', size=(250, 150))\n\
165f.Show()\n\
166";
167
168void MyFrame::OnPyFrame(wxCommandEvent& event)
169{
170 // For simple Python code that doesn't have to interact with the
171 // C++ code in any way, you can execute it with PyRun_SimpleString.
172
173
174 // First, whenever you do anyting with Python objects or code, you
175 // *MUST* aquire the Global Interpreter Lock and block other
176 // Python threads from running.
177 bool blocked = wxPyBeginBlockThreads();
178
179 // Execute the code in the __main__ module
180 PyRun_SimpleString(python_code1);
181
182 // Finally, release the GIL and let other Python threads run.
183 wxPyEndBlockThreads(blocked);
184}
185
186
187void MyFrame::RedirectStdio()
188{
189 // This is a helpful little tidbit to help debugging and such. It
190 // redirects Python's stdout and stderr to a window that will popup
191 // only on demand when something is printed, like a traceback.
192 char* python_redirect = "\
193import sys\n\
194from wxPython.wx import wxPyOnDemandOutputWindow\n\
195output = wxPyOnDemandOutputWindow()\n\
196sys.stdin = sys.stderr = output\n\
197";
198 bool blocked = wxPyBeginBlockThreads();
199 PyRun_SimpleString(python_redirect);
200 wxPyEndBlockThreads(blocked);
201}
202
203
204
205
206char* python_code2 = "\
207import embedded_sample\n\
208\n\
209def makeWindow(parent):\n\
210 win = embedded_sample.MyPanel(parent)\n\
211 return win\n\
212";
213
214wxWindow* MyFrame::DoPythonStuff(wxWindow* parent)
215{
216 // More complex embedded situations will require passing C++ objects to
217 // Python and/or returning objects from Python to be used in C++. This
218 // sample shows one way to do it. NOTE: The above code could just have
219 // easily come from a file, or the whole thing could be in the Python
220 // module that is imported and manipulated directly in this C++ code. See
221 // the Python API for more details.
222
223 wxWindow* window = NULL;
224 PyObject* result;
225
226 // As always, first grab the GIL
227 bool blocked = wxPyBeginBlockThreads();
228
229 // Now make a dictionary to serve as the global namespace when the code is
230 // executed. Put a reference to the builtins module in it. (Yes, the
231 // names are supposed to be different, I don't know why...)
232 PyObject* globals = PyDict_New();
233 PyObject* builtins = PyImport_ImportModule("__builtin__");
234 PyDict_SetItemString(globals, "__builtins__", builtins);
235 Py_DECREF(builtins);
236
237 // Execute the code to make the makeWindow function
238 result = PyRun_String(python_code2, Py_file_input, globals, globals);
239 // Was there an exception?
240 if (! result) {
241 PyErr_Print();
242 wxPyEndBlockThreads();
243 return NULL;
244 }
245 Py_DECREF(result);
246
247 // Now there should be an object named 'makeWindow' in the dictionary that
248 // we can grab a pointer to:
249 PyObject* func = PyDict_GetItemString(globals, "makeWindow");
250 wxASSERT(PyCallable_Check(func));
251
252 // Now build an argument tuple and call the Python function. Notice the
253 // use of another wxPython API to take a wxWindows object and build a
254 // wxPython object that wraps it.
255 PyObject* arg = wxPyMake_wxObject(parent);
256 wxASSERT(arg != NULL);
257
258 PyObject* tuple = PyTuple_New(1);
259 PyTuple_SET_ITEM(tuple, 0, arg);
260 result = PyEval_CallObject(func, tuple);
261
262 // Was there an exception?
263 if (! result)
264 PyErr_Print();
265 else {
266 // Otherwise, get the returned window out of Python-land and
267 // into C++-ville...
268 bool error = SWIG_GetPtrObj(result, (void**)&window, "_wxWindow_p");
269 wxASSERT_MSG(!error, wxT("Returned object was not a wxWindow!"));
270 Py_DECREF(result);
271 }
272
273 // Release the python objects we still have
274 Py_DECREF(globals);
275 Py_DECREF(tuple);
276
277 // Finally, after all Python stuff is done, release the GIL
278 wxPyEndBlockThreads(blocked);
279
280 return window;
281}
282
283
284//----------------------------------------------------------------------
285
286
287
288
289
290