Moved the WindowFromPoint functionality to the server so that we can
properly take into account the window region.

diff --git a/include/wine/server_protocol.h b/include/wine/server_protocol.h
index 7bcfd73..76fa6be 100644
--- a/include/wine/server_protocol.h
+++ b/include/wine/server_protocol.h
@@ -2515,6 +2515,22 @@
 
 
 
+struct get_window_children_from_point_request
+{
+    struct request_header __header;
+    user_handle_t  parent;
+    int            x;
+    int            y;
+};
+struct get_window_children_from_point_reply
+{
+    struct reply_header __header;
+    int            count;
+    /* VARARG(children,user_handles); */
+};
+
+
+
 struct get_window_tree_request
 {
     struct request_header __header;
@@ -3227,6 +3243,7 @@
     REQ_set_window_info,
     REQ_get_window_parents,
     REQ_get_window_children,
+    REQ_get_window_children_from_point,
     REQ_get_window_tree,
     REQ_set_window_rectangles,
     REQ_get_window_rectangles,
@@ -3410,6 +3427,7 @@
     struct set_window_info_request set_window_info_request;
     struct get_window_parents_request get_window_parents_request;
     struct get_window_children_request get_window_children_request;
+    struct get_window_children_from_point_request get_window_children_from_point_request;
     struct get_window_tree_request get_window_tree_request;
     struct set_window_rectangles_request set_window_rectangles_request;
     struct get_window_rectangles_request get_window_rectangles_request;
@@ -3591,6 +3609,7 @@
     struct set_window_info_reply set_window_info_reply;
     struct get_window_parents_reply get_window_parents_reply;
     struct get_window_children_reply get_window_children_reply;
+    struct get_window_children_from_point_reply get_window_children_from_point_reply;
     struct get_window_tree_reply get_window_tree_reply;
     struct set_window_rectangles_reply set_window_rectangles_reply;
     struct get_window_rectangles_reply get_window_rectangles_reply;
@@ -3628,6 +3647,6 @@
     struct set_global_windows_reply set_global_windows_reply;
 };
 
-#define SERVER_PROTOCOL_VERSION 146
+#define SERVER_PROTOCOL_VERSION 147
 
 #endif /* __WINE_WINE_SERVER_PROTOCOL_H */
diff --git a/server/protocol.def b/server/protocol.def
index df2bf68..a2864be 100644
--- a/server/protocol.def
+++ b/server/protocol.def
@@ -1777,6 +1777,17 @@
 @END
 
 
+/* Get a list of the window children that contain a given point */
+@REQ(get_window_children_from_point)
+    user_handle_t  parent;        /* parent window */
+    int            x;             /* point in parent coordinates */
+    int            y;
+@REPLY
+    int            count;         /* total count of children */
+    VARARG(children,user_handles); /* children handles */
+@END
+
+
 /* Get window tree information from a window handle */
 @REQ(get_window_tree)
     user_handle_t  handle;        /* handle to the window */
diff --git a/server/region.c b/server/region.c
index ab6645c..e0035e9 100644
--- a/server/region.c
+++ b/server/region.c
@@ -758,3 +758,20 @@
     dst->extents.bottom = max(src1->extents.bottom, src2->extents.bottom);
     return dst;
 }
+
+/* check if the given point is inside the region */
+int point_in_region( struct region *region, int x, int y )
+{
+    const rectangle_t *ptr, *end;
+
+    for (ptr = region->rects, end = region->rects + region->num_rects; ptr < end; ptr++)
+    {
+        if (ptr->top > y) return 0;
+        if (ptr->bottom <= y) continue;
+        /* now we are in the correct band */
+        if (ptr->left > x) return 0;
+        if (ptr->right <= x) continue;
+        return 1;
+    }
+    return 0;
+}
diff --git a/server/request.h b/server/request.h
index 86c7419..92e6d57 100644
--- a/server/request.h
+++ b/server/request.h
@@ -244,6 +244,7 @@
 DECL_HANDLER(set_window_info);
 DECL_HANDLER(get_window_parents);
 DECL_HANDLER(get_window_children);
+DECL_HANDLER(get_window_children_from_point);
 DECL_HANDLER(get_window_tree);
 DECL_HANDLER(set_window_rectangles);
 DECL_HANDLER(get_window_rectangles);
@@ -426,6 +427,7 @@
     (req_handler)req_set_window_info,
     (req_handler)req_get_window_parents,
     (req_handler)req_get_window_children,
