Last updated: First published:
Styling View Transitions
The animations introduced by the View Transition API are controlled by CSS rules. The way to make the animations fit your expectations is to alter the default animation by overriding the styles for the pseudo elements.
To do this efficiently you should know what the View Transition API defines as defaults and how to easily change these styles if needed. To simplify styling, level 2 of the API introduced helpers in form of the view-transition-class property and the :active-view-transition* pseudo-classes.
There are three parts that you influence with CSS when it comes to view transitions:
- Tell the View Transition API which DOM elements to setup for individual animations
- Setup animations for elements participating in a view transition
- Trigger CSS for things other than animations
If you want a DOM element to participate in view transitions with an individual animation, you assign a value to the view-transition-name CSS property of the element. The browser will then generate three or four pseudo-elements for each visible DOM element with a view transition name. These are a transition group, an image pair, and at least one of the old or new image pseudos.
For a single view transition name, x, there are different animations that you might want to style.
If your image-pair has both, an old and a new image, you can control how the old image is replaced by the new image during the transition. Three animations are involved:
- the animation of the view-transition-old(x)pseudo-element,
- the animation of the view-transition-new(x)pseudo-element and
- the animation of the view-transition-group(x)pseudo-element
The first two typically work together in a cross-fade or in a combined effect that removes the old image and inserts the new one. The animation of the view-transition-group() aligns differences in size, transformation, and position of the old and new image.
If your image-pair has only a single image, the browser places the view-transition-group() at the same position and does not define a morphing animation because there is nothing to align. You will still have the animation for the old or new image, respectively. Instead of a morph animation you have an entry animation if the old image is missing or an exit animation if the new image is missing.
Using an animation shorthand↗, CSS rules for the element with view transition name x could look like this:
::view-transition-old(x) {  animation: 333ms linear both slide-out;}::view-transition-new(x) {  animation: 333ms linear both slide-in;}::view-transition-group(x) {  animation: none; /* switch off morph transition */}The animation shorthand is quite expressive. You can define several animations at once wit h different durations and start delays.
There might be good reasons for defining completely new animations for your view-transitions.
But when you assign to shorthand properties, like animation, you completely override the definitions from the user agent stylesheet. And then you have to specify all values or be sure that the CSS default values fit your expectations.
If you only want to override the duration and the timing function of an animation defined by the view transition API, you might be better of with:
::view-transition-group(x) {  animation-duration: 333ms;  animation-timing-function: linear;}That way you keep the former definitions and only change what is important to you. For an example on reusing browser-defined keyframes, see the notes on reusing user agent stylesheets.
Sometimes you want the animations for new and old images behave different for morph and entry/exit animations. this two cases can be distinguished with the :only-child CSS pseudo-class.
For example, assume you have an element that pops in with an entry animation and pops out with an exit animation. You do not want an animation if the element is on both sides of the transition, i.e. neither exit nor entry.
:is(::view-transition-old(x), ::view-transition-new(x)):not(:only-child) {  animation-name: none;}While the typical task will be redefining animations when styling view transitions, you can also use view transitions to trigger other style changes. You can statically set the properties of the pseudo-elements created by the View Transition API. For example you could add a border.
::view-transition-old(x) {  border: 1pt solid red;}You can also style other DOM elements that are not created by the View Transition API, but keep in mind that those elements are typically hidden behind the view transition pseudo-elements during a view transition.
When the View Transition API inserts an animation, it has to give some default values to them.
Here is what the API automatically assign in its user agent stylesheet↗ for a view transition name x. In keyframe names, -ua stands for user agent.
If both, the old image and the new image for x exist, the API defines a morph animation from the old to the new image. The browser generates specific keyframes for each view transition group:
::view-transition-group(x) {  animation-name: -ua-view-transition-group-anim-x;}@keyframes -ua-view-transition-group-anim-x {  0% {    backdrop-filter: <old backdrop filter>;    transform: <matrix to exactly cover the old element>;    height: <old height>;    width: <old width>;  }  100% {    backdrop-filter: <new backdrop filter>;    transform: <matrix to exactly cover the new element>;    height: <new height>;    width: <new width>;  }}The transform is used to move the deck with the new image on top of the old image from the old image’s size, transform, and position to the new image’s size, transform, and position. I.e. if the original element of the view-transition-name has some CSS transformation applied on the old or new page, like rotate or skew, this will also be honored by the generated transform.
The ::view-transition-group-children pseudo-element was only introduced recently. The spec does not yet describe the animations that the API applies to this element. So details might change.
It was added to make clipping the children of containers more precise. Border-width, border-radius, and corner-shape play an important role here.
Just like with group animations, the transition begins with the values of the old element and gradually morphs them into the values of the new element.
::view-transition-group-children(x) {  animation-name: -ua-view-transition-group-children-anim-x;}@keyframes -ua-view-transition-group-children-anim-x {  0% {    border-width: <old border-width>;    border-radius: <old border-radius>;    corner-shape: <old corner-shape>;  }  100% {    border-width: <new border-width>;    border-radius: <new border-radius>;    corner-shape: <new corner-shape>;  }}Border-style and border-color do not need to change. They are set to solid and transparent.
The typical cross-fade effect of view transitions is implemented by animations defined on the pseudo-elements for the old and new images. The definitions are the same for all view transition names.
::view-transition-old(x) {  animation-name:    -ua-view-transition-fade-out,    -ua-mix-blend-mode-plus-lighter;}::view-transition-new(x) {  animation-name:    -ua-view-transition-fade-in,    -ua-mix-blend-mode-plus-lighter;}@keyframes -ua-view-transition-fade-out {  0% {    opacity: 1;  }  100% {    opacity: 0;  }}@keyframes -ua-view-transition-fade-in {  0% {    opacity: 0;  }  100% {    opacity: 1;  }}@keyframes -ua-mix-blend-mode-plus-lighter {  0% {    mix-blend-mode: plus-lighter;  }  100% {    mix-blend-mode: plus-lighter;  }}-ua-mix-blend-mode-plus-lighter is only added if the image pair has both images. For some background on -ua-mix-blend-mode-plus-lighter see the section about mix blend modes.
The View Transition API does not assign any animations to the ::view-transition-image-pair(x) pseudo-elements as those elements are mainly added for render isolation and morph 1:1 with the ::view-transition-group(x) elements.
Of course, you can add animations to those elements. This has the benefit that you can modify the effect of the morph animation without the need to override the API-generated animations for ::view-transition-group(x).
The values for animation-duration, animation-delay, and animation-fill-mode are inherited from parent in the pseudo-element tree.
The API sets the defaults to 0.25s, 0s, and both.
Recently, all other animation properties but animation-name where added to the list.
::view-transition-group(*),::view-transition-group-children(*),::view-transition-image-pair(*),::view-transition-old(*),::view-transition-new(*) {  animation-duration: inherit;  animation-delay: inherit;  animation-fill-mode: inherit;
  /* newly added */  animation-timing-function: inherit;  animation-iteration-count: inherit;  animation-direction: inherit;  animation-play-state: inherit;}If the browser only supports the short list, the animation-timing-function, animation-iteration-count, animation-direction, and animation-play-state properties keep their default values: ease, 1, normal, and running respectively.
