// Micro-GLUT, bare essentials // Mac version! // Single-file GLUT subset. // By Ingemar Ragnemalm 2012-2015 // I wrote this since GLUT seems not to have been updated to support // creation of a 3.2 context on the Mac, plus that GLUT contains too // much deprecated material, and is impractical to modify. With a // single-file solution like this, there are no installation problems // at all and it is also easy to hack around in a local copy as needed. // Furthermore, this also removes all old deprecated code, and gives // better transparency. (Hack away!) // To do: iOS port. (Preliminary versions exists.) // Some additional features. Text rendering? Menus? (OK) Multi-window? // Changes: // Several additions for the lab 3 version: Simple menu, glutTimerFunc and my own glutrepeatingTimerFunc. // glutKeyboardUpFunc, glutInitDisplayMode. // 120209: Some more additions, like glutMotionFunc and GLUT_RIGHT_BUTTON. // 120228: Added glutSetWindowTitle, glutKeyIsDown, glutInitContextVersion // NOTE: glutInitContextVersion is now required for 3.2 code! // NOTE: This statement is incorrect; MicroGLUT still only supports 3.2 and with no extra calls. // Note however the GL3ONLY define. // 120301: Resizable window. Correct vertical window position. // 120808: Fixed a bug that caused error messages. (Calling finishLaunching twice.) // 120822: Stencil now uses 8 instead of 32 // 120905: Two corrections suggested by Marcus StenbŠck. // 120913: Added 2-button emulation with CTRL // 130103: Added [m_context makeCurrentContext]; in all user callbacks. (GL calls had no effect.) // 130127: Added basic popup menu support // 130214: Added glutSpecialFunc and glutSpecialUpFunc. Sets focus to window when created. // 130220: Modified idle support, added fake visibility support. // 130325: Corrected name of glutWarpPointer // 130330: Added GLUT_MULTISAMPLE // 130331: Added glutChangeMenuEntry // 130513: Added GLUT_WINDOW_WIDTH and GLUT_WINDOW_HEIGHT support to glutGet (which is still lacking a whole bunch of other types). Modified glutWarpPointer. // 130907: Added isFlipped to make mouse coordinates reasonable (and conform with Linux version). // 131011: Makes the context active before calling gReshape in case the reshape function calls OpenGL calls. // 140213: Rewrote glutWarpPointer. // 150205: Added glutPositionWindow and glutFullScreen plus glutExitFullScreen. // 150209: Small correction to full-screen mode, restores windows name! // 150424: Rewrote glutWarpPointer again! // 150520: Added glutExit(). // 150618: Added glutMouseIsDown() (not in the old GLUT API but a nice extension!). // Added #ifdefs to produce errors if compiled on the wrong platform! // 150820: Minor bug fix in glutReshapeWindow. // 150827: Hacked around the needlessly deprecated convertBaseToScreen. Also skipped // CGSetLocalEventsSuppressionInterval which MIGHT not be needed... // 150918: More keys map properly to the GLUT constants, like GLUT_KEY_ESC. // Rewrote glutCheckLoop, now it seems to work fine. Tested with the "triangle" demo that lacks main loop. // Added GLUT_QUIT_FLAG for glutGet. // 150919: Added GLUT_MOUSE_POSITION_X and GLUT_MOUSE_POSITION_Y to glutGet. // 150923: Changed the keyboard special key constants to avoid all usable ASCII codes. This makes the "special key" calls unnecessary and keyboard handling simpler. We can stop using "special key" callbacks alltogether. // Some minor changes not documented. // 171019: Fixed a bug in context version handling. // 180208: Added activateIgnoringOtherApps to bring it to front when launched. // #include removed. Might be needed on older OSX! // 181206: Changed the GlutView to inherit from NSOpenGLView instead of NSView. // 200926: Replaced a number of constants that have been renamed by Apple. // Also put in GL_SILENCE_DEPRECATION. // 210220: Change in drawRect, avoiding problem with random freezes. Also set swap to 1 = sync. // Incompatibility in mouse coordinates; global or local? // Should be local! (I think they are now!) // This silences most warnings but not the NS ones. // -Wno-deprecated-declarations on command-line will silence all. #define GL_SILENCE_DEPRECATION #import #include #include #include #include #include #include #include "MicroGlut.h" // If this is compiled on something else than a Mac, tell me! #ifndef __APPLE__ ERROR! This (MicroGlut.m) is the Mac version of MicroGlut which will not work on other platforms! #endif // Comment out to support GL2, requiring glutInitContextVersion for newer. //#define GL3ONLY // Vital internal variables void (*gDisplay)(void); void (*gReshape)(int width, int height); void (*gKey)(unsigned char key, int x, int y); void (*gSpecialKey)(unsigned char key, int x, int y); void (*gKeyUp)(unsigned char key, int x, int y); void (*gSpecialKeyUp)(unsigned char key, int x, int y); void (*gMouseMoved)(int x, int y); void (*gMouseDragged)(int x, int y); void (*gMouseFunc)(int button, int state, int x, int y); unsigned int gContextInitMode = GLUT_RGBA | GLUT_DOUBLE | GLUT_DEPTH; void (*gIdle)(void); char updatePending = 1; char gRunning = 1; char gKeymap[256]; char gButtonPressed[10] = {0,0,0,0,0,0,0,0,0,0}; int gContextVersionMajor = 0; int gContextVersionMinor = 0; char gTitle[255] = "MicroGlut window"; // Saved title, so we can restore after full screen! // ----------- // Globals (was in GLViewDataPtr) NSOpenGLContext *m_context; float lastWidth, lastHeight; NSView *theView; // Full screen support NSRect gSavedWindowPosition; char gFullScreen = 0; // See glutReshapeWindow, glutPositionWindow and glutFullScreen void MakeContext(NSView *view) { NSOpenGLPixelFormat *fmt; int zdepth, sdepth, i; int profile = 0, profileVersion = 0; if (gContextInitMode & GLUT_DEPTH) zdepth = 32; else zdepth = 0; if (gContextInitMode & GLUT_STENCIL) sdepth = 8; else sdepth = 0; NSOpenGLPixelFormatAttribute attrs[] = { // NSOpenGLPFAPixelBuffer, // NSOpenGLPFAAccelerated, // NSOpenGLPFADoubleBuffer, NSOpenGLPFADepthSize, zdepth, NSOpenGLPFAStencilSize, sdepth, 0,0,0,0,0,0,0 // profile, profileVersion, // multi, }; i = 4; // Def this out about for disabling profile selection support, // if it is on we are defaulting to GL 1/2 #ifndef GL3ONLY if (gContextVersionMajor >= 3) #endif // I ignore the minor version for now, 3.2 is all we can choose currently { profile = NSOpenGLPFAOpenGLProfile; profileVersion = NSOpenGLProfileVersion3_2Core; attrs[i++] = profile; attrs[i++] = profileVersion; } if (gContextInitMode & GLUT_DOUBLE) { attrs[i++] = NSOpenGLPFADoubleBuffer; } if (gContextInitMode & GLUT_MULTISAMPLE) { attrs[i++] = NSOpenGLPFAMultisample; } // Save view (should be packaged with context for multi-window application - to do) theView = view; // Init GL context fmt = [[NSOpenGLPixelFormat alloc] initWithAttributes: &attrs[0]]; m_context = [[NSOpenGLContext alloc] initWithFormat: fmt shareContext: nil]; [fmt release]; // [m_context makeCurrentContext]; int iSwap = 1; [m_context setValues:&iSwap forParameter:NSOpenGLCPSwapInterval]; // [m_context setView: theView]; [m_context makeCurrentContext]; } // ---------------------- Menus ---------------------- // MicroGlut menu support // This is a subset of the GLUT menu support, focused on what seems to be used // in current GLUT-using demos. If nobody uses it, don't add it (unless it is // a feature that you really want people to learn using). typedef void (*MenuProc)(int value); @interface TPopupMenu : NSMenu {@public MenuProc funcCallback;} -(void)selectMenuItem:(id *)sender; @end @implementation TPopupMenu -(void)selectMenuItem:(id *)sender { funcCallback( [((NSMenuItem *) sender) tag]); } @end TPopupMenu *currentMenu; TPopupMenu *buttons[10]; // Menu by button id - for a button, what menu should be used, if any? TPopupMenu *menuList[10]; // List of menus by (internal) menu id int menuCount = -1; int glutCreateMenu(void (*func)(int value)) { currentMenu = [[TPopupMenu alloc] initWithTitle: @""]; currentMenu->funcCallback = func; menuCount += 1; menuList[menuCount] = currentMenu; return menuCount; } void glutAddMenuEntry(char *name, int value) { NSMenuItem *menuItem1; menuItem1 = NSMenuItem.alloc; // string to NSString NSString * s = [NSString stringWithCString: name encoding: NSASCIIStringEncoding]; [menuItem1 initWithTitle: s action: nil keyEquivalent: @""]; [menuItem1 setTarget: currentMenu]; // Who will handle it? [menuItem1 setAction: @selector(selectMenuItem:)]; [menuItem1 setTag: value]; // Save value [currentMenu addItem: menuItem1]; } void glutChangeToMenuEntry(int index, char *name, int value) { NSMenuItem *menuItem1; menuItem1 = [currentMenu itemAtIndex: index-1]; if (menuItem1 != NULL) { // string to NSString NSString * s = [NSString stringWithCString: name encoding: NSASCIIStringEncoding]; [menuItem1 setTarget: currentMenu]; // Who will handle it? [menuItem1 setAction: @selector(selectMenuItem:)]; [menuItem1 setTag: value]; // Save value [menuItem1 setTitle: s]; } } void glutAttachMenu(int button) { // set a data item for the button buttons[button] = currentMenu; } void glutAddSubMenu(char *name, int menu) { NSMenuItem *item; NSString * s = [NSString stringWithCString: name encoding: NSASCIIStringEncoding]; item = [currentMenu addItemWithTitle: s action: nil keyEquivalent: @""]; [currentMenu setSubmenu: menuList[menu] forItem: item]; } void glutDetachMenu(int button) { // reset a data item for the button buttons[button] = nil; } // Check name on this! void glutSetMenu(int menu) { currentMenu = menuList[menu]; } int glutGetMenu(void) { int i; for (i = 0; i <= menuCount; i++) if (menuList[i] == currentMenu) return i; return -1; } // End of MicroGlut button support static void doKeyboardEvent(NSEvent *theEvent, void (*func)(unsigned char key, int x, int y), void (*specialfunc)(unsigned char key, int x, int y), int keyMapValue) { char *chars; chars = (char *)[[theEvent characters] cStringUsingEncoding: NSMacOSRomanStringEncoding]; NSPoint mouseDownPos = [theEvent locationInWindow]; if (chars != NULL) { if (func != NULL) // Change 120913 func(chars[0], mouseDownPos.x, mouseDownPos.y); // TO DO: x and y gKeymap[(unsigned int)chars[0]] = keyMapValue; } else { char code; switch( [theEvent keyCode] ) { case 0x35: code = GLUT_KEY_ESC; break; case 0x30: code = GLUT_KEY_TAB; break; case 0x24: code = GLUT_KEY_RETURN; break; case 0x31: code = GLUT_KEY_SPACE; break; case 0x29: code = GLUT_KEY_SEMICOLON; break; case 0x2B: code = GLUT_KEY_COMMA; break; case 0x2F: code = GLUT_KEY_DECIMAL; break; case 0x32: code = GLUT_KEY_GRAVE; break; case 0x27: code = GLUT_KEY_QUOTE; break; case 0x21: code = GLUT_KEY_LBRACKET; break; case 0x1E: code = GLUT_KEY_RBRACKET; break; case 0x2A: code = GLUT_KEY_BACKSLASH; break; case 0x2C: code = GLUT_KEY_SLASH; break; case 0x18: case 0x51: code = GLUT_KEY_EQUAL; break; case 126: code = GLUT_KEY_UP; break; case 125: code = GLUT_KEY_DOWN; break; case 124: code = GLUT_KEY_RIGHT; break; case 123: code = GLUT_KEY_LEFT; break; case 122: code = GLUT_KEY_F1; break; case 120: code = GLUT_KEY_F2; break; case 99: code = GLUT_KEY_F3; break; case 118: code = GLUT_KEY_F4; break; case 96: code = GLUT_KEY_F5; break; case 97: code = GLUT_KEY_F6; break; case 98: code = GLUT_KEY_F7; break; case 115: code = GLUT_KEY_HOME; break; // ? case 116: code = GLUT_KEY_PAGE_UP; break; case 119: code = GLUT_KEY_END; break; // ? case 121: code = GLUT_KEY_PAGE_DOWN; break; case 117: code = GLUT_KEY_INSERT; break; // Looks more like DEL? default: code = [theEvent keyCode]; } if (specialfunc != NULL) // Change 130114 specialfunc(code, mouseDownPos.x, mouseDownPos.y); // TO DO: x and y else // If no special, send to normal (future preferred way) if (func != NULL) // Change 150114 func(code, mouseDownPos.x, mouseDownPos.y); // TO DO: x and y // NOTE: This was a bug until I modified the special key constants! We can now check gKeymap with normal ASCII and special codes with the same table! gKeymap[(int)code] = keyMapValue; } } // -------------------- View ------------------------ @interface GlutView : NSOpenGLView { } // Changed NSView to NSOpenGLView, and it worked. -(void)drawRect:(NSRect)rect; -(void)keyDown:(NSEvent *)theEvent; -(void)keyUp:(NSEvent *)theEvent; -(void)mouseMoved:(NSEvent *)theEvent; -(void)mouseDragged:(NSEvent *)theEvent; -(void)mouseDown:(NSEvent *)theEvent; -(void)mouseUp:(NSEvent *)theEvent; -(void)rightMouseDown:(NSEvent *)theEvent; -(void)rightMouseUp:(NSEvent *)theEvent; -(void)windowDidresize:(NSNotification *)note; -(BOOL)isFlipped; //-(void)setLayer:(CALayer*)layer; // Mojave??? @end #define Pi 3.1415 // Mouse position is saved so it can be retrieved at any time with glutGet NSPoint gMousePosition; @implementation GlutView -(void) mouseMoved:(NSEvent *)theEvent { // NSPoint p; [m_context makeCurrentContext]; gMousePosition = [theEvent locationInWindow]; gMousePosition = [self convertPoint: gMousePosition fromView: nil]; if (gMouseMoved != nil) { gMouseMoved(gMousePosition.x, gMousePosition.y); } } -(void) mouseDragged:(NSEvent *)theEvent { NSPoint p; [m_context makeCurrentContext]; if (gMouseDragged != nil) { p = [theEvent locationInWindow]; p = [self convertPoint: p fromView: nil]; gMouseDragged((int)p.x, (int)p.y); } } // Clone of above, but necessary for supporting the alternative button. Thanks to Marcus StenbŠck! -(void) rightMouseDragged:(NSEvent *)theEvent { NSPoint p; [m_context makeCurrentContext]; if (gMouseDragged != nil) { p = [theEvent locationInWindow]; p = [self convertPoint: p fromView: nil]; gMouseDragged((int)p.x, (int)p.y); } } // Remember if last press on the left (default) button was modified to // a "right" with CTRL char gLeftIsRight = 0; -(void) mouseDown:(NSEvent *)theEvent { NSPoint p; [m_context makeCurrentContext]; gButtonPressed[0] = 1; // Could maybe be modified by NSControlKeyMask...? if (gMouseFunc != nil) { // Convert location in window to location in view p = [theEvent locationInWindow]; // printf("mouseDown before convertPoint %f %f \n", p.x, p.y); p = [self convertPoint: p fromView: nil]; // printf("mouseDown %f %f \n", p.x, p.y); // if ([NSEvent modifierFlags] & NSControlKeyMask) if ([NSEvent modifierFlags] & NSEventModifierFlagControl) { gMouseFunc(GLUT_RIGHT_BUTTON, GLUT_DOWN, p.x, p.y); gLeftIsRight = 1; } else { gMouseFunc(GLUT_LEFT_BUTTON, GLUT_DOWN, p.x, p.y); gLeftIsRight = 0; } } // else // if ([NSEvent modifierFlags] & NSControlKeyMask) if ([NSEvent modifierFlags] & NSEventModifierFlagControl) { if (buttons[GLUT_RIGHT_BUTTON] != nil) [NSMenu popUpContextMenu: buttons[GLUT_RIGHT_BUTTON] withEvent: theEvent forView: (NSButton *)self]; } else { if (buttons[GLUT_LEFT_BUTTON] != nil) [NSMenu popUpContextMenu: buttons[GLUT_LEFT_BUTTON] withEvent: theEvent forView: (NSButton *)self]; } } -(void) mouseUp:(NSEvent *)theEvent { NSPoint p; [m_context makeCurrentContext]; gButtonPressed[0] = 0; if (gMouseFunc != nil) { // Convert location in window to location in view p = [theEvent locationInWindow]; p = [self convertPoint: p fromView: nil]; // Assuming that the user won't release CTRL - then it looks like different buttons if (gLeftIsRight) gMouseFunc(GLUT_RIGHT_BUTTON, GLUT_UP, p.x, p.y); else gMouseFunc(GLUT_LEFT_BUTTON, GLUT_UP, p.x, p.y); } } -(void) rightMouseDown:(NSEvent *)theEvent { NSPoint p; [m_context makeCurrentContext]; gButtonPressed[1] = 1; if (gMouseFunc != nil) { // Convert location in window to location in view p = [theEvent locationInWindow]; p = [self convertPoint: p fromView: nil]; gMouseFunc(GLUT_RIGHT_BUTTON, GLUT_DOWN, p.x, p.y); } else if (buttons[GLUT_RIGHT_BUTTON] != nil) [NSMenu popUpContextMenu: buttons[GLUT_RIGHT_BUTTON] withEvent: theEvent forView: (NSButton *)self]; } -(void) rightMouseUp:(NSEvent *)theEvent { NSPoint p; [m_context makeCurrentContext]; gButtonPressed[1] = 0; if (gMouseFunc != nil) { // Convert location in window to location in view p = [theEvent locationInWindow]; p = [self convertPoint: p fromView: nil]; gMouseFunc(GLUT_RIGHT_BUTTON, GLUT_UP, p.x, p.y); } } -(void)keyDown:(NSEvent *)theEvent { [m_context makeCurrentContext]; doKeyboardEvent(theEvent, gKey, gSpecialKey, 1); } -(void)keyUp:(NSEvent *)theEvent { [m_context makeCurrentContext]; doKeyboardEvent(theEvent, gKeyUp, gSpecialKeyUp, 0); } - (BOOL)acceptsFirstResponder { return YES; } - (BOOL)becomeFirstResponder { return YES; } - (BOOL)resignFirstResponder { return YES; } -(void)drawRect:(NSRect)rect { [m_context clearDrawable]; [m_context setView: theView]; [m_context makeCurrentContext]; if (([theView frame].size.width != lastWidth) || ([theView frame].size.height != lastHeight)) { lastWidth = [theView frame].size.width; lastHeight = [theView frame].size.height; // Only needed on resize: // [m_context clearDrawable]; // glViewport(0, 0, [theView frame].size.width, [theView frame].size.height); if (gReshape != NULL) { // [m_context setView: theView]; // Make the view current in case gReshape calls OpenGL // [m_context makeCurrentContext]; // 131011 gReshape([theView frame].size.width, [theView frame].size.height); } } // Draw updatePending = 0; // Did not help if (gDisplay != NULL) gDisplay(); [m_context flushBuffer]; // [NSOpenGLContext clearCurrentContext]; } -(void)windowWillClose:(NSNotification *)note { [[NSApplication sharedApplication] terminate:self]; } -(void)windowDidresize:(NSNotification *)note { // Call gReshape now or in drawRect? } -(BOOL)isFlipped { return YES; } // This was supposed to help for NSView but it didn't. //-(void)setLayer:(CALayer*)layer; //{ // [super setLayer:layer]; // [m_context update]; //} @end // -------------------- Timer ------------------------ // Data for timer @interface TimerInfoRec : NSObject { @public void (*func)(int arg); int arg; @private } @end @implementation TimerInfoRec @end // Mini-mini class for the timer @interface TimerController : NSObject { } -(void)timerFireMethod:(NSTimer *)t; @end NSTimer *gTimer; TimerController *myTimerController; NSView *view; // Timer! @implementation TimerController -(void)timerFireMethod:(NSTimer *)t; { TimerInfoRec *tr; if (t.userInfo != nil) // One-shot timer with a TimerInfoRec { tr = t.userInfo; tr->func(tr->arg); [tr release]; // ((TimerInfoRec *)(t.userInfo))->func(((TimerInfoRec *)(t.userInfo))->arg); // free((TimerInfoRec *)(t.userInfo)); } else { [view setNeedsDisplay: YES]; updatePending = 1; } } @end void glutPostRedisplay() { [theView setNeedsDisplay: YES]; updatePending = 1; } //void home() //{ // p = Bundle.main.path(forResource: "???", ofType: "???"); //} // home() #include void home() { CFBundleRef mainBundle = CFBundleGetMainBundle(); CFURLRef resourcesURL = CFBundleCopyResourcesDirectoryURL(mainBundle); char path[PATH_MAX]; if (!CFURLGetFileSystemRepresentation(resourcesURL, TRUE, (UInt8 *)path, PATH_MAX)) { // error! return; } CFRelease(resourcesURL); chdir(path); // printf("Current Path: %s\n", path); } // ------------------ Main program --------------------- @interface MGApplication : NSApplication @end @implementation MGApplication /* Invoked from the Quit menu item */ - (void)terminate:(id)sender { gRunning = 0; } @end NSApplication *myApp; NSView *view; NSAutoreleasePool *pool; NSWindow *window; static struct timeval timeStart; void CreateMenu() { NSMenu *mainMenu, *theMiniMenu; NSMenuItem *menuItem2, *dummyItem; // Create main menu = menu bar mainMenu = NSMenu.alloc; [mainMenu initWithTitle: @""]; [NSApp setMainMenu: mainMenu]; // Create the custom menu theMiniMenu = NSMenu.alloc; [theMiniMenu initWithTitle: @"The MiniMenu"]; // Create a menu item with standard message menuItem2 = NSMenuItem.alloc; [menuItem2 initWithTitle: @"Hide" action: @selector(hide:) keyEquivalent: @"h"]; [menuItem2 setKeyEquivalentModifierMask: NSEventModifierFlagCommand]; [theMiniMenu addItem: menuItem2]; // Create a menu item with standard message menuItem2 = NSMenuItem.alloc; [menuItem2 initWithTitle: @"Hide others" action: @selector(hideOtherApplications:) keyEquivalent: @"h"]; [menuItem2 setKeyEquivalentModifierMask: NSEventModifierFlagCommand | NSEventModifierFlagOption]; [theMiniMenu addItem: menuItem2]; // Create a menu item with standard message menuItem2 = NSMenuItem.alloc; [menuItem2 initWithTitle: @"Show all" action: @selector(unhideAllApplications:) keyEquivalent: @"h"]; [menuItem2 setKeyEquivalentModifierMask: NSEventModifierFlagCommand | NSEventModifierFlagControl]; [theMiniMenu addItem: menuItem2]; // Create a menu item with standard message menuItem2 = NSMenuItem.alloc; [menuItem2 initWithTitle: @"Quit" action: @selector(terminate:) keyEquivalent: @"q"]; [menuItem2 setKeyEquivalentModifierMask: NSEventModifierFlagCommand]; [theMiniMenu addItem: menuItem2]; // Adding a menu is done with a dummy item to connect the menu to its parent dummyItem = NSMenuItem.alloc; [dummyItem initWithTitle: @"" action: nil keyEquivalent: @""]; [dummyItem setSubmenu: theMiniMenu]; [mainMenu addItem: dummyItem]; } void glutInit(int *argcp, char **argv) { pool = [NSAutoreleasePool new]; myApp = [MGApplication sharedApplication]; [NSApp setActivationPolicy: NSApplicationActivationPolicyRegular]; // Thanks to Marcus StenbŠck gRunning = 1; home(); gettimeofday(&timeStart, NULL); CreateMenu(); myTimerController = [TimerController alloc]; // Apple REALLY doesn't like us to mess with the mouse pointer. Now this is deprecated... // and some sources say it isn't needed. // CGSetLocalEventsSuppressionInterval(0.0); // For glutWarpPointer int i; for (i = 0; i < 256; i++) gKeymap[i] = 0; } int gWindowPosX = 10; int gWindowPosY = 50; int gWindowWidth = 400; int gWindowHeight = 400; void glutInitWindowPosition (int x, int y) { gWindowPosX = x; gWindowPosY = y; } void glutInitWindowSize (int width, int height) { gWindowWidth = width; gWindowHeight = height; } int glutCreateWindow (const char *windowTitle) { NSRect frame = NSMakeRect(gWindowPosX, NSScreen.mainScreen.frame.size.height - gWindowPosY-gWindowHeight, gWindowWidth, gWindowHeight); window = [NSWindow alloc]; [window initWithContentRect:frame // styleMask:NSTitledWindowMask | NSClosableWindowMask | NSMiniaturizableWindowMask | NSResizableWindowMask styleMask:NSWindowStyleMaskTitled | NSWindowStyleMaskClosable | NSWindowStyleMaskMiniaturizable | NSWindowStyleMaskResizable backing:NSBackingStoreBuffered defer:false]; [window setTitle: [[NSString alloc] initWithCString:windowTitle encoding:NSMacOSRomanStringEncoding]]; strcpy(gTitle, windowTitle); // Save title in case of full screen (which clears it) view = [GlutView alloc]; [view initWithFrame: frame]; [window setAcceptsMouseMovedEvents:YES]; // OpenGL init! MakeContext(view); // Moved from main loop [NSApp activateIgnoringOtherApps:YES]; // Added 180208 [window setDelegate: (GlutView*)view]; [window makeKeyAndOrderFront: nil]; [window makeFirstResponder: view]; // Added 130214 [window setContentView: view]; return 0; // Fake placeholder } // MAIN LOOP /* void oglutMainLoop() { [window setContentView: view]; [myApp finishLaunching]; NSEvent *event; while (gRunning) { [pool release]; pool = [NSAutoreleasePool new]; if (updatePending || gIdle != NULL) // If it is, then the setNeedsDisplay below has been called and we will get an update event - but must not block! // (If there was an update coming I think it should not block, but at least this works.) event = [myApp nextEventMatchingMask: NSEventMaskAny untilDate: [NSDate dateWithTimeIntervalSinceNow: 0.0] inMode: NSDefaultRunLoopMode dequeue: true ]; else event = [myApp nextEventMatchingMask: NSEventMaskAny untilDate: [NSDate distantFuture] inMode: NSDefaultRunLoopMode dequeue: true ]; [myApp sendEvent: event]; [myApp updateWindows]; if (gIdle != NULL) if (!updatePending) gIdle(); // Did not help // if (updatePending) // [theView setNeedsDisplay: YES]; } } */ char inMainLoop = 0; char finished = 0; void internalCheckLoop() { NSEvent *event; // [myApp runOnce]; // ??? pool = [NSAutoreleasePool new]; if (updatePending || gIdle != NULL || !inMainLoop) // If it is, then the setNeedsDisplay below has been called and we will get an update event - but must not block! // (If there was an update coming I think it should not block, but at least this works.) event = [myApp nextEventMatchingMask: NSEventMaskAny untilDate: [NSDate dateWithTimeIntervalSinceNow: 0.0] inMode: NSDefaultRunLoopMode dequeue: true ]; else event = [myApp nextEventMatchingMask: NSEventMaskAny untilDate: [NSDate distantFuture] inMode: NSDefaultRunLoopMode dequeue: true ]; [myApp sendEvent: event]; [myApp updateWindows]; if (gIdle != NULL) if (!updatePending) gIdle(); // Did not help? if (updatePending) [theView setNeedsDisplay: YES]; [pool release]; } void glutCheckLoop() { inMainLoop = 0; // if (!inMainLoop) [window setContentView: view]; // Why isn't this a job for window creation??? if (!finished) { [myApp finishLaunching]; finished = 1; } // [NSOpenGLContext clearCurrentContext]; // [m_context makeCurrentContext]; // [m_context setView: theView]; [m_context flushBuffer]; internalCheckLoop(); [m_context setView: theView]; // [m_context makeCurrentContext]; } void glutMainLoop() { inMainLoop = 1; // [window setContentView: view]; [myApp finishLaunching]; finished = 1; while (gRunning) { internalCheckLoop(); } inMainLoop = 0; } // END OF MAIN LOOP void glutTimerFunc(int millis, void (*func)(int arg), int arg) { TimerInfoRec *timerInfo = [TimerInfoRec alloc]; timerInfo->arg = arg; timerInfo->func = func; gTimer = [NSTimer scheduledTimerWithTimeInterval: millis/1000.0 target: myTimerController selector: @selector(timerFireMethod:) userInfo: timerInfo repeats: NO]; } // Added by Ingemar void glutRepeatingTimer(int millis) { gTimer = [NSTimer scheduledTimerWithTimeInterval: millis/1000.0 target: myTimerController selector: @selector(timerFireMethod:) userInfo: nil repeats: YES]; } // Bad name, will be removed void glutRepeatingTimerFunc(int millis) { glutRepeatingTimer(millis); } void glutDisplayFunc(void (*func)(void)) { gDisplay = func; } void glutReshapeFunc(void (*func)(int width, int height)) { gReshape = func; } void glutKeyboardFunc(void (*func)(unsigned char key, int x, int y)) { gKey = func; } void glutKeyboardUpFunc(void (*func)(unsigned char key, int x, int y)) { gKeyUp = func; } void glutSpecialFunc(void (*func)(unsigned char key, int x, int y)) { gSpecialKey = func; } void glutSpecialUpFunc(void (*func)(unsigned char key, int x, int y)) { gSpecialKeyUp = func; } void glutPassiveMotionFunc(void (*func)(int x, int y)) { gMouseMoved = func; } void glutMotionFunc(void (*func)(int x, int y)) { gMouseDragged = func; } void glutMouseFunc(void (*func)(int button, int state, int x, int y)) { gMouseFunc = func; } // You can safely skip this void glutSwapBuffers() { // [m_context flushBuffer]; glFlush(); glFinish(); } int glutGet(int type) { struct timeval tv; switch (type) { case GLUT_QUIT_FLAG: return !gRunning; break; case GLUT_WINDOW_WIDTH: return lastWidth; break; case GLUT_WINDOW_HEIGHT: return lastHeight; break; case GLUT_ELAPSED_TIME: gettimeofday(&tv, NULL); return (tv.tv_usec - timeStart.tv_usec) / 1000 + (tv.tv_sec - timeStart.tv_sec)*1000; break; case GLUT_MOUSE_POSITION_X: return gMousePosition.x; break; case GLUT_MOUSE_POSITION_Y: return gMousePosition.y; break; } return 0; } void glutInitDisplayMode(unsigned int mode) { gContextInitMode = mode; } void glutIdleFunc(void (*func)(void)) { // glutIdleFunc not recommended. gIdle = func; } void glutReshapeWindow(int width, int height) { NSRect r; if (gFullScreen) { [window setFrame: gSavedWindowPosition display: true]; gFullScreen = 0; } r = [window frame]; r.size.width = width; r.size.height = height; // Bug fix 150820 // [window setFrame: r display: true]; [window setContentSize: r.size]; } void glutPositionWindow(int x, int y) { NSRect r; if (gFullScreen) { [window setFrame: gSavedWindowPosition display: true]; gFullScreen = 0; } r = [window frame]; r.origin.x = x; r.origin.y = NSScreen.mainScreen.frame.size.height - gWindowPosY - r.size.height; [window setFrame: r display: true]; } void glutSetWindowTitle(char *title) { strcpy(gTitle, title); [window setTitle: [NSString stringWithUTF8String: title]]; } char glutKeyIsDown(unsigned char c) { return gKeymap[(unsigned int)c]; } char glutMouseIsDown(unsigned char c) { return gButtonPressed[(unsigned int)c]; } void glutInitContextVersion(int major, int minor) { gContextVersionMajor = major; gContextVersionMinor = minor; } // glutInitContextFlags(int flags); Add this? // Visibility: Just call back immediately and claim we are visible! void glutVisibilityFunc(void (*visibility)(int status)) { visibility(GLUT_VISIBLE); } void PostMouseEvent(CGMouseButton button, CGEventType type, const CGPoint point) { CGEventRef theEvent = CGEventCreateMouseEvent(NULL /*event source?*/, type, point, button); CGEventSetType(theEvent, type); CGEventPost(kCGHIDEventTap, theEvent); CFRelease(theEvent); } // Rewritten 140213. Seems to work. // 150423: But... why not now??? Rewrote again, and now it works... again! // Some debugging junk outcommented below, should tidy up after more testing. void glutWarpPointer(int x, int y) { NSPoint mp; CGPoint pt; NSRect r; mp.x = 0; mp.y = 0; // mp = [window convertBaseToScreen: mp]; // Deprecation hell below: //The NSWindow class provides these methods for converting between window local coordinates and screen global coordinates: // „ convertRectToScreen: // „ convertRectFromScreen: //Use them instead of the deprecated convertBaseToScreen: and convertScreenToBase: methods. NSRect mpmp; mpmp.size.height = 0; mpmp.size.width = 0; mpmp.origin = mp; mpmp = [window convertRectToScreen: mpmp]; mp = mpmp.origin; // Flip to downwards Y mp.y = NSScreen.mainScreen.frame.size.height - mp.y; // Get the view size r = [[window contentView] frame]; // Add the x, y offset, minus the view frame height to move to upper left. pt.x = x + mp.x; pt.y = y - r.size.height + mp.y; // PostMouseEvent( 1, kCGEventMouseMoved, pt); // CGSetLocalEventsSuppressionInterval(0.0); -> glutInit CGWarpMouseCursorPosition(pt); CGAssociateMouseAndMouseCursorPosition(true); } // Should be used for auto-show-hide on activate/deactivate char hidden = 0; void glutShowCursor() { if (hidden) { [NSCursor unhide]; hidden = 0; } } void glutHideCursor() { if (!hidden) { [NSCursor hide]; hidden = 1; } } void glutFullScreen() { if (gFullScreen == 0) { gFullScreen = 1; gSavedWindowPosition = [window frame]; [NSMenu setMenuBarVisible: FALSE]; [window setStyleMask: NSWindowStyleMaskBorderless]; [window setFrame: NSScreen.mainScreen.frame display: true]; [window makeKeyAndOrderFront: nil]; [window makeFirstResponder: view]; } } // Any reason why we should wait with resizing? // This page says we should: https://www.opengl.org/documentation/specs/glut/spec3/node24.html // This call is not in the original GLUT but IMHO better than relying on glutReshapeWindow or glutPositionWindow to exit! // (These calls also exit full screen mode though.) void glutExitFullScreen() { if (gFullScreen != 0) { gFullScreen = 0; [NSMenu setMenuBarVisible: TRUE]; // [window setStyleMask: NSTitledWindowMask | NSClosableWindowMask | NSMiniaturizableWindowMask | NSResizableWindowMask]; // Back to normal [window setStyleMask: NSWindowStyleMaskTitled | NSWindowStyleMaskClosable | NSWindowStyleMaskMiniaturizable | NSWindowStyleMaskResizable]; // Back to normal [window setFrame: gSavedWindowPosition display: true]; [window makeKeyAndOrderFront: nil]; [window makeFirstResponder: view]; [window setTitle: [NSString stringWithUTF8String: gTitle]]; } } // When I am at it... this is what I really want! void glutToggleFullScreen() { if (gFullScreen) glutExitFullScreen(); else glutFullScreen(); } void glutExit() { gRunning = 0; } // Placeholders void glutSetWindow(int win) {} int glutGetWindow(void) { return 0; }