1928 lines
73 KiB
JavaScript
1928 lines
73 KiB
JavaScript
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
||
/* exported WindowManager */
|
||
|
||
const { Clutter, Gio, GLib, GObject, Meta, Shell, St } = imports.gi;
|
||
|
||
const AltTab = imports.ui.altTab;
|
||
const AppFavorites = imports.ui.appFavorites;
|
||
const Dialog = imports.ui.dialog;
|
||
const WorkspaceSwitcherPopup = imports.ui.workspaceSwitcherPopup;
|
||
const InhibitShortcutsDialog = imports.ui.inhibitShortcutsDialog;
|
||
const Main = imports.ui.main;
|
||
const ModalDialog = imports.ui.modalDialog;
|
||
const WindowMenu = imports.ui.windowMenu;
|
||
const PadOsd = imports.ui.padOsd;
|
||
const EdgeDragAction = imports.ui.edgeDragAction;
|
||
const CloseDialog = imports.ui.closeDialog;
|
||
const SwitchMonitor = imports.ui.switchMonitor;
|
||
const IBusManager = imports.misc.ibusManager;
|
||
const WorkspaceAnimation = imports.ui.workspaceAnimation;
|
||
|
||
const { loadInterfaceXML } = imports.misc.fileUtils;
|
||
|
||
var SHELL_KEYBINDINGS_SCHEMA = 'org.gnome.shell.keybindings';
|
||
var MINIMIZE_WINDOW_ANIMATION_TIME = 400;
|
||
var MINIMIZE_WINDOW_ANIMATION_MODE = Clutter.AnimationMode.EASE_OUT_EXPO;
|
||
var SHOW_WINDOW_ANIMATION_TIME = 150;
|
||
var DIALOG_SHOW_WINDOW_ANIMATION_TIME = 100;
|
||
var DESTROY_WINDOW_ANIMATION_TIME = 150;
|
||
var DIALOG_DESTROY_WINDOW_ANIMATION_TIME = 100;
|
||
var WINDOW_ANIMATION_TIME = 250;
|
||
var SCROLL_TIMEOUT_TIME = 150;
|
||
var DIM_BRIGHTNESS = -0.3;
|
||
var DIM_TIME = 500;
|
||
var UNDIM_TIME = 250;
|
||
var APP_MOTION_THRESHOLD = 30;
|
||
|
||
var ONE_SECOND = 1000; // in ms
|
||
|
||
var MIN_NUM_WORKSPACES = 2;
|
||
|
||
const GSD_WACOM_BUS_NAME = 'org.gnome.SettingsDaemon.Wacom';
|
||
const GSD_WACOM_OBJECT_PATH = '/org/gnome/SettingsDaemon/Wacom';
|
||
|
||
const GsdWacomIface = loadInterfaceXML('org.gnome.SettingsDaemon.Wacom');
|
||
const GsdWacomProxy = Gio.DBusProxy.makeProxyWrapper(GsdWacomIface);
|
||
|
||
const WINDOW_DIMMER_EFFECT_NAME = "gnome-shell-window-dimmer";
|
||
|
||
Gio._promisify(Shell, 'util_start_systemd_unit');
|
||
Gio._promisify(Shell, 'util_stop_systemd_unit');
|
||
|
||
var DisplayChangeDialog = GObject.registerClass(
|
||
class DisplayChangeDialog extends ModalDialog.ModalDialog {
|
||
_init(wm) {
|
||
super._init();
|
||
|
||
this._wm = wm;
|
||
|
||
this._countDown = Meta.MonitorManager.get_display_configuration_timeout();
|
||
|
||
// Translators: This string should be shorter than 30 characters
|
||
let title = _('Keep these display settings?');
|
||
let description = this._formatCountDown();
|
||
|
||
this._content = new Dialog.MessageDialogContent({ title, description });
|
||
this.contentLayout.add_child(this._content);
|
||
|
||
/* Translators: this and the following message should be limited in length,
|
||
to avoid ellipsizing the labels.
|
||
*/
|
||
this._cancelButton = this.addButton({
|
||
label: _('Revert Settings'),
|
||
action: this._onFailure.bind(this),
|
||
key: Clutter.KEY_Escape,
|
||
});
|
||
this._okButton = this.addButton({
|
||
label: _('Keep Changes'),
|
||
action: this._onSuccess.bind(this),
|
||
default: true,
|
||
});
|
||
|
||
this._timeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, ONE_SECOND, this._tick.bind(this));
|
||
GLib.Source.set_name_by_id(this._timeoutId, '[gnome-shell] this._tick');
|
||
}
|
||
|
||
close(timestamp) {
|
||
if (this._timeoutId > 0) {
|
||
GLib.source_remove(this._timeoutId);
|
||
this._timeoutId = 0;
|
||
}
|
||
|
||
super.close(timestamp);
|
||
}
|
||
|
||
_formatCountDown() {
|
||
const fmt = ngettext(
|
||
'Settings changes will revert in %d second',
|
||
'Settings changes will revert in %d seconds',
|
||
this._countDown);
|
||
return fmt.format(this._countDown);
|
||
}
|
||
|
||
_tick() {
|
||
this._countDown--;
|
||
|
||
if (this._countDown == 0) {
|
||
/* mutter already takes care of failing at timeout */
|
||
this._timeoutId = 0;
|
||
this.close();
|
||
return GLib.SOURCE_REMOVE;
|
||
}
|
||
|
||
this._content.description = this._formatCountDown();
|
||
return GLib.SOURCE_CONTINUE;
|
||
}
|
||
|
||
_onFailure() {
|
||
this._wm.complete_display_change(false);
|
||
this.close();
|
||
}
|
||
|
||
_onSuccess() {
|
||
this._wm.complete_display_change(true);
|
||
this.close();
|
||
}
|
||
});
|
||
|
||
var WindowDimmer = GObject.registerClass(
|
||
class WindowDimmer extends Clutter.BrightnessContrastEffect {
|
||
_init() {
|
||
super._init({
|
||
name: WINDOW_DIMMER_EFFECT_NAME,
|
||
enabled: false,
|
||
});
|
||
}
|
||
|
||
_syncEnabled(dimmed) {
|
||
let animating = this.actor.get_transition(`@effects.${this.name}.brightness`) !== null;
|
||
|
||
this.enabled = Meta.prefs_get_attach_modal_dialogs() && (animating || dimmed);
|
||
}
|
||
|
||
setDimmed(dimmed, animate) {
|
||
let val = 127 * (1 + (dimmed ? 1 : 0) * DIM_BRIGHTNESS);
|
||
let color = Clutter.Color.new(val, val, val, 255);
|
||
|
||
this.actor.ease_property(`@effects.${this.name}.brightness`, color, {
|
||
mode: Clutter.AnimationMode.LINEAR,
|
||
duration: (dimmed ? DIM_TIME : UNDIM_TIME) * (animate ? 1 : 0),
|
||
onStopped: () => this._syncEnabled(dimmed),
|
||
});
|
||
|
||
this._syncEnabled(dimmed);
|
||
}
|
||
});
|
||
|
||
function getWindowDimmer(actor) {
|
||
let effect = actor.get_effect(WINDOW_DIMMER_EFFECT_NAME);
|
||
|
||
if (!effect) {
|
||
effect = new WindowDimmer();
|
||
actor.add_effect(effect);
|
||
}
|
||
return effect;
|
||
}
|
||
|
||
/*
|
||
* When the last window closed on a workspace is a dialog or splash
|
||
* screen, we assume that it might be an initial window shown before
|
||
* the main window of an application, and give the app a grace period
|
||
* where it can map another window before we remove the workspace.
|
||
*/
|
||
var LAST_WINDOW_GRACE_TIME = 1000;
|
||
|
||
var WorkspaceTracker = class {
|
||
constructor(wm) {
|
||
this._wm = wm;
|
||
|
||
this._workspaces = [];
|
||
this._checkWorkspacesId = 0;
|
||
|
||
this._pauseWorkspaceCheck = false;
|
||
|
||
let tracker = Shell.WindowTracker.get_default();
|
||
tracker.connect('startup-sequence-changed', this._queueCheckWorkspaces.bind(this));
|
||
|
||
let workspaceManager = global.workspace_manager;
|
||
workspaceManager.connect('notify::n-workspaces',
|
||
this._nWorkspacesChanged.bind(this));
|
||
workspaceManager.connect('workspaces-reordered', () => {
|
||
this._workspaces.sort((a, b) => a.index() - b.index());
|
||
});
|
||
global.window_manager.connect('switch-workspace',
|
||
this._queueCheckWorkspaces.bind(this));
|
||
|
||
global.display.connect('window-entered-monitor',
|
||
this._windowEnteredMonitor.bind(this));
|
||
global.display.connect('window-left-monitor',
|
||
this._windowLeftMonitor.bind(this));
|
||
|
||
this._workspaceSettings = new Gio.Settings({ schema_id: 'org.gnome.mutter' });
|
||
this._workspaceSettings.connect('changed::dynamic-workspaces', this._queueCheckWorkspaces.bind(this));
|
||
|
||
this._nWorkspacesChanged();
|
||
}
|
||
|
||
blockUpdates() {
|
||
this._pauseWorkspaceCheck = true;
|
||
}
|
||
|
||
unblockUpdates() {
|
||
this._pauseWorkspaceCheck = false;
|
||
}
|
||
|
||
_checkWorkspaces() {
|
||
let workspaceManager = global.workspace_manager;
|
||
let i;
|
||
let emptyWorkspaces = [];
|
||
|
||
if (!Meta.prefs_get_dynamic_workspaces()) {
|
||
this._checkWorkspacesId = 0;
|
||
return false;
|
||
}
|
||
|
||
// Update workspaces only if Dynamic Workspace Management has not been paused by some other function
|
||
if (this._pauseWorkspaceCheck)
|
||
return true;
|
||
|
||
for (i = 0; i < this._workspaces.length; i++) {
|
||
let lastRemoved = this._workspaces[i]._lastRemovedWindow;
|
||
if ((lastRemoved &&
|
||
(lastRemoved.get_window_type() == Meta.WindowType.SPLASHSCREEN ||
|
||
lastRemoved.get_window_type() == Meta.WindowType.DIALOG ||
|
||
lastRemoved.get_window_type() == Meta.WindowType.MODAL_DIALOG)) ||
|
||
this._workspaces[i]._keepAliveId)
|
||
emptyWorkspaces[i] = false;
|
||
else
|
||
emptyWorkspaces[i] = true;
|
||
}
|
||
|
||
let sequences = Shell.WindowTracker.get_default().get_startup_sequences();
|
||
for (i = 0; i < sequences.length; i++) {
|
||
let index = sequences[i].get_workspace();
|
||
if (index >= 0 && index <= workspaceManager.n_workspaces)
|
||
emptyWorkspaces[index] = false;
|
||
}
|
||
|
||
let windows = global.get_window_actors();
|
||
for (i = 0; i < windows.length; i++) {
|
||
let actor = windows[i];
|
||
let win = actor.get_meta_window();
|
||
|
||
if (win.is_on_all_workspaces())
|
||
continue;
|
||
|
||
let workspaceIndex = win.get_workspace().index();
|
||
emptyWorkspaces[workspaceIndex] = false;
|
||
}
|
||
|
||
// If we don't have an empty workspace at the end, add one
|
||
if (!emptyWorkspaces[emptyWorkspaces.length - 1]) {
|
||
workspaceManager.append_new_workspace(false, global.get_current_time());
|
||
emptyWorkspaces.push(true);
|
||
}
|
||
|
||
// Enforce minimum number of workspaces
|
||
while (emptyWorkspaces.length < MIN_NUM_WORKSPACES) {
|
||
workspaceManager.append_new_workspace(false, global.get_current_time());
|
||
emptyWorkspaces.push(true);
|
||
}
|
||
|
||
let lastIndex = emptyWorkspaces.length - 1;
|
||
let lastEmptyIndex = emptyWorkspaces.lastIndexOf(false) + 1;
|
||
let activeWorkspaceIndex = workspaceManager.get_active_workspace_index();
|
||
emptyWorkspaces[activeWorkspaceIndex] = false;
|
||
|
||
// Delete empty workspaces except for the last one; do it from the end
|
||
// to avoid index changes
|
||
for (i = lastIndex; i >= 0; i--) {
|
||
if (workspaceManager.n_workspaces === MIN_NUM_WORKSPACES)
|
||
break;
|
||
if (emptyWorkspaces[i] && i != lastEmptyIndex)
|
||
workspaceManager.remove_workspace(this._workspaces[i], global.get_current_time());
|
||
}
|
||
|
||
this._checkWorkspacesId = 0;
|
||
return false;
|
||
}
|
||
|
||
keepWorkspaceAlive(workspace, duration) {
|
||
if (workspace._keepAliveId)
|
||
GLib.source_remove(workspace._keepAliveId);
|
||
|
||
workspace._keepAliveId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, duration, () => {
|
||
workspace._keepAliveId = 0;
|
||
this._queueCheckWorkspaces();
|
||
return GLib.SOURCE_REMOVE;
|
||
});
|
||
GLib.Source.set_name_by_id(workspace._keepAliveId, '[gnome-shell] this._queueCheckWorkspaces');
|
||
}
|
||
|
||
_windowRemoved(workspace, window) {
|
||
workspace._lastRemovedWindow = window;
|
||
this._queueCheckWorkspaces();
|
||
let id = GLib.timeout_add(GLib.PRIORITY_DEFAULT, LAST_WINDOW_GRACE_TIME, () => {
|
||
if (workspace._lastRemovedWindow == window) {
|
||
workspace._lastRemovedWindow = null;
|
||
this._queueCheckWorkspaces();
|
||
}
|
||
return GLib.SOURCE_REMOVE;
|
||
});
|
||
GLib.Source.set_name_by_id(id, '[gnome-shell] this._queueCheckWorkspaces');
|
||
}
|
||
|
||
_windowLeftMonitor(metaDisplay, monitorIndex, _metaWin) {
|
||
// If the window left the primary monitor, that
|
||
// might make that workspace empty
|
||
if (monitorIndex == Main.layoutManager.primaryIndex)
|
||
this._queueCheckWorkspaces();
|
||
}
|
||
|
||
_windowEnteredMonitor(metaDisplay, monitorIndex, _metaWin) {
|
||
// If the window entered the primary monitor, that
|
||
// might make that workspace non-empty
|
||
if (monitorIndex == Main.layoutManager.primaryIndex)
|
||
this._queueCheckWorkspaces();
|
||
}
|
||
|
||
_queueCheckWorkspaces() {
|
||
if (this._checkWorkspacesId == 0)
|
||
this._checkWorkspacesId = Meta.later_add(Meta.LaterType.BEFORE_REDRAW, this._checkWorkspaces.bind(this));
|
||
}
|
||
|
||
_nWorkspacesChanged() {
|
||
let workspaceManager = global.workspace_manager;
|
||
let oldNumWorkspaces = this._workspaces.length;
|
||
let newNumWorkspaces = workspaceManager.n_workspaces;
|
||
|
||
if (oldNumWorkspaces == newNumWorkspaces)
|
||
return false;
|
||
|
||
if (newNumWorkspaces > oldNumWorkspaces) {
|
||
let w;
|
||
|
||
// Assume workspaces are only added at the end
|
||
for (w = oldNumWorkspaces; w < newNumWorkspaces; w++)
|
||
this._workspaces[w] = workspaceManager.get_workspace_by_index(w);
|
||
|
||
for (w = oldNumWorkspaces; w < newNumWorkspaces; w++) {
|
||
this._workspaces[w].connectObject(
|
||
'window-added', this._queueCheckWorkspaces.bind(this),
|
||
'window-removed', this._windowRemoved.bind(this), this);
|
||
}
|
||
} else {
|
||
// Assume workspaces are only removed sequentially
|
||
// (e.g. 2,3,4 - not 2,4,7)
|
||
let removedIndex;
|
||
let removedNum = oldNumWorkspaces - newNumWorkspaces;
|
||
for (let w = 0; w < oldNumWorkspaces; w++) {
|
||
let workspace = workspaceManager.get_workspace_by_index(w);
|
||
if (this._workspaces[w] != workspace) {
|
||
removedIndex = w;
|
||
break;
|
||
}
|
||
}
|
||
|
||
let lostWorkspaces = this._workspaces.splice(removedIndex, removedNum);
|
||
lostWorkspaces.forEach(workspace => workspace.disconnectObject(this));
|
||
}
|
||
|
||
this._queueCheckWorkspaces();
|
||
|
||
return false;
|
||
}
|
||
};
|
||
|
||
var TilePreview = GObject.registerClass(
|
||
class TilePreview extends St.Widget {
|
||
_init() {
|
||
super._init();
|
||
global.window_group.add_actor(this);
|
||
|
||
this._reset();
|
||
this._showing = false;
|
||
}
|
||
|
||
open(window, tileRect, monitorIndex) {
|
||
let windowActor = window.get_compositor_private();
|
||
if (!windowActor)
|
||
return;
|
||
|
||
global.window_group.set_child_below_sibling(this, windowActor);
|
||
|
||
if (this._rect && this._rect.equal(tileRect))
|
||
return;
|
||
|
||
let changeMonitor = this._monitorIndex == -1 ||
|
||
this._monitorIndex != monitorIndex;
|
||
|
||
this._monitorIndex = monitorIndex;
|
||
this._rect = tileRect;
|
||
|
||
let monitor = Main.layoutManager.monitors[monitorIndex];
|
||
|
||
this._updateStyle(monitor);
|
||
|
||
if (!this._showing || changeMonitor) {
|
||
const monitorRect = new Meta.Rectangle({
|
||
x: monitor.x,
|
||
y: monitor.y,
|
||
width: monitor.width,
|
||
height: monitor.height,
|
||
});
|
||
let [, rect] = window.get_frame_rect().intersect(monitorRect);
|
||
this.set_size(rect.width, rect.height);
|
||
this.set_position(rect.x, rect.y);
|
||
this.opacity = 0;
|
||
}
|
||
|
||
this._showing = true;
|
||
this.show();
|
||
this.ease({
|
||
x: tileRect.x,
|
||
y: tileRect.y,
|
||
width: tileRect.width,
|
||
height: tileRect.height,
|
||
opacity: 255,
|
||
duration: WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
});
|
||
}
|
||
|
||
close() {
|
||
if (!this._showing)
|
||
return;
|
||
|
||
this._showing = false;
|
||
this.ease({
|
||
opacity: 0,
|
||
duration: WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
onComplete: () => this._reset(),
|
||
});
|
||
}
|
||
|
||
_reset() {
|
||
this.hide();
|
||
this._rect = null;
|
||
this._monitorIndex = -1;
|
||
}
|
||
|
||
_updateStyle(monitor) {
|
||
let styles = ['tile-preview'];
|
||
if (this._monitorIndex == Main.layoutManager.primaryIndex)
|
||
styles.push('on-primary');
|
||
if (this._rect.x == monitor.x)
|
||
styles.push('tile-preview-left');
|
||
if (this._rect.x + this._rect.width == monitor.x + monitor.width)
|
||
styles.push('tile-preview-right');
|
||
|
||
this.style_class = styles.join(' ');
|
||
}
|
||
});
|
||
|
||
var AppSwitchAction = GObject.registerClass({
|
||
Signals: { 'activated': {} },
|
||
}, class AppSwitchAction extends Clutter.GestureAction {
|
||
_init() {
|
||
super._init();
|
||
this.set_n_touch_points(3);
|
||
|
||
global.display.connect('grab-op-begin', () => {
|
||
this.cancel();
|
||
});
|
||
}
|
||
|
||
vfunc_gesture_prepare(_actor) {
|
||
if (Main.actionMode != Shell.ActionMode.NORMAL) {
|
||
this.cancel();
|
||
return false;
|
||
}
|
||
|
||
return this.get_n_current_points() <= 4;
|
||
}
|
||
|
||
vfunc_gesture_begin(_actor) {
|
||
// in milliseconds
|
||
const LONG_PRESS_TIMEOUT = 250;
|
||
|
||
let nPoints = this.get_n_current_points();
|
||
let event = this.get_last_event(nPoints - 1);
|
||
|
||
if (nPoints == 3) {
|
||
this._longPressStartTime = event.get_time();
|
||
} else if (nPoints == 4) {
|
||
// Check whether the 4th finger press happens after a 3-finger long press,
|
||
// this only needs to be checked on the first 4th finger press
|
||
if (this._longPressStartTime != null &&
|
||
event.get_time() < this._longPressStartTime + LONG_PRESS_TIMEOUT) {
|
||
this.cancel();
|
||
} else {
|
||
this._longPressStartTime = null;
|
||
this.emit('activated');
|
||
}
|
||
}
|
||
|
||
return this.get_n_current_points() <= 4;
|
||
}
|
||
|
||
vfunc_gesture_progress(_actor) {
|
||
if (this.get_n_current_points() == 3) {
|
||
for (let i = 0; i < this.get_n_current_points(); i++) {
|
||
let [startX, startY] = this.get_press_coords(i);
|
||
let [x, y] = this.get_motion_coords(i);
|
||
|
||
if (Math.abs(x - startX) > APP_MOTION_THRESHOLD ||
|
||
Math.abs(y - startY) > APP_MOTION_THRESHOLD)
|
||
return false;
|
||
}
|
||
}
|
||
|
||
return true;
|
||
}
|
||
});
|
||
|
||
var ResizePopup = GObject.registerClass(
|
||
class ResizePopup extends St.Widget {
|
||
_init() {
|
||
super._init({ layout_manager: new Clutter.BinLayout() });
|
||
this._label = new St.Label({
|
||
style_class: 'resize-popup',
|
||
x_align: Clutter.ActorAlign.CENTER,
|
||
y_align: Clutter.ActorAlign.CENTER,
|
||
x_expand: true,
|
||
y_expand: true,
|
||
});
|
||
this.add_child(this._label);
|
||
Main.uiGroup.add_actor(this);
|
||
}
|
||
|
||
set(rect, displayW, displayH) {
|
||
/* Translators: This represents the size of a window. The first number is
|
||
* the width of the window and the second is the height. */
|
||
let text = _("%d × %d").format(displayW, displayH);
|
||
this._label.set_text(text);
|
||
|
||
this.set_position(rect.x, rect.y);
|
||
this.set_size(rect.width, rect.height);
|
||
}
|
||
});
|
||
|
||
var WindowManager = class {
|
||
constructor() {
|
||
this._shellwm = global.window_manager;
|
||
|
||
this._minimizing = new Set();
|
||
this._unminimizing = new Set();
|
||
this._mapping = new Set();
|
||
this._resizing = new Set();
|
||
this._resizePending = new Set();
|
||
this._destroying = new Set();
|
||
|
||
this._dimmedWindows = [];
|
||
|
||
this._skippedActors = new Set();
|
||
|
||
this._allowedKeybindings = {};
|
||
|
||
this._isWorkspacePrepended = false;
|
||
this._canScroll = true; // limiting scrolling speed
|
||
|
||
this._shellwm.connect('kill-window-effects', (shellwm, actor) => {
|
||
this._minimizeWindowDone(shellwm, actor);
|
||
this._mapWindowDone(shellwm, actor);
|
||
this._destroyWindowDone(shellwm, actor);
|
||
this._sizeChangeWindowDone(shellwm, actor);
|
||
});
|
||
|
||
this._shellwm.connect('switch-workspace', this._switchWorkspace.bind(this));
|
||
this._shellwm.connect('show-tile-preview', this._showTilePreview.bind(this));
|
||
this._shellwm.connect('hide-tile-preview', this._hideTilePreview.bind(this));
|
||
this._shellwm.connect('show-window-menu', this._showWindowMenu.bind(this));
|
||
this._shellwm.connect('minimize', this._minimizeWindow.bind(this));
|
||
this._shellwm.connect('unminimize', this._unminimizeWindow.bind(this));
|
||
this._shellwm.connect('size-change', this._sizeChangeWindow.bind(this));
|
||
this._shellwm.connect('size-changed', this._sizeChangedWindow.bind(this));
|
||
this._shellwm.connect('map', this._mapWindow.bind(this));
|
||
this._shellwm.connect('destroy', this._destroyWindow.bind(this));
|
||
this._shellwm.connect('filter-keybinding', this._filterKeybinding.bind(this));
|
||
this._shellwm.connect('confirm-display-change', this._confirmDisplayChange.bind(this));
|
||
this._shellwm.connect('create-close-dialog', this._createCloseDialog.bind(this));
|
||
this._shellwm.connect('create-inhibit-shortcuts-dialog', this._createInhibitShortcutsDialog.bind(this));
|
||
|
||
this._workspaceSwitcherPopup = null;
|
||
this._tilePreview = null;
|
||
|
||
this.allowKeybinding('switch-to-session-1', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-2', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-3', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-4', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-5', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-6', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-7', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-8', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-9', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-10', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-11', Shell.ActionMode.ALL);
|
||
this.allowKeybinding('switch-to-session-12', Shell.ActionMode.ALL);
|
||
|
||
this.setCustomKeybindingHandler('switch-to-workspace-left',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-right',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-up',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-down',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-last',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-left',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-right',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-up',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-down',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-1',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-2',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-3',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-4',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-5',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-6',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-7',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-8',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-9',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-10',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-11',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-to-workspace-12',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-1',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-2',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-3',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-4',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-5',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-6',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-7',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-8',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-9',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-10',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-11',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-12',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('move-to-workspace-last',
|
||
Shell.ActionMode.NORMAL,
|
||
this._showWorkspaceSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-applications',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-group',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-applications-backward',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-group-backward',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-windows',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-windows-backward',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('cycle-windows',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('cycle-windows-backward',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('cycle-group',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('cycle-group-backward',
|
||
Shell.ActionMode.NORMAL,
|
||
this._startSwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-panels',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW |
|
||
Shell.ActionMode.LOCK_SCREEN |
|
||
Shell.ActionMode.UNLOCK_SCREEN |
|
||
Shell.ActionMode.LOGIN_SCREEN,
|
||
this._startA11ySwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-panels-backward',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW |
|
||
Shell.ActionMode.LOCK_SCREEN |
|
||
Shell.ActionMode.UNLOCK_SCREEN |
|
||
Shell.ActionMode.LOGIN_SCREEN,
|
||
this._startA11ySwitcher.bind(this));
|
||
this.setCustomKeybindingHandler('switch-monitor',
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._startSwitcher.bind(this));
|
||
|
||
this.addKeybinding('open-application-menu',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.POPUP,
|
||
this._toggleAppMenu.bind(this));
|
||
|
||
this.addKeybinding('toggle-message-tray',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW |
|
||
Shell.ActionMode.POPUP,
|
||
this._toggleCalendar.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-1',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-2',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-3',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-4',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-5',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-6',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-7',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-8',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
this.addKeybinding('switch-to-application-9',
|
||
new Gio.Settings({ schema_id: SHELL_KEYBINDINGS_SCHEMA }),
|
||
Meta.KeyBindingFlags.IGNORE_AUTOREPEAT,
|
||
Shell.ActionMode.NORMAL |
|
||
Shell.ActionMode.OVERVIEW,
|
||
this._switchToApplication.bind(this));
|
||
|
||
global.stage.connect('scroll-event', (stage, event) => {
|
||
const allowedModes = Shell.ActionMode.NORMAL;
|
||
if ((allowedModes & Main.actionMode) === 0)
|
||
return Clutter.EVENT_PROPAGATE;
|
||
|
||
if (this._workspaceAnimation.canHandleScrollEvent(event))
|
||
return Clutter.EVENT_PROPAGATE;
|
||
|
||
if ((event.get_state() & global.display.compositor_modifiers) === 0)
|
||
return Clutter.EVENT_PROPAGATE;
|
||
|
||
return this.handleWorkspaceScroll(event);
|
||
});
|
||
|
||
global.display.connect('show-resize-popup', this._showResizePopup.bind(this));
|
||
global.display.connect('show-pad-osd', this._showPadOsd.bind(this));
|
||
global.display.connect('show-osd', (display, monitorIndex, iconName, label) => {
|
||
let icon = Gio.Icon.new_for_string(iconName);
|
||
Main.osdWindowManager.show(monitorIndex, icon, label, null);
|
||
});
|
||
|
||
this._gsdWacomProxy = new GsdWacomProxy(Gio.DBus.session, GSD_WACOM_BUS_NAME,
|
||
GSD_WACOM_OBJECT_PATH,
|
||
(proxy, error) => {
|
||
if (error)
|
||
log(error.message);
|
||
});
|
||
|
||
global.display.connect('pad-mode-switch', (display, pad, _group, _mode) => {
|
||
let labels = [];
|
||
|
||
// FIXME: Fix num buttons
|
||
for (let i = 0; i < 50; i++) {
|
||
let str = display.get_pad_action_label(pad, Meta.PadActionType.BUTTON, i);
|
||
labels.push(str ?? '');
|
||
}
|
||
|
||
this._gsdWacomProxy?.SetOLEDLabelsAsync(
|
||
pad.get_device_node(), labels).catch(logError);
|
||
});
|
||
|
||
global.display.connect('init-xserver', (display, task) => {
|
||
IBusManager.getIBusManager().restartDaemon(['--xim']);
|
||
|
||
this._startX11Services(task);
|
||
|
||
return true;
|
||
});
|
||
global.display.connect('x11-display-closing', () => {
|
||
if (!Meta.is_wayland_compositor())
|
||
return;
|
||
|
||
this._stopX11Services(null);
|
||
|
||
IBusManager.getIBusManager().restartDaemon();
|
||
});
|
||
|
||
Main.overview.connect('showing', () => {
|
||
for (let i = 0; i < this._dimmedWindows.length; i++)
|
||
this._undimWindow(this._dimmedWindows[i]);
|
||
});
|
||
Main.overview.connect('hiding', () => {
|
||
for (let i = 0; i < this._dimmedWindows.length; i++)
|
||
this._dimWindow(this._dimmedWindows[i]);
|
||
});
|
||
|
||
this._windowMenuManager = new WindowMenu.WindowMenuManager();
|
||
|
||
if (Main.sessionMode.hasWorkspaces)
|
||
this._workspaceTracker = new WorkspaceTracker(this);
|
||
|
||
let appSwitchAction = new AppSwitchAction();
|
||
appSwitchAction.connect('activated', this._switchApp.bind(this));
|
||
global.stage.add_action_full('app-switch', Clutter.EventPhase.CAPTURE, appSwitchAction);
|
||
|
||
let mode = Shell.ActionMode.NORMAL;
|
||
let topDragAction = new EdgeDragAction.EdgeDragAction(St.Side.TOP, mode);
|
||
topDragAction.connect('activated', () => {
|
||
let currentWindow = global.display.focus_window;
|
||
if (currentWindow)
|
||
currentWindow.unmake_fullscreen();
|
||
});
|
||
|
||
let updateUnfullscreenGesture = () => {
|
||
let currentWindow = global.display.focus_window;
|
||
topDragAction.enabled = currentWindow && currentWindow.is_fullscreen();
|
||
};
|
||
|
||
global.display.connect('notify::focus-window', updateUnfullscreenGesture);
|
||
global.display.connect('in-fullscreen-changed', updateUnfullscreenGesture);
|
||
updateUnfullscreenGesture();
|
||
|
||
global.stage.add_action_full('unfullscreen', Clutter.EventPhase.CAPTURE, topDragAction);
|
||
|
||
this._workspaceAnimation =
|
||
new WorkspaceAnimation.WorkspaceAnimationController();
|
||
|
||
this._shellwm.connect('kill-switch-workspace', () => {
|
||
this._workspaceAnimation.cancelSwitchAnimation();
|
||
this._switchWorkspaceDone();
|
||
});
|
||
}
|
||
|
||
async _startX11Services(task) {
|
||
let status = true;
|
||
try {
|
||
await Shell.util_start_systemd_unit(
|
||
'gnome-session-x11-services-ready.target', 'fail', null);
|
||
} catch (e) {
|
||
// Ignore NOT_SUPPORTED error, which indicates we are not systemd
|
||
// managed and gnome-session will have taken care of everything
|
||
// already.
|
||
// Note that we do log cancellation from here.
|
||
if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.NOT_SUPPORTED)) {
|
||
log(`Error starting X11 services: ${e.message}`);
|
||
status = false;
|
||
}
|
||
} finally {
|
||
task.return_boolean(status);
|
||
}
|
||
}
|
||
|
||
async _stopX11Services(cancellable) {
|
||
try {
|
||
await Shell.util_stop_systemd_unit(
|
||
'gnome-session-x11-services.target', 'fail', cancellable);
|
||
} catch (e) {
|
||
// Ignore NOT_SUPPORTED error, which indicates we are not systemd
|
||
// managed and gnome-session will have taken care of everything
|
||
// already.
|
||
// Note that we do log cancellation from here.
|
||
if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.NOT_SUPPORTED))
|
||
log(`Error stopping X11 services: ${e.message}`);
|
||
}
|
||
}
|
||
|
||
_showPadOsd(display, device, settings, imagePath, editionMode, monitorIndex) {
|
||
this._currentPadOsd = new PadOsd.PadOsd(device, settings, imagePath, editionMode, monitorIndex);
|
||
this._currentPadOsd.connect('closed', () => (this._currentPadOsd = null));
|
||
|
||
return this._currentPadOsd;
|
||
}
|
||
|
||
_lookupIndex(windows, metaWindow) {
|
||
for (let i = 0; i < windows.length; i++) {
|
||
if (windows[i].metaWindow == metaWindow)
|
||
return i;
|
||
}
|
||
return -1;
|
||
}
|
||
|
||
_switchApp() {
|
||
let windows = global.get_window_actors().filter(actor => {
|
||
let win = actor.metaWindow;
|
||
let workspaceManager = global.workspace_manager;
|
||
let activeWorkspace = workspaceManager.get_active_workspace();
|
||
return !win.is_override_redirect() &&
|
||
win.located_on_workspace(activeWorkspace);
|
||
});
|
||
|
||
if (windows.length == 0)
|
||
return;
|
||
|
||
let focusWindow = global.display.focus_window;
|
||
let nextWindow;
|
||
|
||
if (focusWindow == null) {
|
||
nextWindow = windows[0].metaWindow;
|
||
} else {
|
||
let index = this._lookupIndex(windows, focusWindow) + 1;
|
||
|
||
if (index >= windows.length)
|
||
index = 0;
|
||
|
||
nextWindow = windows[index].metaWindow;
|
||
}
|
||
|
||
Main.activateWindow(nextWindow);
|
||
}
|
||
|
||
insertWorkspace(pos) {
|
||
let workspaceManager = global.workspace_manager;
|
||
|
||
if (!Meta.prefs_get_dynamic_workspaces())
|
||
return;
|
||
|
||
workspaceManager.append_new_workspace(false, global.get_current_time());
|
||
|
||
let windows = global.get_window_actors().map(a => a.meta_window);
|
||
|
||
// To create a new workspace, we slide all the windows on workspaces
|
||
// below us to the next workspace, leaving a blank workspace for us
|
||
// to recycle.
|
||
windows.forEach(window => {
|
||
// If the window is attached to an ancestor, we don't need/want
|
||
// to move it
|
||
if (window.get_transient_for() != null)
|
||
return;
|
||
// Same for OR windows
|
||
if (window.is_override_redirect())
|
||
return;
|
||
// Sticky windows don't need moving, in fact moving would
|
||
// unstick them
|
||
if (window.on_all_workspaces)
|
||
return;
|
||
// Windows on workspaces below pos don't need moving
|
||
let index = window.get_workspace().index();
|
||
if (index < pos)
|
||
return;
|
||
window.change_workspace_by_index(index + 1, true);
|
||
});
|
||
|
||
// If the new workspace was inserted before the active workspace,
|
||
// activate the workspace to which its windows went
|
||
let activeIndex = workspaceManager.get_active_workspace_index();
|
||
if (activeIndex >= pos) {
|
||
let newWs = workspaceManager.get_workspace_by_index(activeIndex + 1);
|
||
this._blockAnimations = true;
|
||
newWs.activate(global.get_current_time());
|
||
this._blockAnimations = false;
|
||
}
|
||
}
|
||
|
||
keepWorkspaceAlive(workspace, duration) {
|
||
if (!this._workspaceTracker)
|
||
return;
|
||
|
||
this._workspaceTracker.keepWorkspaceAlive(workspace, duration);
|
||
}
|
||
|
||
skipNextEffect(actor) {
|
||
this._skippedActors.add(actor);
|
||
}
|
||
|
||
setCustomKeybindingHandler(name, modes, handler) {
|
||
if (Meta.keybindings_set_custom_handler(name, handler))
|
||
this.allowKeybinding(name, modes);
|
||
}
|
||
|
||
addKeybinding(name, settings, flags, modes, handler) {
|
||
let action = global.display.add_keybinding(name, settings, flags, handler);
|
||
if (action != Meta.KeyBindingAction.NONE)
|
||
this.allowKeybinding(name, modes);
|
||
return action;
|
||
}
|
||
|
||
removeKeybinding(name) {
|
||
if (global.display.remove_keybinding(name))
|
||
this.allowKeybinding(name, Shell.ActionMode.NONE);
|
||
}
|
||
|
||
allowKeybinding(name, modes) {
|
||
this._allowedKeybindings[name] = modes;
|
||
}
|
||
|
||
_shouldAnimate() {
|
||
const overviewOpen = Main.overview.visible && !Main.overview.closing;
|
||
return !(overviewOpen || this._workspaceAnimation.gestureActive);
|
||
}
|
||
|
||
_shouldAnimateActor(actor, types) {
|
||
if (this._skippedActors.delete(actor))
|
||
return false;
|
||
|
||
if (!this._shouldAnimate())
|
||
return false;
|
||
|
||
if (!actor.get_texture())
|
||
return false;
|
||
|
||
let type = actor.meta_window.get_window_type();
|
||
return types.includes(type);
|
||
}
|
||
|
||
_minimizeWindow(shellwm, actor) {
|
||
const types = [
|
||
Meta.WindowType.NORMAL,
|
||
Meta.WindowType.MODAL_DIALOG,
|
||
Meta.WindowType.DIALOG,
|
||
];
|
||
if (!this._shouldAnimateActor(actor, types)) {
|
||
shellwm.completed_minimize(actor);
|
||
return;
|
||
}
|
||
|
||
actor.set_scale(1.0, 1.0);
|
||
|
||
this._minimizing.add(actor);
|
||
|
||
if (actor.meta_window.is_monitor_sized()) {
|
||
actor.ease({
|
||
opacity: 0,
|
||
duration: MINIMIZE_WINDOW_ANIMATION_TIME,
|
||
mode: MINIMIZE_WINDOW_ANIMATION_MODE,
|
||
onStopped: () => this._minimizeWindowDone(shellwm, actor),
|
||
});
|
||
} else {
|
||
let xDest, yDest, xScale, yScale;
|
||
let [success, geom] = actor.meta_window.get_icon_geometry();
|
||
if (success) {
|
||
xDest = geom.x;
|
||
yDest = geom.y;
|
||
xScale = geom.width / actor.width;
|
||
yScale = geom.height / actor.height;
|
||
} else {
|
||
let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()];
|
||
if (!monitor) {
|
||
this._minimizeWindowDone();
|
||
return;
|
||
}
|
||
xDest = monitor.x;
|
||
yDest = monitor.y;
|
||
if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL)
|
||
xDest += monitor.width;
|
||
xScale = 0;
|
||
yScale = 0;
|
||
}
|
||
|
||
actor.ease({
|
||
scale_x: xScale,
|
||
scale_y: yScale,
|
||
x: xDest,
|
||
y: yDest,
|
||
duration: MINIMIZE_WINDOW_ANIMATION_TIME,
|
||
mode: MINIMIZE_WINDOW_ANIMATION_MODE,
|
||
onStopped: () => this._minimizeWindowDone(shellwm, actor),
|
||
});
|
||
}
|
||
}
|
||
|
||
_minimizeWindowDone(shellwm, actor) {
|
||
if (this._minimizing.delete(actor)) {
|
||
actor.remove_all_transitions();
|
||
actor.set_scale(1.0, 1.0);
|
||
actor.set_opacity(255);
|
||
actor.set_pivot_point(0, 0);
|
||
|
||
shellwm.completed_minimize(actor);
|
||
}
|
||
}
|
||
|
||
_unminimizeWindow(shellwm, actor) {
|
||
const types = [
|
||
Meta.WindowType.NORMAL,
|
||
Meta.WindowType.MODAL_DIALOG,
|
||
Meta.WindowType.DIALOG,
|
||
];
|
||
if (!this._shouldAnimateActor(actor, types)) {
|
||
shellwm.completed_unminimize(actor);
|
||
return;
|
||
}
|
||
|
||
this._unminimizing.add(actor);
|
||
|
||
if (actor.meta_window.is_monitor_sized()) {
|
||
actor.opacity = 0;
|
||
actor.set_scale(1.0, 1.0);
|
||
actor.ease({
|
||
opacity: 255,
|
||
duration: MINIMIZE_WINDOW_ANIMATION_TIME,
|
||
mode: MINIMIZE_WINDOW_ANIMATION_MODE,
|
||
onStopped: () => this._unminimizeWindowDone(shellwm, actor),
|
||
});
|
||
} else {
|
||
let [success, geom] = actor.meta_window.get_icon_geometry();
|
||
if (success) {
|
||
actor.set_position(geom.x, geom.y);
|
||
actor.set_scale(geom.width / actor.width,
|
||
geom.height / actor.height);
|
||
} else {
|
||
let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()];
|
||
if (!monitor) {
|
||
actor.show();
|
||
this._unminimizeWindowDone();
|
||
return;
|
||
}
|
||
actor.set_position(monitor.x, monitor.y);
|
||
if (Clutter.get_default_text_direction() == Clutter.TextDirection.RTL)
|
||
actor.x += monitor.width;
|
||
actor.set_scale(0, 0);
|
||
}
|
||
|
||
let rect = actor.meta_window.get_buffer_rect();
|
||
let [xDest, yDest] = [rect.x, rect.y];
|
||
|
||
actor.show();
|
||
actor.ease({
|
||
scale_x: 1,
|
||
scale_y: 1,
|
||
x: xDest,
|
||
y: yDest,
|
||
duration: MINIMIZE_WINDOW_ANIMATION_TIME,
|
||
mode: MINIMIZE_WINDOW_ANIMATION_MODE,
|
||
onStopped: () => this._unminimizeWindowDone(shellwm, actor),
|
||
});
|
||
}
|
||
}
|
||
|
||
_unminimizeWindowDone(shellwm, actor) {
|
||
if (this._unminimizing.delete(actor)) {
|
||
actor.remove_all_transitions();
|
||
actor.set_scale(1.0, 1.0);
|
||
actor.set_opacity(255);
|
||
actor.set_pivot_point(0, 0);
|
||
|
||
shellwm.completed_unminimize(actor);
|
||
}
|
||
}
|
||
|
||
_sizeChangeWindow(shellwm, actor, whichChange, oldFrameRect, _oldBufferRect) {
|
||
const types = [Meta.WindowType.NORMAL];
|
||
const shouldAnimate =
|
||
this._shouldAnimateActor(actor, types) &&
|
||
oldFrameRect.width > 0 &&
|
||
oldFrameRect.height > 0;
|
||
|
||
if (shouldAnimate)
|
||
this._prepareAnimationInfo(shellwm, actor, oldFrameRect, whichChange);
|
||
else
|
||
shellwm.completed_size_change(actor);
|
||
}
|
||
|
||
_prepareAnimationInfo(shellwm, actor, oldFrameRect, _change) {
|
||
// Position a clone of the window on top of the old position,
|
||
// while actor updates are frozen.
|
||
let actorContent = actor.paint_to_content(oldFrameRect);
|
||
let actorClone = new St.Widget({ content: actorContent });
|
||
actorClone.set_offscreen_redirect(Clutter.OffscreenRedirect.ALWAYS);
|
||
actorClone.set_position(oldFrameRect.x, oldFrameRect.y);
|
||
actorClone.set_size(oldFrameRect.width, oldFrameRect.height);
|
||
|
||
actor.freeze();
|
||
|
||
if (this._clearAnimationInfo(actor)) {
|
||
log(`Old animationInfo removed from actor ${actor}`);
|
||
this._shellwm.completed_size_change(actor);
|
||
}
|
||
|
||
actor.connectObject('destroy',
|
||
() => this._clearAnimationInfo(actor), actorClone);
|
||
|
||
this._resizePending.add(actor);
|
||
actor.__animationInfo = {
|
||
clone: actorClone,
|
||
oldRect: oldFrameRect,
|
||
frozen: true,
|
||
};
|
||
}
|
||
|
||
_sizeChangedWindow(shellwm, actor) {
|
||
if (!actor.__animationInfo)
|
||
return;
|
||
if (this._resizing.has(actor))
|
||
return;
|
||
|
||
let actorClone = actor.__animationInfo.clone;
|
||
let targetRect = actor.meta_window.get_frame_rect();
|
||
let sourceRect = actor.__animationInfo.oldRect;
|
||
|
||
let scaleX = targetRect.width / sourceRect.width;
|
||
let scaleY = targetRect.height / sourceRect.height;
|
||
|
||
this._resizePending.delete(actor);
|
||
this._resizing.add(actor);
|
||
|
||
Main.uiGroup.add_child(actorClone);
|
||
|
||
// Now scale and fade out the clone
|
||
actorClone.ease({
|
||
x: targetRect.x,
|
||
y: targetRect.y,
|
||
scale_x: scaleX,
|
||
scale_y: scaleY,
|
||
opacity: 0,
|
||
duration: WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
});
|
||
|
||
actor.translation_x = -targetRect.x + sourceRect.x;
|
||
actor.translation_y = -targetRect.y + sourceRect.y;
|
||
|
||
// Now set scale the actor to size it as the clone.
|
||
actor.scale_x = 1 / scaleX;
|
||
actor.scale_y = 1 / scaleY;
|
||
|
||
// Scale it to its actual new size
|
||
actor.ease({
|
||
scale_x: 1,
|
||
scale_y: 1,
|
||
translation_x: 0,
|
||
translation_y: 0,
|
||
duration: WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
onStopped: () => this._sizeChangeWindowDone(shellwm, actor),
|
||
});
|
||
|
||
// ease didn't animate and cleared the info, we are done
|
||
if (!actor.__animationInfo)
|
||
return;
|
||
|
||
// Now unfreeze actor updates, to get it to the new size.
|
||
// It's important that we don't wait until the animation is completed to
|
||
// do this, otherwise our scale will be applied to the old texture size.
|
||
actor.thaw();
|
||
actor.__animationInfo.frozen = false;
|
||
}
|
||
|
||
_clearAnimationInfo(actor) {
|
||
if (actor.__animationInfo) {
|
||
actor.__animationInfo.clone.destroy();
|
||
if (actor.__animationInfo.frozen)
|
||
actor.thaw();
|
||
|
||
delete actor.__animationInfo;
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
_sizeChangeWindowDone(shellwm, actor) {
|
||
if (this._resizing.delete(actor)) {
|
||
actor.remove_all_transitions();
|
||
actor.scale_x = 1.0;
|
||
actor.scale_y = 1.0;
|
||
actor.translation_x = 0;
|
||
actor.translation_y = 0;
|
||
this._clearAnimationInfo(actor);
|
||
this._shellwm.completed_size_change(actor);
|
||
}
|
||
|
||
if (this._resizePending.delete(actor)) {
|
||
this._clearAnimationInfo(actor);
|
||
this._shellwm.completed_size_change(actor);
|
||
}
|
||
}
|
||
|
||
_checkDimming(window) {
|
||
const shouldDim = window.has_attached_dialogs();
|
||
|
||
if (shouldDim && !window._dimmed) {
|
||
window._dimmed = true;
|
||
this._dimmedWindows.push(window);
|
||
this._dimWindow(window);
|
||
} else if (!shouldDim && window._dimmed) {
|
||
window._dimmed = false;
|
||
this._dimmedWindows =
|
||
this._dimmedWindows.filter(win => win != window);
|
||
this._undimWindow(window);
|
||
}
|
||
}
|
||
|
||
_dimWindow(window) {
|
||
let actor = window.get_compositor_private();
|
||
if (!actor)
|
||
return;
|
||
let dimmer = getWindowDimmer(actor);
|
||
if (!dimmer)
|
||
return;
|
||
dimmer.setDimmed(true, this._shouldAnimate());
|
||
}
|
||
|
||
_undimWindow(window) {
|
||
let actor = window.get_compositor_private();
|
||
if (!actor)
|
||
return;
|
||
let dimmer = getWindowDimmer(actor);
|
||
if (!dimmer)
|
||
return;
|
||
dimmer.setDimmed(false, this._shouldAnimate());
|
||
}
|
||
|
||
_waitForOverviewToHide() {
|
||
if (!Main.overview.visible)
|
||
return Promise.resolve();
|
||
|
||
return new Promise(resolve => {
|
||
const id = Main.overview.connect('hidden', () => {
|
||
Main.overview.disconnect(id);
|
||
resolve();
|
||
});
|
||
});
|
||
}
|
||
|
||
async _mapWindow(shellwm, actor) {
|
||
actor._windowType = actor.meta_window.get_window_type();
|
||
actor.meta_window.connectObject('notify::window-type', () => {
|
||
let type = actor.meta_window.get_window_type();
|
||
if (type === actor._windowType)
|
||
return;
|
||
if (type === Meta.WindowType.MODAL_DIALOG ||
|
||
actor._windowType === Meta.WindowType.MODAL_DIALOG) {
|
||
let parent = actor.get_meta_window().get_transient_for();
|
||
if (parent)
|
||
this._checkDimming(parent);
|
||
}
|
||
|
||
actor._windowType = type;
|
||
}, actor);
|
||
actor.meta_window.connect('unmanaged', window => {
|
||
let parent = window.get_transient_for();
|
||
if (parent)
|
||
this._checkDimming(parent);
|
||
});
|
||
|
||
if (actor.meta_window.is_attached_dialog())
|
||
this._checkDimming(actor.get_meta_window().get_transient_for());
|
||
|
||
const types = [
|
||
Meta.WindowType.NORMAL,
|
||
Meta.WindowType.DIALOG,
|
||
Meta.WindowType.MODAL_DIALOG,
|
||
];
|
||
if (!this._shouldAnimateActor(actor, types)) {
|
||
shellwm.completed_map(actor);
|
||
return;
|
||
}
|
||
|
||
switch (actor._windowType) {
|
||
case Meta.WindowType.NORMAL:
|
||
actor.set_pivot_point(0.5, 1.0);
|
||
actor.scale_x = 0.01;
|
||
actor.scale_y = 0.05;
|
||
actor.opacity = 0;
|
||
actor.show();
|
||
this._mapping.add(actor);
|
||
|
||
await this._waitForOverviewToHide();
|
||
actor.ease({
|
||
opacity: 255,
|
||
scale_x: 1,
|
||
scale_y: 1,
|
||
duration: SHOW_WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_EXPO,
|
||
onStopped: () => this._mapWindowDone(shellwm, actor),
|
||
});
|
||
break;
|
||
case Meta.WindowType.MODAL_DIALOG:
|
||
case Meta.WindowType.DIALOG:
|
||
actor.set_pivot_point(0.5, 0.5);
|
||
actor.scale_y = 0;
|
||
actor.opacity = 0;
|
||
actor.show();
|
||
this._mapping.add(actor);
|
||
|
||
await this._waitForOverviewToHide();
|
||
actor.ease({
|
||
opacity: 255,
|
||
scale_x: 1,
|
||
scale_y: 1,
|
||
duration: DIALOG_SHOW_WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
onStopped: () => this._mapWindowDone(shellwm, actor),
|
||
});
|
||
break;
|
||
default:
|
||
shellwm.completed_map(actor);
|
||
}
|
||
}
|
||
|
||
_mapWindowDone(shellwm, actor) {
|
||
if (this._mapping.delete(actor)) {
|
||
actor.remove_all_transitions();
|
||
actor.opacity = 255;
|
||
actor.set_pivot_point(0, 0);
|
||
actor.scale_y = 1;
|
||
actor.scale_x = 1;
|
||
actor.translation_y = 0;
|
||
actor.translation_x = 0;
|
||
shellwm.completed_map(actor);
|
||
}
|
||
}
|
||
|
||
_destroyWindow(shellwm, actor) {
|
||
let window = actor.meta_window;
|
||
window.disconnectObject(actor);
|
||
if (window._dimmed) {
|
||
this._dimmedWindows =
|
||
this._dimmedWindows.filter(win => win != window);
|
||
}
|
||
|
||
if (window.is_attached_dialog())
|
||
this._checkDimming(window.get_transient_for());
|
||
|
||
const types = [
|
||
Meta.WindowType.NORMAL,
|
||
Meta.WindowType.DIALOG,
|
||
Meta.WindowType.MODAL_DIALOG,
|
||
];
|
||
if (!this._shouldAnimateActor(actor, types)) {
|
||
shellwm.completed_destroy(actor);
|
||
return;
|
||
}
|
||
|
||
switch (actor.meta_window.window_type) {
|
||
case Meta.WindowType.NORMAL:
|
||
actor.set_pivot_point(0.5, 0.5);
|
||
this._destroying.add(actor);
|
||
|
||
actor.ease({
|
||
opacity: 0,
|
||
scale_x: 0.8,
|
||
scale_y: 0.8,
|
||
duration: DESTROY_WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
onStopped: () => this._destroyWindowDone(shellwm, actor),
|
||
});
|
||
break;
|
||
case Meta.WindowType.MODAL_DIALOG:
|
||
case Meta.WindowType.DIALOG:
|
||
actor.set_pivot_point(0.5, 0.5);
|
||
this._destroying.add(actor);
|
||
|
||
if (window.is_attached_dialog()) {
|
||
let parent = window.get_transient_for();
|
||
parent.connectObject('unmanaged', () => {
|
||
actor.remove_all_transitions();
|
||
this._destroyWindowDone(shellwm, actor);
|
||
}, actor);
|
||
}
|
||
|
||
actor.ease({
|
||
scale_y: 0,
|
||
duration: DIALOG_DESTROY_WINDOW_ANIMATION_TIME,
|
||
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
onStopped: () => this._destroyWindowDone(shellwm, actor),
|
||
});
|
||
break;
|
||
default:
|
||
shellwm.completed_destroy(actor);
|
||
}
|
||
}
|
||
|
||
_destroyWindowDone(shellwm, actor) {
|
||
if (this._destroying.delete(actor)) {
|
||
const parent = actor.get_meta_window()?.get_transient_for();
|
||
parent?.disconnectObject(actor);
|
||
shellwm.completed_destroy(actor);
|
||
}
|
||
}
|
||
|
||
_filterKeybinding(shellwm, binding) {
|
||
if (Main.actionMode == Shell.ActionMode.NONE)
|
||
return true;
|
||
|
||
// There's little sense in implementing a keybinding in mutter and
|
||
// not having it work in NORMAL mode; handle this case generically
|
||
// so we don't have to explicitly allow all builtin keybindings in
|
||
// NORMAL mode.
|
||
if (Main.actionMode == Shell.ActionMode.NORMAL &&
|
||
binding.is_builtin())
|
||
return false;
|
||
|
||
return !(this._allowedKeybindings[binding.get_name()] & Main.actionMode);
|
||
}
|
||
|
||
_switchWorkspace(shellwm, from, to, direction) {
|
||
if (!Main.sessionMode.hasWorkspaces || !this._shouldAnimate()) {
|
||
shellwm.completed_switch_workspace();
|
||
return;
|
||
}
|
||
|
||
this._switchInProgress = true;
|
||
|
||
this._workspaceAnimation.animateSwitch(from, to, direction, () => {
|
||
this._shellwm.completed_switch_workspace();
|
||
this._switchInProgress = false;
|
||
});
|
||
}
|
||
|
||
_switchWorkspaceDone() {
|
||
if (!this._switchInProgress)
|
||
return;
|
||
|
||
this._shellwm.completed_switch_workspace();
|
||
this._switchInProgress = false;
|
||
}
|
||
|
||
_showTilePreview(shellwm, window, tileRect, monitorIndex) {
|
||
if (!this._tilePreview)
|
||
this._tilePreview = new TilePreview();
|
||
this._tilePreview.open(window, tileRect, monitorIndex);
|
||
}
|
||
|
||
_hideTilePreview() {
|
||
if (!this._tilePreview)
|
||
return;
|
||
this._tilePreview.close();
|
||
}
|
||
|
||
_showWindowMenu(shellwm, window, menu, rect) {
|
||
this._windowMenuManager.showWindowMenuForWindow(window, menu, rect);
|
||
}
|
||
|
||
_startSwitcher(display, window, binding) {
|
||
let constructor = null;
|
||
switch (binding.get_name()) {
|
||
case 'switch-applications':
|
||
case 'switch-applications-backward':
|
||
case 'switch-group':
|
||
case 'switch-group-backward':
|
||
constructor = AltTab.AppSwitcherPopup;
|
||
break;
|
||
case 'switch-windows':
|
||
case 'switch-windows-backward':
|
||
constructor = AltTab.WindowSwitcherPopup;
|
||
break;
|
||
case 'cycle-windows':
|
||
case 'cycle-windows-backward':
|
||
constructor = AltTab.WindowCyclerPopup;
|
||
break;
|
||
case 'cycle-group':
|
||
case 'cycle-group-backward':
|
||
constructor = AltTab.GroupCyclerPopup;
|
||
break;
|
||
case 'switch-monitor':
|
||
constructor = SwitchMonitor.SwitchMonitorPopup;
|
||
break;
|
||
}
|
||
|
||
if (!constructor)
|
||
return;
|
||
|
||
/* prevent a corner case where both popups show up at once */
|
||
if (this._workspaceSwitcherPopup != null)
|
||
this._workspaceSwitcherPopup.destroy();
|
||
|
||
let tabPopup = new constructor();
|
||
|
||
if (!tabPopup.show(binding.is_reversed(), binding.get_name(), binding.get_mask()))
|
||
tabPopup.destroy();
|
||
}
|
||
|
||
_startA11ySwitcher(display, window, binding) {
|
||
Main.ctrlAltTabManager.popup(binding.is_reversed(), binding.get_name(), binding.get_mask());
|
||
}
|
||
|
||
_allowFavoriteShortcuts() {
|
||
return Main.sessionMode.hasOverview;
|
||
}
|
||
|
||
_switchToApplication(display, window, binding) {
|
||
if (!this._allowFavoriteShortcuts())
|
||
return;
|
||
|
||
let [, , , target] = binding.get_name().split('-');
|
||
let apps = AppFavorites.getAppFavorites().getFavorites();
|
||
let app = apps[target - 1];
|
||
if (app) {
|
||
Main.overview.hide();
|
||
app.activate();
|
||
}
|
||
}
|
||
|
||
_toggleAppMenu() {
|
||
Main.panel.toggleAppMenu();
|
||
}
|
||
|
||
_toggleCalendar() {
|
||
Main.panel.toggleCalendar();
|
||
}
|
||
|
||
_showWorkspaceSwitcher(display, window, binding) {
|
||
let workspaceManager = display.get_workspace_manager();
|
||
|
||
if (!Main.sessionMode.hasWorkspaces)
|
||
return;
|
||
|
||
if (workspaceManager.n_workspaces == 1)
|
||
return;
|
||
|
||
let [action,,, target] = binding.get_name().split('-');
|
||
let newWs;
|
||
let direction;
|
||
let vertical = workspaceManager.layout_rows == -1;
|
||
let rtl = Clutter.get_default_text_direction() == Clutter.TextDirection.RTL;
|
||
|
||
if (action == 'move') {
|
||
// "Moving" a window to another workspace doesn't make sense when
|
||
// it cannot be unstuck, and is potentially confusing if a new
|
||
// workspaces is added at the start/end
|
||
if (window.is_always_on_all_workspaces() ||
|
||
(Meta.prefs_get_workspaces_only_on_primary() &&
|
||
window.get_monitor() != Main.layoutManager.primaryIndex))
|
||
return;
|
||
}
|
||
|
||
if (target == 'last') {
|
||
if (vertical)
|
||
direction = Meta.MotionDirection.DOWN;
|
||
else if (rtl)
|
||
direction = Meta.MotionDirection.LEFT;
|
||
else
|
||
direction = Meta.MotionDirection.RIGHT;
|
||
newWs = workspaceManager.get_workspace_by_index(workspaceManager.n_workspaces - 1);
|
||
} else if (isNaN(target)) {
|
||
// Prepend a new workspace dynamically
|
||
let prependTarget;
|
||
if (vertical)
|
||
prependTarget = 'up';
|
||
else if (rtl)
|
||
prependTarget = 'right';
|
||
else
|
||
prependTarget = 'left';
|
||
if (workspaceManager.get_active_workspace_index() === 0 &&
|
||
action === 'move' && target === prependTarget &&
|
||
this._isWorkspacePrepended === false) {
|
||
this.insertWorkspace(0);
|
||
this._isWorkspacePrepended = true;
|
||
}
|
||
|
||
direction = Meta.MotionDirection[target.toUpperCase()];
|
||
newWs = workspaceManager.get_active_workspace().get_neighbor(direction);
|
||
} else if ((target > 0) && (target <= workspaceManager.n_workspaces)) {
|
||
target--;
|
||
newWs = workspaceManager.get_workspace_by_index(target);
|
||
|
||
if (workspaceManager.get_active_workspace_index() > target) {
|
||
if (vertical)
|
||
direction = Meta.MotionDirection.UP;
|
||
else if (rtl)
|
||
direction = Meta.MotionDirection.RIGHT;
|
||
else
|
||
direction = Meta.MotionDirection.LEFT;
|
||
} else {
|
||
if (vertical) // eslint-disable-line no-lonely-if
|
||
direction = Meta.MotionDirection.DOWN;
|
||
else if (rtl)
|
||
direction = Meta.MotionDirection.LEFT;
|
||
else
|
||
direction = Meta.MotionDirection.RIGHT;
|
||
}
|
||
}
|
||
|
||
if (workspaceManager.layout_rows == -1 &&
|
||
direction != Meta.MotionDirection.UP &&
|
||
direction != Meta.MotionDirection.DOWN)
|
||
return;
|
||
|
||
if (workspaceManager.layout_columns == -1 &&
|
||
direction != Meta.MotionDirection.LEFT &&
|
||
direction != Meta.MotionDirection.RIGHT)
|
||
return;
|
||
|
||
if (action == 'switch')
|
||
this.actionMoveWorkspace(newWs);
|
||
else
|
||
this.actionMoveWindow(window, newWs);
|
||
|
||
if (!Main.overview.visible) {
|
||
if (this._workspaceSwitcherPopup == null) {
|
||
this._workspaceTracker.blockUpdates();
|
||
this._workspaceSwitcherPopup = new WorkspaceSwitcherPopup.WorkspaceSwitcherPopup();
|
||
this._workspaceSwitcherPopup.connect('destroy', () => {
|
||
this._workspaceTracker.unblockUpdates();
|
||
this._workspaceSwitcherPopup = null;
|
||
this._isWorkspacePrepended = false;
|
||
});
|
||
}
|
||
this._workspaceSwitcherPopup.display(newWs.index());
|
||
}
|
||
}
|
||
|
||
actionMoveWorkspace(workspace) {
|
||
if (!Main.sessionMode.hasWorkspaces)
|
||
return;
|
||
|
||
if (!workspace.active)
|
||
workspace.activate(global.get_current_time());
|
||
}
|
||
|
||
actionMoveWindow(window, workspace) {
|
||
if (!Main.sessionMode.hasWorkspaces)
|
||
return;
|
||
|
||
if (!workspace.active) {
|
||
// This won't have any effect for "always sticky" windows
|
||
// (like desktop windows or docks)
|
||
|
||
this._workspaceAnimation.movingWindow = window;
|
||
window.change_workspace(workspace);
|
||
|
||
global.display.clear_mouse_mode();
|
||
workspace.activate_with_focus(window, global.get_current_time());
|
||
}
|
||
}
|
||
|
||
handleWorkspaceScroll(event) {
|
||
if (!this._canScroll)
|
||
return Clutter.EVENT_PROPAGATE;
|
||
|
||
if (event.type() !== Clutter.EventType.SCROLL)
|
||
return Clutter.EVENT_PROPAGATE;
|
||
|
||
const direction = event.get_scroll_direction();
|
||
if (direction === Clutter.ScrollDirection.SMOOTH)
|
||
return Clutter.EVENT_PROPAGATE;
|
||
|
||
const workspaceManager = global.workspace_manager;
|
||
const vertical = workspaceManager.layout_rows === -1;
|
||
const rtl = Clutter.get_default_text_direction() === Clutter.TextDirection.RTL;
|
||
const activeWs = workspaceManager.get_active_workspace();
|
||
let ws;
|
||
switch (direction) {
|
||
case Clutter.ScrollDirection.UP:
|
||
if (vertical)
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.UP);
|
||
else if (rtl)
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.RIGHT);
|
||
else
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.LEFT);
|
||
break;
|
||
case Clutter.ScrollDirection.DOWN:
|
||
if (vertical)
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.DOWN);
|
||
else if (rtl)
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.LEFT);
|
||
else
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.RIGHT);
|
||
break;
|
||
case Clutter.ScrollDirection.LEFT:
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.LEFT);
|
||
break;
|
||
case Clutter.ScrollDirection.RIGHT:
|
||
ws = activeWs.get_neighbor(Meta.MotionDirection.RIGHT);
|
||
break;
|
||
default:
|
||
return Clutter.EVENT_PROPAGATE;
|
||
}
|
||
this.actionMoveWorkspace(ws);
|
||
|
||
this._canScroll = false;
|
||
GLib.timeout_add(GLib.PRIORITY_DEFAULT,
|
||
SCROLL_TIMEOUT_TIME, () => {
|
||
this._canScroll = true;
|
||
return GLib.SOURCE_REMOVE;
|
||
});
|
||
|
||
return Clutter.EVENT_STOP;
|
||
}
|
||
|
||
_confirmDisplayChange() {
|
||
let dialog = new DisplayChangeDialog(this._shellwm);
|
||
dialog.open();
|
||
}
|
||
|
||
_createCloseDialog(shellwm, window) {
|
||
return new CloseDialog.CloseDialog(window);
|
||
}
|
||
|
||
_createInhibitShortcutsDialog(shellwm, window) {
|
||
return new InhibitShortcutsDialog.InhibitShortcutsDialog(window);
|
||
}
|
||
|
||
_showResizePopup(display, show, rect, displayW, displayH) {
|
||
if (show) {
|
||
if (!this._resizePopup)
|
||
this._resizePopup = new ResizePopup();
|
||
|
||
this._resizePopup.set(rect, displayW, displayH);
|
||
} else {
|
||
if (!this._resizePopup)
|
||
return;
|
||
|
||
this._resizePopup.destroy();
|
||
this._resizePopup = null;
|
||
}
|
||
}
|
||
};
|