CSS Borders and Box Decorations Module Level 4

Editor’s Draft,

More details about this document
This version:
https://drafts.csswg.org/css-borders-4/
Latest published version:
https://www.w3.org/TR/css-borders-4/
Feedback:
CSSWG Issues Repository
Inline In Spec
Editors:
Elika J. Etemad / fantasai (Apple)
Lea Verou (Invited Expert)
(Invited Expert)
Noam Rosenthal (Google)
Former Editor:
(W3C)
Suggest an Edit for this Spec:
GitHub Editor
Test Suite:
https://wpt.fyi/results/css/css-borders/

Abstract

This module contains the features of CSS relating to the borders and decorations of boxes on the page.

CSS is a language for describing the rendering of structured documents (such as HTML and XML) on screen, on paper, etc.

Status of this document

This is a public copy of the editors’ draft. It is provided for discussion only and may change at any moment. Its publication here does not imply endorsement of its contents by W3C. Don’t cite this document other than as work in progress.

Please send feedback by filing issues in GitHub (preferred), including the spec code “css-borders” in the title, like this: “[css-borders] …summary of comment…”. All issues and comments are archived. Alternately, feedback can be sent to the (archived) public mailing list www-style@w3.org.

This document is governed by the 18 August 2025 W3C Process Document.

1. Introduction

The properties of this module deal with the decoration of the border area. It also defines decorations that can be applied to the box.

1.1. Module Interactions

This specification extends the parts related to borders and box decorations of CSS Backgrounds and Borders Module Level 3 [CSS3BG].

It provides specifications for the added corner-*-shape and border-shape properties, as well as logical shorthands for border-*-radius, box-shadow-* longhands, and partial borders via the border-limit and border-*-clip properties.

All properties in this module apply to the ::first-letter pseudo-element. The border-radius properties also apply to the ::first-line pseudo-element. The UA may (but is not required to) apply the border-image or box-shadow properties to ::first-line. The UA must not apply the border-color/style/width properties to ::first-line. [CSS2]

1.2. Value Definitions

This specification follows the CSS property definition conventions from [CSS2] using the value definition syntax from [CSS-VALUES-3]. Value types not defined in this specification are defined in CSS Values & Units [CSS-VALUES-3]. Combination with other CSS modules may expand the definitions of these value types. For example, combining with CSS Images allows for using CSS gradients as background-image or border-image values. [CSS-IMAGES-3]

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 they have not been repeated explicitly.

2. Borders

The border can either be a predefined style (solid line, double line, dotted line, pseudo-3D border, etc.) or it can be an image. In the former case, various properties define the style (border-style), color (border-color), and thickness (border-width) of the border.

Tests

Tests for features Not yet incorporated from Backgrounds 3


2.1. Line Colors: the border-color properties

Name: border-top-color, border-right-color, border-bottom-color, border-left-color, border-block-start-color, border-block-end-color, border-inline-start-color, border-inline-end-color
Value: <color> | <image-1D>
Initial: currentcolor
Applies to: all elements except ruby base containers and ruby annotation containers
Inherited: no
Percentages: N/A
Computed value: the computed color and/or a one-dimensional image function
Canonical order: per grammar
Animation type: see prose
Logical property group: border-color

These properties set the foreground color of the border specified by the border-style properties.

The stripes defined by <image-1D> follow the shape of the border on the side to which they apply, and are drawn in bands starting from the padding edge and progressing outwards. The border width at each point defines the total width of the stripes at that point.

Using multiple colors for each side:
.foo {
  border: 30px solid;
  border-color: stripes(dodgerblue, skyblue) stripes(yellow, gold) stripes(lightgreen, limegreen) stripes(indianred, orange);
}

Sample rendering:

The same border colors with border-style: dotted:

Name: border-color
Value: [ <color> | <image-1D> ]{1,4}
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

border-color is a shorthand property for the four physical border-*-color properties. The four values set the top, right, bottom and left border, respectively. A missing left is the same as right, a missing bottom is the same as top, and a missing right is also the same as top. This is resolved individually for each list item.

The flow-relative properties border-block-start-color, border-block-end-color, border-inline-start-color, and border-inline-end-color correspond to the physical properties border-top-color, border-bottom-color, border-left-color, and border-right-color. The mapping depends on the element’s writing-mode, direction, and text-orientation.

Name: border-block-color, border-inline-color
Value: <'border-top-color'>{1,2}
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

These two shorthand properties set the border-block-start-color & border-block-end-color and border-inline-start-color & border-inline-end-color, respectively. The first value represents the start side color, and the second value represents the end side color. If only one value is given, it applies to both the start and end sides.

Tests

2.2. Line Patterns: the border-style properties

Name: border-top-style, border-right-style, border-bottom-style, border-left-style, border-block-start-style, border-block-end-style, border-inline-start-style, border-inline-end-style
Value: <line-style>
Initial: none
Applies to: all elements except ruby base containers and ruby annotation containers
Inherited: no
Percentages: N/A
Computed value: specified keyword
Canonical order: per grammar
Animation type: discrete
Logical property group: border-style

These properties control whether a border appears, and if it does what style it’s drawn in (if it is not overridden by a border image).

The flow-relative properties border-block-start-style, border-block-end-style, border-inline-start-style, and border-inline-end-style correspond to the physical properties border-top-style, border-bottom-style, border-left-style, and border-right-style. The mapping depends on the element’s writing-mode, direction, and text-orientation.

Name: border-block-style, border-inline-style
Value: <'border-top-style'>{1,2}
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

These two shorthand properties set the border-block-start-style & border-block-end-style and border-inline-start-style & border-inline-end-style, respectively. The first value represents the start side style, and the second value represents the end side style. If only one value is given, it applies to both the start and end sides.

Name: border-style
Value: <'border-top-style'>{1,4}
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

border-style is a shorthand property for the four physical border-*-style properties. The four values set the top, right, bottom and left border, respectively. A missing left is the same as right, a missing bottom is the same as top, and a missing right is also the same as top. This is resolved individually for each list item.

The style is specified as a <line-style> keyword, where

<line-style> = none | hidden | dotted | dashed | solid | double | groove | ridge | inset | outset

Values have the following meanings:

none
No border. Color and width are ignored (i.e., the border has width 0). Note this means that the initial value of border-image-width will also resolve to zero.
hidden
Same as none, but has different behavior in the border conflict resolution rules for border-collapsed tables [CSS2].
dotted
A series of round dots.
dashed
A series of square-ended dashes.
solid
A single line segment.
double
Two parallel solid lines with some space between them. (The thickness of the lines is not specified, but the sum of the lines and the space must equal border-width.)
groove
Looks as if it were carved in the canvas. (This is typically achieved by creating a “shadow” from two colors that are slightly lighter and darker than the specified border-color.)
ridge
Looks as if it were coming out of the canvas.
inset
Looks as if the content on the inside of the border is sunken into the canvas. Treated as ridge in the collapsing border model. [CSS2]
outset
Looks as if the content on the inside of the border is raised out of the canvas. Treated as groove in the collapsing border model. [CSS2]

Borders are drawn in front of the element’s background, but behind the element’s content (in case it overlaps).

Example renderings of the predefined border styles.

Note: Border colors close to black or white may need different color calculations than colors in between in order to create the required “3D” effect of groove, ridge, inset, or outset.

Note: There is no control over the spacing of the dots and dashes, nor over the length of the dashes. Implementations are encouraged to choose a spacing that makes the corners symmetrical.

Tests

2.3. Line Thickness: the border-width properties

Name: border-top-width, border-right-width, border-bottom-width, border-left-width, border-block-start-width, border-block-end-width, border-inline-start-width, border-inline-end-width
Value: <line-width>
Initial: medium
Applies to: all elements except ruby base containers and ruby annotation containers
Inherited: no
Percentages: N/A
Computed value: absolute length, snapped as a border width; zero if the border style is none or hidden
Canonical order: per grammar
Animation type: by computed value
Logical property group: border-width

These properties specify the thickness of the border, i.e. the border width. Where

<line-width> = <length [0,∞]> | thin | medium | thick

Negative values are invalid. The thin, medium, and thick keywords are equivalent to 1px, 3px, and 5px, respectively.

The flow-relative properties border-block-start-width, border-block-end-width, border-inline-start-width, and border-inline-end-width correspond to the physical properties border-top-width, border-bottom-width, border-left-width, and border-right-width. The mapping depends on the element’s writing-mode, direction, and text-orientation.

Name: border-block-width, border-inline-width
Value: <'border-top-width'>{1,2}
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

These two shorthand properties set the border-block-start-width & border-block-end-width and border-inline-start-width & border-inline-end-width, respectively. The first value represents the start side width, and the second value represents the end side width. If only one value is given, it applies to both the start and end sides.

Name: border-width
Value: <'border-top-width'>{1,4}
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

border-width is a shorthand property for the four physical border-*-width properties. The four values set the top, right, bottom and left border, respectively. A missing left is the same as right, a missing bottom is the same as top, and a missing right is also the same as top. This is resolved individually for each list item.

Note: Although the initial width is medium, the initial style is none; therefore the used initial width is 0.

Tests

2.4. Border Shorthand Properties

Name: border-top, border-right, border-bottom, border-left, border-block-start, border-block-end, border-inline-start, border-inline-end
Value: <line-width> || <line-style> || <color>
Initial: See individual properties
Applies to: all elements except ruby base containers and ruby annotation containers
Inherited: no
Percentages: N/A
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

These shorthand properties set the border-width, border-color, and border-style of one side of the borders of a box. Omitted values are set to their initial values.

The flow-relative properties border-block-start, border-block-end, border-inline-start, and border-inline-end correspond to the physical properties border-top, border-bottom, border-left, and border-right. The mapping depends on the element’s writing-mode, direction, and text-orientation.

Name: border-block, border-inline
Value: <'border-block-start'>
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

These two shorthand properties set the border-block-start & border-block-end or border-inline-start & border-inline-end, respectively, both to the same style.

Name: border
Value: <line-width> || <line-style> || <color>
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar
Tests

The border property is a shorthand property for setting the same border-width, border-color, and border-style for all four borders of a box. Unlike the shorthand margin and padding properties, the border property cannot set different values on the four borders. To do so, one or more of the other border properties must be used.

The border shorthand also resets border-image to its initial value. It is therefore recommended that authors use the border shorthand, rather than other shorthands or the individual properties, to override any border settings earlier in the cascade. This will ensure that border-image has also been reset to allow the new styles to take effect.

Note: The CSS Working Group intends for the border shorthand to reset all border properties in future levels of CSS as well. For example, if a border-characters property is introduced in the future to allow glyphs as borders, it will also be reset by the border shorthand. By using the border shorthand to reset borders, authors can be guaranteed a “blank canvas” no matter what properties are introduced in the future.

