From d4168f3c36aba80343ad45ca7e065b2bb3bb868e Mon Sep 17 00:00:00 2001 From: Uli Schlachter Date: Sat, 10 Oct 2015 20:10:23 +0200 Subject: [PATCH 1/2] screen_getbycoord: Change definition of 'closest screen' Instead of comparing only the top-left corner of the screen to the provided coordinate, this now compares the screen in a more intuitive way, e.g. coordinates inside of the screen have a distance of zero. Signed-off-by: Uli Schlachter --- objects/screen.c | 45 ++++++++++++++++++++++++++++++++++++++------- 1 file changed, 38 insertions(+), 7 deletions(-) diff --git a/objects/screen.c b/objects/screen.c index 4ec161c8e..57be3561b 100644 --- a/objects/screen.c +++ b/objects/screen.c @@ -35,7 +35,6 @@ #include "objects/client.h" #include "objects/drawin.h" -#include #include #include @@ -285,6 +284,38 @@ screen_scan(void) screen_scan_x11(); } +/** Return the squared distance of the given screen to the coordinates. + * \param screen The screen + * \param x X coordinate + * \param y Y coordinate + * \return Squared distance of the point to the screen. + */ +static unsigned int +screen_get_distance_squared(screen_t *s, int x, int y) +{ + int sx = s->geometry.x, sy = s->geometry.y; + int sheight = s->geometry.height, swidth = s->geometry.width; + unsigned int dist_x, dist_y; + + /* Calculate distance in X coordinate */ + if (x < sx) + dist_x = sx - x; + else if (x < sx + swidth) + dist_x = 0; + else + dist_x = x - sx - swidth; + + /* Calculate distance in Y coordinate */ + if (y < sy) + dist_y = sy - y; + else if (y < sy + sheight) + dist_y = 0; + else + dist_y = y - sy - sheight; + + return dist_x * dist_x + dist_y * dist_y; +} + /** Return the first screen number where the coordinates belong to. * \param x X coordinate * \param y Y coordinate @@ -298,15 +329,15 @@ screen_getbycoord(int x, int y) return *s; /* No screen found, find nearest screen. */ - screen_t * nearest_screen = globalconf.screens.tab[0]; - int nearest_dist = INT_MAX; + screen_t *nearest_screen = globalconf.screens.tab[0]; + unsigned int nearest_dist = UINT_MAX; foreach(s, globalconf.screens) { - int dist = sqrt(pow((*s)->geometry.x - x, 2) + pow((*s)->geometry.y - y, 2)); - if( dist < nearest_dist ) + unsigned int dist_sq = screen_get_distance_squared(*s, x, y); + if(dist_sq < nearest_dist) { - nearest_dist = dist; - nearest_screen = (*s); + nearest_dist = dist_sq; + nearest_screen = *s; } } return nearest_screen; From b1b28528879e3f0873906a1f64d7482b1a1f573e Mon Sep 17 00:00:00 2001 From: Uli Schlachter Date: Sat, 10 Oct 2015 20:16:49 +0200 Subject: [PATCH 2/2] awful.screen.getbycoord: Use the same algorithm as in C We recently changed the C function screen_getbycoord() and this commit makes awful.screen.getbycoord() use the same algorithm. Signed-off-by: Uli Schlachter --- lib/awful/screen.lua | 38 ++++++++++++++++++++++++++++---------- 1 file changed, 28 insertions(+), 10 deletions(-) diff --git a/lib/awful/screen.lua b/lib/awful/screen.lua index 57cd3439a..b8f667d5a 100644 --- a/lib/awful/screen.lua +++ b/lib/awful/screen.lua @@ -26,24 +26,42 @@ data.padding = {} screen.mouse_per_screen = {} +-- @param s Screen number +-- @param x X coordinate of point +-- @param y Y coordinate of point +-- @return The squared distance of the screen to the provided point +function screen.getdistance_sq(s, x, y) + local geom = capi.screen[s].geometry + local dist_x, dist_y = 0, 0 + if x < geom.x then + dist_x = geom.x - x + elseif x >= geom.x + geom.width then + dist_x = x - geom.x - geom.width + end + if y < geom.y then + dist_y = geom.y - y + elseif y >= geom.y + geom.height then + dist_y = y - geom.y - geom.height + end + return dist_x * dist_x + dist_y * dist_y +end + --- -- Return Xinerama screen number corresponding to the given (pixel) coordinates. -- The number returned can be used as an index into the global -- `screen` table/object. -- @param x The x coordinate -- @param y The y coordinate --- @param[opt] default The default return value. If unspecified, 1 is returned. -function screen.getbycoord(x, y, default) - for i = 1, capi.screen:count() do - local geometry = capi.screen[i].geometry - if((x < 0 or (x >= geometry.x and x < geometry.x + geometry.width)) - and (y < 0 or (y >= geometry.y and y < geometry.y + geometry.height))) then - return i +function screen.getbycoord(x, y) + local s = 1 + local dist = screen.getdistance_sq(s, x, y) + for i = 2, capi.screen:count() do + local d = screen.getdistance_sq(i, x, y) + if d < dist then + s, dist = i, d end end - -- No caller expects a nil result here, so just make something up - if default == nil then return 1 end - return default + return s end --- Give the focus to a screen, and move pointer to last known position on this