[gnome-shell/wip/exalm/gestures: 5/15] workspacesView: Use SwipeTracker
- From: Alexander Mikhaylenko <alexm src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-shell/wip/exalm/gestures: 5/15] workspacesView: Use SwipeTracker
- Date: Fri, 25 Oct 2019 20:23:07 +0000 (UTC)
commit 7001f22b1d1cb23bf0a5ed8faba9ec041d12300d
Author: Alexander Mikhaylenko <alexm gnome org>
Date: Mon Jul 8 13:47:04 2019 +0500
workspacesView: Use SwipeTracker
Replace existing panning, touchpad scrolling and four-finger gesture by
SwipeTracker.
Change programmatic workspace animation to use easeOutCubic interpolator
to match the gesture.
Also change the dragging distance to always match the current monitor.
Fixes touchpad parts of https://gitlab.gnome.org/GNOME/gnome-shell/issues/1338
https://gitlab.gnome.org/GNOME/gnome-shell/merge_requests/605
js/ui/workspacesView.js | 194 ++++++++++++++++++++++--------------------------
1 file changed, 89 insertions(+), 105 deletions(-)
---
diff --git a/js/ui/workspacesView.js b/js/ui/workspacesView.js
index decfc50c46..8122e42590 100644
--- a/js/ui/workspacesView.js
+++ b/js/ui/workspacesView.js
@@ -4,7 +4,7 @@
const { Clutter, Gio, GObject, Meta, Shell, St } = imports.gi;
const Main = imports.ui.main;
-const WindowManager = imports.ui.windowManager;
+const SwipeTracker = imports.ui.swipeTracker;
const Workspace = imports.ui.workspace;
var WORKSPACE_SWITCH_TIME = 250;
@@ -79,7 +79,6 @@ class WorkspacesView extends WorkspacesViewBase {
super._init(monitorIndex);
this._animating = false; // tweening
- this._scrolling = false; // swipe-scrolling
this._gestureActive = false; // touch(pad) gestures
this._workspaces = [];
@@ -144,7 +143,7 @@ class WorkspacesView extends WorkspacesViewBase {
}
_scrollToActive() {
- this.updateWorkspaceActors(true);
+ this.updateWorkspaceActors(!this._gestureActive);
}
// Update workspace actors parameters
@@ -171,7 +170,7 @@ class WorkspacesView extends WorkspacesViewBase {
if (showAnimation) {
let easeParams = Object.assign(params, {
duration: WORKSPACE_SWITCH_TIME,
- mode: Clutter.AnimationMode.EASE_OUT_QUAD
+ mode: Clutter.AnimationMode.EASE_OUT_CUBIC
});
// we have to call _updateVisibility() once before the
// animation and once afterwards - it does not really
@@ -198,7 +197,7 @@ class WorkspacesView extends WorkspacesViewBase {
for (let w = 0; w < this._workspaces.length; w++) {
let workspace = this._workspaces[w];
- if (this._animating || this._scrolling || this._gestureActive) {
+ if (this._animating || this._gestureActive) {
workspace.show();
} else {
if (this._inDrag)
@@ -247,17 +246,6 @@ class WorkspacesView extends WorkspacesViewBase {
workspaceManager.disconnect(this._updateWorkspacesId);
}
- startSwipeScroll() {
- this._scrolling = true;
- }
-
- endSwipeScroll() {
- this._scrolling = false;
-
- // Make sure title captions etc are shown as necessary
- this._updateVisibility();
- }
-
startTouchGesture() {
this._gestureActive = true;
}
@@ -379,12 +367,6 @@ class ExtraWorkspaceView extends WorkspacesViewBase {
updateWorkspaceActors(_showAnimation) {
}
- startSwipeScroll() {
- }
-
- endSwipeScroll() {
- }
-
startTouchGesture() {
}
@@ -405,7 +387,6 @@ class WorkspacesDisplay extends St.Widget {
this.connect('notify::allocation', this._updateWorkspacesActualGeometry.bind(this));
let workspaceManager = global.workspace_manager;
- let activeWorkspaceIndex = workspaceManager.get_active_workspace_index();
this._scrollAdjustment = adjustment;
this._scrollAdjustment.connect('notify::value',
@@ -430,42 +411,17 @@ class WorkspacesDisplay extends St.Widget {
});
Main.overview.addAction(clickAction);
this.bind_property('mapped', clickAction, 'enabled', GObject.BindingFlags.SYNC_CREATE);
-
- let panAction = new Clutter.PanAction({ threshold_trigger_edge: Clutter.GestureTriggerEdge.AFTER });
- panAction.connect('pan', this._onPan.bind(this));
- panAction.connect('gesture-begin', () => {
- if (this._workspacesOnlyOnPrimary) {
- let event = Clutter.get_current_event();
- if (this._getMonitorIndexForEvent(event) != this._primaryIndex)
- return false;
- }
-
- this._startSwipeScroll();
- return true;
- });
- panAction.connect('gesture-cancel', () => {
- clickAction.release();
- this._endSwipeScroll();
- });
- panAction.connect('gesture-end', () => {
- clickAction.release();
- this._endSwipeScroll();
- });
- Main.overview.addAction(panAction);
- this.bind_property('mapped', panAction, 'enabled', GObject.BindingFlags.SYNC_CREATE);
+ this._clickAction = clickAction;
let allowedModes = Shell.ActionMode.OVERVIEW;
- let switchGesture = new WindowManager.WorkspaceSwitchAction(allowedModes);
- switchGesture.connect('motion', this._onSwitchWorkspaceMotion.bind(this));
- switchGesture.connect('activated', this._onSwitchWorkspaceActivated.bind(this));
- switchGesture.connect('cancel', this._endTouchGesture.bind(this));
- Main.overview.addAction(switchGesture);
- this.bind_property('mapped', switchGesture, 'enabled', GObject.BindingFlags.SYNC_CREATE);
-
- switchGesture = new WindowManager.TouchpadWorkspaceSwitchAction(global.stage, allowedModes);
- switchGesture.connect('motion', this._onSwitchWorkspaceMotion.bind(this));
- switchGesture.connect('activated', this._onSwitchWorkspaceActivated.bind(this));
- switchGesture.connect('cancel', this._endTouchGesture.bind(this));
+ this._swipeTracker = new SwipeTracker.SwipeTracker(Main.layoutManager.overviewGroup, allowedModes);
+ this._swipeTracker.connect('begin', this._switchWorkspaceBegin.bind(this));
+ this._swipeTracker.connect('update', this._switchWorkspaceUpdate.bind(this));
+ this._swipeTracker.connect('end', this._switchWorkspaceEnd.bind(this));
+ this._swipeTracker.enabled = this.mapped;
+ this.connect('notify::mapped', () => {
+ this._swipeTracker.enabled = this.mapped;
+ });
this._primaryIndex = Main.layoutManager.primaryIndex;
this._workspacesViews = [];
@@ -482,7 +438,6 @@ class WorkspacesDisplay extends St.Widget {
this._fullGeometry = null;
- this._scrolling = false; // swipe-scrolling
this._gestureActive = false; // touch(pad) gestures
this._animatingScroll = false; // programmatically updating the adjustment
@@ -513,7 +468,7 @@ class WorkspacesDisplay extends St.Widget {
}
_activeWorkspaceChanged(_wm, _from, _to, _direction) {
- if (this._scrolling)
+ if (this._gestureActive)
return;
this._scrollToActive();
@@ -530,80 +485,108 @@ class WorkspacesDisplay extends St.Widget {
}
_updateScrollAdjustment(index) {
- if (this._scrolling || this._gestureActive)
+ if (this._gestureActive)
return;
this._animatingScroll = true;
this._scrollAdjustment.ease(index, {
- mode: Clutter.AnimationMode.EASE_OUT_QUAD,
+ mode: Clutter.AnimationMode.EASE_OUT_CUBIC,
duration: WORKSPACE_SWITCH_TIME,
onComplete: () => (this._animatingScroll = false)
});
}
- _onPan(action) {
- let [dist_, dx, dy] = action.get_motion_delta(0);
- let adjustment = this._scrollAdjustment;
+ _directionForProgress(progress) {
if (global.workspace_manager.layout_rows == -1)
- adjustment.value -= (dy / this.height) * adjustment.page_size;
+ return (progress > 0) ? Meta.MotionDirection.DOWN : Meta.MotionDirection.UP;
else if (this.text_direction == Clutter.TextDirection.RTL)
- adjustment.value += (dx / this.width) * adjustment.page_size;
+ return (progress > 0) ? Meta.MotionDirection.LEFT : Meta.MotionDirection.RIGHT;
else
- adjustment.value -= (dx / this.width) * adjustment.page_size;
- return false;
+ return (progress > 0) ? Meta.MotionDirection.RIGHT : Meta.MotionDirection.LEFT;
}
- _startSwipeScroll() {
- for (let i = 0; i < this._workspacesViews.length; i++)
- this._workspacesViews[i].startSwipeScroll();
- this._scrolling = true;
- }
+ _switchWorkspaceBegin(tracker, monitor) {
+ if (this._workspacesOnlyOnPrimary && monitor != this._primaryIndex)
+ return;
- _endSwipeScroll() {
- for (let i = 0; i < this._workspacesViews.length; i++)
- this._workspacesViews[i].endSwipeScroll();
- this._scrolling = false;
- this._scrollToActive();
- }
+ let adjustment = this._scrollAdjustment;
+ if (this._gestureActive)
+ adjustment.remove_transition('value');
+
+ let horiz = (global.workspace_manager.layout_rows != -1);
+ tracker.orientation = horiz ? Clutter.Orientation.HORIZONTAL : Clutter.Orientation.VERTICAL;
- _startTouchGesture() {
for (let i = 0; i < this._workspacesViews.length; i++)
this._workspacesViews[i].startTouchGesture();
- this._gestureActive = true;
- }
- _endTouchGesture() {
- for (let i = 0; i < this._workspacesViews.length; i++)
- this._workspacesViews[i].endTouchGesture();
- this._gestureActive = false;
- this._scrollToActive();
- }
+ let workspaceManager = global.workspace_manager;
+ let activeWs = workspaceManager.get_active_workspace();
+
+ let monitors = Main.layoutManager.monitors;
+ let geometry = (monitor == this._primaryIndex) ? this._fullGeometry : monitors[monitor];
+ let distance = (global.workspace_manager.layout_rows == -1) ? geometry.height : geometry.width;
+
+ let progress = adjustment.value / adjustment.page_size - activeWs.index();
+ let points = [];
+
+ let dir = this._directionForProgress(-1);
+ if (activeWs.get_neighbor(dir) != activeWs)
+ points.push(-1);
+
+ points.push(0);
- _onSwitchWorkspaceMotion(action, xRel, yRel) {
- // We don't have a way to hook into start of touchpad actions,
- // luckily this is safe to call repeatedly.
- this._startTouchGesture();
+ dir = this._directionForProgress(1);
+ if (activeWs.get_neighbor(dir) != activeWs)
+ points.push(1);
+ tracker.confirmSwipe(distance, points, progress, 0);
+
+ this._gestureActive = true;
+ }
+
+ _switchWorkspaceUpdate(_tracker, progress) {
let workspaceManager = global.workspace_manager;
let active = workspaceManager.get_active_workspace_index();
let adjustment = this._scrollAdjustment;
- if (workspaceManager.layout_rows == -1)
- adjustment.value = (active - yRel / this.height) * adjustment.page_size;
- else if (this.text_direction == Clutter.TextDirection.RTL)
- adjustment.value = (active + xRel / this.width) * adjustment.page_size;
- else
- adjustment.value = (active - xRel / this.width) * adjustment.page_size;
+ adjustment.value = (active + progress) * adjustment.page_size;
}
- _onSwitchWorkspaceActivated(action, direction) {
+ _switchWorkspaceEnd(_tracker, duration, endProgress) {
+ this._clickAction.release();
+
let workspaceManager = global.workspace_manager;
let activeWorkspace = workspaceManager.get_active_workspace();
- let newWs = activeWorkspace.get_neighbor(direction);
- if (newWs != activeWorkspace)
- newWs.activate(global.get_current_time());
+ let newWs = activeWorkspace;
+ if (endProgress != 0) {
+ let direction = this._directionForProgress(endProgress);
+ newWs = activeWorkspace.get_neighbor(direction);
+ }
- this._endTouchGesture();
+ if (duration == 0) {
+ if (newWs != activeWorkspace)
+ newWs.activate(global.get_current_time());
+
+ this._endTouchGesture();
+ return;
+ }
+
+ let active = workspaceManager.get_active_workspace_index();
+ this._scrollAdjustment.ease(active + endProgress, {
+ mode: Clutter.AnimationMode.EASE_OUT_CUBIC,
+ duration: duration,
+ onComplete: () => {
+ if (newWs != activeWorkspace)
+ newWs.activate(global.get_current_time());
+ this._endTouchGesture();
+ }
+ });
+ }
+
+ _endTouchGesture() {
+ for (let i = 0; i < this._workspacesViews.length; i++)
+ this._workspacesViews[i].endTouchGesture();
+ this._gestureActive = false;
}
vfunc_navigate_focus(from, direction) {
@@ -683,8 +666,6 @@ class WorkspacesDisplay extends St.Widget {
else
view = new WorkspacesView(i);
- view.connect('scroll-event', this._onScrollEvent.bind(this));
-
// HACK: Avoid spurious allocation changes while updating views
view.hide();
@@ -792,6 +773,9 @@ class WorkspacesDisplay extends St.Widget {
}
_onScrollEvent(actor, event) {
+ if (this._swipeTracker.canHandleScrollEvent(event))
+ return Clutter.EVENT_PROPAGATE;
+
if (!this.mapped)
return Clutter.EVENT_PROPAGATE;
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]