166 lines
5.6 KiB
Lua
166 lines
5.6 KiB
Lua
---------------------------------------------------------------------------
|
|
--- Handle client shapes.
|
|
--
|
|
-- @author Uli Schlachter <psychon@znc.in>
|
|
-- @copyright 2014 Uli Schlachter
|
|
-- @submodule client
|
|
---------------------------------------------------------------------------
|
|
|
|
-- Grab environment we need
|
|
local surface = require("gears.surface")
|
|
local cairo = require("lgi").cairo
|
|
local capi =
|
|
{
|
|
client = client,
|
|
}
|
|
|
|
local shape = {}
|
|
shape.update = {}
|
|
|
|
--- Get one of a client's shapes and transform it to include window decorations.
|
|
-- @function awful.client.shape.get_transformed
|
|
-- @client c The client whose shape should be retrieved
|
|
-- @tparam string shape_name Either "bounding" or "clip"
|
|
function shape.get_transformed(c, shape_name)
|
|
local border = shape_name == "bounding" and c.border_width or 0
|
|
local shape_img = surface.load_silently(c["client_shape_" .. shape_name], false)
|
|
local _shape = c._shape
|
|
if not (shape_img or _shape) then return end
|
|
|
|
-- Get information about various sizes on the client
|
|
local geom = c:geometry()
|
|
local _, t = c:titlebar_top()
|
|
local _, b = c:titlebar_bottom()
|
|
local _, l = c:titlebar_left()
|
|
local _, r = c:titlebar_right()
|
|
|
|
-- Figure out the size of the shape that we need
|
|
local img_width = geom.width + 2*border
|
|
local img_height = geom.height + 2*border
|
|
local result = cairo.ImageSurface(cairo.Format.A1, img_width, img_height)
|
|
local cr = cairo.Context(result)
|
|
|
|
-- Fill everything (this paints the titlebars and border).
|
|
-- The `cr:paint()` below will have painted the whole surface, so
|
|
-- everything inside the client is currently meant to be visible
|
|
cr:paint()
|
|
|
|
if shape_img then
|
|
-- Draw the client's shape in the middle
|
|
cr:set_operator(cairo.Operator.SOURCE)
|
|
cr:set_source_surface(shape_img, border + l, border + t)
|
|
cr:rectangle(border + l, border + t, geom.width - l - r, geom.height - t - b)
|
|
cr:fill()
|
|
end
|
|
|
|
if _shape then
|
|
-- Draw the shape to an intermediate surface
|
|
cr:push_group()
|
|
-- Intersect what is drawn so far with the shape set by Lua.
|
|
if shape_name == "clip" then
|
|
-- Correct for the border offset
|
|
cr:translate(-c.border_width, -c.border_width)
|
|
end
|
|
-- Always call the shape with the size of the bounding shape
|
|
_shape(cr, geom.width + 2*c.border_width, geom.height + 2*c.border_width)
|
|
-- Now fill the "selected" part
|
|
cr:set_operator(cairo.Operator.SOURCE)
|
|
cr:set_source_rgba(1, 1, 1, 1)
|
|
cr:fill_preserve()
|
|
if shape_name == "clip" then
|
|
-- Remove an area of size c.border_width again (We use 2*bw since
|
|
-- half of that is on the outside)
|
|
cr:set_source_rgba(0, 0, 0, 0)
|
|
cr:set_line_width(2*c.border_width)
|
|
cr:stroke()
|
|
end
|
|
-- Combine the result with what we already have
|
|
cr:pop_group_to_source()
|
|
cr:set_operator(cairo.Operator.IN)
|
|
cr:paint()
|
|
end
|
|
|
|
return result
|
|
end
|
|
|
|
--- Update all of a client's shapes from the shapes the client set itself.
|
|
-- @function awful.client.shape.update.all
|
|
-- @client c The client to act on
|
|
function shape.update.all(c)
|
|
local shape_fun = c._shape
|
|
|
|
if not shape_fun then
|
|
c.shape_bounding = nil
|
|
c.shape_clip = nil
|
|
shape.update.bounding(c)
|
|
shape.update.clip(c)
|
|
return
|
|
end
|
|
|
|
local geo = c:geometry()
|
|
local bw = c.border_width
|
|
|
|
-- First handle the bounding shape (things including the border)
|
|
local img = cairo.ImageSurface(cairo.Format.A1, geo.width + 2*bw, geo.height + 2*bw)
|
|
local cr = cairo.Context(img)
|
|
|
|
-- We just draw the shape in its full size
|
|
shape_fun(cr, geo.width + 2*bw, geo.height + 2*bw)
|
|
cr:set_operator(cairo.Operator.SOURCE)
|
|
cr:fill()
|
|
c.shape_bounding = img._native
|
|
img:finish()
|
|
|
|
-- Now handle the clip shape (things excluding the border)
|
|
img = cairo.ImageSurface(cairo.Format.A1, geo.width, geo.height)
|
|
cr = cairo.Context(img)
|
|
|
|
-- We give the shape the same arguments as for the bounding shape and draw
|
|
-- it in its full size (the translate is to compensate for the smaller
|
|
-- surface)
|
|
cr:translate(-bw, -bw)
|
|
shape_fun(cr, geo.width + 2*bw, geo.height + 2*bw)
|
|
cr:set_operator(cairo.Operator.SOURCE)
|
|
cr:fill_preserve()
|
|
-- Now we remove an area of width 'bw' again around the shape (We use 2*bw
|
|
-- since half of that is on the outside and only half on the inside)
|
|
cr:set_source_rgba(0, 0, 0, 0)
|
|
cr:set_line_width(2*bw)
|
|
cr:stroke()
|
|
c.shape_clip = img._native
|
|
img:finish()
|
|
end
|
|
|
|
--- Update a client's bounding shape from the shape the client set itself.
|
|
-- @function awful.client.shape.update.bounding
|
|
-- @client c The client to act on
|
|
function shape.update.bounding(c)
|
|
local res = shape.get_transformed(c, "bounding")
|
|
c.shape_bounding = res and res._native
|
|
-- Free memory
|
|
if res then
|
|
res:finish()
|
|
end
|
|
end
|
|
|
|
--- Update a client's clip shape from the shape the client set itself.
|
|
-- @function awful.client.shape.update.clip
|
|
-- @client c The client to act on
|
|
function shape.update.clip(c)
|
|
local res = shape.get_transformed(c, "clip")
|
|
c.shape_clip = res and res._native
|
|
-- Free memory
|
|
if res then
|
|
res:finish()
|
|
end
|
|
end
|
|
|
|
capi.client.connect_signal("property::shape_client_bounding", shape.update.bounding)
|
|
capi.client.connect_signal("property::shape_client_clip", shape.update.clip)
|
|
capi.client.connect_signal("property::size", shape.update.all)
|
|
capi.client.connect_signal("property::border_width", shape.update.all)
|
|
|
|
return shape
|
|
|
|
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|