Compare commits
	
		
			12 Commits
		
	
	
		
			wip/exalm/
			...
			wip/notif-
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					672a4a56bd | ||
| 
						 | 
					50421ede37 | ||
| 
						 | 
					826a7fa02a | ||
| 
						 | 
					d7844bc7f4 | ||
| 
						 | 
					485444a451 | ||
| 
						 | 
					e5318d5235 | ||
| 
						 | 
					e6e15489aa | ||
| 
						 | 
					0951d36e8f | ||
| 
						 | 
					c38939902f | ||
| 
						 | 
					97974a3f2a | ||
| 
						 | 
					8b6cab741d | ||
| 
						 | 
					2aad0eac31 | 
@@ -414,10 +414,7 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
/* Buttons */
 | 
			
		||||
 | 
			
		||||
.candidate-page-button,
 | 
			
		||||
.notification-button,
 | 
			
		||||
.notification-icon-button,
 | 
			
		||||
.hotplug-notification-item,
 | 
			
		||||
.hotplug-resident-eject-button,
 | 
			
		||||
.modal-dialog-button,
 | 
			
		||||
.app-view-control {
 | 
			
		||||
    border: 1px solid #8b8b8b;
 | 
			
		||||
@@ -431,17 +428,12 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.candidate-page-button:hover,
 | 
			
		||||
.notification-button:hover,
 | 
			
		||||
.notification-icon-button:hover,
 | 
			
		||||
.hotplug-notification-item:hover,
 | 
			
		||||
.hotplug-resident-eject-button:hover,
 | 
			
		||||
.modal-dialog-button:hover {
 | 
			
		||||
    background-gradient-start: rgba(255, 255, 255, 0.3);
 | 
			
		||||
    background-gradient-end: rgba(255, 255, 255, 0.1);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-button:focus,
 | 
			
		||||
.notification-icon-button:focus,
 | 
			
		||||
.hotplug-notification-item:focus,
 | 
			
		||||
.modal-dialog-button:focus,
 | 
			
		||||
.app-view-control:focus {
 | 
			
		||||
@@ -455,10 +447,7 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
 | 
			
		||||
.candidate-page-button:active,
 | 
			
		||||
.candidate-page-button:pressed,
 | 
			
		||||
.notification-button:active,
 | 
			
		||||
.notification-icon-button:active,
 | 
			
		||||
.hotplug-notification-item:active,
 | 
			
		||||
.hotplug-resident-eject-button:active,
 | 
			
		||||
.modal-dialog-button:active,
 | 
			
		||||
.modal-dialog-button:pressed,
 | 
			
		||||
.app-view-control:checked {
 | 
			
		||||
@@ -467,8 +456,6 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.candidate-page-button:insensitive,
 | 
			
		||||
.notification-button:insensitive,
 | 
			
		||||
.notification-icon-button:insensitive,
 | 
			
		||||
.modal-dialog-button:insensitive {
 | 
			
		||||
    border-color: #666666;
 | 
			
		||||
    color: #9f9f9f;
 | 
			
		||||
@@ -480,7 +467,6 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
 | 
			
		||||
#searchEntry,
 | 
			
		||||
.modal-dialog-button,
 | 
			
		||||
.notification-button,
 | 
			
		||||
.hotplug-notification-item,
 | 
			
		||||
.app-view-controls,
 | 
			
		||||
#screenShieldNotifications {
 | 
			
		||||
@@ -1535,36 +1521,69 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
    color: #999999;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.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;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification, #notification-container {
 | 
			
		||||
    font-size: 11pt;
 | 
			
		||||
    width: 34em;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification.multi-line-notification {
 | 
			
		||||
    padding-bottom: 8px;
 | 
			
		||||
.notification-main-button,
 | 
			
		||||
.notification-button {
 | 
			
		||||
    background: rgba(0,0,0,0.9);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.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;
 | 
			
		||||
.notification-main-button {
 | 
			
		||||
    border-radius: 10px 10px 0px 0px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* We use row-span = 2 for the image cell, which prevents its height preferences to be
 | 
			
		||||
   taken into account during allocation, so its height ends up being limited by the height
 | 
			
		||||
   of the content in the other rows. To avoid showing a stretched image, we set the minimum
 | 
			
		||||
   height of the table to be ICON_SIZE + IMAGE_SIZE + spacing-rows = 24 + 125 + 10 = 159 */
 | 
			
		||||
.notification-with-image {
 | 
			
		||||
    min-height: 159px;
 | 
			
		||||
.notification-main-content {
 | 
			
		||||
    padding: 8px;
 | 
			
		||||
    spacing: 8px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-close-button {
 | 
			
		||||
    padding: 8px;
 | 
			
		||||
    border-radius: 4px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-action-area {
 | 
			
		||||
    padding: 8px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-action-area,
 | 
			
		||||
.notification-button {
 | 
			
		||||
    border-top: 1px solid #666;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-button {
 | 
			
		||||
    padding: 8px 0px;
 | 
			
		||||
    border-right: 1px solid #666;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-main-button:hover,
 | 
			
		||||
.notification-button:hover,
 | 
			
		||||
.notification-close-button:hover {
 | 
			
		||||
    background: rgba(100,100,100,0.9);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-main-button:active,
 | 
			
		||||
.notification-button:active {
 | 
			
		||||
    background: rgba(255,255,255,0.1);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-button:last-child {
 | 
			
		||||
    border-right-width: 0px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-title-box {
 | 
			
		||||
    spacing: 8px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-scrollview:ltr > StScrollBar {
 | 
			
		||||
    padding-left: 6px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-scrollview:rtl > StScrollBar {
 | 
			
		||||
    padding-right: 6px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.summary-boxpointer {
 | 
			
		||||
@@ -1606,47 +1625,6 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
    -st-vfade-offset: 24px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-scrollview:ltr > StScrollBar {
 | 
			
		||||
    padding-left: 6px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-scrollview:rtl > StScrollBar {
 | 
			
		||||
    padding-right: 6px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-body {
 | 
			
		||||
    spacing: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-actions {
 | 
			
		||||
    padding-top: 18px;
 | 
			
		||||
    spacing: 10px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-button {
 | 
			
		||||
    -st-natural-width: 140px;
 | 
			
		||||
    padding: 4px 4px 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-button:focus {
 | 
			
		||||
    -st-natural-width: 138px;
 | 
			
		||||
    padding: 3px 4px 4px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-icon-button {
 | 
			
		||||
    border-radius: 5px;
 | 
			
		||||
    padding: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-icon-button:focus {
 | 
			
		||||
    padding: 4px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notification-icon-button > StIcon {
 | 
			
		||||
    icon-size: 16px;
 | 
			
		||||
    padding: 8px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.secondary-icon {
 | 
			
		||||
    icon-size: 1.09em;
 | 
			
		||||
}
 | 
			
		||||
@@ -1669,45 +1647,6 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
    padding: 2px 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotplug-resident-box {
 | 
			
		||||
    spacing: 8px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotplug-resident-mount {
 | 
			
		||||
    spacing: 8px;
 | 
			
		||||
    border-radius: 4px;
 | 
			
		||||
 | 
			
		||||
    color: #ccc;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotplug-resident-mount:hover {
 | 
			
		||||
    background-gradient-direction: horizontal;
 | 
			
		||||
    background-gradient-start: rgba(255, 255, 255, 0.1);
 | 
			
		||||
    background-gradient-end: rgba(255, 255, 255, 0);
 | 
			
		||||
 | 
			
		||||
    color: #fff;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotplug-resident-mount-label {
 | 
			
		||||
    color: inherit;
 | 
			
		||||
    padding-left: 6px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotplug-resident-mount-icon {
 | 
			
		||||
    icon-size: 24px;
 | 
			
		||||
    padding-left: 6px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotplug-resident-eject-icon {
 | 
			
		||||
    icon-size: 16px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotplug-resident-eject-button {
 | 
			
		||||
    padding: 7px;
 | 
			
		||||
    border-radius: 5px;
 | 
			
		||||
    color: #ccc;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.chat-log-message {
 | 
			
		||||
    color: #888888;
 | 
			
		||||
}
 | 
			
		||||
@@ -1747,7 +1686,11 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
    padding-right: 4px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.chat-notification-scrollview{
 | 
			
		||||
.chat-notification-body-box {
 | 
			
		||||
    spacing: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.chat-notification-scrollview {
 | 
			
		||||
    max-height: 22em;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -2666,8 +2609,7 @@ StScrollBar StButton#vhandle:active {
 | 
			
		||||
    padding-bottom: 0px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#screenShieldNotifications .notification-button,
 | 
			
		||||
#screenShieldNotifications .notification-icon-button {
 | 
			
		||||
#screenShieldNotifications .notification-button {
 | 
			
		||||
    border: 1px rgba(255,255,255,0.5);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -170,17 +170,6 @@ const AutorunManager = new Lang.Class({
 | 
			
		||||
        this._transDispatcher = new AutorunTransientDispatcher(this);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _ensureResidentSource: function() {
 | 
			
		||||
        if (this._residentSource)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        this._residentSource = new AutorunResidentSource(this);
 | 
			
		||||
        let destroyId = this._residentSource.connect('destroy', Lang.bind(this, function() {
 | 
			
		||||
            this._residentSource.disconnect(destroyId);
 | 
			
		||||
            this._residentSource = null;
 | 
			
		||||
        }));
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    enable: function() {
 | 
			
		||||
        this._scanMounts();
 | 
			
		||||
 | 
			
		||||
@@ -189,17 +178,12 @@ const AutorunManager = new Lang.Class({
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    disable: function() {
 | 
			
		||||
        if (this._residentSource)
 | 
			
		||||
            this._residentSource.destroy();
 | 
			
		||||
        this._volumeMonitor.disconnect(this._mountAddedId);
 | 
			
		||||
        this._volumeMonitor.disconnect(this._mountRemovedId);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _processMount: function(mount, hotplug) {
 | 
			
		||||
        let discoverer = new ContentTypeDiscoverer(Lang.bind(this, function(mount, apps, contentTypes) {
 | 
			
		||||
            this._ensureResidentSource();
 | 
			
		||||
            this._residentSource.addMount(mount, apps);
 | 
			
		||||
 | 
			
		||||
            if (hotplug)
 | 
			
		||||
                this._transDispatcher.addMount(mount, apps, contentTypes);
 | 
			
		||||
        }));
 | 
			
		||||
@@ -224,8 +208,6 @@ const AutorunManager = new Lang.Class({
 | 
			
		||||
 | 
			
		||||
    _onMountRemoved: function(monitor, mount) {
 | 
			
		||||
        this._transDispatcher.removeMount(mount);
 | 
			
		||||
        if (this._residentSource)
 | 
			
		||||
            this._residentSource.removeMount(mount);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    ejectMount: function(mount) {
 | 
			
		||||
@@ -288,153 +270,6 @@ const AutorunManager = new Lang.Class({
 | 
			
		||||
    },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const AutorunResidentSource = new Lang.Class({
 | 
			
		||||
    Name: 'AutorunResidentSource',
 | 
			
		||||
    Extends: MessageTray.Source,
 | 
			
		||||
 | 
			
		||||
    _init: function(manager) {
 | 
			
		||||
        this.parent(_("Removable Devices"), 'media-removable');
 | 
			
		||||
        this.resident = true;
 | 
			
		||||
 | 
			
		||||
        this._mounts = [];
 | 
			
		||||
 | 
			
		||||
        this._manager = manager;
 | 
			
		||||
        this._notification = new AutorunResidentNotification(this._manager, this);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _createPolicy: function() {
 | 
			
		||||
        return new MessageTray.NotificationPolicy({ showInLockScreen: false });
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    buildRightClickMenu: function() {
 | 
			
		||||
        return null;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    addMount: function(mount, apps) {
 | 
			
		||||
        if (!shouldAutorunMount(mount, false))
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        let filtered = this._mounts.filter(function (element) {
 | 
			
		||||
            return (element.mount == mount);
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        if (filtered.length != 0)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        let element = { mount: mount, apps: apps };
 | 
			
		||||
        this._mounts.push(element);
 | 
			
		||||
        this._redisplay();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    removeMount: function(mount) {
 | 
			
		||||
        this._mounts =
 | 
			
		||||
            this._mounts.filter(function (element) {
 | 
			
		||||
                return (element.mount != mount);
 | 
			
		||||
            });
 | 
			
		||||
 | 
			
		||||
        this._redisplay();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _redisplay: function() {
 | 
			
		||||
        if (this._mounts.length == 0) {
 | 
			
		||||
            this._notification.destroy();
 | 
			
		||||
            this.destroy();
 | 
			
		||||
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this._notification.updateForMounts(this._mounts);
 | 
			
		||||
 | 
			
		||||
        // add ourselves as a source, and push the notification
 | 
			
		||||
        if (!Main.messageTray.contains(this)) {
 | 
			
		||||
            Main.messageTray.add(this);
 | 
			
		||||
            this.pushNotification(this._notification);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const AutorunResidentNotification = new Lang.Class({
 | 
			
		||||
    Name: 'AutorunResidentNotification',
 | 
			
		||||
    Extends: MessageTray.Notification,
 | 
			
		||||
 | 
			
		||||
    _init: function(manager, source) {
 | 
			
		||||
        this.parent(source, source.title, null, { customContent: true });
 | 
			
		||||
 | 
			
		||||
        // set the notification as resident
 | 
			
		||||
        this.setResident(true);
 | 
			
		||||
 | 
			
		||||
        this._layout = new St.BoxLayout ({ style_class: 'hotplug-resident-box',
 | 
			
		||||
                                           vertical: true });
 | 
			
		||||
        this._manager = manager;
 | 
			
		||||
 | 
			
		||||
        this.addActor(this._layout,
 | 
			
		||||
                      { x_expand: true,
 | 
			
		||||
                        x_fill: true });
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    updateForMounts: function(mounts) {
 | 
			
		||||
        // remove all the layout content
 | 
			
		||||
        this._layout.destroy_all_children();
 | 
			
		||||
 | 
			
		||||
        for (let idx = 0; idx < mounts.length; idx++) {
 | 
			
		||||
            let element = mounts[idx];
 | 
			
		||||
 | 
			
		||||
            let actor = this._itemForMount(element.mount, element.apps);
 | 
			
		||||
            this._layout.add(actor, { x_fill: true,
 | 
			
		||||
                                      expand: true });
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _itemForMount: function(mount, apps) {
 | 
			
		||||
        let item = new St.BoxLayout();
 | 
			
		||||
 | 
			
		||||
        // prepare the mount button content
 | 
			
		||||
        let mountLayout = new St.BoxLayout();
 | 
			
		||||
 | 
			
		||||
        let mountIcon = new St.Icon({ gicon: mount.get_icon(),
 | 
			
		||||
                                      style_class: 'hotplug-resident-mount-icon' });
 | 
			
		||||
        mountLayout.add_actor(mountIcon);
 | 
			
		||||
 | 
			
		||||
        let labelBin = new St.Bin({ y_align: St.Align.MIDDLE });
 | 
			
		||||
        let mountLabel =
 | 
			
		||||
            new St.Label({ text: mount.get_name(),
 | 
			
		||||
                           style_class: 'hotplug-resident-mount-label',
 | 
			
		||||
                           track_hover: true,
 | 
			
		||||
                           reactive: true });
 | 
			
		||||
        labelBin.add_actor(mountLabel);
 | 
			
		||||
        mountLayout.add_actor(labelBin);
 | 
			
		||||
 | 
			
		||||
        let mountButton = new St.Button({ child: mountLayout,
 | 
			
		||||
                                          x_align: St.Align.START,
 | 
			
		||||
                                          x_fill: true,
 | 
			
		||||
                                          style_class: 'hotplug-resident-mount',
 | 
			
		||||
                                          button_mask: St.ButtonMask.ONE });
 | 
			
		||||
        item.add(mountButton, { x_align: St.Align.START,
 | 
			
		||||
                                expand: true });
 | 
			
		||||
 | 
			
		||||
        let ejectIcon = 
 | 
			
		||||
            new St.Icon({ icon_name: 'media-eject-symbolic',
 | 
			
		||||
                          style_class: 'hotplug-resident-eject-icon' });
 | 
			
		||||
 | 
			
		||||
        let ejectButton =
 | 
			
		||||
            new St.Button({ style_class: 'hotplug-resident-eject-button',
 | 
			
		||||
                            button_mask: St.ButtonMask.ONE,
 | 
			
		||||
                            child: ejectIcon });
 | 
			
		||||
        item.add(ejectButton, { x_align: St.Align.END });
 | 
			
		||||
 | 
			
		||||
        // now connect signals
 | 
			
		||||
        mountButton.connect('clicked', Lang.bind(this, function(actor, event) {
 | 
			
		||||
            startAppForMount(apps[0], mount);
 | 
			
		||||
        }));
 | 
			
		||||
 | 
			
		||||
        ejectButton.connect('clicked', Lang.bind(this, function() {
 | 
			
		||||
            this._manager.ejectMount(mount);
 | 
			
		||||
        }));
 | 
			
		||||
 | 
			
		||||
        return item;
 | 
			
		||||
    },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const AutorunTransientDispatcher = new Lang.Class({
 | 
			
		||||
    Name: 'AutorunTransientDispatcher',
 | 
			
		||||
 | 
			
		||||
@@ -559,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;
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -102,15 +102,6 @@ const TelepathyClient = new Lang.Class({
 | 
			
		||||
        this._tpClient.set_handle_channels_func(
 | 
			
		||||
            Lang.bind(this, this._handleChannels));
 | 
			
		||||
 | 
			
		||||
        // Watch subscription requests and connection errors
 | 
			
		||||
        this._subscriptionSource = null;
 | 
			
		||||
        this._accountSource = null;
 | 
			
		||||
 | 
			
		||||
        // Workaround for gjs not supporting GPtrArray in signals.
 | 
			
		||||
        // See BGO bug #653941 for context.
 | 
			
		||||
        this._tpClient.set_contact_list_changed_func(
 | 
			
		||||
            Lang.bind(this, this._contactListChanged));
 | 
			
		||||
 | 
			
		||||
        // Allow other clients (such as Empathy) to pre-empt our channels if
 | 
			
		||||
        // needed
 | 
			
		||||
        this._tpClient.set_delegated_channels_callback(
 | 
			
		||||
@@ -124,17 +115,12 @@ const TelepathyClient = new Lang.Class({
 | 
			
		||||
            throw new Error('Couldn\'t register Telepathy client. Error: \n' + e);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this._accountManagerValidityChangedId = this._accountManager.connect('account-validity-changed',
 | 
			
		||||
                                                                             Lang.bind(this, this._accountValidityChanged));
 | 
			
		||||
 | 
			
		||||
        if (!this._accountManager.is_prepared(Tp.AccountManager.get_feature_quark_core()))
 | 
			
		||||
            this._accountManager.prepare_async(null, Lang.bind(this, this._accountManagerPrepared));
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    disable: function() {
 | 
			
		||||
        this._tpClient.unregister();
 | 
			
		||||
        this._accountManager.disconnect(this._accountManagerValidityChangedId);
 | 
			
		||||
        this._accountManagerValidityChangedId = 0;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _observeChannels: function(observer, account, conn, channels,
 | 
			
		||||
@@ -219,33 +205,6 @@ const TelepathyClient = new Lang.Class({
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _displayRoomInvitation: function(conn, channel, dispatchOp, context) {
 | 
			
		||||
        // We can only approve the rooms if we have been invited to it
 | 
			
		||||
        let selfContact = channel.group_get_self_contact();
 | 
			
		||||
        if (selfContact == null) {
 | 
			
		||||
            context.fail(new Tp.Error({ code: Tp.Error.INVALID_ARGUMENT,
 | 
			
		||||
                                        message: 'Not invited to the room' }));
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        let [invited, inviter, reason, msg] = channel.group_get_local_pending_contact_info(selfContact);
 | 
			
		||||
        if (!invited) {
 | 
			
		||||
            context.fail(new Tp.Error({ code: Tp.Error.INVALID_ARGUMENT,
 | 
			
		||||
                                        message: 'Not invited to the room' }));
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // FIXME: We don't have a 'chat room' icon (bgo #653737) use
 | 
			
		||||
        // system-users for now as Empathy does.
 | 
			
		||||
        let source = new ApproverSource(dispatchOp, _("Invitation"),
 | 
			
		||||
                                        Gio.icon_new_for_string('system-users'));
 | 
			
		||||
        Main.messageTray.add(source);
 | 
			
		||||
 | 
			
		||||
        let notif = new RoomInviteNotification(source, dispatchOp, channel, inviter);
 | 
			
		||||
        source.notify(notif);
 | 
			
		||||
        context.accept();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _approveChannels: function(approver, account, conn, channels,
 | 
			
		||||
                               dispatchOp, context) {
 | 
			
		||||
        let channel = channels[0];
 | 
			
		||||
@@ -259,10 +218,6 @@ const TelepathyClient = new Lang.Class({
 | 
			
		||||
 | 
			
		||||
        if (chanType == Tp.IFACE_CHANNEL_TYPE_TEXT)
 | 
			
		||||
            this._approveTextChannel(account, conn, channel, dispatchOp, context);
 | 
			
		||||
        else if (chanType == Tp.IFACE_CHANNEL_TYPE_CALL)
 | 
			
		||||
            this._approveCall(account, conn, channel, dispatchOp, context);
 | 
			
		||||
        else if (chanType == Tp.IFACE_CHANNEL_TYPE_FILE_TRANSFER)
 | 
			
		||||
            this._approveFileTransfer(account, conn, channel, dispatchOp, context);
 | 
			
		||||
        else
 | 
			
		||||
            context.fail(new Tp.Error({ code: Tp.Error.INVALID_ARGUMENT,
 | 
			
		||||
                                        message: 'Unsupported channel type' }));
 | 
			
		||||
@@ -283,45 +238,9 @@ const TelepathyClient = new Lang.Class({
 | 
			
		||||
                }}));
 | 
			
		||||
 | 
			
		||||
            context.accept();
 | 
			
		||||
        } else {
 | 
			
		||||
            this._displayRoomInvitation(conn, channel, dispatchOp, context);
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _approveCall: function(account, conn, channel, dispatchOp, context) {
 | 
			
		||||
        let isVideo = false;
 | 
			
		||||
 | 
			
		||||
        let props = channel.borrow_immutable_properties();
 | 
			
		||||
 | 
			
		||||
        if (props[Tp.PROP_CHANNEL_TYPE_CALL_INITIAL_VIDEO])
 | 
			
		||||
          isVideo = true;
 | 
			
		||||
 | 
			
		||||
        // We got the TpContact
 | 
			
		||||
        let source = new ApproverSource(dispatchOp, _("Call"), isVideo ?
 | 
			
		||||
                                        Gio.icon_new_for_string('camera-web') :
 | 
			
		||||
                                        Gio.icon_new_for_string('audio-input-microphone'));
 | 
			
		||||
        Main.messageTray.add(source);
 | 
			
		||||
 | 
			
		||||
        let notif = new AudioVideoNotification(source, dispatchOp, channel,
 | 
			
		||||
            channel.get_target_contact(), isVideo);
 | 
			
		||||
        source.notify(notif);
 | 
			
		||||
        context.accept();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _approveFileTransfer: function(account, conn, channel, dispatchOp, context) {
 | 
			
		||||
        // Use the icon of the file being transferred
 | 
			
		||||
        let gicon = Gio.content_type_get_icon(channel.get_mime_type());
 | 
			
		||||
 | 
			
		||||
        // We got the TpContact
 | 
			
		||||
        let source = new ApproverSource(dispatchOp, _("File Transfer"), gicon);
 | 
			
		||||
        Main.messageTray.add(source);
 | 
			
		||||
 | 
			
		||||
        let notif = new FileTransferNotification(source, dispatchOp, channel,
 | 
			
		||||
            channel.get_target_contact());
 | 
			
		||||
        source.notify(notif);
 | 
			
		||||
        context.accept();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _delegatedChannelsCb: function(client, channels) {
 | 
			
		||||
        // Nothing to do as we don't make a distinction between observed and
 | 
			
		||||
        // handled channels.
 | 
			
		||||
@@ -329,105 +248,7 @@ const TelepathyClient = new Lang.Class({
 | 
			
		||||
 | 
			
		||||
    _accountManagerPrepared: function(am, result) {
 | 
			
		||||
        am.prepare_finish(result);
 | 
			
		||||
 | 
			
		||||
        let accounts = am.get_valid_accounts();
 | 
			
		||||
        for (let i = 0; i < accounts.length; i++) {
 | 
			
		||||
            this._accountValidityChanged(am, accounts[i], true);
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _accountValidityChanged: function(am, account, valid) {
 | 
			
		||||
        if (!valid)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        // It would be better to connect to "status-changed" but we cannot.
 | 
			
		||||
        // See discussion in https://bugzilla.gnome.org/show_bug.cgi?id=654159
 | 
			
		||||
        account.connect("notify::connection-status",
 | 
			
		||||
                        Lang.bind(this, this._accountConnectionStatusNotifyCb));
 | 
			
		||||
 | 
			
		||||
        account.connect('notify::connection',
 | 
			
		||||
                        Lang.bind(this, this._connectionChanged));
 | 
			
		||||
        this._connectionChanged(account);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _connectionChanged: function(account) {
 | 
			
		||||
        let conn = account.get_connection();
 | 
			
		||||
        if (conn == null)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        this._tpClient.grab_contact_list_changed(conn);
 | 
			
		||||
        if (conn.get_contact_list_state() == Tp.ContactListState.SUCCESS) {
 | 
			
		||||
            this._contactListChanged(conn, conn.dup_contact_list(), []);
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _contactListChanged: function(conn, added, removed) {
 | 
			
		||||
        for (let i = 0; i < added.length; i++) {
 | 
			
		||||
            let contact = added[i];
 | 
			
		||||
 | 
			
		||||
            contact.connect('subscription-states-changed',
 | 
			
		||||
                            Lang.bind(this, this._subscriptionStateChanged));
 | 
			
		||||
            this._subscriptionStateChanged(contact);
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _subscriptionStateChanged: function(contact) {
 | 
			
		||||
        if (contact.get_publish_state() != Tp.SubscriptionState.ASK)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        /* Implicitly accept publish requests if contact is already subscribed */
 | 
			
		||||
        if (contact.get_subscribe_state() == Tp.SubscriptionState.YES ||
 | 
			
		||||
            contact.get_subscribe_state() == Tp.SubscriptionState.ASK) {
 | 
			
		||||
 | 
			
		||||
            contact.authorize_publication_async(function(src, result) {
 | 
			
		||||
                src.authorize_publication_finish(result)});
 | 
			
		||||
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /* Display notification to ask user to accept/reject request */
 | 
			
		||||
        let source = this._ensureAppSource();
 | 
			
		||||
 | 
			
		||||
        let notif = new SubscriptionRequestNotification(source, contact);
 | 
			
		||||
        source.notify(notif);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _accountConnectionStatusNotifyCb: function(account) {
 | 
			
		||||
        let connectionError = account.connection_error;
 | 
			
		||||
 | 
			
		||||
        if (account.connection_status != Tp.ConnectionStatus.DISCONNECTED ||
 | 
			
		||||
            connectionError == Tp.error_get_dbus_name(Tp.Error.CANCELLED)) {
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        let notif = this._accountNotifications[account.get_object_path()];
 | 
			
		||||
        if (notif)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        /* Display notification that account failed to connect */
 | 
			
		||||
        let source = this._ensureAppSource();
 | 
			
		||||
 | 
			
		||||
        notif = new AccountNotification(source, account, connectionError);
 | 
			
		||||
        this._accountNotifications[account.get_object_path()] = notif;
 | 
			
		||||
        notif.connect('destroy', Lang.bind(this, function() {
 | 
			
		||||
            delete this._accountNotifications[account.get_object_path()];
 | 
			
		||||
        }));
 | 
			
		||||
        source.notify(notif);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _ensureAppSource: function() {
 | 
			
		||||
        if (this._appSource == null) {
 | 
			
		||||
            this._appSource = new MessageTray.Source(_("Chat"), 'empathy');
 | 
			
		||||
            this._appSource.policy = new MessageTray.NotificationApplicationPolicy('empathy');
 | 
			
		||||
 | 
			
		||||
            Main.messageTray.add(this._appSource);
 | 
			
		||||
            this._appSource.connect('destroy', Lang.bind(this, function () {
 | 
			
		||||
                this._appSource = null;
 | 
			
		||||
            }));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return this._appSource;
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const ChatSource = new Lang.Class({
 | 
			
		||||
@@ -545,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() {
 | 
			
		||||
@@ -737,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 += ' <i>(' + GLib.markup_escape_text(message, -1) + ')</i>';
 | 
			
		||||
@@ -764,8 +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.setResident(true);
 | 
			
		||||
        this.parent(source, source.title, null, { secondaryGIcon: source.getSecondaryIcon() });
 | 
			
		||||
 | 
			
		||||
        this._responseEntry = new St.Entry({ style_class: 'chat-response',
 | 
			
		||||
                                             can_focus: true });
 | 
			
		||||
@@ -781,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);
 | 
			
		||||
@@ -826,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 ?
 | 
			
		||||
@@ -864,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)
 | 
			
		||||
@@ -896,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;
 | 
			
		||||
 | 
			
		||||
@@ -910,14 +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.updated();
 | 
			
		||||
        this._bodyBox.add_child(this._lastMessageBox);
 | 
			
		||||
 | 
			
		||||
        let timestamp = props.timestamp;
 | 
			
		||||
        this._history.unshift({ actor: body, time: timestamp,
 | 
			
		||||
@@ -1052,7 +871,7 @@ const ChatNotification = new Lang.Class({
 | 
			
		||||
                                   group: 'meta',
 | 
			
		||||
                                   styles: ['chat-meta-message'] });
 | 
			
		||||
 | 
			
		||||
        this.update(newAlias, null, { customContent: true });
 | 
			
		||||
        this.update(newAlias);
 | 
			
		||||
 | 
			
		||||
        this._filterMessages();
 | 
			
		||||
    },
 | 
			
		||||
@@ -1105,359 +924,4 @@ const ChatNotification = new Lang.Class({
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const ApproverSource = new Lang.Class({
 | 
			
		||||
    Name: 'ApproverSource',
 | 
			
		||||
    Extends: MessageTray.Source,
 | 
			
		||||
 | 
			
		||||
    _init: function(dispatchOp, text, gicon) {
 | 
			
		||||
        this._gicon = gicon;
 | 
			
		||||
 | 
			
		||||
        this.parent(text);
 | 
			
		||||
 | 
			
		||||
        this._dispatchOp = dispatchOp;
 | 
			
		||||
 | 
			
		||||
        // Destroy the source if the channel dispatch operation is invalidated
 | 
			
		||||
        // as we can't approve any more.
 | 
			
		||||
        this._invalidId = dispatchOp.connect('invalidated',
 | 
			
		||||
                                             Lang.bind(this, function(domain, code, msg) {
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _createPolicy: function() {
 | 
			
		||||
        return new MessageTray.NotificationApplicationPolicy('empathy');
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    destroy: function() {
 | 
			
		||||
        if (this._invalidId != 0) {
 | 
			
		||||
            this._dispatchOp.disconnect(this._invalidId);
 | 
			
		||||
            this._invalidId = 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this.parent();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    getIcon: function() {
 | 
			
		||||
        return this._gicon;
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const RoomInviteNotification = new Lang.Class({
 | 
			
		||||
    Name: 'RoomInviteNotification',
 | 
			
		||||
    Extends: MessageTray.Notification,
 | 
			
		||||
 | 
			
		||||
    _init: function(source, dispatchOp, channel, inviter) {
 | 
			
		||||
        this.parent(source,
 | 
			
		||||
                    /* translators: argument is a room name like
 | 
			
		||||
                     * room@jabber.org for example. */
 | 
			
		||||
                    _("Invitation to %s").format(channel.get_identifier()),
 | 
			
		||||
                    null,
 | 
			
		||||
                    { customContent: true });
 | 
			
		||||
        this.setResident(true);
 | 
			
		||||
 | 
			
		||||
        /* translators: first argument is the name of a contact and the second
 | 
			
		||||
         * one the name of a room. "Alice is inviting you to join room@jabber.org
 | 
			
		||||
         * for example. */
 | 
			
		||||
        this.addBody(_("%s is inviting you to join %s").format(inviter.get_alias(), channel.get_identifier()));
 | 
			
		||||
 | 
			
		||||
        this.addAction(_("Decline"), Lang.bind(this, function() {
 | 
			
		||||
            dispatchOp.leave_channels_async(Tp.ChannelGroupChangeReason.NONE, '', function(src, result) {
 | 
			
		||||
                src.leave_channels_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
        this.addAction(_("Accept"), Lang.bind(this, function() {
 | 
			
		||||
            dispatchOp.handle_with_time_async('', global.get_current_time(), function(src, result) {
 | 
			
		||||
                src.handle_with_time_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
// Audio Video
 | 
			
		||||
const AudioVideoNotification = new Lang.Class({
 | 
			
		||||
    Name: 'AudioVideoNotification',
 | 
			
		||||
    Extends: MessageTray.Notification,
 | 
			
		||||
 | 
			
		||||
    _init: function(source, dispatchOp, channel, contact, isVideo) {
 | 
			
		||||
        let title = '';
 | 
			
		||||
 | 
			
		||||
        if (isVideo)
 | 
			
		||||
             /* translators: argument is a contact name like Alice for example. */
 | 
			
		||||
            title = _("Video call from %s").format(contact.get_alias());
 | 
			
		||||
        else
 | 
			
		||||
             /* translators: argument is a contact name like Alice for example. */
 | 
			
		||||
            title = _("Call from %s").format(contact.get_alias());
 | 
			
		||||
 | 
			
		||||
        this.parent(source, title, null, { customContent: true });
 | 
			
		||||
        this.setResident(true);
 | 
			
		||||
 | 
			
		||||
        this.setUrgency(MessageTray.Urgency.CRITICAL);
 | 
			
		||||
 | 
			
		||||
        this.addAction(_("Decline"), Lang.bind(this, function() {
 | 
			
		||||
            dispatchOp.leave_channels_async(Tp.ChannelGroupChangeReason.NONE, '', function(src, result) {
 | 
			
		||||
                src.leave_channels_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
        /* translators: this is a button label (verb), not a noun */
 | 
			
		||||
        this.addAction(_("Answer"), Lang.bind(this, function() {
 | 
			
		||||
            dispatchOp.handle_with_time_async('', global.get_current_time(), function(src, result) {
 | 
			
		||||
                src.handle_with_time_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
// File Transfer
 | 
			
		||||
const FileTransferNotification = new Lang.Class({
 | 
			
		||||
    Name: 'FileTransferNotification',
 | 
			
		||||
    Extends: MessageTray.Notification,
 | 
			
		||||
 | 
			
		||||
    _init: function(source, dispatchOp, channel, contact) {
 | 
			
		||||
        this.parent(source,
 | 
			
		||||
                    /* To translators: The first parameter is
 | 
			
		||||
                     * the contact's alias and the second one is the
 | 
			
		||||
                     * file name. The string will be something
 | 
			
		||||
                     * like: "Alice is sending you test.ogg"
 | 
			
		||||
                     */
 | 
			
		||||
                    _("%s is sending you %s").format(contact.get_alias(),
 | 
			
		||||
                                                     channel.get_filename()),
 | 
			
		||||
                    null,
 | 
			
		||||
                    { customContent: true });
 | 
			
		||||
        this.setResident(true);
 | 
			
		||||
 | 
			
		||||
        this.addAction(_("Decline"), Lang.bind(this, function() {
 | 
			
		||||
            dispatchOp.leave_channels_async(Tp.ChannelGroupChangeReason.NONE, '', function(src, result) {
 | 
			
		||||
                src.leave_channels_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
        this.addAction(_("Accept"), Lang.bind(this, function() {
 | 
			
		||||
            dispatchOp.handle_with_time_async('', global.get_current_time(), function(src, result) {
 | 
			
		||||
                src.handle_with_time_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
// Subscription request
 | 
			
		||||
const SubscriptionRequestNotification = new Lang.Class({
 | 
			
		||||
    Name: 'SubscriptionRequestNotification',
 | 
			
		||||
    Extends: MessageTray.Notification,
 | 
			
		||||
 | 
			
		||||
    _init: function(source, contact) {
 | 
			
		||||
        this.parent(source,
 | 
			
		||||
                    /* To translators: The parameter is the contact's alias */
 | 
			
		||||
                    _("%s would like permission to see when you are online").format(contact.get_alias()),
 | 
			
		||||
                    null, { customContent: true });
 | 
			
		||||
 | 
			
		||||
        this._contact = contact;
 | 
			
		||||
        this._connection = contact.get_connection();
 | 
			
		||||
 | 
			
		||||
        let layout = new St.BoxLayout({ vertical: false });
 | 
			
		||||
 | 
			
		||||
        // Display avatar
 | 
			
		||||
        let iconBox = new St.Bin({ style_class: 'avatar-box' });
 | 
			
		||||
        iconBox._size = 48;
 | 
			
		||||
 | 
			
		||||
        let textureCache = St.TextureCache.get_default();
 | 
			
		||||
        let file = contact.get_avatar_file();
 | 
			
		||||
 | 
			
		||||
        if (file) {
 | 
			
		||||
            let uri = file.get_uri();
 | 
			
		||||
            let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor;
 | 
			
		||||
            iconBox.child = textureCache.load_uri_async(uri, iconBox._size, iconBox._size, scaleFactor);
 | 
			
		||||
        }
 | 
			
		||||
        else {
 | 
			
		||||
            iconBox.child = new St.Icon({ icon_name: 'avatar-default',
 | 
			
		||||
                                          icon_size: iconBox._size });
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        layout.add(iconBox);
 | 
			
		||||
 | 
			
		||||
        // subscription request message
 | 
			
		||||
        let label = new St.Label({ style_class: 'subscription-message',
 | 
			
		||||
                                   text: contact.get_publish_request() });
 | 
			
		||||
 | 
			
		||||
        layout.add(label);
 | 
			
		||||
 | 
			
		||||
        this.addActor(layout);
 | 
			
		||||
 | 
			
		||||
        this.addAction(_("Decline"), Lang.bind(this, function() {
 | 
			
		||||
            contact.remove_async(function(src, result) {
 | 
			
		||||
                src.remove_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
        }));
 | 
			
		||||
        this.addAction(_("Accept"), Lang.bind(this, function() {
 | 
			
		||||
            // Authorize the contact and request to see his status as well
 | 
			
		||||
            contact.authorize_publication_async(function(src, result) {
 | 
			
		||||
                src.authorize_publication_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
 | 
			
		||||
            contact.request_subscription_async('', function(src, result) {
 | 
			
		||||
                src.request_subscription_finish(result);
 | 
			
		||||
            });
 | 
			
		||||
        }));
 | 
			
		||||
 | 
			
		||||
        this._changedId = contact.connect('subscription-states-changed',
 | 
			
		||||
            Lang.bind(this, this._subscriptionStatesChangedCb));
 | 
			
		||||
        this._invalidatedId = this._connection.connect('invalidated',
 | 
			
		||||
            Lang.bind(this, this.destroy));
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    destroy: function() {
 | 
			
		||||
        if (this._changedId != 0) {
 | 
			
		||||
            this._contact.disconnect(this._changedId);
 | 
			
		||||
            this._changedId = 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (this._invalidatedId != 0) {
 | 
			
		||||
            this._connection.disconnect(this._invalidatedId);
 | 
			
		||||
            this._invalidatedId = 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this.parent();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _subscriptionStatesChangedCb: function(contact, subscribe, publish, msg) {
 | 
			
		||||
        // Destroy the notification if the subscription request has been
 | 
			
		||||
        // answered
 | 
			
		||||
        if (publish != Tp.SubscriptionState.ASK)
 | 
			
		||||
            this.destroy();
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
// Messages from empathy/libempathy/empathy-utils.c
 | 
			
		||||
// create_errors_to_message_hash()
 | 
			
		||||
 | 
			
		||||
/* Translator note: these should be the same messages that are
 | 
			
		||||
 * used in Empathy, so just copy and paste from there. */
 | 
			
		||||
let _connectionErrorMessages = {};
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.NETWORK_ERROR)]
 | 
			
		||||
  = _("Network error");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.AUTHENTICATION_FAILED)]
 | 
			
		||||
  = _("Authentication failed");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.ENCRYPTION_ERROR)]
 | 
			
		||||
  = _("Encryption error");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_NOT_PROVIDED)]
 | 
			
		||||
  = _("Certificate not provided");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_UNTRUSTED)]
 | 
			
		||||
  = _("Certificate untrusted");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_EXPIRED)]
 | 
			
		||||
  = _("Certificate expired");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_NOT_ACTIVATED)]
 | 
			
		||||
  = _("Certificate not activated");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_HOSTNAME_MISMATCH)]
 | 
			
		||||
  = _("Certificate hostname mismatch");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_FINGERPRINT_MISMATCH)]
 | 
			
		||||
  = _("Certificate fingerprint mismatch");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_SELF_SIGNED)]
 | 
			
		||||
  = _("Certificate self-signed");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CANCELLED)]
 | 
			
		||||
  = _("Status is set to offline");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.ENCRYPTION_NOT_AVAILABLE)]
 | 
			
		||||
  = _("Encryption is not available");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_INVALID)]
 | 
			
		||||
  = _("Certificate is invalid");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CONNECTION_REFUSED)]
 | 
			
		||||
  = _("Connection has been refused");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CONNECTION_FAILED)]
 | 
			
		||||
  = _("Connection can't be established");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CONNECTION_LOST)]
 | 
			
		||||
  = _("Connection has been lost");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.ALREADY_CONNECTED)]
 | 
			
		||||
  = _("This account is already connected to the server");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CONNECTION_REPLACED)]
 | 
			
		||||
  = _("Connection has been replaced by a new connection using the same resource");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.REGISTRATION_EXISTS)]
 | 
			
		||||
  = _("The account already exists on the server");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.SERVICE_BUSY)]
 | 
			
		||||
  = _("Server is currently too busy to handle the connection");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_REVOKED)]
 | 
			
		||||
  = _("Certificate has been revoked");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_INSECURE)]
 | 
			
		||||
  = _("Certificate uses an insecure cipher algorithm or is cryptographically weak");
 | 
			
		||||
_connectionErrorMessages[Tp.error_get_dbus_name(Tp.Error.CERT_LIMIT_EXCEEDED)]
 | 
			
		||||
  = _("The length of the server certificate, or the depth of the server certificate chain, exceed the limits imposed by the cryptography library");
 | 
			
		||||
_connectionErrorMessages['org.freedesktop.DBus.Error.NoReply']
 | 
			
		||||
  = _("Internal error");
 | 
			
		||||
 | 
			
		||||
const AccountNotification = new Lang.Class({
 | 
			
		||||
    Name: 'AccountNotification',
 | 
			
		||||
    Extends: MessageTray.Notification,
 | 
			
		||||
 | 
			
		||||
    _init: function(source, account, connectionError) {
 | 
			
		||||
        this.parent(source,
 | 
			
		||||
                    /* translators: argument is the account name, like
 | 
			
		||||
                     * name@jabber.org for example. */
 | 
			
		||||
                    _("Unable to connect to %s").format(account.get_display_name()),
 | 
			
		||||
                    this._getMessage(connectionError));
 | 
			
		||||
 | 
			
		||||
        this._account = account;
 | 
			
		||||
 | 
			
		||||
        this.addAction(_("View account"), Lang.bind(this, function() {
 | 
			
		||||
            let cmd = 'empathy-accounts --select-account=' +
 | 
			
		||||
                account.get_path_suffix();
 | 
			
		||||
            let app_info = Gio.app_info_create_from_commandline(cmd, null, 0);
 | 
			
		||||
            app_info.launch([], global.create_app_launch_context(0, -1));
 | 
			
		||||
        }));
 | 
			
		||||
 | 
			
		||||
        this._enabledId = account.connect('notify::enabled',
 | 
			
		||||
                                          Lang.bind(this, function() {
 | 
			
		||||
                                              if (!account.is_enabled())
 | 
			
		||||
                                                  this.destroy();
 | 
			
		||||
                                          }));
 | 
			
		||||
 | 
			
		||||
        this._invalidatedId = account.connect('invalidated',
 | 
			
		||||
                                              Lang.bind(this, this.destroy));
 | 
			
		||||
 | 
			
		||||
        this._connectionStatusId = account.connect('notify::connection-status',
 | 
			
		||||
            Lang.bind(this, function() {
 | 
			
		||||
                let status = account.connection_status;
 | 
			
		||||
                if (status == Tp.ConnectionStatus.CONNECTED) {
 | 
			
		||||
                    this.destroy();
 | 
			
		||||
                } else if (status == Tp.ConnectionStatus.DISCONNECTED) {
 | 
			
		||||
                    let connectionError = account.connection_error;
 | 
			
		||||
 | 
			
		||||
                    if (connectionError == Tp.error_get_dbus_name(Tp.Error.CANCELLED))
 | 
			
		||||
                        this.destroy();
 | 
			
		||||
                    else
 | 
			
		||||
                        this.update(this.title, this._getMessage(connectionError));
 | 
			
		||||
                }
 | 
			
		||||
            }));
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _getMessage: function(connectionError) {
 | 
			
		||||
        let message;
 | 
			
		||||
        if (connectionError in _connectionErrorMessages) {
 | 
			
		||||
            message = _connectionErrorMessages[connectionError];
 | 
			
		||||
        } else {
 | 
			
		||||
            message = _("Unknown reason");
 | 
			
		||||
        }
 | 
			
		||||
        return message;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    destroy: function() {
 | 
			
		||||
        if (this._enabledId != 0) {
 | 
			
		||||
            this._account.disconnect(this._enabledId);
 | 
			
		||||
            this._enabledId = 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (this._invalidatedId != 0) {
 | 
			
		||||
            this._account.disconnect(this._invalidatedId);
 | 
			
		||||
            this._invalidatedId = 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (this._connectionStatusId != 0) {
 | 
			
		||||
            this._account.disconnect(this._connectionStatusId);
 | 
			
		||||
            this._connectionStatusId = 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this.parent();
 | 
			
		||||
    }
 | 
			
		||||
});
 | 
			
		||||
const Component = TelepathyClient;
 | 
			
		||||
 
 | 
			
		||||
@@ -111,7 +111,6 @@ const FocusGrabber = new Lang.Class({
 | 
			
		||||
        if (this._focused)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        this._prevFocusedWindow = global.display.focus_window;
 | 
			
		||||
        this._prevKeyFocusActor = global.stage.get_key_focus();
 | 
			
		||||
 | 
			
		||||
        this._focusActorChangedId = global.stage.connect('notify::key-focus', Lang.bind(this, this._focusActorChanged));
 | 
			
		||||
@@ -234,6 +233,10 @@ const URLHighlighter = new Lang.Class({
 | 
			
		||||
        }));
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    hasText: function() {
 | 
			
		||||
        return !!this._text;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    setMarkup: function(text, allowMarkup) {
 | 
			
		||||
        text = text ? _fixMarkup(text, allowMarkup) : '';
 | 
			
		||||
        this._text = text;
 | 
			
		||||
@@ -440,24 +443,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() and
 | 
			
		||||
// addBody() methods. 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
 | 
			
		||||
@@ -473,8 +463,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
 | 
			
		||||
@@ -482,15 +470,12 @@ const NotificationApplicationPolicy = new Lang.Class({
 | 
			
		||||
const Notification = new Lang.Class({
 | 
			
		||||
    Name: 'Notification',
 | 
			
		||||
 | 
			
		||||
    ICON_SIZE: 24,
 | 
			
		||||
 | 
			
		||||
    IMAGE_SIZE: 125,
 | 
			
		||||
    ICON_SIZE: 32,
 | 
			
		||||
 | 
			
		||||
    _init: function(source, title, banner, params) {
 | 
			
		||||
        this.source = source;
 | 
			
		||||
        this.title = title;
 | 
			
		||||
        this.urgency = Urgency.NORMAL;
 | 
			
		||||
        this.resident = false;
 | 
			
		||||
        // 'transient' is a reserved keyword in JS, so we have to use an alternate variable name
 | 
			
		||||
        this.isTransient = false;
 | 
			
		||||
        this.isMusic = false;
 | 
			
		||||
@@ -499,59 +484,89 @@ 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._imageBin = null;
 | 
			
		||||
        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. I am a bad artist:
 | 
			
		||||
        //
 | 
			
		||||
        //      ,. this._iconBin         ,. this._titleLabel
 | 
			
		||||
        //      |        ,-- this._second|ryIconBin
 | 
			
		||||
        // .----|--------|---------------|-----------.
 | 
			
		||||
        // | .----. | .----.-------------------. | X |
 | 
			
		||||
        // | |    | | |    |                   |-|----- this._titleBox
 | 
			
		||||
        // | '....' | '....'...................' |   |
 | 
			
		||||
        // |        |                            |   |- this._hbox
 | 
			
		||||
        // |        |        this._bodyBin       |   |-.
 | 
			
		||||
        // |________|____________________________|___| |- this.actor
 | 
			
		||||
        // | this._actionArea                        |-'
 | 
			
		||||
        // |_________________________________________|
 | 
			
		||||
        // | this._buttonBox                         |
 | 
			
		||||
        // |_________________________________________|
 | 
			
		||||
 | 
			
		||||
        this.actor = new St.BoxLayout({ vertical: true,
 | 
			
		||||
                                        style_class: 'notification',
 | 
			
		||||
                                        accessible_role: Atk.Role.NOTIFICATION });
 | 
			
		||||
        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);
 | 
			
		||||
        this._mainButton = new St.Button({ style_class: 'notification-main-button',
 | 
			
		||||
                                           can_focus: true,
 | 
			
		||||
                                           x_fill: true, y_fill: true });
 | 
			
		||||
        this._mainButton.connect('clicked', Lang.bind(this, this._onClicked));
 | 
			
		||||
        this.actor.add_child(this._mainButton);
 | 
			
		||||
 | 
			
		||||
        // 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, title/body and close button
 | 
			
		||||
        this._hbox = new St.BoxLayout({ style_class: 'notification-main-content' });
 | 
			
		||||
        this._mainButton.child = this._hbox;
 | 
			
		||||
 | 
			
		||||
        // This is an empty cell that overlaps with this._bannerBox cell to ensure
 | 
			
		||||
        // that this._bannerBox cell expands horizontally, while not forcing the
 | 
			
		||||
        // this._imageBin that is also in col: 2 to expand horizontally.
 | 
			
		||||
        this._table.add(new St.Bin(), { row: 0,
 | 
			
		||||
                                        col: 2,
 | 
			
		||||
                                        y_expand: false,
 | 
			
		||||
                                        y_fill: false });
 | 
			
		||||
        this._iconBin = new St.Bin({ y_align: St.Align.START });
 | 
			
		||||
        this._hbox.add_child(this._iconBin);
 | 
			
		||||
 | 
			
		||||
        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._titleBodyBox = new St.BoxLayout({ style_class: 'notification-title-body-box',
 | 
			
		||||
                                                vertical: true });
 | 
			
		||||
        this._titleBodyBox.set_x_expand(true);
 | 
			
		||||
        this._hbox.add_child(this._titleBodyBox);
 | 
			
		||||
 | 
			
		||||
        this._closeButton = new St.Button({ style_class: 'notification-close-button',
 | 
			
		||||
                                            can_focus: true });
 | 
			
		||||
        this._closeButton.set_y_align(Clutter.ActorAlign.START);
 | 
			
		||||
        this._closeButton.set_y_expand(true);
 | 
			
		||||
        this._closeButton.child = new St.Icon({ icon_name: 'window-close-symbolic', icon_size: 16 });
 | 
			
		||||
        this._closeButton.connect('clicked', Lang.bind(this, this._onCloseClicked));
 | 
			
		||||
        this._hbox.add_child(this._closeButton);
 | 
			
		||||
 | 
			
		||||
        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._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._bodyBin.child = this._bodyUrlHighlighter.actor;
 | 
			
		||||
 | 
			
		||||
        this._actionAreaBin = new St.Bin({ style_class: 'notification-action-area',
 | 
			
		||||
                                           x_expand: true, y_expand: true });
 | 
			
		||||
        this.actor.add_child(this._actionAreaBin);
 | 
			
		||||
 | 
			
		||||
        this._buttonBox = new St.BoxLayout({ style_class: 'notification-button-box',
 | 
			
		||||
                                             x_expand: true, y_expand: true });
 | 
			
		||||
        global.focus_manager.add_group(this._buttonBox);
 | 
			
		||||
        this.actor.add_child(this._buttonBox);
 | 
			
		||||
 | 
			
		||||
        // If called with only one argument we assume the caller
 | 
			
		||||
        // will call .update() later on. This is the case of
 | 
			
		||||
@@ -559,6 +574,31 @@ 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);
 | 
			
		||||
 | 
			
		||||
        if (this.expanded) {
 | 
			
		||||
            this._titleLabel.clutter_text.line_wrap = true;
 | 
			
		||||
            this._titleLabel.clutter_text.ellipsize = Pango.EllipsizeMode.NONE;
 | 
			
		||||
            this._bodyUrlHighlighter.actor.clutter_text.line_wrap = true;
 | 
			
		||||
            this._bodyUrlHighlighter.actor.clutter_text.ellipsize = Pango.EllipsizeMode.NONE;
 | 
			
		||||
        } else {
 | 
			
		||||
            this._titleLabel.clutter_text.line_wrap = false;
 | 
			
		||||
            this._titleLabel.clutter_text.ellipsize = Pango.EllipsizeMode.END;
 | 
			
		||||
            this._bodyUrlHighlighter.actor.clutter_text.line_wrap = false;
 | 
			
		||||
            this._bodyUrlHighlighter.actor.clutter_text.ellipsize = Pango.EllipsizeMode.END;
 | 
			
		||||
        }
 | 
			
		||||
        this.enableScrolling(this.expanded);
 | 
			
		||||
 | 
			
		||||
        this._bodyUrlHighlighter.actor.visible = this._bodyUrlHighlighter.hasText();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    // update:
 | 
			
		||||
@@ -570,52 +610,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 (params.clear)
 | 
			
		||||
            this.unsetImage();
 | 
			
		||||
 | 
			
		||||
        if (!this._scrollArea && !this._actionArea && !this._imageBin)
 | 
			
		||||
            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,
 | 
			
		||||
@@ -624,29 +643,29 @@ 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;
 | 
			
		||||
        title = title ? _fixMarkup(title.replace(/\n/g, ' '), false) : '';
 | 
			
		||||
        this._titleLabel.clutter_text.set_markup('<b>' + title + '</b>');
 | 
			
		||||
 | 
			
		||||
        let titleDirection;
 | 
			
		||||
        if (Pango.find_base_dir(title, -1) == Pango.Direction.RTL)
 | 
			
		||||
            this._titleDirection = Clutter.TextDirection.RTL;
 | 
			
		||||
            titleDirection = Clutter.TextDirection.RTL;
 | 
			
		||||
        else
 | 
			
		||||
            this._titleDirection = Clutter.TextDirection.LTR;
 | 
			
		||||
            titleDirection = Clutter.TextDirection.LTR;
 | 
			
		||||
 | 
			
		||||
        // Let the title's text direction control the overall direction
 | 
			
		||||
        // of the notification - in case where different scripts are used
 | 
			
		||||
@@ -654,24 +673,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(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) {
 | 
			
		||||
@@ -680,71 +684,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._updateLastColumnSettings();
 | 
			
		||||
        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();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    // addBody:
 | 
			
		||||
    // @text: the text
 | 
			
		||||
    // @markup: %true if @text contains pango markup
 | 
			
		||||
    // @style: style to use when adding the actor containing the text
 | 
			
		||||
    //
 | 
			
		||||
    // Adds a multi-line label containing @text to the notification.
 | 
			
		||||
    //
 | 
			
		||||
    // Return value: the newly-added label
 | 
			
		||||
    addBody: function(text, markup, style) {
 | 
			
		||||
        let label = new URLHighlighter(text, true, markup);
 | 
			
		||||
 | 
			
		||||
        this.addActor(label.actor, style);
 | 
			
		||||
        return label.actor;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _addBannerBody: function() {
 | 
			
		||||
        if (this.bannerBodyText && !this._bannerBodyAdded) {
 | 
			
		||||
            this._bannerBodyAdded = true;
 | 
			
		||||
            this.addBody(this.bannerBodyText, this.bannerBodyMarkup);
 | 
			
		||||
        }
 | 
			
		||||
        let scrollPolicy = enableScrolling ? Gtk.PolicyType.AUTOMATIC : Gtk.PolicyType.NEVER;
 | 
			
		||||
        this._bodyScrollArea.vscrollbar_policy = scrollPolicy;
 | 
			
		||||
        this._bodyScrollArea.enable_mouse_scrolling = enableScrolling;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    // scrollTo:
 | 
			
		||||
@@ -752,112 +703,32 @@ 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) {
 | 
			
		||||
            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._updateLastColumnSettings();
 | 
			
		||||
        this.updated();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _updateLastColumnSettings: function() {
 | 
			
		||||
        if (this._scrollArea)
 | 
			
		||||
            this._table.child_set(this._scrollArea, { col: this._imageBin ? 2 : 1,
 | 
			
		||||
                                                      col_span: this._imageBin ? 1 : 2 });
 | 
			
		||||
    setActionArea: function(actor) {
 | 
			
		||||
        if (this._actionArea)
 | 
			
		||||
            this._table.child_set(this._actionArea, { col: this._imageBin ? 2 : 1,
 | 
			
		||||
                                                      col_span: this._imageBin ? 1 : 2 });
 | 
			
		||||
    },
 | 
			
		||||
            this._actionArea.destroy();
 | 
			
		||||
 | 
			
		||||
    setImage: function(image) {
 | 
			
		||||
        this.unsetImage();
 | 
			
		||||
 | 
			
		||||
        if (!image)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        this._imageBin = new St.Bin({ opacity: 230,
 | 
			
		||||
                                      child: image,
 | 
			
		||||
                                      visible: this.expanded });
 | 
			
		||||
 | 
			
		||||
        this._table.add_style_class_name('multi-line-notification');
 | 
			
		||||
        this._table.add_style_class_name('notification-with-image');
 | 
			
		||||
        this._addBannerBody();
 | 
			
		||||
        this._updateLastColumnSettings();
 | 
			
		||||
        this._table.add(this._imageBin, { row: 1,
 | 
			
		||||
                                          col: 1,
 | 
			
		||||
                                          row_span: 2,
 | 
			
		||||
                                          x_expand: false,
 | 
			
		||||
                                          y_expand: false,
 | 
			
		||||
                                          x_fill: false,
 | 
			
		||||
                                          y_fill: false });
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    unsetImage: function() {
 | 
			
		||||
        if (this._imageBin) {
 | 
			
		||||
            this._table.remove_style_class_name('notification-with-image');
 | 
			
		||||
            this._table.remove_actor(this._imageBin);
 | 
			
		||||
            this._imageBin = null;
 | 
			
		||||
            this._updateLastColumnSettings();
 | 
			
		||||
            if (!this._scrollArea && !this._actionArea)
 | 
			
		||||
                this._table.remove_style_class_name('multi-line-notification');
 | 
			
		||||
        }
 | 
			
		||||
        this._actionArea = actor;
 | 
			
		||||
        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();
 | 
			
		||||
 | 
			
		||||
            if (!this.resident) {
 | 
			
		||||
                // We don't hide a resident notification when the user invokes one of its actions,
 | 
			
		||||
                // because it is common for such notifications to update themselves with new
 | 
			
		||||
                // information based on the action. We'd like to display the updated information
 | 
			
		||||
                // in place, rather than pop-up a new notification.
 | 
			
		||||
                this.emit('done-displaying');
 | 
			
		||||
                this.destroy();
 | 
			
		||||
            }
 | 
			
		||||
            this.emit('done-displaying');
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        }));
 | 
			
		||||
 | 
			
		||||
        this.updated();
 | 
			
		||||
        this._sync();
 | 
			
		||||
        return button;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
@@ -870,8 +741,7 @@ const Notification = new Lang.Class({
 | 
			
		||||
    // the notification.
 | 
			
		||||
    addAction: function(label, callback) {
 | 
			
		||||
        let button = new St.Button({ style_class: 'notification-button',
 | 
			
		||||
                                     label: label,
 | 
			
		||||
                                     can_focus: true });
 | 
			
		||||
                                     x_expand: true, label: label, can_focus: true });
 | 
			
		||||
 | 
			
		||||
        return this.addButton(button, callback);
 | 
			
		||||
    },
 | 
			
		||||
@@ -880,10 +750,6 @@ const Notification = new Lang.Class({
 | 
			
		||||
        this.urgency = urgency;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    setResident: function(resident) {
 | 
			
		||||
        this.resident = resident;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    setTransient: function(isTransient) {
 | 
			
		||||
        this.isTransient = isTransient;
 | 
			
		||||
    },
 | 
			
		||||
@@ -911,113 +777,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;
 | 
			
		||||
@@ -1050,83 +809,27 @@ 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._imageBin)
 | 
			
		||||
            this._imageBin.show();
 | 
			
		||||
        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.emit('expanded');
 | 
			
		||||
        this._sync();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    collapseCompleted: function() {
 | 
			
		||||
        if (this._destroyed)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        this.expanded = false;
 | 
			
		||||
 | 
			
		||||
        // Hide additional content that we keep hidden in banner mode
 | 
			
		||||
        if (this._imageBin)
 | 
			
		||||
            this._imageBin.hide();
 | 
			
		||||
        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() {
 | 
			
		||||
        this.emit('clicked');
 | 
			
		||||
        // We hide all types of notifications once the user clicks on them because the common
 | 
			
		||||
        // outcome of clicking should be the relevant window being brought forward and the user's
 | 
			
		||||
        // attention switching to the window.
 | 
			
		||||
        this.emit('done-displaying');
 | 
			
		||||
        if (!this.resident)
 | 
			
		||||
            this.destroy();
 | 
			
		||||
        this.destroy();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _onCloseClicked: function() {
 | 
			
		||||
        this.destroy();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _onDestroy: function() {
 | 
			
		||||
@@ -1294,7 +997,7 @@ const Source = new Lang.Class({
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    get indicatorCount() {
 | 
			
		||||
        let notifications = this.notifications.filter(function(n) { return !n.isTransient && !n.resident; });
 | 
			
		||||
        let notifications = this.notifications.filter(function(n) { return !n.isTransient; });
 | 
			
		||||
        return notifications.length;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
@@ -1307,7 +1010,7 @@ const Source = new Lang.Class({
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    get isClearable() {
 | 
			
		||||
        return !this.trayIcon && !this.isChat && !this.resident;
 | 
			
		||||
        return !this.trayIcon && !this.isChat;
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    countUpdated: function() {
 | 
			
		||||
@@ -1451,10 +1154,9 @@ const Source = new Lang.Class({
 | 
			
		||||
    open: function() {
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    destroyNonResidentNotifications: function() {
 | 
			
		||||
    destroyNotifications: function() {
 | 
			
		||||
        for (let i = this.notifications.length - 1; i >= 0; i--)
 | 
			
		||||
            if (!this.notifications[i].resident)
 | 
			
		||||
                this.notifications[i].destroy();
 | 
			
		||||
            this.notifications[i].destroy();
 | 
			
		||||
 | 
			
		||||
        this.countUpdated();
 | 
			
		||||
    },
 | 
			
		||||
@@ -1786,6 +1488,9 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
                                                   layout_manager: new Clutter.BinLayout() });
 | 
			
		||||
        this._notificationWidget.connect('key-release-event', Lang.bind(this, this._onNotificationKeyRelease));
 | 
			
		||||
        this._notificationWidget.connect('notify::hover', Lang.bind(this, this._onNotificationHoverChanged));
 | 
			
		||||
        this._notificationWidget.connect('notify::height', Lang.bind(this, function() {
 | 
			
		||||
            this._notificationWidget.translation_y = -this._notificationWidget.height;
 | 
			
		||||
        }));
 | 
			
		||||
 | 
			
		||||
        this._notificationBin = new St.Bin({ y_expand: true });
 | 
			
		||||
        this._notificationBin.set_y_align(Clutter.ActorAlign.START);
 | 
			
		||||
@@ -1829,11 +1534,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
        this._clickedSummaryItemMouseButton = -1;
 | 
			
		||||
        this._clickedSummaryItemAllocationChangedId = 0;
 | 
			
		||||
 | 
			
		||||
        this._closeButton = Util.makeCloseButton();
 | 
			
		||||
        this._closeButton.hide();
 | 
			
		||||
        this._closeButton.connect('clicked', Lang.bind(this, this._closeNotification));
 | 
			
		||||
        this._notificationWidget.add_actor(this._closeButton);
 | 
			
		||||
 | 
			
		||||
        this._userActiveWhileNotificationShown = false;
 | 
			
		||||
 | 
			
		||||
        this.idleMonitor = Meta.IdleMonitor.get_core();
 | 
			
		||||
@@ -1864,7 +1564,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
        this._keyboardVisible = false;
 | 
			
		||||
        this._notificationState = State.HIDDEN;
 | 
			
		||||
        this._notificationTimeoutId = 0;
 | 
			
		||||
        this._notificationExpandedId = 0;
 | 
			
		||||
        this._summaryBoxPointerState = State.HIDDEN;
 | 
			
		||||
        this._summaryBoxPointerTimeoutId = 0;
 | 
			
		||||
        this._desktopCloneState = State.HIDDEN;
 | 
			
		||||
@@ -1888,7 +1587,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
        Main.layoutManager.trayBox.add_actor(this._notificationWidget);
 | 
			
		||||
        Main.layoutManager.trackChrome(this.actor);
 | 
			
		||||
        Main.layoutManager.trackChrome(this._notificationWidget);
 | 
			
		||||
        Main.layoutManager.trackChrome(this._closeButton);
 | 
			
		||||
 | 
			
		||||
        global.screen.connect('in-fullscreen-changed', Lang.bind(this, this._updateState));
 | 
			
		||||
        Main.layoutManager.connect('hot-corners-changed', Lang.bind(this, this._hotCornersChanged));
 | 
			
		||||
@@ -2040,14 +1738,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
        this._updateState();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _closeNotification: function() {
 | 
			
		||||
        if (this._notificationState == State.SHOWN) {
 | 
			
		||||
            this._closeButton.hide();
 | 
			
		||||
            this._notification.emit('done-displaying');
 | 
			
		||||
            this._notification.destroy();
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    contains: function(source) {
 | 
			
		||||
        return this._sources.has(source);
 | 
			
		||||
    },
 | 
			
		||||
@@ -2613,7 +2303,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
        this._notificationBin.child = this._notification.actor;
 | 
			
		||||
 | 
			
		||||
        this._notificationWidget.opacity = 0;
 | 
			
		||||
        this._notificationWidget.y = 0;
 | 
			
		||||
        this._notificationWidget.show();
 | 
			
		||||
 | 
			
		||||
        this._updateShowingNotification();
 | 
			
		||||
@@ -2648,23 +2337,16 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
        // We tween all notifications to full opacity. This ensures that both new notifications and
 | 
			
		||||
        // notifications that might have been in the process of hiding get full opacity.
 | 
			
		||||
        //
 | 
			
		||||
        // We tween any notification showing in the banner mode to the appropriate height
 | 
			
		||||
        // (which is banner height or expanded height, depending on the notification state)
 | 
			
		||||
        // This ensures that both new notifications and notifications in the banner mode that might
 | 
			
		||||
        // have been in the process of hiding are shown with the correct height.
 | 
			
		||||
        //
 | 
			
		||||
        // We use this._showNotificationCompleted() onComplete callback to extend the time the updated
 | 
			
		||||
        // notification is being shown.
 | 
			
		||||
 | 
			
		||||
        let tweenParams = { opacity: 255,
 | 
			
		||||
                            y: -this._notificationWidget.height,
 | 
			
		||||
                            time: ANIMATION_TIME,
 | 
			
		||||
                            transition: 'easeOutQuad',
 | 
			
		||||
                            onComplete: this._showNotificationCompleted,
 | 
			
		||||
                            onCompleteScope: this
 | 
			
		||||
                          };
 | 
			
		||||
 | 
			
		||||
        this._tween(this._notificationWidget, '_notificationState', State.SHOWN, tweenParams);
 | 
			
		||||
        this._tween(this._notificationWidget, '_notificationState', State.SHOWN,
 | 
			
		||||
                    { opacity: 255,
 | 
			
		||||
                      time: ANIMATION_TIME,
 | 
			
		||||
                      transition: 'easeOutQuad',
 | 
			
		||||
                      onComplete: this._showNotificationCompleted,
 | 
			
		||||
                      onCompleteScope: this
 | 
			
		||||
                    });
 | 
			
		||||
   },
 | 
			
		||||
 | 
			
		||||
    _showNotificationCompleted: function() {
 | 
			
		||||
@@ -2712,10 +2394,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
    _hideNotification: function(animate) {
 | 
			
		||||
        this._notificationFocusGrabber.ungrabFocus();
 | 
			
		||||
 | 
			
		||||
        if (this._notificationExpandedId) {
 | 
			
		||||
            this._notification.disconnect(this._notificationExpandedId);
 | 
			
		||||
            this._notificationExpandedId = 0;
 | 
			
		||||
        }
 | 
			
		||||
        if (this._notificationClickedId) {
 | 
			
		||||
            this._notification.disconnect(this._notificationClickedId);
 | 
			
		||||
            this._notificationClickedId = 0;
 | 
			
		||||
@@ -2729,8 +2407,7 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
 | 
			
		||||
        if (animate) {
 | 
			
		||||
            this._tween(this._notificationWidget, '_notificationState', State.HIDDEN,
 | 
			
		||||
                        { y: this.actor.height,
 | 
			
		||||
                          opacity: 0,
 | 
			
		||||
                        { opacity: 0,
 | 
			
		||||
                          time: ANIMATION_TIME,
 | 
			
		||||
                          transition: 'easeOutQuad',
 | 
			
		||||
                          onComplete: this._hideNotificationCompleted,
 | 
			
		||||
@@ -2738,7 +2415,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
                        });
 | 
			
		||||
        } else {
 | 
			
		||||
            Tweener.removeTweens(this._notificationWidget);
 | 
			
		||||
            this._notificationWidget.y = this.actor.height;
 | 
			
		||||
            this._notificationWidget.opacity = 0;
 | 
			
		||||
            this._notificationState = State.HIDDEN;
 | 
			
		||||
            this._hideNotificationCompleted();
 | 
			
		||||
@@ -2753,7 +2429,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
        if (notification.isTransient)
 | 
			
		||||
            notification.destroy(NotificationDestroyedReason.EXPIRED);
 | 
			
		||||
 | 
			
		||||
        this._closeButton.hide();
 | 
			
		||||
        this._pointerInNotification = false;
 | 
			
		||||
        this._notificationRemoved = false;
 | 
			
		||||
        this._notificationBin.child = null;
 | 
			
		||||
@@ -2768,10 +2443,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _expandNotification: function(autoExpanding) {
 | 
			
		||||
        if (!this._notificationExpandedId)
 | 
			
		||||
            this._notificationExpandedId =
 | 
			
		||||
                this._notification.connect('expanded',
 | 
			
		||||
                                           Lang.bind(this, this._onNotificationExpanded));
 | 
			
		||||
        // Don't animate changes in notifications that are auto-expanding.
 | 
			
		||||
        this._notification.expand(!autoExpanding);
 | 
			
		||||
 | 
			
		||||
@@ -2780,31 +2451,6 @@ const MessageTray = new Lang.Class({
 | 
			
		||||
            this._ensureNotificationFocused();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _onNotificationExpanded: function() {
 | 
			
		||||
        let expandedY = - this._notificationWidget.height;
 | 
			
		||||
        this._closeButton.show();
 | 
			
		||||
 | 
			
		||||
        // Don't animate the notification to its new position if it has shrunk:
 | 
			
		||||
        // there will be a very visible "gap" that breaks the illusion.
 | 
			
		||||
        if (this._notificationWidget.y < expandedY) {
 | 
			
		||||
            this._notificationWidget.y = expandedY;
 | 
			
		||||
        } else if (this._notification.y != expandedY) {
 | 
			
		||||
            // Tween also opacity here, to override a possible tween that's
 | 
			
		||||
            // currently hiding the notification.
 | 
			
		||||
            Tweener.addTween(this._notificationWidget,
 | 
			
		||||
                             { y: expandedY,
 | 
			
		||||
                               opacity: 255,
 | 
			
		||||
                               time: ANIMATION_TIME,
 | 
			
		||||
                               transition: 'easeOutQuad',
 | 
			
		||||
                               // HACK: Drive the state machine here better,
 | 
			
		||||
                               // instead of overwriting tweens
 | 
			
		||||
                               onComplete: Lang.bind(this, function() {
 | 
			
		||||
                                   this._notificationState = State.SHOWN;
 | 
			
		||||
                               }),
 | 
			
		||||
                             });
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _ensureNotificationFocused: function() {
 | 
			
		||||
        this._notificationFocusGrabber.grabFocus();
 | 
			
		||||
    },
 | 
			
		||||
 
 | 
			
		||||
@@ -91,21 +91,6 @@ const rewriteRules = {
 | 
			
		||||
    ]
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const STANDARD_TRAY_ICON_IMPLEMENTATIONS = {
 | 
			
		||||
    'bluetooth-applet': 'bluetooth',
 | 
			
		||||
    'gnome-volume-control-applet': 'volume', // renamed to gnome-sound-applet
 | 
			
		||||
                                             // when moved to control center
 | 
			
		||||
    'gnome-sound-applet': 'volume',
 | 
			
		||||
    'nm-applet': 'network',
 | 
			
		||||
    'gnome-power-manager': 'battery',
 | 
			
		||||
    'keyboard': 'keyboard',
 | 
			
		||||
    'a11y-keyboard': 'a11y',
 | 
			
		||||
    'kbd-scrolllock': 'keyboard',
 | 
			
		||||
    'kbd-numlock': 'keyboard',
 | 
			
		||||
    'kbd-capslock': 'keyboard',
 | 
			
		||||
    'ibus-ui-gtk': 'keyboard'
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
    Name: 'FdoNotificationDaemon',
 | 
			
		||||
 | 
			
		||||
@@ -334,13 +319,14 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _makeButton: function(id, label, useActionIcons) {
 | 
			
		||||
        let button = new St.Button({ can_focus: true });
 | 
			
		||||
        let button = new St.Button({ can_focus: true,
 | 
			
		||||
                                     x_expand: true,
 | 
			
		||||
                                     style_class: 'notification-button' });
 | 
			
		||||
        let iconName = id.endsWith('-symbolic') ? id : id + '-symbolic';
 | 
			
		||||
 | 
			
		||||
        if (useActionIcons && Gtk.IconTheme.get_default().has_icon(iconName)) {
 | 
			
		||||
            button.add_style_class_name('notification-icon-button');
 | 
			
		||||
            button.child = new St.Icon({ icon_name: iconName });
 | 
			
		||||
            button.child = new St.Icon({ icon_name: iconName, icon_size: 16 });
 | 
			
		||||
        } else {
 | 
			
		||||
            button.add_style_class_name('notification-button');
 | 
			
		||||
            button.label = label;
 | 
			
		||||
        }
 | 
			
		||||
        return button;
 | 
			
		||||
@@ -379,8 +365,6 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
        let gicon = this._iconForNotificationData(icon, hints);
 | 
			
		||||
        let gimage = this._imageForNotificationData(hints);
 | 
			
		||||
 | 
			
		||||
        let image = null;
 | 
			
		||||
 | 
			
		||||
        // If an icon is not specified, we use 'image-data' or 'image-path' hint for an icon
 | 
			
		||||
        // and don't show a large image. There are currently many applications that use
 | 
			
		||||
        // notify_notification_set_icon_from_pixbuf() from libnotify, which in turn sets
 | 
			
		||||
@@ -389,10 +373,7 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
        // So the logic here does the right thing for this case. If both an icon and either
 | 
			
		||||
        // one of 'image-data' or 'image-path' are specified, we show both an icon and
 | 
			
		||||
        // a large image.
 | 
			
		||||
        if (gicon && gimage)
 | 
			
		||||
            image = new St.Icon({ gicon: gimage,
 | 
			
		||||
                                  icon_size: notification.IMAGE_SIZE });
 | 
			
		||||
        else if (!gicon && gimage)
 | 
			
		||||
        if (!gicon && gimage)
 | 
			
		||||
            gicon = gimage;
 | 
			
		||||
        else if (!gicon)
 | 
			
		||||
            gicon = this._fallbackIconForNotificationData(hints);
 | 
			
		||||
@@ -402,7 +383,6 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
                                             clear: true,
 | 
			
		||||
                                             soundFile: hints['sound-file'],
 | 
			
		||||
                                             soundName: hints['sound-name'] });
 | 
			
		||||
        notification.setImage(image);
 | 
			
		||||
 | 
			
		||||
        let hasDefaultAction = false;
 | 
			
		||||
 | 
			
		||||
@@ -442,7 +422,6 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
                notification.setUrgency(MessageTray.Urgency.CRITICAL);
 | 
			
		||||
                break;
 | 
			
		||||
        }
 | 
			
		||||
        notification.setResident(hints.resident == true);
 | 
			
		||||
        // 'transient' is a reserved keyword in JS, so we have to retrieve the value
 | 
			
		||||
        // of the 'transient' hint with hints['transient'] rather than hints.transient
 | 
			
		||||
        notification.setTransient(hints['transient'] == true);
 | 
			
		||||
@@ -470,7 +449,6 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
            'body-markup',
 | 
			
		||||
            // 'icon-multi',
 | 
			
		||||
            'icon-static',
 | 
			
		||||
            'persistence',
 | 
			
		||||
            'sound',
 | 
			
		||||
        ];
 | 
			
		||||
    },
 | 
			
		||||
@@ -492,7 +470,7 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
        for (let i = 0; i < this._sources.length; i++) {
 | 
			
		||||
            let source = this._sources[i];
 | 
			
		||||
            if (source.app == tracker.focus_app) {
 | 
			
		||||
                source.destroyNonResidentNotifications();
 | 
			
		||||
                source.destroyNotifications();
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
@@ -509,10 +487,6 @@ const FdoNotificationDaemon = new Lang.Class({
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _onTrayIconAdded: function(o, icon) {
 | 
			
		||||
        let wmClass = icon.wm_class ? icon.wm_class.toLowerCase() : '';
 | 
			
		||||
        if (STANDARD_TRAY_ICON_IMPLEMENTATIONS[wmClass] !== undefined)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        let source = this._getSource(icon.title || icon.wm_class || C_("program", "Unknown"), icon.pid, null, null, icon);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
@@ -584,7 +558,7 @@ const FdoNotificationDaemonSource = new Lang.Class({
 | 
			
		||||
            this.iconUpdated();
 | 
			
		||||
 | 
			
		||||
        let tracker = Shell.WindowTracker.get_default();
 | 
			
		||||
        if (notification.resident && this.app && tracker.focus_app == this.app)
 | 
			
		||||
        if (this.app && tracker.focus_app == this.app)
 | 
			
		||||
            this.pushNotification(notification);
 | 
			
		||||
        else
 | 
			
		||||
            this.notify(notification);
 | 
			
		||||
@@ -651,7 +625,7 @@ const FdoNotificationDaemonSource = new Lang.Class({
 | 
			
		||||
 | 
			
		||||
    open: function() {
 | 
			
		||||
        this.openApp();
 | 
			
		||||
        this.destroyNonResidentNotifications();
 | 
			
		||||
        this.destroyNotifications();
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    _lastNotificationRemoved: function() {
 | 
			
		||||
 
 | 
			
		||||
@@ -21,10 +21,6 @@ struct _ShellTpClientPrivate
 | 
			
		||||
  ShellTpClientHandleChannelsImpl handle_channels_impl;
 | 
			
		||||
  gpointer user_data_handle_channels;
 | 
			
		||||
  GDestroyNotify destroy_handle_channels;
 | 
			
		||||
 | 
			
		||||
  ShellTpClientContactListChangedImpl contact_list_changed_impl;
 | 
			
		||||
  gpointer user_data_contact_list_changed;
 | 
			
		||||
  GDestroyNotify destroy_contact_list_changed;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
@@ -83,16 +79,6 @@ struct _ShellTpClientPrivate
 | 
			
		||||
 * Signature of the implementation of the HandleChannels method.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ShellTpClientContactListChangedImpl:
 | 
			
		||||
 * @connection: a #TpConnection having %TP_CONNECTION_FEATURE_CORE prepared
 | 
			
		||||
 * if possible
 | 
			
		||||
 * @added: (element-type TelepathyGLib.Contact): a #GPtrArray of added #TpContact
 | 
			
		||||
 * @removed: (element-type TelepathyGLib.Contact): a #GPtrArray of removed #TpContact
 | 
			
		||||
 *
 | 
			
		||||
 * Signature of the implementation of the ContactListChanged method.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
static void
 | 
			
		||||
shell_tp_client_init (ShellTpClient *self)
 | 
			
		||||
{
 | 
			
		||||
@@ -226,13 +212,6 @@ shell_tp_client_dispose (GObject *object)
 | 
			
		||||
      self->priv->user_data_handle_channels = NULL;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  if (self->priv->destroy_contact_list_changed != NULL)
 | 
			
		||||
    {
 | 
			
		||||
      self->priv->destroy_contact_list_changed (self->priv->user_data_contact_list_changed);
 | 
			
		||||
      self->priv->destroy_contact_list_changed = NULL;
 | 
			
		||||
      self->priv->user_data_contact_list_changed = NULL;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  if (dispose != NULL)
 | 
			
		||||
    dispose (object);
 | 
			
		||||
}
 | 
			
		||||
@@ -290,40 +269,3 @@ shell_tp_client_set_handle_channels_func (ShellTpClient *self,
 | 
			
		||||
  self->priv->user_data_handle_channels = user_data;
 | 
			
		||||
  self->priv->destroy_handle_channels = destroy;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void
 | 
			
		||||
shell_tp_client_set_contact_list_changed_func (ShellTpClient *self,
 | 
			
		||||
    ShellTpClientContactListChangedImpl contact_list_changed_impl,
 | 
			
		||||
    gpointer user_data,
 | 
			
		||||
    GDestroyNotify destroy)
 | 
			
		||||
{
 | 
			
		||||
  g_assert (self->priv->contact_list_changed_impl == NULL);
 | 
			
		||||
 | 
			
		||||
  self->priv->contact_list_changed_impl = contact_list_changed_impl;
 | 
			
		||||
  self->priv->user_data_handle_channels = user_data;
 | 
			
		||||
  self->priv->destroy_handle_channels = destroy;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void
 | 
			
		||||
on_contact_list_changed (TpConnection *conn,
 | 
			
		||||
                         GPtrArray *added,
 | 
			
		||||
                         GPtrArray *removed,
 | 
			
		||||
                         gpointer user_data)
 | 
			
		||||
{
 | 
			
		||||
  ShellTpClient *self = (ShellTpClient *) user_data;
 | 
			
		||||
 | 
			
		||||
  g_assert (self->priv->contact_list_changed_impl != NULL);
 | 
			
		||||
 | 
			
		||||
  self->priv->contact_list_changed_impl (conn,
 | 
			
		||||
      added, removed,
 | 
			
		||||
      self->priv->user_data_contact_list_changed);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void
 | 
			
		||||
shell_tp_client_grab_contact_list_changed (ShellTpClient *self,
 | 
			
		||||
                                           TpConnection *conn)
 | 
			
		||||
{
 | 
			
		||||
  g_signal_connect (conn, "contact-list-changed",
 | 
			
		||||
                    G_CALLBACK (on_contact_list_changed),
 | 
			
		||||
                    self);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -86,19 +86,5 @@ void shell_tp_client_set_handle_channels_func (ShellTpClient *self,
 | 
			
		||||
    gpointer user_data,
 | 
			
		||||
    GDestroyNotify destroy);
 | 
			
		||||
 | 
			
		||||
typedef void (*ShellTpClientContactListChangedImpl) (
 | 
			
		||||
    TpConnection *connection,
 | 
			
		||||
    GPtrArray *added,
 | 
			
		||||
    GPtrArray *removed,
 | 
			
		||||
    gpointer user_data);
 | 
			
		||||
 | 
			
		||||
void shell_tp_client_set_contact_list_changed_func (ShellTpClient *self,
 | 
			
		||||
    ShellTpClientContactListChangedImpl contact_list_changed_impl,
 | 
			
		||||
    gpointer user_data,
 | 
			
		||||
    GDestroyNotify destroy);
 | 
			
		||||
 | 
			
		||||
void shell_tp_client_grab_contact_list_changed (ShellTpClient *self,
 | 
			
		||||
    TpConnection *conn);
 | 
			
		||||
 | 
			
		||||
G_END_DECLS
 | 
			
		||||
#endif /* __SHELL_TP_CLIENT_H__ */
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user