For example, the first rule below is equivalent to the set of five rules shown after it:
p { border: solid red }
p {
  border-top: solid red;
  border-right: solid red;
  border-bottom: solid red;
  border-left: solid red;
  border-image: none;
}

Since, to some extent, the properties have overlapping functionality, the order in which the rules are specified is important.

Consider this example:
blockquote {
  border-color: red;
  border-left: double;
  color: black
}

In the above example, the color of the left border is black, while the other borders are red. This is due to border-left setting the width, style, and color. Since the color value is not given by the border-left property, it will be taken from the color property. The fact that the color property is set after the border-left property is not relevant.

3. Corners

The padding edge (inner border) radius is the outer border radius minus the corresponding border thickness. In the case where this results in a negative value, the inner radius is zero. (In such cases the center of the border’s inner curve might not coincide with that of its outer curve.) Likewise the content edge radius is the padding edge radius minus the corresponding padding, or if that is negative, zero. The border and padding thicknesses in the curved region are thus interpolated from the adjoining sides, and when two adjoining borders are of different thicknesses the corner will show a smooth transition between the thicker and thinner borders.

All border styles (solid, dotted, inset, etc.) follow the curve of the border.

The effect of a rounded corner when the two borders it connects are of unequal thickness (left) and the effect of a rounded corner on borders that are thicker than the radius of the corner (right).

Note: If the center of a corner’s outer curve is past an opposite padding edge (in the border area of a side opposite the corner), the inner curve will not be a full quarter ellipse.

p { width: 70px; height: 70px; border: solid 30px;
border-color: orange orange silver silver;
border-top-right-radius: 100%; }
The curved corner is an arc from the top left corner
					          sweeping across the top right corner to the bottom right corner,
					          describing a quarter-ellipse;
					          but since the opposite sides have a border thickness
					          the padding edge curve starts inward from the outer arc's endpoints.
Where the border-radius curve extends into the opposite sides' borders, the arc of the padding edge is less than 90°.

The margin edge, being outside the border edge, calculates its radius by adding the corresponding margin thickness to each border radius. However, in order to create a sharper corner when the border radius is small (and thus ensure continuity between round and sharp corners), when the border radius is less than the margin, the margin is multiplied by the proportion 1 + (r-1)3, where r is the ratio of the border radius to the margin, in calculating the corner radii of the margin box shape.

3.1. Corner Clipping

Although border images are not affected by border-radius, other effects that clip painting or event handling to the border, padding, or content edge must clip to their respective curves. For example, backgrounds clip to the curve specified by background-clip, overflow values other than visible to the curved padding edge (when overflow on both axes is not visible), replaced element content to the curved content edge, pointer events to the curved border edge, etc.

Note: As border-radius reduces the interactive area of an element authors should make sure the remaining interactive area conforms to recommended minima for the platforms they target; in particular, conforming to recommended minimum touch target sizes may require larger widths and heights when border-radius is used.

This example adds appropriate padding, so that the contents do not overflow the corners. Note that there is no border, but the background will still have rounded corners.
DIV {
    background: black;
    color: white;
    border-radius: 1em;
    padding: 1em }

3.2. Color and Style Transitions

Color and style transitions must be contained within the segment of the border that intersects the smallest rectangle that contains both border radii as well as the center of the inner curve (which may be a point representing the corner of the padding edge, if the border radii are smaller than the border width).

If one of these borders is zero-width, then the other border takes up the entire transitional area. Otherwise, the center of color and style transitions between adjoining borders is a point along the curve that is a continuous monotonic function of the ratio of the border widths. However it is not defined what these transitions look like or what function maps from this ratio to a point on the curve.

Given these corner shapes, color and style transitions must be contained within the green region. In case D the rectangle defined by the border radii does not include the center of the inner curve (which is a sharp corner), so the transition region is expanded to include that corner. Transitions may take up the entire transition region, but are not required to: For example, a gradient color transition between two solid border styles might take up only the region bounded by the tips of the outer radii and the tips of the inner radii (represented in case D by the dark green region).

3.3. Overlapping Curves

Corner curves must not overlap: When the sum of any two adjacent border radii exceeds the size of the border box, UAs must proportionally reduce the used values of all border radii until none of them overlap. The algorithm for reducing radii is as follows:

Let f = min(Li/Si), where i ∈ {top, right, bottom, left}, Si is the sum of the two corresponding radii of the corners on side i, and Ltop = Lbottom = the width of the box, and Lleft = Lright = the height of the box. If f < 1, then all corner radii are reduced by multiplying them by f.

Note: This formula ensures that quarter circles remain quarter circles and large radii remain larger than smaller ones, but it may reduce corners that were already small enough, which may make borders of nearby elements that should look the same look different.

If the curve interferes with UI elements such as scrollbars, the UA may further reduce the used value of the affected border radii (and only the affected border radii) as much as necessary, but no more.

For example, the borders A of the figure below might be the result of
box-sizing: border-box;
width: 6em;
height: 2.5em;
border-radius: 0.5em 2em 0.5em 2em

The height (2.5em) is enough for the specified radii (0.5em plus 2.0em). However, if the height is only 2em,

box-sizing: border-box;
width: 6em;
height: 2em;
border-radius: 0.5em 2em 0.5em 2em

all corners need to be reduced by a factor 0.8 to make them fit. The used border radii thus are 0.4em (instead of 0.5em) and 1.6em (instead of 2em). See borders B in the figure.

rectangle with two tiny rounded corners and two very large ones, on opposite corners
These rounded corner might be the result of width: 6em; height: 2.5em; border-radius: 0.5em 2em 0.5em 2em for A; and ditto but with height: 2em for B.

3.4. Effect on Tables

The border-radius properties do apply to table, inline-table, and table-cell boxes in separated borders mode (border-collapse: separate). When border-collapse is collapse, they have no effect.

Tests

3.5. Corner Sizing: the border-*-*-radius properties

Name: border-top-left-radius, border-top-right-radius, border-bottom-right-radius, border-bottom-left-radius, border-start-start-radius, border-start-end-radius, border-end-start-radius, border-end-end-radius
Value: <length-percentage [0,∞]>{1,2}
Initial: 0
Applies to: all elements (but see prose)
Inherited: no
Percentages: Refer to corresponding dimension of the border box.
Computed value: pair of computed <length-percentage> values
Canonical order: per grammar
Animation type: by computed value
Logical property group: border-radius

The two <length-percentage> values of the border-*-radius properties define the radii of a quarter ellipse that defines the shape of the corner of the outer border edge (see the diagram below). The first value is the horizontal radius, the second the vertical radius. If the second value is omitted it is copied from the first. If either length is zero, the corner is square, not rounded. Percentages for the horizontal radius refer to the width of the border box, whereas percentages for the vertical radius refer to the height of the border box. Negative values are invalid.

Diagram of the inscribed ellipse
The two values of border-top-left-radius: 55pt 25pt define the curvature of the corner.
This example draws ovals of 15em wide and 10em high:
DIV.standout {
    width: 13em;
    height: 8em;
    border: solid black 1em;
    border-radius: 7.5em 5em }

The flow-relative properties border-start-start-radius, border-start-end-radius, border-end-start-radius, and border-end-end-radius correspond to the physical properties border-top-left-radius, border-bottom-left-radius, border-top-right-radius, and border-bottom-right-radius. The mapping depends on the element’s writing-mode, direction, and text-orientation, with the first start/end giving the block axis side, and the second the inline-axis side (i.e. patterned as 'border-block-inline-radius').

3.6. Corner Sizing Shorthands: the border-radius and border-*-radius shorthand properties

3.6.1. Sizing The Corners Of One Side: The border-top-radius, border-right-radius, border-bottom-radius, border-left-radius, border-block-start-radius, border-block-end-radius, border-inline-start-radius, border-inline-end-radius shorthands

Name: border-top-radius, border-right-radius, border-bottom-radius, border-left-radius, border-block-start-radius, border-block-end-radius, border-inline-start-radius, border-inline-end-radius
Value: <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]?
Initial: 0
Applies to: all elements (but see prose)
Inherited: no
Percentages: Refer to corresponding dimension of the border box.
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

The border-*-radius shorthands set the two border-*-*-radius longhand properties of the related side. If values are given before and after the slash, then the values before the slash set the horizontal radius and the values after the slash set the vertical radius. If there is no slash, then the values set both radii equally. The two values for the radii are given in the order top-left, top-right for border-top-radius, top-right, bottom-right for border-right-radius, bottom-left, bottom-right for border-bottom-radius, top-left, bottom-left for border-left-radius, start-start, start-end for border-block-start-radius, end-start, end-end for border-block-end-radius start-start, end-start for border-inline-start-radius, and start-end, end-end for border-inline-end-radius. If the second value is omitted it is copied from the first.

Tests

3.6.2. Sizing All Corners At Once: The border-radius shorthand

Name: border-radius
Value: <length-percentage [0,∞]>{1,4} [ / <length-percentage [0,∞]>{1,4} ]?
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

The border-radius shorthand sets all four border-*-radius properties. If values are given before and after the slash, then the values before the slash set the horizontal radii and the values after the slash set the vertical radii. If there is no slash, then the values set the radii in both axes equally. The four values for each radii are given in the order top-left, top-right, bottom-right, bottom-left. If bottom-left is omitted it is the same as top-right. If bottom-right is omitted it is the same as top-left. If top-right is omitted it is the same as top-left.

border-radius: 4em;

is equivalent to

border-top-left-radius:     4em;
border-top-right-radius:    4em;
border-bottom-right-radius: 4em;
border-bottom-left-radius:  4em;

and

border-radius: 2em 1em 4em / 0.5em 3em;

is equivalent to

border-top-left-radius:     2em 0.5em;
border-top-right-radius:    1em 3em;
border-bottom-right-radius: 4em 0.5em;
border-bottom-left-radius:  1em 3em;

3.7. Corner Shaping: the corner-*-shape properties

By default, non-zero border-radius values define a quarter-ellipse corner shape that rounds the affected corners, filling the corner area defined by the border-radius for that corner. However in some cases, other corner shapes are desired. The corner-*-shape properties (and their shorthands) specify exactly what corner shapes a box will use for the region defined by its border-*-radius values.

The different corner shapes can all be expressed as different parameters to a superellipse. A superellipse is a generalization of an ellipse, and based on its `k` parameter can express all the shapes between a square, an ellipse, and a notch.

How does a superellipse work?

A unit circle is defined by the equation:

x 2 + y 2 = 1

The circle is made from all points (x,y) that satisfy the equation. A given ellipse can then be produced by scaling this shape in the X and/or Y axis.

The unit superellipse equation just changes the 2 exponent into a variable. For this spec’s purposes, we’ll write it as 2K:

x 2 K + y 2 K = 1

The K in this equation is the superellipse() argument.

K can be any value; setting K to 1 gives the standard circle/ellipse equation, but other values define the entire family of superellipse curves:

