Skip to content

Commit

Permalink
Update proposal to add 'virtual pointers'
Browse files Browse the repository at this point in the history
  • Loading branch information
Perksey committed Dec 16, 2024
1 parent 8dba362 commit 10fc9e3
Showing 1 changed file with 57 additions and 28 deletions.
85 changes: 57 additions & 28 deletions documentation/proposals/Proposal - Multi-Backend Input.md
Original file line number Diff line number Diff line change
Expand Up @@ -277,8 +277,9 @@ public readonly record struct ConnectionEvent(IInputDevice Device, long Timestam
public readonly record struct KeyChangedEvent(IKeyboard Keyboard, long Timestamp, Button<KeyName> Key, Button<KeyName> Previous, bool IsRepeat, KeyModifiers Modifiers);
public readonly record struct KeyCharEvent(IKeyboard Keyboard, long Timestamp, char? Character);
public readonly record struct ButtonChangedEvent<T>(IButtonDevice<T> Device, long Timestamp, Button<T> Button, Button<T> Previous) where T : struct, Enum;
public readonly record struct PointChangedEvent(IPointer Pointer, long Timestamp, TargetPoint? OldPoint, TargetPoint Point);
public readonly record struct PointChangedEvent(IPointer Pointer, long Timestamp, TargetPoint? OldPoint, TargetPoint? NewPoint);
public readonly record struct PointerGripChangedEvent(IPointer Pointer, long Timestamp, float GripPressure, float Delta);
public readonly record struct PointerTargetChangedEvent(IPointer Pointer, long Timestamp, IPointerTarget Target, bool IsAdded, Box3F<float> OldBounds, Box3F<float> NewBounds);

This comment has been minimized.

Copy link
@domportera

domportera Dec 19, 2024

just for my own clarity - what case is this intended to cover? a cursor traveling across multiple displays? physically changing displays? changing display resolution?

This comment has been minimized.

Copy link
@Perksey

Perksey Dec 20, 2024

Author Member

For mice that aren't in raw input mode, the window is a target. The window can change size. As a result, the target can change size. The latter two of your questions could indeed cause the window size to change, and as a result the target size.

public readonly record struct MouseScrollEvent(IMouse Mouse, long Timestamp, TargetPoint Point, Vector2 WheelPosition, Vector2 Delta);
public readonly record struct PointerClickEvent(IPointer Pointer, long Timestamp, TargetPoint Point, MouseButton Button);
public readonly record struct JoystickHatMoveEvent(IJoystick Joystick, long Timestamp, Vector2 Value, Vector2 Delta);
Expand Down Expand Up @@ -372,23 +373,13 @@ A target is defined as follows:
public interface IPointerTarget
{
/// <summary>
/// The minimum position of points on this target, where <see cref="float.NegativeInfinity" /> represents the lack
/// of a lower bound on a particular axis and <c>0</c> represents an unused axis if <see cref="MaxPosition" /> is
/// also <c>0</c> for that axis.
/// The boundary in which positions of points on this target shall fall. For <see cref="Box3F{T}.Min" />,
/// <see cref="float.NegativeInfinity" /> shall represent the lack of a lower bound on a particular axis. For
/// For <see cref="Box3F{T}.Max" />, <see cref="float.PositiveInfinity" /> shall represent the lack of a lower bound
/// on a particular axis. <c>0</c> represents an unused axis that axis is <c>0</c> on both
/// <see cref="Box3F{T}.Min" /> and <see cref="Box3F{T}.Max" />.
/// </summary>
Vector3 MinPosition { get; }

/// <summary>
/// The maximum position of points on this target, where <see cref="float.PositiveInfinity" /> represents the lack
/// of an upper bound on a particular axis and <c>0</c> represents an unused axis if <see cref="MinPosition" /> is
/// also <c>0</c> for that axis.
/// </summary>
Vector3 MaxPosition { get; }

/// <summary>
/// An optional name describing the target.
/// </summary>
string? Name { get; }
Box3F<float> Bounds { get; }

/// <summary>
/// Gets the number of points with which the given pointer is pointing at this target.
Expand All @@ -415,6 +406,10 @@ public interface IPointerTarget
}
```

**FUTURE IMPROVEMENT:** This interface could be expanded to provide rotation of the target itself as well, representing
a full `Transform` structure for the space. At this time, this was not deemed necessary for inclusion, but should be a
trivial extension to add in the future.

**INFORMATIVE TEXT**: Furthermore, it is our eventual goal to be able to support considering VR hands as pointer devices
through raycasting. Such a future proposal will involve a way to create a child target within the bounds of this target
a `IPointerTarget` from that which represents the 3D world (i.e. the entire VR world is a target, and the point
Expand Down Expand Up @@ -445,31 +440,39 @@ public enum TargetPointFlags
/// <summary>
/// Represents a point on a target at which a pointer is pointing.
/// </summary>
/// <param name="Id">
/// An integral identifier for the point. This point must be the only point for the device currently pointing at a
/// target with this identifier at any given time. If this point ceases to point at the target, then the identifier
/// becomes free for another device point. This means that this identifier can just be an index, but may be globally
/// unique depending on the backend's capabilities.
/// </param>
/// <param name="Flags">Flags describing the state of the point.</param>
/// <param name="Position">The absolute position on the target at which the pointer is pointing.</param>
/// <param name="NormalizedPosition">
/// The normalized position on the target at which the pointer is pointing, if applicable. If this is not available
/// (e.g. due to the target being infinitely large a.k.a. "unbounded"), then this property shall have a value of
/// <c>default</c>.
/// </param>
/// <param name="Orientation">
/// The angle at which the pointer is pointing at the point on the target. An identity quaternion shall be interpreted
/// as the point directly perpendicular to and facing towards the target. This shall carry an identity quaternion if
/// there is no orientation available.
/// <param name="Pointer">
/// A ray representing the distance and angle at which the pointer is pointing at the point on the target. A ray with an

This comment has been minimized.

Copy link
@domportera

domportera Dec 19, 2024

I'm confused by distance being a Vector3 - in what scenarios would distance not be simply represented by the Z axis of a 2D device?

This comment has been minimized.

Copy link
@Perksey

Perksey Dec 20, 2024

Author Member

Distance is now a component of the ray representing the angle at which the device (e.g. a pen) is pointing at the screen. We could explode this into two separate fields where the distance component is one-dimensional, but this then precludes future devices that may have more than one-dimensional distance tracking abilities (and not to mention, by exploding the ray we lose the nice abstraction that is the ray)

/// orientation equivalent to an identity quaternion shall be interpreted as the point directly perpendicular to and
/// facing towards the target, with this being the default value should this information be unavailable. If distance
/// information is unavailable, this shall be equivalent to a <c>default</c> vector.
/// </param>
/// <param name="Distance">The distance of the pointer from the point the pointer is pointing at.</param>
/// <param name="Pressure">
/// The pressure applied to the point on the target by the pointer, between <c>0.0</c> representing the minimum amount
/// of pressure and <c>1.0</c> representing the maximum amount of pressure. This shall be <c>1.0</c> if such data is
/// unavailable but the point is otherwise valid.
/// </param>
/// <param name="Target">The pointer being pointed at.</param>
public readonly record struct TargetPoint(
int Id,
TargetPointFlags Flags,
Vector3 Position,
Vector3 NormalizedPosition,
Quaternion Orientation,
Vector3 Distance,
float Pressure
Ray3F<float> Pointer,
float Pressure,
IPointerTarget Target
) {
public bool IsValid => (Flags & Flags.PointingAtTarget) != Flags.NotPointingAtTarget;
}
Expand All @@ -480,11 +483,9 @@ The `PointerState` shall be defined as follows:
public class PointerState
{
public ButtonReadOnlyList<PointerButton> Buttons { get; }
public InputReadOnlyList<PointerStatePoint> Points { get; }
public InputReadOnlyList<TargetPoint> Points { get; }
public float GripPressure { get; }
}

public readonly record struct PointerStatePoint(IPointerTarget Target, TargetPoint Point);
```

`Points` represents the `TargetPoint`s this pointer is pointing at on its "native targets" i.e. that which is enumerated
Expand All @@ -501,15 +502,43 @@ The handler for pointer inputs shall be defined as follows:
```cs
public interface IPointerInputHandler : IButtonInputHandler<PointerButton>
{
void HandleTargetChanged(PointerTargetChangedEvent @event);
void HandlePointChanged(PointChangedEvent @event);
void HandleGripChanged(PointerGripChangedEvent @event);
}
```

`HandleTargetChanged` must be called when properties on an `IPointerTarget` within `IPointer.Targets` changes, or when
an `IPointerTarget` is added or removed to/from `IPointer.Targets`. `IsAdded` shall be `true` if it has been added,
`false` if it has been removed.

`HandlePointChanged` must be called when a point within `PointerState.Points` changes.

`HandleGripChanged` must be called when `PointerState.GripPressure` changes.

These device interfaces and related APIs are designed to mirror physical hardware that the user uses to point at a
target. However, there are many cases where applications would work better with an abstraction that creates "virtual
pointers" for each point, rather that the points being spread across many logical devices. For this we propose the
following addendum to the `Pointers` class:

```cs
public partial class Pointers
{
public IReadOnlyList<ContextPoint> Points { get; }

This comment has been minimized.

Copy link
@domportera

domportera Dec 19, 2024

I'm not sure this covers the needs entirely. I like the ContextPoint struct (as you well know 😅), but having a single list seems like an inefficient or confusing way to access that data in the case of having more than one provider of such context points. It doesn't harm anything having it, but in absence of alternatives or a defined way of populating such points from the user side, it might just be extra noise.

I think we can safely exclude this portion of the proposal for the time being since we've decided it will be a non-breaking extension anyway. Not sure what the most convenient/flexible/thorough approach will be and I don't want to hold up proceedings while we figure that out

This comment has been minimized.

Copy link
@Perksey

Perksey Dec 20, 2024

Author Member

Good plan, thanks.

}

public readonly record struct ContextPoint(IPointer Device, TargetPoint Point, ButtonReadOnlyList<PointerButton> Buttons, float GripPressure)
{
public int Id { get; }
}
```

`Id` shall be an identifier that mixes `Point.Id` and `Device.Id` in a way that ensures the identifier is unique across
the whole context.

**FUTURE IMPROVEMENT:** The `Pointers` class is also expected to be the site of gesture recognition when proposed in the
future.

`PointerButton` shall be defined as follows:
```cs
public enum PointerButton
Expand Down

0 comments on commit 10fc9e3

Please sign in to comment.