Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Change PointerHandle.motion and friends to take relative coords #1223

Open
wants to merge 3 commits into
base: master
Choose a base branch
from

Conversation

colinmarc
Copy link
Contributor

@colinmarc colinmarc commented Nov 10, 2023

The existing API is helpful in that it makes it harder to mess up the math, but it's not flexible enough to support having different windows with different scaling factors. That's relevant if, for example, you want to force a scale factor of 1 for Xwayland windows.

The existing API takes Option<(<D as SeatHandler>::PointerFocus, Point<i32, Logical>)>, with the point being the surface origin, rather than the relative coordinates of the point on that surface. Then, internally, it converts into relative coordinates to generate the wayland event. A user familiar with wayland but not reading the documentation closely might think that the point should be the relative coords, not the surface origin (this happened to me).

Additionally, if you have different scale factors for different windows, it's impossible (or very difficult) to get the calculation right, since smithay is assuming that the coordinate space for the surface is the same as the global compositor space.

@colinmarc colinmarc force-pushed the pointer-motion-coords branch 3 times, most recently from 4236e1e to f7fb1c4 Compare November 10, 2023 20:20
@colinmarc
Copy link
Contributor Author

I can't get anvil to run on my machine for (it appears unrelated) reasons, so I haven't tested that part.

It occurs to me that this will probably result in confusing behavior for existing compositors who update without knowing. I don't really know how to mitigate that.

@colinmarc
Copy link
Contributor Author

Hm, this is kinda lackluster. Taking another stab at this with a newtype to flush out cases I missed.

@colinmarc colinmarc force-pushed the pointer-motion-coords branch 2 times, most recently from e574574 to 5ceb562 Compare November 12, 2023 14:57
@codecov-commenter
Copy link

codecov-commenter commented Nov 12, 2023

Codecov Report

Attention: Patch coverage is 34.17722% with 104 lines in your changes missing coverage. Please review.

Project coverage is 21.76%. Comparing base (003ca51) to head (e3a9927).
Report is 325 commits behind head on master.

Files Patch % Lines
src/input/pointer/mod.rs 36.53% 33 Missing ⚠️
src/wayland/tablet_manager/tablet_tool.rs 0.00% 16 Missing ⚠️
anvil/src/input_handler.rs 34.78% 15 Missing ⚠️
src/desktop/wayland/popup/grab.rs 0.00% 8 Missing ⚠️
anvil/src/shell/element.rs 0.00% 6 Missing ⚠️
anvil/src/shell/grabs.rs 0.00% 6 Missing ⚠️
src/wayland/selection/data_device/dnd_grab.rs 0.00% 6 Missing ⚠️
...c/wayland/selection/data_device/server_dnd_grab.rs 0.00% 6 Missing ⚠️
anvil/src/shell/xdg.rs 0.00% 2 Missing ⚠️
src/desktop/wayland/layer.rs 0.00% 2 Missing ⚠️
... and 2 more
Additional details and impacted files
@@           Coverage Diff            @@
##           master    #1223    +/-   ##
========================================
  Coverage   21.76%   21.76%            
========================================
  Files         155      156     +1     
  Lines       24373    24490   +117     
========================================
+ Hits         5305     5331    +26     
- Misses      19068    19159    +91     
Flag Coverage Δ
wlcs-buffer 18.74% <0.00%> (-0.05%) ⬇️
wlcs-core 18.40% <0.00%> (-0.05%) ⬇️
wlcs-output 7.64% <0.00%> (+<0.01%) ⬆️
wlcs-pointer-input 20.60% <34.17%> (+0.06%) ⬆️

Flags with carried forward coverage won't be shown. Click here to find out more.

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

@colinmarc
Copy link
Contributor Author

colinmarc commented Nov 12, 2023

Ok, I think this is much better now. I introduced a type, RelativePoint, to add semantic meaning to Option<(<D as SeatHandler>::PointerFocus, Point<i32, Logical>)>, which was a bit of a mouthful.

I find the new API is a bit more intuitive (I got confused that motion wanted the surface origin), but I know it's opinionated, so please tell me if you'd prefer a different approach. One disadvantage is that Window.surface_under still returns Option((surface, surface_origin)), but I added RelativePoint::on_focused_surface(pos, surface, surface_origin) to make it easy to follow the happy path without doing any math yourself.

