diff --git a/proto/meson.build b/proto/meson.build index a05e26e..41925aa 100644 --- a/proto/meson.build +++ b/proto/meson.build @@ -15,8 +15,10 @@ wayland_scanner_client = generator( ) client_protocols = [ - 'wayfire-shell.xml', - 'virtual-keyboard-unstable-v1.xml' + [wl_protocol_dir, 'stable/xdg-shell/xdg-shell.xml'], + ['wayfire-shell.xml'], + ['wlr-layer-shell-unstable-v1.xml'], + ['virtual-keyboard-unstable-v1.xml'] ] wl_protos_src = [] diff --git a/proto/wlr-layer-shell-unstable-v1.xml b/proto/wlr-layer-shell-unstable-v1.xml new file mode 100644 index 0000000..fb4f6b2 --- /dev/null +++ b/proto/wlr-layer-shell-unstable-v1.xml @@ -0,0 +1,285 @@ + + + + Copyright © 2017 Drew DeVault + + Permission to use, copy, modify, distribute, and sell this + software and its documentation for any purpose is hereby granted + without fee, provided that the above copyright notice appear in + all copies and that both that copyright notice and this permission + notice appear in supporting documentation, and that the name of + the copyright holders not be used in advertising or publicity + pertaining to distribution of the software without specific, + written prior permission. The copyright holders make no + representations about the suitability of this software for any + purpose. It is provided "as is" without express or implied + warranty. + + THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS + SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND + FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY + SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN + AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, + ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF + THIS SOFTWARE. + + + + + Clients can use this interface to assign the surface_layer role to + wl_surfaces. Such surfaces are assigned to a "layer" of the output and + rendered with a defined z-depth respective to each other. They may also be + anchored to the edges and corners of a screen and specify input handling + semantics. This interface should be suitable for the implementation of + many desktop shell components, and a broad number of other applications + that interact with the desktop. + + + + + Create a layer surface for an existing surface. This assigns the role of + layer_surface, or raises a protocol error if another role is already + assigned. + + Creating a layer surface from a wl_surface which has a buffer attached + or committed is a client error, and any attempts by a client to attach + or manipulate a buffer prior to the first layer_surface.configure call + must also be treated as errors. + + You may pass NULL for output to allow the compositor to decide which + output to use. Generally this will be the one that the user most + recently interacted with. + + Clients can specify a namespace that defines the purpose of the layer + surface. + + + + + + + + + + + + + + + + + These values indicate which layers a surface can be rendered in. They + are ordered by z depth, bottom-most first. Traditional shell surfaces + will typically be rendered between the bottom and top layers. + Fullscreen shell surfaces are typically rendered at the top layer. + Multiple surfaces can share a single layer, and ordering within a + single layer is undefined. + + + + + + + + + + + + An interface that may be implemented by a wl_surface, for surfaces that + are designed to be rendered as a layer of a stacked desktop-like + environment. + + Layer surface state (size, anchor, exclusive zone, margin, interactivity) + is double-buffered, and will be applied at the time wl_surface.commit of + the corresponding wl_surface is called. + + + + + Sets the size of the surface in surface-local coordinates. The + compositor will display the surface centered with respect to its + anchors. + + If you pass 0 for either value, the compositor will assign it and + inform you of the assignment in the configure event. You must set your + anchor to opposite edges in the dimensions you omit; not doing so is a + protocol error. Both values are 0 by default. + + Size is double-buffered, see wl_surface.commit. + + + + + + + + Requests that the compositor anchor the surface to the specified edges + and corners. If two orthoginal edges are specified (e.g. 'top' and + 'left'), then the anchor point will be the intersection of the edges + (e.g. the top left corner of the output); otherwise the anchor point + will be centered on that edge, or in the center if none is specified. + + Anchor is double-buffered, see wl_surface.commit. + + + + + + + Requests that the compositor avoids occluding an area of the surface + with other surfaces. The compositor's use of this information is + implementation-dependent - do not assume that this region will not + actually be occluded. + + A positive value is only meaningful if the surface is anchored to an + edge, rather than a corner. The zone is the number of surface-local + coordinates from the edge that are considered exclusive. + + Surfaces that do not wish to have an exclusive zone may instead specify + how they should interact with surfaces that do. If set to zero, the + surface indicates that it would like to be moved to avoid occluding + surfaces with a positive excluzive zone. If set to -1, the surface + indicates that it would not like to be moved to accomodate for other + surfaces, and the compositor should extend it all the way to the edges + it is anchored to. + + For example, a panel might set its exclusive zone to 10, so that + maximized shell surfaces are not shown on top of it. A notification + might set its exclusive zone to 0, so that it is moved to avoid + occluding the panel, but shell surfaces are shown underneath it. A + wallpaper or lock screen might set their exclusive zone to -1, so that + they stretch below or over the panel. + + The default value is 0. + + Exclusive zone is double-buffered, see wl_surface.commit. + + + + + + + Requests that the surface be placed some distance away from the anchor + point on the output, in surface-local coordinates. Setting this value + for edges you are not anchored to has no effect. + + The exclusive zone includes the margin. + + Margin is double-buffered, see wl_surface.commit. + + + + + + + + + + Set to 1 to request that the seat send keyboard events to this layer + surface. For layers below the shell surface layer, the seat will use + normal focus semantics. For layers above the shell surface layers, the + seat will always give exclusive keyboard focus to the top-most layer + which has keyboard interactivity set to true. + + Layer surfaces receive pointer, touch, and tablet events normally. If + you do not want to receive them, set the input region on your surface + to an empty region. + + Events is double-buffered, see wl_surface.commit. + + + + + + + This assigns an xdg_popup's parent to this layer_surface. This popup + should have been created via xdg_surface::get_popup with the parent set + to NULL, and this request must be invoked before committing the popup's + initial state. + + See the documentation of xdg_popup for more details about what an + xdg_popup is and how it is used. + + + + + + + When a configure event is received, if a client commits the + surface in response to the configure event, then the client + must make an ack_configure request sometime before the commit + request, passing along the serial of the configure event. + + If the client receives multiple configure events before it + can respond to one, it only has to ack the last configure event. + + A client is not required to commit immediately after sending + an ack_configure request - it may even ack_configure several times + before its next surface commit. + + A client may send multiple ack_configure requests before committing, but + only the last request sent before a commit indicates which configure + event the client really is responding to. + + + + + + + This request destroys the layer surface. + + + + + + The configure event asks the client to resize its surface. + + Clients should arrange their surface for the new states, and then send + an ack_configure request with the serial sent in this configure event at + some point before committing the new surface. + + The client is free to dismiss all but the last configure event it + received. + + The width and height arguments specify the size of the window in + surface-local coordinates. + + The size is a hint, in the sense that the client is free to ignore it if + it doesn't resize, pick a smaller size (to satisfy aspect ratio or + resize in steps of NxM pixels). If the client picks a smaller size and + is anchored to two opposite anchors (e.g. 'top' and 'bottom'), the + surface will be centered on this axis. + + If the width or height arguments are zero, it means the client should + decide its own window dimension. + + + + + + + + + The closed event is sent by the compositor when the surface will no + longer be shown. The output may have been destroyed or the user may + have asked for it to be removed. Further changes to the surface will be + ignored. The client should destroy the resource after receiving this + event, and create a new surface if they so choose. + + + + + + + + + + + + + + + + + diff --git a/src/wayland-window.cpp b/src/wayland-window.cpp index 7671b68..932e5bd 100644 --- a/src/wayland-window.cpp +++ b/src/wayland-window.cpp @@ -21,8 +21,14 @@ namespace wf { display->wf_manager = (zwf_shell_manager_v1*) wl_registry_bind(registry, name, - &zwf_shell_manager_v1_interface, - std::min(version, 1u)); + &zwf_shell_manager_v1_interface, std::min(version, 1u)); + } + + if (strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) + { + display->layer_shell = + (zwlr_layer_shell_v1*) wl_registry_bind(registry, name, + &zwlr_layer_shell_v1_interface, std::min(version, 1u)); } if (strcmp(interface, zwp_virtual_keyboard_manager_v1_interface.name) == 0) @@ -44,6 +50,23 @@ namespace wf ®istry_remove_object }; + static void layer_shell_handle_configure(void *data, struct zwlr_layer_surface_v1 *zwlr_layer_surface, + uint32_t serial, uint32_t width, uint32_t height) + { + zwlr_layer_surface_v1_ack_configure(zwlr_layer_surface, serial); + } + + static void layer_shell_handle_close(void *data, struct zwlr_layer_surface_v1 *surface) + { + zwlr_layer_surface_v1_destroy(surface); + } + + static struct zwlr_layer_surface_v1_listener layer_surface_listener = + { + &layer_shell_handle_configure, + &layer_shell_handle_close + }; + WaylandDisplay::WaylandDisplay() { auto gdk_display = gdk_display_get_default(); @@ -58,9 +81,10 @@ namespace wf wl_registry *registry = wl_display_get_registry(display); wl_registry_add_listener(registry, ®istry_listener, this); + wl_display_dispatch(display); wl_display_roundtrip(display); - if (!vk_manager || !seat || !wf_manager) + if (!vk_manager || !seat || (!wf_manager && !layer_shell)) { std::cerr << "Compositor doesn't support the virtual-keyboard-v1 " << "and/or the wayfire-shell protocols, exiting" << std::endl; @@ -74,6 +98,63 @@ namespace wf return instance; } + void WaylandWindow::initWayfireShell(WaylandDisplay display, int x, int y, int width, int height) + { + this->show_all(); + auto gdk_window = this->get_window()->gobj(); + auto surface = gdk_wayland_window_get_wl_surface(gdk_window); + + if (!surface) + { + std::cerr << "Error: created window was not a wayland surface" << std::endl; + std::exit(-1); + } + + wf_surface = zwf_shell_manager_v1_get_wm_surface(display.wf_manager, surface, + ZWF_WM_SURFACE_V1_ROLE_DESKTOP_WIDGET, nullptr); + zwf_wm_surface_v1_set_keyboard_mode(wf_surface, ZWF_WM_SURFACE_V1_KEYBOARD_FOCUS_MODE_NO_FOCUS); + zwf_wm_surface_v1_configure(wf_surface, x, y); + + } + + void WaylandWindow::initLayerShell(WaylandDisplay display, int width, int height) + { + auto gtk_window = this->gobj(); + auto gtk_widget = GTK_WIDGET(gtk_window); + gtk_widget_realize(gtk_widget); + + auto gdk_window = this->get_window()->gobj(); + gdk_wayland_window_set_use_custom_surface(gdk_window); + auto surface = gdk_wayland_window_get_wl_surface(gdk_window); + + if (!surface) + { + std::cerr << "Error: created window was not a wayland surface" << std::endl; + std::exit(-1); + } + + uint32_t layer = ZWLR_LAYER_SHELL_V1_LAYER_TOP; + layer_surface = zwlr_layer_shell_v1_get_layer_surface(display.layer_shell, + surface, NULL, layer, "wf-osk"); + if (!layer_surface) + { + std::cerr << "Error: could not create layer surface" << std::endl; + std::exit(-1); + } + + zwlr_layer_surface_v1_add_listener(layer_surface, &layer_surface_listener, nullptr); + zwlr_layer_surface_v1_set_keyboard_interactivity(layer_surface, 0); + zwlr_layer_surface_v1_set_size(layer_surface, width, height); + zwlr_layer_surface_v1_set_anchor(layer_surface, 0); + + wl_surface_commit(surface); + auto gdk_display = gdk_display_get_default(); + auto wl_display = gdk_wayland_display_get_wl_display(gdk_display); + wl_display_roundtrip(wl_display); + + this->show_all(); + } + WaylandWindow::WaylandWindow(int x, int y, int width, int height) : Gtk::Window() { @@ -83,21 +164,16 @@ namespace wf this->set_size_request(width, height); this->set_default_size(width, height); this->set_type_hint(Gdk::WINDOW_TYPE_HINT_DOCK); - this->show_all(); - auto gdk_window = this->get_window()->gobj(); - auto surface = gdk_wayland_window_get_wl_surface(gdk_window); - - if (!surface) + if (display.wf_manager) { - std::cerr << "Error: created window was not a wayland surface" << std::endl; + initWayfireShell(display, x, y, width, height); + } else if (display.layer_shell) + { + initLayerShell(display, width, height); + } else { + std::cerr << "Error: cannot find any supported shell protocol" << std::endl; std::exit(-1); } - - wm_surface = zwf_shell_manager_v1_get_wm_surface(display.wf_manager, - surface, ZWF_WM_SURFACE_V1_ROLE_DESKTOP_WIDGET, NULL); - zwf_wm_surface_v1_set_keyboard_mode(wm_surface, - ZWF_WM_SURFACE_V1_KEYBOARD_FOCUS_MODE_NO_FOCUS); - zwf_wm_surface_v1_configure(wm_surface, x, y); } } diff --git a/src/wayland-window.hpp b/src/wayland-window.hpp index 4fea46f..62de2fb 100644 --- a/src/wayland-window.hpp +++ b/src/wayland-window.hpp @@ -2,6 +2,7 @@ #include #include +#include #include namespace wf @@ -15,12 +16,18 @@ namespace wf wl_seat *seat = nullptr; zwf_shell_manager_v1 *wf_manager = nullptr; + zwlr_layer_shell_v1 *layer_shell = nullptr; zwp_virtual_keyboard_manager_v1 *vk_manager = nullptr; }; class WaylandWindow : public Gtk::Window { - zwf_wm_surface_v1 *wm_surface; + zwf_wm_surface_v1 *wf_surface; + zwlr_layer_surface_v1 *layer_surface; + + void initWayfireShell(WaylandDisplay display, int x, int y, int width, int height); + void initLayerShell(WaylandDisplay display, int width, int height); + public: WaylandWindow(int x, int y, int width, int height); };