CSS Display Module Level 3

Status: ED
ED: http://dev.w3.org/csswg/css-display
Shortname: css-display
Group: csswg
Level: 3
Editor: Tab Atkins Jr., Google, http://xanthir.com/contact/
Abstract: This module contains the features of CSS relating to the 'display' property and other box-generation details.
Ignored Terms: show, display-model, display-role

Introduction

This section is not normative. The 'display' property, introduced in CSS 2.1, defines what kind of boxes an element generates (and whether it generates boxes at all), and how it lays out its contents. These concepts are actually rather independent, though they're conflated by the 'display' property. This causes some pain when a property value intended to affect one aspect (such as setting an element to ''display:none'' to suppress box generation) affects another aspect (such as losing the memory of what it was before ''display:none'', so that it can be set back to that value later). This specification subsumes the CSS 2.1 definition of the 'display' property, and redefines it to be a shorthand property for a small family of longhands, each controlling an independent aspect of an element's "display".

Module interactions

This specification transforms the 'display' property into a shorthand property, and defines several longhand properties that it expands into or effects. This module replaces and extends the definition of the 'display' property defined in [[!CSS21]] section 9.2.4. None of the properties in this module apply to the ::first-line or ::first-letter pseudo-elements.

Values

This specification follows the CSS property definition conventions from [[!CSS21]]. Value types not defined in this specification are defined in CSS LevelĀ 2 RevisionĀ 1 [[!CSS21]]. Other CSS modules may expand the definitions of these value types. In addition to the property-specific values listed in their definitions, all properties defined in this specification also accept the CSS-wide keywords as their property value. For readability it has not been repeated explicitly.

The Display Properties

The 'display' shorthand and its associated family of properties control the layout mode of elements (how the element determines the sizes and positions of itself and its descendants), and what boxes they and their descendants generate.

Setting the layout mode: the 'display-inside' property

	Name: display-inside
	Value: auto | block | table | flex | grid | ruby
	Initial: auto
	Applies to: all elements
	Inherited: no
	Percentages: n/a
	Computed value: a keyword
	Media: all
	
auto
If the element's computed 'display-outside' value is ''inline-level'', the element is an inline element, and lays out its contents using inline layout. [[!CSS21]] If the element's computed 'display-outside' value is an layout-specific display role, this elements acts as normal for its given 'display-outside' value. Otherwise, this value computes to ''display-inside/block''.
block
The element lays out its contents using block layout. [[!CSS21]]
table
The element lays out its contents using table layout. [[!CSS21]]
flex
The element lays out its contents using flex layout. [[!CSS3-FLEXBOX]]
grid
The element lays out its contents using grid layout. [[!CSS3-GRID-LAYOUT]]
ruby
The element lays out its contents using ruby layout. [[!CSS3RUBY]]

Interacting with the layout mode: the 'display-outside' property

	Name: display-outside
	Value: block-level | inline-level | none | table-row-group | table-header-group | table-footer-group | table-row | table-cell | table-column-group | table-column | table-caption | ruby-base | ruby-text | ruby-base-container | ruby-text-container
	Initial: inline-level
	Applies to: all elements
	Inherited: no
	Percentages: n/a
	Computed value: as specified
	Media: all
	
block-level
The element is block-level, and participates in a block formatting context. Other formatting contexts, such as flex formatting contexts, may also work with block-level elements. [[!CSS21]]
inline-level
The element is inline-level, and participates in an inline formatting context. [[!CSS21]]
none
The element generates no boxes, and does not participate in any formatting context. Note: This value exists for legacy reasons, and interacts with the separate 'display-box' property. It is recommended that 'display-box' be used to suppress an element, so that the element's display type is automatically preserved for when it's no longer suppressed.
table-row-group, table-header-group, table-footer-group, table-row, table-cell, table-column-group, table-column, table-caption
The element is an internal table element, and participates in a table layout context. [[!CSS21]] These values are all layout-specific display roles.
ruby-base, ruby-text, ruby-base-container, ruby-text-container
The element is an internal ruby element, and participates in a ruby layout context. [[!CSS3RUBY]] These values are all layout-specific display roles.
run-in
The element is a run-in element. Run-in elements act like inlines or blocks, depending on the surrounding elements. This is described more fully in a later section.
Some values of 'display-outside' are specialized for particular formatting contexts, and don't have meaning outside of those specific contexts. These values are called layout-specific display roles. Generally, a layout-specific display role will generate wrapper boxes around itself to ensure that it ends up in the correct formatting context, if it's not placed in an appropriate formatting context; the details of this are specified by each layout mode.

Is fantasai's proposal for a run-in model sane enough to include in this spec?

Additional stuff: the 'display-extras' property

	Name: display-extras
	Value: none | list-item
	Initial: none
	Applies to: all elements
	Inherited: no
	Percentages: n/a
	Computed value: as specified
	Media: all
	
list-item
The element generates a ::marker pseudo-element and is considered a list item.

This property is probably dumb, and at the very least has a dumb name. Better names? If I define more one-off weird box-generation details like this, should I merge them into a single "extras" property like this, or just have them all be separate properties?

The 'display' shorthand property

	Name: display
	Value: inline | block | list-item | inline-list-item | inline-block | table | inline-table | table-cell | table-caption | flex | inline-flex | grid | inline-grid | [ <<'display-inside'>> || <<'display-outside'>> || <<'display-extras'>> ]
	Initial: see individual properties
	Applies to: see individual properties
	Inherited: see individual properties
	Computed value: see individual properties
	Animatable: see individual properties
	Media: see individual properties
	
The single-keyword values listed explicitly in the grammar above are handled specially, for legacy reasons. All other single-keyword values, and all other values in general, are handled as normal for shorthands.
inline
Expands identically to ''inline-level auto''.
block
Expands identically to ''block-level block''.
list-item
Expands identically to ''block-level block list-item''.
inline-list-item
Expands identically to ''inline-level block list-item''.
inline-block
Expands identically to ''inline-level block''.
table
Expands identically to ''block-level table''.
inline-table
Expands identically to ''inline-level table''.
table-caption
Expands identically to ''table-caption block''.
table-cell
Expands identically to ''table-cell block''.
flex
Expands identically to ''block-level flex''.
inline-flex
Expands identically to ''inline-level flex''.
grid
Expands identically to ''block-level grid''.
inline-grid
Expands identically to ''inline-level grid''.

Controlling box generation: the 'display-box' property

	Name: display-box
	Value: normal | none | contents
	Initial: normal
	Applies to: all elements
	Inherited: no
	Percentages: n/a
	Computed value: see prose
	Media: all
	
The 'display-box' property is not part of the 'display' shorthand, so that 'display' can be safely set without accidentally overriding whether an element is being suppressed or not. If the computed value of 'display-outside' is ''display-outside/none'', the computed value of 'display-box' is ''display-box/none''. Otherwise, the computed value is the specified value.
normal
The element generates boxes as normal, per its other 'display-*' properties.
none
The element generates no boxes at all.
contents
The element itself does not generate any boxes, but its children and pseudo-elements still generate boxes as normal. For the purposes of box generation and layout, the element must be treated as if it had been replaced with its children and pseudo-elements in the document tree.

Is there a need for a value that suppresses box generation for layout purposes, but still generates them for the purposes of animations/counters/etc.?

''contents'' currently only has an effect on box generation and layout. Other things that care about the document tree are unaffected, like counter scopes. Is this what we want?

Run-In Layout

A run-in element is an element with ''display-outside: run-in''. A run-in element acts differently based on whether its siblings are inlines or blocks; if it precedes a block, it merges into the block. This is useful for formatting compact headlines, definitions, and other similar things, where the appropriate DOM structure is to have a headline preceding the following prose, but the desired display is an inline headline laying out with the text.
For example, dictionary definitions are often formatted so that the word is inline with the definition:
		<dl class='dict'>
			<dt>dictionary
			<dd>a book that lists the words of a language in alphabetical
			    order and gives their meaning, or that gives the equivalent
			    words in a different language.
			<dt>glossary
			<dd>an alphabetical list of terms or words found in or relating
			    to a specific subject, text, or dialect, with explanations; a
			    brief dictionary.
		</dl>
		<style>
		.dict > dt {
			display: run-in;
		}
		.dict > dt::after {
			display: run-in;
			content: ": "
		}
		</style>
		
Which is formatted as:
		dictionary: a book that lists the words of a language
		in alphabetical order and explains their meaning.

		glossary: an alphabetical list of terms or words found
		in or relating to a specific subject, text, or dialect,
		with explanations; a brief dictionary.
		
Whenever there is a sequence of one or more contiguous sibling run-in elements, if the sequence is followed by an element with ''display-inside: block'' and ''display-outside: block-level'', all of the elements in the sequence generate inline-level boxes inside the following block, as if they were ''display-outside: inline-level'', after its ''::marker'' pseudo-element's boxes (if any), but preceding any other boxes generates by the contents of the block (including the box generated by the ''::before'' pseudo-element, if any). Otherwise, the behavior depends on how many sibling run-in elements are in the sequence:
only one
The run-in element generates a block-level box, as if it were ''display-outside: block-level''.
more than one
Each run-in element generates an inline-level box, as if it were ''display-outside: inline-level''. An anonymous block box is generated around the entire sequence of boxes thus generated.

Acknowledgments

We would like to thank the many people who have attempted to separate out the disparate details of box generation over the years, most particularly Bert Bos, whose last attempt with 'display-model' and 'display-role' didn't get anywhere, but primed us for the current spec. We would also like to thank the many JavaScript libraries such as jQuery which have hacked around the "what 'display' should I give it when you call .show()?" problem, making it extremely clear that something needed to be done on our part.