This has the extra benefit that the calculation pos - surface_origin.to_f64() is only done there, instead of all over the library.

@colinmarc colinmarc changed the title Change PointerHandle.motion to take relative coords Change PointerHandle.motion and friends to take relative coords Nov 12, 2023
focus: Some((root.into(), (0, 0).into())),
focus: Some(RelativePoint {
surface: root.into(),
loc: (0f64, 0f64).into(),
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wanted to highlight this as something I really wasn't sure about.

@colinmarc colinmarc force-pushed the pointer-motion-coords branch 2 times, most recently from 0880050 to c624329 Compare November 12, 2023 15:23
@Drakulix
Copy link
Member

That's relevant if, for example, you want to force a scale factor of 1 for Xwayland windows.

Before giving this a proper review, note that I would love to have proper API in smithay to support overriding scale factors of windows and/or globally for Xwayland.

Copy link
Member

@Drakulix Drakulix left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Couple of nitpicks. I like the general approach, this seems more flexible without introducing too much friction in the api.

/// Represents a point on a surface, relative to that surface's origin in
/// global compositor space.
#[derive(Debug, Clone)]
pub struct RelativePoint<S> {
Copy link
Member

@Drakulix Drakulix Nov 13, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am not a huge fan of this name, as it is used to denote the focus of the pointer and the relative location. I don't think specifying the relative aspect in the name is most important, if surface and loc have good documentation.

I am open to some bike shedding over the name, but I am leaning towards something like FocusedPoint or FocusPoint?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's what I originally had, but I realized I had to move it out of the input::pointer mod because it's used by grab code. Once I moved it into utils, it seemed appropriate to give it more of a general name (Focus being an input-specific concept).

Happy to place/name it whatever you like, just wanted to add that context. My suggestion given your constraints would be SurfaceCoordinate or SurfacePoint or similar.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

SurfacePoint

That is also a good one, but given a focused element doesn't have to be a surface, RelativePoint does seem to capture this quite well. 😅

src/utils/relative_point.rs Outdated Show resolved Hide resolved
anvil/src/winit.rs Outdated Show resolved Hide resolved
let event = MotionEvent {
location: event.location - surface_location.to_f64(),
location: focus.loc.to_f64(),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So we are not using the MotionEvent location anymore at all? I quite dislike that to be honest, because that means we pass in multiple locations, but ignore some of them. In that case please change the signature of motion to not take a MotionEvent, but directly the serial and time.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

event.location is still used to set self.location, which is provided by PointerHandle.current_location, for example.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

event.location is still used to set self.location

Right... Maybe we should then at least rename MotionEvent::location to global_location or absolute_location to further differenciate the two values?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done in e3a9927.

The existing API was helpful in that it made it harder to mess up the math,
but not flexible enough to support having different windows with different
scaling factors. That's relevent if, for example, you want to force a scale
factor of 1 for Xwayland windows.
@colinmarc colinmarc force-pushed the pointer-motion-coords branch from c624329 to 180b08e Compare November 28, 2023 11:18
@colinmarc
Copy link
Contributor Author

Sorry for the delay! Updated to address comments.

@colinmarc
Copy link
Contributor Author

colinmarc commented Nov 28, 2023

Renaming MotionEvent.location made me realize that this line may no longer be correct:

new_event.location -= loc.to_f64();

But I'm not sure.

@Drakulix
Copy link
Member

Renaming MotionEvent.location made me realize that this line may no longer be correct:

new_event.location -= loc.to_f64();

But I'm not sure.

It is correct, the issue is that PointerTarget::enter/motion both get a MotionEvent with relative coordinates, while the PointerHandle and grabs don't. But both take the same MotionEvent-type, which means the last commit made the situation both better and worse, depending on which part of the API you look at.

And I think this the core issue, that makes this api feel so weird. We probably don't want to use the same type here, as this is confusing at best. This fits quite nicely with a discussion we recently had of introducing Local or Relative coordinates as opposed to Global coordinates in addition to the Logical/Physical coordinate spaces. I'll get back to you on that one.

@colinmarc
Copy link
Contributor Author

I took a stab at making MotionEvent generic over relative/global coordinate spaces:

#[derive(Debug)]
pub struct GlobalOrigin;

#[derive(Debug)]
pub struct SurfaceOrigin;

/// Pointer motion event
#[derive(Debug)]
pub struct MotionEvent<Origin> {
    /// Location of the pointer. If `Origin` is `SurfaceOrigin`, the coordinates
    /// are relative to the origin of the surface. If `Origin` is `GlobalOrigin`, it is a
    /// location in global compositor space.
    pub location: Point<f64, Logical>,
    /// Serial of the event
    pub serial: Serial,
    /// Timestamp of the event, with millisecond granularity
    pub time: u32,

    _origin: std::marker::PhantomData<Origin>,
}

The problem is still that lots of functions take an optional focus, but the MotionEvent is not optional:

    /// Notify that the pointer moved
    ///
    /// You provide the new location of the pointer, in the form of:
    ///
    /// - The surface that the cursor is on top of, and
    /// - The coordinates of the cursor relative to the origin of that surface
    ///
    /// A value of None indicates that no surface is focused, and the
    /// coordinates of the MotionEvent will be ignored.
    ///
    /// This will internally take care of notifying the appropriate client
    /// objects of enter/motion/leave events.
    #[instrument(level = "trace", parent = &self.span, skip(self, data, focus))]
    pub fn motion(&self, data: &mut D, focus: Option<D::PointerFocus>, event: &MotionEvent<SurfaceOrigin>) {
        // ...
    }

Here, if focus is None, then the location field of the event is completely ignored. It's the same problem, but subtly different, and even more finicky, because you need to construct a MotionEvent in the backend code of your compositor (anvil example), and it's not clear what should be set for location if it's a relative one and there's no focus.

I see two possibilities:

  1. Ditching MotionEvent altogether, adding serial etc as arguments in their own right, and keeping RelativePoint (from the current version of this PR)
  2. Removing location from MotionEvent but otherwise keeping this PR as-is.

I think I like 2 the best. MotionEvent keeps its semantic meaning with a timestamp and serial, which is to say it's semi-opaque and gets passed through everywhere. But the RelativePoint gets more relative as we go down the tree.

@Drakulix
Copy link
Member

Drakulix commented Mar 5, 2024

I see two possibilities:

1. Ditching `MotionEvent` altogether, adding `serial` etc as arguments in their own right, and keeping `RelativePoint` (from the [current version](https://github.com/Smithay/smithay/pull/1223/commits/9b5411ac287e595b18444ffcfa574700a5d5bed6#diff-4d2f1d428c6ceedf69db74c16fda79aa7ce3e8317c08456336a67f67b7ebcd13) of this PR)

2. Removing `location` from `MotionEvent` but otherwise keeping this PR as-is.

I think I like 2 the best. MotionEvent keeps its semantic meaning with a timestamp and serial, which is to say it's semi-opaque and gets passed through everywhere. But the RelativePoint gets more relative as we go down the tree.

I agree, that 2. sounds like a good solution.

@ryanabx
Copy link

ryanabx commented Apr 30, 2024

Removing location from MotionEvent but otherwise keeping this PR as-is.

I think I like 2 the best. MotionEvent keeps its semantic meaning with a timestamp and serial, which is to say it's semi-opaque and gets passed through everywhere. But the RelativePoint gets more relative as we go down the tree.

Naively trying to understand this -- If MotionEvent doesn't have location, how do we gather where the pointer moved in fn motion...

@colinmarc
Copy link
Contributor Author

Naively trying to understand this -- If MotionEvent doesn't have location, how do we gather where the pointer moved in fn motion...

The RelativePoint has a location, but it's not in the global compositor space, it's relative to the surface. Turns out some of the APIs (grab intercepting for example) rely on getting the global cursor location out of the PointerHandle, so I got stuck.

@chrisduerr
Copy link
Contributor

I don't think RelativePoint contributes much here, especially since every point is relative.

With MotionEvent things are still just as confusing anyway, since what you want is probably position relative to the touch down surface, while the focus will potentially be on a separate surface. So I'd say location should only be in the event and not coupled to the focus.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants