Merge pull request #2811 from sigprof/no_overlap-unselected-tags-v2

awful.placement: Fix no_overlap with unselected tags
This commit is contained in:
Emmanuel Lepage Vallée 2019-07-10 17:20:27 -04:00 committed by GitHub
commit 0ac0a77aab
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 290 additions and 88 deletions

View File

@ -894,6 +894,39 @@ function placement.no_offscreen(c, args)
return fix_new_geometry(geometry, args, true) return fix_new_geometry(geometry, args, true)
end end
-- Check whether the client is on at least one of selected tags (similar to
-- `c:isvisible()`, but without checks for the hidden or minimized state).
local function client_on_selected_tags(c)
if c.sticky then
return true
else
for _, t in pairs(c:tags()) do
if t.selected then
return true
end
end
return false
end
end
-- Check whether the client would be visible if the specified set of tags is
-- selected (using the same set of conditions as `c:isvisible()`, but checking
-- against the specified tags instead of currently selected ones).
local function client_visible_on_tags(c, tags)
if c.hidden or c.minimized then
return false
elseif c.sticky then
return true
else
for _, t in pairs(c:tags()) do
if gtable.hasitem(tags, t) then
return true
end
end
return false
end
end
--- Place the client where there's place available with minimum overlap. --- Place the client where there's place available with minimum overlap.
--@DOC_awful_placement_no_overlap_EXAMPLE@ --@DOC_awful_placement_no_overlap_EXAMPLE@
-- @param c The client. -- @param c The client.
@ -905,8 +938,22 @@ function placement.no_overlap(c, args)
args = add_context(args, "no_overlap") args = add_context(args, "no_overlap")
local geometry = geometry_common(c, args) local geometry = geometry_common(c, args)
local screen = get_screen(c.screen or a_screen.getbycoord(geometry.x, geometry.y)) local screen = get_screen(c.screen or a_screen.getbycoord(geometry.x, geometry.y))
local cls = client.visible(screen) local cls, curlay
local curlay = layout.get() if client_on_selected_tags(c) then
cls = client.visible(screen)
curlay = layout.get()
else
-- When placing a client on unselected tags, place it as if all tags of
-- that client are selected.
local tags = c:tags()
cls = {}
for _, other_c in pairs(capi.client.get(screen)) do
if client_visible_on_tags(other_c, tags) then
table.insert(cls, other_c)
end
end
curlay = tags[1] and tags[1].layout
end
local areas = { screen.workarea } local areas = { screen.workarea }
for _, cl in pairs(cls) do for _, cl in pairs(cls) do
if cl ~= c if cl ~= c

View File

