smithay/wayland/compositor/
mod.rs

1//! Utilities for handling surfaces, subsurfaces and regions
2//!
3//! This module provides automatic handling of surfaces, subsurfaces
4//! and region Wayland objects, by registering an implementation for
5//! for the [`wl_compositor`](wayland_server::protocol::wl_compositor)
6//! and [`wl_subcompositor`](wayland_server::protocol::wl_subcompositor) globals.
7//!
8//! ## Why use this implementation
9//!
10//! This implementation does a simple job: it stores in a coherent way the state of
11//! surface trees with subsurfaces, to provide you direct access to the tree
12//! structure and all surface attributes, and handles the application of double-buffered
13//! state.
14//!
15//! As such, you can, given a root surface with a role requiring it to be displayed,
16//! you can iterate over the whole tree of subsurfaces to recover all the metadata you
17//! need to display the subsurface tree.
18//!
19//! This implementation will not do anything more than present you the metadata specified by the
20//! client in a coherent and practical way. All the logic regarding drawing itself, and
21//! the positioning of windows (surface trees) one relative to another is out of its scope.
22//!
23//! ## How to use it
24//!
25//! ### Initialization
26//!
27//! To initialize this implementation create the [`CompositorState`], store it inside your `State` struct
28//! and implement the [`CompositorHandler`], as shown in this example:
29//!
30//! ```
31//! # extern crate wayland_server;
32//! # #[macro_use] extern crate smithay;
33//! use smithay::delegate_compositor;
34//! use smithay::wayland::compositor::{CompositorState, CompositorClientState, CompositorHandler};
35//!
36//! # struct State { compositor_state: CompositorState }
37//! # struct ClientState { compositor_state: CompositorClientState }
38//! # impl wayland_server::backend::ClientData for ClientState {}
39//! # let mut display = wayland_server::Display::<State>::new().unwrap();
40//! // Create the compositor state
41//! let compositor_state = CompositorState::new::<State>(
42//!     &display.handle(),
43//! );
44//!
45//! // insert the CompositorState into your state
46//! // ..
47//!
48//! // implement the necessary traits
49//! impl CompositorHandler for State {
50//!    fn compositor_state(&mut self) -> &mut CompositorState {
51//!        &mut self.compositor_state
52//!    }
53//!
54//!    fn client_compositor_state<'a>(&self, client: &'a wayland_server::Client) -> &'a CompositorClientState {
55//!        &client.get_data::<ClientState>().unwrap().compositor_state    
56//!    }
57//!
58//!    fn commit(&mut self, surface: &wayland_server::protocol::wl_surface::WlSurface) {
59//!        // called on every buffer commit.
60//!        // .. your implementation ..
61//!    }
62//! }
63//! delegate_compositor!(State);
64//!
65//! // You're now ready to go!
66//! ```
67//!
68//! ### Use the surface states
69//!
70//! The main access to surface states is done through the [`with_states`] function, which
71//! gives you access to the [`SurfaceData`] instance associated with this surface. It acts
72//! as a general purpose container for associating state to a surface, double-buffered or
73//! not. See its documentation for more details.
74//!
75//! ### State application and hooks
76//!
77//! On commit of a surface several steps are taken to update the state of the surface. Actions
78//! are taken by smithay in the following order:
79//!
80//! 1. Pre Commit hooks registered to this surface are invoked. Such hooks can be registered using
81//!    the [`add_pre_commit_hook`] function. They are typically used by protocol extensions that
82//!    add state to a surface and need to check on commit that client did not request an
83//!    illegal state before it is applied on commit.
84//! 2. The pending state is either applied and made current, or cached for later application
85//!    is the surface is a synchronize subsurface. If the current state is applied, state
86//!    of the synchronized children subsurface are applied as well at this point.
87//! 3. Post Commit hooks registered to this surface are invoked. Such hooks can be registered using
88//!    the [`add_post_commit_hook`] function. They are typically used by abstractions that further process
89//!    the state.
90//! 4. Your implementation of [`CompositorHandler::commit`] is invoked, so that you can access
91//!    the new current state of the surface. The state of sync children subsurfaces of your
92//!    surface may have changed as well, so this is the place to check it, using functions
93//!    like [`with_surface_tree_upward`] or [`with_surface_tree_downward`]. On the other hand,
94//!    if the surface is a sync subsurface, its current state will note have changed as
95//!    the result of that commit. You can check if it is using [`is_sync_subsurface`].
96//! 5. If the surface is destroyed, destruction hooks are invoked. Such hooks can be registered
97//!    using the [`add_destruction_hook`] function. They are typically used to cleanup associated
98//!    state.
99//!
100//! ### Surface roles
101//!
102//! The wayland protocol specifies that a surface needs to be assigned a role before it can
103//! be displayed. Furthermore, a surface can only have a single role during its whole lifetime.
104//! Smithay represents this role as a `&'static str` identifier, that can only be set once
105//! on a surface. See [`give_role`] and [`get_role`] for details. This module manages the
106//! subsurface role, which is identified by the string `"subsurface"`.
107
108mod cache;
109mod handlers;
110mod transaction;
111mod tree;
112
113use std::cell::RefCell;
114use std::sync::atomic::{AtomicU32, Ordering};
115use std::sync::Arc;
116use std::{any::Any, sync::Mutex};
117
118pub use self::cache::{Cacheable, CachedState, MultiCache};
119pub use self::handlers::{RegionUserData, SubsurfaceCachedState, SubsurfaceUserData, SurfaceUserData};
120use self::transaction::TransactionQueue;
121pub use self::transaction::{Barrier, Blocker, BlockerState};
122pub use self::tree::{AlreadyHasRole, TraversalAction};
123use self::tree::{PrivateSurfaceData, SuggestedSurfaceState};
124pub use crate::utils::hook::HookId;
125use crate::utils::Transform;
126use crate::utils::{user_data::UserDataMap, Buffer, Logical, Point, Rectangle};
127use wayland_server::backend::GlobalId;
128use wayland_server::protocol::wl_compositor::WlCompositor;
129use wayland_server::protocol::wl_subcompositor::WlSubcompositor;
130use wayland_server::protocol::{wl_buffer, wl_callback, wl_output, wl_region, wl_surface::WlSurface};
131use wayland_server::{Client, DisplayHandle, GlobalDispatch, Resource};
132
133/// The role of a subsurface surface.
134pub const SUBSURFACE_ROLE: &str = "subsurface";
135
136/// Description of a part of a surface that
137/// should be considered damaged and needs to be redrawn
138#[derive(Debug, PartialEq, Eq)]
139pub enum Damage {
140    /// A rectangle containing the damaged zone, in surface coordinates
141    Surface(Rectangle<i32, Logical>),
142    /// A rectangle containing the damaged zone, in buffer coordinates
143    ///
144    /// Note: Buffer scaling must be taken into consideration
145    Buffer(Rectangle<i32, Buffer>),
146}
147
148/// The state container associated with a surface
149///
150/// This general-purpose container provides 2 main storages:
151///
152/// - the `data_map` storage has typemap semantics and allows you
153///   to associate and access non-buffered data to the surface
154/// - the `cached_state` storages allows you to associate state to
155///   the surface that follows the double-buffering semantics associated
156///   with the `commit` procedure of surfaces, also with typemap-like
157///   semantics
158///
159/// See the respective documentation of each container for its usage.
160///
161/// By default, all surfaces have a [`SurfaceAttributes`] cached state,
162/// and subsurface also have a [`SubsurfaceCachedState`] state as well.
163#[derive(Debug)]
164pub struct SurfaceData {
165    /// The current role of the surface.
166    ///
167    /// If `None` if the surface has not yet been assigned a role
168    pub role: Option<&'static str>,
169    /// The non-buffered typemap storage of this surface
170    pub data_map: UserDataMap,
171    /// The double-buffered typemap storage of this surface
172    pub cached_state: MultiCache,
173}
174
175/// New buffer assignation for a surface
176#[derive(Debug)]
177pub enum BufferAssignment {
178    /// The surface no longer has a buffer attached to it
179    Removed,
180    /// A new buffer has been attached
181    NewBuffer(wl_buffer::WlBuffer),
182}
183
184/// General state associated with a surface
185///
186/// The fields `buffer`, `damage` and `frame_callbacks` should be
187/// reset (by clearing their contents) once you have adequately
188/// processed them, as their contents are aggregated from commit to commit.
189#[derive(Debug)]
190pub struct SurfaceAttributes {
191    /// Buffer defining the contents of the surface
192    ///
193    /// You are free to set this field to `None` to avoid processing it several
194    /// times. It'll be set to `Some(...)` if the user attaches a buffer (or `NULL`) to
195    /// the surface, and be left to `None` if the user does not attach anything.
196    pub buffer: Option<BufferAssignment>,
197
198    /// Location of the new buffer relative to the previous one
199    ///
200    /// The x and y arguments specify the location of the new pending buffer's upper left corner,
201    /// relative to the current buffer's upper left corner, in surface-local coordinates.
202    ///
203    /// In other words, the x and y, combined with the new surface size define in which directions
204    /// the surface's size changes.
205    ///
206    /// You are free to set this field to `None` to avoid processing it several times.
207    pub buffer_delta: Option<Point<i32, Logical>>,
208
209    /// Scale of the contents of the buffer, for higher-resolution contents.
210    ///
211    /// If it matches the one of the output displaying this surface, no change
212    /// is necessary.
213    pub buffer_scale: i32,
214    /// Transform under which interpret the contents of the buffer
215    ///
216    /// If it matches the one of the output displaying this surface, no change
217    /// is necessary.
218    pub buffer_transform: wl_output::Transform,
219    /// Region of the surface that is guaranteed to be opaque
220    ///
221    /// By default the whole surface is potentially transparent
222    pub opaque_region: Option<RegionAttributes>,
223    /// Region of the surface that is sensitive to user input
224    ///
225    /// By default the whole surface should be sensitive
226    pub input_region: Option<RegionAttributes>,
227    /// Damage rectangle
228    ///
229    /// Hint provided by the client to suggest that only this part
230    /// of the surface was changed and needs to be redrawn
231    pub damage: Vec<Damage>,
232    /// The frame callbacks associated with this surface for the commit
233    ///
234    /// The server must send the notifications so that a client
235    /// will not send excessive updates, while still allowing
236    /// the highest possible update rate for clients that wait for the reply
237    /// before drawing again. The server should give some time for the client
238    /// to draw and commit after sending the frame callback events to let it
239    /// hit the next output refresh.
240    ///
241    /// A server should avoid signaling the frame callbacks if the
242    /// surface is not visible in any way, e.g. the surface is off-screen,
243    /// or completely obscured by other opaque surfaces.
244    ///
245    /// An example possibility would be to trigger it once the frame
246    /// associated with this commit has been displayed on the screen.
247    pub frame_callbacks: Vec<wl_callback::WlCallback>,
248
249    pub(crate) client_scale: u32,
250}
251
252impl Default for SurfaceAttributes {
253    fn default() -> SurfaceAttributes {
254        SurfaceAttributes {
255            buffer: None,
256            buffer_delta: None,
257            buffer_scale: 1,
258            buffer_transform: wl_output::Transform::Normal,
259            opaque_region: None,
260            input_region: None,
261            damage: Vec::new(),
262            frame_callbacks: Vec::new(),
263            client_scale: 1,
264        }
265    }
266}
267
268/// Kind of a rectangle part of a region
269#[derive(Copy, Clone, Debug)]
270pub enum RectangleKind {
271    /// This rectangle should be added to the region
272    Add,
273    /// The intersection of this rectangle with the region should
274    /// be removed from the region
275    Subtract,
276}
277
278/// Description of the contents of a region
279///
280/// A region is defined as an union and difference of rectangle.
281///
282/// This struct contains an ordered `Vec` containing the rectangles defining
283/// a region. They should be added or subtracted in this order to compute the
284/// actual contents of the region.
285#[derive(Clone, Debug, Default)]
286pub struct RegionAttributes {
287    /// List of rectangle part of this region
288    pub rects: Vec<(RectangleKind, Rectangle<i32, Logical>)>,
289}
290
291impl RegionAttributes {
292    /// Checks whether given point is inside the region.
293    pub fn contains<P: Into<Point<i32, Logical>>>(&self, point: P) -> bool {
294        let point: Point<i32, Logical> = point.into();
295        let mut contains = false;
296        for (kind, rect) in &self.rects {
297            if rect.contains(point) {
298                match kind {
299                    RectangleKind::Add => contains = true,
300                    RectangleKind::Subtract => contains = false,
301                }
302            }
303        }
304        contains
305    }
306}
307
308/// Access the data of a surface tree from bottom to top
309///
310/// You provide three closures, a "filter", a "processor" and a "post filter".
311///
312/// The first closure is initially called on a surface to determine if its children
313/// should be processed as well. It returns a `TraversalAction<T>` reflecting that.
314///
315/// The second closure is supposed to do the actual processing. The processing closure for
316/// a surface may be called after the processing closure of some of its children, depending
317/// on the stack ordering the client requested. Here the surfaces are processed in the same
318/// order as they are supposed to be drawn: from the farthest of the screen to the nearest.
319///
320/// The third closure is called once all the subtree of a node has been processed, and gives
321/// an opportunity for early-stopping. If it returns `true` the processing will continue,
322/// while if it returns `false` it'll stop.
323///
324/// The arguments provided to the closures are, in this order:
325///
326/// - The surface object itself
327/// - a mutable reference to its surface attribute data
328/// - a mutable reference to its role data,
329/// - a custom value that is passed in a fold-like manner, but only from the output of a parent
330///   to its children. See [`TraversalAction`] for details.
331///
332/// If the surface not managed by the `CompositorGlobal` that provided this token, this
333/// will panic (having more than one compositor is not supported).
334pub fn with_surface_tree_upward<F1, F2, F3, T>(
335    surface: &WlSurface,
336    initial: T,
337    filter: F1,
338    processor: F2,
339    post_filter: F3,
340) where
341    F1: FnMut(&WlSurface, &SurfaceData, &T) -> TraversalAction<T>,
342    F2: FnMut(&WlSurface, &SurfaceData, &T),
343    F3: FnMut(&WlSurface, &SurfaceData, &T) -> bool,
344{
345    PrivateSurfaceData::map_tree(surface, &initial, filter, processor, post_filter, false);
346}
347
348/// Access the data of a surface tree from top to bottom
349///
350/// Behavior is the same as [`with_surface_tree_upward`], but the processing is done in the reverse order,
351/// from the nearest of the screen to the deepest.
352///
353/// This would typically be used to find out which surface of a subsurface tree has been clicked for example.
354pub fn with_surface_tree_downward<F1, F2, F3, T>(
355    surface: &WlSurface,
356    initial: T,
357    filter: F1,
358    processor: F2,
359    post_filter: F3,
360) where
361    F1: FnMut(&WlSurface, &SurfaceData, &T) -> TraversalAction<T>,
362    F2: FnMut(&WlSurface, &SurfaceData, &T),
363    F3: FnMut(&WlSurface, &SurfaceData, &T) -> bool,
364{
365    PrivateSurfaceData::map_tree(surface, &initial, filter, processor, post_filter, true);
366}
367
368/// Retrieve the parent of this surface
369///
370/// Returns `None` is this surface is a root surface
371pub fn get_parent(surface: &WlSurface) -> Option<WlSurface> {
372    PrivateSurfaceData::get_parent(surface)
373}
374
375/// Retrieve the children of this surface
376pub fn get_children(surface: &WlSurface) -> Vec<WlSurface> {
377    PrivateSurfaceData::get_children(surface)
378}
379
380/// Check if this subsurface is a synchronized subsurface
381pub fn is_sync_subsurface(surface: &WlSurface) -> bool {
382    self::handlers::is_effectively_sync(surface)
383}
384
385/// Get the current role of this surface
386pub fn get_role(surface: &WlSurface) -> Option<&'static str> {
387    PrivateSurfaceData::get_role(surface)
388}
389
390/// Register that this surface has given role
391///
392/// Fails if the surface already has a role.
393pub fn give_role(surface: &WlSurface, role: &'static str) -> Result<(), AlreadyHasRole> {
394    PrivateSurfaceData::set_role(surface, role)
395}
396
397/// Access the states associated to this surface
398pub fn with_states<F, T>(surface: &WlSurface, f: F) -> T
399where
400    F: FnOnce(&SurfaceData) -> T,
401{
402    PrivateSurfaceData::with_states(surface, f)
403}
404
405/// Send the `scale` and `transform` preferences for the given surface when it supports them.
406///
407/// The new state is only send when it differs from the already cached one on the calling thread.
408pub fn send_surface_state(surface: &WlSurface, data: &SurfaceData, scale: i32, transform: Transform) {
409    if surface.version() < 6 {
410        return;
411    }
412
413    // NOTE we insert default for checks below to work properly.
414    let mut storage = data
415        .data_map
416        .get_or_insert(|| RefCell::new(SuggestedSurfaceState::default()))
417        .borrow_mut();
418
419    if storage.scale != scale {
420        surface.preferred_buffer_scale(scale);
421        storage.scale = scale;
422    }
423
424    let transform = transform.into();
425    if storage.transform != transform {
426        surface.preferred_buffer_transform(transform);
427        storage.transform = transform;
428    }
429}
430
431/// Retrieve the metadata associated with a `wl_region`
432///
433/// If the region is not managed by the `CompositorGlobal` that provided this token, this
434/// will panic (having more than one compositor is not supported).
435pub fn get_region_attributes(region: &wl_region::WlRegion) -> RegionAttributes {
436    match region.data::<RegionUserData>() {
437        Some(data) => data.inner.lock().unwrap().clone(),
438        None => panic!("Accessing the data of foreign regions is not supported."),
439    }
440}
441
442/// Register a pre-commit hook to be invoked on surface commit
443///
444/// It'll be invoked on surface commit, *before* the new state is merged into the current state.
445///
446/// Protocol implementations should use this for error checking, but they should **not** apply
447/// state changes here, since the commit may be further arbitrarily delayed by blockers. Use a
448/// post-commit hook to apply state changes (i.e. copy last acked state to current).
449///
450/// Compositors should use this for adding blockers if needed, e.g. the DMA-BUF readiness blocker.
451pub fn add_pre_commit_hook<D, F>(surface: &WlSurface, hook: F) -> HookId
452where
453    F: Fn(&mut D, &DisplayHandle, &WlSurface) + Send + Sync + 'static,
454    D: 'static,
455{
456    let (user_state_type_id, user_state_type) = surface.data::<SurfaceUserData>().unwrap().user_state_type;
457    assert_eq!(
458        std::any::TypeId::of::<D>(),
459        user_state_type_id,
460        "D has to equal D used in CompositorState::new<D>(), {} != {}",
461        std::any::type_name::<D>(),
462        user_state_type,
463    );
464
465    let hook = move |state: &mut dyn Any, dh: &DisplayHandle, surface: &WlSurface| {
466        let state = state.downcast_mut::<D>().unwrap();
467        hook(state, dh, surface);
468    };
469    PrivateSurfaceData::add_pre_commit_hook(surface, hook)
470}
471
472/// Register a post-commit hook to be invoked on surface commit
473///
474/// It'll be invoked on surface commit, *after* the new state is merged into the current state,
475/// after all commit blockers complete.
476///
477/// Protocol implementations should apply state changes here, i.e. copy last acked state into
478/// current.
479pub fn add_post_commit_hook<D, F>(surface: &WlSurface, hook: F) -> HookId
480where
481    F: Fn(&mut D, &DisplayHandle, &WlSurface) + Send + Sync + 'static,
482    D: 'static,
483{
484    let (user_state_type_id, user_state_type) = surface.data::<SurfaceUserData>().unwrap().user_state_type;
485    assert_eq!(
486        std::any::TypeId::of::<D>(),
487        user_state_type_id,
488        "D has to equal D used in CompositorState::new<D>(), {} != {}",
489        std::any::type_name::<D>(),
490        user_state_type,
491    );
492
493    let hook = move |state: &mut dyn Any, dh: &DisplayHandle, surface: &WlSurface| {
494        let state = state.downcast_mut::<D>().unwrap();
495        hook(state, dh, surface);
496    };
497    PrivateSurfaceData::add_post_commit_hook(surface, hook)
498}
499
500/// Register a destruction hook to be invoked on surface destruction
501///
502/// It'll be invoked when the surface is destroyed (either explicitly by the client or on
503/// client disconnect).
504///
505/// D generic is the compositor state, same as used in `CompositorState::new<D>()`
506pub fn add_destruction_hook<D, F>(surface: &WlSurface, hook: F) -> HookId
507where
508    F: Fn(&mut D, &WlSurface) + Send + Sync + 'static,
509    D: 'static,
510{
511    let (user_state_type_id, user_state_type) = surface.data::<SurfaceUserData>().unwrap().user_state_type;
512    assert_eq!(
513        std::any::TypeId::of::<D>(),
514        user_state_type_id,
515        "D has to equal D used in CompositorState::new<D>(), {} != {}",
516        std::any::type_name::<D>(),
517        user_state_type,
518    );
519
520    let hook = move |state: &mut dyn Any, surface: &WlSurface| {
521        let state = state.downcast_mut::<D>().unwrap();
522        hook(state, surface);
523    };
524    PrivateSurfaceData::add_destruction_hook(surface, hook)
525}
526
527/// Unregister a pre-commit hook
528pub fn remove_pre_commit_hook(surface: &WlSurface, hook_id: HookId) {
529    PrivateSurfaceData::remove_pre_commit_hook(surface, hook_id)
530}
531
532/// Unregister a post-commit hook
533pub fn remove_post_commit_hook(surface: &WlSurface, hook_id: HookId) {
534    PrivateSurfaceData::remove_post_commit_hook(surface, hook_id)
535}
536
537/// Unregister a destruction hook
538pub fn remove_destruction_hook(surface: &WlSurface, hook_id: HookId) {
539    PrivateSurfaceData::remove_destruction_hook(surface, hook_id)
540}
541
542/// Adds a blocker for the currently queued up state changes of the given surface.
543///
544/// Blockers will delay the pending state to be applied on the next commit until
545/// all of them return the state `Released`. Any blocker returning `Cancelled` will
546/// discard all changes.
547///
548/// The module will only evaluate blocker states on commit. If a blocker
549/// becomes ready later, a call to [`CompositorClientState::blocker_cleared`] is necessary
550/// to trigger a re-evaluation.
551pub fn add_blocker(surface: &WlSurface, blocker: impl Blocker + Send + 'static) {
552    PrivateSurfaceData::add_blocker(surface, blocker)
553}
554
555/// Handler trait for compositor
556pub trait CompositorHandler {
557    /// [CompositorState] getter
558    fn compositor_state(&mut self) -> &mut CompositorState;
559    /// [CompositorClientState] getter
560    ///
561    /// The compositor implementation needs some state to be client specific.
562    /// Downstream is expected to store this inside its `ClientData` implementation(s)
563    /// to ensure automatic cleanup of the state, when the client disconnects.
564    fn client_compositor_state<'a>(&self, client: &'a Client) -> &'a CompositorClientState;
565
566    /// New surface handler.
567    ///
568    /// This handler can be used to setup hooks (see [`add_pre_commit_hook`]/[`add_post_commit_hook`]/[`add_destruction_hook`]),
569    /// but not much else. The surface has no role or attached data at this point and cannot be rendered.
570    fn new_surface(&mut self, surface: &WlSurface) {
571        let _ = surface;
572    }
573
574    /// New subsurface handler.
575    ///
576    /// This handler can be used to run extra logic when subsurface is getting created. This
577    /// is an addition to [`new_surface`], which will be run for the subsurface surface anyway.
578    ///
579    /// When your compositor knows beforehand where it'll position subsurfaces it can send
580    /// [`send_surface_state`] to them.
581    ///
582    /// [`new_surface`]: Self::new_surface
583    fn new_subsurface(&mut self, surface: &WlSurface, parent: &WlSurface) {
584        let _ = surface;
585        let _ = parent;
586    }
587
588    /// Surface commit handler
589    ///
590    /// This is called when any changed state from a commit actually becomes visible.
591    /// That might be some time after the actual commit has taken place, if the
592    /// state changes are delayed by an added blocker (see [`add_blocker`]).
593    ///
594    /// If you need to handle a commit as soon as it occurs, you might want to consider
595    /// using a pre-commit hook (see [`add_pre_commit_hook`]).
596    fn commit(&mut self, surface: &WlSurface);
597
598    /// The surface was destroyed.
599    ///
600    /// This allows the compositor to clean up any uses of the surface.
601    fn destroyed(&mut self, _surface: &WlSurface) {}
602}
603
604/// State of a compositor
605#[derive(Debug)]
606pub struct CompositorState {
607    compositor: GlobalId,
608    subcompositor: GlobalId,
609    surfaces: Vec<WlSurface>,
610}
611
612/// Per-client state of a compositor
613#[derive(Debug)]
614pub struct CompositorClientState {
615    queue: Mutex<Option<TransactionQueue>>,
616    scale_override: Arc<AtomicU32>,
617}
618
619impl Default for CompositorClientState {
620    fn default() -> Self {
621        CompositorClientState {
622            queue: Mutex::new(None),
623            scale_override: Arc::new(AtomicU32::new(1)),
624        }
625    }
626}
627
628impl CompositorClientState {
629    /// To be called, when a previously added blocker (via [`add_blocker`])
630    /// got `Released` or `Cancelled` from being `Pending` previously for any
631    /// surface belonging to this client.
632    pub fn blocker_cleared<D: CompositorHandler + 'static>(&self, state: &mut D, dh: &DisplayHandle) {
633        let transactions = if let Some(queue) = self.queue.lock().unwrap().as_mut() {
634            queue.take_ready()
635        } else {
636            Vec::new()
637        };
638
639        for transaction in transactions {
640            transaction.apply(dh, state)
641        }
642    }
643
644    /// Set an additionally mapping between smithay's `Logical` coordinate space
645    /// and this clients logical coordinate space.
646    ///
647    /// This is used in the same way as if the client was setting the
648    /// surface.buffer_scale on every surface i.e a value of 2.0 will make
649    /// the windows appear smaller on a regular DPI monitor.
650    ///
651    /// Only the minimal set of protocols used by xwayland are guaranteed to be supported.
652    ///
653    /// Buffer sizes are unaffected.
654    pub fn set_client_scale(&self, new_scale: u32) {
655        self.scale_override.store(new_scale, Ordering::Release);
656    }
657
658    /// Get the scale factor of the additional mapping between smithay's `Logical`
659    /// coordinate space and this clients logical coordinate space.
660    ///
661    /// This is mainly intended to support out-of-tree protocol implementations.
662    pub fn client_scale(&self) -> u32 {
663        self.scale_override.load(Ordering::Acquire)
664    }
665
666    pub(crate) fn clone_client_scale(&self) -> Arc<AtomicU32> {
667        self.scale_override.clone()
668    }
669}
670
671impl CompositorState {
672    /// Create new [`wl_compositor`] version 5 and [`wl_subcompositor`] globals.
673    ///
674    /// It returns the two global handles, in case you wish to remove these globals from
675    /// the event loop in the future.
676    ///
677    /// [`wl_compositor`]: wayland_server::protocol::wl_compositor
678    /// [`wl_subcompositor`]: wayland_server::protocol::wl_subcompositor
679    pub fn new<D>(display: &DisplayHandle) -> Self
680    where
681        D: GlobalDispatch<WlCompositor, ()> + GlobalDispatch<WlSubcompositor, ()> + 'static,
682    {
683        Self::new_with_version::<D>(display, 5)
684    }
685
686    /// The same as [`new`], but binds at least version 6 of [`wl_compositor`].
687    ///
688    /// This means that for clients to scale and apply transformation with
689    /// non-default values [`send_surface_state`] must be used.
690    ///
691    /// [`new`]: Self::new
692    /// [`wl_compositor`]: wayland_server::protocol::wl_compositor
693    pub fn new_v6<D>(display: &DisplayHandle) -> Self
694    where
695        D: GlobalDispatch<WlCompositor, ()> + GlobalDispatch<WlSubcompositor, ()> + 'static,
696    {
697        Self::new_with_version::<D>(display, 6)
698    }
699
700    fn new_with_version<D>(display: &DisplayHandle, version: u32) -> Self
701    where
702        D: GlobalDispatch<WlCompositor, ()> + GlobalDispatch<WlSubcompositor, ()> + 'static,
703    {
704        let compositor = display.create_global::<D, WlCompositor, ()>(version, ());
705        let subcompositor = display.create_global::<D, WlSubcompositor, ()>(1, ());
706
707        CompositorState {
708            compositor,
709            subcompositor,
710            surfaces: Vec::new(),
711        }
712    }
713
714    /// Get id of compositor global
715    pub fn compositor_global(&self) -> GlobalId {
716        self.compositor.clone()
717    }
718
719    /// Get id of subcompositor global
720    pub fn subcompositor_global(&self) -> GlobalId {
721        self.subcompositor.clone()
722    }
723}
724
725#[allow(missing_docs)] // TODO
726#[macro_export]
727macro_rules! delegate_compositor {
728    ($(@<$( $lt:tt $( : $clt:tt $(+ $dlt:tt )* )? ),+>)? $ty: ty) => {
729        $crate::reexports::wayland_server::delegate_global_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
730            $crate::reexports::wayland_server::protocol::wl_compositor::WlCompositor: ()
731        ] => $crate::wayland::compositor::CompositorState);
732        $crate::reexports::wayland_server::delegate_global_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
733            $crate::reexports::wayland_server::protocol::wl_subcompositor::WlSubcompositor: ()
734        ] => $crate::wayland::compositor::CompositorState);
735
736        $crate::reexports::wayland_server::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
737            $crate::reexports::wayland_server::protocol::wl_compositor::WlCompositor: ()
738        ] => $crate::wayland::compositor::CompositorState);
739        $crate::reexports::wayland_server::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
740            $crate::reexports::wayland_server::protocol::wl_surface::WlSurface: $crate::wayland::compositor::SurfaceUserData
741        ] => $crate::wayland::compositor::CompositorState);
742        $crate::reexports::wayland_server::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
743            $crate::reexports::wayland_server::protocol::wl_region::WlRegion: $crate::wayland::compositor::RegionUserData
744        ] => $crate::wayland::compositor::CompositorState);
745        $crate::reexports::wayland_server::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
746            $crate::reexports::wayland_server::protocol::wl_callback::WlCallback: ()
747        ] => $crate::wayland::compositor::CompositorState);
748            // WlSubcompositor
749        $crate::reexports::wayland_server::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
750            $crate::reexports::wayland_server::protocol::wl_subcompositor::WlSubcompositor: ()
751        ] => $crate::wayland::compositor::CompositorState);
752        $crate::reexports::wayland_server::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [
753            $crate::reexports::wayland_server::protocol::wl_subsurface::WlSubsurface: $crate::wayland::compositor::SubsurfaceUserData
754        ] => $crate::wayland::compositor::CompositorState);
755    };
756}
757
758#[cfg(test)]
759mod tests {
760    use super::*;
761
762    #[test]
763    fn region_attributes_empty() {
764        let region = RegionAttributes { rects: vec![] };
765        assert!(!region.contains((0, 0)));
766    }
767
768    #[test]
769    fn region_attributes_add() {
770        let region = RegionAttributes {
771            rects: vec![(RectangleKind::Add, Rectangle::from_size((10, 10).into()))],
772        };
773
774        assert!(region.contains((0, 0)));
775    }
776
777    #[test]
778    fn region_attributes_add_subtract() {
779        let region = RegionAttributes {
780            rects: vec![
781                (RectangleKind::Add, Rectangle::from_size((10, 10).into())),
782                (RectangleKind::Subtract, Rectangle::from_size((5, 5).into())),
783            ],
784        };
785
786        assert!(!region.contains((0, 0)));
787        assert!(region.contains((5, 5)));
788    }
789
790    #[test]
791    fn region_attributes_add_subtract_add() {
792        let region = RegionAttributes {
793            rects: vec![
794                (RectangleKind::Add, Rectangle::from_size((10, 10).into())),
795                (RectangleKind::Subtract, Rectangle::from_size((5, 5).into())),
796                (RectangleKind::Add, Rectangle::new((2, 2).into(), (2, 2).into())),
797            ],
798        };
799
800        assert!(!region.contains((0, 0)));
801        assert!(region.contains((5, 5)));
802        assert!(region.contains((2, 2)));
803    }
804}