387 lines
9.7 KiB
Lua
387 lines
9.7 KiB
Lua
local gears_obj = require("gears.object")
|
|
local grect = require("gears.geometry").rectangle
|
|
|
|
local clients = {}
|
|
|
|
local client, meta = awesome._shim_fake_class()
|
|
|
|
rawset(client, "_autotags", true)
|
|
|
|
-- Keep an history of the geometry for validation and images
|
|
local function push_geometry(c)
|
|
table.insert(c._old_geo, c:geometry())
|
|
end
|
|
|
|
-- Undo the last move, but keep it in history
|
|
-- local function pop_geometry(c)
|
|
-- CURRENTLY UNUSED
|
|
-- end
|
|
|
|
local function titlebar_meta(c)
|
|
for _, position in ipairs {"top", "bottom", "left", "right" } do
|
|
c["titlebar_"..position] = function(size) --luacheck: no unused
|
|
return drawin{}
|
|
end
|
|
end
|
|
end
|
|
|
|
local properties = {}
|
|
|
|
-- Emit the request::geometry signal to make immobilized and awful.ewmh work.
|
|
for _, prop in ipairs {
|
|
"maximized", "maximized_horizontal", "maximized_vertical", "fullscreen" } do
|
|
properties["get_"..prop] = function(self)
|
|
return self._private[prop] or false
|
|
end
|
|
|
|
properties["set_"..prop] = function(self, value)
|
|
self._private[prop] = value or false
|
|
|
|
if value then
|
|
self:emit_signal("request::geometry", prop, nil)
|
|
end
|
|
|
|
self:emit_signal("property::"..prop, value)
|
|
end
|
|
end
|
|
|
|
function properties.get_screen(self)
|
|
return self._private.screen or screen[1]
|
|
end
|
|
|
|
function properties.set_screen(self, s)
|
|
s = screen[s]
|
|
self._private.screen = s
|
|
|
|
if self.x < s.geometry.x or self.x > s.geometry.x+s.geometry.width then
|
|
self:geometry { x = s.geometry.x }
|
|
end
|
|
|
|
if self.y < s.geometry.y or self.y > s.geometry.y+s.geometry.height then
|
|
self:geometry { y = s.geometry.y }
|
|
end
|
|
|
|
self:emit_signal("property::screen")
|
|
end
|
|
|
|
function properties:get_first_tag()
|
|
return self:tags()[1]
|
|
end
|
|
|
|
-- Create fake clients to move around
|
|
function client.gen_fake(args)
|
|
local ret = gears_obj()
|
|
awesome._forward_class(ret, client)
|
|
|
|
ret._private = {}
|
|
|
|
-- Make sure the layer properties are not `nil`
|
|
local defaults = {
|
|
ontop = false,
|
|
below = false,
|
|
above = false,
|
|
sticky = false,
|
|
urgent = false,
|
|
focusable = true,
|
|
hidden = args.hidden or false,
|
|
minimized = args.minimized or false,
|
|
type = args.type or "normal",
|
|
icon_sizes = args.icon_sizes or {{16,16}},
|
|
name = args.name or "Example Client",
|
|
size_hints = args.size_hints or {},
|
|
_border_width = args.border_width or 1,
|
|
|
|
-- This is a hack because there's a `:is_transient_for(c2)` method
|
|
-- and a `transient_for` property. It will cause a stack overflow
|
|
-- since the auto-alias will kick in if the property is allowed to
|
|
-- be `nil`.
|
|
transient_for = args.transient_for or false,
|
|
}
|
|
|
|
ret.valid = true
|
|
ret._tags = args and args.tags or nil
|
|
ret._private._struts = { top = 0, right = 0, left = 0, bottom = 0 }
|
|
|
|
--TODO v5: remove this. This was a private API and thus doesn't need to be
|
|
-- officially deprecated.
|
|
ret.data = ret._private
|
|
|
|
-- Apply all properties
|
|
for k,v in pairs(args or {}) do
|
|
ret._private[k] = v
|
|
end
|
|
|
|
-- Tests should always set a geometry, but just in case
|
|
for _, v in ipairs{"x","y","width","height"} do
|
|
rawset(ret, v, math.floor(ret[v] or ret._private[v] or 1))
|
|
assert((not args[v]) or ret[v] == args[v])
|
|
end
|
|
|
|
-- Emulate capi.client.geometry
|
|
function ret:geometry(new)
|
|
local new_full = new and {
|
|
x = math.floor(new.x or ret.x),
|
|
y = math.floor(new.y or ret.y),
|
|
width = math.floor(new.width or ret.width),
|
|
height = math.floor(new.height or ret.height),
|
|
} or nil
|
|
|
|
if new and not grect.are_equal(ret, new_full) then
|
|
for k,v in pairs(new) do
|
|
rawset(ret, k, v)
|
|
ret:emit_signal("property::"..k, v)
|
|
end
|
|
ret:emit_signal("property::geometry", ret:geometry())
|
|
push_geometry(ret)
|
|
end
|
|
|
|
return {
|
|
x = math.floor(ret.x),
|
|
y = math.floor(ret.y),
|
|
width = math.floor(ret.width),
|
|
height = math.floor(ret.height),
|
|
label = ret._label,
|
|
}
|
|
end
|
|
|
|
function ret:isvisible()
|
|
if ret.minimized or ret.hidden then return false end
|
|
|
|
local vis = false
|
|
|
|
for _, tag in ipairs(ret:tags()) do
|
|
vis = vis or tag.selected
|
|
end
|
|
|
|
return vis
|
|
end
|
|
|
|
-- Used for screenshots
|
|
function ret:set_label(text)
|
|
ret._old_geo[#ret._old_geo]._label = text
|
|
end
|
|
|
|
-- Used for screenshots, hide the current client position
|
|
function ret:_hide()
|
|
ret._old_geo[#ret._old_geo]._hide = true
|
|
end
|
|
|
|
function ret:_hide_all()
|
|
for _, geo in ipairs(ret._old_geo) do
|
|
geo._hide = true
|
|
end
|
|
end
|
|
|
|
function ret:get_xproperty()
|
|
return nil
|
|
end
|
|
|
|
function ret:set_xproperty(prop, value)
|
|
ret[prop] = value
|
|
end
|
|
|
|
function ret:get_icon(_)
|
|
return require("beautiful").awesome_icon
|
|
end
|
|
|
|
function ret:raise()
|
|
--TODO
|
|
end
|
|
|
|
function ret:lower()
|
|
--TODO
|
|
end
|
|
|
|
function ret:apply_size_hints(w, h)
|
|
return w or ret.width, h or ret.height
|
|
end
|
|
|
|
function ret:kill()
|
|
local old_tags = ret:tags() or {}
|
|
|
|
for k, c in ipairs(clients) do
|
|
if c == ret then
|
|
ret:emit_signal("unmanaged", c)
|
|
ret.valid = false
|
|
table.remove(clients, k)
|
|
break
|
|
end
|
|
end
|
|
|
|
ret._tags = {}
|
|
|
|
for _, t in ipairs(old_tags) do
|
|
ret:emit_signal("untagged", t)
|
|
t:emit_signal("untagged", ret)
|
|
t:emit_signal("property::tags")
|
|
end
|
|
|
|
assert(not ret.valid)
|
|
end
|
|
|
|
function ret:swap(other)
|
|
local idx1, idx2 = nil, nil
|
|
for k, c in ipairs(clients) do
|
|
if c == ret then
|
|
idx1 = k
|
|
elseif c == other then
|
|
idx2 = k
|
|
end
|
|
end
|
|
|
|
if not (idx1 and idx2) then return end
|
|
|
|
clients[idx1], clients[idx2] = other, ret
|
|
ret:emit_signal("swapped", other, true)
|
|
other:emit_signal("swapped", ret, false)
|
|
client.emit_signal("list")
|
|
end
|
|
|
|
titlebar_meta(ret)
|
|
|
|
function ret:tags(new) --FIXME
|
|
if new then
|
|
ret._tags = new
|
|
ret:emit_signal("property::tags")
|
|
end
|
|
|
|
if ret._tags then
|
|
return ret._tags
|
|
end
|
|
|
|
if client._autotags then
|
|
for _, t in ipairs(root._tags) do
|
|
if t.screen == ret.screen then
|
|
return {t}
|
|
end
|
|
end
|
|
end
|
|
|
|
return {}
|
|
end
|
|
|
|
function ret:struts(new)
|
|
for k, v in pairs(new or {}) do
|
|
ret._private._struts[k] = v
|
|
end
|
|
|
|
return ret._private._struts
|
|
end
|
|
|
|
-- Set a dummy one for now since set_screen will corrupt it.
|
|
ret._old_geo = {}
|
|
|
|
-- Set the screen attributes.
|
|
local s = args.screen
|
|
|
|
-- Pick the screen from the geometry.
|
|
if not s then
|
|
for s2 in screen do
|
|
local geo = s2.geometry
|
|
if geo.x >= ret.x and geo.y >= ret.y and ret.x < geo.x+geo.width
|
|
and ret.y < geo.y+geo.height then
|
|
s = s2
|
|
end
|
|
end
|
|
end
|
|
|
|
-- This will happen if the screen coords are not {0,0} and the client had
|
|
-- an unspecified position.
|
|
s = s or screen[1]
|
|
|
|
properties.set_screen(ret, s)
|
|
|
|
-- Set the geometry *again* because the screen possibly overwrote it.
|
|
for _, v in ipairs{"x","y","width","height"} do
|
|
ret[v] = args[v] or ret[v]
|
|
end
|
|
|
|
-- Record the geometry
|
|
ret._old_geo = {}
|
|
push_geometry(ret)
|
|
ret._old_geo[1]._hide = args.hide_first
|
|
|
|
-- Good enough for the geometry and border
|
|
ret.drawin = ret
|
|
ret.drawable = ret
|
|
|
|
-- Declare the deprecated buttons and keys methods.
|
|
function ret:_keys(new)
|
|
if new then
|
|
ret._private.keys = new
|
|
end
|
|
|
|
return ret._private.keys or {}
|
|
end
|
|
|
|
function ret:_buttons(new)
|
|
if new then
|
|
ret._private.buttons = new
|
|
end
|
|
|
|
return ret._private.buttons or {}
|
|
end
|
|
|
|
-- Add to the client list
|
|
table.insert(clients, ret)
|
|
|
|
client.focus = ret
|
|
|
|
setmetatable(ret, {
|
|
__index = function(self, key)
|
|
if properties["get_"..key] then
|
|
return properties["get_"..key](self)
|
|
elseif defaults[key] ~= nil then
|
|
return defaults[key]
|
|
elseif self._private[key] ~= nil then
|
|
return self._private[key]
|
|
end
|
|
|
|
return meta.__index(self, key)
|
|
end,
|
|
__newindex = function(self, key, value)
|
|
if properties["set_"..key] then
|
|
return properties["set_"..key](self, value)
|
|
end
|
|
|
|
if defaults[key] ~= nil then
|
|
defaults[key] = value
|
|
else
|
|
meta.__newindex(self, key, value)
|
|
end
|
|
|
|
ret:emit_signal("property::"..key, value)
|
|
--client.emit_signal("property::"..key, ret, value)
|
|
end
|
|
})
|
|
|
|
client.emit_signal("request::manage", ret)
|
|
|
|
--TODO v6 remove this.
|
|
client.emit_signal("manage", ret)
|
|
|
|
assert(not args.screen or (args.screen == ret.screen))
|
|
|
|
return ret
|
|
end
|
|
|
|
function client.get(s)
|
|
if not s then return clients end
|
|
|
|
local s2 = screen[s]
|
|
|
|
local ret = {}
|
|
|
|
for _,c in ipairs(clients) do
|
|
if c.screen == s2 then
|
|
table.insert(ret, c)
|
|
end
|
|
end
|
|
|
|
return ret
|
|
end
|
|
|
|
return client
|
|
|
|
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|