+    (req_handler)req_get_window_children_from_point,
     (req_handler)req_get_window_tree,
     (req_handler)req_set_window_rectangles,
     (req_handler)req_get_window_rectangles,
diff --git a/server/trace.c b/server/trace.c
index f9445bc..1389608 100644
--- a/server/trace.c
+++ b/server/trace.c
@@ -2105,6 +2105,20 @@
     dump_varargs_user_handles( cur_size );
 }
 
+static void dump_get_window_children_from_point_request( const struct get_window_children_from_point_request *req )
+{
+    fprintf( stderr, " parent=%p,", req->parent );
+    fprintf( stderr, " x=%d,", req->x );
+    fprintf( stderr, " y=%d", req->y );
+}
+
+static void dump_get_window_children_from_point_reply( const struct get_window_children_from_point_reply *req )
+{
+    fprintf( stderr, " count=%d,", req->count );
+    fprintf( stderr, " children=" );
+    dump_varargs_user_handles( cur_size );
+}
+
 static void dump_get_window_tree_request( const struct get_window_tree_request *req )
 {
     fprintf( stderr, " handle=%p", req->handle );
@@ -2677,6 +2691,7 @@
     (dump_func)dump_set_window_info_request,
     (dump_func)dump_get_window_parents_request,
     (dump_func)dump_get_window_children_request,
+    (dump_func)dump_get_window_children_from_point_request,
     (dump_func)dump_get_window_tree_request,
     (dump_func)dump_set_window_rectangles_request,
     (dump_func)dump_get_window_rectangles_request,
@@ -2856,6 +2871,7 @@
     (dump_func)dump_set_window_info_reply,
     (dump_func)dump_get_window_parents_reply,
     (dump_func)dump_get_window_children_reply,
+    (dump_func)dump_get_window_children_from_point_reply,
     (dump_func)dump_get_window_tree_reply,
     (dump_func)0,
     (dump_func)dump_get_window_rectangles_reply,
@@ -3035,6 +3051,7 @@
     "set_window_info",
     "get_window_parents",
     "get_window_children",
+    "get_window_children_from_point",
     "get_window_tree",
     "set_window_rectangles",
     "get_window_rectangles",
diff --git a/server/user.h b/server/user.h
index 80ccddf..638ffc6 100644
--- a/server/user.h
+++ b/server/user.h
@@ -83,6 +83,7 @@
                                        const struct region *src2 );
 extern struct region *union_region( struct region *dst, const struct region *src1,
                                     const struct region *src2 );
+extern int point_in_region( struct region *region, int x, int y );
 static inline struct region *create_empty_region(void) { return create_region( NULL, 0 ); }
 
 /* window functions */
diff --git a/server/window.c b/server/window.c
index fc08531..00a004e 100644
--- a/server/window.c
+++ b/server/window.c
@@ -66,9 +66,9 @@
     struct window_class *class;       /* window class */
     atom_t           atom;            /* class atom */
     user_handle_t    last_active;     /* last active popup */
-    rectangle_t      window_rect;     /* window rectangle */
-    rectangle_t      client_rect;     /* client rectangle */
-    struct region   *win_region;      /* window region (for shaped windows) */
+    rectangle_t      window_rect;     /* window rectangle (relative to parent client area) */
+    rectangle_t      client_rect;     /* client rectangle (relative to parent client area) */
+    struct region   *win_region;      /* region for shaped windows (relative to window rect) */
     unsigned int     style;           /* window style */
     unsigned int     ex_style;        /* window extended style */
     unsigned int     id;              /* window id */
@@ -83,6 +83,14 @@
     char             extra_bytes[1];  /* extra bytes storage */
 };
 
+/* growable array of user handles */
+struct user_handle_array
+{
+    user_handle_t *handles;
+    int            count;
+    int            total;
+};
+
 static struct window *top_window;  /* top-level (desktop) window */
 
 /* global window pointers */
@@ -145,6 +153,26 @@
     }
 }
 
+/* append a user handle to a handle array */
+static int add_handle_to_array( struct user_handle_array *array, user_handle_t handle )
+{
+    if (array->count >= array->total)
+    {
+        int new_total = max( array->total * 2, 32 );
+        user_handle_t *new_array = realloc( array->handles, new_total * sizeof(*new_array) );
+        if (!new_array)
+        {
+            free( array->handles );
+            set_error( STATUS_NO_MEMORY );
+            return 0;
+        }
+        array->handles = new_array;
+        array->total = new_total;
+    }
+    array->handles[array->count++] = handle;
+    return 1;
+}
+
 /* set a window property */
 static void set_property( struct window *win, atom_t atom, obj_handle_t handle,
                           enum property_type type )
