--------------------------------------------------------------------------- --- 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