altTab: Re-implement 'switch-windows' keybinding
Now that we use the new 'switch-applications' keybinding for the application-based alt-tab popup, we can use the 'switch-windows' keybinding for a more traditional switcher. Based heavily on the alternate-tab extension from Giovanni Campagna. https://bugzilla.gnome.org/show_bug.cgi?id=688913
This commit is contained in:
parent
525d3c2619
commit
e725f8a0fe
@ -186,6 +186,33 @@ value here is from the GsmPresenceStatus enumeration.</_summary>
|
|||||||
</key>
|
</key>
|
||||||
</schema>
|
</schema>
|
||||||
|
|
||||||
|
<enum id="org.gnome.shell.window-switcher.AppIconMode">
|
||||||
|
<value value="1" nick="thumbnail-only"/>
|
||||||
|
<value value="2" nick="app-icon-only"/>
|
||||||
|
<value value="3" nick="both"/>
|
||||||
|
</enum>
|
||||||
|
<schema id="org.gnome.shell.window-switcher"
|
||||||
|
path="/org/gnome/shell/window-switcher/"
|
||||||
|
gettext-domain="@GETTEXT_PACKAGE@">
|
||||||
|
<key name="app-icon-mode" enum="org.gnome.shell.window-switcher.AppIconMode">
|
||||||
|
<default>'both'</default>
|
||||||
|
<_summary>The application icon mode.</_summary>
|
||||||
|
<_description>
|
||||||
|
Configures how the windows are shown in the switcher. Valid possibilities
|
||||||
|
are 'thumbnail-only' (shows a thumbnail of the window), 'app-icon-only'
|
||||||
|
(shows only the application icon) or 'both'.
|
||||||
|
</_description>
|
||||||
|
</key>
|
||||||
|
<key type="b" name="current-workspace-only">
|
||||||
|
<default>false</default>
|
||||||
|
<summary>Limit switcher to current workspace.</summary>
|
||||||
|
<description>
|
||||||
|
If true, only windows from the current workspace are shown in the switcher.
|
||||||
|
Otherwise, all windows are included.
|
||||||
|
</description>
|
||||||
|
</key>
|
||||||
|
</schema>
|
||||||
|
|
||||||
<schema id="org.gnome.shell.overrides" path="/org/gnome/shell/overrides/"
|
<schema id="org.gnome.shell.overrides" path="/org/gnome/shell/overrides/"
|
||||||
gettext-domain="@GETTEXT_PACKAGE@">
|
gettext-domain="@GETTEXT_PACKAGE@">
|
||||||
<key name="attach-modal-dialogs" type="b">
|
<key name="attach-modal-dialogs" type="b">
|
||||||
|
170
js/ui/altTab.js
170
js/ui/altTab.js
@ -1,6 +1,7 @@
|
|||||||
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
||||||
|
|
||||||
const Clutter = imports.gi.Clutter;
|
const Clutter = imports.gi.Clutter;
|
||||||
|
const Gio = imports.gi.Gio;
|
||||||
const Lang = imports.lang;
|
const Lang = imports.lang;
|
||||||
const Mainloop = imports.mainloop;
|
const Mainloop = imports.mainloop;
|
||||||
const Meta = imports.gi.Meta;
|
const Meta = imports.gi.Meta;
|
||||||
@ -18,8 +19,18 @@ const THUMBNAIL_DEFAULT_SIZE = 256;
|
|||||||
const THUMBNAIL_POPUP_TIME = 500; // milliseconds
|
const THUMBNAIL_POPUP_TIME = 500; // milliseconds
|
||||||
const THUMBNAIL_FADE_TIME = 0.1; // seconds
|
const THUMBNAIL_FADE_TIME = 0.1; // seconds
|
||||||
|
|
||||||
|
const WINDOW_PREVIEW_SIZE = 128;
|
||||||
|
const APP_ICON_SIZE = 96;
|
||||||
|
const APP_ICON_SIZE_SMALL = 48;
|
||||||
|
|
||||||
const iconSizes = [96, 64, 48, 32, 22];
|
const iconSizes = [96, 64, 48, 32, 22];
|
||||||
|
|
||||||
|
const AppIconMode = {
|
||||||
|
THUMBNAIL_ONLY: 1,
|
||||||
|
APP_ICON_ONLY: 2,
|
||||||
|
BOTH: 3,
|
||||||
|
};
|
||||||
|
|
||||||
function _createWindowClone(window, size) {
|
function _createWindowClone(window, size) {
|
||||||
let windowTexture = window.get_texture();
|
let windowTexture = window.get_texture();
|
||||||
let [width, height] = windowTexture.get_size();
|
let [width, height] = windowTexture.get_size();
|
||||||
@ -372,6 +383,58 @@ const AppSwitcherPopup = new Lang.Class({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const WindowSwitcherPopup = new Lang.Class({
|
||||||
|
Name: 'WindowSwitcherPopup',
|
||||||
|
Extends: SwitcherPopup.SwitcherPopup,
|
||||||
|
|
||||||
|
_getWindowList: function() {
|
||||||
|
let settings = new Gio.Settings({ schema: 'org.gnome.shell.window-switcher' });
|
||||||
|
let workspace = settings.get_boolean('current-workspace-only') ? global.screen.get_active_workspace()
|
||||||
|
: null;
|
||||||
|
return global.display.get_tab_list(Meta.TabList.NORMAL, global.screen, workspace);
|
||||||
|
},
|
||||||
|
|
||||||
|
_createSwitcher: function() {
|
||||||
|
let windows = this._getWindowList();
|
||||||
|
|
||||||
|
if (windows.length == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
this._switcherList = new WindowList(windows);
|
||||||
|
this._items = this._switcherList.icons;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
|
||||||
|
_initialSelection: function(backward, binding) {
|
||||||
|
if (binding == 'switch-windows-backward' || backward)
|
||||||
|
this._select(this._items.length - 1);
|
||||||
|
else if (this._items.length == 1)
|
||||||
|
this._select(0);
|
||||||
|
else
|
||||||
|
this._select(1);
|
||||||
|
},
|
||||||
|
|
||||||
|
_keyPressHandler: function(keysym, backwards, action) {
|
||||||
|
if (action == Meta.KeyBindingAction.SWITCH_WINDOWS) {
|
||||||
|
this._select(backwards ? this._previous() : this._next());
|
||||||
|
} else if (action == Meta.KeyBindingAction.SWITCH_WINDOWS_BACKWARD) {
|
||||||
|
this._select(this._previous());
|
||||||
|
} else {
|
||||||
|
if (keysym == Clutter.Left)
|
||||||
|
this._select(this._previous());
|
||||||
|
else if (keysym == Clutter.Right)
|
||||||
|
this._select(this._next());
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
_finish: function() {
|
||||||
|
this.parent();
|
||||||
|
|
||||||
|
Main.activateWindow(this._items[this._selectedIndex].window);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const AppIcon = new Lang.Class({
|
const AppIcon = new Lang.Class({
|
||||||
Name: 'AppIcon',
|
Name: 'AppIcon',
|
||||||
|
|
||||||
@ -646,3 +709,110 @@ const ThumbnailList = new Lang.Class({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const WindowIcon = new Lang.Class({
|
||||||
|
Name: 'WindowIcon',
|
||||||
|
|
||||||
|
_init: function(window) {
|
||||||
|
this.window = window;
|
||||||
|
|
||||||
|
this.actor = new St.BoxLayout({ style_class: 'alt-tab-app',
|
||||||
|
vertical: true });
|
||||||
|
this._icon = new St.Widget({ layout_manager: new Clutter.BinLayout() });
|
||||||
|
|
||||||
|
this.actor.add(this._icon, { x_fill: false, y_fill: false } );
|
||||||
|
this.label = new St.Label({ text: window.get_title() });
|
||||||
|
|
||||||
|
let tracker = Shell.WindowTracker.get_default();
|
||||||
|
this.app = tracker.get_window_app(window);
|
||||||
|
|
||||||
|
let mutterWindow = this.window.get_compositor_private();
|
||||||
|
let size;
|
||||||
|
|
||||||
|
this._icon.destroy_all_children();
|
||||||
|
|
||||||
|
let settings = new Gio.Settings({ schema: 'org.gnome.shell.window-switcher' });
|
||||||
|
switch (settings.get_enum('app-icon-mode')) {
|
||||||
|
case AppIconMode.THUMBNAIL_ONLY:
|
||||||
|
size = WINDOW_PREVIEW_SIZE;
|
||||||
|
this._icon.add_actor(_createWindowClone(mutterWindow, WINDOW_PREVIEW_SIZE));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AppIconMode.BOTH:
|
||||||
|
size = WINDOW_PREVIEW_SIZE;
|
||||||
|
this._icon.add_actor(_createWindowClone(mutterWindow, WINDOW_PREVIEW_SIZE));
|
||||||
|
|
||||||
|
if (this.app)
|
||||||
|
this._icon.add_actor(this._createAppIcon(this.app,
|
||||||
|
APP_ICON_SIZE_SMALL));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AppIconMode.APP_ICON_ONLY:
|
||||||
|
size = APP_ICON_SIZE;
|
||||||
|
this._icon.add_actor(this._createAppIcon(this.app, size));
|
||||||
|
}
|
||||||
|
|
||||||
|
this._icon.set_size(size, size);
|
||||||
|
},
|
||||||
|
|
||||||
|
_createAppIcon: function(app, size) {
|
||||||
|
let appIcon = app ? app.create_icon_texture(size)
|
||||||
|
: new St.Icon({ icon_name: 'icon-missing',
|
||||||
|
icon_size: size });
|
||||||
|
appIcon.x_expand = appIcon.y_expand = true;
|
||||||
|
appIcon.x_align = appIcon.y_align = Clutter.ActorAlign.END;
|
||||||
|
|
||||||
|
return appIcon;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const WindowList = new Lang.Class({
|
||||||
|
Name: 'WindowList',
|
||||||
|
Extends: SwitcherPopup.SwitcherList,
|
||||||
|
|
||||||
|
_init : function(windows) {
|
||||||
|
this.parent(true);
|
||||||
|
|
||||||
|
this._label = new St.Label({ x_align: Clutter.ActorAlign.CENTER,
|
||||||
|
y_align: Clutter.ActorAlign.CENTER });
|
||||||
|
this.actor.add_actor(this._label);
|
||||||
|
|
||||||
|
this.windows = windows;
|
||||||
|
this.icons = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < windows.length; i++) {
|
||||||
|
let win = windows[i];
|
||||||
|
let icon = new WindowIcon(win);
|
||||||
|
|
||||||
|
this.addItem(icon.actor, icon.label);
|
||||||
|
this.icons.push(icon);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
_getPreferredHeight: function(actor, forWidth, alloc) {
|
||||||
|
this.parent(actor, forWidth, alloc);
|
||||||
|
|
||||||
|
let spacing = this.actor.get_theme_node().get_padding(St.Side.BOTTOM);
|
||||||
|
let [labelMin, labelNat] = this._label.get_preferred_height(-1);
|
||||||
|
alloc.min_size += labelMin + spacing;
|
||||||
|
alloc.natural_size += labelNat + spacing;
|
||||||
|
},
|
||||||
|
|
||||||
|
_allocateTop: function(actor, box, flags) {
|
||||||
|
let childBox = new Clutter.ActorBox();
|
||||||
|
childBox.x1 = box.x1;
|
||||||
|
childBox.x2 = box.x2;
|
||||||
|
childBox.y2 = box.y2;
|
||||||
|
childBox.y1 = childBox.y2 - this._label.height;
|
||||||
|
this._label.allocate(childBox, flags);
|
||||||
|
|
||||||
|
let spacing = this.actor.get_theme_node().get_padding(St.Side.BOTTOM);
|
||||||
|
box.y2 -= this._label.height + spacing;
|
||||||
|
this.parent(actor, box, flags);
|
||||||
|
},
|
||||||
|
|
||||||
|
highlight: function(index, justOutline) {
|
||||||
|
this.parent(index, justOutline);
|
||||||
|
|
||||||
|
this._label.set_text(index == -1 ? '' : this.icons[index].label.text);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
@ -148,6 +148,12 @@ const WindowManager = new Lang.Class({
|
|||||||
this.setCustomKeybindingHandler('switch-group-backward',
|
this.setCustomKeybindingHandler('switch-group-backward',
|
||||||
Main.KeybindingMode.NORMAL,
|
Main.KeybindingMode.NORMAL,
|
||||||
Lang.bind(this, this._startAppSwitcher));
|
Lang.bind(this, this._startAppSwitcher));
|
||||||
|
this.setCustomKeybindingHandler('switch-windows',
|
||||||
|
Main.KeybindingMode.NORMAL,
|
||||||
|
Lang.bind(this, this._startWindowSwitcher));
|
||||||
|
this.setCustomKeybindingHandler('switch-windows-backward',
|
||||||
|
Main.KeybindingMode.NORMAL,
|
||||||
|
Lang.bind(this, this._startWindowSwitcher));
|
||||||
this.setCustomKeybindingHandler('switch-panels',
|
this.setCustomKeybindingHandler('switch-panels',
|
||||||
Main.KeybindingMode.NORMAL |
|
Main.KeybindingMode.NORMAL |
|
||||||
Main.KeybindingMode.OVERVIEW |
|
Main.KeybindingMode.OVERVIEW |
|
||||||
@ -625,6 +631,19 @@ const WindowManager = new Lang.Class({
|
|||||||
tabPopup.destroy();
|
tabPopup.destroy();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_startWindowSwitcher : function(display, screen, window, binding) {
|
||||||
|
/* prevent a corner case where both popups show up at once */
|
||||||
|
if (this._workspaceSwitcherPopup != null)
|
||||||
|
this._workspaceSwitcherPopup.destroy();
|
||||||
|
|
||||||
|
let tabPopup = new AltTab.WindowSwitcherPopup();
|
||||||
|
|
||||||
|
let modifiers = binding.get_modifiers();
|
||||||
|
let backwards = modifiers & Meta.VirtualModifier.SHIFT_MASK;
|
||||||
|
if (!tabPopup.show(backwards, binding.get_name(), binding.get_mask()))
|
||||||
|
tabPopup.destroy();
|
||||||
|
},
|
||||||
|
|
||||||
_startA11ySwitcher : function(display, screen, window, binding) {
|
_startA11ySwitcher : function(display, screen, window, binding) {
|
||||||
let modifiers = binding.get_modifiers();
|
let modifiers = binding.get_modifiers();
|
||||||
let backwards = modifiers & Meta.VirtualModifier.SHIFT_MASK;
|
let backwards = modifiers & Meta.VirtualModifier.SHIFT_MASK;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user