propose-patch
[org.ibex.core.git] / src / org / ibex / plat / Darwin.cc
1 // Copyright 2003 Adam Megacz, see the COPYING file for licensing [LGPL]
2 // Authors: Brian Alliet and Evan Jones
3 #ifndef __APPLE_CC__
4 #define FSF_GCC
5 #define __APPLE_CC__
6 #else
7 #define APPLE_GCC
8 #endif
9
10 #define __private_extern__
11 #include <mach-o/dyld.h>
12
13 #include "POSIX.cc"
14 #include "OpenGL.cc"
15
16 #include <java/lang/Object.h>
17 #include <java/lang/Error.h>
18
19 #include <org/ibex/plat/Darwin.h>
20 #include <org/ibex/plat/Darwin$CarbonSurface.h>
21 #include <org/ibex/plat/Darwin$GLCarbonSurface.h>
22 #include <org/ibex/plat/Darwin$GLCarbonPixelBuffer.h>
23 #include <org/ibex/plat/Darwin$CarbonMessage.h>
24 #include <org/ibex/plat/Darwin$CarbonOpenGL.h>
25 #include <org/ibex/plat/Darwin$FileDialogHelper.h>
26 #include <org/ibex/plat/GCJ$Retainer.h>
27 #include <org/ibex/HTTP$Proxy.h>
28 #include <org/ibex/util/Semaphore.h>
29
30 #include <stdlib.h>
31 #include <pthread.h>
32
33 #include "DarwinCarbonHeaders.h"
34
35 #define Ibex_CARBON_NO_BUNDLE_HACK
36 #ifdef Ibex_CARBON_NO_BUNDLE_HACK
37 extern "C" {
38     OSErr CPSEnableForegroundOperation(ProcessSerialNumber *psn);
39     OSErr CPSSetFrontProcess(ProcessSerialNumber *psn);
40 }
41 #endif
42
43 static const mach_header* CarbonHandle = NULL;
44 static const mach_header* AGLHandle = NULL;
45 static const mach_header* SCHandle = NULL;
46 #define CARBON_LIBRARY_PATH "/System/Library/Frameworks/Carbon.framework/Versions/A/Carbon"
47 #define AGL_LIBRARY_PATH "/System/Library/Frameworks/AGL.framework/Versions/A/AGL"
48 #define SC_LIBRARY_PATH "/System/Library/Frameworks/SystemConfiguration.framework/Versions/A/SystemConfiguration"
49
50 static void* dlsym(char* symbol) {
51   if (CarbonHandle == NULL) CarbonHandle = NSAddImage(CARBON_LIBRARY_PATH, NSADDIMAGE_OPTION_NONE);
52   if (AGLHandle == NULL) AGLHandle = NSAddImage(AGL_LIBRARY_PATH, NSADDIMAGE_OPTION_NONE);
53   if (SCHandle == NULL) SCHandle = NSAddImage(SC_LIBRARY_PATH, NSADDIMAGE_OPTION_NONE);
54   void* ret = NSAddressOfSymbol(NSLookupSymbolInImage(CarbonHandle, symbol,
55                                                       NSLOOKUPSYMBOLINIMAGE_OPTION_BIND));
56   if (ret == NULL) ret = NSAddressOfSymbol(NSLookupSymbolInImage(AGLHandle, symbol,
57                                                                  NSLOOKUPSYMBOLINIMAGE_OPTION_BIND));
58   if (ret == NULL) ret = NSAddressOfSymbol(NSLookupSymbolInImage(SCHandle, symbol,
59                                                                  NSLOOKUPSYMBOLINIMAGE_OPTION_BIND));
60   /*printf("linking symbol %s to address %x\n", symbol, ret);
61   fflush(stdout);*/
62   return ret;
63 }
64
65 #define declare_weak(symbol) typeof(symbol) *symbol##_weak = NULL
66 #define WC(func) \
67     (*((func##_weak != NULL) ? func##_weak : func##_weak = (typeof(func##_weak))dlsym("_" #func)))
68
69 declare_weak(GetRegionBounds);
70 declare_weak(RegisterToolboxObjectClass);
71 declare_weak(GetWindowProperty);
72 declare_weak(SetWindowProperty);
73 declare_weak(CreateCustomWindow);
74 declare_weak(InstallWindowContentPaintProc);
75 declare_weak(AEGetNthPtr);
76 declare_weak(CFArrayGetCount);
77 declare_weak(CFArrayGetTypeID);
78 declare_weak(CFArrayGetValueAtIndex);
79 declare_weak(CFDictionaryGetValue);
80 declare_weak(CFGetTypeID);
81 declare_weak(CFNumberGetTypeID);
82 declare_weak(CFNumberGetValue);
83 declare_weak(CFRelease);
84 declare_weak(CFStringCompare);
85 declare_weak(CFStringCreateWithCString);
86 declare_weak(CFStringCreateWithCharacters);
87 declare_weak(CFStringGetCharacters);
88 declare_weak(CFStringGetLength);
89 declare_weak(CFStringGetTypeID);
90 declare_weak(CFURLCopyScheme);
91 declare_weak(CFURLCreateWithString);
92 declare_weak(CGDisplayPixelsHigh);
93 declare_weak(CGDisplayPixelsWide);
94 declare_weak(CallNextEventHandler);
95 declare_weak(CollapseWindow);
96 declare_weak(ConstrainWindowToScreen);
97 declare_weak(CreateEvent);
98 declare_weak(CreateNewWindow);
99 declare_weak(DisposeEventHandlerUPP);
100 declare_weak(DisposeNavEventUPP);
101 declare_weak(DisposeWindow);
102 declare_weak(ExitToShell);
103 declare_weak(FSRefMakePath);
104 declare_weak(FrontWindow);
105 declare_weak(Gestalt);
106 declare_weak(GetApplicationEventTarget);
107 declare_weak(GetCurrentProcess);
108 declare_weak(GetCurrentScrap);
109 declare_weak(GetEventClass);
110 declare_weak(GetEventKind);
111 declare_weak(GetEventParameter);
112 declare_weak(GetMainEventQueue);
113 declare_weak(GetScrapFlavorData);
114 declare_weak(GetScrapFlavorSize);
115 declare_weak(GetWindowBounds);
116 declare_weak(GetWindowEventTarget);
117 declare_weak(GetWindowPort);
118 declare_weak(HideWindow);
119 declare_weak(InstallEventHandler);
120 declare_weak(IsWindowCollapsed);
121 declare_weak(LSOpenCFURLRef);
122 declare_weak(NavCreateGetFileDialog);
123 declare_weak(NavCreatePutFileDialog);
124 declare_weak(NavDialogDispose);
125 declare_weak(NavDialogGetReply);
126 declare_weak(NavDialogGetUserAction);
127 declare_weak(NavDialogRun);
128 declare_weak(NavDisposeReply);
129 declare_weak(NavGetDefaultDialogCreationOptions);
130 declare_weak(NewEventHandlerUPP);
131 declare_weak(NewNavEventUPP);
132 declare_weak(PostEventToQueue);
133 declare_weak(PutScrapFlavor);
134 declare_weak(QuitApplicationEventLoop);
135 declare_weak(ReleaseEvent);
136 declare_weak(RunApplicationEventLoop);
137 declare_weak(SCDynamicStoreCopyProxies);
138 declare_weak(SelectWindow);
139 declare_weak(SendBehind);
140 declare_weak(SetEventParameter);
141 declare_weak(SetThemeCursor);
142 declare_weak(SetWindowBounds);
143 declare_weak(SetWindowResizeLimits);
144 declare_weak(SetWindowTitleWithCFString);
145 declare_weak(ShowWindow);
146 declare_weak(ZoomWindowIdeal);
147 declare_weak(__CFStringMakeConstantString);
148 declare_weak(aglChoosePixelFormat);
149 declare_weak(aglCreateContext);
150 declare_weak(aglDestroyContext);
151 declare_weak(aglSetCurrentContext);
152 declare_weak(aglSetDrawable);
153 declare_weak(aglSurfaceTexture);
154 declare_weak(aglUpdateContext);
155 declare_weak(CPSEnableForegroundOperation);
156 declare_weak(CPSSetFrontProcess);
157 declare_weak(kCFAllocatorDefault);
158 declare_weak(NewWindowPaintUPP);
159
160 #define GetWindowEventTarget WC(GetWindowEventTarget)
161 #define InstallEventHandler WC(InstallEventHandler)
162 #define __CFStringMakeConstantString WC(__CFStringMakeConstantString)
163
164 using namespace org::ibex::plat;
165 using gnu::gcj::RawData;
166 using org::ibex::util::Semaphore;
167 using java::lang::Object;
168
169 namespace org { namespace ibex { namespace plat {
170
171
172 #pragma mark ----- Blit Locks ------
173 static pthread_mutex_t blit_mutex;
174 static pthread_cond_t blit_cond;
175
176 static inline void blit_lock() { if(pthread_mutex_lock(&blit_mutex) != 0) abort(); }
177 static inline void blit_unlock() { if(pthread_mutex_unlock(&blit_mutex) != 0) abort(); }
178 static inline void blit_wait() { if(pthread_cond_wait(&blit_cond,&blit_mutex) != 0) abort(); }
179 static inline void blit_signal() { if(pthread_cond_signal(&blit_cond) != 0) abort(); }
180
181 void Darwin$CarbonSurface::blitLock() { blit_lock(); }
182 void Darwin$CarbonSurface::blitUnlock() { blit_unlock(); }
183 void Darwin$CarbonSurface::blitWait() { blit_wait(); }
184
185
186 template <bool CHECK> static inline int CompileTimeCheck() { const int something_is_wrong=1; something_is_wrong++; return 0; }
187 template <> static inline int CompileTimeCheck<true>() { return 0; }
188 const static int unichar_check = CompileTimeCheck<sizeof(jchar)==sizeof(UniChar)>();
189
190 static void funcFailed(const char *func,int r);
191 static inline void funcFailed(const char *func) { funcFailed(func,-1); }
192
193 static inline void checkStatus(OSStatus r, char *func) { if(r != noErr) funcFailed(func,r); }
194 static inline void checkStatus(GLboolean b, char *func) { if(!b) funcFailed(func,-1); }
195 static inline void checkStatus(void *p, char *func) { if(!p) funcFailed(func,-1); }
196
197 jstring cfStringToJString(CFStringRef s) {
198     CFIndex length = WC(CFStringGetLength)(s);
199     CFRange range = CFRangeMake(0,length);
200     jstring js = JvAllocString(length);
201     UniChar *buf = (UniChar*)JvGetStringChars(js);
202     WC(CFStringGetCharacters)(s,range,buf);
203     return js;
204 }
205
206 #pragma mark ------ SmartCFString ------
207 class SmartCFString {
208     private:
209         CFStringRef p;
210         void release() { if(p) WC(CFRelease)(p); }
211         void checkNull() { if(!p) funcFailed("CFString function"); }
212     public:
213         // Constructors
214         SmartCFString() : p(0) { }
215         SmartCFString(const char *s) : p(0) { *this = s; }
216         SmartCFString(jstring js) : p(0) { *this = js; }
217         SmartCFString(CFStringRef cf) : p(0) { *this = cf; }
218         // Destructor
219         ~SmartCFString() { release(); }
220         // Assignment
221         SmartCFString& operator= (const char *s) {
222             release();
223             if(!s) s = "(null)";
224             p = WC(CFStringCreateWithCString)(WC(kCFAllocatorDefault),s,kCFStringEncodingISOLatin1);
225             checkNull();
226             return *this;
227         }
228         SmartCFString& operator= (jstring js) {
229             if(!js) return *this = "(null)";
230             release();
231             UniChar *buf = (UniChar*) JvGetStringChars(js);
232             CFIndex length = js->length();
233             p = WC(CFStringCreateWithCharacters)(WC(kCFAllocatorDefault),buf,length);
234             checkNull();
235             return *this;
236         }
237         SmartCFString& operator= (CFStringRef cf) {
238             if(cf == NULL) return *this = "(null)";
239             release();
240             p = cf;
241             return *this;
242         }
243         operator CFStringRef() { return p; }
244         operator jstring() { return getJString(); }
245         
246         jstring getJString() { return cfStringToJString(p); }
247         
248         bool equals(const char *s) {
249             SmartCFString cfs(s);
250             return equals(cfs);
251         }
252         
253         bool equals(CFStringRef cfs) {
254             return WC(CFStringCompare)(p,cfs,0) == kCFCompareEqualTo;
255         }
256 };
257
258 // CHECKME: Is just making up your own four char codes really correct?
259 const static UInt32 kEventClassCarbonMessage = 'ibexa';
260 const static UInt32 kEventCarbonMessage = 'ibexb';
261 const static UInt32 kEventParamCarbonMessage = 'ibexc';
262
263 static OSStatus carbonMessageEventHandler(EventHandlerCallRef handler, EventRef e, void *userData);
264 static EventHandlerUPP carbonMessageEventHandlerUPP;
265
266 static void fileDialogEventHandler(NavEventCallbackMessage callBackSelector, NavCBRecPtr callBackParms, void *userData);
267 static NavEventUPP fileDialogEventHandlerUPP;
268
269 static OSStatus windowEventHandler(EventHandlerCallRef handler, EventRef e, void *userData);
270 static EventHandlerUPP windowEventHandlerUPP;
271
272 static OSStatus paintProc(GDHandle device,GrafPtr qdContext,WindowRef window,RgnHandle inClientPaintRgn,RgnHandle outSystemPaintRgn,void *userData);
273 static WindowPaintUPP paintProcUPP;
274
275 jboolean Darwin::isJaguar() {
276     SInt32 version;
277     OSStatus r = WC(Gestalt)(gestaltSystemVersion, &version);
278     checkStatus(r,"Gestalt");
279     return version >= 0x1020;
280 }
281
282 void Darwin$CarbonMessage::natInit() {
283     OSStatus r;
284     
285     EventTypeSpec eventTypes = { kEventClassCarbonMessage, kEventCarbonMessage };
286     r = InstallEventHandler(WC(GetApplicationEventTarget)(),carbonMessageEventHandlerUPP,1,&eventTypes,NULL,NULL);
287     checkStatus(r,"InstallEventHandler");
288 }
289
290 void Darwin$CarbonMessage::add(Darwin$CarbonMessage *msg) {
291     EventRef event;
292     OSStatus r;
293     
294     GCJ$Retainer::retain(msg);
295
296     r = WC(CreateEvent)(WC(kCFAllocatorDefault),kEventClassCarbonMessage,kEventCarbonMessage,0,kEventAttributeNone,&event);
297     checkStatus(r,"CreateEvent");
298     r = WC(SetEventParameter)(event,kEventParamCarbonMessage,typeVoidPtr,sizeof(msg),(void*)&msg);
299     checkStatus(r,"SetEventParameter");
300     r = WC(PostEventToQueue)(WC(GetMainEventQueue)(),event,kEventPriorityHigh);
301     checkStatus(r,"PostEventToQueue");
302     WC(ReleaseEvent)(event);
303 }
304
305
306 static OSStatus carbonMessageEventHandler(EventHandlerCallRef handler, EventRef e, void *userData) {
307     UInt32 eKind = WC(GetEventKind)(e);
308     UInt32 eClass = WC(GetEventClass)(e);
309     OSStatus r;
310     Darwin$CarbonMessage *msg;
311     if(eClass != kEventClassCarbonMessage || eKind != kEventCarbonMessage)
312         return eventNotHandledErr;
313     r = WC(GetEventParameter)(e,kEventParamCarbonMessage,typeVoidPtr,NULL,sizeof(msg),NULL,&msg);
314     checkStatus(r,"GetEventParameter");
315     msg->perform();
316     GCJ$Retainer::release(msg);
317     return noErr;
318 }
319
320
321 #pragma mark ------ Utility Functions ------
322
323 static void funcFailed(const char *func,int r){
324     fprintf(stderr,"%s() failed (%d)\n",func,r);
325     exit(EXIT_FAILURE);
326
327
328 #pragma mark ------ CarbonPicture Methods ------ 
329
330 #pragma mark ----- Carbon Surface Methods ----
331
332
333 void Darwin$CarbonSurface::natSyncCursor(jint n) {
334     ThemeCursor c;
335     // see Carbon.java for what these numbers mean
336     switch(n) {
337         case 1: c = kThemeWatchCursor;
338         case 2: c = kThemePlusCursor;
339         case 3: c = kThemeIBeamCursor;
340         case 4: c = kThemePointingHandCursor;
341         case 5: c = kThemeOpenHandCursor;
342         case 6: c = kThemeResizeLeftRightCursor;
343         default: c = kThemeArrowCursor;
344     }
345     WC(SetThemeCursor)(c);
346 }
347    
348 void Darwin$CarbonSurface::natSetInvisible(jboolean b) {
349     WindowRef window = (WindowRef) rawWindowRef;
350     fprintf(stderr,"Making window %s\n",b?"invisible":"visible");
351     if(b) WC(HideWindow)(window);
352     else WC(ShowWindow)(window);
353 }
354
355 void Darwin$CarbonSurface::nat_setMaximized(jboolean b) {
356     WindowRef window = (WindowRef) rawWindowRef;
357     Point ideal = { 10000, 10000 };
358     OSStatus r = WC(ZoomWindowIdeal)(window,(b?inZoomOut:inZoomIn),&ideal);
359     checkStatus(r,"ZoomWindowIdeal");
360 }
361
362 void Darwin$CarbonSurface::nat_setMinimized(jboolean b) {
363     WindowRef window = (WindowRef) rawWindowRef;
364     if((WC(IsWindowCollapsed)(window) ? 1 : 0) == (b ? 1 : 0)) return;
365     OSStatus r = WC(CollapseWindow)(window,b);
366     checkStatus(r,"CollapseWindow");
367 }
368
369 void Darwin$CarbonSurface::natSetTitleBarText(jstring js) {
370     SmartCFString s = js;
371     WindowRef window = (WindowRef) rawWindowRef;
372     WC(SetWindowTitleWithCFString)(window,s);
373 }
374
375 void Darwin$CarbonSurface::natToBack() {
376     WindowRef window = (WindowRef) rawWindowRef;
377     WC(SendBehind)(window,NULL);
378 }
379
380 void Darwin$CarbonSurface::natToFront() {
381     WindowRef window = (WindowRef) rawWindowRef;
382     fprintf(stderr,"SelectWindow)()\n");
383     WC(SelectWindow)(window);
384 }
385
386 #pragma mark ---- Window Event Handler ----
387 static const EventTypeSpec eventTypeSpecs[] = {
388     // kEventClassCommand
389     // { kEventClassCommand, ??? },
390     
391     // kEventClassWindow
392     { kEventClassWindow, kEventWindowUpdate },
393     { kEventClassWindow, kEventWindowBoundsChanged },
394     { kEventClassWindow, kEventWindowBoundsChanging },
395     { kEventClassWindow, kEventWindowActivated },
396     { kEventClassWindow, kEventWindowDeactivated },
397     { kEventClassWindow, kEventWindowZoomed },
398     { kEventClassWindow, kEventWindowCollapsed },
399     { kEventClassWindow, kEventWindowExpanded },
400     { kEventClassWindow, kEventWindowClose },
401     { kEventClassWindow, kEventWindowClosed },
402     { kEventClassWindow, kEventWindowDrawFrame },
403     { kEventClassWindow, kEventWindowDrawPart },
404
405     // kEventClassControl
406     { kEventClassControl, kEventControlApplyBackground },
407     
408     // kEventClassKeyboard
409     { kEventClassKeyboard, kEventRawKeyDown },
410     { kEventClassKeyboard, kEventRawKeyRepeat },
411     { kEventClassKeyboard, kEventRawKeyUp },
412     { kEventClassKeyboard, kEventRawKeyModifiersChanged },
413     
414     // kEventClassMouse
415     { kEventClassMouse, kEventMouseDown },
416     { kEventClassMouse, kEventMouseUp },
417     { kEventClassMouse, kEventMouseMoved },
418     { kEventClassMouse, kEventMouseDragged },
419     { kEventClassMouse, kEventMouseWheelMoved },
420 };
421     
422 static OSStatus windowEventHandler(EventHandlerCallRef handler, EventRef e, void *userData) {
423   Darwin$CarbonSurface *surface = (Darwin$CarbonSurface*)userData;
424   UInt32 eKind = WC(GetEventKind)(e);
425   UInt32 eClass = WC(GetEventClass)(e);
426   OSStatus r;
427   char* ec = (char*)&eClass;
428   if (surface == NULL) return eventNotHandledErr;
429     
430
431     switch(eClass) {
432         case kEventClassCommand:
433             switch(eKind) {
434                 // TODO: handle menu items
435             }
436             break;
437         case kEventClassKeyboard:
438             switch(eKind) {
439                 case kEventRawKeyDown:
440                 case kEventRawKeyRepeat: 
441                 case kEventRawKeyUp: {
442                     UInt32 keyCode;
443                     jstring js;
444                     
445                     r = WC(GetEventParameter)(e,kEventParamKeyCode,typeUInt32,NULL,sizeof(keyCode),NULL,&keyCode);
446                     checkStatus(r,"GetEventParameter");
447                     
448                     switch(keyCode) {
449                         // These values were obtained by experimentation. I can't find any constants for them
450                         // in the header files
451                         case 126: js = JvNewStringLatin1("up"); break;
452                         case 125: js = JvNewStringLatin1("down"); break;
453                         case 124: js = JvNewStringLatin1("right"); break;
454                         case 123: js = JvNewStringLatin1("left"); break;
455                         case 122: js = JvNewStringLatin1("f1"); break;
456                         case 120: js = JvNewStringLatin1("f2"); break;
457                         case 99:  js = JvNewStringLatin1("f3"); break;
458                         case 118: js = JvNewStringLatin1("f4"); break;
459                         case 96:  js = JvNewStringLatin1("f5"); break;
460                         case 97:  js = JvNewStringLatin1("f6"); break;
461                         case 98:  js = JvNewStringLatin1("f7"); break;
462                         case 100: js = JvNewStringLatin1("f8"); break;
463                         case 101: js = JvNewStringLatin1("f9"); break;
464                         case 109: js = JvNewStringLatin1("f10"); break;
465                         case 103: js = JvNewStringLatin1("f11"); break;
466                         case 111: js = JvNewStringLatin1("f12"); break;
467                         case 105: js = JvNewStringLatin1("f13"); break;
468                         case 114: js = JvNewStringLatin1("insert"); break;
469                         case 117: js = JvNewStringLatin1("delete"); break;
470                         case 116: js = JvNewStringLatin1("page_up"); break;
471                         case 121: js = JvNewStringLatin1("page_down"); break;
472                         case 115: js = JvNewStringLatin1("home"); break;
473                         case 119: js = JvNewStringLatin1("end"); break;
474                         case 71:  js = JvNewStringLatin1("num_lock"); break;
475                         case 53:  js = JvNewStringLatin1("escape"); break;
476                         case 51:  js = JvNewStringLatin1("back_space"); break;
477                         case 36:  js = JvNewStringLatin1("enter"); break;
478                         case 48:  js = JvNewStringLatin1("tab"); break;
479                         case 76:  js = JvNewStringLatin1("enter"); break; // number pad enter
480                         default: {                            
481                             UInt32 size;
482                             UInt32 modifiers = surface->modifiers;
483                             r = WC(GetEventParameter)(e,kEventParamKeyUnicodes,typeUnicodeText,NULL,0,&size,NULL);
484                             checkStatus(r,"GetEventParameter");
485                             if(size == 0 || (modifiers & controlKey && size>sizeof(UniChar))) return eventNotHandledErr;
486                             
487                             js = JvAllocString(size/sizeof(UniChar));
488                             UniChar *buf = (UniChar*)JvGetStringChars(js);
489                             r = WC(GetEventParameter)(e,kEventParamKeyUnicodes,typeUnicodeText,NULL,size,NULL,buf);
490                             checkStatus(r,"GetEventParameter");
491
492                             if(!buf[0]) return eventNotHandledErr; // shouldn't happen
493                             // odd, when the ctrl key is pressed a-"`" become 1-31, this brings them back to the corect values
494                             if(modifiers & controlKey && buf[0] < 32) buf[0] += 0x60;
495                             break;
496                         }
497                     }
498                     
499                     if(eKind == kEventRawKeyUp)
500                         surface->KeyReleased(js);
501                     else
502                         surface->KeyPressed(js);
503                     return noErr;
504                 }
505                 case kEventRawKeyModifiersChanged: {
506                     const static struct {
507                         UInt32 mask;
508                         jstring ibexKey;
509                     } modifiersTable[] = {
510                         { shiftKey,  JvNewStringLatin1("shift")     },
511                         { alphaLock, JvNewStringLatin1("caps_lock") },
512                         { controlKey,  JvNewStringLatin1("control")   },
513                         { optionKey,  JvNewStringLatin1("alt")       },
514                         { kEventKeyModifierNumLockMask,  JvNewStringLatin1("num_lock")  },
515                         { 0, 0 }                    
516                     };
517                     
518                     UInt32 oldModifiers = (UInt32) surface->modifiers;
519                     UInt32 newModifiers;
520                     r = WC(GetEventParameter)(e,kEventParamKeyModifiers,typeUInt32,NULL,sizeof(newModifiers),NULL,&newModifiers);
521                     checkStatus(r,"GetEventParameter");
522                     surface->modifiers = (jint) newModifiers;
523                     UInt32 changedModifiers = oldModifiers ^ newModifiers;
524                     
525                     for(int i=0;modifiersTable[i].mask;i++) {
526                         UInt32 mask = modifiersTable[i].mask;
527                         if(!(changedModifiers & mask)) continue;
528                         if(newModifiers & mask)
529                             surface->KeyPressed(modifiersTable[i].ibexKey);
530                         else
531                             surface->KeyReleased(modifiersTable[i].ibexKey);
532                     }
533                     return noErr;
534                 }
535             }
536             break;
537         case kEventClassMouse:
538             // The default handler gets first dibs on mouse events
539             // (this catches the titlebar, resize box, etc)
540             r = WC(CallNextEventHandler)(handler, e);
541             if(r != eventNotHandledErr && eKind != kEventMouseMoved && eKind != kEventMouseDragged) return r;
542             
543             switch(eKind) {
544                 case kEventMouseMoved:
545                 case kEventMouseDragged: {
546                     WindowRef window = (WindowRef) surface->rawWindowRef;
547                     Point p;
548                     Rect rect;
549                     
550                     r = WC(GetEventParameter)(e,kEventParamMouseLocation,typeQDPoint,NULL,sizeof(p),NULL,&p);
551                     checkStatus(r,"GetEventParameter");
552                     r = WC(GetWindowBounds)(window,kWindowContentRgn,&rect);
553                     checkStatus(r,"GetWindowBounds");
554                     surface->Move(p.h-rect.left,p.v-rect.top);
555                     return noErr;
556                 }
557                 case kEventMouseDown: 
558                 case kEventMouseUp: {
559                     EventMouseButton button;
560                     UInt32 clickCount;
561                     jint ibexButton;
562                     r = WC(GetEventParameter)(e,kEventParamMouseButton,typeMouseButton,NULL,sizeof(button),NULL,&button);
563                     checkStatus(r,"GetEventParameter");
564                     r = WC(GetEventParameter)(e,kEventParamClickCount,typeUInt32,NULL,sizeof(clickCount),NULL,&clickCount);
565                     checkStatus(r,"GetEventParameter");
566                     
567                     switch(button) {
568                         case kEventMouseButtonPrimary:   ibexButton = 1; break;
569                         case kEventMouseButtonSecondary: ibexButton = 2; break;
570                         case kEventMouseButtonTertiary:  ibexButton = 3; break;
571                         default: return noErr;
572                     }
573                     if(eKind == kEventMouseDown) {
574                         surface->Press(ibexButton);
575                     } else {
576                         surface->Release(ibexButton);
577                         while(clickCount > 1) {
578                             surface->DoubleClick(ibexButton);
579                             clickCount-=2;
580                         }
581                         if(clickCount) surface->Click(ibexButton);
582                     }
583                     return noErr;
584                 }
585                 case kEventMouseWheelMoved: {
586                     EventMouseWheelAxis axis;
587                     SInt32 delta;
588                     r = WC(GetEventParameter)(e,kEventParamMouseWheelAxis,typeMouseWheelAxis,NULL,sizeof(axis),NULL,&axis);
589                     checkStatus(r,"GetEventParameter");
590                     if(axis != kEventMouseWheelAxisY) break;
591                     r = WC(GetEventParameter)(e,kEventParamMouseWheelDelta,typeSInt32,NULL,sizeof(delta),NULL,&delta);
592                     checkStatus(r,"GetEventParameter");
593                     fprintf(stderr,"kEventMouseWheelMoved: delta: %d",delta);
594                     // surface->MouseWheelMoved(...) IMPROVMENT: mouse wheel support in ibex
595                     return noErr;
596                 }
597             }
598             break;
599         
600         case kEventClassWindow: {
601             WindowRef window;
602             r = WC(GetEventParameter)(e,kEventParamDirectObject,typeWindowRef,NULL,sizeof(window),NULL,&window);
603             checkStatus(r,"kEventClassWindow/WC(GetEventParameter");
604             
605             if((RawData*)window != surface->rawWindowRef) Darwin::abort(JvNewStringLatin1("window != surface->window"));
606             
607             switch(eKind) {
608                 case kEventWindowBoundsChanging: {
609                     UInt32 attr;
610                     Rect rect;
611                     
612                     r = WC(GetEventParameter)(e,kEventParamAttributes,typeUInt32,NULL,sizeof(attr),NULL,&attr);
613                     checkStatus(r,"kEventWindowBoundsChanged/GetEventParameter");
614                     
615                     r = WC(GetEventParameter)(e,kEventParamCurrentBounds,typeQDRectangle,NULL,sizeof(rect),NULL,&rect);
616                     checkStatus(r,"kEventWindowBoundsChanging/GetEventParameter"); 
617                     
618                     if(attr & kWindowBoundsChangeSizeChanged) {
619                         jint w = rect.right-rect.left;
620                         jint h = rect.bottom-rect.top;
621                             
622                         blit_lock();
623                         
624                         surface->winWidth = w;
625                         surface->winHeight = h;
626                         surface->pendingResize = true;
627                         surface->needsReshape();
628                             
629                         blit_unlock();
630     
631                         surface->SizeChange(w,h);
632                         if(attr & kWindowBoundsChangeUserResize && surface->maximized)
633                                 surface->Maximized(false);
634                     }
635                     
636                     if(attr & kWindowBoundsChangeOriginChanged) {
637                         surface->PosChange(rect.left,rect.top);
638                     }
639                     
640                     return noErr;
641                 }
642                 case kEventWindowBoundsChanged: {
643                     UInt32 attr;
644                     r = WC(GetEventParameter)(e,kEventParamAttributes,typeUInt32,NULL,sizeof(attr),NULL,&attr);
645                     checkStatus(r,"kEventWindowBoundsChanged/GetEventParameter");
646                     
647                     if(attr & kWindowBoundsChangeSizeChanged) {
648                         blit_lock();
649
650                         surface->pendingResize = false;
651                         
652                         blit_signal();
653                         blit_unlock();
654                     }
655                     return noErr;
656                 }
657                 case kEventWindowActivated:
658                 case kEventWindowDeactivated: {
659                     surface->Focused(eKind == kEventWindowActivated);
660                     return noErr;
661                 }
662                 case kEventWindowZoomed: {
663                     surface->Maximized(true);
664                     return noErr;
665                 }
666                 case kEventWindowCollapsed: {
667                     surface->Minimized(true);
668                     return noErr;
669                 }
670                 case kEventWindowExpanded: {
671                     surface->Minimized(false);
672                     return noErr;
673                 }
674                 case kEventWindowClose: {
675                     surface->Close();
676                     return noErr;
677                 }
678                 case kEventWindowClosed: {
679                     GCJ$Retainer::release(surface);
680                     return noErr;
681                 }
682                 case kEventWindowUpdate: {
683                     fprintf(stderr,"kEventWindowUpdate: Should not be here\n");
684                     abort();
685                 }
686             }
687         }
688         break;
689     }
690     return eventNotHandledErr;
691 }
692
693 static OSStatus paintProc (
694    GDHandle device,
695    GrafPtr qdContext,
696    WindowRef window,
697    RgnHandle inClientPaintRgn,
698    RgnHandle outSystemPaintRgn,
699    void *userData
700 ) {
701     Darwin$CarbonSurface *surface = (Darwin$CarbonSurface*) userData;
702     Rect rect;
703     WC(GetRegionBounds)(inClientPaintRgn, &rect);
704     surface->Dirty(rect.left,rect.top,rect.right-rect.left,rect.bottom-rect.top);
705     return noErr;
706 }
707
708 void Darwin$CarbonSurface::natInit(jboolean framed) {
709     WindowRef window;
710     Rect rect;
711     WindowClass wc = framed ? kDocumentWindowClass : kPlainWindowClass;
712     // FIXME: unframed windows should appear in the window menu
713     // This probably needs a hack similar to whats in Cocoa.mm
714     WindowAttributes attr =  kWindowStandardHandlerAttribute|
715       (framed ? kWindowInWindowMenuAttribute|kWindowStandardDocumentAttributes|kWindowLiveResizeAttribute : 0);
716     OSStatus r;
717     rect.top = rect.left = 0;
718     rect.bottom = rect.right = 10;
719     winWidth = winHeight = 10;
720     
721     r = WC(CreateNewWindow)(wc, attr, &rect, &window);
722     checkStatus(r,"CreateNewWindow");
723
724     rawWindowRef = (RawData*) window;
725
726     GCJ$Retainer::retain(this); // Need to account for the EventHandlers pointer to us
727     
728     r = InstallWindowEventHandler(window,windowEventHandlerUPP,sizeof(eventTypeSpecs)/sizeof(EventTypeSpec),eventTypeSpecs,this,NULL);
729     checkStatus(r,"InstallWindowEventHandler");
730
731     r = WC(InstallWindowContentPaintProc)(window,paintProcUPP,kWindowPaintProcOptionsNone,this);
732     checkStatus(r,"InstallWindowContentPaintProc");
733     
734     WC(ShowWindow)(window);        
735 }
736
737 void Darwin$CarbonSurface::natDispose() {
738     WindowRef window = (WindowRef) rawWindowRef;
739     WC(DisposeWindow)(window);
740 }
741
742 void Darwin$CarbonSurface::natSetIcon(org::ibex::Picture *_p) {
743 }
744
745 void Darwin$CarbonSurface::natSetLocation() {
746     WindowRef window = (WindowRef) rawWindowRef;
747     Rect rect;
748     jint x = root->x;
749     jint y = root->y;
750     OSStatus r = WC(GetWindowBounds)(window,kWindowStructureRgn,&rect);
751     checkStatus(r,"GetWindowBounds");
752     rect.bottom = y + (rect.bottom - rect.top);
753     rect.right = x + (rect.right - rect.left);
754     rect.top = y;
755     rect.left = x;
756     r = WC(SetWindowBounds)(window,kWindowStructureRgn,&rect);
757     checkStatus(r,"SetWindowBounds");
758     r = WC(ConstrainWindowToScreen)(window,kWindowStructureRgn,kWindowConstrainMoveRegardlessOfFit,NULL,NULL);
759     checkStatus(r,"ConstrainWindowToScreen");
760 }
761
762 void Darwin$CarbonSurface::natSetSize(jint w, jint h) {
763     WindowRef window = (WindowRef) rawWindowRef;
764     Rect rect;
765     OSStatus r = WC(GetWindowBounds)(window,kWindowContentRgn,&rect);
766     checkStatus(r,"GetWindowBounds");
767     rect.bottom = rect.top + h;
768     rect.right = rect.left + w;
769     r = WC(SetWindowBounds)(window,kWindowContentRgn,&rect);
770     checkStatus(r,"SetWindowBounds");
771     r = WC(ConstrainWindowToScreen)(window,kWindowStructureRgn,kWindowConstrainMoveRegardlessOfFit,NULL,NULL);
772     checkStatus(r,"ConstrainWindowToScreen");
773 }
774
775 void Darwin$CarbonSurface::natSetLimits(jint minw, jint minh, jint maxw, jint maxh) {
776     WindowRef window = (WindowRef) rawWindowRef;
777     const int maxMax = 32767;
778     const int minMinW = 80;
779     const int minMinH = 20;
780     HISize min,max;
781     min.width  = minw > maxMax ? maxMax : (minw < minMinW ? minMinW : minw);
782     min.height = minh > maxMax ? maxMax : (minh < minMinH ? minMinH : minh);
783     max.width  = maxw > maxMax ? maxMax : (maxw < minMinW ? minMinW : maxw);
784     max.height = maxh > maxMax ? maxMax : (maxh < minMinH ? minMinH : maxh);
785     OSStatus r = WC(SetWindowResizeLimits)(window,&min,&max);
786     checkStatus(r,"SetWindowResizeLimits");
787 }
788
789
790 #pragma mark ------ Carbon Methods ------
791 void fileDialogEventHandler(NavEventCallbackMessage callBackSelector, NavCBRecPtr callBackParms, void *userData) {
792     Darwin$FileDialogHelper *helper = (Darwin$FileDialogHelper*) userData;
793     NavDialogRef dialog = callBackParms->context;
794     OSStatus r;
795     switch(callBackSelector) {
796         case kNavCBUserAction: {
797             NavUserAction action = WC(NavDialogGetUserAction)(dialog);
798             if(action == kNavUserActionNone || action == kNavUserActionCancel) {
799                 helper->fileName = 0;
800             } else {
801                 NavReplyRecord reply;
802                 r = WC(NavDialogGetReply)(dialog,&reply);
803                 checkStatus(r,"NavDialogGetReply");
804
805                 AEKeyword keyword;
806                 FSRef ref;
807                 char buf[4096];
808                 r = WC(AEGetNthPtr)(&reply.selection,1,typeFSRef,&keyword,NULL,&ref,sizeof(ref),NULL);
809                 checkStatus(r,"AEGetNthPtr");
810                 r = WC(FSRefMakePath)(&ref,(UInt8*)buf,sizeof(buf)-1);
811                 checkStatus(r,"FSRefMakePath");
812                 helper->fileName = JvNewStringLatin1(buf);
813                 if(helper->save) helper->saveName = cfStringToJString(reply.saveFileName);
814                 r = WC(NavDisposeReply)(&reply);
815                 checkStatus(r,"NavDialogGetReply");
816             }
817             helper->sem->release();
818             break;
819         }
820         case kNavCBTerminate:
821             WC(NavDialogDispose)(dialog);
822             break;
823     }
824 }
825
826 void Darwin::natFileDialog(Darwin$FileDialogHelper *helper,jstring suggestion_, jboolean write) {
827     NavDialogRef dlg;
828     SmartCFString suggestion = suggestion_;
829     CFStringRef message = CFSTR("By selecting a file in this dialog you are giving this Ibex application permission to access that file.");
830     OSStatus r;
831     WindowRef window = WC(FrontWindow)();
832     NavDialogCreationOptions options;
833     
834     WC(NavGetDefaultDialogCreationOptions)(&options);
835     options.optionFlags =
836         (options.optionFlags|kNavAllFilesInPopup|kNavSelectAllReadableItem|kNavDontUseCustomFrame|kNavDontConfirmReplacement)
837         &~(kNavAllowStationery|kNavAllowMultipleFiles);
838     options.clientName = CFSTR("Ibex");
839     if(write)
840         options.saveFileName  = suggestion;
841     options.message = message;
842     options.modality = window ? kWindowModalityWindowModal : kWindowModalityAppModal;
843     options.parentWindow = window;
844     
845     if(write)
846         r = WC(NavCreatePutFileDialog)(&options,0,0,fileDialogEventHandlerUPP,helper,&dlg);
847     else
848         r = WC(NavCreateGetFileDialog)(&options,NULL,fileDialogEventHandlerUPP,NULL,NULL,helper,&dlg);
849     checkStatus(r,"NavCreate(Get/Put)FileDialog");
850     
851     WC(NavDialogRun)(dlg);
852 }
853
854 jstring Darwin::natGetClipBoard() {
855     ScrapRef scrap;
856     OSStatus r;
857     Size size,size2;
858
859     r = WC(GetCurrentScrap)(&scrap);
860     checkStatus(r,"GetCurrentScrap");
861     
862     r = WC(GetScrapFlavorSize)( scrap, kScrapFlavorTypeUnicode, &size);
863     if(r == scrapFlavorNotFoundErr) return JvNewStringLatin1("");
864     checkStatus(r,"GetScrapFlavorSize");
865     
866     unsigned int length = size/sizeof(UniChar);
867     
868     jstring js = JvAllocString(length);
869     UniChar *buf = (UniChar*) JvGetStringChars(js);
870     size2 = size;
871     r = WC(GetScrapFlavorData)(scrap,kScrapFlavorTypeUnicode,&size2,buf);
872     if(r == scrapFlavorNotFoundErr);
873     checkStatus(r,"GetScrapFlavorData");
874     if(size2 != size) return JvNewStringLatin1("");
875     
876     return js;
877 }
878
879 void Darwin::natSetClipBoard(jstring js) {
880     unsigned int length = js->length();
881     ScrapRef scrap;
882     OSStatus r;
883
884     r = WC(GetCurrentScrap)(&scrap);
885     checkStatus(r,"GetCurrentScrap");
886     
887     r = WC(PutScrapFlavor)(scrap,kScrapFlavorTypeUnicode,0,sizeof(UniChar)*length,JvGetStringChars(js));
888     checkStatus(r,"PutScrapFlavor");
889 }
890
891 HTTP$Proxy *Darwin::natDetectProxy() {
892     using org::ibex::HTTP$Proxy;
893     HTTP$Proxy *p=0;
894     CFStringRef string;
895     CFNumberRef number;
896     SmartCFString smartString;
897     CFArrayRef exceptionList;
898     int i;
899     
900     CFDictionaryRef proxyInfo = WC(SCDynamicStoreCopyProxies)(NULL);
901     if(proxyInfo == NULL) return 0;
902     
903 #define doproto(proto,var) \
904     number = (CFNumberRef) WC(CFDictionaryGetValue)(proxyInfo,kSCPropNetProxies ## proto ## Enable);    \
905     if(number != NULL && WC(CFGetTypeID)(number) != WC(CFNumberGetTypeID)()) number = NULL;                 \
906     if(number && WC(CFNumberGetValue)(number,kCFNumberIntType,&i) && i) {                               \
907         string = (CFStringRef) WC(CFDictionaryGetValue)(proxyInfo, kSCPropNetProxies ## proto ## Proxy);\
908         if(string != NULL && WC(CFGetTypeID)(string) != WC(CFStringGetTypeID)()) string = NULL;             \
909         number = (CFNumberRef) WC(CFDictionaryGetValue)(proxyInfo, kSCPropNetProxies ## proto ## Port); \
910         if(number != NULL && WC(CFGetTypeID)(number) != WC(CFNumberGetTypeID)()) number = NULL;    \
911         if(string && number && WC(CFNumberGetValue)(number,kCFNumberIntType,&i) && i) {        \
912             if(!p) p = new HTTP$Proxy();                                                        \
913             p->var ## ProxyHost = cfStringToJString(string);                           \
914             p->var ## ProxyPort = i;                                                   \
915         }                                                                                  \
916     }                                                                                      
917 doproto(HTTP,http)
918 doproto(HTTPS,https)
919 doproto(SOCKS,socks)
920 #undef doproto
921
922     exceptionList = (CFArrayRef) WC(CFDictionaryGetValue)(proxyInfo,kSCPropNetProxiesExceptionsList);
923     if(exceptionList != NULL && WC(CFGetTypeID)(exceptionList) != WC(CFArrayGetTypeID)()) exceptionList = NULL;
924     if(p && exceptionList && WC(CFArrayGetCount)(exceptionList)) {
925         CFIndex count = WC(CFArrayGetCount)(exceptionList);
926         p->excluded = (JArray<java::lang::String*>*)
927             JvNewObjectArray(count,&java::lang::String::class$,0);
928         for(i=0;i<count;i++) {
929             string = (CFStringRef) WC(CFArrayGetValueAtIndex)(exceptionList,i);
930             if(string != NULL && WC(CFGetTypeID)(string) != WC(CFStringGetTypeID)()) string = NULL;
931             elements(p->excluded)[i] = string ? cfStringToJString(string) : JvNewStringLatin1("(null)");
932         }
933     }
934     WC(CFRelease)(proxyInfo);
935     
936     return p;
937 /*
938     exceptionList = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesExceptionsList];
939     if(p && exceptionList && [exceptionList count]) {
940         NSLog(@"excl: %@",exceptionList);
941         p->excluded = (JArray<java::lang::String*>*)
942             JvNewObjectArray([exceptionList count],&java::lang::String::class$,0);
943         for(i=0;i<[exceptionList count];i++)
944             elements(p->excluded)[i] = [[exceptionList objectAtIndex: i] jstring];
945     }
946     return p;
947     */    
948             
949 /*
950     using org::ibex::Proxy;
951     AutoARP pool;
952     Proxy *p=0;
953     NSString *host;
954     NSNumber *port;
955     NSArray *exceptionList;
956     unsigned int i;
957     NSDictionary *proxyInfo = (NSDictionary*)WC(SCDynamicStoreCopyProxies)(NULL);
958     
959     if(proxyInfo == NULL) return 0;
960     if([[proxyInfo objectForKey: (NSString*) kSCPropNetProxiesHTTPEnable] boolValue]) {
961         host = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesHTTPProxy];
962         port = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesHTTPPort];
963         if(host && [port intValue]) {
964             if(!p) p = new Proxy();
965             p->httpProxyHost = [host jstring];
966             p->httpProxyPort = [port intValue];
967         }
968     }
969     if([[proxyInfo objectForKey: (NSString*) kSCPropNetProxiesHTTPSEnable] boolValue]) {
970         host = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesHTTPSProxy];
971         port = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesHTTPSPort];
972         if(host && [port intValue]) {
973             if(!p) p = new Proxy();
974             p->httpsProxyHost = [host jstring];
975             p->httpsProxyPort = [port intValue];
976         }
977     }
978     if([[proxyInfo objectForKey: (NSString*) kSCPropNetProxiesSOCKSEnable] boolValue]) {
979         host = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesSOCKSProxy];
980         port = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesSOCKSPort];
981         if(host && [port intValue]) {
982             if(!p) p = new Proxy();
983             p->socksProxyHost = [host jstring];
984             p->socksProxyPort = [port intValue];
985         }
986     }
987    
988     exceptionList = [proxyInfo objectForKey: (NSString*) kSCPropNetProxiesExceptionsList];
989     if(p && exceptionList && [exceptionList count]) {
990         NSLog(@"excl: %@",exceptionList);
991         p->excluded = (JArray<java::lang::String*>*)
992             JvNewObjectArray([exceptionList count],&java::lang::String::class$,0);
993         for(i=0;i<[exceptionList count];i++)
994             elements(p->excluded)[i] = [[exceptionList objectAtIndex: i] jstring];
995     }
996     return p;
997 */
998 }
999
1000 jint Darwin::cgScreenWidth() {
1001     return WC(CGDisplayPixelsWide)(((CGDirectDisplayID)0));
1002 }
1003
1004 jint Darwin::cgScreenHeight() {
1005     return WC(CGDisplayPixelsHigh)(((CGDirectDisplayID)0));
1006 }
1007
1008 void Darwin::_newBrowserWindow(jstring js) {
1009     SmartCFString cfs = js;
1010     CFURLRef url = WC(CFURLCreateWithString)(WC(kCFAllocatorDefault),cfs,NULL);
1011     SmartCFString scheme = WC(CFURLCopyScheme)(url);
1012     if(scheme.equals(CFStringRef("http")))
1013         WC(LSOpenCFURLRef)(url,NULL);
1014     WC(CFRelease)(url);
1015 }
1016
1017 void Darwin::natInit() {     
1018     OSStatus r;
1019     
1020     if(pthread_mutex_init(&blit_mutex,NULL) != 0) funcFailed("pthread_mutex_init");
1021     if(pthread_cond_init(&blit_cond,NULL) != 0) funcFailed("pthread_cond_init");
1022     
1023     carbonMessageEventHandlerUPP = WC(NewEventHandlerUPP)(carbonMessageEventHandler);
1024     windowEventHandlerUPP = WC(NewEventHandlerUPP)(windowEventHandler);
1025     fileDialogEventHandlerUPP = WC(NewNavEventUPP)(fileDialogEventHandler);
1026     paintProcUPP = WC(NewWindowPaintUPP)(paintProc);
1027
1028     #ifdef Ibex_CARBON_NO_BUNDLE_HACK
1029     {        
1030         ProcessSerialNumber currentProcess = { 0, kCurrentProcess };
1031         
1032         ::fprintf(stderr,"Doing Ibex_CARBON_NO_BUNDLE_HACK\n");
1033         r = WC(GetCurrentProcess)(&currentProcess);
1034         checkStatus(r,"GetCurrentProcess");
1035         r = WC(CPSEnableForegroundOperation)( &currentProcess );
1036         checkStatus(r,"CPSEnableForegroundOperation");
1037         r = WC(CPSSetFrontProcess)(&currentProcess);
1038         checkStatus(r,"CPSSetFrontProcess");        
1039     }
1040     #else
1041     {
1042         IBNibRef nib;
1043         r = CreateNibReference(CFSTR("MainMenu"), &nib);
1044         checkStatus(r,"CreateNibReference");
1045         r = SetMenuBarFromNib(nib, CFSTR("MenuBar"));
1046         checkStatus(r,"SetMenuBarFromNib");
1047         DisposeNibReference(nib);
1048         
1049         // FEATURE: Install menu event handler
1050     }
1051     #endif
1052 }
1053
1054 void Darwin::runApplicationEventLoop() {
1055     WC(RunApplicationEventLoop)();
1056     WC(ExitToShell)();
1057 }
1058
1059 #pragma mark ------ OpenGL Functions -----
1060
1061 void Darwin$CarbonOpenGL::activateSharedContext() {
1062     AGLContext ctx = (AGLContext) rawSharedContext;
1063     WC(aglSetCurrentContext)(ctx);
1064 }
1065
1066 jboolean Darwin$CarbonOpenGL::initPixelFormat() {
1067     GLint attr[] = {
1068         AGL_NO_RECOVERY,
1069         AGL_RGBA,
1070         AGL_DEPTH_SIZE, 32,
1071         AGL_RED_SIZE, 8,
1072         AGL_GREEN_SIZE, 8,
1073         AGL_RED_SIZE, 8,
1074         AGL_ALPHA_SIZE, 8,
1075         AGL_NONE
1076     };
1077     
1078     rawPixelFormat = (RawData*) WC(aglChoosePixelFormat)(NULL,0,attr);
1079     return rawPixelFormat != 0;
1080 }
1081
1082 void Darwin$CarbonOpenGL::initSharedContext() {
1083     AGLPixelFormat fmt =  (AGLPixelFormat) rawPixelFormat;
1084     rawSharedContext = (RawData*) WC(aglCreateContext)(fmt,NULL);
1085     checkStatus(rawSharedContext,"aglCreateContext");
1086 }
1087
1088 void Darwin$GLCarbonPixelBuffer::natInit() {
1089     WindowClass wc = kPlainWindowClass;
1090     WindowAttributes attr = 0;
1091     WindowRef window;
1092     Rect rect;
1093     OSStatus r;
1094     AGLContext ctx,shared;
1095     AGLPixelFormat fmt;
1096     GLboolean b;
1097     GLuint tex;
1098     GLuint target;
1099     
1100     rect.top = rect.left = 0;
1101     rect.right = width + rect.left;
1102     rect.bottom = height + rect.top; 
1103     
1104     r = WC(CreateNewWindow)(wc,attr,&rect,&window);
1105     checkStatus(r,"CreateNewWindow");
1106         
1107     shared = (AGLContext) gl->rawSharedContext;
1108     fmt = (AGLPixelFormat) gl->rawPixelFormat;
1109     ctx = WC(aglCreateContext)(fmt,shared);
1110     checkStatus(ctx, "aglCreateContext");
1111     
1112     b = WC(aglSetDrawable)(ctx,WC(GetWindowPort)(window));
1113     checkStatus(b,"aglSetDrawable");
1114     
1115     WC(aglSetCurrentContext)(ctx);
1116     WC(aglUpdateContext)(ctx);
1117     drawableInit(width,height);
1118     glClear(GL_COLOR_BUFFER_BIT);
1119
1120     WC(aglSetCurrentContext)(shared);
1121     target = rectTexture ? GL_TEXTURE_RECTANGLE_EXT : GL_TEXTURE_2D;
1122     glEnable(target);
1123     glGenTextures(1,&tex);
1124     glBindTexture(target,tex);
1125     glTexParameterf(target, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
1126     glTexParameterf(target, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
1127     WC(aglSurfaceTexture) (shared, target, GL_RGBA, ctx);
1128     checkGLError();
1129     glDisable(target);
1130     
1131     //WC(ShowWindow)(window);
1132     textureName = (jint) tex;
1133     rawWindowRef = (RawData*) window;
1134     rawCTX = (RawData*) ctx;
1135 }
1136
1137 void Darwin$GLCarbonPixelBuffer::activateContext() {
1138     AGLContext ctx = (AGLContext) rawCTX;
1139     WC(aglSetCurrentContext)(ctx);
1140 }
1141
1142 void Darwin$GLCarbonPixelBuffer::natCleanup(RawData* rawWindowRef, RawData* rawCTX) {
1143     WindowRef window = (WindowRef) rawWindowRef;
1144     AGLContext ctx = (AGLContext) rawCTX;
1145     WC(aglDestroyContext)(ctx);
1146     WC(DisposeWindow)(window);
1147 }
1148
1149 void Darwin$GLCarbonSurface::clear() {
1150     AGLContext ctx = (AGLContext) rawCTX;
1151     WC(aglSetCurrentContext)(ctx);
1152     glColor4f(1.0f,1.0f,1.0f,1.0f);
1153
1154     glMatrixMode (GL_MODELVIEW);
1155     glPushMatrix ();
1156     glLoadIdentity ();
1157     glMatrixMode (GL_PROJECTION);
1158     glPushMatrix ();
1159     glLoadIdentity ();
1160     glBegin (GL_QUADS);
1161     glVertex3i (-1, -1, -1);
1162     glVertex3i (1, -1, -1);
1163     glVertex3i (1, 1, -1);
1164     glVertex3i (-1, 1, -1);
1165     glEnd ();
1166     glPopMatrix ();
1167     glMatrixMode (GL_MODELVIEW);
1168     glPopMatrix ();
1169 }
1170
1171 void Darwin$GLCarbonSurface::flush() {
1172     AGLContext ctx = (AGLContext) rawCTX;
1173     WC(aglSetCurrentContext)(ctx);
1174     glFlush();
1175 }
1176
1177 // blit_lock is assumed to be held
1178 void Darwin$GLCarbonSurface::reshape(jint w, jint h) {
1179     AGLContext ctx = (AGLContext) rawCTX;
1180     WC(aglSetCurrentContext)(ctx);
1181     WC(aglUpdateContext)(ctx);
1182     
1183     glViewport(0,0,w,h);
1184     glMatrixMode(GL_PROJECTION);
1185     glLoadIdentity(); 
1186     glOrtho(0,w,h,0,-1,1);   
1187     glMatrixMode(GL_MODELVIEW);
1188     glLoadIdentity();
1189     glTranslatef (0.375f, 0.375f, 0.0f);
1190     checkGLError();
1191 }
1192
1193 // blit_lock is assumed to be held
1194 void Darwin$GLCarbonSurface::natBlit(Darwin$GLCarbonPixelBuffer *db, jint sx1, jint sy1, jint dx1, jint dy1, jint dx2, jint dy2) {
1195     AGLContext ctx = (AGLContext) rawCTX;
1196     int sx2 = sx1 + (dx2-dx1);
1197     int sy2 = sy1 + (dy2-dy1);
1198     jint w, h;
1199     db->activateContext();
1200     glFlush();
1201     checkGLError();
1202
1203     WC(aglSetCurrentContext)(ctx);
1204     checkGLError();
1205         
1206     if(db->rectTexture) {
1207         glEnable(GL_TEXTURE_RECTANGLE_EXT);
1208         checkGLError();
1209         glBindTexture(GL_TEXTURE_RECTANGLE_EXT, db->textureName);
1210         checkGLError();
1211         glBegin(GL_QUADS); 
1212             glTexCoord2i (sx1, sy1   );
1213             glVertex3i   (dx1, dy1, 0);   
1214             glTexCoord2i (sx2, sy1   ); 
1215             glVertex3i   (dx2, dy1, 0);
1216             glTexCoord2i (sx2, sy2   ); 
1217             glVertex3i   (dx2, dy2, 0);
1218             glTexCoord2i (sx1, sy2   );
1219             glVertex3i   (dx1, dy2, 0);
1220         glEnd();
1221         checkGLError();
1222         glDisable(GL_TEXTURE_RECTANGLE_EXT);
1223         checkGLError();
1224     } else {
1225         float tx1,ty1,tx2,ty2; // normalized texture coords
1226         tx1 = (float) sx1 / (float) db->width;
1227         ty1 = (float) sy1 / (float) db->height;
1228         tx2 = (float) sx2 / (float) db->width;
1229         ty2 = (float) sy2 / (float) db->height;
1230
1231         glColor4f(1.0f,1.0f,1.0f,1.0f);
1232         glEnable(GL_TEXTURE_2D);
1233         glBindTexture(GL_TEXTURE_2D, db->textureName);    
1234         checkGLError();
1235         glBegin(GL_QUADS); 
1236             glTexCoord2f (tx1, ty1   );
1237             glVertex3i   (dx1, dy1, 0);
1238             glTexCoord2f (tx2, ty1   ); 
1239             glVertex3i   (dx2, dy1, 0);
1240             glTexCoord2f (tx2, ty2   ); 
1241             glVertex3i   (dx2, dy2, 0);
1242             glTexCoord2f (tx1, ty2   );
1243             glVertex3i   (dx1, dy2, 0);
1244         glEnd();
1245         glDisable(GL_TEXTURE_2D);
1246         checkGLError();
1247     }
1248 }
1249
1250 void Darwin$GLCarbonSurface::natInit() {
1251     WindowRef window = (WindowRef) rawWindowRef;
1252     AGLContext ctx,shared;
1253     AGLPixelFormat fmt;
1254     GLboolean b;
1255     
1256     shared = (AGLContext) gl->rawSharedContext;
1257     fmt = (AGLPixelFormat) gl->rawPixelFormat;
1258     ctx = WC(aglCreateContext)(fmt,shared);
1259     checkStatus(ctx, "aglCreateContext");
1260     
1261     b = WC(aglSetDrawable)(ctx,WC(GetWindowPort)(window));
1262     checkStatus(b,"aglSetDrawable");    
1263
1264     WC(aglSetCurrentContext)(ctx);
1265     glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST);
1266     
1267     glClearColor (1.0f, 1.0f, 1.0f, 1.0f);
1268     glClearDepth( 0.0f );
1269     
1270     needsReshape();
1271     
1272     rawCTX = (RawData*)ctx;
1273 }
1274
1275 void Darwin$GLCarbonSurface::natDispose() {
1276     AGLContext ctx = (AGLContext) rawCTX;
1277     WC(aglDestroyContext)(ctx);
1278     Darwin$CarbonSurface::natDispose();
1279 }
1280
1281 } } } // end namepsace org::ibex::plat
1282
1283
1284