| @@ -61,10 +61,6 @@ public: | |||||
| setVisible(false); | setVisible(false); | ||||
| } | } | ||||
| #if DISTRHO_OS_WINDOWS | |||||
| Rectangle<int> getBounds(); | |||||
| #endif | |||||
| private: | private: | ||||
| class Private; | class Private; | ||||
| Private* const kPrivate; | Private* const kPrivate; | ||||
| @@ -27,13 +27,12 @@ ImageAboutWindow::ImageAboutWindow(App* app, Window* parent, const Image& image) | |||||
| { | { | ||||
| #if DISTRHO_OS_WINDOWS | #if DISTRHO_OS_WINDOWS | ||||
| // FIXME | // FIXME | ||||
| Window::setSize(image.getWidth()+1, image.getHeight()+1); | |||||
| Window::setSize(image.getWidth(), image.getHeight()+1); | |||||
| #else | #else | ||||
| Window::setSize(image.getWidth(), image.getHeight()); | Window::setSize(image.getWidth(), image.getHeight()); | ||||
| #endif | #endif | ||||
| Window::setWindowTitle("About"); | Window::setWindowTitle("About"); | ||||
| Window::show(); | |||||
| } | } | ||||
| void ImageAboutWindow::onDisplay() | void ImageAboutWindow::onDisplay() | ||||
| @@ -214,19 +214,10 @@ void Widget::onReshape(int width, int height) | |||||
| { | { | ||||
| glEnable(GL_BLEND); | glEnable(GL_BLEND); | ||||
| glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); | glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); | ||||
| glMatrixMode(GL_PROJECTION); | glMatrixMode(GL_PROJECTION); | ||||
| glLoadIdentity(); | glLoadIdentity(); | ||||
| #if DISTRHO_OS_WINDOWS | |||||
| Rectangle<int> r(fParent->getBounds()); | |||||
| glOrtho(r.getX(), r.getWidth(), r.getHeight(), r.getY(), -1.0f, 1.0f); | |||||
| glViewport(0, 0, r.getWidth() - r.getX(), r.getHeight() - r.getY()); | |||||
| #else | |||||
| glOrtho(0, width, height, 0, 0.0f, 1.0f); | glOrtho(0, width, height, 0, 0.0f, 1.0f); | ||||
| glViewport(0, 0, width, height); | glViewport(0, 0, width, height); | ||||
| #endif | |||||
| glMatrixMode(GL_MODELVIEW); | glMatrixMode(GL_MODELVIEW); | ||||
| glLoadIdentity(); | glLoadIdentity(); | ||||
| } | } | ||||
| @@ -53,6 +53,7 @@ public: | |||||
| fChildFocus(nullptr), | fChildFocus(nullptr), | ||||
| fVisible(false), | fVisible(false), | ||||
| fClosed(false), | fClosed(false), | ||||
| fResizable(false), | |||||
| #if DISTRHO_OS_WINDOWS | #if DISTRHO_OS_WINDOWS | ||||
| hwnd(0) | hwnd(0) | ||||
| #elif DISTRHO_OS_LINUX | #elif DISTRHO_OS_LINUX | ||||
| @@ -85,7 +86,6 @@ public: | |||||
| #if DISTRHO_OS_WINDOWS | #if DISTRHO_OS_WINDOWS | ||||
| hwnd = impl->hwnd; | hwnd = impl->hwnd; | ||||
| #elif DISTRHO_OS_LINUX | #elif DISTRHO_OS_LINUX | ||||
| xDisplay = impl->display; | xDisplay = impl->display; | ||||
| xWindow = impl->win; | xWindow = impl->win; | ||||
| @@ -116,17 +116,32 @@ public: | |||||
| void exec() | void exec() | ||||
| { | { | ||||
| fClosed = false; | fClosed = false; | ||||
| show(); | |||||
| if (fParent != nullptr) | if (fParent != nullptr) | ||||
| { | { | ||||
| fParent->fChildFocus = this; | |||||
| #if DISTRHO_OS_WINDOWS | #if DISTRHO_OS_WINDOWS | ||||
| EnableWindow(fParent->hwnd, FALSE); | |||||
| // Center this window | |||||
| PuglInternals* parentImpl = fParent->kView->impl; | |||||
| RECT curRect; | |||||
| RECT parentRect; | |||||
| GetWindowRect(hwnd, &curRect); | |||||
| GetWindowRect(parentImpl->hwnd, &parentRect); | |||||
| int x = parentRect.left+(parentRect.right-curRect.right)/2; | |||||
| int y = parentRect.top+(parentRect.bottom-curRect.bottom)/2; | |||||
| SetWindowPos(hwnd, 0, x, y, 0, 0, SWP_NOACTIVATE|SWP_NOOWNERZORDER|SWP_NOSIZE|SWP_NOZORDER); | |||||
| UpdateWindow(hwnd); | |||||
| #endif | #endif | ||||
| fParent->fChildFocus = this; | |||||
| fParent->show(); | fParent->show(); | ||||
| } | } | ||||
| show(); | |||||
| while (isVisible() && ! fClosed) | while (isVisible() && ! fClosed) | ||||
| { | { | ||||
| idle(); | idle(); | ||||
| @@ -142,9 +157,6 @@ public: | |||||
| if (fParent != nullptr) | if (fParent != nullptr) | ||||
| { | { | ||||
| fParent->fChildFocus = nullptr; | fParent->fChildFocus = nullptr; | ||||
| #if DISTRHO_OS_WINDOWS | |||||
| EnableWindow(fParent->hwnd, TRUE); | |||||
| #endif | |||||
| } | } | ||||
| } | } | ||||
| @@ -224,14 +236,15 @@ public: | |||||
| void setSize(unsigned int width, unsigned int height) | void setSize(unsigned int width, unsigned int height) | ||||
| { | { | ||||
| #if DISTRHO_OS_WINDOWS | #if DISTRHO_OS_WINDOWS | ||||
| RECT rcClient; | |||||
| RECT rcWindow; | |||||
| POINT ptDiff; | |||||
| GetClientRect(hwnd, &rcClient); | |||||
| GetWindowRect(hwnd, &rcWindow); | |||||
| ptDiff.x = (rcWindow.right - rcWindow.left) - rcClient.right; | |||||
| ptDiff.y = (rcWindow.bottom - rcWindow.top) - rcClient.bottom; | |||||
| SetWindowPos(hwnd, 0, 0, 0, width + ptDiff.x, height + ptDiff.y, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOZORDER); | |||||
| int winFlags = WS_POPUPWINDOW | WS_CAPTION; | |||||
| if (fResizable) | |||||
| winFlags |= WS_SIZEBOX; | |||||
| RECT wr = { 0, 0, (long)width, (long)height }; | |||||
| AdjustWindowRectEx(&wr, winFlags, FALSE, WS_EX_TOPMOST); | |||||
| SetWindowPos(hwnd, 0, 0, 0, wr.right-wr.left, wr.bottom-wr.top, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOZORDER); | |||||
| UpdateWindow(hwnd); | UpdateWindow(hwnd); | ||||
| #elif DISTRHO_OS_LINUX | #elif DISTRHO_OS_LINUX | ||||
| XSizeHints sizeHints; | XSizeHints sizeHints; | ||||
| @@ -284,16 +297,6 @@ public: | |||||
| fWidgets.remove(widget); | fWidgets.remove(widget); | ||||
| } | } | ||||
| #if DISTRHO_OS_WINDOWS | |||||
| Rectangle<int> getBounds() | |||||
| { | |||||
| RECT rcClient; | |||||
| GetClientRect(hwnd, &rcClient); | |||||
| return Rectangle<int>(rcClient.left, rcClient.top, rcClient.right, rcClient.bottom); | |||||
| } | |||||
| #endif | |||||
| protected: | protected: | ||||
| void onDisplay() | void onDisplay() | ||||
| { | { | ||||
| @@ -422,6 +425,7 @@ private: | |||||
| Private* fChildFocus; | Private* fChildFocus; | ||||
| bool fVisible; | bool fVisible; | ||||
| bool fClosed; | bool fClosed; | ||||
| bool fResizable; | |||||
| std::list<Widget*> fWidgets; | std::list<Widget*> fWidgets; | ||||
| @@ -563,13 +567,6 @@ void Window::removeWidget(Widget* widget) | |||||
| kPrivate->removeWidget(widget); | kPrivate->removeWidget(widget); | ||||
| } | } | ||||
| #if DISTRHO_OS_WINDOWS | |||||
| Rectangle<int> Window::getBounds() | |||||
| { | |||||
| return kPrivate->getBounds(); | |||||
| } | |||||
| #endif | |||||
| // ------------------------------------------------- | // ------------------------------------------------- | ||||
| END_NAMESPACE_DISTRHO | END_NAMESPACE_DISTRHO | ||||
| @@ -54,7 +54,9 @@ | |||||
| # define PUGL_API | # define PUGL_API | ||||
| #endif | #endif | ||||
| #ifndef __cplusplus | |||||
| #ifdef __cplusplus | |||||
| extern "C" { | |||||
| #else | |||||
| # include <stdbool.h> | # include <stdbool.h> | ||||
| #endif | #endif | ||||
| @@ -224,8 +226,7 @@ puglCreate(PuglNativeWindow parent, | |||||
| int width, | int width, | ||||
| int height, | int height, | ||||
| bool resizable, | bool resizable, | ||||
| bool addToDesktop = true, | |||||
| const char* x11Display = nullptr); | |||||
| bool addToDesktop = true); | |||||
| /** | /** | ||||
| Set the handle to be passed to all callbacks. | Set the handle to be passed to all callbacks. | ||||
| @@ -245,6 +246,12 @@ puglSetHandle(PuglView* view, PuglHandle handle); | |||||
| PUGL_API PuglHandle | PUGL_API PuglHandle | ||||
| puglGetHandle(PuglView* view); | puglGetHandle(PuglView* view); | ||||
| /** | |||||
| Return the timestamp (if any) of the currently-processing event. | |||||
| */ | |||||
| PUGL_API uint32_t | |||||
| puglGetEventTimestamp(PuglView* view); | |||||
| /** | /** | ||||
| Get the currently active modifiers (PuglMod flags). | Get the currently active modifiers (PuglMod flags). | ||||
| @@ -338,4 +345,8 @@ puglDestroy(PuglView* view); | |||||
| @} | @} | ||||
| */ | */ | ||||
| #ifdef __cplusplus | |||||
| } /* extern "C" */ | |||||
| #endif | |||||
| #endif /* PUGL_H_INCLUDED */ | #endif /* PUGL_H_INCLUDED */ | ||||
| @@ -39,11 +39,13 @@ struct PuglViewImpl { | |||||
| PuglInternals* impl; | PuglInternals* impl; | ||||
| int width; | |||||
| int height; | |||||
| int mods; | |||||
| bool ignoreKeyRepeat; | |||||
| bool redisplay; | |||||
| int width; | |||||
| int height; | |||||
| int mods; | |||||
| bool mouse_in_view; | |||||
| bool ignoreKeyRepeat; | |||||
| bool redisplay; | |||||
| uint32_t event_timestamp_ms; | |||||
| }; | }; | ||||
| void | void | ||||
| @@ -58,13 +60,19 @@ puglGetHandle(PuglView* view) | |||||
| return view->handle; | return view->handle; | ||||
| } | } | ||||
| uint32_t | |||||
| puglGetEventTimestamp(PuglView* view) | |||||
| { | |||||
| return view->event_timestamp_ms; | |||||
| } | |||||
| int | int | ||||
| puglGetModifiers(PuglView* view) | puglGetModifiers(PuglView* view) | ||||
| { | { | ||||
| return view->mods; | return view->mods; | ||||
| } | } | ||||
| static inline void | |||||
| void | |||||
| puglDefaultReshape(PuglView* view, int width, int height) | puglDefaultReshape(PuglView* view, int width, int height) | ||||
| { | { | ||||
| glMatrixMode(GL_PROJECTION); | glMatrixMode(GL_PROJECTION); | ||||
| @@ -24,12 +24,64 @@ | |||||
| #include "pugl_internal.h" | #include "pugl_internal.h" | ||||
| @interface PuglWindow : NSWindow | |||||
| { | |||||
| @public | |||||
| PuglView* puglview; | |||||
| } | |||||
| - (id) initWithContentRect:(NSRect)contentRect | |||||
| styleMask:(unsigned int)aStyle | |||||
| backing:(NSBackingStoreType)bufferingType | |||||
| defer:(BOOL)flag; | |||||
| - (void) setPuglview:(PuglView*)view; | |||||
| - (BOOL) windowShouldClose:(id)sender; | |||||
| - (void) becomeKeyWindow:(id)sender; | |||||
| - (BOOL) canBecomeKeyWindow:(id)sender; | |||||
| @end | |||||
| @implementation PuglWindow | |||||
| - (id)initWithContentRect:(NSRect)contentRect | |||||
| styleMask:(unsigned int)aStyle | |||||
| backing:(NSBackingStoreType)bufferingType | |||||
| defer:(BOOL)flag | |||||
| { | |||||
| NSWindow* result = [super initWithContentRect:contentRect | |||||
| styleMask:(NSClosableWindowMask | | |||||
| NSTitledWindowMask | | |||||
| NSResizableWindowMask) | |||||
| backing:NSBackingStoreBuffered defer:NO]; | |||||
| [result setAcceptsMouseMovedEvents:YES]; | |||||
| [result setLevel: CGShieldingWindowLevel() + 1]; | |||||
| return result; | |||||
| } | |||||
| - (void)setPuglview:(PuglView*)view | |||||
| { | |||||
| puglview = view; | |||||
| [self setContentSize:NSMakeSize(view->width, view->height) ]; | |||||
| } | |||||
| - (BOOL)windowShouldClose:(id)sender | |||||
| { | |||||
| if (puglview->closeFunc) | |||||
| puglview->closeFunc(puglview); | |||||
| return YES; | |||||
| } | |||||
| @end | |||||
| @interface PuglOpenGLView : NSOpenGLView | @interface PuglOpenGLView : NSOpenGLView | ||||
| { | { | ||||
| int colorBits; | int colorBits; | ||||
| int depthBits; | int depthBits; | ||||
| @public | @public | ||||
| PuglView* view; | |||||
| PuglView* puglview; | |||||
| NSTrackingArea* trackingArea; | |||||
| } | } | ||||
| - (id) initWithFrame:(NSRect)frame | - (id) initWithFrame:(NSRect)frame | ||||
| @@ -38,6 +90,7 @@ | |||||
| - (void) reshape; | - (void) reshape; | ||||
| - (void) drawRect:(NSRect)rect; | - (void) drawRect:(NSRect)rect; | ||||
| - (void) mouseMoved:(NSEvent*)event; | - (void) mouseMoved:(NSEvent*)event; | ||||
| - (void) mouseDragged:(NSEvent*)event; | |||||
| - (void) mouseDown:(NSEvent*)event; | - (void) mouseDown:(NSEvent*)event; | ||||
| - (void) mouseUp:(NSEvent*)event; | - (void) mouseUp:(NSEvent*)event; | ||||
| - (void) rightMouseDown:(NSEvent*)event; | - (void) rightMouseDown:(NSEvent*)event; | ||||
| @@ -92,33 +145,37 @@ | |||||
| int width = bounds.size.width; | int width = bounds.size.width; | ||||
| int height = bounds.size.height; | int height = bounds.size.height; | ||||
| if (view->reshapeFunc) { | |||||
| view->reshapeFunc(view, width, height); | |||||
| } else { | |||||
| puglDefaultReshape(view, width, height); | |||||
| } | |||||
| if (puglview) { | |||||
| /* NOTE: Apparently reshape gets called when the GC gets around to | |||||
| deleting the view (?), so we must have reset puglview to NULL when | |||||
| this comes around. | |||||
| */ | |||||
| if (puglview->reshapeFunc) { | |||||
| puglview->reshapeFunc(puglview, width, height); | |||||
| } else { | |||||
| puglDefaultReshape(puglview, width, height); | |||||
| } | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| puglview->width = width; | |||||
| puglview->height = height; | |||||
| } | |||||
| } | } | ||||
| - (void) drawRect:(NSRect)rect | - (void) drawRect:(NSRect)rect | ||||
| { | { | ||||
| glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); | |||||
| glLoadIdentity(); | |||||
| if (self->view->displayFunc) { | |||||
| self->view->displayFunc(self->view); | |||||
| } | |||||
| puglDisplay(puglview); | |||||
| glFlush(); | glFlush(); | ||||
| glSwapAPPLE(); | glSwapAPPLE(); | ||||
| } | } | ||||
| static int | |||||
| getModifiers(unsigned modifierFlags) | |||||
| static unsigned | |||||
| getModifiers(PuglView* view, NSevent* ev) | |||||
| { | { | ||||
| int mods = 0; | |||||
| const unsigned modifierFlags = [ev modifierFlags] | |||||
| view->event_timestamp_ms = fmod([ev timestamp] * 1000.0, UINT32_MAX); | |||||
| unsigned mods = 0; | |||||
| mods |= (modifierFlags & NSShiftKeyMask) ? PUGL_MOD_SHIFT : 0; | mods |= (modifierFlags & NSShiftKeyMask) ? PUGL_MOD_SHIFT : 0; | ||||
| mods |= (modifierFlags & NSControlKeyMask) ? PUGL_MOD_CTRL : 0; | mods |= (modifierFlags & NSControlKeyMask) ? PUGL_MOD_CTRL : 0; | ||||
| mods |= (modifierFlags & NSAlternateKeyMask) ? PUGL_MOD_ALT : 0; | mods |= (modifierFlags & NSAlternateKeyMask) ? PUGL_MOD_ALT : 0; | ||||
| @@ -126,99 +183,135 @@ getModifiers(unsigned modifierFlags) | |||||
| return mods; | return mods; | ||||
| } | } | ||||
| -(void)updateTrackingAreas | |||||
| { | |||||
| if (trackingArea != nil) { | |||||
| [self removeTrackingArea:trackingArea]; | |||||
| [trackingArea release]; | |||||
| } | |||||
| const int opts = (NSTrackingMouseEnteredAndExited | | |||||
| NSTrackingMouseMoved | | |||||
| NSTrackingActiveAlways); | |||||
| trackingArea = [ [NSTrackingArea alloc] initWithRect:[self bounds] | |||||
| options:opts | |||||
| owner:self | |||||
| userInfo:nil]; | |||||
| [self addTrackingArea:trackingArea]; | |||||
| } | |||||
| - (void)mouseEntered:(NSEvent*)theEvent | |||||
| { | |||||
| [self updateTrackingAreas]; | |||||
| } | |||||
| - (void)mouseExited:(NSEvent*)theEvent | |||||
| { | |||||
| } | |||||
| - (void) mouseMoved:(NSEvent*)event | - (void) mouseMoved:(NSEvent*)event | ||||
| { | { | ||||
| if (view->motionFunc) { | |||||
| if (puglview->motionFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | NSPoint loc = [event locationInWindow]; | ||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->motionFunc(view, loc.x, loc.y); | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->motionFunc(puglview, loc.x, puglview->height - loc.y); | |||||
| } | |||||
| } | |||||
| - (void) mouseDragged:(NSEvent*)event | |||||
| { | |||||
| if (puglview->motionFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->motionFunc(puglview, loc.x, puglview->height - loc.y); | |||||
| } | } | ||||
| } | } | ||||
| - (void) mouseDown:(NSEvent*)event | - (void) mouseDown:(NSEvent*)event | ||||
| { | { | ||||
| if (view->mouseFunc) { | |||||
| if (puglview->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | NSPoint loc = [event locationInWindow]; | ||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 1, true, loc.x, loc.y); | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->mouseFunc(puglview, 1, true, loc.x, puglview->height - loc.y); | |||||
| } | } | ||||
| } | } | ||||
| - (void) mouseUp:(NSEvent*)event | - (void) mouseUp:(NSEvent*)event | ||||
| { | { | ||||
| if (view->mouseFunc) { | |||||
| if (puglview->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | NSPoint loc = [event locationInWindow]; | ||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 1, false, loc.x, loc.y); | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->mouseFunc(puglview, 1, false, loc.x, puglview->height - loc.y); | |||||
| } | } | ||||
| [self updateTrackingAreas]; | |||||
| } | } | ||||
| - (void) rightMouseDown:(NSEvent*)event | - (void) rightMouseDown:(NSEvent*)event | ||||
| { | { | ||||
| if (view->mouseFunc) { | |||||
| if (puglview->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | NSPoint loc = [event locationInWindow]; | ||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 3, true, loc.x, loc.y); | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->mouseFunc(puglview, 3, true, loc.x, puglview->height - loc.y); | |||||
| } | } | ||||
| } | } | ||||
| - (void) rightMouseUp:(NSEvent*)event | - (void) rightMouseUp:(NSEvent*)event | ||||
| { | { | ||||
| if (view->mouseFunc) { | |||||
| if (puglview->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | NSPoint loc = [event locationInWindow]; | ||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 3, false, loc.x, loc.y); | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->mouseFunc(puglview, 3, false, loc.x, puglview->height - loc.y); | |||||
| } | } | ||||
| } | } | ||||
| - (void) scrollWheel:(NSEvent*)event | - (void) scrollWheel:(NSEvent*)event | ||||
| { | { | ||||
| if (view->scrollFunc) { | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->scrollFunc(view, [event deltaX], [event deltaY]); | |||||
| if (puglview->scrollFunc) { | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->scrollFunc(puglview, [event deltaX], [event deltaY]); | |||||
| } | } | ||||
| [self updateTrackingAreas]; | |||||
| } | } | ||||
| - (void) keyDown:(NSEvent*)event | - (void) keyDown:(NSEvent*)event | ||||
| { | { | ||||
| if (view->keyboardFunc && !(view->ignoreKeyRepeat && [event isARepeat])) { | |||||
| if (puglview->keyboardFunc && !(puglview->ignoreKeyRepeat && [event isARepeat])) { | |||||
| NSString* chars = [event characters]; | NSString* chars = [event characters]; | ||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->keyboardFunc(view, true, [chars characterAtIndex:0]); | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->keyboardFunc(puglview, true, [chars characterAtIndex:0]); | |||||
| } | } | ||||
| } | } | ||||
| - (void) keyUp:(NSEvent*)event | - (void) keyUp:(NSEvent*)event | ||||
| { | { | ||||
| if (view->keyboardFunc) { | |||||
| if (puglview->keyboardFunc) { | |||||
| NSString* chars = [event characters]; | NSString* chars = [event characters]; | ||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->keyboardFunc(view, false, [chars characterAtIndex:0]); | |||||
| puglview->mods = getModifiers(puglview, event); | |||||
| puglview->keyboardFunc(puglview, false, [chars characterAtIndex:0]); | |||||
| } | } | ||||
| } | } | ||||
| - (void) flagsChanged:(NSEvent*)event | - (void) flagsChanged:(NSEvent*)event | ||||
| { | { | ||||
| if (view->specialFunc) { | |||||
| int mods = getModifiers([event modifierFlags]); | |||||
| if ((mods & PUGL_MOD_SHIFT) != (view->mods & PUGL_MOD_SHIFT)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_SHIFT, PUGL_KEY_SHIFT); | |||||
| } else if ((mods & PUGL_MOD_CTRL) != (view->mods & PUGL_MOD_CTRL)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_CTRL, PUGL_KEY_CTRL); | |||||
| } else if ((mods & PUGL_MOD_ALT) != (view->mods & PUGL_MOD_ALT)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_ALT, PUGL_KEY_ALT); | |||||
| } else if ((mods & PUGL_MOD_SUPER) != (view->mods & PUGL_MOD_SUPER)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_SUPER, PUGL_KEY_SUPER); | |||||
| if (puglview->specialFunc) { | |||||
| const unsigned mods = getModifiers(puglview, [event modifierFlags]); | |||||
| if ((mods & PUGL_MOD_SHIFT) != (puglview->mods & PUGL_MOD_SHIFT)) { | |||||
| puglview->specialFunc(puglview, mods & PUGL_MOD_SHIFT, PUGL_KEY_SHIFT); | |||||
| } else if ((mods & PUGL_MOD_CTRL) != (puglview->mods & PUGL_MOD_CTRL)) { | |||||
| puglview->specialFunc(puglview, mods & PUGL_MOD_CTRL, PUGL_KEY_CTRL); | |||||
| } else if ((mods & PUGL_MOD_ALT) != (puglview->mods & PUGL_MOD_ALT)) { | |||||
| puglview->specialFunc(puglview, mods & PUGL_MOD_ALT, PUGL_KEY_ALT); | |||||
| } else if ((mods & PUGL_MOD_SUPER) != (puglview->mods & PUGL_MOD_SUPER)) { | |||||
| puglview->specialFunc(puglview, mods & PUGL_MOD_SUPER, PUGL_KEY_SUPER); | |||||
| } | } | ||||
| view->mods = mods; | |||||
| puglview->mods = mods; | |||||
| } | } | ||||
| } | } | ||||
| @end | @end | ||||
| struct PuglInternalsImpl { | struct PuglInternalsImpl { | ||||
| PuglOpenGLView* view; | |||||
| NSModalSession session; | |||||
| PuglOpenGLView* glview; | |||||
| id window; | id window; | ||||
| }; | }; | ||||
| @@ -228,8 +321,7 @@ puglCreate(PuglNativeWindow parent, | |||||
| int width, | int width, | ||||
| int height, | int height, | ||||
| bool resizable, | bool resizable, | ||||
| bool addToDesktop, | |||||
| const char* x11Display) | |||||
| bool addToDesktop) | |||||
| { | { | ||||
| PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | ||||
| PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | ||||
| @@ -243,46 +335,37 @@ puglCreate(PuglNativeWindow parent, | |||||
| [NSAutoreleasePool new]; | [NSAutoreleasePool new]; | ||||
| [NSApplication sharedApplication]; | [NSApplication sharedApplication]; | ||||
| [NSApp setActivationPolicy:NSApplicationActivationPolicyRegular]; | |||||
| NSString* titleString = [[NSString alloc] | NSString* titleString = [[NSString alloc] | ||||
| initWithBytes:title | initWithBytes:title | ||||
| length:strlen(title) | length:strlen(title) | ||||
| encoding:NSUTF8StringEncoding]; | encoding:NSUTF8StringEncoding]; | ||||
| id window = [[[NSWindow alloc] | |||||
| initWithContentRect:NSMakeRect(0, 0, 512, 512) | |||||
| styleMask:NSTitledWindowMask | |||||
| backing:NSBackingStoreBuffered | |||||
| defer:NO] | |||||
| autorelease]; | |||||
| id window = [[PuglWindow new]retain]; | |||||
| [window cascadeTopLeftFromPoint:NSMakePoint(20, 20)]; | |||||
| [window setPuglview:view]; | |||||
| [window setTitle:titleString]; | [window setTitle:titleString]; | ||||
| [window setAcceptsMouseMovedEvents:YES]; | |||||
| impl->view = [PuglOpenGLView new]; | |||||
| impl->glview = [PuglOpenGLView new]; | |||||
| impl->window = window; | impl->window = window; | ||||
| impl->view->view = view; | |||||
| impl->glview->puglview = view; | |||||
| [window setContentView:impl->view]; | |||||
| [window setContentView:impl->glview]; | |||||
| [NSApp activateIgnoringOtherApps:YES]; | [NSApp activateIgnoringOtherApps:YES]; | ||||
| [window makeFirstResponder:impl->view]; | |||||
| [window makeFirstResponder:impl->glview]; | |||||
| impl->session = [NSApp beginModalSessionForWindow:view->impl->window]; | |||||
| [window makeKeyAndOrderFront:window]; | |||||
| return view; | return view; | ||||
| // unused | |||||
| (void)addToDesktop; | |||||
| (void)x11Display; | |||||
| } | } | ||||
| void | void | ||||
| puglDestroy(PuglView* view) | puglDestroy(PuglView* view) | ||||
| { | { | ||||
| [NSApp endModalSession:view->impl->session]; | |||||
| [view->impl->view release]; | |||||
| view->impl->glview->puglview = NULL; | |||||
| [view->impl->window close]; | |||||
| [view->impl->glview release]; | |||||
| [view->impl->window release]; | |||||
| free(view->impl); | free(view->impl); | ||||
| free(view); | free(view); | ||||
| } | } | ||||
| @@ -290,30 +373,15 @@ puglDestroy(PuglView* view) | |||||
| void | void | ||||
| puglDisplay(PuglView* view) | puglDisplay(PuglView* view) | ||||
| { | { | ||||
| glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); | |||||
| glLoadIdentity(); | |||||
| if (view->displayFunc) { | if (view->displayFunc) { | ||||
| view->displayFunc(view); | view->displayFunc(view); | ||||
| } | } | ||||
| glFlush(); | |||||
| view->redisplay = false; | |||||
| } | } | ||||
| PuglStatus | PuglStatus | ||||
| puglProcessEvents(PuglView* view) | puglProcessEvents(PuglView* view) | ||||
| { | { | ||||
| NSInteger response = [NSApp runModalSession:view->impl->session]; | |||||
| if (response != NSRunContinuesResponse) { | |||||
| if (view->closeFunc) { | |||||
| view->closeFunc(view); | |||||
| } | |||||
| } | |||||
| if (view->redisplay) { | |||||
| puglDisplay(view); | |||||
| } | |||||
| [view->impl->glview setNeedsDisplay: YES]; | |||||
| return PUGL_SUCCESS; | return PUGL_SUCCESS; | ||||
| } | } | ||||
| @@ -327,5 +395,5 @@ puglPostRedisplay(PuglView* view) | |||||
| PuglNativeWindow | PuglNativeWindow | ||||
| puglGetNativeWindow(PuglView* view) | puglGetNativeWindow(PuglView* view) | ||||
| { | { | ||||
| return (PuglNativeWindow)view->impl->view; | |||||
| return (PuglNativeWindow)view->impl->glview; | |||||
| } | } | ||||
| @@ -22,6 +22,9 @@ | |||||
| #include <windowsx.h> | #include <windowsx.h> | ||||
| #include <GL/gl.h> | #include <GL/gl.h> | ||||
| #include <stdio.h> | |||||
| #include <stdlib.h> | |||||
| #include "pugl_internal.h" | #include "pugl_internal.h" | ||||
| #ifndef WM_MOUSEWHEEL | #ifndef WM_MOUSEWHEEL | ||||
| @@ -30,11 +33,17 @@ | |||||
| #ifndef WM_MOUSEHWHEEL | #ifndef WM_MOUSEHWHEEL | ||||
| # define WM_MOUSEHWHEEL 0x020E | # define WM_MOUSEHWHEEL 0x020E | ||||
| #endif | #endif | ||||
| #ifndef WHEEL_DELTA | |||||
| # define WHEEL_DELTA 120 | |||||
| #endif | |||||
| const int LOCAL_CLOSE_MSG = WM_USER + 50; | |||||
| struct PuglInternalsImpl { | struct PuglInternalsImpl { | ||||
| HWND hwnd; | |||||
| HDC hdc; | |||||
| HGLRC hglrc; | |||||
| HWND hwnd; | |||||
| HDC hdc; | |||||
| HGLRC hglrc; | |||||
| WNDCLASS wc; | |||||
| }; | }; | ||||
| LRESULT CALLBACK | LRESULT CALLBACK | ||||
| @@ -46,8 +55,7 @@ puglCreate(PuglNativeWindow parent, | |||||
| int width, | int width, | ||||
| int height, | int height, | ||||
| bool resizable, | bool resizable, | ||||
| bool addToDesktop, | |||||
| const char* x11Display) | |||||
| bool addToDesktop) | |||||
| { | { | ||||
| PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | ||||
| PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | ||||
| @@ -59,30 +67,47 @@ puglCreate(PuglNativeWindow parent, | |||||
| view->width = width; | view->width = width; | ||||
| view->height = height; | view->height = height; | ||||
| WNDCLASS wc; | |||||
| wc.style = CS_OWNDC; | |||||
| wc.lpfnWndProc = wndProc; | |||||
| wc.cbClsExtra = 0; | |||||
| wc.cbWndExtra = 0; | |||||
| wc.hInstance = 0; | |||||
| wc.hIcon = LoadIcon(NULL, IDI_APPLICATION); | |||||
| wc.hCursor = LoadCursor(NULL, IDC_ARROW); | |||||
| wc.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH); | |||||
| wc.lpszMenuName = NULL; | |||||
| wc.lpszClassName = "Pugl"; | |||||
| RegisterClass(&wc); | |||||
| impl->hwnd = CreateWindow( | |||||
| "Pugl", title, | |||||
| (addToDesktop ? WS_VISIBLE : 0) | (parent ? WS_CHILD : (WS_OVERLAPPEDWINDOW & ~WS_THICKFRAME & ~WS_MAXIMIZEBOX)), | |||||
| CW_USEDEFAULT, CW_USEDEFAULT, width, height, | |||||
| // FIXME: This is nasty, and pugl should not have static anything. | |||||
| // Should class be a parameter? Does this make sense on other platforms? | |||||
| static int wc_count = 0; | |||||
| char classNameBuf[256]; | |||||
| _snprintf(classNameBuf, sizeof(classNameBuf), "%s_%d\n", title, wc_count++); | |||||
| impl->wc.style = CS_OWNDC; | |||||
| impl->wc.lpfnWndProc = wndProc; | |||||
| impl->wc.cbClsExtra = 0; | |||||
| impl->wc.cbWndExtra = 0; | |||||
| impl->wc.hInstance = 0; | |||||
| impl->wc.hIcon = LoadIcon(NULL, IDI_APPLICATION); | |||||
| impl->wc.hCursor = LoadCursor(NULL, IDC_ARROW); | |||||
| impl->wc.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH); | |||||
| impl->wc.lpszMenuName = NULL; | |||||
| impl->wc.lpszClassName = classNameBuf; | |||||
| RegisterClass(&impl->wc); | |||||
| int winFlags = WS_POPUPWINDOW | WS_CAPTION; | |||||
| if (resizable) { | |||||
| winFlags |= WS_SIZEBOX; | |||||
| } | |||||
| // Adjust the overall window size to accomodate our requested client size | |||||
| RECT wr = { 0, 0, width, height }; | |||||
| AdjustWindowRectEx( | |||||
| &wr, winFlags, FALSE, WS_EX_TOPMOST); | |||||
| impl->hwnd = CreateWindowEx( | |||||
| WS_EX_TOPMOST, | |||||
| classNameBuf, title, | |||||
| (addToDesktop ? WS_VISIBLE : 0) | (parent ? WS_CHILD : winFlags), | |||||
| 0, 0, wr.right-wr.left, wr.bottom-wr.top, | |||||
| (HWND)parent, NULL, NULL, NULL); | (HWND)parent, NULL, NULL, NULL); | ||||
| if (!impl->hwnd) { | if (!impl->hwnd) { | ||||
| free(impl); | free(impl); | ||||
| free(view); | free(view); | ||||
| return NULL; | return NULL; | ||||
| } | } | ||||
| SetWindowLongPtr(impl->hwnd, GWL_USERDATA, (LONG)view); | SetWindowLongPtr(impl->hwnd, GWL_USERDATA, (LONG)view); | ||||
| impl->hdc = GetDC(impl->hwnd); | impl->hdc = GetDC(impl->hwnd); | ||||
| @@ -107,10 +132,6 @@ puglCreate(PuglNativeWindow parent, | |||||
| view->height = height; | view->height = height; | ||||
| return view; | return view; | ||||
| // unused | |||||
| (void)resizable; | |||||
| (void)x11Display; | |||||
| } | } | ||||
| void | void | ||||
| @@ -120,11 +141,12 @@ puglDestroy(PuglView* view) | |||||
| wglDeleteContext(view->impl->hglrc); | wglDeleteContext(view->impl->hglrc); | ||||
| ReleaseDC(view->impl->hwnd, view->impl->hdc); | ReleaseDC(view->impl->hwnd, view->impl->hdc); | ||||
| DestroyWindow(view->impl->hwnd); | DestroyWindow(view->impl->hwnd); | ||||
| UnregisterClass(view->impl->wc.lpszClassName, NULL); | |||||
| free(view->impl); | free(view->impl); | ||||
| free(view); | free(view); | ||||
| } | } | ||||
| void | |||||
| static void | |||||
| puglReshape(PuglView* view, int width, int height) | puglReshape(PuglView* view, int width, int height) | ||||
| { | { | ||||
| wglMakeCurrent(view->impl->hdc, view->impl->hglrc); | wglMakeCurrent(view->impl->hdc, view->impl->hglrc); | ||||
| @@ -192,6 +214,13 @@ keySymToSpecial(int sym) | |||||
| static void | static void | ||||
| processMouseEvent(PuglView* view, int button, bool press, LPARAM lParam) | processMouseEvent(PuglView* view, int button, bool press, LPARAM lParam) | ||||
| { | { | ||||
| view->event_timestamp_ms = GetMessageTime(); | |||||
| if (press) { | |||||
| SetCapture(view->impl->hwnd); | |||||
| } else { | |||||
| ReleaseCapture(); | |||||
| } | |||||
| if (view->mouseFunc) { | if (view->mouseFunc) { | ||||
| view->mouseFunc(view, button, press, | view->mouseFunc(view, button, press, | ||||
| GET_X_LPARAM(lParam), | GET_X_LPARAM(lParam), | ||||
| @@ -221,7 +250,11 @@ handleMessage(PuglView* view, UINT message, WPARAM wParam, LPARAM lParam) | |||||
| case WM_CREATE: | case WM_CREATE: | ||||
| case WM_SHOWWINDOW: | case WM_SHOWWINDOW: | ||||
| case WM_SIZE: | case WM_SIZE: | ||||
| puglReshape(view, view->width, view->height); | |||||
| RECT rect; | |||||
| GetClientRect(view->impl->hwnd, &rect); | |||||
| puglReshape(view, rect.right, rect.bottom); | |||||
| view->width = rect.right; | |||||
| view->height = rect.bottom; | |||||
| break; | break; | ||||
| case WM_PAINT: | case WM_PAINT: | ||||
| BeginPaint(view->impl->hwnd, &ps); | BeginPaint(view->impl->hwnd, &ps); | ||||
| @@ -230,8 +263,7 @@ handleMessage(PuglView* view, UINT message, WPARAM wParam, LPARAM lParam) | |||||
| break; | break; | ||||
| case WM_MOUSEMOVE: | case WM_MOUSEMOVE: | ||||
| if (view->motionFunc) { | if (view->motionFunc) { | ||||
| view->motionFunc( | |||||
| view, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)); | |||||
| view->motionFunc(view, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)); | |||||
| } | } | ||||
| break; | break; | ||||
| case WM_LBUTTONDOWN: | case WM_LBUTTONDOWN: | ||||
| @@ -265,6 +297,7 @@ handleMessage(PuglView* view, UINT message, WPARAM wParam, LPARAM lParam) | |||||
| } | } | ||||
| break; | break; | ||||
| case WM_KEYDOWN: | case WM_KEYDOWN: | ||||
| view->event_timestamp_ms = (GetMessageTime()); | |||||
| if (view->ignoreKeyRepeat && (lParam & (1 << 30))) { | if (view->ignoreKeyRepeat && (lParam & (1 << 30))) { | ||||
| break; | break; | ||||
| } // else nobreak | } // else nobreak | ||||
| @@ -278,6 +311,7 @@ handleMessage(PuglView* view, UINT message, WPARAM wParam, LPARAM lParam) | |||||
| } | } | ||||
| break; | break; | ||||
| case WM_QUIT: | case WM_QUIT: | ||||
| case LOCAL_CLOSE_MSG: | |||||
| if (view->closeFunc) { | if (view->closeFunc) { | ||||
| view->closeFunc(view); | view->closeFunc(view); | ||||
| } | } | ||||
| @@ -298,8 +332,9 @@ puglProcessEvents(PuglView* view) | |||||
| handleMessage(view, msg.message, msg.wParam, msg.lParam); | handleMessage(view, msg.message, msg.wParam, msg.lParam); | ||||
| } | } | ||||
| if (view->redisplay) { | if (view->redisplay) { | ||||
| puglDisplay(view); | |||||
| InvalidateRect(view->impl->hwnd, NULL, FALSE); | |||||
| } | } | ||||
| return PUGL_SUCCESS; | return PUGL_SUCCESS; | ||||
| @@ -314,14 +349,10 @@ wndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) | |||||
| PostMessage(hwnd, WM_SHOWWINDOW, TRUE, 0); | PostMessage(hwnd, WM_SHOWWINDOW, TRUE, 0); | ||||
| return 0; | return 0; | ||||
| case WM_CLOSE: | case WM_CLOSE: | ||||
| PostQuitMessage(0); | |||||
| PostMessage(hwnd, LOCAL_CLOSE_MSG, wParam, lParam); | |||||
| return 0; | return 0; | ||||
| case WM_DESTROY: | case WM_DESTROY: | ||||
| return 0; | return 0; | ||||
| case WM_MOUSEWHEEL: | |||||
| case WM_MOUSEHWHEEL: | |||||
| PostMessage(hwnd, message, wParam, lParam); | |||||
| return 0; | |||||
| default: | default: | ||||
| if (view) { | if (view) { | ||||
| return handleMessage(view, message, wParam, lParam); | return handleMessage(view, message, wParam, lParam); | ||||
| @@ -23,9 +23,22 @@ | |||||
| #include <stdlib.h> | #include <stdlib.h> | ||||
| #include <string.h> | #include <string.h> | ||||
| #include "pugl_x11.h" | |||||
| #include <GL/gl.h> | |||||
| #include <GL/glx.h> | |||||
| #include <X11/Xatom.h> | |||||
| #include <X11/Xlib.h> | |||||
| #include <X11/keysym.h> | |||||
| #include "pugl_internal.h" | #include "pugl_internal.h" | ||||
| struct PuglInternalsImpl { | |||||
| Display* display; | |||||
| int screen; | |||||
| Window win; | |||||
| GLXContext ctx; | |||||
| Bool doubleBuffered; | |||||
| }; | |||||
| /** | /** | ||||
| Attributes for single-buffered RGBA with at least | Attributes for single-buffered RGBA with at least | ||||
| 4 bits per color and a 16 bit depth buffer. | 4 bits per color and a 16 bit depth buffer. | ||||
| @@ -58,20 +71,19 @@ puglCreate(PuglNativeWindow parent, | |||||
| int width, | int width, | ||||
| int height, | int height, | ||||
| bool resizable, | bool resizable, | ||||
| bool addToDesktop, | |||||
| const char* x11Display) | |||||
| bool addToDesktop) | |||||
| { | { | ||||
| PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | ||||
| PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | ||||
| if (!view || !impl) { | if (!view || !impl) { | ||||
| return nullptr; | |||||
| return NULL; | |||||
| } | } | ||||
| view->impl = impl; | view->impl = impl; | ||||
| view->width = width; | view->width = width; | ||||
| view->height = height; | view->height = height; | ||||
| impl->display = XOpenDisplay(x11Display); | |||||
| impl->display = XOpenDisplay(0); | |||||
| impl->screen = DefaultScreen(impl->display); | impl->screen = DefaultScreen(impl->display); | ||||
| XVisualInfo* vi = glXChooseVisual(impl->display, impl->screen, attrListDbl); | XVisualInfo* vi = glXChooseVisual(impl->display, impl->screen, attrListDbl); | ||||
| @@ -88,7 +100,7 @@ puglCreate(PuglNativeWindow parent, | |||||
| glXQueryVersion(impl->display, &glxMajor, &glxMinor); | glXQueryVersion(impl->display, &glxMajor, &glxMinor); | ||||
| printf("GLX-Version %d.%d\n", glxMajor, glxMinor); | printf("GLX-Version %d.%d\n", glxMajor, glxMinor); | ||||
| impl->ctx = glXCreateContext(impl->display, vi, nullptr, GL_TRUE); | |||||
| impl->ctx = glXCreateContext(impl->display, vi, 0, GL_TRUE); | |||||
| Window xParent = parent | Window xParent = parent | ||||
| ? (Window)parent | ? (Window)parent | ||||
| @@ -108,7 +120,7 @@ puglCreate(PuglNativeWindow parent, | |||||
| impl->win = XCreateWindow( | impl->win = XCreateWindow( | ||||
| impl->display, xParent, | impl->display, xParent, | ||||
| 0, 0, (unsigned int)view->width, (unsigned int)view->height, 0, vi->depth, InputOutput, vi->visual, | |||||
| 0, 0, view->width, view->height, 0, vi->depth, InputOutput, vi->visual, | |||||
| CWBorderPixel | CWColormap | CWEventMask, &attr); | CWBorderPixel | CWColormap | CWEventMask, &attr); | ||||
| XSizeHints sizeHints; | XSizeHints sizeHints; | ||||
| @@ -138,7 +150,7 @@ puglCreate(PuglNativeWindow parent, | |||||
| if (glXIsDirect(impl->display, impl->ctx)) { | if (glXIsDirect(impl->display, impl->ctx)) { | ||||
| printf("DRI enabled\n"); | printf("DRI enabled\n"); | ||||
| } else { | } else { | ||||
| printf("no DRI available\n"); | |||||
| printf("No DRI available\n"); | |||||
| } | } | ||||
| XFree(vi); | XFree(vi); | ||||
| @@ -232,8 +244,10 @@ keySymToSpecial(KeySym sym) | |||||
| } | } | ||||
| static void | static void | ||||
| setModifiers(PuglView* view, int xstate) | |||||
| setModifiers(PuglView* view, unsigned xstate, unsigned xtime) | |||||
| { | { | ||||
| view->event_timestamp_ms = xtime; | |||||
| view->mods = 0; | view->mods = 0; | ||||
| view->mods |= (xstate & ShiftMask) ? PUGL_MOD_SHIFT : 0; | view->mods |= (xstate & ShiftMask) ? PUGL_MOD_SHIFT : 0; | ||||
| view->mods |= (xstate & ControlMask) ? PUGL_MOD_CTRL : 0; | view->mods |= (xstate & ControlMask) ? PUGL_MOD_CTRL : 0; | ||||
| @@ -264,16 +278,15 @@ puglProcessEvents(PuglView* view) | |||||
| break; | break; | ||||
| } | } | ||||
| puglDisplay(view); | puglDisplay(view); | ||||
| view->redisplay = false; | |||||
| break; | break; | ||||
| case MotionNotify: | case MotionNotify: | ||||
| setModifiers(view, (int)event.xmotion.state); | |||||
| setModifiers(view, event.xmotion.state, event.xmotion.time); | |||||
| if (view->motionFunc) { | if (view->motionFunc) { | ||||
| view->motionFunc(view, event.xmotion.x, event.xmotion.y); | view->motionFunc(view, event.xmotion.x, event.xmotion.y); | ||||
| } | } | ||||
| break; | break; | ||||
| case ButtonPress: | case ButtonPress: | ||||
| setModifiers(view, (int)event.xbutton.state); | |||||
| setModifiers(view, event.xbutton.state, event.xbutton.time); | |||||
| if (event.xbutton.button >= 4 && event.xbutton.button <= 7) { | if (event.xbutton.button >= 4 && event.xbutton.button <= 7) { | ||||
| if (view->scrollFunc) { | if (view->scrollFunc) { | ||||
| float dx = 0, dy = 0; | float dx = 0, dy = 0; | ||||
| @@ -289,23 +302,23 @@ puglProcessEvents(PuglView* view) | |||||
| } | } | ||||
| // nobreak | // nobreak | ||||
| case ButtonRelease: | case ButtonRelease: | ||||
| setModifiers(view, (int)event.xbutton.state); | |||||
| setModifiers(view, event.xbutton.state, event.xbutton.time); | |||||
| if (view->mouseFunc && | if (view->mouseFunc && | ||||
| (event.xbutton.button < 4 || event.xbutton.button > 7)) { | (event.xbutton.button < 4 || event.xbutton.button > 7)) { | ||||
| view->mouseFunc(view, | view->mouseFunc(view, | ||||
| (int)event.xbutton.button, event.type == ButtonPress, | |||||
| event.xbutton.button, event.type == ButtonPress, | |||||
| event.xbutton.x, event.xbutton.y); | event.xbutton.x, event.xbutton.y); | ||||
| } | } | ||||
| break; | break; | ||||
| case KeyPress: { | case KeyPress: { | ||||
| setModifiers(view, (int)event.xkey.state); | |||||
| setModifiers(view, event.xkey.state, event.xkey.time); | |||||
| KeySym sym; | KeySym sym; | ||||
| char str[5]; | char str[5]; | ||||
| int n = XLookupString(&event.xkey, str, 4, &sym, nullptr); | |||||
| int n = XLookupString(&event.xkey, str, 4, &sym, NULL); | |||||
| PuglKey key = keySymToSpecial(sym); | PuglKey key = keySymToSpecial(sym); | ||||
| if (!key && view->keyboardFunc) { | if (!key && view->keyboardFunc) { | ||||
| if (n == 1) { | if (n == 1) { | ||||
| view->keyboardFunc(view, true, (uint32_t)str[0]); | |||||
| view->keyboardFunc(view, true, str[0]); | |||||
| } else { | } else { | ||||
| fprintf(stderr, "warning: Unknown key %X\n", (int)sym); | fprintf(stderr, "warning: Unknown key %X\n", (int)sym); | ||||
| } | } | ||||
| @@ -314,7 +327,7 @@ puglProcessEvents(PuglView* view) | |||||
| } | } | ||||
| } break; | } break; | ||||
| case KeyRelease: { | case KeyRelease: { | ||||
| setModifiers(view, (int)event.xkey.state); | |||||
| setModifiers(view, event.xkey.state, event.xkey.time); | |||||
| bool repeated = false; | bool repeated = false; | ||||
| if (view->ignoreKeyRepeat && | if (view->ignoreKeyRepeat && | ||||
| XEventsQueued(view->impl->display, QueuedAfterReading)) { | XEventsQueued(view->impl->display, QueuedAfterReading)) { | ||||
| @@ -330,10 +343,10 @@ puglProcessEvents(PuglView* view) | |||||
| if (!repeated && view->keyboardFunc) { | if (!repeated && view->keyboardFunc) { | ||||
| KeySym sym = XKeycodeToKeysym( | KeySym sym = XKeycodeToKeysym( | ||||
| view->impl->display, (KeyCode)event.xkey.keycode, 0); | |||||
| view->impl->display, event.xkey.keycode, 0); | |||||
| PuglKey special = keySymToSpecial(sym); | PuglKey special = keySymToSpecial(sym); | ||||
| if (!special) { | if (!special) { | ||||
| view->keyboardFunc(view, false, (uint32_t)sym); | |||||
| view->keyboardFunc(view, false, sym); | |||||
| } else if (view->specialFunc) { | } else if (view->specialFunc) { | ||||
| view->specialFunc(view, false, special); | view->specialFunc(view, false, special); | ||||
| } | } | ||||
| @@ -369,11 +382,5 @@ puglPostRedisplay(PuglView* view) | |||||
| PuglNativeWindow | PuglNativeWindow | ||||
| puglGetNativeWindow(PuglView* view) | puglGetNativeWindow(PuglView* view) | ||||
| { | { | ||||
| return static_cast<PuglNativeWindow>(view->impl->win); | |||||
| } | |||||
| PuglInternals* | |||||
| puglGetInternalsImpl(PuglView* view) | |||||
| { | |||||
| return view->impl; | |||||
| return view->impl->win; | |||||
| } | } | ||||
| @@ -1,46 +0,0 @@ | |||||
| /* | |||||
| Copyright 2012 David Robillard <http://drobilla.net> | |||||
| Copyright 2011-2012 Ben Loftis, Harrison Consoles | |||||
| Permission to use, copy, modify, and/or distribute this software for any | |||||
| purpose with or without fee is hereby granted, provided that the above | |||||
| copyright notice and this permission notice appear in all copies. | |||||
| THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |||||
| WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |||||
| MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |||||
| ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |||||
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |||||
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |||||
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |||||
| */ | |||||
| /** | |||||
| @file pugl_x11.h X11 Pugl Implementation (PuglInternalsImpl only). | |||||
| */ | |||||
| #ifndef PUGL_X11_H_INCLUDED | |||||
| #define PUGL_X11_H_INCLUDED | |||||
| #include "pugl.h" | |||||
| #include <GL/gl.h> | |||||
| #include <GL/glx.h> | |||||
| #include <X11/Xatom.h> | |||||
| #include <X11/Xlib.h> | |||||
| #include <X11/keysym.h> | |||||
| struct PuglInternalsImpl { | |||||
| Display* display; | |||||
| int screen; | |||||
| Window win; | |||||
| GLXContext ctx; | |||||
| Bool doubleBuffered; | |||||
| }; | |||||
| typedef struct PuglInternalsImpl PuglInternals; | |||||
| PuglInternals* | |||||
| puglGetInternalsImpl(PuglView* view); | |||||
| #endif /* PUGL_X11_H_INCLUDED */ | |||||
| @@ -1,341 +0,0 @@ | |||||
| /* | |||||
| Copyright 2012 David Robillard <http://drobilla.net> | |||||
| Permission to use, copy, modify, and/or distribute this software for any | |||||
| purpose with or without fee is hereby granted, provided that the above | |||||
| copyright notice and this permission notice appear in all copies. | |||||
| THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |||||
| WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |||||
| MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |||||
| ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |||||
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |||||
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |||||
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |||||
| */ | |||||
| /** | |||||
| @file pugl.h API for Pugl, a minimal portable API for OpenGL. | |||||
| */ | |||||
| #ifndef PUGL_H_INCLUDED | |||||
| #define PUGL_H_INCLUDED | |||||
| #include <stdint.h> | |||||
| /* | |||||
| This API is pure portable C and contains no platform specific elements, or | |||||
| even a GL dependency. However, unfortunately GL includes vary across | |||||
| platforms so they are included here to allow for pure portable programs. | |||||
| */ | |||||
| #ifdef __APPLE__ | |||||
| # include "OpenGL/gl.h" | |||||
| #else | |||||
| # ifdef _WIN32 | |||||
| # include <windows.h> /* Broken Windows GL headers require this */ | |||||
| # endif | |||||
| # include "GL/gl.h" | |||||
| #endif | |||||
| #ifdef PUGL_SHARED | |||||
| # ifdef _WIN32 | |||||
| # define PUGL_LIB_IMPORT __declspec(dllimport) | |||||
| # define PUGL_LIB_EXPORT __declspec(dllexport) | |||||
| # else | |||||
| # define PUGL_LIB_IMPORT __attribute__((visibility("default"))) | |||||
| # define PUGL_LIB_EXPORT __attribute__((visibility("default"))) | |||||
| # endif | |||||
| # ifdef PUGL_INTERNAL | |||||
| # define PUGL_API PUGL_LIB_EXPORT | |||||
| # else | |||||
| # define PUGL_API PUGL_LIB_IMPORT | |||||
| # endif | |||||
| #else | |||||
| # define PUGL_API | |||||
| #endif | |||||
| #ifndef __cplusplus | |||||
| # include <stdbool.h> | |||||
| #endif | |||||
| /** | |||||
| @defgroup pugl Pugl | |||||
| A minimal portable API for OpenGL. | |||||
| @{ | |||||
| */ | |||||
| /** | |||||
| An OpenGL view. | |||||
| */ | |||||
| typedef struct PuglViewImpl PuglView; | |||||
| /** | |||||
| A native window handle. | |||||
| On X11, this is a Window. | |||||
| On OSX, this is an NSView*. | |||||
| On Windows, this is a HWND. | |||||
| */ | |||||
| typedef intptr_t PuglNativeWindow; | |||||
| /** | |||||
| Return status code. | |||||
| */ | |||||
| typedef enum { | |||||
| PUGL_SUCCESS = 0 | |||||
| } PuglStatus; | |||||
| /** | |||||
| Convenience symbols for ASCII control characters. | |||||
| */ | |||||
| typedef enum { | |||||
| PUGL_CHAR_BACKSPACE = 0x08, | |||||
| PUGL_CHAR_ESCAPE = 0x1B, | |||||
| PUGL_CHAR_DELETE = 0x7F | |||||
| } PuglChar; | |||||
| /** | |||||
| Special (non-Unicode) keyboard keys. | |||||
| */ | |||||
| typedef enum { | |||||
| PUGL_KEY_F1 = 1, | |||||
| PUGL_KEY_F2, | |||||
| PUGL_KEY_F3, | |||||
| PUGL_KEY_F4, | |||||
| PUGL_KEY_F5, | |||||
| PUGL_KEY_F6, | |||||
| PUGL_KEY_F7, | |||||
| PUGL_KEY_F8, | |||||
| PUGL_KEY_F9, | |||||
| PUGL_KEY_F10, | |||||
| PUGL_KEY_F11, | |||||
| PUGL_KEY_F12, | |||||
| PUGL_KEY_LEFT, | |||||
| PUGL_KEY_UP, | |||||
| PUGL_KEY_RIGHT, | |||||
| PUGL_KEY_DOWN, | |||||
| PUGL_KEY_PAGE_UP, | |||||
| PUGL_KEY_PAGE_DOWN, | |||||
| PUGL_KEY_HOME, | |||||
| PUGL_KEY_END, | |||||
| PUGL_KEY_INSERT, | |||||
| PUGL_KEY_SHIFT, | |||||
| PUGL_KEY_CTRL, | |||||
| PUGL_KEY_ALT, | |||||
| PUGL_KEY_SUPER | |||||
| } PuglKey; | |||||
| /** | |||||
| Keyboard modifier flags. | |||||
| */ | |||||
| typedef enum { | |||||
| PUGL_MOD_SHIFT = 1, /**< Shift key */ | |||||
| PUGL_MOD_CTRL = 1 << 1, /**< Control key */ | |||||
| PUGL_MOD_ALT = 1 << 2, /**< Alt/Option key */ | |||||
| PUGL_MOD_SUPER = 1 << 3 /**< Mod4/Command/Windows key */ | |||||
| } PuglMod; | |||||
| /** | |||||
| Handle for opaque user data. | |||||
| */ | |||||
| typedef void* PuglHandle; | |||||
| /** | |||||
| A function called when the window is closed. | |||||
| */ | |||||
| typedef void (*PuglCloseFunc)(PuglView* view); | |||||
| /** | |||||
| A function called to draw the view contents with OpenGL. | |||||
| */ | |||||
| typedef void (*PuglDisplayFunc)(PuglView* view); | |||||
| /** | |||||
| A function called when a key is pressed or released. | |||||
| @param view The view the event occured in. | |||||
| @param press True if the key was pressed, false if released. | |||||
| @param key Unicode point of the key pressed. | |||||
| */ | |||||
| typedef void (*PuglKeyboardFunc)(PuglView* view, bool press, uint32_t key); | |||||
| /** | |||||
| A function called when the pointer moves. | |||||
| @param view The view the event occured in. | |||||
| @param x The window-relative x coordinate of the pointer. | |||||
| @param y The window-relative y coordinate of the pointer. | |||||
| */ | |||||
| typedef void (*PuglMotionFunc)(PuglView* view, int x, int y); | |||||
| /** | |||||
| A function called when a mouse button is pressed or released. | |||||
| @param view The view the event occured in. | |||||
| @param button The button number (1 = left, 2 = middle, 3 = right). | |||||
| @param press True if the key was pressed, false if released. | |||||
| @param x The window-relative x coordinate of the pointer. | |||||
| @param y The window-relative y coordinate of the pointer. | |||||
| */ | |||||
| typedef void (*PuglMouseFunc)( | |||||
| PuglView* view, int button, bool press, int x, int y); | |||||
| /** | |||||
| A function called when the view is resized. | |||||
| @param view The view being resized. | |||||
| @param width The new view width. | |||||
| @param height The new view height. | |||||
| */ | |||||
| typedef void (*PuglReshapeFunc)(PuglView* view, int width, int height); | |||||
| /** | |||||
| A function called on scrolling (e.g. mouse wheel or track pad). | |||||
| The distances used here are in "lines", a single tick of a clicking mouse | |||||
| wheel. For example, @p dy = 1.0 scrolls 1 line up. Some systems and | |||||
| devices support finer resolution and/or higher values for fast scrolls, | |||||
| so programs should handle any value gracefully. | |||||
| @param view The view being scrolled. | |||||
| @param dx The scroll x distance. | |||||
| @param dx The scroll y distance. | |||||
| */ | |||||
| typedef void (*PuglScrollFunc)(PuglView* view, float dx, float dy); | |||||
| /** | |||||
| A function called when a special key is pressed or released. | |||||
| This callback allows the use of keys that do not have unicode points. Note | |||||
| that some non-printable keys | |||||
| @param view The view the event occured in. | |||||
| @param press True if the key was pressed, false if released. | |||||
| @param key The key pressed. | |||||
| */ | |||||
| typedef void (*PuglSpecialFunc)(PuglView* view, bool press, PuglKey key); | |||||
| /** | |||||
| Create a new GL window. | |||||
| @param parent Parent window, or 0 for top level. | |||||
| @param title Window title, or NULL. | |||||
| @param width Window width in pixels. | |||||
| @param height Window height in pixels. | |||||
| @param resizable Whether window should be user resizable. | |||||
| */ | |||||
| PUGL_API PuglView* | |||||
| puglCreate(PuglNativeWindow parent, | |||||
| const char* title, | |||||
| int width, | |||||
| int height, | |||||
| bool resizable, | |||||
| bool addToDesktop = true, | |||||
| const char* x11Display = nullptr); | |||||
| /** | |||||
| Set the handle to be passed to all callbacks. | |||||
| This is generally a pointer to a struct which contains all necessary state. | |||||
| Everything needed in callbacks should be here, not in static variables. | |||||
| Note the lack of this facility makes GLUT unsuitable for plugins or | |||||
| non-trivial programs; this mistake is largely why Pugl exists. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetHandle(PuglView* view, PuglHandle handle); | |||||
| /** | |||||
| Get the handle to be passed to all callbacks. | |||||
| */ | |||||
| PUGL_API PuglHandle | |||||
| puglGetHandle(PuglView* view); | |||||
| /** | |||||
| Get the currently active modifiers (PuglMod flags). | |||||
| This should only be called from an event handler. | |||||
| */ | |||||
| PUGL_API int | |||||
| puglGetModifiers(PuglView* view); | |||||
| /** | |||||
| Ignore synthetic repeated key events. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglIgnoreKeyRepeat(PuglView* view, bool ignore); | |||||
| /** | |||||
| Set the function to call when the window is closed. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetCloseFunc(PuglView* view, PuglCloseFunc closeFunc); | |||||
| /** | |||||
| Set the display function which should draw the UI using GL. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetDisplayFunc(PuglView* view, PuglDisplayFunc displayFunc); | |||||
| /** | |||||
| Set the function to call on keyboard events. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetKeyboardFunc(PuglView* view, PuglKeyboardFunc keyboardFunc); | |||||
| /** | |||||
| Set the function to call on mouse motion. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetMotionFunc(PuglView* view, PuglMotionFunc motionFunc); | |||||
| /** | |||||
| Set the function to call on mouse button events. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetMouseFunc(PuglView* view, PuglMouseFunc mouseFunc); | |||||
| /** | |||||
| Set the function to call on scroll events. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetScrollFunc(PuglView* view, PuglScrollFunc scrollFunc); | |||||
| /** | |||||
| Set the function to call on special events. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetSpecialFunc(PuglView* view, PuglSpecialFunc specialFunc); | |||||
| /** | |||||
| Set the function to call when the window size changes. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglSetReshapeFunc(PuglView* view, PuglReshapeFunc reshapeFunc); | |||||
| /** | |||||
| Return the native window handle. | |||||
| */ | |||||
| PUGL_API PuglNativeWindow | |||||
| puglGetNativeWindow(PuglView* view); | |||||
| /** | |||||
| Process all pending window events. | |||||
| This handles input events as well as rendering, so it should be called | |||||
| regularly and rapidly enough to keep the UI responsive. | |||||
| */ | |||||
| PUGL_API PuglStatus | |||||
| puglProcessEvents(PuglView* view); | |||||
| /** | |||||
| Request a redisplay on the next call to puglProcessEvents(). | |||||
| */ | |||||
| PUGL_API void | |||||
| puglPostRedisplay(PuglView* view); | |||||
| /** | |||||
| Destroy a GL window. | |||||
| */ | |||||
| PUGL_API void | |||||
| puglDestroy(PuglView* view); | |||||
| /** | |||||
| @} | |||||
| */ | |||||
| #endif /* PUGL_H_INCLUDED */ | |||||
| @@ -1,135 +0,0 @@ | |||||
| /* | |||||
| Copyright 2012 David Robillard <http://drobilla.net> | |||||
| Permission to use, copy, modify, and/or distribute this software for any | |||||
| purpose with or without fee is hereby granted, provided that the above | |||||
| copyright notice and this permission notice appear in all copies. | |||||
| THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |||||
| WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |||||
| MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |||||
| ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |||||
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |||||
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |||||
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |||||
| */ | |||||
| /** | |||||
| @file pugl_internal.h Private platform-independent definitions. | |||||
| Note this file contains function definitions, so it must be compiled into | |||||
| the final binary exactly once. Each platform specific implementation file | |||||
| including it once should achieve this. | |||||
| */ | |||||
| #include "pugl.h" | |||||
| typedef struct PuglInternalsImpl PuglInternals; | |||||
| struct PuglViewImpl { | |||||
| PuglHandle handle; | |||||
| PuglCloseFunc closeFunc; | |||||
| PuglDisplayFunc displayFunc; | |||||
| PuglKeyboardFunc keyboardFunc; | |||||
| PuglMotionFunc motionFunc; | |||||
| PuglMouseFunc mouseFunc; | |||||
| PuglReshapeFunc reshapeFunc; | |||||
| PuglScrollFunc scrollFunc; | |||||
| PuglSpecialFunc specialFunc; | |||||
| PuglInternals* impl; | |||||
| int width; | |||||
| int height; | |||||
| int mods; | |||||
| bool ignoreKeyRepeat; | |||||
| bool redisplay; | |||||
| }; | |||||
| void | |||||
| puglSetHandle(PuglView* view, PuglHandle handle) | |||||
| { | |||||
| view->handle = handle; | |||||
| } | |||||
| PuglHandle | |||||
| puglGetHandle(PuglView* view) | |||||
| { | |||||
| return view->handle; | |||||
| } | |||||
| int | |||||
| puglGetModifiers(PuglView* view) | |||||
| { | |||||
| return view->mods; | |||||
| } | |||||
| static inline void | |||||
| puglDefaultReshape(PuglView* view, int width, int height) | |||||
| { | |||||
| glMatrixMode(GL_PROJECTION); | |||||
| glLoadIdentity(); | |||||
| glOrtho(0, width, height, 0, 0, 1); | |||||
| glViewport(0, 0, width, height); | |||||
| glMatrixMode(GL_MODELVIEW); | |||||
| glLoadIdentity(); | |||||
| return; | |||||
| // unused | |||||
| (void)view; | |||||
| } | |||||
| void | |||||
| puglIgnoreKeyRepeat(PuglView* view, bool ignore) | |||||
| { | |||||
| view->ignoreKeyRepeat = ignore; | |||||
| } | |||||
| void | |||||
| puglSetCloseFunc(PuglView* view, PuglCloseFunc closeFunc) | |||||
| { | |||||
| view->closeFunc = closeFunc; | |||||
| } | |||||
| void | |||||
| puglSetDisplayFunc(PuglView* view, PuglDisplayFunc displayFunc) | |||||
| { | |||||
| view->displayFunc = displayFunc; | |||||
| } | |||||
| void | |||||
| puglSetKeyboardFunc(PuglView* view, PuglKeyboardFunc keyboardFunc) | |||||
| { | |||||
| view->keyboardFunc = keyboardFunc; | |||||
| } | |||||
| void | |||||
| puglSetMotionFunc(PuglView* view, PuglMotionFunc motionFunc) | |||||
| { | |||||
| view->motionFunc = motionFunc; | |||||
| } | |||||
| void | |||||
| puglSetMouseFunc(PuglView* view, PuglMouseFunc mouseFunc) | |||||
| { | |||||
| view->mouseFunc = mouseFunc; | |||||
| } | |||||
| void | |||||
| puglSetReshapeFunc(PuglView* view, PuglReshapeFunc reshapeFunc) | |||||
| { | |||||
| view->reshapeFunc = reshapeFunc; | |||||
| } | |||||
| void | |||||
| puglSetScrollFunc(PuglView* view, PuglScrollFunc scrollFunc) | |||||
| { | |||||
| view->scrollFunc = scrollFunc; | |||||
| } | |||||
| void | |||||
| puglSetSpecialFunc(PuglView* view, PuglSpecialFunc specialFunc) | |||||
| { | |||||
| view->specialFunc = specialFunc; | |||||
| } | |||||
| @@ -1,331 +0,0 @@ | |||||
| /* | |||||
| Copyright 2012 David Robillard <http://drobilla.net> | |||||
| Permission to use, copy, modify, and/or distribute this software for any | |||||
| purpose with or without fee is hereby granted, provided that the above | |||||
| copyright notice and this permission notice appear in all copies. | |||||
| THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |||||
| WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |||||
| MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |||||
| ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |||||
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |||||
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |||||
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |||||
| */ | |||||
| /** | |||||
| @file pugl_osx.m OSX/Cocoa Pugl Implementation. | |||||
| */ | |||||
| #include <stdlib.h> | |||||
| #import <Cocoa/Cocoa.h> | |||||
| #include "pugl_internal.h" | |||||
| @interface PuglOpenGLView : NSOpenGLView | |||||
| { | |||||
| int colorBits; | |||||
| int depthBits; | |||||
| @public | |||||
| PuglView* view; | |||||
| } | |||||
| - (id) initWithFrame:(NSRect)frame | |||||
| colorBits:(int)numColorBits | |||||
| depthBits:(int)numDepthBits; | |||||
| - (void) reshape; | |||||
| - (void) drawRect:(NSRect)rect; | |||||
| - (void) mouseMoved:(NSEvent*)event; | |||||
| - (void) mouseDown:(NSEvent*)event; | |||||
| - (void) mouseUp:(NSEvent*)event; | |||||
| - (void) rightMouseDown:(NSEvent*)event; | |||||
| - (void) rightMouseUp:(NSEvent*)event; | |||||
| - (void) keyDown:(NSEvent*)event; | |||||
| - (void) keyUp:(NSEvent*)event; | |||||
| - (void) flagsChanged:(NSEvent*)event; | |||||
| @end | |||||
| @implementation PuglOpenGLView | |||||
| - (id) initWithFrame:(NSRect)frame | |||||
| colorBits:(int)numColorBits | |||||
| depthBits:(int)numDepthBits | |||||
| { | |||||
| colorBits = numColorBits; | |||||
| depthBits = numDepthBits; | |||||
| NSOpenGLPixelFormatAttribute pixelAttribs[16] = { | |||||
| NSOpenGLPFADoubleBuffer, | |||||
| NSOpenGLPFAAccelerated, | |||||
| NSOpenGLPFAColorSize, | |||||
| colorBits, | |||||
| NSOpenGLPFADepthSize, | |||||
| depthBits, | |||||
| 0 | |||||
| }; | |||||
| NSOpenGLPixelFormat* pixelFormat = [[NSOpenGLPixelFormat alloc] | |||||
| initWithAttributes:pixelAttribs]; | |||||
| if (pixelFormat) { | |||||
| self = [super initWithFrame:frame pixelFormat:pixelFormat]; | |||||
| [pixelFormat release]; | |||||
| if (self) { | |||||
| [[self openGLContext] makeCurrentContext]; | |||||
| [self reshape]; | |||||
| } | |||||
| } else { | |||||
| self = nil; | |||||
| } | |||||
| return self; | |||||
| } | |||||
| - (void) reshape | |||||
| { | |||||
| [[self openGLContext] update]; | |||||
| NSRect bounds = [self bounds]; | |||||
| int width = bounds.size.width; | |||||
| int height = bounds.size.height; | |||||
| if (view->reshapeFunc) { | |||||
| view->reshapeFunc(view, width, height); | |||||
| } else { | |||||
| puglDefaultReshape(view, width, height); | |||||
| } | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| } | |||||
| - (void) drawRect:(NSRect)rect | |||||
| { | |||||
| glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); | |||||
| glLoadIdentity(); | |||||
| if (self->view->displayFunc) { | |||||
| self->view->displayFunc(self->view); | |||||
| } | |||||
| glFlush(); | |||||
| glSwapAPPLE(); | |||||
| } | |||||
| static int | |||||
| getModifiers(unsigned modifierFlags) | |||||
| { | |||||
| int mods = 0; | |||||
| mods |= (modifierFlags & NSShiftKeyMask) ? PUGL_MOD_SHIFT : 0; | |||||
| mods |= (modifierFlags & NSControlKeyMask) ? PUGL_MOD_CTRL : 0; | |||||
| mods |= (modifierFlags & NSAlternateKeyMask) ? PUGL_MOD_ALT : 0; | |||||
| mods |= (modifierFlags & NSCommandKeyMask) ? PUGL_MOD_SUPER : 0; | |||||
| return mods; | |||||
| } | |||||
| - (void) mouseMoved:(NSEvent*)event | |||||
| { | |||||
| if (view->motionFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->motionFunc(view, loc.x, loc.y); | |||||
| } | |||||
| } | |||||
| - (void) mouseDown:(NSEvent*)event | |||||
| { | |||||
| if (view->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 1, true, loc.x, loc.y); | |||||
| } | |||||
| } | |||||
| - (void) mouseUp:(NSEvent*)event | |||||
| { | |||||
| if (view->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 1, false, loc.x, loc.y); | |||||
| } | |||||
| } | |||||
| - (void) rightMouseDown:(NSEvent*)event | |||||
| { | |||||
| if (view->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 3, true, loc.x, loc.y); | |||||
| } | |||||
| } | |||||
| - (void) rightMouseUp:(NSEvent*)event | |||||
| { | |||||
| if (view->mouseFunc) { | |||||
| NSPoint loc = [event locationInWindow]; | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->mouseFunc(view, 3, false, loc.x, loc.y); | |||||
| } | |||||
| } | |||||
| - (void) scrollWheel:(NSEvent*)event | |||||
| { | |||||
| if (view->scrollFunc) { | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->scrollFunc(view, [event deltaX], [event deltaY]); | |||||
| } | |||||
| } | |||||
| - (void) keyDown:(NSEvent*)event | |||||
| { | |||||
| if (view->keyboardFunc && !(view->ignoreKeyRepeat && [event isARepeat])) { | |||||
| NSString* chars = [event characters]; | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->keyboardFunc(view, true, [chars characterAtIndex:0]); | |||||
| } | |||||
| } | |||||
| - (void) keyUp:(NSEvent*)event | |||||
| { | |||||
| if (view->keyboardFunc) { | |||||
| NSString* chars = [event characters]; | |||||
| view->mods = getModifiers([event modifierFlags]); | |||||
| view->keyboardFunc(view, false, [chars characterAtIndex:0]); | |||||
| } | |||||
| } | |||||
| - (void) flagsChanged:(NSEvent*)event | |||||
| { | |||||
| if (view->specialFunc) { | |||||
| int mods = getModifiers([event modifierFlags]); | |||||
| if ((mods & PUGL_MOD_SHIFT) != (view->mods & PUGL_MOD_SHIFT)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_SHIFT, PUGL_KEY_SHIFT); | |||||
| } else if ((mods & PUGL_MOD_CTRL) != (view->mods & PUGL_MOD_CTRL)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_CTRL, PUGL_KEY_CTRL); | |||||
| } else if ((mods & PUGL_MOD_ALT) != (view->mods & PUGL_MOD_ALT)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_ALT, PUGL_KEY_ALT); | |||||
| } else if ((mods & PUGL_MOD_SUPER) != (view->mods & PUGL_MOD_SUPER)) { | |||||
| view->specialFunc(view, mods & PUGL_MOD_SUPER, PUGL_KEY_SUPER); | |||||
| } | |||||
| view->mods = mods; | |||||
| } | |||||
| } | |||||
| @end | |||||
| struct PuglInternalsImpl { | |||||
| PuglOpenGLView* view; | |||||
| NSModalSession session; | |||||
| id window; | |||||
| }; | |||||
| PuglView* | |||||
| puglCreate(PuglNativeWindow parent, | |||||
| const char* title, | |||||
| int width, | |||||
| int height, | |||||
| bool resizable, | |||||
| bool addToDesktop, | |||||
| const char* x11Display) | |||||
| { | |||||
| PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | |||||
| PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | |||||
| if (!view || !impl) { | |||||
| return NULL; | |||||
| } | |||||
| view->impl = impl; | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| [NSAutoreleasePool new]; | |||||
| [NSApplication sharedApplication]; | |||||
| [NSApp setActivationPolicy:NSApplicationActivationPolicyRegular]; | |||||
| NSString* titleString = [[NSString alloc] | |||||
| initWithBytes:title | |||||
| length:strlen(title) | |||||
| encoding:NSUTF8StringEncoding]; | |||||
| id window = [[[NSWindow alloc] | |||||
| initWithContentRect:NSMakeRect(0, 0, 512, 512) | |||||
| styleMask:NSTitledWindowMask | |||||
| backing:NSBackingStoreBuffered | |||||
| defer:NO] | |||||
| autorelease]; | |||||
| [window cascadeTopLeftFromPoint:NSMakePoint(20, 20)]; | |||||
| [window setTitle:titleString]; | |||||
| [window setAcceptsMouseMovedEvents:YES]; | |||||
| impl->view = [PuglOpenGLView new]; | |||||
| impl->window = window; | |||||
| impl->view->view = view; | |||||
| [window setContentView:impl->view]; | |||||
| [NSApp activateIgnoringOtherApps:YES]; | |||||
| [window makeFirstResponder:impl->view]; | |||||
| impl->session = [NSApp beginModalSessionForWindow:view->impl->window]; | |||||
| return view; | |||||
| // unused | |||||
| (void)addToDesktop; | |||||
| (void)x11Display; | |||||
| } | |||||
| void | |||||
| puglDestroy(PuglView* view) | |||||
| { | |||||
| [NSApp endModalSession:view->impl->session]; | |||||
| [view->impl->view release]; | |||||
| free(view->impl); | |||||
| free(view); | |||||
| } | |||||
| void | |||||
| puglDisplay(PuglView* view) | |||||
| { | |||||
| glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); | |||||
| glLoadIdentity(); | |||||
| if (view->displayFunc) { | |||||
| view->displayFunc(view); | |||||
| } | |||||
| glFlush(); | |||||
| view->redisplay = false; | |||||
| } | |||||
| PuglStatus | |||||
| puglProcessEvents(PuglView* view) | |||||
| { | |||||
| NSInteger response = [NSApp runModalSession:view->impl->session]; | |||||
| if (response != NSRunContinuesResponse) { | |||||
| if (view->closeFunc) { | |||||
| view->closeFunc(view); | |||||
| } | |||||
| } | |||||
| if (view->redisplay) { | |||||
| puglDisplay(view); | |||||
| } | |||||
| return PUGL_SUCCESS; | |||||
| } | |||||
| void | |||||
| puglPostRedisplay(PuglView* view) | |||||
| { | |||||
| view->redisplay = true; | |||||
| } | |||||
| PuglNativeWindow | |||||
| puglGetNativeWindow(PuglView* view) | |||||
| { | |||||
| return (PuglNativeWindow)view->impl->view; | |||||
| } | |||||
| @@ -1,344 +0,0 @@ | |||||
| /* | |||||
| Copyright 2012 David Robillard <http://drobilla.net> | |||||
| Permission to use, copy, modify, and/or distribute this software for any | |||||
| purpose with or without fee is hereby granted, provided that the above | |||||
| copyright notice and this permission notice appear in all copies. | |||||
| THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |||||
| WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |||||
| MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |||||
| ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |||||
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |||||
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |||||
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |||||
| */ | |||||
| /** | |||||
| @file pugl_win.cpp Windows/WGL Pugl Implementation. | |||||
| */ | |||||
| #include <windows.h> | |||||
| #include <windowsx.h> | |||||
| #include <GL/gl.h> | |||||
| #include "pugl_internal.h" | |||||
| #ifndef WM_MOUSEWHEEL | |||||
| # define WM_MOUSEWHEEL 0x020A | |||||
| #endif | |||||
| #ifndef WM_MOUSEHWHEEL | |||||
| # define WM_MOUSEHWHEEL 0x020E | |||||
| #endif | |||||
| struct PuglInternalsImpl { | |||||
| HWND hwnd; | |||||
| HDC hdc; | |||||
| HGLRC hglrc; | |||||
| }; | |||||
| LRESULT CALLBACK | |||||
| wndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam); | |||||
| PuglView* | |||||
| puglCreate(PuglNativeWindow parent, | |||||
| const char* title, | |||||
| int width, | |||||
| int height, | |||||
| bool resizable, | |||||
| bool addToDesktop, | |||||
| const char* x11Display) | |||||
| { | |||||
| PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | |||||
| PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | |||||
| if (!view || !impl) { | |||||
| return NULL; | |||||
| } | |||||
| view->impl = impl; | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| WNDCLASS wc; | |||||
| wc.style = CS_OWNDC; | |||||
| wc.lpfnWndProc = wndProc; | |||||
| wc.cbClsExtra = 0; | |||||
| wc.cbWndExtra = 0; | |||||
| wc.hInstance = 0; | |||||
| wc.hIcon = LoadIcon(NULL, IDI_APPLICATION); | |||||
| wc.hCursor = LoadCursor(NULL, IDC_ARROW); | |||||
| wc.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH); | |||||
| wc.lpszMenuName = NULL; | |||||
| wc.lpszClassName = "Pugl"; | |||||
| RegisterClass(&wc); | |||||
| impl->hwnd = CreateWindow( | |||||
| "Pugl", title, | |||||
| (addToDesktop ? WS_VISIBLE : 0) | (parent ? WS_CHILD : (WS_POPUPWINDOW | WS_CAPTION)), | |||||
| 0, 0, width, height, | |||||
| (HWND)parent, NULL, NULL, NULL); | |||||
| if (!impl->hwnd) { | |||||
| free(impl); | |||||
| free(view); | |||||
| return NULL; | |||||
| } | |||||
| SetWindowLongPtr(impl->hwnd, GWL_USERDATA, (LONG)view); | |||||
| impl->hdc = GetDC(impl->hwnd); | |||||
| PIXELFORMATDESCRIPTOR pfd; | |||||
| ZeroMemory(&pfd, sizeof(pfd)); | |||||
| pfd.nSize = sizeof(pfd); | |||||
| pfd.nVersion = 1; | |||||
| pfd.dwFlags = PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER; | |||||
| pfd.iPixelType = PFD_TYPE_RGBA; | |||||
| pfd.cColorBits = 24; | |||||
| pfd.cDepthBits = 16; | |||||
| pfd.iLayerType = PFD_MAIN_PLANE; | |||||
| int format = ChoosePixelFormat(impl->hdc, &pfd); | |||||
| SetPixelFormat(impl->hdc, format, &pfd); | |||||
| impl->hglrc = wglCreateContext(impl->hdc); | |||||
| wglMakeCurrent(impl->hdc, impl->hglrc); | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| return view; | |||||
| // unused | |||||
| (void)resizable; | |||||
| (void)x11Display; | |||||
| } | |||||
| void | |||||
| puglDestroy(PuglView* view) | |||||
| { | |||||
| wglMakeCurrent(NULL, NULL); | |||||
| wglDeleteContext(view->impl->hglrc); | |||||
| ReleaseDC(view->impl->hwnd, view->impl->hdc); | |||||
| DestroyWindow(view->impl->hwnd); | |||||
| free(view->impl); | |||||
| free(view); | |||||
| } | |||||
| void | |||||
| puglReshape(PuglView* view, int width, int height) | |||||
| { | |||||
| wglMakeCurrent(view->impl->hdc, view->impl->hglrc); | |||||
| if (view->reshapeFunc) { | |||||
| view->reshapeFunc(view, width, height); | |||||
| } else { | |||||
| puglDefaultReshape(view, width, height); | |||||
| } | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| } | |||||
| void | |||||
| puglDisplay(PuglView* view) | |||||
| { | |||||
| wglMakeCurrent(view->impl->hdc, view->impl->hglrc); | |||||
| glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); | |||||
| glLoadIdentity(); | |||||
| if (view->displayFunc) { | |||||
| view->displayFunc(view); | |||||
| } | |||||
| glFlush(); | |||||
| SwapBuffers(view->impl->hdc); | |||||
| view->redisplay = false; | |||||
| } | |||||
| static PuglKey | |||||
| keySymToSpecial(int sym) | |||||
| { | |||||
| switch (sym) { | |||||
| case VK_F1: return PUGL_KEY_F1; | |||||
| case VK_F2: return PUGL_KEY_F2; | |||||
| case VK_F3: return PUGL_KEY_F3; | |||||
| case VK_F4: return PUGL_KEY_F4; | |||||
| case VK_F5: return PUGL_KEY_F5; | |||||
| case VK_F6: return PUGL_KEY_F6; | |||||
| case VK_F7: return PUGL_KEY_F7; | |||||
| case VK_F8: return PUGL_KEY_F8; | |||||
| case VK_F9: return PUGL_KEY_F9; | |||||
| case VK_F10: return PUGL_KEY_F10; | |||||
| case VK_F11: return PUGL_KEY_F11; | |||||
| case VK_F12: return PUGL_KEY_F12; | |||||
| case VK_LEFT: return PUGL_KEY_LEFT; | |||||
| case VK_UP: return PUGL_KEY_UP; | |||||
| case VK_RIGHT: return PUGL_KEY_RIGHT; | |||||
| case VK_DOWN: return PUGL_KEY_DOWN; | |||||
| case VK_PRIOR: return PUGL_KEY_PAGE_UP; | |||||
| case VK_NEXT: return PUGL_KEY_PAGE_DOWN; | |||||
| case VK_HOME: return PUGL_KEY_HOME; | |||||
| case VK_END: return PUGL_KEY_END; | |||||
| case VK_INSERT: return PUGL_KEY_INSERT; | |||||
| case VK_SHIFT: return PUGL_KEY_SHIFT; | |||||
| case VK_CONTROL: return PUGL_KEY_CTRL; | |||||
| case VK_MENU: return PUGL_KEY_ALT; | |||||
| case VK_LWIN: return PUGL_KEY_SUPER; | |||||
| case VK_RWIN: return PUGL_KEY_SUPER; | |||||
| } | |||||
| return (PuglKey)0; | |||||
| } | |||||
| static void | |||||
| processMouseEvent(PuglView* view, int button, bool press, LPARAM lParam) | |||||
| { | |||||
| if (view->mouseFunc) { | |||||
| view->mouseFunc(view, button, press, | |||||
| GET_X_LPARAM(lParam), | |||||
| GET_Y_LPARAM(lParam)); | |||||
| } | |||||
| } | |||||
| static void | |||||
| setModifiers(PuglView* view) | |||||
| { | |||||
| view->mods = 0; | |||||
| view->mods |= (GetKeyState(VK_SHIFT) < 0) ? PUGL_MOD_SHIFT : 0; | |||||
| view->mods |= (GetKeyState(VK_CONTROL) < 0) ? PUGL_MOD_CTRL : 0; | |||||
| view->mods |= (GetKeyState(VK_MENU) < 0) ? PUGL_MOD_ALT : 0; | |||||
| view->mods |= (GetKeyState(VK_LWIN) < 0) ? PUGL_MOD_SUPER : 0; | |||||
| view->mods |= (GetKeyState(VK_RWIN) < 0) ? PUGL_MOD_SUPER : 0; | |||||
| } | |||||
| static LRESULT | |||||
| handleMessage(PuglView* view, UINT message, WPARAM wParam, LPARAM lParam) | |||||
| { | |||||
| PAINTSTRUCT ps; | |||||
| PuglKey key; | |||||
| setModifiers(view); | |||||
| switch (message) { | |||||
| case WM_CREATE: | |||||
| case WM_SHOWWINDOW: | |||||
| case WM_SIZE: | |||||
| puglReshape(view, view->width, view->height); | |||||
| break; | |||||
| case WM_PAINT: | |||||
| BeginPaint(view->impl->hwnd, &ps); | |||||
| puglDisplay(view); | |||||
| EndPaint(view->impl->hwnd, &ps); | |||||
| break; | |||||
| case WM_MOUSEMOVE: | |||||
| if (view->motionFunc) { | |||||
| view->motionFunc( | |||||
| view, GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)); | |||||
| } | |||||
| break; | |||||
| case WM_LBUTTONDOWN: | |||||
| processMouseEvent(view, 1, true, lParam); | |||||
| break; | |||||
| case WM_MBUTTONDOWN: | |||||
| processMouseEvent(view, 2, true, lParam); | |||||
| break; | |||||
| case WM_RBUTTONDOWN: | |||||
| processMouseEvent(view, 3, true, lParam); | |||||
| break; | |||||
| case WM_LBUTTONUP: | |||||
| processMouseEvent(view, 1, false, lParam); | |||||
| break; | |||||
| case WM_MBUTTONUP: | |||||
| processMouseEvent(view, 2, false, lParam); | |||||
| break; | |||||
| case WM_RBUTTONUP: | |||||
| processMouseEvent(view, 3, false, lParam); | |||||
| break; | |||||
| case WM_MOUSEWHEEL: | |||||
| if (view->scrollFunc) { | |||||
| view->scrollFunc( | |||||
| view, 0, (int16_t)HIWORD(wParam) / (float)WHEEL_DELTA); | |||||
| } | |||||
| break; | |||||
| case WM_MOUSEHWHEEL: | |||||
| if (view->scrollFunc) { | |||||
| view->scrollFunc( | |||||
| view, (int16_t)HIWORD(wParam) / float(WHEEL_DELTA), 0); | |||||
| } | |||||
| break; | |||||
| case WM_KEYDOWN: | |||||
| if (view->ignoreKeyRepeat && (lParam & (1 << 30))) { | |||||
| break; | |||||
| } // else nobreak | |||||
| case WM_KEYUP: | |||||
| if ((key = keySymToSpecial(wParam))) { | |||||
| if (view->specialFunc) { | |||||
| view->specialFunc(view, message == WM_KEYDOWN, key); | |||||
| } | |||||
| } else if (view->keyboardFunc) { | |||||
| view->keyboardFunc(view, message == WM_KEYDOWN, wParam); | |||||
| } | |||||
| break; | |||||
| case WM_QUIT: | |||||
| if (view->closeFunc) { | |||||
| view->closeFunc(view); | |||||
| } | |||||
| break; | |||||
| default: | |||||
| return DefWindowProc( | |||||
| view->impl->hwnd, message, wParam, lParam); | |||||
| } | |||||
| return 0; | |||||
| } | |||||
| PuglStatus | |||||
| puglProcessEvents(PuglView* view) | |||||
| { | |||||
| MSG msg; | |||||
| while (PeekMessage(&msg, view->impl->hwnd, 0, 0, PM_REMOVE)) { | |||||
| handleMessage(view, msg.message, msg.wParam, msg.lParam); | |||||
| } | |||||
| if (view->redisplay) { | |||||
| puglDisplay(view); | |||||
| } | |||||
| return PUGL_SUCCESS; | |||||
| } | |||||
| LRESULT CALLBACK | |||||
| wndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) | |||||
| { | |||||
| PuglView* view = (PuglView*)GetWindowLongPtr(hwnd, GWL_USERDATA); | |||||
| switch (message) { | |||||
| case WM_CREATE: | |||||
| PostMessage(hwnd, WM_SHOWWINDOW, TRUE, 0); | |||||
| return 0; | |||||
| case WM_CLOSE: | |||||
| PostQuitMessage(0); | |||||
| return 0; | |||||
| case WM_DESTROY: | |||||
| return 0; | |||||
| case WM_MOUSEWHEEL: | |||||
| case WM_MOUSEHWHEEL: | |||||
| PostMessage(hwnd, message, wParam, lParam); | |||||
| return 0; | |||||
| default: | |||||
| if (view) { | |||||
| return handleMessage(view, message, wParam, lParam); | |||||
| } else { | |||||
| return DefWindowProc(hwnd, message, wParam, lParam); | |||||
| } | |||||
| } | |||||
| } | |||||
| void | |||||
| puglPostRedisplay(PuglView* view) | |||||
| { | |||||
| view->redisplay = true; | |||||
| } | |||||
| PuglNativeWindow | |||||
| puglGetNativeWindow(PuglView* view) | |||||
| { | |||||
| return (PuglNativeWindow)view->impl->hwnd; | |||||
| } | |||||
| @@ -1,379 +0,0 @@ | |||||
| /* | |||||
| Copyright 2012 David Robillard <http://drobilla.net> | |||||
| Copyright 2011-2012 Ben Loftis, Harrison Consoles | |||||
| Permission to use, copy, modify, and/or distribute this software for any | |||||
| purpose with or without fee is hereby granted, provided that the above | |||||
| copyright notice and this permission notice appear in all copies. | |||||
| THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |||||
| WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |||||
| MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |||||
| ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |||||
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |||||
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |||||
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |||||
| */ | |||||
| /** | |||||
| @file pugl_x11.c X11 Pugl Implementation. | |||||
| */ | |||||
| #include <stdio.h> | |||||
| #include <stdlib.h> | |||||
| #include <string.h> | |||||
| #include "pugl_x11.h" | |||||
| #include "pugl_internal.h" | |||||
| /** | |||||
| Attributes for single-buffered RGBA with at least | |||||
| 4 bits per color and a 16 bit depth buffer. | |||||
| */ | |||||
| static int attrListSgl[] = { | |||||
| GLX_RGBA, | |||||
| GLX_RED_SIZE, 4, | |||||
| GLX_GREEN_SIZE, 4, | |||||
| GLX_BLUE_SIZE, 4, | |||||
| GLX_DEPTH_SIZE, 16, | |||||
| None | |||||
| }; | |||||
| /** | |||||
| Attributes for double-buffered RGBA with at least | |||||
| 4 bits per color and a 16 bit depth buffer. | |||||
| */ | |||||
| static int attrListDbl[] = { | |||||
| GLX_RGBA, GLX_DOUBLEBUFFER, | |||||
| GLX_RED_SIZE, 4, | |||||
| GLX_GREEN_SIZE, 4, | |||||
| GLX_BLUE_SIZE, 4, | |||||
| GLX_DEPTH_SIZE, 16, | |||||
| None | |||||
| }; | |||||
| PuglView* | |||||
| puglCreate(PuglNativeWindow parent, | |||||
| const char* title, | |||||
| int width, | |||||
| int height, | |||||
| bool resizable, | |||||
| bool addToDesktop, | |||||
| const char* x11Display) | |||||
| { | |||||
| PuglView* view = (PuglView*)calloc(1, sizeof(PuglView)); | |||||
| PuglInternals* impl = (PuglInternals*)calloc(1, sizeof(PuglInternals)); | |||||
| if (!view || !impl) { | |||||
| return nullptr; | |||||
| } | |||||
| view->impl = impl; | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| impl->display = XOpenDisplay(x11Display); | |||||
| impl->screen = DefaultScreen(impl->display); | |||||
| XVisualInfo* vi = glXChooseVisual(impl->display, impl->screen, attrListDbl); | |||||
| if (!vi) { | |||||
| vi = glXChooseVisual(impl->display, impl->screen, attrListSgl); | |||||
| impl->doubleBuffered = False; | |||||
| printf("singlebuffered rendering will be used, no doublebuffering available\n"); | |||||
| } else { | |||||
| impl->doubleBuffered = True; | |||||
| printf("doublebuffered rendering available\n"); | |||||
| } | |||||
| int glxMajor, glxMinor; | |||||
| glXQueryVersion(impl->display, &glxMajor, &glxMinor); | |||||
| printf("GLX-Version %d.%d\n", glxMajor, glxMinor); | |||||
| impl->ctx = glXCreateContext(impl->display, vi, nullptr, GL_TRUE); | |||||
| Window xParent = parent | |||||
| ? (Window)parent | |||||
| : RootWindow(impl->display, impl->screen); | |||||
| Colormap cmap = XCreateColormap( | |||||
| impl->display, xParent, vi->visual, AllocNone); | |||||
| XSetWindowAttributes attr; | |||||
| memset(&attr, 0, sizeof(XSetWindowAttributes)); | |||||
| attr.colormap = cmap; | |||||
| attr.border_pixel = 0; | |||||
| attr.event_mask = ExposureMask | KeyPressMask | KeyReleaseMask | |||||
| | ButtonPressMask | ButtonReleaseMask | |||||
| | PointerMotionMask | StructureNotifyMask; | |||||
| impl->win = XCreateWindow( | |||||
| impl->display, xParent, | |||||
| 0, 0, (unsigned int)view->width, (unsigned int)view->height, 0, vi->depth, InputOutput, vi->visual, | |||||
| CWBorderPixel | CWColormap | CWEventMask, &attr); | |||||
| XSizeHints sizeHints; | |||||
| memset(&sizeHints, 0, sizeof(sizeHints)); | |||||
| if (!resizable) { | |||||
| sizeHints.flags = PMinSize|PMaxSize; | |||||
| sizeHints.min_width = width; | |||||
| sizeHints.min_height = height; | |||||
| sizeHints.max_width = width; | |||||
| sizeHints.max_height = height; | |||||
| XSetNormalHints(impl->display, impl->win, &sizeHints); | |||||
| } | |||||
| if (title) { | |||||
| XStoreName(impl->display, impl->win, title); | |||||
| } | |||||
| if (!parent) { | |||||
| Atom wmDelete = XInternAtom(impl->display, "WM_DELETE_WINDOW", True); | |||||
| XSetWMProtocols(impl->display, impl->win, &wmDelete, 1); | |||||
| } | |||||
| if (addToDesktop) { | |||||
| XMapRaised(impl->display, impl->win); | |||||
| } | |||||
| if (glXIsDirect(impl->display, impl->ctx)) { | |||||
| printf("DRI enabled\n"); | |||||
| } else { | |||||
| printf("no DRI available\n"); | |||||
| } | |||||
| XFree(vi); | |||||
| return view; | |||||
| } | |||||
| void | |||||
| puglDestroy(PuglView* view) | |||||
| { | |||||
| if (!view) { | |||||
| return; | |||||
| } | |||||
| glXDestroyContext(view->impl->display, view->impl->ctx); | |||||
| XDestroyWindow(view->impl->display, view->impl->win); | |||||
| XCloseDisplay(view->impl->display); | |||||
| free(view->impl); | |||||
| free(view); | |||||
| } | |||||
| static void | |||||
| puglReshape(PuglView* view, int width, int height) | |||||
| { | |||||
| glXMakeCurrent(view->impl->display, view->impl->win, view->impl->ctx); | |||||
| if (view->reshapeFunc) { | |||||
| view->reshapeFunc(view, width, height); | |||||
| } else { | |||||
| puglDefaultReshape(view, width, height); | |||||
| } | |||||
| view->width = width; | |||||
| view->height = height; | |||||
| } | |||||
| static void | |||||
| puglDisplay(PuglView* view) | |||||
| { | |||||
| glXMakeCurrent(view->impl->display, view->impl->win, view->impl->ctx); | |||||
| glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); | |||||
| glLoadIdentity(); | |||||
| if (view->displayFunc) { | |||||
| view->displayFunc(view); | |||||
| } | |||||
| glFlush(); | |||||
| if (view->impl->doubleBuffered) { | |||||
| glXSwapBuffers(view->impl->display, view->impl->win); | |||||
| } | |||||
| view->redisplay = false; | |||||
| } | |||||
| static PuglKey | |||||
| keySymToSpecial(KeySym sym) | |||||
| { | |||||
| switch (sym) { | |||||
| case XK_F1: return PUGL_KEY_F1; | |||||
| case XK_F2: return PUGL_KEY_F2; | |||||
| case XK_F3: return PUGL_KEY_F3; | |||||
| case XK_F4: return PUGL_KEY_F4; | |||||
| case XK_F5: return PUGL_KEY_F5; | |||||
| case XK_F6: return PUGL_KEY_F6; | |||||
| case XK_F7: return PUGL_KEY_F7; | |||||
| case XK_F8: return PUGL_KEY_F8; | |||||
| case XK_F9: return PUGL_KEY_F9; | |||||
| case XK_F10: return PUGL_KEY_F10; | |||||
| case XK_F11: return PUGL_KEY_F11; | |||||
| case XK_F12: return PUGL_KEY_F12; | |||||
| case XK_Left: return PUGL_KEY_LEFT; | |||||
| case XK_Up: return PUGL_KEY_UP; | |||||
| case XK_Right: return PUGL_KEY_RIGHT; | |||||
| case XK_Down: return PUGL_KEY_DOWN; | |||||
| case XK_Page_Up: return PUGL_KEY_PAGE_UP; | |||||
| case XK_Page_Down: return PUGL_KEY_PAGE_DOWN; | |||||
| case XK_Home: return PUGL_KEY_HOME; | |||||
| case XK_End: return PUGL_KEY_END; | |||||
| case XK_Insert: return PUGL_KEY_INSERT; | |||||
| case XK_Shift_L: return PUGL_KEY_SHIFT; | |||||
| case XK_Shift_R: return PUGL_KEY_SHIFT; | |||||
| case XK_Control_L: return PUGL_KEY_CTRL; | |||||
| case XK_Control_R: return PUGL_KEY_CTRL; | |||||
| case XK_Alt_L: return PUGL_KEY_ALT; | |||||
| case XK_Alt_R: return PUGL_KEY_ALT; | |||||
| case XK_Super_L: return PUGL_KEY_SUPER; | |||||
| case XK_Super_R: return PUGL_KEY_SUPER; | |||||
| } | |||||
| return (PuglKey)0; | |||||
| } | |||||
| static void | |||||
| setModifiers(PuglView* view, int xstate) | |||||
| { | |||||
| view->mods = 0; | |||||
| view->mods |= (xstate & ShiftMask) ? PUGL_MOD_SHIFT : 0; | |||||
| view->mods |= (xstate & ControlMask) ? PUGL_MOD_CTRL : 0; | |||||
| view->mods |= (xstate & Mod1Mask) ? PUGL_MOD_ALT : 0; | |||||
| view->mods |= (xstate & Mod4Mask) ? PUGL_MOD_SUPER : 0; | |||||
| } | |||||
| PuglStatus | |||||
| puglProcessEvents(PuglView* view) | |||||
| { | |||||
| XEvent event; | |||||
| while (XPending(view->impl->display) > 0) { | |||||
| XNextEvent(view->impl->display, &event); | |||||
| switch (event.type) { | |||||
| case MapNotify: | |||||
| puglReshape(view, view->width, view->height); | |||||
| break; | |||||
| case ConfigureNotify: | |||||
| if ((event.xconfigure.width != view->width) || | |||||
| (event.xconfigure.height != view->height)) { | |||||
| puglReshape(view, | |||||
| event.xconfigure.width, | |||||
| event.xconfigure.height); | |||||
| } | |||||
| break; | |||||
| case Expose: | |||||
| if (event.xexpose.count != 0) { | |||||
| break; | |||||
| } | |||||
| puglDisplay(view); | |||||
| view->redisplay = false; | |||||
| break; | |||||
| case MotionNotify: | |||||
| setModifiers(view, (int)event.xmotion.state); | |||||
| if (view->motionFunc) { | |||||
| view->motionFunc(view, event.xmotion.x, event.xmotion.y); | |||||
| } | |||||
| break; | |||||
| case ButtonPress: | |||||
| setModifiers(view, (int)event.xbutton.state); | |||||
| if (event.xbutton.button >= 4 && event.xbutton.button <= 7) { | |||||
| if (view->scrollFunc) { | |||||
| float dx = 0, dy = 0; | |||||
| switch (event.xbutton.button) { | |||||
| case 4: dy = 1.0f; break; | |||||
| case 5: dy = -1.0f; break; | |||||
| case 6: dx = -1.0f; break; | |||||
| case 7: dx = 1.0f; break; | |||||
| } | |||||
| view->scrollFunc(view, dx, dy); | |||||
| } | |||||
| break; | |||||
| } | |||||
| // nobreak | |||||
| case ButtonRelease: | |||||
| setModifiers(view, (int)event.xbutton.state); | |||||
| if (view->mouseFunc && | |||||
| (event.xbutton.button < 4 || event.xbutton.button > 7)) { | |||||
| view->mouseFunc(view, | |||||
| (int)event.xbutton.button, event.type == ButtonPress, | |||||
| event.xbutton.x, event.xbutton.y); | |||||
| } | |||||
| break; | |||||
| case KeyPress: { | |||||
| setModifiers(view, (int)event.xkey.state); | |||||
| KeySym sym; | |||||
| char str[5]; | |||||
| int n = XLookupString(&event.xkey, str, 4, &sym, nullptr); | |||||
| PuglKey key = keySymToSpecial(sym); | |||||
| if (!key && view->keyboardFunc) { | |||||
| if (n == 1) { | |||||
| view->keyboardFunc(view, true, (uint32_t)str[0]); | |||||
| } else { | |||||
| fprintf(stderr, "warning: Unknown key %X\n", (int)sym); | |||||
| } | |||||
| } else if (view->specialFunc) { | |||||
| view->specialFunc(view, true, key); | |||||
| } | |||||
| } break; | |||||
| case KeyRelease: { | |||||
| setModifiers(view, (int)event.xkey.state); | |||||
| bool repeated = false; | |||||
| if (view->ignoreKeyRepeat && | |||||
| XEventsQueued(view->impl->display, QueuedAfterReading)) { | |||||
| XEvent next; | |||||
| XPeekEvent(view->impl->display, &next); | |||||
| if (next.type == KeyPress && | |||||
| next.xkey.time == event.xkey.time && | |||||
| next.xkey.keycode == event.xkey.keycode) { | |||||
| XNextEvent(view->impl->display, &event); | |||||
| repeated = true; | |||||
| } | |||||
| } | |||||
| if (!repeated && view->keyboardFunc) { | |||||
| KeySym sym = XKeycodeToKeysym( | |||||
| view->impl->display, (KeyCode)event.xkey.keycode, 0); | |||||
| PuglKey special = keySymToSpecial(sym); | |||||
| if (!special) { | |||||
| view->keyboardFunc(view, false, (uint32_t)sym); | |||||
| } else if (view->specialFunc) { | |||||
| view->specialFunc(view, false, special); | |||||
| } | |||||
| } | |||||
| } break; | |||||
| case ClientMessage: | |||||
| if (!strcmp(XGetAtomName(view->impl->display, | |||||
| event.xclient.message_type), | |||||
| "WM_PROTOCOLS")) { | |||||
| if (view->closeFunc) { | |||||
| view->closeFunc(view); | |||||
| } | |||||
| } | |||||
| break; | |||||
| default: | |||||
| break; | |||||
| } | |||||
| } | |||||
| if (view->redisplay) { | |||||
| puglDisplay(view); | |||||
| } | |||||
| return PUGL_SUCCESS; | |||||
| } | |||||
| void | |||||
| puglPostRedisplay(PuglView* view) | |||||
| { | |||||
| view->redisplay = true; | |||||
| } | |||||
| PuglNativeWindow | |||||
| puglGetNativeWindow(PuglView* view) | |||||
| { | |||||
| return static_cast<PuglNativeWindow>(view->impl->win); | |||||
| } | |||||
| PuglInternals* | |||||
| puglGetInternalsImpl(PuglView* view) | |||||
| { | |||||
| return view->impl; | |||||
| } | |||||
| @@ -1,46 +0,0 @@ | |||||
| /* | |||||
| Copyright 2012 David Robillard <http://drobilla.net> | |||||
| Copyright 2011-2012 Ben Loftis, Harrison Consoles | |||||
| Permission to use, copy, modify, and/or distribute this software for any | |||||
| purpose with or without fee is hereby granted, provided that the above | |||||
| copyright notice and this permission notice appear in all copies. | |||||
| THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |||||
| WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |||||
| MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |||||
| ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |||||
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |||||
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |||||
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |||||
| */ | |||||
| /** | |||||
| @file pugl_x11.h X11 Pugl Implementation (PuglInternalsImpl only). | |||||
| */ | |||||
| #ifndef PUGL_X11_H_INCLUDED | |||||
| #define PUGL_X11_H_INCLUDED | |||||
| #include "pugl.h" | |||||
| #include <GL/gl.h> | |||||
| #include <GL/glx.h> | |||||
| #include <X11/Xatom.h> | |||||
| #include <X11/Xlib.h> | |||||
| #include <X11/keysym.h> | |||||
| struct PuglInternalsImpl { | |||||
| Display* display; | |||||
| int screen; | |||||
| Window win; | |||||
| GLXContext ctx; | |||||
| Bool doubleBuffered; | |||||
| }; | |||||
| typedef struct PuglInternalsImpl PuglInternals; | |||||
| PuglInternals* | |||||
| puglGetInternalsImpl(PuglView* view); | |||||
| #endif /* PUGL_X11_H_INCLUDED */ | |||||