halley-wl 0.3.2

Wayland backend and rendering implementation for the Halley Wayland compositor.
use super::*;
use crate::compositor::{focus, fullscreen, interaction, monitor, spawn, workspace};
use smithay::backend::allocator::dmabuf::Dmabuf;
use smithay::backend::input::TabletToolDescriptor;
use smithay::input::pointer::PointerHandle;
use smithay::reexports::wayland_protocols::xdg::decoration::zv1::server::zxdg_toplevel_decoration_v1::Mode as XdgDecorationMode;
use smithay::reexports::wayland_protocols::xdg::shell::server::xdg_toplevel;
use smithay::reexports::wayland_server::{
    Client, Resource, protocol::wl_surface::WlSurface,
};
use smithay::wayland::compositor::{add_blocker, with_states};
use smithay::wayland::dmabuf::{DmabufFeedback, DmabufGlobal, DmabufHandler, ImportNotifier};
use smithay::wayland::drm_syncobj::{DrmSyncPoint, DrmSyncobjCachedState, DrmSyncobjHandler};
use smithay::wayland::fractional_scale::FractionalScaleHandler;
use smithay::wayland::output::OutputHandler;
use smithay::wayland::selection::primary_selection::{
    PrimarySelectionHandler, PrimarySelectionState,
};
use smithay::wayland::selection::wlr_data_control::DataControlState;
use smithay::wayland::shell::wlr_layer::{
    LayerSurface, LayerSurfaceConfigure, WlrLayerShellHandler, WlrLayerShellState,
};
use smithay::wayland::tablet_manager::TabletSeatHandler;

pub(super) fn initial_toplevel_size(
    st: &mut Halley,
    toplevel: &ToplevelSurface,
    intent: &spawn::rules::InitialWindowIntent,
) -> spawn::reveal::InitialToplevelSize {
    let mut ctx = st.spawn_ctx();
    spawn::reveal::initial_toplevel_size(&mut ctx, toplevel, intent)
}

pub(super) fn constrain_layer_popup(
    st: &mut Halley,
    popup: &PopupSurface,
    positioner: PositionerState,
) {
    let mut ctx = st.layer_shell_ctx();
    monitor::layer_shell::constrain_layer_popup(&mut ctx, popup, positioner);
}

impl SeatHandler for Halley {
    type KeyboardFocus = WlSurface;
    type PointerFocus = WlSurface;
    type TouchFocus = WlSurface;

    fn seat_state(&mut self) -> &mut SeatState<Self> {
        &mut self.platform.seat_state
    }

    fn focus_changed(&mut self, seat: &Seat<Self>, focused: Option<&WlSurface>) {
        let now = Instant::now();
        fullscreen::system::on_seat_focus_changed(&mut self.fullscreen_ctx(), focused, now);
        focus::system::on_seat_focus_changed(&self.focus_ctx(), seat, focused);
    }

    fn cursor_image(&mut self, _seat: &Seat<Self>, image: CursorImageStatus) {
        self.platform.cursor_manager.set_cursor_image(image);
        crate::compositor::platform::refresh_cursor_surface_outputs(self);
        self.runtime.tty_redraw_all = true;
        self.request_maintenance();
    }
}

delegate_seat!(Halley);
delegate_cursor_shape!(Halley);
delegate_pointer_constraints!(Halley);
delegate_relative_pointer!(Halley);
delegate_drm_syncobj!(Halley);
delegate_idle_notify!(Halley);

impl TabletSeatHandler for Halley {
    fn tablet_tool_image(&mut self, _tool: &TabletToolDescriptor, image: CursorImageStatus) {
        self.platform.cursor_manager.set_cursor_image(image);
        crate::compositor::platform::refresh_cursor_surface_outputs(self);
        self.runtime.tty_redraw_all = true;
        self.request_maintenance();
    }
}

impl SelectionHandler for Halley {
    type SelectionUserData = ();
}

impl IdleNotifierHandler for Halley {
    fn idle_notifier_state(&mut self) -> &mut IdleNotifierState<Self> {
        &mut self.platform.idle_notifier_state
    }
}

impl DataDeviceHandler for Halley {
    fn data_device_state(&self) -> &DataDeviceState {
        &self.platform.data_device_state
    }
}

impl ClientDndGrabHandler for Halley {}

impl ServerDndGrabHandler for Halley {
    fn send(&mut self, _mime_type: String, _fd: std::os::unix::io::OwnedFd, _seat: Seat<Self>) {}
}

delegate_data_device!(Halley);

impl CompositorHandler for Halley {
    fn compositor_state(&mut self) -> &mut CompositorState {
        &mut self.platform.compositor_state
    }

