mirror of
https://github.com/juce-framework/JUCE.git
synced 2026-01-10 23:44:24 +00:00
1984 lines
64 KiB
Text
1984 lines
64 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;
|
|
|
|
#if JUCE_MODULE_AVAILABLE_juce_opengl && ! defined (JUCE_OSX_OPENGL_RENDERER)
|
|
//#define JUCE_OSX_OPENGL_RENDERER 1
|
|
#endif
|
|
|
|
//==============================================================================
|
|
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;
|
|
|
|
- (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);
|
|
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
virtual void drawOpenGL();
|
|
#endif
|
|
|
|
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 updateModifiers (NSUInteger);
|
|
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, usingOpenGL, 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) 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
|
|
|
|
|
|
//==============================================================================
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
|
|
@interface JuceOpenGLLayer : CAOpenGLLayer
|
|
{
|
|
NSViewComponentPeer* owner;
|
|
}
|
|
|
|
- (JuceOpenGLLayer*) initWithPeer: (NSViewComponentPeer*) owner;
|
|
- (void) dealloc;
|
|
|
|
- (void) drawInCGLContext: (CGLContextObj) glContext
|
|
pixelFormat: (CGLPixelFormatObj) pixelFormat
|
|
forLayerTime: (CFTimeInterval) timeInterval
|
|
displayTime: (const CVTimeStamp*) timeStamp;
|
|
@end
|
|
|
|
@implementation JuceOpenGLLayer
|
|
|
|
- (JuceOpenGLLayer*) initWithPeer: (NSViewComponentPeer*) owner_
|
|
{
|
|
[super init];
|
|
owner = owner_;
|
|
return self;
|
|
}
|
|
|
|
- (void) dealloc
|
|
{
|
|
[super dealloc];
|
|
}
|
|
|
|
- (void) drawInCGLContext: (CGLContextObj) glContext
|
|
pixelFormat: (CGLPixelFormatObj) pixelFormat
|
|
forLayerTime: (CFTimeInterval) timeInterval
|
|
displayTime: (const CVTimeStamp*) timeStamp
|
|
{
|
|
owner->drawOpenGL();
|
|
}
|
|
|
|
@end
|
|
#endif
|
|
|
|
//==============================================================================
|
|
//==============================================================================
|
|
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 (const NSUInteger flags)
|
|
{
|
|
int m = 0;
|
|
|
|
if ((flags & NSShiftKeyMask) != 0) m |= ModifierKeys::shiftModifier;
|
|
if ((flags & NSControlKeyMask) != 0) m |= ModifierKeys::ctrlModifier;
|
|
if ((flags & NSAlternateKeyMask) != 0) m |= ModifierKeys::altModifier;
|
|
if ((flags & NSCommandKeyMask) != 0) m |= ModifierKeys::commandModifier;
|
|
|
|
currentModifiers = currentModifiers.withOnlyMouseButtons().withFlags (m);
|
|
}
|
|
|
|
void NSViewComponentPeer::updateModifiers (NSEvent* e)
|
|
{
|
|
updateModifiers ([e modifierFlags]);
|
|
}
|
|
|
|
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
|
|
{
|
|
NSViewComponentPeer::updateModifiers ([NSEvent modifierFlags]);
|
|
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
|
|
usingOpenGL (false),
|
|
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)
|
|
{
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
if (usingOpenGL)
|
|
[[view layer] setNeedsDisplay: true];
|
|
#endif
|
|
[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];
|
|
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
if (usingOpenGL)
|
|
[[view layer] setFrame: CGRectMake (0, 0, [view frame].size.width, [view frame].size.height)];
|
|
#endif
|
|
}
|
|
}
|
|
|
|
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();
|
|
}
|
|
|
|
static void getClipRects (RectangleList& clip, NSView* view,
|
|
const int xOffset, const int yOffset, const int clipW, const int clipH)
|
|
{
|
|
const NSRect* rects = nullptr;
|
|
NSInteger numRects = 0;
|
|
[view getRectsBeingDrawn: &rects count: &numRects];
|
|
|
|
const Rectangle<int> clipBounds (clipW, clipH);
|
|
const CGFloat viewH = [view frame].size.height;
|
|
|
|
for (int i = 0; i < numRects; ++i)
|
|
clip.addWithoutMerging (clipBounds.getIntersection (Rectangle<int> (roundToInt (rects[i].origin.x) + xOffset,
|
|
roundToInt (viewH - (rects[i].origin.y + rects[i].size.height)) + yOffset,
|
|
roundToInt (rects[i].size.width),
|
|
roundToInt (rects[i].size.height))));
|
|
}
|
|
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
void NSViewComponentPeer::drawOpenGL()
|
|
{
|
|
if (! component->isOpaque())
|
|
OpenGLHelpers::clear (Colours::transparentBlack);
|
|
|
|
OpenGLRenderer context (OpenGLFrameBuffer::getCurrentFrameBufferTarget(),
|
|
component->getWidth(), component->getHeight());
|
|
|
|
insideDrawRect = true;
|
|
handlePaint (context);
|
|
insideDrawRect = false;
|
|
}
|
|
#endif
|
|
|
|
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
|
|
{
|
|
const int xOffset = -roundToInt (r.origin.x);
|
|
const int yOffset = -roundToInt ([view frame].size.height - (r.origin.y + r.size.height));
|
|
const int clipW = (int) (r.size.width + 0.5f);
|
|
const int clipH = (int) (r.size.height + 0.5f);
|
|
|
|
RectangleList clip;
|
|
getClipRects (clip, view, xOffset, yOffset, clipW, clipH);
|
|
|
|
if (! clip.isEmpty())
|
|
{
|
|
Image temp (getComponent()->isOpaque() ? Image::RGB : Image::ARGB,
|
|
clipW, clipH, ! getComponent()->isOpaque());
|
|
|
|
{
|
|
ScopedPointer<LowLevelGraphicsContext> context (component->getLookAndFeel()
|
|
.createGraphicsContext (temp, Point<int> (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, clipW, clipH), image);
|
|
CGImageRelease (image);
|
|
}
|
|
}
|
|
}
|
|
|
|
StringArray NSViewComponentPeer::getAvailableRenderingEngines()
|
|
{
|
|
StringArray s (ComponentPeer::getAvailableRenderingEngines());
|
|
|
|
#if USE_COREGRAPHICS_RENDERING
|
|
s.add ("CoreGraphics Renderer");
|
|
#endif
|
|
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
s.add ("OpenGL Renderer");
|
|
#endif
|
|
|
|
return s;
|
|
}
|
|
|
|
int NSViewComponentPeer::getCurrentRenderingEngine() const
|
|
{
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
if (usingOpenGL) return 2;
|
|
#endif
|
|
|
|
return usingCoreGraphics ? 1 : 0;
|
|
}
|
|
|
|
void NSViewComponentPeer::setCurrentRenderingEngine (int index)
|
|
{
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
if (index == 2)
|
|
{
|
|
usingCoreGraphics = false;
|
|
usingOpenGL = true;
|
|
|
|
JuceOpenGLLayer* glLayer = [[JuceOpenGLLayer alloc] initWithPeer: this];
|
|
[view setLayer: glLayer];
|
|
[view setWantsLayer: YES];
|
|
[glLayer release];
|
|
}
|
|
else
|
|
{
|
|
usingOpenGL = false;
|
|
|
|
[view setLayer: nil];
|
|
[view setWantsLayer: NO];
|
|
}
|
|
#endif
|
|
|
|
#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];
|
|
}
|
|
#elif JUCE_SUPPORT_CARBON
|
|
if (enableOrDisable)
|
|
{
|
|
SetSystemUIMode (kUIModeAllSuppressed, allowMenusAndBars ? kUIOptionAutoShowMenuBar : 0);
|
|
kioskModeComponent->setBounds (Desktop::getInstance().getMainMonitorArea (false));
|
|
}
|
|
else
|
|
{
|
|
SetSystemUIMode (kUIModeNormal, 0);
|
|
}
|
|
#else
|
|
// If you're targeting OSes earlier than 10.6 and want to use this feature,
|
|
// you'll need to enable JUCE_SUPPORT_CARBON.
|
|
jassertfalse;
|
|
#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
|
|
{
|
|
#if JUCE_OSX_OPENGL_RENDERER
|
|
if (usingOpenGL)
|
|
[[view layer] setNeedsDisplayInRect: CGRectMake ((CGFloat) area.getX(), [view frame].size.height - (CGFloat) area.getBottom(),
|
|
(CGFloat) area.getWidth(), (CGFloat) area.getHeight())];
|
|
else
|
|
#endif
|
|
[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;
|