summaryrefslogtreecommitdiff
path: root/FL/core/pen_events.H
blob: 48ce470348918fa269d77d5fbb13f7317e7878f7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
//
// Pen event header file for the Fast Light Tool Kit (FLTK).
//
// Copyright 2025 by Bill Spitzak and others.
//
// This library is free software. Distribution and use rights are outlined in
// the file "COPYING" which should have been included with this file.  If this
// file is missing or damaged, see the license at:
//
//     https://www.fltk.org/COPYING.php
//
// Please see the following page on how to report bugs and issues:
//
//     https://www.fltk.org/bugs.php
//

/**
 \file FL/core/pen_events.H
 \brief Pen event handling variables and functions.
*/

#ifndef Fl_core_pen_events_H
#define Fl_core_pen_events_H

#include <FL/fl_config.h>             // build configuration
#include <FL/Fl_Export.H>             // for FL_EXPORT
#include <FL/core/function_types.H>   // widget callbacks and services

#include <cstdint>

class Fl_Widget;

namespace Fl {

/** FLTK Pen/Stylus/Tablet input driver API. */
namespace Pen {

/**
 \defgroup fl_pen_events Pen and tablet event handling
 \ingroup fl_events
 \brief This chapter documents the Fl::Pen namespace API, declared in <FL/core/pen_events.H>

 The FL::Pen namespace contains everything needed to work with a pen type input
 device, either in connection with an external tablet, or as a stylus for
 drawing directly onto a screen.

 To receive pen input, call Fl::Pen::subscribe() for one or more widgets. The
 widget will receive a Fl::Pen::ENTER event when the stylus enters the widget
 area. By returning 1 to Fl::Pen::ENTER, all further pen events are sent to
 this widget, and no mouse events are generated until Fl::Pen::LEAVE.

 Returning 0 Fl::Pen::ENTER tells FLTK to suppress further pen events until
 Fl::Pen::LEAVE, and convert them into mouse events instead.

 Pen events also set Fl::event_x(), Fl::event_y(), Fl::event_x_root(),
 Fl::event_y_root(), Fl::event_is_click(), and Fl::event_clicks().

 @{
 */

/**
 \brief Bitfield of traits.
 This is used in Fl::Pen::driver_traits() and Fl::Pen::pen_traits().
 */
enum class Trait : uint32_t {
  /// No bits set
  NONE              = 0x0000,
  /// Set if FLTK supports tablets and pens on this platform
  DRIVER_AVAILABLE  = 0x0001,
  /// Set after the system detected a pen, stylus, or tablet. This bit may not be
  /// set until a pen is brought into proximity of the tablet.
  DETECTED          = 0x0002,
  /// If set, this is a digitizer for a display; if clear, this is a standalone tablet
  DISPLAY           = 0x0004,
  /// Driver provides different device IDs for different pens
  PEN_ID            = 0x0008,
  /// Pen may have an eraser tip
  ERASER            = 0x0010,
  /// Pen returns a pressure value
  PRESSURE          = 0x0020,
  /// Pen returns a barrel pressure value (tangential pressure)
  BARREL_PRESSURE   = 0x0040,
  /// Pen returns tilt in X direction
  TILT_X            = 0x0080,
  /// Pen returns tilt in Y direction
  TILT_Y            = 0x0100,
  /// Pen returns a twist value
  TWIST             = 0x0200,
  /// Pen returns a proximity value
  PROXIMITY         = 0x0400,
};

/**
 \brief Bitwise OR operator for Trait enum.
 \param lhs Left-hand side trait flags
 \param rhs Right-hand side trait flags
 \return Combined trait flags
 */
inline constexpr Trait operator|(Trait lhs, Trait rhs) {
  return static_cast<Trait>(static_cast<uint32_t>(lhs) | static_cast<uint32_t>(rhs));
}

/**
 \brief Bitwise AND operator for Trait enum.
 \param lhs Left-hand side trait flags
 \param rhs Right-hand side trait flags
 \return Intersection of trait flags
 */
inline constexpr Trait operator&(Trait lhs, Trait rhs) {
  return static_cast<Trait>(static_cast<uint32_t>(lhs) & static_cast<uint32_t>(rhs));
}

/**
 \brief Bitwise OR assignment operator for Trait enum.
 \param lhs Left-hand side trait flags (modified in place)
 \param rhs Right-hand side trait flags
 \return Reference to modified lhs
 */
inline Trait& operator|=(Trait& lhs, Trait rhs) {
  lhs = lhs | rhs;
  return lhs;
}


/**
 \brief Bitfield of pen state flags.
 \see event_state(), event_trigger()
 */
enum class State : uint32_t {
  /// No button pressed
  NONE              = 0x0000,
  /// The tip hovers over the surface but does not touch it
  TIP_HOVERS        = 0x0001,
  /// The tip touches the surface
  TIP_DOWN          = 0x0002,
  /// The eraser hovers over the surface but does not touch it
  ERASER_HOVERS     = 0x0004,
  /// The eraser touches the surface
  ERASER_DOWN       = 0x0008,
  /// Barrel button 0, usually the lower button on a pen, is pressed
  BUTTON0           = 0x0100,
  /// Barrel button 1, usually the upper button on a pen, is pressed
  BUTTON1           = 0x0200,
  /// Barrel button 2 is pressed
  BUTTON2           = 0x0400,
  /// Barrel button 3 is pressed
  BUTTON3           = 0x0800,
  /// Mask for all buttons, tip, and eraser down
  ANY_DOWN          = BUTTON0 | BUTTON1 | BUTTON2 | BUTTON3 | TIP_DOWN | ERASER_DOWN,
};

/**
 \brief Bitwise OR operator for State enum.
 \param lhs Left-hand side state flags
 \param rhs Right-hand side state flags
 \return Combined state flags
 */
inline constexpr State operator|(State lhs, State rhs) {
  return static_cast<State>(static_cast<uint32_t>(lhs) | static_cast<uint32_t>(rhs));
}

/**
 \brief Bitwise AND operator for State enum.
 \param lhs Left-hand side state flags
 \param rhs Right-hand side state flags
 \return Intersection of state flags
 */
inline constexpr State operator&(State lhs, State rhs) {
  return static_cast<State>(static_cast<uint32_t>(lhs) & static_cast<uint32_t>(rhs));
}

/**
 \brief Bitwise OR assignment operator for State enum.
 \param lhs Left-hand side state flags (modified in place)
 \param rhs Right-hand side state flags
 \return Reference to modified lhs
 */
inline State& operator|=(State& lhs, State rhs) {
  lhs = lhs | rhs;
  return lhs;
}


/**
 \brief List of pen events.
 These events extend the standard Fl_Event enumeration.
 \see enum Fl_Event
 */
enum Event {
  /**
   Pen entered the proximity of the tablet with a new pen.
   */
  DETECTED = 0x1000,