    fn client_compositor_state<'a>(&self, client: &'a Client) -> &'a CompositorClientState {
        &client
            .get_data::<ClientState>()
            .expect("ClientState missing")
            .compositor_state
    }

    fn commit(&mut self, surface: &WlSurface) {
        on_commit_buffer_handler::<Self>(surface);
        self.install_drm_syncobj_blocker(surface);
        self.platform.popup_manager.commit(surface);
        if crate::render::handle_cursor_surface_commit(
            self.platform.cursor_manager.cursor_image(),
            surface,
        ) {
            crate::compositor::platform::refresh_cursor_surface_outputs(self);
            let output_name = self
                .input
                .interaction_state
                .last_pointer_screen_global
                .and_then(|(sx, sy)| self.monitor_for_screen(sx, sy))
                .unwrap_or_else(|| self.model.monitor_state.current_monitor.clone());
            self.request_tty_redraw_for_monitor(output_name.as_str());
            self.request_maintenance();
        }
        workspace::lifecycle::on_surface_commit(
            &mut self.surface_lifecycle_ctx(),
            surface,
            Instant::now(),
        );
    }
}

delegate_compositor!(Halley);
delegate_viewporter!(Halley);

impl FractionalScaleHandler for Halley {
    fn new_fractional_scale(&mut self, surface: WlSurface) {
        monitor::state::refresh_surface_preferred_scale(self, &surface);
    }
}

impl ShmHandler for Halley {
    fn shm_state(&self) -> &ShmState {
        &self.platform.shm_state
    }
}

impl BufferHandler for Halley {
    fn buffer_destroyed(
        &mut self,
        _buffer: &smithay::reexports::wayland_server::protocol::wl_buffer::WlBuffer,
    ) {
    }
}

delegate_shm!(Halley);

impl DmabufHandler for Halley {
    fn dmabuf_state(&mut self) -> &mut smithay::wayland::dmabuf::DmabufState {
        &mut self.platform.dmabuf_state
    }

    fn dmabuf_imported(&mut self, global: &DmabufGlobal, dmabuf: Dmabuf, notifier: ImportNotifier) {
        if self.platform.dmabuf_global != Some(*global) {
            notifier.failed();
            return;
        }

        let Some(importer) = self.platform.dmabuf_importer.as_ref() else {
            notifier.failed();
            return;
        };

        if importer.import_dmabuf(&dmabuf).is_ok() {
            let _ = notifier.successful::<Self>();
        } else {
            notifier.failed();
        }
    }

    fn new_surface_feedback(
        &mut self,
        surface: &WlSurface,
        global: &DmabufGlobal,
    ) -> Option<DmabufFeedback> {
        if self.platform.dmabuf_global != Some(*global) {
            return None;
        }

        let monitor = monitor::state::monitor_for_surface_or_current(self, surface);
        self.platform
            .dmabuf_output_feedbacks
            .get(monitor.as_str())
            .cloned()
    }
}

impl DrmSyncobjHandler for Halley {
    fn drm_syncobj_state(&mut self) -> Option<&mut DrmSyncobjState> {
        self.platform.drm_syncobj_state.as_mut()
    }
}

impl PointerConstraintsHandler for Halley {
    fn new_constraint(&mut self, surface: &WlSurface, pointer: &PointerHandle<Self>) {
        // Keep this path free of fresh absolute pointer motion. Recomputing focus here
        // regressed Tiny Glade's right-click camera lock by injecting a motion event at
        // lock creation time; use the existing pointer focus/location instead.
        let focus = pointer
            .current_focus()
            .map(|focused| (focused, pointer.current_location()));
        let Some(surface_origin) = interaction::pointer::retarget_pointer_focus_for_constraint(
            self,
            surface,
            pointer,
            focus.as_ref(),
        ) else {
            return;
        };
        interaction::pointer::activate_pointer_constraint_for_surface_at(
            self,
            surface,
            Some(surface_origin),
        );
    }

    fn cursor_position_hint(
        &mut self,
        surface: &WlSurface,
        pointer: &PointerHandle<Self>,
        location: smithay::utils::Point<f64, smithay::utils::Logical>,
    ) {
        interaction::pointer::cursor_position_hint(
            &mut self.pointer_ctx(),
            surface,
            pointer,
            location,
        );
    }
}

#[cfg(test)]
mod tests {}

impl PrimarySelectionHandler for Halley {
    fn primary_selection_state(&self) -> &PrimarySelectionState {
        &self.platform.primary_selection_state
    }
}

delegate_primary_selection!(Halley);

impl DataControlHandler for Halley {
    fn data_control_state(&self) -> &DataControlState {
        &self.platform.data_control_state
    }
}

delegate_data_control!(Halley);