mirror of
https://codeberg.org/dwl/dwl.git
synced 2025-11-08 13:29:53 -05:00
implement foreign-toplevel-management
This commit is contained in:
parent
577d8da6d1
commit
09d42d9834
5 changed files with 468 additions and 6 deletions
182
dwl.c
182
dwl.c
|
|
@ -22,6 +22,7 @@
|
|||
#include <wlr/types/wlr_data_device.h>
|
||||
#include <wlr/types/wlr_drm.h>
|
||||
#include <wlr/types/wlr_export_dmabuf_v1.h>
|
||||
#include <wlr/types/wlr_foreign_toplevel_management_v1.h>
|
||||
#include <wlr/types/wlr_fractional_scale_v1.h>
|
||||
#include <wlr/types/wlr_gamma_control_v1.h>
|
||||
#include <wlr/types/wlr_idle_inhibit_v1.h>
|
||||
|
|
@ -71,7 +72,7 @@
|
|||
#define MIN(A, B) ((A) < (B) ? (A) : (B))
|
||||
#define ROUND(X) ((int)((X < 0) ? (X - 0.5) : (X + 0.5)))
|
||||
#define CLEANMASK(mask) (mask & ~WLR_MODIFIER_CAPS)
|
||||
#define VISIBLEON(C, M) ((M) && (C)->mon == (M) && ((C)->tags & (M)->tagset[(M)->seltags]))
|
||||
#define VISIBLEON(C, M) ((M) && (C)->mon == (M) && ((C)->tags & (M)->tagset[(M)->seltags]) && (!(C)->isminimized))
|
||||
#define LENGTH(X) (sizeof X / sizeof X[0])
|
||||
#define END(A) ((A) + LENGTH(A))
|
||||
#define TAGMASK ((1u << TAGCOUNT) - 1)
|
||||
|
|
@ -120,12 +121,19 @@ typedef struct {
|
|||
struct wl_listener commit;
|
||||
struct wl_listener map;
|
||||
struct wl_listener maximize;
|
||||
struct wl_listener minimize;
|
||||
struct wl_listener unmap;
|
||||
struct wl_listener destroy;
|
||||
struct wl_listener set_title;
|
||||
struct wl_listener fullscreen;
|
||||
struct wl_listener set_decoration_mode;
|
||||
struct wl_listener destroy_decoration;
|
||||
struct wlr_foreign_toplevel_handle_v1 *foreign_toplevel;
|
||||
struct wl_listener foreign_activate_request;
|
||||
struct wl_listener foreign_close_request;
|
||||
struct wl_listener foreign_fullscreen_request;
|
||||
struct wl_listener foreign_destroy;
|
||||
struct wl_listener foreign_minimize;
|
||||
struct wlr_box prev; /* layout-relative, includes border */
|
||||
struct wlr_box bounds;
|
||||
#ifdef XWAYLAND
|
||||
|
|
@ -137,7 +145,7 @@ typedef struct {
|
|||
#endif
|
||||
unsigned int bw;
|
||||
uint32_t tags;
|
||||
int isfloating, isurgent, isfullscreen;
|
||||
int isfloating, isurgent, isfullscreen, isminimized;
|
||||
uint32_t resize; /* configure serial of a pending resize */
|
||||
} Client;
|
||||
|
||||
|
|
@ -295,6 +303,7 @@ static void locksession(struct wl_listener *listener, void *data);
|
|||
static void maplayersurfacenotify(struct wl_listener *listener, void *data);
|
||||
static void mapnotify(struct wl_listener *listener, void *data);
|
||||
static void maximizenotify(struct wl_listener *listener, void *data);
|
||||
static void minimizenotify(struct wl_listener *listener, void *data);
|
||||
static void monocle(Monitor *m);
|
||||
static void motionabsolute(struct wl_listener *listener, void *data);
|
||||
static void motionnotify(uint32_t time, struct wlr_input_device *device, double sx,
|
||||
|
|
@ -318,6 +327,7 @@ static void setcursor(struct wl_listener *listener, void *data);
|
|||
static void setcursorshape(struct wl_listener *listener, void *data);
|
||||
static void setfloating(Client *c, int floating);
|
||||
static void setfullscreen(Client *c, int fullscreen);
|
||||
static void setminimized(Client *c, int minimized);
|
||||
static void setgamma(struct wl_listener *listener, void *data);
|
||||
static void setlayout(const Arg *arg);
|
||||
static void setmfact(const Arg *arg);
|
||||
|
|
@ -332,6 +342,7 @@ static void tagmon(const Arg *arg);
|
|||
static void tile(Monitor *m);
|
||||
static void togglefloating(const Arg *arg);
|
||||
static void togglefullscreen(const Arg *arg);
|
||||
static void toggleminimized(const Arg *arg);
|
||||
static void toggletag(const Arg *arg);
|
||||
static void toggleview(const Arg *arg);
|
||||
static void unlocksession(struct wl_listener *listener, void *data);
|
||||
|
|
@ -347,6 +358,11 @@ static Monitor *xytomon(double x, double y);
|
|||
static void xytonode(double x, double y, struct wlr_surface **psurface,
|
||||
Client **pc, LayerSurface **pl, double *nx, double *ny);
|
||||
static void zoom(const Arg *arg);
|
||||
static void handle_foreign_activate_request(struct wl_listener *listener, void *data);
|
||||
static void handle_foreign_close_request(struct wl_listener *listener, void *data);
|
||||
static void handle_foreign_destroy(struct wl_listener *listener, void *data);
|
||||
static void handle_foreign_fullscreen_request(struct wl_listener *listener, void *data);
|
||||
static void handle_foreign_minimize(struct wl_listener *listener, void *data);
|
||||
|
||||
/* variables */
|
||||
static const char broken[] = "broken";
|
||||
|
|
@ -392,6 +408,8 @@ static struct wlr_scene_rect *locked_bg;
|
|||
static struct wlr_session_lock_v1 *cur_lock;
|
||||
static struct wl_listener lock_listener = {.notify = locksession};
|
||||
|
||||
static struct wlr_foreign_toplevel_manager_v1 *foreign_toplevel_manager;
|
||||
|
||||
static struct wlr_seat *seat;
|
||||
static KeyboardGroup kb_group = {0};
|
||||
static KeyboardGroup vkb_group = {0};
|
||||
|
|
@ -458,6 +476,9 @@ applyrules(Client *c)
|
|||
if (!(title = client_get_title(c)))
|
||||
title = broken;
|
||||
|
||||
wlr_foreign_toplevel_handle_v1_set_app_id(c->foreign_toplevel, appid);
|
||||
wlr_foreign_toplevel_handle_v1_set_title(c->foreign_toplevel, title);
|
||||
|
||||
for (r = rules; r < END(rules); r++) {
|
||||
if ((!r->title || strstr(title, r->title))
|
||||
&& (!r->id || strstr(appid, r->id))) {
|
||||
|
|
@ -975,7 +996,8 @@ createnotify(struct wl_listener *listener, void *data)
|
|||
c->bw = borderpx;
|
||||
|
||||
wlr_xdg_toplevel_set_wm_capabilities(xdg_surface->toplevel,
|
||||
WLR_XDG_TOPLEVEL_WM_CAPABILITIES_FULLSCREEN);
|
||||
WLR_XDG_TOPLEVEL_WM_CAPABILITIES_FULLSCREEN |
|
||||
WLR_XDG_TOPLEVEL_WM_CAPABILITIES_MINIMIZE);
|
||||
|
||||
LISTEN(&xdg_surface->events.destroy, &c->destroy, destroynotify);
|
||||
LISTEN(&xdg_surface->surface->events.commit, &c->commit, commitnotify);
|
||||
|
|
@ -985,6 +1007,8 @@ createnotify(struct wl_listener *listener, void *data)
|
|||
fullscreennotify);
|
||||
LISTEN(&xdg_surface->toplevel->events.request_maximize, &c->maximize,
|
||||
maximizenotify);
|
||||
LISTEN(&xdg_surface->toplevel->events.request_minimize, &c->minimize,
|
||||
minimizenotify);
|
||||
LISTEN(&xdg_surface->toplevel->events.set_title, &c->set_title, updatetitle);
|
||||
}
|
||||
|
||||
|
|
@ -1288,6 +1312,8 @@ focusclient(Client *c, int lift)
|
|||
client_set_border_color(old_c, bordercolor);
|
||||
|
||||
client_activate_surface(old, 0);
|
||||
if (old_c->foreign_toplevel)
|
||||
wlr_foreign_toplevel_handle_v1_set_activated(old_c->foreign_toplevel, 0);
|
||||
}
|
||||
}
|
||||
printstatus();
|
||||
|
|
@ -1306,6 +1332,8 @@ focusclient(Client *c, int lift)
|
|||
|
||||
/* Activate the new client */
|
||||
client_activate_surface(client_surface(c), 1);
|
||||
if (c->foreign_toplevel)
|
||||
wlr_foreign_toplevel_handle_v1_set_activated(c->foreign_toplevel, 1);
|
||||
}
|
||||
|
||||
void
|
||||
|
|
@ -1599,6 +1627,20 @@ mapnotify(struct wl_listener *listener, void *data)
|
|||
c->border[i]->node.data = c;
|
||||
}
|
||||
|
||||
c->foreign_toplevel =
|
||||
wlr_foreign_toplevel_handle_v1_create(foreign_toplevel_manager);
|
||||
|
||||
LISTEN(&c->foreign_toplevel->events.request_activate, &c->foreign_activate_request,
|
||||
handle_foreign_activate_request);
|
||||
LISTEN(&c->foreign_toplevel->events.request_close, &c->foreign_close_request,
|
||||
handle_foreign_close_request);
|
||||
LISTEN(&c->foreign_toplevel->events.request_fullscreen, &c->foreign_fullscreen_request,
|
||||
handle_foreign_fullscreen_request);
|
||||
LISTEN(&c->foreign_toplevel->events.destroy, &c->foreign_destroy,
|
||||
handle_foreign_destroy);
|
||||
LISTEN(&c->foreign_toplevel->events.request_minimize, &c->foreign_minimize,
|
||||
handle_foreign_minimize);
|
||||
|
||||
/* Initialize client geometry with room for border */
|
||||
client_set_tiled(c, WLR_EDGE_TOP | WLR_EDGE_BOTTOM | WLR_EDGE_LEFT | WLR_EDGE_RIGHT);
|
||||
c->geom.width += 2 * c->bw;
|
||||
|
|
@ -1645,6 +1687,24 @@ maximizenotify(struct wl_listener *listener, void *data)
|
|||
wlr_xdg_surface_schedule_configure(c->surface.xdg);
|
||||
}
|
||||
|
||||
void
|
||||
minimizenotify(struct wl_listener *listener, void *data)
|
||||
{
|
||||
/* This event is raised when a client would like to maximize itself,
|
||||
* typically because the user clicked on the maximize button on
|
||||
* client-side decorations. dwl doesn't support maximization, but
|
||||
* to conform to xdg-shell protocol we still must send a configure.
|
||||
* Since xdg-shell protocol v5 we should ignore request of unsupported
|
||||
* capabilities, just schedule a empty configure when the client uses <5
|
||||
* protocol version
|
||||
* wlr_xdg_surface_schedule_configure() is used to send an empty reply. */
|
||||
Client *c = wl_container_of(listener, c, minimize);
|
||||
setminimized(c, !c->isminimized);
|
||||
if (wl_resource_get_version(c->surface.xdg->toplevel->resource)
|
||||
< XDG_TOPLEVEL_WM_CAPABILITIES_SINCE_VERSION)
|
||||
wlr_xdg_surface_schedule_configure(c->surface.xdg);
|
||||
}
|
||||
|
||||
void
|
||||
monocle(Monitor *m)
|
||||
{
|
||||
|
|
@ -1928,6 +1988,7 @@ printstatus(void)
|
|||
printf("%s appid %s\n", m->wlr_output->name, appid ? appid : broken);
|
||||
printf("%s fullscreen %d\n", m->wlr_output->name, c->isfullscreen);
|
||||
printf("%s floating %d\n", m->wlr_output->name, c->isfloating);
|
||||
printf("%s minimized %d\n", m->wlr_output->name, c->isminimized);
|
||||
sel = c->tags;
|
||||
} else {
|
||||
printf("%s title \n", m->wlr_output->name);
|
||||
|
|
@ -2181,6 +2242,25 @@ setfullscreen(Client *c, int fullscreen)
|
|||
printstatus();
|
||||
}
|
||||
|
||||
void
|
||||
setminimized(Client *c, int minimized)
|
||||
{
|
||||
if (c->isminimized == minimized)
|
||||
return;
|
||||
|
||||
c->isminimized = minimized;
|
||||
if (!c->mon)
|
||||
return;
|
||||
client_set_minimized(client_surface(c), minimized);
|
||||
// wlr_scene_node_reparent(&c->scene->node,
|
||||
// layers[c->isminimized ? LyrBottom
|
||||
// : c->isfullscreen ? LyrFS
|
||||
// : c->isfloating ? LyrFloat
|
||||
// : LyrTile]);
|
||||
focusclient(focustop(c->mon), 1);
|
||||
arrange(c->mon);
|
||||
}
|
||||
|
||||
void
|
||||
setgamma(struct wl_listener *listener, void *data)
|
||||
{
|
||||
|
|
@ -2232,14 +2312,20 @@ setmon(Client *c, Monitor *m, uint32_t newtags)
|
|||
c->prev = c->geom;
|
||||
|
||||
/* Scene graph sends surface leave/enter events on move and resize */
|
||||
if (oldmon)
|
||||
if (oldmon) {
|
||||
if (c->foreign_toplevel)
|
||||
wlr_foreign_toplevel_handle_v1_output_leave(c->foreign_toplevel, oldmon->wlr_output);
|
||||
arrange(oldmon);
|
||||
}
|
||||
if (m) {
|
||||
/* Make sure window actually overlaps with the monitor */
|
||||
resize(c, c->geom, 0);
|
||||
c->tags = newtags ? newtags : m->tagset[m->seltags]; /* assign tags of target monitor */
|
||||
if (c->foreign_toplevel)
|
||||
wlr_foreign_toplevel_handle_v1_output_enter(c->foreign_toplevel, m->wlr_output);
|
||||
setfullscreen(c, c->isfullscreen); /* This will call arrange(c->mon) */
|
||||
setfloating(c, c->isfloating);
|
||||
setminimized(c, c->isminimized);
|
||||
}
|
||||
focusclient(focustop(selmon), 1);
|
||||
}
|
||||
|
|
@ -2351,6 +2437,9 @@ setup(void)
|
|||
gamma_control_mgr = wlr_gamma_control_manager_v1_create(dpy);
|
||||
LISTEN_STATIC(&gamma_control_mgr->events.set_gamma, setgamma);
|
||||
|
||||
/* Initializes foreign toplevel management */
|
||||
foreign_toplevel_manager = wlr_foreign_toplevel_manager_v1_create(dpy);
|
||||
|
||||
/* Creates an output layout, which a wlroots utility for working with an
|
||||
* arrangement of screens in a physical layout. */
|
||||
output_layout = wlr_output_layout_create();
|
||||
|
|
@ -2620,6 +2709,14 @@ togglefullscreen(const Arg *arg)
|
|||
setfullscreen(sel, !sel->isfullscreen);
|
||||
}
|
||||
|
||||
void
|
||||
toggleminimized(const Arg *arg)
|
||||
{
|
||||
Client *sel = focustop(selmon);
|
||||
if (sel)
|
||||
setminimized(sel, !sel->isminimized);
|
||||
}
|
||||
|
||||
void
|
||||
toggletag(const Arg *arg)
|
||||
{
|
||||
|
|
@ -2691,6 +2788,11 @@ unmapnotify(struct wl_listener *listener, void *data)
|
|||
wl_list_remove(&c->flink);
|
||||
}
|
||||
|
||||
if (c->foreign_toplevel) {
|
||||
wlr_foreign_toplevel_handle_v1_destroy(c->foreign_toplevel);
|
||||
c->foreign_toplevel = NULL;
|
||||
}
|
||||
|
||||
wlr_scene_node_destroy(&c->scene->node);
|
||||
printstatus();
|
||||
motionnotify(0, NULL, 0, 0, 0, 0);
|
||||
|
|
@ -2802,6 +2904,12 @@ void
|
|||
updatetitle(struct wl_listener *listener, void *data)
|
||||
{
|
||||
Client *c = wl_container_of(listener, c, set_title);
|
||||
if (c->foreign_toplevel) {
|
||||
const char *title;
|
||||
if (!(title = client_get_title(c)))
|
||||
title = broken;
|
||||
wlr_foreign_toplevel_handle_v1_set_title(c->foreign_toplevel, title);
|
||||
}
|
||||
if (c == focustop(c->mon))
|
||||
printstatus();
|
||||
}
|
||||
|
|
@ -2816,6 +2924,8 @@ urgent(struct wl_listener *listener, void *data)
|
|||
return;
|
||||
|
||||
c->isurgent = 1;
|
||||
if (c->isminimized)
|
||||
setminimized(c, 0);
|
||||
printstatus();
|
||||
|
||||
if (client_surface(c)->mapped)
|
||||
|
|
@ -2929,6 +3039,64 @@ zoom(const Arg *arg)
|
|||
arrange(selmon);
|
||||
}
|
||||
|
||||
void
|
||||
handle_foreign_activate_request(struct wl_listener *listener, void *data)
|
||||
{
|
||||
Client *c = wl_container_of(listener, c, foreign_activate_request);
|
||||
|
||||
if (!c)
|
||||
return;
|
||||
if (c->isminimized)
|
||||
setminimized(c, 0);
|
||||
if (c->mon == selmon) {
|
||||
c->tags = c->mon->tagset[c->mon->seltags];
|
||||
} else {
|
||||
setmon(c, selmon, 0);
|
||||
}
|
||||
focusclient(c, 1);
|
||||
arrange(c->mon);
|
||||
}
|
||||
|
||||
void
|
||||
handle_foreign_close_request(struct wl_listener *listener, void *data)
|
||||
{
|
||||
Client *c = wl_container_of(listener, c, foreign_close_request);
|
||||
|
||||
if (c)
|
||||
client_send_close(c);
|
||||
}
|
||||
|
||||
void
|
||||
handle_foreign_fullscreen_request(struct wl_listener *listener, void *data) {
|
||||
Client *c = wl_container_of(listener, c, foreign_fullscreen_request);
|
||||
struct wlr_foreign_toplevel_handle_v1_fullscreen_event *event = data;
|
||||
|
||||
if (c)
|
||||
setfullscreen(c, event->fullscreen);
|
||||
}
|
||||
|
||||
void
|
||||
handle_foreign_destroy(struct wl_listener *listener, void *data)
|
||||
{
|
||||
Client *c = wl_container_of(listener, c, foreign_destroy);
|
||||
|
||||
wl_list_remove(&c->foreign_activate_request.link);
|
||||
wl_list_remove(&c->foreign_close_request.link);
|
||||
wl_list_remove(&c->foreign_fullscreen_request.link);
|
||||
wl_list_remove(&c->foreign_destroy.link);
|
||||
wl_list_remove(&c->foreign_minimize.link);
|
||||
}
|
||||
|
||||
void
|
||||
handle_foreign_minimize(struct wl_listener *listener, void *data)
|
||||
{
|
||||
Client *c = wl_container_of(listener, c, foreign_minimize);
|
||||
struct wlr_foreign_toplevel_handle_v1_minimized_event *event = data;
|
||||
|
||||
if (c)
|
||||
setminimized(c, event->minimized);
|
||||
}
|
||||
|
||||
#ifdef XWAYLAND
|
||||
void
|
||||
activatex11(struct wl_listener *listener, void *data)
|
||||
|
|
@ -2936,8 +3104,11 @@ activatex11(struct wl_listener *listener, void *data)
|
|||
Client *c = wl_container_of(listener, c, activate);
|
||||
|
||||
/* Only "managed" windows can be activated */
|
||||
if (!client_is_unmanaged(c))
|
||||
if (!client_is_unmanaged(c)) {
|
||||
if (c->isminimized)
|
||||
setminimized(c, 0);
|
||||
wlr_xwayland_surface_activate(c->surface.xwayland, 1);
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
|
|
@ -2986,6 +3157,7 @@ createnotifyx11(struct wl_listener *listener, void *data)
|
|||
LISTEN(&xsurface->events.request_activate, &c->activate, activatex11);
|
||||
LISTEN(&xsurface->events.request_configure, &c->configure, configurex11);
|
||||
LISTEN(&xsurface->events.request_fullscreen, &c->fullscreen, fullscreennotify);
|
||||
LISTEN(&xsurface->events.request_minimize, &c->foreign_minimize, handle_foreign_minimize);
|
||||
LISTEN(&xsurface->events.set_hints, &c->set_hints, sethints);
|
||||
LISTEN(&xsurface->events.set_title, &c->set_title, updatetitle);
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue