aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorIvan Molodetskikh <yalterz@gmail.com>2023-08-15 16:19:05 +0400
committerIvan Molodetskikh <yalterz@gmail.com>2023-08-15 16:19:05 +0400
commit2465bba0858662d45ea1e5a321c6fbe11036d99f (patch)
tree98ffb9605df0fec87c2e966d845479bfc6970387 /src
parentb0d129951c122f36f5b85a405cbdc1d38ae0c421 (diff)
downloadniri-2465bba0858662d45ea1e5a321c6fbe11036d99f.tar.gz
niri-2465bba0858662d45ea1e5a321c6fbe11036d99f.tar.bz2
niri-2465bba0858662d45ea1e5a321c6fbe11036d99f.zip
Add client cursor support
Diffstat (limited to 'src')
-rw-r--r--src/handlers/mod.rs28
-rw-r--r--src/niri.rs60
2 files changed, 55 insertions, 33 deletions
diff --git a/src/handlers/mod.rs b/src/handlers/mod.rs
index fcde0585..cc80a841 100644
--- a/src/handlers/mod.rs
+++ b/src/handlers/mod.rs
@@ -2,13 +2,13 @@ mod compositor;
mod layer_shell;
mod xdg_shell;
-//
-// Wl Seat
+use smithay::input::pointer::CursorImageStatus;
use smithay::input::{Seat, SeatHandler, SeatState};
use smithay::reexports::wayland_server::protocol::wl_surface::WlSurface;
use smithay::reexports::wayland_server::Resource;
use smithay::wayland::data_device::{
- set_data_device_focus, ClientDndGrabHandler, DataDeviceHandler, ServerDndGrabHandler,
+ set_data_device_focus, ClientDndGrabHandler, DataDeviceHandler, DataDeviceState,
+ ServerDndGrabHandler,
};
use smithay::{delegate_data_device, delegate_output, delegate_seat};
@@ -22,11 +22,10 @@ impl SeatHandler for Niri {
&mut self.seat_state
}
- fn cursor_image(
- &mut self,
- _seat: &Seat<Self>,
- _image: smithay::input::pointer::CursorImageStatus,
- ) {
+ fn cursor_image(&mut self, _seat: &Seat<Self>, image: CursorImageStatus) {
+ self.cursor_image = image;
+ // FIXME: more granular
+ self.queue_redraw_all();
}
fn focus_changed(&mut self, seat: &Seat<Self>, focused: Option<&WlSurface>) {
@@ -35,27 +34,16 @@ impl SeatHandler for Niri {
set_data_device_focus(dh, seat, client);
}
}
-
delegate_seat!(Niri);
-//
-// Wl Data Device
-//
-
impl DataDeviceHandler for Niri {
type SelectionUserData = ();
- fn data_device_state(&self) -> &smithay::wayland::data_device::DataDeviceState {
+ fn data_device_state(&self) -> &DataDeviceState {
&self.data_device_state
}
}
-
impl ClientDndGrabHandler for Niri {}
impl ServerDndGrabHandler for Niri {}
-
delegate_data_device!(Niri);
-//
-// Wl Output & Xdg Output
-//
-
delegate_output!(Niri);
diff --git a/src/niri.rs b/src/niri.rs
index cfb2456f..1dab2b6e 100644
--- a/src/niri.rs
+++ b/src/niri.rs
@@ -1,10 +1,12 @@
use std::collections::HashMap;
use std::os::unix::io::AsRawFd;
-use std::sync::Arc;
+use std::sync::{Arc, Mutex};
use std::time::Duration;
use smithay::backend::renderer::element::solid::{SolidColorBuffer, SolidColorRenderElement};
-use smithay::backend::renderer::element::surface::WaylandSurfaceRenderElement;
+use smithay::backend::renderer::element::surface::{
+ render_elements_from_surface_tree, WaylandSurfaceRenderElement,
+};
use smithay::backend::renderer::element::{render_elements, AsRenderElements};
use smithay::backend::renderer::gles::GlesRenderer;
use smithay::backend::renderer::ImportAll;
@@ -12,6 +14,7 @@ use smithay::desktop::{
layer_map_for_output, LayerSurface, PopupManager, Space, Window, WindowSurfaceType,
};
use smithay::input::keyboard::XkbConfig;
+use smithay::input::pointer::{CursorImageAttributes, CursorImageStatus};
use smithay::input::{Seat, SeatState};
use smithay::output::Output;
use smithay::reexports::calloop::generic::Generic;
@@ -22,8 +25,8 @@ use smithay::reexports::wayland_server::backend::{
};
use smithay::reexports::wayland_server::protocol::wl_surface::WlSurface;
use smithay::reexports::wayland_server::{Display, DisplayHandle};
-use smithay::utils::{Logical, Point, Scale, SERIAL_COUNTER};
-use smithay::wayland::compositor::{CompositorClientState, CompositorState};
+use smithay::utils::{IsAlive, Logical, Point, Scale, SERIAL_COUNTER};
+use smithay::wayland::compositor::{with_states, CompositorClientState, CompositorState};
use smithay::wayland::data_device::DataDeviceState;
use smithay::wayland::output::OutputManagerState;
use smithay::wayland::shell::wlr_layer::{Layer, WlrLayerShellState};
@@ -68,6 +71,7 @@ pub struct Niri {
pub seat: Seat<Self>,
pub pointer_buffer: SolidColorBuffer,
+ pub cursor_image: CursorImageStatus,
}
pub struct OutputState {
@@ -171,6 +175,7 @@ impl Niri {
seat,
pointer_buffer,
+ cursor_image: CursorImageStatus::Default,
}
}
@@ -319,16 +324,45 @@ impl Niri {
state.queued_redraw = Some(idle);
}
- pub fn pointer_element(&mut self, output: &Output) -> OutputRenderElements<GlesRenderer> {
+ pub fn pointer_element(
+ &mut self,
+ renderer: &mut GlesRenderer,
+ output: &Output,
+ ) -> Vec<OutputRenderElements<GlesRenderer>> {
let output_pos = self.global_space.output_geometry(output).unwrap().loc;
let pointer_pos = self.seat.get_pointer().unwrap().current_location() - output_pos.to_f64();
- OutputRenderElements::Pointer(SolidColorRenderElement::from_buffer(
- &self.pointer_buffer,
- pointer_pos.to_physical_precise_round(1.),
- 1.,
- 1.,
- ))
+ if let CursorImageStatus::Surface(surface) = &mut self.cursor_image {
+ if !surface.alive() {
+ self.cursor_image = CursorImageStatus::Default;
+ }
+ }
+
+ match &self.cursor_image {
+ CursorImageStatus::Hidden => vec![],
+ CursorImageStatus::Default => vec![OutputRenderElements::DefaultPointer(
+ SolidColorRenderElement::from_buffer(
+ &self.pointer_buffer,
+ pointer_pos.to_physical_precise_round(1.),
+ 1.,
+ 1.,
+ ),
+ )],
+ CursorImageStatus::Surface(surface) => {
+ let hotspot = with_states(surface, |states| {
+ states
+ .data_map
+ .get::<Mutex<CursorImageAttributes>>()
+ .unwrap()
+ .lock()
+ .unwrap()
+ .hotspot
+ });
+ let pos = (pointer_pos - hotspot.to_f64()).to_physical_precise_round(1.);
+
+ render_elements_from_surface_tree(renderer, surface, pos, 1., 1.)
+ }
+ }
}
fn redraw(&mut self, backend: &mut dyn Backend, output: &Output) {
@@ -354,7 +388,7 @@ impl Niri {
.partition(|s| matches!(s.layer(), Layer::Background | Layer::Bottom));
// The pointer goes on the top.
- let mut elements = vec![self.pointer_element(output)];
+ let mut elements = self.pointer_element(renderer, output);
// Then the upper layer-shell elements.
elements.extend(
@@ -423,7 +457,7 @@ render_elements! {
pub OutputRenderElements<R> where R: ImportAll;
Monitor = MonitorRenderElement<R>,
Wayland = WaylandSurfaceRenderElement<R>,
- Pointer = SolidColorRenderElement,
+ DefaultPointer = SolidColorRenderElement,
}
#[derive(Default)]