client: Allow clients to have shapes

Also fixes awful.client.shape docs

Closes #1507
This commit is contained in:
copycat-killer 2017-02-02 21:12:47 +01:00 committed by Emmanuel Lepage Vallee
parent cbd22eea50
commit 1a5f6b7ad2
3 changed files with 103 additions and 19 deletions

View File

@ -9,6 +9,7 @@
-- Grab environment we need -- Grab environment we need
local util = require("awful.util") local util = require("awful.util")
local spawn = require("awful.spawn") local spawn = require("awful.spawn")
local set_shape = require("awful.client.shape").update.all
local object = require("gears.object") local object = require("gears.object")
local grect = require("gears.geometry").rectangle local grect = require("gears.geometry").rectangle
local pairs = pairs local pairs = pairs
@ -1155,6 +1156,15 @@ function client.object.is_transient_for(self, c2)
return nil return nil
end end
--- Set the client shape.
-- @property shape
-- @tparam gears.shape A gears.shape compatible function.
-- @see gears.shape
function client.object.set_shape(self, shape)
client.property.set(self, "_shape", shape)
set_shape(self)
end
-- Register standards signals -- Register standards signals
--- The last geometry when client was floating. --- The last geometry when client was floating.

View File

@ -18,13 +18,14 @@ local shape = {}
shape.update = {} shape.update = {}
--- Get one of a client's shapes and transform it to include window decorations. --- Get one of a client's shapes and transform it to include window decorations.
-- @function awful.shape.get_transformed -- @function awful.client.shape.get_transformed
-- @client c The client whose shape should be retrieved -- @client c The client whose shape should be retrieved
-- @tparam string shape_name Either "bounding" or "clip" -- @tparam string shape_name Either "bounding" or "clip"
function shape.get_transformed(c, shape_name) function shape.get_transformed(c, shape_name)
local border = shape_name == "bounding" and c.border_width or 0 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_img = surface.load_silently(c["client_shape_" .. shape_name], false)
if not shape_img then return end local _shape = c._shape
if not (shape_img or _shape) then return end
-- Get information about various sizes on the client -- Get information about various sizes on the client
local geom = c:geometry() local geom = c:geometry()
@ -39,20 +40,99 @@ function shape.get_transformed(c, shape_name)
local result = cairo.ImageSurface(cairo.Format.A1, img_width, img_height) local result = cairo.ImageSurface(cairo.Format.A1, img_width, img_height)
local cr = cairo.Context(result) local cr = cairo.Context(result)
-- Fill everything (this paints the titlebars and border) -- 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() cr:paint()
if shape_img then
-- Draw the client's shape in the middle -- Draw the client's shape in the middle
cr:set_operator(cairo.Operator.SOURCE) cr:set_operator(cairo.Operator.SOURCE)
cr:set_source_surface(shape_img, border + l, border + t) 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:rectangle(border + l, border + t, geom.width - l - r, geom.height - t - b)
cr:fill() 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 return result
end 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. --- Update a client's bounding shape from the shape the client set itself.
-- @function awful.shape.update.bounding -- @function awful.client.shape.update.bounding
-- @client c The client to act on -- @client c The client to act on
function shape.update.bounding(c) function shape.update.bounding(c)
local res = shape.get_transformed(c, "bounding") local res = shape.get_transformed(c, "bounding")
@ -64,7 +144,7 @@ function shape.update.bounding(c)
end end
--- Update a client's clip shape from the shape the client set itself. --- Update a client's clip shape from the shape the client set itself.
-- @function awful.shape.update.clip -- @function awful.client.shape.update.clip
-- @client c The client to act on -- @client c The client to act on
function shape.update.clip(c) function shape.update.clip(c)
local res = shape.get_transformed(c, "clip") local res = shape.get_transformed(c, "clip")
@ -75,14 +155,6 @@ function shape.update.clip(c)
end end
end end
--- Update all of a client's shapes from the shapes the client set itself.
-- @function awful.shape.update.all
-- @client c The client to act on
function shape.update.all(c)
shape.update.bounding(c)
shape.update.clip(c)
end
capi.client.connect_signal("property::shape_client_bounding", shape.update.bounding) 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::shape_client_clip", shape.update.clip)
capi.client.connect_signal("property::size", shape.update.all) capi.client.connect_signal("property::size", shape.update.all)

View File

@ -1,6 +1,7 @@
local test_client = require("_client") local test_client = require("_client")
local placement = require("awful.placement") local placement = require("awful.placement")
local amouse = require("awful.mouse") local amouse = require("awful.mouse")
local rounded_rect = require("gears.shape").rounded_rect
local steps = {} local steps = {}
@ -8,7 +9,6 @@ table.insert(steps, function(count)
if count == 1 then -- Setup. if count == 1 then -- Setup.
test_client("foobar", "foobar") test_client("foobar", "foobar")
elseif #client.get() > 0 then elseif #client.get() > 0 then
client.get()[1] : geometry { client.get()[1] : geometry {
x = 200, x = 200,
y = 200, y = 200,
@ -16,6 +16,8 @@ table.insert(steps, function(count)
height = 300, height = 300,
} }
client.get()[1].shape = rounded_rect
return true return true
end end
end) end)