1. Introduction
This spec describes the CSS and JS mechanics of the single-page transition API.
2. Transitions as an enhancement
This section is non-normative.
A key part of this API design is the view that an animated transition is an enhancement to a DOM change.
If a transition cannot run, or is skipped, the DOM change should still happen. If the DOM change should also be skipped, then that should be handled by another feature. The signal
on NavigateEvent
is an example of a feature developers could use to handle this.
Although the transition API allows DOM changes to be asynchronous via the updateDOM
callback, the transition API is not responsible for queuing or otherwise scheduling the DOM changes, beyond the scheduling needed for the transition itself. Some asynchronous DOM changes can happen concurrently (e.g if they’re happening within independent components), whereas others need to queue, or abort an earlier change. This is best left to a feature or framework that has a more holistic view of the update.
3. CSS properties
3.1. page-transition-tag
Name: | page-transition-tag |
---|---|
Value: | none | <custom-ident> |
Initial: | none |
Applies to: | all elements |
Inherited: | no |
Percentages: | n/a |
Computed value: | as specified |
Canonical order: | per grammar |
Animation type: | discrete |
The page-transition-tag property "tags" an element as participating in a page transition.
- none
-
The element will not participate in a page transition.
- <custom-ident>
-
The element can participate in a page transition, as either an outgoing or incoming element, with a page transition tag equal to the <custom-ident>'s value.
The value none is invalid as a <custom-ident>.
The root element participates in a page transition by default using the following style in the user-agent origin.
html {
page-transition-tag : root;
}
4. Pseudo-elements
While the UA is animating a page transition, it creates the following page-transition pseudo-elements, to represent the various items being animated.
The ::page-transition pseudo-element acts as a grouping element for other page-transition pseudo-elements and has the document’s root element as its originating element.
For example, :root::page-transition selector matches this pseudo-element, but div::page-transition does not.
Other page-transition pseudo-elements take a <pt-tag-selector> argument to specify which elements tagged with page-transition-tag are affected.
There can be multiple pseudo-elements of the same type, one for each page-transition-tag participating in a transition.
The <pt-tag-selector> is defined as follows:
<pt-tag-selector> = '*' | <custom-ident>
A value of * makes the corresponding selector apply to all pseudo elements of the specified type. The specificity of a page-transition selector with a * argument is zero.
The <custom-ident> value makes the corresponding selector apply to exactly one pseudo element of the specified type, namely the pseudo-element that is created as a result of the page-transition-tag property on an element with the same <custom-ident> value. The specificity of a page-transition selector with a <custom-ident> argument is the same as for other pseudo-elements, and is equivalent to a type selector.
The following describes all of the page-transition pseudo-elements and their function:
- ::page-transition
-
This pseudo-element is the grouping container of all the other page-transition pseudo-elements. Its originating element is the document’s root element.
The following user-agent origin styles apply to this element:
html : :page-transition{ position : fixed; inset : 0 ; } Note: This pseudo-element provides a containing block for all ::page-transition-container pseudo-elements. The aim of the style is to size the pseudo-element to cover the large viewport size and position all ::page-transition-container pseudo-elements relative to the origin of the large viewport.
- ::page-transition-container( <pt-tag-selector> )
-
One of these pseudo-elements exists for each page-transition-tag in a page transition, and holds the rest of the pseudo-elements corresponding to this page-transition-tag.
Its originating element is the ::page-transition pseudo-element.
The following user-agent origin styles apply to this element:
html : :page-transition-container ( *) { position : absolute; top : 0 ; left : 0 ; animation-duration : 0.25 s ; animation-fill-mode : both; } Note: The aim of the style is to position the element relative to its ::page-transition parent.
In addition to above, styles in the user-agent origin animate this pseudo-element’s width and height from the size of the outgoing element’s border box to that of the incoming element’s border box. Also the element’s transform is animated from the outgoing element’s screen space transform to the incoming element’s screen space transform. This style is generated dynamically since the values of animated properties are determined at the time that the transition begins.
The selector for this and subsequently defined pseudo-elements is likely to change to indicate position in the pseudo-tree hierarchy.
- ::page-transition-image-wrapper( <pt-tag-selector> )
-
One of these pseudo-elements exists for each page-transition-tag being in a page transition, and holds the images of the outgoing and incoming elements.
Its originating element is the ::page-transition-container() pseudo-element with the same tag.
The following user-agent origin styles apply to this element:
html : :page-transition-image-wrapper ( *) { position : absolute; inset : 0 ; animation-duration : inherit; animation-fill-mode : inherit; } In addition to above, styles in the user-agent origin add ''isolation: isolate'' to this pseudo-element if it has both ::page-transition-incoming-image and ::page-transition-outgoing-image as descendants.
Note: The aim of the style is to position the element to occupy the same space as its ::page-transition-container element and provide isolation for blending.
Isolation is only necessary to get the right cross-fade between incoming and outgoing image pixels. Would it be simpler to always add it and try to optimize in the implementation?
- ::page-transition-outgoing-image( <pt-tag-selector> )
-
One of these pseudo-elements exists for each element in the outgoing DOM being animated by the page transition, and is a replaced element displaying the outgoing element’s snapshot image. It has natural dimensions equal to the snapshot’s size.
Its originating element is the ::page-transition-image-wrapper() pseudo-element with the same tag.
The following user-agent origin styles apply to this element:
html : :page-transition-outgoing-image ( *) { position : absolute; inset-block-start : 0 ; inline-size : 100 % ; block-size : auto; animation-duration : inherit; animation-fill-mode : inherit; } Note: The aim of the style is to match the element’s inline size while retaining the aspect ratio. It is also placed at the block start.
In addition to above, styles in the user-agent origin add mix-blend-mode:plus-lighter to this pseudo element if the ancestor ::page-transition-image-wrapper has both ::page-transition-incoming-image and ::page-transition-outgoing-image as descendants.
Note: mix-blend-mode value of plus-lighter ensures that the blending of identical pixels from the outgoing and incoming images results in the same color value as those pixels.
Additional user-agent origin styles added to animate these pseudo-elements are detailed in Animate a page transition.
- ::page-transition-incoming-image( <pt-tag-selector> )
-
Identical to ::page-transition-outgoing-image(), except it deals with the incoming element instead.
The precise tree structure, and in particular the order of sibling pseudo-elements, is defined in the Create transition pseudo-elements algorithm.
5. Concepts
5.1. Phases
Phases represent an ordered sequence of states. Since phases are ordered, prose can refer to phases before a particular phase, meaning they appear earlier in the sequence, or after a particular phase, meaning they appear later in the sequence.
The initial phase is the first item in the sequence.
5.2. The page-transition layer stacking layer
This specification introduces a stacking layer to the Elaborate description of Stacking Contexts.
The ::page-transition pseudo-element generates a new stacking context called page-transition layer with the following characteristics:
-
Its parent stacking context is the root stacking context.
-
If the page-transition pseudo-element exists, a new stacking context is created for the root and top layer elements. The page-transition layer is a sibling of this stacking context.
-
The page-transition layer paints after the stacking context for the root and top layer elements.
Note: The intent of the feature is to be able to capture the contents of the page, which includes the top layer elements. In order to accomplish that, the page-transition layer cannot be a part of the captured top layer context, since that results in a circular dependency. Instead, this stacking context is a sibling of other page contents.
Do we need to clarify that the stacking context for the root and top layer elements has filters and effects coming from the root element’s style?
5.3. Captured elements
A captured element is a struct with the following:
- outgoing image
-
an image or null. Initially null.
- outgoing styles
-
a set of styles or null. Initially null.
The type of "a set of styles" needs to be linked or defined.
- incoming element
-
an element or null. Initially null.
5.4. Additions to Document
A Document
additionally has:
- active DOM transition
-
a
DOMTransition
or null. Initially null. - transition suppressing rendering
-
a boolean. Initially false.
6. API
6.1. Additions to Document
partial interface Document {DOMTransition createTransition (DOMTransitionInit ); };
init dictionary {
DOMTransitionInit required UpdateDOMCallback ; };
updateDOM callback =
UpdateDOMCallback Promise <any > ();
6.1.1. createTransition()
createTransition(init)
are as follows:
-
Let transition be a new
DOMTransition
object in this’s relevant Realm. -
Set transition’s DOM update callback to init[
updateDOM
]. -
Let document be this’s relevant global object’s associated document.
-
If document’s active DOM transition is not null, then skip the page transition document’s active DOM transition with an "
AbortError
"DOMException
in this’s relevant Realm.Note: This can result in two asynchronous DOM update callbacks running concurrently. One for the document’s current active DOM transition, and another for this transition. As per the design of this feature, it’s assumed that the developer is using another feature or framework to correctly schedule these DOM changes.
-
Set document’s active DOM transition to transition.
Note: The process continues in perform an outgoing capture.
-
Return transition.
document. createTransition({ updateDOM() { coolFramework. changeTheDOMToPageB(); } });
If more precise management is needed, however, transition elements can be managed in script:
async function doTransition() { // Specify "outgoing" elements. The tag is used to match against // "incoming" elements they should transition to, and to refer to // the transitioning pseudo-element. document. querySelector( '.old-message' ). style. pageTransitionTag= 'message' ; const transition= document. createTransition({ async updateDOM() { // This callback is invoked by the browser when "outgoing" // capture finishes and the DOM can be switched to the new // state. No frames are rendered until this callback returns. // DOM changes may be asynchronous await coolFramework. changeTheDOMToPageB(); // Tagging elements during the updateDOM() callback marks them as // "incoming", to be matched up with the same-tagged "outgoing" // elements marked previously and transitioned between. document. querySelector( '.new-message' ). style. pageTransitionTag= 'message' ; }, }); // When ready resolves, all pseudo-elements for this transition have // been generated. // They can now be accessed in script to set up custom animations. await transition. ready; document. documentElement. animate( keyframes, { ... animationOptions, pseudoElement: '::page-transition-container(message)' , }); // When the finished promise resolves, that means the transition is // finished. await transition. finished; }
6.2. The DOMTransition
interface
interface {
DOMTransition undefined skipTransition ();readonly attribute Promise <undefined >;
finished readonly attribute Promise <undefined >;
ready readonly attribute Promise <undefined >; };
domUpdated
DOMTransition
represents and controls
a single same-document transition. That is, it controls a transition where the
starting and ending document are the same, possibly with changes to the
document’s DOM structure. A DOMTransition
has the following:
- tagged elements
-
a map, whose keys are page transition tags and whose values are captured elements. Initially a new map.
- phase
-
One of the following phases:
-
"
pending-capture
". -
"
dom-update-callback-called
". -
"
animating
". -
"
done
".
-
- DOM update callback
-
an
UpdateDOMCallback
or null. Initially null. - ready promise
-
a
Promise
. Initially a new promise in this’s relevant Realm. - DOM updated promise
-
a
Promise
. Initially a new promise in this’s relevant Realm. - finished promise
-
a
Promise
. Initially a new promise in this’s relevant Realm.
The finished
getter steps are to return this’s finished promise.
The ready
getter steps are to return this’s ready promise.
The domUpdated
getter steps are to return this’s DOM updated promise.
6.2.1. skipTransition()
skipTransition()
are:
-
If this's phase is not "
done
", then skip the page transition for this with an "AbortError
"DOMException
.
7. Algorithms
7.1. Monkey patches to rendering
-
For each fully active
Document
in docs, perform pending transition operations for thatDocument
.
Note: These steps will be added to the update the rendering in the HTML spec. As such, the prose style is written to match other steps in that algorithm.
-
For each
Document
in docs with a transition suppressing rendering of true:Define this behavior. Lifecycle updates can still be triggered via script APIs which query style or layout information but no visual updates are presented to the user. Is this the same behavior as render-blocking?
How should input be handled when in this state? The last frame presented to the user will not reflect the DOM state as it asynchronously switches to the new version.
Note: The aim is to prevent unintended DOM updates from being presented to the user after a cached snapshot for the elements has been captured. We wait for one rendering opportunity after prepare to present DOM mutations made by the author before prepare to be presented to the user. This is also the content captured in snapshots.
Note: These steps will be added to the update the rendering in the HTML spec. As such, the prose style is written to match other steps in that algorithm.
7.2. Perform pending transition operations
Document
document,
perform the following steps:
-
If document’s active DOM transition is not null, then:
-
If document’s active DOM transition's phase is "
pending-capture
", then perform an outgoing capture with document’s active DOM transition. -
Otherwise, if document’s active DOM transition's phase is "
animating
", then update transition DOM for document’s active DOM transition.
-
7.3. Perform an outgoing capture
DOMTransition
transition,
perform the following steps:
-
Let taggedElements be transition’s tagged elements.
-
Let usedTransitionTags be a new set of strings.
-
Let document be transition’s relevant global object’s associated document.
-
For each element of every
Element
and pseudo-element connected to document, in paint order:The link for "paint order" doesn’t seem right. Is there a more canonical definition?
-
Let transitionTag be the computed value of page-transition-tag for element.
-
If transitionTag is none, or element is not rendered, then continue.
-
If any of the following is true:
-
usedTransitionTags contains transitionTag.
-
element is not element’s root and element does not have layout containment.
-
element is not element’s root and element allows fragmentation.
Then skip the page transition for transition with an "
InvalidStateError
"DOMException
in transition’s relevant Realm, and return. -
-
Append transitionTag to usedTransitionTags.
-
Let capture be a new captured element struct.
-
Set capture’s outgoing image to the result of capturing the image of element.
-
Set capture’s outgoing styles to the following:
- transform
-
A CSS transform that would place element from the layout viewport origin to its current quad.
This value is identity for the root element.
- width
- height
-
The width and height of element’s border box.
This value is the bounds of the initial containing block for the root element.
- object-view-box
-
An object-view-box value that, when applied to the outgoing image, will cause the view box to coincide with element’s border box in the image.
- writing-mode
-
The writing-mode of element.
- direction
-
The direction of element.
-
Set taggedElements[transitionTag] to capture.
-
-
Set document’s transition suppressing rendering to true.
-
Queue a global task on the DOM manipulation task source, given transition’s relevant global object, to execute the following steps:
Note: A task is queued here because the texture read back in capturing the image may be async, although the render steps in the HTML spec act as if it’s synchronous.
-
If transition’s phase is "
done
", then abort these steps.Note: This happens if transition was skipped before this point.
-
Call the DOM update callback of transition.
-
React to transition’s DOM updated promise:
-
If the promise does not settle within an implementation-defined timeout, then:
-
If transition’s phase is "
done
", then return.Note: This happens if transition was skipped before this point.
-
Skip the page transition transition with a "
TimeoutError
"DOMException
.
-
-
If the promise was fulfilled, then:
-
If transition’s phase is "
done
", then return.Note: This happens if transition was skipped before this point.
-
Set transition suppressing rendering to false.
-
Set usedTransitionTags to a new set.
-
For each element of every
Element
and pseudo-element connected to document, in paint order:The link for "paint order" doesn’t seem right. Is there a more canonical definition?
-
Let transitionTag be the computed value of page-transition-tag for element.
-
If transitionTag is none, or element is not rendered, then continue.
-
If any of the following is true:
-
usedTransitionTags contains transitionTag.
-
element is not element’s root and element does not have layout containment.
-
element is not element’s root and element allows fragmentation.
Then skip the page transition transition with an "
InvalidStateError
"DOMException
, and return. -
-
Append transitionTag to usedTransitionTags.
-
If taggedElements[transitionTag] does not exist, then set taggedElements[transitionTag] to a new captured element struct.
-
Let capture be taggedElements[transitionTag].
-
Let capture’s incoming element item be element.
-
-
Create transition pseudo-elements for transition.
-
Animate a page transition transition.
Note: This will require running document lifecycle phases to compute information calculated during style/layout.
Note: The frame-by-frame parts of the animation are handled in update transition DOM.
-
-
If the promise was rejected with reason r, then:
-
If transition’s phase is "
done
", then return.Note: This happens if transition was skipped before this point.
-
Skip the page transition transition with r.
-
-
-
7.4. Skip the page transition
DOMTransition
transition with reason reason:
-
Let document be transition’s relevant global object’s associated document.
-
Assert: document’s active DOM transition is transition.
-
If transition’s phase is before "
dom-update-callback-called
", then call the DOM update callback of transition. -
Set transition suppressing rendering to false.
-
If transition’s phase is equal to or after "
animating
", then:-
Remove all associated page-transition pseudo-elements from document.
-
-
Set transition’s phase to "
done
". -
Set document’s active DOM transition to null.
-
Reject transition’s ready promise with reason.
-
Reject transition’s finished promise with reason.
7.5. Capture the image
Element
element, perform the following steps.
They return an image.
-
Render the referenced element and its descendants, at the same size that they would be in the document, over an infinite transparent canvas with the following characteristics:
-
The origin of element’s ink overflow rectangle is anchored to canvas origin.
-
If the referenced element has a transform applied to it (or its ancestors), then the transform is ignored.
Note: This transform is applied to the snapshot using the
transform
property of the associated ::page-transition-container pseudo-element. -
For each descendant of shadow-including descendant
Element
and pseudo-element of element, if descendant has a computed value of page-transition-tag that is not none, then skip painting descendant.Note: This is necessary since the descendant will generate its own snapshot which will be displayed and animated independently.
Refactor this so the algorithm takes a set of elements that will be captured. This centralizes the logic for deciding if an element should be included or not.
-
-
Let interestRectangle be the result of computing the interest rectangle for element.
Note: The interestRectangle is the subset of element’s ink overflow rectangle that should be captured. This is required for cases where an element’s ink overflow rectangle needs to be clipped because of hardware constraints. For example, if it exceeds the maximum texture size.
-
Return the portion of the canvas within interestRectangle as an image. The natural size of the image is equal to the interestRectangle bounds.
7.6. Update transition DOM
DOMTransition
transition:
-
Let document be transition’s relevant global object’s associated document.
-
For each page-transition pseudo-elements associated with transition, check whether there is an active animation associated with this pseudo-element.
-
If no page-transition pseudo-elements has an active animation:
There needs to be a definition/link for "active animation".
-
Set transition’s phase to "
done
". -
Remove all associated page-transition pseudo-elements from document.
-
Set document’s active DOM transition to null.
-
Resolve transition’sfinished promise.
-
Return.
-
-
For each tag -> capturedElement of transition’s tagged elements:
-
If capturedElement has an "incoming element", run capture the image on capturedElement’s "incoming element" and update the displayed image for ::page-transition-incoming-image with the tag tag.
At the user-agent origin, set incoming’s object-view-box property to a value that when applied to incoming, will cause the view box to coincide with "incoming element"'s border box in the image.
-
...
Also clarify updating the animation based on new bounds/transform to get c0 continuity.
-
7.7. Compute the interest rectangle
Element
el, perform the following steps.
They return a rectangle.
-
If el is the document’s root element, then return a rectangle that is the intersection of the layout viewport, including the size of rendered scrollbars (if any), with el’s ink overflow rectangle.
-
If el’s ink overflow area does not exceed an implementation-defined maximum size, then return a rectangle that is equal to el’s ink overflow rectangle.
-
Otherwise:
Define the algorithm used to clip the snapshot when it exceeds max size.
7.8. Animate a page transition
DOMTransition
transition:
-
Generate a <keyframe> named "page-transition-fade-out" in user-agent origin as follows:
@keyframes page-transition-fade-out{ to{ opacity : 0 ; } } -
Generate a <keyframe> named "page-transition-fade-in" in user-agent origin as follows:
@keyframes page-transition-fade-in{ from{ opacity : 0 ; } } -
Apply the following styles in user-agent origin:
html : :page-transition-outgoing-image ( *) { animation-name : page-transition-fade-out; } html::page-transition-incoming-image ( *) { animation-name : page-transition-fade-in; } -
For each tag -> capturedElement of transition’s tagged elements:
-
If neither of capturedElement’s outgoing image or incoming element is null:
-
Let transform be capturedElement’s outgoing styles's transform property.
-
Let width be capturedElement’s outgoing styles's width property.
-
Let height be capturedElement’s outgoing styles's height property.
-
Generate a <keyframe> named "page-transition-container-anim-tag" in user-agent origin as follows:
@keyframes page-transition-container-anim-|tag|{ from{ transform : |transform|; width : |width|; height : |height|; } } -
-
Apply the following styles in user-agent origin:
html : :page-transition-container ( |tag|) { animation-name : page-transition-container-anim-|tag|; }
-
-
Set transition’s phase to "
animating
".
How are keyframes scoped to user-agent origin? We could decide
scope based on whether animation-name
in the computed style
came from a developer or UA stylesheet.
We should retarget the animation if computed properties for incoming elements change.
7.9. Create transition pseudo-elements
DOMTransition
transition:
-
Let transitionRoot be the result of creating a new ::page-transition pseudo-element.
-
For each transitionTag → capturedElement of transition’s tagged elements:
-
Let container be the result of creating a new ::page-transition-container pseudo-element with the tag transitionTag.
-
Append container to transitionRoot.
This should be better defined. I’m not sure if pseudo-elements have defined ways to modify their DOM.
-
Let width, height, transform, writingMode, and direction be null.
-
If capturedElement’s incoming element is null, then:
-
Set width to capturedElement’s outgoing styles width property.
-
Set height to capturedElement’s outgoing styles height property.
-
Set transform to capturedElement’s outgoing styles transform property.
-
Set writingMode to capturedElement’s outgoing styles writing-mode property.
-
Set direction to capturedElement’s outgoing styles direction property.
-
-
Otherwise:
-
Set width to the current width of capturedElement’s incoming element's border box.
-
Set height to the current height of capturedElement’s incoming element's border box.
-
Set transform to a transform that maps the capturedElement’s incoming element's border box from document origin to its quad in layout viewport.
-
Set writingMode to the computed value of writing-mode on capturedElement’s incoming element.
-
Set direction to the computed value of direction on capturedElement’s incoming element.
-
-
At the user-agent origin, set container’s width, height, transform, writing-mode, and direction properties to width, height, transform, writingMode, and direction.
-
Let imageWrapper be a new ::page-transition-image-wrapper pseudo-element with the tag transitionTag.
-
Append imageWrapper to container.
-
If capturedElement’s outgoing image is not null, then:
-
Let outgoing be a new ::page-transition-outgoing-image replaced element pseudo-element, with the tag transitionTag, displaying capturedElement’s outgoing image.
-
Append outgoing to imageWrapper.
-
At the user-agent origin, set outgoing’s object-view-box property to capturedElement’s outgoing styles object-view-box property.
-
-
If capturedElement’s incoming element is not null, then:
-
Let incoming be a new ::page-transition-incoming-image replaced element pseudo-element, with the tag transitionTag, displaying the capture the image of capturedElement’s incoming element.
-
Append incoming to imageWrapper.
-
At the user-agent origin, set incoming’s object-view-box property to a value that when applied to incoming, will cause the view box to coincide with incoming element's border box in the image.
The incoming element and its contents (the flat tree descendants of the element, including both text and elements, or the replaced content of a replaced element), except the page-transition pseudo-elements, are not painted (as if they had visibility: hidden) and do not respond to hit-testing (as if they had pointer-events: none) until incoming exists.
-
-
DOMTransition
transition:
-
Assert: transition’s phase is before "
dom-update-callback-called
". -
Let callbackPromise be the result of invoking transition’s DOM update callback.
-
Set transition’s phase to "
dom-update-callback-called
". -
React to callbackPromise:
-
If callbackPromise was fulfilled, then resolve transition’s DOM updated promise.
-
If callbackPromise was rejected with reason r, then reject transition’s DOM updated promise with r.
-