Bug 814 - Enhance PointerEvent / Deprecate MouseEvent
Summary: Enhance PointerEvent / Deprecate MouseEvent
Status: CONFIRMED
Alias: None
Product: Newt
Classification: JogAmp
Component: core (show other bugs)
Version: tbd
Hardware: All all
: --- enhancement
Assignee: Julien Gouesse
URL:
Depends on: 807 813
Blocks: 595 1427
  Show dependency treegraph
 
Reported: 2013-08-09 14:49 CEST by Sven Gothel
Modified: 2023-04-20 13:25 CEST (History)
1 user (show)

See Also:
Type: FEATURE
SCM Refs:
Workaround: ---


Attachments

Note You need to log in before you can comment on or make changes to this bug.
Description Sven Gothel 2013-08-09 14:49:22 CEST
Currently PointerEvent is still called MouseEvent, rename it!

PointerEvent currently misses:
  tiltXYZ[]
to match
   http://www.w3.org/Submission/pointer-events/#pointerevent-interface

+++

Enhance PointerEvent to satisfy 'joystick' controllers.
- Add  PointerTypes: 
   - JoystickA(PointerClass.Offscreen) (analogue)
   - JoystickD(PointerClass.Offscreen) (digital)
   - ???
   - JoystickD == JoyPad ?

- Add 3rd axis for abs coord, z.

- Rel. movement:
  'float moveXYZ[]' and 'float moveScale'
  similar to rotate* !

  Optional:
    Abs. position may be calculated via current position
    and moveScale * moveXYZ[].    
    Resulting position may exceed window coordinates.
    Requires a device context to keep track of position,
    i.e. PointerDevice (see below).

    Hence moveScale is optional.

- Send event to focused window only or
  consider auto-assignment - see Bug 813

- PointerDevice extends InputDevice:
  - Pool of static PointerDevice's accessible by user to
    query (position) and configure (origin, ..).

  - Unique by it's Pointer-ID (as already used in PointerEvent)

  - Keep track of absolute position (see above).

  - Used to fire 'added' and 'removed' events.
Comment 1 Julien Gouesse 2013-08-09 19:00:56 CEST
We have to handle the following kind of events:
- button events (pressed, released) [digital]
- axis events (float) [analog]
- sliding axis events (float, like 2 axis: abscissa, ordinate) [analog]
- hat switch or POV (point-of-view) hat events (9 directions) [analog]
- vector3 events (float, like 3 axis: abscissa, ordinate, applicate) [analog]
- plug in/out events (when a gamepad/mouse/keyboard is added or removed)

We have to handle some effects:
- rumbler
- force feedback

We have to represent all these devices:
- keyboard
- mouse
- gamepad

For example, a mouse event should indicate which mouse has just generated it, nothing forces the end user to plug in a single mouse and a single keyboard even though it is the general case.

