]> git.mdlowis.com Git - proto/labwc.git/commitdiff
src/xdg-popup.c: delay popup unconstrain until after first commit
authorConsolatis <35009135+Consolatis@users.noreply.github.com>
Sat, 24 Feb 2024 22:16:55 +0000 (23:16 +0100)
committerJohan Malm <johanmalm@users.noreply.github.com>
Sun, 25 Feb 2024 11:49:03 +0000 (11:49 +0000)
Fixes: #1372
src/xdg-popup.c

index 496c5c0c170146126f9998c789376d9f9149342b..65c13e8df544cb2ff8ba76b1f8ae25ebb59b86e1 100644 (file)
@@ -16,15 +16,17 @@ struct xdg_popup {
        struct view *parent_view;
        struct wlr_xdg_popup *wlr_popup;
 
+       struct wl_listener commit;
        struct wl_listener destroy;
        struct wl_listener new_popup;
 };
 
 static void
-popup_unconstrain(struct view *view, struct wlr_xdg_popup *popup)
+popup_unconstrain(struct xdg_popup *popup)
 {
+       struct view *view = popup->parent_view;
        struct server *server = view->server;
-       struct wlr_box *popup_box = &popup->current.geometry;
+       struct wlr_box *popup_box = &popup->wlr_popup->current.geometry;
        struct wlr_output_layout *output_layout = server->output_layout;
        struct wlr_output *wlr_output = wlr_output_layout_output_at(
                output_layout, view->current.x + popup_box->x,
@@ -39,7 +41,7 @@ popup_unconstrain(struct view *view, struct wlr_xdg_popup *popup)
                .width = output_box.width,
                .height = output_box.height,
        };
-       wlr_xdg_popup_unconstrain_from_box(popup, &output_toplevel_box);
+       wlr_xdg_popup_unconstrain_from_box(popup->wlr_popup, &output_toplevel_box);
 }
 
 static void
@@ -48,9 +50,28 @@ handle_xdg_popup_destroy(struct wl_listener *listener, void *data)
        struct xdg_popup *popup = wl_container_of(listener, popup, destroy);
        wl_list_remove(&popup->destroy.link);
        wl_list_remove(&popup->new_popup.link);
+
+       /* Usually already removed unless there was no commit at all */
+       if (popup->commit.notify) {
+               wl_list_remove(&popup->commit.link);
+       }
        free(popup);
 }
 
+static void
+handle_xdg_popup_commit(struct wl_listener *listener, void *data)
+{
+       struct xdg_popup *popup = wl_container_of(listener, popup, commit);
+
+       if (popup->wlr_popup->base->initial_commit) {
+               popup_unconstrain(popup);
+
+               /* Prevent getting called over and over again */
+               wl_list_remove(&popup->commit.link);
+               popup->commit.notify = NULL;
+       }
+}
+
 static void
 popup_handle_new_xdg_popup(struct wl_listener *listener, void *data)
 {
@@ -75,9 +96,13 @@ xdg_popup_create(struct view *view, struct wlr_xdg_popup *wlr_popup)
 
        popup->destroy.notify = handle_xdg_popup_destroy;
        wl_signal_add(&wlr_popup->base->events.destroy, &popup->destroy);
+
        popup->new_popup.notify = popup_handle_new_xdg_popup;
        wl_signal_add(&wlr_popup->base->events.new_popup, &popup->new_popup);
 
+       popup->commit.notify = handle_xdg_popup_commit;
+       wl_signal_add(&wlr_popup->base->surface->events.commit, &popup->commit);
+
        /*
         * We must add xdg popups to the scene graph so they get rendered. The
         * wlroots scene graph provides a helper for this, but to use it we must
@@ -100,6 +125,4 @@ xdg_popup_create(struct view *view, struct wlr_xdg_popup *wlr_popup)
                wlr_scene_xdg_surface_create(parent_tree, wlr_popup->base);
        node_descriptor_create(wlr_popup->base->surface->data,
                LAB_NODE_DESC_XDG_POPUP, view);
-
-       popup_unconstrain(view, wlr_popup);
 }