]> git.saurik.com Git - wxWidgets.git/blob - wxPython/samples/embedded/embedded.cpp
typos fixed
[wxWidgets.git] / wxPython / samples / embedded / embedded.cpp
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
38 class MyApp : public wxApp
39 {
40 public:
41 virtual bool OnInit();
42 virtual ~MyApp();
43 void Init_wxPython();
44 };
45
46
47 class MyFrame : public wxFrame
48 {
49 public:
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
56 private:
57 DECLARE_EVENT_TABLE()
58 };
59
60 //----------------------------------------------------------------------
61 // MyApp methods
62
63
64 bool 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
74 void 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
96 MyApp::~MyApp()
97 {
98 // Restore the thread state and tell Python to cleanup after itself.
99 wxPyEndAllowThreads(true);
100 Py_Finalize();
101 }
102
103 IMPLEMENT_APP(MyApp)
104
105 //----------------------------------------------------------------------
106
107 enum
108 {
109 ID_EXIT=1001,
110 ID_PYFRAME
111 };
112
113
114 BEGIN_EVENT_TABLE(MyFrame, wxFrame)
115 EVT_MENU(ID_EXIT, MyFrame::OnExit)
116 EVT_MENU(ID_PYFRAME, MyFrame::OnPyFrame)
117 END_EVENT_TABLE()
118
119
120
121 MyFrame::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
152 void MyFrame::OnExit(wxCommandEvent& event)
153 {
154 Close();
155 }
156
157
158 //----------------------------------------------------------------------
159 // This is were the fun begins...
160
161
162 char* python_code1 = "\
163 from wxPython.wx import wxFrame\n\
164 f = wxFrame(None, -1, 'Hello from wxPython!', size=(250, 150))\n\
165 f.Show()\n\
166 ";
167
168 void 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
187 void 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 = "\
193 import sys\n\
194 from wxPython.wx import wxPyOnDemandOutputWindow\n\
195 output = wxPyOnDemandOutputWindow()\n\
196 sys.stdin = sys.stderr = output\n\
197 ";
198 bool blocked = wxPyBeginBlockThreads();
199 PyRun_SimpleString(python_redirect);
200 wxPyEndBlockThreads(blocked);
201 }
202
203
204
205
206 char* python_code2 = "\
207 import embedded_sample\n\
208 \n\
209 def makeWindow(parent):\n\
210 win = embedded_sample.MyPanel(parent)\n\
211 return win\n\
212 ";
213
214 wxWindow* 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