Due to these default inherits you might want to prefer setting the individual animation-... properties instead of using the animation shortcut, which would reset animation-duration, animation-delay and animation-fill-mode to 0s, 0s, and none if not explicitly stated otherwise.
For example you can set the duration of all unchanged morph and cross-fade animations with a single rule:
::view-transition-group(*) {  animation-duration: 500ms; /* all groups, which pass down to the image-pairs, which pass down to the old and new images. */}Using inherit in your own definitions is a good idea as it allows for control over several animations at once.
Of course styling view transition pseudo-elements is not limited to animations. The browser’s user agent stylesheet also contains examples for such styling:
::view-transition {  position: absolute;  inset: 0;}
::view-transition-group(*) {  position: absolute;  top: 0;  left: 0;}
::view-transition-group-children(*) {  position: absolute;  inset: 0px;  border-style: solid;  border-color: transparent;}
::view-transition-image-pair(*) {  position: absolute;  inset: 0;}
::view-transition-old(*),::view-transition-new(*) {  position: absolute;  inset-block-start: 0;  inline-size: 100%;  block-size: auto;}In the examples above you saw the patterns for addressing an individual pseudo-element: The view transition name is used as a parameter of the pseudo-element selector:
::view-transition-<pseudo-element>(<view-transition-name>) {  /* property definitions */}Beside addressing individual pseudo-elements it is also possible to address the pseudo elements of all view transition names at once using * instead of a view transition name:
::view-transition-<pseudo-element>(*) {  /* property definitions */}Also seen above: You can address the root of all view transition pseudo-elements, although it has no name:
::view-transition {  /* property definitions */}When you have CSS rules for a large number of view transition names, you can use the (*) pattern to address a pseudo-element for all transition names and use specific rules with the ((name-x)) pattern to handle the exception. There might be situations where you wish for a better developer experience. This is where view transition classes come in.
Similar to the view-transition-name property, you can assign view-transition-class values to DOM elements. While view transition names have to be unique in the DOM, view transition class values need not and typically will not be unique.
When looking at how the view transition classes are used, the similarity to CSS classes is rather obvious. You use them inside the pseudo-element selectors, prefixed by a dot (.).
If multiple class names are used to identify a pseudo element, all must match. The following CSS rule can be used to address all view transition groups that have the nav-link view transition class:
::view-transition-group(*.nav-link) {  /* ... */}Or shorter, without the star:
::view-transition-group(.nav-link) {  /* ... */}This is an example on how to assign the class:
#navbar a {  view-transition-class: nav-link;}#prev,#next {  view-transition-class: nav-link some-other-class;}This might best be combined with dynamically (per script) added view transition names or tooling where you declaratively assign view transition names.
Often you want to support different animations for a given view transition pseudo-element and select one of those alternatives depending on some condition. The concept that the View Transition API provides for this use case is called active view transition types.
While it is active, a view transition might have a set of identifiers called types that can be used in CSS pseudo-classes to select different rules. During the view transition, the set can be altered at any time by assigning to the types property of the viewTransition object.
This mechanism can be used to increase isolation and/or composability of independent view transition use cases.
The initial set of types can be set when calling startViewTransition() or it might be specified using the types property inside a @view-transition rule.
document.startViewTransition({ types: ["boom"], update: changeTheDOM });or
@view-transition {  navigation: auto;  types: boom, var(--special);}Another typical pattern for cross-document view transitions is to (conditionally) set the types in the pagereveal or pageswap handlers.
listener(event) {  ...  event.viewTransition.types.add("effect");  ...}Types set during pageswap are only available on the old page. You can not use them to guide the styling of ::view-transition pseudos because that CSS is always taken from the new page. But you can use the pageswap types to conditionally set or remove view transition names on the old page.
Types set during pagereveal can be used for both: setting and removing view transition names on the new page and for styling the view transition pseudo elements.
The pseudo-class that can check for types is :active-view-transition-type(). It takes a comma separated list of types. If at least one of these types is set on the active view transition, the pseudo-class matches the documents root element.
Thus, the typical patterns to use :active-view-transition-type() are:
Directly prepend the pseudo-class to a pseudo-element selector…
:active-view-transition-type(type)::view-transition-group(main) {  /* ... */}…or equivalently use it with a nested CSS rule.
:active-view-transition-type(type) {  &::view-transition-group(main) {    /* ... */  }}There is an additional pseudo-class that does not take type parameters.
:active-view-transition {  /* ... */}This selector will match the the :root element while a view transition is active. You can use this for example to change the shape of the pointer during view transitions.
While view transition types are a Level 2 feature, they can be polyfilled if you want to use them with Level 1 implementations.
When you navigate forth and back through the pages of this site, you see an example on how types are used to select different animation.
- Go to the page about flickers during morph animations. You’ll see that it slides in from the right, as you go forward within the site’s order of pages.
- Select the page again from the global site navigation (have first to open it on mobile). You see a different effect.
- Press the browser’s back key or button to return to this page. The page slides in from the left as you go backwards in the site’s order of pages.
See the Turn-Signal documentation on how this effect is achieved and what pitfalls to avoid.
The pseudo-classes can not only be used to define different animations for different situations. It is also possible to let the existence of a view transition group depend on view transitions types. So on some transitions, an element can have its own animation and on others it is just part of some parent and its animation.
You can find an example for this behavior on this site. When you visit the page about flickers during morph animations and visit it directly again be selecting it in the site navigation, you will see that for a navigation to the same page, the <main> section does not have its own view transition. Also when you click the images on that page, you start view transitions where the <main> section is not animated.
See the Turn-Signal documentation on how this effect is achieved and how you could get even more specific and cancel single images instead of whole transition groups.
In the View Transition Draft Spec↗ the pseudo-elements are prefixed with the the :root pseudo-class selector.
The scope of view transitions is the whole document. Until (element) scoped view transitions change this in the future, all pseudo-elements that the API generates are children of the document’s root element and can be targeted using the CSS :root selector. So ::view-transition-group(x) and :root::view-transition-group(x) currently are the same thing.
Note that :root has a different meaning inside shadow DOMs. But that is a whole different story.
With scoped view transitions, :root::view-transition* will still select only pseudo-elements of the document root, while ::view-transition* will target pseudo-elements of any element. Should I prefix all rules with :root? Right now, I do not. Scoped view transitions will be another (smaller) game changer, and I expect to refactor things anyway once they become available.
When you use these patterns in your own stylesheets, be aware that the specificities differ: :root::view-transition-group(x) with a specificity of (0, 1, 1) is more specific than ::view-transition-group(x) with a specificity of (0, 0, 1).
Now you might conclude that you also need to prefix your view transition pseudo-elements with :root if you want to override the defaults. That is not the case. The user agent stylesheet has the least important origin and is overridden by user stylesheet rules independent of specificity.
