Author: Daniel Libby
Achieving low latency is critical for delivering great inking experiences on the Web. Ink on the Web is generally produced by consuming PointerEvents and rendering strokes to the application view, whether that be 2D or WebGL canvas, or less commonly, SVG or even HTML.
There are a number of progressive enhancements to this programming model that are aimed at reducing latency.
-
Offscreen canvas and Input for Workers
This helps web developers separate their pointer handling from other main thread logic and execution, which allows for more timely and consistent delivery of these input events.
-
PointerEvent getPredictedPoints()
This API returns some number of points that are predicted to occur in the future. Rendering accurately predicted points reduces the user-perceived latency. This comes with some tradeoffs in determining when and how far to predict, as inaccurate prediction may lead to artifacts.
-
This allows canvas views to be decoupled from the composition of the rest of the HTML content. Additionally, this allows User Agents to reduce latency by bypassing the operating system's compositor. This is achieved by using a hardware overlay in order to present the contents of the canvas.
-
Given a desynchronized canvas, rendering multiple pointermove events per frame can result in an input event dispatched in the middle of a frame to show up on the screen one frame earlier than it otherwise would have. Without this pointer updates are aligned to the beginning of the frame (see Aligning Input Events).
Desynchronized canvas is subject to hardware limitations that may not consistently provide latency improvements that applications are depending on for great inking experiences.
-
No alpha blending of hardware overlays. On Windows a hardware overlay is not able to blend with other composed content beneath it. Due to this, scenarios like inking on top of a document will not benefit from the latency improvements associated with hardware overlays.
-
Orientation matching requirements. If the device is not in the default/primary orientation the backing buffer must be created with the same orientation in order to be eligible for an hardware overlay. Doing so for 90 degree rotation means the width and height of the buffer must be swapped and the rendered content must be transformed at some layer (either app or User Agent) before it reaches the screen.
There are typically two types of representation of an ink stroke: 'wet ink', rendered while the pen is in contact with the screen, and 'dry ink', which is rendered once the pen is lifted. For applications such as annotating documents, wet ink is generally rendered segment-by-segment via canvas, but it is desirable for dry ink to become part of the document's view.
Desynchronized canvas is inherently unable to synchronize with other HTML content which makes the process of drying ink difficult to impossible to implement without some visual artifacts. When the pen is lifted the application will stop drawing the stroke from the canvas and 'dry' the stroke into the document view (e.g. as SVG in HTML). When desynchronized canvas is used in this scenario, there is no guarantee that the dried content shows up in the same frame as the wet stoke is erased. This may end up producing one frame with no ink content, or one frame where both wet and dry ink are visible, which results in a visual artifact of a flash of darker for non-opaque strokes.
Operating system compositors typically introduce a frame of latency in order to compose all of the windows together. During this frame of latency, input may be delivered to an application, but that input has no chance of being displayed to the user until the next frame that the system composes, due to this pipelining. System compositors may have the capability to provide an earlier rendering of this input on behalf of the application. We propose exposing this functionality to the Web so that web applications can achieve latency parity with native applications on supported systems. This would also be a progressive enhancement in the same vein as others covered previously.
In order for the system to be able to draw the subsequent points with enough fidelity that the user does not see the difference, the application needs to describe the last rendered point with sufficient details. If the system knows the last rendered point, it can produce the segments of the ink trail for input that has been delievered, but not yet rendered (or at least has not hit the end of the rendering pipeline).
An app renders complex ink using delivered Pointer events, while user continues interaction and OS is working on delivering input to an app.
OS can render ink stroke based on incoming user input using last rendered point information and stroke styles set by an app, at the same time as delivering input to an app.
As Pointer events gets delivered to an app, application continues rendering ink, seamlessly replacing OS ink with application rendered strokes.
- Provide a progressive enhancement over other latency improvements.
- Allow web applications to improve latency without delegating full input / output stack to the OS.
- Provide a compact API surface to connect the last few stroke segments with the OS rendering of the incoming input.
- Co-exist with desynchronized canvas — one of the purposes of desynchronized canvas is to bypass the system compositor.
- Co-exist with input prediction — since the operating system will be rendering points before the application sees them, the application should no longer perform prediction, as doing so may result in a visual 'fork' between the application's rendering and the system's rendering.
- Take over all ink rendering on top of the application — this is a progressive enhancement for the "last few pixels" ink rendering, it is not a replacement for a full I/O stack used to present ink in web applications.
- Provide rich presentation and styling options — in order to improve latency this proposal suggests to limit presentation option for the "last few pixels" to the bare minimum - color, (stroke) diameter, opacity. Applications like Microsoft Office support complex brushes (e.g. Galaxy brush) that should be rendered using existing code - and this proposal will be used in additive manner to add solid color "fluid" ink, that "dries" into complex presentation as input gets to the application.
const renderer = new InkRenderer();
const minExpectedImprovement = 8;
try {
let presenter = await navigator.ink.requestPresenter('delegated-ink-trail', canvas);
// With pointerraw events and javascript prediction, we can reduce latency
// by 16+ ms, so fallback if the InkPresenter is not capable enough to
// provide benefit
if (presenter.expectedImprovement < minExpectedImprovement)
throw new Error("Little to no expected improvement, falling back");
renderer.setPresenter(presenter);
window.addEventListener("pointermove", evt => {
renderer.renderInkPoint(evt);
});
} catch(e) {
// Ink presenter not available, use desynchronized canvas, prediction,
// and pointerrawmove instead
renderer.usePrediction = true;
renderer.desynchronized = true;
window.addEventListener("pointerrawmove", evt => {
renderer.renderInkPoint(evt);
});
}
class InkRenderer {
constructor() {}
renderInkPoint(evt) {
// Render segments for any coalesced events delivered, for best possible
// segment quality.
let events = evt.getCoalescedEvents();
events.forEach(event => {
this.renderStrokeSegment(event.x, event.y);
});
// Render the actual dispatched PointerEvent, and let the
// DelegatedInkTrailPresenter know about this rendering (along with
// style information of the stroke).
this.renderStrokeSegment(evt.x, evt.y);
if (this.presenter) {
this.presenterStyle = { color: "rgba(0, 0, 255, 0.5)", diameter: 4 * evt.pressure };
this.presenter.updateInkTrailStartPoint(evt, this.presenterStyle);
}
}
void setPresenter(presenter) {
this.presenter = presenter;
}
renderStrokeSegment(x, y) {
// application specific code to draw
// the stroke on 2d canvas for example
}
}
Proposed WebIDL
partial interface Navigator {
[SameObject] readonly attribute Ink ink;
};
interface Ink {
Promise<DelegatedInkTrailPresenter> requestPresenter(DOMString type, optional Element? presentationArea = null);
}
dictionary InkTrailStyle {
DOMString color;
unrestricted double diameter;
}
interface DelegatedInkTrailPresenter {
void updateInkTrailStartPoint(PointerEvent evt, InkTrailStyle style);
readonly attribute Element? presentationArea;
readonly attribute unsigned long expectedImprovement;
}
This proposal provides infrastructure for authors to request a DelegatedInkTrailPresenter
interface from User Agents that support it. It is intentionally designed so that in the future, a generic InkPresenter
interface could be introduced that DelegatedInkTrailPresenter
would inherit from, providing greater extensibility. requestPresenter
is made to be extensible via a parameter so that it can easily be extended to other types of ink presentation as well. For example, this could include fully delegated ink presentation, or a presenter than can handle complex tips via WebGL shaders.
The requestPresenter
method also takes in an optional presentationArea
parameter, which is used by the User Agent to limit the visible area where the provided InkPresenter will take effect. This is necessary to prevent ink presentation outside of the provided area. If this argument is not provided, this will default to the containing viewport. This area is the provided element's border-box in client coordinates, which does not require authors to recalculate if the element is moved or scrolled.
The DelegatedInkTrailPresenter
updateInkTrailStartPoint
method is the main addition of this proposal. Authors should use this method to indicate to the User Agent which PointerEvent was used as the last rendered point for the current frame. The PointerEvent passed to updateInkTrailStartPoint
must be a trusted event and should be the last point that was used by the application to render ink to the view. updateInkTrailStartPoint
also accepts all relevant properties of rendering the ink stroke so that the User Agent can closely match the ink rendered by the application. The trusted PointerEvent and style information are used together by the User Agent as the starting point for the delegated ink trail that will be rendered for the subsequent frame that is produced.
Note that if two or more pens are simultaneously drawing ink strokes and requesting delegated ink trails, whether in a single presentation area or multiple, a "last write wins" strategy is used to determine which stroke will receive the delegated ink trail for a particular frame. This will result in whichever pointerevent has a later timestamp receiving the delegated ink trail. This is expected to be a low usage scenario and therefore is acceptable to not produce delegated ink trails for all strokes, but this can be revisited in the future if it becomes a more common scenario.
updateInkTrailStartPoint
accepts an InkTrailStyle
dictionary to describe how the delegated ink trail should appear when produced by the User Agent. Initially it will accept color
and diameter
, where diameter
describes the width of the ink trail drawn by the User Agent in CSS pixels. It is made extensible so that in the future other properties could also be used to describe the trail, such as opacity or more complex brushes.
The presentationArea
attribute reflects the argument passed to requestPresenter
. Once an InkPresenter has been created this cannot be changed.
The expectedImprovement
attribute exists to provide site authors with information regarding the perceived latency improvements they can expect by using this API. The attribute will return the expected average number of milliseconds that perceived latency will be improved by using the API, including prediction.
We considered a few different locations for where the method updateInkTrailStartPoint
should live, but each had drawbacks:
-
On PointerEvent itself
This felt a bit too specific, as PointerEvent is not scoped to pen, and is not associated with ink today.
-
On a canvas rendering context
This could be sufficient, as a lot of ink is rendered on the Web via canvas, but it would exclude SVG renderers.
-
Put it somewhere on Element
This seemed a bit too generic and scoping to a new namespace seemed appropriate.