Improve macOS window behavior.

Don't deactivate the application when the main window is hidden.
Such behavior provides some unwanted windows reordering in the
current workspace when the window is hidden by Cmd+W.

Ignore app activation by applicationDidBecomeActive: notification
for a short period of time after a user notification for other app
instance was received (the system sends them sometimes and the main
window is shown + activated for a wrong instance of the application).
This commit is contained in:
John Preston 2017-04-12 14:58:37 +03:00
parent 734b426518
commit 50ea4e316e
7 changed files with 135 additions and 85 deletions

View File

@ -51,13 +51,17 @@ QString strNeedToRefresh2() {
NSString *fullname; NSString *fullname;
NSURL *app; NSURL *app;
NSImage *icon; NSImage *icon;
} }
@property (nonatomic, retain) NSString *fullname; @property (nonatomic, retain) NSString *fullname;
@property (nonatomic, retain) NSURL *app; @property (nonatomic, retain) NSURL *app;
@property (nonatomic, retain) NSImage *icon; @property (nonatomic, retain) NSImage *icon;
@end
@end // @interface OpenWithApp
@implementation OpenWithApp @implementation OpenWithApp
@synthesize fullname, app, icon; @synthesize fullname, app, icon;
- (void) dealloc { - (void) dealloc {
@ -67,7 +71,7 @@ QString strNeedToRefresh2() {
[super dealloc]; [super dealloc];
} }
@end @end // @implementation OpenWithApp
@interface OpenFileWithInterface : NSObject { @interface OpenFileWithInterface : NSObject {
} }
@ -77,7 +81,7 @@ QString strNeedToRefresh2() {
- (void) itemChosen:(id)sender; - (void) itemChosen:(id)sender;
- (void) dealloc; - (void) dealloc;
@end @end // @interface OpenFileWithInterface
@implementation OpenFileWithInterface { @implementation OpenFileWithInterface {
NSString *toOpen; NSString *toOpen;
@ -89,6 +93,7 @@ QString strNeedToRefresh2() {
NSMutableArray *apps; NSMutableArray *apps;
NSMenu *menu; NSMenu *menu;
} }
- (void) fillAppByUrl:(NSURL*)url bundle:(NSString**)bundle name:(NSString**)name version:(NSString**)version icon:(NSImage**)icon { - (void) fillAppByUrl:(NSURL*)url bundle:(NSString**)bundle name:(NSString**)name version:(NSString**)version icon:(NSImage**)icon {
@ -233,13 +238,13 @@ QString strNeedToRefresh2() {
[super dealloc]; [super dealloc];
} }
@end @end // @implementation OpenFileWithInterface
@interface NSURL(CompareUrls) @interface NSURL(CompareUrls)
- (BOOL) isEquivalent:(NSURL *)aURL; - (BOOL) isEquivalent:(NSURL *)aURL;
@end @end // @interface NSURL(CompareUrls)
@implementation NSURL(CompareUrls) @implementation NSURL(CompareUrls)
@ -253,7 +258,7 @@ QString strNeedToRefresh2() {
return YES; return YES;
} }
@end @end // @implementation NSURL(CompareUrls)
@interface ChooseApplicationDelegate : NSObject<NSOpenSavePanelDelegate> { @interface ChooseApplicationDelegate : NSObject<NSOpenSavePanelDelegate> {
} }
@ -264,7 +269,7 @@ QString strNeedToRefresh2() {
- (void) menuDidClose; - (void) menuDidClose;
- (void) dealloc; - (void) dealloc;
@end @end // @interface ChooseApplicationDelegate
@implementation ChooseApplicationDelegate { @implementation ChooseApplicationDelegate {
BOOL onlyRecommended; BOOL onlyRecommended;
@ -275,6 +280,7 @@ QString strNeedToRefresh2() {
NSImageView *icon; NSImageView *icon;
NSString *recom; NSString *recom;
NSView *accessory; NSView *accessory;
} }
- (id) init:(NSArray *)recommendedApps withPanel:(NSOpenPanel *)creator withSelector:(NSPopUpButton *)menu withGood:(NSTextField *)goodLabel withBad:(NSTextField *)badLabel withIcon:(NSImageView *)badIcon withAccessory:(NSView *)acc { - (id) init:(NSArray *)recommendedApps withPanel:(NSOpenPanel *)creator withSelector:(NSPopUpButton *)menu withGood:(NSTextField *)goodLabel withBad:(NSTextField *)badLabel withIcon:(NSImageView *)badIcon withAccessory:(NSView *)acc {
@ -379,7 +385,7 @@ QString strNeedToRefresh2() {
[super dealloc]; [super dealloc];
} }
@end @end // @implementation ChooseApplicationDelegate
namespace Platform { namespace Platform {
namespace File { namespace File {

View File

@ -22,6 +22,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "platform/platform_main_window.h" #include "platform/platform_main_window.h"
#include "platform/mac/specific_mac_p.h" #include "platform/mac/specific_mac_p.h"
#include "base/timer.h"
namespace Platform { namespace Platform {
@ -65,9 +66,6 @@ public slots:
void psMacDelete(); void psMacDelete();
void psMacSelectAll(); void psMacSelectAll();
private slots:
void onHideAfterFullScreen();
protected: protected:
bool eventFilter(QObject *obj, QEvent *evt) override; bool eventFilter(QObject *obj, QEvent *evt) override;
@ -119,7 +117,7 @@ private:
mutable bool psIdle; mutable bool psIdle;
mutable QTimer psIdleTimer; mutable QTimer psIdleTimer;
QTimer _hideAfterFullScreenTimer; base::Timer _hideAfterFullScreenTimer;
QMenuBar psMainMenu; QMenuBar psMainMenu;
QAction *psLogout = nullptr; QAction *psLogout = nullptr;

View File

@ -37,6 +37,15 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include <IOKit/hidsystem/ev_keymap.h> #include <IOKit/hidsystem/ev_keymap.h>
#include <SPMediaKeyTap.h> #include <SPMediaKeyTap.h>
namespace {
// When we close a window that is fullscreen we first leave the fullscreen
// mode and after that hide the window. This is a timeout for elaving the
// fullscreen mode, after that we'll hide the window no matter what.
constexpr auto kHideAfterFullscreenTimeoutMs = 3000;
} // namespace
@interface MainWindowObserver : NSObject { @interface MainWindowObserver : NSObject {
} }
@ -48,7 +57,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
- (void) windowWillEnterFullScreen:(NSNotification *)aNotification; - (void) windowWillEnterFullScreen:(NSNotification *)aNotification;
- (void) windowWillExitFullScreen:(NSNotification *)aNotification; - (void) windowWillExitFullScreen:(NSNotification *)aNotification;
@end @end // @interface MainWindowObserver
namespace Platform { namespace Platform {
@ -105,8 +114,7 @@ public:
} // namespace Platform } // namespace Platform
@implementation MainWindowObserver { @implementation MainWindowObserver {
MainWindow::Private *_private;
MainWindow::Private *_private;
} }
@ -141,7 +149,7 @@ MainWindow::Private *_private;
_private->willExitFullScreen(); _private->willExitFullScreen();
} }
@end @end // @implementation MainWindowObserver
namespace Platform { namespace Platform {
@ -239,8 +247,7 @@ MainWindow::MainWindow()
trayImg = st::macTrayIcon.instance(QColor(0, 0, 0, 180), dbisOne); trayImg = st::macTrayIcon.instance(QColor(0, 0, 0, 180), dbisOne);
trayImgSel = st::macTrayIcon.instance(QColor(255, 255, 255), dbisOne); trayImgSel = st::macTrayIcon.instance(QColor(255, 255, 255), dbisOne);
_hideAfterFullScreenTimer.setSingleShot(true); _hideAfterFullScreenTimer.setCallback([this] { hideAndDeactivate(); });
connect(&_hideAfterFullScreenTimer, SIGNAL(timeout()), this, SLOT(onHideAfterFullScreen()));
} }
void MainWindow::closeWithoutDestroy() { void MainWindow::closeWithoutDestroy() {
@ -248,7 +255,7 @@ void MainWindow::closeWithoutDestroy() {
auto isFullScreen = (([nsWindow styleMask] & NSFullScreenWindowMask) == NSFullScreenWindowMask); auto isFullScreen = (([nsWindow styleMask] & NSFullScreenWindowMask) == NSFullScreenWindowMask);
if (isFullScreen) { if (isFullScreen) {
_hideAfterFullScreenTimer.start(3000); _hideAfterFullScreenTimer.callOnce(kHideAfterFullscreenTimeoutMs);
[nsWindow toggleFullScreen:nsWindow]; [nsWindow toggleFullScreen:nsWindow];
} else { } else {
hideAndDeactivate(); hideAndDeactivate();
@ -257,8 +264,7 @@ void MainWindow::closeWithoutDestroy() {
void MainWindow::stateChangedHook(Qt::WindowState state) { void MainWindow::stateChangedHook(Qt::WindowState state) {
if (_hideAfterFullScreenTimer.isActive()) { if (_hideAfterFullScreenTimer.isActive()) {
_hideAfterFullScreenTimer.stop(); _hideAfterFullScreenTimer.callOnce(0);
QTimer::singleShot(0, this, SLOT(onHideAfterFullScreen()));
} }
} }
@ -278,14 +284,8 @@ void MainWindow::titleVisibilityChangedHook() {
updateTitleCounter(); updateTitleCounter();
} }
void MainWindow::onHideAfterFullScreen() {
hideAndDeactivate();
}
void MainWindow::hideAndDeactivate() { void MainWindow::hideAndDeactivate() {
hide(); hide();
NSWindow *nsWindow = [reinterpret_cast<NSView*>(winId()) window];
[[NSApplication sharedApplication] hide: nsWindow];
} }
QImage MainWindow::psTrayIcon(bool selected) const { QImage MainWindow::psTrayIcon(bool selected) const {

View File

@ -21,6 +21,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#pragma once #pragma once
#include "platform/platform_notifications_manager.h" #include "platform/platform_notifications_manager.h"
#include "base/weak_unique_ptr.h"
namespace Platform { namespace Platform {
namespace Notifications { namespace Notifications {
@ -28,7 +29,7 @@ namespace Notifications {
bool SkipAudio(); bool SkipAudio();
bool SkipToast(); bool SkipToast();
class Manager : public Window::Notifications::NativeManager { class Manager : public Window::Notifications::NativeManager, public base::enable_weak_from_this {
public: public:
Manager(Window::Notifications::System *system); Manager(Window::Notifications::System *system);
~Manager(); ~Manager();

View File

@ -26,6 +26,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "mainwindow.h" #include "mainwindow.h"
#include "base/task_queue.h" #include "base/task_queue.h"
#include <thread>
#include <Cocoa/Cocoa.h> #include <Cocoa/Cocoa.h>
namespace { namespace {
@ -54,19 +55,18 @@ NSImage *qt_mac_create_nsimage(const QPixmap &pm);
@interface NotificationDelegate : NSObject<NSUserNotificationCenterDelegate> { @interface NotificationDelegate : NSObject<NSUserNotificationCenterDelegate> {
} }
- (id) initWithManager:(std::shared_ptr<Manager*>)manager; - (id) initWithManager:(base::weak_unique_ptr<Manager>)manager;
- (void)userNotificationCenter:(NSUserNotificationCenter *)center didActivateNotification:(NSUserNotification *)notification; - (void) userNotificationCenter:(NSUserNotificationCenter*)center didActivateNotification:(NSUserNotification*)notification;
- (BOOL)userNotificationCenter:(NSUserNotificationCenter *)center shouldPresentNotification:(NSUserNotification *)notification; - (BOOL) userNotificationCenter:(NSUserNotificationCenter*)center shouldPresentNotification:(NSUserNotification*)notification;
@end @end // @interface NotificationDelegate
@implementation NotificationDelegate { @implementation NotificationDelegate {
base::weak_unique_ptr<Manager> _manager;
std::weak_ptr<Manager*> _manager;
} }
- (id) initWithManager:(std::shared_ptr<Manager*>)manager { - (id) initWithManager:(base::weak_unique_ptr<Manager>)manager {
if (self = [super init]) { if (self = [super init]) {
_manager = manager; _manager = manager;
} }
@ -77,14 +77,27 @@ std::weak_ptr<Manager*> _manager;
NSDictionary *notificationUserInfo = [notification userInfo]; NSDictionary *notificationUserInfo = [notification userInfo];
NSNumber *launchIdObject = [notificationUserInfo objectForKey:@"launch"]; NSNumber *launchIdObject = [notificationUserInfo objectForKey:@"launch"];
auto notificationLaunchId = launchIdObject ? [launchIdObject unsignedLongLongValue] : 0ULL; auto notificationLaunchId = launchIdObject ? [launchIdObject unsignedLongLongValue] : 0ULL;
DEBUG_LOG(("Received notification with instance %1").arg(notificationLaunchId)); DEBUG_LOG(("Received notification with instance %1, mine: %2").arg(notificationLaunchId).arg(Global::LaunchId()));
if (notificationLaunchId != Global::LaunchId()) { // other app instance notification if (notificationLaunchId != Global::LaunchId()) { // other app instance notification
base::TaskQueue::Main().Put([] {
// Usually we show and activate main window when the application
// is activated (receives applicationDidBecomeActive: notification).
//
// This is used for window show in Cmd+Tab switching to the application.
//
// But when a notification arrives sometimes macOS still activates the app
// and we receive applicationDidBecomeActive: notification even if the
// notification was sent by another instance of the application. In that case
// we set a flag for a couple of seconds to ignore this app activation.
objc_ignoreApplicationActivationRightNow();
});
return; return;
} }
NSNumber *peerObject = [notificationUserInfo objectForKey:@"peer"]; NSNumber *peerObject = [notificationUserInfo objectForKey:@"peer"];
auto notificationPeerId = peerObject ? [peerObject unsignedLongLongValue] : 0ULL; auto notificationPeerId = peerObject ? [peerObject unsignedLongLongValue] : 0ULL;
if (!notificationPeerId) { if (!notificationPeerId) {
LOG(("App Error: A notification with unknown peer was received"));
return; return;
} }
@ -92,23 +105,27 @@ std::weak_ptr<Manager*> _manager;
auto notificationMsgId = msgObject ? [msgObject intValue] : 0; auto notificationMsgId = msgObject ? [msgObject intValue] : 0;
if (notification.activationType == NSUserNotificationActivationTypeReplied) { if (notification.activationType == NSUserNotificationActivationTypeReplied) {
auto notificationReply = QString::fromUtf8([[[notification response] string] UTF8String]); auto notificationReply = QString::fromUtf8([[[notification response] string] UTF8String]);
if (auto manager = _manager.lock()) { base::TaskQueue::Main().Put([manager = _manager, notificationPeerId, notificationMsgId, notificationReply] {
(*manager)->notificationReplied(notificationPeerId, notificationMsgId, notificationReply); if (manager) {
} manager->notificationReplied(notificationPeerId, notificationMsgId, notificationReply);
}
});
} else if (notification.activationType == NSUserNotificationActivationTypeContentsClicked) { } else if (notification.activationType == NSUserNotificationActivationTypeContentsClicked) {
if (auto manager = _manager.lock()) { base::TaskQueue::Main().Put([manager = _manager, notificationPeerId, notificationMsgId] {
(*manager)->notificationActivated(notificationPeerId, notificationMsgId); if (manager) {
} manager->notificationActivated(notificationPeerId, notificationMsgId);
}
});
} }
[center removeDeliveredNotification: notification]; [center removeDeliveredNotification: notification];
} }
- (BOOL)userNotificationCenter:(NSUserNotificationCenter *)center shouldPresentNotification:(NSUserNotification *)notification { - (BOOL) userNotificationCenter:(NSUserNotificationCenter *)center shouldPresentNotification:(NSUserNotification *)notification {
return YES; return YES;
} }
@end @end // @implementation NotificationDelegate
namespace Platform { namespace Platform {
namespace Notifications { namespace Notifications {
@ -156,6 +173,7 @@ void CustomNotificationShownHook(QWidget *widget) {
class Manager::Private : public QObject, private base::Subscriber { class Manager::Private : public QObject, private base::Subscriber {
public: public:
Private(Manager *manager); Private(Manager *manager);
void showNotification(PeerData *peer, MsgId msgId, const QString &title, const QString &subtitle, const QString &msg, bool hideNameAndPhoto, bool hideReplyButton); void showNotification(PeerData *peer, MsgId msgId, const QString &title, const QString &subtitle, const QString &msg, bool hideNameAndPhoto, bool hideReplyButton);
void clearAll(); void clearAll();
void clearFromHistory(History *history); void clearFromHistory(History *history);
@ -164,14 +182,12 @@ public:
~Private(); ~Private();
private: private:
std::shared_ptr<Manager*> _guarded;
NotificationDelegate *_delegate = nullptr; NotificationDelegate *_delegate = nullptr;
}; };
Manager::Private::Private(Manager *manager) Manager::Private::Private(Manager *manager)
: _guarded(std::make_shared<Manager*>(manager)) : _delegate([[NotificationDelegate alloc] initWithManager:manager]) {
, _delegate([[NotificationDelegate alloc] initWithManager:_guarded]) {
updateDelegate(); updateDelegate();
subscribe(Global::RefWorkMode(), [this](DBIWorkMode mode) { subscribe(Global::RefWorkMode(), [this](DBIWorkMode mode) {
// We need to update the delegate _after_ the tray icon change was done in Qt. // We need to update the delegate _after_ the tray icon change was done in Qt.
@ -225,7 +241,6 @@ void Manager::Private::clearAll() {
[center removeDeliveredNotification:notification]; [center removeDeliveredNotification:notification];
} }
} }
[center removeAllDeliveredNotifications];
} }
void Manager::Private::clearFromHistory(History *history) { void Manager::Private::clearFromHistory(History *history) {

View File

@ -31,6 +31,7 @@ bool objc_idleSupported();
bool objc_idleTime(TimeMs &idleTime); bool objc_idleTime(TimeMs &idleTime);
void objc_start(); void objc_start();
void objc_ignoreApplicationActivationRightNow();
void objc_finish(); void objc_finish();
bool objc_execUpdater(); bool objc_execUpdater();
void objc_execTelegram(const QString &crashreport); void objc_execTelegram(const QString &crashreport);

View File

@ -26,6 +26,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "platform/mac/mac_utilities.h" #include "platform/mac/mac_utilities.h"
#include "styles/style_window.h" #include "styles/style_window.h"
#include "lang.h" #include "lang.h"
#include "base/timer.h"
#include <Cocoa/Cocoa.h> #include <Cocoa/Cocoa.h>
#include <CoreFoundation/CFURL.h> #include <CoreFoundation/CFURL.h>
@ -33,6 +34,12 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include <IOKit/hidsystem/ev_keymap.h> #include <IOKit/hidsystem/ev_keymap.h>
#include <SPMediaKeyTap.h> #include <SPMediaKeyTap.h>
namespace {
constexpr auto kIgnoreActivationTimeoutMs = 1500;
} // namespace
using Platform::Q2NSString; using Platform::Q2NSString;
using Platform::NSlang; using Platform::NSlang;
using Platform::NS2QString; using Platform::NS2QString;
@ -48,10 +55,11 @@ using Platform::NS2QString;
- (id)debugQuickLookObject; - (id)debugQuickLookObject;
@end @end // @interface qVisualize
@implementation qVisualize { @implementation qVisualize {
NSString *value; NSString *value;
} }
+ (id)bytearr:(const QByteArray &)arr { + (id)bytearr:(const QByteArray &)arr {
@ -78,59 +86,68 @@ using Platform::NS2QString;
return value; return value;
} }
@end @end // @implementation qVisualize
@interface ApplicationDelegate : NSObject<NSApplicationDelegate> { @interface ApplicationDelegate : NSObject<NSApplicationDelegate> {
SPMediaKeyTap *keyTap;
BOOL watchingMediaKeys;
} }
- (BOOL)applicationShouldHandleReopen:(NSApplication *)theApplication hasVisibleWindows:(BOOL)flag; - (BOOL) applicationShouldHandleReopen:(NSApplication *)theApplication hasVisibleWindows:(BOOL)flag;
- (void)applicationDidFinishLaunching:(NSNotification *)aNotification; - (void) applicationDidFinishLaunching:(NSNotification *)aNotification;
- (void)applicationDidBecomeActive:(NSNotification *)aNotification; - (void) applicationDidBecomeActive:(NSNotification *)aNotification;
- (void)receiveWakeNote:(NSNotification*)note; - (void) receiveWakeNote:(NSNotification*)note;
- (void)setWatchingMediaKeys:(BOOL)watching;
- (BOOL)isWatchingMediaKeys;
- (void)mediaKeyTap:(SPMediaKeyTap*)keyTap receivedMediaKeyEvent:(NSEvent*)event;
@end - (void) setWatchingMediaKeys:(bool)watching;
- (bool) isWatchingMediaKeys;
- (void) mediaKeyTap:(SPMediaKeyTap*)keyTap receivedMediaKeyEvent:(NSEvent*)event;
- (void) ignoreApplicationActivationRightNow;
@end // @interface ApplicationDelegate
ApplicationDelegate *_sharedDelegate = nil; ApplicationDelegate *_sharedDelegate = nil;
@implementation ApplicationDelegate { @implementation ApplicationDelegate {
SPMediaKeyTap *_keyTap;
bool _watchingMediaKeys;
bool _ignoreActivation;
base::Timer _ignoreActivationStop;
} }
- (BOOL)applicationShouldHandleReopen:(NSApplication *)theApplication hasVisibleWindows:(BOOL)flag { - (BOOL) applicationShouldHandleReopen:(NSApplication *)theApplication hasVisibleWindows:(BOOL)flag {
if (App::wnd() && App::wnd()->isHidden()) App::wnd()->showFromTray(); if (App::wnd() && App::wnd()->isHidden()) App::wnd()->showFromTray();
return YES; return YES;
} }
- (void)applicationDidFinishLaunching:(NSNotification *)aNotification { - (void) applicationDidFinishLaunching:(NSNotification *)aNotification {
keyTap = nullptr; _keyTap = nullptr;
watchingMediaKeys = false; _watchingMediaKeys = false;
_ignoreActivation = false;
_ignoreActivationStop.setCallback([self] {
_ignoreActivation = false;
});
#ifndef OS_MAC_STORE #ifndef OS_MAC_STORE
if ([SPMediaKeyTap usesGlobalMediaKeyTap]) { if ([SPMediaKeyTap usesGlobalMediaKeyTap]) {
keyTap = [[SPMediaKeyTap alloc] initWithDelegate:self]; _keyTap = [[SPMediaKeyTap alloc] initWithDelegate:self];
} else { } else {
LOG(("Media key monitoring disabled")); LOG(("Media key monitoring disabled"));
} }
#endif // else for !OS_MAC_STORE #endif // else for !OS_MAC_STORE
} }
- (void)applicationDidBecomeActive:(NSNotification *)aNotification { - (void) applicationDidBecomeActive:(NSNotification *)aNotification {
if (auto messenger = Messenger::InstancePointer()) { if (auto messenger = Messenger::InstancePointer()) {
messenger->handleAppActivated(); if (!_ignoreActivation) {
if (auto window = App::wnd()) { messenger->handleAppActivated();
if (window->isHidden()) { if (auto window = App::wnd()) {
window->showFromTray(); if (window->isHidden()) {
window->showFromTray();
}
} }
} }
} }
} }
- (void)receiveWakeNote:(NSNotification*)aNotification { - (void) receiveWakeNote:(NSNotification*)aNotification {
if (auto messenger = Messenger::InstancePointer()) { if (auto messenger = Messenger::InstancePointer()) {
messenger->checkLocalTime(); messenger->checkLocalTime();
} }
@ -139,38 +156,43 @@ ApplicationDelegate *_sharedDelegate = nil;
Media::Player::DetachFromDeviceByTimer(); Media::Player::DetachFromDeviceByTimer();
} }
- (void)setWatchingMediaKeys:(BOOL)watching { - (void) setWatchingMediaKeys:(bool)watching {
if (watchingMediaKeys != watching) { if (_watchingMediaKeys != watching) {
watchingMediaKeys = watching; _watchingMediaKeys = watching;
if (keyTap) { if (_keyTap) {
#ifndef OS_MAC_STORE #ifndef OS_MAC_STORE
if (watchingMediaKeys) { if (_watchingMediaKeys) {
[keyTap startWatchingMediaKeys]; [_keyTap startWatchingMediaKeys];
} else { } else {
[keyTap stopWatchingMediaKeys]; [_keyTap stopWatchingMediaKeys];
} }
#endif // else for !OS_MAC_STORE #endif // else for !OS_MAC_STORE
} }
} }
} }
- (BOOL)isWatchingMediaKeys { - (bool) isWatchingMediaKeys {
return watchingMediaKeys; return _watchingMediaKeys;
} }
- (void)mediaKeyTap:(SPMediaKeyTap*)keyTap receivedMediaKeyEvent:(NSEvent*)e { - (void) mediaKeyTap:(SPMediaKeyTap*)keyTap receivedMediaKeyEvent:(NSEvent*)e {
if (e && [e type] == NSSystemDefined && [e subtype] == SPSystemDefinedEventMediaKeys) { if (e && [e type] == NSSystemDefined && [e subtype] == SPSystemDefinedEventMediaKeys) {
objc_handleMediaKeyEvent(e); objc_handleMediaKeyEvent(e);
} }
} }
@end - (void) ignoreApplicationActivationRightNow {
_ignoreActivation = true;
_ignoreActivationStop.callOnce(kIgnoreActivationTimeoutMs);
}
@end // @implementation ApplicationDelegate
namespace Platform { namespace Platform {
void SetWatchingMediaKeys(bool watching) { void SetWatchingMediaKeys(bool watching) {
if (_sharedDelegate) { if (_sharedDelegate) {
[_sharedDelegate setWatchingMediaKeys:(watching ? YES : NO)]; [_sharedDelegate setWatchingMediaKeys:watching];
} }
} }
@ -327,12 +349,19 @@ void objc_start() {
name: NSWorkspaceDidWakeNotification object: NULL]; name: NSWorkspaceDidWakeNotification object: NULL];
} }
void objc_ignoreApplicationActivationRightNow() {
if (_sharedDelegate) {
[_sharedDelegate ignoreApplicationActivationRightNow];
}
}
namespace { namespace {
NSURL *_downloadPathUrl = nil; NSURL *_downloadPathUrl = nil;
} }
void objc_finish() { void objc_finish() {
[_sharedDelegate release]; [_sharedDelegate release];
_sharedDelegate = nil;
if (_downloadPathUrl) { if (_downloadPathUrl) {
[_downloadPathUrl stopAccessingSecurityScopedResource]; [_downloadPathUrl stopAccessingSecurityScopedResource];
_downloadPathUrl = nil; _downloadPathUrl = nil;