|  |  | @@ -33,59 +33,79 @@ extern NSMenu* createNSMenu (const PopupMenu&, const String& name, int topLevelM | 
		
	
		
			
			|  |  |  | class SystemTrayIconComponent::Pimpl  : private Timer | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | public: | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | Pimpl (SystemTrayIconComponent& iconComp, const Image& im) | 
		
	
		
			
			|  |  |  | : owner (iconComp), statusIcon (imageToNSImage (im)) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | static SystemTrayViewClass cls; | 
		
	
		
			
			|  |  |  | view = [cls.createInstance() init]; | 
		
	
		
			
			|  |  |  | SystemTrayViewClass::setOwner (view, this); | 
		
	
		
			
			|  |  |  | SystemTrayViewClass::setImage (view, statusIcon); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | setIconSize(); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | statusItem = [[[NSStatusBar systemStatusBar] statusItemWithLength: NSSquareStatusItemLength] retain]; | 
		
	
		
			
			|  |  |  | [statusItem setView: view]; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | SystemTrayViewClass::frameChanged (view, SEL(), nullptr); | 
		
	
		
			
			|  |  |  | static ButtonEventForwarderClass cls; | 
		
	
		
			
			|  |  |  | eventForwarder.reset ([cls.createInstance() init]); | 
		
	
		
			
			|  |  |  | ButtonEventForwarderClass::setOwner (eventForwarder.get(), this); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | configureIcon(); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | statusItem.reset ([[[NSStatusBar systemStatusBar] statusItemWithLength: NSSquareStatusItemLength] retain]); | 
		
	
		
			
			|  |  |  | auto button = [statusItem.get() button]; | 
		
	
		
			
			|  |  |  | button.image = statusIcon.get(); | 
		
	
		
			
			|  |  |  | button.target = eventForwarder.get(); | 
		
	
		
			
			|  |  |  | button.action = @selector (handleEvent:); | 
		
	
		
			
			|  |  |  | #if defined (MAC_OS_X_VERSION_10_12) && MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_12 | 
		
	
		
			
			|  |  |  | [button sendActionOn: NSEventMaskLeftMouseDown | NSEventMaskRightMouseDown | NSEventMaskScrollWheel]; | 
		
	
		
			
			|  |  |  | #else | 
		
	
		
			
			|  |  |  | [button sendActionOn: NSLeftMouseDownMask | NSRightMouseDownMask | NSScrollWheelMask]; | 
		
	
		
			
			|  |  |  | #endif | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | [[NSNotificationCenter defaultCenter]  addObserver: view | 
		
	
		
			
			|  |  |  | selector: @selector (frameChanged:) | 
		
	
		
			
			|  |  |  | name: NSWindowDidMoveNotification | 
		
	
		
			
			|  |  |  | object: nil]; | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | void updateIcon (const Image& newImage) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | statusIcon.reset (imageToNSImage (newImage)); | 
		
	
		
			
			|  |  |  | configureIcon(); | 
		
	
		
			
			|  |  |  | [statusItem.get() button].image = statusIcon.get(); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | ~Pimpl() override | 
		
	
		
			
			|  |  |  | void setHighlighted (bool shouldHighlight) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | [[NSNotificationCenter defaultCenter]  removeObserver: view]; | 
		
	
		
			
			|  |  |  | [[NSStatusBar systemStatusBar] removeStatusItem: statusItem]; | 
		
	
		
			
			|  |  |  | SystemTrayViewClass::setOwner (view, nullptr); | 
		
	
		
			
			|  |  |  | SystemTrayViewClass::setImage (view, nil); | 
		
	
		
			
			|  |  |  | [statusItem release]; | 
		
	
		
			
			|  |  |  | [view release]; | 
		
	
		
			
			|  |  |  | [statusIcon release]; | 
		
	
		
			
			|  |  |  | [[statusItem.get() button] setHighlighted: shouldHighlight]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void updateIcon (const Image& newImage) | 
		
	
		
			
			|  |  |  | void showMenu (const PopupMenu& menu) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | [statusIcon release]; | 
		
	
		
			
			|  |  |  | statusIcon = imageToNSImage (newImage); | 
		
	
		
			
			|  |  |  | setIconSize(); | 
		
	
		
			
			|  |  |  | SystemTrayViewClass::setImage (view, statusIcon); | 
		
	
		
			
			|  |  |  | [statusItem setView: view]; | 
		
	
		
			
			|  |  |  | if (NSMenu* m = createNSMenu (menu, "MenuBarItem", -2, -3, true)) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | setHighlighted (true); | 
		
	
		
			
			|  |  |  | stopTimer(); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | // There's currently no good alternative to this... | 
		
	
		
			
			|  |  |  | #if defined __clang__ && defined (MAC_OS_X_VERSION_10_14) && MAC_OS_X_VERSION_MIN_REQUIRED <= MAC_OS_X_VERSION_10_14 | 
		
	
		
			
			|  |  |  | #define IGNORE_POPUP_DEPRECATION 1 | 
		
	
		
			
			|  |  |  | #pragma clang diagnostic push | 
		
	
		
			
			|  |  |  | #pragma clang diagnostic ignored "-Wdeprecated-declarations" | 
		
	
		
			
			|  |  |  | #endif | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | [statusItem.get() popUpStatusItemMenu: m]; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | #if IGNORE_POPUP_DEPRECATION | 
		
	
		
			
			|  |  |  | #pragma clang diagnostic pop | 
		
	
		
			
			|  |  |  | #endif | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | startTimer (1); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void setHighlighted (bool shouldHighlight) | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | NSStatusItem* getStatusItem() | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | isHighlighted = shouldHighlight; | 
		
	
		
			
			|  |  |  | [view setNeedsDisplay: true]; | 
		
	
		
			
			|  |  |  | return statusItem.get(); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void handleStatusItemAction (NSEvent* e) | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | void handleEvent() | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | auto e = [NSApp currentEvent]; | 
		
	
		
			
			|  |  |  | NSEventType type = [e type]; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | const bool isLeft  = (type == NSEventTypeLeftMouseDown  || type == NSEventTypeLeftMouseUp); | 
		
	
		
			
			|  |  |  | const bool isRight = (type == NSEventTypeRightMouseDown || type == NSEventTypeRightMouseUp); | 
		
	
		
			
			|  |  |  | const bool isLeft  = (type == NSEventTypeLeftMouseDown); | 
		
	
		
			
			|  |  |  | const bool isRight = (type == NSEventTypeRightMouseDown); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | if (owner.isCurrentlyBlockedByAnotherModalComponent()) | 
		
	
		
			
			|  |  |  | { | 
		
	
	
		
			
				|  |  | @@ -104,22 +124,22 @@ public: | 
		
	
		
			
			|  |  |  | auto mouseSource = Desktop::getInstance().getMainMouseSource(); | 
		
	
		
			
			|  |  |  | auto pressure = (float) e.pressure; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | if (isLeft || isRight)  // Only mouse up is sent by the OS, so simulate a down/up | 
		
	
		
			
			|  |  |  | if (isLeft || isRight) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | setHighlighted (true); | 
		
	
		
			
			|  |  |  | startTimer (150); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | owner.mouseDown (MouseEvent (mouseSource, {}, | 
		
	
		
			
			|  |  |  | eventMods.withFlags (isLeft ? ModifierKeys::leftButtonModifier | 
		
	
		
			
			|  |  |  | : ModifierKeys::rightButtonModifier), | 
		
	
		
			
			|  |  |  | pressure, MouseInputSource::invalidOrientation, MouseInputSource::invalidRotation, | 
		
	
		
			
			|  |  |  | MouseInputSource::invalidTiltX, MouseInputSource::invalidTiltY, | 
		
	
		
			
			|  |  |  | &owner, &owner, now, {}, now, 1, false)); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | owner.mouseUp (MouseEvent (mouseSource, {}, eventMods.withoutMouseButtons(), pressure, | 
		
	
		
			
			|  |  |  | MouseInputSource::invalidOrientation, MouseInputSource::invalidRotation, | 
		
	
		
			
			|  |  |  | MouseInputSource::invalidTiltX, MouseInputSource::invalidTiltY, | 
		
	
		
			
			|  |  |  | &owner, &owner, now, {}, now, 1, false)); | 
		
	
		
			
			|  |  |  | owner.mouseDown ({ mouseSource, {}, | 
		
	
		
			
			|  |  |  | eventMods.withFlags (isLeft ? ModifierKeys::leftButtonModifier | 
		
	
		
			
			|  |  |  | : ModifierKeys::rightButtonModifier), | 
		
	
		
			
			|  |  |  | pressure, | 
		
	
		
			
			|  |  |  | MouseInputSource::invalidOrientation, MouseInputSource::invalidRotation, | 
		
	
		
			
			|  |  |  | MouseInputSource::invalidTiltX, MouseInputSource::invalidTiltY, | 
		
	
		
			
			|  |  |  | &owner, &owner, now, {}, now, 1, false }); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | owner.mouseUp   ({ mouseSource, {}, | 
		
	
		
			
			|  |  |  | eventMods.withoutMouseButtons(), | 
		
	
		
			
			|  |  |  | pressure, | 
		
	
		
			
			|  |  |  | MouseInputSource::invalidOrientation, MouseInputSource::invalidRotation, | 
		
	
		
			
			|  |  |  | MouseInputSource::invalidTiltX, MouseInputSource::invalidTiltY, | 
		
	
		
			
			|  |  |  | &owner, &owner, now, {}, now, 1, false }); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | else if (type == NSEventTypeMouseMoved) | 
		
	
		
			
			|  |  |  | { | 
		
	
	
		
			
				|  |  | @@ -131,28 +151,12 @@ public: | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void showMenu (const PopupMenu& menu) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | if (NSMenu* m = createNSMenu (menu, "MenuBarItem", -2, -3, true)) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | setHighlighted (true); | 
		
	
		
			
			|  |  |  | stopTimer(); | 
		
	
		
			
			|  |  |  | [statusItem popUpStatusItemMenu: m]; | 
		
	
		
			
			|  |  |  | startTimer (1); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | SystemTrayIconComponent& owner; | 
		
	
		
			
			|  |  |  | NSStatusItem* statusItem = nil; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | private: | 
		
	
		
			
			|  |  |  | NSImage* statusIcon = nil; | 
		
	
		
			
			|  |  |  | NSControl* view = nil; | 
		
	
		
			
			|  |  |  | bool isHighlighted = false; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void setIconSize() | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | void configureIcon() | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | [statusIcon setSize: NSMakeSize (20.0f, 20.0f)]; | 
		
	
		
			
			|  |  |  | [statusIcon.get() setSize: NSMakeSize (20.0f, 20.0f)]; | 
		
	
		
			
			|  |  |  | [statusIcon.get() setTemplate: true]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void timerCallback() override | 
		
	
	
		
			
				|  |  | @@ -161,79 +165,49 @@ private: | 
		
	
		
			
			|  |  |  | setHighlighted (false); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | struct SystemTrayViewClass : public ObjCClass<NSControl> | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | class ButtonEventForwarderClass   : public ObjCClass<NSObject> | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | SystemTrayViewClass()  : ObjCClass<NSControl> ("JUCESystemTrayView_") | 
		
	
		
			
			|  |  |  | public: | 
		
	
		
			
			|  |  |  | ButtonEventForwarderClass() : ObjCClass<NSObject> ("JUCEButtonEventForwarderClass_") | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | addIvar<Pimpl*> ("owner"); | 
		
	
		
			
			|  |  |  | addIvar<NSImage*> ("image"); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | addMethod (@selector (mouseDown:),      handleEventDown, "v@:@"); | 
		
	
		
			
			|  |  |  | addMethod (@selector (rightMouseDown:), handleEventDown, "v@:@"); | 
		
	
		
			
			|  |  |  | addMethod (@selector (drawRect:),       drawRect,        "v@:@"); | 
		
	
		
			
			|  |  |  | addMethod (@selector (frameChanged:),   frameChanged,    "v@:@"); | 
		
	
		
			
			|  |  |  | addMethod (@selector (handleEvent:), handleEvent, "v@:@"); | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | registerClass(); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | static Pimpl* getOwner (id self)                { return getIvar<Pimpl*> (self, "owner"); } | 
		
	
		
			
			|  |  |  | static NSImage* getImage (id self)              { return getIvar<NSImage*> (self, "image"); } | 
		
	
		
			
			|  |  |  | static void setOwner (id self, Pimpl* owner)    { object_setInstanceVariable (self, "owner", owner); } | 
		
	
		
			
			|  |  |  | static void setImage (id self, NSImage* image)  { object_setInstanceVariable (self, "image", image); } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | static void frameChanged (id self, SEL, NSNotification*) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | if (auto* owner = getOwner (self)) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | NSRect r = [[[owner->statusItem view] window] frame]; | 
		
	
		
			
			|  |  |  | NSRect sr = [[[NSScreen screens] objectAtIndex: 0] frame]; | 
		
	
		
			
			|  |  |  | r.origin.y = sr.size.height - r.origin.y - r.size.height; | 
		
	
		
			
			|  |  |  | owner->owner.setBounds (convertToRectInt (r)); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | private: | 
		
	
		
			
			|  |  |  | static void handleEventDown (id self, SEL, NSEvent* e) | 
		
	
		
			
			|  |  |  | static void handleEvent (id self, SEL, id) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | if (auto* owner = getOwner (self)) | 
		
	
		
			
			|  |  |  | owner->handleStatusItemAction (e); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | static void drawRect (id self, SEL, NSRect) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | NSRect bounds = [self bounds]; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | if (auto* owner = getOwner (self)) | 
		
	
		
			
			|  |  |  | [owner->statusItem drawStatusBarBackgroundInRect: bounds | 
		
	
		
			
			|  |  |  | withHighlight: owner->isHighlighted]; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | if (NSImage* const im = getImage (self)) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | NSSize imageSize = [im size]; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | [im drawInRect: NSMakeRect (bounds.origin.x + ((bounds.size.width  - imageSize.width)  / 2.0f), | 
		
	
		
			
			|  |  |  | bounds.origin.y + ((bounds.size.height - imageSize.height) / 2.0f), | 
		
	
		
			
			|  |  |  | imageSize.width, imageSize.height) | 
		
	
		
			
			|  |  |  | fromRect: NSZeroRect | 
		
	
		
			
			|  |  |  | operation: NSCompositingOperationSourceOver | 
		
	
		
			
			|  |  |  | fraction: 1.0f]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | owner->handleEvent(); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | }; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | SystemTrayIconComponent& owner; | 
		
	
		
			
			|  |  |  | std::unique_ptr<NSStatusItem, NSObjectDeleter> statusItem; | 
		
	
		
			
			|  |  |  | std::unique_ptr<NSObject, NSObjectDeleter> eventForwarder; | 
		
	
		
			
			|  |  |  | std::unique_ptr<NSImage, NSObjectDeleter> statusIcon; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (Pimpl) | 
		
	
		
			
			|  |  |  | }; | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | //============================================================================== | 
		
	
		
			
			|  |  |  | void SystemTrayIconComponent::setIconImage (const Image& newImage) | 
		
	
		
			
			|  |  |  | void SystemTrayIconComponent::setIconImage (const Image&, const Image& templateImage) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | if (newImage.isValid()) | 
		
	
		
			
			|  |  |  | if (templateImage.isValid()) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | if (pimpl == nullptr) | 
		
	
		
			
			|  |  |  | pimpl.reset (new Pimpl (*this, newImage)); | 
		
	
		
			
			|  |  |  | pimpl.reset (new Pimpl (*this, templateImage)); | 
		
	
		
			
			|  |  |  | else | 
		
	
		
			
			|  |  |  | pimpl->updateIcon (newImage); | 
		
	
		
			
			|  |  |  | pimpl->updateIcon (templateImage); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | else | 
		
	
		
			
			|  |  |  | { | 
		
	
	
		
			
				|  |  | @@ -264,7 +238,7 @@ void SystemTrayIconComponent::hideInfoBubble() | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void* SystemTrayIconComponent::getNativeHandle() const | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | return pimpl != nullptr ? pimpl->statusItem : nullptr; | 
		
	
		
			
			|  |  |  | return pimpl != nullptr ? pimpl->getStatusItem() : nullptr; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  |  | 
		
	
		
			
			|  |  |  | void SystemTrayIconComponent::showDropdownMenu (const PopupMenu& menu) | 
		
	
	
		
			
				|  |  | 
 |