// Copyright (c) 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef UI_MESSAGE_CENTER_PUBLIC_CPP_NOTIFICATION_H_ #define UI_MESSAGE_CENTER_PUBLIC_CPP_NOTIFICATION_H_ #include #include #include #include "base/memory/ref_counted.h" #include "base/optional.h" #include "base/strings/string16.h" #include "base/time/time.h" #include "base/values.h" #include "third_party/skia/include/core/SkColor.h" #include "ui/gfx/color_palette.h" #include "ui/gfx/image/image.h" #include "ui/gfx/paint_vector_icon.h" #include "ui/gfx/skia_util.h" #include "ui/gfx/vector_icon_types.h" #include "ui/message_center/public/cpp/message_center_public_export.h" #include "ui/message_center/public/cpp/notification_delegate.h" #include "ui/message_center/public/cpp/notification_types.h" #include "ui/message_center/public/cpp/notifier_id.h" #include "url/gurl.h" namespace gfx { struct VectorIcon; } // namespace gfx namespace message_center { // Represents an individual item in NOTIFICATION_TYPE_MULTIPLE notifications. struct MESSAGE_CENTER_PUBLIC_EXPORT NotificationItem { base::string16 title; base::string16 message; }; enum class SettingsButtonHandler { NONE = 0, // No button. This is the default. Does not affect native // settings button (like on Android). INLINE = 1, // Button shown, settings inline. DELEGATE = 2, // Button shown, notification's delegate handles action. }; enum class SystemNotificationWarningLevel { NORMAL, WARNING, CRITICAL_WARNING }; // Represents a button to be shown as part of a notification. struct MESSAGE_CENTER_PUBLIC_EXPORT ButtonInfo { explicit ButtonInfo(const base::string16& title); ButtonInfo(const ButtonInfo& other); ButtonInfo(); ~ButtonInfo(); ButtonInfo& operator=(const ButtonInfo& other); // Title that should be displayed on the notification button. base::string16 title; // Icon that should be displayed on the notification button. Optional. On some // platforms, a mask will be applied to the icon, to match the visual // requirements of the notification. As with Android, MD notifications don't // display this icon. gfx::Image icon; // The placeholder string that should be displayed in the input field for // text input type buttons until the user has entered a response themselves. // If the value is null, there is no input field associated with the button. base::Optional placeholder; }; enum class FullscreenVisibility { NONE = 0, // Don't show the notification over fullscreen (default). OVER_USER = 1, // Show over the current fullscreened client window. // windows (like Chrome OS login). }; // Represents rich features available for notifications. class MESSAGE_CENTER_PUBLIC_EXPORT RichNotificationData { public: RichNotificationData(); RichNotificationData(const RichNotificationData& other); ~RichNotificationData(); // Priority of the notification. This must be one of the NotificationPriority // values defined in notification_types.h. int priority = DEFAULT_PRIORITY; // Whether the notification should remain on screen indefinitely. bool never_timeout = false; // Time indicating when the notification was shown. Defaults to the time at // which the RichNotificationData instance is constructed. base::Time timestamp; // Context message to display below the notification's content. Optional. May // not be used for notifications that have an explicit origin URL set. base::string16 context_message; // Large image to display on the notification. Optional. gfx::Image image; // Small badge to display on the notification to illustrate the source of the // notification. Optional. gfx::Image small_image; #if defined(OS_CHROMEOS) // If true, we simply use the raw |small_image| icon, ignoring accent color // styling. For example, this is used with raw icons received from Android. bool ignore_accent_color_for_small_image = false; #endif // defined(OS_CHROMEOS) // Vector version of |small_image|. // Used by Notification::GenerateMaskedSmallIcon. // If not available, |small_image| will be used by the method. Optional. // // Due to the restriction of CreateVectorIcon, this should be a pointer to // globally defined VectorIcon instance e.g. kNotificationCapsLockIcon. // gfx::Image created by gfx::CreateVectorIcon internally stores reference to // VectorIcon, so the VectorIcon should live longer than gfx::Image instance. // As a temporary solution to this problem, we make this variable a pointer // and only pass globally defined constants. // TODO(tetsui): Remove the pointer, after fixing VectorIconSource not to // retain VectorIcon reference. https://crbug.com/760866 const gfx::VectorIcon* vector_small_image = &gfx::kNoneIcon; // Items to display on the notification. Only applicable for notifications // that have type NOTIFICATION_TYPE_MULTIPLE. std::vector items; // Progress, in range of [0-100], of NOTIFICATION_TYPE_PROGRESS notifications. // Values outside of the range (e.g. -1) will show an infinite loading // progress bar. int progress = 0; // Status text string shown in NOTIFICATION_TYPE_PROGRESS notifications. // If MD style notification is not enabled, this attribute is ignored. base::string16 progress_status; // Buttons that should show up on the notification. A maximum of 16 buttons // is supported by the current implementation, but this may differ between // platforms. std::vector buttons; // Whether updates to the visible notification should be announced to users // depending on visual assistance systems. bool should_make_spoken_feedback_for_popup_updates = true; #if defined(OS_CHROMEOS) // Flag if the notification is pinned. If true, the notification is pinned // and the user can't remove it. bool pinned = false; #endif // defined(OS_CHROMEOS) // Vibration pattern to play when displaying the notification. There must be // an odd number of entries in this pattern when it's set: numbers of // milliseconds to vibrate separated by numbers of milliseconds to pause. std::vector vibration_pattern; // Whether the vibration pattern and other applicable announcement mechanisms // should be considered when updating the notification. bool renotify = false; // Whether all announcement mechansims should be suppressed when displaying // the notification. bool silent = false; // An accessible description of the notification's contents. base::string16 accessible_name; // Unified theme color used in new style notification. // Usually, it should not be set directly. // For system notification, ash::CreateSystemNotification with // SystemNotificationWarningLevel should be used. base::Optional accent_color; // Controls whether a settings button should appear on the notification. See // enum definition. TODO(estade): turn this into a boolean. See // crbug.com/780342 SettingsButtonHandler settings_button_handler = SettingsButtonHandler::NONE; // Controls whether a snooze button should appear on the notification. bool should_show_snooze_button = false; FullscreenVisibility fullscreen_visibility = FullscreenVisibility::NONE; }; class MESSAGE_CENTER_PUBLIC_EXPORT Notification { public: // Creates a new notification. // // |type|: Type of the notification that dictates the layout. // |id|: Identifier of the notification. Showing a notification that shares // its profile and identifier with an already visible notification will // replace the former one // |title|: Title of the notification. // |message|: Body text of the notification. May not be used for certain // values of |type|, for example list-style notifications. // |icon|: Icon to show alongside of the notification. // |display_source|: Textual representation of who's shown the notification. // |origin_url|: URL of the website responsible for showing the notification. // |notifier_id|: NotifierId instance representing the system responsible for // showing the notification. // |optional_fields|: Rich data that can be used to assign more elaborate // features to notifications. // |delegate|: Delegate that will influence the behaviour of this notification // and receives events on its behalf. May be omitted. Notification(NotificationType type, const std::string& id, const base::string16& title, const base::string16& message, const gfx::Image& icon, const base::string16& display_source, const GURL& origin_url, const NotifierId& notifier_id, const RichNotificationData& optional_fields, scoped_refptr delegate); // Creates a copy of the |other| notification. The delegate, if any, will be // identical for both the Notification instances. The |id| of the notification // will be replaced by the given value. Notification(const std::string& id, const Notification& other); // Creates a copy of the |other| notification. The delegate will be replaced // by |delegate|. Notification(scoped_refptr delegate, const Notification& other); // Creates a copy of the |other| notification. The delegate, if any, will be // identical for both the Notification instances. Notification(const Notification& other); Notification& operator=(const Notification& other); virtual ~Notification(); // Performs a deep copy of |notification|, including images and (optionally) // the body image, small image, and icon images which are not supported on all // platforms. static std::unique_ptr DeepCopy( const Notification& notification, bool include_body_image, bool include_small_image, bool include_icon_images); NotificationType type() const { return type_; } void set_type(NotificationType type) { type_ = type; } // Uniquely identifies a notification in the message center. For // notification front ends that support multiple profiles, this id should // identify a unique profile + frontend_notification_id combination. You can // Use this id against the MessageCenter interface but not the // NotificationUIManager interface. const std::string& id() const { return id_; } const base::string16& title() const { return title_; } void set_title(const base::string16& title) { title_ = title; } const base::string16& message() const { return message_; } void set_message(const base::string16& message) { message_ = message; } // The origin URL of the script which requested the notification. // Can be empty if the notification is requested by an extension or // Chrome app. const GURL& origin_url() const { return origin_url_; } void set_origin_url(const GURL& origin_url) { origin_url_ = origin_url; } // A display string for the source of the notification. const base::string16& display_source() const { return display_source_; } const NotifierId& notifier_id() const { return notifier_id_; } void set_profile_id(const std::string& profile_id) { notifier_id_.profile_id = profile_id; } // Begin unpacked values from optional_fields. int priority() const { return optional_fields_.priority; } void set_priority(int priority) { optional_fields_.priority = priority; } // This vibration_pattern property currently has no effect on // non-Android platforms. const std::vector& vibration_pattern() const { return optional_fields_.vibration_pattern; } void set_vibration_pattern(const std::vector& vibration_pattern) { optional_fields_.vibration_pattern = vibration_pattern; } // This property currently only works in platforms that support native // notifications. // It determines whether the sound and vibration effects should signal // if the notification is replacing another notification. bool renotify() const { return optional_fields_.renotify; } void set_renotify(bool renotify) { optional_fields_.renotify = renotify; } // This property currently has no effect on non-Android platforms. bool silent() const { return optional_fields_.silent; } void set_silent(bool silent) { optional_fields_.silent = silent; } base::Time timestamp() const { return optional_fields_.timestamp; } void set_timestamp(const base::Time& timestamp) { optional_fields_.timestamp = timestamp; } const base::string16 context_message() const { return optional_fields_.context_message; } void set_context_message(const base::string16& context_message) { optional_fields_.context_message = context_message; } // Decides if the notification origin should be used as a context message bool UseOriginAsContextMessage() const; const std::vector& items() const { return optional_fields_.items; } void set_items(const std::vector& items) { optional_fields_.items = items; } int progress() const { return optional_fields_.progress; } void set_progress(int progress) { optional_fields_.progress = progress; } base::string16 progress_status() const { return optional_fields_.progress_status; } void set_progress_status(const base::string16& progress_status) { optional_fields_.progress_status = progress_status; } // End unpacked values. // Images fetched asynchronously. const gfx::Image& icon() const { return icon_; } void set_icon(const gfx::Image& icon) { icon_ = icon; } const gfx::Image& image() const { return optional_fields_.image; } void set_image(const gfx::Image& image) { optional_fields_.image = image; } const gfx::Image& small_image() const { return optional_fields_.small_image; } void set_small_image(const gfx::Image& image) { optional_fields_.small_image = image; } const gfx::VectorIcon& vector_small_image() const { return *optional_fields_.vector_small_image; } // Due to the restriction of CreateVectorIcon, this should be a pointer to // globally defined VectorIcon instance e.g. kNotificationCapsLockIcon. // See detailed comment in RichNotificationData::vector_small_image. void set_vector_small_image(const gfx::VectorIcon& image) { optional_fields_.vector_small_image = ℑ } // Mask the color of |small_image| to the given |color|. // If |vector_small_image| is available, it returns the vector image // filled by the |color|. // Otherwise, it uses alpha channel of the rasterized |small_image| for // masking. gfx::Image GenerateMaskedSmallIcon(int dip_size, SkColor color) const; // Buttons, with icons fetched asynchronously. const std::vector& buttons() const { return optional_fields_.buttons; } void set_buttons(const std::vector& buttons) { optional_fields_.buttons = buttons; } void SetButtonIcon(size_t index, const gfx::Image& icon); // Used to keep the order of notifications with the same timestamp. // The notification with lesser serial_number is considered 'older'. unsigned serial_number() { return serial_number_; } // Gets and sets whether the notifiction should remain onscreen permanently. bool never_timeout() const { return optional_fields_.never_timeout; } void set_never_timeout(bool never_timeout) { optional_fields_.never_timeout = never_timeout; } bool pinned() const { #if defined(OS_CHROMEOS) return optional_fields_.pinned; #else return false; #endif // defined(OS_CHROMEOS) } #if defined(OS_CHROMEOS) void set_pinned(bool pinned) { optional_fields_.pinned = pinned; } #endif // defined(OS_CHROMEOS) // Gets a text for spoken feedback. const base::string16& accessible_name() const { return optional_fields_.accessible_name; } base::Optional accent_color() const { return optional_fields_.accent_color; } void set_accent_color(SkColor accent_color) { optional_fields_.accent_color = accent_color; } bool should_show_settings_button() const { return optional_fields_.settings_button_handler != SettingsButtonHandler::NONE; } void set_settings_button_handler(SettingsButtonHandler handler) { optional_fields_.settings_button_handler = handler; } bool should_show_snooze_button() const { return optional_fields_.should_show_snooze_button; } FullscreenVisibility fullscreen_visibility() const { return optional_fields_.fullscreen_visibility; } void set_fullscreen_visibility(FullscreenVisibility visibility) { optional_fields_.fullscreen_visibility = visibility; } NotificationDelegate* delegate() const { return delegate_.get(); } const RichNotificationData& rich_notification_data() const { return optional_fields_; } void set_delegate(scoped_refptr delegate) { DCHECK(!delegate_); delegate_ = delegate; } // Set the priority to SYSTEM. The system priority user needs to call this // method explicitly, to avoid setting it accidentally. void SetSystemPriority(); #if defined(OS_CHROMEOS) void set_system_notification_warning_level( SystemNotificationWarningLevel warning_level) { system_notification_warning_level_ = warning_level; } SystemNotificationWarningLevel system_notification_warning_level() const { return system_notification_warning_level_; } #endif // defined(OS_CHROMEOS) const std::string& custom_view_type() const { return custom_view_type_; } void set_custom_view_type(const std::string& custom_view_type) { DCHECK_EQ(type(), NotificationType::NOTIFICATION_TYPE_CUSTOM); custom_view_type_ = custom_view_type; } protected: // The type of notification we'd like displayed. NotificationType type_; std::string id_; base::string16 title_; base::string16 message_; // Image data for the associated icon, used by Ash when available. gfx::Image icon_; // The display string for the source of the notification. Could be // the same as |origin_url_|, or the name of an extension. // Expected to be a localized user facing string. base::string16 display_source_; private: // The origin URL of the script which requested the notification. // Can be empty if requested through a chrome app or extension or if // it's a system notification. GURL origin_url_; NotifierId notifier_id_; RichNotificationData optional_fields_; // TODO(estade): these book-keeping fields should be moved into // NotificationList. unsigned serial_number_; // A proxy object that allows access back to the JavaScript object that // represents the notification, for firing events. scoped_refptr delegate_; // For custom notifications this determines which factory will be used for // creating the view for this notification. The type should match the type // used to register the factory in MessageViewFactory. std::string custom_view_type_; #if defined(OS_CHROMEOS) // The warning level of a system notification. SystemNotificationWarningLevel system_notification_warning_level_ = SystemNotificationWarningLevel::NORMAL; #endif // defined(OS_CHROMEOS) }; } // namespace message_center #endif // UI_MESSAGE_CENTER_PUBLIC_CPP_NOTIFICATION_H_