2021-04-08 20:17:10 +02:00
|
|
|
local awful = require("awful")
|
2021-08-04 11:56:25 +02:00
|
|
|
local gears = require("gears")
|
2021-04-08 20:17:10 +02:00
|
|
|
|
|
|
|
local helpers = require(tostring(...):match(".*bling") .. ".helpers")
|
|
|
|
|
2021-08-04 11:56:25 +02:00
|
|
|
local Scratchpad = { mt = {} }
|
2021-04-08 20:17:10 +02:00
|
|
|
|
|
|
|
--- Creates a new scratchpad object based on the argument
|
|
|
|
--
|
2021-08-04 11:56:25 +02:00
|
|
|
-- @param args A table of possible arguments
|
2021-04-08 20:17:10 +02:00
|
|
|
-- @return The new scratchpad object
|
2021-08-04 11:56:25 +02:00
|
|
|
function Scratchpad:new(args)
|
|
|
|
args = args or {}
|
|
|
|
args.awestore = args.awestore or {}
|
|
|
|
args.in_anim = false
|
|
|
|
local ret = gears.object {}
|
|
|
|
gears.table.crush(ret, Scratchpad)
|
|
|
|
gears.table.crush(ret, args)
|
|
|
|
return ret
|
2021-04-08 20:17:10 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--- Find all clients that satisfy the the rule
|
|
|
|
--
|
|
|
|
-- @return A list of all clients that satisfy the rule
|
2021-04-09 02:44:03 +02:00
|
|
|
function Scratchpad:find() return helpers.client.find(self.rule) end
|
|
|
|
|
2021-04-08 20:17:10 +02:00
|
|
|
--- Applies the objects scratchpad properties to a given client
|
|
|
|
--
|
2021-06-18 14:49:45 +02:00
|
|
|
-- @param c A client to which to apply the properties
|
2021-04-08 20:17:10 +02:00
|
|
|
function Scratchpad:apply(c)
|
|
|
|
if not c or not c.valid then return end
|
|
|
|
c.floating = self.floating
|
|
|
|
c.sticky = self.sticky
|
2021-04-09 02:44:03 +02:00
|
|
|
c:geometry({
|
|
|
|
x = self.geometry.x + awful.screen.focused().geometry.x,
|
|
|
|
y = self.geometry.y + awful.screen.focused().geometry.y,
|
|
|
|
width = self.geometry.width,
|
|
|
|
height = self.geometry.height
|
|
|
|
})
|
2021-04-08 20:17:10 +02:00
|
|
|
if self.autoclose then
|
2021-08-04 11:56:25 +02:00
|
|
|
c:connect_signal("unfocus", function(c1)
|
|
|
|
c1.sticky = false -- client won't turn off if sticky
|
|
|
|
helpers.client.turn_off(c1)
|
2021-04-08 20:17:10 +02:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Turns the scratchpad on
|
|
|
|
function Scratchpad:turn_on()
|
|
|
|
local matches = self:find()
|
2021-08-04 11:56:25 +02:00
|
|
|
local c = matches[1]
|
|
|
|
if c and not self.in_anim and c.first_tag and c.first_tag.selected then
|
|
|
|
c:raise()
|
|
|
|
client.focus = c
|
|
|
|
return
|
|
|
|
end
|
|
|
|
if c and not self.in_anim then
|
2021-04-08 20:17:10 +02:00
|
|
|
-- if a client was found, turn it on
|
|
|
|
if self.reapply then self:apply(c) end
|
|
|
|
-- c.sticky was set to false in turn_off so it has to be reapplied anyway
|
2021-04-09 02:44:03 +02:00
|
|
|
c.sticky = self.sticky
|
2021-04-16 21:46:42 +02:00
|
|
|
local new_y = c.y
|
|
|
|
local new_x = c.x
|
|
|
|
|
|
|
|
-- Get the tweens
|
|
|
|
local anim_x = self.awestore.x
|
|
|
|
local anim_y = self.awestore.y
|
|
|
|
|
|
|
|
-- Subscribe
|
|
|
|
if anim_x then
|
|
|
|
anim_x:subscribe(function(x)
|
|
|
|
if c and c.valid then c.x = x end
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = true
|
2021-04-16 21:46:42 +02:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
if anim_y then
|
|
|
|
anim_y:subscribe(function(y)
|
|
|
|
if c and c.valid then c.y = y end
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = true
|
2021-04-16 21:46:42 +02:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
2021-04-08 20:17:10 +02:00
|
|
|
helpers.client.turn_on(c)
|
2021-08-04 11:56:25 +02:00
|
|
|
self:emit_signal("turn_on", c)
|
2021-04-16 21:46:42 +02:00
|
|
|
|
|
|
|
-- Unsubscribe
|
|
|
|
if anim_x then
|
|
|
|
anim_x:set(new_x)
|
|
|
|
local unsub_x
|
|
|
|
unsub_x = anim_x.ended:subscribe(
|
|
|
|
function()
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = false
|
2021-04-16 21:46:42 +02:00
|
|
|
unsub_x()
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
if anim_y then
|
|
|
|
anim_y:set(new_y)
|
|
|
|
local unsub_y
|
|
|
|
unsub_y = anim_y.ended:subscribe(
|
|
|
|
function()
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = false
|
2021-04-16 21:46:42 +02:00
|
|
|
unsub_y()
|
|
|
|
end)
|
|
|
|
end
|
2021-04-08 20:17:10 +02:00
|
|
|
return
|
2021-08-04 11:56:25 +02:00
|
|
|
end
|
|
|
|
if not c then
|
2021-04-08 20:17:10 +02:00
|
|
|
-- if no client was found, spawn one, find the corresponding window,
|
2021-04-16 21:46:42 +02:00
|
|
|
-- apply the properties only once (until the next closing)
|
2021-04-08 20:17:10 +02:00
|
|
|
local pid = awful.spawn.with_shell(self.command)
|
2021-08-04 11:56:25 +02:00
|
|
|
self:emit_signal("spawn")
|
|
|
|
local function inital_apply(c1)
|
|
|
|
if helpers.client.is_child_of(c1, pid) then
|
|
|
|
self:apply(c1)
|
|
|
|
self:emit_signal("inital_apply", c1)
|
2021-04-08 20:17:10 +02:00
|
|
|
client.disconnect_signal("manage", inital_apply)
|
2021-08-04 11:56:25 +02:00
|
|
|
end
|
2021-04-08 20:17:10 +02:00
|
|
|
end
|
|
|
|
client.connect_signal("manage", inital_apply)
|
|
|
|
return
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Turns the scratchpad off
|
|
|
|
function Scratchpad:turn_off()
|
|
|
|
local matches = self:find()
|
|
|
|
local c = matches[1]
|
2021-08-04 11:56:25 +02:00
|
|
|
if c and not self.in_anim then
|
2021-04-08 20:17:10 +02:00
|
|
|
c.sticky = false
|
2021-04-16 21:46:42 +02:00
|
|
|
|
|
|
|
-- Get the tweens
|
|
|
|
local anim_x = self.awestore.x
|
|
|
|
local anim_y = self.awestore.y
|
|
|
|
|
|
|
|
-- Subscribe
|
|
|
|
if anim_x then
|
|
|
|
anim_x:subscribe(function(x)
|
|
|
|
if c and c.valid then c.x = x end
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = true
|
2021-04-16 21:46:42 +02:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
if anim_y then
|
|
|
|
anim_y:subscribe(function(y)
|
|
|
|
if c and c.valid then c.y = y end
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = true
|
2021-04-16 21:46:42 +02:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Unsubscribe
|
|
|
|
if anim_x then
|
|
|
|
anim_x:set(anim_x:initial())
|
|
|
|
local unsub
|
|
|
|
unsub = anim_x.ended:subscribe(
|
|
|
|
function()
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = false
|
2021-04-16 21:46:42 +02:00
|
|
|
helpers.client.turn_off(c)
|
2021-08-04 11:56:25 +02:00
|
|
|
self:emit_signal("turn_off", c)
|
2021-04-16 21:46:42 +02:00
|
|
|
unsub()
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
if anim_y then
|
|
|
|
anim_y:set(anim_y:initial())
|
|
|
|
|
|
|
|
local unsub
|
|
|
|
unsub = anim_y.ended:subscribe(
|
|
|
|
function()
|
2021-08-04 11:56:25 +02:00
|
|
|
self.in_anim = false
|
2021-04-16 21:46:42 +02:00
|
|
|
helpers.client.turn_off(c)
|
2021-08-04 11:56:25 +02:00
|
|
|
self:emit_signal("turn_off", c)
|
2021-04-16 21:46:42 +02:00
|
|
|
unsub()
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
2021-08-04 11:56:25 +02:00
|
|
|
if not anim_x and not anim_y then
|
|
|
|
helpers.client.turn_off(c)
|
|
|
|
self:emit_signal("turn_off", c)
|
|
|
|
end
|
2021-04-08 20:17:10 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Turns the scratchpad off if it is focused otherwise it raises the scratchpad
|
|
|
|
function Scratchpad:toggle()
|
|
|
|
local is_turn_off = false
|
2021-06-18 14:49:45 +02:00
|
|
|
local matches = self:find()
|
2021-04-08 20:17:10 +02:00
|
|
|
if self.dont_focus_before_close then
|
2021-06-18 14:49:45 +02:00
|
|
|
if matches[1] then
|
|
|
|
local current_tag = matches[1].screen.selected_tag
|
|
|
|
for k, tag in pairs(matches[1]:tags()) do
|
|
|
|
if tag == current_tag then
|
|
|
|
is_turn_off = true
|
|
|
|
break
|
|
|
|
else
|
|
|
|
is_turn_off = false
|
|
|
|
end
|
|
|
|
end
|
2021-04-08 20:17:10 +02:00
|
|
|
end
|
|
|
|
else
|
2021-04-09 02:44:03 +02:00
|
|
|
is_turn_off = client.focus and
|
|
|
|
awful.rules.match(client.focus, self.rule)
|
2021-04-08 20:17:10 +02:00
|
|
|
end
|
2021-04-09 02:44:03 +02:00
|
|
|
|
2021-04-08 20:17:10 +02:00
|
|
|
if is_turn_off then
|
2021-06-18 14:49:45 +02:00
|
|
|
if self.disable_floating_on_close and matches[1].floating == true then
|
|
|
|
matches[1].floating = false
|
|
|
|
end
|
2021-04-08 20:17:10 +02:00
|
|
|
self:turn_off()
|
|
|
|
else
|
|
|
|
self:turn_on()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-08-04 11:56:25 +02:00
|
|
|
--- Make the module callable without putting a `:new` at the end of it
|
|
|
|
--
|
|
|
|
-- @param args A table of possible arguments
|
|
|
|
-- @return The new scratchpad object
|
|
|
|
function Scratchpad.mt:__call(...)
|
|
|
|
return Scratchpad:new(...)
|
|
|
|
end
|
|
|
|
|
|
|
|
return setmetatable(Scratchpad, Scratchpad.mt)
|
|
|
|
|