Accept additional notification actions

Change to the existing API definition: The first action with type `button` seen will be displayed on the notification, the rest listed as additional actions (shown when holding down on the primary action button)
This commit is contained in:
Zhuo Lu 2018-01-15 20:50:03 -08:00
parent 647d04cf92
commit 844ee0a3f4
3 changed files with 48 additions and 8 deletions

View file

@ -9,6 +9,7 @@
#include <string> #include <string>
#include <vector> #include <vector>
#include <map>
#include "base/mac/scoped_nsobject.h" #include "base/mac/scoped_nsobject.h"
#include "brightray/browser/notification.h" #include "brightray/browser/notification.h"
@ -28,6 +29,7 @@ class CocoaNotification : public Notification {
void NotificationDisplayed(); void NotificationDisplayed();
void NotificationReplied(const std::string& reply); void NotificationReplied(const std::string& reply);
void NotificationButtonClicked(); void NotificationButtonClicked();
void NotificationAdditionalActionClicked(NSUserNotificationAction* action);
NSUserNotification* notification() const { return notification_; } NSUserNotification* notification() const { return notification_; }
@ -35,7 +37,8 @@ class CocoaNotification : public Notification {
void LogAction(const char* action); void LogAction(const char* action);
base::scoped_nsobject<NSUserNotification> notification_; base::scoped_nsobject<NSUserNotification> notification_;
int action_index_; std::map<std::string, unsigned> additional_action_indices_;
unsigned action_index_;
DISALLOW_COPY_AND_ASSIGN(CocoaNotification); DISALLOW_COPY_AND_ASSIGN(CocoaNotification);
}; };

View file

@ -59,14 +59,31 @@ void CocoaNotification::Show(const NotificationOptions& options) {
[notification_ setHasActionButton:false]; [notification_ setHasActionButton:false];
int i = 0; int i = 0;
action_index_ = UINT_MAX;
NSMutableArray* additionalActions = [[[NSMutableArray alloc] init] autorelease];
for (const auto& action : options.actions) { for (const auto& action : options.actions) {
if (action.type == base::ASCIIToUTF16("button")) { if (action.type == base::ASCIIToUTF16("button")) {
[notification_ setHasActionButton:true]; if (action_index_ == UINT_MAX) {
[notification_ setActionButtonTitle:base::SysUTF16ToNSString(action.text)]; // First button observed is the displayed action
action_index_ = i; [notification_ setHasActionButton:true];
[notification_ setActionButtonTitle:base::SysUTF16ToNSString(action.text)];
action_index_ = i;
} else {
// All of the rest are appended to the list of additional actions
NSString* actionIdentifier = [NSString stringWithFormat:@"%@Action%d", identifier, i];
NSUserNotificationAction* notificationAction =
[NSUserNotificationAction actionWithIdentifier:actionIdentifier
title:base::SysUTF16ToNSString(action.text)];
[additionalActions addObject:notificationAction];
additional_action_indices_.insert(std::make_pair(base::SysNSStringToUTF8(actionIdentifier), i));
}
} }
i++; i++;
} }
if ([additionalActions count] > 0 &&
[notification_ respondsToSelector:@selector(setAdditionalActions:)]) {
[notification_ setAdditionalActions:additionalActions]; // Requires macOS 10.10
}
if (options.has_reply) { if (options.has_reply) {
[notification_ setResponsePlaceholder:base::SysUTF16ToNSString(options.reply_placeholder)]; [notification_ setResponsePlaceholder:base::SysUTF16ToNSString(options.reply_placeholder)];
@ -108,6 +125,23 @@ void CocoaNotification::NotificationButtonClicked() {
this->LogAction("button clicked"); this->LogAction("button clicked");
} }
void CocoaNotification::NotificationAdditionalActionClicked(NSUserNotificationAction* action) {
if (delegate()) {
unsigned index = action_index_;
std::string identifier = base::SysNSStringToUTF8(action.identifier);
for (const auto& it : additional_action_indices_) {
if (it.first == identifier) {
index = it.second;
break;
}
}
delegate()->NotificationAction(index);
}
this->LogAction("button clicked");
}
void CocoaNotification::LogAction(const char* action) { void CocoaNotification::LogAction(const char* action) {
if (getenv("ELECTRON_DEBUG_NOTIFICATIONS")) { if (getenv("ELECTRON_DEBUG_NOTIFICATIONS")) {
NSString* identifier = [notification_ valueForKey:@"identifier"]; NSString* identifier = [notification_ valueForKey:@"identifier"];

View file

@ -34,12 +34,15 @@
} }
if (notification) { if (notification) {
if (notif.activationType == NSUserNotificationActivationTypeReplied) { // Ref: https://developer.apple.com/documentation/foundation/nsusernotificationactivationtype?language=objc
notification->NotificationReplied([notif.response.string UTF8String]); if (notif.activationType == NSUserNotificationActivationTypeContentsClicked) {
notification->NotificationClicked();
} else if (notif.activationType == NSUserNotificationActivationTypeActionButtonClicked) { } else if (notif.activationType == NSUserNotificationActivationTypeActionButtonClicked) {
notification->NotificationButtonClicked(); notification->NotificationButtonClicked();
} else if (notif.activationType == NSUserNotificationActivationTypeContentsClicked) { } else if (notif.activationType == NSUserNotificationActivationTypeReplied) {
notification->NotificationClicked(); notification->NotificationReplied([notif.response.string UTF8String]);
} else if (notif.activationType == NSUserNotificationActivationTypeAdditionalActionClicked) {
notification->NotificationAdditionalActionClicked([notif additionalActivationAction]);
} }
} }
} }