2011-09-28 09:16:26 -04:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2008-10-31 18:09:20 +00:00
|
|
|
|
2011-02-09 03:12:10 +01:00
|
|
|
const Cairo = imports.cairo;
|
2008-10-31 18:09:20 +00:00
|
|
|
const Clutter = imports.gi.Clutter;
|
2010-05-05 23:05:42 +02:00
|
|
|
const Gio = imports.gi.Gio;
|
2011-05-15 18:55:23 +02:00
|
|
|
const GLib = imports.gi.GLib;
|
2012-03-27 15:26:20 +02:00
|
|
|
const Gtk = imports.gi.Gtk;
|
2009-05-07 09:54:21 -04:00
|
|
|
const Lang = imports.lang;
|
2008-11-21 13:54:25 +00:00
|
|
|
const Mainloop = imports.mainloop;
|
2011-10-04 17:14:29 +02:00
|
|
|
const Meta = imports.gi.Meta;
|
2010-05-25 10:21:22 -04:00
|
|
|
const Pango = imports.gi.Pango;
|
2008-11-19 23:04:53 +00:00
|
|
|
const Shell = imports.gi.Shell;
|
2009-09-30 10:02:08 -04:00
|
|
|
const St = imports.gi.St;
|
2009-08-11 11:32:58 -04:00
|
|
|
const Signals = imports.signals;
|
2012-02-27 17:31:10 +01:00
|
|
|
const Atk = imports.gi.Atk;
|
2008-10-31 18:09:20 +00:00
|
|
|
|
2012-08-28 23:10:44 +02:00
|
|
|
|
2013-06-18 07:35:41 -04:00
|
|
|
const Animation = imports.ui.animation;
|
2012-08-28 23:10:44 +02:00
|
|
|
const Config = imports.misc.config;
|
2011-02-07 11:29:34 -05:00
|
|
|
const CtrlAltTab = imports.ui.ctrlAltTab;
|
2012-02-11 11:14:43 +01:00
|
|
|
const DND = imports.ui.dnd;
|
2010-05-12 17:07:41 -04:00
|
|
|
const Overview = imports.ui.overview;
|
2010-05-20 11:18:46 -04:00
|
|
|
const PopupMenu = imports.ui.popupMenu;
|
2010-06-22 23:02:26 +02:00
|
|
|
const PanelMenu = imports.ui.panelMenu;
|
2013-05-09 17:35:57 -04:00
|
|
|
const RemoteMenu = imports.ui.remoteMenu;
|
2008-11-09 18:01:59 +00:00
|
|
|
const Main = imports.ui.main;
|
2010-05-20 11:18:46 -04:00
|
|
|
const Tweener = imports.ui.tweener;
|
2008-10-31 23:09:46 +00:00
|
|
|
|
2009-08-11 11:32:58 -04:00
|
|
|
const PANEL_ICON_SIZE = 24;
|
|
|
|
|
2011-01-05 15:47:27 +01:00
|
|
|
const BUTTON_DND_ACTIVATION_TIMEOUT = 250;
|
|
|
|
|
2011-03-12 00:37:43 +01:00
|
|
|
const SPINNER_ANIMATION_TIME = 0.2;
|
2010-06-10 16:07:33 +04:00
|
|
|
|
2011-02-09 03:12:10 +01:00
|
|
|
// To make sure the panel corners blend nicely with the panel,
|
|
|
|
// we draw background and borders the same way, e.g. drawing
|
|
|
|
// them as filled shapes from the outside inwards instead of
|
|
|
|
// using cairo stroke(). So in order to give the border the
|
|
|
|
// appearance of being drawn on top of the background, we need
|
|
|
|
// to blend border and background color together.
|
|
|
|
// For that purpose we use the following helper methods, taken
|
|
|
|
// from st-theme-node-drawing.c
|
|
|
|
function _norm(x) {
|
|
|
|
return Math.round(x / 255);
|
|
|
|
}
|
|
|
|
|
|
|
|
function _over(srcColor, dstColor) {
|
|
|
|
let src = _premultiply(srcColor);
|
|
|
|
let dst = _premultiply(dstColor);
|
|
|
|
let result = new Clutter.Color();
|
|
|
|
|
|
|
|
result.alpha = src.alpha + _norm((255 - src.alpha) * dst.alpha);
|
|
|
|
result.red = src.red + _norm((255 - src.alpha) * dst.red);
|
|
|
|
result.green = src.green + _norm((255 - src.alpha) * dst.green);
|
|
|
|
result.blue = src.blue + _norm((255 - src.alpha) * dst.blue);
|
|
|
|
|
|
|
|
return _unpremultiply(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
function _premultiply(color) {
|
|
|
|
return new Clutter.Color({ red: _norm(color.red * color.alpha),
|
|
|
|
green: _norm(color.green * color.alpha),
|
|
|
|
blue: _norm(color.blue * color.alpha),
|
|
|
|
alpha: color.alpha });
|
|
|
|
};
|
|
|
|
|
|
|
|
function _unpremultiply(color) {
|
|
|
|
if (color.alpha == 0)
|
|
|
|
return new Clutter.Color();
|
|
|
|
|
|
|
|
let red = Math.min((color.red * 255 + 127) / color.alpha, 255);
|
|
|
|
let green = Math.min((color.green * 255 + 127) / color.alpha, 255);
|
|
|
|
let blue = Math.min((color.blue * 255 + 127) / color.alpha, 255);
|
|
|
|
return new Clutter.Color({ red: red, green: green,
|
|
|
|
blue: blue, alpha: color.alpha });
|
|
|
|
};
|
|
|
|
|
2011-11-20 18:56:27 +01:00
|
|
|
const TextShadower = new Lang.Class({
|
|
|
|
Name: 'TextShadower',
|
2009-12-26 12:00:36 -05:00
|
|
|
|
|
|
|
_init: function() {
|
|
|
|
this.actor = new Shell.GenericContainer();
|
|
|
|
this.actor.connect('get-preferred-width', Lang.bind(this, this._getPreferredWidth));
|
|
|
|
this.actor.connect('get-preferred-height', Lang.bind(this, this._getPreferredHeight));
|
|
|
|
this.actor.connect('allocate', Lang.bind(this, this._allocate));
|
|
|
|
|
|
|
|
this._label = new St.Label();
|
|
|
|
this.actor.add_actor(this._label);
|
|
|
|
for (let i = 0; i < 4; i++) {
|
|
|
|
let actor = new St.Label({ style_class: 'label-shadow' });
|
2010-05-25 10:21:22 -04:00
|
|
|
actor.clutter_text.ellipsize = Pango.EllipsizeMode.END;
|
2009-12-26 12:00:36 -05:00
|
|
|
this.actor.add_actor(actor);
|
|
|
|
}
|
|
|
|
this._label.raise_top();
|
|
|
|
},
|
|
|
|
|
|
|
|
setText: function(text) {
|
|
|
|
let children = this.actor.get_children();
|
|
|
|
for (let i = 0; i < children.length; i++)
|
|
|
|
children[i].set_text(text);
|
|
|
|
},
|
|
|
|
|
|
|
|
_getPreferredWidth: function(actor, forHeight, alloc) {
|
|
|
|
let [minWidth, natWidth] = this._label.get_preferred_width(forHeight);
|
2010-05-25 09:34:01 -04:00
|
|
|
alloc.min_size = minWidth + 2;
|
|
|
|
alloc.natural_size = natWidth + 2;
|
2009-12-26 12:00:36 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
_getPreferredHeight: function(actor, forWidth, alloc) {
|
|
|
|
let [minHeight, natHeight] = this._label.get_preferred_height(forWidth);
|
2010-05-25 09:34:01 -04:00
|
|
|
alloc.min_size = minHeight + 2;
|
|
|
|
alloc.natural_size = natHeight + 2;
|
2009-12-26 12:00:36 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
_allocate: function(actor, box, flags) {
|
|
|
|
let children = this.actor.get_children();
|
|
|
|
|
|
|
|
let availWidth = box.x2 - box.x1;
|
|
|
|
let availHeight = box.y2 - box.y1;
|
|
|
|
|
|
|
|
let [minChildWidth, minChildHeight, natChildWidth, natChildHeight] =
|
|
|
|
this._label.get_preferred_size();
|
|
|
|
|
2010-05-25 09:34:01 -04:00
|
|
|
let childWidth = Math.min(natChildWidth, availWidth - 2);
|
|
|
|
let childHeight = Math.min(natChildHeight, availHeight - 2);
|
2009-12-26 12:00:36 -05:00
|
|
|
|
|
|
|
for (let i = 0; i < children.length; i++) {
|
|
|
|
let child = children[i];
|
|
|
|
let childBox = new Clutter.ActorBox();
|
|
|
|
// The order of the labels here is arbitrary, except
|
2013-02-22 12:23:56 +01:00
|
|
|
// we know the "real" label is at the end because Clutter.Actor
|
2009-12-26 12:00:36 -05:00
|
|
|
// sorts by Z order
|
|
|
|
switch (i) {
|
|
|
|
case 0: // top
|
|
|
|
childBox.x1 = 1;
|
|
|
|
childBox.y1 = 0;
|
|
|
|
break;
|
|
|
|
case 1: // right
|
|
|
|
childBox.x1 = 2;
|
|
|
|
childBox.y1 = 1;
|
|
|
|
break;
|
|
|
|
case 2: // bottom
|
|
|
|
childBox.x1 = 1;
|
|
|
|
childBox.y1 = 2;
|
|
|
|
break;
|
|
|
|
case 3: // left
|
|
|
|
childBox.x1 = 0;
|
|
|
|
childBox.y1 = 1;
|
|
|
|
break;
|
|
|
|
case 4: // center
|
|
|
|
childBox.x1 = 1;
|
|
|
|
childBox.y1 = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
childBox.x2 = childBox.x1 + childWidth;
|
|
|
|
childBox.y2 = childBox.y1 + childHeight;
|
|
|
|
child.allocate(childBox, flags);
|
|
|
|
}
|
|
|
|
}
|
2011-11-20 18:56:27 +01:00
|
|
|
});
|
2009-12-26 12:00:36 -05:00
|
|
|
|
|
|
|
/**
|
2010-05-05 10:03:48 -04:00
|
|
|
* AppMenuButton:
|
2009-12-26 12:00:36 -05:00
|
|
|
*
|
|
|
|
* This class manages the "application menu" component. It tracks the
|
|
|
|
* currently focused application. However, when an app is launched,
|
|
|
|
* this menu also handles startup notification for it. So when we
|
|
|
|
* have an active startup notification, we switch modes to display that.
|
|
|
|
*/
|
2011-11-20 15:38:48 +01:00
|
|
|
const AppMenuButton = new Lang.Class({
|
|
|
|
Name: 'AppMenuButton',
|
|
|
|
Extends: PanelMenu.Button,
|
2010-05-05 10:03:48 -04:00
|
|
|
|
2012-08-28 23:10:44 +02:00
|
|
|
_init: function(panel) {
|
2012-01-05 19:00:06 +01:00
|
|
|
this.parent(0.0, null, true);
|
2011-05-15 18:55:23 +02:00
|
|
|
|
2012-02-27 17:31:10 +01:00
|
|
|
this.actor.accessible_role = Atk.Role.MENU;
|
|
|
|
|
2010-10-07 01:31:22 +04:00
|
|
|
this._startingApps = [];
|
2009-08-11 11:32:58 -04:00
|
|
|
|
2012-08-28 23:10:44 +02:00
|
|
|
this._menuManager = panel.menuManager;
|
2010-10-07 01:31:22 +04:00
|
|
|
this._targetApp = null;
|
2011-12-08 21:55:51 +01:00
|
|
|
this._appMenuNotifyId = 0;
|
2011-12-15 01:25:35 -05:00
|
|
|
this._actionGroupNotifyId = 0;
|
2009-08-11 11:32:58 -04:00
|
|
|
|
2010-05-05 10:03:48 -04:00
|
|
|
let bin = new St.Bin({ name: 'appMenu' });
|
2013-04-16 16:57:46 +02:00
|
|
|
bin.connect('style-changed', Lang.bind(this, this._onStyleChanged));
|
2011-06-09 11:50:24 -04:00
|
|
|
this.actor.add_actor(bin);
|
2011-03-08 20:46:43 +03:00
|
|
|
|
2011-11-16 00:23:04 +00:00
|
|
|
this.actor.bind_property("reactive", this.actor, "can-focus", 0);
|
2011-03-08 20:46:43 +03:00
|
|
|
this.actor.reactive = false;
|
|
|
|
|
2010-02-28 15:36:13 -05:00
|
|
|
this._container = new Shell.GenericContainer();
|
2010-05-05 10:03:48 -04:00
|
|
|
bin.set_child(this._container);
|
|
|
|
this._container.connect('get-preferred-width', Lang.bind(this, this._getContentPreferredWidth));
|
|
|
|
this._container.connect('get-preferred-height', Lang.bind(this, this._getContentPreferredHeight));
|
|
|
|
this._container.connect('allocate', Lang.bind(this, this._contentAllocate));
|
2010-03-12 15:57:01 -05:00
|
|
|
|
2012-10-30 18:30:21 +01:00
|
|
|
let textureCache = St.TextureCache.get_default();
|
|
|
|
textureCache.connect('icon-theme-changed',
|
|
|
|
Lang.bind(this, this._onIconThemeChanged));
|
|
|
|
|
2009-12-26 12:00:36 -05:00
|
|
|
this._iconBox = new Shell.Slicer({ name: 'appMenuIcon' });
|
2011-02-18 23:00:04 +01:00
|
|
|
this._iconBox.connect('style-changed',
|
|
|
|
Lang.bind(this, this._onIconBoxStyleChanged));
|
2011-03-29 11:49:50 +02:00
|
|
|
this._iconBox.connect('notify::allocation',
|
|
|
|
Lang.bind(this, this._updateIconBoxClip));
|
2010-02-28 15:36:13 -05:00
|
|
|
this._container.add_actor(this._iconBox);
|
2013-08-12 08:36:38 -04:00
|
|
|
|
2013-06-06 17:26:42 -04:00
|
|
|
this._hbox = new St.BoxLayout({ style_class: 'panel-status-menu-box' });
|
2013-08-12 08:36:38 -04:00
|
|
|
this._container.add_actor(this._hbox);
|
|
|
|
|
2009-12-26 12:00:36 -05:00
|
|
|
this._label = new TextShadower();
|
2013-08-12 08:36:38 -04:00
|
|
|
this._label.actor.y_align = Clutter.ActorAlign.CENTER;
|
|
|
|
this._hbox.add_actor(this._label.actor);
|
2013-10-07 16:01:07 +02:00
|
|
|
this._arrow = PopupMenu.unicodeArrow(St.Side.BOTTOM);
|
2013-06-06 17:26:42 -04:00
|
|
|
this._hbox.add_actor(this._arrow);
|
2009-08-11 11:32:58 -04:00
|
|
|
|
2011-03-29 11:49:50 +02:00
|
|
|
this._iconBottomClip = 0;
|
|
|
|
|
2010-05-12 23:30:14 +02:00
|
|
|
this._visible = !Main.overview.visible;
|
|
|
|
if (!this._visible)
|
2011-02-16 15:17:46 -05:00
|
|
|
this.actor.hide();
|
2013-08-13 07:41:44 -04:00
|
|
|
this._overviewHidingId = Main.overview.connect('hiding', Lang.bind(this, this._sync));
|
|
|
|
this._overviewShowingId = Main.overview.connect('showing', Lang.bind(this, this._sync));
|
2009-08-11 11:32:58 -04:00
|
|
|
|
2011-03-12 00:37:43 +01:00
|
|
|
this._stop = true;
|
2013-04-15 18:40:00 -04:00
|
|
|
|
2013-04-16 16:57:46 +02:00
|
|
|
this._spinner = null;
|
2010-06-10 16:07:33 +04:00
|
|
|
|
2009-10-15 19:28:29 -04:00
|
|
|
let tracker = Shell.WindowTracker.get_default();
|
apps: Ensure running apps override new .desktop file data
This patch fixes the "apps vanish from alt-TAB bug".
If a "package system" rips away and possibly replaces .desktop files
at some random time, we have historically used inotify to detect this
and reread state (in a racy way, but...). In GNOME 2, this was
generally not too problematic because the menu widget was totally
separate from the list of windows - and the data they operate on was
disjoint as well.
In GNOME 3 we unify these, and this creates architectural problems
because the windows are tied to the app.
What this patch tries to do is, when rereading the application state,
if we have a running application, we keep that app around instead of
making a new instance. This ensures we preserve any state such as the
set of open windows.
This requires moving the running state into ShellAppSystem. Adjust
callers as necessary, and while we're at it drop the unused "contexts"
stuff.
This is just a somewhat quick band-aid; a REAL fix would require us
having low-level control over application installation. As long as
we're on top of random broken tar+wget wrappers, it will be gross.
A slight future improvement to this patch would add an explicit
"merge" between the old and new data. I think probably we always keep
around the ShellApp corresponding to a given ID, but replace its
GMenuTreeEntry.
https://bugzilla.gnome.org/show_bug.cgi?id=657990
2011-09-03 10:32:06 -04:00
|
|
|
let appSys = Shell.AppSystem.get_default();
|
2013-04-22 09:49:14 +10:00
|
|
|
this._focusAppNotifyId =
|
|
|
|
tracker.connect('notify::focus-app', Lang.bind(this, this._focusAppChanged));
|
|
|
|
this._appStateChangedSignalId =
|
|
|
|
appSys.connect('app-state-changed', Lang.bind(this, this._onAppStateChanged));
|
|
|
|
this._switchWorkspaceNotifyId =
|
|
|
|
global.window_manager.connect('switch-workspace', Lang.bind(this, this._sync));
|
2010-10-07 01:31:22 +04:00
|
|
|
|
2009-08-11 11:32:58 -04:00
|
|
|
this._sync();
|
2010-05-12 23:30:14 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
show: function() {
|
2013-01-27 15:45:04 +01:00
|
|
|
if (this._visible)
|
2010-05-12 23:30:14 +02:00
|
|
|
return;
|
|
|
|
|
2011-02-16 15:17:46 -05:00
|
|
|
this._visible = true;
|
2012-03-13 19:14:18 +01:00
|
|
|
this.actor.reactive = true;
|
2013-08-13 07:41:44 -04:00
|
|
|
this.actor.show();
|
2013-09-10 09:53:33 +02:00
|
|
|
Tweener.removeTweens(this.actor);
|
2010-05-12 23:30:14 +02:00
|
|
|
Tweener.addTween(this.actor,
|
|
|
|
{ opacity: 255,
|
|
|
|
time: Overview.ANIMATION_TIME,
|
2011-02-16 15:17:46 -05:00
|
|
|
transition: 'easeOutQuad' });
|
2010-05-12 23:30:14 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
hide: function() {
|
|
|
|
if (!this._visible)
|
|
|
|
return;
|
|
|
|
|
2011-02-16 15:17:46 -05:00
|
|
|
this._visible = false;
|
2011-05-07 03:12:52 -04:00
|
|
|
this.actor.reactive = false;
|
2013-09-10 09:53:33 +02:00
|
|
|
Tweener.removeTweens(this.actor);
|
2010-05-12 23:30:14 +02:00
|
|
|
Tweener.addTween(this.actor,
|
|
|
|
{ opacity: 0,
|
|
|
|
time: Overview.ANIMATION_TIME,
|
|
|
|
transition: 'easeOutQuad',
|
|
|
|
onComplete: function() {
|
|
|
|
this.actor.hide();
|
|
|
|
},
|
|
|
|
onCompleteScope: this });
|
2009-08-11 11:32:58 -04:00
|
|
|
},
|
|
|
|
|
2013-04-16 16:57:46 +02:00
|
|
|
_onStyleChanged: function(actor) {
|
|
|
|
let node = actor.get_theme_node();
|
|
|
|
let [success, icon] = node.lookup_url('spinner-image', false);
|
|
|
|
if (!success || this._spinnerIcon == icon)
|
|
|
|
return;
|
|
|
|
this._spinnerIcon = icon;
|
2013-06-18 07:35:41 -04:00
|
|
|
this._spinner = new Animation.AnimatedIcon(this._spinnerIcon, PANEL_ICON_SIZE);
|
2013-08-12 08:36:38 -04:00
|
|
|
this._hbox.add_actor(this._spinner.actor);
|
2013-04-16 16:57:46 +02:00
|
|
|
this._spinner.actor.hide();
|
|
|
|
},
|
|
|
|
|
2011-02-18 23:00:04 +01:00
|
|
|
_onIconBoxStyleChanged: function() {
|
|
|
|
let node = this._iconBox.get_theme_node();
|
2011-03-29 11:49:50 +02:00
|
|
|
this._iconBottomClip = node.get_length('app-icon-bottom-clip');
|
|
|
|
this._updateIconBoxClip();
|
|
|
|
},
|
|
|
|
|
2013-07-19 15:39:48 -04:00
|
|
|
_syncIcon: function() {
|
2013-08-13 07:41:44 -04:00
|
|
|
if (!this._targetApp)
|
|
|
|
return;
|
|
|
|
|
2013-07-19 15:39:48 -04:00
|
|
|
let icon = this._targetApp.get_faded_icon(2 * PANEL_ICON_SIZE, this._iconBox.text_direction);
|
|
|
|
this._iconBox.set_child(icon);
|
|
|
|
},
|
|
|
|
|
2012-10-30 18:30:21 +01:00
|
|
|
_onIconThemeChanged: function() {
|
|
|
|
if (this._iconBox.child == null)
|
|
|
|
return;
|
|
|
|
|
2013-07-19 15:39:48 -04:00
|
|
|
this._syncIcon();
|
2012-10-30 18:30:21 +01:00
|
|
|
},
|
|
|
|
|
2011-03-29 11:49:50 +02:00
|
|
|
_updateIconBoxClip: function() {
|
|
|
|
let allocation = this._iconBox.allocation;
|
|
|
|
if (this._iconBottomClip > 0)
|
2011-02-18 23:00:04 +01:00
|
|
|
this._iconBox.set_clip(0, 0,
|
2011-03-29 11:49:50 +02:00
|
|
|
allocation.x2 - allocation.x1,
|
|
|
|
allocation.y2 - allocation.y1 - this._iconBottomClip);
|
2011-02-18 23:00:04 +01:00
|
|
|
else
|
|
|
|
this._iconBox.remove_clip();
|
|
|
|
},
|
|
|
|
|
2010-06-10 16:07:33 +04:00
|
|
|
stopAnimation: function() {
|
2011-03-12 00:37:43 +01:00
|
|
|
if (this._stop)
|
|
|
|
return;
|
2010-06-10 16:07:33 +04:00
|
|
|
|
2011-03-12 00:37:43 +01:00
|
|
|
this._stop = true;
|
2013-04-16 16:57:46 +02:00
|
|
|
|
|
|
|
if (this._spinner == null)
|
|
|
|
return;
|
|
|
|
|
2011-03-12 00:37:43 +01:00
|
|
|
Tweener.addTween(this._spinner.actor,
|
|
|
|
{ opacity: 0,
|
|
|
|
time: SPINNER_ANIMATION_TIME,
|
|
|
|
transition: "easeOutQuad",
|
|
|
|
onCompleteScope: this,
|
|
|
|
onComplete: function() {
|
2012-11-05 23:11:27 +01:00
|
|
|
this._spinner.stop();
|
2011-03-12 00:37:43 +01:00
|
|
|
this._spinner.actor.opacity = 255;
|
|
|
|
this._spinner.actor.hide();
|
|
|
|
}
|
|
|
|
});
|
2010-06-10 16:07:33 +04:00
|
|
|
},
|
|
|
|
|
|
|
|
startAnimation: function() {
|
|
|
|
this._stop = false;
|
2013-04-16 16:57:46 +02:00
|
|
|
|
|
|
|
if (this._spinner == null)
|
|
|
|
return;
|
|
|
|
|
2012-11-05 23:11:27 +01:00
|
|
|
this._spinner.play();
|
2010-06-10 16:07:33 +04:00
|
|
|
this._spinner.actor.show();
|
|
|
|
},
|
|
|
|
|
2010-05-05 10:03:48 -04:00
|
|
|
_getContentPreferredWidth: function(actor, forHeight, alloc) {
|
2010-02-28 15:36:13 -05:00
|
|
|
let [minSize, naturalSize] = this._iconBox.get_preferred_width(forHeight);
|
2010-03-01 11:30:31 -05:00
|
|
|
alloc.min_size = minSize;
|
|
|
|
alloc.natural_size = naturalSize;
|
2013-08-12 08:36:38 -04:00
|
|
|
[minSize, naturalSize] = this._hbox.get_preferred_width(forHeight);
|
2010-03-01 11:30:31 -05:00
|
|
|
alloc.min_size = alloc.min_size + Math.max(0, minSize - Math.floor(alloc.min_size / 2));
|
|
|
|
alloc.natural_size = alloc.natural_size + Math.max(0, naturalSize - Math.floor(alloc.natural_size / 2));
|
2010-02-28 15:36:13 -05:00
|
|
|
},
|
|
|
|
|
2010-05-05 10:03:48 -04:00
|
|
|
_getContentPreferredHeight: function(actor, forWidth, alloc) {
|
2010-02-28 15:36:13 -05:00
|
|
|
let [minSize, naturalSize] = this._iconBox.get_preferred_height(forWidth);
|
|
|
|
alloc.min_size = minSize;
|
|
|
|
alloc.natural_size = naturalSize;
|
2013-08-12 08:36:38 -04:00
|
|
|
[minSize, naturalSize] = this._hbox.get_preferred_height(forWidth);
|
2010-02-28 15:36:13 -05:00
|
|
|
if (minSize > alloc.min_size)
|
|
|
|
alloc.min_size = minSize;
|
|
|
|
if (naturalSize > alloc.natural_size)
|
|
|
|
alloc.natural_size = naturalSize;
|
|
|
|
},
|
|
|
|
|
2010-05-05 10:03:48 -04:00
|
|
|
_contentAllocate: function(actor, box, flags) {
|
2010-02-28 15:36:13 -05:00
|
|
|
let allocWidth = box.x2 - box.x1;
|
|
|
|
let allocHeight = box.y2 - box.y1;
|
|
|
|
let childBox = new Clutter.ActorBox();
|
|
|
|
|
|
|
|
let [minWidth, minHeight, naturalWidth, naturalHeight] = this._iconBox.get_preferred_size();
|
|
|
|
|
2012-02-13 20:37:28 -05:00
|
|
|
let direction = this.actor.get_text_direction();
|
2010-03-01 11:30:31 -05:00
|
|
|
|
2010-02-28 15:36:13 -05:00
|
|
|
let yPadding = Math.floor(Math.max(0, allocHeight - naturalHeight) / 2);
|
|
|
|
childBox.y1 = yPadding;
|
2010-03-01 11:30:31 -05:00
|
|
|
childBox.y2 = childBox.y1 + Math.min(naturalHeight, allocHeight);
|
2012-02-13 20:37:28 -05:00
|
|
|
if (direction == Clutter.TextDirection.LTR) {
|
2010-03-01 11:30:31 -05:00
|
|
|
childBox.x1 = 0;
|
|
|
|
childBox.x2 = childBox.x1 + Math.min(naturalWidth, allocWidth);
|
|
|
|
} else {
|
|
|
|
childBox.x1 = Math.max(0, allocWidth - naturalWidth);
|
|
|
|
childBox.x2 = allocWidth;
|
|
|
|
}
|
2010-02-28 15:36:13 -05:00
|
|
|
this._iconBox.allocate(childBox, flags);
|
|
|
|
|
2010-03-01 11:30:31 -05:00
|
|
|
let iconWidth = childBox.x2 - childBox.x1;
|
|
|
|
|
2013-08-12 08:36:38 -04:00
|
|
|
[minWidth, naturalWidth] = this._hbox.get_preferred_width(-1);
|
2010-02-28 15:36:13 -05:00
|
|
|
|
2013-08-12 08:36:38 -04:00
|
|
|
childBox.y1 = 0;
|
|
|
|
childBox.y2 = allocHeight;
|
2010-03-01 11:30:31 -05:00
|
|
|
|
2012-02-13 20:37:28 -05:00
|
|
|
if (direction == Clutter.TextDirection.LTR) {
|
2010-03-01 11:30:31 -05:00
|
|
|
childBox.x1 = Math.floor(iconWidth / 2);
|
|
|
|
childBox.x2 = Math.min(childBox.x1 + naturalWidth, allocWidth);
|
|
|
|
} else {
|
|
|
|
childBox.x2 = allocWidth - Math.floor(iconWidth / 2);
|
|
|
|
childBox.x1 = Math.max(0, childBox.x2 - naturalWidth);
|
|
|
|
}
|
2013-08-12 08:36:38 -04:00
|
|
|
this._hbox.allocate(childBox, flags);
|
2009-12-26 12:00:36 -05:00
|
|
|
},
|
|
|
|
|
apps: Ensure running apps override new .desktop file data
This patch fixes the "apps vanish from alt-TAB bug".
If a "package system" rips away and possibly replaces .desktop files
at some random time, we have historically used inotify to detect this
and reread state (in a racy way, but...). In GNOME 2, this was
generally not too problematic because the menu widget was totally
separate from the list of windows - and the data they operate on was
disjoint as well.
In GNOME 3 we unify these, and this creates architectural problems
because the windows are tied to the app.
What this patch tries to do is, when rereading the application state,
if we have a running application, we keep that app around instead of
making a new instance. This ensures we preserve any state such as the
set of open windows.
This requires moving the running state into ShellAppSystem. Adjust
callers as necessary, and while we're at it drop the unused "contexts"
stuff.
This is just a somewhat quick band-aid; a REAL fix would require us
having low-level control over application installation. As long as
we're on top of random broken tar+wget wrappers, it will be gross.
A slight future improvement to this patch would add an explicit
"merge" between the old and new data. I think probably we always keep
around the ShellApp corresponding to a given ID, but replace its
GMenuTreeEntry.
https://bugzilla.gnome.org/show_bug.cgi?id=657990
2011-09-03 10:32:06 -04:00
|
|
|
_onAppStateChanged: function(appSys, app) {
|
2010-06-10 16:07:33 +04:00
|
|
|
let state = app.state;
|
2010-10-07 01:31:22 +04:00
|
|
|
if (state != Shell.AppState.STARTING) {
|
|
|
|
this._startingApps = this._startingApps.filter(function(a) {
|
|
|
|
return a != app;
|
|
|
|
});
|
2010-06-10 16:07:33 +04:00
|
|
|
} else if (state == Shell.AppState.STARTING) {
|
2010-10-07 01:31:22 +04:00
|
|
|
this._startingApps.push(app);
|
2010-06-10 16:07:33 +04:00
|
|
|
}
|
|
|
|
// For now just resync on all running state changes; this is mainly to handle
|
|
|
|
// cases where the focused window's application changes without the focus
|
|
|
|
// changing. An example case is how we map OpenOffice.org based on the window
|
|
|
|
// title which is a dynamic property.
|
|
|
|
this._sync();
|
|
|
|
},
|
|
|
|
|
2012-03-13 19:20:37 +01:00
|
|
|
_focusAppChanged: function() {
|
2009-10-15 19:28:29 -04:00
|
|
|
let tracker = Shell.WindowTracker.get_default();
|
|
|
|
let focusedApp = tracker.focus_app;
|
2010-07-02 15:36:56 -04:00
|
|
|
if (!focusedApp) {
|
|
|
|
// If the app has just lost focus to the panel, pretend
|
|
|
|
// nothing happened; otherwise you can't keynav to the
|
|
|
|
// app menu.
|
Rework window / actor focus handling
The duality of the Clutter's key focus and mutter's window focus has long been
a problem for us in lots of case, and caused us to create large and complicated
hacks to get around the issue, including GrabHelper's focus grab model.
Instead of doing this, tie basic focus management into the core of gnome-shell,
instead of requiring complex "application-level" management to get it done
right.
Do this by making sure that only one of an actor or window can be focused at
the same time, and apply the appropriate logic to drop one or the other,
reactively.
Modals are considered a special case, as we grab all keyboard events, but at
the X level, the client window still has focus. Make sure to not do any input
synchronization when we have a modal.
At the same time, remove the FOCUSED input mode, as it's no longer necessary.
https://bugzilla.gnome.org/show_bug.cgi?id=700735
2013-05-18 00:18:13 -04:00
|
|
|
if (global.stage.key_focus != null)
|
2010-07-02 15:36:56 -04:00
|
|
|
return;
|
|
|
|
}
|
2012-03-13 19:20:37 +01:00
|
|
|
this._sync();
|
|
|
|
},
|
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
_findTargetApp: function() {
|
|
|
|
let workspace = global.screen.get_active_workspace();
|
2012-03-13 19:20:37 +01:00
|
|
|
let tracker = Shell.WindowTracker.get_default();
|
|
|
|
let focusedApp = tracker.focus_app;
|
2013-08-13 07:41:44 -04:00
|
|
|
if (focusedApp && focusedApp.is_on_workspace(workspace))
|
|
|
|
return focusedApp;
|
|
|
|
|
2012-03-13 19:20:37 +01:00
|
|
|
for (let i = 0; i < this._startingApps.length; i++)
|
|
|
|
if (this._startingApps[i].is_on_workspace(workspace))
|
2013-08-13 07:41:44 -04:00
|
|
|
return this._startingApps[i];
|
2010-07-02 15:36:56 -04:00
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
return null;
|
|
|
|
},
|
2011-03-08 20:46:43 +03:00
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
_sync: function() {
|
|
|
|
let targetApp = this._findTargetApp();
|
2011-11-16 00:12:58 +00:00
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
if (this._targetApp != targetApp) {
|
|
|
|
if (this._appMenuNotifyId) {
|
|
|
|
this._targetApp.disconnect(this._appMenuNotifyId);
|
|
|
|
this._appMenuNotifyId = 0;
|
|
|
|
}
|
|
|
|
if (this._actionGroupNotifyId) {
|
|
|
|
this._targetApp.disconnect(this._actionGroupNotifyId);
|
|
|
|
this._actionGroupNotifyId = 0;
|
|
|
|
}
|
2011-03-08 20:46:43 +03:00
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
this._targetApp = targetApp;
|
2011-03-08 20:46:43 +03:00
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
if (this._targetApp) {
|
|
|
|
this._appMenuNotifyId = this._targetApp.connect('notify::menu', Lang.bind(this, this._sync));
|
|
|
|
this._actionGroupNotifyId = this._targetApp.connect('notify::action-group', Lang.bind(this, this._sync));
|
|
|
|
this._label.setText(this._targetApp.get_name());
|
|
|
|
this.actor.set_accessible_name(this._targetApp.get_name());
|
2011-05-15 18:55:23 +02:00
|
|
|
}
|
2011-03-08 20:46:43 +03:00
|
|
|
}
|
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
let visible = (this._targetApp != null && !Main.overview.visibleTarget);
|
|
|
|
if (visible)
|
|
|
|
this.show();
|
|
|
|
else
|
|
|
|
this.hide();
|
2010-06-10 16:07:33 +04:00
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
let isBusy = (this._targetApp != null &&
|
|
|
|
(this._targetApp.get_state() == Shell.AppState.STARTING ||
|
|
|
|
this._targetApp.get_state() == Shell.AppState.BUSY));
|
|
|
|
if (isBusy)
|
2011-03-08 20:46:43 +03:00
|
|
|
this.startAnimation();
|
2011-05-15 18:55:23 +02:00
|
|
|
else
|
2013-08-13 07:41:44 -04:00
|
|
|
this.stopAnimation();
|
2009-08-11 11:32:58 -04:00
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
this.actor.reactive = (visible && !isBusy);
|
|
|
|
|
|
|
|
this._syncIcon();
|
|
|
|
this._maybeSetMenu();
|
2009-08-11 11:32:58 -04:00
|
|
|
this.emit('changed');
|
2011-05-15 18:55:23 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
_maybeSetMenu: function() {
|
|
|
|
let menu;
|
|
|
|
|
2013-08-13 07:41:44 -04:00
|
|
|
if (this._targetApp == null) {
|
|
|
|
menu = null;
|
|
|
|
} else if (this._targetApp.action_group && this._targetApp.menu) {
|
2013-05-09 17:35:57 -04:00
|
|
|
if (this.menu instanceof RemoteMenu.RemoteMenu &&
|
2011-05-15 18:55:23 +02:00
|
|
|
this.menu.actionGroup == this._targetApp.action_group)
|
|
|
|
return;
|
|
|
|
|
2013-05-09 17:35:57 -04:00
|
|
|
menu = new RemoteMenu.RemoteMenu(this.actor, this._targetApp.menu, this._targetApp.action_group);
|
2012-12-24 18:28:17 +01:00
|
|
|
menu.connect('activate', Lang.bind(this, function() {
|
|
|
|
let win = this._targetApp.get_windows()[0];
|
|
|
|
win.check_alive(global.get_current_time());
|
|
|
|
}));
|
|
|
|
|
2011-05-15 18:55:23 +02:00
|
|
|
} else {
|
2013-08-13 07:41:44 -04:00
|
|
|
if (this.menu && this.menu.isDummyQuitMenu)
|
2011-05-15 18:55:23 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
// fallback to older menu
|
|
|
|
menu = new PopupMenu.PopupMenu(this.actor, 0.0, St.Side.TOP, 0);
|
2012-12-04 14:52:34 -05:00
|
|
|
menu.isDummyQuitMenu = true;
|
2011-05-15 18:55:23 +02:00
|
|
|
menu.addAction(_("Quit"), Lang.bind(this, function() {
|
|
|
|
this._targetApp.request_quit();
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
this.setMenu(menu);
|
2013-08-13 07:41:44 -04:00
|
|
|
if (menu)
|
|
|
|
this._menuManager.addMenu(menu);
|
2013-04-22 09:49:14 +10:00
|
|
|
},
|
|
|
|
|
|
|
|
destroy: function() {
|
|
|
|
if (this._appStateChangedSignalId > 0) {
|
|
|
|
let appSys = Shell.AppSystem.get_default();
|
|
|
|
appSys.disconnect(this._appStateChangedSignalId);
|
|
|
|
this._appStateChangedSignalId = 0;
|
|
|
|
}
|
|
|
|
if (this._focusAppNotifyId > 0) {
|
|
|
|
let tracker = Shell.WindowTracker.get_default();
|
|
|
|
tracker.disconnect(this._focusAppNotifyId);
|
|
|
|
this._focusAppNotifyId = 0;
|
|
|
|
}
|
|
|
|
if (this._overviewHidingId > 0) {
|
|
|
|
Main.overview.disconnect(this._overviewHidingId);
|
|
|
|
this._overviewHidingId = 0;
|
|
|
|
}
|
|
|
|
if (this._overviewShowingId > 0) {
|
|
|
|
Main.overview.disconnect(this._overviewShowingId);
|
|
|
|
this._overviewShowingId = 0;
|
|
|
|
}
|
|
|
|
if (this._switchWorkspaceNotifyId > 0) {
|
|
|
|
global.window_manager.disconnect(this._switchWorkspaceNotifyId);
|
|
|
|
this._switchWorkspaceNotifyId = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.parent();
|
2009-08-11 11:32:58 -04:00
|
|
|
}
|
2011-11-20 15:38:48 +01:00
|
|
|
});
|
2009-08-11 11:32:58 -04:00
|
|
|
|
2010-05-05 10:03:48 -04:00
|
|
|
Signals.addSignalMethods(AppMenuButton.prototype);
|
2009-08-11 11:32:58 -04:00
|
|
|
|
2011-11-20 15:38:48 +01:00
|
|
|
const ActivitiesButton = new Lang.Class({
|
|
|
|
Name: 'ActivitiesButton',
|
|
|
|
Extends: PanelMenu.Button,
|
2011-07-14 08:56:14 -04:00
|
|
|
|
2011-07-14 08:53:44 -04:00
|
|
|
_init: function() {
|
2012-12-03 14:32:06 -05:00
|
|
|
this.parent(0.0, null, true);
|
2012-02-27 17:31:10 +01:00
|
|
|
this.actor.accessible_role = Atk.Role.TOGGLE_BUTTON;
|
2011-07-14 08:56:14 -04:00
|
|
|
|
2011-07-14 08:35:55 -04:00
|
|
|
this.actor.name = 'panelActivities';
|
|
|
|
|
2011-07-14 08:53:44 -04:00
|
|
|
/* Translators: If there is no suitable word for "Activities"
|
|
|
|
in your language, you can use the word for "Overview". */
|
2013-08-03 08:30:46 -04:00
|
|
|
this._label = new St.Label({ text: _("Activities"),
|
|
|
|
y_align: Clutter.ActorAlign.CENTER });
|
2013-03-01 16:00:37 -05:00
|
|
|
this.actor.add_actor(this._label);
|
2011-07-14 08:35:55 -04:00
|
|
|
|
2012-02-17 18:16:53 +01:00
|
|
|
this.actor.label_actor = this._label;
|
|
|
|
|
2011-07-14 08:56:14 -04:00
|
|
|
this.actor.connect('captured-event', Lang.bind(this, this._onCapturedEvent));
|
|
|
|
this.actor.connect_after('button-release-event', Lang.bind(this, this._onButtonRelease));
|
|
|
|
this.actor.connect_after('key-release-event', Lang.bind(this, this._onKeyRelease));
|
2011-07-14 08:53:44 -04:00
|
|
|
|
|
|
|
Main.overview.connect('showing', Lang.bind(this, function() {
|
2011-07-14 08:56:14 -04:00
|
|
|
this.actor.add_style_pseudo_class('overview');
|
2012-03-10 02:06:03 +01:00
|
|
|
this.actor.add_accessible_state (Atk.StateType.CHECKED);
|
2011-07-14 08:53:44 -04:00
|
|
|
}));
|
|
|
|
Main.overview.connect('hiding', Lang.bind(this, function() {
|
2011-07-14 08:56:14 -04:00
|
|
|
this.actor.remove_style_pseudo_class('overview');
|
2012-03-10 02:06:03 +01:00
|
|
|
this.actor.remove_accessible_state (Atk.StateType.CHECKED);
|
2011-07-14 08:53:44 -04:00
|
|
|
}));
|
|
|
|
|
|
|
|
this._xdndTimeOut = 0;
|
|
|
|
},
|
|
|
|
|
|
|
|
handleDragOver: function(source, actor, x, y, time) {
|
|
|
|
if (source != Main.xdndHandler)
|
2012-02-11 11:14:43 +01:00
|
|
|
return DND.DragMotionResult.CONTINUE;
|
2011-07-14 08:53:44 -04:00
|
|
|
|
|
|
|
if (this._xdndTimeOut != 0)
|
|
|
|
Mainloop.source_remove(this._xdndTimeOut);
|
|
|
|
this._xdndTimeOut = Mainloop.timeout_add(BUTTON_DND_ACTIVATION_TIMEOUT,
|
2013-03-04 17:02:02 -05:00
|
|
|
Lang.bind(this, this._xdndToggleOverview, actor));
|
2012-02-11 11:14:43 +01:00
|
|
|
|
|
|
|
return DND.DragMotionResult.CONTINUE;
|
2011-07-14 08:53:44 -04:00
|
|
|
},
|
|
|
|
|
2011-07-14 08:56:14 -04:00
|
|
|
_onCapturedEvent: function(actor, event) {
|
|
|
|
if (event.type() == Clutter.EventType.BUTTON_PRESS) {
|
2013-03-01 16:00:37 -05:00
|
|
|
if (!Main.overview.shouldToggleByCornerOrButton())
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_STOP;
|
2011-07-14 08:56:14 -04:00
|
|
|
}
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-07-14 08:56:14 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
_onButtonRelease: function() {
|
2012-12-03 14:32:06 -05:00
|
|
|
Main.overview.toggle();
|
2013-09-20 14:09:29 +02:00
|
|
|
this.menu.close();
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-07-14 08:56:14 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
_onKeyRelease: function(actor, event) {
|
|
|
|
let symbol = event.get_key_symbol();
|
|
|
|
if (symbol == Clutter.KEY_Return || symbol == Clutter.KEY_space) {
|
|
|
|
Main.overview.toggle();
|
|
|
|
}
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-07-14 08:56:14 -04:00
|
|
|
},
|
|
|
|
|
2013-03-04 17:02:02 -05:00
|
|
|
_xdndToggleOverview: function(actor) {
|
2011-07-14 08:53:44 -04:00
|
|
|
let [x, y, mask] = global.get_pointer();
|
|
|
|
let pickedActor = global.stage.get_actor_at_pos(Clutter.PickMode.REACTIVE, x, y);
|
|
|
|
|
2013-03-04 17:02:02 -05:00
|
|
|
if (pickedActor == this.actor && Main.overview.shouldToggleByCornerOrButton())
|
|
|
|
Main.overview.toggle();
|
2011-07-14 08:53:44 -04:00
|
|
|
|
|
|
|
Mainloop.source_remove(this._xdndTimeOut);
|
|
|
|
this._xdndTimeOut = 0;
|
2013-11-29 01:45:39 +01:00
|
|
|
return GLib.SOURCE_REMOVE;
|
2011-07-14 08:53:44 -04:00
|
|
|
}
|
2011-11-20 18:56:27 +01:00
|
|
|
});
|
2011-02-09 03:12:10 +01:00
|
|
|
|
2011-11-20 18:56:27 +01:00
|
|
|
const PanelCorner = new Lang.Class({
|
|
|
|
Name: 'PanelCorner',
|
2011-02-09 03:12:10 +01:00
|
|
|
|
2012-12-13 22:35:00 +01:00
|
|
|
_init: function(side) {
|
2011-02-09 03:12:10 +01:00
|
|
|
this._side = side;
|
2011-08-28 16:07:25 -04:00
|
|
|
|
2011-02-09 03:12:10 +01:00
|
|
|
this.actor = new St.DrawingArea({ style_class: 'panel-corner' });
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor.connect('style-changed', Lang.bind(this, this._styleChanged));
|
2011-02-09 03:12:10 +01:00
|
|
|
this.actor.connect('repaint', Lang.bind(this, this._repaint));
|
|
|
|
},
|
|
|
|
|
2011-08-28 16:07:25 -04:00
|
|
|
_findRightmostButton: function(container) {
|
|
|
|
if (!container.get_children)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
let children = container.get_children();
|
|
|
|
|
|
|
|
if (!children || children.length == 0)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
// Start at the back and work backward
|
2012-07-17 19:27:43 +02:00
|
|
|
let index;
|
|
|
|
for (index = children.length - 1; index >= 0; index--) {
|
|
|
|
if (children[index].visible)
|
|
|
|
break;
|
|
|
|
}
|
2011-08-28 16:07:25 -04:00
|
|
|
if (index < 0)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (!(children[index].has_style_class_name('panel-menu')) &&
|
|
|
|
!(children[index].has_style_class_name('panel-button')))
|
|
|
|
return this._findRightmostButton(children[index]);
|
|
|
|
|
|
|
|
return children[index];
|
|
|
|
},
|
|
|
|
|
|
|
|
_findLeftmostButton: function(container) {
|
|
|
|
if (!container.get_children)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
let children = container.get_children();
|
|
|
|
|
|
|
|
if (!children || children.length == 0)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
// Start at the front and work forward
|
2012-07-17 19:27:43 +02:00
|
|
|
let index;
|
|
|
|
for (index = 0; index < children.length; index++) {
|
|
|
|
if (children[index].visible)
|
|
|
|
break;
|
|
|
|
}
|
2011-08-28 16:07:25 -04:00
|
|
|
if (index == children.length)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (!(children[index].has_style_class_name('panel-menu')) &&
|
|
|
|
!(children[index].has_style_class_name('panel-button')))
|
|
|
|
return this._findLeftmostButton(children[index]);
|
|
|
|
|
|
|
|
return children[index];
|
|
|
|
},
|
|
|
|
|
2012-12-13 22:35:00 +01:00
|
|
|
setStyleParent: function(box) {
|
2011-09-10 03:41:39 +02:00
|
|
|
let side = this._side;
|
|
|
|
|
2012-12-13 22:35:00 +01:00
|
|
|
let rtlAwareContainer = box instanceof St.BoxLayout;
|
2011-09-10 03:41:39 +02:00
|
|
|
if (rtlAwareContainer &&
|
2012-12-13 22:35:00 +01:00
|
|
|
box.get_text_direction() == Clutter.TextDirection.RTL) {
|
2011-09-10 03:41:39 +02:00
|
|
|
if (this._side == St.Side.LEFT)
|
|
|
|
side = St.Side.RIGHT;
|
|
|
|
else if (this._side == St.Side.RIGHT)
|
|
|
|
side = St.Side.LEFT;
|
|
|
|
}
|
2011-08-28 16:07:25 -04:00
|
|
|
|
2011-09-10 03:41:39 +02:00
|
|
|
let button;
|
|
|
|
if (side == St.Side.LEFT)
|
2012-12-13 22:35:00 +01:00
|
|
|
button = this._findLeftmostButton(box);
|
2011-09-10 03:41:39 +02:00
|
|
|
else if (side == St.Side.RIGHT)
|
2012-12-13 22:35:00 +01:00
|
|
|
button = this._findRightmostButton(box);
|
2011-08-28 16:07:25 -04:00
|
|
|
|
|
|
|
if (button) {
|
2011-09-07 12:23:43 -04:00
|
|
|
if (this._button && this._buttonStyleChangedSignalId) {
|
2011-08-28 16:07:25 -04:00
|
|
|
this._button.disconnect(this._buttonStyleChangedSignalId);
|
2011-09-07 12:23:43 -04:00
|
|
|
this._button.style = null;
|
|
|
|
}
|
2011-08-28 16:07:25 -04:00
|
|
|
|
|
|
|
this._button = button;
|
|
|
|
|
|
|
|
button.connect('destroy', Lang.bind(this,
|
|
|
|
function() {
|
|
|
|
if (this._button == button) {
|
|
|
|
this._button = null;
|
|
|
|
this._buttonStyleChangedSignalId = 0;
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
|
|
|
// Synchronize the locate button's pseudo classes with this corner
|
|
|
|
this._buttonStyleChangedSignalId = button.connect('style-changed', Lang.bind(this,
|
|
|
|
function(actor) {
|
|
|
|
let pseudoClass = button.get_style_pseudo_class();
|
|
|
|
this.actor.set_style_pseudo_class(pseudoClass);
|
|
|
|
}));
|
2011-09-07 12:23:43 -04:00
|
|
|
|
|
|
|
// The corner doesn't support theme transitions, so override
|
|
|
|
// the .panel-button default
|
2013-01-12 01:29:10 +01:00
|
|
|
button.style = 'transition-duration: 0ms';
|
2011-08-28 16:07:25 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2011-02-09 03:12:10 +01:00
|
|
|
_repaint: function() {
|
|
|
|
let node = this.actor.get_theme_node();
|
|
|
|
|
|
|
|
let cornerRadius = node.get_length("-panel-corner-radius");
|
2012-02-08 14:31:42 +01:00
|
|
|
let borderWidth = node.get_length('-panel-corner-border-width');
|
2011-02-09 03:12:10 +01:00
|
|
|
|
|
|
|
let backgroundColor = node.get_color('-panel-corner-background-color');
|
2012-02-08 14:31:42 +01:00
|
|
|
let borderColor = node.get_color('-panel-corner-border-color');
|
2011-02-09 03:12:10 +01:00
|
|
|
|
2013-01-07 15:07:40 -05:00
|
|
|
let overlap = borderColor.alpha != 0;
|
|
|
|
let offsetY = overlap ? 0 : borderWidth;
|
2012-09-21 22:10:07 +02:00
|
|
|
|
2011-02-09 03:12:10 +01:00
|
|
|
let cr = this.actor.get_context();
|
|
|
|
cr.setOperator(Cairo.Operator.SOURCE);
|
|
|
|
|
2012-09-21 22:10:07 +02:00
|
|
|
cr.moveTo(0, offsetY);
|
2011-02-09 03:12:10 +01:00
|
|
|
if (this._side == St.Side.LEFT)
|
|
|
|
cr.arc(cornerRadius,
|
2012-02-08 14:31:42 +01:00
|
|
|
borderWidth + cornerRadius,
|
2011-02-09 03:12:10 +01:00
|
|
|
cornerRadius, Math.PI, 3 * Math.PI / 2);
|
|
|
|
else
|
|
|
|
cr.arc(0,
|
2012-02-08 14:31:42 +01:00
|
|
|
borderWidth + cornerRadius,
|
2011-02-09 03:12:10 +01:00
|
|
|
cornerRadius, 3 * Math.PI / 2, 2 * Math.PI);
|
2012-09-21 22:10:07 +02:00
|
|
|
cr.lineTo(cornerRadius, offsetY);
|
2011-02-09 03:12:10 +01:00
|
|
|
cr.closePath();
|
|
|
|
|
|
|
|
let savedPath = cr.copyPath();
|
|
|
|
|
|
|
|
let xOffsetDirection = this._side == St.Side.LEFT ? -1 : 1;
|
2012-02-08 14:31:42 +01:00
|
|
|
let over = _over(borderColor, backgroundColor);
|
2011-02-09 03:12:10 +01:00
|
|
|
Clutter.cairo_set_source_color(cr, over);
|
|
|
|
cr.fill();
|
|
|
|
|
2013-01-07 15:07:40 -05:00
|
|
|
if (overlap) {
|
|
|
|
let offset = borderWidth;
|
|
|
|
Clutter.cairo_set_source_color(cr, backgroundColor);
|
2012-09-21 22:10:07 +02:00
|
|
|
|
2013-01-07 15:07:40 -05:00
|
|
|
cr.save();
|
|
|
|
cr.translate(xOffsetDirection * offset, - offset);
|
|
|
|
cr.appendPath(savedPath);
|
|
|
|
cr.fill();
|
|
|
|
cr.restore();
|
|
|
|
}
|
2011-02-09 03:12:10 +01:00
|
|
|
|
2013-01-07 15:07:40 -05:00
|
|
|
cr.$dispose();
|
2011-02-09 03:12:10 +01:00
|
|
|
},
|
|
|
|
|
2011-07-21 10:49:04 -04:00
|
|
|
_styleChanged: function() {
|
2011-02-09 03:12:10 +01:00
|
|
|
let node = this.actor.get_theme_node();
|
|
|
|
|
|
|
|
let cornerRadius = node.get_length("-panel-corner-radius");
|
2012-02-08 14:31:42 +01:00
|
|
|
let borderWidth = node.get_length('-panel-corner-border-width');
|
2011-02-09 03:12:10 +01:00
|
|
|
|
2012-02-08 14:31:42 +01:00
|
|
|
this.actor.set_size(cornerRadius, borderWidth + cornerRadius);
|
|
|
|
this.actor.set_anchor_point(0, borderWidth);
|
2011-02-09 03:12:10 +01:00
|
|
|
}
|
2011-11-20 18:56:27 +01:00
|
|
|
});
|
2011-02-09 03:12:10 +01:00
|
|
|
|
2013-06-06 17:27:32 -04:00
|
|
|
const AggregateMenu = new Lang.Class({
|
|
|
|
Name: 'AggregateMenu',
|
|
|
|
Extends: PanelMenu.Button,
|
|
|
|
|
|
|
|
_init: function() {
|
2013-09-14 13:43:45 -04:00
|
|
|
this.parent(0.0, _("Settings"), false);
|
2013-07-01 14:19:12 -04:00
|
|
|
this.menu.actor.add_style_class_name('aggregate-menu');
|
2013-06-06 17:27:32 -04:00
|
|
|
|
|
|
|
this._indicators = new St.BoxLayout({ style_class: 'panel-status-indicators-box' });
|
|
|
|
this.actor.add_child(this._indicators);
|
|
|
|
|
|
|
|
this._network = new imports.ui.status.network.NMApplet();
|
2013-09-04 01:22:46 +02:00
|
|
|
if (Config.HAVE_BLUETOOTH) {
|
|
|
|
this._bluetooth = new imports.ui.status.bluetooth.Indicator();
|
|
|
|
} else {
|
|
|
|
this._bluetooth = null;
|
|
|
|
}
|
|
|
|
|
2013-06-06 17:27:32 -04:00
|
|
|
this._power = new imports.ui.status.power.Indicator();
|
2013-05-27 17:54:50 -04:00
|
|
|
this._rfkill = new imports.ui.status.rfkill.Indicator();
|
2013-06-06 17:27:32 -04:00
|
|
|
this._volume = new imports.ui.status.volume.Indicator();
|
2013-04-23 19:26:05 -04:00
|
|
|
this._brightness = new imports.ui.status.brightness.Indicator();
|
2013-06-06 17:27:32 -04:00
|
|
|
this._system = new imports.ui.status.system.Indicator();
|
2013-07-17 15:26:06 -04:00
|
|
|
this._screencast = new imports.ui.status.screencast.Indicator();
|
2014-01-28 17:41:26 +00:00
|
|
|
this._location = new imports.ui.status.location.Indicator();
|
2013-06-06 17:27:32 -04:00
|
|
|
|
2013-07-17 15:26:06 -04:00
|
|
|
this._indicators.add_child(this._screencast.indicators);
|
2014-01-28 17:41:26 +00:00
|
|
|
this._indicators.add_child(this._location.indicators);
|
2013-06-06 17:27:32 -04:00
|
|
|
this._indicators.add_child(this._network.indicators);
|
2013-09-04 01:22:46 +02:00
|
|
|
if (this._bluetooth) {
|
|
|
|
this._indicators.add_child(this._bluetooth.indicators);
|
|
|
|
}
|
2013-05-27 17:54:50 -04:00
|
|
|
this._indicators.add_child(this._rfkill.indicators);
|
2013-06-06 17:27:32 -04:00
|
|
|
this._indicators.add_child(this._volume.indicators);
|
2013-06-17 18:19:20 -04:00
|
|
|
this._indicators.add_child(this._power.indicators);
|
2013-10-07 16:01:07 +02:00
|
|
|
this._indicators.add_child(PopupMenu.unicodeArrow(St.Side.BOTTOM));
|
2013-06-06 17:27:32 -04:00
|
|
|
|
|
|
|
this.menu.addMenuItem(this._volume.menu);
|
2013-04-23 19:26:05 -04:00
|
|
|
this.menu.addMenuItem(this._brightness.menu);
|
2013-06-06 17:27:32 -04:00
|
|
|
this.menu.addMenuItem(new PopupMenu.PopupSeparatorMenuItem());
|
|
|
|
this.menu.addMenuItem(this._network.menu);
|
2013-09-04 01:22:46 +02:00
|
|
|
if (this._bluetooth) {
|
|
|
|
this.menu.addMenuItem(this._bluetooth.menu);
|
|
|
|
}
|
2013-05-27 17:54:50 -04:00
|
|
|
this.menu.addMenuItem(this._rfkill.menu);
|
2013-06-06 17:27:32 -04:00
|
|
|
this.menu.addMenuItem(this._power.menu);
|
|
|
|
this.menu.addMenuItem(new PopupMenu.PopupSeparatorMenuItem());
|
|
|
|
this.menu.addMenuItem(this._system.menu);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2012-08-28 23:10:44 +02:00
|
|
|
const PANEL_ITEM_IMPLEMENTATIONS = {
|
|
|
|
'activities': ActivitiesButton,
|
2013-06-06 17:27:32 -04:00
|
|
|
'aggregateMenu': AggregateMenu,
|
2012-08-28 23:10:44 +02:00
|
|
|
'appMenu': AppMenuButton,
|
|
|
|
'dateMenu': imports.ui.dateMenu.DateMenuButton,
|
|
|
|
'a11y': imports.ui.status.accessibility.ATIndicator,
|
2012-08-13 19:04:03 +02:00
|
|
|
'a11yGreeter': imports.ui.status.accessibility.ATGreeterIndicator,
|
2012-08-28 23:10:44 +02:00
|
|
|
'keyboard': imports.ui.status.keyboard.InputSourceIndicator,
|
|
|
|
};
|
|
|
|
|
2011-11-20 18:56:27 +01:00
|
|
|
const Panel = new Lang.Class({
|
|
|
|
Name: 'Panel',
|
2008-10-31 18:09:20 +00:00
|
|
|
|
2010-05-05 10:03:48 -04:00
|
|
|
_init : function() {
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor = new Shell.GenericContainer({ name: 'panel',
|
|
|
|
reactive: true });
|
2009-11-10 12:13:58 -05:00
|
|
|
this.actor._delegate = this;
|
|
|
|
|
2012-09-21 22:19:27 +02:00
|
|
|
this._sessionStyle = null;
|
|
|
|
|
2012-08-26 16:05:46 +02:00
|
|
|
this.statusArea = {};
|
2011-04-06 09:26:15 -04:00
|
|
|
|
2013-04-26 15:48:36 +02:00
|
|
|
this.menuManager = new PopupMenu.PopupMenuManager(this, { keybindingMode: Shell.KeyBindingMode.TOPBAR_POPUP });
|
2010-05-20 11:18:46 -04:00
|
|
|
|
2009-11-17 17:46:20 -05:00
|
|
|
this._leftBox = new St.BoxLayout({ name: 'panelLeft' });
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor.add_actor(this._leftBox);
|
2009-11-17 17:46:20 -05:00
|
|
|
this._centerBox = new St.BoxLayout({ name: 'panelCenter' });
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor.add_actor(this._centerBox);
|
2009-11-17 17:46:20 -05:00
|
|
|
this._rightBox = new St.BoxLayout({ name: 'panelRight' });
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor.add_actor(this._rightBox);
|
2009-08-11 11:16:25 -04:00
|
|
|
|
2012-12-13 22:35:00 +01:00
|
|
|
this._leftCorner = new PanelCorner(St.Side.LEFT);
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor.add_actor(this._leftCorner.actor);
|
2011-08-28 16:07:25 -04:00
|
|
|
|
2012-12-13 22:35:00 +01:00
|
|
|
this._rightCorner = new PanelCorner(St.Side.RIGHT);
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor.add_actor(this._rightCorner.actor);
|
2010-05-25 10:21:22 -04:00
|
|
|
|
2011-07-21 10:49:04 -04:00
|
|
|
this.actor.connect('get-preferred-width', Lang.bind(this, this._getPreferredWidth));
|
|
|
|
this.actor.connect('get-preferred-height', Lang.bind(this, this._getPreferredHeight));
|
|
|
|
this.actor.connect('allocate', Lang.bind(this, this._allocate));
|
2011-10-04 17:14:29 +02:00
|
|
|
this.actor.connect('button-press-event', Lang.bind(this, this._onButtonPress));
|
2009-08-11 11:16:25 -04:00
|
|
|
|
2013-02-05 15:24:46 +01:00
|
|
|
Main.overview.connect('showing', Lang.bind(this, function () {
|
|
|
|
this.actor.add_style_pseudo_class('overview');
|
|
|
|
}));
|
|
|
|
Main.overview.connect('hiding', Lang.bind(this, function () {
|
|
|
|
this.actor.remove_style_pseudo_class('overview');
|
|
|
|
}));
|
|
|
|
|
layout: Add sessionGroup / systemGroup to better-define layers
In order to build a better transition animation from the lock screen, we
need to split the world into layers, as per this reference:
https://raw.github.com/gnome-design-team/gnome-mockups/master/system-lock-login-boot/system-layers2.png
Everything that pertains to the user's session is in the "session group",
which includes the window group, overview, message tray (for now),
keyboard, OSDs, menus, etc.
For implementation sake, we did not match this mockup exactly. The new layers
look like this, from top to bottom:
* Stage
* Magnifier (clones the uiGroup)
* uiGroup
* overlayGroup
* menuGroup
* panelGroup
* screenShieldGroup
* sessionGroup
* top_window_group
* other boxes (trayBox, keyboardBox, etc.)
* other groups (osdGroup, switcherPopupGroup, etc.)
* overviewGroup
* window_group
* systemGroup
The "session startup" animation now only zooms in the sessionGroup.
The panel is now outside the session, as it needs to sit above the screen
shield. This also means that it's not zoomed in as part of startup. I think
this is OK.
This also means that the lightboxes that the screen shield uses to fade out
the screen have to go in a new group, above the panel. This is known as the
overlayGroup, which has no relation to the old mutter group of the same name.
We also change the screen shield to put the lockDialogGroup in the system
group, and put the lockScreenGroup in the screenShieldGroup, which means
that the layer stacking is correct. Note that we don't hide the session
group in the lock screen yet, which is something I want to do.
Since not a lot of items need to be in the uiGroup anymore, we've removed
the Main.uiGroup fallback; others should use sessionGroup instead, when
appropriate.
2014-01-13 15:56:21 -05:00
|
|
|
Main.layoutManager.panelGroup.add_child(this.actor);
|
2012-11-20 11:51:56 +01:00
|
|
|
Main.ctrlAltTabManager.addGroup(this.actor, _("Top Bar"), 'emblem-system-symbolic',
|
2011-02-07 11:29:34 -05:00
|
|
|
{ sortGroup: CtrlAltTab.SortGroup.TOP });
|
2012-09-01 09:42:53 -03:00
|
|
|
|
2012-09-06 21:22:57 +02:00
|
|
|
Main.sessionMode.connect('updated', Lang.bind(this, this._updatePanel));
|
2012-09-01 09:42:53 -03:00
|
|
|
this._updatePanel();
|
2008-12-01 19:51:43 +00:00
|
|
|
},
|
|
|
|
|
2011-07-21 10:49:04 -04:00
|
|
|
_getPreferredWidth: function(actor, forHeight, alloc) {
|
|
|
|
alloc.min_size = -1;
|
|
|
|
alloc.natural_size = Main.layoutManager.primaryMonitor.width;
|
|
|
|
},
|
|
|
|
|
|
|
|
_getPreferredHeight: function(actor, forWidth, alloc) {
|
|
|
|
// We don't need to implement this; it's forced by the CSS
|
|
|
|
alloc.min_size = -1;
|
|
|
|
alloc.natural_size = -1;
|
|
|
|
},
|
|
|
|
|
|
|
|
_allocate: function(actor, box, flags) {
|
|
|
|
let allocWidth = box.x2 - box.x1;
|
|
|
|
let allocHeight = box.y2 - box.y1;
|
|
|
|
|
|
|
|
let [leftMinWidth, leftNaturalWidth] = this._leftBox.get_preferred_width(-1);
|
|
|
|
let [centerMinWidth, centerNaturalWidth] = this._centerBox.get_preferred_width(-1);
|
|
|
|
let [rightMinWidth, rightNaturalWidth] = this._rightBox.get_preferred_width(-1);
|
|
|
|
|
|
|
|
let sideWidth, centerWidth;
|
|
|
|
centerWidth = centerNaturalWidth;
|
|
|
|
sideWidth = (allocWidth - centerWidth) / 2;
|
|
|
|
|
|
|
|
let childBox = new Clutter.ActorBox();
|
|
|
|
|
|
|
|
childBox.y1 = 0;
|
|
|
|
childBox.y2 = allocHeight;
|
2012-02-13 20:37:28 -05:00
|
|
|
if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) {
|
2011-07-21 10:49:04 -04:00
|
|
|
childBox.x1 = allocWidth - Math.min(Math.floor(sideWidth),
|
|
|
|
leftNaturalWidth);
|
|
|
|
childBox.x2 = allocWidth;
|
|
|
|
} else {
|
|
|
|
childBox.x1 = 0;
|
|
|
|
childBox.x2 = Math.min(Math.floor(sideWidth),
|
|
|
|
leftNaturalWidth);
|
|
|
|
}
|
|
|
|
this._leftBox.allocate(childBox, flags);
|
|
|
|
|
|
|
|
childBox.x1 = Math.ceil(sideWidth);
|
|
|
|
childBox.y1 = 0;
|
|
|
|
childBox.x2 = childBox.x1 + centerWidth;
|
|
|
|
childBox.y2 = allocHeight;
|
|
|
|
this._centerBox.allocate(childBox, flags);
|
|
|
|
|
|
|
|
childBox.y1 = 0;
|
|
|
|
childBox.y2 = allocHeight;
|
2012-02-13 20:37:28 -05:00
|
|
|
if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) {
|
2011-07-21 10:49:04 -04:00
|
|
|
childBox.x1 = 0;
|
|
|
|
childBox.x2 = Math.min(Math.floor(sideWidth),
|
|
|
|
rightNaturalWidth);
|
|
|
|
} else {
|
|
|
|
childBox.x1 = allocWidth - Math.min(Math.floor(sideWidth),
|
|
|
|
rightNaturalWidth);
|
|
|
|
childBox.x2 = allocWidth;
|
|
|
|
}
|
|
|
|
this._rightBox.allocate(childBox, flags);
|
|
|
|
|
2012-10-12 13:51:48 -04:00
|
|
|
let cornerMinWidth, cornerMinHeight;
|
|
|
|
let cornerWidth, cornerHeight;
|
|
|
|
|
|
|
|
[cornerMinWidth, cornerWidth] = this._leftCorner.actor.get_preferred_width(-1);
|
|
|
|
[cornerMinHeight, cornerHeight] = this._leftCorner.actor.get_preferred_height(-1);
|
2011-07-21 10:49:04 -04:00
|
|
|
childBox.x1 = 0;
|
|
|
|
childBox.x2 = cornerWidth;
|
|
|
|
childBox.y1 = allocHeight;
|
|
|
|
childBox.y2 = allocHeight + cornerHeight;
|
|
|
|
this._leftCorner.actor.allocate(childBox, flags);
|
|
|
|
|
2012-10-12 13:51:48 -04:00
|
|
|
[cornerMinWidth, cornerWidth] = this._rightCorner.actor.get_preferred_width(-1);
|
|
|
|
[cornerMinHeight, cornerHeight] = this._rightCorner.actor.get_preferred_height(-1);
|
2011-07-21 10:49:04 -04:00
|
|
|
childBox.x1 = allocWidth - cornerWidth;
|
|
|
|
childBox.x2 = allocWidth;
|
|
|
|
childBox.y1 = allocHeight;
|
|
|
|
childBox.y2 = allocHeight + cornerHeight;
|
|
|
|
this._rightCorner.actor.allocate(childBox, flags);
|
|
|
|
},
|
|
|
|
|
2011-10-04 17:14:29 +02:00
|
|
|
_onButtonPress: function(actor, event) {
|
2013-02-19 20:27:11 -05:00
|
|
|
if (Main.modalCount > 0)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2013-02-19 20:27:11 -05:00
|
|
|
|
2011-10-04 17:14:29 +02:00
|
|
|
if (event.get_source() != actor)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-10-04 17:14:29 +02:00
|
|
|
|
|
|
|
let button = event.get_button();
|
|
|
|
if (button != 1)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-10-04 17:14:29 +02:00
|
|
|
|
|
|
|
let focusWindow = global.display.focus_window;
|
|
|
|
if (!focusWindow)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-10-04 17:14:29 +02:00
|
|
|
|
|
|
|
let dragWindow = focusWindow.is_attached_dialog() ? focusWindow.get_transient_for()
|
|
|
|
: focusWindow;
|
|
|
|
if (!dragWindow)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-10-04 17:14:29 +02:00
|
|
|
|
|
|
|
let rect = dragWindow.get_outer_rect();
|
|
|
|
let [stageX, stageY] = event.get_coords();
|
|
|
|
|
|
|
|
let allowDrag = dragWindow.maximized_vertically &&
|
|
|
|
stageX > rect.x && stageX < rect.x + rect.width;
|
|
|
|
|
|
|
|
if (!allowDrag)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-10-04 17:14:29 +02:00
|
|
|
|
|
|
|
global.display.begin_grab_op(global.screen,
|
|
|
|
dragWindow,
|
|
|
|
Meta.GrabOp.MOVING,
|
|
|
|
false, /* pointer grab */
|
|
|
|
true, /* frame action */
|
|
|
|
button,
|
|
|
|
event.get_state(),
|
|
|
|
event.get_time(),
|
|
|
|
stageX, stageY);
|
|
|
|
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_STOP;
|
2011-10-04 17:14:29 +02:00
|
|
|
},
|
|
|
|
|
2013-04-26 16:14:55 +02:00
|
|
|
toggleAppMenu: function() {
|
2012-08-28 23:10:44 +02:00
|
|
|
let indicator = this.statusArea.appMenu;
|
|
|
|
if (!indicator) // appMenu not supported by current session mode
|
|
|
|
return;
|
|
|
|
|
|
|
|
let menu = indicator.menu;
|
2013-04-26 16:14:55 +02:00
|
|
|
if (!indicator.actor.reactive)
|
2012-05-18 16:04:47 +02:00
|
|
|
return;
|
2012-03-27 15:26:20 +02:00
|
|
|
|
2013-04-26 16:14:55 +02:00
|
|
|
menu.toggle();
|
|
|
|
if (menu.isOpen)
|
|
|
|
menu.actor.navigate_focus(null, Gtk.DirectionType.TAB_FORWARD, false);
|
2012-03-27 15:26:20 +02:00
|
|
|
},
|
|
|
|
|
2012-09-01 09:42:53 -03:00
|
|
|
set boxOpacity(value) {
|
|
|
|
let isReactive = value > 0;
|
|
|
|
|
|
|
|
this._leftBox.opacity = value;
|
|
|
|
this._leftBox.reactive = isReactive;
|
|
|
|
this._centerBox.opacity = value;
|
|
|
|
this._centerBox.reactive = isReactive;
|
|
|
|
this._rightBox.opacity = value;
|
|
|
|
this._rightBox.reactive = isReactive;
|
|
|
|
},
|
|
|
|
|
|
|
|
get boxOpacity() {
|
|
|
|
return this._leftBox.opacity;
|
|
|
|
},
|
|
|
|
|
|
|
|
_updatePanel: function() {
|
2012-08-28 23:10:44 +02:00
|
|
|
let panel = Main.sessionMode.panel;
|
2012-09-01 09:42:53 -03:00
|
|
|
this._hideIndicators();
|
|
|
|
this._updateBox(panel.left, this._leftBox);
|
|
|
|
this._updateBox(panel.center, this._centerBox);
|
|
|
|
this._updateBox(panel.right, this._rightBox);
|
2012-09-21 22:19:27 +02:00
|
|
|
|
|
|
|
if (this._sessionStyle)
|
|
|
|
this._removeStyleClassName(this._sessionStyle);
|
|
|
|
|
|
|
|
this._sessionStyle = Main.sessionMode.panelStyle;
|
|
|
|
if (this._sessionStyle)
|
|
|
|
this._addStyleClassName(this._sessionStyle);
|
2012-12-13 22:35:00 +01:00
|
|
|
|
|
|
|
if (this.actor.get_text_direction() == Clutter.TextDirection.RTL) {
|
|
|
|
this._leftCorner.setStyleParent(this._rightBox);
|
|
|
|
this._rightCorner.setStyleParent(this._leftBox);
|
|
|
|
} else {
|
|
|
|
this._leftCorner.setStyleParent(this._leftBox);
|
|
|
|
this._rightCorner.setStyleParent(this._rightBox);
|
|
|
|
}
|
2012-08-28 23:10:44 +02:00
|
|
|
},
|
|
|
|
|
2012-09-01 09:42:53 -03:00
|
|
|
_hideIndicators: function() {
|
|
|
|
for (let role in PANEL_ITEM_IMPLEMENTATIONS) {
|
|
|
|
let indicator = this.statusArea[role];
|
|
|
|
if (!indicator)
|
|
|
|
continue;
|
2012-09-06 21:22:57 +02:00
|
|
|
if (indicator.menu)
|
|
|
|
indicator.menu.close();
|
2012-09-04 18:27:50 +02:00
|
|
|
indicator.container.hide();
|
2012-09-01 09:42:53 -03:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_ensureIndicator: function(role) {
|
|
|
|
let indicator = this.statusArea[role];
|
|
|
|
if (!indicator) {
|
|
|
|
let constructor = PANEL_ITEM_IMPLEMENTATIONS[role];
|
|
|
|
if (!constructor) {
|
|
|
|
// This icon is not implemented (this is a bug)
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
indicator = new constructor(this);
|
|
|
|
this.statusArea[role] = indicator;
|
|
|
|
}
|
|
|
|
return indicator;
|
|
|
|
},
|
|
|
|
|
|
|
|
_updateBox: function(elements, box) {
|
|
|
|
let nChildren = box.get_n_children();
|
|
|
|
|
|
|
|
for (let i = 0; i < elements.length; i++) {
|
|
|
|
let role = elements[i];
|
|
|
|
let indicator = this._ensureIndicator(role);
|
|
|
|
if (indicator == null)
|
|
|
|
continue;
|
2011-04-06 09:26:15 -04:00
|
|
|
|
2012-09-01 09:42:53 -03:00
|
|
|
this._addToPanelBox(role, indicator, i + nChildren, box);
|
2010-07-24 13:57:53 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-08-28 23:10:44 +02:00
|
|
|
_addToPanelBox: function(role, indicator, position, box) {
|
2012-09-04 18:27:50 +02:00
|
|
|
let container = indicator.container;
|
2012-09-01 09:42:53 -03:00
|
|
|
container.show();
|
|
|
|
|
|
|
|
let parent = container.get_parent();
|
2012-09-04 18:27:50 +02:00
|
|
|
if (parent)
|
2012-09-01 09:42:53 -03:00
|
|
|
parent.remove_actor(container);
|
|
|
|
|
|
|
|
box.insert_child_at_index(container, position);
|
2012-08-28 23:10:44 +02:00
|
|
|
if (indicator.menu)
|
|
|
|
this.menuManager.addMenu(indicator.menu);
|
|
|
|
this.statusArea[role] = indicator;
|
|
|
|
let destroyId = indicator.connect('destroy', Lang.bind(this, function(emitter) {
|
|
|
|
delete this.statusArea[role];
|
|
|
|
emitter.disconnect(destroyId);
|
2012-09-01 09:42:53 -03:00
|
|
|
container.destroy();
|
2012-08-28 23:10:44 +02:00
|
|
|
}));
|
2011-08-25 11:46:25 -04:00
|
|
|
},
|
|
|
|
|
2012-08-28 23:10:44 +02:00
|
|
|
addToStatusArea: function(role, indicator, position, box) {
|
2012-08-26 16:05:46 +02:00
|
|
|
if (this.statusArea[role])
|
2011-08-22 23:19:13 +02:00
|
|
|
throw new Error('Extension point conflict: there is already a status indicator for role ' + role);
|
|
|
|
|
|
|
|
if (!(indicator instanceof PanelMenu.Button))
|
|
|
|
throw new TypeError('Status indicator must be an instance of PanelMenu.Button');
|
|
|
|
|
2012-08-28 23:10:44 +02:00
|
|
|
position = position || 0;
|
|
|
|
let boxes = {
|
|
|
|
left: this._leftBox,
|
|
|
|
center: this._centerBox,
|
|
|
|
right: this._rightBox
|
|
|
|
};
|
|
|
|
let boxContainer = boxes[box] || this._rightBox;
|
2012-09-01 09:42:53 -03:00
|
|
|
this.statusArea[role] = indicator;
|
2012-08-28 23:10:44 +02:00
|
|
|
this._addToPanelBox(role, indicator, position, boxContainer);
|
2011-08-22 23:19:13 +02:00
|
|
|
return indicator;
|
2012-09-21 22:19:27 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
_addStyleClassName: function(className) {
|
|
|
|
this.actor.add_style_class_name(className);
|
|
|
|
this._rightCorner.actor.add_style_class_name(className);
|
|
|
|
this._leftCorner.actor.add_style_class_name(className);
|
|
|
|
},
|
|
|
|
|
|
|
|
_removeStyleClassName: function(className) {
|
|
|
|
this.actor.remove_style_class_name(className);
|
|
|
|
this._rightCorner.actor.remove_style_class_name(className);
|
|
|
|
this._leftCorner.actor.remove_style_class_name(className);
|
2012-09-01 09:42:53 -03:00
|
|
|
}
|
2011-11-20 18:56:27 +01:00
|
|
|
});
|