(Note that most literature on superellipses will write the equation with a simpler x K exponent. The x 2 K form was chosen here to make the argument ranges easier to reason about: all possible values are valid, the symmetrical shapes are just positive/negative, the "middle" bevel is 0, etc.)

To allow full expression as well as interpolation, the corner-shape properties can provide the superellipse parameter directly using the superellipse() function, or use one of the supplied keywords which represent commonly used parameters. See the <corner-shape-value> definition for details.

Tests
Name: corner-top-left-shape, corner-top-right-shape, corner-bottom-right-shape, corner-bottom-left-shape, corner-start-start-shape, corner-start-end-shape, corner-end-start-shape, corner-end-end-shape
Value: <corner-shape-value>
Initial: round
Applies to: all elements where border-radius can apply
Inherited: no
Percentages: n/a
Computed value: the corresponding superellipse() value
Canonical order: per grammar
Animation type: see superellipse interpolation
Logical property group: corner-shape

The corner-*-*-shape longhand properties set the corner shape for the given corner.

The flow-relative longhands (corner-start-start-shape, etc) correspond to the physical longhands (corner-top-left-shape, etc) depending on the element’s writing-mode, direction, and text-orientation. The first start/end gives the block axis side, and the second the inline-axis side (i.e. patterned as corner-block-inline-shape).

<corner-shape-value> = round | scoop | bevel | notch | square | squircle |
                       <superellipse()>
superellipse() = superellipse(<number [-∞,∞]> | infinity | -infinity)
round
The corner shape is a quarter of a convex ellipse. Equivalent to superellipse(1).

Note: This is the initial value of corner-shape properties, as it was the behavior of border-radius before corner-shape existed.

squircle
The corner shape is a quarter of a "squircle", a convex curve between round and square. Equivalent to superellipse(2).
square
The corner shape is a convex 90deg angle. Equivalent to superellipse(infinity).

Note: This looks identical to the "normal" square corner you get from border-radius: 0, but it can smoothly animate with the other corner-shape values.

bevel
The corner shape is a straight diagonal line, neither convex nor concave. Equivalent to superellipse(0).
scoop
The corner shape is a concave quarter-ellipse. Equivalent to superellipse(-1).
notch
The corner shape is a concave 90deg angle. Equivalent to superellipse(-infinity).
superellipse(K)
The corner shape is a quarter of a superellipse. The argument K is the superellipse parameter, and it defines a superellipse using an exponent of 2K.

See the note in § 3.7 Corner Shaping: the corner-*-shape properties for an explanation of the mathematical definition of a superellipse, and what various K values mean. See § 3.9.4 Rendering corner-shape for precise details of how the superellipse is computed and rendered.

Different superellipse() values for the top right corner: infinity, 1, 0, -1, and -infinity.

Note: If border-radius is not specified (or is set to 0), the corner area is zero-sized as well, and corner-shape won’t have any effect.

corner-*-shape does not alter the overflow rules for border-*-radius, except insofar as it shapes the corners differently; elements are still clipped by the shaped border as normal.

The curve specified by corner-*-shape defines the outer edge of the border. The inner edge of the border follows the curve of the outer edge (in a way that’s not necessarily expressible as a superellipse curve), with a nearly consistent distance from the outer edge throughout, (or a linearly increasing distance if the border-width of the two border edges meeting at the the corner are not uniform).

corner-*-shape also affects the rendering of box-shadow, and how the overflow clip edge is shaped when it’s extended from the box, but these do not directly follow the corner-*-shape path like the inner border edge does. Instead, it scales the corner-shape path in an axis-aligned manner.

3.8. Corner Shaping Shorthands: the corner-shape and corner-*-shape shorthand properties

3.8.1. Shaping The Corners Of One Side: The corner-*-shape shorthands:

Name: corner-top-shape, corner-right-shape, corner-bottom-shape, corner-left-shape, corner-block-start-shape, corner-block-end-shape, corner-inline-start-shape, corner-inline-end-shape
Value: <'corner-top-left-shape'>{1,2}
Initial: see individual properties
Applies to: see individual properties
Inherited: see individual properties
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

The corner-*-shape/etc shorthands set the two corner-*-*-shape properties of the related side. If only one value is given, the second value defaults to the same value.

For the physical shorthands (corner-top-shape, etc), the values are either in left/right order, or top/bottom order, whichever axis is meaningful for the property. That is, corner-top-shape: round square sets corner-top-left-shape: round; corner-top-right-shape: square;.

For the logical shorthands (corner-block-start-shape, etc), the values are always in start/end order in the other axis. That is, corner-block-start-shape: round square sets corner-start-start-shape: round; corner-start-end-shape: square;.

3.8.2. Shaping All Corners At Once: The corner-shape shorthand

Name: corner-shape
Value: <'corner-top-left-shape'>{1,4}
Initial: round
Applies to: all elements where border-radius can apply
Inherited: no
Percentages: see individual properties
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

The corner-shape property specifies the shape of the box’s corners, within the region specified by border-radius. The four values set the top, right, bottom and left shape, respectively. A missing left is the same as right, a missing bottom is the same as top, and a missing right is also the same as top.

3.9. Corner Sizing & Shaping Shorthands: the corner and corner-* shorthand properties

3.9.1. Sizing & shaping a corner: The corner-top-left, corner-top-right, corner-bottom-right, corner-bottom-left, corner-start-start, corner-start-end, corner-end-start, corner-end-end shorthands

Name: corner-top-left, corner-top-right, corner-bottom-left, corner-bottom-right, corner-start-start, corner-start-end, corner-end-start, corner-end-end
Value: <'border-top-left-radius'> || <'corner-top-left-shape'>
Initial: 0
Applies to: all elements (but see prose)
Inherited: no
Percentages: Refer to corresponding dimension of the border box.
Computed value: see individual properties
Canonical order: per grammar
Animation type: see individual properties

The corner-*-* shorthands set the two corner-*-*-shape and border-*-*-radius longhand properties of the related side. See the corresponding corner-*-*-shape and border-*-*-radius properties for further details.

3.9.2. Sizing & shaping rhe Corners Of One Side: The corner-top, corner-right, corner-bottom, corner-left, corner-block-start, corner-block-end, corner-inline-start, corner-inline-end shorthands

Name: corner-top, corner-right, corner-bottom, corner-left, corner-block-start, corner-block-end, corner-inline-start, corner-inline-end
Value: <'border-top-radius'> || <'corner-top-shape'>
Initial: 0
Applies to: all elements (but see prose)
Inherited: no
Percentages: Refer to corresponding dimension of the border box.
Computed value: see individual properties
Animation type: see individual properties
Canonical order: per grammar

The corner-* shorthands set the two corner-*-shape longhand properties and two border-*-radius longhand properties of the related side. See the corresponding corner-*-shape and border-*-radius properties for further details.

3.9.3. Sizing & Shaping All Corners At Once: The corner shorthand

Name: corner
Value: <'border-radius'> || <'corner-shape'>
Initial: 0
Applies to: all elements (but see prose)
Inherited: no
Percentages: Refer to corresponding dimension of the border box.
Computed value: see individual properties
Canonical order: per grammar
Animation type: see individual properties

The corner shorthand sets the corner-*-shape and border-*-radius longhands all together.

3.9.4. Rendering corner-shape

When rendering elements with shaped corners, the element’s path needs to be offset, based on border, outline, box-shadow, overflow-clip-margin and more.

When rendering borders or outlines, the offset is aligned to the curve of the element’s shape, while when rendering box-shadow or offsetting for overflow-clip-margin, the offset is aligned to the axis.

Adjusting corner shapes
Borders are aligned to the curve, shadows and clip are aligned to the axis.

An element element’s outer contour is the border contour path given element and element’s border edge.

An element element’s inner contour is the border contour path given element and element’s padding edge.

An element’s border is rendered in the area between its outer contour and its inner contour.

An element’s outline follows the outer contour with the used outline-width and outline-offset. The precise way in which it is rendered is implementation-defined.

An element’s overflow area is shaped by its inner contour. An element’s overflow clip edge is shaped by the border contour path given element, and element’s padding edge, and element’s used overflow-clip-margin.

Each shadow of element’s 'box shadow' is shaped by the border contour path given element, and element’s border edge, and the shadow’s used box-shadow-spread.

To compute an element element’s border contour path given an edge targetEdge and an optional number spread (default 0):
  1. Let outerLeft, outerTop, outerRight, outerBottom be element’s unshaped border edge, outset by spread.

  2. Let topLeftHorizontalRadius, topLeftVericalRadius, topRightHorizontalRadius, topRightVerticalRadius, bottomRightHorizontalRadius, bottomRightVerticalRadius, bottomLeftHorizontalRadius, and bottomLeftVerticalRadius be element border edge’s radii, scaled by element’s opposite corner scale factor and outset-adjusted.

  3. Let topLeftShape, topRightShape, bottomRightShape, and bottomLeftShape be element’s computed corner-*-shape values.

  4. Let targetLeft, targetTop, targetRight, targetBottom unshaped targetEdge.

  5. Let path be a new path [SVG2].

  6. Add corner to path given path, the rectangle (outerRight - topRightHorizontalRadius, outerTop, topRightHorizontalRadius, topRightVerticalRadius), targetEdge, 0, targetTop - outerTop, outerRight - targetRight, and topRightShape.

  7. Add corner to path given path, the rectangle (outerRight - bottomRightHorizontalRadius, outerBottom - bottomRightVerticalRadius, bottomRightHorizontalRadius, bottomRightVerticalRadius), targetEdge, 1, outerRight - targetRight, outerBottom - targetBottom, and bottomRightShape.

  8. Add corner to path given path, the rectangle (outerLeft, outerBottom - bottomLeftVerticalRadius, bottomLeftHorizontalRadius, bottomLeftVerticalRadius), targetEdge, 2, outerBottom - targetBottom, targetLeft - outerLeft, and bottomLeftShape.

  9. Add corner to path given path, the rectangle (outerLeft, outerTop, topLeftHorizontalRadius, topLeftVericalRadius), targetEdge, 3, targetLeft - outerLeft, targetTop - outerTop, and topLeftShape.

  10. Return path.

