baseViewIcon: Introduce base class for view icons
Right now, only AppIcon supports being dragged. In the future, however, both app and folder icons will be reorderable, and to avoid copying the same code between FolderIcon and AppIcon, add a new base class BaseViewIcon that contains the shared code between them. Adding this new base class also has the side effect that it already allows for folder icons to be dragged, although full support for that will come in next commits. Because the Dash icons are not drop targets themselves, add a tiny DashIcon class, which is an AppDisplay.AppIcon subclass, and disable all DND drop code from it. https://gitlab.gnome.org/GNOME/gnome-shell/merge_requests/603
This commit is contained in:
parent
0596848c27
commit
55eb949def
@ -886,7 +886,8 @@ var AllView = class AllView extends BaseAppView {
|
|||||||
if (index == -1)
|
if (index == -1)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (source.view instanceof FolderView) {
|
if ((source instanceof AppIcon) &&
|
||||||
|
(source.view instanceof FolderView)) {
|
||||||
source.view.removeApp(source.app);
|
source.view.removeApp(source.app);
|
||||||
source = this._items[source.id];
|
source = this._items[source.id];
|
||||||
|
|
||||||
@ -1462,37 +1463,243 @@ var FolderView = class FolderView extends BaseAppView {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var FolderIcon = class FolderIcon {
|
var BaseViewIcon = class BaseViewIcon {
|
||||||
|
constructor(params, buttonParams) {
|
||||||
|
buttonParams = Params.parse(buttonParams, {
|
||||||
|
pivot_point: new Clutter.Point({x: 0.5, y: 0.5}),
|
||||||
|
reactive: true,
|
||||||
|
can_focus: true,
|
||||||
|
x_fill: true,
|
||||||
|
y_fill: true
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
this.actor = new St.Button(buttonParams);
|
||||||
|
this.actor._delegate = this;
|
||||||
|
|
||||||
|
// Get the isDraggable property without passing it on to the BaseIcon:
|
||||||
|
params = Params.parse(params, {
|
||||||
|
isDraggable: true,
|
||||||
|
hideWhileDragging: false
|
||||||
|
}, true);
|
||||||
|
let isDraggable = params['isDraggable'];
|
||||||
|
delete params['isDraggable'];
|
||||||
|
|
||||||
|
this._hasDndHover = false;
|
||||||
|
|
||||||
|
if (isDraggable) {
|
||||||
|
this._draggable = DND.makeDraggable(this.actor);
|
||||||
|
this._draggable.connect('drag-begin', () => {
|
||||||
|
this._dragging = true;
|
||||||
|
this.scaleAndFade();
|
||||||
|
Main.overview.beginItemDrag(this);
|
||||||
|
});
|
||||||
|
this._draggable.connect('drag-cancelled', () => {
|
||||||
|
this._dragging = false;
|
||||||
|
Main.overview.cancelledItemDrag(this);
|
||||||
|
});
|
||||||
|
this._draggable.connect('drag-end', () => {
|
||||||
|
this._dragging = false;
|
||||||
|
this.undoScaleAndFade();
|
||||||
|
Main.overview.endItemDrag(this);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
Main.overview.connect('item-drag-begin', this._onDragBegin.bind(this));
|
||||||
|
Main.overview.connect('item-drag-end', this._onDragEnd.bind(this));
|
||||||
|
|
||||||
|
this.actor.connect('destroy', this._onDestroy.bind(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
_onDestroy() {
|
||||||
|
if (this._draggable && this._dragging) {
|
||||||
|
Main.overview.endItemDrag(this);
|
||||||
|
this.draggable = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_createIcon(iconSize) {
|
||||||
|
throw new GObject.NotImplementedError(`_createIcon in ${this.constructor.name}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
_canDropAt(source) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Should be overriden by subclasses
|
||||||
|
_setHoveringByDnd(isHovering) {
|
||||||
|
if (isHovering)
|
||||||
|
this.actor.add_style_pseudo_class('drop');
|
||||||
|
else
|
||||||
|
this.actor.remove_style_pseudo_class('drop');
|
||||||
|
}
|
||||||
|
|
||||||
|
_onDragBegin() {
|
||||||
|
this._dragMonitor = {
|
||||||
|
dragMotion: this._onDragMotion.bind(this),
|
||||||
|
};
|
||||||
|
DND.addDragMonitor(this._dragMonitor);
|
||||||
|
}
|
||||||
|
|
||||||
|
_onDragMotion(dragEvent) {
|
||||||
|
let target = dragEvent.targetActor;
|
||||||
|
let hoveringActor = target == this.actor || this.actor.contains(target);
|
||||||
|
let canDrop = this._canDropAt(dragEvent.source);
|
||||||
|
let hasDndHover = hoveringActor && canDrop;
|
||||||
|
|
||||||
|
if (this._hasDndHover != hasDndHover) {
|
||||||
|
this._setHoveringByDnd(hasDndHover);
|
||||||
|
this._hasDndHover = hasDndHover;
|
||||||
|
}
|
||||||
|
|
||||||
|
return DND.DragMotionResult.CONTINUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
_onDragEnd() {
|
||||||
|
this.actor.remove_style_pseudo_class('drop');
|
||||||
|
DND.removeDragMonitor(this._dragMonitor);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleDragOver(source, actor, x, y, time) {
|
||||||
|
if (source == this)
|
||||||
|
return DND.DragMotionResult.NO_DROP;
|
||||||
|
|
||||||
|
if (!this._canDropAt(source))
|
||||||
|
return DND.DragMotionResult.CONTINUE;
|
||||||
|
|
||||||
|
return DND.DragMotionResult.MOVE_DROP;
|
||||||
|
}
|
||||||
|
|
||||||
|
acceptDrop(source, actor, x, y, time) {
|
||||||
|
source.undoScaleAndFade();
|
||||||
|
|
||||||
|
this._setHoveringByDnd(false);
|
||||||
|
|
||||||
|
if (!this._canDropAt(source))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDragActor() {
|
||||||
|
let iconParams = {
|
||||||
|
createIcon: this._createIcon.bind(this),
|
||||||
|
showLabel: (this._icon.label != null),
|
||||||
|
setSizeManually: true
|
||||||
|
};
|
||||||
|
|
||||||
|
let icon = new IconGrid.BaseIcon(this.name, iconParams);
|
||||||
|
icon.setIconSize(this.icon.iconSize);
|
||||||
|
|
||||||
|
let bin = new St.Bin({ style_class: this.actor.style_class });
|
||||||
|
bin.set_child(icon);
|
||||||
|
|
||||||
|
return bin;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDragActorSource() {
|
||||||
|
return this._icon.icon;
|
||||||
|
}
|
||||||
|
|
||||||
|
_scaleIn() {
|
||||||
|
this.actor.scale_x = 0;
|
||||||
|
this.actor.scale_y = 0;
|
||||||
|
this.actor.pivot_point = new Clutter.Point({ x: 0.5, y: 0.5 });
|
||||||
|
|
||||||
|
Tweener.addTween(this.actor, {
|
||||||
|
scale_x: 1,
|
||||||
|
scale_y: 1,
|
||||||
|
time: APP_ICON_SCALE_IN_TIME,
|
||||||
|
delay: APP_ICON_SCALE_IN_DELAY,
|
||||||
|
transition: (t, b, c, d) => {
|
||||||
|
// Similar to easeOutElastic, but less aggressive.
|
||||||
|
t /= d;
|
||||||
|
let p = 0.5;
|
||||||
|
return b + c * (Math.pow(2, -11 * t) * Math.sin(2 * Math.PI * (t - p / 4) / p) + 1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_unscheduleScaleIn() {
|
||||||
|
if (this._scaleInId != 0) {
|
||||||
|
this.actor.disconnect(this._scaleInId);
|
||||||
|
this._scaleInId = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scheduleScaleIn() {
|
||||||
|
if (this._scaleInId != 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (this.actor.mapped) {
|
||||||
|
this._scaleIn();
|
||||||
|
} else {
|
||||||
|
this._scaleInId = this.actor.connect('notify::mapped', () => {
|
||||||
|
this._unscheduleScaleIn();
|
||||||
|
this._scaleIn();
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scaleAndFade() {
|
||||||
|
this.actor.save_easing_state();
|
||||||
|
this.actor.reactive = false;
|
||||||
|
this.actor.scale_x = 0.75;
|
||||||
|
this.actor.scale_y = 0.75;
|
||||||
|
this.actor.opacity = 128;
|
||||||
|
this.actor.restore_easing_state();
|
||||||
|
}
|
||||||
|
|
||||||
|
undoScaleAndFade() {
|
||||||
|
this.actor.save_easing_state();
|
||||||
|
this.actor.reactive = true;
|
||||||
|
this.actor.scale_x = 1.0;
|
||||||
|
this.actor.scale_y = 1.0;
|
||||||
|
this.actor.opacity = 255;
|
||||||
|
this.actor.restore_easing_state();
|
||||||
|
}
|
||||||
|
|
||||||
|
get icon() {
|
||||||
|
return this._icon;
|
||||||
|
}
|
||||||
|
|
||||||
|
get id() {
|
||||||
|
return this._id;
|
||||||
|
}
|
||||||
|
|
||||||
|
get name() {
|
||||||
|
return this._name;
|
||||||
|
}
|
||||||
|
|
||||||
|
get view() {
|
||||||
|
return this._view;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var FolderIcon = class FolderIcon extends BaseViewIcon {
|
||||||
constructor(id, path, parentView) {
|
constructor(id, path, parentView) {
|
||||||
this.id = id;
|
super({ hideWhileDragging: true }, {
|
||||||
this.name = '';
|
style_class: 'app-well-app app-folder',
|
||||||
this._parentView = parentView;
|
toggle_mode: true
|
||||||
|
});
|
||||||
|
|
||||||
|
this._id = id;
|
||||||
|
this._name = '';
|
||||||
|
this._view = parentView;
|
||||||
|
|
||||||
this._folder = new Gio.Settings({ schema_id: 'org.gnome.desktop.app-folders.folder',
|
this._folder = new Gio.Settings({ schema_id: 'org.gnome.desktop.app-folders.folder',
|
||||||
path: path });
|
path: path });
|
||||||
this.actor = new St.Button({ style_class: 'app-well-app app-folder',
|
|
||||||
button_mask: St.ButtonMask.ONE,
|
|
||||||
toggle_mode: true,
|
|
||||||
can_focus: true,
|
|
||||||
x_fill: true,
|
|
||||||
y_fill: true });
|
|
||||||
this.actor._delegate = this;
|
|
||||||
// whether we need to update arrow side, position etc.
|
// whether we need to update arrow side, position etc.
|
||||||
this._popupInvalidated = false;
|
this._popupInvalidated = false;
|
||||||
|
|
||||||
this.icon = new IconGrid.BaseIcon('', {
|
this._icon = new IconGrid.BaseIcon('', {
|
||||||
createIcon: this._createIcon.bind(this),
|
createIcon: this._createIcon.bind(this),
|
||||||
setSizeManually: true
|
setSizeManually: true
|
||||||
});
|
});
|
||||||
this.actor.set_child(this.icon);
|
this.actor.set_child(this.icon);
|
||||||
this.actor.label_actor = this.icon.label;
|
this.actor.label_actor = this.icon.label;
|
||||||
|
|
||||||
this.view = new FolderView(this._folder, parentView);
|
this._folderView = new FolderView(this._folder, parentView);
|
||||||
|
|
||||||
Main.overview.connect('item-drag-begin',
|
|
||||||
this._onDragBegin.bind(this));
|
|
||||||
Main.overview.connect('item-drag-end',
|
|
||||||
this._onDragEnd.bind(this));
|
|
||||||
|
|
||||||
this.actor.connect('clicked', this.open.bind(this));
|
this.actor.connect('clicked', this.open.bind(this));
|
||||||
this.actor.connect('destroy', this.onDestroy.bind(this));
|
this.actor.connect('destroy', this.onDestroy.bind(this));
|
||||||
@ -1506,10 +1713,10 @@ var FolderIcon = class FolderIcon {
|
|||||||
}
|
}
|
||||||
|
|
||||||
onDestroy() {
|
onDestroy() {
|
||||||
this.view.actor.destroy();
|
this._folderView.actor.destroy();
|
||||||
|
|
||||||
if (this._spaceReadySignalId) {
|
if (this._spaceReadySignalId) {
|
||||||
this._parentView.disconnect(this._spaceReadySignalId);
|
this.view.disconnect(this._spaceReadySignalId);
|
||||||
this._spaceReadySignalId = 0;
|
this._spaceReadySignalId = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1519,38 +1726,22 @@ var FolderIcon = class FolderIcon {
|
|||||||
|
|
||||||
open() {
|
open() {
|
||||||
this._ensurePopup();
|
this._ensurePopup();
|
||||||
this.view.actor.vscroll.adjustment.value = 0;
|
this._folderView.actor.vscroll.adjustment.value = 0;
|
||||||
this._openSpaceForPopup();
|
this._openSpaceForPopup();
|
||||||
}
|
}
|
||||||
|
|
||||||
getAppIds() {
|
getAppIds() {
|
||||||
return this.view.getAllItems().map(item => item.id);
|
return this._folderView.getAllItems().map(item => item.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
_onDragBegin() {
|
_onDragBegin() {
|
||||||
this._dragMonitor = {
|
super._onDragBegin();
|
||||||
dragMotion: this._onDragMotion.bind(this),
|
this.view.inhibitEventBlocker();
|
||||||
};
|
|
||||||
DND.addDragMonitor(this._dragMonitor);
|
|
||||||
|
|
||||||
this._parentView.inhibitEventBlocker();
|
|
||||||
}
|
|
||||||
|
|
||||||
_onDragMotion(dragEvent) {
|
|
||||||
let target = dragEvent.targetActor;
|
|
||||||
|
|
||||||
if (!this.actor.contains(target) || !this._canDropAt(dragEvent.source))
|
|
||||||
this.actor.remove_style_pseudo_class('drop');
|
|
||||||
else
|
|
||||||
this.actor.add_style_pseudo_class('drop');
|
|
||||||
|
|
||||||
return DND.DragMotionResult.CONTINUE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_onDragEnd() {
|
_onDragEnd() {
|
||||||
this.actor.remove_style_pseudo_class('drop');
|
super._onDragEnd();
|
||||||
this._parentView.uninhibitEventBlocker();
|
this.view.uninhibitEventBlocker();
|
||||||
DND.removeDragMonitor(this._dragMonitor);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_canDropAt(source) {
|
_canDropAt(source) {
|
||||||
@ -1590,57 +1781,57 @@ var FolderIcon = class FolderIcon {
|
|||||||
|
|
||||||
_updateName() {
|
_updateName() {
|
||||||
let name = _getFolderName(this._folder);
|
let name = _getFolderName(this._folder);
|
||||||
if (this.name == name)
|
if (this._name == name)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
this.name = name;
|
this._name = name;
|
||||||
this.icon.label.text = this.name;
|
this.icon.label.text = name;
|
||||||
this.emit('name-changed');
|
this.emit('name-changed');
|
||||||
}
|
}
|
||||||
|
|
||||||
_redisplay() {
|
_redisplay() {
|
||||||
this._updateName();
|
this._updateName();
|
||||||
this.actor.visible = this.view.getAllItems().length > 0;
|
this.actor.visible = this._folderView.getAllItems().length > 0;
|
||||||
this.icon.update();
|
this.icon.update();
|
||||||
this.emit('apps-changed');
|
this.emit('apps-changed');
|
||||||
}
|
}
|
||||||
|
|
||||||
_createIcon(iconSize) {
|
_createIcon(iconSize) {
|
||||||
return this.view.createFolderIcon(iconSize, this);
|
return this._folderView.createFolderIcon(iconSize, this);
|
||||||
}
|
}
|
||||||
|
|
||||||
_popupHeight() {
|
_popupHeight() {
|
||||||
let usedHeight = this.view.usedHeight() + this._popup.getOffset(St.Side.TOP) + this._popup.getOffset(St.Side.BOTTOM);
|
let usedHeight = this._folderView.usedHeight() + this._popup.getOffset(St.Side.TOP) + this._popup.getOffset(St.Side.BOTTOM);
|
||||||
return usedHeight;
|
return usedHeight;
|
||||||
}
|
}
|
||||||
|
|
||||||
_openSpaceForPopup() {
|
_openSpaceForPopup() {
|
||||||
this._spaceReadySignalId = this._parentView.connect('space-ready', () => {
|
this._spaceReadySignalId = this.view.connect('space-ready', () => {
|
||||||
this._parentView.disconnect(this._spaceReadySignalId);
|
this.view.disconnect(this._spaceReadySignalId);
|
||||||
this._spaceReadySignalId = 0;
|
this._spaceReadySignalId = 0;
|
||||||
this._popup.popup();
|
this._popup.popup();
|
||||||
this._updatePopupPosition();
|
this._updatePopupPosition();
|
||||||
});
|
});
|
||||||
this._parentView.openSpaceForPopup(this, this._boxPointerArrowside, this.view.nRowsDisplayedAtOnce());
|
this.view.openSpaceForPopup(this, this._boxPointerArrowside, this._folderView.nRowsDisplayedAtOnce());
|
||||||
}
|
}
|
||||||
|
|
||||||
_calculateBoxPointerArrowSide() {
|
_calculateBoxPointerArrowSide() {
|
||||||
let spaceTop = this.actor.y - this._parentView.getCurrentPageY();
|
let spaceTop = this.actor.y - this.view.getCurrentPageY();
|
||||||
let spaceBottom = this._parentView.actor.height - (spaceTop + this.actor.height);
|
let spaceBottom = this.view.actor.height - (spaceTop + this.actor.height);
|
||||||
|
|
||||||
return spaceTop > spaceBottom ? St.Side.BOTTOM : St.Side.TOP;
|
return spaceTop > spaceBottom ? St.Side.BOTTOM : St.Side.TOP;
|
||||||
}
|
}
|
||||||
|
|
||||||
_updatePopupSize() {
|
_updatePopupSize() {
|
||||||
// StWidget delays style calculation until needed, make sure we use the correct values
|
// StWidget delays style calculation until needed, make sure we use the correct values
|
||||||
this.view._grid.ensure_style();
|
this._folderView._grid.ensure_style();
|
||||||
|
|
||||||
let offsetForEachSide = Math.ceil((this._popup.getOffset(St.Side.TOP) +
|
let offsetForEachSide = Math.ceil((this._popup.getOffset(St.Side.TOP) +
|
||||||
this._popup.getOffset(St.Side.BOTTOM) -
|
this._popup.getOffset(St.Side.BOTTOM) -
|
||||||
this._popup.getCloseButtonOverlap()) / 2);
|
this._popup.getCloseButtonOverlap()) / 2);
|
||||||
// Add extra padding to prevent boxpointer decorations and close button being cut off
|
// Add extra padding to prevent boxpointer decorations and close button being cut off
|
||||||
this.view.setPaddingOffsets(offsetForEachSide);
|
this._folderView.setPaddingOffsets(offsetForEachSide);
|
||||||
this.view.adaptToSize(this._parentAvailableWidth, this._parentAvailableHeight);
|
this._folderView.adaptToSize(this._parentAvailableWidth, this._parentAvailableHeight);
|
||||||
}
|
}
|
||||||
|
|
||||||
_updatePopupPosition() {
|
_updatePopupPosition() {
|
||||||
@ -1659,7 +1850,7 @@ var FolderIcon = class FolderIcon {
|
|||||||
this._boxPointerArrowside = this._calculateBoxPointerArrowSide();
|
this._boxPointerArrowside = this._calculateBoxPointerArrowSide();
|
||||||
if (!this._popup) {
|
if (!this._popup) {
|
||||||
this._popup = new AppFolderPopup(this, this._boxPointerArrowside);
|
this._popup = new AppFolderPopup(this, this._boxPointerArrowside);
|
||||||
this._parentView.addFolderPopup(this._popup);
|
this.view.addFolderPopup(this._popup);
|
||||||
this._popup.connect('open-state-changed', (popup, isOpen) => {
|
this._popup.connect('open-state-changed', (popup, isOpen) => {
|
||||||
if (!isOpen)
|
if (!isOpen)
|
||||||
this.actor.checked = false;
|
this.actor.checked = false;
|
||||||
@ -1676,7 +1867,7 @@ var FolderIcon = class FolderIcon {
|
|||||||
this._parentAvailableWidth = width;
|
this._parentAvailableWidth = width;
|
||||||
this._parentAvailableHeight = height;
|
this._parentAvailableHeight = height;
|
||||||
if (this._popup)
|
if (this._popup)
|
||||||
this.view.adaptToSize(width, height);
|
this._folderView.adaptToSize(width, height);
|
||||||
this._popupInvalidated = true;
|
this._popupInvalidated = true;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -1685,7 +1876,7 @@ Signals.addSignalMethods(FolderIcon.prototype);
|
|||||||
var AppFolderPopup = class AppFolderPopup {
|
var AppFolderPopup = class AppFolderPopup {
|
||||||
constructor(source, side) {
|
constructor(source, side) {
|
||||||
this._source = source;
|
this._source = source;
|
||||||
this._view = source.view;
|
this._view = source._folderView;
|
||||||
this._arrowSide = side;
|
this._arrowSide = side;
|
||||||
|
|
||||||
this._isOpen = false;
|
this._isOpen = false;
|
||||||
@ -1847,20 +2038,16 @@ var AppFolderPopup = class AppFolderPopup {
|
|||||||
};
|
};
|
||||||
Signals.addSignalMethods(AppFolderPopup.prototype);
|
Signals.addSignalMethods(AppFolderPopup.prototype);
|
||||||
|
|
||||||
var AppIcon = class AppIcon {
|
var AppIcon = class AppIcon extends BaseViewIcon {
|
||||||
constructor(app, view, iconParams = {}) {
|
constructor(app, parentView, iconParams = {}) {
|
||||||
this.app = app;
|
super(iconParams, {
|
||||||
this.id = app.get_id();
|
|
||||||
this.name = app.get_name();
|
|
||||||
this._view = view;
|
|
||||||
|
|
||||||
this.actor = new St.Button({ style_class: 'app-well-app',
|
|
||||||
pivot_point: new Clutter.Point({x: 0.5, y: 0.5}),
|
|
||||||
reactive: true,
|
|
||||||
button_mask: St.ButtonMask.ONE | St.ButtonMask.TWO,
|
button_mask: St.ButtonMask.ONE | St.ButtonMask.TWO,
|
||||||
can_focus: true,
|
style_class: 'app-well-app'
|
||||||
x_fill: true,
|
});
|
||||||
y_fill: true });
|
this.app = app;
|
||||||
|
this._id = app.get_id();
|
||||||
|
this._name = app.get_name();
|
||||||
|
this._view = parentView;
|
||||||
|
|
||||||
this._dot = new St.Widget({ style_class: 'app-well-app-running-dot',
|
this._dot = new St.Widget({ style_class: 'app-well-app-running-dot',
|
||||||
layout_manager: new Clutter.BinLayout(),
|
layout_manager: new Clutter.BinLayout(),
|
||||||
@ -1877,14 +2064,11 @@ var AppIcon = class AppIcon {
|
|||||||
this.actor._delegate = this;
|
this.actor._delegate = this;
|
||||||
this._scaleInId = 0;
|
this._scaleInId = 0;
|
||||||
|
|
||||||
// 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'];
|
delete iconParams['isDraggable'];
|
||||||
|
|
||||||
iconParams['createIcon'] = this._createIcon.bind(this);
|
iconParams['createIcon'] = this._createIcon.bind(this);
|
||||||
iconParams['setSizeManually'] = true;
|
iconParams['setSizeManually'] = true;
|
||||||
this.icon = new IconGrid.BaseIcon(app.get_name(), iconParams);
|
this._icon = new IconGrid.BaseIcon(app.get_name(), iconParams);
|
||||||
this._iconContainer.add_child(this.icon);
|
this._iconContainer.add_child(this.icon);
|
||||||
|
|
||||||
this.actor.label_actor = this.icon.label;
|
this.actor.label_actor = this.icon.label;
|
||||||
@ -1898,26 +2082,8 @@ var AppIcon = class AppIcon {
|
|||||||
this._menu = null;
|
this._menu = null;
|
||||||
this._menuManager = new PopupMenu.PopupMenuManager(this.actor);
|
this._menuManager = new PopupMenu.PopupMenuManager(this.actor);
|
||||||
|
|
||||||
if (isDraggable) {
|
if (this._draggable)
|
||||||
this._draggable = DND.makeDraggable(this.actor);
|
this._draggable.connect('drag-begin', this._removeMenuTimeout.bind(this));
|
||||||
this._draggable.connect('drag-begin', () => {
|
|
||||||
this._dragging = true;
|
|
||||||
this.scaleAndFade();
|
|
||||||
this._removeMenuTimeout();
|
|
||||||
Main.overview.beginItemDrag(this);
|
|
||||||
});
|
|
||||||
this._draggable.connect('drag-cancelled', () => {
|
|
||||||
this._dragging = false;
|
|
||||||
Main.overview.cancelledItemDrag(this);
|
|
||||||
});
|
|
||||||
this._draggable.connect('drag-end', () => {
|
|
||||||
this._dragging = false;
|
|
||||||
this.undoScaleAndFade();
|
|
||||||
Main.overview.endItemDrag(this);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
this.actor.connect('destroy', this._onDestroy.bind(this));
|
|
||||||
|
|
||||||
this._menuTimeoutId = 0;
|
this._menuTimeoutId = 0;
|
||||||
this._stateChangedId = this.app.connect('notify::state', () => {
|
this._stateChangedId = this.app.connect('notify::state', () => {
|
||||||
@ -1927,12 +2093,10 @@ var AppIcon = class AppIcon {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_onDestroy() {
|
_onDestroy() {
|
||||||
|
super._onDestroy();
|
||||||
|
|
||||||
if (this._stateChangedId > 0)
|
if (this._stateChangedId > 0)
|
||||||
this.app.disconnect(this._stateChangedId);
|
this.app.disconnect(this._stateChangedId);
|
||||||
if (this._draggable && this._dragging) {
|
|
||||||
Main.overview.endItemDrag(this);
|
|
||||||
this.draggable = null;
|
|
||||||
}
|
|
||||||
this._stateChangedId = 0;
|
this._stateChangedId = 0;
|
||||||
this._removeMenuTimeout();
|
this._removeMenuTimeout();
|
||||||
}
|
}
|
||||||
@ -2075,46 +2239,6 @@ var AppIcon = class AppIcon {
|
|||||||
this.icon.animateZoomOut();
|
this.icon.animateZoomOut();
|
||||||
}
|
}
|
||||||
|
|
||||||
_scaleIn() {
|
|
||||||
this.actor.scale_x = 0;
|
|
||||||
this.actor.scale_y = 0;
|
|
||||||
this.actor.pivot_point = new Clutter.Point({ x: 0.5, y: 0.5 });
|
|
||||||
|
|
||||||
Tweener.addTween(this.actor, {
|
|
||||||
scale_x: 1,
|
|
||||||
scale_y: 1,
|
|
||||||
time: APP_ICON_SCALE_IN_TIME,
|
|
||||||
delay: APP_ICON_SCALE_IN_DELAY,
|
|
||||||
transition: (t, b, c, d) => {
|
|
||||||
// Similar to easeOutElastic, but less aggressive.
|
|
||||||
t /= d;
|
|
||||||
let p = 0.5;
|
|
||||||
return b + c * (Math.pow(2, -11 * t) * Math.sin(2 * Math.PI * (t - p / 4) / p) + 1);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_unscheduleScaleIn() {
|
|
||||||
if (this._scaleInId != 0) {
|
|
||||||
this.actor.disconnect(this._scaleInId);
|
|
||||||
this._scaleInId = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
scheduleScaleIn() {
|
|
||||||
if (this._scaleInId != 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (this.actor.mapped) {
|
|
||||||
this._scaleIn();
|
|
||||||
} else {
|
|
||||||
this._scaleInId = this.actor.connect('notify::mapped', () => {
|
|
||||||
this._unscheduleScaleIn();
|
|
||||||
this._scaleIn();
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
shellWorkspaceLaunch(params) {
|
shellWorkspaceLaunch(params) {
|
||||||
params = Params.parse(params, { workspace: -1,
|
params = Params.parse(params, { workspace: -1,
|
||||||
timestamp: 0 });
|
timestamp: 0 });
|
||||||
@ -2122,46 +2246,9 @@ var AppIcon = class AppIcon {
|
|||||||
this.app.open_new_window(params.workspace);
|
this.app.open_new_window(params.workspace);
|
||||||
}
|
}
|
||||||
|
|
||||||
getDragActor() {
|
|
||||||
let iconParams = { createIcon: this._createIcon.bind(this),
|
|
||||||
showLabel: (this.icon.label != null),
|
|
||||||
setSizeManually: true };
|
|
||||||
let icon = new IconGrid.BaseIcon(this.name, iconParams);
|
|
||||||
icon.setIconSize(this.icon.iconSize);
|
|
||||||
return icon;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns the original actor that should align with the actor
|
|
||||||
// we show as the item is being dragged.
|
|
||||||
getDragActorSource() {
|
|
||||||
return this.icon.icon;
|
|
||||||
}
|
|
||||||
|
|
||||||
shouldShowTooltip() {
|
shouldShowTooltip() {
|
||||||
return this.actor.hover && (!this._menu || !this._menu.isOpen);
|
return this.actor.hover && (!this._menu || !this._menu.isOpen);
|
||||||
}
|
}
|
||||||
|
|
||||||
scaleAndFade() {
|
|
||||||
this.actor.save_easing_state();
|
|
||||||
this.actor.reactive = false;
|
|
||||||
this.actor.scale_x = 0.75;
|
|
||||||
this.actor.scale_y = 0.75;
|
|
||||||
this.actor.opacity = 128;
|
|
||||||
this.actor.restore_easing_state();
|
|
||||||
}
|
|
||||||
|
|
||||||
undoScaleAndFade() {
|
|
||||||
this.actor.save_easing_state();
|
|
||||||
this.actor.reactive = true;
|
|
||||||
this.actor.scale_x = 1.0;
|
|
||||||
this.actor.scale_y = 1.0;
|
|
||||||
this.actor.opacity = 255;
|
|
||||||
this.actor.restore_easing_state();
|
|
||||||
}
|
|
||||||
|
|
||||||
get view() {
|
|
||||||
return this._view;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
Signals.addSignalMethods(AppIcon.prototype);
|
Signals.addSignalMethods(AppIcon.prototype);
|
||||||
|
|
||||||
|
@ -25,6 +25,32 @@ function getAppFromSource(source) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var DashIcon = class DashIcon extends AppDisplay.AppIcon {
|
||||||
|
constructor(app) {
|
||||||
|
super(app, null, {
|
||||||
|
setSizeManually: true,
|
||||||
|
showLabel: false
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Disable all DnD methods
|
||||||
|
_onDragBegin() {
|
||||||
|
}
|
||||||
|
|
||||||
|
_onDragEnd() {
|
||||||
|
}
|
||||||
|
|
||||||
|
handleDragOver() {
|
||||||
|
return DND.DragMotionResult.CONTINUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
acceptDrop() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// A container like StBin, but taking the child's scale into account
|
// A container like StBin, but taking the child's scale into account
|
||||||
// when requesting a size
|
// when requesting a size
|
||||||
var DashItemContainer = GObject.registerClass(
|
var DashItemContainer = GObject.registerClass(
|
||||||
@ -475,9 +501,7 @@ var Dash = class Dash {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_createAppItem(app) {
|
_createAppItem(app) {
|
||||||
let appIcon = new AppDisplay.AppIcon(app, null,
|
let appIcon = new DashIcon(app);
|
||||||
{ setSizeManually: true,
|
|
||||||
showLabel: false });
|
|
||||||
|
|
||||||
appIcon.connect('menu-state-changed',
|
appIcon.connect('menu-state-changed',
|
||||||
(appIcon, opened) => {
|
(appIcon, opened) => {
|
||||||
|
Loading…
Reference in New Issue
Block a user