dnd: Make the draggable much faster
It turns out that picking a 3200x1200 scene on notebook chipsets every time the mouse is moved isn't exactly the fastest thing. Defer picking to an idle to ensure that it won't get in the way of keeping up with mouse events. https://bugzilla.gnome.org/show_bug.cgi?id=703443
This commit is contained in:
parent
67f10ea7eb
commit
317b9a9c87
55
js/ui/dnd.js
55
js/ui/dnd.js
@ -1,6 +1,7 @@
|
|||||||
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
||||||
|
|
||||||
const Clutter = imports.gi.Clutter;
|
const Clutter = imports.gi.Clutter;
|
||||||
|
const GLib = imports.gi.GLib;
|
||||||
const Gtk = imports.gi.Gtk;
|
const Gtk = imports.gi.Gtk;
|
||||||
const St = imports.gi.St;
|
const St = imports.gi.St;
|
||||||
const Lang = imports.lang;
|
const Lang = imports.lang;
|
||||||
@ -358,25 +359,12 @@ const _Draggable = new Lang.Class({
|
|||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
|
|
||||||
_updateDragPosition : function (event) {
|
_updateDragHover : function () {
|
||||||
let [stageX, stageY] = event.get_coords();
|
|
||||||
this._dragX = stageX;
|
|
||||||
this._dragY = stageY;
|
|
||||||
|
|
||||||
// If we are dragging, update the position
|
|
||||||
if (this._dragActor) {
|
|
||||||
this._dragActor.set_position(stageX + this._dragOffsetX,
|
|
||||||
stageY + this._dragOffsetY);
|
|
||||||
|
|
||||||
let target = this._dragActor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL,
|
let target = this._dragActor.get_stage().get_actor_at_pos(Clutter.PickMode.ALL,
|
||||||
stageX, stageY);
|
this._dragX, this._dragY);
|
||||||
|
|
||||||
// We call observers only once per motion with the innermost
|
|
||||||
// target actor. If necessary, the observer can walk the
|
|
||||||
// parent itself.
|
|
||||||
let dragEvent = {
|
let dragEvent = {
|
||||||
x: stageX,
|
x: this._dragX,
|
||||||
y: stageY,
|
y: this._dragY,
|
||||||
dragActor: this._dragActor,
|
dragActor: this._dragActor,
|
||||||
source: this.actor._delegate,
|
source: this.actor._delegate,
|
||||||
targetActor: target
|
targetActor: target
|
||||||
@ -387,13 +375,14 @@ const _Draggable = new Lang.Class({
|
|||||||
let result = motionFunc(dragEvent);
|
let result = motionFunc(dragEvent);
|
||||||
if (result != DragMotionResult.CONTINUE) {
|
if (result != DragMotionResult.CONTINUE) {
|
||||||
global.set_cursor(DRAG_CURSOR_MAP[result]);
|
global.set_cursor(DRAG_CURSOR_MAP[result]);
|
||||||
return true;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
while (target) {
|
while (target) {
|
||||||
if (target._delegate && target._delegate.handleDragOver) {
|
if (target._delegate && target._delegate.handleDragOver) {
|
||||||
let [r, targX, targY] = target.transform_stage_point(stageX, stageY);
|
let [r, targX, targY] = target.transform_stage_point(this._dragX, this._dragY);
|
||||||
// We currently loop through all parents on drag-over even if one of the children has handled it.
|
// We currently loop through all parents on drag-over even if one of the children has handled it.
|
||||||
// We can check the return value of the function and break the loop if it's true if we don't want
|
// We can check the return value of the function and break the loop if it's true if we don't want
|
||||||
// to continue checking the parents.
|
// to continue checking the parents.
|
||||||
@ -401,17 +390,34 @@ const _Draggable = new Lang.Class({
|
|||||||
this._dragActor,
|
this._dragActor,
|
||||||
targX,
|
targX,
|
||||||
targY,
|
targY,
|
||||||
event.get_time());
|
0);
|
||||||
if (result != DragMotionResult.CONTINUE) {
|
if (result != DragMotionResult.CONTINUE) {
|
||||||
global.set_cursor(DRAG_CURSOR_MAP[result]);
|
global.set_cursor(DRAG_CURSOR_MAP[result]);
|
||||||
return true;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
target = target.get_parent();
|
target = target.get_parent();
|
||||||
}
|
}
|
||||||
global.set_cursor(Shell.Cursor.DND_IN_DRAG);
|
global.set_cursor(Shell.Cursor.DND_IN_DRAG);
|
||||||
}
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
_queueUpdateDragHover: function() {
|
||||||
|
if (this._updateHoverId)
|
||||||
|
GLib.source_remove(this._updateHoverId);
|
||||||
|
|
||||||
|
this._updateHoverId = GLib.idle_add(GLib.PRIORITY_DEFAULT,
|
||||||
|
Lang.bind(this, this._updateDragHover));
|
||||||
|
},
|
||||||
|
|
||||||
|
_updateDragPosition : function (event) {
|
||||||
|
let [stageX, stageY] = event.get_coords();
|
||||||
|
this._dragX = stageX;
|
||||||
|
this._dragY = stageY;
|
||||||
|
this._dragActor.set_position(stageX + this._dragOffsetX,
|
||||||
|
stageY + this._dragOffsetY);
|
||||||
|
|
||||||
|
this._queueUpdateDragHover();
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -511,6 +517,11 @@ const _Draggable = new Lang.Class({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_cancelDrag: function(eventTime) {
|
_cancelDrag: function(eventTime) {
|
||||||
|
if (this._updateHoverId) {
|
||||||
|
GLib.source_remove(this._updateHoverId);
|
||||||
|
this._updateHoverId = 0;
|
||||||
|
}
|
||||||
|
|
||||||
this.emit('drag-cancelled', eventTime);
|
this.emit('drag-cancelled', eventTime);
|
||||||
this._dragInProgress = false;
|
this._dragInProgress = false;
|
||||||
let [snapBackX, snapBackY, snapBackScale] = this._getRestoreLocation();
|
let [snapBackX, snapBackY, snapBackScale] = this._getRestoreLocation();
|
||||||
|
Loading…
Reference in New Issue
Block a user