Forked from gitaarik/restore_floating_clients.lua
Last active
January 23, 2025 02:15
-
-
Save corpix/1487f5a5aa03b9e890f47ad9d01edb8e to your computer and use it in GitHub Desktop.
Awesome WM script that restores floating clients geometry (width / height / position) when switching layouts or unmaximizing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
-- Restore last used floating client geometry (with / height / position) when | |
-- switching layouts and unmaximizing clients. | |
-- | |
-- When unmaximizing a never-before floating client, it will nicely center the | |
-- client in the middle of the screen. | |
-- | |
local capi = { | |
client = client, | |
tag = tag, | |
awesome = awesome | |
} | |
local awful = require("awful") | |
local gears = require("gears") | |
local json = require("cjson") | |
local floating_geometry = {} | |
function floating_geometry:identify(c) | |
return (c.class or c.name) .. "_" .. c.type | |
end | |
-- Save geometries to file | |
function floating_geometry:save_geometries() | |
local file = io.open(self.opts.file, "w") | |
if file then | |
file:write(json.encode(self.class_geometries)) | |
file:close() | |
end | |
end | |
-- Load geometries from file | |
function floating_geometry:load_geometries() | |
local file = io.open(self.opts.file, "r") | |
if file then | |
local content = file:read("*all") | |
file:close() | |
if content then | |
self.class_geometries = json.decode(content) | |
end | |
end | |
end | |
function floating_geometry:flush_geometries() | |
for _, c in ipairs(capi.client.get()) do | |
local identifier = self:identify(c) | |
if identifier and self.floating_client_geometries[c.window] then | |
self.class_geometries[identifier] = self.floating_client_geometries[c.window] | |
end | |
end | |
end | |
-- When the geometry of a client changes, and the layout is floating, then | |
-- store the client's maximized state. And if the client is also not maximized, | |
-- then also store the client's geometry. This will be used to restore the | |
-- client's geometry when they become floating again. | |
function floating_geometry:on_property_geometry(c) | |
self.clients_screens[c.window] = c.screen | |
if (capi.awesome.startup or awful.layout.get(awful.screen.focused()) ~= awful.layout.suit.floating) | |
then | |
-- Don't store anything during startup, or when we're not in a floating | |
-- layout. | |
return | |
end | |
-- Store the unmaximized state | |
self.unmaximized_state[c.window] = not c.maximized | |
if c.maximized then | |
-- Don't store geometry for maximized clients | |
return | |
end | |
if self.floating_client_geometries[c.window] then | |
-- Store the previous client geometry, before saving new one, for | |
-- the unmaximize trigger. | |
self.prev_floating_client_geometries[c.window] = self.floating_client_geometries[c.window] | |
end | |
local geometry = c:geometry() | |
-- Store client geometry | |
self.floating_client_geometries[c.window] = geometry | |
end | |
-- When the layout changes, then if the layout is floating, restore the | |
-- geometry of clients that were not maximized the last time the floating | |
-- layout was used, if there is any previously stored geometry to restore. | |
function floating_geometry:on_property_layout(t) | |
if t.layout == awful.layout.suit.floating | |
then | |
for k, c in ipairs(t:clients()) | |
do | |
if (self.floating_client_geometries[c.window] and self.unmaximized_state[c.window]) | |
then | |
-- Restore client geometry | |
c:geometry(self.floating_client_geometries[c.window]) | |
else | |
-- If the client was maximized in the last used floating layout, | |
-- or if there's no geometry to restore, just set the client to | |
-- maximized. | |
c.maximized = true | |
end | |
end | |
else | |
-- If the layout is not floating, then remove the maximized state from | |
-- all clients, otherwise those layouts don't work properly. | |
for k, c in ipairs(t:clients()) do c.maximized = false end | |
end | |
end | |
-- When changing a maximized client to unmaximized in a floating layout, then | |
-- restore the floating client geometry to what it was in the previous floating | |
-- state. | |
function floating_geometry:on_property_maximized(c) | |
if (c.maximized or awful.layout.get(awful.screen.focused()) ~= awful.layout.suit.floating) | |
then | |
-- If the client is maximized, or the layout is not floating, don't | |
-- restore anything. | |
return | |
end | |
if self.prev_floating_client_geometries[c.window] then | |
-- Restore client geometry | |
c:geometry(self.prev_floating_client_geometries[c.window]) | |
else | |
-- If there is no previous geometry state, then center the client nicely in the | |
-- middle of the screen, so that it's not off bounds or in a corner or | |
-- extremely small or anything. Then it's easy to move / resize the floating | |
-- client to your wish. | |
local g = c.screen.geometry | |
c:geometry({ | |
x = g.width / 6, | |
y = g.height / 6, | |
width = g.width / 1.5, | |
height = g.height / 1.5 | |
}) | |
-- Restore the client screen, because this seems to get lost due to the | |
-- `c:geometry()` call. This issue only occurs when you have multiple | |
-- screens in use. | |
c.screen = self.clients_screens[c.window] | |
end | |
end | |
-- When a new client appears, reset the geometry states. Because this signal | |
-- comes after the geometry signal, and the geometry signal in this case stores | |
-- incorrect geometries for a new client. So since this signal comes after the | |
-- geometry signal, we can reset it, so a new client starts with a blank slate. | |
function floating_geometry:on_manage (c) | |
if (capi.awesome.startup or awful.layout.get(awful.screen.focused()) ~= awful.layout.suit.floating) | |
then | |
return | |
end | |
self.floating_client_geometries[c.window] = nil | |
self.prev_floating_client_geometries[c.window] = nil | |
local identifier = self:identify(c) | |
if self.opts.exclude and self.opts.exclude(c) | |
then | |
return | |
end | |
if identifier and self.class_geometries[identifier] then | |
c:geometry(self.class_geometries[identifier]) | |
end | |
end | |
function floating_geometry:on_startup() | |
self:load_geometries() | |
end | |
function floating_geometry:on_exit() | |
self:save_geometries() | |
end | |
function floating_geometry.new(opts) | |
opts = opts or {} | |
if not opts.file | |
then | |
opts.file = os.getenv("HOME") .. "/.config/awesome/floating-geometry.json" | |
end | |
-- | |
local obj = gears.object { class = floating_geometry } | |
obj.opts = opts | |
-- The following variables contain different states of clients, with the | |
-- `client.window` in the key and the relevant state in the value. | |
-- Store the floating clients geometries (with / height / position), so we can | |
-- restore the clients when the layout changes. | |
obj.floating_client_geometries = {} | |
-- Store geometries by window class/name for persistence across sessions | |
obj.class_geometries = {} | |
-- Store the previous floating clients geometries. We need this to restore from | |
-- maximized, because the maximized signal happens after the geometry signal. | |
-- So the state we want to revert to gets overridden before we can restore it. | |
-- Therefore we also store the previous floating client geometries. | |
obj.prev_floating_client_geometries = {} | |
-- Store whether a client was NOT maximized in the last used floating layout. | |
-- We use this to determin whether a client should become maximized or floating | |
-- when switching to floating layout. | |
obj.unmaximized_state = {} | |
-- Store on which screen a client is. This is needed because when setting the | |
-- geometry of a client, the client seems to get moved to the first screen | |
-- regardless of which screen it was on. | |
obj.clients_screens = {} | |
obj.flush_geometries_timer = gears.timer { | |
timeout = 5, | |
call_now = false, | |
autostart = true, | |
callback = function() obj:flush_geometries() end | |
} | |
obj.save_geometries_timer = gears.timer { | |
timeout = 15, | |
call_now = false, | |
autostart = true, | |
callback = function() obj:save_geometries() end | |
} | |
capi.client.connect_signal("manage", function(c) obj:on_manage(c) end) | |
capi.client.connect_signal("property::maximized", function(c) obj:on_property_maximized(c) end) | |
capi.client.connect_signal("property::geometry", function(c) obj:on_property_geometry(c) end) | |
capi.tag.connect_signal("property::layout", function(t) obj:on_property_layout(t) end) | |
capi.awesome.connect_signal("startup", function() obj:on_startup() end) | |
capi.awesome.connect_signal("exit", function() obj:on_exit() end) | |
end | |
function floating_geometry.init(opts) | |
instance = floating_geometry.new(opts) | |
return instance | |
end | |
return floating_geometry |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment