Transitioning Top-Layer Entries And The Display Property In CSS<\/h1>\nBrecht De Ruyte<\/address>\n 2025-01-29T10:00:00+00:00
\n 2025-03-06T17:04:34+00:00
\n <\/header>\n
Animating from and to display: none;<\/code> was something we could only achieve with JavaScript to change classes or create other hacks. The reason why we couldn\u2019t do this in CSS is explained in the new CSS Transitions Level 2 specification<\/a>:<\/p>\n\u201cIn Level 1 of this specification, transitions can only start during a style change event for elements that have a defined before-change style established by the previous style change event. That means a transition could not be started on an element that was not being rendered for the previous style change event.\u201d<\/p><\/blockquote>\n
In simple terms, this means that we couldn\u2019t start a transition on an element that is hidden or that has just been created.<\/p>\n
What Does transition-behavior: allow-discrete<\/code> Do?<\/h3>\nallow-discrete<\/code> is a bit of a strange name for a CSS property value, right? We are going on about transitioning display: none<\/code>, so why isn\u2019t this named transition-behavior: allow-display<\/code> instead? The reason is that this does a bit more than handling the CSS display<\/code> property, as there are other \u201cdiscrete\u201d properties in CSS. A simple rule of thumb is that discrete properties do not transition but usually flip right away between two states. Other examples of discrete properties are visibility<\/code> and mix-blend-mode<\/code>. I\u2019ll include an example of these at the end of this article.<\/p>\nTo summarise, setting the transition-behavior<\/code> property to allow-discrete<\/code> allows us to tell the browser it can swap the values of a discrete property (e.g., display<\/code>, visibility<\/code>, and mix-blend-mode<\/code>) at the 50% mark instead of the 0% mark of a transition.<\/p>\nWhat Does @starting-style<\/code> Do?<\/h3>\nThe @starting-style<\/code> rule defines the styles of an element right before it is rendered to the page. This is highly needed in combination with transition-behavior<\/code> and this is why:<\/p>\nWhen an item is added to the DOM or is initially set to display: none<\/code>, it needs some sort of \u201cstarting style\u201d from which it needs to transition. To take the example further, popovers and dialog elements<\/a> are added to a top layer which is a layer that is outside of your document flow, you can kind of look at it as a sibling of the <html><\/code> element in your page\u2019s structure. Now, when opening this dialog or popover, they get created inside that top layer, so they don\u2019t have any styles to start transitioning from, which is why we set @starting-style<\/code>. Don\u2019t worry if all of this sounds a bit confusing. The demos might make it more clearly. The important thing to know is that we can give the browser something to start the animation with since it otherwise has nothing to animate from.<\/p>\nA Note On Browser Support<\/h3>\n
At the moment of writing, the transition-behavior<\/code> is available in Chrome, Edge, Safari, and Firefox. It\u2019s the same for @starting-style<\/code>, but Firefox currently does not support animating from display: none<\/code>. But remember that everything in this article can be perfectly used as a progressive enhancement.<\/p>\nNow that we have the theory of all this behind us, let\u2019s get practical. I\u2019ll be covering three use cases in this article:<\/p>\n
\n- Animating from and to
display: none<\/code> in the DOM.<\/li>\n- Animating dialogs and popovers entering and exiting the top layer.<\/li>\n
- More \u201cdiscrete properties\u201d we can handle.\n
<\/li>\n<\/ul>\n
\n
\n 2025-03-06T17:04:34+00:00
\n <\/header>\n
display: none;<\/code> was something we could only achieve with JavaScript to change classes or create other hacks. The reason why we couldn\u2019t do this in CSS is explained in the new CSS Transitions Level 2 specification<\/a>:<\/p>\n\u201cIn Level 1 of this specification, transitions can only start during a style change event for elements that have a defined before-change style established by the previous style change event. That means a transition could not be started on an element that was not being rendered for the previous style change event.\u201d<\/p><\/blockquote>\n
In simple terms, this means that we couldn\u2019t start a transition on an element that is hidden or that has just been created.<\/p>\n
What Does transition-behavior: allow-discrete<\/code> Do?<\/h3>\nallow-discrete<\/code> is a bit of a strange name for a CSS property value, right? We are going on about transitioning display: none<\/code>, so why isn\u2019t this named transition-behavior: allow-display<\/code> instead? The reason is that this does a bit more than handling the CSS display<\/code> property, as there are other \u201cdiscrete\u201d properties in CSS. A simple rule of thumb is that discrete properties do not transition but usually flip right away between two states. Other examples of discrete properties are visibility<\/code> and mix-blend-mode<\/code>. I\u2019ll include an example of these at the end of this article.<\/p>\nTo summarise, setting the transition-behavior<\/code> property to allow-discrete<\/code> allows us to tell the browser it can swap the values of a discrete property (e.g., display<\/code>, visibility<\/code>, and mix-blend-mode<\/code>) at the 50% mark instead of the 0% mark of a transition.<\/p>\nWhat Does @starting-style<\/code> Do?<\/h3>\nThe @starting-style<\/code> rule defines the styles of an element right before it is rendered to the page. This is highly needed in combination with transition-behavior<\/code> and this is why:<\/p>\nWhen an item is added to the DOM or is initially set to display: none<\/code>, it needs some sort of \u201cstarting style\u201d from which it needs to transition. To take the example further, popovers and dialog elements<\/a> are added to a top layer which is a layer that is outside of your document flow, you can kind of look at it as a sibling of the <html><\/code> element in your page\u2019s structure. Now, when opening this dialog or popover, they get created inside that top layer, so they don\u2019t have any styles to start transitioning from, which is why we set @starting-style<\/code>. Don\u2019t worry if all of this sounds a bit confusing. The demos might make it more clearly. The important thing to know is that we can give the browser something to start the animation with since it otherwise has nothing to animate from.<\/p>\nA Note On Browser Support<\/h3>\n
At the moment of writing, the transition-behavior<\/code> is available in Chrome, Edge, Safari, and Firefox. It\u2019s the same for @starting-style<\/code>, but Firefox currently does not support animating from display: none<\/code>. But remember that everything in this article can be perfectly used as a progressive enhancement.<\/p>\nNow that we have the theory of all this behind us, let\u2019s get practical. I\u2019ll be covering three use cases in this article:<\/p>\n
\n- Animating from and to
display: none<\/code> in the DOM.<\/li>\n- Animating dialogs and popovers entering and exiting the top layer.<\/li>\n
- More \u201cdiscrete properties\u201d we can handle.\n
<\/li>\n<\/ul>\n
\n