2013-02-14 14:41:38 -05:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2019-01-31 15:07:06 +01:00
|
|
|
/* exported ControlsManager */
|
2013-02-14 14:41:38 -05:00
|
|
|
|
2021-04-15 12:00:12 +02:00
|
|
|
const { Clutter, Gio, GLib, GObject, Meta, Shell, St } = imports.gi;
|
2013-02-14 14:41:38 -05:00
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
const AppDisplay = imports.ui.appDisplay;
|
2013-02-25 18:05:45 -05:00
|
|
|
const Dash = imports.ui.dash;
|
2021-02-15 17:39:14 -03:00
|
|
|
const Layout = imports.ui.layout;
|
2021-01-02 17:41:54 -03:00
|
|
|
const Main = imports.ui.main;
|
2020-12-30 17:45:09 -03:00
|
|
|
const Overview = imports.ui.overview;
|
2021-01-15 09:59:58 -03:00
|
|
|
const SearchController = imports.ui.searchController;
|
2021-01-14 20:23:15 -03:00
|
|
|
const Util = imports.misc.util;
|
2021-01-02 17:41:54 -03:00
|
|
|
const WindowManager = imports.ui.windowManager;
|
2021-01-14 20:23:15 -03:00
|
|
|
const WorkspaceThumbnail = imports.ui.workspaceThumbnail;
|
|
|
|
const WorkspacesView = imports.ui.workspacesView;
|
|
|
|
|
|
|
|
const SMALL_WORKSPACE_RATIO = 0.15;
|
2021-01-22 10:36:37 -03:00
|
|
|
const DASH_MAX_HEIGHT_RATIO = 0.15;
|
2013-02-15 18:25:36 -05:00
|
|
|
|
2020-06-28 02:45:18 +02:00
|
|
|
const A11Y_SCHEMA = 'org.gnome.desktop.a11y.keyboard';
|
|
|
|
|
2020-12-30 17:45:09 -03:00
|
|
|
var SIDE_CONTROLS_ANIMATION_TIME = Overview.ANIMATION_TIME;
|
|
|
|
|
2020-12-30 17:32:03 -03:00
|
|
|
var ControlsState = {
|
|
|
|
HIDDEN: 0,
|
|
|
|
WINDOW_PICKER: 1,
|
|
|
|
APP_GRID: 2,
|
|
|
|
};
|
|
|
|
|
2020-12-30 17:16:18 -03:00
|
|
|
var ControlsManagerLayout = GObject.registerClass(
|
|
|
|
class ControlsManagerLayout extends Clutter.BoxLayout {
|
2021-01-14 20:23:15 -03:00
|
|
|
_init(searchEntry, appDisplay, workspacesDisplay, workspacesThumbnails,
|
2021-01-15 09:59:58 -03:00
|
|
|
searchController, dash, stateAdjustment) {
|
2020-12-30 17:16:18 -03:00
|
|
|
super._init({ orientation: Clutter.Orientation.VERTICAL });
|
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
this._appDisplay = appDisplay;
|
|
|
|
this._workspacesDisplay = workspacesDisplay;
|
|
|
|
this._workspacesThumbnails = workspacesThumbnails;
|
2020-12-30 17:32:03 -03:00
|
|
|
this._stateAdjustment = stateAdjustment;
|
2020-12-30 17:16:18 -03:00
|
|
|
this._searchEntry = searchEntry;
|
2021-01-15 09:59:58 -03:00
|
|
|
this._searchController = searchController;
|
2020-12-30 17:16:18 -03:00
|
|
|
this._dash = dash;
|
2020-12-30 17:32:03 -03:00
|
|
|
|
2021-02-10 18:09:04 -03:00
|
|
|
this._cachedWorkspaceBoxes = new Map();
|
2021-02-15 17:39:14 -03:00
|
|
|
this._postAllocationCallbacks = [];
|
2021-02-10 18:09:04 -03:00
|
|
|
|
2020-12-30 17:32:03 -03:00
|
|
|
stateAdjustment.connect('notify::value', () => this.layout_changed());
|
2022-07-30 21:59:33 -07:00
|
|
|
|
|
|
|
this._workAreaBox = new Clutter.ActorBox();
|
|
|
|
global.display.connectObject(
|
|
|
|
'workareas-changed', () => this._updateWorkAreaBox(),
|
2023-02-02 15:52:03 +01:00
|
|
|
this);
|
|
|
|
Main.layoutManager.connectObject(
|
|
|
|
'monitors-changed', () => this._updateWorkAreaBox(),
|
2022-07-30 21:59:33 -07:00
|
|
|
this);
|
|
|
|
this._updateWorkAreaBox();
|
2020-12-30 17:16:18 -03:00
|
|
|
}
|
|
|
|
|
2022-07-30 21:59:33 -07:00
|
|
|
_updateWorkAreaBox() {
|
|
|
|
const monitor = Main.layoutManager.primaryMonitor;
|
2022-09-08 12:31:15 +02:00
|
|
|
if (!monitor)
|
|
|
|
return;
|
|
|
|
|
2022-07-30 21:59:33 -07:00
|
|
|
const workArea = Main.layoutManager.getWorkAreaForMonitor(monitor.index);
|
|
|
|
const startX = workArea.x - monitor.x;
|
|
|
|
const startY = workArea.y - monitor.y;
|
|
|
|
this._workAreaBox.set_origin(startX, startY);
|
|
|
|
this._workAreaBox.set_size(workArea.width, workArea.height);
|
|
|
|
}
|
|
|
|
|
|
|
|
_computeWorkspacesBoxForState(state, box, searchHeight, dashHeight, thumbnailsHeight) {
|
2022-03-05 13:14:19 +01:00
|
|
|
const workspaceBox = box.copy();
|
2021-01-14 20:23:15 -03:00
|
|
|
const [width, height] = workspaceBox.get_size();
|
2022-07-30 21:59:33 -07:00
|
|
|
const {y1: startY} = this._workAreaBox;
|
2022-07-31 17:41:00 -07:00
|
|
|
const {spacing} = this;
|
|
|
|
const {expandFraction} = this._workspacesThumbnails;
|
2021-01-14 20:23:15 -03:00
|
|
|
|
|
|
|
switch (state) {
|
|
|
|
case ControlsState.HIDDEN:
|
2022-07-30 21:59:33 -07:00
|
|
|
workspaceBox.set_origin(...this._workAreaBox.get_origin());
|
|
|
|
workspaceBox.set_size(...this._workAreaBox.get_size());
|
2021-01-14 20:23:15 -03:00
|
|
|
break;
|
|
|
|
case ControlsState.WINDOW_PICKER:
|
2022-03-05 13:14:19 +01:00
|
|
|
workspaceBox.set_origin(0,
|
2021-03-09 17:31:30 +01:00
|
|
|
startY + searchHeight + spacing +
|
2021-02-16 22:03:29 +01:00
|
|
|
thumbnailsHeight + spacing * expandFraction);
|
2021-01-14 20:23:15 -03:00
|
|
|
workspaceBox.set_size(width,
|
|
|
|
height -
|
|
|
|
dashHeight - spacing -
|
|
|
|
searchHeight - spacing -
|
2021-02-16 22:03:29 +01:00
|
|
|
thumbnailsHeight - spacing * expandFraction);
|
2021-01-14 20:23:15 -03:00
|
|
|
break;
|
|
|
|
case ControlsState.APP_GRID:
|
2022-03-05 13:14:19 +01:00
|
|
|
workspaceBox.set_origin(0, startY + searchHeight + spacing);
|
2021-01-14 20:23:15 -03:00
|
|
|
workspaceBox.set_size(
|
|
|
|
width,
|
2021-02-22 13:10:02 +01:00
|
|
|
Math.round(height * SMALL_WORKSPACE_RATIO));
|
2021-01-14 20:23:15 -03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return workspaceBox;
|
|
|
|
}
|
|
|
|
|
2022-07-30 21:59:33 -07:00
|
|
|
_getAppDisplayBoxForState(state, box, searchHeight, dashHeight, appGridBox) {
|
2022-03-05 13:14:19 +01:00
|
|
|
const [width, height] = box.get_size();
|
2022-07-30 21:59:33 -07:00
|
|
|
const {y1: startY} = this._workAreaBox;
|
2021-01-20 19:25:13 -03:00
|
|
|
const appDisplayBox = new Clutter.ActorBox();
|
2022-07-31 17:41:00 -07:00
|
|
|
const {spacing} = this;
|
2021-01-20 19:25:13 -03:00
|
|
|
|
|
|
|
switch (state) {
|
|
|
|
case ControlsState.HIDDEN:
|
|
|
|
case ControlsState.WINDOW_PICKER:
|
2022-03-05 13:14:19 +01:00
|
|
|
appDisplayBox.set_origin(0, box.y2);
|
2021-01-20 19:25:13 -03:00
|
|
|
break;
|
|
|
|
case ControlsState.APP_GRID:
|
2022-03-05 13:14:19 +01:00
|
|
|
appDisplayBox.set_origin(0,
|
2021-03-09 17:31:30 +01:00
|
|
|
startY + searchHeight + spacing + appGridBox.get_height());
|
2021-01-20 19:25:13 -03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
appDisplayBox.set_size(width,
|
|
|
|
height -
|
|
|
|
searchHeight - spacing -
|
|
|
|
appGridBox.get_height() - spacing -
|
|
|
|
dashHeight);
|
|
|
|
|
|
|
|
return appDisplayBox;
|
|
|
|
}
|
|
|
|
|
2021-02-15 17:39:14 -03:00
|
|
|
_runPostAllocation() {
|
|
|
|
if (this._postAllocationCallbacks.length === 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this._postAllocationCallbacks.forEach(cb => cb());
|
|
|
|
this._postAllocationCallbacks = [];
|
|
|
|
}
|
|
|
|
|
2020-12-30 17:16:18 -03:00
|
|
|
vfunc_set_container(container) {
|
|
|
|
this._container = container;
|
2021-07-08 20:42:55 +02:00
|
|
|
if (container)
|
|
|
|
this.hookup_style(container);
|
2020-12-30 17:16:18 -03:00
|
|
|
}
|
|
|
|
|
2021-03-11 15:26:14 +01:00
|
|
|
vfunc_get_preferred_width(_container, _forHeight) {
|
|
|
|
// The MonitorConstraint will allocate us a fixed size anyway
|
|
|
|
return [0, 0];
|
|
|
|
}
|
|
|
|
|
|
|
|
vfunc_get_preferred_height(_container, _forWidth) {
|
|
|
|
// The MonitorConstraint will allocate us a fixed size anyway
|
|
|
|
return [0, 0];
|
|
|
|
}
|
|
|
|
|
2022-03-05 13:14:19 +01:00
|
|
|
vfunc_allocate(container, box) {
|
2020-12-30 17:16:18 -03:00
|
|
|
const childBox = new Clutter.ActorBox();
|
|
|
|
|
|
|
|
const { spacing } = this;
|
|
|
|
|
2022-07-30 21:59:33 -07:00
|
|
|
const startY = this._workAreaBox.y1;
|
2022-03-05 13:14:19 +01:00
|
|
|
box.y1 += startY;
|
|
|
|
const [width, height] = box.get_size();
|
2020-12-30 17:16:18 -03:00
|
|
|
let availableHeight = height;
|
|
|
|
|
|
|
|
// Search entry
|
2021-03-09 17:31:30 +01:00
|
|
|
let [searchHeight] = this._searchEntry.get_preferred_height(width);
|
2022-03-05 13:14:19 +01:00
|
|
|
childBox.set_origin(0, startY);
|
2020-12-30 17:16:18 -03:00
|
|
|
childBox.set_size(width, searchHeight);
|
|
|
|
this._searchEntry.allocate(childBox);
|
|
|
|
|
|
|
|
availableHeight -= searchHeight + spacing;
|
|
|
|
|
|
|
|
// Dash
|
2022-03-05 13:14:19 +01:00
|
|
|
const maxDashHeight = Math.round(box.get_height() * DASH_MAX_HEIGHT_RATIO);
|
2021-01-22 10:36:37 -03:00
|
|
|
this._dash.setMaxSize(width, maxDashHeight);
|
|
|
|
|
|
|
|
let [, dashHeight] = this._dash.get_preferred_height(width);
|
|
|
|
dashHeight = Math.min(dashHeight, maxDashHeight);
|
2022-03-05 13:14:19 +01:00
|
|
|
childBox.set_origin(0, startY + height - dashHeight);
|
2020-12-30 17:16:18 -03:00
|
|
|
childBox.set_size(width, dashHeight);
|
|
|
|
this._dash.allocate(childBox);
|
|
|
|
|
|
|
|
availableHeight -= dashHeight + spacing;
|
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
// Workspace Thumbnails
|
|
|
|
let thumbnailsHeight = 0;
|
|
|
|
if (this._workspacesThumbnails.visible) {
|
2021-02-16 22:03:29 +01:00
|
|
|
const { expandFraction } = this._workspacesThumbnails;
|
2021-01-14 20:23:15 -03:00
|
|
|
[thumbnailsHeight] =
|
|
|
|
this._workspacesThumbnails.get_preferred_height(width);
|
|
|
|
thumbnailsHeight = Math.min(
|
2021-02-16 22:03:29 +01:00
|
|
|
thumbnailsHeight * expandFraction,
|
2021-01-14 20:23:15 -03:00
|
|
|
height * WorkspaceThumbnail.MAX_THUMBNAIL_SCALE);
|
2022-03-05 13:14:19 +01:00
|
|
|
childBox.set_origin(0, startY + searchHeight + spacing);
|
2021-01-14 20:23:15 -03:00
|
|
|
childBox.set_size(width, thumbnailsHeight);
|
|
|
|
this._workspacesThumbnails.allocate(childBox);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Workspaces
|
2022-07-30 21:59:33 -07:00
|
|
|
let params = [box, searchHeight, dashHeight, thumbnailsHeight];
|
2021-01-14 20:23:15 -03:00
|
|
|
const transitionParams = this._stateAdjustment.getStateTransitionParams();
|
|
|
|
|
2021-02-10 18:09:04 -03:00
|
|
|
// Update cached boxes
|
|
|
|
for (const state of Object.values(ControlsState)) {
|
|
|
|
this._cachedWorkspaceBoxes.set(
|
workspacesView: Interpolate against relative workspace boxes
The overview transition consists of getting the initial and final
states of the overview adjustment, derivating various other internal
states from them (such as the fit mode, opacities, translations, etc),
and finally interpolating the allocation boxes.
When interpolating between the fit mode, WorkspacesView uses the current
allocation box to derivate the SINGLE and ALL fit mode boxes. However,
that creates a curved path during overview transitions. What we really
want to do here is calculate the fit mode box relative to the corresponding
overview state. For example:
+----------------+----------+------------------------+
| Overview State | Fit Mode | Workspaces geometry |
+----------------+----------+------------------------+
| HIDDEN | SINGLE | Cover entire screen |
| WINDOW PICKER | SINGLE | Between minimap & Dash |
| APP GRID | ALL | 15% screen height |
+----------------+----------+------------------------+
Using the table above as the reference, when the overview transitions
between WINDOW PICKER and APP GRID, we must interpolate between
(SINGLE fit mode @ between minimap & Dash) and (ALL fit mode @ 15% screen
height). That way, we always interpolate the final boxes, which corrects
the odd path that workspaces follow during this transition.
Make the WorkspacesView of the primary monitor use these cached boxes
when the overview is in the middle of a transition, and the fit modes of
the initial and final state differ, to calculate the workspaces positions.
Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/1624>
2021-02-02 16:34:59 -03:00
|
|
|
state, this._computeWorkspacesBoxForState(state, ...params));
|
2021-02-10 18:09:04 -03:00
|
|
|
}
|
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
let workspacesBox;
|
|
|
|
if (!transitionParams.transitioning) {
|
2021-02-10 18:09:04 -03:00
|
|
|
workspacesBox = this._cachedWorkspaceBoxes.get(transitionParams.currentState);
|
2021-01-14 20:23:15 -03:00
|
|
|
} else {
|
2021-02-10 18:09:04 -03:00
|
|
|
const initialBox = this._cachedWorkspaceBoxes.get(transitionParams.initialState);
|
|
|
|
const finalBox = this._cachedWorkspaceBoxes.get(transitionParams.finalState);
|
2021-01-14 20:23:15 -03:00
|
|
|
workspacesBox = initialBox.interpolate(finalBox, transitionParams.progress);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._workspacesDisplay.allocate(workspacesBox);
|
|
|
|
|
|
|
|
// AppDisplay
|
2021-03-09 17:37:22 +01:00
|
|
|
if (this._appDisplay.visible) {
|
|
|
|
const workspaceAppGridBox =
|
|
|
|
this._cachedWorkspaceBoxes.get(ControlsState.APP_GRID);
|
|
|
|
|
2022-07-30 21:59:33 -07:00
|
|
|
params = [box, searchHeight, dashHeight, workspaceAppGridBox];
|
2021-03-09 17:37:22 +01:00
|
|
|
let appDisplayBox;
|
|
|
|
if (!transitionParams.transitioning) {
|
|
|
|
appDisplayBox =
|
|
|
|
this._getAppDisplayBoxForState(transitionParams.currentState, ...params);
|
|
|
|
} else {
|
|
|
|
const initialBox =
|
|
|
|
this._getAppDisplayBoxForState(transitionParams.initialState, ...params);
|
|
|
|
const finalBox =
|
|
|
|
this._getAppDisplayBoxForState(transitionParams.finalState, ...params);
|
|
|
|
|
|
|
|
appDisplayBox = initialBox.interpolate(finalBox, transitionParams.progress);
|
|
|
|
}
|
2021-01-14 20:23:15 -03:00
|
|
|
|
2021-03-09 17:37:22 +01:00
|
|
|
this._appDisplay.allocate(appDisplayBox);
|
2021-01-20 19:25:13 -03:00
|
|
|
}
|
2020-12-30 17:32:03 -03:00
|
|
|
|
2021-01-15 09:59:58 -03:00
|
|
|
// Search
|
2022-03-05 13:14:19 +01:00
|
|
|
childBox.set_origin(0, startY + searchHeight + spacing);
|
|
|
|
childBox.set_size(width, availableHeight);
|
2020-12-30 17:32:03 -03:00
|
|
|
|
2021-01-15 09:59:58 -03:00
|
|
|
this._searchController.allocate(childBox);
|
2021-02-15 17:39:14 -03:00
|
|
|
|
|
|
|
this._runPostAllocation();
|
|
|
|
}
|
|
|
|
|
|
|
|
ensureAllocation() {
|
|
|
|
this.layout_changed();
|
|
|
|
return new Promise(
|
|
|
|
resolve => this._postAllocationCallbacks.push(resolve));
|
2020-12-30 17:16:18 -03:00
|
|
|
}
|
workspacesView: Interpolate against relative workspace boxes
The overview transition consists of getting the initial and final
states of the overview adjustment, derivating various other internal
states from them (such as the fit mode, opacities, translations, etc),
and finally interpolating the allocation boxes.
When interpolating between the fit mode, WorkspacesView uses the current
allocation box to derivate the SINGLE and ALL fit mode boxes. However,
that creates a curved path during overview transitions. What we really
want to do here is calculate the fit mode box relative to the corresponding
overview state. For example:
+----------------+----------+------------------------+
| Overview State | Fit Mode | Workspaces geometry |
+----------------+----------+------------------------+
| HIDDEN | SINGLE | Cover entire screen |
| WINDOW PICKER | SINGLE | Between minimap & Dash |
| APP GRID | ALL | 15% screen height |
+----------------+----------+------------------------+
Using the table above as the reference, when the overview transitions
between WINDOW PICKER and APP GRID, we must interpolate between
(SINGLE fit mode @ between minimap & Dash) and (ALL fit mode @ 15% screen
height). That way, we always interpolate the final boxes, which corrects
the odd path that workspaces follow during this transition.
Make the WorkspacesView of the primary monitor use these cached boxes
when the overview is in the middle of a transition, and the fit modes of
the initial and final state differ, to calculate the workspaces positions.
Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/1624>
2021-02-02 16:34:59 -03:00
|
|
|
|
|
|
|
getWorkspacesBoxForState(state) {
|
|
|
|
return this._cachedWorkspaceBoxes.get(state);
|
|
|
|
}
|
2020-12-30 17:16:18 -03:00
|
|
|
});
|
|
|
|
|
2021-02-12 00:20:31 +01:00
|
|
|
var OverviewAdjustment = GObject.registerClass({
|
|
|
|
Properties: {
|
|
|
|
'gesture-in-progress': GObject.ParamSpec.boolean(
|
|
|
|
'gesture-in-progress', 'Gesture in progress', 'Gesture in progress',
|
|
|
|
GObject.ParamFlags.READWRITE,
|
|
|
|
false),
|
|
|
|
},
|
|
|
|
}, class OverviewAdjustment extends St.Adjustment {
|
2021-01-15 15:28:23 -03:00
|
|
|
_init(actor) {
|
|
|
|
super._init({
|
|
|
|
actor,
|
|
|
|
value: ControlsState.WINDOW_PICKER,
|
|
|
|
lower: ControlsState.HIDDEN,
|
|
|
|
upper: ControlsState.APP_GRID,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
getStateTransitionParams() {
|
|
|
|
const currentState = this.value;
|
|
|
|
|
|
|
|
const transition = this.get_transition('value');
|
|
|
|
let initialState = transition
|
|
|
|
? transition.get_interval().peek_initial_value()
|
|
|
|
: currentState;
|
|
|
|
let finalState = transition
|
|
|
|
? transition.get_interval().peek_final_value()
|
|
|
|
: currentState;
|
|
|
|
|
|
|
|
if (initialState > finalState) {
|
|
|
|
initialState = Math.ceil(initialState);
|
|
|
|
finalState = Math.floor(finalState);
|
|
|
|
} else {
|
|
|
|
initialState = Math.floor(initialState);
|
|
|
|
finalState = Math.ceil(finalState);
|
|
|
|
}
|
|
|
|
|
|
|
|
const length = Math.abs(finalState - initialState);
|
|
|
|
const progress = length > 0
|
|
|
|
? Math.abs((currentState - initialState) / length)
|
|
|
|
: 1;
|
|
|
|
|
|
|
|
return {
|
2021-02-12 00:20:31 +01:00
|
|
|
transitioning: transition !== null || this.gestureInProgress,
|
2021-01-15 15:28:23 -03:00
|
|
|
currentState,
|
|
|
|
initialState,
|
|
|
|
finalState,
|
|
|
|
progress,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
var ControlsManager = GObject.registerClass(
|
|
|
|
class ControlsManager extends St.Widget {
|
2020-12-30 17:02:14 -03:00
|
|
|
_init() {
|
2019-07-16 11:24:13 +02:00
|
|
|
super._init({
|
2020-12-30 17:16:18 -03:00
|
|
|
style_class: 'controls-manager',
|
2019-07-16 11:24:13 +02:00
|
|
|
x_expand: true,
|
|
|
|
y_expand: true,
|
2019-08-20 23:43:54 +02:00
|
|
|
clip_to_allocation: true,
|
2019-07-16 11:24:13 +02:00
|
|
|
});
|
|
|
|
|
2021-01-01 14:00:30 -03:00
|
|
|
this._ignoreShowAppsButtonToggle = false;
|
|
|
|
|
2020-12-30 17:02:14 -03:00
|
|
|
this._searchEntry = new St.Entry({
|
|
|
|
style_class: 'search-entry',
|
|
|
|
/* Translators: this is the text displayed
|
|
|
|
in the search entry when no search is
|
|
|
|
active; it should not exceed ~30
|
|
|
|
characters. */
|
|
|
|
hint_text: _('Type to search'),
|
|
|
|
track_hover: true,
|
|
|
|
can_focus: true,
|
|
|
|
});
|
|
|
|
this._searchEntry.set_offscreen_redirect(Clutter.OffscreenRedirect.ALWAYS);
|
2021-02-15 17:38:36 -03:00
|
|
|
this._searchEntryBin = new St.Bin({
|
2020-12-30 17:02:14 -03:00
|
|
|
child: this._searchEntry,
|
|
|
|
x_align: Clutter.ActorAlign.CENTER,
|
|
|
|
});
|
|
|
|
|
2013-02-25 18:05:45 -05:00
|
|
|
this.dash = new Dash.Dash();
|
|
|
|
|
2019-07-08 13:03:20 +05:00
|
|
|
let workspaceManager = global.workspace_manager;
|
|
|
|
let activeWorkspaceIndex = workspaceManager.get_active_workspace_index();
|
|
|
|
|
|
|
|
this._workspaceAdjustment = new St.Adjustment({
|
2020-06-16 22:03:07 +02:00
|
|
|
actor: this,
|
2019-07-08 13:03:20 +05:00
|
|
|
value: activeWorkspaceIndex,
|
|
|
|
lower: 0,
|
|
|
|
page_increment: 1,
|
|
|
|
page_size: 1,
|
|
|
|
step_increment: 0,
|
|
|
|
upper: workspaceManager.n_workspaces,
|
|
|
|
});
|
|
|
|
|
2021-01-15 15:28:23 -03:00
|
|
|
this._stateAdjustment = new OverviewAdjustment(this);
|
2021-01-14 20:23:15 -03:00
|
|
|
this._stateAdjustment.connect('notify::value', this._update.bind(this));
|
2020-12-30 17:32:03 -03:00
|
|
|
|
2021-08-16 00:36:59 +02:00
|
|
|
workspaceManager.connectObject(
|
|
|
|
'notify::n-workspaces', () => this._updateAdjustment(), this);
|
2019-07-08 13:03:20 +05:00
|
|
|
|
2021-01-15 09:59:58 -03:00
|
|
|
this._searchController = new SearchController.SearchController(
|
|
|
|
this._searchEntry,
|
2021-01-14 20:23:15 -03:00
|
|
|
this.dash.showAppsButton);
|
2021-01-15 09:59:58 -03:00
|
|
|
this._searchController.connect('notify::search-active', this._onSearchChanged.bind(this));
|
2021-01-14 20:23:15 -03:00
|
|
|
|
workspaceThumbnail: Update monitor index on 'monitors-changed'
For the primary monitor workspace thumbnail, we must keep the monitor
index in sync with what is currently the primary monitor index,
otherwise we might end up trying to move windows to non-existing
monitors.
For example, if the primary monitor index was 1 when the thumbnail box
was created, but later, the primary monitor index changed to 0, with the
other monitor being turned off, moving a window to one of the workspaces
on the workspace thumbnail, gnome-shell would attempt to move it to the
monitor with the index the primary monitor had in the past, with the
problem being that that monitor no longer exists.
Fix this by listening on the 'monitors-changed' signal on the layout
manager, and update the monitor index of the primary workspace
thumbnails box. Make sure to connect to the signal before creating the
thumbnails box, as the thumbnails box itself will listen to the signal
and recreate its actual thumbnails, and it must do this with the up to
date monitor index.
Closes: https://gitlab.gnome.org/GNOME/gnome-shell/-/issues/4075
Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/1810>
2021-04-14 17:12:12 +02:00
|
|
|
Main.layoutManager.connect('monitors-changed', () => {
|
|
|
|
this._thumbnailsBox.setMonitorIndex(Main.layoutManager.primaryIndex);
|
|
|
|
});
|
2021-02-26 14:09:15 +01:00
|
|
|
this._thumbnailsBox = new WorkspaceThumbnail.ThumbnailsBox(
|
|
|
|
this._workspaceAdjustment, Main.layoutManager.primaryIndex);
|
2021-02-16 22:03:29 +01:00
|
|
|
this._thumbnailsBox.connect('notify::should-show', () => {
|
|
|
|
this._thumbnailsBox.show();
|
|
|
|
this._thumbnailsBox.ease_property('expand-fraction',
|
|
|
|
this._thumbnailsBox.should_show ? 1 : 0, {
|
|
|
|
duration: SIDE_CONTROLS_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
onComplete: () => this._updateThumbnailsBox(),
|
|
|
|
});
|
|
|
|
});
|
2021-02-16 20:40:43 +01:00
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
this._workspacesDisplay = new WorkspacesView.WorkspacesDisplay(
|
workspacesView: Interpolate against relative workspace boxes
The overview transition consists of getting the initial and final
states of the overview adjustment, derivating various other internal
states from them (such as the fit mode, opacities, translations, etc),
and finally interpolating the allocation boxes.
When interpolating between the fit mode, WorkspacesView uses the current
allocation box to derivate the SINGLE and ALL fit mode boxes. However,
that creates a curved path during overview transitions. What we really
want to do here is calculate the fit mode box relative to the corresponding
overview state. For example:
+----------------+----------+------------------------+
| Overview State | Fit Mode | Workspaces geometry |
+----------------+----------+------------------------+
| HIDDEN | SINGLE | Cover entire screen |
| WINDOW PICKER | SINGLE | Between minimap & Dash |
| APP GRID | ALL | 15% screen height |
+----------------+----------+------------------------+
Using the table above as the reference, when the overview transitions
between WINDOW PICKER and APP GRID, we must interpolate between
(SINGLE fit mode @ between minimap & Dash) and (ALL fit mode @ 15% screen
height). That way, we always interpolate the final boxes, which corrects
the odd path that workspaces follow during this transition.
Make the WorkspacesView of the primary monitor use these cached boxes
when the overview is in the middle of a transition, and the fit modes of
the initial and final state differ, to calculate the workspaces positions.
Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/1624>
2021-02-02 16:34:59 -03:00
|
|
|
this,
|
2020-12-30 17:45:09 -03:00
|
|
|
this._workspaceAdjustment,
|
|
|
|
this._stateAdjustment);
|
2021-01-14 20:23:15 -03:00
|
|
|
this._appDisplay = new AppDisplay.AppDisplay();
|
2013-02-15 18:25:36 -05:00
|
|
|
|
2021-02-15 17:38:36 -03:00
|
|
|
this.add_child(this._searchEntryBin);
|
2021-01-14 20:23:15 -03:00
|
|
|
this.add_child(this._appDisplay);
|
2020-12-30 17:16:18 -03:00
|
|
|
this.add_child(this.dash);
|
2021-01-15 09:59:58 -03:00
|
|
|
this.add_child(this._searchController);
|
2021-01-14 20:23:15 -03:00
|
|
|
this.add_child(this._thumbnailsBox);
|
|
|
|
this.add_child(this._workspacesDisplay);
|
2013-02-25 18:05:45 -05:00
|
|
|
|
2021-02-15 17:38:36 -03:00
|
|
|
this.layout_manager = new ControlsManagerLayout(
|
|
|
|
this._searchEntryBin,
|
2021-01-14 20:23:15 -03:00
|
|
|
this._appDisplay,
|
|
|
|
this._workspacesDisplay,
|
|
|
|
this._thumbnailsBox,
|
2021-01-15 09:59:58 -03:00
|
|
|
this._searchController,
|
2021-01-14 20:23:15 -03:00
|
|
|
this.dash,
|
|
|
|
this._stateAdjustment);
|
2013-02-17 23:45:24 -05:00
|
|
|
|
2020-12-30 17:45:09 -03:00
|
|
|
this.dash.showAppsButton.connect('notify::checked',
|
|
|
|
this._onShowAppsButtonToggled.bind(this));
|
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
Main.ctrlAltTabManager.addGroup(
|
|
|
|
this.appDisplay,
|
2022-12-16 18:33:10 -03:00
|
|
|
_('Apps'),
|
2021-01-14 20:23:15 -03:00
|
|
|
'view-app-grid-symbolic', {
|
|
|
|
proxy: this,
|
|
|
|
focusCallback: () => {
|
|
|
|
this.dash.showAppsButton.checked = true;
|
|
|
|
this.appDisplay.navigate_focus(
|
|
|
|
null, St.DirectionType.TAB_FORWARD, false);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Main.ctrlAltTabManager.addGroup(
|
|
|
|
this._workspacesDisplay,
|
|
|
|
_('Windows'),
|
|
|
|
'focus-windows-symbolic', {
|
|
|
|
proxy: this,
|
|
|
|
focusCallback: () => {
|
|
|
|
this.dash.showAppsButton.checked = false;
|
|
|
|
this._workspacesDisplay.navigate_focus(
|
|
|
|
null, St.DirectionType.TAB_FORWARD, false);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2020-06-28 02:45:18 +02:00
|
|
|
this._a11ySettings = new Gio.Settings({ schema_id: A11Y_SCHEMA });
|
|
|
|
|
2021-04-15 12:00:12 +02:00
|
|
|
this._lastOverlayKeyTime = 0;
|
2020-06-28 02:45:18 +02:00
|
|
|
global.display.connect('overlay-key', () => {
|
|
|
|
if (this._a11ySettings.get_boolean('stickykeys-enable'))
|
|
|
|
return;
|
|
|
|
|
2017-11-15 17:15:49 +00:00
|
|
|
const { initialState, finalState, transitioning } =
|
|
|
|
this._stateAdjustment.getStateTransitionParams();
|
|
|
|
|
2021-04-15 12:00:12 +02:00
|
|
|
const time = GLib.get_monotonic_time() / 1000;
|
|
|
|
const timeDiff = time - this._lastOverlayKeyTime;
|
|
|
|
this._lastOverlayKeyTime = time;
|
|
|
|
|
|
|
|
const shouldShift = St.Settings.get().enable_animations
|
|
|
|
? transitioning && finalState > initialState
|
|
|
|
: Main.overview.visible && timeDiff < Overview.ANIMATION_TIME;
|
|
|
|
|
|
|
|
if (shouldShift)
|
2017-11-15 17:15:49 +00:00
|
|
|
this._shiftState(Meta.MotionDirection.UP);
|
|
|
|
else
|
|
|
|
Main.overview.toggle();
|
2020-06-28 02:45:18 +02:00
|
|
|
});
|
|
|
|
|
2022-01-24 17:00:36 +01:00
|
|
|
// connect_after to give search controller first dibs on the event
|
|
|
|
global.stage.connect_after('key-press-event', (actor, event) => {
|
|
|
|
if (this._searchController.searchActive)
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
|
|
|
if (global.stage.key_focus &&
|
|
|
|
!this.contains(global.stage.key_focus))
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
|
|
|
const { finalState } =
|
|
|
|
this._stateAdjustment.getStateTransitionParams();
|
|
|
|
let keynavDisplay;
|
|
|
|
|
|
|
|
if (finalState === ControlsState.WINDOW_PICKER)
|
|
|
|
keynavDisplay = this._workspacesDisplay;
|
|
|
|
else if (finalState === ControlsState.APP_GRID)
|
|
|
|
keynavDisplay = this._appDisplay;
|
|
|
|
|
|
|
|
if (!keynavDisplay)
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
|
|
|
const symbol = event.get_key_symbol();
|
|
|
|
if (symbol === Clutter.KEY_Tab || symbol === Clutter.KEY_Down) {
|
|
|
|
keynavDisplay.navigate_focus(
|
|
|
|
null, St.DirectionType.TAB_FORWARD, false);
|
|
|
|
return Clutter.EVENT_STOP;
|
|
|
|
} else if (symbol === Clutter.KEY_ISO_Left_Tab) {
|
|
|
|
keynavDisplay.navigate_focus(
|
|
|
|
null, St.DirectionType.TAB_BACKWARD, false);
|
|
|
|
return Clutter.EVENT_STOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
});
|
|
|
|
|
2021-01-02 17:41:54 -03:00
|
|
|
Main.wm.addKeybinding(
|
|
|
|
'toggle-application-view',
|
|
|
|
new Gio.Settings({ schema_id: WindowManager.SHELL_KEYBINDINGS_SCHEMA }),
|
|
|
|
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
|
|
|
Shell.ActionMode.NORMAL | Shell.ActionMode.OVERVIEW,
|
|
|
|
this._toggleAppsPage.bind(this));
|
|
|
|
|
2020-06-13 13:26:09 +02:00
|
|
|
Main.wm.addKeybinding('shift-overview-up',
|
|
|
|
new Gio.Settings({ schema_id: WindowManager.SHELL_KEYBINDINGS_SCHEMA }),
|
|
|
|
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
|
|
|
Shell.ActionMode.NORMAL | Shell.ActionMode.OVERVIEW,
|
|
|
|
() => this._shiftState(Meta.MotionDirection.UP));
|
|
|
|
|
|
|
|
Main.wm.addKeybinding('shift-overview-down',
|
|
|
|
new Gio.Settings({ schema_id: WindowManager.SHELL_KEYBINDINGS_SCHEMA }),
|
|
|
|
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
|
|
|
Shell.ActionMode.NORMAL | Shell.ActionMode.OVERVIEW,
|
|
|
|
() => this._shiftState(Meta.MotionDirection.DOWN));
|
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
this._update();
|
2023-02-02 15:54:05 +01:00
|
|
|
|
|
|
|
this.connect('destroy', this._onDestroy.bind(this));
|
2021-01-14 20:23:15 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
_getFitModeForState(state) {
|
|
|
|
switch (state) {
|
|
|
|
case ControlsState.HIDDEN:
|
|
|
|
case ControlsState.WINDOW_PICKER:
|
|
|
|
return WorkspacesView.FitMode.SINGLE;
|
|
|
|
case ControlsState.APP_GRID:
|
|
|
|
return WorkspacesView.FitMode.ALL;
|
|
|
|
default:
|
|
|
|
return WorkspacesView.FitMode.SINGLE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_getThumbnailsBoxParams() {
|
|
|
|
const { initialState, finalState, progress } =
|
|
|
|
this._stateAdjustment.getStateTransitionParams();
|
|
|
|
|
|
|
|
const paramsForState = s => {
|
|
|
|
let opacity, scale, translationY;
|
|
|
|
switch (s) {
|
|
|
|
case ControlsState.HIDDEN:
|
|
|
|
case ControlsState.WINDOW_PICKER:
|
|
|
|
opacity = 255;
|
|
|
|
scale = 1;
|
|
|
|
translationY = 0;
|
|
|
|
break;
|
|
|
|
case ControlsState.APP_GRID:
|
|
|
|
opacity = 0;
|
|
|
|
scale = 0.5;
|
|
|
|
translationY = this._thumbnailsBox.height / 2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
opacity = 255;
|
|
|
|
scale = 1;
|
|
|
|
translationY = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return { opacity, scale, translationY };
|
|
|
|
};
|
|
|
|
|
|
|
|
const initialParams = paramsForState(initialState);
|
|
|
|
const finalParams = paramsForState(finalState);
|
|
|
|
|
|
|
|
return [
|
|
|
|
Util.lerp(initialParams.opacity, finalParams.opacity, progress),
|
|
|
|
Util.lerp(initialParams.scale, finalParams.scale, progress),
|
|
|
|
Util.lerp(initialParams.translationY, finalParams.translationY, progress),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
_updateThumbnailsBox(animate = false) {
|
2021-02-16 20:40:43 +01:00
|
|
|
const { shouldShow } = this._thumbnailsBox;
|
2021-01-15 09:59:58 -03:00
|
|
|
const { searchActive } = this._searchController;
|
2021-01-14 20:23:15 -03:00
|
|
|
const [opacity, scale, translationY] = this._getThumbnailsBoxParams();
|
|
|
|
|
2021-02-16 20:40:43 +01:00
|
|
|
const thumbnailsBoxVisible = shouldShow && !searchActive && opacity !== 0;
|
2021-01-14 20:23:15 -03:00
|
|
|
if (thumbnailsBoxVisible) {
|
|
|
|
this._thumbnailsBox.opacity = 0;
|
|
|
|
this._thumbnailsBox.visible = thumbnailsBoxVisible;
|
|
|
|
}
|
|
|
|
|
|
|
|
const params = {
|
2021-01-15 09:50:39 -03:00
|
|
|
opacity: searchActive ? 0 : opacity,
|
2021-01-14 20:23:15 -03:00
|
|
|
duration: animate ? SIDE_CONTROLS_ANIMATION_TIME : 0,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
onComplete: () => (this._thumbnailsBox.visible = thumbnailsBoxVisible),
|
|
|
|
};
|
|
|
|
|
2021-01-15 09:50:39 -03:00
|
|
|
if (!searchActive) {
|
2021-01-14 20:23:15 -03:00
|
|
|
params.scale_x = scale;
|
|
|
|
params.scale_y = scale;
|
|
|
|
params.translation_y = translationY;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._thumbnailsBox.ease(params);
|
|
|
|
}
|
|
|
|
|
2021-04-10 13:27:01 -03:00
|
|
|
_updateAppDisplayVisibility(stateTransitionParams = null) {
|
|
|
|
if (!stateTransitionParams)
|
|
|
|
stateTransitionParams = this._stateAdjustment.getStateTransitionParams();
|
|
|
|
|
2021-06-13 00:41:52 +05:30
|
|
|
const { initialState, finalState } = stateTransitionParams;
|
|
|
|
const state = Math.max(initialState, finalState);
|
2021-04-10 13:27:01 -03:00
|
|
|
|
2021-03-09 17:37:22 +01:00
|
|
|
this._appDisplay.visible =
|
2021-04-10 13:27:01 -03:00
|
|
|
state > ControlsState.WINDOW_PICKER &&
|
2021-03-09 17:37:22 +01:00
|
|
|
!this._searchController.searchActive;
|
|
|
|
}
|
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
_update() {
|
|
|
|
const params = this._stateAdjustment.getStateTransitionParams();
|
|
|
|
|
|
|
|
const fitMode = Util.lerp(
|
|
|
|
this._getFitModeForState(params.initialState),
|
|
|
|
this._getFitModeForState(params.finalState),
|
|
|
|
params.progress);
|
|
|
|
|
|
|
|
const { fitModeAdjustment } = this._workspacesDisplay;
|
|
|
|
fitModeAdjustment.value = fitMode;
|
|
|
|
|
|
|
|
this._updateThumbnailsBox();
|
2021-04-10 13:27:01 -03:00
|
|
|
this._updateAppDisplayVisibility(params);
|
2021-01-14 20:23:15 -03:00
|
|
|
}
|
|
|
|
|
2021-01-15 09:50:39 -03:00
|
|
|
_onSearchChanged() {
|
2021-01-15 09:59:58 -03:00
|
|
|
const { searchActive } = this._searchController;
|
2021-01-14 20:23:15 -03:00
|
|
|
|
2021-01-15 09:50:39 -03:00
|
|
|
if (!searchActive) {
|
2021-03-09 17:37:22 +01:00
|
|
|
this._updateAppDisplayVisibility();
|
2021-01-14 20:23:15 -03:00
|
|
|
this._workspacesDisplay.reactive = true;
|
|
|
|
this._workspacesDisplay.setPrimaryWorkspaceVisible(true);
|
|
|
|
} else {
|
2021-01-15 09:59:58 -03:00
|
|
|
this._searchController.show();
|
2021-01-14 20:23:15 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
this._updateThumbnailsBox(true);
|
|
|
|
|
|
|
|
this._appDisplay.ease({
|
2021-01-15 09:50:39 -03:00
|
|
|
opacity: searchActive ? 0 : 255,
|
2021-01-14 20:23:15 -03:00
|
|
|
duration: SIDE_CONTROLS_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2021-03-09 17:37:22 +01:00
|
|
|
onComplete: () => this._updateAppDisplayVisibility(),
|
2021-01-14 20:23:15 -03:00
|
|
|
});
|
|
|
|
this._workspacesDisplay.ease({
|
2021-01-15 09:50:39 -03:00
|
|
|
opacity: searchActive ? 0 : 255,
|
2021-01-14 20:23:15 -03:00
|
|
|
duration: SIDE_CONTROLS_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
onComplete: () => {
|
2021-01-15 09:50:39 -03:00
|
|
|
this._workspacesDisplay.reactive = !searchActive;
|
|
|
|
this._workspacesDisplay.setPrimaryWorkspaceVisible(!searchActive);
|
2021-01-14 20:23:15 -03:00
|
|
|
},
|
|
|
|
});
|
2021-01-15 09:59:58 -03:00
|
|
|
this._searchController.ease({
|
2021-01-15 09:50:39 -03:00
|
|
|
opacity: searchActive ? 255 : 0,
|
2021-01-14 20:23:15 -03:00
|
|
|
duration: SIDE_CONTROLS_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2021-01-15 09:59:58 -03:00
|
|
|
onComplete: () => (this._searchController.visible = searchActive),
|
2021-01-14 20:23:15 -03:00
|
|
|
});
|
2019-07-08 13:03:20 +05:00
|
|
|
}
|
|
|
|
|
2020-12-30 17:45:09 -03:00
|
|
|
_onShowAppsButtonToggled() {
|
2021-01-01 14:00:30 -03:00
|
|
|
if (this._ignoreShowAppsButtonToggle)
|
|
|
|
return;
|
|
|
|
|
2020-12-30 17:45:09 -03:00
|
|
|
const checked = this.dash.showAppsButton.checked;
|
|
|
|
|
|
|
|
const value = checked
|
|
|
|
? ControlsState.APP_GRID : ControlsState.WINDOW_PICKER;
|
|
|
|
this._stateAdjustment.remove_transition('value');
|
|
|
|
this._stateAdjustment.ease(value, {
|
|
|
|
duration: SIDE_CONTROLS_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-01-02 17:41:54 -03:00
|
|
|
_toggleAppsPage() {
|
|
|
|
if (Main.overview.visible) {
|
|
|
|
const checked = this.dash.showAppsButton.checked;
|
|
|
|
this.dash.showAppsButton.checked = !checked;
|
|
|
|
} else {
|
|
|
|
Main.overview.show(ControlsState.APP_GRID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-13 13:26:09 +02:00
|
|
|
_shiftState(direction) {
|
|
|
|
let { currentState, finalState } = this._stateAdjustment.getStateTransitionParams();
|
|
|
|
|
|
|
|
if (direction === Meta.MotionDirection.DOWN)
|
|
|
|
finalState = Math.max(finalState - 1, ControlsState.HIDDEN);
|
|
|
|
else if (direction === Meta.MotionDirection.UP)
|
|
|
|
finalState = Math.min(finalState + 1, ControlsState.APP_GRID);
|
|
|
|
|
|
|
|
if (finalState === currentState)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (currentState === ControlsState.HIDDEN &&
|
|
|
|
finalState === ControlsState.WINDOW_PICKER) {
|
|
|
|
Main.overview.show();
|
|
|
|
} else if (finalState === ControlsState.HIDDEN) {
|
|
|
|
Main.overview.hide();
|
|
|
|
} else {
|
|
|
|
this._stateAdjustment.ease(finalState, {
|
|
|
|
duration: SIDE_CONTROLS_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
onComplete: () => {
|
|
|
|
this.dash.showAppsButton.checked =
|
|
|
|
finalState === ControlsState.APP_GRID;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-08 13:03:20 +05:00
|
|
|
_updateAdjustment() {
|
|
|
|
let workspaceManager = global.workspace_manager;
|
|
|
|
let newNumWorkspaces = workspaceManager.n_workspaces;
|
|
|
|
let activeIndex = workspaceManager.get_active_workspace_index();
|
|
|
|
|
|
|
|
this._workspaceAdjustment.upper = newNumWorkspaces;
|
|
|
|
|
|
|
|
// A workspace might have been inserted or removed before the active
|
|
|
|
// one, causing the adjustment to go out of sync, so update the value
|
2020-04-28 01:04:08 +02:00
|
|
|
this._workspaceAdjustment.remove_transition('value');
|
2019-07-08 13:03:20 +05:00
|
|
|
this._workspaceAdjustment.value = activeIndex;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2020-12-30 17:02:14 -03:00
|
|
|
|
2021-01-14 20:23:15 -03:00
|
|
|
vfunc_unmap() {
|
|
|
|
super.vfunc_unmap();
|
2023-02-02 15:54:05 +01:00
|
|
|
this._workspacesDisplay?.hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
_onDestroy() {
|
|
|
|
delete this._searchEntryBin;
|
|
|
|
delete this._appDisplay;
|
|
|
|
delete this.dash;
|
|
|
|
delete this._searchController;
|
|
|
|
delete this._thumbnailsBox;
|
|
|
|
delete this._workspacesDisplay;
|
2021-01-14 20:23:15 -03:00
|
|
|
}
|
|
|
|
|
2022-11-21 12:32:05 +01:00
|
|
|
prepareToEnterOverview() {
|
|
|
|
this._searchController.prepareToEnterOverview();
|
|
|
|
this._workspacesDisplay.prepareToEnterOverview();
|
|
|
|
}
|
|
|
|
|
|
|
|
prepareToLeaveOverview() {
|
|
|
|
this._workspacesDisplay.prepareToLeaveOverview();
|
|
|
|
}
|
|
|
|
|
2021-01-02 17:19:37 -03:00
|
|
|
animateToOverview(state, callback) {
|
2021-01-02 17:28:44 -03:00
|
|
|
this._ignoreShowAppsButtonToggle = true;
|
|
|
|
|
2021-01-01 14:00:30 -03:00
|
|
|
this._stateAdjustment.value = ControlsState.HIDDEN;
|
2021-01-02 17:19:37 -03:00
|
|
|
this._stateAdjustment.ease(state, {
|
2021-01-01 14:00:30 -03:00
|
|
|
duration: Overview.ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2021-01-01 14:18:18 -03:00
|
|
|
onStopped: () => {
|
|
|
|
if (callback)
|
|
|
|
callback();
|
|
|
|
},
|
2021-01-01 14:00:30 -03:00
|
|
|
});
|
2021-01-02 17:28:44 -03:00
|
|
|
|
|
|
|
this.dash.showAppsButton.checked =
|
|
|
|
state === ControlsState.APP_GRID;
|
|
|
|
|
|
|
|
this._ignoreShowAppsButtonToggle = false;
|
2021-01-01 13:39:07 -03:00
|
|
|
}
|
|
|
|
|
2021-01-01 14:18:18 -03:00
|
|
|
animateFromOverview(callback) {
|
2021-01-01 14:00:30 -03:00
|
|
|
this._ignoreShowAppsButtonToggle = true;
|
|
|
|
|
|
|
|
this._stateAdjustment.ease(ControlsState.HIDDEN, {
|
|
|
|
duration: Overview.ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
onStopped: () => {
|
|
|
|
this.dash.showAppsButton.checked = false;
|
|
|
|
this._ignoreShowAppsButtonToggle = false;
|
2021-01-01 14:18:18 -03:00
|
|
|
|
|
|
|
if (callback)
|
|
|
|
callback();
|
2021-01-01 14:00:30 -03:00
|
|
|
},
|
|
|
|
});
|
2021-01-01 13:39:07 -03:00
|
|
|
}
|
|
|
|
|
workspacesView: Interpolate against relative workspace boxes
The overview transition consists of getting the initial and final
states of the overview adjustment, derivating various other internal
states from them (such as the fit mode, opacities, translations, etc),
and finally interpolating the allocation boxes.
When interpolating between the fit mode, WorkspacesView uses the current
allocation box to derivate the SINGLE and ALL fit mode boxes. However,
that creates a curved path during overview transitions. What we really
want to do here is calculate the fit mode box relative to the corresponding
overview state. For example:
+----------------+----------+------------------------+
| Overview State | Fit Mode | Workspaces geometry |
+----------------+----------+------------------------+
| HIDDEN | SINGLE | Cover entire screen |
| WINDOW PICKER | SINGLE | Between minimap & Dash |
| APP GRID | ALL | 15% screen height |
+----------------+----------+------------------------+
Using the table above as the reference, when the overview transitions
between WINDOW PICKER and APP GRID, we must interpolate between
(SINGLE fit mode @ between minimap & Dash) and (ALL fit mode @ 15% screen
height). That way, we always interpolate the final boxes, which corrects
the odd path that workspaces follow during this transition.
Make the WorkspacesView of the primary monitor use these cached boxes
when the overview is in the middle of a transition, and the fit modes of
the initial and final state differ, to calculate the workspaces positions.
Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/1624>
2021-02-02 16:34:59 -03:00
|
|
|
getWorkspacesBoxForState(state) {
|
|
|
|
return this.layoutManager.getWorkspacesBoxForState(state);
|
|
|
|
}
|
|
|
|
|
2021-01-04 16:23:45 +01:00
|
|
|
gestureBegin(tracker) {
|
|
|
|
const baseDistance = global.screen_height;
|
|
|
|
const progress = this._stateAdjustment.value;
|
|
|
|
const points = [
|
|
|
|
ControlsState.HIDDEN,
|
|
|
|
ControlsState.WINDOW_PICKER,
|
|
|
|
ControlsState.APP_GRID,
|
|
|
|
];
|
|
|
|
|
|
|
|
const transition = this._stateAdjustment.get_transition('value');
|
|
|
|
const cancelProgress = transition
|
|
|
|
? transition.get_interval().peek_final_value()
|
|
|
|
: Math.round(progress);
|
2021-04-29 15:51:11 +02:00
|
|
|
this._stateAdjustment.remove_transition('value');
|
2021-01-04 16:23:45 +01:00
|
|
|
|
|
|
|
tracker.confirmSwipe(baseDistance, points, progress, cancelProgress);
|
2022-11-21 12:32:05 +01:00
|
|
|
this.prepareToEnterOverview();
|
2021-01-04 16:23:45 +01:00
|
|
|
this._stateAdjustment.gestureInProgress = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
gestureProgress(progress) {
|
|
|
|
this._stateAdjustment.value = progress;
|
|
|
|
}
|
|
|
|
|
|
|
|
gestureEnd(target, duration, onComplete) {
|
|
|
|
if (target === ControlsState.HIDDEN)
|
2022-11-21 12:32:05 +01:00
|
|
|
this.prepareToLeaveOverview();
|
2021-01-04 16:23:45 +01:00
|
|
|
|
|
|
|
this.dash.showAppsButton.checked =
|
|
|
|
target === ControlsState.APP_GRID;
|
|
|
|
|
|
|
|
this._stateAdjustment.remove_transition('value');
|
|
|
|
this._stateAdjustment.ease(target, {
|
|
|
|
duration,
|
2021-02-28 19:57:53 +05:00
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_CUBIC,
|
2021-01-04 16:23:45 +01:00
|
|
|
onComplete,
|
|
|
|
});
|
|
|
|
|
|
|
|
this._stateAdjustment.gestureInProgress = false;
|
|
|
|
}
|
|
|
|
|
2021-02-15 17:39:14 -03:00
|
|
|
async runStartupAnimation(callback) {
|
|
|
|
this._ignoreShowAppsButtonToggle = true;
|
|
|
|
|
2022-11-21 12:32:05 +01:00
|
|
|
this.prepareToEnterOverview();
|
2021-02-15 17:39:14 -03:00
|
|
|
|
|
|
|
this._stateAdjustment.value = ControlsState.HIDDEN;
|
|
|
|
this._stateAdjustment.ease(ControlsState.WINDOW_PICKER, {
|
|
|
|
duration: Overview.ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
});
|
|
|
|
|
|
|
|
this.dash.showAppsButton.checked = false;
|
|
|
|
this._ignoreShowAppsButtonToggle = false;
|
|
|
|
|
|
|
|
// Set the opacity here to avoid a 1-frame flicker
|
|
|
|
this.opacity = 0;
|
|
|
|
|
|
|
|
// We can't run the animation before the first allocation happens
|
|
|
|
await this.layout_manager.ensureAllocation();
|
|
|
|
|
|
|
|
const { STARTUP_ANIMATION_TIME } = Layout;
|
|
|
|
|
|
|
|
// Opacity
|
|
|
|
this.ease({
|
|
|
|
opacity: 255,
|
|
|
|
duration: STARTUP_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.LINEAR,
|
|
|
|
});
|
|
|
|
|
|
|
|
// Search bar falls from the ceiling
|
|
|
|
const { primaryMonitor } = Main.layoutManager;
|
|
|
|
const [, y] = this._searchEntryBin.get_transformed_position();
|
|
|
|
const yOffset = y - primaryMonitor.y;
|
|
|
|
|
|
|
|
this._searchEntryBin.translation_y = -(yOffset + this._searchEntryBin.height);
|
|
|
|
this._searchEntryBin.ease({
|
|
|
|
translation_y: 0,
|
|
|
|
duration: STARTUP_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
});
|
|
|
|
|
|
|
|
// The Dash rises from the bottom. This is the last animation to finish,
|
|
|
|
// so run the callback there.
|
2022-02-16 20:02:48 +01:00
|
|
|
this.dash.translation_y = this.dash.height + this.dash.margin_bottom;
|
2021-02-15 17:39:14 -03:00
|
|
|
this.dash.ease({
|
|
|
|
translation_y: 0,
|
|
|
|
delay: STARTUP_ANIMATION_TIME,
|
|
|
|
duration: STARTUP_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
onComplete: () => callback(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-06-01 15:54:59 -07:00
|
|
|
get searchController() {
|
|
|
|
return this._searchController;
|
|
|
|
}
|
|
|
|
|
2020-12-30 17:02:14 -03:00
|
|
|
get searchEntry() {
|
|
|
|
return this._searchEntry;
|
|
|
|
}
|
2021-01-14 20:23:15 -03:00
|
|
|
|
|
|
|
get appDisplay() {
|
|
|
|
return this._appDisplay;
|
|
|
|
}
|
2019-07-16 11:24:13 +02:00
|
|
|
});
|