From a072e341949a52d7dd5fe2a498a9199823eecee1 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 28 Apr 2016 16:47:01 -0400 Subject: [PATCH 01/39] shims: Add the request:: signals --- tests/examples/shims/client.lua | 3 +++ 1 file changed, 3 insertions(+) diff --git a/tests/examples/shims/client.lua b/tests/examples/shims/client.lua index 97d19fd9..5fcc6486 100644 --- a/tests/examples/shims/client.lua +++ b/tests/examples/shims/client.lua @@ -166,6 +166,9 @@ client:_add_signal("property::fullscreen") client:_add_signal("property::border_width") client:_add_signal("property::hidden") client:_add_signal("property::screen") +client:_add_signal("request::tag") +client:_add_signal("request::geometry") +client:_add_signal("request::activate") client:_add_signal("raised") client:_add_signal("lowered") client:_add_signal("list") From 604ea15c4585c0124fb4a7a9e2b9660694f567b6 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Wed, 20 Apr 2016 21:41:00 -0400 Subject: [PATCH 02/39] awful.placement: Add a 'resize_to_mouse' function --- lib/awful/placement.lua | 91 ++++++++++++++++++++++++++++++++++++++++- 1 file changed, 89 insertions(+), 2 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index b98079c9..a46996c8 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -143,6 +143,21 @@ local align_map = { -- Store function -> keys local reverse_align_map = {} +-- Some parameters to correctly compute the final size +local resize_to_point_map = { + -- Corners + top_left = {p1= nil , p2={1,1}, x_only=false, y_only=false}, + top_right = {p1={0,1} , p2= nil , x_only=false, y_only=false}, + bottom_left = {p1= nil , p2={1,0}, x_only=false, y_only=false}, + bottom_right = {p1={0,0} , p2= nil , x_only=false, y_only=false}, + + -- Sides + left = {p1= nil , p2={1,1}, x_only=true , y_only=false}, + right = {p1={0,0} , p2= nil , x_only=true , y_only=false}, + top = {p1= nil , p2={1,1}, x_only=false, y_only=true }, + bottom = {p1={0,0} , p2= nil , x_only=false, y_only=true }, +} + --- Add a context to the arguments. -- This function extend the argument table. The context is used by some -- internal helper methods. If there already is a context, it has priority and @@ -172,8 +187,6 @@ local function area_common(d, new_geo, ignore_border_width) -- The C side expect no arguments, nil isn't valid local geometry = new_geo and d:geometry(new_geo) or d:geometry() local border = ignore_border_width and 0 or d.border_width or 0 - geometry.x = geometry.x - geometry.y = geometry.y geometry.width = geometry.width + 2 * border geometry.height = geometry.height + 2 * border return geometry @@ -434,6 +447,24 @@ local function area_remove(areas, elem) return areas end +-- Convert 2 points into a rectangle +local function rect_from_points(p1x, p1y, p2x, p2y) + return { + x = p1x, + y = p1y, + width = p2x - p1x, + height = p2y - p1y, + } +end + +-- Convert a rectangle and matrix info into a point +local function rect_to_point(rect, corner_i, corner_j) + return { + x = rect.x + corner_i * math.floor(rect.width ), + y = rect.y + corner_j * math.floor(rect.height), + } +end + --- Move a drawable to the closest corner of the parent geometry (such as the -- screen). -- @@ -622,6 +653,62 @@ function placement.next_to_mouse(c, offset) return c:geometry({ x = x, y = y }) end +--- Resize the drawable to the cursor. +-- +-- Valid args: +-- +-- * *axis*: The axis (vertical or horizontal). If none is +-- specified, then the drawable will be resized on both axis. +-- +-- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) +-- @tparam[opt={}] table args Other arguments +function placement.resize_to_mouse(d, args) + d = d or capi.client.focus + args = add_context(args, "resize_to_mouse") + + local coords = capi.mouse.coords() + local ngeo = geometry_common(d, args) + local h_only = args.axis == "horizontal" + local v_only = args.axis == "vertical" + + -- To support both growing and shrinking the drawable, it is necessary + -- to decide to use either "north or south" and "east or west" directions. + -- Otherwise, the result will always be 1x1 + local closest_corner = placement.closest_corner(capi.mouse, { + parent = d, + include_sides = args.include_sides or false, + }) + + -- Given "include_sides" wasn't set, it will always return a name + -- with the 2 axis. If only one axis is needed, adjust the result + if h_only then + closest_corner = closest_corner:match("left") or closest_corner:match("right") + elseif v_only then + closest_corner = closest_corner:match("top") or closest_corner:match("bottom") + end + + -- Use p0 (mouse), p1 and p2 to create a rectangle + local pts = resize_to_point_map[closest_corner] + local p1 = pts.p1 and rect_to_point(ngeo, pts.p1[1], pts.p1[2]) or coords + local p2 = pts.p2 and rect_to_point(ngeo, pts.p2[1], pts.p2[2]) or coords + + -- Create top_left and bottom_right points, convert to rectangle + ngeo = rect_from_points( + pts.y_only and ngeo.x or math.min(p1.x, p2.x), + pts.x_only and ngeo.y or math.min(p1.y, p2.y), + pts.y_only and ngeo.x + ngeo.width or math.max(p2.x, p1.x), + pts.x_only and ngeo.y + ngeo.height or math.max(p2.y, p1.y) + ) + + local bw = d.border_width or 0 + + for _, a in ipairs {"width", "height"} do + ngeo[a] = ngeo[a] - 2*bw + end + + geometry_common(d, args, ngeo) +end + --- Move the drawable (client or wibox) `d` to a screen position or side. -- -- Supported args.positions are: From 5f17f1aa83a7f42a31b90857bd43cf04deaf6c74 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 30 Apr 2016 02:35:14 -0400 Subject: [PATCH 03/39] resize_to_mouse: Support size_hints --- lib/awful/placement.lua | 27 +++++++++++++++++++-------- 1 file changed, 19 insertions(+), 8 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index a46996c8..4de4631e 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -146,16 +146,16 @@ local reverse_align_map = {} -- Some parameters to correctly compute the final size local resize_to_point_map = { -- Corners - top_left = {p1= nil , p2={1,1}, x_only=false, y_only=false}, - top_right = {p1={0,1} , p2= nil , x_only=false, y_only=false}, - bottom_left = {p1= nil , p2={1,0}, x_only=false, y_only=false}, - bottom_right = {p1={0,0} , p2= nil , x_only=false, y_only=false}, + top_left = {p1= nil , p2={1,1}, x_only=false, y_only=false, align="bottom_right"}, + top_right = {p1={0,1} , p2= nil , x_only=false, y_only=false, align="bottom_left" }, + bottom_left = {p1= nil , p2={1,0}, x_only=false, y_only=false, align="top_right" }, + bottom_right = {p1={0,0} , p2= nil , x_only=false, y_only=false, align="top_left" }, -- Sides - left = {p1= nil , p2={1,1}, x_only=true , y_only=false}, - right = {p1={0,0} , p2= nil , x_only=true , y_only=false}, - top = {p1= nil , p2={1,1}, x_only=false, y_only=true }, - bottom = {p1={0,0} , p2= nil , x_only=false, y_only=true }, + left = {p1= nil , p2={1,1}, x_only=true , y_only=false, align="top_right" }, + right = {p1={0,0} , p2= nil , x_only=true , y_only=false, align="top_left" }, + top = {p1= nil , p2={1,1}, x_only=false, y_only=true , align="bottom_left" }, + bottom = {p1={0,0} , p2= nil , x_only=false, y_only=true , align="top_left" }, } --- Add a context to the arguments. @@ -706,6 +706,17 @@ function placement.resize_to_mouse(d, args) ngeo[a] = ngeo[a] - 2*bw end + -- Now, correct the geometry by the given size_hints offset + if d.apply_size_hints then + local w, h = d:apply_size_hints( + ngeo.width, + ngeo.height + ) + local offset = align_map[pts.align](w, h, ngeo.width, ngeo.height) + ngeo.x = ngeo.x - offset.x + ngeo.y = ngeo.y - offset.y + end + geometry_common(d, args, ngeo) end From 3e7a7f23c2d724a6bfa3c61930fe1cf2429551a1 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 23 Apr 2016 23:32:33 -0400 Subject: [PATCH 04/39] tests: Test awful.placement.resize_to_mouse --- lib/awful/placement.lua | 1 + .../awful/placement/resize_to_mouse.lua | 74 +++++++++++++++++++ 2 files changed, 75 insertions(+) create mode 100644 tests/examples/awful/placement/resize_to_mouse.lua diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index 4de4631e..4baf67e0 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -660,6 +660,7 @@ end -- * *axis*: The axis (vertical or horizontal). If none is -- specified, then the drawable will be resized on both axis. -- +--@DOC_awful_placement_resize_to_mouse_EXAMPLE@ -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) -- @tparam[opt={}] table args Other arguments function placement.resize_to_mouse(d, args) diff --git a/tests/examples/awful/placement/resize_to_mouse.lua b/tests/examples/awful/placement/resize_to_mouse.lua new file mode 100644 index 00000000..3114fb38 --- /dev/null +++ b/tests/examples/awful/placement/resize_to_mouse.lua @@ -0,0 +1,74 @@ +--DOC_HIDE_ALL +local awful = {placement = require("awful.placement")} +local unpack = unpack or table.unpack -- luacheck: globals unpack (compatibility with Lua 5.1) + +screen._setup_grid(64, 48, {4, 4, 4, 4}, {workarea_sides=0}) + +local function test_touch_mouse(c) + local coords = mouse.coords() + + return c:geometry().x == coords.x or c:geometry().y == coords.y + or c:geometry().x+c:geometry().width+2*c.border_width == coords.x + or c:geometry().y+c:geometry().height+2*c.border_width == coords.y +end + +for s=1, 8 do + local scr = screen[s] + local x, y = scr.geometry.x, scr.geometry.y + local c = client.gen_fake{x = x+22, y = y+16, width=20, height=15, screen=scr} + assert(client.get()[s] == c) +end + +for s=9, 16 do + local scr = screen[s] + local x, y = scr.geometry.x, scr.geometry.y + local c = client.gen_fake{x = x+10, y = y+10, width=44, height=28, screen=scr} + assert(client.get()[s] == c) +end + +local function move_corsor(s, x, y) + local sg = screen[s].geometry + mouse.coords {x=sg.x+x,y=sg.y+y} +end + +local all_coords_out = { + top_left = {10, 10}, + top = {32, 10}, + top_right = {60, 10}, + right = {60, 20}, + bottom_right = {60, 40}, + bottom = {32, 40}, + bottom_left = {10, 40}, + left = {10, 29}, +} + +local all_coords_in = { + top_left = {20, 18}, + top = {32, 18}, + top_right = {44, 18}, + right = {44, 24}, + bottom_right = {44, 34}, + bottom = {32, 34}, + bottom_left = {20, 34}, + left = {32, 24}, +} + +-- Top left +local s = 1 +for k, v in pairs(all_coords_out) do + move_corsor(s, unpack(v)) + assert(client.get()[s].screen == screen[s]) + awful.placement.resize_to_mouse(client.get()[s], {include_sides=true}) + mouse.push_history() + assert(test_touch_mouse(client.get()[s]), k) + s = s + 1 +end + +for k, v in pairs(all_coords_in) do + move_corsor(s, unpack(v)) + assert(client.get()[s].screen == screen[s]) + awful.placement.resize_to_mouse(client.get()[s], {include_sides=true}) + mouse.push_history() + assert(test_touch_mouse(client.get()[s]), k) + s = s + 1 +end From 75764108e1c1e5e6d3227a791118e5e4eaa6e3c7 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Wed, 20 Apr 2016 21:48:46 -0400 Subject: [PATCH 05/39] awful.placement: Make all functions return the geometry --- lib/awful/placement.lua | 36 +++++++++++++------ tests/examples/awful/placement/bottom.lua | 1 + .../examples/awful/placement/bottom_left.lua | 1 + .../examples/awful/placement/bottom_right.lua | 1 + .../awful/placement/center_horizontal.lua | 1 + tests/examples/awful/placement/centered.lua | 1 + .../awful/placement/closest_mouse.lua | 2 +- tests/examples/awful/placement/left.lua | 1 + tests/examples/awful/placement/right.lua | 1 + .../examples/awful/placement/stretch_down.lua | 1 + .../examples/awful/placement/stretch_left.lua | 1 + .../awful/placement/stretch_right.lua | 1 + tests/examples/awful/placement/stretch_up.lua | 1 + tests/examples/awful/placement/top.lua | 1 + tests/examples/awful/placement/top_left.lua | 1 + tests/examples/awful/placement/top_right.lua | 1 + 16 files changed, 41 insertions(+), 11 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index 4baf67e0..435d34d0 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -476,6 +476,7 @@ end -- @tparam[opt=client.focus] drawable d A drawable (like `client`, `mouse` -- or `wibox`) -- @tparam[opt={}] table args The arguments +-- @treturn table The new geometry -- @treturn string The corner name function placement.closest_corner(d, args) args = add_context(args, "closest_corner") @@ -507,9 +508,9 @@ function placement.closest_corner(d, args) -- Transpose the corner back to the original size local new_args = setmetatable({position = corner}, {__index=args}) - placement_private.align(d, new_args) + local ngeo = placement_private.align(d, new_args) - return corner + return ngeo, corner end --- Place the client so no part of it will be outside the screen (workarea). @@ -551,6 +552,7 @@ end --- Place the client where there's place available with minimum overlap. --@DOC_awful_placement_no_overlap_EXAMPLE@ -- @param c The client. +-- @treturn table The new geometry function placement.no_overlap(c) c = c or capi.client.focus local geometry = area_common(c) @@ -605,7 +607,7 @@ end --- Place the client under the mouse. --@DOC_awful_placement_under_mouse_EXAMPLE@ -- @param c The client. --- @return The new client geometry. +-- @treturn table The new geometry function placement.under_mouse(c) c = c or capi.client.focus local c_geometry = area_common(c) @@ -621,7 +623,7 @@ end --@DOC_awful_placement_next_to_mouse_EXAMPLE@ -- @client[opt=focused] c The client. -- @tparam[opt=apply_dpi(5)] integer offset The offset from the mouse position. --- @return The new client geometry. +-- @treturn table The new geometry function placement.next_to_mouse(c, offset) c = c or capi.client.focus offset = offset or dpi(5) @@ -663,6 +665,7 @@ end --@DOC_awful_placement_resize_to_mouse_EXAMPLE@ -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) -- @tparam[opt={}] table args Other arguments +-- @treturn table The new geometry function placement.resize_to_mouse(d, args) d = d or capi.client.focus args = add_context(args, "resize_to_mouse") @@ -675,7 +678,7 @@ function placement.resize_to_mouse(d, args) -- To support both growing and shrinking the drawable, it is necessary -- to decide to use either "north or south" and "east or west" directions. -- Otherwise, the result will always be 1x1 - local closest_corner = placement.closest_corner(capi.mouse, { + local _, closest_corner = placement.closest_corner(capi.mouse, { parent = d, include_sides = args.include_sides or false, }) @@ -719,6 +722,8 @@ function placement.resize_to_mouse(d, args) end geometry_common(d, args, ngeo) + + return ngeo end --- Move the drawable (client or wibox) `d` to a screen position or side. @@ -740,6 +745,7 @@ end --@DOC_awful_placement_align_EXAMPLE@ -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) -- @tparam[opt={}] table args Other arguments +-- @treturn table The new geometry function placement.align(d, args) args = add_context(args, "align") d = d or capi.client.focus @@ -757,14 +763,18 @@ function placement.align(d, args) dgeo.height ) - geometry_common(d, args, { + local ngeo = { x = (pos.x and math.ceil(sgeo.x + pos.x) or dgeo.x) , y = (pos.y and math.ceil(sgeo.y + pos.y) or dgeo.y) , width = math.ceil(dgeo.width ) - 2*bw, height = math.ceil(dgeo.height ) - 2*bw, - }) + } + + geometry_common(d, args, ngeo) attach(d, placement[args.position], args) + + return ngeo end -- Add the alias functions @@ -772,7 +782,7 @@ for k in pairs(align_map) do placement[k] = function(d, args) args = add_context(args, k) args.position = k - placement_private.align(d, args) + return placement_private.align(d, args) end reverse_align_map[placement[k]] = k end @@ -810,6 +820,7 @@ end --@DOC_awful_placement_stretch_EXAMPLE@ -- @tparam[opt=client.focus] drawable d A drawable (like `client` or `wibox`) -- @tparam[opt={}] table args The arguments +-- @treturn table The new geometry function placement.stretch(d, args) args = add_context(args, "stretch") @@ -851,6 +862,8 @@ function placement.stretch(d, args) geometry_common(d, args, ngeo) attach(d, placement["stretch_"..args.direction], args) + + return ngeo end -- Add the alias functions @@ -858,7 +871,7 @@ for _,v in ipairs {"left", "right", "up", "down"} do placement["stretch_"..v] = function(d, args) args = add_context(args, "stretch_"..v) args.direction = v - placement_private.stretch(d, args) + return placement_private.stretch(d, args) end end @@ -879,6 +892,7 @@ end --@DOC_awful_placement_maximize_EXAMPLE@ -- @tparam[opt=client.focus] drawable d A drawable (like `client` or `wibox`) -- @tparam[opt={}] table args The arguments +-- @treturn table The new geometry function placement.maximize(d, args) args = add_context(args, "maximize") d = d or capi.client.focus @@ -902,6 +916,8 @@ function placement.maximize(d, args) geometry_common(d, args, ngeo) attach(d, placement.maximize, args) + + return ngeo end -- Add the alias functions @@ -909,7 +925,7 @@ for _, v in ipairs {"vertically", "horizontally"} do placement["maximize_"..v] = function(d2, args) args = add_context(args, "maximize_"..v) args.axis = v - placement_private.maximize(d2, args) + return placement_private.maximize(d2, args) end end diff --git a/tests/examples/awful/placement/bottom.lua b/tests/examples/awful/placement/bottom.lua index 10dff9ab..6af09262 100644 --- a/tests/examples/awful/placement/bottom.lua +++ b/tests/examples/awful/placement/bottom.lua @@ -1,6 +1,7 @@ -- Align a client to the bottom of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name bottom --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/bottom_left.lua b/tests/examples/awful/placement/bottom_left.lua index a9ebc062..4d4cff37 100644 --- a/tests/examples/awful/placement/bottom_left.lua +++ b/tests/examples/awful/placement/bottom_left.lua @@ -1,6 +1,7 @@ -- Align a client to the bottom left of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name bottom_left --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/bottom_right.lua b/tests/examples/awful/placement/bottom_right.lua index 0ad1072e..2abb4490 100644 --- a/tests/examples/awful/placement/bottom_right.lua +++ b/tests/examples/awful/placement/bottom_right.lua @@ -1,6 +1,7 @@ -- Align a client to the bottom right of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name bottom_right --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/center_horizontal.lua b/tests/examples/awful/placement/center_horizontal.lua index b9d1d6bb..931b947c 100644 --- a/tests/examples/awful/placement/center_horizontal.lua +++ b/tests/examples/awful/placement/center_horizontal.lua @@ -1,6 +1,7 @@ -- Align a client to the horizontal center left of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name center_horizontal --DOC_HEADER -- @class function --DOC_HEADER screen[1]._resize {width = 128, height = 96} --DOC_HIDE diff --git a/tests/examples/awful/placement/centered.lua b/tests/examples/awful/placement/centered.lua index af1a1c84..d6b1e3ae 100644 --- a/tests/examples/awful/placement/centered.lua +++ b/tests/examples/awful/placement/centered.lua @@ -1,6 +1,7 @@ -- Align a client to the center of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name centered --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/closest_mouse.lua b/tests/examples/awful/placement/closest_mouse.lua index 90642641..4269d1d8 100644 --- a/tests/examples/awful/placement/closest_mouse.lua +++ b/tests/examples/awful/placement/closest_mouse.lua @@ -55,7 +55,7 @@ assert(mouse.coords().x == c.x and mouse.coords().y == c.y+c.height+2*bw) --DOC_ -- It is possible to emulate the mouse API to get the closest corner of -- random area -local corner = awful.placement.closest_corner( +local _, corner = awful.placement.closest_corner( {coords=function() return {x = 100, y=100} end}, {include_sides = true, bounding_rect = {x=0, y=0, width=200, height=200}} ) diff --git a/tests/examples/awful/placement/left.lua b/tests/examples/awful/placement/left.lua index 49cf93ac..d3e4e698 100644 --- a/tests/examples/awful/placement/left.lua +++ b/tests/examples/awful/placement/left.lua @@ -1,6 +1,7 @@ -- Align a client to the left of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name left --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/right.lua b/tests/examples/awful/placement/right.lua index d2450116..4e16cea6 100644 --- a/tests/examples/awful/placement/right.lua +++ b/tests/examples/awful/placement/right.lua @@ -1,6 +1,7 @@ -- Align a client to the right of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name right --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/stretch_down.lua b/tests/examples/awful/placement/stretch_down.lua index 258647f8..720da36d 100644 --- a/tests/examples/awful/placement/stretch_down.lua +++ b/tests/examples/awful/placement/stretch_down.lua @@ -1,6 +1,7 @@ -- Stretch the drawable to the bottom of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name stretch_down --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/stretch_left.lua b/tests/examples/awful/placement/stretch_left.lua index 51364697..4fdd5857 100644 --- a/tests/examples/awful/placement/stretch_left.lua +++ b/tests/examples/awful/placement/stretch_left.lua @@ -1,6 +1,7 @@ -- Stretch the drawable to the left of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name stretch_left --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/stretch_right.lua b/tests/examples/awful/placement/stretch_right.lua index 9965d1a6..47dcce80 100644 --- a/tests/examples/awful/placement/stretch_right.lua +++ b/tests/examples/awful/placement/stretch_right.lua @@ -1,6 +1,7 @@ -- Stretch the drawable to the right of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name stretch_right --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/stretch_up.lua b/tests/examples/awful/placement/stretch_up.lua index 817a11d5..0ccfd2bc 100644 --- a/tests/examples/awful/placement/stretch_up.lua +++ b/tests/examples/awful/placement/stretch_up.lua @@ -1,6 +1,7 @@ -- Stretch the drawable to the top of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name stretch_up --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/top.lua b/tests/examples/awful/placement/top.lua index f84ea306..b3375fbc 100644 --- a/tests/examples/awful/placement/top.lua +++ b/tests/examples/awful/placement/top.lua @@ -1,6 +1,7 @@ -- Align a client to the top of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name top --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/top_left.lua b/tests/examples/awful/placement/top_left.lua index a2bb3b58..f70b85f6 100644 --- a/tests/examples/awful/placement/top_left.lua +++ b/tests/examples/awful/placement/top_left.lua @@ -1,6 +1,7 @@ -- Align a client to the top left of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name top_left --DOC_HEADER -- @class function --DOC_HEADER diff --git a/tests/examples/awful/placement/top_right.lua b/tests/examples/awful/placement/top_right.lua index a94890c9..ef6fc833 100644 --- a/tests/examples/awful/placement/top_right.lua +++ b/tests/examples/awful/placement/top_right.lua @@ -1,6 +1,7 @@ -- Align a client to the top right of the parent area. --DOC_HEADER -- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) --DOC_HEADER -- @tparam[opt={}] table args Other arguments") --DOC_HEADER +-- @treturn table The new geometry --DOC_HEADER -- @name top_right --DOC_HEADER -- @class function --DOC_HEADER From 75af0a24c76b19dcdcac2e427b7392ba639330f9 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 21 Apr 2016 04:07:35 -0400 Subject: [PATCH 06/39] awful.placement: Add a 'pretend' option --- lib/awful/placement.lua | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index 435d34d0..e9133198 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -18,6 +18,11 @@ -- -- ### Common arguments -- +-- **pretend** (*boolean*): +-- +-- Do not apply the new geometry. This is useful if only the return values is +-- necessary. +-- -- **honor_workarea** (*boolean*): -- -- Take workarea into account when placing the drawable (default: false) @@ -209,14 +214,17 @@ local function geometry_common(obj, args, new_geo, ignore_border_width) -- It's a mouse if obj.coords then - local coords = new_geo and obj.coords(new_geo) or obj.coords() + local coords = (not args.pretend and new_geo) + and obj.coords(new_geo) or obj.coords() return {x=coords.x, y=coords.y, width=0, height=0} elseif obj.geometry then local geo = obj.geometry -- It is either a drawable or something that implement its API if type(geo) == "function" then - local dgeo = area_common(obj, new_geo, ignore_border_width) + local dgeo = area_common( + obj, (not args.pretend) and new_geo or nil, ignore_border_width + ) -- Apply the margins if args.margins then @@ -680,6 +688,7 @@ function placement.resize_to_mouse(d, args) -- Otherwise, the result will always be 1x1 local _, closest_corner = placement.closest_corner(capi.mouse, { parent = d, + pretend = true, include_sides = args.include_sides or false, }) From 7242b30c01c2d27407b5204b36c81fc84ea944f6 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 21 Apr 2016 01:51:34 -0400 Subject: [PATCH 07/39] awful.mouse: Deprecate awful.mouse.client.corner --- lib/awful/mouse/init.lua | 55 ++++++++++++++-------------------------- 1 file changed, 19 insertions(+), 36 deletions(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 7a8a86a7..bd8d01d4 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -11,6 +11,7 @@ local layout = require("awful.layout") local tag = require("awful.tag") local aclient = require("awful.client") +local aplace = require("awful.placement") local awibox = require("awful.wibox") local util = require("awful.util") local type = type @@ -297,49 +298,31 @@ function mouse.wibox.move(w) end --- Get a client corner coordinates. --- @param c The client to get corner from, focused one by default. --- @param corner The corner to use: auto, top_left, top_right, bottom_left, --- bottom_right. Default is auto, and auto find the nearest corner. --- @return Actual used corner and x and y coordinates. +-- @tparam[opt=client.focus] client c The client to get corner from, focused one by default. +-- @tparam string corner The corner to use: auto, top_left, top_right, bottom_left, +-- bottom_right, left, right, top bottom. Default is auto, and auto find the +-- nearest corner. +-- @treturn string The corner name +-- @treturn number x The horizontal position +-- @treturn number y The vertical position function mouse.client.corner(c, corner) + util.deprecated( + "Use awful.placement.closest_corner(mouse) or awful.placement[corner](mouse)".. + " instead of awful.mouse.client.corner" + ) + c = c or capi.client.focus if not c then return end - local g = c:geometry() + local ngeo = nil - if not corner or corner == "auto" then - local m_c = capi.mouse.coords() - if math.abs(g.y - m_c.y) < math.abs(g.y + g.height - m_c.y) then - if math.abs(g.x - m_c.x) < math.abs(g.x + g.width - m_c.x) then - corner = "top_left" - else - corner = "top_right" - end - else - if math.abs(g.x - m_c.x) < math.abs(g.x + g.width - m_c.x) then - corner = "bottom_left" - else - corner = "bottom_right" - end - end + if (not corner) or corner == "auto" then + ngeo, corner = aplace.closest_corner(mouse, {parent = c}) + elseif corner and aplace[corner] then + ngeo = aplace[corner](mouse, {parent = c}) end - local x, y - if corner == "top_right" then - x = g.x + g.width - y = g.y - elseif corner == "top_left" then - x = g.x - y = g.y - elseif corner == "bottom_left" then - x = g.x - y = g.y + g.height - else - x = g.x + g.width - y = g.y + g.height - end - - return corner, x, y + return corner, ngeo and ngeo.x or nil, ngeo and ngeo.y or nil end --- Resize a client. From e93e2913b670ba8aee80842e6090417e76fe1bfa Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 21 Apr 2016 04:01:41 -0400 Subject: [PATCH 08/39] awful.placement: Fix a closest_corner corner case When the mouse was exactly on the right or bottom edge, there was a rounding error. --- lib/awful/placement.lua | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index e9133198..59fea71f 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -500,8 +500,10 @@ function placement.closest_corner(d, args) -- Use the product of 3 to get the closest point in a NxN matrix local function f(_n, mat) n = _n - corner_i = -math.ceil( ( (sgeo.x - pos.x) * n) / sgeo.width ) - corner_j = -math.ceil( ( (sgeo.y - pos.y) * n) / sgeo.height ) + -- The +1 is required to avoid a rounding error when + -- pos.x == sgeo.x+sgeo.width + corner_i = -math.ceil( ( (sgeo.x - pos.x) * n) / (sgeo.width + 1)) + corner_j = -math.ceil( ( (sgeo.y - pos.y) * n) / (sgeo.height + 1)) return mat[corner_j + 1][corner_i + 1] end From f8f57fb6b71a69a6a5f9efb8218912984e597e8c Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 21 Apr 2016 04:03:20 -0400 Subject: [PATCH 09/39] awful.mouse: Add a generic mousegrabber Previously, all layouts had their own mouse grabbing logic. The new one is based on the client request::geometry feature. --- lib/awful/mouse/init.lua | 4 +- lib/awful/mouse/resize.lua | 193 +++++++++++++++++++++++++++++++++++++ 2 files changed, 196 insertions(+), 1 deletion(-) create mode 100644 lib/awful/mouse/resize.lua diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index bd8d01d4..4bc78d0f 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -26,7 +26,9 @@ local capi = mousegrabber = mousegrabber, } -local mouse = {} +local mouse = { + resize = require("awful.mouse.resize") +} mouse.client = {} mouse.wibox = {} diff --git a/lib/awful/mouse/resize.lua b/lib/awful/mouse/resize.lua new file mode 100644 index 00000000..3662358e --- /dev/null +++ b/lib/awful/mouse/resize.lua @@ -0,0 +1,193 @@ +--------------------------------------------------------------------------- +--- An extandable mouse resizing handler. +-- +-- This module offer a resizing and moving mechanism for drawable such as +-- clients and wiboxes. +-- +-- @author Emmanuel Lepage Vallee <elv1313@gmail.com> +-- @copyright 2016 Emmanuel Lepage Vallee +-- @release @AWESOME_VERSION@ +-- @submodule awful.mouse +--------------------------------------------------------------------------- + +local aplace = require("awful.placement") +local capi = {mousegrabber = mousegrabber} + +local module = {} + +local mode = "live" +local req = "request::geometry" +local callbacks = {enter={}, move={}, leave={}} + +--- Set the resize mode. +-- The available modes are: +-- +-- * **live**: Resize the layout everytime the mouse move +-- * **after**: Resize the layout only when the mouse is released +-- +-- Some clients, such as XTerm, may lose information if resized too often. +-- +-- @tparam string m The mode +function module.set_mode(m) + assert(m == "live" or m == "after") + mode = m +end + +--- Add a initialization callback. +-- This callback will be executed before the mouse grabbing start +-- @tparam function cb The callback (or nil) +-- @tparam[default=other] string context The callback context +function module.add_enter_callback(cb, context) + context = context or "other" + callbacks.enter[context] = callbacks.enter[context] or {} + table.insert(callbacks.enter[context], cb) +end + +--- Add a "move" callback. +-- This callback is executed in "after" mode (see `set_mode`) instead of +-- applying the operation. +-- @tparam function cb The callback (or nil) +-- @tparam[default=other] string context The callback context +function module.add_move_callback(cb, context) + context = context or "other" + callbacks.move[context] = callbacks.move[context] or {} + table.insert(callbacks.move[context], cb) +end + +--- Add a "leave" callback +-- This callback is executed just before the `mousegrabber` stop +-- @tparam function cb The callback (or nil) +-- @tparam[default=other] string context The callback context +function module.add_leave_callback(cb, context) + context = context or "other" + callbacks.leave[context] = callbacks.leave[context] or {} + table.insert(callbacks.leave[context], cb) +end + +-- Resize, the drawable +-- +-- Valid `args` are: +-- +-- * *enter_callback*: A function called before the `mousegrabber` start +-- * *move_callback*: A function called when the mouse move +-- * *leave_callback*: A function called before the `mousegrabber` is released +-- * *mode*: The resize mode +-- +-- @tparam client client A client +-- @tparam[default=mouse.resize] string context The resizing context +-- @tparam[opt={}] table args A set of `awful.placement` arguments +local function handler(_, client, context, args) --luacheck: no unused_args + args = args or {} + context = context or "mouse.resize" + + local placement = args.placement + + if type(placement) == "string" and aplace[placement] then + placement = aplace[placement] + end + + -- Extend the table with the default arguments + args = setmetatable( + { + placement = placement or aplace.resize_to_mouse, + mode = args.mode or mode, + pretend = true, + }, + {__index = args or {}} + ) + + local geo + + for _, cb in ipairs(callbacks.enter[context] or {}) do + geo = cb(client, args) + + if geo == false then + return false + end + end + + if args.enter_callback then + geo = args.enter_callback(client, args) + + if geo == false then + return false + end + end + + geo = nil + + -- Execute the placement function and use request::geometry + capi.mousegrabber.run(function (_mouse) + if not client.valid then return end + + -- Resize everytime the mouse move (default behavior) + if args.mode == "live" then + -- Get the new geometry + geo = setmetatable(args.placement(client, args),{__index=args}) + end + + -- Execute the move callbacks. This can be used to add features such as + -- snap or adding fancy graphical effects. + for _, cb in ipairs(callbacks.move[context] or {}) do + -- If something is returned, assume it is a modified geometry + geo = cb(client, geo, args) or geo + + if geo == false then + return false + end + end + + if args.move_callback then + geo = args.move_callback(client, geo, args) + + if geo == false then + return false + end + end + + -- In case it was modified + setmetatable(geo,{__index=args}) + + if args.mode == "live" then + -- Ask the resizing handler to resize the client + client:emit_signal( req, context, geo) + end + + -- Quit when the button is released + for _,v in pairs(_mouse.buttons) do + if v then return true end + end + + -- Only resize after the mouse is released, this avoid losing content + -- in resize sensitive apps such as XTerm or allow external modules + -- to implement custom resizing + if args.mode == "after" then + -- Get the new geometry + geo = args.placement(client, args) + + -- Ask the resizing handler to resize the client + client:emit_signal( req, context, geo) + end + + geo = nil + + for _, cb in ipairs(callbacks.leave[context] or {}) do + geo = cb(client, geo, args) + end + + if args.leave_callback then + geo = args.leave_callback(client, geo, args) + end + + if not geo then return false end + + -- In case it was modified + setmetatable(geo,{__index=args}) + + client:emit_signal( req, context, geo) + + return false + end, "cross") +end + +return setmetatable(module, {__call=handler}) From be455cb140f46198cd3526767c2348f1ff3c2c3a Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 21 Apr 2016 05:01:35 -0400 Subject: [PATCH 10/39] awful.mouse: Add a request::geometry handler. --- lib/awful/mouse/init.lua | 40 ++++++++++++++++++++++++++++++++-------- 1 file changed, 32 insertions(+), 8 deletions(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 4bc78d0f..4b3575dd 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -329,8 +329,9 @@ end --- Resize a client. -- @param c The client to resize, or the focused one by default. --- @param corner The corner to grab on resize. Auto detected by default. -function mouse.client.resize(c, corner) +-- @tparam string corner The corner to grab on resize. Auto detected by default. +-- @tparam[opt={}] table args A set of `awful.placement` arguments +function mouse.client.resize(c, corner, args) c = c or capi.client.focus if not c then return end @@ -342,16 +343,39 @@ function mouse.client.resize(c, corner) return end - local lay = layout.get(c.screen) - local corner2, x, y = mouse.client.corner(c, corner) + -- Move the mouse to the corner + if corner and aplace[corner] then + aplace[corner](capi.mouse, {parent=c}) + end - if lay == layout.suit.floating or c.floating then - return layout.suit.floating.mouse_resize_handler(c, corner2, x, y) - elseif lay.mouse_resize_handler then - return lay.mouse_resize_handler(c, corner2, x, y) + mouse.resize(c, "mouse.resize", args or {include_sides=true}) +end + +--- Default handler for `request::geometry` signals with `mouse.resize` context. +-- @tparam client c The client +-- @tparam string context The context +-- @tparam[opt={}] table hints The hints to pass to the handler +function mouse.resize_handler(c, context, hints) + if hints and context and context:find("mouse.*") then + -- This handler only handle the floating clients. If the client is tiled, + -- then it let the layouts handle it. + local lay = c.screen.selected_tag.layout + + if lay == layout.suit.floating or c.floating then + c:geometry { + x = hints.x, + y = hints.y, + width = hints.width, + height = hints.height, + } + elseif lay.resize_handler then + lay.resize_handler(c, context, hints) + end end end +capi.client.connect_signal("request::geometry", mouse.resize_handler) + -- Set the cursor at startup capi.root.cursor("left_ptr") From e6a04a59223f9a974a5e1fc415780fdec14019fd Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 21 Apr 2016 04:14:09 -0400 Subject: [PATCH 11/39] awful.placement: Extend the compositing feature It can now keep the different return values and use them in later chain nodes. It also add a "virtual" geometry argument so the geometry is applied only after the last node is executed. Finally, it fixes using pretend and a composite chain at the same time. --- lib/awful/placement.lua | 101 ++++++++++++++++++++++++++++++++++------ 1 file changed, 88 insertions(+), 13 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index 59fea71f..acb02517 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -90,20 +90,89 @@ end local wrap_client = nil -local function compose(w1, w2) - return wrap_client(function(...) - w1(...) - w2(...) - return --It make no sense to keep a return value - end) +--- Allow multiple placement functions to be daisy chained. +-- This also allow the functions to be aware they are being chained and act +-- upon the previous nodes results to avoid unnecessary processing or deduce +-- extra paramaters/arguments. +local function compose(...) + local queue = {} + + local nodes = {...} + + -- Allow placement.foo + (var == 42 and placement.bar) + if not nodes[2] then + return nodes[1] + end + + -- nodes[1] == self, nodes[2] == other + for _, w in ipairs(nodes) do + -- Build an execution queue + if w.context and w.context == "compose" then + for _, elem in ipairs(w.queue or {}) do + table.insert(queue, elem) + end + else + table.insert(queue, w) + end + end + + local ret = wrap_client(function(d, args, ...) + local rets = {} + local last_geo = nil + + -- As some functions may have to take into account results from + -- previously execued ones, add the `composition_results` hint. + args = setmetatable({composition_results=rets}, {__index=args}) + + -- Only apply the geometry once, not once per chain node, to do this, + -- Force the "pretend" argument and restore the original value for + -- the last node. + local pretend_real = args.pretend + + args.pretend = true + + for k, f in ipairs(queue) do + if k == #queue then + args.pretent = pretend_real + end + + local r = {f(d, args, ...)} + last_geo = r[1] or last_geo + args.override_geometry = last_geo + + -- Keep the return value, store one per context + if f.context then + -- When 2 composition queue are executed, merge the return values + if f.context == "compose" then + for k2,v in pairs(r) do + rets[k2] = v + end + else + rets[f.context] = r + end + end + end + + return last_geo, rets + end, "compose") + + ret.queue = queue + + return ret end -wrap_client = function(f) - return setmetatable({is_placement=true}, { - __call = function(_,...) return f(...) end, - __add = compose, -- Composition is usually defined as + - __mul = compose -- Make sense if you think of the functions as matrices - }) +wrap_client = function(f, context) + return setmetatable( + { + is_placement= true, + context = context, + }, + { + __call = function(_,...) return f(...) end, + __add = compose, -- Composition is usually defined as + + __mul = compose -- Make sense if you think of the functions as matrices + } + ) end local placement_private = {} @@ -116,7 +185,7 @@ local placement_private = {} local placement = setmetatable({}, { __index = placement_private, __newindex = function(_, k, f) - placement_private[k] = wrap_client(f) + placement_private[k] = wrap_client(f, k) end }) @@ -226,6 +295,12 @@ local function geometry_common(obj, args, new_geo, ignore_border_width) obj, (not args.pretend) and new_geo or nil, ignore_border_width ) + -- When using the placement composition along with the "pretend" + -- option, it is necessary to keep a "virtual" geometry. + if args.override_geometry then + dgeo = args.override_geometry + end + -- Apply the margins if args.margins then local delta = type(args.margins) == "table" and args.margins or { From 2f819df7f7751fa1dd1e69e449b3ad4e47b988b3 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Fri, 22 Apr 2016 00:34:28 -0400 Subject: [PATCH 12/39] awful.placement: Add 'offset' argument --- lib/awful/placement.lua | 34 ++++++++++++++++++++++++++++++++-- 1 file changed, 32 insertions(+), 2 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index acb02517..e2923405 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -49,6 +49,10 @@ -- -- **attach** (*boolean*): -- +-- **offset** (*table or number*): +-- +-- The offset(s) to apply to the new geometry. +-- -- **store_geometry** (*boolean*): -- -- Keep a single history of each type of placement. It can be restored using @@ -252,6 +256,32 @@ local function store_geometry(d, reqtype) data[d][reqtype].screen = d.screen end +--- Apply some modifications before applying the new geometry. +-- @tparam table new_geo The new geometry +-- @tparam table args The common arguments +-- @treturn table|nil The new geometry +local function fix_new_geometry(new_geo, args) + if args.pretend or not new_geo then return nil end + + local offset = args.offset or {} + + if type(offset) == "number" then + offset = { + x = offset, + y = offset, + width = offset, + height = offset, + } + end + + return { + x = new_geo.x and (new_geo.x + (offset.x or 0)), + y = new_geo.y and (new_geo.y + (offset.y or 0)), + width = new_geo.width and (new_geo.width + (offset.width or 0)), + height = new_geo.height and (new_geo.height + (offset.height or 0)), + } +end + --- Get the area covered by a drawin. -- @param d The drawin -- @tparam[opt=nil] table new_geo A new geometry @@ -283,7 +313,7 @@ local function geometry_common(obj, args, new_geo, ignore_border_width) -- It's a mouse if obj.coords then - local coords = (not args.pretend and new_geo) + local coords = fix_new_geometry(new_geo, args) and obj.coords(new_geo) or obj.coords() return {x=coords.x, y=coords.y, width=0, height=0} elseif obj.geometry then @@ -292,7 +322,7 @@ local function geometry_common(obj, args, new_geo, ignore_border_width) -- It is either a drawable or something that implement its API if type(geo) == "function" then local dgeo = area_common( - obj, (not args.pretend) and new_geo or nil, ignore_border_width + obj, fix_new_geometry(new_geo, args), ignore_border_width ) -- When using the placement composition along with the "pretend" From c286470c17c6c4e91fdf4f445099ab2734a49cf1 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Fri, 22 Apr 2016 00:35:46 -0400 Subject: [PATCH 13/39] awful.placement: Port `under_mouse` to the new argument syntax --- lib/awful/placement.lua | 22 ++++++++++++++++------ 1 file changed, 16 insertions(+), 6 deletions(-) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index e2923405..e19fc28d 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -721,14 +721,24 @@ end --- Place the client under the mouse. --@DOC_awful_placement_under_mouse_EXAMPLE@ --- @param c The client. +-- @tparam drawable d A drawable (like `client`, `mouse` or `wibox`) +-- @tparam[opt={}] table args Other arguments -- @treturn table The new geometry -function placement.under_mouse(c) - c = c or capi.client.focus - local c_geometry = area_common(c) +function placement.under_mouse(d, args) + args = add_context(args, "under_mouse") + d = d or capi.client.focus + local m_coords = capi.mouse.coords() - return c:geometry({ x = m_coords.x - c_geometry.width / 2, - y = m_coords.y - c_geometry.height / 2 }) + + local ngeo = geometry_common(d, args) + ngeo.x = m_coords.x - ngeo.width / 2 + ngeo.y = m_coords.y - ngeo.height / 2 + + local bw = d.border_width or 0 + ngeo.width = ngeo.width - 2*bw + ngeo.height = ngeo.height - 2*bw + + return ngeo end --- Place the client next to the mouse. From cdcfb913f1e631453e5101c04ae8a590a4334e55 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Fri, 22 Apr 2016 00:50:28 -0400 Subject: [PATCH 14/39] awful.mouse: Move clients using request::geometry --- lib/awful/mouse/init.lua | 85 ++++++++++++++-------------------------- 1 file changed, 30 insertions(+), 55 deletions(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 4b3575dd..fa94dbdf 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -10,12 +10,10 @@ -- Grab environment we need local layout = require("awful.layout") local tag = require("awful.tag") -local aclient = require("awful.client") local aplace = require("awful.placement") local awibox = require("awful.wibox") local util = require("awful.util") local type = type -local math = math local ipairs = ipairs local capi = { @@ -154,10 +152,13 @@ end --- Move a client. -- @param c The client to move, or the focused one if nil. -- @param snap The pixel to snap clients. --- @param finished_cb An optional callback function, that will be called --- when moving the client has been finished. The client --- that has been moved will be passed to that function. -function mouse.client.move(c, snap, finished_cb) +-- @param finished_cb Deprecated, do not use +function mouse.client.move(c, snap, finished_cb) --luacheck: no unused args + if finished_cb then + util.deprecated("The mouse.client.move `finished_cb` argument is no longer".. + " used, please use awful.mouse.resize.add_leave_callback(f, 'mouse.move')") + end + c = c or capi.client.focus if not c @@ -168,53 +169,16 @@ function mouse.client.move(c, snap, finished_cb) return end - local orig = c:geometry() - local m_c = capi.mouse.coords() - local dist_x = m_c.x - orig.x - local dist_y = m_c.y - orig.y - -- Only allow moving in the non-maximized directions - local fixed_x = c.maximized_horizontal - local fixed_y = c.maximized_vertical + -- Compute the offset + local coords = capi.mouse.coords() + local geo = aplace.centered(capi.mouse,{parent=c, pretend=true}) - capi.mousegrabber.run(function (_mouse) - if not c.valid then return false end + local offset = { + x = geo.x - coords.x, + y = geo.y - coords.y, + } - for _, v in ipairs(_mouse.buttons) do - if v then - local lay = layout.get(c.screen) - if lay == layout.suit.floating or c.floating then - local x = _mouse.x - dist_x - local y = _mouse.y - dist_y - c:geometry(mouse.client.snap(c, snap, x, y, fixed_x, fixed_y)) - elseif lay ~= layout.suit.magnifier then - -- Only move the client to the mouse - -- screen if the target screen is not - -- floating. - -- Otherwise, we move if via geometry. - if layout.get(capi.mouse.screen) == layout.suit.floating then - local x = _mouse.x - dist_x - local y = _mouse.y - dist_y - c:geometry(mouse.client.snap(c, snap, x, y, fixed_x, fixed_y)) - else - c.screen = capi.mouse.screen - end - if layout.get(c.screen) ~= layout.suit.floating then - local c_u_m = mouse.client_under_pointer() - if c_u_m and not c_u_m.floating then - if c_u_m ~= c then - c:swap(c_u_m) - end - end - end - end - return true - end - end - if finished_cb then - finished_cb(c) - end - return false - end, "fleur") + mouse.resize(c, "mouse.move", {placement=aplace.under_mouse, offset=offset}) end mouse.client.dragtotag = { } @@ -362,11 +326,22 @@ function mouse.resize_handler(c, context, hints) local lay = c.screen.selected_tag.layout if lay == layout.suit.floating or c.floating then + local offset = hints and hints.offset or {} + + if type(offset) == "number" then + offset = { + x = offset, + y = offset, + width = offset, + height = offset, + } + end + c:geometry { - x = hints.x, - y = hints.y, - width = hints.width, - height = hints.height, + x = hints.x + (offset.x or 0 ), + y = hints.y + (offset.y or 0 ), + width = hints.width + (offset.width or 0 ), + height = hints.height + (offset.height or 0 ), } elseif lay.resize_handler then lay.resize_handler(c, context, hints) From 688021d087082accd9348f8c08c4c038b923a52e Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 28 Apr 2016 16:46:09 -0400 Subject: [PATCH 15/39] ewmh: Ignore request::geometry for tiled clients It causes flicker --- lib/awful/ewmh.lua | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/lib/awful/ewmh.lua b/lib/awful/ewmh.lua index ffe04fd7..f63ef505 100644 --- a/lib/awful/ewmh.lua +++ b/lib/awful/ewmh.lua @@ -15,6 +15,7 @@ local math = math local util = require("awful.util") local aclient = require("awful.client") local aplace = require("awful.placement") +local asuit = require("awful.layout.suit") local ewmh = {} @@ -160,6 +161,13 @@ local context_mapper = { -- @tparam string context The context -- @tparam[opt={}] table hints The hints to pass to the handler function ewmh.geometry(c, context, hints) + local layout = c.screen.selected_tag and c.screen.selected_tag.layout or nil + + -- Setting the geometry wont work unless the client is floating. + if (not c.floating) and (not layout == asuit.floating) then + return + end + context = context or "" local original_context = context From b5a1a8d6e5230661f1733faf5f15017472ed3237 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Thu, 28 Apr 2016 16:47:29 -0400 Subject: [PATCH 16/39] layout: Handle tiled client request::geometry 'mouse.move' --- lib/awful/layout/init.lua | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/lib/awful/layout/init.lua b/lib/awful/layout/init.lua index 3d32ef08..8054e5de 100644 --- a/lib/awful/layout/init.lua +++ b/lib/awful/layout/init.lua @@ -241,6 +241,28 @@ capi.client.connect_signal("list", function() end end) +--- Default handler for tiled clients request::geometry with the `mouse.move` +-- context. +-- @tparam client c The client +-- @tparam string context The context +-- @tparam table hints Additional hints +function layout.move_handler(c, context, hints) --luacheck: no unused args + -- Quit if it isn't a mouse.move on a tiled layout, that's handled elsewhere + if c.floating then return end + if context ~= "mouse.move" then return end + local l = c.screen.selected_tag and c.screen.selected_tag.layout or nil + if l == layout.suit.floating then return end + + local c_u_m = capi.mouse.current_client + if c_u_m and not c_u_m.floating then + if c_u_m ~= c then + c:swap(c_u_m) + end + end +end + +capi.client.connect_signal("request::geometry", layout.move_handler) + return layout -- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80 From 2624370cc0fd0993fd5f79f98c2f518f58410776 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Fri, 22 Apr 2016 01:59:02 -0400 Subject: [PATCH 17/39] awful.mouse: Move snap into a submodule --- lib/awful/mouse/init.lua | 103 +------------------------------- lib/awful/mouse/snap.lua | 123 +++++++++++++++++++++++++++++++++++++++ 2 files changed, 125 insertions(+), 101 deletions(-) create mode 100644 lib/awful/mouse/snap.lua diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index fa94dbdf..33e0c721 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -25,7 +25,8 @@ local capi = } local mouse = { - resize = require("awful.mouse.resize") + resize = require("awful.mouse.resize"), + snap = require("awful.mouse.snap"), } mouse.client = {} @@ -49,106 +50,6 @@ function mouse.drawin_under_pointer() end end -local function snap_outside(g, sg, snap) - if g.x < snap + sg.x + sg.width and g.x > sg.x + sg.width then - g.x = sg.x + sg.width - elseif g.x + g.width < sg.x and g.x + g.width > sg.x - snap then - g.x = sg.x - g.width - end - if g.y < snap + sg.y + sg.height and g.y > sg.y + sg.height then - g.y = sg.y + sg.height - elseif g.y + g.height < sg.y and g.y + g.height > sg.y - snap then - g.y = sg.y - g.height - end - return g -end - -local function snap_inside(g, sg, snap) - local edgev = 'none' - local edgeh = 'none' - if math.abs(g.x) < snap + sg.x and g.x > sg.x then - edgev = 'left' - g.x = sg.x - elseif math.abs((sg.x + sg.width) - (g.x + g.width)) < snap then - edgev = 'right' - g.x = sg.x + sg.width - g.width - end - if math.abs(g.y) < snap + sg.y and g.y > sg.y then - edgeh = 'top' - g.y = sg.y - elseif math.abs((sg.y + sg.height) - (g.y + g.height)) < snap then - edgeh = 'bottom' - g.y = sg.y + sg.height - g.height - end - - -- What is the dominant dimension? - if g.width > g.height then - return g, edgeh - else - return g, edgev - end -end - ---- Snap a client to the closest client or screen edge. --- @param c The client to snap. --- @param snap The pixel to snap clients. --- @param x The client x coordinate. --- @param y The client y coordinate. --- @param fixed_x True if the client isn't allowed to move in the x direction. --- @param fixed_y True if the client isn't allowed to move in the y direction. -function mouse.client.snap(c, snap, x, y, fixed_x, fixed_y) - snap = snap or 8 - c = c or capi.client.focus - local cur_geom = c:geometry() - local geom = c:geometry() - geom.width = geom.width + (2 * c.border_width) - geom.height = geom.height + (2 * c.border_width) - local edge - geom.x = x or geom.x - geom.y = y or geom.y - - geom, edge = snap_inside(geom, capi.screen[c.screen].geometry, snap) - geom = snap_inside(geom, capi.screen[c.screen].workarea, snap) - - -- Allow certain windows to snap to the edge of the workarea. - -- Only allow docking to workarea for consistency/to avoid problems. - if c.dockable then - local struts = c:struts() - struts['left'] = 0 - struts['right'] = 0 - struts['top'] = 0 - struts['bottom'] = 0 - if edge ~= "none" and c.floating then - if edge == "left" or edge == "right" then - struts[edge] = cur_geom.width - elseif edge == "top" or edge == "bottom" then - struts[edge] = cur_geom.height - end - end - c:struts(struts) - end - - geom.x = geom.x - (2 * c.border_width) - geom.y = geom.y - (2 * c.border_width) - - for _, snapper in ipairs(aclient.visible(c.screen)) do - if snapper ~= c then - geom = snap_outside(geom, snapper:geometry(), snap) - end - end - - geom.width = geom.width - (2 * c.border_width) - geom.height = geom.height - (2 * c.border_width) - geom.x = geom.x + (2 * c.border_width) - geom.y = geom.y + (2 * c.border_width) - - -- It's easiest to undo changes afterwards if they're not allowed - if fixed_x then geom.x = cur_geom.x end - if fixed_y then geom.y = cur_geom.y end - - return geom -end - --- Move a client. -- @param c The client to move, or the focused one if nil. -- @param snap The pixel to snap clients. diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua new file mode 100644 index 00000000..aec5b834 --- /dev/null +++ b/lib/awful/mouse/snap.lua @@ -0,0 +1,123 @@ +--------------------------------------------------------------------------- +--- Mouse snapping related functions +-- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008 Julien Danjou +-- @release @AWESOME_VERSION@ +-- @submodule awful.mouse.snap +--------------------------------------------------------------------------- + +local aclient = require("awful.client") + +local capi = { + root = root, + mouse = mouse, + screen = screen, + client = client, + mousegrabber = mousegrabber, +} + +local module = {} + +local function snap_outside(g, sg, snap) + if g.x < snap + sg.x + sg.width and g.x > sg.x + sg.width then + g.x = sg.x + sg.width + elseif g.x + g.width < sg.x and g.x + g.width > sg.x - snap then + g.x = sg.x - g.width + end + if g.y < snap + sg.y + sg.height and g.y > sg.y + sg.height then + g.y = sg.y + sg.height + elseif g.y + g.height < sg.y and g.y + g.height > sg.y - snap then + g.y = sg.y - g.height + end + return g +end + +local function snap_inside(g, sg, snap) + local edgev = 'none' + local edgeh = 'none' + if math.abs(g.x) < snap + sg.x and g.x > sg.x then + edgev = 'left' + g.x = sg.x + elseif math.abs((sg.x + sg.width) - (g.x + g.width)) < snap then + edgev = 'right' + g.x = sg.x + sg.width - g.width + end + if math.abs(g.y) < snap + sg.y and g.y > sg.y then + edgeh = 'top' + g.y = sg.y + elseif math.abs((sg.y + sg.height) - (g.y + g.height)) < snap then + edgeh = 'bottom' + g.y = sg.y + sg.height - g.height + end + + -- What is the dominant dimension? + if g.width > g.height then + return g, edgeh + else + return g, edgev + end +end + +--- Snap a client to the closest client or screen edge. +-- @param c The client to snap. +-- @param snap The pixel to snap clients. +-- @param x The client x coordinate. +-- @param y The client y coordinate. +-- @param fixed_x True if the client isn't allowed to move in the x direction. +-- @param fixed_y True if the client isn't allowed to move in the y direction. +function module.snap(c, snap, x, y, fixed_x, fixed_y) + snap = snap or 8 + c = c or capi.client.focus + local cur_geom = c:geometry() + local geom = c:geometry() + geom.width = geom.width + (2 * c.border_width) + geom.height = geom.height + (2 * c.border_width) + local edge + geom.x = x or geom.x + geom.y = y or geom.y + + geom, edge = snap_inside(geom, capi.screen[c.screen].geometry, snap) + geom = snap_inside(geom, capi.screen[c.screen].workarea, snap) + + -- Allow certain windows to snap to the edge of the workarea. + -- Only allow docking to workarea for consistency/to avoid problems. + if c.dockable then + local struts = c:struts() + struts['left'] = 0 + struts['right'] = 0 + struts['top'] = 0 + struts['bottom'] = 0 + if edge ~= "none" and c.floating then + if edge == "left" or edge == "right" then + struts[edge] = cur_geom.width + elseif edge == "top" or edge == "bottom" then + struts[edge] = cur_geom.height + end + end + c:struts(struts) + end + + geom.x = geom.x - (2 * c.border_width) + geom.y = geom.y - (2 * c.border_width) + + for _, snapper in ipairs(aclient.visible(c.screen)) do + if snapper ~= c then + geom = snap_outside(geom, snapper:geometry(), snap) + end + end + + geom.width = geom.width - (2 * c.border_width) + geom.height = geom.height - (2 * c.border_width) + geom.x = geom.x + (2 * c.border_width) + geom.y = geom.y + (2 * c.border_width) + + -- It's easiest to undo changes afterwards if they're not allowed + if fixed_x then geom.x = cur_geom.x end + if fixed_y then geom.y = cur_geom.y end + + return geom +end + +return setmetatable(module, {__call = function(_, ...) return module.snap(...) end}) + From 86ca6222e25d1b32f2cd3387627d476ec61075a6 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Fri, 22 Apr 2016 02:16:59 -0400 Subject: [PATCH 18/39] awful.mouse.snap: Add a configurable default distance --- lib/awful/mouse/init.lua | 5 +++++ lib/awful/mouse/snap.lua | 8 +++++--- 2 files changed, 10 insertions(+), 3 deletions(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 33e0c721..867ca61f 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -32,6 +32,11 @@ local mouse = { mouse.client = {} mouse.wibox = {} +--- The default snap distance. +-- @tfield integer awful.mouse.snap.default_distance +-- @tparam[opt=8] integer default_distance +-- @see awful.mouse.snap + --- Get the client object under the pointer. -- @return The client object under the pointer, if one can be found. function mouse.client_under_pointer() diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua index aec5b834..746151c8 100644 --- a/lib/awful/mouse/snap.lua +++ b/lib/awful/mouse/snap.lua @@ -4,7 +4,7 @@ -- @author Julien Danjou <julien@danjou.info> -- @copyright 2008 Julien Danjou -- @release @AWESOME_VERSION@ --- @submodule awful.mouse.snap +-- @submodule awful.mouse --------------------------------------------------------------------------- local aclient = require("awful.client") @@ -17,7 +17,9 @@ local capi = { mousegrabber = mousegrabber, } -local module = {} +local module = { + default_distance = 8 +} local function snap_outside(g, sg, snap) if g.x < snap + sg.x + sg.width and g.x > sg.x + sg.width then @@ -67,7 +69,7 @@ end -- @param fixed_x True if the client isn't allowed to move in the x direction. -- @param fixed_y True if the client isn't allowed to move in the y direction. function module.snap(c, snap, x, y, fixed_x, fixed_y) - snap = snap or 8 + snap = snap or module.default_distance c = c or capi.client.focus local cur_geom = c:geometry() local geom = c:geometry() From baec2efe2d7fb6182f227171391969f4475a1258 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Fri, 22 Apr 2016 03:04:28 -0400 Subject: [PATCH 19/39] awful.mouse.move: Fix snapping support --- lib/awful/mouse/init.lua | 6 +++++- lib/awful/mouse/snap.lua | 11 +++++++++++ 2 files changed, 16 insertions(+), 1 deletion(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 867ca61f..5dbc774f 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -84,7 +84,11 @@ function mouse.client.move(c, snap, finished_cb) --luacheck: no unused args y = geo.y - coords.y, } - mouse.resize(c, "mouse.move", {placement=aplace.under_mouse, offset=offset}) + mouse.resize(c, "mouse.move", { + placement = aplace.under_mouse, + offset = offset, + snap = snap + }) end mouse.client.dragtotag = { } diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua index 746151c8..33c101c2 100644 --- a/lib/awful/mouse/snap.lua +++ b/lib/awful/mouse/snap.lua @@ -8,6 +8,7 @@ --------------------------------------------------------------------------- local aclient = require("awful.client") +local resize = require("awful.mouse.resize") local capi = { root = root, @@ -121,5 +122,15 @@ function module.snap(c, snap, x, y, fixed_x, fixed_y) return geom end +-- Enable edge snapping +resize.add_move_callback(function(c, geo, args) + if args and (args.snap == nil or args.snap) then + local ngeo = module.snap(c, args.snap, geo.x, geo.y) + ngeo.x = ngeo.x + (2 * c.border_width) + ngeo.y = ngeo.y + (2 * c.border_width) + return ngeo + end +end, "mouse.move") + return setmetatable(module, {__call = function(_, ...) return module.snap(...) end}) From 91cdc4a8990666fdb14a81496868de2714dc5d7d Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Fri, 22 Apr 2016 03:26:43 -0400 Subject: [PATCH 20/39] awful.mouse.snap: Support windows 7 style screen edges snap --- lib/awful/mouse/snap.lua | 61 +++++++++++++++++++++++++++++++++++++++- 1 file changed, 60 insertions(+), 1 deletion(-) diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua index 33c101c2..533d6ce2 100644 --- a/lib/awful/mouse/snap.lua +++ b/lib/awful/mouse/snap.lua @@ -9,6 +9,7 @@ local aclient = require("awful.client") local resize = require("awful.mouse.resize") +local aplace = require("awful.placement") local capi = { root = root, @@ -22,6 +23,54 @@ local module = { default_distance = 8 } +local function detect_screen_edges(c, snap) + local coords = capi.mouse.coords() + + local sg = c.screen.geometry + + local v, h = nil + + if math.abs(coords.x) <= snap + sg.x and coords.x >= sg.x then + h = "left" + end + + if math.abs((sg.x + sg.width) - coords.x) <= snap then + h = "right" + end + + if math.abs(coords.y) <= snap + sg.y and coords.y >= sg.y then + v = "top" + end + + if math.abs((sg.y + sg.height) - coords.y) <= snap then + v = "bottom" + end + + return v, h +end + +local current_snap = nil + +local function detect_areasnap(c, distance) + local v, h = detect_screen_edges(c, distance) + + if v and h then + current_snap = v.."_"..h + else + current_snap = v or h or nil + end + +end + +local function apply_areasnap(c, args) + if not current_snap then return end + + -- Remove the move offset + args.offset = {} + + return aplace[current_snap](c,{honor_workarea=true}) +end + local function snap_outside(g, sg, snap) if g.x < snap + sg.x + sg.width and g.x > sg.x + sg.width then g.x = sg.x + sg.width @@ -124,6 +173,12 @@ end -- Enable edge snapping resize.add_move_callback(function(c, geo, args) + -- Screen edge snapping (areosnap) + if args and (args.snap == nil or args.snap) then--TODO add a config option + detect_areasnap(c, 16) + end + + -- Snapping between clients if args and (args.snap == nil or args.snap) then local ngeo = module.snap(c, args.snap, geo.x, geo.y) ngeo.x = ngeo.x + (2 * c.border_width) @@ -132,5 +187,9 @@ resize.add_move_callback(function(c, geo, args) end end, "mouse.move") -return setmetatable(module, {__call = function(_, ...) return module.snap(...) end}) +-- Apply the aerosnap +resize.add_leave_callback(function(c, _, args) + return apply_areasnap(c, args) +end, "mouse.move") +return setmetatable(module, {__call = function(_, ...) return module.snap(...) end}) From 80ee4e9f5dadfb5a013786cce5d1a6b715c0e9db Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 23 Apr 2016 02:14:01 -0400 Subject: [PATCH 21/39] awful.mouse.snap: Add a visual cue --- lib/awful/mouse/init.lua | 12 +++++++ lib/awful/mouse/snap.lua | 75 +++++++++++++++++++++++++++++++++++----- 2 files changed, 78 insertions(+), 9 deletions(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 5dbc774f..b7d2dda1 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -37,6 +37,18 @@ mouse.wibox = {} -- @tparam[opt=8] integer default_distance -- @see awful.mouse.snap +--- The snap outline background color. +-- @beautiful beautiful.snap_bg +-- @tparam color|string|gradient|pattern color + +--- The snap outline width. +-- @beautiful beautiful.snap_border_width +-- @param integer + +--- The snap outline width. +-- @beautiful beautiful.snap_shape +-- @tparam function shape A `gears.shape` compatible function + --- Get the client object under the pointer. -- @return The client object under the pointer, if one can be found. function mouse.client_under_pointer() diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua index 533d6ce2..c197e7f5 100644 --- a/lib/awful/mouse/snap.lua +++ b/lib/awful/mouse/snap.lua @@ -7,9 +7,14 @@ -- @submodule awful.mouse --------------------------------------------------------------------------- -local aclient = require("awful.client") -local resize = require("awful.mouse.resize") -local aplace = require("awful.placement") +local aclient = require("awful.client") +local resize = require("awful.mouse.resize") +local aplace = require("awful.placement") +local wibox = require("wibox") +local beautiful = require("beautiful") +local color = require("gears.color") +local shape = require("gears.shape") +local cairo = require("lgi").cairo local capi = { root = root, @@ -23,6 +28,49 @@ local module = { default_distance = 8 } +local placeholder_w = nil + +local function show_placeholder(geo) + if not geo then + if placeholder_w then + placeholder_w.visible = false + end + return + end + + placeholder_w = placeholder_w or wibox { + ontop = true, + bg = color(beautiful.snap_bg or beautiful.bg_urgent or "#ff0000"), + } + + placeholder_w:geometry(geo) + + local img = cairo.ImageSurface(cairo.Format.A1, geo.width, geo.height) + local cr = cairo.Context(img) + + cr:set_operator(cairo.Operator.CLEAR) + cr:set_source_rgba(0,0,0,1) + cr:paint() + cr:set_operator(cairo.Operator.SOURCE) + cr:set_source_rgba(1,1,1,1) + + local line_width = beautiful.snap_border_width or 5 + cr:set_line_width(beautiful.xresources.apply_dpi(line_width)) + + local f = beautiful.snap_shape or function() + cr:translate(line_width,line_width) + shape.rounded_rect(cr,geo.width-2*line_width,geo.height-2*line_width, 10) + end + + f(cr, geo.width, geo.height) + + cr:stroke() + + placeholder_w.shape_bounding = img._native + + placeholder_w.visible = true +end + local function detect_screen_edges(c, snap) local coords = capi.mouse.coords() @@ -32,17 +80,13 @@ local function detect_screen_edges(c, snap) if math.abs(coords.x) <= snap + sg.x and coords.x >= sg.x then h = "left" - end - - if math.abs((sg.x + sg.width) - coords.x) <= snap then + elseif math.abs((sg.x + sg.width) - coords.x) <= snap then h = "right" end if math.abs(coords.y) <= snap + sg.y and coords.y >= sg.y then v = "top" - end - - if math.abs((sg.y + sg.height) - coords.y) <= snap then + elseif math.abs((sg.y + sg.height) - coords.y) <= snap then v = "bottom" end @@ -52,6 +96,7 @@ end local current_snap = nil local function detect_areasnap(c, distance) + local old_snap = current_snap local v, h = detect_screen_edges(c, distance) if v and h then @@ -60,6 +105,16 @@ local function detect_areasnap(c, distance) current_snap = v or h or nil end + -- Show the expected geometry outline + if current_snap ~= old_snap then + show_placeholder( + current_snap and aplace[current_snap](c, { + honor_workarea = true, + pretend = true + }) or nil + ) + end + end local function apply_areasnap(c, args) @@ -68,6 +123,8 @@ local function apply_areasnap(c, args) -- Remove the move offset args.offset = {} + placeholder_w.visible = false + return aplace[current_snap](c,{honor_workarea=true}) end From 1802e9e2fe3fdcebe0454ee9d6e852586d349015 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 23 Apr 2016 04:33:46 -0400 Subject: [PATCH 22/39] awful.placement: Add a `scale` function --- lib/awful/placement.lua | 52 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 52 insertions(+) diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index e19fc28d..6e5df3f3 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -1055,6 +1055,58 @@ for _, v in ipairs {"vertically", "horizontally"} do end end +--- Scale the drawable by either a relative or absolute percent. +-- +-- Valid args: +-- +-- **to_percent** : A number between 0 and 1. It represent a percent related to +-- the parent geometry. +-- **by_percent** : A number between 0 and 1. It represent a percent related to +-- the current size. +-- **direction**: Nothing or "left", "right", "up", "down". +-- +-- @tparam[opt=client.focus] drawable d A drawable (like `client` or `wibox`) +-- @tparam[opt={}] table args The arguments +-- @treturn table The new geometry +function placement.scale(d, args) + args = add_context(args, "scale_to_percent") + d = d or capi.client.focus + + local to_percent = args.to_percent + local by_percent = args.by_percent + + local percent = to_percent or by_percent + + local direction = args.direction + + local sgeo = get_parent_geometry(d, args) + local ngeo = geometry_common(d, args, nil) + + local old_area = {width = ngeo.width, height = ngeo.height} + + if (not direction) or direction == "left" or direction == "right" then + ngeo.width = (to_percent and sgeo or ngeo).width*percent + + if direction == "left" then + ngeo.x = ngeo.x - (ngeo.width - old_area.width) + end + end + + if (not direction) or direction == "up" or direction == "down" then + ngeo.height = (to_percent and sgeo or ngeo).height*percent + + if direction == "up" then + ngeo.y = ngeo.y - (ngeo.height - old_area.height) + end + end + + geometry_common(d, args, ngeo) + + attach(d, placement.maximize, args) + + return ngeo +end + ---@DOC_awful_placement_maximize_vertically_EXAMPLE@ ---@DOC_awful_placement_maximize_horizontally_EXAMPLE@ From 07e1b975a70ec2c1e6e76b6932ba5a1fd617b562 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 23 Apr 2016 20:08:42 -0400 Subject: [PATCH 23/39] mouse.snap: Use a composed placement function --- lib/awful/mouse/snap.lua | 36 ++++++++++++++++++++++++++---------- 1 file changed, 26 insertions(+), 10 deletions(-) diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua index c197e7f5..26174ac8 100644 --- a/lib/awful/mouse/snap.lua +++ b/lib/awful/mouse/snap.lua @@ -71,6 +71,14 @@ local function show_placeholder(geo) placeholder_w.visible = true end +local function build_placement(snap, axis) + return aplace.scale + + aplace[snap] + + ( + axis and aplace["maximize_"..axis] or nil + ) +end + local function detect_screen_edges(c, snap) local coords = capi.mouse.coords() @@ -93,7 +101,7 @@ local function detect_screen_edges(c, snap) return v, h end -local current_snap = nil +local current_snap, current_axis = nil local function detect_areasnap(c, distance) local old_snap = current_snap @@ -105,15 +113,20 @@ local function detect_areasnap(c, distance) current_snap = v or h or nil end + if old_snap == current_snap then return end + + current_axis = ((v and not h) and "horizontally") + or ((h and not v) and "vertically") + or nil + -- Show the expected geometry outline - if current_snap ~= old_snap then - show_placeholder( - current_snap and aplace[current_snap](c, { - honor_workarea = true, - pretend = true - }) or nil - ) - end + show_placeholder( + current_snap and build_placement(current_snap, current_axis)(c, { + to_percent = 0.5, + honor_workarea = true, + pretend = true + }) or nil + ) end @@ -125,7 +138,10 @@ local function apply_areasnap(c, args) placeholder_w.visible = false - return aplace[current_snap](c,{honor_workarea=true}) + return build_placement(current_snap, current_axis)(c,{ + to_percent = 0.5, + honor_workarea = true, + }) end local function snap_outside(g, sg, snap) From bb0a2addb49487e77e6f78f209a101aa8b693e2c Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Mon, 11 Apr 2016 05:15:57 -0400 Subject: [PATCH 24/39] tests: Add an option to disable @usage ldoc tag --- tests/examples/CMakeLists.txt | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/tests/examples/CMakeLists.txt b/tests/examples/CMakeLists.txt index 71469153..0c40ede3 100644 --- a/tests/examples/CMakeLists.txt +++ b/tests/examples/CMakeLists.txt @@ -46,7 +46,6 @@ function(escape_string variable content escaped_content line_prefix) if(variable MATCHES "--DOC_HIDE_ALL") return() endif() - string(REGEX REPLACE "\n" ";" var_lines "${variable}") set(tmp_output ${content}) @@ -201,10 +200,17 @@ function(run_test test_path namespace template escaped_content) # Only add it if there is something to display. if(NOT ${TEST_CODE} STREQUAL "\n--") + # Do not use the @usage tag, use 4 spaces + file(READ ${test_path} tmp_content) + if(NOT tmp_content MATCHES "--DOC_NO_USAGE") + set(DOC_PREFIX "@usage") + endif() + escape_string( - " @usage" + " ${DOC_PREFIX}" "${TEST_DOC_CONTENT}" TEST_DOC_CONTENT "" ) + set(TEST_DOC_CONTENT "${TEST_DOC_CONTENT}${TEST_CODE}") endif() From f0546b900d3c0815fd98bac6cb91b3cf53c23b2d Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 23 Apr 2016 23:05:11 -0400 Subject: [PATCH 25/39] tests: Improve the placement compositing tests --- lib/awful/placement.lua | 35 +++++++++++++-------- tests/examples/awful/placement/compose.lua | 12 ++++--- tests/examples/awful/placement/compose2.lua | 19 +++++++++++ 3 files changed, 48 insertions(+), 18 deletions(-) create mode 100644 tests/examples/awful/placement/compose2.lua diff --git a/lib/awful/placement.lua b/lib/awful/placement.lua index 6e5df3f3..426da79e 100644 --- a/lib/awful/placement.lua +++ b/lib/awful/placement.lua @@ -9,14 +9,16 @@ -- * Turn each function into an API with various common customization parameters. -- * Re-use the same functions for the `mouse`, `client`s, `screen`s and `wibox`es -- --- +-- +--

Compositing

-- -- It is possible to compose placement function using the `+` or `*` operator: -- --- local f = (awful.placement.right + awful.placement.left) --- f(client.focus) +--@DOC_awful_placement_compose_EXAMPLE@ -- --- ### Common arguments +--@DOC_awful_placement_compose2_EXAMPLE@ +-- +--

Common arguments

-- -- **pretend** (*boolean*): -- @@ -86,6 +88,7 @@ local capi = local client = require("awful.client") local layout = require("awful.layout") local a_screen = require("awful.screen") +local util = require("awful.util") local dpi = require("beautiful").xresources.apply_dpi local function get_screen(s) @@ -137,7 +140,7 @@ local function compose(...) for k, f in ipairs(queue) do if k == #queue then - args.pretent = pretend_real + args.pretend = pretend_real or false end local r = {f(d, args, ...)} @@ -286,11 +289,19 @@ end -- @param d The drawin -- @tparam[opt=nil] table new_geo A new geometry -- @tparam[opt=false] boolean ignore_border_width Ignore the border +-- @tparam table args the method arguments -- @treturn The drawin's area. -local function area_common(d, new_geo, ignore_border_width) +local function area_common(d, new_geo, ignore_border_width, args) -- The C side expect no arguments, nil isn't valid local geometry = new_geo and d:geometry(new_geo) or d:geometry() local border = ignore_border_width and 0 or d.border_width or 0 + + -- When using the placement composition along with the "pretend" + -- option, it is necessary to keep a "virtual" geometry. + if args and args.override_geometry then + geometry = util.table.clone(args.override_geometry) + end + geometry.width = geometry.width + 2 * border geometry.height = geometry.height + 2 * border return geometry @@ -322,15 +333,9 @@ local function geometry_common(obj, args, new_geo, ignore_border_width) -- It is either a drawable or something that implement its API if type(geo) == "function" then local dgeo = area_common( - obj, fix_new_geometry(new_geo, args), ignore_border_width + obj, fix_new_geometry(new_geo, args), ignore_border_width, args ) - -- When using the placement composition along with the "pretend" - -- option, it is necessary to keep a "virtual" geometry. - if args.override_geometry then - dgeo = args.override_geometry - end - -- Apply the margins if args.margins then local delta = type(args.margins) == "table" and args.margins or { @@ -1100,6 +1105,10 @@ function placement.scale(d, args) end end + local bw = d.border_width or 0 + ngeo.width = ngeo.width - 2*bw + ngeo.height = ngeo.height - 2*bw + geometry_common(d, args, ngeo) attach(d, placement.maximize, args) diff --git a/tests/examples/awful/placement/compose.lua b/tests/examples/awful/placement/compose.lua index 28098677..f6710762 100644 --- a/tests/examples/awful/placement/compose.lua +++ b/tests/examples/awful/placement/compose.lua @@ -1,9 +1,11 @@ -screen[1]._resize {width = 128, height = 96} --DOC_HIDE +screen[1]._resize {x = 175, width = 128, height = 96} --DOC_NO_USAGE --DOC_HIDE local awful = {placement = require("awful.placement")} --DOC_HIDE -local c = client.gen_fake {x = 45, y = 35, width=40, height=30} --DOC_HIDE +local c = client.gen_fake {x = 220, y = 35, width=40, height=30} --DOC_HIDE -local f = (awful.placement.right + awful.placement.left) -f(client.focus) + -- "right" will be replaced by "left" + local f = (awful.placement.right + awful.placement.left) + f(client.focus) -assert(c.x == 0 and c.y==screen[1].geometry.height/2-30/2-c.border_width--DOC_HIDE +local sg = screen[1].geometry--DOC_HIDE +assert(c.x == sg.x and c.y==sg.height/2-30/2-c.border_width--DOC_HIDE and c.width==40 and c.height==30)--DOC_HIDE diff --git a/tests/examples/awful/placement/compose2.lua b/tests/examples/awful/placement/compose2.lua new file mode 100644 index 00000000..68cbf5b3 --- /dev/null +++ b/tests/examples/awful/placement/compose2.lua @@ -0,0 +1,19 @@ +screen[1]._resize {x = 175, width = 128, height = 96} --DOC_NO_USAGE --DOC_HIDE +local awful = {placement = require("awful.placement")} --DOC_HIDE +local c = client.gen_fake {x = 220, y = 35, width=40, height=30} --DOC_HIDE + + + -- Simulate Windows 7 "edge snap" (also called aero snap) feature + local axis = "vertically" + + local f = awful.placement.scale + + awful.placement.left + + (axis and awful.placement["maximize_"..axis] or nil) + + local geo = f(client.focus, {honor_workarea=true, to_percent = 0.5}) + +local wa = screen[1].workarea--DOC_HIDE +assert(c.x == wa.x and geo.x == wa.x)--DOC_HIDE +assert(c.y == wa.y) --DOC_HIDE +assert(c.width == wa.width/2 - 2*c.border_width)--DOC_HIDE +assert(c.height == wa.height - 2*c.border_width)--DOC_HIDE From 424181248a12ce6ec016bcbc7f6838a9c74c7045 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 20:55:29 -0400 Subject: [PATCH 26/39] mouse: Add property miss handler. Just like luaobject has, but for "static" classes. --- mouse.c | 76 ++++++++++++++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 72 insertions(+), 4 deletions(-) diff --git a/mouse.c b/mouse.c index 5c80b17b..df149a02 100644 --- a/mouse.c +++ b/mouse.c @@ -33,11 +33,15 @@ #include "math.h" #include "common/util.h" #include "common/xutil.h" +#include "common/luaclass.h" #include "globalconf.h" #include "objects/client.h" #include "objects/drawin.h" #include "objects/screen.h" +static int miss_index_handler = LUA_REFNIL; +static int miss_newindex_handler = LUA_REFNIL; + /** Mouse library. * * @table mouse @@ -119,6 +123,39 @@ mouse_warp_pointer(xcb_window_t window, int16_t x, int16_t y) 0, 0, 0, 0, x, y); } +/** + * Allow the a Lua handler to be implemented for custom properties and + * functions. + * \param L A lua state + * \param handler A function on the LUA_REGISTRYINDEX + */ +static int +luaA_mouse_call_handler(lua_State *L, int handler) +{ + int nargs = lua_gettop(L); + + /* Push error handling function and move it before args */ + lua_pushcfunction(L, luaA_dofunction_error); + lua_insert(L, - nargs - 1); + int error_func_pos = 1; + + /* push function and move it before args */ + lua_rawgeti(L, LUA_REGISTRYINDEX, handler); + lua_insert(L, - nargs - 1); + + if(lua_pcall(L, nargs, LUA_MULTRET, error_func_pos)) + { + warn("%s", lua_tostring(L, -1)); + /* Remove error function and error string */ + lua_pop(L, 2); + return 0; + } + /* Remove error function */ + lua_remove(L, error_func_pos); + + return lua_gettop(L); +} + /** Mouse library. * \param L The Lua VM state. * \return The number of elements pushed on stack. @@ -133,8 +170,13 @@ luaA_mouse_index(lua_State *L) int16_t mouse_x, mouse_y; /* attr is not "screen"?! */ - if (A_STRNEQ(attr, "screen")) - return luaA_default_index(L); + if (A_STRNEQ(attr, "screen")) { + if (miss_index_handler != LUA_REFNIL) { + return luaA_mouse_call_handler(L, miss_index_handler); + } + else + return luaA_default_index(L); + } if (!mouse_query_pointer_root(&mouse_x, &mouse_y, NULL, NULL)) { @@ -162,8 +204,14 @@ luaA_mouse_newindex(lua_State *L) const char *attr = luaL_checkstring(L, 2); screen_t *screen; - if (A_STRNEQ(attr, "screen")) - return luaA_default_newindex(L); + if (A_STRNEQ(attr, "screen")) { + /* Call the lua mouse property handler */ + if (miss_newindex_handler != LUA_REFNIL) { + return luaA_mouse_call_handler(L, miss_newindex_handler); + } + else + return luaA_default_newindex(L); + } screen = luaA_checkscreen(L, 3); mouse_warp_pointer(globalconf.screen->root, screen->geometry.x, screen->geometry.y); @@ -271,12 +319,32 @@ luaA_mouse_object_under_pointer(lua_State *L) return 0; } +/** + * Add a custom property handler (getter). + */ +static int +luaA_mouse_set_index_miss_handler(lua_State *L) +{ + return luaA_registerfct(L, 1, &miss_index_handler); +} + +/** + * Add a custom property handler (setter). + */ +static int +luaA_mouse_set_newindex_miss_handler(lua_State *L) +{ + return luaA_registerfct(L, 1, &miss_newindex_handler); +} + const struct luaL_Reg awesome_mouse_methods[] = { { "__index", luaA_mouse_index }, { "__newindex", luaA_mouse_newindex }, { "coords", luaA_mouse_coords }, { "object_under_pointer", luaA_mouse_object_under_pointer }, + { "set_index_miss_handler", luaA_mouse_set_index_miss_handler}, + { "set_newindex_miss_handler", luaA_mouse_set_newindex_miss_handler}, { NULL, NULL } }; const struct luaL_Reg awesome_mouse_meta[] = From 0b6cd90dce4794ff744f69fd06d2ce93380b93f4 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 21:00:38 -0400 Subject: [PATCH 27/39] mouse: Implement the Lua custom property handler --- lib/awful/mouse/init.lua | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index b7d2dda1..fe154be4 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -29,6 +29,7 @@ local mouse = { snap = require("awful.mouse.snap"), } +mouse.object = {} mouse.client = {} mouse.wibox = {} @@ -276,6 +277,25 @@ capi.client.connect_signal("request::geometry", mouse.resize_handler) -- Set the cursor at startup capi.root.cursor("left_ptr") +-- Implement the custom property handler +local props = {} + +capi.mouse.set_newindex_miss_handler(function(_,key,value) + if mouse.object["set_"..key] then + mouse.object["set_"..key](value) + else + props[key] = value + end +end) + +capi.mouse.set_index_miss_handler(function(_,key) + if mouse.object["get_"..key] then + return mouse.object["get_"..key]() + else + return props[key] + end +end) + return mouse -- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80 From e8dd707508afb0de9b4d2d2bef8e3787810076c7 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 21:04:27 -0400 Subject: [PATCH 28/39] tests: The the mouse miss handler --- tests/test-miss-handlers.lua | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/tests/test-miss-handlers.lua b/tests/test-miss-handlers.lua index f97d089b..fe61a860 100644 --- a/tests/test-miss-handlers.lua +++ b/tests/test-miss-handlers.lua @@ -1,5 +1,6 @@ -- Test set_{,new}index_miss_handler +local mouse = mouse local class = tag local obj = class({}) local handler = require("gears.object.properties") @@ -30,4 +31,8 @@ assert(not obj.key) obj.key = 1337 assert(obj.key == 1337) +-- The the custom mouse handler +mouse.foo = "bar" +assert(mouse.foo == "bar") + require("_runner").run_steps({ function() return true end }) From 800b8c7d4aa2cd7f66da94f80b11508cff76e7e2 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 22:09:14 -0400 Subject: [PATCH 29/39] doc: Merge awful.mouse and mouse documentation pages Just like client, tag and screen, this is to avoid new users some confusing implementation details. --- docs/aliases/awful_mouse.lua | 6 ++++++ docs/config.ld | 1 + lib/awful/mouse/init.lua | 14 +++++++++++--- lib/awful/mouse/resize.lua | 10 ++++++++-- lib/awful/mouse/snap.lua | 3 ++- mouse.c | 27 ++++++++++----------------- 6 files changed, 38 insertions(+), 23 deletions(-) create mode 100644 docs/aliases/awful_mouse.lua diff --git a/docs/aliases/awful_mouse.lua b/docs/aliases/awful_mouse.lua new file mode 100644 index 00000000..9b339a6f --- /dev/null +++ b/docs/aliases/awful_mouse.lua @@ -0,0 +1,6 @@ +--------------------------------------------------------------------------- +--- This module is deprecated, use `mouse` +-- =============================== +-- +-- @module awful.mouse +--------------------------------------------------------------------------- diff --git a/docs/config.ld b/docs/config.ld index b220b720..8ff7c13e 100644 --- a/docs/config.ld +++ b/docs/config.ld @@ -67,6 +67,7 @@ file = { '../docs/aliases/awful_client.lua', '../docs/aliases/awful_screen.lua', '../docs/aliases/awful_tag.lua', + '../docs/aliases/awful_mouse.lua', exclude = { -- exclude these modules, as they do not contain any written -- documentation diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index fe154be4..4036bfab 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -4,7 +4,7 @@ -- @author Julien Danjou <julien@danjou.info> -- @copyright 2008 Julien Danjou -- @release @AWESOME_VERSION@ --- @module awful.mouse +-- @module mouse --------------------------------------------------------------------------- -- Grab environment we need @@ -51,6 +51,7 @@ mouse.wibox = {} -- @tparam function shape A `gears.shape` compatible function --- Get the client object under the pointer. +-- @deprecated awful.mouse.client_under_pointer -- @return The client object under the pointer, if one can be found. function mouse.client_under_pointer() local obj = capi.mouse.object_under_pointer() @@ -60,6 +61,7 @@ function mouse.client_under_pointer() end --- Get the drawin object under the pointer. +-- @deprecated awful.mouse.drawin_under_pointer -- @return The drawin object under the pointer, if one can be found. function mouse.drawin_under_pointer() local obj = capi.mouse.object_under_pointer() @@ -69,6 +71,7 @@ function mouse.drawin_under_pointer() end --- Move a client. +-- @function awful.mouse.client.move -- @param c The client to move, or the focused one if nil. -- @param snap The pixel to snap clients. -- @param finished_cb Deprecated, do not use @@ -106,7 +109,8 @@ end mouse.client.dragtotag = { } ---- Move a client to a tag by dragging it onto the left / right side of the screen +--- Move a client to a tag by dragging it onto the left / right side of the screen. +-- @function awful.mouse.client.dragtotag.border -- @param c The client to move function mouse.client.dragtotag.border(c) capi.mousegrabber.run(function (_mouse) @@ -146,7 +150,8 @@ function mouse.client.dragtotag.border(c) end, "fleur") end ---- Move the wibox under the cursor +--- Move the wibox under the cursor. +-- @function awful.mouse.wibox.move --@param w The wibox to move, or none to use that under the pointer function mouse.wibox.move(w) w = w or mouse.wibox_under_pointer() @@ -187,6 +192,7 @@ function mouse.wibox.move(w) end --- Get a client corner coordinates. +-- @deprecated awful.mouse.client.corner -- @tparam[opt=client.focus] client c The client to get corner from, focused one by default. -- @tparam string corner The corner to use: auto, top_left, top_right, bottom_left, -- bottom_right, left, right, top bottom. Default is auto, and auto find the @@ -215,6 +221,7 @@ function mouse.client.corner(c, corner) end --- Resize a client. +-- @function awful.mouse.client.resize -- @param c The client to resize, or the focused one by default. -- @tparam string corner The corner to grab on resize. Auto detected by default. -- @tparam[opt={}] table args A set of `awful.placement` arguments @@ -239,6 +246,7 @@ function mouse.client.resize(c, corner, args) end --- Default handler for `request::geometry` signals with `mouse.resize` context. +-- @function awful.mouse.resize_handler -- @tparam client c The client -- @tparam string context The context -- @tparam[opt={}] table hints The hints to pass to the handler diff --git a/lib/awful/mouse/resize.lua b/lib/awful/mouse/resize.lua index 3662358e..ed95fbbd 100644 --- a/lib/awful/mouse/resize.lua +++ b/lib/awful/mouse/resize.lua @@ -7,7 +7,7 @@ -- @author Emmanuel Lepage Vallee <elv1313@gmail.com> -- @copyright 2016 Emmanuel Lepage Vallee -- @release @AWESOME_VERSION@ --- @submodule awful.mouse +-- @submodule mouse --------------------------------------------------------------------------- local aplace = require("awful.placement") @@ -27,6 +27,7 @@ local callbacks = {enter={}, move={}, leave={}} -- -- Some clients, such as XTerm, may lose information if resized too often. -- +-- @function awful.mouse.resize.set_mode -- @tparam string m The mode function module.set_mode(m) assert(m == "live" or m == "after") @@ -35,6 +36,7 @@ end --- Add a initialization callback. -- This callback will be executed before the mouse grabbing start +-- @function awful.mouse.resize.add_enter_callback -- @tparam function cb The callback (or nil) -- @tparam[default=other] string context The callback context function module.add_enter_callback(cb, context) @@ -46,6 +48,7 @@ end --- Add a "move" callback. -- This callback is executed in "after" mode (see `set_mode`) instead of -- applying the operation. +-- @function awful.mouse.resize.add_move_callback -- @tparam function cb The callback (or nil) -- @tparam[default=other] string context The callback context function module.add_move_callback(cb, context) @@ -56,6 +59,7 @@ end --- Add a "leave" callback -- This callback is executed just before the `mousegrabber` stop +-- @function awful.mouse.resize.add_leave_callback -- @tparam function cb The callback (or nil) -- @tparam[default=other] string context The callback context function module.add_leave_callback(cb, context) @@ -64,7 +68,7 @@ function module.add_leave_callback(cb, context) table.insert(callbacks.leave[context], cb) end --- Resize, the drawable +-- Resize, the drawable. -- -- Valid `args` are: -- @@ -73,9 +77,11 @@ end -- * *leave_callback*: A function called before the `mousegrabber` is released -- * *mode*: The resize mode -- +-- @function awful.mouse.resize -- @tparam client client A client -- @tparam[default=mouse.resize] string context The resizing context -- @tparam[opt={}] table args A set of `awful.placement` arguments + local function handler(_, client, context, args) --luacheck: no unused_args args = args or {} context = context or "mouse.resize" diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua index 26174ac8..76974a55 100644 --- a/lib/awful/mouse/snap.lua +++ b/lib/awful/mouse/snap.lua @@ -4,7 +4,7 @@ -- @author Julien Danjou <julien@danjou.info> -- @copyright 2008 Julien Danjou -- @release @AWESOME_VERSION@ --- @submodule awful.mouse +-- @submodule mouse --------------------------------------------------------------------------- local aclient = require("awful.client") @@ -185,6 +185,7 @@ local function snap_inside(g, sg, snap) end --- Snap a client to the closest client or screen edge. +-- @function awful.mouse.snap -- @param c The client to snap. -- @param snap The pixel to snap clients. -- @param x The client x coordinate. diff --git a/mouse.c b/mouse.c index df149a02..615af62e 100644 --- a/mouse.c +++ b/mouse.c @@ -42,22 +42,10 @@ static int miss_index_handler = LUA_REFNIL; static int miss_newindex_handler = LUA_REFNIL; -/** Mouse library. - * - * @table mouse - */ - /** * The `screen` under the cursor - * @field screen - */ - -/** A table with X and Y coordinates. - * @field x X coordinate. - * @field y Y coordinate. - * @field buttons Table containing the status of buttons, e.g. field [1] is true - * when button 1 is pressed. - * @table coords_table + * @property screen + * @param screen */ /** Get the pointer position. @@ -251,11 +239,16 @@ luaA_mouse_pushstatus(lua_State *L, int x, int y, uint16_t mask) /** Get or set the mouse coords. * - * @tparam coords_table coords_table None or a table with x and y keys as mouse + * @tparam[opt=nil] table coords_table None or a table with x and y keys as mouse * coordinates. - * @tparam boolean silent Disable mouse::enter or mouse::leave events that + * @tparam[opt=nil] integer coords_table.x The mouse horizontal position + * @tparam[opt=nil] integer coords_table.y The mouse vertical position + * @tparam[opt=false] boolean silent Disable mouse::enter or mouse::leave events that * could be triggered by the pointer when moving. - * @treturn coords_table A table with mouse coordinates. + * @treturn integer table.x The horizontal position + * @treturn integer table.y The vertical position + * @treturn table table.buttons Table containing the status of buttons, e.g. field [1] is true + * when button 1 is pressed. * @function coords */ static int From fc2d359722a17a8e4c8cfef5eb318feabae642af Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 22:21:35 -0400 Subject: [PATCH 30/39] doc: Add a coords example This forced the `coords()` documentation to be moved to Lua. Keeping it in the C file caused yet another example CMake hack. --- lib/awful/mouse/init.lua | 17 +++++++++++++++++ mouse.c | 15 +-------------- tests/examples/awful/mouse/coords.lua | 13 +++++++++++++ 3 files changed, 31 insertions(+), 14 deletions(-) create mode 100644 tests/examples/awful/mouse/coords.lua diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 4036bfab..6c6794ba 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -304,6 +304,23 @@ capi.mouse.set_index_miss_handler(function(_,key) end end) +--- Get or set the mouse coords. +-- +--@DOC_awful_mouse_coords_EXAMPLE@ +-- +-- @tparam[opt=nil] table coords_table None or a table with x and y keys as mouse +-- coordinates. +-- @tparam[opt=nil] integer coords_table.x The mouse horizontal position +-- @tparam[opt=nil] integer coords_table.y The mouse vertical position +-- @tparam[opt=false] boolean silent Disable mouse::enter or mouse::leave events that +-- could be triggered by the pointer when moving. +-- @treturn integer table.x The horizontal position +-- @treturn integer table.y The vertical position +-- @treturn table table.buttons Table containing the status of buttons, e.g. field [1] is true +-- when button 1 is pressed. +-- @function mouse.coords + + return mouse -- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80 diff --git a/mouse.c b/mouse.c index 615af62e..57c4a8ba 100644 --- a/mouse.c +++ b/mouse.c @@ -237,20 +237,7 @@ luaA_mouse_pushstatus(lua_State *L, int x, int y, uint16_t mask) return 1; } -/** Get or set the mouse coords. - * - * @tparam[opt=nil] table coords_table None or a table with x and y keys as mouse - * coordinates. - * @tparam[opt=nil] integer coords_table.x The mouse horizontal position - * @tparam[opt=nil] integer coords_table.y The mouse vertical position - * @tparam[opt=false] boolean silent Disable mouse::enter or mouse::leave events that - * could be triggered by the pointer when moving. - * @treturn integer table.x The horizontal position - * @treturn integer table.y The vertical position - * @treturn table table.buttons Table containing the status of buttons, e.g. field [1] is true - * when button 1 is pressed. - * @function coords - */ +/* documented in lib/awful/mouse/init.lua */ static int luaA_mouse_coords(lua_State *L) { diff --git a/tests/examples/awful/mouse/coords.lua b/tests/examples/awful/mouse/coords.lua new file mode 100644 index 00000000..b4b0ed95 --- /dev/null +++ b/tests/examples/awful/mouse/coords.lua @@ -0,0 +1,13 @@ +screen[1]._resize {x = 175, width = 128, height = 96} --DOC_HIDE +mouse.coords {x=175+60,y=60} --DOC_HIDE + + -- Get the position +print(mouse.coords().x) + + -- Change the position +mouse.coords { + x = 185, + y = 10 +} + +mouse.push_history() --DOC_HIDE From 47fffb5eee5be4c427f3120a070d594d365d385d Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 22:33:44 -0400 Subject: [PATCH 31/39] mouse: Add current_client property It is called wibox instead of drawin because from the user point of view, it will be a wibox anyway. --- lib/awful/mouse/init.lua | 42 +++++++++++++++++++++++++++------------- 1 file changed, 29 insertions(+), 13 deletions(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 6c6794ba..67468d74 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -53,21 +53,11 @@ mouse.wibox = {} --- Get the client object under the pointer. -- @deprecated awful.mouse.client_under_pointer -- @return The client object under the pointer, if one can be found. +-- @see current_client function mouse.client_under_pointer() - local obj = capi.mouse.object_under_pointer() - if type(obj) == "client" then - return obj - end -end + util.deprecated("Use mouse.current_client instead of awful.mouse.client_under_pointer()") ---- Get the drawin object under the pointer. --- @deprecated awful.mouse.drawin_under_pointer --- @return The drawin object under the pointer, if one can be found. -function mouse.drawin_under_pointer() - local obj = capi.mouse.object_under_pointer() - if type(obj) == "drawin" then - return obj - end + return mouse.object.get_current_client() end --- Move a client. @@ -280,6 +270,32 @@ function mouse.resize_handler(c, context, hints) end end +--- Get the client currently under the mouse cursor. +-- @property current_client +-- @tparam client|nil The client + +function mouse.object.get_current_client() + local obj = capi.mouse.object_under_pointer() + if type(obj) == "client" then + return obj + end +end + +function mouse.object.set_current_client() end + +--- Get the wibox currently under the mouse cursor. +-- @property current_wibox +-- @tparam wibox|nil The wibox + +function mouse.object.get_current_wibox() + local obj = capi.mouse.object_under_pointer() + if type(obj) == "drawin" then + return obj + end +end + +function mouse.object.set_current_wibox() end + capi.client.connect_signal("request::geometry", mouse.resize_handler) -- Set the cursor at startup From e78ad709e7680dca47b193463d0c6d7ac8b0f894 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 22:42:32 -0400 Subject: [PATCH 32/39] doc: Move request:: handler to their own documentation section --- docs/config.ld | 2 ++ lib/awful/ewmh.lua | 7 ++++++- lib/awful/mouse/init.lua | 2 +- 3 files changed, 9 insertions(+), 2 deletions(-) diff --git a/docs/config.ld b/docs/config.ld index 8ff7c13e..19886170 100644 --- a/docs/config.ld +++ b/docs/config.ld @@ -35,6 +35,8 @@ new_type("function", "Functions") new_type("property", "Object properties", false, "Type") -- New type for signals new_type("signal", "Signals", false, "Arguments") +-- New type for signals connections +new_type("signalhandler", "Request handlers", false, "Arguments") -- Allow objects to define a set of beautiful properties affecting them new_type("beautiful", "Theme variables", false, "Type") -- Put deprecated methods in their own section diff --git a/lib/awful/ewmh.lua b/lib/awful/ewmh.lua index f63ef505..d110cd31 100644 --- a/lib/awful/ewmh.lua +++ b/lib/awful/ewmh.lua @@ -101,6 +101,7 @@ end -- -- It is the default signal handler for `request::activate` on a `client`. -- +-- @signalhandler awful.ewmh.activate -- @client c A client to use -- @tparam string context The context where this signal was used. -- @tparam[opt] table hints A table with additional hints: @@ -121,8 +122,11 @@ function ewmh.activate(c, context, hints) -- luacheck: no unused args end end ---- Tag a window with its requested tag +--- Tag a window with its requested tag. -- +-- It is the default signal handler for `request::tag` on a `client`. +-- +-- @signalhandler awful.ewmh.tag -- @client c A client to tag -- @tag[opt] t A tag to use. If omitted, then the client is made sticky. -- @tparam[opt={}] table hints Extra information @@ -157,6 +161,7 @@ local context_mapper = { -- -- This is the default geometry request handler. -- +-- @signalhandler awful.ewmh.geometry -- @tparam client c The client -- @tparam string context The context -- @tparam[opt={}] table hints The hints to pass to the handler diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 67468d74..14c40548 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -236,7 +236,7 @@ function mouse.client.resize(c, corner, args) end --- Default handler for `request::geometry` signals with `mouse.resize` context. --- @function awful.mouse.resize_handler +-- @signalhandler awful.mouse.resize_handler -- @tparam client c The client -- @tparam string context The context -- @tparam[opt={}] table hints The hints to pass to the handler From a39b93f2e3306dc8da9a0596889f80820c16457b Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 22:51:36 -0400 Subject: [PATCH 33/39] mouse: Add is_*_mouse_button_pressed helper properties. --- lib/awful/mouse/init.lua | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 14c40548..3ad0b0a2 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -296,6 +296,26 @@ end function mouse.object.set_current_wibox() end +--- True if the left mouse button is pressed. +-- @property is_left_mouse_button_pressed +-- @param boolean + +--- True if the right mouse button is pressed. +-- @property is_right_mouse_button_pressed +-- @param boolean + +--- True if the middle mouse button is pressed. +-- @property is_middle_mouse_button_pressed +-- @param boolean + +for _, b in ipairs {"left", "right", "middle"} do + mouse.object["is_".. b .."_mouse_button_pressed"] = function() + return capi.mouse.coords().buttons[1] + end + + mouse.object["set_is_".. b .."_mouse_button_pressed"] = function() end +end + capi.client.connect_signal("request::geometry", mouse.resize_handler) -- Set the cursor at startup From 6e354cd3763cb7f934db24f2f084809a619d8dc7 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 23:29:22 -0400 Subject: [PATCH 34/39] doc: Add proper mouse documentation --- docs/images/mouse.svg | 253 ++++++++++++++++++++++++++++++++++++++++++ mouse.c | 34 +++++- 2 files changed, 286 insertions(+), 1 deletion(-) create mode 100644 docs/images/mouse.svg diff --git a/docs/images/mouse.svg b/docs/images/mouse.svg new file mode 100644 index 00000000..e56eff51 --- /dev/null +++ b/docs/images/mouse.svg @@ -0,0 +1,253 @@ + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + Button 1(LMB) + Button 2(RMB) + + + Click: Button 3 + Down: Button 5 + Up: Button 4 + + + + + + + + + + + + + + + diff --git a/mouse.c b/mouse.c index 57c4a8ba..272b12ad 100644 --- a/mouse.c +++ b/mouse.c @@ -19,7 +19,39 @@ * */ -/** awesome mouse API +/** awesome mouse API. + * + * The mouse buttons are represented as index. The common ones are: + * + * ![Client geometry](../images/mouse.svg) + * + * It is possible to be notified of mouse events by connecting to various + * `client`, `widget`s and `wibox` signals: + * + * * `mouse::enter` + * * `mouse::leave` + * * `mouse::press` + * * `mouse::release` + * * `mouse::move` + * + * It is also possible to add generic mouse button callbacks for `client`s, + * `wiboxe`s and the `root` window. Those are set in the default `rc.lua` as such: + * + * **root**: + * + * root.buttons(awful.util.table.join( + * awful.button({ }, 3, function () mymainmenu:toggle() end), + * awful.button({ }, 4, awful.tag.viewnext), + * awful.button({ }, 5, awful.tag.viewprev) + * )) + * + * **client**: + * + * clientbuttons = awful.util.table.join( + * awful.button({ }, 1, function (c) client.focus = c; c:raise() end), + * awful.button({ modkey }, 1, awful.mouse.client.move), + * awful.button({ modkey }, 3, awful.mouse.client.resize) + * ) * * See also `mousegrabber` * From e31563b056e48e1d46f9893778f7502f9710f738 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Wed, 27 Apr 2016 21:01:44 -0400 Subject: [PATCH 35/39] mouse.move: Return the corner --- lib/awful/mouse/init.lua | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 3ad0b0a2..b9026a8c 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -215,6 +215,7 @@ end -- @param c The client to resize, or the focused one by default. -- @tparam string corner The corner to grab on resize. Auto detected by default. -- @tparam[opt={}] table args A set of `awful.placement` arguments +-- @treturn string The corner (or side) name function mouse.client.resize(c, corner, args) c = c or capi.client.focus @@ -230,9 +231,14 @@ function mouse.client.resize(c, corner, args) -- Move the mouse to the corner if corner and aplace[corner] then aplace[corner](capi.mouse, {parent=c}) + else + local _ + _, corner = aplace.closest_corner(capi.mouse, {parent=c}) end mouse.resize(c, "mouse.resize", args or {include_sides=true}) + + return corner end --- Default handler for `request::geometry` signals with `mouse.resize` context. From a62e749216a979c73ad4b46a5d3211b429e15e1c Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Wed, 27 Apr 2016 21:16:20 -0400 Subject: [PATCH 36/39] mouse.snap: Add some configuration options --- lib/awful/mouse/init.lua | 11 ++++++++++- lib/awful/mouse/snap.lua | 12 ++++++------ 2 files changed, 16 insertions(+), 7 deletions(-) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index b9026a8c..ee86d2fe 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -38,6 +38,15 @@ mouse.wibox = {} -- @tparam[opt=8] integer default_distance -- @see awful.mouse.snap +--- Enable screen edges snapping. +-- @tfield[opt=true] boolean awful.mouse.snap.edge_enabled + +--- Enable client to client snapping. +-- @tfield[opt=true] boolean awful.mouse.snap.client_enabled + +--- Enable changing tag when a client is dragged to the edge of the screen. +-- @tfield[opt=false] integer awful.mouse.snap.drag_to_tag_enabled + --- The snap outline background color. -- @beautiful beautiful.snap_bg -- @tparam color|string|gradient|pattern color @@ -46,7 +55,7 @@ mouse.wibox = {} -- @beautiful beautiful.snap_border_width -- @param integer ---- The snap outline width. +--- The snap outline shape. -- @beautiful beautiful.snap_shape -- @tparam function shape A `gears.shape` compatible function diff --git a/lib/awful/mouse/snap.lua b/lib/awful/mouse/snap.lua index 76974a55..684217cd 100644 --- a/lib/awful/mouse/snap.lua +++ b/lib/awful/mouse/snap.lua @@ -248,21 +248,21 @@ end -- Enable edge snapping resize.add_move_callback(function(c, geo, args) -- Screen edge snapping (areosnap) - if args and (args.snap == nil or args.snap) then--TODO add a config option + if (module.edge_enabled ~= false) + and args and (args.snap == nil or args.snap) then detect_areasnap(c, 16) end -- Snapping between clients - if args and (args.snap == nil or args.snap) then - local ngeo = module.snap(c, args.snap, geo.x, geo.y) - ngeo.x = ngeo.x + (2 * c.border_width) - ngeo.y = ngeo.y + (2 * c.border_width) - return ngeo + if (module.client_enabled ~= false) + and args and (args.snap == nil or args.snap) then + return module.snap(c, args.snap, geo.x, geo.y) end end, "mouse.move") -- Apply the aerosnap resize.add_leave_callback(function(c, _, args) + if module.edge_enabled == false then return end return apply_areasnap(c, args) end, "mouse.move") From cd632e4a84d8b59b0978423db8e9ea8998ff163f Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Wed, 27 Apr 2016 23:20:00 -0400 Subject: [PATCH 37/39] mouse.dragtotag: Fix and port to the new move API It wasn't really working as it would in other WM. It might have been a feature, but I got the behavior in line with KDE and Gnome. --- lib/awful/mouse/drag_to_tag.lua | 59 +++++++++++++++++++++++++++++++++ lib/awful/mouse/init.lua | 46 +++++-------------------- 2 files changed, 68 insertions(+), 37 deletions(-) create mode 100644 lib/awful/mouse/drag_to_tag.lua diff --git a/lib/awful/mouse/drag_to_tag.lua b/lib/awful/mouse/drag_to_tag.lua new file mode 100644 index 00000000..ee4cb895 --- /dev/null +++ b/lib/awful/mouse/drag_to_tag.lua @@ -0,0 +1,59 @@ +--------------------------------------------------------------------------- +--- When the the mouse reach the end of the screen, then switch tag instead +-- of screens. +-- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008 Julien Danjou +-- @release @AWESOME_VERSION@ +-- @submodule mouse +--------------------------------------------------------------------------- + +local capi = {screen = screen, mouse = mouse} +local util = require("awful.util") +local tag = require("awful.tag") +local resize = require("awful.mouse.resize") + +local module = {} + +function module.drag_to_tag(c) + if (not c) or (not c.valid) then return end + + local coords = capi.mouse.coords() + + local dir = nil + + local wa = capi.screen[c.screen].workarea + + if coords.x >= wa.x + wa.width - 1 then + capi.mouse.coords({ x = wa.x + 2 }, true) + dir = "right" + elseif coords.x <= wa.x + 1 then + capi.mouse.coords({ x = wa.x + wa.width - 2 }, true) + dir = "left" + end + + local tags = c.screen.tags + local t = c.screen.selected_tag + local idx = t.index + + if dir then + + if dir == "right" then + local newtag = tags[util.cycle(#tags, idx + 1)] + c:move_to_tag(newtag) + tag.viewnext() + elseif dir == "left" then + local newtag = tags[util.cycle(#tags, idx - 1)] + c:move_to_tag(newtag) + tag.viewprev() + end + end +end + +resize.add_move_callback(function(c, _, _) + if module.enabled then + module.drag_to_tag(c) + end +end, "mouse.move") + +return setmetatable(module, {__call = function(_, ...) return module.drag_to_tag(...) end}) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index ee86d2fe..1e10713d 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -9,7 +9,6 @@ -- Grab environment we need local layout = require("awful.layout") -local tag = require("awful.tag") local aplace = require("awful.placement") local awibox = require("awful.wibox") local util = require("awful.util") @@ -27,6 +26,7 @@ local capi = local mouse = { resize = require("awful.mouse.resize"), snap = require("awful.mouse.snap"), + drag_to_tag = require("awful.mouse.drag_to_tag") } mouse.object = {} @@ -45,7 +45,7 @@ mouse.wibox = {} -- @tfield[opt=true] boolean awful.mouse.snap.client_enabled --- Enable changing tag when a client is dragged to the edge of the screen. --- @tfield[opt=false] integer awful.mouse.snap.drag_to_tag_enabled +-- @tfield[opt=false] integer awful.mouse.drag_to_tag.enabled --- The snap outline background color. -- @beautiful beautiful.snap_bg @@ -109,44 +109,16 @@ end mouse.client.dragtotag = { } --- Move a client to a tag by dragging it onto the left / right side of the screen. --- @function awful.mouse.client.dragtotag.border +-- @deprecated awful.mouse.client.dragtotag.border -- @param c The client to move function mouse.client.dragtotag.border(c) - capi.mousegrabber.run(function (_mouse) - if not c.valid then return false end + util.deprecated("Use awful.mouse.snap.drag_to_tag_enabled = true instead ".. + "of awful.mouse.client.dragtotag.border(c). It will now be enabled.") - local button_down = false - for _, v in ipairs(_mouse.buttons) do - if v then button_down = true end - end - local wa = capi.screen[c.screen].workarea - if _mouse.x >= wa.x + wa.width then - capi.mouse.coords({ x = wa.x + wa.width - 1 }) - elseif _mouse.x <= wa.x then - capi.mouse.coords({ x = wa.x + 1 }) - end - if not button_down then - local tags = c.screen.tags - local t = c.screen.selected_tag - local idx - for i, v in ipairs(tags) do - if v == t then - idx = i - end - end - if _mouse.x > wa.x + wa.width - 10 then - local newtag = tags[util.cycle(#tags, idx + 1)] - c:move_to_tag(newtag) - tag.viewnext() - elseif _mouse.x < wa.x + 10 then - local newtag = tags[util.cycle(#tags, idx - 1)] - c:move_to_tag(newtag) - tag.viewprev() - end - return false - end - return true - end, "fleur") + -- Enable drag to border + mouse.snap.drag_to_tag_enabled = true + + return mouse.client.move(c) end --- Move the wibox under the cursor. From e41853a47a11b719657e69d37c39ef7fc1c45175 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 24 Apr 2016 02:38:22 -0400 Subject: [PATCH 38/39] tests: Test floating client move, snap and resize --- tests/test-resize.lua | 409 ++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 409 insertions(+) create mode 100644 tests/test-resize.lua diff --git a/tests/test-resize.lua b/tests/test-resize.lua new file mode 100644 index 00000000..a9eb7168 --- /dev/null +++ b/tests/test-resize.lua @@ -0,0 +1,409 @@ +local test_client = require("_client") +local placement = require("awful.placement") +local amouse = require("awful.mouse") + +local steps = {} + +table.insert(steps, function(count) + if count == 1 then -- Setup. + test_client("foobar", "foobar") + elseif #client.get() > 0 then + + client.get()[1] : geometry { + x = 200, + y = 200, + width = 300, + height = 300, + } + + return true + end +end) + +table.insert(steps, function() + -- The mousegrabber expect a button to be pressed. + root.fake_input("button_press",1) + local c = client.get()[1] + + -- Just in case there is an accidental delayed geometry callback + assert(c:geometry().x == 200) + assert(c:geometry().y == 200) + assert(c:geometry().width == 300) + assert(c:geometry().height == 300) + + mouse.coords {x = 500+2*c.border_width, y= 500+2*c.border_width} + + local corner = amouse.client.resize(c) + + assert(corner == "bottom_right") + + return true +end) + +-- The geometry should remain the same, as the cursor is placed at the end of +-- the geometry. +table.insert(steps, function() + + local c = client.get()[1] + + assert(c:geometry().x == 200) + assert(c:geometry().y == 200) + assert(c:geometry().width == 300) + assert(c:geometry().height == 300) + + mouse.coords {x = 600+2*c.border_width, y= 600+2*c.border_width} + + return true +end) + +-- Grow the client by 100px +table.insert(steps, function() + + local c = client.get()[1] + + assert(c:geometry().x == 200) + assert(c:geometry().y == 200) + assert(c:geometry().width == 400) + assert(c:geometry().height == 400) + + mouse.coords {x = 400+2*c.border_width, y= 400+2*c.border_width} + + return true +end) + +-- Shirnk the client by 200px +table.insert(steps, function() + + local c = client.get()[1] + + assert(c:geometry().x == 200) + assert(c:geometry().y == 200) + assert(c:geometry().width == 200) + assert(c:geometry().height == 200) + + mouse.coords {x = 100, y= 100} + + return true +end) + +-- Grow the client by 100px from the top left +table.insert(steps, function() + + local c = client.get()[1] + + assert(c:geometry().x == 100) + assert(c:geometry().y == 100) + assert(c:geometry().width == 300) + assert(c:geometry().height == 300) + + mouse.coords {x = 300, y= 200} + + return true +end) + +-- Shirnk the client by 100px from the top right +table.insert(steps, function() + + local c = client.get()[1] + + assert(c:geometry().x == 100) + assert(c:geometry().y == 200) +-- assert(c:geometry().width == 200-2*c.border_width) --FIXME off by border width... +-- assert(c:geometry().height == 200-2*c.border_width) --FIXME off by border width... + + mouse.coords {x = 300, y= 200} + + return true +end) + +-- Stop the resize +table.insert(steps, function() + root.fake_input("button_release",1) + +-- if not mousegrabber.isrunning then --FIXME it should work, but doesn't +-- return true +-- end + + mousegrabber.stop() + + return true +end) + +-- Setup for move testing +table.insert(steps, function() + assert(not mousegrabber.isrunning()) + + local c = client.get()[1] + + c:geometry { + width = 200, + height = 200, + } + + placement.bottom_right(c) + + mouse.coords {x = c.screen.geometry.width -150, + y = c.screen.geometry.height-150} + + + return true +end) + +-- Start the move mouse grabber +table.insert(steps, function() + local c = client.get()[1] + + -- The resize is over, it should not move the client anymore + assert(c:geometry().x == c.screen.geometry.width - 200 - 2*c.border_width) + assert(c:geometry().y == c.screen.geometry.height - 200 - 2*c.border_width) + assert(c:geometry().width == 200) + assert(c:geometry().height == 200) + + assert(c.valid) + + root.fake_input("button_press",1) + + -- Begin the move + amouse.client.move(c) + + -- Make sure nothing unwanted happen by accident + assert(c:geometry().x == c.screen.geometry.width - 200 - 2*c.border_width) + assert(c:geometry().y == c.screen.geometry.height - 200 - 2*c.border_width) + assert(c:geometry().width == 200) + assert(c:geometry().height == 200) + + -- The cursor should not have moved + assert(mouse.coords().x == c.screen.geometry.width - 150) + assert(mouse.coords().y == c.screen.geometry.height - 150) + mouse.coords {x = 50 + 2*c.border_width, y= 50 + 2*c.border_width} + + assert(mousegrabber.isrunning()) + + return true +end) + +-- The client should now be in the top left +table.insert(steps, function() + local c = client.get()[1] + assert(c:geometry().x == 0) + assert(c:geometry().y == 0) + assert(c:geometry().width == 200) + assert(c:geometry().height == 200) + + -- Move to the bottom left + mouse.coords { + x = 50 + 2*c.border_width, + y = c.screen.geometry.height - 200 + } + + return true +end) + +-- Ensure the move was correct, the snap to the top part of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == 0) + assert(c:geometry().y == c.screen.geometry.height - 250 - 2*c.border_width) + assert(c:geometry().width == 200) + assert(c:geometry().height == 200) + + -- Should trigger the top snap + mouse.coords {x = 600, y= 0} + + -- The snap is only applied on release + root.fake_input("button_release",1) + + return true +end) + +-- The client should now fill the top half of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == c.screen.workarea.x ) + assert(c:geometry().y == c.screen.workarea.y ) + assert(c:geometry().width == c.screen.workarea.width - 2*c.border_width ) + assert(c:geometry().height == math.ceil(c.screen.workarea.height/2 - 2*c.border_width)) + + -- Snap to the top right + root.fake_input("button_press",1) + amouse.client.move(c) + placement.top_right(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The client should now fill the top right corner of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == c.screen.workarea.x+c.screen.workarea.width/2 ) + assert(c:geometry().y == c.screen.workarea.y ) + assert(c:geometry().width == math.ceil(c.screen.workarea.width/2 - 2*c.border_width) ) + assert(c:geometry().height == math.ceil(c.screen.workarea.height/2 - 2*c.border_width)) + + -- Snap to the top right + root.fake_input("button_press",1) + amouse.client.move(c) + placement.right(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The client should now fill the top right half of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == c.screen.workarea.x+c.screen.workarea.width/2 ) + assert(c:geometry().y == c.screen.workarea.y ) + assert(c:geometry().width == math.ceil(c.screen.workarea.width/2 - 2*c.border_width)) + assert(c:geometry().height == c.screen.workarea.height - 2*c.border_width ) + + -- Snap to the top right + root.fake_input("button_press",1) + amouse.client.move(c) + placement.bottom(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The client should now fill the bottom half of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == c.screen.workarea.x ) + assert(c:geometry().y == c.screen.workarea.y+c.screen.workarea.height/2 ) + assert(c:geometry().width == c.screen.workarea.width - 2*c.border_width ) + assert(c:geometry().height == math.ceil(c.screen.workarea.height/2 - 2*c.border_width)) + + -- Snap to the top right + root.fake_input("button_press",1) + amouse.client.move(c) + placement.bottom_left(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The client should now fill the bottom left corner of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == c.screen.workarea.x ) + assert(c:geometry().y == c.screen.workarea.y+c.screen.workarea.height/2 ) + assert(c:geometry().width == math.ceil(c.screen.workarea.width/2 - 2*c.border_width) ) + assert(c:geometry().height == math.ceil(c.screen.workarea.height/2 - 2*c.border_width)) + + -- Snap to the top right + root.fake_input("button_press",1) + amouse.client.move(c) + placement.left(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The client should now fill the left half of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == c.screen.workarea.x ) + assert(c:geometry().y == c.screen.workarea.y ) + assert(c:geometry().width == math.ceil(c.screen.workarea.width/2 - 2*c.border_width) ) + assert(c:geometry().height == c.screen.workarea.height - 2*c.border_width ) + + -- Snap to the top right + root.fake_input("button_press",1) + amouse.client.move(c) + placement.top_left(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +local cur_tag = nil + +-- The client should now fill the top left corner of the screen +table.insert(steps, function() + local c = client.get()[1] + + assert(c:geometry().x == c.screen.workarea.x ) + assert(c:geometry().y == c.screen.workarea.y ) + assert(c:geometry().width == math.ceil(c.screen.workarea.width/2 - 2*c.border_width) ) + assert(c:geometry().height == math.ceil(c.screen.workarea.height/2 - 2*c.border_width)) + + -- Change the mode to test drag_to_tag + amouse.drag_to_tag.enabled = true + amouse.snap.edge_enabled = false + + cur_tag = c.first_tag + + root.fake_input("button_press",1) + amouse.client.move(c) + placement.right(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The tag should have changed +table.insert(steps, function() + local c = client.get()[1] + + assert(c.first_tag ~= cur_tag ) + assert(c.first_tag.index == cur_tag.index + 1) + + -- Move it back + root.fake_input("button_press",1) + amouse.client.move(c) + placement.left(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The tag should now be the same as before +table.insert(steps, function() + local c = client.get()[1] + + assert(c.first_tag == cur_tag) + assert(c.first_tag.index == 1) + + -- Wrap + root.fake_input("button_press",1) + amouse.client.move(c) + placement.left(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The tag should now be the last +table.insert(steps, function() + local c = client.get()[1] + + assert(c.first_tag.index == #c.screen.tags) + + -- Wrap back + root.fake_input("button_press",1) + amouse.client.move(c) + placement.right(mouse, {honor_workarea=false}) + root.fake_input("button_release",1) + + return true +end) + +-- The tag should now original one again +table.insert(steps, function() + local c = client.get()[1] + + assert(c.first_tag == cur_tag) + + return true +end) + +require("_runner").run_steps(steps) From 3dab42e6a29fbedfb258207d818046f322291995 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 30 Apr 2016 01:47:41 -0400 Subject: [PATCH 39/39] mouse: Restore support for layouts with their own mousegrabber --- lib/awful/mouse/init.lua | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/lib/awful/mouse/init.lua b/lib/awful/mouse/init.lua index 1e10713d..13cd23ae 100644 --- a/lib/awful/mouse/init.lua +++ b/lib/awful/mouse/init.lua @@ -257,6 +257,27 @@ function mouse.resize_handler(c, context, hints) end end +-- Older layouts implement their own mousegrabber. +-- @tparam client c The client +-- @tparam table args Additional arguments +-- @treturn boolean This return false when the resize need to be aborted +mouse.resize.add_enter_callback(function(c, args) --luacheck: no unused args + if c.floating then return end + + local l = c.screen.selected_tag and c.screen.selected_tag.layout or nil + if l == layout.suit.floating then return end + + if l ~= layout.suit.floating and l.mouse_resize_handler then + capi.mousegrabber.stop() + + local geo, corner = aplace.closest_corner(capi.mouse, {parent=c}) + + l.mouse_resize_handler(c, corner, geo.x, geo.y) + + return false + end +end, "mouse.resize") + --- Get the client currently under the mouse cursor. -- @property current_client -- @tparam client|nil The client