App Picker: Prevent drag if favorite-apps is locked.

Because there's nothing (in single-monitor setups) that could
take the drop in this case.

* js/ui/appDisplay.js:
  AllView._loadApps(), FrequentView._loadApps(): Pass
  an isDraggable parameter when creating the AppIcons,
  depending on whether the favorite-apps key is locked.
  AppIcon._init(): Check for isDraggable in the params and
  do not create _draggable if it was specified, to prevent a
  drag from starting.
  AppIcon.popupMenu(): Check _draggable before trying to call
  fakeRelease on it.
* js/ui/dash.js: Dash._createAppItem(): Check AppIcon._draggable
  before trying to connect to its signals.

https://bugzilla.gnome.org/show_bug.cgi?id=741325
This commit is contained in:
Murray Cumming 2015-01-27 22:10:45 +01:00
parent e69cc20fc7
commit 1da9546c40
2 changed files with 61 additions and 25 deletions

View File

@ -366,6 +366,8 @@ const AllView = new Lang.Class({
Extends: BaseAppView, Extends: BaseAppView,
_init: function() { _init: function() {
this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' });
this.parent({ usePagination: true }, null); this.parent({ usePagination: true }, null);
this._scrollView = new St.ScrollView({ style_class: 'all-apps', this._scrollView = new St.ScrollView({ style_class: 'all-apps',
x_expand: true, x_expand: true,
@ -517,9 +519,19 @@ const AllView = new Lang.Class({
this.folderIcons.push(icon); this.folderIcons.push(icon);
})); }));
// Allow dragging of the icon only if the Dash would accept a drop to
// change favorite-apps. There are no other possible drop targets from
// the app picker, so there's no other need for a drag to start,
// at least on single-monitor setups.
// This also disables drag-to-launch on multi-monitor setups,
// but we hope that is not used much.
let favoritesWritable = this._settings.is_writable('favorite-apps');
apps.forEach(Lang.bind(this, function(appId) { apps.forEach(Lang.bind(this, function(appId) {
let app = appSys.lookup_app(appId); let app = appSys.lookup_app(appId);
let icon = new AppIcon(app);
let icon = new AppIcon(app,
{ isDraggable: favoritesWritable });
this.addItem(icon); this.addItem(icon);
})); }));
@ -763,6 +775,9 @@ const FrequentView = new Lang.Class({
_init: function() { _init: function() {
this.parent(null, { fillParent: true }); this.parent(null, { fillParent: true });
this._settings = new Gio.Settings({ schema_id: 'org.gnome.shell' });
this.actor = new St.Widget({ style_class: 'frequent-apps', this.actor = new St.Widget({ style_class: 'frequent-apps',
layout_manager: new Clutter.BinLayout(), layout_manager: new Clutter.BinLayout(),
x_expand: true, y_expand: true }); x_expand: true, y_expand: true });
@ -799,10 +814,19 @@ const FrequentView = new Lang.Class({
if(!hasUsefulData) if(!hasUsefulData)
return; return;
// Allow dragging of the icon only if the Dash would accept a drop to
// change favorite-apps. There are no other possible drop targets from
// the app picker, so there's no other need for a drag to start,
// at least on single-monitor setups.
// This also disables drag-to-launch on multi-monitor setups,
// but we hope that is not used much.
let favoritesWritable = this._settings.is_writable('favorite-apps');
for (let i = 0; i < mostUsed.length; i++) { for (let i = 0; i < mostUsed.length; i++) {
if (!mostUsed[i].get_app_info().should_show()) if (!mostUsed[i].get_app_info().should_show())
continue; continue;
let appIcon = new AppIcon(mostUsed[i]); let appIcon = new AppIcon(mostUsed[i],
{ isDraggable: favoritesWritable });
this._grid.addItem(appIcon, -1); this._grid.addItem(appIcon, -1);
} }
}, },
@ -1528,6 +1552,11 @@ const AppIcon = new Lang.Class({
if (!iconParams) if (!iconParams)
iconParams = {}; iconParams = {};
// Get the isDraggable property without passing it on to the BaseIcon:
let appIconParams = Params.parse(iconParams, { isDraggable: true }, true);
let isDraggable = appIconParams['isDraggable'];
delete iconParams['isDraggable'];
iconParams['createIcon'] = Lang.bind(this, this._createIcon); iconParams['createIcon'] = Lang.bind(this, this._createIcon);
iconParams['setSizeManually'] = true; iconParams['setSizeManually'] = true;
this.icon = new IconGrid.BaseIcon(app.get_name(), iconParams); this.icon = new IconGrid.BaseIcon(app.get_name(), iconParams);
@ -1544,6 +1573,7 @@ const AppIcon = new Lang.Class({
this._menu = null; this._menu = null;
this._menuManager = new PopupMenu.PopupMenuManager(this); this._menuManager = new PopupMenu.PopupMenuManager(this);
if (isDraggable) {
this._draggable = DND.makeDraggable(this.actor); this._draggable = DND.makeDraggable(this.actor);
this._draggable.connect('drag-begin', Lang.bind(this, this._draggable.connect('drag-begin', Lang.bind(this,
function () { function () {
@ -1558,6 +1588,7 @@ const AppIcon = new Lang.Class({
function () { function () {
Main.overview.endItemDrag(this); Main.overview.endItemDrag(this);
})); }));
}
this.actor.connect('destroy', Lang.bind(this, this._onDestroy)); this.actor.connect('destroy', Lang.bind(this, this._onDestroy));
@ -1645,6 +1676,8 @@ const AppIcon = new Lang.Class({
popupMenu: function() { popupMenu: function() {
this._removeMenuTimeout(); this._removeMenuTimeout();
this.actor.fake_release(); this.actor.fake_release();
if (this._draggable)
this._draggable.fakeRelease(); this._draggable.fakeRelease();
if (!this._menu) { if (!this._menu) {

View File

@ -535,6 +535,7 @@ const Dash = new Lang.Class({
let appIcon = new AppDisplay.AppIcon(app, let appIcon = new AppDisplay.AppIcon(app,
{ setSizeManually: true, { setSizeManually: true,
showLabel: false }); showLabel: false });
if (appIcon._draggable) {
appIcon._draggable.connect('drag-begin', appIcon._draggable.connect('drag-begin',
Lang.bind(this, function() { Lang.bind(this, function() {
appIcon.actor.opacity = 50; appIcon.actor.opacity = 50;
@ -543,6 +544,8 @@ const Dash = new Lang.Class({
Lang.bind(this, function() { Lang.bind(this, function() {
appIcon.actor.opacity = 255; appIcon.actor.opacity = 255;
})); }));
}
appIcon.connect('menu-state-changed', appIcon.connect('menu-state-changed',
Lang.bind(this, function(appIcon, opened) { Lang.bind(this, function(appIcon, opened) {
this._itemMenuStateChanged(item, opened); this._itemMenuStateChanged(item, opened);