--- /dev/null
+/* SPDX-License-Identifier: GPL-2.0-only */
+#ifndef LABWC_FOREIGN_TOPLEVEL_INTERNAL_H
+#define LABWC_FOREIGN_TOPLEVEL_INTERNAL_H
+
+#include <stdbool.h>
+#include <wayland-server-core.h>
+#include "foreign-toplevel.h"
+
+struct foreign_toplevel {
+ struct view *view;
+
+ /* *-toplevel implementations */
+ struct wlr_foreign_toplevel {
+ struct wlr_foreign_toplevel_handle_v1 *handle;
+
+ /* Client side events */
+ struct {
+ struct wl_listener request_maximize;
+ struct wl_listener request_minimize;
+ struct wl_listener request_fullscreen;
+ struct wl_listener request_activate;
+ struct wl_listener request_close;
+ struct wl_listener handle_destroy;
+ } on;
+
+ /* Compositor side state updates */
+ struct {
+ struct wl_listener new_app_id;
+ struct wl_listener new_title;
+ struct wl_listener new_outputs;
+ struct wl_listener maximized;
+ struct wl_listener minimized;
+ struct wl_listener fullscreened;
+ struct wl_listener activated;
+ } on_view;
+
+ /* Internal signals */
+ struct {
+ struct wl_listener toplevel_parent;
+ struct wl_listener toplevel_destroy;
+ } on_foreign_toplevel;
+
+ } wlr_toplevel;
+
+ /* TODO: add struct xdg_x11_mapped_toplevel at some point */
+
+ struct {
+ struct wl_signal toplevel_parent; /* struct view *parent */
+ struct wl_signal toplevel_destroy;
+ } events;
+};
+
+void wlr_foreign_toplevel_init(struct foreign_toplevel *toplevel);
+
+void foreign_request_minimize(struct foreign_toplevel *toplevel, bool minimized);
+void foreign_request_maximize(struct foreign_toplevel *toplevel, enum view_axis axis);
+void foreign_request_fullscreen(struct foreign_toplevel *toplevel, bool fullscreen);
+void foreign_request_activate(struct foreign_toplevel *toplevel);
+void foreign_request_close(struct foreign_toplevel *toplevel);
+
+#endif /* LABWC_FOREIGN_TOPLEVEL_INTERNAL_H */
--- /dev/null
+/* SPDX-License-Identifier: GPL-2.0-only */
+#ifndef LABWC_FOREIGN_TOPLEVEL_H
+#define LABWC_FOREIGN_TOPLEVEL_H
+
+struct view;
+struct foreign_toplevel;
+
+struct foreign_toplevel *foreign_toplevel_create(struct view *view);
+void foreign_toplevel_set_parent(struct foreign_toplevel *toplevel,
+ struct foreign_toplevel *parent);
+void foreign_toplevel_destroy(struct foreign_toplevel *toplevel);
+
+#endif /* LABWC_FOREIGN_TOPLEVEL_H */
#include <wlr/types/wlr_buffer.h>
#include <wlr/types/wlr_cursor.h>
#include <wlr/types/wlr_data_device.h>
-#include <wlr/types/wlr_foreign_toplevel_management_v1.h>
#include <wlr/types/wlr_gamma_control_v1.h>
#include <wlr/types/wlr_input_device.h>
#include <wlr/types/wlr_keyboard.h>
void xdg_popup_create(struct view *view, struct wlr_xdg_popup *wlr_popup);
void xdg_shell_init(struct server *server);
-void foreign_toplevel_handle_create(struct view *view);
-void foreign_toplevel_update_outputs(struct view *view);
-
/*
* desktop.c routines deal with a collection of views
*
struct view;
struct wlr_surface;
+struct foreign_toplevel;
/* Common to struct view and struct xwayland_unmanaged */
struct mappable {
struct multi_rect *rect;
} resize_outlines;
- struct foreign_toplevel {
- struct wlr_foreign_toplevel_handle_v1 *handle;
- struct wl_listener maximize;
- struct wl_listener minimize;
- struct wl_listener fullscreen;
- struct wl_listener activate;
- struct wl_listener close;
- struct wl_listener destroy;
- } toplevel;
-
struct mappable mappable;
struct wl_listener destroy;
struct wl_listener request_fullscreen;
struct wl_listener set_title;
+ struct foreign_toplevel *foreign_toplevel;
+
struct {
struct wl_signal new_app_id;
struct wl_signal new_title;
--- /dev/null
+// SPDX-License-Identifier: GPL-2.0-only
+#include <assert.h>
+#include <wayland-server-core.h>
+#include "common/macros.h"
+#include "common/mem.h"
+#include "labwc.h"
+#include "view.h"
+#include "foreign-toplevel-internal.h"
+
+/* Internal API */
+void
+foreign_request_minimize(struct foreign_toplevel *toplevel, bool minimized)
+{
+ view_minimize(toplevel->view, minimized);
+}
+
+void
+foreign_request_maximize(struct foreign_toplevel *toplevel, enum view_axis axis)
+{
+ view_maximize(toplevel->view, axis, /*store_natural_geometry*/ true);
+}
+
+void
+foreign_request_fullscreen(struct foreign_toplevel *toplevel, bool fullscreen)
+{
+ view_set_fullscreen(toplevel->view, fullscreen);
+}
+
+void
+foreign_request_activate(struct foreign_toplevel *toplevel)
+{
+ if (toplevel->view->server->osd_state.cycle_view) {
+ wlr_log(WLR_INFO, "Preventing focus request while in window switcher");
+ return;
+ }
+
+ desktop_focus_view(toplevel->view, /*raise*/ true);
+}
+
+void
+foreign_request_close(struct foreign_toplevel *toplevel)
+{
+ view_close(toplevel->view);
+}
+
+/* Public API */
+struct foreign_toplevel *
+foreign_toplevel_create(struct view *view)
+{
+ assert(view);
+ assert(view->mapped);
+
+ struct foreign_toplevel *toplevel = znew(*toplevel);
+ toplevel->view = view;
+
+ wl_signal_init(&toplevel->events.toplevel_parent);
+ wl_signal_init(&toplevel->events.toplevel_destroy);
+
+ wlr_foreign_toplevel_init(toplevel);
+
+ return toplevel;
+}
+
+void
+foreign_toplevel_set_parent(struct foreign_toplevel *toplevel, struct foreign_toplevel *parent)
+{
+ assert(toplevel);
+ wl_signal_emit_mutable(&toplevel->events.toplevel_parent, parent);
+}
+
+void
+foreign_toplevel_destroy(struct foreign_toplevel *toplevel)
+{
+ assert(toplevel);
+ wl_signal_emit_mutable(&toplevel->events.toplevel_destroy, NULL);
+ assert(!toplevel->wlr_toplevel.handle);
+ free(toplevel);
+}
--- /dev/null
+labwc_sources += files(
+ 'foreign.c',
+ 'wlr-foreign.c',
+)
--- /dev/null
+// SPDX-License-Identifier: GPL-2.0-only
+#include <assert.h>
+#include <wlr/types/wlr_foreign_toplevel_management_v1.h>
+#include "common/macros.h"
+#include "labwc.h"
+#include "view.h"
+#include "foreign-toplevel-internal.h"
+
+/* wlr signals */
+static void
+handle_request_minimize(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel = wl_container_of(
+ listener, toplevel, wlr_toplevel.on.request_minimize);
+ struct wlr_foreign_toplevel_handle_v1_minimized_event *event = data;
+
+ foreign_request_minimize(toplevel, event->minimized);
+}
+
+static void
+handle_request_maximize(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel = wl_container_of(
+ listener, toplevel, wlr_toplevel.on.request_maximize);
+ struct wlr_foreign_toplevel_handle_v1_maximized_event *event = data;
+
+ foreign_request_maximize(toplevel,
+ event->maximized ? VIEW_AXIS_BOTH : VIEW_AXIS_NONE);
+}
+
+static void
+handle_request_fullscreen(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel = wl_container_of(
+ listener, toplevel, wlr_toplevel.on.request_fullscreen);
+ struct wlr_foreign_toplevel_handle_v1_fullscreen_event *event = data;
+
+ /* TODO: This ignores event->output */
+ foreign_request_fullscreen(toplevel, event->fullscreen);
+}
+
+static void
+handle_request_activate(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel = wl_container_of(
+ listener, toplevel, wlr_toplevel.on.request_activate);
+
+ /* In a multi-seat world we would select seat based on event->seat here. */
+ foreign_request_activate(toplevel);
+}
+
+static void
+handle_request_close(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel = wl_container_of(
+ listener, toplevel, wlr_toplevel.on.request_close);
+
+ foreign_request_close(toplevel);
+}
+
+static void
+handle_handle_destroy(struct wl_listener *listener, void *data)
+{
+ struct wlr_foreign_toplevel *wlr_toplevel =
+ wl_container_of(listener, wlr_toplevel, on.handle_destroy);
+
+ /* Client side requests */
+ wl_list_remove(&wlr_toplevel->on.request_maximize.link);
+ wl_list_remove(&wlr_toplevel->on.request_minimize.link);
+ wl_list_remove(&wlr_toplevel->on.request_fullscreen.link);
+ wl_list_remove(&wlr_toplevel->on.request_activate.link);
+ wl_list_remove(&wlr_toplevel->on.request_close.link);
+ wl_list_remove(&wlr_toplevel->on.handle_destroy.link);
+ wlr_toplevel->handle = NULL;
+}
+
+/* Compositor signals */
+static void
+handle_new_app_id(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel =
+ wl_container_of(listener, toplevel, wlr_toplevel.on_view.new_app_id);
+ assert(toplevel->wlr_toplevel.handle);
+
+ const char *app_id = view_get_string_prop(toplevel->view, "app_id");
+ wlr_foreign_toplevel_handle_v1_set_app_id(toplevel->wlr_toplevel.handle, app_id);
+}
+
+static void
+handle_new_title(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel =
+ wl_container_of(listener, toplevel, wlr_toplevel.on_view.new_title);
+ assert(toplevel->wlr_toplevel.handle);
+
+ const char *title = view_get_string_prop(toplevel->view, "title");
+ wlr_foreign_toplevel_handle_v1_set_title(toplevel->wlr_toplevel.handle, title);
+}
+
+static void
+handle_new_outputs(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel =
+ wl_container_of(listener, toplevel, wlr_toplevel.on_view.new_outputs);
+ assert(toplevel->wlr_toplevel.handle);
+
+ /*
+ * Loop over all outputs and notify foreign_toplevel clients about changes.
+ * wlr_foreign_toplevel_handle_v1_output_xxx() keeps track of the active
+ * outputs internally and merges the events. It also listens to output
+ * destroy events so its fine to just relay the current state and let
+ * wlr_foreign_toplevel handle the rest.
+ */
+ struct output *output;
+ wl_list_for_each(output, &toplevel->view->server->outputs, link) {
+ if (view_on_output(toplevel->view, output)) {
+ wlr_foreign_toplevel_handle_v1_output_enter(
+ toplevel->wlr_toplevel.handle, output->wlr_output);
+ } else {
+ wlr_foreign_toplevel_handle_v1_output_leave(
+ toplevel->wlr_toplevel.handle, output->wlr_output);
+ }
+ }
+}
+
+static void
+handle_maximized(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel =
+ wl_container_of(listener, toplevel, wlr_toplevel.on_view.maximized);
+ assert(toplevel->wlr_toplevel.handle);
+
+ wlr_foreign_toplevel_handle_v1_set_maximized(
+ toplevel->wlr_toplevel.handle,
+ toplevel->view->maximized == VIEW_AXIS_BOTH);
+}
+
+static void
+handle_minimized(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel =
+ wl_container_of(listener, toplevel, wlr_toplevel.on_view.minimized);
+ assert(toplevel->wlr_toplevel.handle);
+
+ wlr_foreign_toplevel_handle_v1_set_minimized(
+ toplevel->wlr_toplevel.handle, toplevel->view->minimized);
+}
+
+static void
+handle_fullscreened(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel =
+ wl_container_of(listener, toplevel, wlr_toplevel.on_view.fullscreened);
+ assert(toplevel->wlr_toplevel.handle);
+
+ wlr_foreign_toplevel_handle_v1_set_fullscreen(
+ toplevel->wlr_toplevel.handle, toplevel->view->fullscreen);
+}
+
+static void
+handle_activated(struct wl_listener *listener, void *data)
+{
+ struct foreign_toplevel *toplevel =
+ wl_container_of(listener, toplevel, wlr_toplevel.on_view.activated);
+ assert(toplevel->wlr_toplevel.handle);
+
+ bool *activated = data;
+ wlr_foreign_toplevel_handle_v1_set_activated(
+ toplevel->wlr_toplevel.handle, *activated);
+}
+
+/* Internal signals */
+static void
+handle_toplevel_parent(struct wl_listener *listener, void *data)
+{
+ struct wlr_foreign_toplevel *wlr_toplevel = wl_container_of(
+ listener, wlr_toplevel, on_foreign_toplevel.toplevel_parent);
+ struct foreign_toplevel *parent = data;
+
+ assert(wlr_toplevel->handle);
+
+ /* The wlroots wlr-foreign-toplevel impl ensures parent is reset to NULL on destroy */
+ wlr_foreign_toplevel_handle_v1_set_parent(wlr_toplevel->handle, parent
+ ? parent->wlr_toplevel.handle
+ : NULL);
+}
+
+static void
+handle_toplevel_destroy(struct wl_listener *listener, void *data)
+{
+ struct wlr_foreign_toplevel *wlr_toplevel = wl_container_of(
+ listener, wlr_toplevel, on_foreign_toplevel.toplevel_destroy);
+
+ assert(wlr_toplevel->handle);
+ wlr_foreign_toplevel_handle_v1_destroy(wlr_toplevel->handle);
+
+ /* Compositor side state changes */
+ wl_list_remove(&wlr_toplevel->on_view.new_app_id.link);
+ wl_list_remove(&wlr_toplevel->on_view.new_title.link);
+ wl_list_remove(&wlr_toplevel->on_view.new_outputs.link);
+ wl_list_remove(&wlr_toplevel->on_view.maximized.link);
+ wl_list_remove(&wlr_toplevel->on_view.minimized.link);
+ wl_list_remove(&wlr_toplevel->on_view.fullscreened.link);
+ wl_list_remove(&wlr_toplevel->on_view.activated.link);
+
+ /* Internal signals */
+ wl_list_remove(&wlr_toplevel->on_foreign_toplevel.toplevel_parent.link);
+ wl_list_remove(&wlr_toplevel->on_foreign_toplevel.toplevel_destroy.link);
+}
+
+/* Internal API */
+void
+wlr_foreign_toplevel_init(struct foreign_toplevel *toplevel)
+{
+ struct wlr_foreign_toplevel *wlr_toplevel = &toplevel->wlr_toplevel;
+ struct view *view = toplevel->view;
+
+ assert(view->server->foreign_toplevel_manager);
+
+ wlr_toplevel->handle = wlr_foreign_toplevel_handle_v1_create(
+ view->server->foreign_toplevel_manager);
+ if (!wlr_toplevel->handle) {
+ wlr_log(WLR_ERROR, "cannot create wlr foreign toplevel handle for (%s)",
+ view_get_string_prop(view, "title"));
+ return;
+ }
+
+ /* Client side requests */
+ CONNECT_SIGNAL(wlr_toplevel->handle, &wlr_toplevel->on, request_maximize);
+ CONNECT_SIGNAL(wlr_toplevel->handle, &wlr_toplevel->on, request_minimize);
+ CONNECT_SIGNAL(wlr_toplevel->handle, &wlr_toplevel->on, request_fullscreen);
+ CONNECT_SIGNAL(wlr_toplevel->handle, &wlr_toplevel->on, request_activate);
+ CONNECT_SIGNAL(wlr_toplevel->handle, &wlr_toplevel->on, request_close);
+ wlr_toplevel->on.handle_destroy.notify = handle_handle_destroy;
+ wl_signal_add(&wlr_toplevel->handle->events.destroy, &wlr_toplevel->on.handle_destroy);
+
+ /* Compositor side state changes */
+ CONNECT_SIGNAL(view, &wlr_toplevel->on_view, new_app_id);
+ CONNECT_SIGNAL(view, &wlr_toplevel->on_view, new_title);
+ CONNECT_SIGNAL(view, &wlr_toplevel->on_view, new_outputs);
+ CONNECT_SIGNAL(view, &wlr_toplevel->on_view, maximized);
+ CONNECT_SIGNAL(view, &wlr_toplevel->on_view, minimized);
+ CONNECT_SIGNAL(view, &wlr_toplevel->on_view, fullscreened);
+ CONNECT_SIGNAL(view, &wlr_toplevel->on_view, activated);
+
+ /* Internal signals */
+ CONNECT_SIGNAL(toplevel, &wlr_toplevel->on_foreign_toplevel, toplevel_parent);
+ CONNECT_SIGNAL(toplevel, &wlr_toplevel->on_foreign_toplevel, toplevel_destroy);
+}
+++ /dev/null
-// SPDX-License-Identifier: GPL-2.0-only
-#include <assert.h>
-#include "labwc.h"
-#include "view.h"
-#include "workspaces.h"
-
-static void
-handle_request_minimize(struct wl_listener *listener, void *data)
-{
- struct view *view = wl_container_of(listener, view, toplevel.minimize);
- struct wlr_foreign_toplevel_handle_v1_minimized_event *event = data;
- view_minimize(view, event->minimized);
-}
-
-static void
-handle_request_maximize(struct wl_listener *listener, void *data)
-{
- struct view *view = wl_container_of(listener, view, toplevel.maximize);
- struct wlr_foreign_toplevel_handle_v1_maximized_event *event = data;
- view_maximize(view, event->maximized ? VIEW_AXIS_BOTH : VIEW_AXIS_NONE,
- /*store_natural_geometry*/ true);
-}
-
-static void
-handle_request_fullscreen(struct wl_listener *listener, void *data)
-{
- struct view *view = wl_container_of(listener, view, toplevel.fullscreen);
- struct wlr_foreign_toplevel_handle_v1_fullscreen_event *event = data;
- view_set_fullscreen(view, event->fullscreen);
-}
-
-static void
-handle_request_activate(struct wl_listener *listener, void *data)
-{
- struct view *view = wl_container_of(listener, view, toplevel.activate);
- // struct wlr_foreign_toplevel_handle_v1_activated_event *event = data;
-
- if (view->server->osd_state.cycle_view) {
- wlr_log(WLR_INFO, "Preventing focus request while in window switcher");
- return;
- }
-
- /* In a multi-seat world we would select seat based on event->seat here. */
- desktop_focus_view(view, /*raise*/ true);
-}
-
-static void
-handle_request_close(struct wl_listener *listener, void *data)
-{
- struct view *view = wl_container_of(listener, view, toplevel.close);
- view_close(view);
-}
-
-static void
-handle_destroy(struct wl_listener *listener, void *data)
-{
- struct view *view = wl_container_of(listener, view, toplevel.destroy);
- struct foreign_toplevel *toplevel = &view->toplevel;
- wl_list_remove(&toplevel->maximize.link);
- wl_list_remove(&toplevel->minimize.link);
- wl_list_remove(&toplevel->fullscreen.link);
- wl_list_remove(&toplevel->activate.link);
- wl_list_remove(&toplevel->close.link);
- wl_list_remove(&toplevel->destroy.link);
- toplevel->handle = NULL;
-}
-
-void
-foreign_toplevel_handle_create(struct view *view)
-{
- struct foreign_toplevel *toplevel = &view->toplevel;
-
- toplevel->handle = wlr_foreign_toplevel_handle_v1_create(
- view->server->foreign_toplevel_manager);
- if (!toplevel->handle) {
- wlr_log(WLR_ERROR, "cannot create foreign toplevel handle for (%s)",
- view_get_string_prop(view, "title"));
- return;
- }
-
- toplevel->maximize.notify = handle_request_maximize;
- wl_signal_add(&toplevel->handle->events.request_maximize,
- &toplevel->maximize);
-
- toplevel->minimize.notify = handle_request_minimize;
- wl_signal_add(&toplevel->handle->events.request_minimize,
- &toplevel->minimize);
-
- toplevel->fullscreen.notify = handle_request_fullscreen;
- wl_signal_add(&toplevel->handle->events.request_fullscreen,
- &toplevel->fullscreen);
-
- toplevel->activate.notify = handle_request_activate;
- wl_signal_add(&toplevel->handle->events.request_activate,
- &toplevel->activate);
-
- toplevel->close.notify = handle_request_close;
- wl_signal_add(&toplevel->handle->events.request_close,
- &toplevel->close);
-
- toplevel->destroy.notify = handle_destroy;
- wl_signal_add(&toplevel->handle->events.destroy, &toplevel->destroy);
-}
-
-/*
- * Loop over all outputs and notify foreign_toplevel clients about changes.
- * wlr_foreign_toplevel_handle_v1_output_xxx() keeps track of the active
- * outputs internally and merges the events. It also listens to output
- * destroy events so its fine to just relay the current state and let
- * wlr_foreign_toplevel handle the rest.
- */
-void
-foreign_toplevel_update_outputs(struct view *view)
-{
- assert(view->toplevel.handle);
-
- struct output *output;
- wl_list_for_each(output, &view->server->outputs, link) {
- if (view_on_output(view, output)) {
- wlr_foreign_toplevel_handle_v1_output_enter(
- view->toplevel.handle, output->wlr_output);
- } else {
- wlr_foreign_toplevel_handle_v1_output_leave(
- view->toplevel.handle, output->wlr_output);
- }
- }
-}
wl_list_for_each(view, &server->views, link) {
if (view->workspace == workspace) {
const char *title = view_get_string_prop(view, "title");
- if (!view->toplevel.handle || string_null_or_empty(title)) {
+ if (!view->foreign_toplevel || string_null_or_empty(title)) {
continue;
}
'desktop.c',
'dnd.c',
'edges.c',
- 'foreign.c',
'idle.c',
'interactive.c',
'layers.c',
subdir('common')
subdir('config')
subdir('decorations')
+subdir('foreign-toplevel')
subdir('input')
subdir('menu')
subdir('ssd')
#include <wlr/types/wlr_data_control_v1.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_presentation_time.h>
#include <wlr/types/wlr_single_pixel_buffer_v1.h>
#include <wlr/types/wlr_viewporter.h>
#include <wlr/types/wlr_tablet_v2.h>
+
#if HAVE_XWAYLAND
#include <wlr/xwayland.h>
#include "xwayland-shell-v1-protocol.h"
#include <stdio.h>
#include <strings.h>
#include "common/list.h"
+#include "foreign-toplevel.h"
#include "labwc.h"
#include "view.h"
#include "view-impl-common.h"
*/
enum property ret = window_rules_get_property(view, "skipTaskbar");
if (ret == LAB_PROP_TRUE) {
- if (view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_destroy(view->toplevel.handle);
+ if (view->foreign_toplevel) {
+ foreign_toplevel_destroy(view->foreign_toplevel);
+ view->foreign_toplevel = NULL;
}
}
#include "common/mem.h"
#include "common/parse-bool.h"
#include "common/scene-helpers.h"
+#include "foreign-toplevel.h"
#include "input/keyboard.h"
#include "labwc.h"
#include "menu/menu.h"
if (view->impl->set_activated) {
view->impl->set_activated(view, activated);
}
- if (view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_set_activated(
- view->toplevel.handle, activated);
- }
wl_signal_emit_mutable(&view->events.activated, &activated);
if (new_outputs != view->outputs) {
view->outputs = new_outputs;
- if (view->toplevel.handle) {
- foreign_toplevel_update_outputs(view);
- }
wl_signal_emit_mutable(&view->events.new_outputs, NULL);
desktop_update_top_layer_visiblity(view->server);
}
if (view->minimized == minimized) {
return;
}
- if (view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_set_minimized(
- view->toplevel.handle, minimized);
- }
+
if (view->impl->minimize) {
view->impl->minimize(view, minimized);
}
if (view->impl->maximize) {
view->impl->maximize(view, (maximized == VIEW_AXIS_BOTH));
}
- if (view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_set_maximized(
- view->toplevel.handle, (maximized == VIEW_AXIS_BOTH));
- }
view->maximized = maximized;
wl_signal_emit_mutable(&view->events.maximized, NULL);
if (view->impl->set_fullscreen) {
view->impl->set_fullscreen(view, fullscreen);
}
- if (view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_set_fullscreen(
- view->toplevel.handle, fullscreen);
- }
view->fullscreen = fullscreen;
wl_signal_emit_mutable(&view->events.fullscreened, NULL);
{
assert(view);
const char *title = view_get_string_prop(view, "title");
- if (!view->toplevel.handle || !title) {
+ if (!title) {
return;
}
ssd_update_title(view->ssd);
- wlr_foreign_toplevel_handle_v1_set_title(view->toplevel.handle, title);
+
wl_signal_emit_mutable(&view->events.new_title, NULL);
}
{
assert(view);
const char *app_id = view_get_string_prop(view, "app_id");
- if (!view->toplevel.handle || !app_id) {
+ if (!app_id) {
return;
}
- wlr_foreign_toplevel_handle_v1_set_app_id(
- view->toplevel.handle, app_id);
if (view->ssd_enabled) {
ssd_update_window_icon(view->ssd);
wl_list_remove(&view->set_title.link);
wl_list_remove(&view->destroy.link);
- if (view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_destroy(view->toplevel.handle);
+ if (view->foreign_toplevel) {
+ foreign_toplevel_destroy(view->foreign_toplevel);
+ view->foreign_toplevel = NULL;
}
if (server->grabbed_view == view) {
#include "common/macros.h"
#include "common/mem.h"
#include "decorations.h"
+#include "foreign-toplevel.h"
#include "labwc.h"
#include "menu/menu.h"
#include "node.h"
static void
init_foreign_toplevel(struct view *view)
{
- foreign_toplevel_handle_create(view);
+ assert(!view->foreign_toplevel);
+ view->foreign_toplevel = foreign_toplevel_create(view);
+
struct wlr_xdg_toplevel *toplevel = xdg_toplevel_from_view(view);
if (!toplevel->parent) {
return;
}
struct wlr_xdg_surface *surface = toplevel->parent->base;
struct view *parent = surface->data;
- if (!parent->toplevel.handle) {
+ if (!parent->foreign_toplevel) {
return;
}
- wlr_foreign_toplevel_handle_v1_set_parent(view->toplevel.handle, parent->toplevel.handle);
+ foreign_toplevel_set_parent(view->foreign_toplevel, parent->foreign_toplevel);
}
static void
}
struct wlr_xdg_surface *xdg_surface = xdg_surface_from_view(view);
wlr_scene_node_set_enabled(&view->scene_tree->node, true);
+
+ if (!view->foreign_toplevel) {
+ init_foreign_toplevel(view);
+ /*
+ * Initial outputs will be synced via
+ * view->events.new_outputs on view_moved()
+ */
+ }
+
if (!view->been_mapped) {
if (view_wants_decorations(view)) {
view_set_ssd_mode(view, LAB_SSD_MODE_FULL);
view_moved(view);
}
- if (!view->toplevel.handle) {
- init_foreign_toplevel(view);
- foreign_toplevel_update_outputs(view);
- }
-
view_impl_map(view);
view->been_mapped = true;
}
* than just minimized), destroy the foreign toplevel handle so
* the unmapped view doesn't show up in panels and the like.
*/
- if (client_request && view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_destroy(view->toplevel.handle);
+ if (client_request && view->foreign_toplevel) {
+ foreign_toplevel_destroy(view->foreign_toplevel);
+ view->foreign_toplevel = NULL;
}
}
#include "common/mem.h"
#include "config/rcxml.h"
#include "config/session.h"
+#include "foreign-toplevel.h"
#include "labwc.h"
#include "node.h"
#include "ssd.h"
static void
init_foreign_toplevel(struct view *view)
{
- foreign_toplevel_handle_create(view);
+ assert(!view->foreign_toplevel);
+ view->foreign_toplevel = foreign_toplevel_create(view);
struct wlr_xwayland_surface *surface = xwayland_surface_from_view(view);
if (!surface->parent) {
return;
}
struct view *parent = (struct view *)surface->parent->data;
- if (!parent || !parent->toplevel.handle) {
+ if (!parent || !parent->foreign_toplevel) {
return;
}
- wlr_foreign_toplevel_handle_v1_set_parent(view->toplevel.handle, parent->toplevel.handle);
+ foreign_toplevel_set_parent(view->foreign_toplevel, parent->foreign_toplevel);
}
static void
view->scene_node = &tree->node;
}
+ /*
+ * Exclude unfocusable views from wlr-foreign-toplevel. These
+ * views (notifications, floating toolbars, etc.) should not be
+ * shown in taskbars/docks/etc.
+ */
+ if (!view->foreign_toplevel && view_is_focusable(view)) {
+ init_foreign_toplevel(view);
+ /*
+ * Initial outputs will be synced via
+ * view->events.new_outputs on view_moved()
+ */
+ }
+
if (!view->been_mapped) {
check_natural_geometry(view);
set_initial_position(view, xwayland_surface);
view_moved(view);
}
- /*
- * Exclude unfocusable views from wlr-foreign-toplevel. These
- * views (notifications, floating toolbars, etc.) should not be
- * shown in taskbars/docks/etc.
- */
- if (!view->toplevel.handle && view_is_focusable(view)) {
- init_foreign_toplevel(view);
- foreign_toplevel_update_outputs(view);
- }
-
/* Add commit here, as xwayland map/unmap can change the wlr_surface */
wl_signal_add(&xwayland_surface->surface->events.commit, &view->commit);
view->commit.notify = handle_commit;
* the unmapped view doesn't show up in panels and the like.
*/
out:
- if (client_request && view->toplevel.handle) {
- wlr_foreign_toplevel_handle_v1_destroy(view->toplevel.handle);
+ if (client_request && view->foreign_toplevel) {
+ foreign_toplevel_destroy(view->foreign_toplevel);
+ view->foreign_toplevel = NULL;
}
}