@ -44,25 +44,25 @@ local function default_test(c, geometry)
return true return true
end end
local client_count = 0
local client_data = {} local client_data = {}
local function add_client(args) local function add_client(args)
client_count = client_count + 1 local data = {}
local client_index = client_count table.insert(client_data, data)
local client_index = #client_data
table.insert(tests, function(count) table.insert(tests, function(count)
local name = string.format("client%010d", client_index) local name = string.format("client%010d", client_index)
if count <= 1 then if count <= 1 then
data.prev_client_count = #client.get()
local geometry = args.geometry(mouse.screen.workarea) local geometry = args.geometry(mouse.screen.workarea)
test_client(class, name, nil, nil, nil, { test_client(class, name, args.sn_rules, nil, nil, {
size = { size = {
width = geometry.width, width = geometry.width,
height = geometry.height height = geometry.height
} }
}) })
client_data[client_index] = { geometry = geometry } data.geometry = geometry
return nil return nil
elseif #client.get() >= client_index then elseif #client.get() > data.prev_client_count then
local data = client_data[client_index]
local c = data.c local c = data.c
if not c then if not c then
c = client.get()[1] c = client.get()[1]
@ -75,8 +75,26 @@ local function add_client(args)
end) end)
end end
-- The first 100x100 window should be placed at the top left corner. -- Repeat testing 3 times, placing clients on different tags:
add_client { --
-- - Iteration 1 places clients on the tag 1, which is selected.
--
-- - Iteration 2 places clients on the tag 2, which is unselected; the
-- selected tag 1 remains empty.
--
-- - Iteration 3 places clients on the tag 3, which is unselected; the
-- selected tag 1 contains some clients.
--
for _, tag_num in ipairs{1, 2, 3} do
local sn_rules
if tag_num > 1 then
sn_rules = { tag = root.tags()[tag_num] }
end
-- Put a 100x100 client on the tag 1 before iteration 3.
if tag_num == 3 then
add_client {
geometry = function(wa) geometry = function(wa)
return { return {
width = 100, width = 100,
@ -85,12 +103,31 @@ add_client {
expected_y = wa.y expected_y = wa.y
} }
end end
} }
end
-- The second 100x100 window should be placed to the right of the first window. -- The first 100x100 client should be placed at the top left corner.
-- Note that this assumption fails if the screen is in the portrait orientation add_client {
-- (e.g., the test succeeds with a 600x703 screen and fails with 600x704). sn_rules = sn_rules,
add_client { geometry = function(wa)
return {
width = 100,
height = 100,
expected_x = wa.x,
expected_y = wa.y
}
end
}
-- Remember the first client data for the current iteration.
local first_client_data = client_data[#client_data]
-- The second 100x100 client should be placed to the right of the first
-- client. Note that this assumption fails if the screen is in the portrait
-- orientation (e.g., the test succeeds with a 600x703 screen and fails with
-- 600x704).
add_client {
sn_rules = sn_rules,
geometry = function(wa) geometry = function(wa)
return { return {
width = 100, width = 100,
@ -99,10 +136,107 @@ add_client {
expected_y = wa.y expected_y = wa.y
} }
end end
} }
-- The wide window should be placed below the two 100x100 windows. -- Hide last client.
add_client { do
local data = client_data[#client_data]
table.insert(tests, function()
data.c.hidden = true
return true
end)
end
-- Another 100x100 client should be placed to the right of the first client
-- (the hidden client should be ignored during placement).
add_client {
sn_rules = sn_rules,
geometry = function(wa)
return {
width = 100,
height = 100,
expected_x = wa.x + 100 + 2*border_width,
expected_y = wa.y
}
end
}
-- Minimize last client.
do
local data = client_data[#client_data]
table.insert(tests, function()
data.c.minimized = true
return true
end)
end
-- Another 100x100 client should be placed to the right of the first client
-- (the minimized client should be ignored during placement).
add_client {
sn_rules = sn_rules,
geometry = function(wa)
return {
width = 100,
height = 100,
expected_x = wa.x + 100 + 2*border_width,
expected_y = wa.y
}
end
}
-- Hide last client, and make the first client sticky.
do
local data = client_data[#client_data]
table.insert(tests, function()
data.c.hidden = true
first_client_data.c.sticky = true
return true
end)
end
-- Another 100x100 client should be placed to the right of the first client
-- (the sticky client should be taken into account during placement).
add_client {
sn_rules = sn_rules,
geometry = function(wa)
return {
width = 100,
height = 100,
expected_x = wa.x + 100 + 2*border_width,
expected_y = wa.y
}
end
}
-- Hide last client, and put the first client on the tag 9 (because the
-- first client is sticky, it should remain visible).
do
local data = client_data[#client_data]
table.insert(tests, function()
data.c.hidden = true
first_client_data.c:tags{ root.tags()[9] }
return true
end)
end
-- Another 100x100 client should be placed to the right of the first client
-- (the sticky client should be taken into account during placement even if
-- that client seems to be on an unselected tag).
add_client {
sn_rules = sn_rules,
geometry = function(wa)
return {
width = 100,
height = 100,
expected_x = wa.x + 100 + 2*border_width,
expected_y = wa.y
}
end
}
-- The wide client should be placed below the two 100x100 client.
add_client {
sn_rules = sn_rules,
geometry = function(wa) geometry = function(wa)
return { return {
width = wa.width - 50, width = wa.width - 50,
@ -111,13 +245,14 @@ add_client {
expected_y = wa.y + tb_height + 2*border_width + 100 expected_y = wa.y + tb_height + 2*border_width + 100
} }
end end
} }
-- The first large window which does not completely fit in any free area should -- The first large client which does not completely fit in any free area
-- be placed at the bottom left corner (no_overlap should place it below the -- should be placed at the bottom left corner (no_overlap should place it
-- wide window, and then no_offscreen should shift it up so that it would be -- below the wide client, and then no_offscreen should shift it up so that
-- completely inside the workarea). -- it would be completely inside the workarea).
add_client { add_client {
sn_rules = sn_rules,
geometry = function(wa) geometry = function(wa)
return { return {
width = wa.width - 10, width = wa.width - 10,
@ -126,10 +261,11 @@ add_client {
expected_y = wa.y + 50 - 2*border_width - tb_height expected_y = wa.y + 50 - 2*border_width - tb_height
} }
end end
} }
-- The second large window should be placed at the top right corner. -- The second large client should be placed at the top right corner.
add_client { add_client {
sn_rules = sn_rules,
geometry = function(wa) geometry = function(wa)
return { return {
width = wa.width - 10, width = wa.width - 10,
@ -138,10 +274,11 @@ add_client {
expected_y = wa.y expected_y = wa.y
} }
end end
} }
-- The third large window should be placed at the bottom right corner. -- The third large client should be placed at the bottom right corner.
add_client { add_client {
sn_rules = sn_rules,
geometry = function(wa) geometry = function(wa)
return { return {
width = wa.width - 10, width = wa.width - 10,
@ -150,11 +287,12 @@ add_client {
expected_y = wa.y + 50 - 2*border_width - tb_height expected_y = wa.y + 50 - 2*border_width - tb_height
} }
end end
} }
-- The fourth large window should be placed at the top left corner (the whole -- The fourth large client should be placed at the top left corner (the
-- workarea is occupied now). -- whole workarea is occupied now).
add_client { add_client {
sn_rules = sn_rules,
geometry = function(wa) geometry = function(wa)
return { return {
width = wa.width - 50, width = wa.width - 50,
@ -163,7 +301,24 @@ add_client {
expected_y = wa.y expected_y = wa.y
} }
end end
} }
-- Kill test clients to prepare for the next iteration.
table.insert(tests, function(count)
if count <= 1 then
for _, data in ipairs(client_data) do
if data.c then
data.c:kill()
data.c = nil
end
end
end
if #client.get() == 0 then
return true
end
end)
end
runner.run_steps(tests) runner.run_steps(tests)