  /**
   Pen entered the proximity of the tablet with a known, but changed pen.
   User changed to a different pen (event_id() > 0) or the pen or tablet
   was disconnected (event_id() == -1). Pen IDs, if supported, are assigned by
   the tablet manufacturer.
   */
  CHANGED,

  /**
   Pen entered the proximity of the tablet with a known pen.
   */
  IN_RANGE,

  /**
   Pen left the proximity of the tablet.
   */
  OUT_OF_RANGE,

  /**
   Pen entered the widget area, either by moving in x/y, or by
   a proximity change (pen gets closer to the surface).
   event_trigger() returns 0, TIP_HOVERS, or ERASER_HOVERS.
   */
  ENTER,

  /**
   If no button is pressed, indicates that the pen left the widget area.
   While any pen button is held down, or the pen touches the surface,
   Fl::pushed() is set, and the pushed widgets receives DRAG events, even
   if the pen leaves the widget area. If all buttons are released outside the
   widget area, a LEAVE event is sent as well as LIFT or BUTTON_RELEASE.
   */
  LEAVE,

  /**
   Pen went from hovering to touching the surface.
   event_trigger() returns TIP_DOWN or ERASER_DOWN.
   */
  TOUCH,

  /**
   Pen went from touching to hovering over the surface.
   event_trigger() returns TIP_HOVERS or ERASER_HOVERS.
  */
  LIFT,

  /** Pen moved without touching the surface and no button is pressed. */
  HOVER,

  /** Pen moved while touching the surface, or any button is pressed. */
  DRAW,

  /**
   A pen button was pushed.
   event_trigger() returns BUTTON0, BUTTON1, BUTTON2, or BUTTON3.
   */
  BUTTON_PUSH,

  /**
   A pen button was released.
   event_trigger() returns BUTTON0, BUTTON1, BUTTON2, or BUTTON3.
   */
  BUTTON_RELEASE

};

/**
 \brief Query the traits supported by the pen/tablet driver.

 This function returns a bitfield of traits that are supported by the FLTK driver
 for this platform. If a trait is not supported, the corresponding event value
 will not return a useful value. Note that even if the FLTK driver support a
 trait, the underlying pen device or driver may not. Fl::Pen will return a
 known default for those event values.

 The bitfield returned is static.

 \return a bitfield of supported traits
 \see pen_traits()
 */
FL_EXPORT extern Trait driver_traits();

/**
 \brief Return true if the corresponding bit is set in the driver traits.
 \param[in] bits check for one or more trait bits
 \return true if any bit is set
 */
inline bool driver_traits(Trait bits) {
  return ((driver_traits() & bits) != Trait::NONE);
}

/**
 \brief Query traits of the current pen or stylus.
 The value returned by this function may change when pens change or when more
 information becomes known about the currently used pen.
 \param[in] pen_id a know pen ID as returned from event_pen_id(),
    or 0 for the current pen
 \return a bitfield of supported traits
 */
FL_EXPORT extern Trait pen_traits(int pen_id = 0);

/**
 \brief Return true if the corresponding bit is set in the pen traits.
 \param[in] bits check for one or more trait bits
 \param[in] pen_id a know pen ID as returned from event_pen_id(),
  or 0 for the current pen
 \return true if any bit is set
 */
inline bool pen_traits(Trait bits, int pen_id = 0) {
  return ((pen_traits() & bits) != Trait::NONE);
}

/**
 \brief Receives a Pen::ENTER event when the pen enters this widget.
 Multiple widgets may subscribe to pen events. Additional subscription
 requests for the same widget are ignored.
 \param widget The widget subscribing to pen events.
 */
FL_EXPORT extern void subscribe(Fl_Widget* widget);

/**
 \brief Stop receiving Pen::ENTER for this widget.
 Deleting a widget will automatically unsubscribe it.
 \param widget Widget to unsubscribe from pen events
 */
FL_EXPORT extern void unsubscribe(Fl_Widget* widget);

/**
 Clear the "pushed" state and forward pen events as mouse events.
 Call this if another window is popped up during pen event handling, so
 mouse event handling can resume normal.
 */
FL_EXPORT extern void release();

/// \name Query values during event handling
/// @{

/**
 \brief Returns the pen x and y position inside the handling widget as doubles.
 These functions provide high-precision pen coordinates relative to the widget
 that received the pen event. For integer coordinates, use Fl::event_x() and
 Fl::event_y() instead.
 \return Pen position as floating-point coordinate, defaults to 0.0
 \see Fl::event_x(), Fl::event_y()
 */
FL_EXPORT extern double event_x();
/** \brief Returns pen Y coordinate in widget space, see event_x(). */
FL_EXPORT extern double event_y();

/**
 \brief Returns the pen x and y position in global coordinates as doubles.
 For integer coordinates, use Fl::event_x_root() and Fl::event_y_root().
 \return Pen position as floating-point coordinate in screen space, defaults to 0.0
 \see Fl::event_x_root(), Fl::event_y_root()
 */
FL_EXPORT extern double event_x_root();
/** \brief Returns pen Y coordinate in screen space, see event_x_root(). */
FL_EXPORT extern double event_y_root();

/**
 \brief Returns the ID of the pen used in the last event.
 \return Unique pen identifier, or -1 if pen was removed, defaults to 0
 \see Trait::PEN_ID
 */
FL_EXPORT extern int event_pen_id();

/**
 \brief Returns the pressure between the tip or eraser and the surface.
 \return pressure value from 0.0 (no pressure) to 1.0 (maximum pressure),
      defaults to 1.0.
 \see Trait::PRESSURE
 */
FL_EXPORT extern double event_pressure();

/**
 \brief Returns barrel pressure or tangential pressure.
 \return Pressure value from -1.0 to 1.0 , defaults to 0.0 .
 \see Trait::BARREL_PRESSURE
 */
FL_EXPORT extern double event_barrel_pressure();

/**
 \brief Returns the tilt of the pen in the x and y directions between -1 and 1.

 X-axis tilt returns -1.0 when the pen tilts all the way to the left, 0.0 when
 it is perfectly vertical, and 1.0 all the way to the right. Most pens seem to
 return a maximum range of -0.7 to 0.7.

 Y-axis tilt returns -1.0 when the pen tilts away from the user, and 1.0 when
 it tilts toward the user.

 \return Tilt value from -1.0 to 1.0, defaults to 0.0
 \see Trait::TILT_X, Trait::TILT_Y
 */
FL_EXPORT extern double event_tilt_x();
/** \brief Returns pen Y-axis tilt, see event_tilt_x() */
FL_EXPORT extern double event_tilt_y();

/**
 \brief Returns the pen's axial rotation in degrees.
 The twist angle is reported in degrees, with:
  - 0° : pen’s “top” (the button side) facing upward
  - +180° : twisted halfway clockwise, looking at the end of the pen toward the tip
  - –180° → twisted half way counter clockwise
 So the full range is usually –180° to +180°.
 \return Twist angle in degrees, defaults to 0.0.
 \see Trait::TWIST
 */
FL_EXPORT extern double event_twist();

/**
 \brief Returns the proximity of the pen to the surface between 0 and 1.
 A proximity of 0 is closest to the surface, 1 is farthest away.
 \return Proximity value from 0.0 (touching) to 1.0 (far away), defaults to 0.0 .
 \see Trait::PROXIMITY
 */
FL_EXPORT extern double event_proximity();

/**
 \brief Returns the state of the various buttons and tips.
 \return Current state flags (combination of State values)
 */
FL_EXPORT extern State event_state();

/**
 \brief Return true if any of the bits are set in the event state.
 \param[in] bits check for one or more event state bits
 \return true if any bit is set
 */
inline bool event_state(State bits) {
  return ((event_state() & bits) != State::NONE);
}

/**
 \brief Returns the state change that triggered the event.
 \return a state with one bit set for the action that triggered this event
 */
FL_EXPORT extern State event_trigger();

/** @} */ // group fl_pen_events


} // namespace Pen

} // namespace Fl


/*
 Resources:

 Windows:
 1. Legacy WinTab API (Win2k), Wintab32.dll, wintab.h
    https://developer.wacom.com/en-us/developer-dashboard/downloads
 2. Windows Ink API (Modern, Win10), Windows.UI.Input.Inking (WinRT API), InkCanvas(), etc.
    https://learn.microsoft.com/windows/uwp/design/input/windows-ink
 3. Pointer Input / WM_POINTER API (Win8), WM_POINTERUPDATE, GetPointerPenInfo
    https://learn.microsoft.com/windows/win32/inputmsg/wm-pointerupdate
      return WTInfo(0, 0, NULL) > 0; // Wintab check

 Linux:
 1. Low-level: evdev, /dev/input/event*, libevdev,
    https://www.kernel.org/doc/html/latest/input/event-codes.html
 2. Mid-level: XInput2 (for X11), XI_Motion, XI_ButtonPress
    https://www.x.org/releases/current/doc/inputproto/XI2proto.txt
    https://www.freedesktop.org/wiki/Software/libevdev/
 3. Mid-level: Wayland tablet protocol, tablet-v2 protocol,
    zwp_tablet_tool_v2_listener, zwp_tablet_v2, zwp_tablet_seat_v2
    https://wayland.app/protocols/tablet-v2

 SDL3:
    https://github.com/libsdl-org/SDL/blob/main/include/SDL3/SDL_pen.h
    https://wiki.libsdl.org/SDL3/CategoryPen
 */


#endif // !Fl_core_pen_events_H