blob: ac41d6408aa89a1902b7f08939d43615e97540d0 [file] [log] [blame]
Jack Jansen90ecdf41996-04-16 14:29:15 +00001# This script generates a Python interface for an Apple Macintosh Manager.
2# It uses the "bgen" package to generate C code.
3# The function specifications are generated by scanning the mamager's header file,
4# using the "scantools" package (customized for this particular manager).
5
6import string
7
8# Declarations that change for each manager
9MACHEADERFILE = 'WASTE.h' # The Apple header file
10MODNAME = 'waste' # The name of the module
11OBJECTNAME = 'waste' # The basic name of the objects used here
12KIND = 'Ptr' # Usually 'Ptr' or 'Handle'
13
14# The following is *usually* unchanged but may still require tuning
15MODPREFIX = MODNAME # The prefix for module-wide routines
16OBJECTTYPE = "WEReference" # The C type used to represent them
17OBJECTPREFIX = MODPREFIX + 'Obj' # The prefix for object methods
18INPUTFILE = 'wastegen.py' # The file generated by the scanner
19TYPETESTFILE = 'wastetypetest.py' # Another file generated by the scanner
20OUTPUTFILE = "wastemodule.c" # The file generated by this program
21
22from macsupport import *
23
24# Create the type objects
25WEReference = OpaqueByValueType("WEReference", "wasteObj")
Jack Jansen756522f1996-05-07 15:24:01 +000026ExistingWEReference = OpaqueByValueType("WEReference", "ExistingwasteObj")
Jack Jansen90ecdf41996-04-16 14:29:15 +000027WEObjectReference = OpaqueByValueType("WEObjectReference", "WEOObj")
Jack Jansen90ecdf41996-04-16 14:29:15 +000028StScrpHandle = OpaqueByValueType("StScrpHandle", "ResObj")
Jack Jansen90ecdf41996-04-16 14:29:15 +000029RgnHandle = OpaqueByValueType("RgnHandle", "ResObj")
30EventModifiers = Type("EventModifiers", "h")
31FlavorType = OSTypeType("FlavorType")
32WESelector = OSTypeType("WESelector")
33
Jack Jansen8ae8e4f1996-04-23 16:17:08 +000034OptHandle = OpaqueByValueType("Handle", "OptResObj")
35OptSoupHandle = OpaqueByValueType("WESoupHandle", "OptResObj")
36OptStScrpHandle = OpaqueByValueType("StScrpHandle", "OptResObj")
37
Jack Jansen90ecdf41996-04-16 14:29:15 +000038WEStyleMode = Type("WEStyleMode", "h")
39WEActionKind = Type("WEActionKind", "h")
40WEAlignment = Type("WEAlignment", "b")
Jack Jansen2268af51996-08-06 16:04:22 +000041WEEdge = Type("WEEdge", "b")
Jack Jansen90ecdf41996-04-16 14:29:15 +000042WESoupHandle = OpaqueByValueType("WESoupHandle", "ResObj")
43WERunInfo = OpaqueType("WERunInfo", "RunInfo")
44
45TextStyle = OpaqueType("TextStyle", "TextStyle")
46TextStyle_ptr = TextStyle
47LongPt = OpaqueType("LongPt", "LongPt")
48LongPt_ptr = LongPt
49LongRect = OpaqueType("LongRect", "LongRect")
50LongRect_ptr = LongRect
51
52includestuff = includestuff + """
53#include <%s>""" % MACHEADERFILE + """
Jack Jansen756522f1996-05-07 15:24:01 +000054#include <WEObjectHandlers.h>
Jack Jansen90ecdf41996-04-16 14:29:15 +000055
56/* Exported by Qdmodule.c: */
57extern PyObject *QdRGB_New(RGBColor *);
58extern int QdRGB_Convert(PyObject *, RGBColor *);
59
60/* Forward declaration */
61staticforward PyObject *WEOObj_New(WEObjectReference);
Jack Jansen756522f1996-05-07 15:24:01 +000062staticforward PyObject *ExistingwasteObj_New(WEReference);
Jack Jansen90ecdf41996-04-16 14:29:15 +000063
64/*
65** Parse/generate TextStyle records
66*/
67static
68PyObject *TextStyle_New(itself)
69 TextStylePtr itself;
70{
71
72 return Py_BuildValue("lllO&", (long)itself->tsFont, (long)itself->tsFace, (long)itself->tsSize, QdRGB_New,
73 &itself->tsColor);
74}
75
76static
77TextStyle_Convert(v, p_itself)
78 PyObject *v;
79 TextStylePtr p_itself;
80{
81 long font, face, size;
82
83 if( !PyArg_ParseTuple(v, "lllO&", &font, &face, &size, QdRGB_Convert, &p_itself->tsColor) )
84 return 0;
85 p_itself->tsFont = (short)font;
86 p_itself->tsFace = (Style)face;
87 p_itself->tsSize = (short)size;
88 return 1;
89}
90
91/*
92** Parse/generate RunInfo records
93*/
94static
95PyObject *RunInfo_New(itself)
96 WERunInfo *itself;
97{
98
99 return Py_BuildValue("llhhO&O&", itself->runStart, itself->runEnd, itself->runHeight,
100 itself->runAscent, TextStyle_New, &itself->runStyle, WEOObj_New, itself->runObject);
101}
102
103/* Conversion of long points and rects */
104int
105LongRect_Convert(PyObject *v, LongRect *r)
106{
107 return PyArg_Parse(v, "(llll)", &r->left, &r->top, &r->right, &r->bottom);
108}
109
110PyObject *
111LongRect_New(LongRect *r)
112{
113 return Py_BuildValue("(llll)", r->left, r->top, r->right, r->bottom);
114}
115
116
117LongPt_Convert(PyObject *v, LongPt *p)
118{
119 return PyArg_Parse(v, "(ll)", &p->h, &p->v);
120}
121
122PyObject *
123LongPt_New(LongPt *p)
124{
125 return Py_BuildValue("(ll)", p->h, p->v);
126}
Jack Jansen756522f1996-05-07 15:24:01 +0000127
128/* Stuff for the callbacks: */
129static PyObject *callbackdict;
Jack Jansen25241d91996-05-20 11:30:45 +0000130WENewObjectUPP upp_new_handler;
131WEDisposeObjectUPP upp_dispose_handler;
132WEDrawObjectUPP upp_draw_handler;
133WEClickObjectUPP upp_click_handler;
Jack Jansen756522f1996-05-07 15:24:01 +0000134
135static OSErr
136any_handler(WESelector what, WEObjectReference who, PyObject *args, PyObject **rv)
137{
138 FlavorType tp;
139 PyObject *key, *func;
140
141 if ( args == NULL ) return errAECorruptData;
142
143 tp = WEGetObjectType(who);
144
145 if( (key=Py_BuildValue("O&O&", PyMac_BuildOSType, tp, PyMac_BuildOSType, what)) == NULL)
146 return errAECorruptData;
147 if( (func = PyDict_GetItem(callbackdict, key)) == NULL ) {
148 Py_DECREF(key);
149 return errAEHandlerNotFound;
150 }
151 Py_INCREF(func);
152 *rv = PyEval_CallObject(func, args);
153 Py_DECREF(func);
154 Py_DECREF(key);
155 if ( *rv == NULL ) {
156 fprintf(stderr, "--Exception in callback: ");
157 PyErr_Print();
158 return errAEReplyNotArrived;
159 }
160 return 0;
161}
162
163static pascal OSErr
164my_new_handler(Point *objectSize, WEObjectReference objref)
165{
166 PyObject *args=NULL, *rv=NULL;
167 OSErr err;
168
169 args=Py_BuildValue("(O&)", WEOObj_New, objref);
170 err = any_handler(weNewHandler, objref, args, &rv);
171 if (!err) {
172 if (!PyMac_GetPoint(rv, objectSize) )
173 err = errAECoercionFail;
174 }
175 if ( args ) Py_DECREF(args);
176 if ( rv ) Py_DECREF(rv);
177 return err;
178}
179
180static pascal OSErr
181my_dispose_handler(WEObjectReference objref)
182{
183 PyObject *args=NULL, *rv=NULL;
184 OSErr err;
185
186 args=Py_BuildValue("(O&)", WEOObj_New, objref);
187 err = any_handler(weDisposeHandler, objref, args, &rv);
188 if ( args ) Py_DECREF(args);
189 if ( rv ) Py_DECREF(rv);
190 return err;
191}
192
193static pascal OSErr
194my_draw_handler(Rect *destRect, WEObjectReference objref)
195{
196 PyObject *args=NULL, *rv=NULL;
197 OSErr err;
198
199 args=Py_BuildValue("O&O&", PyMac_BuildRect, destRect, WEOObj_New, objref);
200 err = any_handler(weDrawHandler, objref, args, &rv);
201 if ( args ) Py_DECREF(args);
202 if ( rv ) Py_DECREF(rv);
203 return err;
204}
205
206static pascal Boolean
207my_click_handler(Point hitPt, EventModifiers modifiers,
208 unsigned long clickTime, WEObjectReference objref)
209{
210 PyObject *args=NULL, *rv=NULL;
211 int retvalue;
212 OSErr err;
213
214 args=Py_BuildValue("O&llO&", PyMac_BuildPoint, hitPt,
215 (long)modifiers, (long)clickTime, WEOObj_New, objref);
216 err = any_handler(weClickHandler, objref, args, &rv);
217 if (!err)
218 retvalue = PyInt_AsLong(rv);
219 else
220 retvalue = 0;
221 if ( args ) Py_DECREF(args);
222 if ( rv ) Py_DECREF(rv);
223 return retvalue;
224}
225
226
227"""
228finalstuff = finalstuff + """
229/* Return the object corresponding to the window, or NULL */
230
231PyObject *
232ExistingwasteObj_New(w)
233 WEReference w;
234{
235 PyObject *it = NULL;
236
237 if (w == NULL)
238 it = NULL;
239 else
240 WEGetInfo(weRefCon, (void *)&it, w);
241 if (it == NULL || ((wasteObject *)it)->ob_itself != w)
242 it = Py_None;
243 Py_INCREF(it);
244 return it;
245}
Jack Jansen90ecdf41996-04-16 14:29:15 +0000246"""
247
248class WEMethodGenerator(OSErrMethodGenerator):
249 """Similar to MethodGenerator, but has self as last argument"""
250
251 def parseArgumentList(self, args):
252 args, a0 = args[:-1], args[-1]
253 t0, n0, m0 = a0
254 if m0 != InMode:
255 raise ValueError, "method's 'self' must be 'InMode'"
256 self.itself = Variable(t0, "_self->ob_itself", SelfMode)
257 FunctionGenerator.parseArgumentList(self, args)
258 self.argumentList.append(self.itself)
259
260
261
262class WEObjectDefinition(GlobalObjectDefinition):
263 def outputCheckNewArg(self):
264 Output("""if (itself == NULL) {
265 PyErr_SetString(waste_Error,"Cannot create null WE");
266 return NULL;
267 }""")
Jack Jansen756522f1996-05-07 15:24:01 +0000268 def outputInitStructMembers(self):
269 GlobalObjectDefinition.outputInitStructMembers(self)
270 Output("WESetInfo(weRefCon, (void *)&it, itself);")
Jack Jansen90ecdf41996-04-16 14:29:15 +0000271 def outputFreeIt(self, itselfname):
272 Output("WEDispose(%s);", itselfname)
273
274class WEOObjectDefinition(GlobalObjectDefinition):
275 def outputCheckNewArg(self):
276 Output("""if (itself == NULL) {
277 Py_INCREF(Py_None);
278 return Py_None;
Jack Jansen756522f1996-05-07 15:24:01 +0000279 }""")
280
281variablestuff = """
282 callbackdict = PyDict_New();
283 if (callbackdict == NULL || PyDict_SetItemString(d, "callbacks", callbackdict) != 0)
284 Py_FatalError("can't initialize Waste.callbackdict");
285 upp_new_handler = NewWENewObjectProc(my_new_handler);
Jack Jansen25241d91996-05-20 11:30:45 +0000286 upp_dispose_handler = NewWEDisposeObjectProc(my_dispose_handler);
287 upp_draw_handler = NewWEDrawObjectProc(my_draw_handler);
288 upp_click_handler = NewWEClickObjectProc(my_click_handler);
Jack Jansen756522f1996-05-07 15:24:01 +0000289"""
290
Jack Jansen90ecdf41996-04-16 14:29:15 +0000291
292# From here on it's basically all boiler plate...
293
294# Test types used for existence
Jack Jansen8ae8e4f1996-04-23 16:17:08 +0000295## execfile(TYPETESTFILE)
Jack Jansen90ecdf41996-04-16 14:29:15 +0000296
297# Create the generator groups and link them
Jack Jansen756522f1996-05-07 15:24:01 +0000298module = MacModule(MODNAME, MODPREFIX, includestuff, finalstuff, initstuff, variablestuff)
Jack Jansen90ecdf41996-04-16 14:29:15 +0000299object = WEObjectDefinition(OBJECTNAME, OBJECTPREFIX, OBJECTTYPE)
300object2 = WEOObjectDefinition("WEO", "WEOObj", "WEObjectReference")
301module.addobject(object2)
302module.addobject(object)
303
304# Create the generator classes used to populate the lists
305Function = OSErrFunctionGenerator
306Method = WEMethodGenerator
307Method2 = OSErrMethodGenerator
308
309# Create and populate the lists
310functions = []
311methods = []
312methods2 = []
313execfile(INPUTFILE)
314
Jack Jansen756522f1996-05-07 15:24:01 +0000315# A function written by hand:
316stdhandlers_body = """
317 OSErr err;
318 // install the sample object handlers for pictures and sounds
319#define kTypePicture 'PICT'
320#define kTypeSound 'snd '
321
322 if ( !PyArg_ParseTuple(_args, "") ) return NULL;
323
324 if ((err = WEInstallObjectHandler(kTypePicture, weNewHandler,
325 (UniversalProcPtr) NewWENewObjectProc(HandleNewPicture), NULL)) != noErr)
326 goto cleanup;
327
328 if ((err = WEInstallObjectHandler(kTypePicture, weDisposeHandler,
329 (UniversalProcPtr) NewWEDisposeObjectProc(HandleDisposePicture), NULL)) != noErr)
330 goto cleanup;
331
332 if ((err = WEInstallObjectHandler(kTypePicture, weDrawHandler,
333 (UniversalProcPtr) NewWEDrawObjectProc(HandleDrawPicture), NULL)) != noErr)
334 goto cleanup;
335
336 if ((err = WEInstallObjectHandler(kTypeSound, weNewHandler,
337 (UniversalProcPtr) NewWENewObjectProc(HandleNewSound), NULL)) != noErr)
338 goto cleanup;
339
340 if ((err = WEInstallObjectHandler(kTypeSound, weDrawHandler,
341 (UniversalProcPtr) NewWEDrawObjectProc(HandleDrawSound), NULL)) != noErr)
342 goto cleanup;
343
344 if ((err = WEInstallObjectHandler(kTypeSound, weClickHandler,
345 (UniversalProcPtr) NewWEClickObjectProc(HandleClickSound), NULL)) != noErr)
346 goto cleanup;
347 Py_INCREF(Py_None);
348 return Py_None;
349
350cleanup:
351 return PyMac_Error(err);
352"""
353
354inshandler_body = """
355 OSErr err;
356 FlavorType objectType;
357 WESelector selector;
358 PyObject *py_handler;
359 UniversalProcPtr handler;
360 WEReference we = NULL;
361 PyObject *key;
362
363
364 if ( !PyArg_ParseTuple(_args, "O&O&O|O&",
365 PyMac_GetOSType, &objectType,
366 PyMac_GetOSType, &selector,
367 &py_handler,
Jack Jansen9f2ff911996-10-01 10:48:45 +0000368 WEOObj_Convert, &we) ) return NULL;
Jack Jansen756522f1996-05-07 15:24:01 +0000369
Jack Jansen25241d91996-05-20 11:30:45 +0000370 if ( selector == weNewHandler ) handler = (UniversalProcPtr)upp_new_handler;
371 else if ( selector == weDisposeHandler ) handler = (UniversalProcPtr)upp_dispose_handler;
372 else if ( selector == weDrawHandler ) handler = (UniversalProcPtr)upp_draw_handler;
373 else if ( selector == weClickHandler ) handler = (UniversalProcPtr)upp_click_handler;
Jack Jansen756522f1996-05-07 15:24:01 +0000374 else return PyMac_Error(weUndefinedSelectorErr);
375
376 if ((key = Py_BuildValue("O&O&",
377 PyMac_BuildOSType, objectType,
378 PyMac_BuildOSType, selector)) == NULL )
379 return NULL;
380
381 PyDict_SetItem(callbackdict, key, py_handler);
382
383 err = WEInstallObjectHandler(objectType, selector, handler, we);
384 if ( err ) return PyMac_Error(err);
385 Py_INCREF(Py_None);
386 return Py_None;
387"""
388
389stdhand = ManualGenerator("STDObjectHandlers", stdhandlers_body)
390inshand = ManualGenerator("WEInstallObjectHandler", inshandler_body)
391
Jack Jansen90ecdf41996-04-16 14:29:15 +0000392# add the populated lists to the generator groups
393# (in a different wordl the scan program would generate this)
394for f in functions: module.add(f)
Jack Jansen756522f1996-05-07 15:24:01 +0000395module.add(stdhand)
396module.add(inshand)
Jack Jansen90ecdf41996-04-16 14:29:15 +0000397for f in methods: object.add(f)
398for f in methods2: object2.add(f)
399
400# generate output (open the output file as late as possible)
401SetOutputFileName(OUTPUTFILE)
402module.generate()
403