uid |
---|
Uno.Features.UserInputs |
User inputs are usually propagated using RoutedEvents
. See Uno's routed events documentation to better understand their implementation on Uno.
Routed Event | Android | iOS | Wasm | macOS | Skia WPF | Skia GtK | Tizen | |
---|---|---|---|---|---|---|---|---|
focus events | ||||||||
GotFocus |
Yes | Yes (1) | Yes (1) | ? | Yes | Yes | Yes | Documentation |
LostFocus |
Yes | Yes (1) | Yes (1) | ? | Yes | Yes | Yes | Documentation |
keyboard events | ||||||||
KeyDown |
Hardware Only (2) | Yes (2) | Yes | Yes | Yes | Yes | Yes | Documentation |
KeyUp |
Hardware Only (2) | Yes (2) | Yes | Yes | Yes | Yes | Yes | Documentation |
pointer events | ||||||||
PointerCanceled |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
PointerCaptureLost |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
PointerEntered |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
PointerExited |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
PointerMoved |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
PointerPressed |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
PointerReleased |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
PointerWheelChanged |
No | No | Yes | Yes | Yes | Yes | No | Documentation |
manipulation events | ||||||||
ManipulationStarting |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
ManipulationStarted |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
ManipulationDelta |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
ManipulationInertiaStarting |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
ManipulationCompleted |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
gesture events | ||||||||
Tapped |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
DoubleTapped |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
RightTapped |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
Holding |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
drag and drop | ||||||||
DragStarting |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
DragEnter |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
DragOver |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
DragLeave |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
Drop |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
DropCompleted |
Yes | Yes | Yes | Yes | Yes | Yes | Yes | Documentation |
Notes:
- Focus events:
- iOS: The concept of focus is emulated because it's not supported by the platform, so this event is always bubbling in managed code.
- Wasm: Current implementation is not totally reliable and doesn't support lost focus most of the time.
- Keyboard events:
- Android:
KeyDown
andKeyUp
events are generated only from hardware keyboards (Except for the Editor Action on soft keyboards, those are translated asKeyUp
withKeyCode.Enter
). Some soft keyboard MAY generate those events, but your code shouldn't rely on that. This is a limitation in the Android platform (see note on this link content).Because of those limitations, Key Events are not being implemented as routed events on Android, so
AddHandler
&RemoveHandler
won't work for keyboard events. They won't bubble in managed code. - iOS:
KeyDown
&KeyUp
routed events are generated from only aTextBox
. Only character-related keyboard events are generated. They are implemented as Routed Events and they are always bubbling in managed code. - Skia: Keyboard events are supported from
CoreWindow.KeyUp
andCoreWindow.KeyDown
events, as well asUIElement.KeyUp
andUIElement.KeyDown
events for GTK, WPF and Tizen.
- Android:
These events are the base for all other pointing device related events (i.e. Manipulation, Gesture and drag and dop events). They are directly linked to the native events of each platform:
Touches[Began|Moved|Ended|Cancelled]
on iOSdispatchTouchEvent
anddispatchGenericMotionEvent
on Androidpointer[enter|leave|down|up|move|cancel]
on WebAssembly
On Skia however, they are fully managed events.
Like on WinUI as soon as the system detects that the user wants to scroll, a control gets a PointerCancelled
and that control won't receive
any other pointer event until the user releases the pointer. That behavior can be prevented by setting the ManipulationMode
to something else than System
on a control nested in the ScrollViewer
. (cf. Manipulation events)
Be aware that on iOS this will set DelaysContentTouches
to false
so it means that it will slightly reduce the performance
of the scrolling (cf. documentation).
As those events are tightly coupled to the native events, Uno has to make some compromises:
- On iOS, when tapping with a mouse or a pen on Android, or in few other specific cases (like
PointerCaptureLost
), multiple managed events are raised from a single native event. These have multiple effects:- On UWP if you have a control A and a nested control B, you will get:
but with UNO you will get:
B.PointerEnter A.PointerEnter B.PointerPressed A.PointerPressed
B.PointerEnter B.PointerPressed A.PointerEnter A.PointerPressed
- If you handle the
PointerEnter
on B, the parent control A won't get thePointerEnter
(as expected) nor thePointerPressed
.
- On UWP if you have a control A and a nested control B, you will get:
- On Android with a mouse or a pen, the
PointerEnter
andPointerExit
are going to be raised without taking clipping in consideration. This means that you will get the enter earlier and the exit later than on other platform. - On Android if you have an element with a
RenderTransform
which overlaps one of its sibling element, the element at the top will get the pointer events. - On WASM, iOS and Android, the
RoutedPointerEventArgs.FrameId
will be reset to 0 after 49 days of running time of the application. - Unlike on UWP, controls that are under a
Popup
won't receive the unhandled pointer events. - On non Skia-based platforms, unlike UWP, it's impossible to receive a
PointerReleased
without getting aPointerPressed
before. (For instance if a child control handled the pressed event but not the released event).On WASM as
TextElement
inherits fromUIElement
, it means that unlike UWPTextBlock
won't raise thePointerReleased
event when clicking on aHyperlink
. - Unlike UWP, on the
Hyperlink
theClick
will be raised before thePointerReleased
. - The property
PointerPointProperties.PointerUpdateKind
is not set on Android 5.x and lower (API level < 23) - On Firefox, pressed pointers are reported as fingers. This means you will receive events with
PointerDeviceType == Pen
only for hovering (i.e.Pointer<Enter|Move|Exit>
- note that, as of 2019-11-28, once pressedPointerMove
will be flagged as "touch") and you won't be able to track the barrel button nor the eraser. (cf. https://bugzilla.mozilla.org/show_bug.cgi?id=1449660) - On WASM, if you touch the screen with the pen then you press the barrel button (still while touching the screen), the pointer events will
have the
IsRightButtonPressed
set (in addition of theIsBarrelButtonPressed
). On WinUI and Android you get this flag only if the barrel button was pressed at the moment where you touched the screen, otherwise you will have theIsLeftButtonPressed
and theIsBarrelButtonPressed
. - For pen and fingers, the
Holding
event is not raised after a given delay like on WinUI, but instead we rely on the fact that we usually get a lot of moves for those kind of pointers, so we raise the event only when we get a move that exceed defined thresholds for holding. - On WASM, Shapes must have a non null Fill to receive pointer events (setting the Stroke is not sufficient).
- On WASM if the user scrolls in diagonal (e.g. with a Touchpad), but you mark as
Handled
pointer events only for vertical scrolling, then the events for the horizontal scroll component won't bubble through the parents.
The capture of pointer is handled in managed code only. On WebAssembly Uno however still requests the browser to capture the pointer,
but Uno does not rely on native [got|lost]pointercapture
events.
To differentiate between mouse and touch device type for pointer events, include the following in your app's Info.plist
:
<key>UIApplicationSupportsIndirectInputEvents</key>
<true/>
Without this key the current version of iPadOS reports mouse interaction as normal touch.
They are generated from the PointerXXX events (using the Windows.UI.Input.GestureRecognizer
) and are bubbling in managed only.
They are generated from the PointerXXX events (using the Windows.UI.Input.GestureRecognizer
) and are bubbling in managed only.
Note that Tapped
and DoubleTapped
are not linked in any way to a native equivalent, but are fully interpreted in managed code.
In order to match the WinUI behavior, on WASM the default "Context menu" of the browser is disabled (except for the TextBox
),
no matter if you use / handle the RightTapped
event or not.
Be aware that on some browser (Firefox), user can still request to get the "Context menu" on right click.
While the browser context menu enabled on TextBox
and PasswordBox
by default, it will be disabled when ContextFlyout
is set on the control.
To manually disable the context menu on a UIElement
which represents a HTML <input>
, you can manually set the context-menu-disabled
CSS class:
#if __WASM__
MyInputElement.SetCssClasses("context-menu-disabled");
#endif
Those events are also 100% managed events, built from the PointerXXX events (using the Windows.UI.Input.GestureRecognizer
)
A drag and drop operation can be used to move content within an app, but it can also be used to copy / move / link between apps. While intra-app drag and drop is supported on all platforms without limitations, inter-app drag and drop requires platform specific support. The table and sections below describe supported functionality and limitations for each platform.
From uno app to external | From external app to uno | |
---|---|---|
Android | No | No |
iOS | No | No |
Wasm | No | Yes (Text, Link, Image, File, Html, Rtf) |
macOS | Yes (Text, Link, Image, Html, Rtf) | Yes (Text, Link, Image, File, Html, Rtf) |
Skia WPF | Yes (Text, Link, Image, File, Html, Rtf) | Yes (Text, Link, Image, File, Html, Rtf) |
Skia GTK | No | No |
- "Link" may refer to WebLink, ApplicationLink or Uri formats
- When dragging content from external app to uno, you cannot retrieve the content from the
DataPackage
before theDrop
event. This a limitations of web browsers. Any attempt to read it before theDrop
will result into a timeout exception after a hard coded delay of 10 seconds. - When dragging some uris from external app to uno, only the first uri will be accessible through the WebLink standard format ID.
- Dragging a File (StorageItem) from an Uno Platform App to an external destination is not currently supported.
- When receiving a drop within an Uno Platform App from an external source, key modifiers are not supported
- There is no standard type for WebLink (nor ApplicationLink) on this platform. They are copied to the external app as raw Text, and converted back as WebLink or ApplicationLink) from raw text from the external app when `Uri.IsWellFormedUriString(text, UriKind.Absolute) returns true.
- The image content seems to not being readable by common apps, only another Uno app is able to read it properly.
UWP has the following standard data formats that correspond with a URI/URL:
- Uri, now deprecated in favor of:
- WebLink and
- ApplicationLink
Several platforms such as macOS/iOS/Android do not differentiate between them and only use a single URI/URL class or string.
When applying data to the native clipboard or drag/drop data from a DataPackage, only one URI/URL may be used. Therefore, all URI data formats are merged together into a single URI using the above defined priority. WebLink is considered more specific than ApplicationLink.
When pulling data from the native clipboard or drag/drop data, this single native URI/URL is separated into the equivalent UWP data format (since UWP's direct equivalent standard data format 'Uri' is deprecated). The mapping is as follows:
- WebLink is used if the given URL/URI has a scheme of "http" or "https"
- ApplicationLink is used if not #1
For full compatibility, the Uri format within a DataPackage should still be populated regardless of #1 or #2.
- If you have 2 nested drop targets (i.e. element flagged with
AllowDrop = true
), when the pointer leaves the deepest / top most element but not the parent, the parent element will also raiseDragLeave
and immediately after raiseDragEnter
. - On UWP, the default UI will include a tooltip which indicates the accepted drop action, and a "screenshot" of the dragged element. Currently Uno will display only the tooltip.
- The accepted drop action displayed in the tooltip is not localized.