To add corner to path given a path path, a rectangle cornerRect, a rectangle trimRect, and numbers orientation, startThickness, endThickness, curvature:

  1. If cornerRect is empty, or if curvature is ∞:

    1. Let innerQuad be trimRect’s clockwise quad .

    2. Extend path by drawing a line to innerQuad[(orienation + 1) % 4].

    3. Return.

  2. Let cornerQuad be cornerRect’s clockwise quad.

  3. If curvature is -∞:

    1. Extend path by drawing a line from cornerQuad[0] to cornerQuad[3], trimmed by trimRect.

    2. Extend path by drawing a line from cornerQuad[3] to cornerQuad[2], trimmed by trimRect.

    3. Return.

  4. Let clampedNormalizedHalfCorner be the normalized superellipse half corner given clamp(curvature, -1, 1).

  5. Let equivalentQuadraticControlPointX be clampedNormalizedHalfCorner * 2 - 0.5.

  6. Let curveStartPoint be the aligned corner point given cornerQuad[orienation], the vector (equivalentQuadraticControlPointX, 1 - equivalentQuadraticControlPointX), startThickness, and orientation + 1.

  7. Let curveEndPoint by the aligned corner point given cornerQuad[(orientation + 2) % 4], the vector (equivalentQuadraticControlPointX - 1, -equivalentQuadraticControlPointX), endThickness, and orientation + 3.

  8. Let alignedCornerRect be a rectangle that includes the points curveStartPoint and curveEndPoint.

  9. Let projectionToCornerRect be a transformation matrix, translated by (alignedCornerRect’s x coordinate, alignedCornerRect’s y coordinate), scaled by (alignedCornerRect’s width dimension, alignedCornerRect’s height dimension), translated by (0.5, 0.5), rotated by 90deg * orientation, and translated by (-0.5, -0.5).

  10. Let K be 0.5abs(curvature).

  11. For each T between 0 and 1:

    1. Let A be TK.

    2. Let B be 1 - (1 - T)K.

    3. Let normalizedPoint be (A, B) if curvature is positive, otherwise (B, A).

    4. Let absolutePoint be normalizedPoint, transformed by projectionToCornerRect.

    5. If absolutePoint is within trimRect, extend path through absolutePoint.

    Note: User agents may approximate this algorithm, for instance, by using concatenated Bezier curves, to balance between performance and rendering accuracy.

To compute the aligned corner point given a point originalPoint, a two-component vector offsetFromControlPoint, a number thickness, and a number orientation:

  1. Let length be hypot(offsetFromControlPoint.x, offsetFromControlPoint.y).

  2. Rotate offsetFromControlPoint by 90deg * orientation, and scale by thickness.

  3. Translate originalPoint by offsetFromControlPoint.x / length, offsetFromControlPoint.y / length, and return the result.

The clockwise quad given a rectangle rect, is a quadrilateral with the points (rect’s x coordinate, rect’s y coordinate), (rect’s x coordinate + rect’s width dimension, rect’s y coordinate), (rect’s x coordinate + rect’s width dimension, rect’s y coordinate + rect’s height dimension), (rect’s x coordinate, rect’s y coordinate + rect’s height dimension).

Tests

3.9.5. Constraining opposite radii

When concave corner-shape values are present (the superellipse parameter is negative), diagonally opposite corners might overlap each other.

The following example would create overlapping corners if not constrained.

div {
  corner-shape: scoop;
  border-top-left-radius: 80%;
  border-bottom-right-radius: 80%;
}

To prevent this, the four radii are constrained to prevent overlaps. This is done by computing a hull polygon for each of the opposite corners, and finding the highest downscale factor which, if applied to both corners, would make it so that the polygons would not intersect.

To compute the opposite corner scale factor given an element element:
  1. Let rect be element’s border box.

  2. Let topRightHull be a the normalized inner corner hull given element’s computed corner-top-right-shape, mapped to the rectangle (rect’s width dimension - element’s computed horizontal border-top-right-radius, 0, rect’s computed border-top-right-radius).

  3. Let bottomRightHull be a the normalized inner corner hull given element’s computed corner-bottom-right-shape, rotated by 90deg with (0.5, 0.5) as an origin, and mapped to the rectangle (rect’s width dimension - element’s computed horizontal border-bottom-right-radius, rect’s height dimension - element’s computed vertical border-bottom-right-radius, element’s computed border-bottom-right-radius).

  4. Let bottomLeftHull be a the normalized inner corner hull given element’s computed corner-bottom-right-shape, rotated by 180deg with (0.5, 0.5) as an origin, and mapped to the rectangle (0, rect’s height dimension - element’s computed vertical border-bottom-left-radius, element’s computed border-bottom-left-radius).

  5. Let topLeftHull be a the normalized inner corner hull given element’s computed corner-top-left-shape, rotated by 270deg with (0.5, 0.5) as an origin, mapped to (0, 0, element’s computed border-top-left-radius).

  6. Let scaleFactorA be the highest number which, if both topLeftHull and bottomRightHull were scaled by, using their first point as the origin, those polygons would not intersect.

  7. Let scaleFactorB be the highest number which, if both topRightHull and bottomLeftHull were scaled by, using their first point as the origin, those polygons would not intersect.

  8. Return min(1, scaleFactorA, scaleFactorB).

3.9.6. Interpolating corner shapes

Since a <corner-shape-value> can always be expressed by a superellipse() with an superellipse parameter variable, interpolating between two <corner-shape-value>s is done by interpolating the superellipse parameter itself. Since it uses a log2, interpolating it linearly would result in an effect where concave corners interpolate at a much higher velocity than convex corners. To balance that, the superellipse interpolation formula describes how a superellipse parameter is converted to a value between 0 and 1, and vice versa:

To compute the normalized superellipse half corner given a superellipse parameter s, return the first matching statement, switching on s:
-∞

Return 0.

Return 1.

Otherwise
  1. Let k be 0.5abs(s).

  2. Let convexHalfCorner be 0.5k.

  3. If s is less than 0, return 1 - convexHalfCorner.

  4. Return convexHalfCorner.

To compute the normalized inner corner hull given a superellipse parameter curvature:
  1. If curvature is greater than or equal to zero, return a triangle betwen « (1, 1), (1, 0), (0, 1) ».

  2. Let axisLineA be a line between (1, 0) and (1, 1).

  3. Let axisLineB be a line between (0, 1) and (1, 1).

  4. Let normalizedHalfCorner be the normalized superellipse half corner given curvature.

  5. Let halfCornerPoint be (normalizedHalfCorner, 1 - normalizedHalfCorner).

  6. Let lineFromCenterToHalfCorner be a line between (0, 0) and halfCornerPoint.

  7. Let tangentLine be the line perpendicular to lineFromCenterToHalfCorner, at halfCornerPoint.

  8. Let intersectionA be the intersection between axisLineA and tangentLine.

  9. Let intersectionB be the intersection between axisLineB and tangentLine.

  10. Return a pentagon between the points « (1, 1), (1, 0), intersectionA, intersectionB, (0, 1), (1, 1) ».

To interpolate a superellipse parameter s to an interpolation value between 0 and 1, return the normalized superellipse half corner given s.

To convert a <number [0,1]> interpolationValue back to a superellipse parameter, switch on interpolationValue:
0

Return -∞.

0.5

Return 0.

1

Return ∞.

Otherwise
  1. Let convexHalfCorner be interpolationValue.

  2. If interpolationValue is less than 0.5, set convexHalfCorner to 1 - interpolationValue.

  3. Let k be ln(0.5) / ln(convexHalfCorner).

  4. Let s be log2(k).

  5. If interpolationValue is less than 0.5, return -s.

  6. Return s.

Tests

4. Border Images

Authors can specify an image to be used in place of the border styles. In this case, the border’s design is taken from the sides and corners of an image specified with border-image-source, whose pieces may be sliced, scaled, and stretched in various ways to fit the size of the border image area. The border-image properties do not affect layout: layout of the box, its content, and surrounding content is based on the border-width and border-style properties only.

This example creates a top and bottom border consisting of a whole number of orange diamonds and a left and right border of a single, stretched diamond. The corners are diamonds of a different color. The image to tile is as follows. Apart from the diamonds, it is transparent:

Tile for border

The image is 81 by 81 pixels and has to be divided into 9 equal parts. The style rules could thus be as follows:

DIV {
  border: double orange 1em;
  border-image: url("border.png") 27 round stretch;
 }

The result, when applied to a DIV of 12 by 5em, will be similar to this:

element with a diamond border

This shows a more complicated example, demonstrating how the border image corresponds to the fallback border-style but can also extend beyond the border area. The border image is a wavy green border with an extended corner effect:
Diagram: The border image shows a wavy green border
		          with more exaggerated waves towards the corners,
		          which are capped by a disconnected green circle.
		          Four cuts at 124px offsets from each side
		          divide the image into 124px-wide square corners,
		          124px-wide but thin side slices,
		          and a small center square.
The border-image-source image, with the four border-image-slice cuts at 124px dividing the image into nine parts.

The rest of the border properties then interact to lay out the tiles as follows:

Diagram: The image-less (fallback) rendering
		          has a green double border.
		          The rendering with border-image shows the wavy green border,
		          ith the waves getting longer as they reach the corners.
		          The corner tiles render as 124px-wide squares
		          and the side tiles repeat a whole number of times
		          to fill the space in between.
		          Because of the gradual corner effects,
		          the tiles extend deep into the padding area.
		          The whole border image effect is outset 31px,
		          so that the troughs of the waves align
		          just outside the padding edge.
Diagram of all border-image properties and how they interact, and showing the rendering with and without the border-image in effect.

Here, even though the border-width is 12px, the border-image-width property computes to 124px. The border image area is then outset 31px from the border box and into the margin area. If the border-image fails to load (or if border images are not supported by the UA), the fallback rendering uses a green double border.

Notice that the border shorthand resets border-image. This makes it easy to turn off or reset all border effects:
.notebox {
  border: double orange;
  /* must set 'border' shorthand first, otherwise it erases 'border-image' */
  border-image: url("border.png") 30 round;
  /* but other 'border' properties can be set after */
  border-width: thin thick;
}
...
.sidebar .notebox {
  box-shadow: 0 0 5px gray;
  border-radius: 5px;
  border: none; /* turn off all borders */
  /* 'border' shorthand resets 'border-image' */
}

4.1. Image Source: the border-image-source property

Name: border-image-source
Value: none | <image>
Initial: none
Applies to: All elements, except internal table elements when border-collapse is collapse
Inherited: no
Percentages: N/A
Computed value: the keyword none or the computed <image>
Canonical order: per grammar
Animation type: discrete
Tests

Specifies an image to use as a border in place of the rendering specified by the border-style properties and, if given the fill keyword in border-image-slice, as an additional image backdrop for the element. If the value is none or if the image cannot be displayed (or the property doesn’t apply), the border styles will be used; otherwise the element’s border-style borders are not drawn and this border image is drawn as described in the sections below.

4.2. Image Slicing: the border-image-slice property

Name: border-image-slice
Value: [<number [0,∞]> | <percentage [0,∞]>]{1,4} && fill?
Initial: 100%
Applies to: All elements, except internal table elements when border-collapse is collapse
Inherited: no
Percentages: refer to size of the border image
Computed value: four values, each either a number or percentage; plus a fill keyword if specified
Canonical order: per grammar
Animation type: by computed value
Tests

This property specifies inward offsets from the top, right, bottom, and left edges of the image, dividing it into nine regions: four corners, four edges and a middle. The middle image part is discarded (treated as fully transparent) unless the fill keyword is present. (It is drawn over the background; see Drawing the Border Image.)

If there is only one component value, it applies to all sides. If there are two values, the top and bottom are set to the first value and the right and left are set to the second. If there are three values, the top is set to the first value, the left and right are set to the second, and the bottom is set to the third. If there are four values they apply to the top, right, bottom, and left, respectively.

<percentage [0,∞]>
Percentages are relative to the size of the image: the width of the image for the horizontal offsets, the height for vertical offsets.
<number [0,∞]>
Numbers represent pixels in the image (if the image is a raster image) or vector coordinates (if the image is a vector image).
fill
The fill keyword, if present, causes the middle part of the border-image to be preserved. (By default it is discarded, i.e., treated as empty.)

Negative values are invalid. Computed values larger than the size of the image are interpreted as 100%.

The regions given by the border-image-slice values may overlap. However if the sum of the right and left widths is equal to or greater than the width of the image, the images for the top and bottom edge and the middle part are empty—​which has the same effect as if a nonempty transparent image had been specified for those parts. Analogously for the top and bottom values.

If the image must be sized to determine the slices (for example, for SVG images with no natural dimensions), then it is sized using the default sizing algorithm with no specified size and the border image area as the default object size.

Diagram: two horizontal cuts and two vertical cuts through an image
Diagram illustrating the cuts corresponding to the value 25% 30% 12% 20%

4.3. Drawing Areas: the border-image-width property

Name: border-image-width
Value: [ <length-percentage [0,∞]> | <number [0,∞]> | auto ]{1,4}
Initial: 1
Applies to: All elements, except internal table elements when border-collapse is collapse
Inherited: no
Percentages: Relative to width/height of the border image area
Computed value: four values, each either a number, the keyword auto, or a computed <length-percentage> value
Canonical order: per grammar
Animation type: by computed value
Tests

The border image is drawn inside an area called the border image area. This is an area whose boundaries by default correspond to the border box, see border-image-outset.

The four values of border-image-width specify offsets that are used to divide the border image area into nine regions. The offsets represent inward distances from the top, right, bottom, and left sides of the area, respectively.

If there is only one component value, it applies to all sides. If there are two values, the top and bottom are set to the first value and the right and left are set to the second. If there are three values, the top is set to the first value, the left and right are set to the second, and the bottom is set to the third. If there are four values they apply to the top, right, bottom, and left, respectively.

Values have the following meanings:

<length-percentage [0,∞]>
Percentages refer to the size of the border image area: the width of the area for horizontal offsets, the height for vertical offsets.
<number [0,∞]>
Numbers represent multiples of the corresponding computed border-width.
auto
If auto is specified then the used border-image-width is the natural width or height (whichever is applicable) of the corresponding image slice (see border-image-slice). If the image does not have the required natural dimension then the corresponding computed border-width is used instead.

Negative values are invalid for any border-image-width values.

If two opposite border-image-width offsets are large enough that they overlap, then the used values of all border-image-width offsets are proportionally reduced until they no longer overlap. In mathematical notation: Given Lwidth as the width of the border image area, Lheight as its height, and Wside as the border-image-width offset for the side side, let f = min(Lwidth/(Wleft+Wright), Lheight/(Wtop+Wbottom)). If f < 1, then all W are reduced by multiplying them by f.

4.4. Edge Overhang: the border-image-outset property

Name: border-image-outset
Value: [ <length [0,∞]> | <number [0,∞]> ]{1,4}
Initial: 0
Applies to: All elements, except internal table elements when border-collapse is collapse
Inherited: no
Percentages: N/A
Computed value: four values, each a number or absolute length
Canonical order: per grammar
Animation type: by computed value
Tests

The values specify the amount by which the border image area extends beyond the border box.

If there is only one component value, it applies to all sides. If there are two values, the top and bottom are set to the first value and the right and left are set to the second. If there are three values, the top is set to the first value, the left and right are set to the second, and the bottom is set to the third. If there are four values they apply to the top, right, bottom, and left, respectively.

<length [0,∞]>
Represents an outset of the specified length.
<number [0,∞]>
Represents an outset of the specified multiple of the corresponding computed border-width.

Negative values are invalid.

Portions of the border-image that are rendered outside the border box are ink overflow and do not trigger scrolling. Also such portions are invisible to mouse events and do not capture such events on behalf of the element.

Note: Even though they never cause a scrolling mechanism, outset images may still be clipped by an ancestor or by the viewport.

4.5. Image Tiling: the border-image-repeat property

Name: border-image-repeat
Value: [ stretch | repeat | round | space ]{1,2}
Initial: stretch
Applies to: All elements, except internal table elements when border-collapse is collapse
Inherited: no
Percentages: N/A
Computed value: two keywords, one per axis
Canonical order: per grammar
Animation type: discrete
Tests

This property specifies how the images for the sides and the middle part of the border image are scaled and tiled. The first keyword applies to the horizontal scaling and tiling of the top, middle and bottom parts, the second to the vertical scaling and tiling of the left, middle and right parts; see Drawing the Border Image. If the second keyword is absent, it is assumed to be the same as the first. Values have the following meanings:

stretch
The image is stretched to fill its corresponding region.
repeat
The image is tiled (repeated) to fill its corresponding region.
round
The image is tiled (repeated) to fill its corresponding region. If it does not fill the area with a whole number of tiles, the image is rescaled so that it does.
space
The image is tiled (repeated) to fill its corresponding region. If it does not fill the region with a whole number of tiles, the extra space is distributed around the tiles.

The exact process for scaling and tiling the border-image parts is given in the section below.

4.6. Drawing the Border Image

After the border image given by border-image-source is sliced by the border-image-slice values, the resulting nine images are scaled, positioned, and tiled into their corresponding border image regions in four steps:

  1. Scale to border-image-width.
    • The two images for the top and bottom edges are made as tall as the top and bottom border image regions, respectively, and their width is scaled proportionally.
    • The images for the left and right edges are made as wide as the left and right border image regions, respectively, and their height is scaled proportionally.
    • The corner images are scaled to be as wide and as tall as the their respective border image regions.
    • The middle image’s width is scaled by the same factor as the top image unless that factor is zero or infinity, in which case the scaling factor of the bottom is substituted, and failing that, the width is not scaled. The height of the middle image is scaled by the same factor as the left image unless that factor is zero or infinity, in which case the scaling factor of the right image is substituted, and failing that, the height is not scaled.
  2. Scale to border-image-repeat.
    • If the first keyword is stretch, the top, middle and bottom images are further scaled to be as wide as the middle region of the border image area. The height is not changed any further.
    • If the first keyword is round, the top, middle and bottom images are resized in width, so that exactly a whole number of them fit in the middle region of the border image area, exactly as for round in the background-repeat property.
    • If the first keyword is repeat or space, the top, middle, and bottom images are not changed any further.
    • The effects of stretch, round, repeat, and space for the second keyword are analogous, acting on the height of the left, middle and right images.
  3. Position the first tile.
    • If the first keyword is repeat, the top, middle, and bottom images are centered horizontally in their respective regions. Otherwise the images are placed at the left edge of their respective regions of the border image area.
    • If the second keyword is repeat, the left, middle, and right images are centered vertically in their respective regions. Otherwise the images are placed at the top edge of their respective regions of the border image area.
  4. Tile and draw.
    • The images are then tiled to fill their respective regions.
    • In the case of space, any partial tiles are discarded and the extra space distributed before, after, and between the tiles. (I.e. the gap before the first tile, the gap after the last tile, and the gaps between tiles are equalized.) This can result in empty border-image side regions.
    • The images are drawn at the same stacking level as normal borders: immediately in front of the background layers.
    • The middle image is not drawn unless fill was specified for border-image-source.

4.7. Border Image Shorthand: the border-image property

Name: border-image
Value: <'border-image-source'> || <'border-image-slice'> [ / <'border-image-width'> | / <'border-image-width'>? / <'border-image-outset'> ]? || <'border-image-repeat'>
Initial: See individual properties
Applies to: See individual properties
Inherited: no
Percentages: N/A
Computed value: See individual properties
Canonical order: per grammar
Animation type: See individual properties
Tests

This is a shorthand property for setting border-image-source, border-image-slice, border-image-width, border-image-outset, and border-image-repeat in a single declaration. Omitted values are set to their initial values.

4.8. Effect on Tables

The border-image properties apply to the border of tables and inline tables that have border-collapse set to collapse. However, this specification does not define how such an image border is rendered. In particular, it does not define how the image border interacts with the borders of cells, rows and row groups at the edges of the table (see border conflict resolution in [CSS2]).

It is expected that a future specification will define the rendering. It is recommended that UAs do not apply border images to tables with collapsed borders until then.

5. Partial borders

Not Ready For Implementation

This section is not yet ready for implementation. It exists in this repository to record the ideas and promote discussion.

Before attempting to implement anything of this section, please contact the CSSWG at www-style@w3.org.

CSS borders traditionally cover an entire border edge. Sometimes, however, it can be useful to hide some parts of the border.

Here are two proposals for doing this: the second one is from GCPM, the first one is an attempt to recast it more readably. The names are terrible, known problem, proposals accepted. There is a problem with conceiving this as clipping: if you have dotted borders, you want whole dots always, not parts of dots. So it should be a drawing limit, not a clip.

5.1. Partial Borders: the border-limit property

Name: border-limit
Value: all | [ sides | corners ] <length-percentage [0,∞]>? | [ top | right | bottom | left ] <length-percentage [0,∞]>
Initial: all
Applies to: all elements, except table element when border-collapse is collapse
Inherited: no
Percentages: relative to border-box
Computed value: as specified
Canonical order: per grammar
Animation type: discrete

By default, the entire border is drawn. However, border rendering can be limited to only part of a border. The keyword specifies which part, and the length or percentage specifies how much.

all
The entire border is drawn.
sides
The sides are drawn up to but not including the corners (as defined by the border radii). A length or percentage is measured from the center of each side: 50% draws the middle 50% of the border; by default the entire side is drawn.
corners
The corners are drawn plus the specified distance into the sides if specified. A length is measured from the closest edge of the corner area. A percentage is measured from the absolute corner of the border box.
left
right
For the left and right (vertical) sides, draws the entire side and corner. For the top and bottom (horizontal) sides, draws the left/right portion, as specified. Distances are measured as for corners.
top
bottom
For the top and bottom (horizontal) sides, draws the entire side and corner. For the left and right (vertical) sides, draws the top/bottom portion, as specified. Distances are measured as for corners.

The following example draws only the middle 50% of the sides.

div {
  border: solid;
  border-limit: sides 50%;
}

The following example draws only the curved parts of the corners.

div {
  border: solid;
  border-radius: 1em 2em;
  border-limit: corners;
}

The following example draws only the left 4em of the top border.

div {
  border-top: solid;
  border-limit: left 4em;
}

The following example draws only the first 10px of each corner:

div {
  border: solid;
  border-limit: corners 10px;
}

The following example draws the curved part of the corner plus 5px along the sides:

div {
  border: solid;
  border-radius: 5px;
  border-limit: corners 5px;
}

The following example draws the curved part of the corner and all of the side except the middle 40%.

div {
  border: solid;
  border-radius: 5px;
  border-limit: corners 30%;
}

5.2. The border-clip properties

Name: border-clip, border-clip-top, border-clip-right, border-clip-bottom, border-clip-left
Value: normal | [ <length-percentage [0,∞]> | <flex> ]+
Initial: normal
Applies to: all elements
Inherited: no
Percentages: refer to length of border-edge side
Computed value: normal, or a list consisting of absolute lengths, or percentages as specified
Canonical order: per grammar
Animation type: by computed value

These properties split their respective borders into parts along the border edge. The first part is visible, the second is invisible, the third part is visible, etc. Parts can be specified with lengths, percentages, or flexible lengths (expressed by the fr unit, as per [CSS3GRID]). The normal value means that the border is not split, but shown normally.

border-clip is a shorthand property for the four individual properties.

If the listed parts are shorter than the border, any remaining border is split proportionally between the specified flexible lengths. If there are no flexible lengths, the behavior is as if 1fr had been specified at the end of the list.

If the listed parts are longer than the border, the specified parts will be shown in full until the end of the border. In this case, all flexible lengths will be zero.

For horizontal borders, parts are listed from left to right. For vertical borders, parts are listed from top to bottom.

The exact border parts are determined by laying out the specified border parts with all flexible lengths initially set to zero. Any remaining border is split proportionally between the flexible lengths specified.

border-clip: 10px 1fr 10px;
border-clip-top: 10px 1fr 10px;
border-clip-bottom: 10px 1fr 10px;
border-clip-right: 5px 1fr 5px;
border-clip-left: 5px 1fr 5px;

By making the first part have zero length, the inverse border of the previous example can easily be created:

border-clip-top: 0 10px 1fr 10px;
border-clip-bottom: 0 10px 1fr 10px;
border-clip-right: 0 5px 1fr 5px;
border-clip-left: 0 5px 1fr 5px;
border: thin solid black;
border-clip: 0 1fr; /* hide borders */
border-clip-top: 10px 1fr 10px; /* make certain borders visible */
border-clip-bottom: 10px 1fr 10px;
border-top: thin solid black;
border-bottom: thin solid black;
border-clip-top: 10px;
border-clip-bottom: 10px;
border-top: thin solid black;
border-clip: 10px;

This rendering:

A sentence consists of words¹.
¹ Most often.
can be achieved with this style sheet:
@footnote {
  border-top: thin solid black;
  border-clip: 4em;
}
border: 4px solid black;
border-clip-top: 40px 20px 0 1fr 20px 20px 0 1fr 40px;

In this example, there will be a visible 40px border part on each end of the top border. Inside the 40px border parts, there will be an invisible border part of at least 20px. Inside these invisible border parts, there will be visible border parts, each 20px long with 20px invisible border parts between them.

The fragments are shown in red for illustrative purposes; they should not be visible in compliant UAs.

border: 4px solid black;
border-clip-top: 3fr 10px 2fr 10px 1fr 10px 10px 10px 1fr 10px 2fr 10px 3fr;

All but one of the visible border parts are represented as flexible lengths in this example. The length of these border parts will change when the width of the element changes. Here is one rendering where 1fr ends up being 10px:

Here is another rendering where 1fr ends up being 30px:

The fragments are shown in red for illustrative purposes; they should be black in compliant UAs.

Tests

6. Drop Shadows

6.1. Coloring shadows: the box-shadow-color property

Name: box-shadow-color
Value: <color>#
Initial: currentcolor
Applies to: all elements
Inherited: no
Percentages: N/A
Computed value: list, each item a computed color
Canonical order: per grammar
Animation type: by computed value

The box-shadow-color property defines one or more drop shadow colors. The property accepts a comma-separated list of shadow colors.

See the section “Layering, Layout, and Other Details” for how box-shadow-color interacts with other comma-separated drop shadow properties to form each drop shadow layer.

Tests

6.2. Offsetting shadows: the box-shadow-offset property

Name: box-shadow-offset
Value: [ none | <length>{2} ]#
Initial: none
Applies to: all elements
Inherited: no
Percentages: N/A
Computed value: list, each item either none or a pair of offsets (horizontal and vertical) from the element‘s box
Canonical order: per grammar
Animation type: by computed value, treating none as 0 0 when interpolated with non-none values.

The box-shadow-offset property defines one or more drop shadow offsets. The property accepts a comma-separated list. Each item in that list can either be the none value, which indicates no shadow, or a pair of horizontal and vertical offsets, where both values are described as <length> values.

none
The shadow will not be rendered. The values of other box shadow properties corresponding to this shadow have no effect.
1st <length>
Specifies the horizontal offset of the shadow. A positive value draws a shadow that is offset to the right of the box, a negative length to the left.
2nd <length>
Specifies the vertical offset of the shadow. A positive value offsets the shadow down, a negative one up.

See the section “Layering, Layout, and Other Details” for how box-shadow-offset interacts with other comma-separated drop shadow properties to form each drop shadow layer.

Tests

6.3. Blurring shadows: the box-shadow-blur property

Name: box-shadow-blur
Value: <length [0,∞]>#
Initial: 0
Applies to: all elements
Inherited: no
Percentages: N/A
Computed value: list, each item a <length>
Canonical order: per grammar
Animation type: by computed value

The box-shadow-blur property defines one or more blur radii for drop shadows. The property accepts a comma-separated list of <length> values.

Negative values are invalid. If the blur value is zero, the shadow’s edge is sharp. Otherwise, the larger the value, the more the shadow’s edge is blurred. See Shadow Blurring, below.

See the section “Layering, Layout, and Other Details” for how box-shadow-blur interacts with other comma-separated drop shadow properties to form each drop shadow layer.

Tests

6.4. Spreading shadows: the box-shadow-spread property

Name: box-shadow-spread
Value: <length>#
Initial: 0
Applies to: all elements
Inherited: no
Percentages: N/A
Computed value: list, each item a <length>
Canonical order: per grammar
Animation type: by computed value

The box-shadow-spread property defines one or more spread distances for drop shadows. The property accepts a comma-separated list of <length> values.

Positive values cause the shadow to expand in all directions by the specified radius. Negative values cause the shadow to contract. See Shadow Shape, below.

Note that for inner shadows, expanding the shadow (creating more shadow area) means contracting the shadow’s perimeter shape.

See the section “Layering, Layout, and Other Details” for how box-shadow-spread interacts with other comma-separated drop shadow properties to form each drop shadow layer.

Tests

6.5. Spreading shadows: the box-shadow-position property

Name: box-shadow-position
Value: [ outset | inset ]#
Initial: outset
Applies to: all elements
Inherited: no
Percentages: N/A
Computed value: list, each item one of the keywords
Canonical order: per grammar
Animation type: by computed value

The box-shadow-position property defines one or more drop shadow positions. The property accepts a comma-separated list of outset and inset keywords.

outset
Causes the drop shadow to be an outer box-shadow. That means, one that shadows the box onto the canvas, as if it were lifted above the canvas.
inset
Causes the drop shadow to be an inner box-shadow. That means, one that shadows the canvas onto the box, as if the box were cut out of the canvas and shifted behind it.

See the section “Layering, Layout, and Other Details” for how box-shadow-position interacts with other comma-separated drop shadow properties to form each drop shadow layer.

Tests

6.6. Drop Shadows Shorthand: the box-shadow property

Name: box-shadow
Value: <spread-shadow>#
Initial: none
Applies to: all elements
Inherited: no
Percentages: N/A
Computed value: see individual properties
Canonical order: per grammar
Animation type: see individual properties

The box-shadow property attaches one or more drop-shadows to the box. The property accepts a comma-separated list of shadows, ordered front to back.

Each shadow is given as a <spread-shadow>, outlining the box-shadow-offset, and optional values for the box-shadow-blur, box-shadow-spread, box-shadow-color, and box-shadow-position. Omitted lengths are 0; omitted colors default to transparent when the specified offset is none and to currentcolor otherwise.

<spread-shadow> = <'box-shadow-color'>? && [ <'box-shadow-offset'> [ <'box-shadow-blur'> <'box-shadow-spread'>? ]? ] && <'box-shadow-position'>?

6.6.1. Shadow Shape, Spread, and Knockout

An outer box-shadow casts a shadow as if the border-box of the element were opaque. Assuming a spread distance of zero, its perimeter has the exact same size and shape as the border box. The shadow is drawn outside the border edge only: it is clipped inside the border-box of the element.

An inner box-shadow casts a shadow as if everything outside the padding edge were opaque. Assuming a spread distance of zero, its perimeter has the exact same size and shape as the padding box. The shadow is drawn inside the padding edge only: it is clipped outside the padding box of the element.

If a spread distance is defined, the shadow perimeter defined above is expanded outward (for outer box-shadows) or contracted inward (for inner box-shadows) by outsetting (insetting, for inner shadows) the shadow’s straight edges by the spread distance (and flooring the resulting width/height at zero).

Below are some examples of an orange box with a blue border being given a drop shadow.
border:5px solid blue;
background-color:orange;
width: 144px;
height: 144px;
border-radius: 20px;
border-radius: 0;
box-shadow:
  rgba(0,0,0,0.4)
  10px 10px;
A round-cornered box with a light gray shadow the same shape
					          as the border box offset 10px to the right and 10px down
					          from directly underneath the box. A square-cornered box with a light gray shadow the same shape
					          as the border box offset 10px to the right and 10px down
					          from directly underneath the box.
box-shadow:
  rgba(0,0,0,0.4)
  10px 10px
  inset
A round-cornered box with a light gray shadow the inverse shape
					          of the padding box filling 10px in from the top and left edges
					          (just inside the border). A square-cornered box with a light gray shadow the inverse shape
					          of the padding box filling 10px in from the top and left edges
					          (just inside the border).
box-shadow:
  rgba(0,0,0,0.4)
  10px 10px 0
  10px /* spread */
A round-cornered box with a light gray shadow the same shape
					          as the box but 20px taller and wider and offset so that the
					          top and left edges of the shadow are directly underneath the
					          top and left edges of the box. A square-cornered box with a light gray shadow the same shape
					          as the box but 20px taller and wider and offset so that the
					          top and left edges of the shadow are directly underneath the
					          top and left edges of the box.
box-shadow:
  rgba(0,0,0,0.4)
  10px 10px 0
  10px /* spread */
  inset
A round-cornered box with a light gray shadow the inverse shape
					          of the box but 20px narrower and shorter filling 20px in from
					          the top and left edges (just inside the border). A round-cornered box with a light gray shadow the inverse shape
					          of the box but 20px narrower and shorter filling 20px in from
					          the top and left edges (just inside the border).

To preserve the box’s shape when spread is applied, the corner radii of the shadow are also increased (decreased, for inner shadows) from the border-box (padding-box) radii by adding (subtracting) the spread distance (and flooring at zero). However, in order to create a sharper corner when the border radius is small (and thus ensure continuity between round and sharp corners), when the border radius is less than the spread distance (or in the case of an inner shadow, less than the absolute value of a negative spread distance), the spread distance is first multiplied by the proportion 1 + (r-1)3, where r is the ratio of the border radius to the spread distance, in calculating the corner radii of the spread shadow shape. For example, if the border radius is 10px and the spread distance is 20px (r = .5), the corner radius of the shadow shape will be 10px + 20px × (1 + (.5 - 1)3) = 27.5px rather than 30px. This adjustment is applied independently to the radii in each dimension.

The border-image does not affect the shape of the box-shadow.

6.6.2. Blurring Shadow Edges

A non-zero blur radius indicates that the resulting shadow should be blurred, such as by a Gaussian filter. The exact algorithm is not defined; however the resulting shadow must approximate (with each pixel being within 5% of its expected value) the image that would be generated by applying to the shadow a Gaussian blur with a standard deviation equal to half the blur radius.

Note: This means for a long, straight shadow edge, the blur radius will create a visibly apparent color transition approximately the twice length of the blur radius that is perpendicular to and centered on the shadow’s edge, and that ranges from almost the full shadow color at the endpoint inside the shadow to almost fully transparent at the endpoint outside it.

6.7. Layering, Layout, and Other Details

Drop shadows are declared in the coordinated value list constructed from the box-shadow-* properties, which form a coordinating list property group with box-shadow-offset as the coordinating list base property. See CSS Values 4 § A Coordinating List-Valued Properties.

The shadow effects are applied front-to-back: the first shadow is on top and the others are layered behind. Shadows do not influence layout and may overlap (or be overlapped by) other boxes and text or their shadows. In terms of stacking contexts and the painting order, the outer box-shadows of an element are drawn immediately below the background of that element, and the inner shadows of an element are drawn immediately above the background of that element (below the borders and border image, if any).

Unless otherwise specified, drop shadows are only applied to the principal box. If the affected box has multiple fragments, the shadows are applied as specified in box-decoration-break.

Shadows do not trigger scrolling or increase the size of the scrollable area.

Outer shadows have no effect on internal table elements in the collapsing border model. If a shadow is defined for single border edge in the collapsing border model that has multiple border thicknesses (e.g. an outer shadow on a table where one row has thicker borders than the others, or an inner shadow on a rowspanning table cell that adjoins cells with different border thicknesses), the exact position and rendering of its shadows are undefined.

7. Border Shaping

While corner-shape and border-radius allow some expressiveness to styling a border, they still work with the assumption that the border is rectangular.

The border-shape function augments these capabilities, by enabling the author to use any basic shape to specify the path of the border.

7.1. The border-shape property

Name: border-shape
Value: none | [ <basic-shape> <geometry-box>?]{1,2}
Initial: none
Applies to: all elements
Inherited: no
Percentages: see prose
Computed value: list, each item a computed color
Canonical order: per grammar
Animation type: by computed value

The border-shape property is provided with either a single <basic-shape> or two <basic-shape>s, resulting in one or two paths, respectively.

When two <basic-shape> values are given, the border is rendered as the shape between the two paths. When only a single <basic-shape> is given, the border is rendered as a stroke with the relevant side’s computed border width as the stroke width.

The fill color of the border is the relevant side’s computedborder-color.

When a <geometry-box> is not given, the default computation of percentage differs based on the number of given <basic-shape> values.

When two <basic-shape> values are given, the first (outer) one defaults to the border box and the second (inner) one defaults to the padding box. This allows using the different border-width properties to affect the final shape.

When a single <basic-shape> value is given, the <geometry-box> defaults to the half-border-box value, which allows stroking in a way that matches the default border behavior.

The border-shape property is not compatible with border-radius and corner-shape. When an element’s computed value of border-shape is not none, its border-radius is ignored, as if it was set to 0. corner-shape is implicitly ignored, as it can only work in tandem with border-radius.

A box-shadow follows both the outer border path.

border-shape does not affect geometry or layout, which is still computed using the existing border-width properties.

border-shape does not affect the flow of content inside the box. Note: An author can use border-shape in tandem with shape-inside to create effects that decorate the box and control its text flow at the same time.

The inner border-shape clips the overflow content of the element, in the same manner as border-radius, as described in corner clipping.

how should this affect clipping replaced elements?

An element’s relevant side for border shape is the first side (in the order block-start, inline-start, block-end, and inline-end) that has a non-none border style, or block-start if they’re all none.
  1. If element’s computed border-block-start-style is not none, then return block-start.

  2. If element’s computed border-inline-start-style is not none, then return inline-start.

  3. If element’s computed border-block-end-style is not none, then return block-end.

  4. If element’s computed border-inline-end-style is not none, then return inline-end.

  5. Return block-start.

Tests

Privacy Considerations

No new privacy considerations have been reported on this specification.

Security Considerations

No new security considerations have been reported on this specification.

Changes

Changes since the First Public Working Draft of 22 July 2025

Additions since [CSS3BG]

Acknowledgments

In addition to the many contributors to the [CSS1], [CSS2], and [CSS3BG] predecessors to this module, the editors would like to thank Tab Atkins, Noam Rosenthal, Håkon Wium Lie, and Oriol Brufau for their suggestions and feedback specifically for this Level 4.

Conformance

Document conventions

Conformance requirements are expressed with a combination of descriptive assertions and RFC 2119 terminology. The key words “MUST”, “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”, “SHOULD NOT”, “RECOMMENDED”, “MAY”, and “OPTIONAL” in the normative parts of this document are to be interpreted as described in RFC 2119. However, for readability, these words do not appear in all uppercase letters in this specification.

All of the text of this specification is normative except sections explicitly marked as non-normative, examples, and notes. [RFC2119]

Examples in this specification are introduced with the words “for example” or are set apart from the normative text with class="example", like this:

This is an example of an informative example.

Informative notes begin with the word “Note” and are set apart from the normative text with class="note", like this:

Note, this is an informative note.

Advisements are normative sections styled to evoke special attention and are set apart from other normative text with <strong class="advisement">, like this: UAs MUST provide an accessible alternative.

Tests

Tests relating to the content of this specification may be documented in “Tests” blocks like this one. Any such block is non-normative.


Conformance classes

Conformance to this specification is defined for three conformance classes:

style sheet
A CSS style sheet.
renderer
A UA that interprets the semantics of a style sheet and renders documents that use them.
authoring tool
A UA that writes a style sheet.

A style sheet is conformant to this specification if all of its statements that use syntax defined in this module are valid according to the generic CSS grammar and the individual grammars of each feature defined in this module.

A renderer is conformant to this specification if, in addition to interpreting the style sheet as defined by the appropriate specifications, it supports all the features defined by this specification by parsing them correctly and rendering the document accordingly. However, the inability of a UA to correctly render a document due to limitations of the device does not make the UA non-conformant. (For example, a UA is not required to render color on a monochrome monitor.)

An authoring tool is conformant to this specification if it writes style sheets that are syntactically correct according to the generic CSS grammar and the individual grammars of each feature in this module, and meet all other conformance requirements of style sheets as described in this module.

Partial implementations

So that authors can exploit the forward-compatible parsing rules to assign fallback values, CSS renderers must treat as invalid (and ignore as appropriate) any at-rules, properties, property values, keywords, and other syntactic constructs for which they have no usable level of support. In particular, user agents must not selectively ignore unsupported component values and honor supported values in a single multi-value property declaration: if any value is considered invalid (as unsupported values must be), CSS requires that the entire declaration be ignored.

Implementations of Unstable and Proprietary Features

To avoid clashes with future stable CSS features, the CSSWG recommends following best practices for the implementation of unstable features and proprietary extensions to CSS.

Non-experimental implementations

Once a specification reaches the Candidate Recommendation stage, non-experimental implementations are possible, and implementors should release an unprefixed implementation of any CR-level feature they can demonstrate to be correctly implemented according to spec.

To establish and maintain the interoperability of CSS across implementations, the CSS Working Group requests that non-experimental CSS renderers submit an implementation report (and, if necessary, the testcases used for that implementation report) to the W3C before releasing an unprefixed implementation of any CSS features. Testcases submitted to W3C are subject to review and correction by the CSS Working Group.

Further information on submitting testcases and implementation reports can be found from on the CSS Working Group’s website at http://www.w3.org/Style/CSS/Test/. Questions should be directed to the public-css-testsuite@w3.org mailing list.

Index

Terms defined by this specification

Terms defined by reference

References

Normative References

[CSS-BOX-4]
Elika Etemad. CSS Box Model Module Level 4. URL: https://drafts.csswg.org/css-box-4/
[CSS-BREAK-4]
Rossen Atanassov; Elika Etemad. CSS Fragmentation Module Level 4. URL: https://drafts.csswg.org/css-break-4/
[CSS-CASCADE-5]
Elika Etemad; Miriam Suzanne; Tab Atkins Jr.. CSS Cascading and Inheritance Level 5. URL: https://drafts.csswg.org/css-cascade-5/
[CSS-COLOR-4]
Chris Lilley; Tab Atkins Jr.; Lea Verou. CSS Color Module Level 4. URL: https://drafts.csswg.org/css-color-4/
[CSS-COLOR-5]
Chris Lilley; et al. CSS Color Module Level 5. URL: https://drafts.csswg.org/css-color-5/
[CSS-DISPLAY-4]
Elika Etemad; Tab Atkins Jr.. CSS Display Module Level 4. URL: https://drafts.csswg.org/css-display/
[CSS-GRID-2]
Tab Atkins Jr.; et al. CSS Grid Layout Module Level 2. URL: https://drafts.csswg.org/css-grid-2/
[CSS-IMAGES-3]
Tab Atkins Jr.; Elika Etemad; Lea Verou. CSS Images Module Level 3. URL: https://drafts.csswg.org/css-images-3/
[CSS-IMAGES-4]
Elika Etemad; Tab Atkins Jr.; Lea Verou. CSS Images Module Level 4. URL: https://drafts.csswg.org/css-images-4/
[CSS-MASKING-1]
Dirk Schulze; Brian Birtles; Tab Atkins Jr.. CSS Masking Module Level 1. URL: https://drafts.fxtf.org/css-masking-1/
[CSS-OVERFLOW-3]
Elika Etemad; Florian Rivoal. CSS Overflow Module Level 3. URL: https://drafts.csswg.org/css-overflow-3/
[CSS-OVERFLOW-4]
David Baron; Florian Rivoal; Elika Etemad. CSS Overflow Module Level 4. URL: https://drafts.csswg.org/css-overflow-4/
[CSS-PSEUDO-4]
Elika Etemad; Alan Stearns. CSS Pseudo-Elements Module Level 4. URL: https://drafts.csswg.org/css-pseudo-4/
[CSS-RUBY-1]
Elika Etemad; et al. CSS Ruby Annotation Layout Module Level 1. URL: https://drafts.csswg.org/css-ruby-1/
[CSS-SHAPES-1]
Alan Stearns; Rossen Atanassov; Noam Rosenthal. CSS Shapes Module Level 1. URL: https://drafts.csswg.org/css-shapes/
[CSS-SHAPES-2]
CSS Shapes Module Level 2. Editor's Draft. URL: https://drafts.csswg.org/css-shapes-2/
[CSS-TEXT-4]
Elika Etemad; et al. CSS Text Module Level 4. URL: https://drafts.csswg.org/css-text-4/
[CSS-TRANSFORMS-1]
Simon Fraser; et al. CSS Transforms Module Level 1. URL: https://drafts.csswg.org/css-transforms/
[CSS-UI-4]
Florian Rivoal. CSS Basic User Interface Module Level 4. URL: https://drafts.csswg.org/css-ui-4/
[CSS-VALUES-3]
Tab Atkins Jr.; Elika Etemad. CSS Values and Units Module Level 3. URL: https://drafts.csswg.org/css-values-3/
[CSS-VALUES-4]
Tab Atkins Jr.; Elika Etemad. CSS Values and Units Module Level 4. URL: https://drafts.csswg.org/css-values-4/
[CSS-WRITING-MODES-3]
Elika Etemad; Koji Ishii. CSS Writing Modes Level 3. URL: https://drafts.csswg.org/css-writing-modes-3/
[CSS-WRITING-MODES-4]
Elika Etemad; Koji Ishii. CSS Writing Modes Level 4. URL: https://drafts.csswg.org/css-writing-modes-4/
[CSS2]
Bert Bos; et al. Cascading Style Sheets Level 2 Revision 1 (CSS 2.1) Specification. URL: https://drafts.csswg.org/css2/
[CSS3BG]
Elika Etemad; Brad Kemper. CSS Backgrounds and Borders Module Level 3. URL: https://drafts.csswg.org/css-backgrounds/
[DOM]
Anne van Kesteren. DOM Standard. Living Standard. URL: https://dom.spec.whatwg.org/
[GEOMETRY-1]
Simon Pieters; Chris Harrelson. Geometry Interfaces Module Level 1. URL: https://drafts.fxtf.org/geometry/
[RFC2119]
S. Bradner. Key words for use in RFCs to Indicate Requirement Levels. March 1997. Best Current Practice. URL: https://datatracker.ietf.org/doc/html/rfc2119
[SELECTORS-4]
Elika Etemad; Tab Atkins Jr.. Selectors Level 4. URL: https://drafts.csswg.org/selectors/
[SVG2]
Amelia Bellamy-Royds; et al. Scalable Vector Graphics (SVG) 2. URL: https://svgwg.org/svg2-draft/

Informative References

[CSS-LOGICAL-1]
Rossen Atanassov; Elika Etemad. CSS Logical Properties and Values Level 1. URL: https://drafts.csswg.org/css-logical-1/
[CSS-SIZING-3]
Tab Atkins Jr.; Elika Etemad. CSS Box Sizing Module Level 3. URL: https://drafts.csswg.org/css-sizing-3/
[CSS1]
Håkon Wium Lie; Bert Bos. Cascading Style Sheets, level 1. 13 September 2018. REC. URL: https://www.w3.org/TR/CSS1/
[CSS3GRID]
Tab Atkins Jr.; et al. CSS Grid Layout Module Level 1. URL: https://drafts.csswg.org/css-grid-1/

Property Index

Name Value Initial Applies to Inh. %ages Anim­ation type Canonical order Com­puted value Logical property group
border <line-width> || <line-style> || <color> see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-block <'border-block-start'> see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-block-color <'border-top-color'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-block-end <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-block-end-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-block-end-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-block-end-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-block-end-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-block-start <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-block-start-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-block-start-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-block-start-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-block-start-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-block-style <'border-top-style'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-block-width <'border-top-width'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-bottom <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-bottom-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-bottom-left-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-bottom-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-bottom-right-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-bottom-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-bottom-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-clip normal | [ <length-percentage [0,∞]> | <flex> ]+ normal all elements no refer to length of border-edge side by computed value per grammar normal, or a list consisting of absolute lengths, or percentages as specified
border-clip-bottom normal | [ <length-percentage [0,∞]> | <flex> ]+ normal all elements no refer to length of border-edge side by computed value per grammar normal, or a list consisting of absolute lengths, or percentages as specified
border-clip-left normal | [ <length-percentage [0,∞]> | <flex> ]+ normal all elements no refer to length of border-edge side by computed value per grammar normal, or a list consisting of absolute lengths, or percentages as specified
border-clip-right normal | [ <length-percentage [0,∞]> | <flex> ]+ normal all elements no refer to length of border-edge side by computed value per grammar normal, or a list consisting of absolute lengths, or percentages as specified
border-clip-top normal | [ <length-percentage [0,∞]> | <flex> ]+ normal all elements no refer to length of border-edge side by computed value per grammar normal, or a list consisting of absolute lengths, or percentages as specified
border-color [ <color> | <image-1D> ]{1,4} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-end-end-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-end-start-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-image <'border-image-source'> || <'border-image-slice'> [ / <'border-image-width'> | / <'border-image-width'>? / <'border-image-outset'> ]? || <'border-image-repeat'> See individual properties See individual properties no N/A See individual properties per grammar See individual properties
border-image-outset [ <length [0,∞]> | <number [0,∞]> ]{1,4} 0 All elements, except internal table elements when border-collapse is collapse no N/A by computed value per grammar four values, each a number or absolute length
border-image-repeat [ stretch | repeat | round | space ]{1,2} stretch All elements, except internal table elements when border-collapse is collapse no N/A discrete per grammar two keywords, one per axis
border-image-slice [<number [0,∞]> | <percentage [0,∞]>]{1,4} && fill? 100% All elements, except internal table elements when border-collapse is collapse no refer to size of the border image by computed value per grammar four values, each either a number or percentage; plus a fill keyword if specified
border-image-source none | <image> none All elements, except internal table elements when border-collapse is collapse no N/A discrete per grammar the keyword none or the computed <image>
border-image-width [ <length-percentage [0,∞]> | <number [0,∞]> | auto ]{1,4} 1 All elements, except internal table elements when border-collapse is collapse no Relative to width/height of the border image area by computed value per grammar four values, each either a number, the keyword auto, or a computed <length-percentage> value
border-inline <'border-block-start'> see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-inline-color <'border-top-color'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-inline-end <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-inline-end-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-inline-end-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-inline-end-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-inline-end-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-inline-start <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-inline-start-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-inline-start-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-inline-start-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-inline-start-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-inline-style <'border-top-style'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-inline-width <'border-top-width'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-left <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-left-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-left-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-left-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-left-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-limit all | [ sides | corners ] <length-percentage [0,∞]>? | [ top | right | bottom | left ] <length-percentage [0,∞]> all all elements, except table element when border-collapse is collapse no relative to border-box discrete per grammar as specified
border-radius <length-percentage [0,∞]>{1,4} [ / <length-percentage [0,∞]>{1,4} ]? see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-right <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-right-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-right-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-right-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-right-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-shape none | [ <basic-shape> <geometry-box>?]{1,2} none all elements no see prose by computed value per grammar list, each item a computed color
border-start-end-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-start-start-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-style <'border-top-style'>{1,4} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
border-top <line-width> || <line-style> || <color> See individual properties all elements except ruby base containers and ruby annotation containers no N/A see individual properties per grammar see individual properties
border-top-color <color> | <image-1D> currentcolor all elements except ruby base containers and ruby annotation containers no N/A see prose per grammar the computed color and/or a one-dimensional image function border-color
border-top-left-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-top-radius <length-percentage [0,∞]>{1,2} [ / <length-percentage [0,∞]>{1,2} ]? 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
border-top-right-radius <length-percentage [0,∞]>{1,2} 0 all elements (but see prose) no Refer to corresponding dimension of the border box. by computed value per grammar pair of computed <length-percentage> values border-radius
border-top-style <line-style> none all elements except ruby base containers and ruby annotation containers no N/A discrete per grammar specified keyword border-style
border-top-width <line-width> medium all elements except ruby base containers and ruby annotation containers no N/A by computed value per grammar absolute length, snapped as a border width; zero if the border style is none or hidden border-width
border-width <'border-top-width'>{1,4} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
box-shadow <spread-shadow># none all elements no N/A see individual properties per grammar see individual properties
box-shadow-blur <length [0,∞]># 0 all elements no N/A by computed value per grammar list, each item a <length>
box-shadow-color <color># currentcolor all elements no N/A by computed value per grammar list, each item a computed color
box-shadow-offset [ none | <length>{2} ]# none all elements no N/A by computed value, treating none as 0 0 when interpolated with non-none values. per grammar list, each item either none or a pair of offsets (horizontal and vertical) from the element‘s box
box-shadow-position [ outset | inset ]# outset all elements no N/A by computed value per grammar list, each item one of the keywords
box-shadow-spread <length># 0 all elements no N/A by computed value per grammar list, each item a <length>
corner <'border-radius'> || <'corner-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-block-end <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-block-end-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
corner-block-start <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-block-start-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
corner-bottom <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-bottom-left <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-bottom-left-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-bottom-right <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-bottom-right-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-bottom-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
corner-end-end <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-end-end-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-end-start <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-end-start-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-inline-end <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-inline-end-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
corner-inline-start <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-inline-start-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
corner-left <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-left-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
corner-right <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-right-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties
corner-shape <'corner-top-left-shape'>{1,4} round all elements where border-radius can apply no see individual properties see individual properties per grammar see individual properties
corner-start-end <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-start-end-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-start-start <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-start-start-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-top <'border-top-radius'> || <'corner-top-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-top-left <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-top-left-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-top-right <'border-top-left-radius'> || <'corner-top-left-shape'> 0 all elements (but see prose) no Refer to corresponding dimension of the border box. see individual properties per grammar see individual properties
corner-top-right-shape <corner-shape-value> round all elements where border-radius can apply no n/a see superellipse interpolation per grammar the corresponding superellipse() value corner-shape
corner-top-shape <'corner-top-left-shape'>{1,2} see individual properties see individual properties see individual properties see individual properties see individual properties per grammar see individual properties

Issues Index

Here are two proposals for doing this: the second one is from GCPM, the first one is an attempt to recast it more readably. The names are terrible, known problem, proposals accepted. There is a problem with conceiving this as clipping: if you have dotted borders, you want whole dots always, not parts of dots. So it should be a drawing limit, not a clip.
how should this affect clipping replaced elements?