15 Commits

Author SHA1 Message Date
Dome a8f7ca7e7e Update README.md 2025-07-28 08:41:19 +02:00
Dome d249abe804 Update README.md 2025-07-28 00:53:12 +02:00
Dome 59b2fe1646 Update modern.js 2025-07-28 00:41:54 +02:00
Dome 6e9fb687fe Update legacy.js 2025-07-28 00:41:24 +02:00
Dome f9c1aee610 Update modern.js 2025-07-28 00:40:55 +02:00
Dome 70a0fd9eab Update exceptions.txt 2025-07-28 00:40:38 +02:00
Dome 2fee754776 Update modern.js 2025-07-28 00:24:08 +02:00
Dome 114b7cc265 Update README.md 2025-07-27 23:49:33 +02:00
Dome 8e6d9ab2d9 Update README.md 2025-07-27 23:47:56 +02:00
Dome bcd14cb87d Update modern.js 2025-07-27 21:42:54 +02:00
Dome 307908cd39 Create legacy.js 2025-07-27 21:18:17 +02:00
Dome fcfdf39059 Update extension.js 2025-07-27 21:17:59 +02:00
Dome e7618cf3e5 Update metadata.json 2025-07-27 21:17:35 +02:00
Dome 811db01995 Create modern.js 2025-07-27 21:17:22 +02:00
Dome ae59b60a7a Update prefs.js 2025-07-27 21:16:51 +02:00
12 changed files with 1060 additions and 1016 deletions
+1 -1
View File
@@ -1,6 +1,6 @@
MIT License MIT License
Copyright (c) 2025 Domoel (https://github.com/Domoel/) Copyright (c) 2025 Dome
Permission is hereby granted, free of charge, to any person obtaining a copy Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal of this software and associated documentation files (the "Software"), to deal
-87
View File
@@ -1,87 +0,0 @@
###############################################################################
# SimpleTiling  Makefile
#
# make build → baut beide ZIPPakete
# make build-legacy → nur LegacyZIP (Shell 3.3844)
# make build-modern → nur ModernZIP (Shell 4548)
# make clean → räumt auf
###############################################################################
UUID := simple-tiling@domoel
VERSION := 6
# Dateien/Ordner, die in *beide* Pakete gehören
COMMON_FILES := schemas exceptions.txt locale *.css README.md LICENSE
# PrefDateien (zwei Varianten)
LEGACY_PREFS := prefs_legacy.js
MODERN_PREFS := prefs_modern.js
###############################################################################
# Helfer: copies <file list> <dest>
###############################################################################
define copies
@for f in $(1) ; do \
if [ -e $$f ] ; then \
cp -r $$f $(2)/ ; \
fi ; \
done
endef
.PHONY: build build-legacy build-modern clean
build: build-legacy build-modern
###############################################################################
# LegacyBuild
###############################################################################
build-legacy:
@echo "==> Building LEGACY package (3.3844)…"
@rm -rf build && mkdir -p build/$(UUID)
$(call copies,$(COMMON_FILES),build/$(UUID))
# Schema kompilieren
@glib-compile-schemas build/$(UUID)/schemas
# Haupt und PrefSkript
@cp legacy.js build/$(UUID)/extension.js
@cp $(LEGACY_PREFS) build/$(UUID)/prefs.js
# metadata.json anpassen
@sed -e "s/__UUID__/$(UUID)/g" \
-e "s/__VERSION__/$(VERSION)/g" \
metadata_legacy.json.in > build/$(UUID)/metadata.json
# ZipPaket
@cd build && zip -qr ../$(UUID)-legacy-v$(VERSION).zip .
@rm -rf build
@echo "✓ created $(UUID)-legacy-v$(VERSION).zip"
###############################################################################
# ModernBuild
###############################################################################
build-modern:
@echo "==> Building MODERN package (4548)…"
@rm -rf build && mkdir -p build/$(UUID)
$(call copies,$(COMMON_FILES),build/$(UUID))
# Schema kompilieren
@glib-compile-schemas build/$(UUID)/schemas
# Haupt und PrefSkript
@cp modern.js build/$(UUID)/extension.js
@cp $(MODERN_PREFS) build/$(UUID)/prefs.js
# metadata.json anpassen
@sed -e "s/__UUID__/$(UUID)/g" \
-e "s/__VERSION__/$(VERSION)/g" \
metadata_modern.json.in > build/$(UUID)/metadata.json
# ZipPaket
@cd build && zip -qr ../$(UUID)-modern-v$(VERSION).zip .
@rm -rf build
@echo "✓ created $(UUID)-modern-v$(VERSION).zip"
###############################################################################
clean:
@rm -rf build $(UUID)-legacy-v$(VERSION).zip $(UUID)-modern-v$(VERSION).zip
@echo "BuildOrdner und ZIPs entfernt."
+22 -34
View File
@@ -4,11 +4,12 @@ Simple Tiling
</span> </span>
<h4 align="center"> <h4 align="center">
<span style="display:inline-flex; align-items:center; gap:12px;"> <span style="display:inline-flex; align-items:center; gap:12px;">
A lightweight, opinionated, and automatic tiling window manager for GNOME Shell A lightweight, opinionated, and automatic tiling window manager for GNOME Shell 3.38.
</span> </span>
<p> <p>
[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT) [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
![GNOME Shell Version](https://img.shields.io/badge/GNOME%20Shell-3.38-blue)
<img width="2560" height="1440" alt="Simple-Tiling-v6" src="https://github.com/user-attachments/assets/eb0f7cc3-6a5a-4036-8a1e-8f945c52e55c" /> <img width="2560" height="1440" alt="Simple-Tiling-v6" src="https://github.com/user-attachments/assets/eb0f7cc3-6a5a-4036-8a1e-8f945c52e55c" />
@@ -16,7 +17,7 @@ A lightweight, opinionated, and automatic tiling window manager for GNOME Shell
Simple Tiling is a GNOME Shell extension created for users who want a clean, predictable, and automatic tiling layout without the complexity of larger, more feature-heavy tiling extensions. It is designed to be simple to configure and intuitive to use, focusing on a core set of essential tiling features. Simple Tiling is a GNOME Shell extension created for users who want a clean, predictable, and automatic tiling layout without the complexity of larger, more feature-heavy tiling extensions. It is designed to be simple to configure and intuitive to use, focusing on a core set of essential tiling features.
This extension was built from the ground up to be stable and performant on **GNOME Shell 3.38**. However it is now also supporting modern gnome shells up to **version 48**. This extension was built from the ground up to be stable and performant on **GNOME Shell 3.38**.
## Features ## Features
@@ -35,10 +36,10 @@ This extension was built from the ground up to be stable and performant on **GNO
## Requirements ## Requirements
Please note that this extension has been developed for a very specific environment. However, with the latest updates, I have ensured that modern Gnome Shells and Wayland are also supported. Please note that this extension has been developed for a very specific environment:
* **GNOME Shell Version:** **3.38 - 48** * **GNOME Shell Version:** **3.38**
* **Session Type:** **X11** (Wayland is still in beta but should be fine!). * **Session Type:** **X11** (Wayland is not supported).
* **Monitor Setup:** **Single monitor only.** Multi-monitor support is not yet implemented. * **Monitor Setup:** **Single monitor only.** Multi-monitor support is not yet implemented.
## Installation ## Installation
@@ -49,40 +50,26 @@ Use the [GNOME Shell Extensions website](https://extensions.gnome.org/extension/
#### Manual Installation #### Manual Installation
The repository includes a Makefile that produces readytoinstall ZIP packages for the two supported GNOMEShell lines (a legacy build (Gnome-Shell 3.38 - 44) and a modern build for Gnome-Shell 45+). 1. **Navigate to your extensions folder:**
1. **Clone the Source**
```bash
git clone https://github.com/YourUser/Simple-Tiling.git
cd Simple-Tiling
```
2 · **Create the package that matches your GNOME-Shell version**
Open the Terminal within the Simple-Tiling directory and run
```bash ```bash
make build cd ~/.local/share/gnome-shell/extensions/
``` ```
**Note:** This will create a ready to go .zip archive of both, the modern and the legacy version of the extension ready to be used. Alternativley you can also run "make build-legacy" or "make build-modern" to only compile one of both versions. 3. **Clone the repository directly into a folder named after the extension's UUID:**
3 · **Locate the output** ```bash
```bash git clone https://github.com/Domoel/Simple-Tiling.git simple-tiling@domoel
ls -1 ../simple-tiling@domoel-*-v*.zip ```
``` 5. **Compile the GSettings schema.** This is a mandatory step for the keyboard shortcuts to work.
4 · **Install & enable**
```bash
gnome-extensions install ../simple-tiling@domoel-legacy-v6.zip
gnome-extensions enable simple-tiling@domoel
```
**Note:** You can also unzip the file and put the folder right into your extensions directory (~/.local/share/gnome-shell/extensions/)
5 · **Reload the shell** ```bash
```bash cd ~/.local/share/gnome-shell/extensions/simple-tiling@domoel
Press Alt + F2, type r , hit ↩ (works for X11 and Wayland) glib-compile-schemas schemas/
``` ```
6 · **Clean up (optional)**
```bash 3. **Restart GNOME Shell.** Press `Alt` + `F2`, type `r`, and press `Enter`.
make clean # removes build/ folder and generated ZIPs
``` 5. **Enable the extension** using the GNOME Extensions app or GNOME Tweaks.
**Note:** You have to use "simple-tiling@domoel" as your extension folder / directory. Put all necessary files into this directory. Otherwise the extension will not show up in extension manager. **Note:** You have to use "simple-tiling@domoel" as your extension folder / directory. Put all necessary files into this directory. Otherwise the extension will not show up in extension manager.
@@ -126,6 +113,7 @@ If you have race condition issues between mutter (Gnome WM) and the Simple Tilin
This extension was built to solve a specific need. However, future enhancements could include: This extension was built to solve a specific need. However, future enhancements could include:
* Multi-monitor support. * Multi-monitor support.
* Support for newer Gnome shells
* Additional layout algorithms. * Additional layout algorithms.
* A more detailed settings panel to configure other options via a GUI. * A more detailed settings panel to configure other options via a GUI.
-9
View File
@@ -24,12 +24,3 @@
# --- Start of the Exception List --- # --- Start of the Exception List ---
ulauncher ulauncher
steam
element
totem
extension-manager
timeshift-gtk
gnome-screenshot
org.gnome.NautilusPreviewer
org.gnome.Shell.Extensions
evolution-alarm-notify
+41
View File
@@ -0,0 +1,41 @@
/////////////////////////////////////////////////////////////
// Simple-Tiling GLOBAL CONFIG //
// © 2025 domoel MIT //
//////////////////////////////////////////////////////////
// --- GLOBAL IMPORTS ---
'use strict';
const ExtensionUtils = imports.misc.extensionUtils;
const Config = imports.misc.config;
const Me = ExtensionUtils.getCurrentExtension();
const [SHELL_MAJOR] = Config.PACKAGE_VERSION.split('.').map(n => parseInt(n));
let extension = null;
function init() {
}
// --- SHELL SWITCH ---
async function enable() {
try {
if (SHELL_MAJOR >= 40) {
const module = await import('./modern.js');
extension = new module.default(Me.metadata);
} else {
const { LegacyExtension } = Me.imports.legacy;
extension = new LegacyExtension(Me.metadata);
}
extension.enable();
} catch (e) {
logError(e, `[Simple Tiling] Failed to enable extension`);
}
}
function disable() {
if (extension) {
extension.disable();
extension = null;
}
}
+475 -347
View File
File diff suppressed because it is too large Load Diff
+4 -3
View File
@@ -1,9 +1,10 @@
{ {
"uuid": "__UUID__", "uuid": "simple-tiling@domoel",
"name": "Simple Tiling", "name": "Simple Tiling",
"description": "A Simple Tiling Extension for Gnome Shell.", "description": "A Simple Tiling Extension for Gnome Shell.",
"version": __VERSION__, "version": 6,
"shell-version": [ "shell-version": [
"3.38",
"45", "45",
"46", "46",
"47", "47",
@@ -12,5 +13,5 @@
"settings-schema": "org.gnome.shell.extensions.simple-tiling.domoel", "settings-schema": "org.gnome.shell.extensions.simple-tiling.domoel",
"preferences_ui": "prefs.js", "preferences_ui": "prefs.js",
"url": "https://github.com/Domoel/Simple-Tiling", "url": "https://github.com/Domoel/Simple-Tiling",
"gettext-domain": "__UUID__" "gettext-domain": "simple-tiling-domoel"
} }
-18
View File
@@ -1,18 +0,0 @@
{
"uuid": "__UUID__",
"name": "Simple Tiling",
"description": "A Simple Tiling Extension for Gnome Shell.",
"version": __VERSION__,
"shell-version": [
"3.38",
"40",
"41",
"42",
"43",
"44"
],
"settings-schema": "org.gnome.shell.extensions.simple-tiling.domoel",
"preferences_ui": "prefs.js",
"url": "https://github.com/Domoel/Simple-Tiling",
"gettext-domain": "__UUID__"
}
+296 -201
View File
@@ -1,117 +1,109 @@
///////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////
// SimpleTiling  MODERN (GNOME Shell 45+) // // Simple-Tiling MODERN (for GNOME Shell 40+) //
// ©2025domoel  MIT // // © 2025 domoel MIT //
///////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////
// ── GLOBAL IMPORTS ──────────────────────────────────────── // --- GLOBAL IMPORTS ---
import { Extension } from 'resource:///org/gnome/shell/extensions/extension.js'; import { Extension } from "resource:///org/gnome/shell/extensions/extension.js";
import * as Main from 'resource:///org/gnome/shell/ui/main.js'; import * as Main from "resource:///org/gnome/shell/ui/main.js";
import Meta from 'gi://Meta'; import Meta from "gi://Meta";
import Shell from 'gi://Shell'; import Shell from "gi://Shell";
import Gio from 'gi://Gio'; import Gio from "gi://Gio";
import GLib from 'gi://GLib'; import GLib from "gi://GLib";
import Clutter from 'gi://Clutter';
// ── CONST ──────────────────────────────────────────── const SCHEMA_NAME = "org.gnome.shell.extensions.simple-tiling.domoel";
const WM_SCHEMA = 'org.gnome.desktop.wm.keybindings'; const WM_SCHEMA = "org.gnome.desktop.wm.keybindings";
const TILING_DELAY_MS = 20; // Change Tiling Window Delay const TILING_DELAY_MS = 20; // Change Tiling Window Delay
const CENTERING_DELAY_MS = 5; // Change Centered Window Delay const CENTERING_DELAY_MS = 5; // Change Centered Window Delay
const KEYBINDINGS = { const KEYBINDINGS = {
'swap-master-window': (self) => self._swapWithMaster(), "swap-master-window": (self) => self._swapWithMaster(),
'swap-left-window': (self) => self._swapInDirection('left'), "swap-left-window": (self) => self._swapInDirection("left"),
'swap-right-window': (self) => self._swapInDirection('right'), "swap-right-window": (self) => self._swapInDirection("right"),
'swap-up-window': (self) => self._swapInDirection('up'), "swap-up-window": (self) => self._swapInDirection("up"),
'swap-down-window': (self) => self._swapInDirection('down'), "swap-down-window": (self) => self._swapInDirection("down"),
'focus-left': (self) => self._focusInDirection('left'), "focus-left": (self) => self._focusInDirection("left"),
'focus-right': (self) => self._focusInDirection('right'), "focus-right": (self) => self._focusInDirection("right"),
'focus-up': (self) => self._focusInDirection('up'), "focus-up": (self) => self._focusInDirection("up"),
'focus-down': (self) => self._focusInDirection('down'), "focus-down": (self) => self._focusInDirection("down"),
}; };
// ── HELPERFUNCTION ──────────────────────────────────────── // --- INTERACTIONHANDLER ---
function getPointerXY() {
if (global.get_pointer) {
const [x, y] = global.get_pointer();
return [x, y];
}
const ev = Clutter.get_current_event();
if (ev) {
const coords = ev.get_coords();
if (Array.isArray(coords))
return coords;
}
const device = Clutter.get_default_backend()
.get_default_seat()
.get_pointer();
return device ? device.get_position() : [0, 0];
}
// ── INTERACTIONHANDLER ───────────────────────────────────
class InteractionHandler { class InteractionHandler {
constructor(tiler) { constructor(tiler) {
this.tiler = tiler; this.tiler = tiler;
this._settings = this.tiler.settings; this._settings = this.tiler.settings;
this._wmSettings = new Gio.Settings({ schema: WM_SCHEMA }); this._wmSettings = new Gio.Settings({ schema: WM_SCHEMA });
this._wmKeysToDisable = [];
this._wmKeysToDisable = []; this._savedWmShortcuts = {};
this._savedWmShortcuts = {}; this._grabOpIds = [];
this._grabOpIds = [];
this._settingsChangedId = null; this._settingsChangedId = null;
} }
enable() { enable() {
this._prepareWmShortcuts(); this._prepareWmShortcuts();
if (this._wmKeysToDisable.length) {
if (this._wmKeysToDisable.length) this._wmKeysToDisable.forEach((key) =>
this._wmKeysToDisable.forEach(k => this._wmSettings.set_value(key, new GLib.Variant("as", []))
this._wmSettings.set_value(k, new GLib.Variant('as', []))); );
}
this._bindAllShortcuts(); this._bindAllShortcuts();
this._settingsChangedId = this._settingsChangedId = this._settings.connect("changed", () =>
this._settings.connect('changed', () => this._onSettingsChanged()); this._onSettingsChanged()
this._grabOpIds.push(
global.display.connect('grab-op-begin',
(_, __, win) => { if (this.tiler.windows.includes(win))
this.tiler.grabbedWindow = win; })
); );
this._grabOpIds.push( this._grabOpIds.push(
global.display.connect('grab-op-end', () => this._onGrabEnd()) global.display.connect(
"grab-op-begin",
(display, screen, window) => {
if (this.tiler.windows.includes(window)) {
this.tiler.grabbedWindow = window;
}
}
)
);
this._grabOpIds.push(
global.display.connect("grab-op-end", () => this._onGrabEnd())
); );
} }
disable() { disable() {
if (this._wmKeysToDisable.length) if (this._wmKeysToDisable.length) {
this._wmKeysToDisable.forEach(k => this._wmKeysToDisable.forEach((key) =>
this._wmSettings.set_value(k, this._savedWmShortcuts[k])); this._wmSettings.set_value(key, this._savedWmShortcuts[key])
);
}
this._unbindAllShortcuts(); this._unbindAllShortcuts();
if (this._settingsChangedId) { if (this._settingsChangedId) {
this._settings.disconnect(this._settingsChangedId); this._settings.disconnect(this._settingsChangedId);
this._settingsChangedId = null; this._settingsChangedId = null;
} }
this._grabOpIds.forEach(id => global.display.disconnect(id)); this._grabOpIds.forEach((id) => global.display.disconnect(id));
this._grabOpIds = []; this._grabOpIds = [];
} }
_bind(key, handler) { _bind(key, callback) {
global.display.add_keybinding( global.display.add_keybinding(
key, key,
this._settings, this._settings,
Meta.KeyBindingFlags.NONE, Meta.KeyBindingFlags.NONE,
Shell.ActionMode.NORMAL, Shell.ActionMode.NORMAL,
(..._args) => handler(this) callback
); );
} }
_bindAllShortcuts() { for (const [k,h] of Object.entries(KEYBINDINGS)) this._bind(k, h); } _bindAllShortcuts() {
_unbindAllShortcuts(){ for (const k in KEYBINDINGS) global.display.remove_keybinding(k); } for (const [key, handler] of Object.entries(KEYBINDINGS)) {
this._bind(key, () => handler(this));
}
}
_unbindAllShortcuts() {
for (const key in KEYBINDINGS) {
global.display.remove_keybinding(key);
}
}
_onSettingsChanged() { _onSettingsChanged() {
this._unbindAllShortcuts(); this._unbindAllShortcuts();
@@ -119,168 +111,253 @@ class InteractionHandler {
} }
_prepareWmShortcuts() { _prepareWmShortcuts() {
const schema = this._wmSettings.settings_schema; const schema = this._wmSettings
.get_schema_source()
.lookup(WM_SCHEMA, true);
if (!schema) return; if (!schema) return;
const addKeyIfExists = (keys, key) => {
if (schema.has_key(key)) keys.push(key);
};
const keys = []; const keys = [];
if (schema.has_key("toggle-tiled-left")) {
const add = key => { if (schema.has_key(key)) keys.push(key); }; keys.push("toggle-tiled-left", "toggle-tiled-right");
} else {
if (schema.has_key('toggle-tiled-left')) addKeyIfExists(keys, "tile-left");
keys.push('toggle-tiled-left', 'toggle-tiled-right'); addKeyIfExists(keys, "tile-right");
else {
add('tile-left'); add('tile-right');
} }
if (schema.has_key('toggle-maximized')) if (schema.has_key("toggle-maximized")) {
keys.push('toggle-maximized'); keys.push("toggle-maximized");
else { } else {
add('maximize'); add('unmaximize'); addKeyIfExists(keys, "maximize");
addKeyIfExists(keys, "unmaximize");
} }
if (keys.length) { if (keys.length) {
this._wmKeysToDisable = keys; this._wmKeysToDisable = keys;
keys.forEach(k => this._savedWmShortcuts[k] = keys.forEach(
this._wmSettings.get_value(k)); (key) =>
(this._savedWmShortcuts[key] = this._wmSettings.get_value(
key
))
);
} }
} }
_focusInDirection(direction) { _focusInDirection(direction) {
const src = global.display.get_focus_window(); const sourceWindow = global.display.get_focus_window();
if (!src || !this.tiler.windows.includes(src)) return; if (!sourceWindow || !this.tiler.windows.includes(sourceWindow)) return;
const tgt = this._findTargetInDirection(src, direction);
if (tgt) tgt.activate(global.get_current_time()); const targetWindow = this._findTargetInDirection(
sourceWindow,
direction
);
if (targetWindow) {
targetWindow.activate(global.get_current_time());
}
} }
_swapWithMaster() { _swapWithMaster() {
const w = this.tiler.windows; const windows = this.tiler.windows;
if (w.length < 2) return; if (windows.length < 2) return;
const foc = global.display.get_focus_window(); const focusedWindow = global.display.get_focus_window();
if (!foc || !w.includes(foc)) return; if (!focusedWindow || !windows.includes(focusedWindow)) return;
const idx = w.indexOf(foc); const focusedIndex = windows.indexOf(focusedWindow);
if (idx > 0) [w[0], w[idx]] = [w[idx], w[0]]; if (focusedIndex > 0) {
else [w[0], w[1]] = [w[1], w[0]]; [windows[0], windows[focusedIndex]] = [
windows[focusedIndex],
windows[0],
];
} else if (focusedIndex === 0) {
[windows[0], windows[1]] = [windows[1], windows[0]];
}
this.tiler.tileNow(); this.tiler.tileNow();
w[0]?.activate(global.get_current_time()); if (windows.length > 0) windows[0].activate(global.get_current_time());
} }
_swapInDirection(direction) { _swapInDirection(direction) {
const src = global.display.get_focus_window(); const sourceWindow = global.display.get_focus_window();
if (!src || !this.tiler.windows.includes(src)) return; if (!sourceWindow || !this.tiler.windows.includes(sourceWindow)) return;
let tgt = null; let targetWindow = null;
const idx = this.tiler.windows.indexOf(src); const sourceIndex = this.tiler.windows.indexOf(sourceWindow);
if (idx === 0 && direction==='right' && this.tiler.windows.length>1) if (
tgt = this.tiler.windows[1]; sourceIndex === 0 &&
else direction === "right" &&
tgt = this._findTargetInDirection(src, direction); this.tiler.windows.length > 1
if (!tgt) return; ) {
const tidx = this.tiler.windows.indexOf(tgt); targetWindow = this.tiler.windows[1];
[this.tiler.windows[idx], this.tiler.windows[tidx]] = } else {
[this.tiler.windows[tidx], this.tiler.windows[idx]]; targetWindow = this._findTargetInDirection(sourceWindow, direction);
}
if (!targetWindow) return;
const targetIndex = this.tiler.windows.indexOf(targetWindow);
[this.tiler.windows[sourceIndex], this.tiler.windows[targetIndex]] = [
this.tiler.windows[targetIndex],
this.tiler.windows[sourceIndex],
];
this.tiler.tileNow(); this.tiler.tileNow();
src.activate(global.get_current_time()); sourceWindow.activate(global.get_current_time());
} }
_findTargetInDirection(src, dir) { _findTargetInDirection(source, direction) {
const sRect = src.get_frame_rect(), cand=[]; const sourceRect = source.get_frame_rect();
let candidates = [];
for (const win of this.tiler.windows) { for (const win of this.tiler.windows) {
if (win===src) continue; if (win === source) continue;
const r=win.get_frame_rect(); const targetRect = win.get_frame_rect();
if (dir==='left' && r.x<sRect.x) cand.push(win); switch (direction) {
if (dir==='right'&& r.x>sRect.x) cand.push(win); case "left":
if (dir==='up' && r.y<sRect.y) cand.push(win); if (targetRect.x < sourceRect.x) candidates.push(win);
if (dir==='down' && r.y>sRect.y) cand.push(win); break;
case "right":
if (targetRect.x > sourceRect.x) candidates.push(win);
break;
case "up":
if (targetRect.y < sourceRect.y) candidates.push(win);
break;
case "down":
if (targetRect.y > sourceRect.y) candidates.push(win);
break;
}
} }
if (!cand.length) return null; if (candidates.length === 0) return null;
let best=null, min=Infinity; let bestTarget = null;
for (const w of cand) { let minDeviation = Infinity;
const r=w.get_frame_rect(); for (const win of candidates) {
const dev = (dir==='left'||dir==='right') const targetRect = win.get_frame_rect();
? Math.abs(sRect.y - r.y) let deviation;
: Math.abs(sRect.x - r.x); if (direction === "left" || direction === "right") {
if (dev<min){min=dev; best=w;} deviation = Math.abs(sourceRect.y - targetRect.y);
} else {
deviation = Math.abs(sourceRect.x - targetRect.x);
}
if (deviation < minDeviation) {
minDeviation = deviation;
bestTarget = win;
}
} }
return best; return bestTarget;
} }
_onGrabEnd() { _onGrabEnd() {
const grabbed = this.tiler.grabbedWindow; const grabbedWindow = this.tiler.grabbedWindow;
if (!grabbed) return; if (!grabbedWindow) return;
const tgt = this._findTargetUnderPointer(grabbed); const targetWindow = this._findTargetUnderPointer(grabbedWindow);
if (tgt) { if (targetWindow) {
const a = this.tiler.windows.indexOf(grabbed); const sourceIndex = this.tiler.windows.indexOf(grabbedWindow);
const b = this.tiler.windows.indexOf(tgt); const targetIndex = this.tiler.windows.indexOf(targetWindow);
[this.tiler.windows[a], this.tiler.windows[b]] = [
[this.tiler.windows[b], this.tiler.windows[a]]; this.tiler.windows[sourceIndex],
this.tiler.windows[targetIndex],
] = [
this.tiler.windows[targetIndex],
this.tiler.windows[sourceIndex],
];
} }
this.tiler.queueTile(); this.tiler.queueTile();
this.tiler.grabbedWindow = null; this.tiler.grabbedWindow = null;
} }
_findTargetUnderPointer(exclude) { _findTargetUnderPointer(excludeWindow) {
const [x,y] = getPointerXY(); let [pointerX, pointerY] = global.get_pointer();
const wins = global.get_window_actors() let windows = global
.map(a=>a.meta_window) .get_window_actors()
.filter(w=>w && w!==exclude && .map((actor) => actor.meta_window)
this.tiler.windows.includes(w) && (()=>{const f=w.get_frame_rect(); .filter((win) => {
return x>=f.x && x<f.x+f.width && if (
y>=f.y && y<f.y+f.height;})()); !win ||
if (wins.length) return wins[wins.length-1]; win === excludeWindow ||
!this.tiler.windows.includes(win)
let best=null, max=0, sRect=exclude.get_frame_rect(); )
for (const w of this.tiler.windows) { return false;
if (w===exclude) continue; let frame = win.get_frame_rect();
const r=w.get_frame_rect(); return (
const ovX=Math.max(0, Math.min(sRect.x+sRect.width, r.x+r.width)-Math.max(sRect.x,r.x)); pointerX >= frame.x &&
const ovY=Math.max(0, Math.min(sRect.y+sRect.height,r.y+r.height)-Math.max(sRect.y,r.y)); pointerX < frame.x + frame.width &&
const area=ovX*ovY; pointerY >= frame.y &&
if (area>max){max=area; best=w;} pointerY < frame.y + frame.height
);
});
if (windows.length > 0) {
return windows[windows.length - 1];
} }
return best;
let bestTarget = null;
let maxOverlap = 0;
const sourceFrame = excludeWindow.get_frame_rect();
for (const win of this.tiler.windows) {
if (win === excludeWindow) continue;
const targetFrame = win.get_frame_rect();
const overlapX = Math.max(
0,
Math.min(
sourceFrame.x + sourceFrame.width,
targetFrame.x + targetFrame.width
) - Math.max(sourceFrame.x, targetFrame.x)
);
const overlapY = Math.max(
0,
Math.min(
sourceFrame.y + sourceFrame.height,
targetFrame.y + targetFrame.height
) - Math.max(sourceFrame.y, targetFrame.y)
);
const overlapArea = overlapX * overlapY;
if (overlapArea > maxOverlap) {
maxOverlap = overlapArea;
bestTarget = win;
}
}
return bestTarget;
} }
} }
// ── TILER ──────────────────────────────────────────────── // --- TILER ---
class Tiler { class Tiler {
constructor(extension) { constructor(extension) {
this._extension = extension; this._extension = extension;
this.settings = this._extension.getSettings(); this.settings = this._extension.getSettings();
this.windows = []; this.windows = [];
this.grabbedWindow = null; this.grabbedWindow = null;
this._signalIds = new Map(); this._signalIds = new Map();
this._tileInProgress = false; this._tileInProgress = false;
this._innerGap = this.settings.get_int('inner-gap'); this._innerGap = this.settings.get_int("inner-gap");
this._outerGapVertical= this.settings.get_int('outer-gap-vertical'); this._outerGapVertical = this.settings.get_int("outer-gap-vertical");
this._outerGapHorizontal = this.settings.get_int('outer-gap-horizontal'); this._outerGapHorizontal = this.settings.get_int(
"outer-gap-horizontal"
);
this._tilingDelay = TILING_DELAY_MS; this._tilingDelay = TILING_DELAY_MS;
this._centeringDelay = CENTERING_DELAY_MS; this._centeringDelay = CENTERING_DELAY_MS;
this._exceptions = []; this._exceptions = [];
this._interactionHandler = new InteractionHandler(this); this._interactionHandler = new InteractionHandler(this);
this._tileTimeoutId = null; this._tileTimeoutId = null;
this._centerTimeoutIds= []; this._centerTimeoutIds = [];
} }
enable() { enable() {
this._loadExceptions(); this._loadExceptions();
this._workspaceManager = global.workspace_manager; this._workspaceManager = global.workspace_manager;
this._signalIds.set('workspace-changed', { this._signalIds.set("workspace-changed", {
object: this._workspaceManager, object: this._workspaceManager,
id: this._workspaceManager.connect('active-workspace-changed', id: this._workspaceManager.connect("active-workspace-changed", () =>
()=>this._onActiveWorkspaceChanged()) this._onActiveWorkspaceChanged()
),
}); });
this._connectToWorkspace(); this._connectToWorkspace();
this._interactionHandler.enable(); this._interactionHandler.enable();
this._signalIds.set("settings-changed", {
this._signalIds.set('settings-changed', {
object: this.settings, object: this.settings,
id: this.settings.connect('changed', ()=>this._onSettingsChanged()) id: this.settings.connect("changed", () =>
this._onSettingsChanged()
),
}); });
} }
@@ -289,38 +366,47 @@ class Tiler {
GLib.source_remove(this._tileTimeoutId); GLib.source_remove(this._tileTimeoutId);
this._tileTimeoutId = null; this._tileTimeoutId = null;
} }
this._centerTimeoutIds.forEach(id=>GLib.source_remove(id)); this._centerTimeoutIds.forEach((id) => GLib.source_remove(id));
this._centerTimeoutIds = []; this._centerTimeoutIds = [];
this._interactionHandler.disable(); this._interactionHandler.disable();
this._disconnectFromWorkspace(); this._disconnectFromWorkspace();
for (const [, signal] of this._signalIds) {
for (const [,sig] of this._signalIds) { try {
try { sig.object.disconnect(sig.id); } catch {} signal.object.disconnect(signal.id);
} catch (e) {}
} }
this._signalIds.clear(); this._signalIds.clear();
this.windows = []; this.windows = [];
} }
_onSettingsChanged() { _onSettingsChanged() {
this._innerGap = this.settings.get_int('inner-gap'); this._innerGap = this.settings.get_int("inner-gap");
this._outerGapVertical = this.settings.get_int('outer-gap-vertical'); this._outerGapVertical = this.settings.get_int("outer-gap-vertical");
this._outerGapHorizontal= this.settings.get_int('outer-gap-horizontal'); this._outerGapHorizontal = this.settings.get_int(
"outer-gap-horizontal"
);
this.queueTile(); this.queueTile();
} }
_loadExceptions() { _loadExceptions() {
const file = Gio.File.new_for_path(this._extension.path + '/exceptions.txt'); const file = Gio.File.new_for_path(
if (!file.query_exists(null)) { this._exceptions=[]; return; } this._extension.path + "/exceptions.txt"
);
const [ok,data] = file.load_contents(null); if (!file.query_exists(null)) {
if (!ok) { this._exceptions=[]; return; } this._exceptions = [];
return;
const txt = new TextDecoder('utf-8').decode(data); }
this._exceptions = txt.split('\n') const [ok, data] = file.load_contents(null);
.map(l=>l.trim()) if (ok) {
.filter(l=>l && !l.startsWith('#')) this._exceptions = GLib.locale_from_utf8(data)
.map(l=>l.toLowerCase()); .split("\n")
.map((l) => l.trim())
.filter((l) => l && !l.startsWith("#"))
.map((l) => l.toLowerCase());
} else {
this._exceptions = [];
}
} }
_isException(win) { _isException(win) {
@@ -591,8 +677,17 @@ class Tiler {
} }
} }
// ── EXTENSIONWRAPPER ─────────────────────────────────── // --- MODERN EXTENSION WRAPPER ---
export default class ModernExtension extends Extension { export default class ModernExtension extends Extension {
enable() { this.tiler = new Tiler(this); this.tiler.enable(); } enable() {
disable() { this.tiler?.disable(); this.tiler = null; } this.tiler = new Tiler(this);
this.tiler.enable();
}
disable() {
if (this.tiler) {
this.tiler.disable();
this.tiler = null;
}
}
} }
+221
View File
@@ -0,0 +1,221 @@
///////////////////////////////////////////////////////
// --- Extension Settings Menu for Simple Tiling --- //
// --- © 2025 domoel MIT --- //
///////////////////////////////////////////////////////
// --- GLOBAL IMPORTS ---
'use strict';
const { Gtk, GObject, Gio, GLib } = imports.gi;
const ExtensionUtils = imports.misc.extensionUtils;
// --- Global Version Checkup ---
let Adw;
try {
Adw = imports.gi.Adw;
} catch (e) {
Adw = null;
}
// --- DEFINITIONS ---
// --- Definitions for GNOME Shell 40+ ---
const ModernPrefs = Adw ? class extends ExtensionPreferences {
fillPreferencesWindow(window) {
const settings = this.getSettings();
const page = new Adw.PreferencesPage();
window.add(page);
// --- Window Gaps ---
const groupGaps = new Adw.PreferencesGroup({ title: 'Window Gaps' });
page.add(groupGaps);
const rowInnerGap = new Adw.SpinRow({
title: 'Inner Gap',
subtitle: 'The gap between windows in pixels.',
adjustment: new Gtk.Adjustment({ lower: 0, upper: 100, step_increment: 1 }),
});
groupGaps.add(rowInnerGap);
settings.bind('inner-gap', rowInnerGap, 'value', Gio.SettingsBindFlags.DEFAULT);
const rowOuterHGap = new Adw.SpinRow({
title: 'Outer Gap (horizontal)',
subtitle: 'The gap to the left and right screen edges.',
adjustment: new Gtk.Adjustment({ lower: 0, upper: 100, step_increment: 1 }),
});
groupGaps.add(rowOuterHGap);
settings.bind('outer-gap-horizontal', rowOuterHGap, 'value', Gio.SettingsBindFlags.DEFAULT);
const rowOuterVGap = new Adw.SpinRow({
title: 'Outer Gap (vertical)',
subtitle: 'The gap to the top and bottom screen edges.',
adjustment: new Gtk.Adjustment({ lower: 0, upper: 100, step_increment: 1 }),
});
groupGaps.add(rowOuterVGap);
settings.bind('outer-gap-vertical', rowOuterVGap, 'value', Gio.SettingsBindFlags.DEFAULT);
// --- Window Behavior ---
const groupBehavior = new Adw.PreferencesGroup({ title: 'Window Behavior' });
page.add(groupBehavior);
const rowNewWindow = new Adw.ComboRow({
title: 'Open new windows as',
subtitle: 'Determines if a new window is added as master or stack window.',
model: new Gtk.StringList({ strings: ['Stack Window (Default)', 'Master Window'] }),
});
groupBehavior.add(rowNewWindow);
const mapping = new Gio.SettingsBindMapping({
settings: settings, key: 'new-window-behavior', property: 'selected',
get_mapping: (value, variant_type) => value === 'master' ? 1 : 0,
set_mapping: (value, param_type) => new GLib.Variant('s', value === 1 ? 'master' : 'stack'),
});
settings.bind_with_mapping('new-window-behavior', rowNewWindow, 'selected', Gio.SettingsBindFlags.DEFAULT, mapping);
// --- Keybindings ---
const groupKeys = new Adw.PreferencesGroup({ title: 'Keybindings' });
page.add(groupKeys);
const rowKeys = new Adw.ActionRow({
title: 'Configure Shortcuts',
subtitle: 'All shortcuts can be configured in GNOME\'s main Keyboard settings.',
});
groupKeys.add(rowKeys);
const button = new Gtk.Button({ label: 'Open Keyboard Settings', valign: Gtk.Align.CENTER });
button.connect('clicked', () => {
const appInfo = Gio.AppInfo.create_from_commandline(
'gnome-control-center keyboard', null, Gio.AppInfoCreateFlags.NONE
);
appInfo.launch([], null);
});
rowKeys.add_suffix(button);
rowKeys.set_activatable_widget(button);
}
} : null;
// --- Definitions for GNOME Shell 3.38 ---
const buildLegacyPrefsWidget = () => {
const settings = ExtensionUtils.getSettings("org.gnome.shell.extensions.simple-tiling.domoel");
const prefsWidget = new Gtk.Box({
orientation: Gtk.Orientation.VERTICAL,
margin_top: 20, margin_bottom: 20, margin_start: 20, margin_end: 20,
spacing: 18,
visible: true,
});
// --- Keybindings ---
const keysTitle = new Gtk.Label({ label: "<b>Keybindings</b>", use_markup: true, halign: Gtk.Align.START, visible: true });
const keysFrame = new Gtk.Frame({ label_widget: keysTitle, shadow_type: Gtk.ShadowType.NONE, visible: true });
let keysBox = new Gtk.Box({ orientation: Gtk.Orientation.VERTICAL, margin: 12, spacing: 6, visible: true });
keysFrame.add(keysBox);
let store = new Gtk.ListStore();
store.set_column_types([ GObject.TYPE_STRING, GObject.TYPE_STRING, GObject.TYPE_INT, GObject.TYPE_INT ]);
const COLUMN_ID = 0, COLUMN_DESC = 1, COLUMN_KEY = 2, COLUMN_MODS = 3;
const addKeybinding = (id, desc) => {
let [key, mods] = [0, 0];
const strv = settings.get_strv(id);
if (strv && strv[0]) [key, mods] = Gtk.accelerator_parse(strv[0]);
let iter = store.append();
store.set(iter, [COLUMN_ID, COLUMN_DESC, COLUMN_KEY, COLUMN_MODS], [id, desc, key, mods]);
};
addKeybinding("swap-master-window", "Swap current window with master");
addKeybinding("swap-up-window", "Swap current window with window above");
addKeybinding("swap-down-window", "Swap current window with window below");
addKeybinding("swap-left-window", "Swap current window with window to the left");
addKeybinding("swap-right-window", "Swap current window with window to the right");
addKeybinding("focus-up", "Focus window above");
addKeybinding("focus-down", "Focus window below");
addKeybinding("focus-left", "Focus window to the left");
addKeybinding("focus-right", "Focus window to the right");
let treeView = new Gtk.TreeView({ model: store, headers_visible: false, hexpand: true, visible: true });
keysBox.add(treeView);
let descRenderer = new Gtk.CellRendererText();
let descColumn = new Gtk.TreeViewColumn({ expand: true });
descColumn.pack_start(descRenderer, true);
descColumn.add_attribute(descRenderer, "text", COLUMN_DESC);
treeView.append_column(descColumn);
let accelRenderer = new Gtk.CellRendererAccel({ "accel-mode": Gtk.CellRendererAccelMode.GTK, editable: true });
accelRenderer.connect("accel-edited", (r, path, key, mods) => {
let [ok, iter] = store.get_iter_from_string(path);
if (ok) {
store.set(iter, [COLUMN_KEY, COLUMN_MODS], [key, mods]);
settings.set_strv(store.get_value(iter, COLUMN_ID), [ Gtk.accelerator_name(key, mods) ]);
}
});
accelRenderer.connect("accel-cleared", (r, path) => {
let [ok, iter] = store.get_iter_from_string(path);
if (ok) {
store.set(iter, [COLUMN_KEY, COLUMN_MODS], [0, 0]);
settings.set_strv(store.get_value(iter, COLUMN_ID), []);
}
});
let accelColumn = new Gtk.TreeViewColumn();
accelColumn.pack_end(accelRenderer, false);
accelColumn.add_attribute(accelRenderer, "accel-key", COLUMN_KEY);
accelColumn.add_attribute(accelRenderer, "accel-mods", COLUMN_MODS);
treeView.append_column(accelColumn);
prefsWidget.add(keysFrame);
// --- Window Gaps ---
const gapsTitle = new Gtk.Label({ label: "<b>Window Gaps</b>", use_markup: true, halign: Gtk.Align.START, visible: true });
const gapsFrame = new Gtk.Frame({ label_widget: gapsTitle, shadow_type: Gtk.ShadowType.NONE, visible: true });
const gapsGrid = new Gtk.Grid({ margin: 12, column_spacing: 12, row_spacing: 12, visible: true });
gapsFrame.add(gapsGrid);
const addSpinButtonRow = (desc, key, pos) => {
const label = new Gtk.Label({ label: desc, halign: Gtk.Align.START, visible: true });
gapsGrid.attach(label, 0, pos, 1, 1);
const adj = new Gtk.Adjustment({ lower: 0, upper: 50, step_increment: 1 });
const spin = new Gtk.SpinButton({ adjustment: adj, climb_rate: 1, digits: 0, halign: Gtk.Align.END, visible: true });
settings.bind(key, spin, "value", Gio.SettingsBindFlags.DEFAULT);
gapsGrid.attach(spin, 1, pos, 1, 1);
};
addSpinButtonRow("Inner Gap", "inner-gap", 0);
addSpinButtonRow("Outer Gap (horizontal)", "outer-gap-horizontal", 1);
addSpinButtonRow("Outer Gap (vertical)", "outer-gap-vertical", 2);
prefsWidget.add(gapsFrame);
// --- Window Behavior ---
const behaviorTitle = new Gtk.Label({ label: "<b>Window Behavior</b>", use_markup: true, halign: Gtk.Align.START, visible: true });
const behaviorFrame = new Gtk.Frame({ label_widget: behaviorTitle, shadow_type: Gtk.ShadowType.NONE, visible: true });
const behaviorGrid = new Gtk.Grid({ margin: 12, column_spacing: 12, row_spacing: 12, visible: true });
behaviorFrame.add(behaviorGrid);
const label = new Gtk.Label({ label: "Open new windows as", halign: Gtk.Align.START, visible: true });
behaviorGrid.attach(label, 0, 0, 1, 1);
const combo = new Gtk.ComboBoxText({ visible: true, halign: Gtk.Align.END });
combo.append("stack", "Stack Window (Default)");
combo.append("master", "Master Window");
combo.set_active_id(settings.get_string("new-window-behavior"));
combo.connect("changed", () => {
settings.set_string("new-window-behavior", combo.get_active_id());
});
behaviorGrid.attach(combo, 1, 0, 1, 1);
prefsWidget.add(behaviorFrame);
return prefsWidget;
};
// --- MAIN ENTRY POINTS (called by GNOME Shell) ---
function init() {}
function buildPrefsWidget() {
return buildLegacyPrefsWidget();
}
if (Adw) {
var defaultExport = ModernPrefs;
}
-227
View File
@@ -1,227 +0,0 @@
///////////////////////////////////////////////////////////////
// SimpleTiling  LEGACY MENU (GNOME Shell 3.38  44) //
// © 2025domoel  MIT //
/////////////////////////////////////////////////////////////
// ── GLOBAL IMPORTS ────────────────────────────────────────
"use strict";
const { Gtk, GObject, Gio } = imports.gi;
const ExtensionUtils = imports.misc.extensionUtils;
const SCHEMA_NAME = "org.gnome.shell.extensions.simple-tiling.domoel";
// ── DEFINITIONS ────────────────────────────────────────────
const COLUMN_ID = 0;
const COLUMN_DESC = 1;
const COLUMN_KEY = 2;
const COLUMN_MODS = 3;
function init() {}
function buildPrefsWidget() {
const settings = ExtensionUtils.getSettings(SCHEMA_NAME);
const prefsWidget = new Gtk.Box({
orientation: Gtk.Orientation.VERTICAL,
margin_top: 20,
margin_bottom: 20,
margin_start: 20,
margin_end: 20,
spacing: 18,
visible: true,
});
// ── KEYBINDINGS ────────────────────────────────────────────
const keysTitle = new Gtk.Label({
label: "<b>Keybindings</b>",
use_markup: true,
halign: Gtk.Align.START,
visible: true,
});
const keysFrame = new Gtk.Frame({
label_widget: keysTitle,
shadow_type: Gtk.ShadowType.NONE,
visible: true,
});
let keysBox = new Gtk.Box({
orientation: Gtk.Orientation.VERTICAL,
margin: 12,
spacing: 6,
visible: true,
});
keysFrame.add(keysBox);
let store = new Gtk.ListStore();
store.set_column_types([
GObject.TYPE_STRING,
GObject.TYPE_STRING,
GObject.TYPE_INT,
GObject.TYPE_INT,
]);
addKeybinding(store, settings, "swap-master-window", "Swap current window with master");
addKeybinding(store, settings, "swap-up-window", "Swap current window with window above");
addKeybinding(store, settings, "swap-down-window", "Swap current window with window below");
addKeybinding(store, settings, "swap-left-window", "Swap current window with window to the left");
addKeybinding(store, settings, "swap-right-window", "Swap current window with window to the right");
addKeybinding(store, settings, "focus-up", "Focus window above");
addKeybinding(store, settings, "focus-down", "Focus window below");
addKeybinding(store, settings, "focus-left", "Focus window to the left");
addKeybinding(store, settings, "focus-right", "Focus window to the right");
let treeView = new Gtk.TreeView({
model: store,
headers_visible: false,
hexpand: true,
visible: true,
});
keysBox.add(treeView);
let descRenderer = new Gtk.CellRendererText();
let descColumn = new Gtk.TreeViewColumn({ expand: true });
descColumn.pack_start(descRenderer, true);
descColumn.add_attribute(descRenderer, "text", COLUMN_DESC);
treeView.append_column(descColumn);
let accelRenderer = new Gtk.CellRendererAccel({
"accel-mode": Gtk.CellRendererAccelMode.GTK,
editable: true,
});
let accelColumn = new Gtk.TreeViewColumn();
accelColumn.pack_end(accelRenderer, false);
accelColumn.add_attribute(accelRenderer, "accel-key", COLUMN_KEY);
accelColumn.add_attribute(accelRenderer, "accel-mods", COLUMN_MODS);
treeView.append_column(accelColumn);
accelRenderer.connect("accel-edited", (r, path, key, mods) => {
let [ok, iter] = store.get_iter_from_string(path);
if (ok) {
store.set(iter, [COLUMN_KEY, COLUMN_MODS], [key, mods]);
settings.set_strv(store.get_value(iter, COLUMN_ID), [
Gtk.accelerator_name(key, mods),
]);
}
});
accelRenderer.connect("accel-cleared", (r, path) => {
let [ok, iter] = store.get_iter_from_string(path);
if (ok) {
store.set(iter, [COLUMN_KEY, COLUMN_MODS], [0, 0]);
settings.set_strv(store.get_value(iter, COLUMN_ID), []);
}
});
prefsWidget.add(keysFrame);
// ── WINDOW GAPS ────────────────────────────────────────────
const gapsTitle = new Gtk.Label({
label: "<b>Window Gaps</b>",
use_markup: true,
halign: Gtk.Align.START,
visible: true,
});
const gapsFrame = new Gtk.Frame({
label_widget: gapsTitle,
shadow_type: Gtk.ShadowType.NONE,
visible: true,
});
const gapsGrid = new Gtk.Grid({
margin: 12,
column_spacing: 12,
row_spacing: 12,
visible: true,
});
gapsFrame.add(gapsGrid);
addSpinButtonRow(gapsGrid, settings, "Inner Gap", "inner-gap", 0);
addSpinButtonRow(gapsGrid, settings, "Outer Gap (horizontal)", "outer-gap-horizontal", 1);
addSpinButtonRow(gapsGrid, settings, "Outer Gap (vertical)", "outer-gap-vertical", 2);
prefsWidget.add(gapsFrame);
// ── WINDOW BEHAVIOR ────────────────────────────────────────────
const behaviorTitle = new Gtk.Label({
label: "<b>Window Behavior</b>",
use_markup: true,
halign: Gtk.Align.START,
visible: true,
});
const behaviorFrame = new Gtk.Frame({
label_widget: behaviorTitle,
shadow_type: Gtk.ShadowType.NONE,
visible: true,
});
const behaviorGrid = new Gtk.Grid({
margin: 12,
column_spacing: 12,
row_spacing: 12,
visible: true,
});
behaviorFrame.add(behaviorGrid);
addComboBoxRow(
behaviorGrid,
settings,
"Open new windows as",
"new-window-behavior",
0
);
prefsWidget.add(behaviorFrame);
prefsWidget.show_all();
return prefsWidget;
}
function addKeybinding(model, settings, id, desc) {
let [key, mods] = [0, 0];
const strv = settings.get_strv(id);
if (strv && strv[0]) {
[key, mods] = Gtk.accelerator_parse(strv[0]);
}
let iter = model.append();
model.set(
iter,
[COLUMN_ID, COLUMN_DESC, COLUMN_KEY, COLUMN_MODS],
[id, desc, key, mods]
);
}
function addSpinButtonRow(grid, settings, desc, key, pos) {
const label = new Gtk.Label({
label: desc,
halign: Gtk.Align.START,
visible: true,
});
grid.attach(label, 0, pos, 1, 1);
const adj = new Gtk.Adjustment({ lower: 0, upper: 100, step_increment: 1 });
const spin = new Gtk.SpinButton({
adjustment: adj,
climb_rate: 1,
digits: 0,
halign: Gtk.Align.END,
visible: true,
});
settings.bind(key, spin, "value", Gio.SettingsBindFlags.DEFAULT);
grid.attach(spin, 1, pos, 1, 1);
}
function addComboBoxRow(grid, settings, desc, key, pos) {
const label = new Gtk.Label({
label: desc,
halign: Gtk.Align.START,
visible: true,
});
grid.attach(label, 0, pos, 1, 1);
const combo = new Gtk.ComboBoxText({
visible: true,
halign: Gtk.Align.END,
});
combo.append("stack", "Stack Window (Default)");
combo.append("master", "Master Window");
combo.set_active_id(settings.get_string(key));
combo.connect("changed", () => {
settings.set_string(key, combo.get_active_id());
});
grid.attach(combo, 1, pos, 1, 1);
}
-89
View File
@@ -1,89 +0,0 @@
///////////////////////////////////////////////////////////////
// SimpleTiling  MODERN MENU (GNOME Shell 45+) //
// © 2025domoel  MIT //
/////////////////////////////////////////////////////////////
// ── GLOBAL IMPORTS ────────────────────────────────────────
import { ExtensionPreferences } from 'resource:///org/gnome/shell/extensions/extension.js';
import Adw from 'gi://Adw';
import Gio from 'gi://Gio';
import Gtk from 'gi://Gtk';
export default class SimpleTilingPrefs extends ExtensionPreferences {
fillPreferencesWindow(window) {
const settings = this.getSettings();
const page = new Adw.PreferencesPage();
window.add(page);
// ── WINDOW GAPS ────────────────────────────────────────────
const groupGaps = new Adw.PreferencesGroup({
title: 'Window Gaps',
description: 'Adjust spacing between windows and screen edges.'
});
page.add(groupGaps);
const rowInnerGap = new Adw.SpinRow({
title: 'Inner Gap',
subtitle: 'Space between tiled windows (pixels)',
adjustment: new Gtk.Adjustment({ lower: 0, upper: 100, step_increment: 1 }),
});
groupGaps.add(rowInnerGap);
settings.bind('inner-gap', rowInnerGap, 'value', Gio.SettingsBindFlags.DEFAULT);
const rowOuterH = new Adw.SpinRow({
title: 'Outer Gap (horizontal)',
subtitle: 'Left / right screen edges (pixels)',
adjustment: new Gtk.Adjustment({ lower: 0, upper: 100, step_increment: 1 }),
});
groupGaps.add(rowOuterH);
settings.bind('outer-gap-horizontal', rowOuterH, 'value', Gio.SettingsBindFlags.DEFAULT);
const rowOuterV = new Adw.SpinRow({
title: 'Outer Gap (vertical)',
subtitle: 'Top / bottom screen edges (pixels)',
adjustment: new Gtk.Adjustment({ lower: 0, upper: 100, step_increment: 1 }),
});
groupGaps.add(rowOuterV);
settings.bind('outer-gap-vertical', rowOuterV, 'value', Gio.SettingsBindFlags.DEFAULT);
// ── WINDOW BEHAVIOR ────────────────────────────────────────────
const groupBehavior = new Adw.PreferencesGroup({ title: 'Window Behavior' });
page.add(groupBehavior);
const rowNewWindow = new Adw.ComboRow({
title: 'Open new windows as',
subtitle: 'Whether a new window starts as Master or Stack',
model: new Gtk.StringList({
strings: ['Stack Window (Default)', 'Master Window'],
}),
});
groupBehavior.add(rowNewWindow);
rowNewWindow.selected = settings.get_string('new-window-behavior') === 'master' ? 1 : 0;
rowNewWindow.connect('notify::selected', () => {
const newVal = rowNewWindow.selected === 1 ? 'master' : 'stack';
settings.set_string('new-window-behavior', newVal);
});
// ── KEYBINDINGS ────────────────────────────────────────────
const groupKeys = new Adw.PreferencesGroup({ title: 'Keybindings' });
page.add(groupKeys);
const rowKeys = new Adw.ActionRow({
title: 'Configure Shortcuts',
subtitle: 'Adjust all shortcuts in GNOME Keyboard settings.',
});
groupKeys.add(rowKeys);
const btnOpenKeyboard = new Gtk.Button({ label: 'Open Keyboard Settings' });
btnOpenKeyboard.connect('clicked', () => {
const appInfo = Gio.AppInfo.create_from_commandline(
'gnome-control-center keyboard', null, Gio.AppInfoCreateFlags.NONE
);
appInfo.launch([], null);
});
rowKeys.add_suffix(btnOpenKeyboard);
rowKeys.set_activatable_widget(btnOpenKeyboard);
}
}