/** \page FLTK-devel Development of the FLTK library - \subpage wayland-devel - \subpage bundled-libs - \subpage development */ /** \page wayland-devel The Wayland backend for its developer This chapter describes how the Wayland backend of FLTK works from a developer's viewpoint. \section wayland-intro Introduction to Wayland Wayland usage involves communication via a Unix domain socket between a client application and another process called the Wayland compositor which creates, moves, resizes and draws windows on the display. Diverse Wayland compositors exist. They can follow rather diverse logics. For example, FreeBSD offers Sway which is a tiling compositor where the display is always entirely filled with whatever resizable windows are mapped at any given time. Compositors follow either the client-side decoration (CSD) rule where client apps draw window titlebars, or the server-side decoration (SSD) rule where the compositor draws titlebars. FLTK supports both CSD and SSD compositors. It uses a library called \c libdecor charged of determining whether a CSD or a SSD compositor is active, and of drawing titlebars in the first case. Wayland is divided in various protocols that a given compositor may or may not support, although they all support the \c core protocol. Each protocol adds functionality not available in the core protocol. Wayland Explorer lists all protocols. The core protocol allows a client app to discover what protocols the connected compositor supports. Protocols can be stable, which means they have a defined API that will not change but can be expanded, or unstable. For example, mapping a window on a display is not done by the core protocol but by the xdg shell protocol which is stable. The names of symbols used by unstable protocols always begin with letter 'z'. For example, FLTK uses unstable protocol Text input to support CJK input methods; its symbol names begin with \c zwp_text_input_v3. Wayland makes intensive use of the listener mechanism. A listener is a small array of pointers to FLTK-defined callback functions associated to a Wayland-defined object; Wayland calls these functions when defined events occur (more at \ref wayland-listeners below). Wayland differs noticeably from X11 in that rendering is left to clients: Wayland provides no drawing API. Instead, Wayland provides objects of type struct wl_buffer which encapsulate a memory array of pixel values shared between the client and the compositor. The client app is expected to draw to that memory buffer with whatever means it chooses, and to instruct the compositor to map those pixels to the display when the drawing is complete. The Wayland platform of FLTK draws with the Cairo library to \c Fl_Window's and \c Fl_Image_Surface's, and with OpenGL to \c Fl_Gl_Window's. Wayland differs also from X11 in that the position of a window in the display is completely hidden to the client app. This prevents function \c Fl_Window::position() from having any effect on a top-level window. Wayland also prevents a client app from knowing whether a window is minimized: \c Fl_Window::show() has no effect on a minimized window. Subwindows can be positioned as usual relatively to their parent window. Wayland allows to create popup windows positioned relatively to a previously mapped other window. This allows FLTK to position adequately menu and tooltip windows (see \ref menu-windows). FLTK uses also popups for the small, yellow windows that display the new scale factor value when it's changed: these are created as short-lived popups centered above \c Fl::first_window(). Wayland uses a trick of its own to handle lists of linked records. It defines type struct wl_list and a few macros (\c wl_list_init(), \c wl_list_for_each(), \c wl_list_insert(), \c wl_list_for_each_safe(), \c wl_list_remove()) to manage linked lists. Records put in these lists must contain a member variable of type struct wl_list used to link records together and often named 'link'. Access to such a list is possible memorizing a value of type struct wl_list computed by macro \c wl_list_init(). Macro wl_list_for_each(arg1, arg2, arg3) allows to run through all list elements with: - \c arg1 is a pointer variable of the type of elements of the linked list; - \c arg2 is the address of a variable of type struct wl_list identifying the targeted list; - \c arg3 is the name of the member variable of these elements used to link them together. For example, \c wl_list_for_each() can be used as follows to scan the linked list of all displays of the system (see \ref wayland-output "Fl_Wayland_Screen_Driver::output"): \code Fl_Wayland_Screen_Driver::output *output; Fl_Wayland_Screen_Driver *scr_driver = (Fl_Wayland_Screen_Driver*)Fl::screen_driver(); wl_list_for_each(output, &(scr_driver->outputs), link) { // … work with output, an item of the linked list of all displays in the system … } \endcode Overall, and ignoring for now OpenGL usage, FLTK interacts with Wayland as follows : - When opening the display: FLTK calls \c Fl::add_fd() in \c FL_READ mode to associate a callback function to the socket connecting the client and the compositor. - Client to compositor: FLTK calls C functions of the \c libwayland-client.so, \c libwayland-cursor.so and \c libxkbcommon.so shared libraries and of the libdecor library. These send suitable messages to the compositor writing to the socket. The names of these functions begin with \c wl_, \c xkb_ or \c libdecor_. - Compositor to client: the callback function runs when there are data to read in the socket; it calls \c wl_display_dispatch() which interprets the read data and calls corresponding listeners. The core protocol defines also a number of mostly opaque structures whose names begin with \c wl_. The names of symbols and types defined by the other protocols FLTK uses begin with \c xdg_, \c zwp_text_input_v3, \c zxdg_toplevel_decoration_, \c gtk_shell1_ and \c gtk_surface1_. FLTK defines a few structures holding Wayland-related data. The names of FLTK-defined structures don't begin with \c wl_. For example, struct wld_window (see \ref wld_window) is used to store all Wayland-specific data associated to a mapped Fl_Window. \section wayland-build Building libfltk as a Wayland client Classes \c Fl_Wayland_Window_Driver, \c Fl_Wayland_Screen_Driver, \c Fl_Wayland_Graphics_Driver, \c Fl_Wayland_Copy_Surface_Driver, \c Fl_Wayland_Image_Surface_Driver and \c Fl_Wayland_Gl_Window_Driver and file \c fl_wayland_platform_init.cxx contain all the Wayland-specific code of the FLTK library. This code is located at \c src/drivers/Wayland/ in the FLTK source tree. A single C++ source file generally contains all the code of a given class. The code related to copy, paste and drag-and-drop operations, however, is gathered in file \c fl_wayland_clipboard_dnd.cxx and contains a few member functions of class \c Fl_Wayland_Screen_Driver. Furthermore, class \c Fl_Unix_System_Driver is used by both the Wayland and the X11 FLTK platforms. File FL/fl_config.h defines preprocessor variables \c FLTK_USE_WAYLAND and \c FLTK_USE_CAIRO. The public C API to Wayland, xkb, EGL and libdecor libraries are obtained with \code #include #include #include #include #include // for BTN_LEFT, BTN_RIGHT, BTN_MIDDLE #include "../../../libdecor/src/libdecor.h" #include "../../../libdecor/src/libdecor-plugin.h" #if HAVE_GL # include # include #endif // HAVE_GL \endcode as necessary. File \c README.Wayland.txt details what software packages are needed on Debian-based, Fedora and FreeBSD systems for FLTK to use Wayland. Wayland protocols are packaged as XML files accompanied by a utility program, \c wayland-scanner, able to generate a header file and a necessary glue C source file from a given XML file. For example, for FLTK to use the XDG shell protocol, these commands are run at build time to generate a .c file that will be compiled into libfltk and a header file that FLTK code will include: \code PROTOCOLS=`pkg-config --variable=pkgdatadir wayland-protocols` wayland-scanner private-code $PROTOCOLS/stable/xdg-shell/xdg-shell.xml xdg-shell-protocol.c wayland-scanner client-header $PROTOCOLS/stable/xdg-shell/xdg-shell.xml xdg-shell-client-protocol.h \endcode Similar operations are performed for FLTK to use protocols XDG decoration, Text input and GTK Shell. \section wayland-x11-hybrid The hybrid Wayland/X11 platform The Wayland platform of FLTK is normally a two-legged hybrid able to use either Wayland or X11 and to choose between these possibilities at run-time, without any change to the client application. The Wayland/X11 hybrid is essentially a version of the FLTK library containing both all Wayland-specific and all X11-specific code. That's reflected in file FL/fl_config.h which defines both \c FLTK_USE_WAYLAND and \c FLTK_USE_X11. This creates the constraint that Wayland and X11 cannot use the same type name for different purposes or the same symbol name. That is why function fl_xid(const Fl_Window*) is deprecated in FLTK 1.4 and replaced by \c fl_wl_xid() for Wayland and \c fl_x11_xid() for X11. Also, global variable Window fl_window is not used by the Wayland platform which instead uses static struct wld_window *Fl_Wayland_Window_Driver:: wld_window. The FLTK library contains also a short source file, \c fl_wayland_platform_init.cxx, that determines, at startup time, whether the app will run as a Wayland or as an X11 client. Function \c attempt_wayland() therein performs this choice as follows : - if the app defines a global bool variable called \c fl_disable_wayland and this variable is true, the X11 leg is chosen; - if environment variable FLTK_BACKEND is defined to string "wayland", the Wayland leg is chosen; - if environment variable FLTK_BACKEND is defined to string "x11", the X11 leg is chosen; - otherwise, a connection to a Wayland compositor is attempted; if it's successful, the Wayland leg is chosen; if it's not, the X11 leg is chosen. The first condition listed above is meant to facilitate transition to FLTK 1.4 of source code written for FLTK 1.3 and containing X11-specific code : it's enough to put \code FL_EXPORT bool fl_disable_wayland = true; \endcode anywhere in the source code, for the app to run with 1.4, using the x11 leg of the hybrid platform, without any other change in the source code nor to the application's environment. In special situations, such as with embedded systems equipped with the Wayland software but lacking the X11 library, it's possible to build the FLTK library such as it contains only the Wayland backend. This is achieved building FLTK with cmake -DFLTK_BACKEND_X11=OFF or with configure --disable-x11. In that case, FL/fl_config.h does not define \c FLTK_USE_X11. The rest of this chapter describes what happens when the Wayland leg has been chosen. \section wayland-listeners Listeners A Wayland 'listener' is a small array of pointers to FLTK-defined callback functions associated to a Wayland-defined object; Wayland calls these functions when defined events occur, and transmits relevant information to the client app as parameters of these calls. Each listener is associated to its corresponding Wayland object, usually right after the object's creation, by a call to a specific Wayland function named following the form \c wl_XXX_add_listener(). For example, this code: \code static void surface_enter(……) { …… } // called when a surface enters a display static void surface_leave(……) { …… } // called when a surface leaves a display static struct wl_surface_listener surface_listener = { surface_enter, surface_leave, }; some_pointer_type pter_to_data; struct wl_surface *my_wl_surface; my_wl_surface = wl_compositor_create_surface(scr_driver->wl_compositor); wl_surface_add_listener(my_wl_surface, &surface_listener, pter_to_data); \endcode creates a Wayland object of type struct wl_surface, and associates it with a 2-member listener called \c surface_listener. After this, Wayland is expected to call the 2 listener members, \c surface_enter or \c surface_leave, each time \c my_wl_surface will enter or leave, respectively, a display. The arguments of these calls, not detailed here, allow the member functions to identify which surface enters or leaves which display. The \c wl_surface_add_listener() call above also associates \c pter_to_data to \c my_wl_surface as user data. The \c wl_surface object's "user data" can be obtained later calling function \c wl_surface_get_user_data(). Wayland function \c wl_proxy_get_listener() returns a pointer to a Wayland object's listener provided that object is transmitted cast to type struct wl_proxy *. This gives a handy way to distinguish FLTK-created Wayland objects from objects of other origin: the listener of an FLTK-created object is a known FLTK listener. For example, function \c Fl_Wayland_Window_Driver::surface_to_window() uses this possibility calling wl_proxy_get_listener( (struct wl_proxy *)wl_surface ) for any object of type struct wl_surface: if that object was created as in the example above, this call returns a pointer to FLTK's \c surface_listener static variable. \section wayland-connection Opening a Wayland connection Establishing a Wayland connection requires environment variable \c XDG_RUNTIME_DIR to be defined and to point to a directory containing a socket connected to a Wayland compositor. This variable is usually set by the login procedure of Wayland-friendly desktops. The name of the Wayland socket is determined as follows: - the client may call Fl::display(const char *display_name) before \c fl_open_display() runs or use the \c -display command line argument and transmit there the socket name; - environment variable \c WAYLAND_DISPLAY can be defined to the socket name; - otherwise, \c "wayland-0" is used. What socket is selected determines what compositor will be used by the client application. Function \c Fl_Wayland_Screen_Driver::open_display_platform() establishes the connection to the Wayland socket identified above calling \c wl_display_connect(NULL) which returns a struct wl_display pointer or NULL in case of failure. Such NULL return is the hint that allows the FLTK display opening procedure of the Wayland/X11 hybrid to recognize when Wayland access is not possible and to fallback to X11. Then, function \c wl_registry_add_listener() associates a 2-member listener, whose 1st member, \c registry_handle_global(), will be called by Wayland a number of times to indicate each time a protocol supported by the compositor or a system feature such as displays and keyboards. This code allows to run the client until all calls to \c registry_handle_global() have occurred: \code static void sync_done(void *data, struct wl_callback *cb, uint32_t time) { *(struct wl_callback **)data = NULL; wl_callback_destroy(cb); } static const struct wl_callback_listener sync_listener = { sync_done }; struct wl_callback *registry_cb = wl_display_sync(wl_display); wl_callback_add_listener(registry_cb, &sync_listener, ®istry_cb); while (registry_cb) wl_display_dispatch(wl_display); \endcode A pointer to an object of type struct wl_callback created by function \c wl_display_sync() is assigned to variable \c registry_cb. Then a 1-member listener is attached to this object. Wayland will run this listener's member function, \c sync_done(), after all calls to \c registry_handle_global() have occurred. Function \c sync_done() sets to null variable \c registry_cb and destroys the \c wl_callback. Finally, function \c wl_display_dispatch() is called as long as variable \c registry_cb is not null. This makes Wayland process all its pending requests until \c sync_done() runs. The prototype of function \c registry_handle_global is: \code static void registry_handle_global(void *user_data, struct wl_registry *wl_registry, uint32_t id, const char *interface, uint32_t version) \endcode Each time Wayland calls \c registry_handle_global(), \c interface and \c version give the name and version of a component or feature of the Wayland system. It's necessary to call each time function \c wl_registry_bind() which returns a pointer to a Wayland structure that will be the client's access point to the corresponding Wayland protocol or system feature. This pointer is stored in a dedicated member variable of the unique \c Fl_Wayland_Screen_Driver object of an FLTK app, or of another object accessible from this object. For example, when \c interface equals "wl_compositor", the value returned by \c wl_registry_bind() is stored as member \c wl_compositor of the \c Fl_Wayland_Screen_Driver object. \c registry_handle_global() also identifies whether the Mutter, Weston, or KWin compositor is connected and stores this information in static member variable \c Fl_Wayland_Screen_Driver::compositor. Wayland calls \c registry_handle_global() with its parameter \c interface equals to "wl_output" once for each screen connected to the system. Each time, an object of type struct wl_output is created, to which a 4-member listener is associated by function \c wl_output_add_listener(). The 3rd member of this 4-function listener, \c output_done(), runs after all initialization steps of the screen have completed and turns to \c true member \c done of a record of type \c struct \ref wayland-output "Fl_Wayland_Screen_Driver::output" associated to the screen. Function \c sync_done() mentioned above therefore also calls \c wl_display_dispatch() until the \c done member of all \c Fl_Wayland_Screen_Driver::output records are \c true. Overall, after return from function \c sync_done(), FLTK has been made aware of all optional protocols and features of its connected Wayland compositor, and has initialized all screens of the system. Finally, function \c wl_display_get_fd() is called to obtain the file descriptor of the Wayland socket and a call to Fl::add_fd() makes FLTK listen to this descriptor in \c FL_READ mode and associates function \c wayland_socket_callback() from file \c Fl_Wayland_Screen_Driver.cxx with it. This function calls \c wl_display_dispatch() which reads and interprets data available from the file descriptor, and calls corresponding listeners. The \c wl_display_dispatch() call is repeated as long as data are available for reading. The event loop is run by function \c Fl_Unix_System_Driver::wait() which is used by both the Wayland and X11 FLTK backends. Among various tasks, this function waits for data arriving on the file descriptors FLTK is listening. Overall, the event loop of the Wayland backend is nearly exactly the same as that used by the X11 backend. The Wayland backend differs only in the callback function handling data read from the Wayland connection socket, and in overridden functions \c Fl_Wayland_Screen_Driver::poll_or_select_with_delay() and \c Fl_Wayland_Screen_Driver::poll_or_select(). \section wayland-surface Wayland windows and surfaces Wayland defines objects called surfaces of type struct wl_surface. A Wayland surface "has a rectangular area which may be displayed on zero or more displays, present buffers, receive user input, and define a local coordinate system". In other words, surface is the name Wayland uses for a window. Buffers allow the client app to draw to surfaces (see \ref wayland-buffer). FLTK creates a surface each time an Fl_Window is show()'n calling function \c wl_compositor_create_surface(). Static member function Fl_Wayland_Window_Driver::surface_to_window(struct wl_surface *) gives the \c Fl_Window* corresponding to the surface given in argument. FLTK recognizes 4 distinct kinds of surfaces named DECORATED, UNFRAMED, POPUP and SUBWINDOW. DECORATED are toplevel windows with a titlebar. UNFRAMED have no titlebar. POPUP correspond to menus and tooltips, SUBWINDOW to an Fl_Window embedded in another Fl_Window. Function \c Fl_Wayland_Window_Driver::makeWindow() creates all these surfaces, creates for each a record of type struct wld_window (see \ref wld_window), and stores the window kind in member variable \c kind of this record. Member variable \c xid of the window's \c Fl_X record stores the adress of this record. Except for SUBWINDOW's, each surface needs a Wayland object of type struct xdg_surface used to make it become a mapped window and stored in member \c xdg_surface of the window's \ref wld_window record. For DECORATED windows, this object is created inside libdecor and transmitted to FLTK by function \c libdecor_frame_get_xdg_surface(). For UNFRAMED and POPUP windows, it's created by function \c xdg_wm_base_get_xdg_surface(). Finally, each surface is also associated to one more Wayland object whose type varies with the window's kind. These explain this part of the \ref wld_window record: \code union { struct libdecor_frame *frame; // created for DECORATED by libdecor_decorate() struct wl_subsurface *subsurface; // created for SUBWINDOW by wl_subcompositor_get_subsurface() struct xdg_popup *xdg_popup; // created for POPUP by xdg_surface_get_popup() struct xdg_toplevel *xdg_toplevel; // created for UNFRAMED by xdg_surface_get_toplevel() }; \endcode Except for SUBWINDOW's, each surface is associated to a 'configure' function that Wayland calls one or more times when the window is going to be mapped on the display. The 'configure' function of DECORATED surfaces is \c handle_configure() which is the 1st member of a 4-member listener named \c libdecor_frame_iface associated to a decorated window when it's created calling \c libdecor_decorate(). Finally, a call to \c libdecor_frame_map() triggers the process of mapping the newly created DECORATED surface on a display. Wayland calls \c handle_configure() twice during this process. The first \c handle_configure() run allows to set the window's \c xdg_surface object which is returned by function \c libdecor_frame_get_xdg_surface(). FLTK distinguishes the first from the second run of \c handle_configure() by looking at the \c xdg_surface member variable that's NULL at the beginning of the 1st run and not NULL later. Wayland calls \c handle_configure() also during operations such as resizing, minimizing (see below). With the help of a few calls to libdecor functions, FLTK obtains in this function all needed information about the size and state of the mapped window. The 'configure' functions of UNFRAMED and POPUP surfaces are \c xdg_surface_configure(), \c xdg_toplevel_configure() and \c popup_configure(). The mapping process of these surfaces is triggered by a call to \c wl_surface_commit(). These 'configure' functions transmit effective window size information to FLTK. Also, they are where the window's \c Fl_Window_Driver::wait_for_expose_value member variable is set to 0 to indicate that the window has been mapped to display. \b Caution: there are some small differences between how and when the various Wayland compositors call \c handle_configure(). When a decorated window changes size, whatever the cause of it, Wayland calls \c handle_configure() which sets member variable \c Fl_Wayland_Window_Driver::in_handle_configure to true and calls the window's virtual \c resize() function which ultimately runs \c Fl_Wayland_Window_Driver::resize() which calls Fl_Group::resize() to perform FLTK's resize operations and \c Fl_Wayland_Graphics_Driver::buffer_release() to delete the existing window buffer that's not adequate for the new window size. At the end of the run of \c handle_configure(), \c in_handle_configure is set back to false. When the window size change is caused by the app itself calling the window's \c resize() function, \c Fl_Wayland_Window_Driver::in_handle_configure is false. This allows \c Fl_Wayland_Window_Driver::resize() to detect that Wayland needs be informed of the desired size change, which gets done by a call to \c libdecor_frame_commit(). Wayland later calls \c handle_configure() and events described above unfold. Wayland generally does not provide a way to control where the compositor should map a window in the system displays. Nevertheless, for multi-display systems, Wayland allows to control on what display should the compositor map a fullscreen window. That is done inside function \c handle_configure() which calls \c libdecor_frame_set_fullscreen() for DECORATED windows and inside function \c xdg_toplevel_configure() which calls \c xdg_toplevel_set_fullscreen() for UNFRAMED. The struct wl_output pointer for the targeted display is transmitted as 2nd argument of these calls. \section menu-windows Menu windows and other popups Menu windows, tiny menu title windows, and tooltip windows are implemented using Wayland's popup mechanism which allows to position a popup window relatively to a previously mapped window, itself a popup or another kind of window, with the restriction that any popup must overlap or at least touch that other window. Member function \c Fl_Wayland_Window_Driver::makeWindow calls member function \c Fl_Wayland_Window_Driver::process_menu_or_tooltip to create all popups. This function gets called after FLTK has computed using a given algorithm the desired \c (x,y) position of the popup window's top-left corner, using coordinates centered on the top-left corner of the toplevel window from which the popup originates. This algorithm is able to prevent popups from being positioned beyond the screen borders under the assumption that the position of a toplevel window inside a screen is known. While this assumption holds for other platforms, it does not for the Wayland platform. The FLTK code for the Wayland platform therefore modifies the algorithm that FLTK uses to compute the position of menu windows. The key information used by this algorithm is obtained by member function \c Fl_Window_Driver::menu_window_area which computes the coordinates of the rectangle where menu windows are allowed to be positioned. Under other platforms, this function just returns the origin and size of the work area of the screen in use. In contrast, the Wayland platform handles two situations differently : - For menu windows that are not taller than the display in use, the Wayland-overridden member function \c Fl_Wayland_Window_Driver::menu_window_area returns large negative origin and large width and height values. This lets the standard FLTK algorithm position the menu relatively to its window of origin without concern about screen limits, and relies on Wayland's constraint mechanism described below to prevent the menu from going beyond these limits, without FLTK having to know where they are. - Menu windows taller than the screen where they are mapped need special handling described in detail in a comment above the source code of function \c Fl_Wayland_Window_Driver::process_menu_or_tooltip. Function \c Fl_Wayland_Window_Driver::process_menu_or_tooltip first computes \c origin_win, pointer to the \c Fl_Window relatively to which the popup is to be positioned. Window \c origin_win is the parent menu window when the popup is a sub-menu; it's the tiny windowtitle when the popup is a menu with a title; otherwise, it's the window containing the point of origin of the popup. An object of type struct xdg_positioner created by function \c xdg_wm_base_create_positioner() is used to express the rules that will determine the popup position relatively to \c origin_win as follows: - Function \c xdg_positioner_set_anchor_rect() determines a rectangle in \c origin_win relatively to which the popup is to be positioned. When the popup to be created is a menu window spawned by an Fl_Menu_Bar, that rectangle is the full area of the menu title window. Otherwise, that rectangle is an adequately located point. - Function \c xdg_positioner_set_size() sets the popup size. - The xdg_positioner_set_anchor(positioner, XDG_POSITIONER_ANCHOR_BOTTOM_LEFT); and xdg_positioner_set_gravity(positioner, XDG_POSITIONER_GRAVITY_BOTTOM_RIGHT); calls position the popup so that its top-left corner is initially below and at right of the bottom-left corner of the \c origin_win 's anchor rectangle. - The call to \c xdg_positioner_set_offset() further changes the popup vertical position. - The call to \c xdg_positioner_set_constraint_adjustment() uses constraint flags \c XDG_POSITIONER_CONSTRAINT_ADJUSTMENT_SLIDE_X and \c XDG_POSITIONER_CONSTRAINT_ADJUSTMENT_SLIDE_Y which mean that the compositor will move the popup horizontally and vertically if its initial position would make it expand beyond the edges of the screen. Furthermore, flag XDG_POSITIONER_CONSTRAINT_ADJUSTMENT_FLIP_Y is added when the popup is a menu window spawned by an Fl_Menu_Bar; this has the popup flipped above the Fl_Menu_Bar if there's not enough screen room below it for the popup. - Finally, a call to function \c xdg_surface_get_popup() creates the popup accounting for position rules listed above. The positioner is then deleted by \c xdg_positioner_destroy(), a listener is associated to the popup surface with \c xdg_popup_add_listener(), and a call to \c wl_surface_commit() triggers the mapping of the popup on the display. Overall, the expected coordinates of the top-left corner of the popup relatively to \c origin_win are popup_x, popup_y. They are memorized in a record of FLTK-defined type struct win_positioner that's associated to the popup listener. When the compositor maps the popup, function \c popup_configure, the first element of the popup listener, runs and receives as arguments the coordinates of the popup top left and its size. These values account for the positioning constraints of the popup which may have moved it to avoid screen borders. This function can therefore detect whether constraints applied have modified the effective popup location in comparison to the expected coordinates which are available as member variables of the struct win_positioner record mentioned above. That's key to the handling by FLTK of tall menu windows. Groups of popups containing a menutitle, the associated menuwindow, and optionally a submenu window and that don't belong to an Fl_Menu_Bar are mapped in a different order: the menuwindow is mapped first, and the menutitle is mapped second above it as a child popup. Function \c Fl_Window_Driver::is_floating_title() detects when such a menutitle is created, static member variable \c previous_floatingtitle is assigned the value of this menutitle, and the menutitle is mapped only after the menuwindow has been mapped, as a child of it. This positions better the popup group in the display relatively to where the popup was created. \section wayland-graphics-driver Fl_Wayland_Graphics_Driver and Fl_Cairo_Graphics_Driver The Wayland platform of FLTK uses an \c Fl_Wayland_Graphics_Driver object for all its on-screen drawing operations. This object is created by function \c Fl_Graphics_Driver::newMainGraphicsDriver() called by \c Fl_Display_Device::display_device() when the library opens the display. New \c Fl_Wayland_Graphics_Driver objects are also created for each \c Fl_Image_Surface and each \c Fl_Copy_Surface used, and deleted when these objects are deleted. Class \c Fl_Wayland_Graphics_Driver derives from class \c Fl_Cairo_Graphics_Driver which implements all the FLTK drawing API for a Cairo surface. Function \c Fl_Wayland_Graphics_Driver::cairo_init() creates the Cairo surface used by each \c Fl_Wayland_Graphics_Driver object by calling \c cairo_image_surface_create_for_data() for the window's or offscreen's \c draw_buffer (see below). Class \c Fl_Cairo_Graphics_Driver is also used by the X11 leg of the hybrid Wayland-X11 platform because this leg draws to the display with an \c Fl_X11_Cairo_Graphics_Driver object which derives from class \c Fl_Cairo_Graphics_Driver. Finally, \c Fl_Cairo_Graphics_Driver is also used, in the form of an object from its derived class \c Fl_PostScript_Graphics_Driver, when the hybrid Wayland-X11 platform draws PostScript, or when the classic X11 platform uses Pango and draws PostScript. This happens when classes \c Fl_PostScript_File_Device and \c Fl_Printer are used. \section wayland-buffer Wayland buffers Wayland uses buffers, objects of type struct wl_buffer, to draw to surfaces. In principle, one or more buffers can be associated to a surface, and functions \c wl_surface_attach() and \c wl_surface_commit() are called to first attach one such buffer to the surface and then inform the compositor to map this buffer's graphics content on the display. Wayland buffers can use various memory layouts. FLTK uses WL_SHM_FORMAT_ARGB8888, which is the same layout as what Cairo calls CAIRO_FORMAT_ARGB32. FLTK calls function \c Fl_Wayland_Window_Driver::make_current() before drawing to any Fl_Window. Member \c buffer of this Fl_Window's struct wld_window (see \ref wld_window) is NULL when the window has just been created or resized. In that case, FLTK calls \c Fl_Wayland_Graphics_Driver::create_wld_buffer() which returns a pointer to a struct wld_buffer containing - a Wayland buffer, member \c wl_buffer; - a Cairo image surface, created by a call to \c Fl_Wayland_Graphics_Driver::cairo_init(). Each of these two objects encapsulates a byte array of the same size and the same memory layout destined to contain the Fl_Window's graphics. The Cairo image surface object is where FLTK draws. The Wayland buffer is what Wayland maps on the display. FLTK copies the Cairo surface's byte array to the Wayland buffer's byte array before beginning the mapping operation. If \c width and \c height are a window's dimensions in pixels, \code int stride = cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, width); int size = stride * height; \endcode give \c size, the common size of both byte arrays. The effective creation of the \c wl_buffer object is delayed until function \c Fl_Wayland_Graphics_Driver::buffer_commit() gets called. Section \ref wayland-buffer-factory below details how FLTK creates \c wl_buffer objects. The struct Fl_Wayland_Graphics_Driver::wld_buffer (see \ref wld_buffer) contains a pointer to the byte array of the Cairo image surface (member \c draw_buffer.buffer), information about the Wayland buffer (members \c wl_buffer and \c data), the common size of the Cairo surface's and Wayland buffer's byte arrays (member \c draw_buffer.data_size), and other information. A pointer to this struct Fl_Wayland_Graphics_Driver::wld_buffer is memorized as member \c buffer of the Fl_Window's \ref wld_window. All drawing operations to the Fl_Window then modify the content of the Cairo image surface. Function \c Fl_Wayland_Window_Driver::flush() is in charge of sending FLTK graphics data to the display. That is done by calling function \c Fl_Wayland_Graphics_Driver::buffer_commit() which creates the struct wl_buffer object calling \c create_shm_buffer() if that was not done before, copies the byte array of the Cairo surface to the Wayland buffer's starting memory address, and calls functions \c wl_surface_attach() and \c wl_surface_commit(). Before calling Fl_Window::flush(), FLTK has computed a damaged region. If that region is not null, \c Fl_Wayland_Graphics_Driver::buffer_commit() copies only the damaged parts of the Cairo surface to the Wayland buffer and calls function \c wl_surface_damage_buffer() for these parts to inform the compositor of what parts of the surface need its attention.

Throttling redraw operations

An important detail here is that FLTK uses Wayland's synchronization mechanism to make sure the surface's \c wl_buffer is not changed while the compositor is using it and to refrain from calling \c wl_surface_commit() more frequently than the system can process it. This 2-step mechanism works as follows: - Fl_Wayland_Graphics_Driver::buffer_commit() first calls function \c wl_surface_frame() to obtain a pointer to a struct wl_callback object and stores it as member \c frame_cb of the surface's \ref wld_window. Then it calls \c wl_callback_add_listener() to associate this object to the FLTK-defined, callback function \c surface_frame_done(). It next calls \c wl_surface_commit(). Together, these 3 calls instruct Wayland to start mapping the buffer content to the display and to call \c surface_frame_done() later, when it will have become ready for another mapping operation. - Later, \c surface_frame_done() runs and destroys the \c wl_callback object by function \c wl_callback_destroy() and sets member \c frame_cb to NULL. Member variable \c draw_buffer_needs_commit of the \ref wld_buffer is also important in this mechanism : it informs FLTK that the graphics buffer has changed and needs being committed. This variable is turned \c true every time a graphics operation changes the buffer content and turned \c false when the buffer gets committed. This procedure ensures that FLTK never changes the surface's Wayland buffer while it's being used by the compositor and never calls \c wl_surface_commit() before Wayland gets ready for a new commit because \c Fl_Wayland_Window_Driver::flush() calls \c Fl_Wayland_Graphics_Driver::buffer_commit() only if \c frame_cb is NULL. If it's not NULL, the exact content of function \c surface_frame_done() : \code static void surface_frame_done(void *data, struct wl_callback *cb, uint32_t time) { struct wld_window *window = (struct wld_window *)data; wl_callback_destroy(cb); window->frame_cb = NULL; if (window->buffer && window->buffer->draw_buffer_needs_commit) { Fl_Wayland_Graphics_Driver::buffer_commit(window); } } \endcode has the effect that when the mapping operation eventually completes, Wayland runs \c surface_frame_done(), which calls anew \c Fl_Wayland_Graphics_Driver::buffer_commit() if the buffer's \c draw_buffer_needs_commit member is true. The net result is that the screen shows the most recent surface content. This synchronization mechanism is also used when performing an interactive window resize operation. During such operation, the compositor informs the client an interactive resize is being performed and sends window resize commands at high rate (~60 Hz) to the client via the socket. Libdecor turns on flag \c LIBDECOR_WINDOW_STATE_RESIZING to inform the client, and runs function \c handle_configure() for each received resize command. Before calling Fl_Group::resize() and later Fl_Window::draw(), \c handle_configure() tests whether \c window->frame_cb is NULL. When it's not because a previous resize operation is being performed, the current resize command is skipped. At the end of the interactive resize, flag \c LIBDECOR_WINDOW_STATE_RESIZING is off and Wayland sends a final resize command which is not skipped. Overall, this ensures the client program resizes its window as frequently as it can without falling behind resize commands sent by the compositor. To account for a bug in Mutter (issue #878), the \c window->frame_cb object is not created when a toplevel window is being resized and is entirely covered by one subwindow.

Progressive window drawing

FLTK supports progressive drawing when an app calls function Fl_Window::make_current() at any time and then calls the FLTK drawing API. This is made possible in function \c Fl_Wayland_Window_Driver::make_current() with \code // to support progressive drawing if ( (!Fl_Wayland_Window_Driver::in_flush_) && window->buffer && (!window->frame_cb) && window->buffer->draw_buffer_needs_commit && (!wait_for_expose_value) ) { Fl_Wayland_Graphics_Driver::buffer_commit(window); } \endcode Thus, \c buffer_commit() runs only when \c frame_cb is NULL. If an app rapidly performs calls to \c Fl_Window::make_current() and to drawing functions, FLTK will copy \c draw_buffer to the Wayland buffer and instruct Wayland to map it to the display when \c frame_cb is NULL which means that the compositor is ready to start performing a mapping operation. This occurs when the progressive drawing operation begins. Later, \c frame_cb is generally found non NULL when \c Fl_Wayland_Window_Driver::make_current() runs because the compositor is busy processing the previous Wayland buffer. When the compositor has completed this processing, the client app runs \c surface_frame_done() which, provided member variable \c draw_buffer_needs_commit is true, calls \c Fl_Wayland_Graphics_Driver::buffer_commit(). This makes the compositor map the Wayland buffer in its new, more advanced, state. An example of progressive drawing is given by FLTK's mandelbrot test app. When set to fullscreen, this app can be seen to progressively fill its window from top to bottom by blocks of lines, each block appearing when the compositor is ready to map a new buffer. When the compositor is not ready, the app does not block but continues computing and drawing in memory but not on display more lines of the desired Mandelbrot graph.

Wayland buffer deletion

Each \ref wld_buffer record contains boolean member \c in_use which is set to \c true just before the buffer gets committed, and boolean member \c released which is set to \c true when FLTK no longer needs the buffer and calls \c Fl_Wayland_Graphics_Driver::buffer_release(). FLTK's buffer-creating function, \c Fl_Wayland_Graphics_Driver::create_shm_buffer(), attaches a 1-member listener to each buffer which Wayland calls after a commit operation to indicate the client is allowed to re-use the buffer. This listener's member function, \c buffer_release_listener(), turns to false member \c in_use of the buffer's \ref wld_buffer record. Since the two events 'FLTK no longer needs the buffer' and 'the client is allowed to re-use the buffer' can arrive in any order, FLTK deletes the struct wl_buffer object by running \c do_buffer_release() only after both events happened, that is, when \c in_use is \c false and \c released is \c true. That's why function \c do_buffer_release() is called by both functions \c Fl_Wayland_Graphics_Driver::buffer_release() and \c buffer_release_listener(). \section wayland-buffer-factory Buffer factories Wayland names buffer factory a software procedure that constructs objects of type struct wl_buffer for use by a client application. FLTK creates a \c wl_buffer object each time an Fl_Window is mapped on a display or resized. That's done by member function \c Fl_Wayland_Graphics_Driver::create_shm_buffer() which follows this 3-step procedure to create a "buffer factory" for FLTK and to construct Wayland buffers from it: - Libdecor function os_create_anonymous_file(off_t size) creates an adequate file and mmap's it. This file lives in RAM because it is created by function \c memfd_create(). FLTK sets this file size to 10 MB unless the size of the buffer to be created is larger; in that case the anonymous file is sized to twice the buffer size. - Wayland function \c wl_shm_create_pool() shares this mmap'ed memory with the Wayland compositor and returns an object of type struct wl_shm_pool which encapsulates this memory. A record of type struct Fl_Wayland_Graphics_Driver::wld_shm_pool_data is created and associated to the newly created \c wl_shm_pool by \c wl_shm_pool_set_user_data(). This record stores the starting address (\c pool_memory) and size (\c pool_size) of the pool's encapsulated memory. The record also contains member \c buffers of type struct wl_list which stores the access point to the linked list of \c wl_buffer objects that will be created from the \c wl_shm_pool. - A variable named \c chunk_offset represents the offset within the pool's shared memory available for the buffer being constructed. It equals 0 when the pool has just been created and is updated as detailed below when one or more buffers have been previously created from the pool. A record of type struct Fl_Wayland_Graphics_Driver::wld_buffer is created. This record will contain (member \c wl_buffer) the address of a \c wl_buffer object that's created by function \c wl_shm_pool_create_buffer(). This \c wl_buffer object encapsulates a section of a given size of the pool's shared memory beginning at offset \c chunk_offset in it. Quantity pool_memory + chunk_offset is therefore the address of the beginning of the mmap'ed memory section encapsulated by this \c wl_buffer. Member \c shm_pool of the newly constructed \c Fl_Wayland_Graphics_Driver::wld_buffer object is set to the address of the current \c wl_shm_pool object. This record is added to the head of the linked list of current pool's buffers by a call to \c wl_list_insert(). At that point, a struct Fl_Wayland_Graphics_Driver::wld_buffer record is part of the linked list of all such records corresponding to \c wl_buffer objects created from the same \c wl_shm_pool object, and member \c shm_pool of this record gives the address of this \c wl_shm_pool. When a new struct Fl_Wayland_Graphics_Driver::wld_buffer record is to be created, \code struct wld_shm_pool_data *pool_data = (struct wld_shm_pool_data *)wl_shm_pool_get_user_data(pool); struct Fl_Wayland_Graphics_Driver::wld_buffer *record = wl_container_of(pool_data->buffers.next, record, link); int chunk_offset = ((char*)record->data - pool_data->pool_memory) + record->data_size; \endcode gives the offset within the current pool's mmap'ed memory available for a new \c wl_buffer. Macro \c wl_container_of() gives the address of a record belonging to a linked list of records of the same type. A window's \c wl_buffer is re-filled by graphics data and committed each time the window gets redrawn, and is set to be destroyed by function \c Fl_Wayland_Graphics_Driver::buffer_release() when \c Fl_Window::hide() runs or the window is resized. When the \c wl_buffer is no longer in use, function \c do_buffer_release() gets called as explained above. It destroys the \c wl_buffer with \c wl_buffer_destroy(), and removes the corresponding \c Fl_Wayland_Graphics_Driver::wld_buffer record from the linked list of buffers from the same \c wl_shm_pool. Since new \c Fl_Wayland_Graphics_Driver::wld_buffer records are added at the head of the linked list, and since the record at the head of this list is used to compute the offset within the pool's mmap'ed memory available for a new \c wl_buffer, destruction of the last created \c wl_buffer allows to re-use the destroyed buffer's pool's memory for a new \c wl_buffer. When function \c do_buffer_release() finds the list of buffers from a given pool empty, two situations can occur. 1) This pool is the current pool. Its mmap'ed memory will be re-used from offset 0 to create future \c wl_buffer objects. 2) This pool is not current. It gets destroyed with \c wl_shm_pool_destroy(), the pool's mmap'ed memory is munmap'ed, and the pool's associated struct wld_shm_pool_data is freed. In situation 1) above, the next \c wl_buffer to be created can need more memory than the current pool's memory size. If so, the current pool gets destroyed and replaced by a new, larger pool. If the sum of \c chunk_offset plus the buffer size is larger than the current pool's size when function \c create_shm_buffer() is called, \c chunk_offset is reset to 0, and a new \c wl_shm_pool object is created and used by FLTK's "buffer factory". This mechanism allows to access new mmap'ed memory when \c chunk_offset reaches the end of the previous mmap'ed section. Wayland uses also \c wl_buffer objects to support cursors. FLTK uses the "buffer factory" described here when creating custom cursors (see \ref custom-cursor) with function Fl_Wayland_Window_Driver::set_cursor(const Fl_RGB_Image *,…) which calls \c create_shm_buffer() via \c set_cursor_4args(), \c custom_offscreen() and \c create_wld_buffer(). In contrast, standard shaped-cursors (e.g., FL_CURSOR_INSERT) use their own "buffer factory" inside Wayland functions such as \c wl_cursor_theme_get_cursor(). Therefore, the fact that the \c wl_buffer objects behind standard cursors are never destroyed doesn't prevent disused struct wl_shm_pool objects from being freed because those buffers come from a distinct "buffer factory". The "buffer factory" described here is also used by function \c offscreen_from_text() when displaying dragged text in a DnD operation. \section wayland-display Displays and HighDPI support Wayland uses the concept of seat of type struct wl_seat which encompasses displays, a keyboard, a mouse, and a trackpad. Although Wayland may be in principle able to deal with several seats, FLTK's Wayland platform is conceived for one seat only. That seat may contain one or more displays, which Wayland calls outputs, of type struct wl_output. As written above, function \c registry_handle_global() discovers the available seat at start-up time. This function also associates a listener to each display connected to the system by calling function \c wl_output_add_listener(). This listener's member functions run at program startup when Wayland discovers its displays (see \ref wayland-connection). Member \c output_mode runs also when the display is resized and member \c output_scale also when the Wayland scale factor (see below) is changed. FLTK defines type \c struct \ref wayland-output "Fl_Wayland_Screen_Driver::output" to store display size and scaling information. One such record is created for each display. These records are put in a struct wl_list accessible from member \c outputs of the single \c Fl_Wayland_Screen_Driver object. FLTK uses 2 distinct scaling parameters for each display: - int wld_scale;. This member variable of the \c struct \ref wayland-output "Fl_Wayland_Screen_Driver::output" record typically equals 1 for standard, and 2 for HighDPI displays. The effect of value \c n of variable \c wld_scale is that 1 Wayland graphics unit represents a block of \c nxn pixels. Another effect is that a drawing buffer for a surface of size WxH units contains W * n * H * n * 4 bytes. Member function \c output_scale() mentionned above sets this value for each system's display at startup time. Member function \c Fl_Wayland_Graphics_Driver::buffer_commit() informs the Wayland compositor of the value of \c wld_scale calling \c wl_surface_set_buffer_scale() which is enough to make FLTK apps HighDPI-aware. Under the gnome and KDE desktops, this parameter is visible in the "Settings" app, "Displays" section, "Scale" parameter which is 200% on HighDPI displays. - float gui_scale;. This other member variable is where FLTK's own GUI scaling mechanism with ctrl/+/-/0/ keystrokes and with environment variable FLTK_SCALING_FACTOR operates: when FLTK is scaled at 150%, \c gui_scale is assigned value 1.5. Function Fl_Wayland_Screen_Driver::scale(int n, float f) assigns value \c f to the \c gui_scale member variable of display # \c n. This variable is used by function \c Fl_Wayland_Window_Driver::make_current() when it calls \c Fl_Wayland_Graphics_Driver::set_buffer() that scales the graphics driver by this factor with \c cairo_scale(). Overall, an FLTK object, say an Fl_Window, of size \c WxH FLTK units occupies int(W * gui_scale) * wld_scale x int(H * gui_scale) * wld_scale pixels on the display. When an \c Fl_Window is to be show()'n, \c Fl_Wayland_Window_Driver::makeWindow() creates a struct wl_surface with \c wl_compositor_create_surface() and associates it calling \c wl_surface_add_listener() with a 2-member listener called \c surface_listener encharged of managing as follows the list of displays where this \c wl_surface will map. The \c Fl_Window possesses an initially empty linked list of displays accessible at member \c outputs of the window's \ref wld_window record. When the \c Fl_Window, or more exactly its associated struct wl_surface is mapped on a display, member \c surface_enter() of \c surface_listener runs. This function adds the display where the surface belongs to the end of the linked list of displays for this surface. When a surface is dragged or enlarged across the edge of a display in a multi-display system and expands on a second display, \c surface_enter() runs again, and this surface's list of displays contains 2 items. When a surface leaves a display, member \c surface_leave() of \c surface_listener runs. It removes that display from the surface's list of displays. Each time the first item of a surface's list of displays changes, function \c change_scale() is called and applies that display's \c gui_scale value to that surface calling \c Fl_Window_Driver::screen_num(int). When a window is unmapped by function \c Fl_Wayland_Window_Driver::hide(), the surface's list of displays is emptied.

Fractional scaling

The KWin compositor, and gnome too if specially set, allow to use fractional scaling that can take intermediate values between 100% and 200%. Wayland implements this rendering all wl_surface's as if the scaling was at 200%, and downsizing them to the desired fractional scale value at the compositing stage. Seen from FLTK, everything runs as when wld_scale = 2. These commands make gnome accept fractional scaling, and turn that off: \code gsettings set org.gnome.mutter experimental-features "['scale-monitor-framebuffer']" gsettings reset org.gnome.mutter experimental-features \endcode \section wayland-mouse Mouse and trackpad handling FLTK receives information about mouse and pointer events via a 'listener' made up of 5 pointers to functions which Wayland calls when events listed in table below occur. These functions receive from Wayland enough information in their parameters to generate corresponding FLTK events, that is, calls to Fl::handle(int event_type, Fl_Window *).
listener functioncalled by Wayland whenresulting FLTK events
\c pointer_enterpointer enters a windowFL_ENTER
\c pointer_leavepointer leaves a windowFL_LEAVE
\c pointer_motionpointer moves inside a windowFL_MOVE
\c pointer_buttonstate of mouse buttons changesFL_PUSH, FL_RELEASE
\c pointer_axistrackpad is moved vertically or horizontally FL_MOUSEWHEEL
\c pointer_listener is installed by a call to function \c wl_pointer_add_listener() made by function \c seat_capabilities() which is itself another 'listener' made up of 2 function pointers \code static struct wl_seat_listener seat_listener = { seat_capabilities, seat_name }; \endcode installed by a call to function \c wl_seat_add_listener() made by function \c registry_handle_global() when it receives a \c "wl_seat" interface.

Handling middle mouse button clicks on window titlebars

The gnome desktop, via its \c gnome-tweaks application, allows to determine what happens when a middle mouse button click occurs on a window titlebar. To obey this setting, FLTK implements part of the GTK Shell protocol as follows. Mutter, gnome's Wayland compositor, declares its support of the GTK Shell protocol calling \c registry_handle_global() with its \c interface argument equal to \c "gtk_shell1". FLTK initializes then a static global variable \c gtk_shell of type struct gtk_shell1*. Member functions of \c pointer_listener mentionned above run for all mouse events on all \c wl_surface objects. The table above describes what these functions do for mouse events on FLTK-created \c wl_surface objects. But they also run for the libdecor-created \c wl_surface objects corresponding to window titlebars. Thus, member function \c pointer_enter() runs when the mouse enters a titlebar. It calls \c Fl_Wayland_Screen_Driver::event_coords_from_surface() which calls \c Fl_Wayland_Window_Driver::surface_to_window() which, as mentionned above, can distinguish FLTK-created from non FLTK-created \c wl_surface objects. This allows \c pointer_enter() to identify the entered surface as a titlebar and to assign static global variable \c gtk_shell_surface with the titlebar's \c wl_surface when the mouse enters a titlebar. Similarly, member function \c pointer_leave() sets \c gtk_shell_surface to NULL when the mouse leaves this titlebar. When there's a click on a titlebar, member function \c pointer_button() runs this code \code if (gtk_shell_surface && state == WL_POINTER_BUTTON_STATE_PRESSED && button == BTN_MIDDLE) { struct gtk_surface1 *gtk_surface = gtk_shell1_get_gtk_surface(gtk_shell, gtk_shell_surface); gtk_surface1_titlebar_gesture(gtk_surface, serial, seat->wl_seat, GTK_SURFACE1_GESTURE_MIDDLE_CLICK); gtk_surface1_release(gtk_surface); return; } \endcode which ensures that what \c gnome-tweaks has assigned to middle-click events is executed. At this point, FLTK obeys what \c libdecor decides for right-click (display the window menu) and double-click (maximize the window) events on titlebars which may diverge from \c gnome-tweaks settings. \section wayland-cursor Wayland cursors Wayland defines types struct wl_cursor and struct wl_cursor_theme to hold cursor-related data. FLTK uses function \c init_cursors() from file \c Fl_Wayland_Screen_Driver.cxx to obtain the 'cursor theme' name using function \c libdecor_get_cursor_settings() of library \c libdecor. Function \c wl_cursor_theme_load() then returns a pointer to an object of type struct wl_cursor_theme stored in member variable \c cursor_theme of the \ref wayland-seat "Fl_Wayland_Screen_Driver::seat" record. Function \c init_cursors() is itself called by a 'listener' called \c seat_capabilities() installed when function \c registry_handle_global() receives a \c "wl_seat" interface, at program startup. It is also called when the value of the Wayland scaling factor changes: \c output_done() calls \c try_update_cursor() calls \c init_cursors(). Function \c output_done() belongs to a 'listener' installed when function \c registry_handle_global() receives a \c "wl_output" interface. Each time \c Fl_Window::cursor(Fl_Cursor) runs, FLTK calls \c Fl_Wayland_Window_Driver::set_cursor(Fl_Cursor) which calls \c wl_cursor_theme_get_cursor() to set the current cursor shape to one of the standard shapes from the \c Fl_Cursor enumeration. This Wayland function selects a cursor shape based on the current \c wl_cursor_theme object and a cursor name and returns a pointer to a struct wl_cursor. Under the gnome desktop, cursor names are the files of directory \c /usr/share/icons/XXXX/cursors/ where \c XXXX is the 'gnome cursor theme' (default= Adwaita). For example, what FLTK calls \c FL_CURSOR_INSERT corresponds to file \c xterm therein. The full correspondance between \c Fl_Cursor values and names of files therein is found in function \c Fl_Wayland_Window_Driver::set_cursor(Fl_Cursor). FLTK stores in member variable \c default_cursor of the \ref wayland-seat "Fl_Wayland_Screen_Driver::seat" record a pointer to the currently used \c wl_cursor object, and the current \c Fl_Cursor value in member \c standard_cursor_ of the \c Fl_Wayland_Window_Driver object. Finally, function do_set_cursor() of file \c Fl_Wayland_Screen_Driver.cxx makes the system pointer use the current \c wl_cursor object to draw its shape on screen. That's done with a call to \c wl_pointer_set_cursor() and a few other functions. \anchor custom-cursor

Custom cursor shapes

To support custom cursors, FLTK presently uses a non-public type, struct cursor_image, defined in file \c Fl_Wayland_Window_Driver.cxx as follows: \code struct cursor_image { struct wl_cursor_image image; struct wl_cursor_theme *theme; struct wl_buffer *buffer; int offset; }; \endcode This definition has been copied to the FLTK source code from file wayland-cursor.c of the Wayland project source code because it's not accessible via Wayland header files. It shows that a pointer to a \c cursor_image object can also be viewed as a pointer to the embedded struct wl_cursor_image object, this one being part of the public Wayland API. It also shows that a struct cursor_image object has an associated struct wl_buffer object used to contain the cursor's graphics. Function Fl_Wayland_Window_Driver::set_cursor(const Fl_RGB_Image *rgb, int hotx, int hoty) gives FLTK support of custom cursor shapes. It calls \c Fl_Wayland_Window_Driver::set_cursor_4args() that creates a \c cursor_image object, allocates the corresponding \c wl_buffer by a call to \c Fl_Wayland_Graphics_Driver::create_shm_buffer() via \c custom_offscreen() and \c create_wld_buffer() and draws the cursor shape into that buffer using the offscreen-drawing method of FLTK. The public type struct wl_cursor is essentially an array of \c wl_cursor_image objects and a name: \code struct wl_cursor { unsigned int image_count; struct wl_cursor_image **images; char *name; }; \endcode Function \c Fl_Wayland_Window_Driver::set_cursor_4args() also creates a struct wl_cursor object containing a single \c wl_cursor_image, which is in fact the \c cursor_image. Finally, a struct custom_cursor_ (see \ref wld_window) is allocated and used to memorize the struct wl_cursor and the cursor's image and hotspot. A pointer to this struct custom_cursor_ object is stored in member \c custom_cursor of the window's \ref wld_window. Function \c Fl_Wayland_Window_Driver::set_cursor_4args() is also called when a window with a custom cursor is moved between distinct displays or when a display is rescaled to adapt the cursor size to the new display's scale factor. Member function \c Fl_Wayland_Window_Driver::delete_cursor_() is used to delete any custom cursor shape. This occurs when a window associated to a custom cursor is un-mapped and when such a window gets associated to a standard cursor or to a new custom cursor. \section wayland-text Keyboard support The "Mouse handling" section above mentionned function \c seat_capabilities() that Wayland calls when the app discovers its "seat". Presence of flag \c WL_SEAT_CAPABILITY_KEYBOARD in argument \c capabilities of this function indicates that a keyboard is available. In that case, a call to \c wl_seat_get_keyboard() returns a pointer stored in member \c wl_keyboard of the \ref wayland-seat "Fl_Wayland_Screen_Driver::seat" object, and a call to \c wl_keyboard_add_listener() installs a 6-member listener of type struct wl_keyboard_listener. These 6 FLTK-defined, callback functions are used as follows. Function \c wl_keyboard_keymap() runs when the app starts and also if the keyboard layout is changed during run-time. It allows initialization of access to this keyboard. Noticeably, member \c xkb_state of type struct xkb_state* of the current \ref wayland-seat "Fl_Wayland_Screen_Driver::seat" record is adequately initialized. Functions \c wl_keyboard_enter() and \c wl_keyboard_leave(), called when focus enters and leaves a surface, send \c FL_FOCUS and \c FL_UNFOCUS events to the \c Fl_Window object corresponding to this surface. Function \c wl_keyboard_key() runs each time a keyboard key is pressed or released. Its argument \c key, to which 8 must be added, provides the keycode via function \c xkb_state_key_get_one_sym() and then the corresponding text via function \c xkb_state_key_get_utf8() which is put in \c Fl::e_text. Then, a few calls to functions whose name begin with \c xkb_compose_ are necessary to support dead and compose keys. Finally a call to \c Fl::handle() sends an \c FL_KEYDOWN or \c FL_KEYUP event to the appropriate \c Fl_Window. Also, function \c wl_keyboard_key() uses global variable Fl_Int_Vector key_vector to record all currently pressed keys. This is the base of the implementation of \c Fl_Wayland_Screen_Driver::event_key(int). Function \c wl_keyboard_modifiers() runs when a modifier key (e.g., shift, control) is pressed or released. Calls to functions \c xkb_state_update_mask() and \c xkb_state_mod_name_is_active() allow FLTK to set \c Fl::e_state adequately. Function \c wl_keyboard_repeat_info() does not run, for now, because this would require version 4 of the wl_keyboard object which is at version 2 in all tested Wayland compositors. \section wayland-text-input Support of text input methods When the connected Wayland compositor supports text input methods, function \c registry_handle_global() gets called with its \c interface argument equal to \c zwp_text_input_manager_v3_interface.name. The following call to \c wl_registry_bind() returns a pointer to type struct zwp_text_input_manager_v3 that is stored as member \c text_input_base of the \c Fl_Wayland_Screen_Driver object. Later, when function \c seat_capabilities() runs, \c text_input_base is found not NULL, which triggers a call to function \c zwp_text_input_manager_v3_get_text_input() returning a value of type struct zwp_text_input_v3 * and stored as member \c text_input of the \ref wayland-seat "Fl_Wayland_Screen_Driver::seat" object. Next, a call to \c zwp_text_input_v3_add_listener() associates this \c text_input with a 6-member listener of type struct zwp_text_input_v3_listener. These 6 FLTK-defined, callback functions are used as follows. Functions \c text_input_enter() and \c text_input_leave() are called when text input enters or leaves a surface (which corresponds to an \c Fl_Window). Functions \c text_input_preedit_string() and \c text_input_commit_string() are called when the text input method asks the client app to insert 'marked' text or regular text, respectively. Complex text input often begins by inserting temporary text which is said to be 'marked' before replacing it with the text that will stay in the document. FLTK underlines marked text to distinguish it from regular text. Functions \c text_input_delete_surrounding_text() and \c text_input_done() have no effect at present, without this preventing input methods that have been tested with FLTK from working satisfactorily. It's necessary to inform text input methods of the current location of the insertion point in the active surface. This information allows them to map their auxiliary windows next to the insertion point, where they are expected to appear. The flow of information on this topic is as follows: - The two FLTK widgets supporting text input, Fl_Input_ and Fl_Text_Display, transmit to FLTK the window coordinates of the bottom of the current insertion point and the line height each time they change calling function \c fl_set_spot(). - fl_set_spot() calls the platform override of virtual member function \c Fl_Screen_Driver::set_spot(). Under Wayland, this just calls \c Fl_Wayland_Screen_Driver::insertion_point_location(int x, int y, int height) which calls \c zwp_text_input_v3_set_cursor_rectangle() to inform the text input method about the surface position and size of the insertion point and also memorizes this information in static member variables of class \c Fl_Wayland_Screen_Driver. - Callback function \c text_input_enter() calls \c Fl_Wayland_Screen_Driver::insertion_point_location(int *x, int *y, int *height) which gives it the stored position information, and then calls \c zwp_text_input_v3_set_cursor_rectangle() to inform the text input method about the position of the insertion point. \section wayland-libdecor Interface with libdecor FLTK uses a library called libdecor to determine whether the Wayland compositor uses CSD or SSD mode, and also to draw window titlebars when in CSD mode (see \ref bundled-libdecor). \c Libdecor is conceived to be present in a shared library linked to the Wayland client application which itself, and if the running Wayland compositor uses CSD mode, loads another shared library intended to draw titlebars in a way that best matches the Desktop. As of late 2023, libdecor is at version 0.2.0 and contains two titlebar-drawing plugins: - \c libdecor-gtk intended for the Gnome desktop; - \c libdecor-cairo for other situations. Because \c libdecor is not yet in major Linux packages, or only at version 0.1.x, FLTK bundles the most recent source code of \c libdecor and its plugins. This code is included in libfltk. FLTK uses \c libdecor-gtk when software package \c libgtk-3-dev is present in the build system, and \c libdecor-cairo otherwise. As of late 2023, libdecor version 0.2.0 is available in very recent Linux distributions in packages \c libdecor-0-dev and \c libdecor-0-plugin-1-gtk. If they are installed on the build system, preprocessor variable \c USE_SYSTEM_LIBDECOR is 1, and both \c libdecor and its plugin are loaded at run-time from shared libraries. When these packages are not available or are at an earlier version, FLTK uses the bundled copy of \c libdecor. When CMake \c FLTK_USE_SYSTEM_LIBDECOR is OFF, FLTK uses the bundled \c libdecor copy even if shared libraries \c libdecor.so and \c libdecor-gtk.so are installed. This option is ON by default. \c Libdecor uses the Wayland protocol XDG decoration to request being decorated by a supporting compositor. If the running compositor supports SSD, \c libdecor doesn't draw window titlebars because the compositor does it. That is what happens with the \c KWin and \c Sway compositors. However, if environment variable \c LIBDECOR_FORCE_CSD is defined to value \c 1 when an FLTK app runs, \c libdecor instructs an SSD-able compositor to refrain from decorating its windows and decorates windows itself. Whatever the value of \c FLTK_USE_SYSTEM_LIBDECOR, FLTK and \c libdecor use environment variable \c LIBDECOR_PLUGIN_DIR as follows: if this variable is defined and points to the name of a directory, this directory is searched for a potential \c libdecor plugin in the form of a shared library; if one is found, FLTK and \c libdecor load it and use it. The \c libdecor source code bundled in FLTK is identical to that of the \c libdecor repository. Nevertheless, FLTK uses this code with some minor changes. For example, except if \c USE_SYSTEM_LIBDECOR is 1, FLTK needs to modify function \c libdecor_new() charged of loading the plugin, to make it use the plugin code that is included in libfltk if none is found as a dynamic library. This is done as follows in file \c libdecor/build/fl_libdecor.c: \code #define libdecor_new libdecor_new_orig #include "../src/libdecor.c" #undef libdecor_new void libdecor_new() { // FLTK rewrite of this function …… } \endcode FLTK compiles file \c fl_libdecor.c which includes \c libdecor.c to the effect that all of the \c libdecor code becomes part of libfltk except that function \c libdecor_new() is substituted by its FLTK rewrite, without file \c libdecor.c being modified at all. This trick is also used to modify function \c libdecor_frame_set_minimized() to bypass a bug in the Weston compositor before version 10. Similarly, FLTK compiles file \c fl_libdecor-plugins.c which includes either \c libdecor-gtk.c or \c libdecor-cairo.c to the effect that the desired plugin becomes part of libfltk. To support function \c Fl_Widget_Surface::draw_decorated_window() that draws a mapped window and its titlebar, FLTK needs to perform two operations: 1) identify what plugin is operating, and 2) call a function that is specific of that plugin and that returns the pixels of the drawn titlebar. FLTK performs operation 1) above using its function \c get_libdecor_plugin_description() of file \c fl_libdecor-plugins.c that returns a human readable string describing the running plugin. Each plugin puts its own string in member \c description of a record of type struct libdecor_plugin_description. Although this type is public in header file \c libdecor-plugin.h, accessing the symbol defined by the plugin to store a pointer to a value of this type is complicated for a reason and solved by a method detailed in a comment before function \c get_libdecor_plugin_description(). Operation 2) above is done by FLTK-defined function \c fl_libdecor_titlebar_buffer() from file \c fl_libdecor-plugins.c. This function calls \c get_libdecor_plugin_description() seen above to get the running plugin's descriptive string. That is "GTK3 plugin" with \c libdecor-gtk. FLTK function \c gtk_titlebar_buffer() is then called, and returns a pointer to the start of a byte buffer containing the titlebar graphics. That is, again, not possible with the public \c libdecor API. Therefore, FLTK copies to \c fl_libdecor-plugins.c the definitions of several types given in \c libdecor-gtk.c or \c libdecor-cairo.c such as type struct border_component. \section wayland-clipboard Copy/Paste/Drag-n-Drop FLTK follows the procedure that is very well described in item "Wayland clipboard and drag & drop" of the \ref wayland-doc. All corresponding source code is in file \c src/drivers/Wayland/fl_wayland_clipboard_dnd.cxx. This part of the \ref wayland-seat "Fl_Wayland_Screen_Driver::seat" record stores pointers to Wayland objects used for clipboard and D-n-D operations: \code struct wl_data_device_manager *data_device_manager; struct wl_data_device *data_device; struct wl_data_source *data_source; \endcode FLTK can copy or paste plain UTF-8 text or image data to/from the clipboard. Images are copied to the clipboard as \c image/bmp mime type. Images in \c image/bmp or \c image/png mime types from the clipboard can be pasted to FLTK apps. Files dropped are received one pathname per line, with no \c '\\n' after the last pathname. \section wayland-egl EGL as support for OpenGL Wayland uses EGL™ to interface OpenGL with the underlying native platform window system. OpenGL-using FLTK apps are therefore linked to \c libwayland-egl.so and \c libEGL.so in addition to \c libGL.so and \c libGLU.so. EGL is initialized calling member function \c Fl_Wayland_Gl_Window_Driver::init() once, the first time the \c Fl_Wayland_Gl_Window_Driver c'tor runs. That is done with calls to eglGetDisplay(), eglInitialize(), and \c eglBindAPI(). Member function \c Fl_Wayland_Gl_Window_Driver::find() calls \c eglChooseConfig() to filter the set of GL configurations that match the \c Fl_Gl_Window's mode(), and puts in the returned \c Fl_Gl_Choice object the first matching configuration. The filtering gets done with bits \c EGL_WINDOW_BIT, to support the creation of window surfaces, and \c EGL_OPENGL_BIT, to support the creation of OpenGL contexts. EGL needs 2 more objects created for each \c Fl_Gl_Window. They have types struct wl_egl_window and \c EGLSurface, and are created by member function \c Fl_Wayland_Gl_Window_Driver::make_current_before() which runs at the beginning of \c Fl_Gl_Window::make_current(). The first argument of the call to \c wl_egl_window_create() therein has type struct wl_surface * and is what connects EGL with the targeted Wayland window. EGL creates with \c eglCreateContext() an object of type \c EGLContext via member function \c Fl_Wayland_Gl_Window_Driver::create_gl_context() called by \c Fl_Gl_Window::make_current(). Types \c EGLContext and \c GLContext are 2 names for the same object. The call to \c eglCreateContext() is made asking for a GL context of version at least 2. This does not prevent from obtaining contexts of higher versions, namely above 3.2, which are compatible with version 2 (the so-called compatibility profile) under all tested Linux systems. FLTK function \c Fl_Gl_Window::make_current() calls overridden function \c Fl_Wayland_Gl_Window_Driver::set_gl_context() which calls EGL function \c eglMakeCurrent() when the cached context changes. FLTK calls function \c Fl_Wayland_Gl_Window_Driver::swap_buffers() each time it wants a GL context to be sent to the display. This function contains some pure GL code to emulate an overlay buffer to support Fl_Gl_Window objects overriding their \c draw_overlay() member function. Then, it calls function \c eglSwapBuffers(). The overridden \c Fl_Wayland_Gl_Window_Driver::resize() function is implemented with calls to \c wl_egl_window_get_attached_size() and \c wl_egl_window_resize(). Class \c Fl_Wayland_Gl_Plugin exists to allow \c libfltk to call functions from \c libfltk_gl, \c libwayland-egl.so or \c libEGL.so and without having \c libfltk force linking any FLTK app with these GL-related libraries. For example, \c Fl_Wayland_Window_Driver::flush() needs to call \c Fl_Gl_Window::valid(0). \section wayland-type FLTK-defined, Wayland-specific types \anchor wld_window

struct wld_window

Defined in \c Fl_Wayland_Window_Driver.H. One such record is created for each shown()'n Fl_Window by \c Fl_Wayland_Window_Driver::makeWindow(). Function \c fl_wl_xid(Fl_Window*) returns a pointer to the struct wld_window of its argument.
struct wld_window {
  Fl_Window *fl_win;
  struct wl_list outputs; // linked list of displays where part or whole of window maps
  struct wl_surface *wl_surface; // the window's surface
  struct wl_callback *frame_cb; // non-NULL until Wayland can process new surface commit
  struct Fl_Wayland_Graphics_Driver::wld_buffer *buffer; // see \ref wld_buffer
  struct xdg_surface *xdg_surface;
  enum Fl_Wayland_Window_Driver::kind kind; // DECORATED or POPUP or SUBWINDOW or UNFRAMED
  union {
    struct libdecor_frame *frame; // for DECORATED windows
    struct wl_subsurface *subsurface; // for SUBWINDOW windows
    struct xdg_popup *xdg_popup; // for POPUP windows
    struct xdg_toplevel *xdg_toplevel; // for UNFRAMED windows
  };
  struct custom_cursor_ {
    struct wl_cursor *wl_cursor;
    const Fl_RGB_Image *rgb;
    int hotx, hoty;
  } *custom_cursor; // non-null when using custom cursor
  int configured_width; // used when negotiating window size with the compositor
  int configured_height;
  int floating_width; // helps restoring size after un-maximizing
  int floating_height;
  int state; // indicates whether window is fullscreen, maximized. Used otherwise for POPUPs
  bool covered; // specially for Mutter and issue #878
}
\anchor draw_buffer

struct Fl_Wayland_Graphics_Driver::draw_buffer

Defined in file \c Fl_Wayland_Graphics_Driver.H. One such record is created when an Fl_Image_Surface object is created. One such record is also embedded inside each struct Fl_Wayland_Graphics_Driver::wld_buffer record (see \ref wld_buffer).
struct Fl_Wayland_Graphics_Driver::draw_buffer {
  unsigned char *buffer; // address of the beginning of the Cairo image surface's byte array
  cairo_t *cairo_; // used when drawing to the Cairo image surface
  size_t data_size; // of buffer and wl_buffer, in bytes
  int stride; // bytes per line
  int width; // in pixels
};
FLTK gives offscreen buffers the platform-dependent type \c Fl_Offscreen which is in fact member \c cairo_ of struct Fl_Wayland_Graphics_Driver::draw_buffer. Thus, a variable with type \c Fl_Offscreen needs be cast to type \c cairo_t*. Static member function struct draw_buffer *offscreen_buffer(Fl_Offscreen) of class \c Fl_Wayland_Graphics_Driver returns the \c draw_buffer record corresponding to an \c Fl_Offscreen value. \anchor wld_buffer

struct Fl_Wayland_Graphics_Driver::wld_buffer

Defined in file \c Fl_Wayland_Graphics_Driver.H. One such record is created by \c Fl_Wayland_Graphics_Driver::create_wld_buffer() when an Fl_Window is show()'n or resized, when a custom cursor shape is created, or when text is dragged.
struct Fl_Wayland_Graphics_Driver::wld_buffer {
  struct draw_buffer draw_buffer; // see \ref draw_buffer
  struct wl_list link; // links all buffers from the same wl_shm_pool
  struct wl_buffer *wl_buffer; // the Wayland buffer
  void *data; // address of the beginning of the Wayland buffer's byte array
  struct wl_shm_pool *shm_pool; // pter to wl_shm_pool from which this wl_buffer comes
  bool draw_buffer_needs_commit; // true when draw_buffer has been modified but not yet committed
  bool in_use; // true while being committed
  bool released; // true after buffer_release() was called
};
\anchor wayland-output

struct Fl_Wayland_Screen_Driver::output

Defined in Fl_Wayland_Screen_Driver.H. One such record is created for each display of the system by function \c registry_handle_global() when it receives a \c "wl_output" interface. These records are kept in a linked list of them all, and an identifier of this linked list is stored in member \c outputs of the unique \c Fl_Wayland_Screen_Driver object FLTK uses. Thus, \code Fl_Wayland_Screen_Driver *scr_driver = (Fl_Wayland_Screen_Driver*)Fl::screen_driver(); struct wl_list list_of_all_displays = scr_driver->outputs; \endcode gives access, the Wayland way, to the linked list of displays in the system.
struct Fl_Wayland_Screen_Driver::output { // one record for each display
  uint32_t id; // an identifier of the display
  int x, y; // logical position of the top-left of display
  int width; // nber of horizontal pixels
  int height; // nber of vertical pixels
  float dpi; // at this point, always 96.
  struct wl_output *wl_output; // the Wayland object for this display
  int wld_scale; // Wayland scale factor
  float gui_scale; // FLTK scale factor
  bool done; // true means record members have been initialized
  struct wl_list link; // links these records together
};
It's possible to get the FLTK-defined record associated to a display from the Wayland-associated object for the same display, say struct wl_output *wl_output, by this call: (struct Fl_Wayland_Screen_Driver::output *)wl_output_get_user_data(wl_output). \anchor wayland-seat

struct Fl_Wayland_Screen_Driver::seat

Defined in file \c Fl_Wayland_Screen_Driver.H. One record is created by function \c registry_handle_global() when it receives a \c "wl_seat" or \c wl_data_device_manager_interface.name interface. A pointer to this struct is stored in member \c seat of the client's unique \c Fl_Wayland_Screen_Driver object.
struct Fl_Wayland_Screen_Driver::seat {
  struct wl_seat *wl_seat;
  struct wl_pointer *wl_pointer;
  struct wl_keyboard *wl_keyboard;
  uint32_t keyboard_enter_serial;
  struct wl_surface *keyboard_surface;
  struct wl_list pointer_outputs;
  struct wl_cursor_theme *cursor_theme;
  struct wl_cursor *default_cursor;
  struct wl_surface *cursor_surface;
  struct wl_surface *pointer_focus;
  int pointer_scale;
  uint32_t serial;
  uint32_t pointer_enter_serial;
  struct wl_data_device_manager *data_device_manager;
  struct wl_data_device *data_device;
  struct wl_data_source *data_source;
  struct xkb_state *xkb_state;
  struct xkb_context *xkb_context;
  struct xkb_keymap *xkb_keymap;
  struct xkb_compose_state *xkb_compose_state;
  char *name;
  struct zwp_text_input_v3 *text_input;
};
\section wayland-doc Documentation resources
The Wayland book Extensive introduction to Wayland programming written by the author of the sway compositor, unfortunately unachieved.
Wayland Explorer Documentation of all Wayland protocols, both stable and unstable. A language-independent syntax is used which makes function names usable from C or C++ not always obvious. Some useful functions seem undocumented here for an unclear reason.
Wayland Protocol Specification Documentation for all functions of the Wayland core protocol.
Wayland clipboard and drag & drop Detailed explanation of how clipboard and drag-and-drop work under Wayland.
Wayland and input methods Blog article introducing to the issue of text input methods under Wayland.
Input Method Hub Entry page for input method support giving newcomers a first understanding of what input methods are and how they are implemented in Wayland.
*/