@@ -377,6 +405,23 @@
     return 1;
 }
 
+/* check if point is inside the window */
+static inline int is_point_in_window( struct window *win, int x, int y )
+{
+    if (!(win->style & WS_VISIBLE)) return 0; /* not visible */
+    if ((win->style & (WS_POPUP|WS_CHILD|WS_DISABLED)) == (WS_CHILD|WS_DISABLED))
+        return 0;  /* disabled child */
+    if ((win->ex_style & (WS_EX_LAYERED|WS_EX_TRANSPARENT)) == (WS_EX_LAYERED|WS_EX_TRANSPARENT))
+        return 0;  /* transparent */
+    if (x < win->window_rect.left || x >= win->window_rect.right ||
+        y < win->window_rect.top || y >= win->window_rect.bottom)
+        return 0;  /* not in window */
+    if (win->win_region &&
+        !point_in_region( win->win_region, x - win->window_rect.left, y - win->window_rect.top ))
+        return 0;  /* not in window region */
+    return 1;
+}
+
 /* find child of 'parent' that contains the given point (in parent-relative coords) */
 static struct window *child_window_from_point( struct window *parent, int x, int y )
 {
@@ -384,16 +429,7 @@
 
     for (ptr = parent->first_child; ptr; ptr = ptr->next)
     {
-        if (!(ptr->style & WS_VISIBLE)) continue; /* not visible -> skip */
-        if ((ptr->style & (WS_POPUP|WS_CHILD|WS_DISABLED)) == (WS_CHILD|WS_DISABLED))
-            continue;  /* disabled child -> skip */
-        if ((ptr->ex_style & (WS_EX_LAYERED|WS_EX_TRANSPARENT)) == (WS_EX_LAYERED|WS_EX_TRANSPARENT))
-            continue;  /* transparent -> skip */
-        if (x < ptr->window_rect.left || x >= ptr->window_rect.right ||
-            y < ptr->window_rect.top || y >= ptr->window_rect.bottom)
-            continue;  /* not in window -> skip */
-
-        /* FIXME: check window region here */
+        if (!is_point_in_window( ptr, x, y )) continue;  /* skip it */
 
         /* if window is minimized or disabled, return at once */
         if (ptr->style & (WS_MINIMIZE|WS_DISABLED)) return ptr;
@@ -408,6 +444,32 @@
     return parent;  /* not found any child */
 }
 
+/* find all children of 'parent' that contain the given point */
+static int get_window_children_from_point( struct window *parent, int x, int y,
+                                           struct user_handle_array *array )
+{
+    struct window *ptr;
+
+    for (ptr = parent->first_child; ptr; ptr = ptr->next)
+    {
+        if (!is_point_in_window( ptr, x, y )) continue;  /* skip it */
+
+        /* if point is in client area, and window is not minimized or disabled, check children */
+        if (!(ptr->style & (WS_MINIMIZE|WS_DISABLED)) &&
+            x >= ptr->client_rect.left && x < ptr->client_rect.right &&
+            y >= ptr->client_rect.top && y < ptr->client_rect.bottom)
+        {
+            if (!get_window_children_from_point( ptr, x - ptr->client_rect.left,
+                                                 y - ptr->client_rect.top, array ))
+                return 0;
+        }
+
+        /* now add window to the array */
+        if (!add_handle_to_array( array, ptr->handle )) return 0;
+    }
+    return 1;
+}
+
 /* find window containing point (in absolute coords) */
 user_handle_t window_from_point( int x, int y )
 {
@@ -418,6 +480,35 @@
     return ret->handle;
 }
 
+/* return list of all windows containing point (in absolute coords) */
+static int all_windows_from_point( struct window *top, int x, int y, struct user_handle_array *array )
+{
+    struct window *ptr;
+
+    /* make point relative to top window */
+    for (ptr = top->parent; ptr && ptr != top_window; ptr = ptr->parent)
+    {
+        x -= ptr->client_rect.left;
+        y -= ptr->client_rect.top;
+    }
+
+    if (!is_point_in_window( top, x, y )) return 1;
+
+    /* if point is in client area, and window is not minimized or disabled, check children */
+    if (!(top->style & (WS_MINIMIZE|WS_DISABLED)) &&
+        x >= top->client_rect.left && x < top->client_rect.right &&
+        y >= top->client_rect.top && y < top->client_rect.bottom)
+    {
+        if (!get_window_children_from_point( top, x - top->client_rect.left,
+                                             y - top->client_rect.top, array ))
+            return 0;
+    }
+    /* now add window to the array */
+    if (!add_handle_to_array( array, top->handle )) return 0;
+    return 1;
+}
+
+
 /* return the thread owning a window */
 struct thread *get_window_thread( user_handle_t handle )
 {
@@ -825,6 +916,27 @@
 }
 
 
+/* get a list of the window children that contain a given point */
+DECL_HANDLER(get_window_children_from_point)
+{
+    struct user_handle_array array;
+    struct window *parent = get_window( req->parent );
+    size_t len;
+
+    if (!parent) return;
+
+    array.handles = NULL;
+    array.count = 0;
+    array.total = 0;
+    if (!all_windows_from_point( parent, req->x, req->y, &array )) return;
+
+    reply->count = array.count;
+    len = min( get_reply_max_size(), array.count * sizeof(user_handle_t) );
+    if (len) set_reply_data_ptr( array.handles, len );
+    else free( array.handles );
+}
+
+
 /* get window tree information from a window handle */
 DECL_HANDLER(get_window_tree)
 {
diff --git a/windows/winpos.c b/windows/winpos.c
index 4140a8d..fa1610a 100644
--- a/windows/winpos.c
+++ b/windows/winpos.c
@@ -338,92 +338,42 @@
 
 
 /***********************************************************************
- *           find_child_from_point
+ *           list_children_from_point
  *
- * Find the child that contains pt. Helper for WindowFromPoint.
- * pt is in parent client coordinates.
- * lparam is the param to pass in the WM_NCHITTEST message.
+ * Get the list of children that can contain point from the server.
+ * Point is in screen coordinates.
+ * Returned list must be freed by caller.
  */
-static HWND find_child_from_point( HWND parent, POINT pt, INT *hittest, LPARAM lparam )
+static HWND *list_children_from_point( HWND hwnd, POINT pt )
 {
-    int i, res;
-    LONG style, exstyle;
-    RECT rectWindow, rectClient;
-    WND *wndPtr;
-    HWND *list = WIN_ListChildren( parent );
-    HWND retvalue = 0;
+    HWND *list;
+    int size = 32;
 
-    if (!list) return 0;
-    for (i = 0; list[i]; i++)
+    for (;;)
     {
-        /* If point is in window, and window is visible, and it  */
-        /* is enabled (or it's a top-level window), then explore */
-        /* its children. Otherwise, go to the next window.       */
+        int count = 0;
 
-        style = GetWindowLongW( list[i], GWL_STYLE );
-        if (!(style & WS_VISIBLE)) continue;  /* not visible -> skip */
-        if ((style & (WS_POPUP | WS_CHILD | WS_DISABLED)) == (WS_CHILD | WS_DISABLED))
-            continue;  /* disabled child -> skip */
-        exstyle = GetWindowLongW( list[i], GWL_EXSTYLE );
-        if ((exstyle & (WS_EX_LAYERED | WS_EX_TRANSPARENT)) == (WS_EX_LAYERED | WS_EX_TRANSPARENT))
-            continue;  /* transparent -> skip */
+        if (!(list = HeapAlloc( GetProcessHeap(), 0, size * sizeof(HWND) ))) break;
 
-        if (!WIN_GetRectangles( list[i], &rectWindow, &rectClient )) continue;
-        if (!PtInRect( &rectWindow, pt )) continue;  /* not in window -> skip */
-
-        /* FIXME: check window region for other processes too */
-        if ((wndPtr = WIN_GetPtr( list[i] )) && wndPtr != WND_OTHER_PROCESS)
+        SERVER_START_REQ( get_window_children_from_point )
         {
-            if (wndPtr->hrgnWnd && !PtInRegion( wndPtr->hrgnWnd,
-                                                pt.x - rectWindow.left, pt.y - rectWindow.top ))
-            {
-                WIN_ReleasePtr( wndPtr );
-                continue;  /* point outside window region -> skip */
-            }
-            WIN_ReleasePtr( wndPtr );
+            req->parent = hwnd;
+            req->x = pt.x;
+            req->y = pt.y;
+            wine_server_set_reply( req, list, (size-1) * sizeof(HWND) );
+            if (!wine_server_call( req )) count = reply->count;
         }
-
-        /* If window is minimized or disabled, return at once */
-        if (style & WS_MINIMIZE)
+        SERVER_END_REQ;
+        if (count && count < size)
         {
-            *hittest = HTCAPTION;
-            retvalue = list[i];
-            break;
+            list[count] = 0;
+            return list;
         }
-        if (style & WS_DISABLED)
-        {
-            *hittest = HTERROR;
-            retvalue = list[i];
-            break;
-        }
-
-        /* If point is in client area, explore children */
-        if (PtInRect( &rectClient, pt ))
-        {
-            POINT new_pt;
-
-            new_pt.x = pt.x - rectClient.left;
-            new_pt.y = pt.y - rectClient.top;
-            if ((retvalue = find_child_from_point( list[i], new_pt, hittest, lparam ))) break;
-        }
-
-        /* Now it's inside window, send WM_NCCHITTEST (if same thread) */
-        if (!WIN_IsCurrentThread( list[i] ))
-        {
-            *hittest = HTCLIENT;
-            retvalue = list[i];
-            break;
-        }
-        if ((res = SendMessageA( list[i], WM_NCHITTEST, 0, lparam )) != HTTRANSPARENT)
-        {
-            *hittest = res;  /* Found the window */
-            retvalue = list[i];
-            break;
-        }
-        /* continue search with next sibling */
+        HeapFree( GetProcessHeap(), 0, list );
+        if (!count) break;
+        size = count + 1;  /* restart with a large enough buffer */
     }
-    HeapFree( GetProcessHeap(), 0, list );
-    return retvalue;
+    return NULL;
 }
 
 
@@ -434,54 +384,50 @@
  */
 HWND WINPOS_WindowFromPoint( HWND hwndScope, POINT pt, INT *hittest )
 {
-    POINT xy = pt;
-    int res;
-    LONG style;
-
-    TRACE("scope %p %ld,%ld\n", hwndScope, pt.x, pt.y);
+    int i, res;
+    HWND ret, *list;
 
     if (!hwndScope) hwndScope = GetDesktopWindow();
-    style = GetWindowLongW( hwndScope, GWL_STYLE );
 
-    *hittest = HTERROR;
-    if (style & WS_DISABLED) return 0;
-
-    MapWindowPoints( GetDesktopWindow(), GetAncestor( hwndScope, GA_PARENT ), &xy, 1 );
-
-    if (!(style & WS_MINIMIZE))
-    {
-        RECT rectClient;
-        if (WIN_GetRectangles( hwndScope, NULL, &rectClient ) && PtInRect( &rectClient, xy ))
-        {
-            HWND ret;
-
-            xy.x -= rectClient.left;
-            xy.y -= rectClient.top;
-            if ((ret = find_child_from_point( hwndScope, xy, hittest, MAKELONG( pt.x, pt.y ) )))
-            {
-                TRACE( "found child %p\n", ret );
-                return ret;
-            }
-        }
-    }
-
-    /* If nothing found, try the scope window */
-    if (!WIN_IsCurrentThread( hwndScope ))
-    {
-        *hittest = HTCLIENT;
-        TRACE( "returning %p\n", hwndScope );
-        return hwndScope;
-    }
-    res = SendMessageA( hwndScope, WM_NCHITTEST, 0, MAKELONG( pt.x, pt.y ) );
-    if (res != HTTRANSPARENT)
-    {
-        *hittest = res;  /* Found the window */
-        TRACE( "returning %p\n", hwndScope );
-        return hwndScope;
-    }
     *hittest = HTNOWHERE;
-    TRACE( "nothing found\n" );
-    return 0;
+
+    if (!(list = list_children_from_point( hwndScope, pt ))) return 0;
+
+    /* now determine the hittest */
+
+    for (i = 0; list[i]; i++)
+    {
+        LONG style = GetWindowLongW( list[i], GWL_STYLE );
+
+        /* If window is minimized or disabled, return at once */
+        if (style & WS_MINIMIZE)
+        {
+            *hittest = HTCAPTION;
+            break;
+        }
+        if (style & WS_DISABLED)
+        {
+            *hittest = HTERROR;
+            break;
+        }
+        /* Send WM_NCCHITTEST (if same thread) */
+        if (!WIN_IsCurrentThread( list[i] ))
+        {
+            *hittest = HTCLIENT;
+            break;
+        }
+        res = SendMessageA( list[i], WM_NCHITTEST, 0, MAKELONG(pt.x,pt.y) );
+        if (res != HTTRANSPARENT)
+        {
+            *hittest = res;  /* Found the window */
+            break;
+        }
+        /* continue search with next window in z-order */
+    }
+    ret = list[i];
+    HeapFree( GetProcessHeap(), 0, list );
+    TRACE( "scope %p (%ld,%ld) returning %p\n", hwndScope, pt.x, pt.y, ret );
+    return ret;
 }