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