1
0
Fork 0
mirror of https://github.com/juce-framework/JUCE.git synced 2026-01-10 23:44:24 +00:00
JUCE/modules/juce_gui_basics/native/juce_mac_NSViewComponentPeer.mm
2011-08-13 21:13:50 +01:00

1861 lines
60 KiB
Text

/*
==============================================================================
This file is part of the JUCE library - "Jules' Utility Class Extensions"
Copyright 2004-11 by Raw Material Software Ltd.
------------------------------------------------------------------------------
JUCE can be redistributed and/or modified under the terms of the GNU General
Public License (Version 2), as published by the Free Software Foundation.
A copy of the license is included in the JUCE distribution, or can be found
online at www.gnu.org/licenses.
JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
------------------------------------------------------------------------------
To release a closed-source product which uses JUCE, commercial licenses are
available: visit www.rawmaterialsoftware.com/juce for more information.
==============================================================================
*/
class NSViewComponentPeer;
typedef void (*AppFocusChangeCallback)();
extern AppFocusChangeCallback appFocusChangeCallback;
typedef bool (*CheckEventBlockedByModalComps) (NSEvent*);
extern CheckEventBlockedByModalComps isEventBlockedByModalComps;
//==============================================================================
END_JUCE_NAMESPACE
@interface NSEvent (JuceDeviceDelta)
- (float) deviceDeltaX;
- (float) deviceDeltaY;
@end
#define JuceNSView MakeObjCClassName(JuceNSView)
@interface JuceNSView : NSView<NSTextInput>
{
@public
NSViewComponentPeer* owner;
NSNotificationCenter* notificationCenter;
String* stringBeingComposed;
bool textWasInserted;
}
- (JuceNSView*) initWithOwner: (NSViewComponentPeer*) owner withFrame: (NSRect) frame;
- (void) dealloc;
- (BOOL) isOpaque;
- (void) drawRect: (NSRect) r;
- (void) mouseDown: (NSEvent*) ev;
- (void) asyncMouseDown: (NSEvent*) ev;
- (void) mouseUp: (NSEvent*) ev;
- (void) asyncMouseUp: (NSEvent*) ev;
- (void) mouseDragged: (NSEvent*) ev;
- (void) mouseMoved: (NSEvent*) ev;
- (void) mouseEntered: (NSEvent*) ev;
- (void) mouseExited: (NSEvent*) ev;
- (void) rightMouseDown: (NSEvent*) ev;
- (void) rightMouseDragged: (NSEvent*) ev;
- (void) rightMouseUp: (NSEvent*) ev;
- (void) otherMouseDown: (NSEvent*) ev;
- (void) otherMouseDragged: (NSEvent*) ev;
- (void) otherMouseUp: (NSEvent*) ev;
- (void) scrollWheel: (NSEvent*) ev;
- (BOOL) acceptsFirstMouse: (NSEvent*) ev;
- (void) frameChanged: (NSNotification*) n;
- (void) viewDidMoveToWindow;
- (void) keyDown: (NSEvent*) ev;
- (void) keyUp: (NSEvent*) ev;
// NSTextInput Methods
- (void) insertText: (id) aString;
- (void) doCommandBySelector: (SEL) aSelector;
- (void) setMarkedText: (id) aString selectedRange: (NSRange) selRange;
- (void) unmarkText;
- (BOOL) hasMarkedText;
- (long) conversationIdentifier;
- (NSAttributedString*) attributedSubstringFromRange: (NSRange) theRange;
- (NSRange) markedRange;
- (NSRange) selectedRange;
- (NSRect) firstRectForCharacterRange: (NSRange) theRange;
- (NSUInteger) characterIndexForPoint: (NSPoint) thePoint;
- (NSArray*) validAttributesForMarkedText;
- (void) flagsChanged: (NSEvent*) ev;
#if MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_X_VERSION_10_5
- (BOOL) performKeyEquivalent: (NSEvent*) ev;
#endif
- (BOOL) becomeFirstResponder;
- (BOOL) resignFirstResponder;
- (BOOL) acceptsFirstResponder;
- (void) asyncRepaint: (id) rect;
- (NSArray*) getSupportedDragTypes;
- (BOOL) sendDragCallback: (int) type sender: (id <NSDraggingInfo>) sender;
- (NSDragOperation) draggingEntered: (id <NSDraggingInfo>) sender;
- (NSDragOperation) draggingUpdated: (id <NSDraggingInfo>) sender;
- (void) draggingEnded: (id <NSDraggingInfo>) sender;
- (void) draggingExited: (id <NSDraggingInfo>) sender;
- (BOOL) prepareForDragOperation: (id <NSDraggingInfo>) sender;
- (BOOL) performDragOperation: (id <NSDraggingInfo>) sender;
- (void) concludeDragOperation: (id <NSDraggingInfo>) sender;
@end
//==============================================================================
#define JuceNSWindow MakeObjCClassName(JuceNSWindow)
#if defined (MAC_OS_X_VERSION_10_6) && MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_6
@interface JuceNSWindow : NSWindow <NSWindowDelegate>
#else
@interface JuceNSWindow : NSWindow
#endif
{
@private
NSViewComponentPeer* owner;
bool isZooming;
}
- (void) setOwner: (NSViewComponentPeer*) owner;
- (BOOL) canBecomeKeyWindow;
- (void) becomeKeyWindow;
- (BOOL) windowShouldClose: (id) window;
- (NSRect) constrainFrameRect: (NSRect) frameRect toScreen: (NSScreen*) screen;
- (NSSize) windowWillResize: (NSWindow*) window toSize: (NSSize) proposedFrameSize;
- (void) zoom: (id) sender;
@end
BEGIN_JUCE_NAMESPACE
//==============================================================================
class NSViewComponentPeer : public ComponentPeer
{
public:
NSViewComponentPeer (Component* const component,
const int windowStyleFlags,
NSView* viewToAttachTo);
~NSViewComponentPeer();
//==============================================================================
void* getNativeHandle() const;
void setVisible (bool shouldBeVisible);
void setTitle (const String& title);
void setPosition (int x, int y);
void setSize (int w, int h);
void setBounds (int x, int y, int w, int h, const bool isNowFullScreen);
const Rectangle<int> getBounds (const bool global) const;
const Rectangle<int> getBounds() const;
const Point<int> getScreenPosition() const;
const Point<int> localToGlobal (const Point<int>& relativePosition);
const Point<int> globalToLocal (const Point<int>& screenPosition);
void setAlpha (float newAlpha);
void setMinimised (bool shouldBeMinimised);
bool isMinimised() const;
void setFullScreen (bool shouldBeFullScreen);
bool isFullScreen() const;
void updateFullscreenStatus();
bool contains (const Point<int>& position, bool trueIfInAChildWindow) const;
bool hasNativeTitleBar() const { return (getStyleFlags() & windowHasTitleBar) != 0; }
const BorderSize<int> getFrameSize() const;
bool setAlwaysOnTop (bool alwaysOnTop);
void toFront (bool makeActiveWindow);
void toBehind (ComponentPeer* other);
void setIcon (const Image& newIcon);
StringArray getAvailableRenderingEngines();
int getCurrentRenderingEngine() const;
void setCurrentRenderingEngine (int index);
/* When you use multiple DLLs which share similarly-named obj-c classes - like
for example having more than one juce plugin loaded into a host, then when a
method is called, the actual code that runs might actually be in a different module
than the one you expect... So any calls to library functions or statics that are
made inside obj-c methods will probably end up getting executed in a different DLL's
memory space. Not a great thing to happen - this obviously leads to bizarre crashes.
To work around this insanity, I'm only allowing obj-c methods to make calls to
virtual methods of an object that's known to live inside the right module's space.
*/
virtual void redirectMouseDown (NSEvent* ev);
virtual void redirectMouseUp (NSEvent* ev);
virtual void redirectMouseDrag (NSEvent* ev);
virtual void redirectMouseMove (NSEvent* ev);
virtual void redirectMouseEnter (NSEvent* ev);
virtual void redirectMouseExit (NSEvent* ev);
virtual void redirectMouseWheel (NSEvent* ev);
void sendMouseEvent (NSEvent* ev);
bool handleKeyEvent (NSEvent* ev, bool isKeyDown);
virtual bool redirectKeyDown (NSEvent* ev);
virtual bool redirectKeyUp (NSEvent* ev);
virtual void redirectModKeyChange (NSEvent* ev);
#if MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_X_VERSION_10_5
virtual bool redirectPerformKeyEquivalent (NSEvent* ev);
#endif
virtual BOOL sendDragCallback (int type, id <NSDraggingInfo> sender);
virtual bool isOpaque();
virtual void drawRect (NSRect r);
virtual bool canBecomeKeyWindow();
virtual void becomeKeyWindow();
virtual bool windowShouldClose();
virtual void redirectMovedOrResized();
virtual void viewMovedToWindow();
virtual NSRect constrainRect (NSRect r);
static void showArrowCursorIfNeeded();
static void updateModifiers (NSEvent* e);
static void updateKeysDown (NSEvent* ev, bool isKeyDown);
static int getKeyCodeFromEvent (NSEvent* ev)
{
const String unmodified (nsStringToJuce ([ev charactersIgnoringModifiers]));
int keyCode = unmodified[0];
if (keyCode == 0x19) // (backwards-tab)
keyCode = '\t';
else if (keyCode == 0x03) // (enter)
keyCode = '\r';
else
keyCode = (int) CharacterFunctions::toUpperCase ((juce_wchar) keyCode);
if (([ev modifierFlags] & NSNumericPadKeyMask) != 0)
{
const int numPadConversions[] = { '0', KeyPress::numberPad0, '1', KeyPress::numberPad1,
'2', KeyPress::numberPad2, '3', KeyPress::numberPad3,
'4', KeyPress::numberPad4, '5', KeyPress::numberPad5,
'6', KeyPress::numberPad6, '7', KeyPress::numberPad7,
'8', KeyPress::numberPad8, '9', KeyPress::numberPad9,
'+', KeyPress::numberPadAdd, '-', KeyPress::numberPadSubtract,
'*', KeyPress::numberPadMultiply, '/', KeyPress::numberPadDivide,
'.', KeyPress::numberPadDecimalPoint, '=', KeyPress::numberPadEquals };
for (int i = 0; i < numElementsInArray (numPadConversions); i += 2)
if (keyCode == numPadConversions [i])
keyCode = numPadConversions [i + 1];
}
return keyCode;
}
static int64 getMouseTime (NSEvent* e)
{
return (Time::currentTimeMillis() - Time::getMillisecondCounter())
+ (int64) ([e timestamp] * 1000.0);
}
static const Point<int> getMousePos (NSEvent* e, NSView* view)
{
NSPoint p = [view convertPoint: [e locationInWindow] fromView: nil];
return Point<int> (roundToInt (p.x), roundToInt ([view frame].size.height - p.y));
}
static int getModifierForButtonNumber (const NSInteger num)
{
return num == 0 ? ModifierKeys::leftButtonModifier
: (num == 1 ? ModifierKeys::rightButtonModifier
: (num == 2 ? ModifierKeys::middleButtonModifier : 0));
}
static unsigned int getNSWindowStyleMask (const int flags) noexcept
{
unsigned int style = (flags & windowHasTitleBar) != 0 ? NSTitledWindowMask
: NSBorderlessWindowMask;
if ((flags & windowHasMinimiseButton) != 0) style |= NSMiniaturizableWindowMask;
if ((flags & windowHasCloseButton) != 0) style |= NSClosableWindowMask;
if ((flags & windowIsResizable) != 0) style |= NSResizableWindowMask;
return style;
}
//==============================================================================
virtual void viewFocusGain();
virtual void viewFocusLoss();
bool isFocused() const;
void grabFocus();
void textInputRequired (const Point<int>& position);
//==============================================================================
void repaint (const Rectangle<int>& area);
void performAnyPendingRepaintsNow();
//==============================================================================
NSWindow* window;
JuceNSView* view;
bool isSharedWindow, fullScreen, insideDrawRect, usingCoreGraphics, recursiveToFrontCall;
static ModifierKeys currentModifiers;
static ComponentPeer* currentlyFocusedPeer;
static Array<int> keysCurrentlyDown;
private:
static void appFocusChanged()
{
keysCurrentlyDown.clear();
if (isValidPeer (currentlyFocusedPeer))
{
if (Process::isForegroundProcess())
{
currentlyFocusedPeer->handleFocusGain();
ModalComponentManager::getInstance()->bringModalComponentsToFront();
}
else
{
currentlyFocusedPeer->handleFocusLoss();
// turn kiosk mode off if we lose focus..
Desktop::getInstance().setKioskModeComponent (nullptr);
}
}
}
static bool checkEventBlockedByModalComps (NSEvent* e)
{
if (Component::getNumCurrentlyModalComponents() == 0)
return false;
NSWindow* const w = [e window];
if (w == nil || [w worksWhenModal])
return false;
bool isKey = false, isInputAttempt = false;
switch ([e type])
{
case NSKeyDown:
case NSKeyUp:
isKey = isInputAttempt = true;
break;
case NSLeftMouseDown:
case NSRightMouseDown:
case NSOtherMouseDown:
isInputAttempt = true;
break;
case NSLeftMouseDragged:
case NSRightMouseDragged:
case NSLeftMouseUp:
case NSRightMouseUp:
case NSOtherMouseUp:
case NSOtherMouseDragged:
if (Desktop::getInstance().getDraggingMouseSource(0) != nullptr)
return false;
break;
case NSMouseMoved:
case NSMouseEntered:
case NSMouseExited:
case NSCursorUpdate:
case NSScrollWheel:
case NSTabletPoint:
case NSTabletProximity:
break;
default:
return false;
}
for (int i = ComponentPeer::getNumPeers(); --i >= 0;)
{
ComponentPeer* const peer = ComponentPeer::getPeer (i);
NSView* const compView = (NSView*) peer->getNativeHandle();
if ([compView window] == w)
{
if (isKey)
{
if (compView == [w firstResponder])
return false;
}
else
{
NSViewComponentPeer* nsViewPeer = dynamic_cast<NSViewComponentPeer*> (peer);
if ((nsViewPeer == nullptr || ! nsViewPeer->isSharedWindow)
? NSPointInRect ([e locationInWindow], NSMakeRect (0, 0, [w frame].size.width, [w frame].size.height))
: NSPointInRect ([compView convertPoint: [e locationInWindow] fromView: nil], [compView bounds]))
return false;
}
}
}
if (isInputAttempt)
{
if (! [NSApp isActive])
[NSApp activateIgnoringOtherApps: YES];
Component* const modal = Component::getCurrentlyModalComponent (0);
if (modal != nullptr)
modal->inputAttemptWhenModal();
}
return true;
}
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (NSViewComponentPeer);
};
//==============================================================================
END_JUCE_NAMESPACE
@implementation JuceNSView
- (JuceNSView*) initWithOwner: (NSViewComponentPeer*) owner_
withFrame: (NSRect) frame
{
[super initWithFrame: frame];
owner = owner_;
stringBeingComposed = nullptr;
textWasInserted = false;
notificationCenter = [NSNotificationCenter defaultCenter];
[notificationCenter addObserver: self
selector: @selector (frameChanged:)
name: NSViewFrameDidChangeNotification
object: self];
if (! owner_->isSharedWindow)
{
[notificationCenter addObserver: self
selector: @selector (frameChanged:)
name: NSWindowDidMoveNotification
object: owner_->window];
}
[self registerForDraggedTypes: [self getSupportedDragTypes]];
return self;
}
- (void) dealloc
{
[notificationCenter removeObserver: self];
delete stringBeingComposed;
[super dealloc];
}
//==============================================================================
- (void) drawRect: (NSRect) r
{
if (owner != nullptr)
owner->drawRect (r);
}
- (BOOL) isOpaque
{
return owner == nullptr || owner->isOpaque();
}
//==============================================================================
- (void) mouseDown: (NSEvent*) ev
{
if (JUCEApplication::isStandaloneApp())
[self asyncMouseDown: ev];
else
// In some host situations, the host will stop modal loops from working
// correctly if they're called from a mouse event, so we'll trigger
// the event asynchronously..
[self performSelectorOnMainThread: @selector (asyncMouseDown:)
withObject: ev
waitUntilDone: NO];
}
- (void) asyncMouseDown: (NSEvent*) ev
{
if (owner != nullptr)
owner->redirectMouseDown (ev);
}
- (void) mouseUp: (NSEvent*) ev
{
if (! JUCEApplication::isStandaloneApp())
[self asyncMouseUp: ev];
else
// In some host situations, the host will stop modal loops from working
// correctly if they're called from a mouse event, so we'll trigger
// the event asynchronously..
[self performSelectorOnMainThread: @selector (asyncMouseUp:)
withObject: ev
waitUntilDone: NO];
}
- (void) asyncMouseUp: (NSEvent*) ev { if (owner != nullptr) owner->redirectMouseUp (ev); }
- (void) mouseDragged: (NSEvent*) ev { if (owner != nullptr) owner->redirectMouseDrag (ev); }
- (void) mouseMoved: (NSEvent*) ev { if (owner != nullptr) owner->redirectMouseMove (ev); }
- (void) mouseEntered: (NSEvent*) ev { if (owner != nullptr) owner->redirectMouseEnter (ev); }
- (void) mouseExited: (NSEvent*) ev { if (owner != nullptr) owner->redirectMouseExit (ev); }
- (void) scrollWheel: (NSEvent*) ev { if (owner != nullptr) owner->redirectMouseWheel (ev); }
- (void) rightMouseDown: (NSEvent*) ev { [self mouseDown: ev]; }
- (void) rightMouseDragged: (NSEvent*) ev { [self mouseDragged: ev]; }
- (void) rightMouseUp: (NSEvent*) ev { [self mouseUp: ev]; }
- (void) otherMouseDown: (NSEvent*) ev { [self mouseDown: ev]; }
- (void) otherMouseDragged: (NSEvent*) ev { [self mouseDragged: ev]; }
- (void) otherMouseUp: (NSEvent*) ev { [self mouseUp: ev]; }
- (BOOL) acceptsFirstMouse: (NSEvent*) ev
{
(void) ev;
return YES;
}
- (void) frameChanged: (NSNotification*) n
{
(void) n;
if (owner != nullptr)
owner->redirectMovedOrResized();
}
- (void) viewDidMoveToWindow
{
if (owner != nullptr)
owner->viewMovedToWindow();
}
- (void) asyncRepaint: (id) rect
{
NSRect* r = (NSRect*) [((NSData*) rect) bytes];
[self setNeedsDisplayInRect: *r];
}
//==============================================================================
- (void) keyDown: (NSEvent*) ev
{
TextInputTarget* const target = owner->findCurrentTextInputTarget();
textWasInserted = false;
if (target != nullptr)
[self interpretKeyEvents: [NSArray arrayWithObject: ev]];
else
deleteAndZero (stringBeingComposed);
if ((! textWasInserted) && (owner == nullptr || ! owner->redirectKeyDown (ev)))
[super keyDown: ev];
}
- (void) keyUp: (NSEvent*) ev
{
if (owner == nullptr || ! owner->redirectKeyUp (ev))
[super keyUp: ev];
}
//==============================================================================
- (void) insertText: (id) aString
{
// This commits multi-byte text when return is pressed, or after every keypress for western keyboards
NSString* newText = [aString isKindOfClass: [NSAttributedString class]] ? [aString string] : aString;
if ([newText length] > 0)
{
TextInputTarget* const target = owner->findCurrentTextInputTarget();
if (target != nullptr)
{
target->insertTextAtCaret (nsStringToJuce (newText));
textWasInserted = true;
}
}
deleteAndZero (stringBeingComposed);
}
- (void) doCommandBySelector: (SEL) aSelector
{
(void) aSelector;
}
- (void) setMarkedText: (id) aString selectedRange: (NSRange) selectionRange
{
(void) selectionRange;
if (stringBeingComposed == 0)
stringBeingComposed = new String();
*stringBeingComposed = nsStringToJuce ([aString isKindOfClass:[NSAttributedString class]] ? [aString string] : aString);
TextInputTarget* const target = owner->findCurrentTextInputTarget();
if (target != nullptr)
{
const Range<int> currentHighlight (target->getHighlightedRegion());
target->insertTextAtCaret (*stringBeingComposed);
target->setHighlightedRegion (currentHighlight.withLength (stringBeingComposed->length()));
textWasInserted = true;
}
}
- (void) unmarkText
{
if (stringBeingComposed != nullptr)
{
TextInputTarget* const target = owner->findCurrentTextInputTarget();
if (target != nullptr)
{
target->insertTextAtCaret (*stringBeingComposed);
textWasInserted = true;
}
}
deleteAndZero (stringBeingComposed);
}
- (BOOL) hasMarkedText
{
return stringBeingComposed != nullptr;
}
- (long) conversationIdentifier
{
return (long) (pointer_sized_int) self;
}
- (NSAttributedString*) attributedSubstringFromRange: (NSRange) theRange
{
TextInputTarget* const target = owner->findCurrentTextInputTarget();
if (target != nullptr)
{
const Range<int> r ((int) theRange.location,
(int) (theRange.location + theRange.length));
return [[[NSAttributedString alloc] initWithString: juceStringToNS (target->getTextInRange (r))] autorelease];
}
return nil;
}
- (NSRange) markedRange
{
return stringBeingComposed != nullptr ? NSMakeRange (0, stringBeingComposed->length())
: NSMakeRange (NSNotFound, 0);
}
- (NSRange) selectedRange
{
TextInputTarget* const target = owner->findCurrentTextInputTarget();
if (target != nullptr)
{
const Range<int> highlight (target->getHighlightedRegion());
if (! highlight.isEmpty())
return NSMakeRange (highlight.getStart(), highlight.getLength());
}
return NSMakeRange (NSNotFound, 0);
}
- (NSRect) firstRectForCharacterRange: (NSRange) theRange
{
(void) theRange;
juce::Component* const comp = dynamic_cast <juce::Component*> (owner->findCurrentTextInputTarget());
if (comp == 0)
return NSMakeRect (0, 0, 0, 0);
const Rectangle<int> bounds (comp->getScreenBounds());
return NSMakeRect (bounds.getX(),
[[[NSScreen screens] objectAtIndex: 0] frame].size.height - bounds.getY(),
bounds.getWidth(),
bounds.getHeight());
}
- (NSUInteger) characterIndexForPoint: (NSPoint) thePoint
{
(void) thePoint;
return NSNotFound;
}
- (NSArray*) validAttributesForMarkedText
{
return [NSArray array];
}
//==============================================================================
- (void) flagsChanged: (NSEvent*) ev
{
if (owner != nullptr)
owner->redirectModKeyChange (ev);
}
#if MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_X_VERSION_10_5
- (BOOL) performKeyEquivalent: (NSEvent*) ev
{
if (owner != nullptr && owner->redirectPerformKeyEquivalent (ev))
return true;
return [super performKeyEquivalent: ev];
}
#endif
- (BOOL) becomeFirstResponder { if (owner != nullptr) owner->viewFocusGain(); return YES; }
- (BOOL) resignFirstResponder { if (owner != nullptr) owner->viewFocusLoss(); return YES; }
- (BOOL) acceptsFirstResponder { return owner != nullptr && owner->canBecomeKeyWindow(); }
//==============================================================================
- (NSArray*) getSupportedDragTypes
{
return [NSArray arrayWithObjects: NSFilenamesPboardType, NSFilesPromisePboardType, /* NSStringPboardType,*/ nil];
}
- (BOOL) sendDragCallback: (int) type sender: (id <NSDraggingInfo>) sender
{
return owner != nullptr && owner->sendDragCallback (type, sender);
}
- (NSDragOperation) draggingEntered: (id <NSDraggingInfo>) sender
{
if ([self sendDragCallback: 0 sender: sender])
return NSDragOperationCopy | NSDragOperationMove | NSDragOperationGeneric;
else
return NSDragOperationNone;
}
- (NSDragOperation) draggingUpdated: (id <NSDraggingInfo>) sender
{
if ([self sendDragCallback: 0 sender: sender])
return NSDragOperationCopy | NSDragOperationMove | NSDragOperationGeneric;
else
return NSDragOperationNone;
}
- (void) draggingEnded: (id <NSDraggingInfo>) sender
{
[self sendDragCallback: 1 sender: sender];
}
- (void) draggingExited: (id <NSDraggingInfo>) sender
{
[self sendDragCallback: 1 sender: sender];
}
- (BOOL) prepareForDragOperation: (id <NSDraggingInfo>) sender
{
(void) sender;
return YES;
}
- (BOOL) performDragOperation: (id <NSDraggingInfo>) sender
{
return [self sendDragCallback: 2 sender: sender];
}
- (void) concludeDragOperation: (id <NSDraggingInfo>) sender
{
(void) sender;
}
@end
//==============================================================================
@implementation JuceNSWindow
- (void) setOwner: (NSViewComponentPeer*) owner_
{
owner = owner_;
isZooming = false;
}
- (BOOL) canBecomeKeyWindow
{
return owner != nullptr && owner->canBecomeKeyWindow();
}
- (void) becomeKeyWindow
{
[super becomeKeyWindow];
if (owner != nullptr)
owner->becomeKeyWindow();
}
- (BOOL) windowShouldClose: (id) window
{
(void) window;
return owner == nullptr || owner->windowShouldClose();
}
- (NSRect) constrainFrameRect: (NSRect) frameRect toScreen: (NSScreen*) screen
{
(void) screen;
if (owner != nullptr)
frameRect = owner->constrainRect (frameRect);
return frameRect;
}
- (NSSize) windowWillResize: (NSWindow*) window toSize: (NSSize) proposedFrameSize
{
(void) window;
if (isZooming)
return proposedFrameSize;
NSRect frameRect = [self frame];
frameRect.origin.y -= proposedFrameSize.height - frameRect.size.height;
frameRect.size = proposedFrameSize;
if (owner != nullptr)
frameRect = owner->constrainRect (frameRect);
if (juce::Component::getCurrentlyModalComponent() != nullptr
&& owner->getComponent()->isCurrentlyBlockedByAnotherModalComponent()
&& owner->hasNativeTitleBar())
juce::Component::getCurrentlyModalComponent()->inputAttemptWhenModal();
return frameRect.size;
}
- (void) zoom: (id) sender
{
isZooming = true;
[super zoom: sender];
isZooming = false;
owner->redirectMovedOrResized();
}
- (void) windowWillMove: (NSNotification*) notification
{
(void) notification;
if (juce::Component::getCurrentlyModalComponent() != nullptr
&& owner->getComponent()->isCurrentlyBlockedByAnotherModalComponent()
&& owner->hasNativeTitleBar())
juce::Component::getCurrentlyModalComponent()->inputAttemptWhenModal();
}
@end
//==============================================================================
//==============================================================================
BEGIN_JUCE_NAMESPACE
//==============================================================================
ModifierKeys NSViewComponentPeer::currentModifiers;
ComponentPeer* NSViewComponentPeer::currentlyFocusedPeer = nullptr;
Array<int> NSViewComponentPeer::keysCurrentlyDown;
//==============================================================================
bool KeyPress::isKeyCurrentlyDown (const int keyCode)
{
if (NSViewComponentPeer::keysCurrentlyDown.contains (keyCode))
return true;
if (keyCode >= 'A' && keyCode <= 'Z'
&& NSViewComponentPeer::keysCurrentlyDown.contains ((int) CharacterFunctions::toLowerCase ((juce_wchar) keyCode)))
return true;
if (keyCode >= 'a' && keyCode <= 'z'
&& NSViewComponentPeer::keysCurrentlyDown.contains ((int) CharacterFunctions::toUpperCase ((juce_wchar) keyCode)))
return true;
return false;
}
void NSViewComponentPeer::updateModifiers (NSEvent* e)
{
int m = 0;
if (([e modifierFlags] & NSShiftKeyMask) != 0) m |= ModifierKeys::shiftModifier;
if (([e modifierFlags] & NSControlKeyMask) != 0) m |= ModifierKeys::ctrlModifier;
if (([e modifierFlags] & NSAlternateKeyMask) != 0) m |= ModifierKeys::altModifier;
if (([e modifierFlags] & NSCommandKeyMask) != 0) m |= ModifierKeys::commandModifier;
currentModifiers = currentModifiers.withOnlyMouseButtons().withFlags (m);
}
void NSViewComponentPeer::updateKeysDown (NSEvent* ev, bool isKeyDown)
{
updateModifiers (ev);
int keyCode = getKeyCodeFromEvent (ev);
if (keyCode != 0)
{
if (isKeyDown)
keysCurrentlyDown.addIfNotAlreadyThere (keyCode);
else
keysCurrentlyDown.removeValue (keyCode);
}
}
ModifierKeys ModifierKeys::getCurrentModifiersRealtime() noexcept
{
return NSViewComponentPeer::currentModifiers;
}
void ModifierKeys::updateCurrentModifiers() noexcept
{
currentModifiers = NSViewComponentPeer::currentModifiers;
}
//==============================================================================
NSViewComponentPeer::NSViewComponentPeer (Component* const component_,
const int windowStyleFlags,
NSView* viewToAttachTo)
: ComponentPeer (component_, windowStyleFlags),
window (nil),
view (nil),
isSharedWindow (viewToAttachTo != nil),
fullScreen (false),
insideDrawRect (false),
#if USE_COREGRAPHICS_RENDERING
usingCoreGraphics (true),
#else
usingCoreGraphics (false),
#endif
recursiveToFrontCall (false)
{
appFocusChangeCallback = appFocusChanged;
isEventBlockedByModalComps = checkEventBlockedByModalComps;
NSRect r = NSMakeRect (0, 0, (CGFloat) component->getWidth(), (CGFloat) component->getHeight());
view = [[JuceNSView alloc] initWithOwner: this withFrame: r];
[view setPostsFrameChangedNotifications: YES];
if (isSharedWindow)
{
window = [viewToAttachTo window];
[viewToAttachTo addSubview: view];
}
else
{
r.origin.x = (CGFloat) component->getX();
r.origin.y = (CGFloat) component->getY();
r.origin.y = [[[NSScreen screens] objectAtIndex: 0] frame].size.height - (r.origin.y + r.size.height);
window = [[JuceNSWindow alloc] initWithContentRect: r
styleMask: getNSWindowStyleMask (windowStyleFlags)
backing: NSBackingStoreBuffered
defer: YES];
[((JuceNSWindow*) window) setOwner: this];
[window orderOut: nil];
[window setDelegate: (JuceNSWindow*) window];
[window setOpaque: component->isOpaque()];
[window setHasShadow: ((windowStyleFlags & windowHasDropShadow) != 0)];
if (component->isAlwaysOnTop())
[window setLevel: NSFloatingWindowLevel];
[window setContentView: view];
[window setAutodisplay: YES];
[window setAcceptsMouseMovedEvents: YES];
// We'll both retain and also release this on closing because plugin hosts can unexpectedly
// close the window for us, and also tend to get cause trouble if setReleasedWhenClosed is NO.
[window setReleasedWhenClosed: YES];
[window retain];
[window setExcludedFromWindowsMenu: (windowStyleFlags & windowIsTemporary) != 0];
[window setIgnoresMouseEvents: (windowStyleFlags & windowIgnoresMouseClicks) != 0];
}
const float alpha = component->getAlpha();
if (alpha < 1.0f)
setAlpha (alpha);
setTitle (component->getName());
}
NSViewComponentPeer::~NSViewComponentPeer()
{
view->owner = nullptr;
[view removeFromSuperview];
[view release];
if (! isSharedWindow)
{
[((JuceNSWindow*) window) setOwner: 0];
[window close];
[window release];
}
}
//==============================================================================
void* NSViewComponentPeer::getNativeHandle() const
{
return view;
}
void NSViewComponentPeer::setVisible (bool shouldBeVisible)
{
if (isSharedWindow)
{
[view setHidden: ! shouldBeVisible];
}
else
{
if (shouldBeVisible)
{
[window orderFront: nil];
handleBroughtToFront();
}
else
{
[window orderOut: nil];
}
}
}
void NSViewComponentPeer::setTitle (const String& title)
{
JUCE_AUTORELEASEPOOL
if (! isSharedWindow)
[window setTitle: juceStringToNS (title)];
}
void NSViewComponentPeer::setPosition (int x, int y)
{
setBounds (x, y, component->getWidth(), component->getHeight(), false);
}
void NSViewComponentPeer::setSize (int w, int h)
{
setBounds (component->getX(), component->getY(), w, h, false);
}
void NSViewComponentPeer::setBounds (int x, int y, int w, int h, bool isNowFullScreen)
{
fullScreen = isNowFullScreen;
NSRect r = NSMakeRect ((CGFloat) x, (CGFloat) y, (CGFloat) jmax (0, w), (CGFloat) jmax (0, h));
if (isSharedWindow)
{
r.origin.y = [[view superview] frame].size.height - (r.origin.y + r.size.height);
if ([view frame].size.width != r.size.width
|| [view frame].size.height != r.size.height)
[view setNeedsDisplay: true];
[view setFrame: r];
}
else
{
r.origin.y = [[[NSScreen screens] objectAtIndex: 0] frame].size.height - (r.origin.y + r.size.height);
[window setFrame: [window frameRectForContentRect: r]
display: true];
}
}
const Rectangle<int> NSViewComponentPeer::getBounds (const bool global) const
{
NSRect r = [view frame];
if (global && [view window] != nil)
{
r = [view convertRect: r toView: nil];
NSRect wr = [[view window] frame];
r.origin.x += wr.origin.x;
r.origin.y += wr.origin.y;
r.origin.y = [[[NSScreen screens] objectAtIndex: 0] frame].size.height - r.origin.y - r.size.height;
}
else
{
r.origin.y = [[view superview] frame].size.height - r.origin.y - r.size.height;
}
return Rectangle<int> (convertToRectInt (r));
}
const Rectangle<int> NSViewComponentPeer::getBounds() const
{
return getBounds (! isSharedWindow);
}
const Point<int> NSViewComponentPeer::getScreenPosition() const
{
return getBounds (true).getPosition();
}
const Point<int> NSViewComponentPeer::localToGlobal (const Point<int>& relativePosition)
{
return relativePosition + getScreenPosition();
}
const Point<int> NSViewComponentPeer::globalToLocal (const Point<int>& screenPosition)
{
return screenPosition - getScreenPosition();
}
NSRect NSViewComponentPeer::constrainRect (NSRect r)
{
if (constrainer != nullptr)
{
NSRect current = [window frame];
current.origin.y = [[[NSScreen screens] objectAtIndex: 0] frame].size.height - current.origin.y - current.size.height;
r.origin.y = [[[NSScreen screens] objectAtIndex: 0] frame].size.height - r.origin.y - r.size.height;
Rectangle<int> pos (convertToRectInt (r));
Rectangle<int> original (convertToRectInt (current));
#if defined (MAC_OS_X_VERSION_10_6) && MAC_OS_X_VERSION_MIN_ALLOWED >= MAC_OS_X_VERSION_10_6
if ([window inLiveResize])
#else
if ([window respondsToSelector: @selector (inLiveResize)]
&& [window performSelector: @selector (inLiveResize)])
#endif
{
constrainer->checkBounds (pos, original,
Desktop::getInstance().getAllMonitorDisplayAreas().getBounds(),
false, false, true, true);
}
else
{
constrainer->checkBounds (pos, original,
Desktop::getInstance().getAllMonitorDisplayAreas().getBounds(),
pos.getY() != original.getY() && pos.getBottom() == original.getBottom(),
pos.getX() != original.getX() && pos.getRight() == original.getRight(),
pos.getY() == original.getY() && pos.getBottom() != original.getBottom(),
pos.getX() == original.getX() && pos.getRight() != original.getRight());
}
r.origin.x = pos.getX();
r.origin.y = [[[NSScreen screens] objectAtIndex: 0] frame].size.height - r.size.height - pos.getY();
r.size.width = pos.getWidth();
r.size.height = pos.getHeight();
}
return r;
}
void NSViewComponentPeer::setAlpha (float newAlpha)
{
if (! isSharedWindow)
{
[window setAlphaValue: (CGFloat) newAlpha];
}
else
{
#if defined (MAC_OS_X_VERSION_10_5) && MAC_OS_X_VERSION_MIN_ALLOWED >= MAC_OS_X_VERSION_10_5
[view setAlphaValue: (CGFloat) newAlpha];
#else
if ([view respondsToSelector: @selector (setAlphaValue:)])
{
// PITA dynamic invocation for 10.4 builds..
NSInvocation* inv = [NSInvocation invocationWithMethodSignature: [view methodSignatureForSelector: @selector (setAlphaValue:)]];
[inv setSelector: @selector (setAlphaValue:)];
[inv setTarget: view];
CGFloat cgNewAlpha = (CGFloat) newAlpha;
[inv setArgument: &cgNewAlpha atIndex: 2];
[inv invoke];
}
#endif
}
}
void NSViewComponentPeer::setMinimised (bool shouldBeMinimised)
{
if (! isSharedWindow)
{
if (shouldBeMinimised)
[window miniaturize: nil];
else
[window deminiaturize: nil];
}
}
bool NSViewComponentPeer::isMinimised() const
{
return [window isMiniaturized];
}
void NSViewComponentPeer::setFullScreen (bool shouldBeFullScreen)
{
if (! isSharedWindow)
{
Rectangle<int> r (lastNonFullscreenBounds);
if (isMinimised())
setMinimised (false);
if (fullScreen != shouldBeFullScreen)
{
if (shouldBeFullScreen && hasNativeTitleBar())
{
fullScreen = true;
[window performZoom: nil];
}
else
{
if (shouldBeFullScreen)
r = component->getParentMonitorArea();
// (can't call the component's setBounds method because that'll reset our fullscreen flag)
if (r != getComponent()->getBounds() && ! r.isEmpty())
setBounds (r.getX(), r.getY(), r.getWidth(), r.getHeight(), shouldBeFullScreen);
}
}
}
}
bool NSViewComponentPeer::isFullScreen() const
{
return fullScreen;
}
bool NSViewComponentPeer::contains (const Point<int>& position, bool trueIfInAChildWindow) const
{
if (! (isPositiveAndBelow (position.getX(), component->getWidth())
&& isPositiveAndBelow (position.getY(), component->getHeight())))
return false;
NSRect frameRect = [view frame];
NSView* v = [view hitTest: NSMakePoint (frameRect.origin.x + position.getX(),
frameRect.origin.y + frameRect.size.height - position.getY())];
if (trueIfInAChildWindow)
return v != nil;
return v == view;
}
const BorderSize<int> NSViewComponentPeer::getFrameSize() const
{
BorderSize<int> b;
if (! isSharedWindow)
{
NSRect v = [view convertRect: [view frame] toView: nil];
NSRect w = [window frame];
b.setTop ((int) (w.size.height - (v.origin.y + v.size.height)));
b.setBottom ((int) v.origin.y);
b.setLeft ((int) v.origin.x);
b.setRight ((int) (w.size.width - (v.origin.x + v.size.width)));
}
return b;
}
bool NSViewComponentPeer::setAlwaysOnTop (bool alwaysOnTop)
{
if (! isSharedWindow)
[window setLevel: alwaysOnTop ? NSFloatingWindowLevel
: NSNormalWindowLevel];
return true;
}
void NSViewComponentPeer::toFront (bool makeActiveWindow)
{
if (isSharedWindow)
[[view superview] addSubview: view
positioned: NSWindowAbove
relativeTo: nil];
if (window != nil && component->isVisible())
{
if (makeActiveWindow)
[window makeKeyAndOrderFront: nil];
else
[window orderFront: nil];
if (! recursiveToFrontCall)
{
recursiveToFrontCall = true;
Desktop::getInstance().getMainMouseSource().forceMouseCursorUpdate();
handleBroughtToFront();
recursiveToFrontCall = false;
}
}
}
void NSViewComponentPeer::toBehind (ComponentPeer* other)
{
NSViewComponentPeer* const otherPeer = dynamic_cast <NSViewComponentPeer*> (other);
jassert (otherPeer != nullptr); // wrong type of window?
if (otherPeer != nullptr)
{
if (isSharedWindow)
{
[[view superview] addSubview: view
positioned: NSWindowBelow
relativeTo: otherPeer->view];
}
else
{
[window orderWindow: NSWindowBelow
relativeTo: [otherPeer->window windowNumber]];
}
}
}
void NSViewComponentPeer::setIcon (const Image& /*newIcon*/)
{
// to do..
}
//==============================================================================
void NSViewComponentPeer::viewFocusGain()
{
if (currentlyFocusedPeer != this)
{
if (ComponentPeer::isValidPeer (currentlyFocusedPeer))
currentlyFocusedPeer->handleFocusLoss();
currentlyFocusedPeer = this;
handleFocusGain();
}
}
void NSViewComponentPeer::viewFocusLoss()
{
if (currentlyFocusedPeer == this)
{
currentlyFocusedPeer = nullptr;
handleFocusLoss();
}
}
bool NSViewComponentPeer::isFocused() const
{
return isSharedWindow ? this == currentlyFocusedPeer
: [window isKeyWindow];
}
void NSViewComponentPeer::grabFocus()
{
if (window != nil)
{
[window makeKeyWindow];
[window makeFirstResponder: view];
viewFocusGain();
}
}
void NSViewComponentPeer::textInputRequired (const Point<int>&)
{
}
bool NSViewComponentPeer::handleKeyEvent (NSEvent* ev, bool isKeyDown)
{
String unicode (nsStringToJuce ([ev characters]));
String unmodified (nsStringToJuce ([ev charactersIgnoringModifiers]));
int keyCode = getKeyCodeFromEvent (ev);
//DBG ("unicode: " + unicode + " " + String::toHexString ((int) unicode[0]));
//DBG ("unmodified: " + unmodified + " " + String::toHexString ((int) unmodified[0]));
if (unicode.isNotEmpty() || keyCode != 0)
{
if (isKeyDown)
{
bool used = false;
while (unicode.length() > 0)
{
juce_wchar textCharacter = unicode[0];
unicode = unicode.substring (1);
if (([ev modifierFlags] & NSCommandKeyMask) != 0)
textCharacter = 0;
used = handleKeyUpOrDown (true) || used;
used = handleKeyPress (keyCode, textCharacter) || used;
}
return used;
}
else
{
if (handleKeyUpOrDown (false))
return true;
}
}
return false;
}
bool NSViewComponentPeer::redirectKeyDown (NSEvent* ev)
{
updateKeysDown (ev, true);
bool used = handleKeyEvent (ev, true);
if (([ev modifierFlags] & NSCommandKeyMask) != 0)
{
// for command keys, the key-up event is thrown away, so simulate one..
updateKeysDown (ev, false);
used = (isValidPeer (this) && handleKeyEvent (ev, false)) || used;
}
// (If we're running modally, don't allow unused keystrokes to be passed
// along to other blocked views..)
if (Component::getCurrentlyModalComponent() != nullptr)
used = true;
return used;
}
bool NSViewComponentPeer::redirectKeyUp (NSEvent* ev)
{
updateKeysDown (ev, false);
return handleKeyEvent (ev, false)
|| Component::getCurrentlyModalComponent() != nullptr;
}
void NSViewComponentPeer::redirectModKeyChange (NSEvent* ev)
{
keysCurrentlyDown.clear();
handleKeyUpOrDown (true);
updateModifiers (ev);
handleModifierKeysChange();
}
#if MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_X_VERSION_10_5
bool NSViewComponentPeer::redirectPerformKeyEquivalent (NSEvent* ev)
{
if ([ev type] == NSKeyDown)
return redirectKeyDown (ev);
else if ([ev type] == NSKeyUp)
return redirectKeyUp (ev);
return false;
}
#endif
//==============================================================================
void NSViewComponentPeer::sendMouseEvent (NSEvent* ev)
{
updateModifiers (ev);
handleMouseEvent (0, getMousePos (ev, view), currentModifiers, getMouseTime (ev));
}
void NSViewComponentPeer::redirectMouseDown (NSEvent* ev)
{
currentModifiers = currentModifiers.withFlags (getModifierForButtonNumber ([ev buttonNumber]));
sendMouseEvent (ev);
}
void NSViewComponentPeer::redirectMouseUp (NSEvent* ev)
{
currentModifiers = currentModifiers.withoutFlags (getModifierForButtonNumber ([ev buttonNumber]));
sendMouseEvent (ev);
showArrowCursorIfNeeded();
}
void NSViewComponentPeer::redirectMouseDrag (NSEvent* ev)
{
currentModifiers = currentModifiers.withFlags (getModifierForButtonNumber ([ev buttonNumber]));
sendMouseEvent (ev);
}
void NSViewComponentPeer::redirectMouseMove (NSEvent* ev)
{
currentModifiers = currentModifiers.withoutMouseButtons();
sendMouseEvent (ev);
showArrowCursorIfNeeded();
}
void NSViewComponentPeer::redirectMouseEnter (NSEvent* ev)
{
Desktop::getInstance().getMainMouseSource().forceMouseCursorUpdate();
currentModifiers = currentModifiers.withoutMouseButtons();
sendMouseEvent (ev);
}
void NSViewComponentPeer::redirectMouseExit (NSEvent* ev)
{
currentModifiers = currentModifiers.withoutMouseButtons();
sendMouseEvent (ev);
}
void NSViewComponentPeer::redirectMouseWheel (NSEvent* ev)
{
updateModifiers (ev);
float x = 0, y = 0;
@try
{
x = [ev deviceDeltaX] * 0.5f;
y = [ev deviceDeltaY] * 0.5f;
}
@catch (...)
{}
if (x == 0 && y == 0)
{
x = [ev deltaX] * 10.0f;
y = [ev deltaY] * 10.0f;
}
handleMouseWheel (0, getMousePos (ev, view), getMouseTime (ev), x, y);
}
void NSViewComponentPeer::showArrowCursorIfNeeded()
{
MouseInputSource& mouse = Desktop::getInstance().getMainMouseSource();
if (mouse.getComponentUnderMouse() == nullptr
&& Desktop::getInstance().findComponentAt (mouse.getScreenPosition()) == nullptr)
{
[[NSCursor arrowCursor] set];
}
}
//==============================================================================
BOOL NSViewComponentPeer::sendDragCallback (const int type, id <NSDraggingInfo> sender)
{
NSString* bestType
= [[sender draggingPasteboard] availableTypeFromArray: [view getSupportedDragTypes]];
if (bestType == nil)
return false;
NSPoint p = [view convertPoint: [sender draggingLocation] fromView: nil];
const Point<int> pos ((int) p.x, (int) ([view frame].size.height - p.y));
NSPasteboard* pasteBoard = [sender draggingPasteboard];
StringArray files;
NSString* iTunesPasteboardType = nsStringLiteral ("CorePasteboardFlavorType 0x6974756E"); // 'itun'
if (bestType == NSFilesPromisePboardType
&& [[pasteBoard types] containsObject: iTunesPasteboardType])
{
id list = [pasteBoard propertyListForType: iTunesPasteboardType];
if ([list isKindOfClass: [NSDictionary class]])
{
NSDictionary* iTunesDictionary = (NSDictionary*) list;
NSArray* tracks = [iTunesDictionary valueForKey: nsStringLiteral ("Tracks")];
NSEnumerator* enumerator = [tracks objectEnumerator];
NSDictionary* track;
while ((track = [enumerator nextObject]) != nil)
{
NSURL* url = [NSURL URLWithString: [track valueForKey: nsStringLiteral ("Location")]];
if ([url isFileURL])
files.add (nsStringToJuce ([url path]));
}
}
}
else
{
id list = [pasteBoard propertyListForType: NSFilenamesPboardType];
if ([list isKindOfClass: [NSArray class]])
{
NSArray* items = (NSArray*) [pasteBoard propertyListForType: NSFilenamesPboardType];
for (unsigned int i = 0; i < [items count]; ++i)
files.add (nsStringToJuce ((NSString*) [items objectAtIndex: i]));
}
}
if (files.size() > 0)
{
switch (type)
{
case 0: return handleFileDragMove (files, pos);
case 1: return handleFileDragExit (files);
case 2: return handleFileDragDrop (files, pos);
default: jassertfalse; break;
}
}
return false;
}
bool NSViewComponentPeer::isOpaque()
{
return component == nullptr || component->isOpaque();
}
void NSViewComponentPeer::drawRect (NSRect r)
{
if (r.size.width < 1.0f || r.size.height < 1.0f)
return;
CGContextRef cg = (CGContextRef) [[NSGraphicsContext currentContext] graphicsPort];
if (! component->isOpaque())
CGContextClearRect (cg, CGContextGetClipBoundingBox (cg));
#if USE_COREGRAPHICS_RENDERING
if (usingCoreGraphics)
{
CoreGraphicsContext context (cg, (float) [view frame].size.height);
insideDrawRect = true;
handlePaint (context);
insideDrawRect = false;
}
else
#endif
{
Image temp (getComponent()->isOpaque() ? Image::RGB : Image::ARGB,
(int) (r.size.width + 0.5f),
(int) (r.size.height + 0.5f),
! getComponent()->isOpaque());
const int xOffset = -roundToInt (r.origin.x);
const int yOffset = -roundToInt ([view frame].size.height - (r.origin.y + r.size.height));
const NSRect* rects = nullptr;
NSInteger numRects = 0;
[view getRectsBeingDrawn: &rects count: &numRects];
const Rectangle<int> clipBounds (temp.getBounds());
RectangleList clip;
for (int i = 0; i < numRects; ++i)
{
clip.addWithoutMerging (clipBounds.getIntersection (Rectangle<int> (roundToInt (rects[i].origin.x) + xOffset,
roundToInt ([view frame].size.height - (rects[i].origin.y + rects[i].size.height)) + yOffset,
roundToInt (rects[i].size.width),
roundToInt (rects[i].size.height))));
}
if (! clip.isEmpty())
{
LowLevelGraphicsSoftwareRenderer context (temp, xOffset, yOffset, clip);
insideDrawRect = true;
handlePaint (context);
insideDrawRect = false;
CGColorSpaceRef colourSpace = CGColorSpaceCreateDeviceRGB();
CGImageRef image = juce_createCoreGraphicsImage (temp, false, colourSpace, false);
CGColorSpaceRelease (colourSpace);
CGContextDrawImage (cg, CGRectMake (r.origin.x, r.origin.y, temp.getWidth(), temp.getHeight()), image);
CGImageRelease (image);
}
}
}
StringArray NSViewComponentPeer::getAvailableRenderingEngines()
{
StringArray s (ComponentPeer::getAvailableRenderingEngines());
#if USE_COREGRAPHICS_RENDERING
s.add ("CoreGraphics Renderer");
#endif
return s;
}
int NSViewComponentPeer::getCurrentRenderingEngine() const
{
return usingCoreGraphics ? 1 : 0;
}
void NSViewComponentPeer::setCurrentRenderingEngine (int index)
{
#if USE_COREGRAPHICS_RENDERING
if (usingCoreGraphics != (index > 0))
{
usingCoreGraphics = index > 0;
[view setNeedsDisplay: true];
}
#endif
}
bool NSViewComponentPeer::canBecomeKeyWindow()
{
return (getStyleFlags() & juce::ComponentPeer::windowIgnoresKeyPresses) == 0;
}
void NSViewComponentPeer::becomeKeyWindow()
{
handleBroughtToFront();
grabFocus();
}
bool NSViewComponentPeer::windowShouldClose()
{
if (! isValidPeer (this))
return YES;
handleUserClosingWindow();
return NO;
}
void NSViewComponentPeer::updateFullscreenStatus()
{
if (hasNativeTitleBar())
{
const Rectangle<int> screen (getFrameSize().subtractedFrom (component->getParentMonitorArea()));
const Rectangle<int> window (component->getScreenBounds());
fullScreen = std::abs (screen.getX() - window.getX()) <= 2
&& std::abs (screen.getY() - window.getY()) <= 2
&& std::abs (screen.getRight() - window.getRight()) <= 2
&& std::abs (screen.getBottom() - window.getBottom()) <= 2;
}
}
void NSViewComponentPeer::redirectMovedOrResized()
{
updateFullscreenStatus();
handleMovedOrResized();
}
void NSViewComponentPeer::viewMovedToWindow()
{
if (isSharedWindow)
window = [view window];
}
//==============================================================================
void Desktop::createMouseInputSources()
{
mouseSources.add (new MouseInputSource (0, true));
}
//==============================================================================
void Desktop::setKioskComponent (Component* kioskModeComponent, bool enableOrDisable, bool allowMenusAndBars)
{
#if defined (MAC_OS_X_VERSION_10_6) && MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_6
if (enableOrDisable)
{
[NSApp setPresentationOptions: (allowMenusAndBars ? (NSApplicationPresentationAutoHideDock | NSApplicationPresentationAutoHideMenuBar)
: (NSApplicationPresentationHideDock | NSApplicationPresentationHideMenuBar))];
kioskModeComponent->setBounds (Desktop::getInstance().getMainMonitorArea (false));
}
else
{
[NSApp setPresentationOptions: NSApplicationPresentationDefault];
}
#else
if (enableOrDisable)
{
SetSystemUIMode (kUIModeAllSuppressed, allowMenusAndBars ? kUIOptionAutoShowMenuBar : 0);
kioskModeComponent->setBounds (Desktop::getInstance().getMainMonitorArea (false));
}
else
{
SetSystemUIMode (kUIModeNormal, 0);
}
#endif
}
//==============================================================================
void NSViewComponentPeer::repaint (const Rectangle<int>& area)
{
if (insideDrawRect)
{
class AsyncRepaintMessage : public CallbackMessage
{
public:
AsyncRepaintMessage (NSViewComponentPeer* const peer_, const Rectangle<int>& rect_)
: peer (peer_), rect (rect_)
{
}
void messageCallback()
{
if (ComponentPeer::isValidPeer (peer))
peer->repaint (rect);
}
private:
NSViewComponentPeer* const peer;
const Rectangle<int> rect;
};
(new AsyncRepaintMessage (this, area))->post();
}
else
{
[view setNeedsDisplayInRect: NSMakeRect ((CGFloat) area.getX(), [view frame].size.height - (CGFloat) area.getBottom(),
(CGFloat) area.getWidth(), (CGFloat) area.getHeight())];
}
}
void NSViewComponentPeer::performAnyPendingRepaintsNow()
{
[view displayIfNeeded];
}
ComponentPeer* Component::createNewPeer (int styleFlags, void* windowToAttachTo)
{
return new NSViewComponentPeer (this, styleFlags, (NSView*) windowToAttachTo);
}
//==============================================================================
const int KeyPress::spaceKey = ' ';
const int KeyPress::returnKey = 0x0d;
const int KeyPress::escapeKey = 0x1b;
const int KeyPress::backspaceKey = 0x7f;
const int KeyPress::leftKey = NSLeftArrowFunctionKey;
const int KeyPress::rightKey = NSRightArrowFunctionKey;
const int KeyPress::upKey = NSUpArrowFunctionKey;
const int KeyPress::downKey = NSDownArrowFunctionKey;
const int KeyPress::pageUpKey = NSPageUpFunctionKey;
const int KeyPress::pageDownKey = NSPageDownFunctionKey;
const int KeyPress::endKey = NSEndFunctionKey;
const int KeyPress::homeKey = NSHomeFunctionKey;
const int KeyPress::deleteKey = NSDeleteFunctionKey;
const int KeyPress::insertKey = -1;
const int KeyPress::tabKey = 9;
const int KeyPress::F1Key = NSF1FunctionKey;
const int KeyPress::F2Key = NSF2FunctionKey;
const int KeyPress::F3Key = NSF3FunctionKey;
const int KeyPress::F4Key = NSF4FunctionKey;
const int KeyPress::F5Key = NSF5FunctionKey;
const int KeyPress::F6Key = NSF6FunctionKey;
const int KeyPress::F7Key = NSF7FunctionKey;
const int KeyPress::F8Key = NSF8FunctionKey;
const int KeyPress::F9Key = NSF9FunctionKey;
const int KeyPress::F10Key = NSF10FunctionKey;
const int KeyPress::F11Key = NSF1FunctionKey;
const int KeyPress::F12Key = NSF12FunctionKey;
const int KeyPress::F13Key = NSF13FunctionKey;
const int KeyPress::F14Key = NSF14FunctionKey;
const int KeyPress::F15Key = NSF15FunctionKey;
const int KeyPress::F16Key = NSF16FunctionKey;
const int KeyPress::numberPad0 = 0x30020;
const int KeyPress::numberPad1 = 0x30021;
const int KeyPress::numberPad2 = 0x30022;
const int KeyPress::numberPad3 = 0x30023;
const int KeyPress::numberPad4 = 0x30024;
const int KeyPress::numberPad5 = 0x30025;
const int KeyPress::numberPad6 = 0x30026;
const int KeyPress::numberPad7 = 0x30027;
const int KeyPress::numberPad8 = 0x30028;
const int KeyPress::numberPad9 = 0x30029;
const int KeyPress::numberPadAdd = 0x3002a;
const int KeyPress::numberPadSubtract = 0x3002b;
const int KeyPress::numberPadMultiply = 0x3002c;
const int KeyPress::numberPadDivide = 0x3002d;
const int KeyPress::numberPadSeparator = 0x3002e;
const int KeyPress::numberPadDecimalPoint = 0x3002f;
const int KeyPress::numberPadEquals = 0x30030;
const int KeyPress::numberPadDelete = 0x30031;
const int KeyPress::playKey = 0x30000;
const int KeyPress::stopKey = 0x30001;
const int KeyPress::fastForwardKey = 0x30002;
const int KeyPress::rewindKey = 0x30003;