pinnacle/src/input.rs

594 lines
24 KiB
Rust
Raw Normal View History

2023-08-01 18:06:35 +02:00
// SPDX-License-Identifier: GPL-3.0-or-later
2023-06-26 00:18:50 +02:00
use std::collections::HashMap;
2023-07-25 01:59:05 +02:00
use crate::{
api::msg::{CallbackId, Modifier, ModifierMask, OutgoingMsg},
focus::FocusTarget,
2023-08-29 05:53:24 +02:00
state::{Backend, WithState},
2023-07-25 01:59:05 +02:00
window::WindowElement,
};
use smithay::{
2023-08-08 17:55:06 +02:00
backend::{
input::{
AbsolutePositionEvent, Axis, AxisSource, ButtonState, Event, InputBackend, InputEvent,
KeyState, KeyboardKeyEvent, PointerAxisEvent, PointerButtonEvent, PointerMotionEvent,
},
session::Session,
},
2023-08-06 03:49:39 +02:00
desktop::{layer_map_for_output, space::SpaceElement},
input::{
keyboard::{keysyms, FilterResult},
2023-08-08 00:23:09 +02:00
pointer::{AxisFrame, ButtonEvent, MotionEvent},
},
reexports::wayland_protocols::xdg::shell::server::xdg_toplevel::ResizeEdge,
2023-06-10 03:29:17 +02:00
utils::{Logical, Point, SERIAL_COUNTER},
2023-08-06 03:49:39 +02:00
wayland::{compositor, seat::WaylandFocus, shell::wlr_layer},
};
2023-08-29 05:53:24 +02:00
use crate::state::State;
pub struct InputState {
2023-06-18 04:02:58 +02:00
/// A hashmap of modifier keys and keycodes to callback IDs
2023-06-25 00:39:40 +02:00
pub keybinds: HashMap<(ModifierMask, u32), CallbackId>,
2023-06-18 04:02:58 +02:00
/// A hashmap of modifier keys and mouse button codes to callback IDs
2023-06-25 00:39:40 +02:00
pub mousebinds: HashMap<(ModifierMask, u32), CallbackId>,
2023-08-16 00:26:17 +02:00
pub reload_keybind: (ModifierMask, u32),
pub kill_keybind: (ModifierMask, u32),
2023-06-18 04:02:58 +02:00
}
impl InputState {
2023-08-16 00:26:17 +02:00
pub fn new(reload_keybind: (ModifierMask, u32), kill_keybind: (ModifierMask, u32)) -> Self {
Self {
keybinds: HashMap::new(),
mousebinds: HashMap::new(),
reload_keybind,
kill_keybind,
}
2023-06-18 04:02:58 +02:00
}
}
#[derive(Debug)]
enum KeyAction {
CallCallback(CallbackId),
Quit,
SwitchVt(i32),
2023-08-16 00:26:17 +02:00
ReloadConfig,
}
2023-08-29 05:53:24 +02:00
impl State {
pub fn process_input_event<B: InputBackend>(&mut self, event: InputEvent<B>) {
match event {
// TODO: rest of input events
// InputEvent::DeviceAdded { device } => todo!(),
// InputEvent::DeviceRemoved { device } => todo!(),
InputEvent::Keyboard { event } => self.keyboard::<B>(event),
InputEvent::PointerMotion { event } => self.pointer_motion::<B>(event),
InputEvent::PointerMotionAbsolute { event } => self.pointer_motion_absolute::<B>(event),
InputEvent::PointerButton { event } => self.pointer_button::<B>(event),
InputEvent::PointerAxis { event } => self.pointer_axis::<B>(event),
_ => (),
}
}
2023-09-02 23:51:39 +02:00
/// Get the [`FocusTarget`] under `point`.
2023-07-28 22:16:25 +02:00
pub fn surface_under<P>(&self, point: P) -> Option<(FocusTarget, Point<i32, Logical>)>
where
P: Into<Point<f64, Logical>>,
{
2023-08-06 03:49:39 +02:00
let point: Point<f64, Logical> = point.into();
let output = self.space.outputs().find(|op| {
self.space
.output_geometry(op)
.expect("called output_geometry on unmapped output (this shouldn't happen here)")
.contains(point.to_i32_round())
})?;
let output_geo = self
.space
.output_geometry(output)
.expect("called output_geometry on unmapped output");
let layers = layer_map_for_output(output);
let top_fullscreen_window = self.focus_state.focus_stack.iter().rev().find(|win| {
win.with_state(|state| {
state.fullscreen_or_maximized.is_fullscreen()
&& state.tags.iter().any(|tag| tag.active())
})
});
2023-08-06 03:49:39 +02:00
// I think I'm going a bit too far with the functional stuff
top_fullscreen_window
.map(|window| (FocusTarget::from(window.clone()), output_geo.loc))
.or_else(|| {
layers
.layer_under(wlr_layer::Layer::Overlay, point)
.or_else(|| layers.layer_under(wlr_layer::Layer::Top, point))
.map(|layer| {
let layer_loc = layers.layer_geometry(layer).expect("no layer geo").loc;
(FocusTarget::from(layer.clone()), output_geo.loc + layer_loc)
})
2023-08-06 03:49:39 +02:00
})
.or_else(|| {
self.space
2023-09-02 23:51:39 +02:00
.elements()
.rev()
.filter(|win| win.is_on_active_tag(self.space.outputs()))
.find_map(|win| {
let loc = self
.space
.element_location(win)
.expect("called elem loc on unmapped win")
- win.geometry().loc;
if win.is_in_input_region(&(point - loc.to_f64())) {
Some((win.clone().into(), loc))
} else {
None
}
})
2023-08-06 03:49:39 +02:00
})
.or_else(|| {
layers
.layer_under(wlr_layer::Layer::Bottom, point)
.or_else(|| layers.layer_under(wlr_layer::Layer::Background, point))
.map(|layer| {
let layer_loc = layers.layer_geometry(layer).expect("no layer geo").loc;
(FocusTarget::from(layer.clone()), output_geo.loc + layer_loc)
})
})
}
2023-08-08 17:55:06 +02:00
fn keyboard<I: InputBackend>(&mut self, event: I::KeyboardKeyEvent) {
let serial = SERIAL_COUNTER.next_serial();
let time = event.time_msec();
let press_state = event.state();
let mut move_mode = false;
2023-08-16 00:26:17 +02:00
let reload_keybind = self.input_state.reload_keybind;
let kill_keybind = self.input_state.kill_keybind;
2023-08-08 17:55:06 +02:00
let action = self
.seat
.get_keyboard()
.expect("Seat has no keyboard") // FIXME: handle err
.input(
self,
event.key_code(),
press_state,
serial,
time,
|state, modifiers, keysym| {
if press_state == KeyState::Pressed {
let mut modifier_mask = Vec::<Modifier>::new();
if modifiers.alt {
modifier_mask.push(Modifier::Alt);
}
if modifiers.shift {
modifier_mask.push(Modifier::Shift);
}
if modifiers.ctrl {
modifier_mask.push(Modifier::Ctrl);
}
if modifiers.logo {
modifier_mask.push(Modifier::Super);
}
2023-08-16 00:26:17 +02:00
let modifier_mask = ModifierMask::from(modifier_mask);
2023-08-08 17:55:06 +02:00
let raw_sym = if keysym.raw_syms().len() == 1 {
keysym.raw_syms()[0]
} else {
keysyms::KEY_NoSymbol
};
2023-08-16 00:26:17 +02:00
2023-08-08 17:55:06 +02:00
if let Some(callback_id) = state
.input_state
.keybinds
2023-08-16 00:26:17 +02:00
.get(&(modifier_mask, raw_sym))
2023-08-08 17:55:06 +02:00
{
return FilterResult::Intercept(KeyAction::CallCallback(*callback_id));
2023-08-16 00:26:17 +02:00
} else if (modifier_mask, raw_sym) == kill_keybind {
return FilterResult::Intercept(KeyAction::Quit);
2023-08-16 00:26:17 +02:00
} else if (modifier_mask, raw_sym) == reload_keybind {
return FilterResult::Intercept(KeyAction::ReloadConfig);
2023-08-08 17:55:06 +02:00
} else if let mut vt @ keysyms::KEY_XF86Switch_VT_1..=keysyms::KEY_XF86Switch_VT_12 =
keysym.modified_sym() {
vt = vt - keysyms::KEY_XF86Switch_VT_1 + 1;
tracing::info!("Switching to vt {vt}");
return FilterResult::Intercept(KeyAction::SwitchVt(vt as i32));
2023-08-08 17:55:06 +02:00
}
}
if keysym.modified_sym() == keysyms::KEY_Control_L {
match press_state {
KeyState::Pressed => {
move_mode = true;
}
KeyState::Released => {
move_mode = false;
}
}
}
FilterResult::Forward
},
);
self.move_mode = move_mode;
match action {
Some(KeyAction::CallCallback(callback_id)) => {
if let Some(stream) = self.api_state.stream.as_ref() {
if let Err(err) = crate::api::send_to_client(
&mut stream.lock().expect("Could not lock stream mutex"),
&OutgoingMsg::CallCallback {
callback_id,
args: None,
},
) {
tracing::error!("error sending msg to client: {err}");
}
}
2023-08-08 17:55:06 +02:00
}
Some(KeyAction::SwitchVt(vt)) => {
2023-08-29 05:53:24 +02:00
if let Backend::Udev(udev) = &mut self.backend {
if let Err(err) = udev.session.change_vt(vt) {
2023-08-08 17:55:06 +02:00
tracing::error!("Failed to switch to vt {vt}: {err}");
}
}
}
Some(KeyAction::Quit) => {
self.loop_signal.stop();
2023-08-08 17:55:06 +02:00
}
2023-08-16 00:26:17 +02:00
Some(KeyAction::ReloadConfig) => {
self.restart_config().expect("failed to restart config");
}
None => {}
2023-08-08 17:55:06 +02:00
}
}
2023-06-10 03:29:17 +02:00
fn pointer_button<I: InputBackend>(&mut self, event: I::PointerButtonEvent) {
let pointer = self.seat.get_pointer().expect("Seat has no pointer"); // FIXME: handle err
let keyboard = self.seat.get_keyboard().expect("Seat has no keyboard"); // FIXME: handle err
2023-06-10 03:29:17 +02:00
// A serial is a number sent with a event that is sent back to the
// server by the clients in further requests. This allows the server to
// keep track of which event caused which requests. It is an AtomicU32
// that increments when next_serial is called.
let serial = SERIAL_COUNTER.next_serial();
// Returns which button on the pointer was used.
let button = event.button_code();
// The state, either released or pressed.
let button_state = event.state();
let pointer_loc = pointer.current_location();
2023-06-10 03:29:17 +02:00
// If the button was clicked, focus on the window below if exists, else
// unfocus on windows.
if ButtonState::Pressed == button_state {
2023-08-08 00:23:09 +02:00
if let Some((focus, window_loc)) = self.surface_under(pointer_loc) {
2023-07-29 04:38:12 +02:00
// tracing::debug!("button click on {window:?}");
2023-06-10 03:29:17 +02:00
const BUTTON_LEFT: u32 = 0x110;
const BUTTON_RIGHT: u32 = 0x111;
if self.move_mode {
if event.button_code() == BUTTON_LEFT {
2023-08-08 00:23:09 +02:00
if let Some(wl_surf) = focus.wl_surface() {
2023-07-28 21:33:14 +02:00
crate::grab::move_grab::move_request_server(
2023-07-25 01:59:05 +02:00
self,
&wl_surf,
&self.seat.clone(),
serial,
BUTTON_LEFT,
2023-07-25 01:59:05 +02:00
);
}
2023-06-10 03:29:17 +02:00
return; // TODO: kinda ugly return here
} else if event.button_code() == BUTTON_RIGHT {
2023-08-08 00:23:09 +02:00
let FocusTarget::Window(window) = focus else { return };
2023-06-10 03:29:17 +02:00
let window_geometry = window.geometry();
let window_x = window_loc.x as f64;
let window_y = window_loc.y as f64;
let window_width = window_geometry.size.w as f64;
let window_height = window_geometry.size.h as f64;
let half_width = window_x + window_width / 2.0;
let half_height = window_y + window_height / 2.0;
let full_width = window_x + window_width;
let full_height = window_y + window_height;
let edges = match pointer_loc {
Point { x, y, .. }
if (window_x..=half_width).contains(&x)
&& (window_y..=half_height).contains(&y) =>
{
ResizeEdge::TopLeft
}
2023-06-10 03:29:17 +02:00
Point { x, y, .. }
if (half_width..=full_width).contains(&x)
&& (window_y..=half_height).contains(&y) =>
{
ResizeEdge::TopRight
}
2023-06-10 03:29:17 +02:00
Point { x, y, .. }
if (window_x..=half_width).contains(&x)
&& (half_height..=full_height).contains(&y) =>
{
ResizeEdge::BottomLeft
}
Point { x, y, .. }
if (half_width..=full_width).contains(&x)
&& (half_height..=full_height).contains(&y) =>
{
ResizeEdge::BottomRight
}
_ => ResizeEdge::None,
};
2023-07-25 01:59:05 +02:00
if let Some(wl_surf) = window.wl_surface() {
2023-07-28 21:33:14 +02:00
crate::grab::resize_grab::resize_request_server(
2023-07-25 01:59:05 +02:00
self,
&wl_surf,
&self.seat.clone(),
serial,
edges.into(),
2023-07-25 01:59:05 +02:00
BUTTON_RIGHT,
);
}
}
2023-06-10 03:29:17 +02:00
} else {
// Move window to top of stack.
2023-08-08 00:23:09 +02:00
if let FocusTarget::Window(window) = &focus {
self.space.raise_element(window, true);
if let WindowElement::X11(surface) = &window {
if !surface.is_override_redirect() {
self.xwm
.as_mut()
.expect("no xwm")
.raise_window(surface)
.expect("failed to raise x11 win");
surface
.set_activated(true)
.expect("failed to set x11 win to activated");
}
2023-07-29 04:38:12 +02:00
}
2023-07-25 01:59:05 +02:00
}
2023-06-10 03:29:17 +02:00
2023-08-08 00:23:09 +02:00
tracing::debug!("wl_surface focus is some? {}", focus.wl_surface().is_some());
2023-07-29 04:38:12 +02:00
// NOTE: *Do not* set keyboard focus to an override redirect window. This leads
// | to wonky things like right-click menus not correctly getting pointer
// | clicks or showing up at all.
2023-08-06 03:49:39 +02:00
// TODO: use update_keyboard_focus from anvil
2023-08-08 00:23:09 +02:00
if !matches!(&focus, FocusTarget::Window(WindowElement::X11(surf)) if surf.is_override_redirect())
{
keyboard.set_focus(self, Some(focus.clone()), serial);
2023-07-29 04:38:12 +02:00
}
2023-06-18 04:02:58 +02:00
2023-06-10 03:29:17 +02:00
self.space.elements().for_each(|window| {
2023-07-25 01:59:05 +02:00
if let WindowElement::Wayland(window) = window {
window.toplevel().send_configure();
}
2023-06-10 03:29:17 +02:00
});
2023-08-08 00:23:09 +02:00
if let FocusTarget::Window(window) = &focus {
let focused_name = match &window {
WindowElement::Wayland(win) => {
compositor::with_states(win.toplevel().wl_surface(), |states| {
let lock = states
.data_map
.get::<smithay::wayland::shell::xdg::XdgToplevelSurfaceData>()
.expect("XdgToplevelSurfaceData wasn't in surface's data map")
.lock()
.expect("failed to acquire lock");
lock.app_id.clone().unwrap_or_default()
})
}
WindowElement::X11(surf) => surf.class(),
};
tracing::debug!("setting keyboard focus to {focused_name}");
}
2023-06-10 03:29:17 +02:00
}
} else {
2023-07-25 01:59:05 +02:00
self.space.elements().for_each(|window| match window {
WindowElement::Wayland(window) => {
window.set_activated(false);
window.toplevel().send_configure();
}
WindowElement::X11(surface) => {
surface
.set_activated(false)
.expect("failed to deactivate x11 win");
// INFO: do i need to configure this?
}
2023-06-10 03:29:17 +02:00
});
keyboard.set_focus(self, None, serial);
}
2023-06-10 03:29:17 +02:00
};
2023-06-10 03:29:17 +02:00
// Send the button event to the client.
pointer.button(
self,
&ButtonEvent {
button,
state: button_state,
serial,
time: event.time_msec(),
},
);
}
fn pointer_axis<I: InputBackend>(&mut self, event: I::PointerAxisEvent) {
let source = event.source();
let horizontal_amount = event
.amount(Axis::Horizontal)
.unwrap_or_else(|| event.amount_discrete(Axis::Horizontal).unwrap_or(0.0) * 3.0);
2023-06-10 03:29:17 +02:00
let vertical_amount = event
.amount(Axis::Vertical)
.unwrap_or_else(|| event.amount_discrete(Axis::Vertical).unwrap_or(0.0) * 3.0);
2023-06-10 03:29:17 +02:00
let horizontal_amount_discrete = event.amount_discrete(Axis::Horizontal);
let vertical_amount_discrete = event.amount_discrete(Axis::Vertical);
2023-06-10 03:29:17 +02:00
let mut frame = AxisFrame::new(event.time_msec()).source(source);
2023-06-10 03:29:17 +02:00
if horizontal_amount != 0.0 {
frame = frame.value(Axis::Horizontal, horizontal_amount);
if let Some(discrete) = horizontal_amount_discrete {
frame = frame.discrete(Axis::Horizontal, discrete as i32);
}
} else if source == AxisSource::Finger {
frame = frame.stop(Axis::Horizontal);
}
2023-06-10 03:29:17 +02:00
if vertical_amount != 0.0 {
frame = frame.value(Axis::Vertical, vertical_amount);
if let Some(discrete) = vertical_amount_discrete {
frame = frame.discrete(Axis::Vertical, discrete as i32);
}
} else if source == AxisSource::Finger {
frame = frame.stop(Axis::Vertical);
}
2023-07-29 04:38:12 +02:00
// tracing::debug!(
// "axis on current focus: {:?}",
// self.seat.get_pointer().unwrap().current_focus()
// );
self.seat
.get_pointer()
.expect("Seat has no pointer")
2023-07-29 04:38:12 +02:00
.axis(self, frame);
2023-06-10 03:29:17 +02:00
}
2023-08-08 02:48:18 +02:00
2023-08-29 05:53:24 +02:00
/// Clamp pointer coordinates inside outputs
fn clamp_coords(&self, pos: Point<f64, Logical>) -> Point<f64, Logical> {
if self.space.outputs().next().is_none() {
return pos;
}
2023-08-08 02:48:18 +02:00
2023-08-29 05:53:24 +02:00
let (pos_x, pos_y) = pos.into();
2023-08-08 02:48:18 +02:00
2023-08-29 05:53:24 +02:00
let nearest_points = self.space.outputs().map(|op| {
let size = self
.space
.output_geometry(op)
.expect("called output_geometry on unmapped output")
.size;
let loc = op.current_location();
let pos_x = pos_x.clamp(loc.x as f64, (loc.x + size.w) as f64);
let pos_y = pos_y.clamp(loc.y as f64, (loc.y + size.h) as f64);
(pos_x, pos_y)
});
let nearest_point = nearest_points.min_by(|(x1, y1), (x2, y2)| {
f64::total_cmp(
&((pos_x - x1).powi(2) + (pos_y - y1).powi(2)).sqrt(),
&((pos_x - x2).powi(2) + (pos_y - y2).powi(2)).sqrt(),
)
});
nearest_point.map(|point| point.into()).unwrap_or(pos)
2023-08-08 02:48:18 +02:00
}
2023-06-10 03:29:17 +02:00
fn pointer_motion_absolute<I: InputBackend>(&mut self, event: I::PointerMotionAbsoluteEvent) {
let Some(output) = self.space.outputs().next() else { return; };
let output_geo = self
.space
.output_geometry(output)
.expect("Output geometry doesn't exist");
2023-06-10 03:29:17 +02:00
let pointer_loc = event.position_transformed(output_geo.size) + output_geo.loc.to_f64();
let serial = SERIAL_COUNTER.next_serial();
let pointer = self.seat.get_pointer().expect("Seat has no pointer"); // FIXME: handle err
2023-06-10 03:29:17 +02:00
// tracing::info!("pointer_loc: {:?}", pointer_loc);
self.pointer_location = pointer_loc;
2023-06-25 00:39:40 +02:00
match self.focus_state.focused_output {
Some(_) => {
if let Some(output) = self
.space
.output_under(self.pointer_location)
.next()
.cloned()
{
self.focus_state.focused_output = Some(output);
}
}
None => {
self.focus_state.focused_output = self.space.outputs().next().cloned();
}
}
2023-08-08 00:23:09 +02:00
let surface_under_pointer = self.surface_under(pointer_loc);
2023-07-28 22:16:25 +02:00
2023-07-29 04:38:12 +02:00
// tracing::debug!("surface_under_pointer: {surface_under_pointer:?}");
// tracing::debug!("pointer focus: {:?}", pointer.current_focus());
2023-06-10 03:29:17 +02:00
pointer.motion(
self,
surface_under_pointer,
&MotionEvent {
location: pointer_loc,
serial,
time: event.time_msec(),
},
);
}
fn pointer_motion<I: InputBackend>(&mut self, event: I::PointerMotionEvent) {
let serial = SERIAL_COUNTER.next_serial();
self.pointer_location += event.delta();
// clamp to screen limits
// this event is never generated by winit
self.pointer_location = self.clamp_coords(self.pointer_location);
2023-06-25 00:39:40 +02:00
match self.focus_state.focused_output {
Some(_) => {
if let Some(output) = self
.space
.output_under(self.pointer_location)
.next()
.cloned()
{
self.focus_state.focused_output = Some(output);
}
}
None => {
self.focus_state.focused_output = self.space.outputs().next().cloned();
}
}
2023-06-10 03:29:17 +02:00
2023-07-28 22:16:25 +02:00
let surface_under = self.surface_under(self.pointer_location);
2023-06-10 03:29:17 +02:00
// tracing::info!("{:?}", self.pointer_location);
if let Some(ptr) = self.seat.get_pointer() {
ptr.motion(
self,
surface_under,
2023-06-10 03:29:17 +02:00
&MotionEvent {
location: self.pointer_location,
serial,
time: event.time_msec(),
},
);
// ptr.relative_motion(
// self,
// under,
// &RelativeMotionEvent {
// delta: event.delta(),
// delta_unaccel: event.delta_unaccel(),
// utime: event.time(),
// },
// )
}
}
}