]>
Commit | Line | Data |
---|---|---|
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 |