macOS: Modernize QCocoaSystemTrayIcon
The code had not been touched in a very long time and was overgrown with weeds from pre-QPA times. We no longer maintain an indirection through QSystemTrayIconSys, which was a remnant from Qt 4 times. The Objective-C helper class used for callbacks has been slimmed down to just a simple delegate, with the actual work done in the QCocoaSystemTrayIcon implementation, further reducing indirection. We no longer use a custom NSView for the status bar item, something that has been deprecated for a long time. Instead we set properties on the NSStatusItem's button. This gives us automatic support for drawing the icon with the right highlight, including in dark mode. Finally, the code has been updated to modern Objective-C syntax. Change-Id: I59706081f6b179035b8216a7a6ebc08a47cec127 Fixes: QTBUG-77189 Reviewed-by: Volker Hilsheimer <volker.hilsheimer@qt.io>
This commit is contained in:
parent
54f8be6cc0
commit
395e2d9bc4
@ -49,14 +49,23 @@
|
|||||||
#include "QtCore/qstring.h"
|
#include "QtCore/qstring.h"
|
||||||
#include "QtGui/qpa/qplatformsystemtrayicon.h"
|
#include "QtGui/qpa/qplatformsystemtrayicon.h"
|
||||||
|
|
||||||
QT_BEGIN_NAMESPACE
|
#include "qcocoamenu.h"
|
||||||
|
|
||||||
class QSystemTrayIconSys;
|
QT_FORWARD_DECLARE_CLASS(QCocoaSystemTrayIcon);
|
||||||
|
|
||||||
|
@interface QT_MANGLE_NAMESPACE(QStatusItemDelegate) : NSObject <NSUserNotificationCenterDelegate>
|
||||||
|
- (instancetype)initWithSysTray:(QCocoaSystemTrayIcon *)platformSystemTray;
|
||||||
|
@property (nonatomic, assign) QCocoaSystemTrayIcon *platformSystemTray;
|
||||||
|
@end
|
||||||
|
|
||||||
|
QT_NAMESPACE_ALIAS_OBJC_CLASS(QStatusItemDelegate);
|
||||||
|
|
||||||
|
QT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
class Q_GUI_EXPORT QCocoaSystemTrayIcon : public QPlatformSystemTrayIcon
|
class Q_GUI_EXPORT QCocoaSystemTrayIcon : public QPlatformSystemTrayIcon
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
QCocoaSystemTrayIcon() : m_sys(nullptr) {}
|
QCocoaSystemTrayIcon() {}
|
||||||
|
|
||||||
void init() override;
|
void init() override;
|
||||||
void cleanup() override;
|
void cleanup() override;
|
||||||
@ -70,8 +79,12 @@ public:
|
|||||||
bool isSystemTrayAvailable() const override;
|
bool isSystemTrayAvailable() const override;
|
||||||
bool supportsMessages() const override;
|
bool supportsMessages() const override;
|
||||||
|
|
||||||
|
void statusItemClicked();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QSystemTrayIconSys *m_sys;
|
NSStatusItem *m_statusItem = nullptr;
|
||||||
|
QStatusItemDelegate *m_delegate = nullptr;
|
||||||
|
QCocoaMenu *m_menu = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
QT_END_NAMESPACE
|
QT_END_NAMESPACE
|
||||||
|
@ -92,67 +92,46 @@
|
|||||||
|
|
||||||
#import <AppKit/AppKit.h>
|
#import <AppKit/AppKit.h>
|
||||||
|
|
||||||
QT_USE_NAMESPACE
|
|
||||||
|
|
||||||
@interface QT_MANGLE_NAMESPACE(QNSStatusItem) : NSObject <NSUserNotificationCenterDelegate>
|
|
||||||
@property (nonatomic, assign) QCocoaMenu *menu;
|
|
||||||
@property (nonatomic, assign) QIcon icon;
|
|
||||||
@property (nonatomic, readonly) NSStatusItem *item;
|
|
||||||
@property (nonatomic, readonly) QRectF geometry;
|
|
||||||
- (instancetype)initWithSysTray:(QCocoaSystemTrayIcon *)systray;
|
|
||||||
- (void)triggerSelector:(id)sender button:(Qt::MouseButton)mouseButton;
|
|
||||||
- (void)doubleClickSelector:(id)sender;
|
|
||||||
@end
|
|
||||||
|
|
||||||
QT_NAMESPACE_ALIAS_OBJC_CLASS(QNSStatusItem);
|
|
||||||
|
|
||||||
@interface QT_MANGLE_NAMESPACE(QNSImageView) : NSImageView
|
|
||||||
@property (nonatomic, assign) BOOL down;
|
|
||||||
@property (nonatomic, assign) QNSStatusItem *parent;
|
|
||||||
@end
|
|
||||||
|
|
||||||
QT_NAMESPACE_ALIAS_OBJC_CLASS(QNSImageView);
|
|
||||||
|
|
||||||
QT_BEGIN_NAMESPACE
|
QT_BEGIN_NAMESPACE
|
||||||
class QSystemTrayIconSys
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
QSystemTrayIconSys(QCocoaSystemTrayIcon *sys) {
|
|
||||||
item = [[QNSStatusItem alloc] initWithSysTray:sys];
|
|
||||||
NSUserNotificationCenter.defaultUserNotificationCenter.delegate = item;
|
|
||||||
}
|
|
||||||
~QSystemTrayIconSys() {
|
|
||||||
[[[item item] view] setHidden: YES];
|
|
||||||
NSUserNotificationCenter *center = NSUserNotificationCenter.defaultUserNotificationCenter;
|
|
||||||
if (center.delegate == item)
|
|
||||||
center.delegate = nil;
|
|
||||||
[item release];
|
|
||||||
}
|
|
||||||
QNSStatusItem *item;
|
|
||||||
};
|
|
||||||
|
|
||||||
void QCocoaSystemTrayIcon::init()
|
void QCocoaSystemTrayIcon::init()
|
||||||
{
|
{
|
||||||
if (!m_sys)
|
m_statusItem = [[NSStatusBar.systemStatusBar statusItemWithLength:NSSquareStatusItemLength] retain];
|
||||||
m_sys = new QSystemTrayIconSys(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
QRect QCocoaSystemTrayIcon::geometry() const
|
m_delegate = [[QStatusItemDelegate alloc] initWithSysTray:this];
|
||||||
{
|
|
||||||
if (!m_sys)
|
|
||||||
return QRect();
|
|
||||||
|
|
||||||
const QRectF geom = [m_sys->item geometry];
|
m_statusItem.button.target = m_delegate;
|
||||||
if (!geom.isNull())
|
m_statusItem.button.action = @selector(statusItemClicked);
|
||||||
return geom.toRect();
|
[m_statusItem.button sendActionOn:NSEventMaskLeftMouseUp | NSEventMaskRightMouseUp | NSEventMaskOtherMouseUp];
|
||||||
else
|
|
||||||
return QRect();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void QCocoaSystemTrayIcon::cleanup()
|
void QCocoaSystemTrayIcon::cleanup()
|
||||||
{
|
{
|
||||||
delete m_sys;
|
NSUserNotificationCenter *center = NSUserNotificationCenter.defaultUserNotificationCenter;
|
||||||
m_sys = nullptr;
|
if (center.delegate == m_delegate)
|
||||||
|
center.delegate = nil;
|
||||||
|
|
||||||
|
[NSStatusBar.systemStatusBar removeStatusItem:m_statusItem];
|
||||||
|
[m_statusItem release];
|
||||||
|
m_statusItem = nil;
|
||||||
|
|
||||||
|
[m_delegate release];
|
||||||
|
m_delegate = nil;
|
||||||
|
|
||||||
|
m_menu = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
QRect QCocoaSystemTrayIcon::geometry() const
|
||||||
|
{
|
||||||
|
if (!m_statusItem)
|
||||||
|
return QRect();
|
||||||
|
|
||||||
|
if (NSWindow *window = m_statusItem.button.window) {
|
||||||
|
if (QCocoaScreen *screen = QCocoaScreen::get(window.screen))
|
||||||
|
return screen->mapFromNative(window.frame).toRect();
|
||||||
|
}
|
||||||
|
|
||||||
|
return QRect();
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool heightCompareFunction (QSize a, QSize b) { return (a.height() < b.height()); }
|
static bool heightCompareFunction (QSize a, QSize b) { return (a.height() < b.height()); }
|
||||||
@ -165,17 +144,15 @@ static QList<QSize> sortByHeight(const QList<QSize> &sizes)
|
|||||||
|
|
||||||
void QCocoaSystemTrayIcon::updateIcon(const QIcon &icon)
|
void QCocoaSystemTrayIcon::updateIcon(const QIcon &icon)
|
||||||
{
|
{
|
||||||
if (!m_sys)
|
if (!m_statusItem)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
m_sys->item.icon = icon;
|
// The recommended maximum title bar icon height is 18 points
|
||||||
|
|
||||||
// The reccomended maximum title bar icon height is 18 points
|
|
||||||
// (device independent pixels). The menu height on past and
|
// (device independent pixels). The menu height on past and
|
||||||
// current OS X versions is 22 points. Provide some future-proofing
|
// current OS X versions is 22 points. Provide some future-proofing
|
||||||
// by deriving the icon height from the menu height.
|
// by deriving the icon height from the menu height.
|
||||||
const int padding = 4;
|
const int padding = 4;
|
||||||
const int menuHeight = [[NSStatusBar systemStatusBar] thickness];
|
const int menuHeight = NSStatusBar.systemStatusBar.thickness;
|
||||||
const int maxImageHeight = menuHeight - padding;
|
const int maxImageHeight = menuHeight - padding;
|
||||||
|
|
||||||
// Select pixmap based on the device pixel height. Ideally we would use
|
// Select pixmap based on the device pixel height. Ideally we would use
|
||||||
@ -230,27 +207,26 @@ void QCocoaSystemTrayIcon::updateIcon(const QIcon &icon)
|
|||||||
|
|
||||||
auto *nsimage = [NSImage imageFromQImage:fullHeightPixmap.toImage()];
|
auto *nsimage = [NSImage imageFromQImage:fullHeightPixmap.toImage()];
|
||||||
[nsimage setTemplate:icon.isMask()];
|
[nsimage setTemplate:icon.isMask()];
|
||||||
[(NSImageView*)[[m_sys->item item] view] setImage: nsimage];
|
m_statusItem.button.image = nsimage;
|
||||||
|
m_statusItem.button.imageScaling = NSImageScaleProportionallyDown;
|
||||||
}
|
}
|
||||||
|
|
||||||
void QCocoaSystemTrayIcon::updateMenu(QPlatformMenu *menu)
|
void QCocoaSystemTrayIcon::updateMenu(QPlatformMenu *menu)
|
||||||
{
|
{
|
||||||
if (!m_sys)
|
// We don't set the menu property of the NSStatusItem here,
|
||||||
return;
|
// as that would prevent us from receiving the action for the
|
||||||
|
// click, and we wouldn't be able to emit the activated signal.
|
||||||
m_sys->item.menu = static_cast<QCocoaMenu *>(menu);
|
// Instead we show the menu manually when the status item is
|
||||||
if (menu && [m_sys->item.menu->nsMenu() numberOfItems] > 0) {
|
// clicked.
|
||||||
[[m_sys->item item] setHighlightMode:YES];
|
m_menu = static_cast<QCocoaMenu *>(menu);
|
||||||
} else {
|
|
||||||
[[m_sys->item item] setHighlightMode:NO];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void QCocoaSystemTrayIcon::updateToolTip(const QString &toolTip)
|
void QCocoaSystemTrayIcon::updateToolTip(const QString &toolTip)
|
||||||
{
|
{
|
||||||
if (!m_sys)
|
if (!m_statusItem)
|
||||||
return;
|
return;
|
||||||
[[[m_sys->item item] view] setToolTip:toolTip.toNSString()];
|
|
||||||
|
m_statusItem.button.toolTip = toolTip.toNSString();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool QCocoaSystemTrayIcon::isSystemTrayAvailable() const
|
bool QCocoaSystemTrayIcon::isSystemTrayAvailable() const
|
||||||
@ -266,175 +242,83 @@ bool QCocoaSystemTrayIcon::supportsMessages() const
|
|||||||
void QCocoaSystemTrayIcon::showMessage(const QString &title, const QString &message,
|
void QCocoaSystemTrayIcon::showMessage(const QString &title, const QString &message,
|
||||||
const QIcon& icon, MessageIcon, int msecs)
|
const QIcon& icon, MessageIcon, int msecs)
|
||||||
{
|
{
|
||||||
if (!m_sys)
|
if (!m_statusItem)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
NSUserNotification *notification = [[NSUserNotification alloc] init];
|
auto *notification = [[NSUserNotification alloc] init];
|
||||||
notification.title = [NSString stringWithUTF8String:title.toUtf8().data()];
|
notification.title = title.toNSString();
|
||||||
notification.informativeText = [NSString stringWithUTF8String:message.toUtf8().data()];
|
notification.informativeText = message.toNSString();
|
||||||
notification.contentImage = [NSImage imageFromQIcon:icon];
|
notification.contentImage = [NSImage imageFromQIcon:icon];
|
||||||
|
|
||||||
NSUserNotificationCenter *center = NSUserNotificationCenter.defaultUserNotificationCenter;
|
NSUserNotificationCenter *center = NSUserNotificationCenter.defaultUserNotificationCenter;
|
||||||
center.delegate = m_sys->item;
|
center.delegate = m_delegate;
|
||||||
[center deliverNotification:notification];
|
|
||||||
|
[center deliverNotification:[notification autorelease]];
|
||||||
|
|
||||||
if (msecs) {
|
if (msecs) {
|
||||||
NSTimeInterval timeout = msecs / 1000.0;
|
NSTimeInterval timeout = msecs / 1000.0;
|
||||||
[center performSelector:@selector(removeDeliveredNotification:) withObject:notification afterDelay:timeout];
|
[center performSelector:@selector(removeDeliveredNotification:) withObject:notification afterDelay:timeout];
|
||||||
}
|
}
|
||||||
[notification release];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void QCocoaSystemTrayIcon::statusItemClicked()
|
||||||
|
{
|
||||||
|
auto *mouseEvent = NSApp.currentEvent;
|
||||||
|
|
||||||
|
auto activationReason = QPlatformSystemTrayIcon::Unknown;
|
||||||
|
|
||||||
|
if (mouseEvent.clickCount == 2) {
|
||||||
|
activationReason = QPlatformSystemTrayIcon::DoubleClick;
|
||||||
|
} else {
|
||||||
|
auto mouseButton = cocoaButton2QtButton(mouseEvent);
|
||||||
|
if (mouseButton == Qt::MidButton)
|
||||||
|
activationReason = QPlatformSystemTrayIcon::MiddleClick;
|
||||||
|
else if (mouseButton == Qt::RightButton)
|
||||||
|
activationReason = QPlatformSystemTrayIcon::Context;
|
||||||
|
else
|
||||||
|
activationReason = QPlatformSystemTrayIcon::Trigger;
|
||||||
|
}
|
||||||
|
|
||||||
|
emit activated(activationReason);
|
||||||
|
|
||||||
|
if (NSMenu *menu = m_menu ? m_menu->nsMenu() : nil)
|
||||||
|
[m_statusItem popUpStatusItemMenu:menu];
|
||||||
|
}
|
||||||
|
|
||||||
QT_END_NAMESPACE
|
QT_END_NAMESPACE
|
||||||
|
|
||||||
@implementation NSStatusItem (Qt)
|
@implementation QStatusItemDelegate
|
||||||
@end
|
|
||||||
|
- (instancetype)initWithSysTray:(QCocoaSystemTrayIcon *)platformSystemTray
|
||||||
|
{
|
||||||
|
if ((self = [super init]))
|
||||||
|
self.platformSystemTray = platformSystemTray;
|
||||||
|
|
||||||
@implementation QNSImageView
|
|
||||||
- (instancetype)initWithParent:(QNSStatusItem *)myParent {
|
|
||||||
self = [super init];
|
|
||||||
self.parent = myParent;
|
|
||||||
self.down = NO;
|
|
||||||
return self;
|
return self;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)menuTrackingDone:(NSNotification *)__unused notification
|
- (void)dealloc
|
||||||
{
|
{
|
||||||
self.down = NO;
|
self.platformSystemTray = nullptr;
|
||||||
|
|
||||||
[self setNeedsDisplay:YES];
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)mousePressed:(NSEvent *)mouseEvent
|
|
||||||
{
|
|
||||||
self.down = YES;
|
|
||||||
int clickCount = [mouseEvent clickCount];
|
|
||||||
[self setNeedsDisplay:YES];
|
|
||||||
|
|
||||||
if (clickCount == 2) {
|
|
||||||
[self menuTrackingDone:nil];
|
|
||||||
[self.parent doubleClickSelector:self];
|
|
||||||
} else {
|
|
||||||
[self.parent triggerSelector:self button:cocoaButton2QtButton(mouseEvent)];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)mouseDown:(NSEvent *)mouseEvent
|
|
||||||
{
|
|
||||||
[self mousePressed:mouseEvent];
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)mouseUp:(NSEvent *)mouseEvent
|
|
||||||
{
|
|
||||||
Q_UNUSED(mouseEvent);
|
|
||||||
[self menuTrackingDone:nil];
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)rightMouseDown:(NSEvent *)mouseEvent
|
|
||||||
{
|
|
||||||
[self mousePressed:mouseEvent];
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)rightMouseUp:(NSEvent *)mouseEvent
|
|
||||||
{
|
|
||||||
Q_UNUSED(mouseEvent);
|
|
||||||
[self menuTrackingDone:nil];
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)otherMouseDown:(NSEvent *)mouseEvent
|
|
||||||
{
|
|
||||||
[self mousePressed:mouseEvent];
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)otherMouseUp:(NSEvent *)mouseEvent
|
|
||||||
{
|
|
||||||
Q_UNUSED(mouseEvent);
|
|
||||||
[self menuTrackingDone:nil];
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)drawRect:(NSRect)rect {
|
|
||||||
[[self.parent item] drawStatusBarBackgroundInRect:rect withHighlight:self.down];
|
|
||||||
[super drawRect:rect];
|
|
||||||
}
|
|
||||||
@end
|
|
||||||
|
|
||||||
@implementation QNSStatusItem {
|
|
||||||
QCocoaSystemTrayIcon *systray;
|
|
||||||
NSStatusItem *item;
|
|
||||||
QNSImageView *imageCell;
|
|
||||||
}
|
|
||||||
|
|
||||||
@synthesize menu = menu;
|
|
||||||
@synthesize icon = icon;
|
|
||||||
|
|
||||||
- (instancetype)initWithSysTray:(QCocoaSystemTrayIcon *)sys
|
|
||||||
{
|
|
||||||
self = [super init];
|
|
||||||
if (self) {
|
|
||||||
item = [[[NSStatusBar systemStatusBar] statusItemWithLength:NSSquareStatusItemLength] retain];
|
|
||||||
menu = nullptr;
|
|
||||||
systray = sys;
|
|
||||||
imageCell = [[QNSImageView alloc] initWithParent:self];
|
|
||||||
[item setView: imageCell];
|
|
||||||
}
|
|
||||||
return self;
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)dealloc {
|
|
||||||
[[NSStatusBar systemStatusBar] removeStatusItem:item];
|
|
||||||
[[NSNotificationCenter defaultCenter] removeObserver:imageCell];
|
|
||||||
imageCell.parent = nil;
|
|
||||||
[imageCell release];
|
|
||||||
[item release];
|
|
||||||
[super dealloc];
|
[super dealloc];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSStatusItem *)item {
|
- (void)statusItemClicked
|
||||||
return item;
|
{
|
||||||
|
self.platformSystemTray->statusItemClicked();
|
||||||
}
|
}
|
||||||
|
|
||||||
- (QRectF)geometry {
|
- (BOOL)userNotificationCenter:(NSUserNotificationCenter *)center shouldPresentNotification:(NSUserNotification *)notification
|
||||||
if (NSWindow *window = item.view.window) {
|
{
|
||||||
if (QCocoaScreen *screen = QCocoaScreen::get(window.screen))
|
|
||||||
return screen->mapFromNative(window.frame);
|
|
||||||
}
|
|
||||||
return QRectF();
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)triggerSelector:(id)sender button:(Qt::MouseButton)mouseButton {
|
|
||||||
Q_UNUSED(sender);
|
|
||||||
if (!systray)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (mouseButton == Qt::MidButton)
|
|
||||||
emit systray->activated(QPlatformSystemTrayIcon::MiddleClick);
|
|
||||||
else
|
|
||||||
emit systray->activated(QPlatformSystemTrayIcon::Trigger);
|
|
||||||
|
|
||||||
if (menu) {
|
|
||||||
NSMenu *m = menu->nsMenu();
|
|
||||||
[[NSNotificationCenter defaultCenter] addObserver:imageCell
|
|
||||||
selector:@selector(menuTrackingDone:)
|
|
||||||
name:NSMenuDidEndTrackingNotification
|
|
||||||
object:m];
|
|
||||||
[item popUpStatusItemMenu: m];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)doubleClickSelector:(id)sender {
|
|
||||||
Q_UNUSED(sender);
|
|
||||||
if (!systray)
|
|
||||||
return;
|
|
||||||
emit systray->activated(QPlatformSystemTrayIcon::DoubleClick);
|
|
||||||
}
|
|
||||||
|
|
||||||
- (BOOL)userNotificationCenter:(NSUserNotificationCenter *)center shouldPresentNotification:(NSUserNotification *)notification {
|
|
||||||
Q_UNUSED(center);
|
Q_UNUSED(center);
|
||||||
Q_UNUSED(notification);
|
Q_UNUSED(notification);
|
||||||
return YES;
|
return YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)userNotificationCenter:(NSUserNotificationCenter *)center didActivateNotification:(NSUserNotification *)notification {
|
- (void)userNotificationCenter:(NSUserNotificationCenter *)center didActivateNotification:(NSUserNotification *)notification
|
||||||
|
{
|
||||||
[center removeDeliveredNotification:notification];
|
[center removeDeliveredNotification:notification];
|
||||||
emit systray->messageClicked();
|
emit self.platformSystemTray->messageClicked();
|
||||||
}
|
}
|
||||||
|
|
||||||
@end
|
@end
|
||||||
|
Loading…
Reference in New Issue
Block a user