gui-wm/swayfx: new package, add 0.3.2

Signed-off-by: Xin Yang <yangmame@icloud.com>
This commit is contained in:
Xin Yang 2024-03-26 09:19:14 +08:00 committed by 梁永祥
parent 33d6762856
commit 396c012ea7
4 changed files with 831 additions and 0 deletions

1
gui-wm/swayfx/Manifest Normal file
View File

@ -0,0 +1 @@
DIST swayfx-0.3.2.tar.gz 5684837 BLAKE2B ffe2a5d4a756f30007fb65824471196b6c51034a4e205558a82f547211714be8b0ae03763f32203c9d99fab4d3d69320f2244975a7c279d813091b8629d0c20a SHA512 0cffa25fc4b4d40621b14d9dc430f1aa9956ad935124cfda7dd1a63cbf997f0d506922b1168c7ad13250fa909cc1328526818f75ec936aa8d6cc40152dfbb358

View File

@ -0,0 +1,691 @@
From c9b99f407feab95418411ab20df5b0bfe167703a Mon Sep 17 00:00:00 2001
From: ShootingStarDragons <ShootingStarDragons@protonmail.com>
Date: Wed, 3 May 2023 14:21:27 +0800
Subject: [PATCH] swayfx-im
---
.gitignore | 1 +
include/sway/input/text_input.h | 24 +++
include/sway/output.h | 4 +
sway/desktop/output.c | 30 +++
sway/desktop/render.c | 17 +-
sway/input/cursor.c | 26 +++
sway/input/text_input.c | 351 +++++++++++++++++++++++++++++++-
sway/tree/container.c | 3 +
sway/tree/view.c | 3 +
9 files changed, 451 insertions(+), 8 deletions(-)
diff --git a/.gitignore b/.gitignore
index ba02e504c..0a1a8eed8 100644
--- a/.gitignore
+++ b/.gitignore
@@ -14,3 +14,4 @@ config-debug
wayland-*-protocol.*
/subprojects/wlroots
subprojects
+.cache
diff --git a/include/sway/input/text_input.h b/include/sway/input/text_input.h
index c70fd935b..f1c08f6be 100644
--- a/include/sway/input/text_input.h
+++ b/include/sway/input/text_input.h
@@ -22,18 +22,37 @@ struct sway_input_method_relay {
struct sway_seat *seat;
struct wl_list text_inputs; // sway_text_input::link
+ struct wl_list input_popups; // sway_input_popup::link
struct wlr_input_method_v2 *input_method; // doesn't have to be present
struct wl_listener text_input_new;
struct wl_listener input_method_new;
struct wl_listener input_method_commit;
+ struct wl_listener input_method_new_popup_surface;
struct wl_listener input_method_grab_keyboard;
struct wl_listener input_method_destroy;
struct wl_listener input_method_keyboard_grab_destroy;
};
+struct sway_input_popup {
+ struct sway_input_method_relay *relay;
+ struct wlr_input_popup_surface_v2 *popup_surface;
+
+ int x, y;
+ bool visible;
+
+ struct wl_list link;
+
+ struct wl_listener popup_map;
+ struct wl_listener popup_unmap;
+ struct wl_listener popup_destroy;
+ struct wl_listener popup_surface_commit;
+
+ struct wl_listener focused_surface_unmap;
+};
+
struct sway_text_input {
struct sway_input_method_relay *relay;
@@ -66,4 +85,9 @@ struct sway_text_input *sway_text_input_create(
struct sway_input_method_relay *relay,
struct wlr_text_input_v3 *text_input);
+bool sway_input_popup_get_position(
+ struct sway_input_popup *popup, int *lx, int *ly);
+
+void sway_input_popup_damage(struct sway_input_popup *popup);
+
#endif
diff --git a/include/sway/output.h b/include/sway/output.h
index 3215c8531..b2e1f60b7 100644
--- a/include/sway/output.h
+++ b/include/sway/output.h
@@ -154,6 +154,10 @@ void output_unmanaged_for_each_surface(struct sway_output *output,
void *user_data);
#endif
+void output_input_popups_for_each_surface(struct sway_output *output,
+ struct wl_list *input_popups, sway_surface_iterator_func_t iterator,
+ void *user_data);
+
void output_drag_icons_for_each_surface(struct sway_output *output,
struct wl_list *drag_icons, sway_surface_iterator_func_t iterator,
void *user_data);
diff --git a/sway/desktop/output.c b/sway/desktop/output.c
index c41088acc..727fda29f 100644
--- a/sway/desktop/output.c
+++ b/sway/desktop/output.c
@@ -22,6 +22,8 @@
#include "sway/desktop/transaction.h"
#include "sway/input/input-manager.h"
#include "sway/input/seat.h"
+#include "sway/input/text_input.h"
+#include "sway/ipc-server.h"
#include "sway/layers.h"
#include "sway/output.h"
#include "sway/server.h"
@@ -254,6 +256,31 @@ void output_unmanaged_for_each_surface(struct sway_output *output,
}
#endif
+void output_input_popups_for_each_surface(struct sway_output *output,
+ struct wl_list *input_popups, sway_surface_iterator_func_t iterator,
+ void *user_data) {
+ struct sway_input_popup *popup;
+ wl_list_for_each(popup, input_popups, link) {
+ int lx, ly;
+ if (!sway_input_popup_get_position(popup, &lx, &ly)) {
+ continue;
+ }
+ if (!popup->popup_surface->mapped || !popup->visible) {
+ continue;
+ }
+
+ // damage the popup if surface is updated
+ sway_input_popup_damage(popup);
+
+ double ox = lx - output->lx;
+ double oy = ly - output->ly;
+
+ output_surface_for_each_surface(output,
+ popup->popup_surface->surface, ox, oy,
+ iterator, user_data);
+ }
+}
+
void output_drag_icons_for_each_surface(struct sway_output *output,
struct wl_list *drag_icons, sway_surface_iterator_func_t iterator,
void *user_data) {
@@ -364,6 +391,9 @@ static void output_for_each_surface(struct sway_output *output,
output_layer_for_each_surface(output,
&output->layers[ZWLR_LAYER_SHELL_V1_LAYER_OVERLAY],
iterator, user_data);
+ output_input_popups_for_each_surface(
+ output, &input_manager_current_seat()->im_relay.input_popups,
+ iterator, user_data);
output_drag_icons_for_each_surface(output, &root->drag_icons,
iterator, user_data);
}
diff --git a/sway/desktop/render.c b/sway/desktop/render.c
index 43abc81c2..d9565e54d 100644
--- a/sway/desktop/render.c
+++ b/sway/desktop/render.c
@@ -526,6 +526,16 @@ static void render_unmanaged(struct sway_output *output,
}
#endif
+static void render_input_popups(struct sway_output *output,
+ pixman_region32_t *damage, struct wl_list *input_popups) {
+ struct render_data data = {
+ .damage = damage,
+ .deco_data = get_undecorated_decoration_data(),
+ };
+ output_input_popups_for_each_surface(output, input_popups,
+ render_surface_iterator, &data);
+}
+
static void render_drag_icons(struct sway_output *output,
pixman_region32_t *damage, struct wl_list *drag_icons) {
struct render_data data = {
@@ -1859,6 +1869,10 @@ void output_render(struct sway_output *output, struct timespec *when,
goto renderer_end;
}
+ struct sway_seat *seat = input_manager_current_seat();
+ struct sway_container *focus = seat_get_focused_container(seat);
+ // here use seat
+
if (output_has_opaque_overlay_layer_surface(output)) {
goto render_overlay;
}
@@ -1986,8 +2000,6 @@ void output_render(struct sway_output *output, struct timespec *when,
render_seatops(output, damage);
- struct sway_seat *seat = input_manager_current_seat();
- struct sway_container *focus = seat_get_focused_container(seat);
if (focus && focus->view) {
struct decoration_data deco_data = {
.alpha = focus->alpha,
@@ -2010,6 +2022,7 @@ void output_render(struct sway_output *output, struct timespec *when,
&output->layers[ZWLR_LAYER_SHELL_V1_LAYER_OVERLAY]);
render_layer_popups(output, damage,
&output->layers[ZWLR_LAYER_SHELL_V1_LAYER_OVERLAY]);
+ render_input_popups(output, damage, &seat->im_relay.input_popups);
render_drag_icons(output, damage, &root->drag_icons);
renderer_end:
diff --git a/sway/input/cursor.c b/sway/input/cursor.c
index a5f1204ba..ed2188cd4 100644
--- a/sway/input/cursor.c
+++ b/sway/input/cursor.c
@@ -37,6 +37,28 @@ static uint32_t get_current_time_msec(void) {
return now.tv_sec * 1000 + now.tv_nsec / 1000000;
}
+static struct wlr_surface *input_popup_surface_at(struct sway_output *output,
+ struct sway_input_method_relay *relay, double ox, double oy, double *sx, double *sy) {
+ struct sway_input_popup *popup;
+ wl_list_for_each(popup, &relay->input_popups, link) {
+ int lx, ly;
+ if (!sway_input_popup_get_position(popup, &lx, &ly)) {
+ continue;
+ }
+ if (!popup->popup_surface->mapped || !popup->visible) {
+ continue;
+ }
+ double _sx = ox - lx;
+ double _sy = oy - ly;
+ struct wlr_surface *sub = wlr_surface_surface_at(
+ popup->popup_surface->surface, _sx, _sy, sx, sy);
+ if (sub) {
+ return sub;
+ }
+ }
+ return NULL;
+}
+
static struct wlr_surface *layer_surface_at(struct sway_output *output,
struct wl_list *layer, double ox, double oy, double *sx, double *sy) {
struct sway_layer_surface *sway_layer;
@@ -122,6 +144,10 @@ struct sway_node *node_at_coords(
return NULL;
}
+ if ((*surface = input_popup_surface_at(output,
+ &seat->im_relay, ox, oy, sx, sy))) {
+ return NULL;
+ }
// check for unmanaged views
#if HAVE_XWAYLAND
struct wl_list *unmanaged = &root->xwayland_unmanaged;
diff --git a/sway/input/text_input.c b/sway/input/text_input.c
index 58911c2de..9ddebb1b3 100644
--- a/sway/input/text_input.c
+++ b/sway/input/text_input.c
@@ -1,8 +1,16 @@
#include <assert.h>
+#include <stdio.h>
#include <stdlib.h>
#include "log.h"
#include "sway/input/seat.h"
#include "sway/input/text_input.h"
+#include "sway/tree/view.h"
+#include "sway/tree/container.h"
+#include "sway/desktop.h"
+#include "sway/output.h"
+#include "sway/layers.h"
+
+#define wlr_layer_surface_v1_try_from_wlr_surface(surface) (wlr_surface_is_layer_surface(surface) ? wlr_layer_surface_v1_from_wlr_surface(surface) : NULL)
static struct sway_text_input *relay_get_focusable_text_input(
struct sway_input_method_relay *relay) {
@@ -26,6 +34,328 @@ static struct sway_text_input *relay_get_focused_text_input(
return NULL;
}
+// damage the popup
+void sway_input_popup_damage(struct sway_input_popup *popup) {
+ if (!popup->visible) {
+ return;
+ }
+
+ struct sway_text_input *text_input =
+ relay_get_focused_text_input(popup->relay);
+ if (text_input == NULL || text_input->input->focused_surface == NULL) {
+ return;
+ }
+
+ struct wlr_surface *focused_surface = text_input->input->focused_surface;
+ struct wlr_layer_surface_v1 *layer_surface = wlr_layer_surface_v1_try_from_wlr_surface(focused_surface);
+ if (layer_surface != NULL) {
+ struct sway_layer_surface *layer =
+ layer_from_wlr_layer_surface_v1(layer_surface);
+ output_damage_surface(layer->layer_surface->output->data,
+ layer->geo.x + popup->x, layer->geo.y + popup->y,
+ popup->popup_surface->surface, true);
+ return;
+ }
+
+ struct sway_view *view = view_from_wlr_surface(
+ text_input->input->focused_surface);
+ if (view->container == NULL) {
+ sway_log(SWAY_INFO, "Tried to damage popup, but view is gone");
+ return;
+ }
+ desktop_damage_surface(popup->popup_surface->surface,
+ view->container->surface_x - view->geometry.x + popup->x,
+ view->container->surface_y - view->geometry.y + popup->y, true);
+}
+
+static void input_popup_update(struct sway_input_popup *popup) {
+ sway_input_popup_damage(popup);
+
+ struct sway_text_input *text_input =
+ relay_get_focused_text_input(popup->relay);
+
+ if (text_input == NULL || text_input->input->focused_surface == NULL) {
+ return;
+ }
+
+ if (!popup->popup_surface->mapped) {
+ return;
+ }
+
+ bool cursor_rect = text_input->input->current.features
+ & WLR_TEXT_INPUT_V3_FEATURE_CURSOR_RECTANGLE;
+ struct wlr_surface *focused_surface = text_input->input->focused_surface;
+ struct wlr_box cursor = text_input->input->current.cursor_rectangle;
+
+ struct wlr_output *output;
+ struct wlr_box output_box;
+ struct wlr_box parent;
+ struct wlr_layer_surface_v1 *layer_surface =
+ wlr_layer_surface_v1_try_from_wlr_surface(focused_surface);
+ if (layer_surface != NULL) {
+ struct sway_layer_surface *layer =
+ layer_from_wlr_layer_surface_v1(layer_surface);
+ output = layer->layer_surface->output;
+ wlr_output_layout_get_box(root->output_layout, output, &output_box);
+ parent = layer->geo;
+ parent.x += output_box.x;
+ parent.y += output_box.y;
+ } else {
+ struct sway_view *view = view_from_wlr_surface(focused_surface);
+ output = wlr_output_layout_output_at(root->output_layout,
+ view->container->surface_x + view->geometry.x,
+ view->container->surface_y + view->geometry.y);
+ wlr_output_layout_get_box(root->output_layout, output, &output_box);
+ parent.x = view->container->surface_x + view->geometry.x;
+ parent.y = view->container->surface_y + view->geometry.y;
+ parent.width = view->geometry.width;
+ parent.height = view->geometry.height;
+ }
+
+ if (!cursor_rect) {
+ cursor.x = 0;
+ cursor.y = 0;
+ cursor.width = parent.width;
+ cursor.height = parent.height;
+ }
+
+ int popup_width = popup->popup_surface->surface->current.width;
+ int popup_height = popup->popup_surface->surface->current.height;
+ int x1 = parent.x + cursor.x;
+ int x2 = parent.x + cursor.x + cursor.width;
+ int y1 = parent.y + cursor.y;
+ int y2 = parent.y + cursor.y + cursor.height;
+ int x = x1;
+ int y = y2;
+
+ int available_right = output_box.x + output_box.width - x1;
+ int available_left = x2 - output_box.x;
+ if (available_right < popup_width && available_left > available_right) {
+ x = x2 - popup_width;
+ }
+
+ int available_down = output_box.y + output_box.height - y2;
+ int available_up = y1 - output_box.y;
+ if (available_down < popup_height && available_up > available_down) {
+ y = y1 - popup_height;
+ }
+
+ popup->x = x - parent.x;
+ popup->y = y - parent.y;
+
+ // Hide popup if cursor position is completely out of bounds
+ bool x1_in_bounds = (cursor.x >= 0 && cursor.x < parent.width);
+ bool y1_in_bounds = (cursor.y >= 0 && cursor.y < parent.height);
+ bool x2_in_bounds = (cursor.x + cursor.width >= 0
+ && cursor.x + cursor.width < parent.width);
+ bool y2_in_bounds = (cursor.y + cursor.height >= 0
+ && cursor.y + cursor.height < parent.height);
+ popup->visible =
+ (x1_in_bounds && y1_in_bounds) || (x2_in_bounds && y2_in_bounds);
+
+ if (cursor_rect) {
+ struct wlr_box box = {
+ .x = x1 - x,
+ .y = y1 - y,
+ .width = cursor.width,
+ .height = cursor.height,
+ };
+ wlr_input_popup_surface_v2_send_text_input_rectangle(
+ popup->popup_surface, &box);
+ }
+
+ sway_input_popup_damage(popup);
+}
+
+static void surface_send_enter_iterator(struct wlr_surface *surface,
+ int x, int y, void *data) {
+ struct wlr_output *wlr_output = data;
+ wlr_surface_send_enter(surface, wlr_output);
+}
+
+static void surface_send_leave_iterator(struct wlr_surface *surface,
+ int x, int y, void *data) {
+ struct wlr_output *wlr_output = data;
+ wlr_surface_send_leave(surface, wlr_output);
+}
+
+static void input_popup_send_outputs(struct sway_input_popup *popup,
+ wlr_surface_iterator_func_t iterator) {
+ struct sway_text_input *text_input =
+ relay_get_focused_text_input(popup->relay);
+ if (text_input == NULL || text_input->input->focused_surface == NULL) {
+ return;
+ }
+ struct wlr_surface *focused_surface = text_input->input->focused_surface;
+ struct wlr_layer_surface_v1 *layer_surface =
+ wlr_layer_surface_v1_try_from_wlr_surface(focused_surface);
+ if (layer_surface != NULL) {
+ struct sway_layer_surface *layer =
+ layer_from_wlr_layer_surface_v1(layer_surface);
+ wlr_surface_for_each_surface(popup->popup_surface->surface,
+ iterator, layer->layer_surface->output);
+ return;
+ }
+ struct sway_view *view = view_from_wlr_surface(focused_surface);
+ for (int i = 0; i < view->container->outputs->length; i++) {
+ struct sway_output *output = view->container->outputs->items[i];
+ wlr_surface_for_each_surface(popup->popup_surface->surface,
+ iterator, output->wlr_output);
+ }
+}
+
+static void handle_im_popup_map(struct wl_listener *listener, void *data) {
+ struct sway_input_popup *popup =
+ wl_container_of(listener, popup, popup_map);
+ input_popup_send_outputs(popup, surface_send_enter_iterator);
+ input_popup_update(popup);
+}
+
+static void handle_im_popup_unmap(struct wl_listener *listener, void *data) {
+ struct sway_input_popup *popup =
+ wl_container_of(listener, popup, popup_unmap);
+ input_popup_send_outputs(popup, surface_send_leave_iterator);
+ input_popup_update(popup);
+}
+
+static void handle_im_popup_destroy(struct wl_listener *listener, void *data) {
+ struct sway_input_popup *popup =
+ wl_container_of(listener, popup, popup_destroy);
+ wl_list_remove(&popup->focused_surface_unmap.link);
+ wl_list_remove(&popup->popup_surface_commit.link);
+ wl_list_remove(&popup->popup_destroy.link);
+ wl_list_remove(&popup->popup_unmap.link);
+ wl_list_remove(&popup->popup_map.link);
+ wl_list_remove(&popup->link);
+ free(popup);
+}
+
+static void handle_im_popup_surface_commit(struct wl_listener *listener,
+ void *data) {
+ struct sway_input_popup *popup =
+ wl_container_of(listener, popup, popup_surface_commit);
+ input_popup_update(popup);
+}
+
+static void handle_im_focused_surface_unmap(
+ struct wl_listener *listener, void *data) {
+ struct sway_input_popup *popup =
+ wl_container_of(listener, popup, focused_surface_unmap);
+ input_popup_update(popup);
+}
+
+static void input_popup_set_focus(struct sway_input_popup *popup,
+ struct wlr_surface *surface) {
+ wl_list_remove(&popup->focused_surface_unmap.link);
+
+ if (surface == NULL) {
+ wl_list_init(&popup->focused_surface_unmap.link);
+ input_popup_update(popup);
+ return;
+ }
+ struct wlr_layer_surface_v1 *layer_surface =
+ wlr_layer_surface_v1_try_from_wlr_surface(surface);
+ if (layer_surface != NULL) {
+ struct sway_layer_surface *layer =
+ layer_from_wlr_layer_surface_v1(layer_surface);
+ wl_signal_add(
+ &layer->layer_surface->events.unmap, &popup->focused_surface_unmap);
+ input_popup_update(popup);
+ return;
+ }
+
+ struct sway_view *view = view_from_wlr_surface(surface);
+ wl_signal_add(&view->events.unmap, &popup->focused_surface_unmap);
+
+ // Since the focus has changed, the popup may have to adjust
+}
+
+static void handle_im_new_popup_surface(struct wl_listener *listener,
+ void *data) {
+ struct sway_input_method_relay *relay = wl_container_of(listener, relay,
+ input_method_new_popup_surface);
+ struct sway_input_popup *popup = calloc(1, sizeof(*popup));
+ popup->relay = relay;
+ popup->popup_surface = data;
+ popup->popup_surface->data = popup;
+
+ wl_signal_add(&popup->popup_surface->events.map, &popup->popup_map);
+ popup->popup_map.notify = handle_im_popup_map;
+ wl_signal_add(
+ &popup->popup_surface->events.unmap, &popup->popup_unmap);
+ popup->popup_unmap.notify = handle_im_popup_unmap;
+ wl_signal_add(
+ &popup->popup_surface->events.destroy, &popup->popup_destroy);
+ popup->popup_destroy.notify = handle_im_popup_destroy;
+ wl_signal_add(&popup->popup_surface->surface->events.commit,
+ &popup->popup_surface_commit);
+ popup->popup_surface_commit.notify = handle_im_popup_surface_commit;
+ wl_list_init(&popup->focused_surface_unmap.link);
+ popup->focused_surface_unmap.notify = handle_im_focused_surface_unmap;
+
+ struct sway_text_input *text_input = relay_get_focused_text_input(relay);
+ if (text_input != NULL) {
+ input_popup_set_focus(popup, text_input->input->focused_surface);
+ } else {
+ input_popup_set_focus(popup, NULL);
+ }
+
+ wl_list_insert(&relay->input_popups, &popup->link);
+}
+
+bool sway_input_popup_get_position(
+ struct sway_input_popup *popup, int *lx, int *ly) {
+ struct sway_text_input *text_input =
+ relay_get_focused_text_input(popup->relay);
+ if (text_input == NULL || text_input->input->focused_surface == NULL) {
+ *lx = 0;
+ *ly = 0;
+ return false;
+ }
+
+ struct wlr_surface *focused_surface = text_input->input->focused_surface;
+ struct wlr_layer_surface_v1 *layer_surface =
+ wlr_layer_surface_v1_try_from_wlr_surface(focused_surface);
+ if (layer_surface != NULL) {
+ struct sway_layer_surface *layer =
+ layer_from_wlr_layer_surface_v1(layer_surface);
+ *lx = layer->geo.x + popup->x;
+ *ly = layer->geo.y + popup->y;
+ return true;
+ }
+
+
+ struct sway_view *view = view_from_wlr_surface(
+ text_input->input->focused_surface);
+ if (view->container == NULL || view == NULL) {
+ sway_log(SWAY_INFO, "Tried to find popup, but view is gone");
+ return false;
+ }
+ *lx = view->container->surface_x - view->geometry.x + popup->x;
+ *ly = view->container->surface_y - view->geometry.y + popup->y;
+ return true;
+}
+
+static void text_input_send_enter(struct sway_text_input *text_input,
+ struct wlr_surface *surface) {
+ wlr_text_input_v3_send_enter(text_input->input, surface);
+ struct sway_input_popup *popup;
+ wl_list_for_each(popup, &text_input->relay->input_popups, link) {
+ input_popup_set_focus(popup, surface);
+ }
+}
+
+static void text_input_send_leave(struct sway_text_input *text_input,
+ struct wlr_surface *surface) {
+ wlr_text_input_v3_send_leave(text_input->input);
+ if (text_input->input->focused_surface == surface) {
+ struct sway_input_popup *popup;
+ wl_list_for_each(popup, &text_input->relay->input_popups, link) {
+ input_popup_set_focus(popup, NULL);
+ }
+ }
+}
+
static void handle_im_commit(struct wl_listener *listener, void *data) {
struct sway_input_method_relay *relay = wl_container_of(listener, relay,
input_method_commit);
@@ -109,7 +439,7 @@ static void handle_im_destroy(struct wl_listener *listener, void *data) {
// the input method returns
text_input_set_pending_focused_surface(text_input,
text_input->input->focused_surface);
- wlr_text_input_v3_send_leave(text_input->input);
+ text_input_send_leave(text_input, text_input->input->focused_surface);
}
}
@@ -133,8 +463,13 @@ static void relay_send_im_state(struct sway_input_method_relay *relay,
input->current.content_type.hint,
input->current.content_type.purpose);
}
+ struct sway_input_popup *popup;
+ wl_list_for_each(popup, &relay->input_popups, link) {
+ // send_text_input_rectangle is called in this function
+ input_popup_update(popup);
+ }
wlr_input_method_v2_send_done(input_method);
- // TODO: pass intent, display popup size
+ // TODO: pass intent
}
static void handle_text_input_enable(struct wl_listener *listener, void *data) {
@@ -274,6 +609,9 @@ static void relay_handle_input_method(struct wl_listener *listener,
wl_signal_add(&relay->input_method->events.commit,
&relay->input_method_commit);
relay->input_method_commit.notify = handle_im_commit;
+ wl_signal_add(&relay->input_method->events.new_popup_surface,
+ &relay->input_method_new_popup_surface);
+ relay->input_method_new_popup_surface.notify = handle_im_new_popup_surface;
wl_signal_add(&relay->input_method->events.grab_keyboard,
&relay->input_method_grab_keyboard);
relay->input_method_grab_keyboard.notify = handle_im_grab_keyboard;
@@ -283,8 +621,7 @@ static void relay_handle_input_method(struct wl_listener *listener,
struct sway_text_input *text_input = relay_get_focusable_text_input(relay);
if (text_input) {
- wlr_text_input_v3_send_enter(text_input->input,
- text_input->pending_focused_surface);
+ text_input_send_enter(text_input, text_input->pending_focused_surface);
text_input_set_pending_focused_surface(text_input, NULL);
}
}
@@ -293,6 +630,7 @@ void sway_input_method_relay_init(struct sway_seat *seat,
struct sway_input_method_relay *relay) {
relay->seat = seat;
wl_list_init(&relay->text_inputs);
+ wl_list_init(&relay->input_popups);
relay->text_input_new.notify = relay_handle_text_input;
wl_signal_add(&server.text_input->events.text_input,
@@ -321,8 +659,9 @@ void sway_input_method_relay_set_focus(struct sway_input_method_relay *relay,
} else if (text_input->input->focused_surface) {
assert(text_input->pending_focused_surface == NULL);
if (surface != text_input->input->focused_surface) {
+ text_input_send_leave(
+ text_input, text_input->input->focused_surface);
relay_disable_text_input(relay, text_input);
- wlr_text_input_v3_send_leave(text_input->input);
} else {
sway_log(SWAY_DEBUG, "IM relay set_focus already focused");
continue;
@@ -333,7 +672,7 @@ void sway_input_method_relay_set_focus(struct sway_input_method_relay *relay,
&& wl_resource_get_client(text_input->input->resource)
== wl_resource_get_client(surface->resource)) {
if (relay->input_method) {
- wlr_text_input_v3_send_enter(text_input->input, surface);
+ text_input_send_enter(text_input, surface);
} else {
text_input_set_pending_focused_surface(text_input, surface);
}
diff --git a/sway/tree/container.c b/sway/tree/container.c
index 2e61ff5c9..14d0e6079 100644
--- a/sway/tree/container.c
+++ b/sway/tree/container.c
@@ -400,6 +400,9 @@ static bool surface_is_popup(struct wlr_surface *surface) {
wlr_subsurface_from_wlr_surface(surface);
surface = subsurface->parent;
}
+ if (wlr_surface_is_input_popup_surface_v2(surface)) {
+ return true;
+ }
struct wlr_xdg_surface *xdg_surface =
wlr_xdg_surface_from_wlr_surface(surface);
return xdg_surface->role == WLR_XDG_SURFACE_ROLE_POPUP;
diff --git a/sway/tree/view.c b/sway/tree/view.c
index 272967f4a..70d02a199 100644
--- a/sway/tree/view.c
+++ b/sway/tree/view.c
@@ -1249,6 +1249,9 @@ struct sway_view *view_from_wlr_surface(struct wlr_surface *wlr_surface) {
if (wlr_surface_is_layer_surface(wlr_surface)) {
return NULL;
}
+ if (wlr_surface_is_input_popup_surface_v2(wlr_surface)) {
+ return NULL;
+ }
const char *role = wlr_surface->role ? wlr_surface->role->name : NULL;
sway_log(SWAY_DEBUG, "Surface of unknown type (role %s): %p",

View File

@ -0,0 +1,36 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE pkgmetadata SYSTEM "https://www.gentoo.org/dtd/metadata.dtd">
<pkgmetadata>
<maintainer type="person">
<email>yangmame@icloud.com</email>
<name>Xin Yang</name>
</maintainer>
<longdescription lang="en">
"SirCmpwn's Wayland window manager" is an i3-compatible window manager
for Wayland.
sway was created to fill the need of an i3-like window manager for
Wayland. The upstream i3 developers have no intention of porting i3 to
Wayland, and projects proposed by others ended up as vaporware. Many
thanks to the i3 folks for providing such a great piece of software, so
good that your users would rather write an entirely new window manager
from scratch that behaved *exactly* like i3 rather than switch to
something else. Swayfx is a fork intending to bring more effects to sway.
</longdescription>
<use>
<flag name="grimshot">Install 'grimshot': script for taking screenshots</flag>
<flag name="swaybar">Install 'swaybar': sway's status bar component</flag>
<flag name="swaynag">Install 'swaynag': shows a message with buttons</flag>
<flag name="tray">Enable support for StatusNotifierItem tray specification</flag>
<flag name="wallpapers">Install sway's default wallpaper image</flag>
<flag name="X">Enable support for X11 applications (XWayland)</flag>
</use>
<upstream>
<remote-id type="github">WillPower3309/swayfx</remote-id>
<maintainer status="active">
<name>William McKinnon</name>
</maintainer>
<bugs-to>https://github.com/WillPower3309/swayfx/issues</bugs-to>
<changelog>https://github.com/WillPower3309/swayfx/releases</changelog>
</upstream>
</pkgmetadata>

View File

@ -0,0 +1,103 @@
# Copyright 1999-2022 Gentoo Authors
# Distributed under the terms of the GNU General Public License v2
EAPI=8
inherit meson optfeature
DESCRIPTION="SwayFX: Sway, but with eye candy!"
HOMEPAGE="https://github.com/WillPower3309/swayfx"
SRC_URI="https://github.com/WillPower3309/swayfx/archive/refs/tags/${PV}.tar.gz -> ${P}.tar.gz"
LICENSE="MIT"
SLOT="0"
KEYWORDS="~amd64"
IUSE="grimshot +man +swaybar +swaynag tray wallpapers X"
DEPEND="
>=dev-libs/json-c-0.13:0=
>=dev-libs/libinput-1.21.0:0=
sys-auth/seatd:=
dev-libs/libpcre
>=dev-libs/wayland-1.20.0
x11-libs/cairo
x11-libs/libxkbcommon
x11-libs/pango
x11-libs/pixman
media-libs/mesa[gles2,libglvnd(+)]
swaybar? ( x11-libs/gdk-pixbuf:2 )
tray? ( || (
sys-apps/systemd
sys-auth/elogind
sys-libs/basu
) )
wallpapers? ( gui-apps/swaybg[gdk-pixbuf(+)] )
X? ( x11-libs/libxcb:0= )
"
DEPEND+="
>=gui-libs/wlroots-0.16:=[X?]
<gui-libs/wlroots-0.17:=[X?]
"
RDEPEND="
x11-misc/xkeyboard-config
grimshot? (
app-misc/jq
gui-apps/grim
gui-apps/slurp
gui-apps/wl-clipboard
x11-libs/libnotify
)
!!gui-wm/sway
${DEPEND}
"
BDEPEND="
>=dev-libs/wayland-protocols-1.24
>=dev-build/meson-0.60.0
virtual/pkgconfig
"
BDEPEND+="man? ( >=app-text/scdoc-1.9.3 )"
REQUIRED_USE="tray? ( swaybar )"
src_prepare() {
eapply "${FILESDIR}/${P}-im.patch"
default
}
src_configure() {
local emesonargs=(
$(meson_feature man man-pages)
$(meson_feature tray)
$(meson_feature X xwayland)
$(meson_feature swaybar gdk-pixbuf)
$(meson_use swaynag)
$(meson_use swaybar)
$(meson_use wallpapers default-wallpaper)
-Dfish-completions=true
-Dzsh-completions=true
-Dbash-completions=true
)
meson_src_configure
}
src_install() {
meson_src_install
if use grimshot; then
doman contrib/grimshot.1
dobin contrib/grimshot
fi
}
pkg_postinst() {
optfeature_header "There are several packages that may be useful with swayfx:"
optfeature "wallpaper utility" gui-apps/swaybg
optfeature "idle management utility" gui-apps/swayidle
optfeature "simple screen locker" gui-apps/swaylock
optfeature "lightweight notification daemon" gui-apps/mako
echo
einfo "For a list of additional addons and tools usable with sway please"
einfo "visit the offical wiki at:"
einfo "https://github.com/swaywm/sway/wiki/Useful-add-ons-for-sway"
einfo "Please note that some of them might not (yet) available on gentoo"
}