An input device must describe its "components" which is very important for gamepads. The developer should be able to listener a specific device. It's already possible to listen to all keyboards which is the case with current key listeners.
Comment 2 Julien Gouesse 2013-08-09 19:04:39 CEST
Generating "mouse" events and moving the mouse pointer with a gamepad or any joystick can be useful when there is no mouse, for video games machines and in assistive technology, for people who can no longer use a mouse because of a muscular weakness.
Comment 3 Julien Gouesse 2013-08-09 19:31:11 CEST
A mouse can be seen as an input device with 2 axis (X & Y), one or multiple buttons and maybe a wheel. The problem is that some native APIs only provide relative data about the mouse movements and the wheel. These "raw" APIs are useful to get more accurate and resolution independent data, especially in games. It allows to decrease the screen resolution without decreasing the accuracy of the mouse. If we use these APIs, we have to compute the absolute position from the relative movement.
Comment 4 Sven Gothel 2013-08-09 23:22:32 CEST
(In reply to comment #1)
> - axis events (float) [analog]
> - sliding axis events (float, like 2 axis: abscissa, ordinate) [analog]
> - hat switch or POV (point-of-view) hat events (9 directions) [analog]
> - vector3 events (float, like 3 axis: abscissa, ordinate, applicate) [analog]

.. can you elaborate on these ? 
Best if you can add a reference/link to documentation.
Comment 5 Sven Gothel 2013-08-09 23:24:20 CEST
(In reply to comment #1)
> We have to handle some effects:
> - rumbler
> - force feedback

I assume these are not input events, but triggers the user 
sent to the InputDevice itself.
Hence InputDevice / PointerDevice may require methods like:
  doRumble(int ms, float strength/frequency)
  ..
or something ?
Comment 6 Julien Gouesse 2013-08-10 00:52:01 CEST
(In reply to comment #4)
> (In reply to comment #1)
> > - axis events (float) [analog]
> > - sliding axis events (float, like 2 axis: abscissa, ordinate) [analog]
> > - hat switch or POV (point-of-view) hat events (9 directions) [analog]
> > - vector3 events (float, like 3 axis: abscissa, ordinate, applicate) [analog]
> 
> .. can you elaborate on these ? 
> Best if you can add a reference/link to documentation.

https://en.wikipedia.org/wiki/File:Joyopis.svg
The main stick [1] can generate axis events.
The green component [7] is a POV hat.
Vector3 events concern natural input interfaces, the only examples that comes to my mind are the Wiimote and the Kinect.

My main source of inspiration is OOIS Joystick listener:
https://github.com/wgois/Object-oriented-Input-System--OIS-/blob/master/includes/OISJoyStick.h
Comment 7 Julien Gouesse 2013-08-10 00:55:30 CEST
(In reply to comment #5)
> (In reply to comment #1)
> > We have to handle some effects:
> > - rumbler
> > - force feedback
> 
> I assume these are not input events, but triggers the user 
> sent to the InputDevice itself.
> Hence InputDevice / PointerDevice may require methods like:
>   doRumble(int ms, float strength/frequency)
>   ..
> or something ?

You're right, haptic effects should be treated in the device itself.
Comment 8 Julien Gouesse 2013-08-11 13:47:10 CEST
(In reply to comment #0)
> Currently PointerEvent is still called MouseEvent, rename it!
> 
On one hand, what will we do with the interfaces dealing with MouseEvent instances? MouseListener handles the callbacks mainly useful for the mice. Should we create another listener for the callbacks used by the gamepads?

On the other hand, MouseEvent.getButton() returns a short, this isn't enough for gamepads, the object representing the device should have a method that uses a short in input and that provides some information about the button in output. Maybe some other kinds of "components" (axis, POV hat, ...) will require some complementary information.
Comment 9 Julien Gouesse 2013-08-11 14:15:48 CEST
Other kinds of devices have to be considered (like it is already the case in JInput):
- fingerstick
- headtracker
- rudder
- stick (traditional joystick, like the one designed for the Apple 2E)
- trackball
- trackpad (?)
- wheel (a steering wheel, not a mouse wheel, see net.java.games.input.Controller)

Should the port type (USB, parallel, ...) be exposed in the API? The port number will have to be exposed. The device should indicate which effects are supported (rumbler(s)?).

Look at net.java.games.input.Component to have an idea of which kind of information have to be returned to identify a "component" of a gamepad.
Comment 10 Julien Gouesse 2013-11-07 22:27:32 CET
mousePressed and mouseReleased should be renamed, why not calling them buttonPressed and buttonReleased?

Instead of creating tons of classes to handle events, PointerEvent can fit into most of our needs except when adding or removing a "pointer device".

PointerDevice can contain a mapping between short button codes and real "components". The value of a component can be relative or absolute.

If every new move can be managed as an axis move, PointerEvent.pointerMoved() is enough for point-of-view controls, sliders and axis. getTranslation() or getMove() would be enough to indicate the moves along 3 axis. A PointerEvent must contain some information about the concerned device, not just the pointer type, the pointer identifier (Pointer-ID) should allow to retrieve some complementary information. PointerEvent.PovDirection could represent the 9 possible directions of a point-of-view control.

By default, multiple mice are seen as a single one by the operating system and the data are expressed in a resolution dependent way. How should we allow the developer to switch to the resolution independent raw data? Should each mouse be exposed as 2 pointer devices?
Comment 11 Julien Gouesse 2013-11-07 22:28:27 CET
Actually, pointerPressed and pointerReleased might be better than buttonPressed and buttonReleased.
Comment 12 Julien Gouesse 2013-11-08 15:05:25 CET
At first, maybe getMove() should be replaced by getD or getDX + getDY + getDZ to avoid any confusion as those names are already used in some other APIs and it clearly indicates that the data are relative (D -> delta).

The mouse data of the poller should be considered as relative only when the cursor is invisible (as it is done in some APIs based on JInput). The poller should store the previous fetched data.

- visible cursor:
  - The new delta is the difference between the current absolute coordinates and the previous unclipped absolute coordinates.
  - The new unclipped absolute coordinates and the new (potentially clipped) absolute coordinates are equal to the current absolute coordinates.

- invisible cursor
  - The new delta is the sum between the current relative coordinates and the previous delta.
  - The new unclipped absolute coordinates and the new (potentially clipped) absolute coordinates are equal to the sum between the current relative coordinates and themselves.

The absolute clipped coordinates have to be clipped later. We have to take care of the overflow too.

Then, the mouse (PointerDevice) can expose both its relative data and its absolute data without appearing twice in the list of devices.

Should the pollers be exposed in the public API too?

N.B: The absolute data depend on the display, monitor or screen. Another conversion might be necessary.
Comment 13 Julien Gouesse 2014-10-20 20:02:32 CEST
This API could be a nice source of inspiration to manipulate USB HID:
https://github.com/nyholku/purejavahidapi
Comment 14 Julien Gouesse 2014-10-21 16:18:42 CEST
We'll have to decide whether using udev (part of systemd) or evdev (inside the GNU Linux kernel) later.
Comment 15 Julien Gouesse 2014-10-21 16:37:52 CEST
Raw Input API will be used under Windows and probably IOKit under Mac OS X. It was already the case in JInput.
Comment 16 Julien Gouesse 2020-04-22 14:35:10 CEST
Actually, I think that I should use libinput for this purpose, it has a udev and a path backend, it works both with Wayland and X11, it provides:
- Keyboard events
- Pointer events
- Touch events
- Gesture events
- Tablet events
- Tablet pad events
- Switch events

A risk of using several different libraries at the same time to manage input events is to provide inconsistent data. I think that it's a safer bet to use the same library to manage the mouse and keyboard events that we support in NEWT from the very beginning and the other events.
Comment 17 Julien Gouesse 2020-04-27 19:43:50 CEST
LibInput provides relative pointer motion events with libinput_event_pointer_get_dx_unaccelerated() and libinput_event_pointer_get_dy_unaccelerated(), it helps to get the absolute and relative pointer motion events from the same API in the software stack that manages the input events. If I succeed in implementing a NEWT backend using it, it will allow to provide relative pointer motion events to some softwares in which providing the absolute pointer motion events doesn't really make sense, typically fullscreen games with no pointer icon, mostly first person shooters. In my humble opinion, computing absolute coordinates from relative coordinates is LibInput's role, using it to avoid having to do it by ourselves seems to be a good idea.

LibInput supports both X11 and Wayland but I have no system to test the latter yet.

I haven't found a way of obtaining any information through LibInput about gamepads whereas a quick test with udevadm showed me that udev sees my gamepad as expected. If I can't use LibInput to manage gamepads, I'll have to use either evdev (with libevdev or directly with ioctl like JInput) and/or udev as I initially planned.
Comment 18 Julien Gouesse 2023-04-20 13:25:58 CEST
I'll try this:
https://github.com/git-moss/purejavahidapi