class PygletToolWindow_Implementation: PygletToolWindow = ObjCSubclass('NSPanel', 'PygletToolWindow') @PygletToolWindow.method(b'@' + NSUIntegerEncoding + b'@@B') def nextEventMatchingMask_untilDate_inMode_dequeue_( self, mask, date, mode, dequeue): if self.inLiveResize(): # Call the idle() method while we're stuck in a live resize event. from pyglet import app if app.event_loop is not None: app.event_loop.idle() event = send_super(self, 'nextEventMatchingMask:untilDate:inMode:dequeue:', mask, date, mode, dequeue, argtypes=[NSUInteger, c_void_p, c_void_p, c_bool]) if event.value == None: return 0 else: return event.value # Need this for set_size to not flash. @PygletToolWindow.method(b'd' + NSRectEncoding) def animationResizeTime_(self, newFrame): return 0.0
class PygletWindow_Implementation: PygletWindow = ObjCSubclass('NSWindow', 'PygletWindow') @PygletWindow.method('B') def canBecomeKeyWindow(self): return True # When the window is being resized, it enters into a mini event loop that # only looks at mouseDragged and mouseUp events, blocking everything else. # Among other things, this makes it impossible to run an NSTimer to call the # idle() function in order to update the view during the resize. So we # override this method, called by the resizing event loop, and call the # idle() function from here. This *almost* works. I can't figure out what # is happening at the very beginning of a resize event. The NSView's # viewWillStartLiveResize method is called and then nothing happens until # the mouse is dragged. I think NSApplication's nextEventMatchingMask_etc # method is being called instead of this one. I don't really feel like # subclassing NSApplication just to fix this. Also, to prevent white flashes # while resizing, we must also call idle() from the view's reshape method. @PygletWindow.method(b'@' + NSUIntegerEncoding + b'@@B') def nextEventMatchingMask_untilDate_inMode_dequeue_( self, mask, date, mode, dequeue): if self.inLiveResize(): # Call the idle() method while we're stuck in a live resize event. from pyglet import app if app.event_loop is not None: app.event_loop.idle() event = send_super(self, 'nextEventMatchingMask:untilDate:inMode:dequeue:', mask, date, mode, dequeue, superclass_name='NSWindow', argtypes=[NSUInteger, c_void_p, c_void_p, c_bool]) if event.value is None: return 0 else: return event.value # Need this for set_size to not flash. @PygletWindow.method(b'd' + NSRectEncoding) def animationResizeTime_(self, newFrame): return 0.0
class PygletTextView_Implementation: PygletTextView = ObjCSubclass('NSTextView', 'PygletTextView') @PygletTextView.method(b'@' + PyObjectEncoding) def initWithCocoaWindow_(self, window): self = ObjCInstance(send_super(self, 'init')) if not self: return None self._window = window # Interpret tab and return as raw characters self.setFieldEditor_(False) self.empty_string = CFSTR("") return self @PygletTextView.method('v') def dealloc(self): self.empty_string.release() @PygletTextView.method('v@') def keyDown_(self, nsevent): array = NSArray.arrayWithObject_(nsevent) self.interpretKeyEvents_(array) @PygletTextView.method('v@') def insertText_(self, text): text = cfstring_to_string(text) self.setString_(self.empty_string) # Don't send control characters (tab, newline) as on_text events. if unicodedata.category(text[0]) != 'Cc': self._window.dispatch_event("on_text", text) @PygletTextView.method('v@') def insertNewline_(self, sender): # Distinguish between carriage return (u'\r') and enter (u'\x03'). # Only the return key press gets sent as an on_text event. event = NSApplication.sharedApplication().currentEvent() chars = event.charactersIgnoringModifiers() ch = chr(chars.characterAtIndex_(0)) if ch == u'\r': self._window.dispatch_event("on_text", u'\r') @PygletTextView.method('v@') def moveUp_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_UP) @PygletTextView.method('v@') def moveDown_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_DOWN) @PygletTextView.method('v@') def moveLeft_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_LEFT) @PygletTextView.method('v@') def moveRight_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_RIGHT) @PygletTextView.method('v@') def moveWordLeft_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_PREVIOUS_WORD) @PygletTextView.method('v@') def moveWordRight_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_NEXT_WORD) @PygletTextView.method('v@') def moveToBeginningOfLine_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_BEGINNING_OF_LINE) @PygletTextView.method('v@') def moveToEndOfLine_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_END_OF_LINE) @PygletTextView.method('v@') def scrollPageUp_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_PREVIOUS_PAGE) @PygletTextView.method('v@') def scrollPageDown_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_NEXT_PAGE) @PygletTextView.method('v@') def scrollToBeginningOfDocument_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion", key.MOTION_BEGINNING_OF_FILE) @PygletTextView.method('v@') def scrollToEndOfDocument_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion", key.MOTION_END_OF_FILE) @PygletTextView.method('v@') def deleteBackward_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_BACKSPACE) @PygletTextView.method('v@') def deleteForward_(self, sender): self._window.dispatch_event("on_text_motion", key.MOTION_DELETE) @PygletTextView.method('v@') def moveUpAndModifySelection_(self, sender): self._window.dispatch_event("on_text_motion_select", key.MOTION_UP) @PygletTextView.method('v@') def moveDownAndModifySelection_(self, sender): self._window.dispatch_event("on_text_motion_select", key.MOTION_DOWN) @PygletTextView.method('v@') def moveLeftAndModifySelection_(self, sender): self._window.dispatch_event("on_text_motion_select", key.MOTION_LEFT) @PygletTextView.method('v@') def moveRightAndModifySelection_(self, sender): self._window.dispatch_event("on_text_motion_select", key.MOTION_RIGHT) @PygletTextView.method('v@') def moveWordLeftAndModifySelection_(self, sender): self._window.dispatch_event("on_text_motion_select", key.MOTION_PREVIOUS_WORD) @PygletTextView.method('v@') def moveWordRightAndModifySelection_(self, sender): self._window.dispatch_event("on_text_motion_select", key.MOTION_NEXT_WORD) @PygletTextView.method('v@') def moveToBeginningOfLineAndModifySelection_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion_select", key.MOTION_BEGINNING_OF_LINE) @PygletTextView.method('v@') def moveToEndOfLineAndModifySelection_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion_select", key.MOTION_END_OF_LINE) @PygletTextView.method('v@') def pageUpAndModifySelection_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion_select", key.MOTION_PREVIOUS_PAGE) @PygletTextView.method('v@') def pageDownAndModifySelection_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion_select", key.MOTION_NEXT_PAGE) @PygletTextView.method('v@') def moveToBeginningOfDocumentAndModifySelection_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion_select", key.MOTION_BEGINNING_OF_FILE) @PygletTextView.method('v@') def moveToEndOfDocumentAndModifySelection_(self, sender): # Mac OS X 10.6 self._window.dispatch_event("on_text_motion_select", key.MOTION_END_OF_FILE)
class PygletDelegate_Implementation: PygletDelegate = ObjCSubclass('NSObject', 'PygletDelegate') @PygletDelegate.method(b'@' + PyObjectEncoding) def initWithWindow_(self, window): self = ObjCInstance(send_super(self, 'init')) if not self: return None # CocoaWindow object. self._window = window window._nswindow.setDelegate_(self) # Register delegate for hide and unhide notifications so that we # can dispatch the corresponding pyglet events. notificationCenter = NSNotificationCenter.defaultCenter() notificationCenter.addObserver_selector_name_object_( self, get_selector('applicationDidHide:'), NSApplicationDidHideNotification, None) notificationCenter.addObserver_selector_name_object_( self, get_selector('applicationDidUnhide:'), NSApplicationDidUnhideNotification, None) # Flag set when we pause exclusive mouse mode if window loses key status. self.did_pause_exclusive_mouse = False return self @PygletDelegate.method('v') def dealloc(self): # Unregister delegate from notification center. notificationCenter = NSNotificationCenter.defaultCenter() notificationCenter.removeObserver_(self) self._window = None send_super(self, 'dealloc') @PygletDelegate.method('v@') def applicationDidHide_(self, notification): self._window.dispatch_event("on_hide") @PygletDelegate.method('v@') def applicationDidUnhide_(self, notification): if self._window._is_mouse_exclusive and quartz.CGCursorIsVisible(): # The cursor should be hidden, but for some reason it's not; # try to force the cursor to hide (without over-hiding). SystemCursor.unhide() SystemCursor.hide() pass self._window.dispatch_event("on_show") @PygletDelegate.method('B@') def windowShouldClose_(self, notification): # The method is not called if [NSWindow close] was used. self._window.dispatch_event("on_close") return False @PygletDelegate.method('v@') def windowDidMove_(self, notification): x, y = self._window.get_location() self._window.dispatch_event("on_move", x, y) @PygletDelegate.method('v@') def windowDidBecomeKey_(self, notification): # Restore exclusive mouse mode if it was active before we lost key status. if self.did_pause_exclusive_mouse: self._window.set_exclusive_mouse(True) self.did_pause_exclusive_mouse = False self._window._nswindow.setMovable_(True) # Mac OS 10.6 # Restore previous mouse visibility settings. self._window.set_mouse_platform_visible() self._window.dispatch_event("on_activate") @PygletDelegate.method('v@') def windowDidResignKey_(self, notification): # Pause exclusive mouse mode if it is active. if self._window._is_mouse_exclusive: self._window.set_exclusive_mouse(False) self.did_pause_exclusive_mouse = True # We need to prevent the window from being unintentionally dragged # (by the call to set_mouse_position in set_exclusive_mouse) when # the window is reactivated by clicking on its title bar. self._window._nswindow.setMovable_(False) # Mac OS X 10.6 # Make sure that cursor is visible. self._window.set_mouse_platform_visible(True) self._window.dispatch_event("on_deactivate") @PygletDelegate.method('v@') def windowDidMiniaturize_(self, notification): self._window.dispatch_event("on_hide") @PygletDelegate.method('v@') def windowDidDeminiaturize_(self, notification): if self._window._is_mouse_exclusive and quartz.CGCursorIsVisible(): # The cursor should be hidden, but for some reason it's not; # try to force the cursor to hide (without over-hiding). SystemCursor.unhide() SystemCursor.hide() pass self._window.dispatch_event("on_show") @PygletDelegate.method('v@') def windowDidExpose_(self, notification): self._window.dispatch_event("on_expose") @PygletDelegate.method('v@') def terminate_(self, sender): NSApp = NSApplication.sharedApplication() NSApp.terminate_(self) @PygletDelegate.method('B@') def validateMenuItem_(self, menuitem): # Disable quitting with command-q when in keyboard exclusive mode. if menuitem.action() == get_selector('terminate:'): return not self._window._is_keyboard_exclusive return True
from pyglet.libs.darwin.cocoapy import ObjCClass, get_NSString, objc, ObjCSubclass NSUserNotificationCenter = ObjCClass("NSUserNotificationCenter") NSUserNotification = ObjCClass("NSUserNotification") NSApplication = ObjCClass("NSApplication") NSUserNotificationCenterDelegate = objc.objc_getProtocol( bytes("NSUserNotificationCenterDelegate")) NSApp = NSApplication.sharedApplication() NotificationDelegateClass = ObjCSubclass('NSObject', 'NotificationDelegate', register=False) @NotificationDelegateClass.method("v@@") def userNotificationCenter_didActivateNotification_(self, notificationCenter, notification): # bring window to front NSApp.activateIgnoringOtherApps_(True) objc.class_addProtocol(NotificationDelegateClass.objc_cls, NSUserNotificationCenterDelegate) NotificationDelegateClass.register() NotificationDelegate = ObjCClass("NotificationDelegate") notificationCenter = NSUserNotificationCenter.defaultUserNotificationCenter() notificationCenter.setDelegate_(NotificationDelegate.alloc().init())
from pyglet.libs.darwin.cocoapy import ObjCClass, get_NSString, objc, ObjCSubclass NSUserNotificationCenter = ObjCClass("NSUserNotificationCenter") NSUserNotification = ObjCClass("NSUserNotification") NSApplication = ObjCClass("NSApplication") NSUserNotificationCenterDelegate = objc.objc_getProtocol(bytes("NSUserNotificationCenterDelegate")) NSApp = NSApplication.sharedApplication() NotificationDelegateClass = ObjCSubclass('NSObject', 'NotificationDelegate', register=False) @NotificationDelegateClass.method("v@@") def userNotificationCenter_didActivateNotification_(self, notificationCenter, notification): # bring window to front NSApp.activateIgnoringOtherApps_(True) objc.class_addProtocol(NotificationDelegateClass.objc_cls, NSUserNotificationCenterDelegate) NotificationDelegateClass.register() NotificationDelegate = ObjCClass("NotificationDelegate") notificationCenter = NSUserNotificationCenter.defaultUserNotificationCenter() notificationCenter.setDelegate_(NotificationDelegate.alloc().init()) def _notify(title, msg): notification = NSUserNotification.alloc().init().autorelease() notification.setTitle_(get_NSString(title)) notification.setInformativeText_(get_NSString(msg)) notificationCenter.deliverNotification_(notification)