3794 lines
131 KiB
Diff
3794 lines
131 KiB
Diff
|
From ba93ee43c6521f0bdde8b21fb49a2906a5e36290 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 23 Mar 2022 19:59:14 +0100
|
||
|
Subject: [PATCH 01/28] build: Remove unused stylesheets
|
||
|
|
||
|
The only reason for installing empty stylesheets is minimizing
|
||
|
build system differences between extensions. That's not a very
|
||
|
good reason and we don't do this for other optional files like
|
||
|
schemas.
|
||
|
|
||
|
Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell-extensions/-/merge_requests/223>
|
||
|
---
|
||
|
extensions/apps-menu/meson.build | 1 +
|
||
|
extensions/auto-move-windows/stylesheet.css | 1 -
|
||
|
extensions/drive-menu/meson.build | 1 +
|
||
|
extensions/launch-new-instance/stylesheet.css | 1 -
|
||
|
extensions/meson.build | 2 +-
|
||
|
extensions/meson.build.template | 1 +
|
||
|
extensions/native-window-placement/stylesheet.css | 1 -
|
||
|
extensions/places-menu/meson.build | 1 +
|
||
|
extensions/screenshot-window-sizer/meson.build | 1 +
|
||
|
extensions/user-theme/stylesheet.css | 1 -
|
||
|
extensions/window-list/meson.build | 1 +
|
||
|
extensions/windowsNavigator/meson.build | 1 +
|
||
|
extensions/workspace-indicator/meson.build | 1 +
|
||
|
13 files changed, 9 insertions(+), 5 deletions(-)
|
||
|
delete mode 100644 extensions/auto-move-windows/stylesheet.css
|
||
|
delete mode 100644 extensions/launch-new-instance/stylesheet.css
|
||
|
delete mode 100644 extensions/native-window-placement/stylesheet.css
|
||
|
delete mode 100644 extensions/user-theme/stylesheet.css
|
||
|
|
||
|
diff --git a/extensions/apps-menu/meson.build b/extensions/apps-menu/meson.build
|
||
|
index 48504f63..6b9bb19c 100644
|
||
|
--- a/extensions/apps-menu/meson.build
|
||
|
+++ b/extensions/apps-menu/meson.build
|
||
|
@@ -3,3 +3,4 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
+extension_data += files('stylesheet.css')
|
||
|
diff --git a/extensions/auto-move-windows/stylesheet.css b/extensions/auto-move-windows/stylesheet.css
|
||
|
deleted file mode 100644
|
||
|
index 25134b65..00000000
|
||
|
--- a/extensions/auto-move-windows/stylesheet.css
|
||
|
+++ /dev/null
|
||
|
@@ -1 +0,0 @@
|
||
|
-/* This extensions requires no special styling */
|
||
|
diff --git a/extensions/drive-menu/meson.build b/extensions/drive-menu/meson.build
|
||
|
index 48504f63..6b9bb19c 100644
|
||
|
--- a/extensions/drive-menu/meson.build
|
||
|
+++ b/extensions/drive-menu/meson.build
|
||
|
@@ -3,3 +3,4 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
+extension_data += files('stylesheet.css')
|
||
|
diff --git a/extensions/launch-new-instance/stylesheet.css b/extensions/launch-new-instance/stylesheet.css
|
||
|
deleted file mode 100644
|
||
|
index 25134b65..00000000
|
||
|
--- a/extensions/launch-new-instance/stylesheet.css
|
||
|
+++ /dev/null
|
||
|
@@ -1 +0,0 @@
|
||
|
-/* This extensions requires no special styling */
|
||
|
diff --git a/extensions/meson.build b/extensions/meson.build
|
||
|
index ca00d01a..6f60b08d 100644
|
||
|
--- a/extensions/meson.build
|
||
|
+++ b/extensions/meson.build
|
||
|
@@ -15,7 +15,7 @@ foreach e : enabled_extensions
|
||
|
metadata_conf.set('url', 'https://gitlab.gnome.org/GNOME/gnome-shell-extensions')
|
||
|
|
||
|
extension_sources = files(e + '/extension.js')
|
||
|
- extension_data = files(e + '/stylesheet.css')
|
||
|
+ extension_data = []
|
||
|
|
||
|
subdir(e)
|
||
|
|
||
|
diff --git a/extensions/meson.build.template b/extensions/meson.build.template
|
||
|
index e83e528b..a9915994 100644
|
||
|
--- a/extensions/meson.build.template
|
||
|
+++ b/extensions/meson.build.template
|
||
|
@@ -4,5 +4,6 @@ extension_data += configure_file(
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
|
||
|
+# extension_data += files('stylesheet.css')
|
||
|
# extension_sources += files('prefs.js')
|
||
|
# extension_schemas += files(metadata_conf.get('gschemaname') + '.gschema.xml')
|
||
|
diff --git a/extensions/native-window-placement/stylesheet.css b/extensions/native-window-placement/stylesheet.css
|
||
|
deleted file mode 100644
|
||
|
index 25134b65..00000000
|
||
|
--- a/extensions/native-window-placement/stylesheet.css
|
||
|
+++ /dev/null
|
||
|
@@ -1 +0,0 @@
|
||
|
-/* This extensions requires no special styling */
|
||
|
diff --git a/extensions/places-menu/meson.build b/extensions/places-menu/meson.build
|
||
|
index d9a59691..cbc2a02b 100644
|
||
|
--- a/extensions/places-menu/meson.build
|
||
|
+++ b/extensions/places-menu/meson.build
|
||
|
@@ -3,5 +3,6 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
+extension_data += files('stylesheet.css')
|
||
|
|
||
|
extension_sources += files('placeDisplay.js')
|
||
|
diff --git a/extensions/screenshot-window-sizer/meson.build b/extensions/screenshot-window-sizer/meson.build
|
||
|
index 585c02da..8257dee0 100644
|
||
|
--- a/extensions/screenshot-window-sizer/meson.build
|
||
|
+++ b/extensions/screenshot-window-sizer/meson.build
|
||
|
@@ -3,5 +3,6 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
+extension_data += files('stylesheet.css')
|
||
|
|
||
|
extension_schemas += files(metadata_conf.get('gschemaname') + '.gschema.xml')
|
||
|
diff --git a/extensions/user-theme/stylesheet.css b/extensions/user-theme/stylesheet.css
|
||
|
deleted file mode 100644
|
||
|
index 6d914832..00000000
|
||
|
--- a/extensions/user-theme/stylesheet.css
|
||
|
+++ /dev/null
|
||
|
@@ -1 +0,0 @@
|
||
|
-/* none used */
|
||
|
diff --git a/extensions/window-list/meson.build b/extensions/window-list/meson.build
|
||
|
index 34d7c3fd..599f45e1 100644
|
||
|
--- a/extensions/window-list/meson.build
|
||
|
+++ b/extensions/window-list/meson.build
|
||
|
@@ -3,6 +3,7 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
+extension_data += files('stylesheet.css')
|
||
|
|
||
|
extension_sources += files('prefs.js', 'windowPicker.js', 'workspaceIndicator.js')
|
||
|
extension_schemas += files(metadata_conf.get('gschemaname') + '.gschema.xml')
|
||
|
diff --git a/extensions/windowsNavigator/meson.build b/extensions/windowsNavigator/meson.build
|
||
|
index 48504f63..6b9bb19c 100644
|
||
|
--- a/extensions/windowsNavigator/meson.build
|
||
|
+++ b/extensions/windowsNavigator/meson.build
|
||
|
@@ -3,3 +3,4 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
+extension_data += files('stylesheet.css')
|
||
|
diff --git a/extensions/workspace-indicator/meson.build b/extensions/workspace-indicator/meson.build
|
||
|
index 71efa039..19858a39 100644
|
||
|
--- a/extensions/workspace-indicator/meson.build
|
||
|
+++ b/extensions/workspace-indicator/meson.build
|
||
|
@@ -3,5 +3,6 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
+extension_data += files('stylesheet.css')
|
||
|
|
||
|
extension_sources += files('prefs.js')
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From b5ec53b2fcc134a667e92bccb93672e0f286e9f2 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 12:38:33 +0100
|
||
|
Subject: [PATCH 02/28] workspace-indicator: Move indicator code into separate
|
||
|
file
|
||
|
|
||
|
Shortly after the window-list extension was added, it gained a
|
||
|
workspace switcher based on the workspace indicator extension.
|
||
|
|
||
|
Duplicating the code wasn't a big issue while the switcher was
|
||
|
a simple menu, but since it gained previews with a fair bit of
|
||
|
custom styling, syncing changes between the two extensions has
|
||
|
become tedious, in particular as the two copies have slightly
|
||
|
diverged over time.
|
||
|
|
||
|
In order to allow the two copies to converge again, the indicator
|
||
|
code needs to be separate from the extension boilerplate, so
|
||
|
split out the code into a separate module.
|
||
|
---
|
||
|
extensions/workspace-indicator/extension.js | 440 +----------------
|
||
|
extensions/workspace-indicator/meson.build | 2 +-
|
||
|
.../workspace-indicator/workspaceIndicator.js | 454 ++++++++++++++++++
|
||
|
po/POTFILES.in | 2 +-
|
||
|
4 files changed, 457 insertions(+), 441 deletions(-)
|
||
|
create mode 100644 extensions/workspace-indicator/workspaceIndicator.js
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/extension.js b/extensions/workspace-indicator/extension.js
|
||
|
index 6974062b..5e1ed8e5 100644
|
||
|
--- a/extensions/workspace-indicator/extension.js
|
||
|
+++ b/extensions/workspace-indicator/extension.js
|
||
|
@@ -1,449 +1,11 @@
|
||
|
// -*- mode: js2; indent-tabs-mode: nil; js2-basic-offset: 4 -*-
|
||
|
/* exported init enable disable */
|
||
|
|
||
|
-const { Clutter, Gio, GObject, Meta, St } = imports.gi;
|
||
|
-
|
||
|
-const DND = imports.ui.dnd;
|
||
|
const ExtensionUtils = imports.misc.extensionUtils;
|
||
|
const Main = imports.ui.main;
|
||
|
-const PanelMenu = imports.ui.panelMenu;
|
||
|
-const PopupMenu = imports.ui.popupMenu;
|
||
|
|
||
|
const Me = ExtensionUtils.getCurrentExtension();
|
||
|
-
|
||
|
-const Gettext = imports.gettext.domain(Me.metadata['gettext-domain']);
|
||
|
-const _ = Gettext.gettext;
|
||
|
-
|
||
|
-const WORKSPACE_SCHEMA = 'org.gnome.desktop.wm.preferences';
|
||
|
-const WORKSPACE_KEY = 'workspace-names';
|
||
|
-
|
||
|
-const TOOLTIP_OFFSET = 6;
|
||
|
-const TOOLTIP_ANIMATION_TIME = 150;
|
||
|
-
|
||
|
-const MAX_THUMBNAILS = 6;
|
||
|
-
|
||
|
-let WindowPreview = GObject.registerClass(
|
||
|
-class WindowPreview extends St.Button {
|
||
|
- _init(window) {
|
||
|
- super._init({
|
||
|
- style_class: 'workspace-indicator-window-preview',
|
||
|
- });
|
||
|
-
|
||
|
- this._delegate = this;
|
||
|
- DND.makeDraggable(this, { restoreOnSuccess: true });
|
||
|
-
|
||
|
- this._window = window;
|
||
|
-
|
||
|
- this.connect('destroy', this._onDestroy.bind(this));
|
||
|
-
|
||
|
- this._sizeChangedId = this._window.connect('size-changed',
|
||
|
- () => this.queue_relayout());
|
||
|
- this._positionChangedId = this._window.connect('position-changed',
|
||
|
- () => {
|
||
|
- this._updateVisible();
|
||
|
- this.queue_relayout();
|
||
|
- });
|
||
|
- this._minimizedChangedId = this._window.connect('notify::minimized',
|
||
|
- this._updateVisible.bind(this));
|
||
|
-
|
||
|
- this._focusChangedId = global.display.connect('notify::focus-window',
|
||
|
- this._onFocusChanged.bind(this));
|
||
|
- this._onFocusChanged();
|
||
|
- }
|
||
|
-
|
||
|
- // needed for DND
|
||
|
- get metaWindow() {
|
||
|
- return this._window;
|
||
|
- }
|
||
|
-
|
||
|
- _onDestroy() {
|
||
|
- this._window.disconnect(this._sizeChangedId);
|
||
|
- this._window.disconnect(this._positionChangedId);
|
||
|
- this._window.disconnect(this._minimizedChangedId);
|
||
|
- global.display.disconnect(this._focusChangedId);
|
||
|
- }
|
||
|
-
|
||
|
- _onFocusChanged() {
|
||
|
- if (global.display.focus_window === this._window)
|
||
|
- this.add_style_class_name('active');
|
||
|
- else
|
||
|
- this.remove_style_class_name('active');
|
||
|
- }
|
||
|
-
|
||
|
- _updateVisible() {
|
||
|
- const monitor = Main.layoutManager.findIndexForActor(this);
|
||
|
- const workArea = Main.layoutManager.getWorkAreaForMonitor(monitor);
|
||
|
- this.visible = this._window.get_frame_rect().overlap(workArea) &&
|
||
|
- this._window.window_type !== Meta.WindowType.DESKTOP &&
|
||
|
- this._window.showing_on_its_workspace();
|
||
|
- }
|
||
|
-});
|
||
|
-
|
||
|
-let WorkspaceLayout = GObject.registerClass(
|
||
|
-class WorkspaceLayout extends Clutter.LayoutManager {
|
||
|
- vfunc_get_preferred_width() {
|
||
|
- return [0, 0];
|
||
|
- }
|
||
|
-
|
||
|
- vfunc_get_preferred_height() {
|
||
|
- return [0, 0];
|
||
|
- }
|
||
|
-
|
||
|
- vfunc_allocate(container, box) {
|
||
|
- const monitor = Main.layoutManager.findIndexForActor(container);
|
||
|
- const workArea = Main.layoutManager.getWorkAreaForMonitor(monitor);
|
||
|
- const hscale = box.get_width() / workArea.width;
|
||
|
- const vscale = box.get_height() / workArea.height;
|
||
|
-
|
||
|
- for (const child of container) {
|
||
|
- const childBox = new Clutter.ActorBox();
|
||
|
- const frameRect = child.metaWindow.get_frame_rect();
|
||
|
- childBox.set_size(
|
||
|
- Math.round(Math.min(frameRect.width, workArea.width) * hscale),
|
||
|
- Math.round(Math.min(frameRect.height, workArea.height) * vscale));
|
||
|
- childBox.set_origin(
|
||
|
- Math.round((frameRect.x - workArea.x) * hscale),
|
||
|
- Math.round((frameRect.y - workArea.y) * vscale));
|
||
|
- child.allocate(childBox);
|
||
|
- }
|
||
|
- }
|
||
|
-});
|
||
|
-
|
||
|
-let WorkspaceThumbnail = GObject.registerClass(
|
||
|
-class WorkspaceThumbnail extends St.Button {
|
||
|
- _init(index) {
|
||
|
- super._init({
|
||
|
- style_class: 'workspace',
|
||
|
- child: new Clutter.Actor({
|
||
|
- layout_manager: new WorkspaceLayout(),
|
||
|
- clip_to_allocation: true,
|
||
|
- }),
|
||
|
- });
|
||
|
-
|
||
|
- this._tooltip = new St.Label({
|
||
|
- style_class: 'dash-label',
|
||
|
- visible: false,
|
||
|
- });
|
||
|
- Main.uiGroup.add_child(this._tooltip);
|
||
|
-
|
||
|
- this.connect('destroy', this._onDestroy.bind(this));
|
||
|
- this.connect('notify::hover', this._syncTooltip.bind(this));
|
||
|
-
|
||
|
- this._index = index;
|
||
|
- this._delegate = this; // needed for DND
|
||
|
-
|
||
|
- this._windowPreviews = new Map();
|
||
|
-
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
- this._workspace = workspaceManager.get_workspace_by_index(index);
|
||
|
-
|
||
|
- this._windowAddedId = this._workspace.connect('window-added',
|
||
|
- (ws, window) => {
|
||
|
- this._addWindow(window);
|
||
|
- });
|
||
|
- this._windowRemovedId = this._workspace.connect('window-removed',
|
||
|
- (ws, window) => {
|
||
|
- this._removeWindow(window);
|
||
|
- });
|
||
|
- this._restackedId = global.display.connect('restacked',
|
||
|
- this._onRestacked.bind(this));
|
||
|
-
|
||
|
- this._workspace.list_windows().forEach(w => this._addWindow(w));
|
||
|
- this._onRestacked();
|
||
|
- }
|
||
|
-
|
||
|
- acceptDrop(source) {
|
||
|
- if (!source.metaWindow)
|
||
|
- return false;
|
||
|
-
|
||
|
- this._moveWindow(source.metaWindow);
|
||
|
- return true;
|
||
|
- }
|
||
|
-
|
||
|
- handleDragOver(source) {
|
||
|
- if (source.metaWindow)
|
||
|
- return DND.DragMotionResult.MOVE_DROP;
|
||
|
- else
|
||
|
- return DND.DragMotionResult.CONTINUE;
|
||
|
- }
|
||
|
-
|
||
|
- _addWindow(window) {
|
||
|
- if (this._windowPreviews.has(window))
|
||
|
- return;
|
||
|
-
|
||
|
- let preview = new WindowPreview(window);
|
||
|
- preview.connect('clicked', (a, btn) => this.emit('clicked', btn));
|
||
|
- this._windowPreviews.set(window, preview);
|
||
|
- this.child.add_child(preview);
|
||
|
- }
|
||
|
-
|
||
|
- _removeWindow(window) {
|
||
|
- let preview = this._windowPreviews.get(window);
|
||
|
- if (!preview)
|
||
|
- return;
|
||
|
-
|
||
|
- this._windowPreviews.delete(window);
|
||
|
- preview.destroy();
|
||
|
- }
|
||
|
-
|
||
|
- _onRestacked() {
|
||
|
- let lastPreview = null;
|
||
|
- let windows = global.get_window_actors().map(a => a.meta_window);
|
||
|
- for (let i = 0; i < windows.length; i++) {
|
||
|
- let preview = this._windowPreviews.get(windows[i]);
|
||
|
- if (!preview)
|
||
|
- continue;
|
||
|
-
|
||
|
- this.child.set_child_above_sibling(preview, lastPreview);
|
||
|
- lastPreview = preview;
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _moveWindow(window) {
|
||
|
- let monitorIndex = Main.layoutManager.findIndexForActor(this);
|
||
|
- if (monitorIndex !== window.get_monitor())
|
||
|
- window.move_to_monitor(monitorIndex);
|
||
|
- window.change_workspace_by_index(this._index, false);
|
||
|
- }
|
||
|
-
|
||
|
- on_clicked() {
|
||
|
- let ws = global.workspace_manager.get_workspace_by_index(this._index);
|
||
|
- if (ws)
|
||
|
- ws.activate(global.get_current_time());
|
||
|
- }
|
||
|
-
|
||
|
- _syncTooltip() {
|
||
|
- if (this.hover) {
|
||
|
- this._tooltip.set({
|
||
|
- text: Meta.prefs_get_workspace_name(this._index),
|
||
|
- visible: true,
|
||
|
- opacity: 0,
|
||
|
- });
|
||
|
-
|
||
|
- const [stageX, stageY] = this.get_transformed_position();
|
||
|
- const thumbWidth = this.allocation.get_width();
|
||
|
- const thumbHeight = this.allocation.get_height();
|
||
|
- const tipWidth = this._tooltip.width;
|
||
|
- const xOffset = Math.floor((thumbWidth - tipWidth) / 2);
|
||
|
- const monitor = Main.layoutManager.findMonitorForActor(this);
|
||
|
- const x = Math.clamp(
|
||
|
- stageX + xOffset,
|
||
|
- monitor.x,
|
||
|
- monitor.x + monitor.width - tipWidth);
|
||
|
- const y = stageY + thumbHeight + TOOLTIP_OFFSET;
|
||
|
- this._tooltip.set_position(x, y);
|
||
|
- }
|
||
|
-
|
||
|
- this._tooltip.ease({
|
||
|
- opacity: this.hover ? 255 : 0,
|
||
|
- duration: TOOLTIP_ANIMATION_TIME,
|
||
|
- mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
|
- onComplete: () => (this._tooltip.visible = this.hover),
|
||
|
- });
|
||
|
- }
|
||
|
-
|
||
|
- _onDestroy() {
|
||
|
- this._tooltip.destroy();
|
||
|
-
|
||
|
- this._workspace.disconnect(this._windowAddedId);
|
||
|
- this._workspace.disconnect(this._windowRemovedId);
|
||
|
- global.display.disconnect(this._restackedId);
|
||
|
- }
|
||
|
-});
|
||
|
-
|
||
|
-let WorkspaceIndicator = GObject.registerClass(
|
||
|
-class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
- _init() {
|
||
|
- super._init(0.0, _('Workspace Indicator'));
|
||
|
-
|
||
|
- let container = new St.Widget({
|
||
|
- layout_manager: new Clutter.BinLayout(),
|
||
|
- x_expand: true,
|
||
|
- y_expand: true,
|
||
|
- });
|
||
|
- this.add_actor(container);
|
||
|
-
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
- this._statusLabel = new St.Label({
|
||
|
- style_class: 'panel-workspace-indicator',
|
||
|
- y_align: Clutter.ActorAlign.CENTER,
|
||
|
- text: this._labelText(),
|
||
|
- });
|
||
|
-
|
||
|
- container.add_actor(this._statusLabel);
|
||
|
-
|
||
|
- this._thumbnailsBox = new St.BoxLayout({
|
||
|
- style_class: 'panel-workspace-indicator-box',
|
||
|
- y_expand: true,
|
||
|
- reactive: true,
|
||
|
- });
|
||
|
-
|
||
|
- container.add_actor(this._thumbnailsBox);
|
||
|
-
|
||
|
- this._workspacesItems = [];
|
||
|
- this._workspaceSection = new PopupMenu.PopupMenuSection();
|
||
|
- this.menu.addMenuItem(this._workspaceSection);
|
||
|
-
|
||
|
- this._workspaceManagerSignals = [
|
||
|
- workspaceManager.connect_after('notify::n-workspaces',
|
||
|
- this._nWorkspacesChanged.bind(this)),
|
||
|
- workspaceManager.connect_after('workspace-switched',
|
||
|
- this._onWorkspaceSwitched.bind(this)),
|
||
|
- workspaceManager.connect('notify::layout-rows',
|
||
|
- this._updateThumbnailVisibility.bind(this)),
|
||
|
- ];
|
||
|
-
|
||
|
- this.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
- this._thumbnailsBox.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
- this._createWorkspacesSection();
|
||
|
- this._updateThumbnails();
|
||
|
- this._updateThumbnailVisibility();
|
||
|
-
|
||
|
- this._settings = new Gio.Settings({ schema_id: WORKSPACE_SCHEMA });
|
||
|
- this._settingsChangedId = this._settings.connect(
|
||
|
- `changed::${WORKSPACE_KEY}`,
|
||
|
- this._updateMenuLabels.bind(this));
|
||
|
- }
|
||
|
-
|
||
|
- _onDestroy() {
|
||
|
- for (let i = 0; i < this._workspaceManagerSignals.length; i++)
|
||
|
- global.workspace_manager.disconnect(this._workspaceManagerSignals[i]);
|
||
|
-
|
||
|
- if (this._settingsChangedId) {
|
||
|
- this._settings.disconnect(this._settingsChangedId);
|
||
|
- this._settingsChangedId = 0;
|
||
|
- }
|
||
|
-
|
||
|
- Main.panel.set_offscreen_redirect(Clutter.OffscreenRedirect.ALWAYS);
|
||
|
-
|
||
|
- super._onDestroy();
|
||
|
- }
|
||
|
-
|
||
|
- _updateThumbnailVisibility() {
|
||
|
- const { workspaceManager } = global;
|
||
|
- const vertical = workspaceManager.layout_rows === -1;
|
||
|
- const useMenu =
|
||
|
- vertical || workspaceManager.n_workspaces > MAX_THUMBNAILS;
|
||
|
- this.reactive = useMenu;
|
||
|
-
|
||
|
- this._statusLabel.visible = useMenu;
|
||
|
- this._thumbnailsBox.visible = !useMenu;
|
||
|
-
|
||
|
- // Disable offscreen-redirect when showing the workspace switcher
|
||
|
- // so that clip-to-allocation works
|
||
|
- Main.panel.set_offscreen_redirect(useMenu
|
||
|
- ? Clutter.OffscreenRedirect.ALWAYS
|
||
|
- : Clutter.OffscreenRedirect.AUTOMATIC_FOR_OPACITY);
|
||
|
- }
|
||
|
-
|
||
|
- _onWorkspaceSwitched() {
|
||
|
- this._currentWorkspace = global.workspace_manager.get_active_workspace_index();
|
||
|
-
|
||
|
- this._updateMenuOrnament();
|
||
|
- this._updateActiveThumbnail();
|
||
|
-
|
||
|
- this._statusLabel.set_text(this._labelText());
|
||
|
- }
|
||
|
-
|
||
|
- _nWorkspacesChanged() {
|
||
|
- this._createWorkspacesSection();
|
||
|
- this._updateThumbnails();
|
||
|
- this._updateThumbnailVisibility();
|
||
|
- }
|
||
|
-
|
||
|
- _updateMenuOrnament() {
|
||
|
- for (let i = 0; i < this._workspacesItems.length; i++) {
|
||
|
- this._workspacesItems[i].setOrnament(i === this._currentWorkspace
|
||
|
- ? PopupMenu.Ornament.DOT
|
||
|
- : PopupMenu.Ornament.NONE);
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _updateActiveThumbnail() {
|
||
|
- let thumbs = this._thumbnailsBox.get_children();
|
||
|
- for (let i = 0; i < thumbs.length; i++) {
|
||
|
- if (i === this._currentWorkspace)
|
||
|
- thumbs[i].add_style_class_name('active');
|
||
|
- else
|
||
|
- thumbs[i].remove_style_class_name('active');
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _labelText(workspaceIndex) {
|
||
|
- if (workspaceIndex === undefined) {
|
||
|
- workspaceIndex = this._currentWorkspace;
|
||
|
- return (workspaceIndex + 1).toString();
|
||
|
- }
|
||
|
- return Meta.prefs_get_workspace_name(workspaceIndex);
|
||
|
- }
|
||
|
-
|
||
|
- _updateMenuLabels() {
|
||
|
- for (let i = 0; i < this._workspacesItems.length; i++)
|
||
|
- this._workspacesItems[i].label.text = this._labelText(i);
|
||
|
- }
|
||
|
-
|
||
|
- _createWorkspacesSection() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this._workspaceSection.removeAll();
|
||
|
- this._workspacesItems = [];
|
||
|
- this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
-
|
||
|
- let i = 0;
|
||
|
- for (; i < workspaceManager.n_workspaces; i++) {
|
||
|
- this._workspacesItems[i] = new PopupMenu.PopupMenuItem(this._labelText(i));
|
||
|
- this._workspaceSection.addMenuItem(this._workspacesItems[i]);
|
||
|
- this._workspacesItems[i].workspaceId = i;
|
||
|
- this._workspacesItems[i].label_actor = this._statusLabel;
|
||
|
- this._workspacesItems[i].connect('activate', (actor, _event) => {
|
||
|
- this._activate(actor.workspaceId);
|
||
|
- });
|
||
|
-
|
||
|
- if (i === this._currentWorkspace)
|
||
|
- this._workspacesItems[i].setOrnament(PopupMenu.Ornament.DOT);
|
||
|
- }
|
||
|
-
|
||
|
- this._statusLabel.set_text(this._labelText());
|
||
|
- }
|
||
|
-
|
||
|
- _updateThumbnails() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this._thumbnailsBox.destroy_all_children();
|
||
|
-
|
||
|
- for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
- let thumb = new WorkspaceThumbnail(i);
|
||
|
- this._thumbnailsBox.add_actor(thumb);
|
||
|
- }
|
||
|
- this._updateActiveThumbnail();
|
||
|
- }
|
||
|
-
|
||
|
- _activate(index) {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- if (index >= 0 && index < workspaceManager.n_workspaces) {
|
||
|
- let metaWorkspace = workspaceManager.get_workspace_by_index(index);
|
||
|
- metaWorkspace.activate(global.get_current_time());
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _onScrollEvent(actor, event) {
|
||
|
- let direction = event.get_scroll_direction();
|
||
|
- let diff = 0;
|
||
|
- if (direction === Clutter.ScrollDirection.DOWN)
|
||
|
- diff = 1;
|
||
|
- else if (direction === Clutter.ScrollDirection.UP)
|
||
|
- diff = -1;
|
||
|
- else
|
||
|
- return;
|
||
|
-
|
||
|
-
|
||
|
- let newIndex = global.workspace_manager.get_active_workspace_index() + diff;
|
||
|
- this._activate(newIndex);
|
||
|
- }
|
||
|
-});
|
||
|
+const { WorkspaceIndicator } = Me.imports.workspaceIndicator;
|
||
|
|
||
|
function init() {
|
||
|
ExtensionUtils.initTranslations();
|
||
|
diff --git a/extensions/workspace-indicator/meson.build b/extensions/workspace-indicator/meson.build
|
||
|
index 19858a39..eb25b9cc 100644
|
||
|
--- a/extensions/workspace-indicator/meson.build
|
||
|
+++ b/extensions/workspace-indicator/meson.build
|
||
|
@@ -5,4 +5,4 @@ extension_data += configure_file(
|
||
|
)
|
||
|
extension_data += files('stylesheet.css')
|
||
|
|
||
|
-extension_sources += files('prefs.js')
|
||
|
+extension_sources += files('prefs.js', 'workspaceIndicator.js')
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
new file mode 100644
|
||
|
index 00000000..b98de047
|
||
|
--- /dev/null
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -0,0 +1,454 @@
|
||
|
+// SPDX-FileCopyrightText: 2011 Erick Pérez Castellanos <erick.red@gmail.com>
|
||
|
+// SPDX-FileCopyrightText: 2011 Giovanni Campagna <gcampagna@src.gnome.org>
|
||
|
+// SPDX-FileCopyrightText: 2017 Florian Müllner <fmuellner@gnome.org>
|
||
|
+//
|
||
|
+// SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
+
|
||
|
+const { Clutter, Gio, GObject, Meta, St } = imports.gi;
|
||
|
+
|
||
|
+const ExtensionUtils = imports.misc.extensionUtils;
|
||
|
+const Me = ExtensionUtils.getCurrentExtension();
|
||
|
+
|
||
|
+const Gettext = imports.gettext.domain(Me.metadata['gettext-domain']);
|
||
|
+const _ = Gettext.gettext;
|
||
|
+
|
||
|
+const DND = imports.ui.dnd;
|
||
|
+const Main = imports.ui.main;
|
||
|
+const PanelMenu = imports.ui.panelMenu;
|
||
|
+const PopupMenu = imports.ui.popupMenu;
|
||
|
+
|
||
|
+const WORKSPACE_SCHEMA = 'org.gnome.desktop.wm.preferences';
|
||
|
+const WORKSPACE_KEY = 'workspace-names';
|
||
|
+
|
||
|
+const TOOLTIP_OFFSET = 6;
|
||
|
+const TOOLTIP_ANIMATION_TIME = 150;
|
||
|
+
|
||
|
+const MAX_THUMBNAILS = 6;
|
||
|
+
|
||
|
+const WindowPreview = GObject.registerClass(
|
||
|
+class WindowPreview extends St.Button {
|
||
|
+ _init(window) {
|
||
|
+ super._init({
|
||
|
+ style_class: 'workspace-indicator-window-preview',
|
||
|
+ });
|
||
|
+
|
||
|
+ this._delegate = this;
|
||
|
+ DND.makeDraggable(this, {restoreOnSuccess: true});
|
||
|
+
|
||
|
+ this._window = window;
|
||
|
+
|
||
|
+ this.connect('destroy', this._onDestroy.bind(this));
|
||
|
+
|
||
|
+ this._sizeChangedId = this._window.connect('size-changed',
|
||
|
+ () => this._checkRelayout());
|
||
|
+ this._positionChangedId = this._window.connect('position-changed',
|
||
|
+ () => this._checkRelayout());
|
||
|
+ this._minimizedChangedId = this._window.connect('notify::minimized',
|
||
|
+ () => this._updateVisible());
|
||
|
+ this._typeChangedId = this._window.connect('notify::window-type',
|
||
|
+ () => this._updateVisible());
|
||
|
+ this._updateVisible();
|
||
|
+
|
||
|
+ this._focusChangedId = global.display.connect('notify::focus-window',
|
||
|
+ this._onFocusChanged.bind(this));
|
||
|
+ this._onFocusChanged();
|
||
|
+ }
|
||
|
+
|
||
|
+ // needed for DND
|
||
|
+ get metaWindow() {
|
||
|
+ return this._window;
|
||
|
+ }
|
||
|
+
|
||
|
+ _onDestroy() {
|
||
|
+ this._window.disconnect(this._sizeChangedId);
|
||
|
+ this._window.disconnect(this._positionChangedId);
|
||
|
+ this._window.disconnect(this._minimizedChangedId);
|
||
|
+ this._window.disconnect(this._typeChangedId);
|
||
|
+ global.display.disconnect(this._focusChangedId);
|
||
|
+ }
|
||
|
+
|
||
|
+ _onFocusChanged() {
|
||
|
+ if (global.display.focus_window === this._window)
|
||
|
+ this.add_style_class_name('active');
|
||
|
+ else
|
||
|
+ this.remove_style_class_name('active');
|
||
|
+ }
|
||
|
+
|
||
|
+ _checkRelayout() {
|
||
|
+ const monitor = Main.layoutManager.findIndexForActor(this);
|
||
|
+ const workArea = Main.layoutManager.getWorkAreaForMonitor(monitor);
|
||
|
+ if (this._window.get_frame_rect().overlap(workArea))
|
||
|
+ this.queue_relayout();
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateVisible() {
|
||
|
+ this.visible = this._window.window_type !== Meta.WindowType.DESKTOP &&
|
||
|
+ this._window.showing_on_its_workspace();
|
||
|
+ }
|
||
|
+});
|
||
|
+
|
||
|
+const WorkspaceLayout = GObject.registerClass(
|
||
|
+class WorkspaceLayout extends Clutter.LayoutManager {
|
||
|
+ vfunc_get_preferred_width() {
|
||
|
+ return [0, 0];
|
||
|
+ }
|
||
|
+
|
||
|
+ vfunc_get_preferred_height() {
|
||
|
+ return [0, 0];
|
||
|
+ }
|
||
|
+
|
||
|
+ vfunc_allocate(container, box) {
|
||
|
+ const monitor = Main.layoutManager.findIndexForActor(container);
|
||
|
+ const workArea = Main.layoutManager.getWorkAreaForMonitor(monitor);
|
||
|
+ const hscale = box.get_width() / workArea.width;
|
||
|
+ const vscale = box.get_height() / workArea.height;
|
||
|
+
|
||
|
+ for (const child of container) {
|
||
|
+ const childBox = new Clutter.ActorBox();
|
||
|
+ const frameRect = child.metaWindow.get_frame_rect();
|
||
|
+ childBox.set_size(
|
||
|
+ Math.round(Math.min(frameRect.width, workArea.width) * hscale),
|
||
|
+ Math.round(Math.min(frameRect.height, workArea.height) * vscale));
|
||
|
+ childBox.set_origin(
|
||
|
+ Math.round((frameRect.x - workArea.x) * hscale),
|
||
|
+ Math.round((frameRect.y - workArea.y) * vscale));
|
||
|
+ child.allocate(childBox);
|
||
|
+ }
|
||
|
+ }
|
||
|
+});
|
||
|
+
|
||
|
+const WorkspaceThumbnail = GObject.registerClass(
|
||
|
+class WorkspaceThumbnail extends St.Button {
|
||
|
+ _init(index) {
|
||
|
+ super._init({
|
||
|
+ style_class: 'workspace',
|
||
|
+ child: new Clutter.Actor({
|
||
|
+ layout_manager: new WorkspaceLayout(),
|
||
|
+ clip_to_allocation: true,
|
||
|
+ x_expand: true,
|
||
|
+ y_expand: true,
|
||
|
+ }),
|
||
|
+ });
|
||
|
+
|
||
|
+ this._tooltip = new St.Label({
|
||
|
+ style_class: 'dash-label',
|
||
|
+ visible: false,
|
||
|
+ });
|
||
|
+ Main.uiGroup.add_child(this._tooltip);
|
||
|
+
|
||
|
+ this.connect('destroy', this._onDestroy.bind(this));
|
||
|
+ this.connect('notify::hover', this._syncTooltip.bind(this));
|
||
|
+
|
||
|
+ this._index = index;
|
||
|
+ this._delegate = this; // needed for DND
|
||
|
+
|
||
|
+ this._windowPreviews = new Map();
|
||
|
+
|
||
|
+ let workspaceManager = global.workspace_manager;
|
||
|
+ this._workspace = workspaceManager.get_workspace_by_index(index);
|
||
|
+
|
||
|
+ this._windowAddedId = this._workspace.connect('window-added',
|
||
|
+ (ws, window) => this._addWindow(window));
|
||
|
+ this._windowRemovedId = this._workspace.connect('window-removed',
|
||
|
+ (ws, window) => this._removeWindow(window));
|
||
|
+
|
||
|
+ this._restackedId = global.display.connect('restacked',
|
||
|
+ this._onRestacked.bind(this));
|
||
|
+
|
||
|
+ this._workspace.list_windows().forEach(w => this._addWindow(w));
|
||
|
+ this._onRestacked();
|
||
|
+ }
|
||
|
+
|
||
|
+ acceptDrop(source) {
|
||
|
+ if (!source.metaWindow)
|
||
|
+ return false;
|
||
|
+
|
||
|
+ this._moveWindow(source.metaWindow);
|
||
|
+ return true;
|
||
|
+ }
|
||
|
+
|
||
|
+ handleDragOver(source) {
|
||
|
+ if (source.metaWindow)
|
||
|
+ return DND.DragMotionResult.MOVE_DROP;
|
||
|
+ else
|
||
|
+ return DND.DragMotionResult.CONTINUE;
|
||
|
+ }
|
||
|
+
|
||
|
+ _addWindow(window) {
|
||
|
+ if (this._windowPreviews.has(window))
|
||
|
+ return;
|
||
|
+
|
||
|
+ let preview = new WindowPreview(window);
|
||
|
+ preview.connect('clicked', (a, btn) => this.emit('clicked', btn));
|
||
|
+ this._windowPreviews.set(window, preview);
|
||
|
+ this.child.add_child(preview);
|
||
|
+ }
|
||
|
+
|
||
|
+ _removeWindow(window) {
|
||
|
+ let preview = this._windowPreviews.get(window);
|
||
|
+ if (!preview)
|
||
|
+ return;
|
||
|
+
|
||
|
+ this._windowPreviews.delete(window);
|
||
|
+ preview.destroy();
|
||
|
+ }
|
||
|
+
|
||
|
+ _onRestacked() {
|
||
|
+ let lastPreview = null;
|
||
|
+ let windows = global.get_window_actors().map(a => a.meta_window);
|
||
|
+ for (let i = 0; i < windows.length; i++) {
|
||
|
+ let preview = this._windowPreviews.get(windows[i]);
|
||
|
+ if (!preview)
|
||
|
+ continue;
|
||
|
+
|
||
|
+ this.child.set_child_above_sibling(preview, lastPreview);
|
||
|
+ lastPreview = preview;
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ _moveWindow(window) {
|
||
|
+ let monitorIndex = Main.layoutManager.findIndexForActor(this);
|
||
|
+ if (monitorIndex !== window.get_monitor())
|
||
|
+ window.move_to_monitor(monitorIndex);
|
||
|
+ window.change_workspace_by_index(this._index, false);
|
||
|
+ }
|
||
|
+
|
||
|
+ on_clicked() {
|
||
|
+ let ws = global.workspace_manager.get_workspace_by_index(this._index);
|
||
|
+ if (ws)
|
||
|
+ ws.activate(global.get_current_time());
|
||
|
+ }
|
||
|
+
|
||
|
+ _syncTooltip() {
|
||
|
+ if (this.hover) {
|
||
|
+ this._tooltip.set({
|
||
|
+ text: Meta.prefs_get_workspace_name(this._index),
|
||
|
+ visible: true,
|
||
|
+ opacity: 0,
|
||
|
+ });
|
||
|
+
|
||
|
+ const [stageX, stageY] = this.get_transformed_position();
|
||
|
+ const thumbWidth = this.allocation.get_width();
|
||
|
+ const thumbHeight = this.allocation.get_height();
|
||
|
+ const tipWidth = this._tooltip.width;
|
||
|
+ const xOffset = Math.floor((thumbWidth - tipWidth) / 2);
|
||
|
+ const monitor = Main.layoutManager.findMonitorForActor(this);
|
||
|
+ const x = Math.clamp(
|
||
|
+ stageX + xOffset,
|
||
|
+ monitor.x,
|
||
|
+ monitor.x + monitor.width - tipWidth);
|
||
|
+ const y = stageY + thumbHeight + TOOLTIP_OFFSET;
|
||
|
+ this._tooltip.set_position(x, y);
|
||
|
+ }
|
||
|
+
|
||
|
+ this._tooltip.ease({
|
||
|
+ opacity: this.hover ? 255 : 0,
|
||
|
+ duration: TOOLTIP_ANIMATION_TIME,
|
||
|
+ mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
|
+ onComplete: () => (this._tooltip.visible = this.hover),
|
||
|
+ });
|
||
|
+ }
|
||
|
+
|
||
|
+ _onDestroy() {
|
||
|
+ this._tooltip.destroy();
|
||
|
+
|
||
|
+ this._workspace.disconnect(this._windowAddedId);
|
||
|
+ this._workspace.disconnect(this._windowRemovedId);
|
||
|
+ global.display.disconnect(this._restackedId);
|
||
|
+ }
|
||
|
+});
|
||
|
+
|
||
|
+var WorkspaceIndicator = GObject.registerClass(
|
||
|
+class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
+ _init() {
|
||
|
+ super._init(0.5, _('Workspace Indicator'));
|
||
|
+
|
||
|
+ let container = new St.Widget({
|
||
|
+ layout_manager: new Clutter.BinLayout(),
|
||
|
+ x_expand: true,
|
||
|
+ y_expand: true,
|
||
|
+ });
|
||
|
+ this.add_child(container);
|
||
|
+
|
||
|
+ let workspaceManager = global.workspace_manager;
|
||
|
+
|
||
|
+ this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
+ this._statusLabel = new St.Label({
|
||
|
+ style_class: 'panel-workspace-indicator',
|
||
|
+ y_align: Clutter.ActorAlign.CENTER,
|
||
|
+ text: this._labelText(),
|
||
|
+ });
|
||
|
+
|
||
|
+ container.add_child(this._statusLabel);
|
||
|
+
|
||
|
+ this._thumbnailsBox = new St.BoxLayout({
|
||
|
+ style_class: 'panel-workspace-indicator-box',
|
||
|
+ y_expand: true,
|
||
|
+ reactive: true,
|
||
|
+ });
|
||
|
+
|
||
|
+ container.add_child(this._thumbnailsBox);
|
||
|
+
|
||
|
+ this._workspacesItems = [];
|
||
|
+ this._workspaceSection = new PopupMenu.PopupMenuSection();
|
||
|
+ this.menu.addMenuItem(this._workspaceSection);
|
||
|
+
|
||
|
+ this._workspaceManagerSignals = [
|
||
|
+ workspaceManager.connect_after('notify::n-workspaces',
|
||
|
+ this._nWorkspacesChanged.bind(this)),
|
||
|
+ workspaceManager.connect_after('workspace-switched',
|
||
|
+ this._onWorkspaceSwitched.bind(this)),
|
||
|
+ workspaceManager.connect('notify::layout-rows',
|
||
|
+ this._updateThumbnailVisibility.bind(this)),
|
||
|
+ ];
|
||
|
+
|
||
|
+ this.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
+ this._thumbnailsBox.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
+ this._createWorkspacesSection();
|
||
|
+ this._updateThumbnails();
|
||
|
+ this._updateThumbnailVisibility();
|
||
|
+
|
||
|
+ this._settings = new Gio.Settings({schema_id: WORKSPACE_SCHEMA});
|
||
|
+ this._settingsChangedId = this._settings.connect(
|
||
|
+ `changed::${WORKSPACE_KEY}`,
|
||
|
+ this._updateMenuLabels.bind(this));
|
||
|
+ }
|
||
|
+
|
||
|
+ _onDestroy() {
|
||
|
+ for (let i = i; i < this._workspaceManagerSignals.length; i++)
|
||
|
+ global.workspace_manager.disconnect(this._workspaceManagerSignals[i]);
|
||
|
+
|
||
|
+ if (this._settingsChangedId) {
|
||
|
+ this._settings.disconnect(this._settingsChangedId);
|
||
|
+ this._settingsChangedId = 0;
|
||
|
+ }
|
||
|
+
|
||
|
+ Main.panel.set_offscreen_redirect(Clutter.OffscreenRedirect.ALWAYS);
|
||
|
+
|
||
|
+ super._onDestroy();
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateThumbnailVisibility() {
|
||
|
+ const {workspaceManager} = global;
|
||
|
+ const vertical = workspaceManager.layout_rows === -1;
|
||
|
+ const useMenu =
|
||
|
+ vertical || workspaceManager.n_workspaces > MAX_THUMBNAILS;
|
||
|
+ this.reactive = useMenu;
|
||
|
+
|
||
|
+ this._statusLabel.visible = useMenu;
|
||
|
+ this._thumbnailsBox.visible = !useMenu;
|
||
|
+
|
||
|
+ // Disable offscreen-redirect when showing the workspace switcher
|
||
|
+ // so that clip-to-allocation works
|
||
|
+ Main.panel.set_offscreen_redirect(useMenu
|
||
|
+ ? Clutter.OffscreenRedirect.ALWAYS
|
||
|
+ : Clutter.OffscreenRedirect.AUTOMATIC_FOR_OPACITY);
|
||
|
+ }
|
||
|
+
|
||
|
+ _onWorkspaceSwitched() {
|
||
|
+ this._currentWorkspace = global.workspace_manager.get_active_workspace_index();
|
||
|
+
|
||
|
+ this._updateMenuOrnament();
|
||
|
+ this._updateActiveThumbnail();
|
||
|
+
|
||
|
+ this._statusLabel.set_text(this._labelText());
|
||
|
+ }
|
||
|
+
|
||
|
+ _nWorkspacesChanged() {
|
||
|
+ this._createWorkspacesSection();
|
||
|
+ this._updateThumbnails();
|
||
|
+ this._updateThumbnailVisibility();
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateMenuOrnament() {
|
||
|
+ for (let i = 0; i < this._workspacesItems.length; i++) {
|
||
|
+ this._workspacesItems[i].setOrnament(i === this._currentWorkspace
|
||
|
+ ? PopupMenu.Ornament.DOT
|
||
|
+ : PopupMenu.Ornament.NO_DOT);
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateActiveThumbnail() {
|
||
|
+ let thumbs = this._thumbnailsBox.get_children();
|
||
|
+ for (let i = 0; i < thumbs.length; i++) {
|
||
|
+ if (i === this._currentWorkspace)
|
||
|
+ thumbs[i].add_style_class_name('active');
|
||
|
+ else
|
||
|
+ thumbs[i].remove_style_class_name('active');
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ _labelText(workspaceIndex) {
|
||
|
+ if (workspaceIndex === undefined) {
|
||
|
+ workspaceIndex = this._currentWorkspace;
|
||
|
+ return (workspaceIndex + 1).toString();
|
||
|
+ }
|
||
|
+ return Meta.prefs_get_workspace_name(workspaceIndex);
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateMenuLabels() {
|
||
|
+ for (let i = 0; i < this._workspacesItems.length; i++)
|
||
|
+ this._workspacesItems[i].label.text = this._labelText(i);
|
||
|
+ }
|
||
|
+
|
||
|
+ _createWorkspacesSection() {
|
||
|
+ let workspaceManager = global.workspace_manager;
|
||
|
+
|
||
|
+ this._workspaceSection.removeAll();
|
||
|
+ this._workspacesItems = [];
|
||
|
+ this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
+
|
||
|
+ let i = 0;
|
||
|
+ for (; i < workspaceManager.n_workspaces; i++) {
|
||
|
+ this._workspacesItems[i] = new PopupMenu.PopupMenuItem(this._labelText(i));
|
||
|
+ this._workspaceSection.addMenuItem(this._workspacesItems[i]);
|
||
|
+ this._workspacesItems[i].workspaceId = i;
|
||
|
+ this._workspacesItems[i].label_actor = this._statusLabel;
|
||
|
+ this._workspacesItems[i].connect('activate', (actor, _event) => {
|
||
|
+ this._activate(actor.workspaceId);
|
||
|
+ });
|
||
|
+
|
||
|
+ this._workspacesItems[i].setOrnament(i === this._currentWorkspace
|
||
|
+ ? PopupMenu.Ornament.DOT
|
||
|
+ : PopupMenu.Ornament.NO_DOT);
|
||
|
+ }
|
||
|
+
|
||
|
+ this._statusLabel.set_text(this._labelText());
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateThumbnails() {
|
||
|
+ let workspaceManager = global.workspace_manager;
|
||
|
+
|
||
|
+ this._thumbnailsBox.destroy_all_children();
|
||
|
+
|
||
|
+ for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
+ let thumb = new WorkspaceThumbnail(i);
|
||
|
+ this._thumbnailsBox.add_child(thumb);
|
||
|
+ }
|
||
|
+ this._updateActiveThumbnail();
|
||
|
+ }
|
||
|
+
|
||
|
+ _activate(index) {
|
||
|
+ let workspaceManager = global.workspace_manager;
|
||
|
+
|
||
|
+ if (index >= 0 && index < workspaceManager.n_workspaces) {
|
||
|
+ let metaWorkspace = workspaceManager.get_workspace_by_index(index);
|
||
|
+ metaWorkspace.activate(global.get_current_time());
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ _onScrollEvent(actor, event) {
|
||
|
+ let direction = event.get_scroll_direction();
|
||
|
+ let diff = 0;
|
||
|
+ if (direction === Clutter.ScrollDirection.DOWN)
|
||
|
+ diff = 1;
|
||
|
+ else if (direction === Clutter.ScrollDirection.UP)
|
||
|
+ diff = -1;
|
||
|
+ else
|
||
|
+ return;
|
||
|
+
|
||
|
+
|
||
|
+ let newIndex = global.workspace_manager.get_active_workspace_index() + diff;
|
||
|
+ this._activate(newIndex);
|
||
|
+ }
|
||
|
+});
|
||
|
diff --git a/po/POTFILES.in b/po/POTFILES.in
|
||
|
index 10b1d517..bd39ab61 100644
|
||
|
--- a/po/POTFILES.in
|
||
|
+++ b/po/POTFILES.in
|
||
|
@@ -17,5 +17,5 @@ extensions/window-list/org.gnome.shell.extensions.window-list.gschema.xml
|
||
|
extensions/window-list/prefs.js
|
||
|
extensions/window-list/workspaceIndicator.js
|
||
|
extensions/windowsNavigator/extension.js
|
||
|
-extensions/workspace-indicator/extension.js
|
||
|
extensions/workspace-indicator/prefs.js
|
||
|
+extensions/workspace-indicator/workspaceIndicator.js
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 8c5020870abefdd1e14584bca80a96e3f50862ff Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 19:09:38 +0100
|
||
|
Subject: [PATCH 03/28] workspace-indicator: Use descendant style selectors
|
||
|
|
||
|
Add a style class to the indicator itself, and only select
|
||
|
descendant elements. This allows using the briefer class names
|
||
|
from the window-list extension without too much risk of conflicts.
|
||
|
---
|
||
|
extensions/workspace-indicator/stylesheet.css | 8 ++++----
|
||
|
extensions/workspace-indicator/workspaceIndicator.js | 6 ++++--
|
||
|
2 files changed, 8 insertions(+), 6 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/stylesheet.css b/extensions/workspace-indicator/stylesheet.css
|
||
|
index 84aaf454..4e12cce4 100644
|
||
|
--- a/extensions/workspace-indicator/stylesheet.css
|
||
|
+++ b/extensions/workspace-indicator/stylesheet.css
|
||
|
@@ -1,20 +1,20 @@
|
||
|
-.panel-workspace-indicator {
|
||
|
+.workspace-indicator .status-label {
|
||
|
padding: 0 8px;
|
||
|
}
|
||
|
|
||
|
-.panel-workspace-indicator-box {
|
||
|
+.workspace-indicator .workspaces-box {
|
||
|
padding: 4px 0;
|
||
|
spacing: 4px;
|
||
|
}
|
||
|
|
||
|
-.panel-workspace-indicator-box .workspace {
|
||
|
+.workspace-indicator .workspace {
|
||
|
width: 40px;
|
||
|
border: 2px solid #000;
|
||
|
border-radius: 2px;
|
||
|
background-color: #595959;
|
||
|
}
|
||
|
|
||
|
-.panel-workspace-indicator-box .workspace.active {
|
||
|
+.workspace-indicator .workspace.active {
|
||
|
border-color: #fff;
|
||
|
}
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index b98de047..101c89c6 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -263,6 +263,8 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
_init() {
|
||
|
super._init(0.5, _('Workspace Indicator'));
|
||
|
|
||
|
+ this.add_style_class_name('workspace-indicator');
|
||
|
+
|
||
|
let container = new St.Widget({
|
||
|
layout_manager: new Clutter.BinLayout(),
|
||
|
x_expand: true,
|
||
|
@@ -274,7 +276,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
this._statusLabel = new St.Label({
|
||
|
- style_class: 'panel-workspace-indicator',
|
||
|
+ style_class: 'status-label',
|
||
|
y_align: Clutter.ActorAlign.CENTER,
|
||
|
text: this._labelText(),
|
||
|
});
|
||
|
@@ -282,7 +284,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
container.add_child(this._statusLabel);
|
||
|
|
||
|
this._thumbnailsBox = new St.BoxLayout({
|
||
|
- style_class: 'panel-workspace-indicator-box',
|
||
|
+ style_class: 'workspaces-box',
|
||
|
y_expand: true,
|
||
|
reactive: true,
|
||
|
});
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 55614bb3ccaea057ca450e53d8d3a5b91ff1ffa0 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 12:48:43 +0100
|
||
|
Subject: [PATCH 04/28] window-list: Use consistent style class prefix
|
||
|
|
||
|
This will eventually allow us to re-use the workspace-indicator
|
||
|
extension without changing anything but the used prefix.
|
||
|
---
|
||
|
extensions/window-list/classic.css | 4 ++--
|
||
|
extensions/window-list/stylesheet.css | 4 ++--
|
||
|
extensions/window-list/workspaceIndicator.js | 2 +-
|
||
|
3 files changed, 5 insertions(+), 5 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/window-list/classic.css b/extensions/window-list/classic.css
|
||
|
index 375a33e1..ab982b92 100644
|
||
|
--- a/extensions/window-list/classic.css
|
||
|
+++ b/extensions/window-list/classic.css
|
||
|
@@ -58,11 +58,11 @@
|
||
|
border-color: #888;
|
||
|
}
|
||
|
|
||
|
-.window-list-window-preview {
|
||
|
+.window-list-workspace-indicator-window-preview {
|
||
|
background-color: #ededed;
|
||
|
border: 1px solid #ccc;
|
||
|
}
|
||
|
|
||
|
-.window-list-window-preview.active {
|
||
|
+.window-list-workspace-indicator-window-preview.active {
|
||
|
background-color: #f6f5f4;
|
||
|
}
|
||
|
diff --git a/extensions/window-list/stylesheet.css b/extensions/window-list/stylesheet.css
|
||
|
index 87813a42..a13f72d8 100644
|
||
|
--- a/extensions/window-list/stylesheet.css
|
||
|
+++ b/extensions/window-list/stylesheet.css
|
||
|
@@ -101,12 +101,12 @@
|
||
|
border-color: #fff;
|
||
|
}
|
||
|
|
||
|
-.window-list-window-preview {
|
||
|
+.window-list-workspace-indicator-window-preview {
|
||
|
background-color: #bebebe;
|
||
|
border: 1px solid #828282;
|
||
|
}
|
||
|
|
||
|
-.window-list-window-preview.active {
|
||
|
+.window-list-workspace-indicator-window-preview.active {
|
||
|
background-color: #d4d4d4;
|
||
|
}
|
||
|
|
||
|
diff --git a/extensions/window-list/workspaceIndicator.js b/extensions/window-list/workspaceIndicator.js
|
||
|
index cdfe5b61..c24f159f 100644
|
||
|
--- a/extensions/window-list/workspaceIndicator.js
|
||
|
+++ b/extensions/window-list/workspaceIndicator.js
|
||
|
@@ -21,7 +21,7 @@ let WindowPreview = GObject.registerClass(
|
||
|
class WindowPreview extends St.Button {
|
||
|
_init(window) {
|
||
|
super._init({
|
||
|
- style_class: 'window-list-window-preview',
|
||
|
+ style_class: 'window-list-workspace-indicator-window-preview',
|
||
|
});
|
||
|
|
||
|
this._delegate = this;
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From df0a023f8c0721b5e2bd7613d32a71e3591381cd Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Fri, 23 Feb 2024 01:59:15 +0100
|
||
|
Subject: [PATCH 05/28] workspace-indicator: Allow overriding base style class
|
||
|
|
||
|
This will allow reusing the code from the window-list extension
|
||
|
without limiting the ability to specify styling that only applies
|
||
|
to one of the extensions.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 13 ++++++++++---
|
||
|
1 file changed, 10 insertions(+), 3 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 101c89c6..ba1e05d7 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -25,11 +25,13 @@ const TOOLTIP_ANIMATION_TIME = 150;
|
||
|
|
||
|
const MAX_THUMBNAILS = 6;
|
||
|
|
||
|
+let baseStyleClassName = '';
|
||
|
+
|
||
|
const WindowPreview = GObject.registerClass(
|
||
|
class WindowPreview extends St.Button {
|
||
|
_init(window) {
|
||
|
super._init({
|
||
|
- style_class: 'workspace-indicator-window-preview',
|
||
|
+ style_class: `${baseStyleClassName}-window-preview`,
|
||
|
});
|
||
|
|
||
|
this._delegate = this;
|
||
|
@@ -260,10 +262,15 @@ class WorkspaceThumbnail extends St.Button {
|
||
|
|
||
|
var WorkspaceIndicator = GObject.registerClass(
|
||
|
class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
- _init() {
|
||
|
+ _init(params = {}) {
|
||
|
super._init(0.5, _('Workspace Indicator'));
|
||
|
|
||
|
- this.add_style_class_name('workspace-indicator');
|
||
|
+ const {
|
||
|
+ baseStyleClass = 'workspace-indicator',
|
||
|
+ } = params;
|
||
|
+
|
||
|
+ baseStyleClassName = baseStyleClass;
|
||
|
+ this.add_style_class_name(baseStyleClassName);
|
||
|
|
||
|
let container = new St.Widget({
|
||
|
layout_manager: new Clutter.BinLayout(),
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 7a5a295b6ff1c82be9f159ac5ffcc95e79bd887a Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Fri, 23 Feb 2024 01:58:50 +0100
|
||
|
Subject: [PATCH 06/28] window-list: Override base style class
|
||
|
|
||
|
Apply the changes from the last commit to the workspace-indicator
|
||
|
copy, and override the base style class from the extension.
|
||
|
|
||
|
This will eventually allow us to share the exact same code between
|
||
|
the two extensions, but still use individual styling if necessary.
|
||
|
---
|
||
|
extensions/window-list/extension.js | 5 ++++-
|
||
|
extensions/window-list/workspaceIndicator.js | 15 ++++++++++++---
|
||
|
2 files changed, 16 insertions(+), 4 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/window-list/extension.js b/extensions/window-list/extension.js
|
||
|
index 0c28692d..41a0c143 100644
|
||
|
--- a/extensions/window-list/extension.js
|
||
|
+++ b/extensions/window-list/extension.js
|
||
|
@@ -774,7 +774,10 @@ class WindowList extends St.Widget {
|
||
|
let indicatorsBox = new St.BoxLayout({ x_align: Clutter.ActorAlign.END });
|
||
|
box.add(indicatorsBox);
|
||
|
|
||
|
- this._workspaceIndicator = new WorkspaceIndicator();
|
||
|
+ this._workspaceIndicator = new WorkspaceIndicator({
|
||
|
+ baseStyleClass: 'window-list-workspace-indicator',
|
||
|
+ });
|
||
|
+
|
||
|
indicatorsBox.add_child(this._workspaceIndicator.container);
|
||
|
|
||
|
this._mutterSettings = new Gio.Settings({ schema_id: 'org.gnome.mutter' });
|
||
|
diff --git a/extensions/window-list/workspaceIndicator.js b/extensions/window-list/workspaceIndicator.js
|
||
|
index c24f159f..1a1d15cd 100644
|
||
|
--- a/extensions/window-list/workspaceIndicator.js
|
||
|
+++ b/extensions/window-list/workspaceIndicator.js
|
||
|
@@ -17,11 +17,13 @@ const TOOLTIP_ANIMATION_TIME = 150;
|
||
|
|
||
|
const MAX_THUMBNAILS = 6;
|
||
|
|
||
|
+let baseStyleClassName = '';
|
||
|
+
|
||
|
let WindowPreview = GObject.registerClass(
|
||
|
class WindowPreview extends St.Button {
|
||
|
_init(window) {
|
||
|
super._init({
|
||
|
- style_class: 'window-list-workspace-indicator-window-preview',
|
||
|
+ style_class: `${baseStyleClassName}-window-preview`,
|
||
|
});
|
||
|
|
||
|
this._delegate = this;
|
||
|
@@ -248,10 +250,17 @@ class WorkspaceThumbnail extends St.Button {
|
||
|
|
||
|
var WorkspaceIndicator = GObject.registerClass(
|
||
|
class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
- _init() {
|
||
|
+ _init(params = {}) {
|
||
|
super._init(0.0, _('Workspace Indicator'), true);
|
||
|
+
|
||
|
+ const {
|
||
|
+ baseStyleClass = 'workspace-indicator',
|
||
|
+ } = params;
|
||
|
+
|
||
|
+ baseStyleClassName = baseStyleClass;
|
||
|
+ this.add_style_class_name(baseStyleClassName);
|
||
|
+
|
||
|
this.setMenu(new PopupMenu.PopupMenu(this, 0.0, St.Side.BOTTOM));
|
||
|
- this.add_style_class_name('window-list-workspace-indicator');
|
||
|
this.remove_style_class_name('panel-button');
|
||
|
this.menu.actor.remove_style_class_name('panel-menu');
|
||
|
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 1d1f915a9cedf0717b2d5f04fd70feead4b109be Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 12:48:43 +0100
|
||
|
Subject: [PATCH 07/28] window-list: Externally adjust workspace menu
|
||
|
|
||
|
In order to use a PanelMenu.Button in the bottom bar, we have
|
||
|
to tweak its menu a bit.
|
||
|
|
||
|
We currently handle this inside the indicator, but that means the
|
||
|
code diverges from the original code in the workspace-indicator
|
||
|
extension.
|
||
|
|
||
|
Avoid this by using a small subclass that handles the adjustments.
|
||
|
---
|
||
|
extensions/window-list/extension.js | 22 ++++++++++++++++++--
|
||
|
extensions/window-list/workspaceIndicator.js | 6 +-----
|
||
|
2 files changed, 21 insertions(+), 7 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/window-list/extension.js b/extensions/window-list/extension.js
|
||
|
index 41a0c143..c58df434 100644
|
||
|
--- a/extensions/window-list/extension.js
|
||
|
+++ b/extensions/window-list/extension.js
|
||
|
@@ -774,10 +774,9 @@ class WindowList extends St.Widget {
|
||
|
let indicatorsBox = new St.BoxLayout({ x_align: Clutter.ActorAlign.END });
|
||
|
box.add(indicatorsBox);
|
||
|
|
||
|
- this._workspaceIndicator = new WorkspaceIndicator({
|
||
|
+ this._workspaceIndicator = new BottomWorkspaceIndicator({
|
||
|
baseStyleClass: 'window-list-workspace-indicator',
|
||
|
});
|
||
|
-
|
||
|
indicatorsBox.add_child(this._workspaceIndicator.container);
|
||
|
|
||
|
this._mutterSettings = new Gio.Settings({ schema_id: 'org.gnome.mutter' });
|
||
|
@@ -1140,6 +1139,25 @@ class WindowList extends St.Widget {
|
||
|
}
|
||
|
});
|
||
|
|
||
|
+const BottomWorkspaceIndicator = GObject.registerClass(
|
||
|
+class BottomWorkspaceIndicator extends WorkspaceIndicator {
|
||
|
+ _init(params) {
|
||
|
+ super._init(params);
|
||
|
+
|
||
|
+ this.remove_style_class_name('panel-button');
|
||
|
+ }
|
||
|
+
|
||
|
+ setMenu(menu) {
|
||
|
+ super.setMenu(menu);
|
||
|
+
|
||
|
+ if (!menu)
|
||
|
+ return;
|
||
|
+
|
||
|
+ this.menu.actor.updateArrowSide(St.Side.BOTTOM);
|
||
|
+ this.menu.actor.remove_style_class_name('panel-menu');
|
||
|
+ }
|
||
|
+});
|
||
|
+
|
||
|
class Extension {
|
||
|
constructor() {
|
||
|
ExtensionUtils.initTranslations();
|
||
|
diff --git a/extensions/window-list/workspaceIndicator.js b/extensions/window-list/workspaceIndicator.js
|
||
|
index 1a1d15cd..4290d58a 100644
|
||
|
--- a/extensions/window-list/workspaceIndicator.js
|
||
|
+++ b/extensions/window-list/workspaceIndicator.js
|
||
|
@@ -251,7 +251,7 @@ class WorkspaceThumbnail extends St.Button {
|
||
|
var WorkspaceIndicator = GObject.registerClass(
|
||
|
class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
_init(params = {}) {
|
||
|
- super._init(0.0, _('Workspace Indicator'), true);
|
||
|
+ super._init(0.0, _('Workspace Indicator'));
|
||
|
|
||
|
const {
|
||
|
baseStyleClass = 'workspace-indicator',
|
||
|
@@ -260,10 +260,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
baseStyleClassName = baseStyleClass;
|
||
|
this.add_style_class_name(baseStyleClassName);
|
||
|
|
||
|
- this.setMenu(new PopupMenu.PopupMenu(this, 0.0, St.Side.BOTTOM));
|
||
|
- this.remove_style_class_name('panel-button');
|
||
|
- this.menu.actor.remove_style_class_name('panel-menu');
|
||
|
-
|
||
|
let container = new St.Widget({
|
||
|
layout_manager: new Clutter.BinLayout(),
|
||
|
x_expand: true,
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 70b2caf35fea16bbbb20333128a9800b29c92e6f Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Thu, 21 Mar 2024 16:49:35 +0100
|
||
|
Subject: [PATCH 08/28] window-list: Handle changes to workspace menu
|
||
|
|
||
|
For now the menu is always set at construction time, however this
|
||
|
will change in the future. Prepare for that by handling the
|
||
|
`menu-set` signal, similar to the top bar.
|
||
|
---
|
||
|
extensions/window-list/extension.js | 9 ++++++++-
|
||
|
1 file changed, 8 insertions(+), 1 deletion(-)
|
||
|
|
||
|
diff --git a/extensions/window-list/extension.js b/extensions/window-list/extension.js
|
||
|
index c58df434..a011bc90 100644
|
||
|
--- a/extensions/window-list/extension.js
|
||
|
+++ b/extensions/window-list/extension.js
|
||
|
@@ -789,7 +789,9 @@ class WindowList extends St.Widget {
|
||
|
this._updateWorkspaceIndicatorVisibility();
|
||
|
|
||
|
this._menuManager = new PopupMenu.PopupMenuManager(this);
|
||
|
- this._menuManager.addMenu(this._workspaceIndicator.menu);
|
||
|
+ this._workspaceIndicator.connect('menu-set',
|
||
|
+ () => this._onWorkspaceMenuSet());
|
||
|
+ this._onWorkspaceMenuSet();
|
||
|
|
||
|
Main.layoutManager.addChrome(this, {
|
||
|
affectsStruts: true,
|
||
|
@@ -884,6 +886,11 @@ class WindowList extends St.Widget {
|
||
|
children[newActive].activate();
|
||
|
}
|
||
|
|
||
|
+ _onWorkspaceMenuSet() {
|
||
|
+ if (this._workspaceIndicator.menu)
|
||
|
+ this._menuManager.addMenu(this._workspaceIndicator.menu);
|
||
|
+ }
|
||
|
+
|
||
|
_updatePosition() {
|
||
|
this.set_position(
|
||
|
this._monitor.x,
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 8915a813c5c5d4ca4eb71da56e4fa29def55abe9 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 15:58:39 +0100
|
||
|
Subject: [PATCH 09/28] workspace-indicator: Don't use SCHEMA/KEY constants
|
||
|
|
||
|
Each constant is only used once, so all they do is disconnect
|
||
|
the actual value from the code that uses it.
|
||
|
|
||
|
The copy in the window-list extension just uses the strings directly,
|
||
|
do the same here.
|
||
|
---
|
||
|
extensions/workspace-indicator/workspaceIndicator.js | 7 ++-----
|
||
|
1 file changed, 2 insertions(+), 5 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index ba1e05d7..60e084e2 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -17,9 +17,6 @@ const Main = imports.ui.main;
|
||
|
const PanelMenu = imports.ui.panelMenu;
|
||
|
const PopupMenu = imports.ui.popupMenu;
|
||
|
|
||
|
-const WORKSPACE_SCHEMA = 'org.gnome.desktop.wm.preferences';
|
||
|
-const WORKSPACE_KEY = 'workspace-names';
|
||
|
-
|
||
|
const TOOLTIP_OFFSET = 6;
|
||
|
const TOOLTIP_ANIMATION_TIME = 150;
|
||
|
|
||
|
@@ -317,9 +314,9 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._updateThumbnails();
|
||
|
this._updateThumbnailVisibility();
|
||
|
|
||
|
- this._settings = new Gio.Settings({schema_id: WORKSPACE_SCHEMA});
|
||
|
+ this._settings = new Gio.Settings({schema_id: 'org.gnome.desktop.wm.preferences'});
|
||
|
this._settingsChangedId = this._settings.connect(
|
||
|
- `changed::${WORKSPACE_KEY}`,
|
||
|
+ 'changed::workspace-names',
|
||
|
this._updateMenuLabels.bind(this));
|
||
|
}
|
||
|
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 50b5b6eba5cad5a3c47043834cdb87aaf7d33b75 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 18:59:23 +0100
|
||
|
Subject: [PATCH 10/28] workspace-indicator: Use existing property
|
||
|
|
||
|
We already track the current workspace index, use that
|
||
|
instead of getting it from the workspace manager again.
|
||
|
---
|
||
|
extensions/workspace-indicator/workspaceIndicator.js | 2 +-
|
||
|
1 file changed, 1 insertion(+), 1 deletion(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 60e084e2..4f2188be 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -454,7 +454,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
return;
|
||
|
|
||
|
|
||
|
- let newIndex = global.workspace_manager.get_active_workspace_index() + diff;
|
||
|
+ const newIndex = this._currentWorkspace + diff;
|
||
|
this._activate(newIndex);
|
||
|
}
|
||
|
});
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From b2848cccdc75a8b354fe4e5693e0cf0e63fdeedf Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 16:14:24 +0100
|
||
|
Subject: [PATCH 11/28] workspace-indicator: Don't use menu section
|
||
|
|
||
|
We never added anything else to the menu, so we can just operate
|
||
|
on the entire menu instead of an intermediate section.
|
||
|
|
||
|
This removes another difference with the window-list copy.
|
||
|
---
|
||
|
extensions/workspace-indicator/workspaceIndicator.js | 12 +++++-------
|
||
|
1 file changed, 5 insertions(+), 7 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 4f2188be..66b71cd6 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -296,8 +296,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
container.add_child(this._thumbnailsBox);
|
||
|
|
||
|
this._workspacesItems = [];
|
||
|
- this._workspaceSection = new PopupMenu.PopupMenuSection();
|
||
|
- this.menu.addMenuItem(this._workspaceSection);
|
||
|
|
||
|
this._workspaceManagerSignals = [
|
||
|
workspaceManager.connect_after('notify::n-workspaces',
|
||
|
@@ -310,7 +308,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
this.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
this._thumbnailsBox.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
- this._createWorkspacesSection();
|
||
|
+ this._updateMenu();
|
||
|
this._updateThumbnails();
|
||
|
this._updateThumbnailVisibility();
|
||
|
|
||
|
@@ -361,7 +359,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
}
|
||
|
|
||
|
_nWorkspacesChanged() {
|
||
|
- this._createWorkspacesSection();
|
||
|
+ this._updateMenu();
|
||
|
this._updateThumbnails();
|
||
|
this._updateThumbnailVisibility();
|
||
|
}
|
||
|
@@ -397,17 +395,17 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._workspacesItems[i].label.text = this._labelText(i);
|
||
|
}
|
||
|
|
||
|
- _createWorkspacesSection() {
|
||
|
+ _updateMenu() {
|
||
|
let workspaceManager = global.workspace_manager;
|
||
|
|
||
|
- this._workspaceSection.removeAll();
|
||
|
+ this.menu.removeAll();
|
||
|
this._workspacesItems = [];
|
||
|
this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
|
||
|
let i = 0;
|
||
|
for (; i < workspaceManager.n_workspaces; i++) {
|
||
|
this._workspacesItems[i] = new PopupMenu.PopupMenuItem(this._labelText(i));
|
||
|
- this._workspaceSection.addMenuItem(this._workspacesItems[i]);
|
||
|
+ this.menu.addMenuItem(this._workspacesItems[i]);
|
||
|
this._workspacesItems[i].workspaceId = i;
|
||
|
this._workspacesItems[i].label_actor = this._statusLabel;
|
||
|
this._workspacesItems[i].connect('activate', (actor, _event) => {
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 21cea529850b861bb813666efa25da70a4e52ac5 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 13:05:15 +0100
|
||
|
Subject: [PATCH 12/28] workspace-indicator: Support showing tooltips above
|
||
|
|
||
|
The indicator is located in the top bar, so tooltips are always
|
||
|
shown below the previews. However supporting showing tooltips
|
||
|
above previews when space permits allows the same code to be
|
||
|
used in the copy that is included with the window-list extension.
|
||
|
---
|
||
|
extensions/workspace-indicator/workspaceIndicator.js | 9 +++++----
|
||
|
1 file changed, 5 insertions(+), 4 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 66b71cd6..bc9e222b 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -227,16 +227,17 @@ class WorkspaceThumbnail extends St.Button {
|
||
|
});
|
||
|
|
||
|
const [stageX, stageY] = this.get_transformed_position();
|
||
|
- const thumbWidth = this.allocation.get_width();
|
||
|
- const thumbHeight = this.allocation.get_height();
|
||
|
- const tipWidth = this._tooltip.width;
|
||
|
+ const [thumbWidth, thumbHeight] = this.allocation.get_size();
|
||
|
+ const [tipWidth, tipHeight] = this._tooltip.get_size();
|
||
|
const xOffset = Math.floor((thumbWidth - tipWidth) / 2);
|
||
|
const monitor = Main.layoutManager.findMonitorForActor(this);
|
||
|
const x = Math.clamp(
|
||
|
stageX + xOffset,
|
||
|
monitor.x,
|
||
|
monitor.x + monitor.width - tipWidth);
|
||
|
- const y = stageY + thumbHeight + TOOLTIP_OFFSET;
|
||
|
+ const y = stageY - monitor.y > thumbHeight + TOOLTIP_OFFSET
|
||
|
+ ? stageY - tipHeight - TOOLTIP_OFFSET // show above
|
||
|
+ : stageY + thumbHeight + TOOLTIP_OFFSET; // show below
|
||
|
this._tooltip.set_position(x, y);
|
||
|
}
|
||
|
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 658df8f7dd63d23f000738fb63fd145a744c0ccd Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 17:37:16 +0100
|
||
|
Subject: [PATCH 13/28] workspace-indicator: Only change top bar redirect when
|
||
|
in top bar
|
||
|
|
||
|
While this is always the case for the workspace indicator, adding
|
||
|
the check will allow to use the same code in the window list.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 23 +++++++++++++++++--
|
||
|
1 file changed, 21 insertions(+), 2 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index bc9e222b..ac270d64 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -309,6 +309,16 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
this.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
this._thumbnailsBox.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
+
|
||
|
+ this._inTopBar = false;
|
||
|
+ this.connect('notify::realized', () => {
|
||
|
+ if (!this.realized)
|
||
|
+ return;
|
||
|
+
|
||
|
+ this._inTopBar = Main.panel.contains(this);
|
||
|
+ this._updateTopBarRedirect();
|
||
|
+ });
|
||
|
+
|
||
|
this._updateMenu();
|
||
|
this._updateThumbnails();
|
||
|
this._updateThumbnailVisibility();
|
||
|
@@ -328,7 +338,9 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._settingsChangedId = 0;
|
||
|
}
|
||
|
|
||
|
- Main.panel.set_offscreen_redirect(Clutter.OffscreenRedirect.ALWAYS);
|
||
|
+ if (this._inTopBar)
|
||
|
+ Main.panel.set_offscreen_redirect(Clutter.OffscreenRedirect.ALWAYS);
|
||
|
+ this._inTopBar = false;
|
||
|
|
||
|
super._onDestroy();
|
||
|
}
|
||
|
@@ -343,9 +355,16 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._statusLabel.visible = useMenu;
|
||
|
this._thumbnailsBox.visible = !useMenu;
|
||
|
|
||
|
+ this._updateTopBarRedirect();
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateTopBarRedirect() {
|
||
|
+ if (!this._inTopBar)
|
||
|
+ return;
|
||
|
+
|
||
|
// Disable offscreen-redirect when showing the workspace switcher
|
||
|
// so that clip-to-allocation works
|
||
|
- Main.panel.set_offscreen_redirect(useMenu
|
||
|
+ Main.panel.set_offscreen_redirect(this._thumbnailsBox.visible
|
||
|
? Clutter.OffscreenRedirect.ALWAYS
|
||
|
: Clutter.OffscreenRedirect.AUTOMATIC_FOR_OPACITY);
|
||
|
}
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 2ed34034a2ee8ec8b2dc7a789ceac5753e76f61e Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 16:13:00 +0100
|
||
|
Subject: [PATCH 14/28] workspace-indicator: Small cleanup
|
||
|
|
||
|
The code to update the menu labels is a bit cleaner in the
|
||
|
window-list extension, so use that.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 19 +++++++++----------
|
||
|
1 file changed, 9 insertions(+), 10 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index ac270d64..9b22102a 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -422,19 +422,18 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._workspacesItems = [];
|
||
|
this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
|
||
|
- let i = 0;
|
||
|
- for (; i < workspaceManager.n_workspaces; i++) {
|
||
|
- this._workspacesItems[i] = new PopupMenu.PopupMenuItem(this._labelText(i));
|
||
|
- this.menu.addMenuItem(this._workspacesItems[i]);
|
||
|
- this._workspacesItems[i].workspaceId = i;
|
||
|
- this._workspacesItems[i].label_actor = this._statusLabel;
|
||
|
- this._workspacesItems[i].connect('activate', (actor, _event) => {
|
||
|
- this._activate(actor.workspaceId);
|
||
|
- });
|
||
|
+ for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
+ const item = new PopupMenu.PopupMenuItem(this._labelText(i));
|
||
|
|
||
|
- this._workspacesItems[i].setOrnament(i === this._currentWorkspace
|
||
|
+ item.connect('activate',
|
||
|
+ () => this._activate(i));
|
||
|
+
|
||
|
+ item.setOrnament(i === this._currentWorkspace
|
||
|
? PopupMenu.Ornament.DOT
|
||
|
: PopupMenu.Ornament.NO_DOT);
|
||
|
+
|
||
|
+ this.menu.addMenuItem(item);
|
||
|
+ this._workspacesItems[i] = item;
|
||
|
}
|
||
|
|
||
|
this._statusLabel.set_text(this._labelText());
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From b367e8a92de8fa8535ff27fcbc9a78c1476c9d19 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 16:13:00 +0100
|
||
|
Subject: [PATCH 15/28] workspace-indicator: Simplify getting status text
|
||
|
|
||
|
Currently the same method is used to get the label text for the
|
||
|
indicator itself and for the menu items.
|
||
|
|
||
|
A method that behaves significantly different depending on whether
|
||
|
a parameter is passed is confusing, so only deal with the indicator
|
||
|
label and directly use the mutter API to get the workspace names
|
||
|
for menu items.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 24 +++++++++----------
|
||
|
1 file changed, 12 insertions(+), 12 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 9b22102a..d8b29f58 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -283,7 +283,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._statusLabel = new St.Label({
|
||
|
style_class: 'status-label',
|
||
|
y_align: Clutter.ActorAlign.CENTER,
|
||
|
- text: this._labelText(),
|
||
|
+ text: this._getStatusText(),
|
||
|
});
|
||
|
|
||
|
container.add_child(this._statusLabel);
|
||
|
@@ -375,7 +375,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._updateMenuOrnament();
|
||
|
this._updateActiveThumbnail();
|
||
|
|
||
|
- this._statusLabel.set_text(this._labelText());
|
||
|
+ this._statusLabel.set_text(this._getStatusText());
|
||
|
}
|
||
|
|
||
|
_nWorkspacesChanged() {
|
||
|
@@ -402,17 +402,16 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
}
|
||
|
}
|
||
|
|
||
|
- _labelText(workspaceIndex) {
|
||
|
- if (workspaceIndex === undefined) {
|
||
|
- workspaceIndex = this._currentWorkspace;
|
||
|
- return (workspaceIndex + 1).toString();
|
||
|
- }
|
||
|
- return Meta.prefs_get_workspace_name(workspaceIndex);
|
||
|
+ _getStatusText() {
|
||
|
+ const current = this._currentWorkspace + 1;
|
||
|
+ return `${current}`;
|
||
|
}
|
||
|
|
||
|
_updateMenuLabels() {
|
||
|
- for (let i = 0; i < this._workspacesItems.length; i++)
|
||
|
- this._workspacesItems[i].label.text = this._labelText(i);
|
||
|
+ for (let i = 0; i < this._workspacesItems.length; i++) {
|
||
|
+ const item = this._workspacesItems[i];
|
||
|
+ item.label.text = Meta.prefs_get_workspace_name(i);
|
||
|
+ }
|
||
|
}
|
||
|
|
||
|
_updateMenu() {
|
||
|
@@ -423,7 +422,8 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
|
||
|
for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
- const item = new PopupMenu.PopupMenuItem(this._labelText(i));
|
||
|
+ const name = Meta.prefs_get_workspace_name(i);
|
||
|
+ const item = new PopupMenu.PopupMenuItem(name);
|
||
|
|
||
|
item.connect('activate',
|
||
|
() => this._activate(i));
|
||
|
@@ -436,7 +436,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._workspacesItems[i] = item;
|
||
|
}
|
||
|
|
||
|
- this._statusLabel.set_text(this._labelText());
|
||
|
+ this._statusLabel.set_text(this._getStatusText());
|
||
|
}
|
||
|
|
||
|
_updateThumbnails() {
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 4918d40d0161a670baee9153599384d38a5166f2 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 16:35:09 +0100
|
||
|
Subject: [PATCH 16/28] workspace-indicator: Include n-workspaces in status
|
||
|
label
|
||
|
|
||
|
The two extensions currently use a slightly different label
|
||
|
in menu mode:
|
||
|
The workspace indicator uses the plain workspace number ("2"),
|
||
|
while the window list includes the number of workspaces ("2 / 4").
|
||
|
|
||
|
The additional information seem useful, as well as the slightly
|
||
|
bigger click/touch target, so copy the window-list behavior.
|
||
|
---
|
||
|
extensions/workspace-indicator/workspaceIndicator.js | 3 ++-
|
||
|
1 file changed, 2 insertions(+), 1 deletion(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index d8b29f58..756758b3 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -403,8 +403,9 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
}
|
||
|
|
||
|
_getStatusText() {
|
||
|
+ const {nWorkspaces} = global.workspace_manager;
|
||
|
const current = this._currentWorkspace + 1;
|
||
|
- return `${current}`;
|
||
|
+ return `${current} / ${nWorkspaces}`;
|
||
|
}
|
||
|
|
||
|
_updateMenuLabels() {
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 8a99594c3a33d1623fcdf4d87f4a1ace8e1952f1 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Thu, 22 Feb 2024 04:45:23 +0100
|
||
|
Subject: [PATCH 17/28] workspace-indicator: Tweak preview style
|
||
|
|
||
|
Sync sizes and padding with the window-list previews.
|
||
|
|
||
|
Tone down the colors a bit, but less then the current window-list
|
||
|
style where workspaces blend too much into the background and
|
||
|
the selection is unclear.
|
||
|
---
|
||
|
extensions/workspace-indicator/stylesheet.css | 13 +++++++------
|
||
|
1 file changed, 7 insertions(+), 6 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/stylesheet.css b/extensions/workspace-indicator/stylesheet.css
|
||
|
index 4e12cce4..68fd534c 100644
|
||
|
--- a/extensions/workspace-indicator/stylesheet.css
|
||
|
+++ b/extensions/workspace-indicator/stylesheet.css
|
||
|
@@ -3,24 +3,25 @@
|
||
|
}
|
||
|
|
||
|
.workspace-indicator .workspaces-box {
|
||
|
- padding: 4px 0;
|
||
|
- spacing: 4px;
|
||
|
+ padding: 5px;
|
||
|
+ spacing: 3px;
|
||
|
}
|
||
|
|
||
|
.workspace-indicator .workspace {
|
||
|
- width: 40px;
|
||
|
+ width: 52px;
|
||
|
border: 2px solid #000;
|
||
|
- border-radius: 2px;
|
||
|
- background-color: #595959;
|
||
|
+ border-radius: 4px;
|
||
|
+ background-color: #3f3f3f;
|
||
|
}
|
||
|
|
||
|
.workspace-indicator .workspace.active {
|
||
|
- border-color: #fff;
|
||
|
+ border-color: #9f9f9f;
|
||
|
}
|
||
|
|
||
|
.workspace-indicator-window-preview {
|
||
|
background-color: #bebebe;
|
||
|
border: 1px solid #828282;
|
||
|
+ border-radius: 1px;
|
||
|
}
|
||
|
|
||
|
.workspace-indicator-window-preview.active {
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 786e03c6da64c9195223ad7648975e9007f7b2ac Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 23:22:58 +0100
|
||
|
Subject: [PATCH 18/28] workspace-indicator: Support light style
|
||
|
|
||
|
The window-list extension already includes light styling for
|
||
|
its copy of the workspace indicator. Just copy that over to
|
||
|
support the light variant here as well.
|
||
|
---
|
||
|
extensions/workspace-indicator/meson.build | 6 +++-
|
||
|
.../workspace-indicator/stylesheet-dark.css | 29 ++++++++++++++++++
|
||
|
.../workspace-indicator/stylesheet-light.css | 26 ++++++++++++++++
|
||
|
extensions/workspace-indicator/stylesheet.css | 30 +------------------
|
||
|
4 files changed, 61 insertions(+), 30 deletions(-)
|
||
|
create mode 100644 extensions/workspace-indicator/stylesheet-dark.css
|
||
|
create mode 100644 extensions/workspace-indicator/stylesheet-light.css
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/meson.build b/extensions/workspace-indicator/meson.build
|
||
|
index eb25b9cc..0bf9f023 100644
|
||
|
--- a/extensions/workspace-indicator/meson.build
|
||
|
+++ b/extensions/workspace-indicator/meson.build
|
||
|
@@ -3,6 +3,10 @@ extension_data += configure_file(
|
||
|
output: metadata_name,
|
||
|
configuration: metadata_conf
|
||
|
)
|
||
|
-extension_data += files('stylesheet.css')
|
||
|
+extension_data += files(
|
||
|
+ 'stylesheet.css',
|
||
|
+ 'stylesheet-dark.css',
|
||
|
+ 'stylesheet-light.css',
|
||
|
+)
|
||
|
|
||
|
extension_sources += files('prefs.js', 'workspaceIndicator.js')
|
||
|
diff --git a/extensions/workspace-indicator/stylesheet-dark.css b/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
new file mode 100644
|
||
|
index 00000000..68fd534c
|
||
|
--- /dev/null
|
||
|
+++ b/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
@@ -0,0 +1,29 @@
|
||
|
+.workspace-indicator .status-label {
|
||
|
+ padding: 0 8px;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator .workspaces-box {
|
||
|
+ padding: 5px;
|
||
|
+ spacing: 3px;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator .workspace {
|
||
|
+ width: 52px;
|
||
|
+ border: 2px solid #000;
|
||
|
+ border-radius: 4px;
|
||
|
+ background-color: #3f3f3f;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator .workspace.active {
|
||
|
+ border-color: #9f9f9f;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-window-preview {
|
||
|
+ background-color: #bebebe;
|
||
|
+ border: 1px solid #828282;
|
||
|
+ border-radius: 1px;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-window-preview.active {
|
||
|
+ background-color: #d4d4d4;
|
||
|
+}
|
||
|
diff --git a/extensions/workspace-indicator/stylesheet-light.css b/extensions/workspace-indicator/stylesheet-light.css
|
||
|
new file mode 100644
|
||
|
index 00000000..da77d90e
|
||
|
--- /dev/null
|
||
|
+++ b/extensions/workspace-indicator/stylesheet-light.css
|
||
|
@@ -0,0 +1,26 @@
|
||
|
+/*
|
||
|
+ * SPDX-FileCopyrightText: 2013 Florian Müllner <fmuellner@gnome.org>
|
||
|
+ * SPDX-FileCopyrightText: 2015 Jakub Steiner <jimmac@gmail.com>
|
||
|
+ *
|
||
|
+ * SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
+ */
|
||
|
+
|
||
|
+@import url("stylesheet-dark.css");
|
||
|
+
|
||
|
+.workspace-indicator .workspace {
|
||
|
+ border: 2px solid #f6f5f4;
|
||
|
+ background-color: #ccc;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator .workspace.active {
|
||
|
+ border-color: #888;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-window-preview {
|
||
|
+ background-color: #ededed;
|
||
|
+ border: 1px solid #ccc;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-window-preview.active {
|
||
|
+ background-color: #f6f5f4;
|
||
|
+}
|
||
|
diff --git a/extensions/workspace-indicator/stylesheet.css b/extensions/workspace-indicator/stylesheet.css
|
||
|
index 68fd534c..b0f7d171 100644
|
||
|
--- a/extensions/workspace-indicator/stylesheet.css
|
||
|
+++ b/extensions/workspace-indicator/stylesheet.css
|
||
|
@@ -1,29 +1 @@
|
||
|
-.workspace-indicator .status-label {
|
||
|
- padding: 0 8px;
|
||
|
-}
|
||
|
-
|
||
|
-.workspace-indicator .workspaces-box {
|
||
|
- padding: 5px;
|
||
|
- spacing: 3px;
|
||
|
-}
|
||
|
-
|
||
|
-.workspace-indicator .workspace {
|
||
|
- width: 52px;
|
||
|
- border: 2px solid #000;
|
||
|
- border-radius: 4px;
|
||
|
- background-color: #3f3f3f;
|
||
|
-}
|
||
|
-
|
||
|
-.workspace-indicator .workspace.active {
|
||
|
- border-color: #9f9f9f;
|
||
|
-}
|
||
|
-
|
||
|
-.workspace-indicator-window-preview {
|
||
|
- background-color: #bebebe;
|
||
|
- border: 1px solid #828282;
|
||
|
- border-radius: 1px;
|
||
|
-}
|
||
|
-
|
||
|
-.workspace-indicator-window-preview.active {
|
||
|
- background-color: #d4d4d4;
|
||
|
-}
|
||
|
+@import url("stylesheet-dark.css");
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 5e9b22449ec55f7f051b32d7d0b6fa9173a929e5 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Wed, 21 Feb 2024 13:08:52 +0100
|
||
|
Subject: [PATCH 19/28] window-list: Use actual copy of workspace-indicator
|
||
|
|
||
|
We are now at a point where the code from the workspace-indicator
|
||
|
extension is usable from the window-list.
|
||
|
|
||
|
However instead of updating the copy, go one step further and
|
||
|
remove it altogether, and copy the required files at build time.
|
||
|
|
||
|
This ensures that future changes are picked up by both extensions
|
||
|
without duplicating any work.
|
||
|
---
|
||
|
extensions/window-list/classic.css | 20 +-
|
||
|
extensions/window-list/meson.build | 29 +-
|
||
|
extensions/window-list/stylesheet.css | 2 +
|
||
|
extensions/window-list/workspaceIndicator.js | 447 -------------------
|
||
|
4 files changed, 31 insertions(+), 467 deletions(-)
|
||
|
delete mode 100644 extensions/window-list/workspaceIndicator.js
|
||
|
|
||
|
diff --git a/extensions/window-list/classic.css b/extensions/window-list/classic.css
|
||
|
index ab982b92..d7ceb062 100644
|
||
|
--- a/extensions/window-list/classic.css
|
||
|
+++ b/extensions/window-list/classic.css
|
||
|
@@ -1,4 +1,5 @@
|
||
|
@import url("stylesheet.css");
|
||
|
+@import url("stylesheet-workspace-switcher-light.css");
|
||
|
|
||
|
#panel.bottom-panel {
|
||
|
border-top-width: 1px;
|
||
|
@@ -47,22 +48,3 @@
|
||
|
color: #888;
|
||
|
box-shadow: none;
|
||
|
}
|
||
|
-
|
||
|
-/* workspace switcher */
|
||
|
-.window-list-workspace-indicator .workspace {
|
||
|
- border: 2px solid #f6f5f4;
|
||
|
- background-color: #ccc;
|
||
|
-}
|
||
|
-
|
||
|
-.window-list-workspace-indicator .workspace.active {
|
||
|
- border-color: #888;
|
||
|
-}
|
||
|
-
|
||
|
-.window-list-workspace-indicator-window-preview {
|
||
|
- background-color: #ededed;
|
||
|
- border: 1px solid #ccc;
|
||
|
-}
|
||
|
-
|
||
|
-.window-list-workspace-indicator-window-preview.active {
|
||
|
- background-color: #f6f5f4;
|
||
|
-}
|
||
|
diff --git a/extensions/window-list/meson.build b/extensions/window-list/meson.build
|
||
|
index 599f45e1..12d2b174 100644
|
||
|
--- a/extensions/window-list/meson.build
|
||
|
+++ b/extensions/window-list/meson.build
|
||
|
@@ -5,7 +5,34 @@ extension_data += configure_file(
|
||
|
)
|
||
|
extension_data += files('stylesheet.css')
|
||
|
|
||
|
-extension_sources += files('prefs.js', 'windowPicker.js', 'workspaceIndicator.js')
|
||
|
+transform_stylesheet = [
|
||
|
+ 'sed', '-E',
|
||
|
+ '-e', 's:^\.(workspace-indicator):.window-list-\\1:',
|
||
|
+ '-e', '/^@import/d',
|
||
|
+ '@INPUT@',
|
||
|
+ ]
|
||
|
+
|
||
|
+workspaceIndicatorSources = [
|
||
|
+ configure_file(
|
||
|
+ input: '../workspace-indicator/workspaceIndicator.js',
|
||
|
+ output: '@PLAINNAME@',
|
||
|
+ copy: true,
|
||
|
+ ),
|
||
|
+ configure_file(
|
||
|
+ input: '../workspace-indicator/stylesheet-dark.css',
|
||
|
+ output: 'stylesheet-workspace-switcher-dark.css',
|
||
|
+ command: transform_stylesheet,
|
||
|
+ capture: true,
|
||
|
+ ),
|
||
|
+ configure_file(
|
||
|
+ input: '../workspace-indicator/stylesheet-light.css',
|
||
|
+ output: 'stylesheet-workspace-switcher-light.css',
|
||
|
+ command: transform_stylesheet,
|
||
|
+ capture: true,
|
||
|
+ ),
|
||
|
+]
|
||
|
+
|
||
|
+extension_sources += files('prefs.js', 'windowPicker.js') + workspaceIndicatorSources
|
||
|
extension_schemas += files(metadata_conf.get('gschemaname') + '.gschema.xml')
|
||
|
|
||
|
if classic_mode_enabled
|
||
|
diff --git a/extensions/window-list/stylesheet.css b/extensions/window-list/stylesheet.css
|
||
|
index a13f72d8..4ba47f07 100644
|
||
|
--- a/extensions/window-list/stylesheet.css
|
||
|
+++ b/extensions/window-list/stylesheet.css
|
||
|
@@ -1,3 +1,5 @@
|
||
|
+@import url("stylesheet-workspace-switcher-dark.css");
|
||
|
+
|
||
|
.window-list {
|
||
|
spacing: 2px;
|
||
|
font-size: 10pt;
|
||
|
diff --git a/extensions/window-list/workspaceIndicator.js b/extensions/window-list/workspaceIndicator.js
|
||
|
deleted file mode 100644
|
||
|
index 4290d58a..00000000
|
||
|
--- a/extensions/window-list/workspaceIndicator.js
|
||
|
+++ /dev/null
|
||
|
@@ -1,447 +0,0 @@
|
||
|
-/* exported WorkspaceIndicator */
|
||
|
-const { Clutter, Gio, GObject, Meta, St } = imports.gi;
|
||
|
-
|
||
|
-const DND = imports.ui.dnd;
|
||
|
-const ExtensionUtils = imports.misc.extensionUtils;
|
||
|
-const Main = imports.ui.main;
|
||
|
-const PanelMenu = imports.ui.panelMenu;
|
||
|
-const PopupMenu = imports.ui.popupMenu;
|
||
|
-
|
||
|
-const Me = ExtensionUtils.getCurrentExtension();
|
||
|
-
|
||
|
-const Gettext = imports.gettext.domain(Me.metadata['gettext-domain']);
|
||
|
-const _ = Gettext.gettext;
|
||
|
-
|
||
|
-const TOOLTIP_OFFSET = 6;
|
||
|
-const TOOLTIP_ANIMATION_TIME = 150;
|
||
|
-
|
||
|
-const MAX_THUMBNAILS = 6;
|
||
|
-
|
||
|
-let baseStyleClassName = '';
|
||
|
-
|
||
|
-let WindowPreview = GObject.registerClass(
|
||
|
-class WindowPreview extends St.Button {
|
||
|
- _init(window) {
|
||
|
- super._init({
|
||
|
- style_class: `${baseStyleClassName}-window-preview`,
|
||
|
- });
|
||
|
-
|
||
|
- this._delegate = this;
|
||
|
- DND.makeDraggable(this, { restoreOnSuccess: true });
|
||
|
-
|
||
|
- this._window = window;
|
||
|
-
|
||
|
- this.connect('destroy', this._onDestroy.bind(this));
|
||
|
-
|
||
|
- this._sizeChangedId = this._window.connect('size-changed',
|
||
|
- () => this.queue_relayout());
|
||
|
- this._positionChangedId = this._window.connect('position-changed',
|
||
|
- () => {
|
||
|
- this._updateVisible();
|
||
|
- this.queue_relayout();
|
||
|
- });
|
||
|
- this._minimizedChangedId = this._window.connect('notify::minimized',
|
||
|
- this._updateVisible.bind(this));
|
||
|
-
|
||
|
- this._focusChangedId = global.display.connect('notify::focus-window',
|
||
|
- this._onFocusChanged.bind(this));
|
||
|
- this._onFocusChanged();
|
||
|
- }
|
||
|
-
|
||
|
- // needed for DND
|
||
|
- get metaWindow() {
|
||
|
- return this._window;
|
||
|
- }
|
||
|
-
|
||
|
- _onDestroy() {
|
||
|
- this._window.disconnect(this._sizeChangedId);
|
||
|
- this._window.disconnect(this._positionChangedId);
|
||
|
- this._window.disconnect(this._minimizedChangedId);
|
||
|
- global.display.disconnect(this._focusChangedId);
|
||
|
- }
|
||
|
-
|
||
|
- _onFocusChanged() {
|
||
|
- if (global.display.focus_window === this._window)
|
||
|
- this.add_style_class_name('active');
|
||
|
- else
|
||
|
- this.remove_style_class_name('active');
|
||
|
- }
|
||
|
-
|
||
|
- _updateVisible() {
|
||
|
- const monitor = Main.layoutManager.findIndexForActor(this);
|
||
|
- const workArea = Main.layoutManager.getWorkAreaForMonitor(monitor);
|
||
|
- this.visible = this._window.get_frame_rect().overlap(workArea) &&
|
||
|
- this._window.window_type !== Meta.WindowType.DESKTOP &&
|
||
|
- this._window.showing_on_its_workspace();
|
||
|
- }
|
||
|
-});
|
||
|
-
|
||
|
-let WorkspaceLayout = GObject.registerClass(
|
||
|
-class WorkspaceLayout extends Clutter.LayoutManager {
|
||
|
- vfunc_get_preferred_width() {
|
||
|
- return [0, 0];
|
||
|
- }
|
||
|
-
|
||
|
- vfunc_get_preferred_height() {
|
||
|
- return [0, 0];
|
||
|
- }
|
||
|
-
|
||
|
- vfunc_allocate(container, box) {
|
||
|
- const monitor = Main.layoutManager.findIndexForActor(container);
|
||
|
- const workArea = Main.layoutManager.getWorkAreaForMonitor(monitor);
|
||
|
- const hscale = box.get_width() / workArea.width;
|
||
|
- const vscale = box.get_height() / workArea.height;
|
||
|
-
|
||
|
- for (const child of container) {
|
||
|
- const childBox = new Clutter.ActorBox();
|
||
|
- const frameRect = child.metaWindow.get_frame_rect();
|
||
|
- childBox.set_size(
|
||
|
- Math.round(Math.min(frameRect.width, workArea.width) * hscale),
|
||
|
- Math.round(Math.min(frameRect.height, workArea.height) * vscale));
|
||
|
- childBox.set_origin(
|
||
|
- Math.round((frameRect.x - workArea.x) * hscale),
|
||
|
- Math.round((frameRect.y - workArea.y) * vscale));
|
||
|
- child.allocate(childBox);
|
||
|
- }
|
||
|
- }
|
||
|
-});
|
||
|
-
|
||
|
-let WorkspaceThumbnail = GObject.registerClass(
|
||
|
-class WorkspaceThumbnail extends St.Button {
|
||
|
- _init(index) {
|
||
|
- super._init({
|
||
|
- style_class: 'workspace',
|
||
|
- child: new Clutter.Actor({
|
||
|
- layout_manager: new WorkspaceLayout(),
|
||
|
- clip_to_allocation: true,
|
||
|
- }),
|
||
|
- });
|
||
|
-
|
||
|
- this._tooltip = new St.Label({
|
||
|
- style_class: 'dash-label',
|
||
|
- visible: false,
|
||
|
- });
|
||
|
- Main.uiGroup.add_child(this._tooltip);
|
||
|
-
|
||
|
- this.connect('destroy', this._onDestroy.bind(this));
|
||
|
- this.connect('notify::hover', this._syncTooltip.bind(this));
|
||
|
-
|
||
|
- this._index = index;
|
||
|
- this._delegate = this; // needed for DND
|
||
|
-
|
||
|
- this._windowPreviews = new Map();
|
||
|
-
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
- this._workspace = workspaceManager.get_workspace_by_index(index);
|
||
|
-
|
||
|
- this._windowAddedId = this._workspace.connect('window-added',
|
||
|
- (ws, window) => {
|
||
|
- this._addWindow(window);
|
||
|
- });
|
||
|
- this._windowRemovedId = this._workspace.connect('window-removed',
|
||
|
- (ws, window) => {
|
||
|
- this._removeWindow(window);
|
||
|
- });
|
||
|
- this._restackedId = global.display.connect('restacked',
|
||
|
- this._onRestacked.bind(this));
|
||
|
-
|
||
|
- this._workspace.list_windows().forEach(w => this._addWindow(w));
|
||
|
- this._onRestacked();
|
||
|
- }
|
||
|
-
|
||
|
- acceptDrop(source) {
|
||
|
- if (!source.metaWindow)
|
||
|
- return false;
|
||
|
-
|
||
|
- this._moveWindow(source.metaWindow);
|
||
|
- return true;
|
||
|
- }
|
||
|
-
|
||
|
- handleDragOver(source) {
|
||
|
- if (source.metaWindow)
|
||
|
- return DND.DragMotionResult.MOVE_DROP;
|
||
|
- else
|
||
|
- return DND.DragMotionResult.CONTINUE;
|
||
|
- }
|
||
|
-
|
||
|
- _addWindow(window) {
|
||
|
- if (this._windowPreviews.has(window))
|
||
|
- return;
|
||
|
-
|
||
|
- let preview = new WindowPreview(window);
|
||
|
- preview.connect('clicked', (a, btn) => this.emit('clicked', btn));
|
||
|
- this._windowPreviews.set(window, preview);
|
||
|
- this.child.add_child(preview);
|
||
|
- }
|
||
|
-
|
||
|
- _removeWindow(window) {
|
||
|
- let preview = this._windowPreviews.get(window);
|
||
|
- if (!preview)
|
||
|
- return;
|
||
|
-
|
||
|
- this._windowPreviews.delete(window);
|
||
|
- preview.destroy();
|
||
|
- }
|
||
|
-
|
||
|
- _onRestacked() {
|
||
|
- let lastPreview = null;
|
||
|
- let windows = global.get_window_actors().map(a => a.meta_window);
|
||
|
- for (let i = 0; i < windows.length; i++) {
|
||
|
- let preview = this._windowPreviews.get(windows[i]);
|
||
|
- if (!preview)
|
||
|
- continue;
|
||
|
-
|
||
|
- this.child.set_child_above_sibling(preview, lastPreview);
|
||
|
- lastPreview = preview;
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _moveWindow(window) {
|
||
|
- let monitorIndex = Main.layoutManager.findIndexForActor(this);
|
||
|
- if (monitorIndex !== window.get_monitor())
|
||
|
- window.move_to_monitor(monitorIndex);
|
||
|
- window.change_workspace_by_index(this._index, false);
|
||
|
- }
|
||
|
-
|
||
|
- on_clicked() {
|
||
|
- let ws = global.workspace_manager.get_workspace_by_index(this._index);
|
||
|
- if (ws)
|
||
|
- ws.activate(global.get_current_time());
|
||
|
- }
|
||
|
-
|
||
|
- _syncTooltip() {
|
||
|
- if (this.hover) {
|
||
|
- this._tooltip.set({
|
||
|
- text: Meta.prefs_get_workspace_name(this._index),
|
||
|
- visible: true,
|
||
|
- opacity: 0,
|
||
|
- });
|
||
|
-
|
||
|
- const [stageX, stageY] = this.get_transformed_position();
|
||
|
- const thumbWidth = this.allocation.get_width();
|
||
|
- const tipWidth = this._tooltip.width;
|
||
|
- const tipHeight = this._tooltip.height;
|
||
|
- const xOffset = Math.floor((thumbWidth - tipWidth) / 2);
|
||
|
- const monitor = Main.layoutManager.findMonitorForActor(this);
|
||
|
- const x = Math.clamp(
|
||
|
- stageX + xOffset,
|
||
|
- monitor.x,
|
||
|
- monitor.x + monitor.width - tipWidth);
|
||
|
- const y = stageY - tipHeight - TOOLTIP_OFFSET;
|
||
|
- this._tooltip.set_position(x, y);
|
||
|
- }
|
||
|
-
|
||
|
- this._tooltip.ease({
|
||
|
- opacity: this.hover ? 255 : 0,
|
||
|
- duration: TOOLTIP_ANIMATION_TIME,
|
||
|
- mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
||
|
- onComplete: () => (this._tooltip.visible = this.hover),
|
||
|
- });
|
||
|
- }
|
||
|
-
|
||
|
- _onDestroy() {
|
||
|
- this._tooltip.destroy();
|
||
|
-
|
||
|
- this._workspace.disconnect(this._windowAddedId);
|
||
|
- this._workspace.disconnect(this._windowRemovedId);
|
||
|
- global.display.disconnect(this._restackedId);
|
||
|
- }
|
||
|
-});
|
||
|
-
|
||
|
-var WorkspaceIndicator = GObject.registerClass(
|
||
|
-class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
- _init(params = {}) {
|
||
|
- super._init(0.0, _('Workspace Indicator'));
|
||
|
-
|
||
|
- const {
|
||
|
- baseStyleClass = 'workspace-indicator',
|
||
|
- } = params;
|
||
|
-
|
||
|
- baseStyleClassName = baseStyleClass;
|
||
|
- this.add_style_class_name(baseStyleClassName);
|
||
|
-
|
||
|
- let container = new St.Widget({
|
||
|
- layout_manager: new Clutter.BinLayout(),
|
||
|
- x_expand: true,
|
||
|
- y_expand: true,
|
||
|
- });
|
||
|
- this.add_actor(container);
|
||
|
-
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
- this._statusLabel = new St.Label({ text: this._getStatusText() });
|
||
|
-
|
||
|
- this._statusBin = new St.Bin({
|
||
|
- style_class: 'status-label-bin',
|
||
|
- x_expand: true,
|
||
|
- y_expand: true,
|
||
|
- child: this._statusLabel,
|
||
|
- });
|
||
|
- container.add_actor(this._statusBin);
|
||
|
-
|
||
|
- this._thumbnailsBox = new St.BoxLayout({
|
||
|
- style_class: 'workspaces-box',
|
||
|
- y_expand: true,
|
||
|
- reactive: true,
|
||
|
- });
|
||
|
- this._thumbnailsBox.connect('scroll-event',
|
||
|
- this._onScrollEvent.bind(this));
|
||
|
- container.add_actor(this._thumbnailsBox);
|
||
|
-
|
||
|
- this._workspacesItems = [];
|
||
|
-
|
||
|
- this._workspaceManagerSignals = [
|
||
|
- workspaceManager.connect('notify::n-workspaces',
|
||
|
- this._nWorkspacesChanged.bind(this)),
|
||
|
- workspaceManager.connect_after('workspace-switched',
|
||
|
- this._onWorkspaceSwitched.bind(this)),
|
||
|
- workspaceManager.connect('notify::layout-rows',
|
||
|
- this._updateThumbnailVisibility.bind(this)),
|
||
|
- ];
|
||
|
-
|
||
|
- this.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
- this._updateMenu();
|
||
|
- this._updateThumbnails();
|
||
|
- this._updateThumbnailVisibility();
|
||
|
-
|
||
|
- this._settings = new Gio.Settings({ schema_id: 'org.gnome.desktop.wm.preferences' });
|
||
|
- this._settingsChangedId = this._settings.connect(
|
||
|
- 'changed::workspace-names', this._updateMenuLabels.bind(this));
|
||
|
- }
|
||
|
-
|
||
|
- _onDestroy() {
|
||
|
- for (let i = 0; i < this._workspaceManagerSignals.length; i++)
|
||
|
- global.workspace_manager.disconnect(this._workspaceManagerSignals[i]);
|
||
|
-
|
||
|
- if (this._settingsChangedId) {
|
||
|
- this._settings.disconnect(this._settingsChangedId);
|
||
|
- this._settingsChangedId = 0;
|
||
|
- }
|
||
|
-
|
||
|
- super._onDestroy();
|
||
|
- }
|
||
|
-
|
||
|
- _updateThumbnailVisibility() {
|
||
|
- const { workspaceManager } = global;
|
||
|
- const vertical = workspaceManager.layout_rows === -1;
|
||
|
- const useMenu =
|
||
|
- vertical || workspaceManager.n_workspaces > MAX_THUMBNAILS;
|
||
|
- this.reactive = useMenu;
|
||
|
-
|
||
|
- this._statusBin.visible = useMenu;
|
||
|
- this._thumbnailsBox.visible = !useMenu;
|
||
|
- }
|
||
|
-
|
||
|
- _onWorkspaceSwitched() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
- this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
-
|
||
|
- this._updateMenuOrnament();
|
||
|
- this._updateActiveThumbnail();
|
||
|
-
|
||
|
- this._statusLabel.set_text(this._getStatusText());
|
||
|
- }
|
||
|
-
|
||
|
- _nWorkspacesChanged() {
|
||
|
- this._updateMenu();
|
||
|
- this._updateThumbnails();
|
||
|
- this._updateThumbnailVisibility();
|
||
|
- }
|
||
|
-
|
||
|
- _updateMenuOrnament() {
|
||
|
- for (let i = 0; i < this._workspacesItems.length; i++) {
|
||
|
- this._workspacesItems[i].setOrnament(i === this._currentWorkspace
|
||
|
- ? PopupMenu.Ornament.DOT
|
||
|
- : PopupMenu.Ornament.NONE);
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _updateActiveThumbnail() {
|
||
|
- let thumbs = this._thumbnailsBox.get_children();
|
||
|
- for (let i = 0; i < thumbs.length; i++) {
|
||
|
- if (i === this._currentWorkspace)
|
||
|
- thumbs[i].add_style_class_name('active');
|
||
|
- else
|
||
|
- thumbs[i].remove_style_class_name('active');
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _getStatusText() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
- let current = workspaceManager.get_active_workspace_index();
|
||
|
- let total = workspaceManager.n_workspaces;
|
||
|
-
|
||
|
- return '%d / %d'.format(current + 1, total);
|
||
|
- }
|
||
|
-
|
||
|
- _updateMenuLabels() {
|
||
|
- for (let i = 0; i < this._workspacesItems.length; i++) {
|
||
|
- let item = this._workspacesItems[i];
|
||
|
- let name = Meta.prefs_get_workspace_name(i);
|
||
|
- item.label.text = name;
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _updateMenu() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this.menu.removeAll();
|
||
|
- this._workspacesItems = [];
|
||
|
- this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
-
|
||
|
- for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
- let name = Meta.prefs_get_workspace_name(i);
|
||
|
- let item = new PopupMenu.PopupMenuItem(name);
|
||
|
- item.workspaceId = i;
|
||
|
-
|
||
|
- item.connect('activate', () => {
|
||
|
- this._activate(item.workspaceId);
|
||
|
- });
|
||
|
-
|
||
|
- if (i === this._currentWorkspace)
|
||
|
- item.setOrnament(PopupMenu.Ornament.DOT);
|
||
|
-
|
||
|
- this.menu.addMenuItem(item);
|
||
|
- this._workspacesItems[i] = item;
|
||
|
- }
|
||
|
-
|
||
|
- this._statusLabel.set_text(this._getStatusText());
|
||
|
- }
|
||
|
-
|
||
|
- _updateThumbnails() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this._thumbnailsBox.destroy_all_children();
|
||
|
-
|
||
|
- for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
- let thumb = new WorkspaceThumbnail(i);
|
||
|
- this._thumbnailsBox.add_actor(thumb);
|
||
|
- }
|
||
|
- this._updateActiveThumbnail();
|
||
|
- }
|
||
|
-
|
||
|
- _activate(index) {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- if (index >= 0 && index < workspaceManager.n_workspaces) {
|
||
|
- let metaWorkspace = workspaceManager.get_workspace_by_index(index);
|
||
|
- metaWorkspace.activate(global.get_current_time());
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _onScrollEvent(actor, event) {
|
||
|
- let direction = event.get_scroll_direction();
|
||
|
- let diff = 0;
|
||
|
- if (direction === Clutter.ScrollDirection.DOWN)
|
||
|
- diff = 1;
|
||
|
- else if (direction === Clutter.ScrollDirection.UP)
|
||
|
- diff = -1;
|
||
|
- else
|
||
|
- return;
|
||
|
-
|
||
|
- let newIndex = this._currentWorkspace + diff;
|
||
|
- this._activate(newIndex);
|
||
|
- }
|
||
|
-});
|
||
|
-
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 61f68777292726c337cf4894be675bbd924376ea Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Tue, 20 Feb 2024 17:39:49 +0100
|
||
|
Subject: [PATCH 20/28] workspace-indicator: Simplify scroll handling
|
||
|
|
||
|
gnome-shell already includes a method for switching workspaces
|
||
|
via scroll events. Use that instead of implementing our own.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 21 ++++---------------
|
||
|
1 file changed, 4 insertions(+), 17 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 756758b3..8e3fec56 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -307,8 +307,10 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._updateThumbnailVisibility.bind(this)),
|
||
|
];
|
||
|
|
||
|
- this.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
- this._thumbnailsBox.connect('scroll-event', this._onScrollEvent.bind(this));
|
||
|
+ this.connect('scroll-event',
|
||
|
+ (a, event) => Main.wm.handleWorkspaceScroll(event));
|
||
|
+ this._thumbnailsBox.connect('scroll-event',
|
||
|
+ (a, event) => Main.wm.handleWorkspaceScroll(event));
|
||
|
|
||
|
this._inTopBar = false;
|
||
|
this.connect('notify::realized', () => {
|
||
|
@@ -460,19 +462,4 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
metaWorkspace.activate(global.get_current_time());
|
||
|
}
|
||
|
}
|
||
|
-
|
||
|
- _onScrollEvent(actor, event) {
|
||
|
- let direction = event.get_scroll_direction();
|
||
|
- let diff = 0;
|
||
|
- if (direction === Clutter.ScrollDirection.DOWN)
|
||
|
- diff = 1;
|
||
|
- else if (direction === Clutter.ScrollDirection.UP)
|
||
|
- diff = -1;
|
||
|
- else
|
||
|
- return;
|
||
|
-
|
||
|
-
|
||
|
- const newIndex = this._currentWorkspace + diff;
|
||
|
- this._activate(newIndex);
|
||
|
- }
|
||
|
});
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 4a3e86c8d54c256318620f2f8e2eea5ea0be181b Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Tue, 27 Feb 2024 21:20:45 +0100
|
||
|
Subject: [PATCH 21/28] workspace-indicator: Handle active indication in
|
||
|
thumbnail
|
||
|
|
||
|
Meta.Workspace has had an `active` property for a while now, so
|
||
|
we can use a property binding instead of tracking the active
|
||
|
workspace ourselves.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 38 ++++++++++++-------
|
||
|
1 file changed, 24 insertions(+), 14 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 8e3fec56..01b831f7 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -116,8 +116,14 @@ class WorkspaceLayout extends Clutter.LayoutManager {
|
||
|
}
|
||
|
});
|
||
|
|
||
|
-const WorkspaceThumbnail = GObject.registerClass(
|
||
|
-class WorkspaceThumbnail extends St.Button {
|
||
|
+const WorkspaceThumbnail = GObject.registerClass({
|
||
|
+ Properties: {
|
||
|
+ 'active': GObject.ParamSpec.boolean(
|
||
|
+ 'active', '', '',
|
||
|
+ GObject.ParamFlags.READWRITE,
|
||
|
+ false),
|
||
|
+ },
|
||
|
+}, class WorkspaceThumbnail extends St.Button {
|
||
|
_init(index) {
|
||
|
super._init({
|
||
|
style_class: 'workspace',
|
||
|
@@ -146,6 +152,10 @@ class WorkspaceThumbnail extends St.Button {
|
||
|
let workspaceManager = global.workspace_manager;
|
||
|
this._workspace = workspaceManager.get_workspace_by_index(index);
|
||
|
|
||
|
+ this._workspace.bind_property('active',
|
||
|
+ this, 'active',
|
||
|
+ GObject.BindingFlags.SYNC_CREATE);
|
||
|
+
|
||
|
this._windowAddedId = this._workspace.connect('window-added',
|
||
|
(ws, window) => this._addWindow(window));
|
||
|
this._windowRemovedId = this._workspace.connect('window-removed',
|
||
|
@@ -158,6 +168,18 @@ class WorkspaceThumbnail extends St.Button {
|
||
|
this._onRestacked();
|
||
|
}
|
||
|
|
||
|
+ get active() {
|
||
|
+ return this.has_style_class_name('active');
|
||
|
+ }
|
||
|
+
|
||
|
+ set active(active) {
|
||
|
+ if (active)
|
||
|
+ this.add_style_class_name('active');
|
||
|
+ else
|
||
|
+ this.remove_style_class_name('active');
|
||
|
+ this.notify('active');
|
||
|
+ }
|
||
|
+
|
||
|
acceptDrop(source) {
|
||
|
if (!source.metaWindow)
|
||
|
return false;
|
||
|
@@ -375,7 +397,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._currentWorkspace = global.workspace_manager.get_active_workspace_index();
|
||
|
|
||
|
this._updateMenuOrnament();
|
||
|
- this._updateActiveThumbnail();
|
||
|
|
||
|
this._statusLabel.set_text(this._getStatusText());
|
||
|
}
|
||
|
@@ -394,16 +415,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
}
|
||
|
}
|
||
|
|
||
|
- _updateActiveThumbnail() {
|
||
|
- let thumbs = this._thumbnailsBox.get_children();
|
||
|
- for (let i = 0; i < thumbs.length; i++) {
|
||
|
- if (i === this._currentWorkspace)
|
||
|
- thumbs[i].add_style_class_name('active');
|
||
|
- else
|
||
|
- thumbs[i].remove_style_class_name('active');
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
_getStatusText() {
|
||
|
const {nWorkspaces} = global.workspace_manager;
|
||
|
const current = this._currentWorkspace + 1;
|
||
|
@@ -451,7 +462,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
let thumb = new WorkspaceThumbnail(i);
|
||
|
this._thumbnailsBox.add_child(thumb);
|
||
|
}
|
||
|
- this._updateActiveThumbnail();
|
||
|
}
|
||
|
|
||
|
_activate(index) {
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 4ffe354adabcb830014196a89e898dc8c2ca6bda Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Tue, 20 Feb 2024 17:27:57 +0100
|
||
|
Subject: [PATCH 22/28] workspace-indicator: Split out WorkspacePreviews
|
||
|
|
||
|
The previews will become a bit more complex soon, so spit them out
|
||
|
into a dedicated class.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 74 ++++++++++++-------
|
||
|
1 file changed, 49 insertions(+), 25 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 01b831f7..a559b8e2 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -280,6 +280,51 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
}
|
||
|
});
|
||
|
|
||
|
+const WorkspacePreviews = GObject.registerClass(
|
||
|
+class WorkspacePreviews extends Clutter.Actor {
|
||
|
+ _init(params) {
|
||
|
+ super._init({
|
||
|
+ ...params,
|
||
|
+ layout_manager: new Clutter.BinLayout(),
|
||
|
+ reactive: true,
|
||
|
+ y_expand: true,
|
||
|
+ });
|
||
|
+
|
||
|
+ this.connect('scroll-event',
|
||
|
+ (a, event) => Main.wm.handleWorkspaceScroll(event));
|
||
|
+ this.connect('destroy', () => this._onDestroy());
|
||
|
+
|
||
|
+ const {workspaceManager} = global;
|
||
|
+
|
||
|
+ this._nWorkspacesChanged =
|
||
|
+ workspaceManager.connect_after('notify::n-workspaces',
|
||
|
+ () => this._updateThumbnails());
|
||
|
+
|
||
|
+ this._thumbnailsBox = new St.BoxLayout({
|
||
|
+ style_class: 'workspaces-box',
|
||
|
+ y_expand: true,
|
||
|
+ });
|
||
|
+ this.add_child(this._thumbnailsBox);
|
||
|
+
|
||
|
+ this._updateThumbnails();
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateThumbnails() {
|
||
|
+ const {nWorkspaces} = global.workspace_manager;
|
||
|
+
|
||
|
+ this._thumbnailsBox.destroy_all_children();
|
||
|
+
|
||
|
+ for (let i = 0; i < nWorkspaces; i++) {
|
||
|
+ const thumb = new WorkspaceThumbnail(i);
|
||
|
+ this._thumbnailsBox.add_child(thumb);
|
||
|
+ }
|
||
|
+ }
|
||
|
+
|
||
|
+ _onDestroy() {
|
||
|
+ global.workspace_manager.disconnect(this._nWorkspacesChanged);
|
||
|
+ }
|
||
|
+});
|
||
|
+
|
||
|
var WorkspaceIndicator = GObject.registerClass(
|
||
|
class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
_init(params = {}) {
|
||
|
@@ -307,16 +352,10 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
y_align: Clutter.ActorAlign.CENTER,
|
||
|
text: this._getStatusText(),
|
||
|
});
|
||
|
-
|
||
|
container.add_child(this._statusLabel);
|
||
|
|
||
|
- this._thumbnailsBox = new St.BoxLayout({
|
||
|
- style_class: 'workspaces-box',
|
||
|
- y_expand: true,
|
||
|
- reactive: true,
|
||
|
- });
|
||
|
-
|
||
|
- container.add_child(this._thumbnailsBox);
|
||
|
+ this._thumbnails = new WorkspacePreviews();
|
||
|
+ container.add_child(this._thumbnails);
|
||
|
|
||
|
this._workspacesItems = [];
|
||
|
|
||
|
@@ -331,8 +370,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
this.connect('scroll-event',
|
||
|
(a, event) => Main.wm.handleWorkspaceScroll(event));
|
||
|
- this._thumbnailsBox.connect('scroll-event',
|
||
|
- (a, event) => Main.wm.handleWorkspaceScroll(event));
|
||
|
|
||
|
this._inTopBar = false;
|
||
|
this.connect('notify::realized', () => {
|
||
|
@@ -344,7 +381,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
});
|
||
|
|
||
|
this._updateMenu();
|
||
|
- this._updateThumbnails();
|
||
|
this._updateThumbnailVisibility();
|
||
|
|
||
|
this._settings = new Gio.Settings({schema_id: 'org.gnome.desktop.wm.preferences'});
|
||
|
@@ -377,7 +413,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this.reactive = useMenu;
|
||
|
|
||
|
this._statusLabel.visible = useMenu;
|
||
|
- this._thumbnailsBox.visible = !useMenu;
|
||
|
+ this._thumbnails.visible = !useMenu;
|
||
|
|
||
|
this._updateTopBarRedirect();
|
||
|
}
|
||
|
@@ -388,7 +424,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
// Disable offscreen-redirect when showing the workspace switcher
|
||
|
// so that clip-to-allocation works
|
||
|
- Main.panel.set_offscreen_redirect(this._thumbnailsBox.visible
|
||
|
+ Main.panel.set_offscreen_redirect(this._thumbnails.visible
|
||
|
? Clutter.OffscreenRedirect.ALWAYS
|
||
|
: Clutter.OffscreenRedirect.AUTOMATIC_FOR_OPACITY);
|
||
|
}
|
||
|
@@ -403,7 +439,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
_nWorkspacesChanged() {
|
||
|
this._updateMenu();
|
||
|
- this._updateThumbnails();
|
||
|
this._updateThumbnailVisibility();
|
||
|
}
|
||
|
|
||
|
@@ -453,17 +488,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._statusLabel.set_text(this._getStatusText());
|
||
|
}
|
||
|
|
||
|
- _updateThumbnails() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this._thumbnailsBox.destroy_all_children();
|
||
|
-
|
||
|
- for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
- let thumb = new WorkspaceThumbnail(i);
|
||
|
- this._thumbnailsBox.add_child(thumb);
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
_activate(index) {
|
||
|
let workspaceManager = global.workspace_manager;
|
||
|
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 1e0f95c9f2f562348086dcd27a8f45430d399902 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Mon, 19 Feb 2024 14:42:04 +0100
|
||
|
Subject: [PATCH 23/28] workspace-indicator: Handle preview overflow
|
||
|
|
||
|
We currently avoid previews from overflowing in most setups by
|
||
|
artificially limiting them to a maximum of six workspaces.
|
||
|
|
||
|
Add some proper handling to also cover cases where space is more
|
||
|
limited, and to allow removing the restriction in the future.
|
||
|
|
||
|
For that, wrap the previews in an auto-scrolling scroll view
|
||
|
and add overflow indicators on each side.
|
||
|
---
|
||
|
.../workspace-indicator/stylesheet-dark.css | 8 +++
|
||
|
.../workspace-indicator/workspaceIndicator.js | 59 ++++++++++++++++++-
|
||
|
2 files changed, 66 insertions(+), 1 deletion(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/stylesheet-dark.css b/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
index 68fd534c..55f8ca5f 100644
|
||
|
--- a/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
+++ b/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
@@ -2,11 +2,19 @@
|
||
|
padding: 0 8px;
|
||
|
}
|
||
|
|
||
|
+.workspace-indicator .workspaces-view {
|
||
|
+ /*max-width: 280px;*/
|
||
|
+}
|
||
|
+
|
||
|
.workspace-indicator .workspaces-box {
|
||
|
padding: 5px;
|
||
|
spacing: 3px;
|
||
|
}
|
||
|
|
||
|
+.workspace-indicator .preview-arrow {
|
||
|
+ icon-size: 1.08em;
|
||
|
+}
|
||
|
+
|
||
|
.workspace-indicator .workspace {
|
||
|
width: 52px;
|
||
|
border: 2px solid #000;
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index a559b8e2..4cebeb20 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -299,12 +299,48 @@ class WorkspacePreviews extends Clutter.Actor {
|
||
|
this._nWorkspacesChanged =
|
||
|
workspaceManager.connect_after('notify::n-workspaces',
|
||
|
() => this._updateThumbnails());
|
||
|
+ this._workspaceSwitchedId =
|
||
|
+ workspaceManager.connect('workspace-switched',
|
||
|
+ () => this._updateScrollPosition());
|
||
|
+
|
||
|
+ this.connect('notify::mapped', () => {
|
||
|
+ if (this.mapped)
|
||
|
+ this._updateScrollPosition();
|
||
|
+ });
|
||
|
|
||
|
this._thumbnailsBox = new St.BoxLayout({
|
||
|
style_class: 'workspaces-box',
|
||
|
y_expand: true,
|
||
|
});
|
||
|
- this.add_child(this._thumbnailsBox);
|
||
|
+
|
||
|
+ const view = new St.ScrollView({
|
||
|
+ style_class: 'workspaces-view hfade',
|
||
|
+ enable_mouse_scrolling: false,
|
||
|
+ hscrollbar_policy: St.PolicyType.EXTERNAL,
|
||
|
+ vscrollbar_policy: St.PolicyType.NEVER,
|
||
|
+ y_expand: true,
|
||
|
+ });
|
||
|
+ this._scrollAdjustment = view.hscroll.adjustment;
|
||
|
+
|
||
|
+ view.add_actor(this._thumbnailsBox);
|
||
|
+
|
||
|
+ this.add_child(view);
|
||
|
+
|
||
|
+ this._arrowStart = new St.Icon({
|
||
|
+ icon_name: 'pan-start-symbolic',
|
||
|
+ style_class: 'preview-arrow',
|
||
|
+ x_align: Clutter.ActorAlign.START,
|
||
|
+ x_expand: true,
|
||
|
+ });
|
||
|
+ this.add_child(this._arrowStart);
|
||
|
+
|
||
|
+ this._arrowEnd = new St.Icon({
|
||
|
+ icon_name: 'pan-end-symbolic',
|
||
|
+ style_class: 'preview-arrow',
|
||
|
+ x_align: Clutter.ActorAlign.END,
|
||
|
+ x_expand: true,
|
||
|
+ });
|
||
|
+ this.add_child(this._arrowEnd);
|
||
|
|
||
|
this._updateThumbnails();
|
||
|
}
|
||
|
@@ -318,6 +354,27 @@ class WorkspacePreviews extends Clutter.Actor {
|
||
|
const thumb = new WorkspaceThumbnail(i);
|
||
|
this._thumbnailsBox.add_child(thumb);
|
||
|
}
|
||
|
+
|
||
|
+ if (this.mapped)
|
||
|
+ this._updateScrollPosition();
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateScrollPosition() {
|
||
|
+ const activeWorkspace = global.workspaceManager.get_active_workspace_index();
|
||
|
+ const {nWorkspaces} = global.workspace_manager;
|
||
|
+ const {pageSize} = this._scrollAdjustment;
|
||
|
+ const [fullSize] = this._thumbnailsBox.get_preferred_width(-1);
|
||
|
+ const wsSize = Math.floor(fullSize / nWorkspaces);
|
||
|
+ const value = (activeWorkspace + 0.5) * wsSize - pageSize / 2;
|
||
|
+ this._scrollAdjustment.set({value});
|
||
|
+ this._updateArrowVisibility();
|
||
|
+ }
|
||
|
+
|
||
|
+ _updateArrowVisibility() {
|
||
|
+ const {value, upper, pageSize} = this._scrollAdjustment;
|
||
|
+
|
||
|
+ this._arrowStart.opacity = value > 0 ? 255 : 0;
|
||
|
+ this._arrowEnd.opacity = value < upper - pageSize ? 255 : 0;
|
||
|
}
|
||
|
|
||
|
_onDestroy() {
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 3c41c8497080d6a3183916594f51fd623e12c1cf Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Sun, 3 Mar 2024 15:05:23 +0100
|
||
|
Subject: [PATCH 24/28] workspace-indicator: Support labels in previews
|
||
|
|
||
|
The space in the top bar is too limited to include the workspace
|
||
|
names. However we'll soon replace the textual menu with a preview
|
||
|
popover. We can use bigger previews there, so we can include the
|
||
|
names to not lose functionality with regards to the current menu.
|
||
|
---
|
||
|
.../workspace-indicator/workspaceIndicator.js | 63 ++++++++++++++++---
|
||
|
1 file changed, 55 insertions(+), 8 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 4cebeb20..5ef5674d 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -122,10 +122,23 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
'active', '', '',
|
||
|
GObject.ParamFlags.READWRITE,
|
||
|
false),
|
||
|
+ 'show-label': GObject.ParamSpec.boolean(
|
||
|
+ 'show-label', '', '',
|
||
|
+ GObject.ParamFlags.READWRITE,
|
||
|
+ false),
|
||
|
},
|
||
|
}, class WorkspaceThumbnail extends St.Button {
|
||
|
_init(index) {
|
||
|
- super._init({
|
||
|
+ super._init();
|
||
|
+
|
||
|
+ const box = new St.BoxLayout({
|
||
|
+ style_class: 'workspace-box',
|
||
|
+ y_expand: true,
|
||
|
+ vertical: true,
|
||
|
+ });
|
||
|
+ this.set_child(box);
|
||
|
+
|
||
|
+ this._preview = new St.Bin({
|
||
|
style_class: 'workspace',
|
||
|
child: new Clutter.Actor({
|
||
|
layout_manager: new WorkspaceLayout(),
|
||
|
@@ -133,7 +146,15 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
x_expand: true,
|
||
|
y_expand: true,
|
||
|
}),
|
||
|
+ y_expand: true,
|
||
|
});
|
||
|
+ box.add_child(this._preview);
|
||
|
+
|
||
|
+ this._label = new St.Label({
|
||
|
+ x_align: Clutter.ActorAlign.CENTER,
|
||
|
+ text: Meta.prefs_get_workspace_name(index),
|
||
|
+ });
|
||
|
+ box.add_child(this._label);
|
||
|
|
||
|
this._tooltip = new St.Label({
|
||
|
style_class: 'dash-label',
|
||
|
@@ -141,9 +162,20 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
});
|
||
|
Main.uiGroup.add_child(this._tooltip);
|
||
|
|
||
|
+ this.bind_property('show-label',
|
||
|
+ this._label, 'visible',
|
||
|
+ GObject.BindingFlags.SYNC_CREATE);
|
||
|
+
|
||
|
this.connect('destroy', this._onDestroy.bind(this));
|
||
|
this.connect('notify::hover', this._syncTooltip.bind(this));
|
||
|
|
||
|
+ this._desktopSettings =
|
||
|
+ new Gio.Settings({schema_id: 'org.gnome.desktop.wm.preferences'});
|
||
|
+ this._namesChangedId =
|
||
|
+ this._desktopSettings.connect('changed::workspace-names', () => {
|
||
|
+ this._label.text = Meta.prefs_get_workspace_name(index);
|
||
|
+ });
|
||
|
+
|
||
|
this._index = index;
|
||
|
this._delegate = this; // needed for DND
|
||
|
|
||
|
@@ -169,14 +201,14 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
}
|
||
|
|
||
|
get active() {
|
||
|
- return this.has_style_class_name('active');
|
||
|
+ return this._preview.has_style_class_name('active');
|
||
|
}
|
||
|
|
||
|
set active(active) {
|
||
|
if (active)
|
||
|
- this.add_style_class_name('active');
|
||
|
+ this._preview.add_style_class_name('active');
|
||
|
else
|
||
|
- this.remove_style_class_name('active');
|
||
|
+ this._preview.remove_style_class_name('active');
|
||
|
this.notify('active');
|
||
|
}
|
||
|
|
||
|
@@ -202,7 +234,7 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
let preview = new WindowPreview(window);
|
||
|
preview.connect('clicked', (a, btn) => this.emit('clicked', btn));
|
||
|
this._windowPreviews.set(window, preview);
|
||
|
- this.child.add_child(preview);
|
||
|
+ this._preview.child.add_child(preview);
|
||
|
}
|
||
|
|
||
|
_removeWindow(window) {
|
||
|
@@ -222,7 +254,7 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
if (!preview)
|
||
|
continue;
|
||
|
|
||
|
- this.child.set_child_above_sibling(preview, lastPreview);
|
||
|
+ this._preview.child.set_child_above_sibling(preview, lastPreview);
|
||
|
lastPreview = preview;
|
||
|
}
|
||
|
}
|
||
|
@@ -241,6 +273,9 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
}
|
||
|
|
||
|
_syncTooltip() {
|
||
|
+ if (this.showLabel)
|
||
|
+ return;
|
||
|
+
|
||
|
if (this.hover) {
|
||
|
this._tooltip.set({
|
||
|
text: Meta.prefs_get_workspace_name(this._index),
|
||
|
@@ -277,11 +312,20 @@ const WorkspaceThumbnail = GObject.registerClass({
|
||
|
this._workspace.disconnect(this._windowAddedId);
|
||
|
this._workspace.disconnect(this._windowRemovedId);
|
||
|
global.display.disconnect(this._restackedId);
|
||
|
+
|
||
|
+ this._desktopSettings.disconnect(this._namesChangedId);
|
||
|
+ this._desktopSettings = null;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
-const WorkspacePreviews = GObject.registerClass(
|
||
|
-class WorkspacePreviews extends Clutter.Actor {
|
||
|
+const WorkspacePreviews = GObject.registerClass({
|
||
|
+ Properties: {
|
||
|
+ 'show-labels': GObject.ParamSpec.boolean(
|
||
|
+ 'show-labels', '', '',
|
||
|
+ GObject.ParamFlags.READWRITE | GObject.ParamFlags.CONSTRUCT_ONLY,
|
||
|
+ false),
|
||
|
+ },
|
||
|
+}, class WorkspacePreviews extends Clutter.Actor {
|
||
|
_init(params) {
|
||
|
super._init({
|
||
|
...params,
|
||
|
@@ -352,6 +396,9 @@ class WorkspacePreviews extends Clutter.Actor {
|
||
|
|
||
|
for (let i = 0; i < nWorkspaces; i++) {
|
||
|
const thumb = new WorkspaceThumbnail(i);
|
||
|
+ this.bind_property('show-labels',
|
||
|
+ thumb, 'show-label',
|
||
|
+ GObject.BindingFlags.SYNC_CREATE);
|
||
|
this._thumbnailsBox.add_child(thumb);
|
||
|
}
|
||
|
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 4c7c7df4965e1a06bf38e4404363b6c134ff56c7 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Tue, 20 Feb 2024 21:43:55 +0100
|
||
|
Subject: [PATCH 25/28] workspace-indicator: Stop handling vertical layouts
|
||
|
|
||
|
Both the regular session and GNOME classic use a horizontal layout
|
||
|
nowadays, so it doesn't seem worth to specifically handle vertical
|
||
|
layouts anymore.
|
||
|
|
||
|
The extension will still work when the layout is changed (by some
|
||
|
other extension), there will simply be a mismatch between horizontal
|
||
|
previews and the actual layout.
|
||
|
---
|
||
|
extensions/workspace-indicator/workspaceIndicator.js | 6 +-----
|
||
|
1 file changed, 1 insertion(+), 5 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 5ef5674d..1171fbd1 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -468,8 +468,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._nWorkspacesChanged.bind(this)),
|
||
|
workspaceManager.connect_after('workspace-switched',
|
||
|
this._onWorkspaceSwitched.bind(this)),
|
||
|
- workspaceManager.connect('notify::layout-rows',
|
||
|
- this._updateThumbnailVisibility.bind(this)),
|
||
|
];
|
||
|
|
||
|
this.connect('scroll-event',
|
||
|
@@ -511,9 +509,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
_updateThumbnailVisibility() {
|
||
|
const {workspaceManager} = global;
|
||
|
- const vertical = workspaceManager.layout_rows === -1;
|
||
|
- const useMenu =
|
||
|
- vertical || workspaceManager.n_workspaces > MAX_THUMBNAILS;
|
||
|
+ const useMenu = workspaceManager.n_workspaces > MAX_THUMBNAILS;
|
||
|
this.reactive = useMenu;
|
||
|
|
||
|
this._statusLabel.visible = useMenu;
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 02bc0cd65e082a390a22e2b2a25bad64aedc2299 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Sun, 3 Mar 2024 15:05:23 +0100
|
||
|
Subject: [PATCH 26/28] workspace-indicator: Also show previews in menu
|
||
|
|
||
|
Since the regular session also switched to horizontal workspaces,
|
||
|
using a vertical menu has been a bit awkward.
|
||
|
|
||
|
Now that our previews have become more flexible, we can use them
|
||
|
in the collapsed state as well as when embedded into the top bar.
|
||
|
---
|
||
|
.../workspace-indicator/stylesheet-dark.css | 26 +++++-
|
||
|
.../workspace-indicator/workspaceIndicator.js | 79 +++----------------
|
||
|
2 files changed, 37 insertions(+), 68 deletions(-)
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/stylesheet-dark.css b/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
index 55f8ca5f..77d187de 100644
|
||
|
--- a/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
+++ b/extensions/workspace-indicator/stylesheet-dark.css
|
||
|
@@ -6,22 +6,46 @@
|
||
|
/*max-width: 280px;*/
|
||
|
}
|
||
|
|
||
|
+.workspace-indicator-menu .workspaces-view {
|
||
|
+ max-width: 480px;
|
||
|
+}
|
||
|
+
|
||
|
.workspace-indicator .workspaces-box {
|
||
|
padding: 5px;
|
||
|
spacing: 3px;
|
||
|
}
|
||
|
|
||
|
+.workspace-indicator-menu .workspaces-box {
|
||
|
+ padding: 5px;
|
||
|
+ spacing: 6px;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-menu .preview-arrow,
|
||
|
.workspace-indicator .preview-arrow {
|
||
|
icon-size: 1.08em;
|
||
|
}
|
||
|
|
||
|
+.workspace-indicator-menu .workspace-box {
|
||
|
+ spacing: 6px;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-menu .workspace,
|
||
|
.workspace-indicator .workspace {
|
||
|
- width: 52px;
|
||
|
border: 2px solid #000;
|
||
|
border-radius: 4px;
|
||
|
background-color: #3f3f3f;
|
||
|
}
|
||
|
|
||
|
+.workspace-indicator .workspace {
|
||
|
+ width: 52px;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-menu .workspace {
|
||
|
+ height: 80px;
|
||
|
+ width: 160px;
|
||
|
+}
|
||
|
+
|
||
|
+.workspace-indicator-menu .workspace.active,
|
||
|
.workspace-indicator .workspace.active {
|
||
|
border-color: #9f9f9f;
|
||
|
}
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 1171fbd1..368d785a 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -432,7 +432,7 @@ const WorkspacePreviews = GObject.registerClass({
|
||
|
var WorkspaceIndicator = GObject.registerClass(
|
||
|
class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
_init(params = {}) {
|
||
|
- super._init(0.5, _('Workspace Indicator'));
|
||
|
+ super(0.5, _('Workspace Indicator'), true);
|
||
|
|
||
|
const {
|
||
|
baseStyleClass = 'workspace-indicator',
|
||
|
@@ -465,7 +465,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
this._workspaceManagerSignals = [
|
||
|
workspaceManager.connect_after('notify::n-workspaces',
|
||
|
- this._nWorkspacesChanged.bind(this)),
|
||
|
+ this._updateThumbnailVisibility.bind(this)),
|
||
|
workspaceManager.connect_after('workspace-switched',
|
||
|
this._onWorkspaceSwitched.bind(this)),
|
||
|
];
|
||
|
@@ -482,24 +482,13 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._updateTopBarRedirect();
|
||
|
});
|
||
|
|
||
|
- this._updateMenu();
|
||
|
this._updateThumbnailVisibility();
|
||
|
-
|
||
|
- this._settings = new Gio.Settings({schema_id: 'org.gnome.desktop.wm.preferences'});
|
||
|
- this._settingsChangedId = this._settings.connect(
|
||
|
- 'changed::workspace-names',
|
||
|
- this._updateMenuLabels.bind(this));
|
||
|
}
|
||
|
|
||
|
_onDestroy() {
|
||
|
for (let i = i; i < this._workspaceManagerSignals.length; i++)
|
||
|
global.workspace_manager.disconnect(this._workspaceManagerSignals[i]);
|
||
|
|
||
|
- if (this._settingsChangedId) {
|
||
|
- this._settings.disconnect(this._settingsChangedId);
|
||
|
- this._settingsChangedId = 0;
|
||
|
- }
|
||
|
-
|
||
|
if (this._inTopBar)
|
||
|
Main.panel.set_offscreen_redirect(Clutter.OffscreenRedirect.ALWAYS);
|
||
|
this._inTopBar = false;
|
||
|
@@ -515,6 +504,10 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._statusLabel.visible = useMenu;
|
||
|
this._thumbnails.visible = !useMenu;
|
||
|
|
||
|
+ this.setMenu(useMenu
|
||
|
+ ? this._createPreviewMenu()
|
||
|
+ : null);
|
||
|
+
|
||
|
this._updateTopBarRedirect();
|
||
|
}
|
||
|
|
||
|
@@ -531,69 +524,21 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
|
||
|
_onWorkspaceSwitched() {
|
||
|
this._currentWorkspace = global.workspace_manager.get_active_workspace_index();
|
||
|
-
|
||
|
- this._updateMenuOrnament();
|
||
|
-
|
||
|
this._statusLabel.set_text(this._getStatusText());
|
||
|
}
|
||
|
|
||
|
- _nWorkspacesChanged() {
|
||
|
- this._updateMenu();
|
||
|
- this._updateThumbnailVisibility();
|
||
|
- }
|
||
|
-
|
||
|
- _updateMenuOrnament() {
|
||
|
- for (let i = 0; i < this._workspacesItems.length; i++) {
|
||
|
- this._workspacesItems[i].setOrnament(i === this._currentWorkspace
|
||
|
- ? PopupMenu.Ornament.DOT
|
||
|
- : PopupMenu.Ornament.NO_DOT);
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
_getStatusText() {
|
||
|
const {nWorkspaces} = global.workspace_manager;
|
||
|
const current = this._currentWorkspace + 1;
|
||
|
return `${current} / ${nWorkspaces}`;
|
||
|
}
|
||
|
|
||
|
- _updateMenuLabels() {
|
||
|
- for (let i = 0; i < this._workspacesItems.length; i++) {
|
||
|
- const item = this._workspacesItems[i];
|
||
|
- item.label.text = Meta.prefs_get_workspace_name(i);
|
||
|
- }
|
||
|
- }
|
||
|
-
|
||
|
- _updateMenu() {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- this.menu.removeAll();
|
||
|
- this._workspacesItems = [];
|
||
|
- this._currentWorkspace = workspaceManager.get_active_workspace_index();
|
||
|
+ _createPreviewMenu() {
|
||
|
+ const menu = new PopupMenu.PopupMenu(this, 0.5, St.Side.TOP);
|
||
|
|
||
|
- for (let i = 0; i < workspaceManager.n_workspaces; i++) {
|
||
|
- const name = Meta.prefs_get_workspace_name(i);
|
||
|
- const item = new PopupMenu.PopupMenuItem(name);
|
||
|
-
|
||
|
- item.connect('activate',
|
||
|
- () => this._activate(i));
|
||
|
-
|
||
|
- item.setOrnament(i === this._currentWorkspace
|
||
|
- ? PopupMenu.Ornament.DOT
|
||
|
- : PopupMenu.Ornament.NO_DOT);
|
||
|
-
|
||
|
- this.menu.addMenuItem(item);
|
||
|
- this._workspacesItems[i] = item;
|
||
|
- }
|
||
|
-
|
||
|
- this._statusLabel.set_text(this._getStatusText());
|
||
|
- }
|
||
|
-
|
||
|
- _activate(index) {
|
||
|
- let workspaceManager = global.workspace_manager;
|
||
|
-
|
||
|
- if (index >= 0 && index < workspaceManager.n_workspaces) {
|
||
|
- let metaWorkspace = workspaceManager.get_workspace_by_index(index);
|
||
|
- metaWorkspace.activate(global.get_current_time());
|
||
|
- }
|
||
|
+ const previews = new WorkspacePreviews({show_labels: true});
|
||
|
+ menu.box.add_child(previews);
|
||
|
+ menu.actor.add_style_class_name(`${baseStyleClassName}-menu`);
|
||
|
+ return menu;
|
||
|
}
|
||
|
});
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From 94b21919c3d86901c24356739fb4baf7b747e990 Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Tue, 20 Feb 2024 22:00:57 +0100
|
||
|
Subject: [PATCH 27/28] workspace-indicator: Make previews configurable
|
||
|
|
||
|
Now that previews scroll when there are too many workspaces,
|
||
|
there is no longer a reason for the 6-workspace limit.
|
||
|
|
||
|
However some users do prefer the menu, so rather than drop it,
|
||
|
turn it into a proper preference.
|
||
|
|
||
|
Closes
|
||
|
https://gitlab.gnome.org/GNOME/gnome-shell-extensions/-/issues/336
|
||
|
---
|
||
|
extensions/window-list/extension.js | 1 +
|
||
|
...e.shell.extensions.window-list.gschema.xml | 4 +++
|
||
|
extensions/workspace-indicator/extension.js | 4 ++-
|
||
|
extensions/workspace-indicator/meson.build | 1 +
|
||
|
extensions/workspace-indicator/prefs.js | 30 +++++++++++++++++++
|
||
|
...extensions.workspace-indicator.gschema.xml | 15 ++++++++++
|
||
|
.../workspace-indicator/workspaceIndicator.js | 14 ++++-----
|
||
|
po/POTFILES.in | 1 +
|
||
|
8 files changed, 62 insertions(+), 8 deletions(-)
|
||
|
create mode 100644 extensions/workspace-indicator/schemas/org.gnome.shell.extensions.workspace-indicator.gschema.xml
|
||
|
|
||
|
diff --git a/extensions/window-list/extension.js b/extensions/window-list/extension.js
|
||
|
index a011bc90..688ca761 100644
|
||
|
--- a/extensions/window-list/extension.js
|
||
|
+++ b/extensions/window-list/extension.js
|
||
|
@@ -776,6 +776,7 @@ class WindowList extends St.Widget {
|
||
|
|
||
|
this._workspaceIndicator = new BottomWorkspaceIndicator({
|
||
|
baseStyleClass: 'window-list-workspace-indicator',
|
||
|
+ settings: ExtensionUtils.getSettings(),
|
||
|
});
|
||
|
indicatorsBox.add_child(this._workspaceIndicator.container);
|
||
|
|
||
|
diff --git a/extensions/window-list/org.gnome.shell.extensions.window-list.gschema.xml b/extensions/window-list/org.gnome.shell.extensions.window-list.gschema.xml
|
||
|
index 299864cf..c5ab9fb1 100644
|
||
|
--- a/extensions/window-list/org.gnome.shell.extensions.window-list.gschema.xml
|
||
|
+++ b/extensions/window-list/org.gnome.shell.extensions.window-list.gschema.xml
|
||
|
@@ -30,5 +30,9 @@
|
||
|
only on the primary one.
|
||
|
</description>
|
||
|
</key>
|
||
|
+ <key name="embed-previews" type="b">
|
||
|
+ <default>true</default>
|
||
|
+ <summary>Show workspace previews in window list</summary>
|
||
|
+ </key>
|
||
|
</schema>
|
||
|
</schemalist>
|
||
|
diff --git a/extensions/workspace-indicator/extension.js b/extensions/workspace-indicator/extension.js
|
||
|
index 5e1ed8e5..ec991993 100644
|
||
|
--- a/extensions/workspace-indicator/extension.js
|
||
|
+++ b/extensions/workspace-indicator/extension.js
|
||
|
@@ -14,7 +14,9 @@ function init() {
|
||
|
let _indicator;
|
||
|
|
||
|
function enable() {
|
||
|
- _indicator = new WorkspaceIndicator();
|
||
|
+ _indicator = new WorkspaceIndicator({
|
||
|
+ settings: ExtensionUtils.getSettings(),
|
||
|
+ });
|
||
|
Main.panel.addToStatusArea('workspace-indicator', _indicator);
|
||
|
}
|
||
|
|
||
|
diff --git a/extensions/workspace-indicator/meson.build b/extensions/workspace-indicator/meson.build
|
||
|
index 0bf9f023..a88db78a 100644
|
||
|
--- a/extensions/workspace-indicator/meson.build
|
||
|
+++ b/extensions/workspace-indicator/meson.build
|
||
|
@@ -8,5 +8,6 @@ extension_data += files(
|
||
|
'stylesheet-dark.css',
|
||
|
'stylesheet-light.css',
|
||
|
)
|
||
|
+extension_schemas += files('schemas/' + metadata_conf.get('gschemaname') + '.gschema.xml')
|
||
|
|
||
|
extension_sources += files('prefs.js', 'workspaceIndicator.js')
|
||
|
diff --git a/extensions/workspace-indicator/prefs.js b/extensions/workspace-indicator/prefs.js
|
||
|
index 567f3e99..d93aa383 100644
|
||
|
--- a/extensions/workspace-indicator/prefs.js
|
||
|
+++ b/extensions/workspace-indicator/prefs.js
|
||
|
@@ -13,6 +13,34 @@ const N_ = e => e;
|
||
|
const WORKSPACE_SCHEMA = 'org.gnome.desktop.wm.preferences';
|
||
|
const WORKSPACE_KEY = 'workspace-names';
|
||
|
|
||
|
+const GeneralGroup = GObject.registerClass(
|
||
|
+class GeneralGroup extends Gtk.Box {
|
||
|
+ _init() {
|
||
|
+ super._init({
|
||
|
+ orientation: Gtk.Orientation.VERTICAL,
|
||
|
+ });
|
||
|
+
|
||
|
+ const row = new Gtk.Box();
|
||
|
+ this.append(row);
|
||
|
+
|
||
|
+ row.append(new Gtk.Label({
|
||
|
+ label: _('Show Previews In Top Bar'),
|
||
|
+ }));
|
||
|
+
|
||
|
+ const sw = new Gtk.Switch({
|
||
|
+ hexpand: true,
|
||
|
+ halign: Gtk.Align.END,
|
||
|
+ });
|
||
|
+ row.append(sw);
|
||
|
+
|
||
|
+ const settings = ExtensionUtils.getSettings();
|
||
|
+
|
||
|
+ settings.bind('embed-previews',
|
||
|
+ sw, 'active',
|
||
|
+ Gio.SettingsBindFlags.DEFAULT);
|
||
|
+ }
|
||
|
+});
|
||
|
+
|
||
|
const WorkspaceSettingsWidget = GObject.registerClass(
|
||
|
class WorkspaceSettingsWidget extends Gtk.ScrolledWindow {
|
||
|
_init() {
|
||
|
@@ -31,6 +59,8 @@ class WorkspaceSettingsWidget extends Gtk.ScrolledWindow {
|
||
|
});
|
||
|
this.set_child(box);
|
||
|
|
||
|
+ box.append(new GeneralGroup());
|
||
|
+
|
||
|
box.append(new Gtk.Label({
|
||
|
label: '<b>%s</b>'.format(_('Workspace Names')),
|
||
|
use_markup: true,
|
||
|
diff --git a/extensions/workspace-indicator/schemas/org.gnome.shell.extensions.workspace-indicator.gschema.xml b/extensions/workspace-indicator/schemas/org.gnome.shell.extensions.workspace-indicator.gschema.xml
|
||
|
new file mode 100644
|
||
|
index 00000000..c7c634ca
|
||
|
--- /dev/null
|
||
|
+++ b/extensions/workspace-indicator/schemas/org.gnome.shell.extensions.workspace-indicator.gschema.xml
|
||
|
@@ -0,0 +1,15 @@
|
||
|
+<!--
|
||
|
+SPDX-FileCopyrightText: 2024 Florian Müllner <fmuellner@gnome.org>
|
||
|
+
|
||
|
+SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
+-->
|
||
|
+
|
||
|
+<schemalist gettext-domain="gnome-shell-extensions">
|
||
|
+ <schema id="org.gnome.shell.extensions.workspace-indicator"
|
||
|
+ path="/org/gnome/shell/extensions/workspace-indicator/">
|
||
|
+ <key name="embed-previews" type="b">
|
||
|
+ <default>true</default>
|
||
|
+ <summary>Show workspace previews in top bar</summary>
|
||
|
+ </key>
|
||
|
+ </schema>
|
||
|
+</schemalist>
|
||
|
diff --git a/extensions/workspace-indicator/workspaceIndicator.js b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
index 368d785a..8c7b2258 100644
|
||
|
--- a/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
+++ b/extensions/workspace-indicator/workspaceIndicator.js
|
||
|
@@ -20,8 +20,6 @@ const PopupMenu = imports.ui.popupMenu;
|
||
|
const TOOLTIP_OFFSET = 6;
|
||
|
const TOOLTIP_ANIMATION_TIME = 150;
|
||
|
|
||
|
-const MAX_THUMBNAILS = 6;
|
||
|
-
|
||
|
let baseStyleClassName = '';
|
||
|
|
||
|
const WindowPreview = GObject.registerClass(
|
||
|
@@ -432,12 +430,15 @@ const WorkspacePreviews = GObject.registerClass({
|
||
|
var WorkspaceIndicator = GObject.registerClass(
|
||
|
class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
_init(params = {}) {
|
||
|
- super(0.5, _('Workspace Indicator'), true);
|
||
|
+ super._init(0.5, _('Workspace Indicator'), true);
|
||
|
|
||
|
const {
|
||
|
baseStyleClass = 'workspace-indicator',
|
||
|
+ settings,
|
||
|
} = params;
|
||
|
|
||
|
+ this._settings = settings;
|
||
|
+
|
||
|
baseStyleClassName = baseStyleClass;
|
||
|
this.add_style_class_name(baseStyleClassName);
|
||
|
|
||
|
@@ -464,8 +465,6 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._workspacesItems = [];
|
||
|
|
||
|
this._workspaceManagerSignals = [
|
||
|
- workspaceManager.connect_after('notify::n-workspaces',
|
||
|
- this._updateThumbnailVisibility.bind(this)),
|
||
|
workspaceManager.connect_after('workspace-switched',
|
||
|
this._onWorkspaceSwitched.bind(this)),
|
||
|
];
|
||
|
@@ -482,6 +481,8 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
this._updateTopBarRedirect();
|
||
|
});
|
||
|
|
||
|
+ this._settings.connect('changed::embed-previews',
|
||
|
+ () => this._updateThumbnailVisibility());
|
||
|
this._updateThumbnailVisibility();
|
||
|
}
|
||
|
|
||
|
@@ -497,8 +498,7 @@ class WorkspaceIndicator extends PanelMenu.Button {
|
||
|
}
|
||
|
|
||
|
_updateThumbnailVisibility() {
|
||
|
- const {workspaceManager} = global;
|
||
|
- const useMenu = workspaceManager.n_workspaces > MAX_THUMBNAILS;
|
||
|
+ const useMenu = !this._settings.get_boolean('embed-previews');
|
||
|
this.reactive = useMenu;
|
||
|
|
||
|
this._statusLabel.visible = useMenu;
|
||
|
diff --git a/po/POTFILES.in b/po/POTFILES.in
|
||
|
index bd39ab61..4d551780 100644
|
||
|
--- a/po/POTFILES.in
|
||
|
+++ b/po/POTFILES.in
|
||
|
@@ -18,4 +18,5 @@ extensions/window-list/prefs.js
|
||
|
extensions/window-list/workspaceIndicator.js
|
||
|
extensions/windowsNavigator/extension.js
|
||
|
extensions/workspace-indicator/prefs.js
|
||
|
+extensions/workspace-indicator/schemas/org.gnome.shell.extensions.workspace-indicator.gschema.xml
|
||
|
extensions/workspace-indicator/workspaceIndicator.js
|
||
|
--
|
||
|
2.44.0
|
||
|
|
||
|
|
||
|
From f8497b510f2778a1ec985b546128096b461aa11e Mon Sep 17 00:00:00 2001
|
||
|
From: =?UTF-8?q?Florian=20M=C3=BCllner?= <fmuellner@gnome.org>
|
||
|
Date: Thu, 21 Mar 2024 17:27:09 +0100
|
||
|
Subject: [PATCH 28/28] window-list: Expose workspace preview option
|
||
|
|
||
|
Now that we have the option, the window-list should expose it
|
||
|
in its preference window like the workspace-indicator.
|
||
|
---
|
||
|
extensions/window-list/prefs.js | 6 ++++++
|
||
|
1 file changed, 6 insertions(+)
|
||
|
|
||
|
diff --git a/extensions/window-list/prefs.js b/extensions/window-list/prefs.js
|
||
|
index aec8cc9d..e870c087 100644
|
||
|
--- a/extensions/window-list/prefs.js
|
||
|
+++ b/extensions/window-list/prefs.js
|
||
|
@@ -102,6 +102,12 @@ class WindowListPrefsWidget extends Gtk.Box {
|
||
|
});
|
||
|
this._settings.bind('display-all-workspaces', check, 'active', Gio.SettingsBindFlags.DEFAULT);
|
||
|
this.append(check);
|
||
|
+
|
||
|
+ check = new Gtk.CheckButton({
|
||
|
+ label: _('Show workspace previews'),
|
||
|
+ });
|
||
|
+ this._settings.bind('embed-previews', check, 'active', Gio.SettingsBindFlags.DEFAULT);
|
||
|
+ this.append(check);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
--
|
||
|
2.44.0
|
||
|
|