2011-09-28 09:16:26 -04:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2008-10-31 23:09:46 +00:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
import Clutter from 'gi://Clutter';
|
|
|
|
import Gio from 'gi://Gio';
|
|
|
|
import GLib from 'gi://GLib';
|
|
|
|
import GObject from 'gi://GObject';
|
|
|
|
import Meta from 'gi://Meta';
|
|
|
|
import Shell from 'gi://Shell';
|
|
|
|
import St from 'gi://St';
|
|
|
|
import * as Signals from '../misc/signals.js';
|
2009-02-02 23:02:16 +00:00
|
|
|
|
2020-06-24 15:28:53 +02:00
|
|
|
// Time for initial animation going into Overview mode;
|
|
|
|
// this is defined here to make it available in imports.
|
2023-07-10 02:53:00 -07:00
|
|
|
export const ANIMATION_TIME = 250;
|
2020-06-24 15:28:53 +02:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
import * as DND from './dnd.js';
|
|
|
|
import * as LayoutManager from './layout.js';
|
|
|
|
import * as Main from './main.js';
|
|
|
|
import * as MessageTray from './messageTray.js';
|
|
|
|
import * as OverviewControls from './overviewControls.js';
|
|
|
|
import * as Params from '../misc/params.js';
|
|
|
|
import * as SwipeTracker from './swipeTracker.js';
|
|
|
|
import * as WindowManager from './windowManager.js';
|
|
|
|
import * as WorkspaceThumbnail from './workspaceThumbnail.js';
|
2008-10-31 23:09:46 +00:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
const DND_WINDOW_SWITCH_TIMEOUT = 750;
|
2012-08-30 02:22:39 +02:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
const OVERVIEW_ACTIVATION_TIMEOUT = 0.5;
|
2013-03-01 16:00:37 -05:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
class ShellInfo {
|
2017-10-31 02:19:44 +01:00
|
|
|
constructor() {
|
2010-07-22 17:49:43 +02:00
|
|
|
this._source = null;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2010-02-09 01:50:50 +03:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
setMessage(text, options) {
|
2019-02-12 15:02:09 +01:00
|
|
|
options = Params.parse(options, {
|
|
|
|
undoCallback: null,
|
|
|
|
forFeedback: false,
|
|
|
|
});
|
2012-11-02 18:06:40 +01:00
|
|
|
|
|
|
|
let undoCallback = options.undoCallback;
|
|
|
|
let forFeedback = options.forFeedback;
|
|
|
|
|
2010-07-22 17:49:43 +02:00
|
|
|
if (this._source == null) {
|
2010-11-30 10:47:28 -05:00
|
|
|
this._source = new MessageTray.SystemNotificationSource();
|
2017-10-31 01:38:18 +01:00
|
|
|
this._source.connect('destroy', () => {
|
|
|
|
this._source = null;
|
|
|
|
});
|
2010-07-22 17:49:43 +02:00
|
|
|
Main.messageTray.add(this._source);
|
|
|
|
}
|
2010-02-09 01:50:50 +03:00
|
|
|
|
2011-03-21 17:43:34 -04:00
|
|
|
let notification = null;
|
|
|
|
if (this._source.notifications.length == 0) {
|
2010-07-22 17:49:43 +02:00
|
|
|
notification = new MessageTray.Notification(this._source, text, null);
|
2012-08-20 16:14:22 +02:00
|
|
|
notification.setTransient(true);
|
2012-11-02 18:06:40 +01:00
|
|
|
notification.setForFeedback(forFeedback);
|
2011-03-21 17:43:34 -04:00
|
|
|
} else {
|
|
|
|
notification = this._source.notifications[0];
|
2023-08-07 00:40:20 +02:00
|
|
|
notification.update(text, null, {clear: true});
|
2011-03-21 17:43:34 -04:00
|
|
|
}
|
2010-03-10 15:14:56 +01:00
|
|
|
|
2013-10-13 23:06:09 -04:00
|
|
|
if (undoCallback)
|
2021-07-08 19:23:38 +02:00
|
|
|
notification.addAction(_('Undo'), () => undoCallback());
|
2010-07-22 17:49:43 +02:00
|
|
|
|
2019-05-13 23:32:31 +02:00
|
|
|
this._source.showNotification(notification);
|
2010-02-09 01:50:50 +03:00
|
|
|
}
|
2023-07-10 02:53:00 -07:00
|
|
|
}
|
2010-02-09 01:50:50 +03:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
const OverviewActor = GObject.registerClass(
|
2019-08-31 03:51:02 +02:00
|
|
|
class OverviewActor extends St.BoxLayout {
|
|
|
|
_init() {
|
|
|
|
super._init({
|
|
|
|
name: 'overview',
|
|
|
|
/* Translators: This is the main view to select
|
|
|
|
activities. See also note for "Activities" string. */
|
2023-08-07 00:34:20 +02:00
|
|
|
accessible_name: _('Overview'),
|
2019-08-20 23:43:54 +02:00
|
|
|
vertical: true,
|
2019-08-31 03:51:02 +02:00
|
|
|
});
|
|
|
|
|
2023-08-07 00:40:20 +02:00
|
|
|
this.add_constraint(new LayoutManager.MonitorConstraint({primary: true}));
|
2019-08-31 03:51:02 +02:00
|
|
|
|
2020-12-30 17:02:14 -03:00
|
|
|
this._controls = new OverviewControls.ControlsManager();
|
2019-10-21 20:44:00 +02:00
|
|
|
this.add_child(this._controls);
|
2019-08-31 03:51:02 +02:00
|
|
|
}
|
|
|
|
|
2022-11-21 12:39:19 +01:00
|
|
|
prepareToEnterOverview() {
|
|
|
|
this._controls.prepareToEnterOverview();
|
|
|
|
}
|
|
|
|
|
|
|
|
prepareToLeaveOverview() {
|
|
|
|
this._controls.prepareToLeaveOverview();
|
|
|
|
}
|
|
|
|
|
2021-01-02 17:19:37 -03:00
|
|
|
animateToOverview(state, callback) {
|
|
|
|
this._controls.animateToOverview(state, callback);
|
2021-01-01 13:39:07 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
animateFromOverview(callback) {
|
2021-01-01 14:18:18 -03:00
|
|
|
this._controls.animateFromOverview(callback);
|
2021-01-01 13:39:07 -03:00
|
|
|
}
|
|
|
|
|
2021-02-15 17:39:14 -03:00
|
|
|
runStartupAnimation(callback) {
|
|
|
|
this._controls.runStartupAnimation(callback);
|
|
|
|
}
|
|
|
|
|
2019-08-31 03:51:02 +02:00
|
|
|
get dash() {
|
|
|
|
return this._controls.dash;
|
|
|
|
}
|
|
|
|
|
2023-06-01 15:54:59 -07:00
|
|
|
get searchController() {
|
|
|
|
return this._controls.searchController;
|
|
|
|
}
|
|
|
|
|
2019-08-31 03:51:02 +02:00
|
|
|
get searchEntry() {
|
2020-12-30 17:02:14 -03:00
|
|
|
return this._controls.searchEntry;
|
2019-08-31 03:51:02 +02:00
|
|
|
}
|
|
|
|
|
2021-01-04 16:23:45 +01:00
|
|
|
get controls() {
|
|
|
|
return this._controls;
|
|
|
|
}
|
2019-08-31 03:51:02 +02:00
|
|
|
});
|
|
|
|
|
2022-03-16 22:36:49 +01:00
|
|
|
const OverviewShownState = {
|
|
|
|
HIDDEN: 'HIDDEN',
|
|
|
|
HIDING: 'HIDING',
|
|
|
|
SHOWING: 'SHOWING',
|
|
|
|
SHOWN: 'SHOWN',
|
|
|
|
};
|
|
|
|
|
|
|
|
const OVERVIEW_SHOWN_TRANSITIONS = {
|
|
|
|
[OverviewShownState.HIDDEN]: {
|
|
|
|
signal: 'hidden',
|
|
|
|
allowedTransitions: [OverviewShownState.SHOWING],
|
|
|
|
},
|
|
|
|
[OverviewShownState.HIDING]: {
|
|
|
|
signal: 'hiding',
|
|
|
|
allowedTransitions:
|
|
|
|
[OverviewShownState.HIDDEN, OverviewShownState.SHOWING],
|
|
|
|
},
|
|
|
|
[OverviewShownState.SHOWING]: {
|
|
|
|
signal: 'showing',
|
|
|
|
allowedTransitions:
|
|
|
|
[OverviewShownState.SHOWN, OverviewShownState.HIDING],
|
|
|
|
},
|
|
|
|
[OverviewShownState.SHOWN]: {
|
|
|
|
signal: 'shown',
|
|
|
|
allowedTransitions: [OverviewShownState.HIDING],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
export class Overview extends Signals.EventEmitter {
|
2017-10-31 02:19:44 +01:00
|
|
|
constructor() {
|
2022-07-04 18:30:44 -04:00
|
|
|
super();
|
|
|
|
|
2012-11-29 17:58:31 +01:00
|
|
|
this._initCalled = false;
|
2020-01-29 17:48:57 +01:00
|
|
|
this._visible = false;
|
2012-09-01 09:42:53 -03:00
|
|
|
|
2017-12-02 01:27:35 +01:00
|
|
|
Main.sessionMode.connect('updated', this._sessionUpdated.bind(this));
|
2012-09-01 09:42:53 -03:00
|
|
|
this._sessionUpdated();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-08-28 10:24:48 -04:00
|
|
|
|
2019-08-31 03:51:02 +02:00
|
|
|
get dash() {
|
|
|
|
return this._overview.dash;
|
|
|
|
}
|
|
|
|
|
|
|
|
get dashIconSize() {
|
|
|
|
logError(new Error('Usage of Overview.\'dashIconSize\' is deprecated, ' +
|
|
|
|
'use \'dash.iconSize\' property instead'));
|
|
|
|
return this.dash.iconSize;
|
|
|
|
}
|
|
|
|
|
2019-08-31 04:04:34 +02:00
|
|
|
get animationInProgress() {
|
|
|
|
return this._animationInProgress;
|
|
|
|
}
|
|
|
|
|
|
|
|
get visible() {
|
|
|
|
return this._visible;
|
|
|
|
}
|
|
|
|
|
|
|
|
get visibleTarget() {
|
|
|
|
return this._visibleTarget;
|
|
|
|
}
|
|
|
|
|
2020-09-16 14:52:19 +08:00
|
|
|
get closing() {
|
|
|
|
return this._animationInProgress && !this._visibleTarget;
|
|
|
|
}
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_createOverview() {
|
2019-08-31 03:51:02 +02:00
|
|
|
if (this._overview)
|
2011-08-28 10:24:48 -04:00
|
|
|
return;
|
2012-09-01 09:42:53 -03:00
|
|
|
|
|
|
|
if (this.isDummy)
|
|
|
|
return;
|
|
|
|
|
2013-03-01 16:00:37 -05:00
|
|
|
this._activationTime = 0;
|
|
|
|
|
2019-08-31 04:04:34 +02:00
|
|
|
this._visible = false; // animating to overview, in overview, animating out
|
2011-01-05 15:39:36 +01:00
|
|
|
this._shown = false; // show() and not hide()
|
|
|
|
this._modal = false; // have a modal grab
|
2019-08-31 04:04:34 +02:00
|
|
|
this._animationInProgress = false;
|
|
|
|
this._visibleTarget = false;
|
2022-03-16 22:36:49 +01:00
|
|
|
this._shownState = OverviewShownState.HIDDEN;
|
2009-06-24 18:24:48 -04:00
|
|
|
|
2009-09-13 14:54:56 -04:00
|
|
|
// During transitions, we raise this to the top to avoid having the overview
|
|
|
|
// area be reactive; it causes too many issues such as double clicks on
|
|
|
|
// Dash elements, or mouseover handlers in the workspaces.
|
2020-03-29 23:51:13 +02:00
|
|
|
this._coverPane = new Clutter.Actor({
|
|
|
|
opacity: 0,
|
|
|
|
reactive: true,
|
|
|
|
});
|
2013-09-29 23:20:27 +02:00
|
|
|
Main.layoutManager.overviewGroup.add_child(this._coverPane);
|
2022-01-12 17:19:53 +01:00
|
|
|
this._coverPane.connect('event', (_actor, event) => {
|
|
|
|
return event.type() === Clutter.EventType.ENTER ||
|
|
|
|
event.type() === Clutter.EventType.LEAVE
|
|
|
|
? Clutter.EVENT_PROPAGATE : Clutter.EVENT_STOP;
|
|
|
|
});
|
2011-02-11 23:53:27 +01:00
|
|
|
this._coverPane.hide();
|
|
|
|
|
|
|
|
// XDND
|
|
|
|
this._dragMonitor = {
|
2019-08-20 23:43:54 +02:00
|
|
|
dragMotion: this._onDragMotion.bind(this),
|
2011-02-11 23:53:27 +01:00
|
|
|
};
|
|
|
|
|
2016-07-02 11:16:22 +02:00
|
|
|
|
|
|
|
Main.layoutManager.overviewGroup.connect('scroll-event',
|
2017-12-02 01:27:35 +01:00
|
|
|
this._onScrollEvent.bind(this));
|
|
|
|
Main.xdndHandler.connect('drag-begin', this._onDragBegin.bind(this));
|
|
|
|
Main.xdndHandler.connect('drag-end', this._onDragEnd.bind(this));
|
2011-02-11 23:53:27 +01:00
|
|
|
|
2018-01-03 15:55:38 +08:00
|
|
|
global.display.connect('restacked', this._onRestacked.bind(this));
|
2012-12-13 11:00:30 -05:00
|
|
|
|
2011-02-11 23:53:27 +01:00
|
|
|
this._windowSwitchTimeoutId = 0;
|
|
|
|
this._windowSwitchTimestamp = 0;
|
|
|
|
this._lastActiveWorkspaceIndex = -1;
|
|
|
|
this._lastHoveredWindow = null;
|
2012-11-29 17:58:31 +01:00
|
|
|
|
|
|
|
if (this._initCalled)
|
|
|
|
this.init();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-02-11 23:53:27 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_sessionUpdated() {
|
2023-08-07 00:40:20 +02:00
|
|
|
const {hasOverview} = Main.sessionMode;
|
2020-02-24 11:19:28 +01:00
|
|
|
if (!hasOverview)
|
|
|
|
this.hide();
|
|
|
|
|
|
|
|
this.isDummy = !hasOverview;
|
2012-09-01 09:42:53 -03:00
|
|
|
this._createOverview();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2012-09-01 09:42:53 -03:00
|
|
|
|
2011-02-11 23:53:27 +01:00
|
|
|
// The members we construct that are implemented in JS might
|
|
|
|
// want to access the overview as Main.overview to connect
|
|
|
|
// signal handlers and so forth. So we create them after
|
|
|
|
// construction in this init() method.
|
2017-10-31 01:03:21 +01:00
|
|
|
init() {
|
2012-11-29 17:58:31 +01:00
|
|
|
this._initCalled = true;
|
|
|
|
|
2011-08-28 10:24:48 -04:00
|
|
|
if (this.isDummy)
|
|
|
|
return;
|
|
|
|
|
2019-08-31 03:51:02 +02:00
|
|
|
this._overview = new OverviewActor();
|
|
|
|
this._overview._delegate = this;
|
|
|
|
Main.layoutManager.overviewGroup.add_child(this._overview);
|
2012-12-10 15:39:15 -05:00
|
|
|
|
2019-08-31 03:51:02 +02:00
|
|
|
this._shellInfo = new ShellInfo();
|
2013-01-24 16:34:06 -05:00
|
|
|
|
2017-12-02 01:27:35 +01:00
|
|
|
Main.layoutManager.connect('monitors-changed', this._relayout.bind(this));
|
2011-06-13 09:54:05 -04:00
|
|
|
this._relayout();
|
2021-01-02 17:10:36 -03:00
|
|
|
|
|
|
|
Main.wm.addKeybinding(
|
|
|
|
'toggle-overview',
|
2023-08-07 00:40:20 +02:00
|
|
|
new Gio.Settings({schema_id: WindowManager.SHELL_KEYBINDINGS_SCHEMA}),
|
2021-01-02 17:10:36 -03:00
|
|
|
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
|
|
|
Shell.ActionMode.NORMAL | Shell.ActionMode.OVERVIEW,
|
|
|
|
this.toggle.bind(this));
|
2021-01-04 16:23:45 +01:00
|
|
|
|
|
|
|
const swipeTracker = new SwipeTracker.SwipeTracker(global.stage,
|
2021-03-01 21:53:55 +05:00
|
|
|
Clutter.Orientation.VERTICAL,
|
2021-01-04 16:23:45 +01:00
|
|
|
Shell.ActionMode.NORMAL | Shell.ActionMode.OVERVIEW,
|
2023-08-07 00:40:20 +02:00
|
|
|
{allowDrag: false, allowScroll: false});
|
2021-01-04 16:23:45 +01:00
|
|
|
swipeTracker.orientation = Clutter.Orientation.VERTICAL;
|
|
|
|
swipeTracker.connect('begin', this._gestureBegin.bind(this));
|
|
|
|
swipeTracker.connect('update', this._gestureUpdate.bind(this));
|
|
|
|
swipeTracker.connect('end', this._gestureEnd.bind(this));
|
|
|
|
this._swipeTracker = swipeTracker;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2009-06-24 18:24:48 -04:00
|
|
|
|
2012-11-02 18:06:40 +01:00
|
|
|
//
|
|
|
|
// options:
|
|
|
|
// - undoCallback (function): the callback to be called if undo support is needed
|
|
|
|
// - forFeedback (boolean): whether the message is for direct feedback of a user action
|
|
|
|
//
|
2017-10-31 01:03:21 +01:00
|
|
|
setMessage(text, options) {
|
2011-08-28 10:24:48 -04:00
|
|
|
if (this.isDummy)
|
|
|
|
return;
|
|
|
|
|
2012-11-02 18:06:40 +01:00
|
|
|
this._shellInfo.setMessage(text, options);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-08-28 10:07:44 -04:00
|
|
|
|
2022-03-16 22:36:49 +01:00
|
|
|
_changeShownState(state) {
|
|
|
|
const {allowedTransitions} =
|
|
|
|
OVERVIEW_SHOWN_TRANSITIONS[this._shownState];
|
|
|
|
|
|
|
|
if (!allowedTransitions.includes(state)) {
|
|
|
|
throw new Error('Invalid overview shown transition from ' +
|
|
|
|
`${this._shownState} to ${state}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._shownState = state;
|
|
|
|
this.emit(OVERVIEW_SHOWN_TRANSITIONS[state].signal);
|
|
|
|
}
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_onDragBegin() {
|
2013-03-04 17:02:02 -05:00
|
|
|
this._inXdndDrag = true;
|
|
|
|
|
2011-01-05 15:47:27 +01:00
|
|
|
DND.addDragMonitor(this._dragMonitor);
|
|
|
|
// Remember the workspace we started from
|
2018-01-03 15:55:38 +08:00
|
|
|
let workspaceManager = global.workspace_manager;
|
|
|
|
this._lastActiveWorkspaceIndex = workspaceManager.get_active_workspace_index();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-01-05 15:47:27 +01:00
|
|
|
|
2022-02-23 16:23:02 +01:00
|
|
|
_onDragEnd() {
|
2013-03-04 17:02:02 -05:00
|
|
|
this._inXdndDrag = false;
|
|
|
|
|
2011-01-05 15:47:27 +01:00
|
|
|
// In case the drag was canceled while in the overview
|
|
|
|
// we have to go back to where we started and hide
|
|
|
|
// the overview
|
2013-03-04 17:02:02 -05:00
|
|
|
if (this._shown) {
|
2018-01-03 15:55:38 +08:00
|
|
|
let workspaceManager = global.workspace_manager;
|
2022-02-23 16:23:02 +01:00
|
|
|
workspaceManager.get_workspace_by_index(this._lastActiveWorkspaceIndex)
|
|
|
|
.activate(global.get_current_time());
|
2013-03-04 17:02:02 -05:00
|
|
|
this.hide();
|
2011-01-05 15:47:27 +01:00
|
|
|
}
|
2011-02-24 11:30:46 +01:00
|
|
|
this._resetWindowSwitchTimeout();
|
2011-01-07 11:12:00 +01:00
|
|
|
this._lastHoveredWindow = null;
|
2011-06-27 18:59:56 +02:00
|
|
|
DND.removeDragMonitor(this._dragMonitor);
|
2011-03-13 19:34:47 +01:00
|
|
|
this.endItemDrag();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-01-05 15:47:27 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_resetWindowSwitchTimeout() {
|
2011-02-24 11:30:46 +01:00
|
|
|
if (this._windowSwitchTimeoutId != 0) {
|
2019-08-19 20:50:33 +02:00
|
|
|
GLib.source_remove(this._windowSwitchTimeoutId);
|
2011-02-24 11:30:46 +01:00
|
|
|
this._windowSwitchTimeoutId = 0;
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-02-24 11:30:46 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_onDragMotion(dragEvent) {
|
2011-01-07 11:12:00 +01:00
|
|
|
let targetIsWindow = dragEvent.targetActor &&
|
|
|
|
dragEvent.targetActor._delegate &&
|
2011-03-21 23:20:36 +01:00
|
|
|
dragEvent.targetActor._delegate.metaWindow &&
|
|
|
|
!(dragEvent.targetActor._delegate instanceof WorkspaceThumbnail.WindowClone);
|
2011-01-07 11:12:00 +01:00
|
|
|
|
2011-02-24 11:25:29 +01:00
|
|
|
this._windowSwitchTimestamp = global.get_current_time();
|
|
|
|
|
2011-01-07 11:12:00 +01:00
|
|
|
if (targetIsWindow &&
|
|
|
|
dragEvent.targetActor._delegate.metaWindow == this._lastHoveredWindow)
|
2011-01-20 16:46:55 +01:00
|
|
|
return DND.DragMotionResult.CONTINUE;
|
|
|
|
|
|
|
|
this._lastHoveredWindow = null;
|
2011-01-07 11:12:00 +01:00
|
|
|
|
2011-02-24 11:30:46 +01:00
|
|
|
this._resetWindowSwitchTimeout();
|
2011-01-05 15:47:27 +01:00
|
|
|
|
2011-01-07 11:12:00 +01:00
|
|
|
if (targetIsWindow) {
|
|
|
|
this._lastHoveredWindow = dragEvent.targetActor._delegate.metaWindow;
|
2019-08-19 20:50:33 +02:00
|
|
|
this._windowSwitchTimeoutId = GLib.timeout_add(
|
|
|
|
GLib.PRIORITY_DEFAULT,
|
|
|
|
DND_WINDOW_SWITCH_TIMEOUT,
|
2017-10-31 01:38:18 +01:00
|
|
|
() => {
|
|
|
|
this._windowSwitchTimeoutId = 0;
|
|
|
|
Main.activateWindow(dragEvent.targetActor._delegate.metaWindow,
|
|
|
|
this._windowSwitchTimestamp);
|
|
|
|
this.hide();
|
|
|
|
this._lastHoveredWindow = null;
|
|
|
|
return GLib.SOURCE_REMOVE;
|
|
|
|
});
|
2014-04-10 19:26:52 +02:00
|
|
|
GLib.Source.set_name_by_id(this._windowSwitchTimeoutId, '[gnome-shell] Main.activateWindow');
|
2011-01-05 15:47:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return DND.DragMotionResult.CONTINUE;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-01-05 15:47:27 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_onScrollEvent(actor, event) {
|
2012-10-22 17:39:19 +02:00
|
|
|
this.emit('scroll-event', event);
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2012-10-22 17:39:19 +02:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_relayout() {
|
2011-06-13 09:54:05 -04:00
|
|
|
// To avoid updating the position and size of the workspaces
|
|
|
|
// we just hide the overview. The positions will be updated
|
|
|
|
// when it is next shown.
|
|
|
|
this.hide();
|
|
|
|
|
2018-07-24 14:45:03 +02:00
|
|
|
this._coverPane.set_position(0, 0);
|
|
|
|
this._coverPane.set_size(global.screen_width, global.screen_height);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2008-12-01 19:51:43 +00:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_onRestacked() {
|
2012-12-13 11:00:30 -05:00
|
|
|
let stack = global.get_window_actors();
|
|
|
|
let stackIndices = {};
|
|
|
|
|
|
|
|
for (let i = 0; i < stack.length; i++) {
|
|
|
|
// Use the stable sequence for an integer to use as a hash key
|
|
|
|
stackIndices[stack[i].get_meta_window().get_stable_sequence()] = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.emit('windows-restacked', stackIndices);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2012-12-13 11:00:30 -05:00
|
|
|
|
2021-01-04 16:23:45 +01:00
|
|
|
_gestureBegin(tracker) {
|
|
|
|
this._overview.controls.gestureBegin(tracker);
|
|
|
|
}
|
|
|
|
|
|
|
|
_gestureUpdate(tracker, progress) {
|
|
|
|
if (!this._shown) {
|
|
|
|
Meta.disable_unredirect_for_display(global.display);
|
|
|
|
|
|
|
|
this._shown = true;
|
|
|
|
this._visible = true;
|
|
|
|
this._visibleTarget = true;
|
|
|
|
this._animationInProgress = true;
|
|
|
|
|
|
|
|
Main.layoutManager.overviewGroup.set_child_above_sibling(
|
|
|
|
this._coverPane, null);
|
|
|
|
this._coverPane.show();
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.SHOWING);
|
2021-01-04 16:23:45 +01:00
|
|
|
|
|
|
|
Main.layoutManager.showOverview();
|
|
|
|
this._syncGrab();
|
|
|
|
}
|
|
|
|
|
|
|
|
this._overview.controls.gestureProgress(progress);
|
|
|
|
}
|
|
|
|
|
|
|
|
_gestureEnd(tracker, duration, endProgress) {
|
|
|
|
let onComplete;
|
|
|
|
if (endProgress === 0) {
|
|
|
|
this._shown = false;
|
2021-02-26 22:55:57 +01:00
|
|
|
this._visibleTarget = false;
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.HIDING);
|
2022-02-07 15:14:06 +01:00
|
|
|
Main.panel.style = `transition-duration: ${duration}ms;`;
|
2021-01-04 16:23:45 +01:00
|
|
|
onComplete = () => this._hideDone();
|
|
|
|
} else {
|
|
|
|
onComplete = () => this._showDone();
|
|
|
|
}
|
|
|
|
|
|
|
|
this._overview.controls.gestureEnd(endProgress, duration, onComplete);
|
|
|
|
}
|
|
|
|
|
2020-06-23 10:30:33 -03:00
|
|
|
beginItemDrag(source) {
|
|
|
|
this.emit('item-drag-begin', source);
|
2017-11-16 17:27:12 +01:00
|
|
|
this._inItemDrag = true;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2009-03-11 15:21:45 -04:00
|
|
|
|
2020-06-23 10:30:33 -03:00
|
|
|
cancelledItemDrag(source) {
|
|
|
|
this.emit('item-drag-cancelled', source);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-03-09 16:40:48 +01:00
|
|
|
|
2020-06-23 10:30:33 -03:00
|
|
|
endItemDrag(source) {
|
2017-11-16 17:27:12 +01:00
|
|
|
if (!this._inItemDrag)
|
|
|
|
return;
|
2020-06-23 10:30:33 -03:00
|
|
|
this.emit('item-drag-end', source);
|
2017-11-16 17:27:12 +01:00
|
|
|
this._inItemDrag = false;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2009-03-11 15:21:45 -04:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
beginWindowDrag(window) {
|
2014-09-03 21:55:05 +02:00
|
|
|
this.emit('window-drag-begin', window);
|
2017-11-16 17:27:12 +01:00
|
|
|
this._inWindowDrag = true;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2010-11-15 21:58:27 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
cancelledWindowDrag(window) {
|
2014-09-03 21:55:05 +02:00
|
|
|
this.emit('window-drag-cancelled', window);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-03-08 14:44:47 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
endWindowDrag(window) {
|
2017-11-16 17:27:12 +01:00
|
|
|
if (!this._inWindowDrag)
|
|
|
|
return;
|
2014-09-03 21:55:05 +02:00
|
|
|
this.emit('window-drag-end', window);
|
2017-11-16 17:27:12 +01:00
|
|
|
this._inWindowDrag = false;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2010-11-15 21:58:27 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
focusSearch() {
|
2013-05-17 17:31:27 +02:00
|
|
|
this.show();
|
2019-08-31 03:51:02 +02:00
|
|
|
this._overview.searchEntry.grab_key_focus();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2013-05-17 17:31:27 +02:00
|
|
|
|
2014-07-03 15:33:09 +02:00
|
|
|
// Checks if the Activities button is currently sensitive to
|
|
|
|
// clicks. The first call to this function within the
|
|
|
|
// OVERVIEW_ACTIVATION_TIMEOUT time of the hot corner being
|
|
|
|
// triggered will return false. This avoids opening and closing
|
|
|
|
// the overview if the user both triggered the hot corner and
|
|
|
|
// clicked the Activities button.
|
2017-10-31 01:03:21 +01:00
|
|
|
shouldToggleByCornerOrButton() {
|
2019-08-31 04:04:34 +02:00
|
|
|
if (this._animationInProgress)
|
2014-07-03 15:33:09 +02:00
|
|
|
return false;
|
2017-11-16 17:27:12 +01:00
|
|
|
if (this._inItemDrag || this._inWindowDrag)
|
2014-07-03 15:33:09 +02:00
|
|
|
return false;
|
2019-08-31 03:51:02 +02:00
|
|
|
if (!this._activationTime ||
|
2018-01-15 15:36:44 +01:00
|
|
|
GLib.get_monotonic_time() / GLib.USEC_PER_SEC - this._activationTime > OVERVIEW_ACTIVATION_TIMEOUT)
|
2014-07-03 15:33:09 +02:00
|
|
|
return true;
|
|
|
|
return false;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2014-07-03 15:33:09 +02:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_syncGrab() {
|
2014-07-03 15:33:09 +02:00
|
|
|
// We delay grab changes during animation so that when removing the
|
|
|
|
// overview we don't have a problem with the release of a press/release
|
|
|
|
// going to an application.
|
2019-08-31 04:04:34 +02:00
|
|
|
if (this._animationInProgress)
|
2014-07-03 15:33:09 +02:00
|
|
|
return true;
|
|
|
|
|
|
|
|
if (this._shown) {
|
|
|
|
let shouldBeModal = !this._inXdndDrag;
|
2019-08-20 04:25:24 +02:00
|
|
|
if (shouldBeModal && !this._modal) {
|
2022-09-29 14:23:31 +02:00
|
|
|
if (global.display.is_grabbed()) {
|
2022-04-26 13:09:01 +02:00
|
|
|
this.hide();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const grab = Main.pushModal(global.stage, {
|
|
|
|
actionMode: Shell.ActionMode.OVERVIEW,
|
|
|
|
});
|
2022-04-30 08:18:39 +02:00
|
|
|
if (grab.get_seat_state() !== Clutter.GrabState.ALL) {
|
2021-11-25 10:49:42 +01:00
|
|
|
Main.popModal(grab);
|
2019-08-20 04:25:24 +02:00
|
|
|
this.hide();
|
|
|
|
return false;
|
2014-07-03 15:33:09 +02:00
|
|
|
}
|
2022-04-30 08:18:39 +02:00
|
|
|
|
|
|
|
this._grab = grab;
|
|
|
|
this._modal = true;
|
2014-07-03 15:33:09 +02:00
|
|
|
}
|
|
|
|
} else {
|
2019-08-20 04:24:37 +02:00
|
|
|
// eslint-disable-next-line no-lonely-if
|
2014-07-03 15:33:09 +02:00
|
|
|
if (this._modal) {
|
2021-11-25 10:49:42 +01:00
|
|
|
Main.popModal(this._grab);
|
|
|
|
this._grab = false;
|
2014-07-03 15:33:09 +02:00
|
|
|
this._modal = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2014-07-03 15:33:09 +02:00
|
|
|
|
|
|
|
// show:
|
|
|
|
//
|
|
|
|
// Animates the overview visible and grabs mouse and keyboard input
|
2021-01-02 17:19:37 -03:00
|
|
|
show(state = OverviewControls.ControlsState.WINDOW_PICKER) {
|
|
|
|
if (state === OverviewControls.ControlsState.HIDDEN)
|
|
|
|
throw new Error('Invalid state, use hide() to hide');
|
|
|
|
|
2014-07-03 15:33:09 +02:00
|
|
|
if (this.isDummy)
|
|
|
|
return;
|
|
|
|
if (this._shown)
|
|
|
|
return;
|
|
|
|
this._shown = true;
|
|
|
|
|
|
|
|
if (!this._syncGrab())
|
|
|
|
return;
|
|
|
|
|
|
|
|
Main.layoutManager.showOverview();
|
2021-01-02 17:19:37 -03:00
|
|
|
this._animateVisible(state);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2014-07-03 15:33:09 +02:00
|
|
|
|
|
|
|
|
2021-01-02 17:19:37 -03:00
|
|
|
_animateVisible(state) {
|
2019-08-31 04:04:34 +02:00
|
|
|
if (this._visible || this._animationInProgress)
|
2011-01-05 15:39:36 +01:00
|
|
|
return;
|
2008-11-28 20:12:20 +00:00
|
|
|
|
2019-08-31 04:04:34 +02:00
|
|
|
this._visible = true;
|
|
|
|
this._animationInProgress = true;
|
|
|
|
this._visibleTarget = true;
|
2018-01-15 15:36:44 +01:00
|
|
|
this._activationTime = GLib.get_monotonic_time() / GLib.USEC_PER_SEC;
|
2008-11-28 20:12:20 +00:00
|
|
|
|
2018-01-03 15:55:38 +08:00
|
|
|
Meta.disable_unredirect_for_display(global.display);
|
2010-07-15 16:21:32 +02:00
|
|
|
|
2019-11-05 20:17:19 +01:00
|
|
|
Main.layoutManager.overviewGroup.set_child_above_sibling(
|
|
|
|
this._coverPane, null);
|
2011-01-21 14:08:42 +01:00
|
|
|
this._coverPane.show();
|
2022-11-06 12:17:15 +01:00
|
|
|
|
|
|
|
this._overview.prepareToEnterOverview();
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.SHOWING);
|
2022-11-06 12:17:15 +01:00
|
|
|
this._overview.animateToOverview(state, () => this._showDone());
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2008-12-01 19:51:43 +00:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_showDone() {
|
2019-08-31 04:04:34 +02:00
|
|
|
this._animationInProgress = false;
|
2014-07-03 15:33:09 +02:00
|
|
|
this._coverPane.hide();
|
|
|
|
|
2022-12-20 13:49:59 +01:00
|
|
|
if (this._shownState !== OverviewShownState.SHOWN)
|
|
|
|
this._changeShownState(OverviewShownState.SHOWN);
|
|
|
|
|
2014-07-03 15:33:09 +02:00
|
|
|
// Handle any calls to hide* while we were showing
|
|
|
|
if (!this._shown)
|
|
|
|
this._animateNotVisible();
|
|
|
|
|
|
|
|
this._syncGrab();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2014-07-03 15:33:09 +02:00
|
|
|
|
2011-01-05 15:39:36 +01:00
|
|
|
// hide:
|
|
|
|
//
|
|
|
|
// Reverses the effect of show()
|
2017-10-31 01:03:21 +01:00
|
|
|
hide() {
|
2011-08-28 10:24:48 -04:00
|
|
|
if (this.isDummy)
|
|
|
|
return;
|
|
|
|
|
2011-01-05 15:39:36 +01:00
|
|
|
if (!this._shown)
|
|
|
|
return;
|
|
|
|
|
2013-03-04 17:54:02 -05:00
|
|
|
let event = Clutter.get_current_event();
|
2013-03-14 11:34:09 -04:00
|
|
|
if (event) {
|
|
|
|
let type = event.type();
|
2019-08-19 21:38:51 +02:00
|
|
|
let button = type == Clutter.EventType.BUTTON_PRESS ||
|
|
|
|
type == Clutter.EventType.BUTTON_RELEASE;
|
2013-03-14 11:34:09 -04:00
|
|
|
let ctrl = (event.get_state() & Clutter.ModifierType.CONTROL_MASK) != 0;
|
|
|
|
if (button && ctrl)
|
|
|
|
return;
|
|
|
|
}
|
2013-02-21 15:57:00 +01:00
|
|
|
|
2011-01-05 15:39:36 +01:00
|
|
|
this._shown = false;
|
2014-11-26 18:05:36 -08:00
|
|
|
|
|
|
|
this._animateNotVisible();
|
2013-05-22 18:05:24 +02:00
|
|
|
this._syncGrab();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-01-05 15:39:36 +01:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_animateNotVisible() {
|
2019-08-31 04:04:34 +02:00
|
|
|
if (!this._visible || this._animationInProgress)
|
2008-12-01 19:51:43 +00:00
|
|
|
return;
|
|
|
|
|
2019-08-31 04:04:34 +02:00
|
|
|
this._animationInProgress = true;
|
|
|
|
this._visibleTarget = false;
|
2010-07-15 16:21:32 +02:00
|
|
|
|
2019-11-05 20:17:19 +01:00
|
|
|
Main.layoutManager.overviewGroup.set_child_above_sibling(
|
|
|
|
this._coverPane, null);
|
2011-01-21 14:08:42 +01:00
|
|
|
this._coverPane.show();
|
2022-11-06 12:17:15 +01:00
|
|
|
|
|
|
|
this._overview.prepareToLeaveOverview();
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.HIDING);
|
2022-11-06 12:17:15 +01:00
|
|
|
this._overview.animateFromOverview(() => this._hideDone());
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2009-05-07 09:47:48 -04:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_hideDone() {
|
2011-08-29 23:55:42 +02:00
|
|
|
// Re-enable unredirection
|
2018-01-03 15:55:38 +08:00
|
|
|
Meta.enable_unredirect_for_display(global.display);
|
2011-08-29 23:55:42 +02:00
|
|
|
|
2013-05-22 18:05:24 +02:00
|
|
|
this._coverPane.hide();
|
2009-03-20 12:06:34 -04:00
|
|
|
|
2019-08-31 04:04:34 +02:00
|
|
|
this._visible = false;
|
|
|
|
this._animationInProgress = false;
|
2008-12-01 19:51:43 +00:00
|
|
|
|
2011-01-05 15:39:36 +01:00
|
|
|
// Handle any calls to show* while we were hiding
|
2020-09-17 15:24:12 +08:00
|
|
|
if (this._shown) {
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.HIDDEN);
|
2021-03-16 18:21:05 +05:30
|
|
|
this._animateVisible(OverviewControls.ControlsState.WINDOW_PICKER);
|
2020-09-17 15:24:12 +08:00
|
|
|
} else {
|
2013-05-22 18:05:24 +02:00
|
|
|
Main.layoutManager.hideOverview();
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.HIDDEN);
|
2020-09-17 15:24:12 +08:00
|
|
|
}
|
2011-01-05 15:39:36 +01:00
|
|
|
|
2021-02-17 00:57:13 +01:00
|
|
|
Main.panel.style = null;
|
|
|
|
|
2013-05-22 18:05:24 +02:00
|
|
|
this._syncGrab();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2014-07-03 15:33:09 +02:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
toggle() {
|
2014-07-03 15:33:09 +02:00
|
|
|
if (this.isDummy)
|
|
|
|
return;
|
|
|
|
|
2019-08-31 04:04:34 +02:00
|
|
|
if (this._visible)
|
2014-07-03 15:33:09 +02:00
|
|
|
this.hide();
|
|
|
|
else
|
|
|
|
this.show();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2014-06-17 19:10:54 +02:00
|
|
|
|
2021-01-02 17:28:44 -03:00
|
|
|
showApps() {
|
2021-03-14 12:07:31 +01:00
|
|
|
this.show(OverviewControls.ControlsState.APP_GRID);
|
2021-01-02 17:28:44 -03:00
|
|
|
}
|
|
|
|
|
2021-06-17 22:09:14 +02:00
|
|
|
selectApp(id) {
|
|
|
|
this.showApps();
|
|
|
|
this._overview.controls.appDisplay.selectApp(id);
|
|
|
|
}
|
|
|
|
|
2021-02-15 17:39:14 -03:00
|
|
|
runStartupAnimation(callback) {
|
2021-03-02 19:52:26 -03:00
|
|
|
Main.panel.style = 'transition-duration: 0ms;';
|
|
|
|
|
2021-02-15 17:39:14 -03:00
|
|
|
this._shown = true;
|
|
|
|
this._visible = true;
|
|
|
|
this._visibleTarget = true;
|
|
|
|
Main.layoutManager.showOverview();
|
2021-04-15 16:58:17 +02:00
|
|
|
// We should call this._syncGrab() here, but moved it to happen after
|
|
|
|
// the animation because of a race in the xserver where the grab
|
|
|
|
// fails when requested very early during startup.
|
2021-02-15 17:39:14 -03:00
|
|
|
|
|
|
|
Meta.disable_unredirect_for_display(global.display);
|
|
|
|
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.SHOWING);
|
2021-02-15 17:39:14 -03:00
|
|
|
|
|
|
|
this._overview.runStartupAnimation(() => {
|
2023-03-01 21:43:15 +01:00
|
|
|
// Overview got hidden during startup animation
|
|
|
|
if (this._shownState !== OverviewShownState.SHOWING) {
|
|
|
|
callback();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-04-15 16:58:17 +02:00
|
|
|
if (!this._syncGrab()) {
|
|
|
|
callback();
|
2023-03-01 21:48:45 +01:00
|
|
|
this.hide();
|
2021-04-15 16:58:17 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-03-02 19:52:26 -03:00
|
|
|
Main.panel.style = null;
|
2022-03-16 22:36:49 +01:00
|
|
|
this._changeShownState(OverviewShownState.SHOWN);
|
2021-02-15 17:39:14 -03:00
|
|
|
callback();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
getShowAppsButton() {
|
2019-08-31 03:51:02 +02:00
|
|
|
logError(new Error('Usage of Overview.\'getShowAppsButton\' is deprecated, ' +
|
|
|
|
'use \'dash.showAppsButton\' property instead'));
|
|
|
|
|
|
|
|
return this.dash.showAppsButton;
|
2008-12-01 19:51:43 +00:00
|
|
|
}
|
2020-01-09 17:44:22 -03:00
|
|
|
|
2023-06-01 15:54:59 -07:00
|
|
|
get searchController() {
|
|
|
|
return this._overview.searchController;
|
|
|
|
}
|
|
|
|
|
2020-01-09 17:44:22 -03:00
|
|
|
get searchEntry() {
|
|
|
|
return this._overview.searchEntry;
|
|
|
|
}
|
2023-07-10 02:53:00 -07:00
|
|
|
}
|