From fbe379c81c3e844b8c8843666f37904332cd79b7 Mon Sep 17 00:00:00 2001 From: "Jasper St. Pierre" Date: Thu, 5 Dec 2013 00:19:53 -0500 Subject: [PATCH] messageTray: Considerably rework how layout is done in notifications Use a series of nested BoxLayouts, Bins, and more instead of an StTable and a custom ShellGenericContainer, and hacky style classes. This also removes the customContent parameter in favor of subclasses setting the child of this._bodyBin instead. We lose a few of the fancy features like showing the first part of the body, ellipsized, next the banner when it will fit, and other layout logic. But since the layout of notifications is changing substantially anyways, I don't feel too bad... --- data/theme/gnome-shell.css | 33 +- js/ui/components/autorunManager.js | 4 +- js/ui/components/telepathyClient.js | 33 +- js/ui/messageTray.js | 478 ++++++++-------------------- 4 files changed, 157 insertions(+), 391 deletions(-) diff --git a/data/theme/gnome-shell.css b/data/theme/gnome-shell.css index 70a830620..4d544535c 100644 --- a/data/theme/gnome-shell.css +++ b/data/theme/gnome-shell.css @@ -1535,9 +1535,15 @@ StScrollBar StButton#vhandle:active { .notification { border-radius: 10px 10px 0px 0px; background: rgba(0,0,0,0.9); - padding: 8px 8px 4px 8px; - spacing-rows: 4px; - spacing-columns: 10px; + padding: 8px; +} + +.notification-hbox { + spacing: 8px; +} + +.notification-title-box { + spacing: 8px; } .notification, #notification-container { @@ -1545,17 +1551,6 @@ StScrollBar StButton#vhandle:active { width: 34em; } -.notification.multi-line-notification { - padding-bottom: 8px; -} - -.notification-unexpanded { - /* We want to force the actor at a specific size, irrespective - of its minimum and preferred size, so we override both */ - min-height: 36px; - height: 36px; -} - .summary-boxpointer { -arrow-border-radius: 15px; -arrow-background-color: rgba(0,0,0,0.9); @@ -1603,10 +1598,6 @@ StScrollBar StButton#vhandle:active { padding-right: 6px; } -.notification-body { - spacing: 5px; -} - .notification-actions { padding-top: 18px; spacing: 10px; @@ -1697,7 +1688,11 @@ StScrollBar StButton#vhandle:active { padding-right: 4px; } -.chat-notification-scrollview{ +.chat-notification-body-box { + spacing: 5px; +} + +.chat-notification-scrollview { max-height: 22em; } diff --git a/js/ui/components/autorunManager.js b/js/ui/components/autorunManager.js index a93ec3ab6..d3a0a4e72 100644 --- a/js/ui/components/autorunManager.js +++ b/js/ui/components/autorunManager.js @@ -394,12 +394,12 @@ const AutorunTransientNotification = new Lang.Class({ Extends: MessageTray.Notification, _init: function(manager, source) { - this.parent(source, source.title, null, { customContent: true }); + this.parent(source, source.title); this._manager = manager; this._box = new St.BoxLayout({ style_class: 'hotplug-transient-box', vertical: true }); - this.addActor(this._box); + this._bodyBin.child = this._box; this._mount = source.mount; diff --git a/js/ui/components/telepathyClient.js b/js/ui/components/telepathyClient.js index 1cb0c2587..9ec4357f9 100644 --- a/js/ui/components/telepathyClient.js +++ b/js/ui/components/telepathyClient.js @@ -366,7 +366,7 @@ const ChatSource = new Lang.Class({ _updateAvatarIcon: function() { this.iconUpdated(); - this._notification.update(this._notification.title, null, { customContent: true }); + this._notification.update(this._notification.title); }, open: function() { @@ -558,7 +558,7 @@ const ChatSource = new Lang.Class({ title = GLib.markup_escape_text(this.title, -1); - this._notification.update(this._notification.title, null, { customContent: true, secondaryGIcon: this.getSecondaryIcon() }); + this._notification.update(this._notification.title, null, { secondaryGIcon: this.getSecondaryIcon() }); if (message) msg += ' (' + GLib.markup_escape_text(message, -1) + ')'; @@ -585,7 +585,7 @@ const ChatNotification = new Lang.Class({ Extends: MessageTray.Notification, _init: function(source) { - this.parent(source, source.title, null, { customContent: true, secondaryGIcon: source.getSecondaryIcon() }); + this.parent(source, source.title, null, { secondaryGIcon: source.getSecondaryIcon() }); this._responseEntry = new St.Entry({ style_class: 'chat-response', can_focus: true }); @@ -601,15 +601,17 @@ const ChatNotification = new Lang.Class({ this.emit('unfocused'); })); - this._createScrollArea(); this._lastGroup = null; + this._bodyBox = new St.BoxLayout({ style_class: 'chat-notification-body-box' }); + this._bodyBin.child = this._bodyBox; + // Keep track of the bottom position for the current adjustment and // force a scroll to the bottom if things change while we were at the // bottom - this._oldMaxScrollValue = this._scrollArea.vscroll.adjustment.value; - this._scrollArea.add_style_class_name('chat-notification-scrollview'); - this._scrollArea.vscroll.adjustment.connect('changed', Lang.bind(this, function(adjustment) { + this._oldMaxScrollValue = this._bodyScrollArea.vscroll.adjustment.value; + this._bodyScrollArea.add_style_class_name('chat-notification-scrollview'); + this._bodyScrollArea.vscroll.adjustment.connect('changed', Lang.bind(this, function(adjustment) { if (adjustment.value == this._oldMaxScrollValue) this.scrollTo(St.Side.BOTTOM); this._oldMaxScrollValue = Math.max(adjustment.lower, adjustment.upper - adjustment.page_size); @@ -646,8 +648,7 @@ const ChatNotification = new Lang.Class({ } if (message.direction == NotificationDirection.RECEIVED) { - this.update(this.source.title, messageBody, { customContent: true, - bannerMarkup: true }); + this.update(this.source.title, messageBody, { bannerMarkup: true }); } let group = (message.direction == NotificationDirection.RECEIVED ? @@ -684,7 +685,7 @@ const ChatNotification = new Lang.Class({ expired[i].actor.destroy(); } - let groups = this._contentArea.get_children(); + let groups = this._bodyBox.get_children(); for (let i = 0; i < groups.length; i++) { let group = groups[i]; if (group.get_n_children() == 0) @@ -716,9 +717,9 @@ const ChatNotification = new Lang.Class({ if (this._timestampTimeoutId) Mainloop.source_remove(this._timestampTimeoutId); - let highlighter = new MessageTray.URLHighlighter(props.body, - true, // line wrap? - true); // allow markup? + let highlighter = new MessageTray.URLHighlighter(); + highlighter.actor.clutter_text.line_wrap = true; + highlighter.setMarkup(props.body, true); let body = highlighter.actor; @@ -730,12 +731,12 @@ const ChatNotification = new Lang.Class({ if (group != this._lastGroup) { this._lastGroup = group; let emptyLine = new St.Label({ style_class: 'chat-empty-line' }); - this.addActor(emptyLine); + this._bodyBox.add_child(emptyLine); } this._lastMessageBox = new St.BoxLayout({ vertical: false }); this._lastMessageBox.add(body, props.childProps); - this.addActor(this._lastMessageBox); + this._bodyBox.add_child(this._lastMessageBox); this.updated(); @@ -872,7 +873,7 @@ const ChatNotification = new Lang.Class({ group: 'meta', styles: ['chat-meta-message'] }); - this.update(newAlias, null, { customContent: true }); + this.update(newAlias); this._filterMessages(); }, diff --git a/js/ui/messageTray.js b/js/ui/messageTray.js index a4442f592..5599635f4 100644 --- a/js/ui/messageTray.js +++ b/js/ui/messageTray.js @@ -159,9 +159,7 @@ const FocusGrabber = new Lang.Class({ const URLHighlighter = new Lang.Class({ Name: 'URLHighlighter', - _init: function(text, lineWrap, allowMarkup) { - if (!text) - text = ''; + _init: function() { this.actor = new St.Label({ reactive: true, style_class: 'url-highlighter' }); this._linkColor = '#ccccff'; this.actor.connect('style-changed', Lang.bind(this, function() { @@ -174,13 +172,7 @@ const URLHighlighter = new Lang.Class({ } } })); - if (lineWrap) { - this.actor.clutter_text.line_wrap = true; - this.actor.clutter_text.line_wrap_mode = Pango.WrapMode.WORD_CHAR; - this.actor.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; - } - this.setMarkup(text, allowMarkup); this.actor.connect('button-press-event', Lang.bind(this, function(actor, event) { // Don't try to URL highlight when invisible. // The MessageTray doesn't actually hide us, so @@ -234,6 +226,10 @@ const URLHighlighter = new Lang.Class({ })); }, + hasText: function() { + return !!this._text; + }, + setMarkup: function(text, allowMarkup) { text = text ? _fixMarkup(text, allowMarkup) : ''; this._text = text; @@ -440,23 +436,11 @@ const NotificationApplicationPolicy = new Lang.Class({ // elements that were added to it or if the @banner text did not // fit fully in the banner mode. When the notification is expanded, // the @banner text from the top line is always removed. The complete -// @banner text is added as the first element in the content section, -// unless 'customContent' parameter with the value 'true' is specified -// in @params. +// @banner text is added to the notification by default. You can change +// what is displayed by setting the child of this._bodyBin. // -// Additional notification content can be added with addActor(). The -// notification content is put inside a scrollview, so if it gets too -// tall, the notification will scroll rather than continue to grow. -// In addition to this main content area, there is also a single-row -// action area, which is not scrolled and can contain a single actor. -// The action area can be set by calling setActionArea() method. There -// is also a convenience method addButton() for adding a button to the -// action area. -// -// If @params contains a 'customContent' parameter with the value %true, -// then @banner will not be shown in the body of the notification when the -// notification is expanded and calls to update() will not clear the content -// unless 'clear' parameter with value %true is explicitly specified. +// You can also add buttons to the notification with addButton(), +// and you can construct simple default buttons with addAction(). // // By default, the icon shown is the same as the source's. // However, if @params contains a 'gicon' parameter, the passed in gicon @@ -472,8 +456,6 @@ const NotificationApplicationPolicy = new Lang.Class({ // // If @params contains a 'clear' parameter with the value %true, then // the content and the action area of the notification will be cleared. -// The content area is also always cleared if 'customContent' is false -// because it might contain the @banner that didn't fit in the banner mode. // // If @params contains 'soundName' or 'soundFile', the corresponding // event sound is played when the notification is shown (if the policy for @@ -495,50 +477,84 @@ const Notification = new Lang.Class({ this.focused = false; this.acknowledged = false; this._destroyed = false; - this._customContent = false; - this.bannerBodyText = null; - this.bannerBodyMarkup = false; - this._bannerBodyAdded = false; - this._titleFitsInBannerMode = true; this._titleDirection = Clutter.TextDirection.DEFAULT; - this._spacing = 0; - this._scrollPolicy = Gtk.PolicyType.AUTOMATIC; this._soundName = null; this._soundFile = null; this._soundPlayed = false; - this.actor = new St.Button({ accessible_role: Atk.Role.NOTIFICATION }); - this.actor.add_style_class_name('notification-unexpanded'); + // Let me draw you a picture: + // + // ,. this._iconBin ,. this._titleLabel + // | ,. this._secondaryIconBin + // .----|--------|---------------|------------. + // | .-----. | .----.-----------------------. | + // | | | | | | |--- this._titleBox + // | '.....' | '....'.......................' | + // | | |- this._hbox + // | | this._bodyBin | + // | | | --- this._vbox + // |_________|________________________________| + // | this._actionArea | + // |__________________________________________| + // | this._buttonBox | + // |__________________________________________| + + this.actor = new St.Button({ style_class: 'notification', + accessible_role: Atk.Role.NOTIFICATION, + x_fill: true, y_fill: true }); this.actor._delegate = this; this.actor.connect('clicked', Lang.bind(this, this._onClicked)); this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); - this._table = new St.Table({ style_class: 'notification', - reactive: true }); - this._table.connect('style-changed', Lang.bind(this, this._styleChanged)); - this.actor.set_child(this._table); + // Separates the notification content, action area and button box + this._vbox = new St.BoxLayout({ style_class: 'notification-vbox', vertical: true }); + this.actor.child = this._vbox; - // The first line should have the title, followed by the - // banner text, but ellipsized if they won't both fit. We can't - // make St.Table or St.BoxLayout do this the way we want (don't - // show banner at all if title needs to be ellipsized), so we - // use Shell.GenericContainer. - this._bannerBox = new Shell.GenericContainer(); - this._bannerBox.connect('get-preferred-width', Lang.bind(this, this._bannerBoxGetPreferredWidth)); - this._bannerBox.connect('get-preferred-height', Lang.bind(this, this._bannerBoxGetPreferredHeight)); - this._bannerBox.connect('allocate', Lang.bind(this, this._bannerBoxAllocate)); - this._table.add(this._bannerBox, { row: 0, - col: 1, - col_span: 2, - x_expand: false, - y_expand: false, - y_fill: false }); + // Separates the icon and title/body + this._hbox = new St.BoxLayout({ style_class: 'notification-hbox' }); + this._vbox.add_child(this._hbox); - this._titleLabel = new St.Label(); - this._bannerBox.add_actor(this._titleLabel); - this._bannerUrlHighlighter = new URLHighlighter(); - this._bannerLabel = this._bannerUrlHighlighter.actor; - this._bannerBox.add_actor(this._bannerLabel); + this._iconBin = new St.Bin(); + this._iconBin.set_y_align(Clutter.ActorAlign.START); + this._hbox.add_child(this._iconBin); + + this._titleBodyBox = new St.BoxLayout({ style_class: 'notification-vbox', + vertical: true }); + this._hbox.add_child(this._titleBodyBox); + + this._titleBox = new St.BoxLayout({ style_class: 'notification-title-box', + x_expand: true, x_align: Clutter.ActorAlign.START }); + this._secondaryIconBin = new St.Bin(); + this._titleBox.add_child(this._secondaryIconBin); + this._titleLabel = new St.Label({ x_expand: true }); + this._titleBox.add_child(this._titleLabel); + this._titleBodyBox.add(this._titleBox); + + this._bodyScrollArea = new St.ScrollView({ style_class: 'notification-scrollview', + hscrollbar_policy: Gtk.PolicyType.NEVER }); + this._titleBodyBox.add(this._bodyScrollArea); + this.enableScrolling(true); + + this._bodyScrollable = new St.BoxLayout(); + this._bodyScrollArea.add_actor(this._bodyScrollable); + + this._bodyBin = new St.Bin(); + this._bodyScrollable.add_actor(this._bodyBin); + + // By default, this._bodyBin contains a URL highlighter. Subclasses + // can override this to provide custom content if they want to. + this._bodyUrlHighlighter = new URLHighlighter(); + this._bodyUrlHighlighter.actor.clutter_text.line_wrap = true; + this._bodyUrlHighlighter.actor.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; + this._bodyBin.child = this._bodyUrlHighlighter.actor; + + this._actionAreaBin = new St.Bin({ x_expand: true, y_expand: true }); + this._vbox.add_child(this._actionAreaBin); + + this._buttonBox = new St.BoxLayout({ style_class: 'notification-actions', + x_expand: true, y_expand: true }); + global.focus_manager.add_group(this._buttonBox); + this._vbox.add_child(this._buttonBox); // If called with only one argument we assume the caller // will call .update() later on. This is the case of @@ -546,6 +562,21 @@ const Notification = new Lang.Class({ // for new and updated notifications if (arguments.length != 1) this.update(title, banner, params); + + this._sync(); + }, + + _sync: function() { + this._actionAreaBin.visible = this.expanded && (this._actionArea != null); + this._buttonBox.visible = this.expanded && (this._buttonBox.get_n_children() > 0); + + this._iconBin.visible = (this._icon != null && this._icon.visible); + this._secondaryIconBin.visible = (this._secondaryIcon != null); + + this._titleLabel.clutter_text.line_wrap = this.expanded; + this._bodyUrlHighlighter.actor.visible = this.expanded && this._bodyUrlHighlighter.hasText(); + + this._bodyScrollArea.visible = this.expanded && (this._bodyBin.child != null && this._bodyBin.child.visible); }, // update: @@ -557,50 +588,31 @@ const Notification = new Lang.Class({ // the title/banner. If @params.clear is %true, it will also // remove any additional actors/action buttons previously added. update: function(title, banner, params) { - params = Params.parse(params, { customContent: false, - gicon: null, + params = Params.parse(params, { gicon: null, secondaryGIcon: null, bannerMarkup: false, clear: false, soundName: null, soundFile: null }); - this._customContent = params.customContent; - let oldFocus = global.stage.key_focus; - if (this._icon && (params.gicon || params.clear)) { - this._icon.destroy(); - this._icon = null; - } - - if (this._secondaryIcon && (params.secondaryGIcon || params.clear)) { - this._secondaryIcon.destroy(); - this._secondaryIcon = null; - } - - // We always clear the content area if we don't have custom - // content because it might contain the @banner that didn't - // fit in the banner mode. - if (this._scrollArea && (!this._customContent || params.clear)) { - if (oldFocus && this._scrollArea.contains(oldFocus)) - this.actor.grab_key_focus(); - - this._scrollArea.destroy(); - this._scrollArea = null; - this._contentArea = null; - } if (this._actionArea && params.clear) { if (oldFocus && this._actionArea.contains(oldFocus)) this.actor.grab_key_focus(); this._actionArea.destroy(); this._actionArea = null; - this._buttonBox = null; } - if (!this._scrollArea && !this._actionArea) - this._table.remove_style_class_name('multi-line-notification'); + if (params.clear) { + this._buttonBox.destroy_all_children(); + } + + if (this._icon && (params.gicon || params.clear)) { + this._icon.destroy(); + this._icon = null; + } if (params.gicon) { this._icon = new St.Icon({ gicon: params.gicon, @@ -609,19 +621,18 @@ const Notification = new Lang.Class({ this._icon = this.source.createIcon(this.ICON_SIZE); } - if (this._icon) { - this._table.add(this._icon, { row: 0, - col: 0, - x_expand: false, - y_expand: false, - y_fill: false, - y_align: St.Align.START }); + if (this._icon) + this._iconBin.child = this._icon; + + if (this._secondaryIcon && (params.secondaryGIcon || params.clear)) { + this._secondaryIcon.destroy(); + this._secondaryIcon = null; } if (params.secondaryGIcon) { this._secondaryIcon = new St.Icon({ gicon: params.secondaryGIcon, style_class: 'secondary-icon' }); - this._bannerBox.add_actor(this._secondaryIcon); + this._secondaryIconBin.child = this._secondaryIcon; } this.title = title; @@ -639,24 +650,9 @@ const Notification = new Lang.Class({ // arguably for action buttons as well. Labels other than the title // will be allocated at the available width, so that their alignment // is done correctly automatically. - this._table.set_text_direction(this._titleDirection); + this.actor.set_text_direction(this._titleDirection); - // Unless the notification has custom content, we save this.bannerBodyText - // to add it to the content of the notification if the notification is - // expandable due to other elements in its content area or due to the banner - // not fitting fully in the single-line mode. - this.bannerBodyText = this._customContent ? null : banner; - this.bannerBodyMarkup = params.bannerMarkup; - this._bannerBodyAdded = false; - - banner = banner ? banner.replace(/\n/g, ' ') : ''; - - this._bannerUrlHighlighter.setMarkup(banner, params.bannerMarkup); - this._bannerLabel.queue_relayout(); - - // Add the bannerBody now if we know for sure we'll need it - if (this.bannerBodyText && this.bannerBodyText.indexOf('\n') > -1) - this._addBannerBody(); + this._bodyUrlHighlighter.setMarkup(banner, params.bannerMarkup); if (this._soundName != params.soundName || this._soundFile != params.soundFile) { @@ -665,56 +661,18 @@ const Notification = new Lang.Class({ this._soundPlayed = false; } - this.updated(); + this._sync(); }, setIconVisible: function(visible) { this._icon.visible = visible; + this._sync(); }, enableScrolling: function(enableScrolling) { - this._scrollPolicy = enableScrolling ? Gtk.PolicyType.AUTOMATIC : Gtk.PolicyType.NEVER; - if (this._scrollArea) { - this._scrollArea.vscrollbar_policy = this._scrollPolicy; - this._scrollArea.enable_mouse_scrolling = enableScrolling; - } - }, - - _createScrollArea: function() { - this._table.add_style_class_name('multi-line-notification'); - this._scrollArea = new St.ScrollView({ style_class: 'notification-scrollview', - vscrollbar_policy: this._scrollPolicy, - hscrollbar_policy: Gtk.PolicyType.NEVER, - visible: this.expanded }); - this._table.add(this._scrollArea, { row: 1, - col: 2 }); - this._contentArea = new St.BoxLayout({ style_class: 'notification-body', - vertical: true }); - this._scrollArea.add_actor(this._contentArea); - // If we know the notification will be expandable, we need to add - // the banner text to the body as the first element. - this._addBannerBody(); - }, - - // addActor: - // @actor: actor to add to the body of the notification - // - // Appends @actor to the notification's body - addActor: function(actor, style) { - if (!this._scrollArea) { - this._createScrollArea(); - } - - this._contentArea.add(actor, style ? style : {}); - this.updated(); - }, - - _addBannerBody: function() { - if (this.bannerBodyText && !this._bannerBodyAdded) { - let label = new URLHighlighter(this.bannerBodyText, true, this.bannerBodyMarkup); - this.addActor(label.actor); - this._bannerBodyAdded = true; - } + let scrollPolicy = enableScrolling ? Gtk.PolicyType.AUTOMATIC : Gtk.PolicyType.NEVER; + this._bodyScrollArea.vscrollbar_policy = scrollPolicy; + this._bodyScrollArea.enable_mouse_scrolling = enableScrolling; }, // scrollTo: @@ -722,53 +680,23 @@ const Notification = new Lang.Class({ // // Scrolls the content area (if scrollable) to the indicated edge scrollTo: function(side) { - let adjustment = this._scrollArea.vscroll.adjustment; + let adjustment = this._bodyScrollArea.vscroll.adjustment; if (side == St.Side.TOP) adjustment.value = adjustment.lower; else if (side == St.Side.BOTTOM) adjustment.value = adjustment.upper; }, - // setActionArea: - // @actor: the actor - // @props: (option) St.Table child properties - // - // Puts @actor into the action area of the notification, replacing - // the previous contents - setActionArea: function(actor, props) { - if (this._actionArea) { + setActionArea: function(actor) { + if (this._actionArea) this._actionArea.destroy(); - this._actionArea = null; - if (this._buttonBox) - this._buttonBox = null; - } else { - this._addBannerBody(); - } + this._actionArea = actor; - this._actionArea.visible = this.expanded; - - if (!props) - props = {}; - props.row = 2; - props.col = 2; - - this._table.add_style_class_name('multi-line-notification'); - this._table.add(this._actionArea, props); - this.updated(); + this._actionAreaBin.child = actor; + this._sync(); }, addButton: function(button, callback) { - if (!this._buttonBox) { - let box = new St.BoxLayout({ style_class: 'notification-actions' }); - this.setActionArea(box, { x_expand: false, - y_expand: false, - x_fill: false, - y_fill: false, - x_align: St.Align.END }); - this._buttonBox = box; - global.focus_manager.add_group(this._buttonBox); - } - this._buttonBox.add(button); button.connect('clicked', Lang.bind(this, function() { callback(); @@ -777,7 +705,7 @@ const Notification = new Lang.Class({ this.destroy(); })); - this.updated(); + this._sync(); return button; }, @@ -827,113 +755,6 @@ const Notification = new Lang.Class({ } }, - _bannerBoxGetPreferredHeight: function(actor, forWidth, alloc) { - [alloc.min_size, alloc.natural_size] = - this._titleLabel.get_preferred_height(forWidth); - }, - - _bannerBoxAllocate: function(actor, box, flags) { - let availWidth = box.x2 - box.x1; - - let [titleMinW, titleNatW] = this._titleLabel.get_preferred_width(-1); - let [titleMinH, titleNatH] = this._titleLabel.get_preferred_height(availWidth); - let [bannerMinW, bannerNatW] = this._bannerLabel.get_preferred_width(availWidth); - - let rtl = (this._titleDirection == Clutter.TextDirection.RTL); - let x = rtl ? availWidth : 0; - - if (this._secondaryIcon) { - let [iconMinW, iconNatW] = this._secondaryIcon.get_preferred_width(-1); - let [iconMinH, iconNatH] = this._secondaryIcon.get_preferred_height(availWidth); - - let secondaryIconBox = new Clutter.ActorBox(); - let secondaryIconBoxW = Math.min(iconNatW, availWidth); - - // allocate secondary icon box - if (rtl) { - secondaryIconBox.x1 = x - secondaryIconBoxW; - secondaryIconBox.x2 = x; - x = x - (secondaryIconBoxW + this._spacing); - } else { - secondaryIconBox.x1 = x; - secondaryIconBox.x2 = x + secondaryIconBoxW; - x = x + secondaryIconBoxW + this._spacing; - } - secondaryIconBox.y1 = 0; - // Using titleNatH ensures that the secondary icon is centered vertically - secondaryIconBox.y2 = titleNatH; - - availWidth = availWidth - (secondaryIconBoxW + this._spacing); - this._secondaryIcon.allocate(secondaryIconBox, flags); - } - - let titleBox = new Clutter.ActorBox(); - let titleBoxW = Math.min(titleNatW, availWidth); - if (rtl) { - titleBox.x1 = availWidth - titleBoxW; - titleBox.x2 = availWidth; - } else { - titleBox.x1 = x; - titleBox.x2 = titleBox.x1 + titleBoxW; - } - titleBox.y1 = 0; - titleBox.y2 = titleNatH; - this._titleLabel.allocate(titleBox, flags); - this._titleFitsInBannerMode = (titleNatW <= availWidth); - - let bannerFits = true; - if (titleBoxW + this._spacing > availWidth) { - this._bannerLabel.opacity = 0; - bannerFits = false; - } else { - let bannerBox = new Clutter.ActorBox(); - - if (rtl) { - bannerBox.x1 = 0; - bannerBox.x2 = titleBox.x1 - this._spacing; - - bannerFits = (bannerBox.x2 - bannerNatW >= 0); - } else { - bannerBox.x1 = titleBox.x2 + this._spacing; - bannerBox.x2 = availWidth; - - bannerFits = (bannerBox.x1 + bannerNatW <= availWidth); - } - bannerBox.y1 = 0; - bannerBox.y2 = titleNatH; - this._bannerLabel.allocate(bannerBox, flags); - - // Make _bannerLabel visible if the entire notification - // fits on one line, or if the notification is currently - // unexpanded and only showing one line anyway. - if (!this.expanded || (bannerFits && this._table.row_count == 1)) - this._bannerLabel.opacity = 255; - } - - // If the banner doesn't fully fit in the banner box, we possibly need to add the - // banner to the body. We can't do that from here though since that will force a - // relayout, so we add it to the main loop. - if (!bannerFits && this._canExpandContent()) - Meta.later_add(Meta.LaterType.BEFORE_REDRAW, - Lang.bind(this, - function() { - if (this._destroyed) - return false; - - if (this._canExpandContent()) { - this._addBannerBody(); - this._table.add_style_class_name('multi-line-notification'); - this.updated(); - } - return false; - })); - }, - - _canExpandContent: function() { - return (this.bannerBodyText && !this._bannerBodyAdded) || - (!this._titleFitsInBannerMode && !this._table.has_style_class_name('multi-line-notification')); - }, - playSound: function() { if (this._soundPlayed) return; @@ -966,69 +787,18 @@ const Notification = new Lang.Class({ } }, - updated: function() { - if (this.expanded) - this.expand(false); - }, - expand: function(animate) { this.expanded = true; - this.actor.remove_style_class_name('notification-unexpanded'); - - // Show additional content that we keep hidden in banner mode - if (this._actionArea) - this._actionArea.show(); - if (this._scrollArea) - this._scrollArea.show(); - - // The banner is never shown when the title did not fit, so this - // can be an if-else statement. - if (!this._titleFitsInBannerMode) { - // Remove ellipsization from the title label and make it wrap so that - // we show the full title when the notification is expanded. - this._titleLabel.clutter_text.line_wrap = true; - this._titleLabel.clutter_text.line_wrap_mode = Pango.WrapMode.WORD_CHAR; - this._titleLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE; - } else if (this._table.row_count > 1 && this._bannerLabel.opacity != 0) { - // We always hide the banner if the notification has additional content. - // - // We don't need to wrap the banner that doesn't fit the way we wrap the - // title that doesn't fit because we won't have a notification with - // row_count=1 that has a banner that doesn't fully fit. We'll either add - // that banner to the content of the notification in _bannerBoxAllocate() - // or the notification will have custom content. - if (animate) - Tweener.addTween(this._bannerLabel, - { opacity: 0, - time: ANIMATION_TIME, - transition: 'easeOutQuad' }); - else - this._bannerLabel.opacity = 0; - } + this._sync(); this.emit('expanded'); }, collapseCompleted: function() { if (this._destroyed) return; + this.expanded = false; - - // Hide additional content that we keep hidden in banner mode - if (this._actionArea) - this._actionArea.hide(); - if (this._scrollArea) - this._scrollArea.hide(); - - // Make sure we don't line wrap the title, and ellipsize it instead. - this._titleLabel.clutter_text.line_wrap = false; - this._titleLabel.clutter_text.ellipsize = Pango.EllipsizeMode.END; - - // Restore banner opacity in case the notification is shown in the - // banner mode again on update. - this._bannerLabel.opacity = 255; - - // Restore height requisition - this.actor.add_style_class_name('notification-unexpanded'); + this._sync(); }, _onClicked: function() {