shape()
        
        
          Limited availability
        
        
        
          
                
              
                
              
                
              
        
        
      
      This feature is not Baseline because it does not work in some of the most widely-used browsers.
The shape() CSS function is used to define a shape for the clip-path and offset-path properties. It combines an initial starting point with a series of shape commands that define the path of the shape. The shape() function is a member of the <basic-shape> data type.
Syntax
/* <fill-rule> */
clip-path: shape(nonzero from 0 0, line to 10px 10px);
/* <move-command>, <line-command>, and close */
offset-path: shape(from 10px 10px, move by 10px 5px, line by 20px 40%, close);
/* <hvline-command> */
offset-path: shape(from 10px 10px, hline by 50px, vline to 5rem);
/* <curve-command> */
offset-path: shape(
  from 10px 10px,
  curve to 80px 80px with 160px 1px / 20% 16px
);
/* <smooth-command> */
offset-path: shape(from 10px 10px, smooth to 100px 50pt);
/* <arc-command> */
offset-path: shape(
  from 5% 0.5rem,
  arc to 80px 1pt of 10% ccw large rotate 25deg
);
/* Using a CSS math function */
offset-path: shape(
  from 5px -5%,
  hline to 50px,
  vline by calc(0% + 160px),
  hline by 70.5px,
  close,
  vline by 60px
);
clip-path: shape(
  evenodd from 10px 10px,
  curve to 60px 20% with 40px 0,
  smooth to 90px 0,
  curve by -20px 60% with 10% 40px / 20% 20px,
  smooth by -40% -10px with -10px 70px,
  close
);
Parameters
- <fill-rule>Optional
- 
Specifies how overlapping regions of a shape should be filled. The possible values include: - 
nonzero: A point is considered inside the shape if a ray drawn from the point crosses more left-to-right than right-to-left path segments, resulting in a non-zero count. This is the default value when<fill-rule>is omitted.
- 
evenodd: A point is considered to be inside the shape if a ray drawn from the point crosses an odd number of path segments. This means that for each time the ray enters the shape, it has not exited an equal number of times, indicating an odd count of entries without corresponding exits.
 Warning: <fill-rule>is not supported inoffset-pathand using it invalidates the property.
- 
- from <coordinate-pair>
- 
Defines the starting point of the first <shape-command>as a pair of coordinates that are measured from the top-left corner of the reference box. The coordinates are specified as space-separated<x> <y><length-percentage>values representing the left offset and top offset, respectively. Percentage values are relative to the width and height of the element's reference box, respectively. Add a comma after this parameter.
- <shape-command>
- 
Specifies a list of one or more comma-separated commands that define the shape, using syntax similar to SVG path commands. Commands include <move-command>,<line-command>,<hv-line-command>,<curve-command>,<smooth-command>,<arc-command>, andclose. Each command's starting point is the previous command's ending point, with the first point of the shape defined by thefrom <coordinate-pair>parameter.The syntax of most shape commands is a keyword providing a directive, such as moveorline, followed by thebyortokeyword, and a set of coordinates.by: Indicates that the<coordinate-pair>is relative to the command's starting point (a "relative" value).to: Indicates that the<coordinate-pair>is relative to the top-left corner of the reference box (an "absolute" value).Note: If a coordinate in a <coordinate-pair>is specified as a percentage, the value is calculated relative to the respective width or height of the reference box.The following <shape-command>s can be specified:<move-command>,<line-command>,<hv-line-command>,<curve-command>,<smooth-command>,<arc-command>, andclose.<move-command>: Specified asmove [by | to] <coordinate-pair>. This command adds a MoveTo command to the list of shape commands. It draws nothing. Rather, it specifies the starting position for the next command. Thebyortokeyword specifies whether the<coordinate-pair>point is relative or absolute, respectively. If the<move-command>follows theclosecommand, it identifies the starting point of the next shape or subpath.<line-command>: Specified asline [by | to] <coordinate-pair>. This command adds a LineTo command to the list of shape commands. It draws a straight line from the command's starting point to its ending point. Thebyortokeyword specifies whether the ending point specified by<coordinate-pair>is relative or absolute, respectively.<hv-line-command>: Specified as[hline | vline] [by | to] <length-percentage>. This command adds a horizontal (hline) or vertical (vline) LineTo command to the list of shape commands. Withhline, a horizontal line is drawn from the command's starting pointtoorbythexposition defined by<length-percentage>. Withvline, a vertical line is drawn from the command's starting pointtoorbytheyposition defined by<length-percentage>. Thebyortokeyword determines the relative or absolute ending point, respectively. This command is equivalent to<line-command>with one coordinate value set by the single<length-percentage>and the other coordinate value remaining unchanged from its starting command.<curve-command>: Specified ascurve [by | to] <coordinate-pair> with <coordinate-pair> [/ <coordinate-pair>]. This command adds a Bézier curve command to the list of shape commands. Thebyortokeyword determines whether the ending point of the curve, specified by the first<coordinate-pair>, is relative or absolute, respectively. Thewithkeyword specifies the control points of the Bézier curve.- If only a single <coordinate-pair>is provided, the command draws a quadratic Bézier curve, which is defined by three points (the start point, control point, and end point).
- If two <coordinate-pair>values are provided, the command draws a cubic Bézier curve, which is defined by four points (the start point, two control points, and the end point).
 <smooth-command>: Specified assmooth [by | to] <coordinate-pair> [with <coordinate-pair>]. This command adds a smooth Bézier curve command to the list of shape commands. Thebyortokeyword determines whether the ending point of the curve, specified by the first<coordinate-pair>, is relative or absolute, respectively.- If with <coordinate-pair>is omitted, the command draws a smooth quadratic Bézier curve, which uses the previous control point and the current endpoint to define the curve.
- If the optional withkeyword is included, it specifies the control points of the curve through<coordinate-pair>, drawing a smooth cubic Bézier curve defined by the previous control point, the current control point, and the current endpoint.
 Smooth curves ensure a continuous transition from the shape, while quadratic curves do not. Smooth quadratic curves maintain a seamless transition using a single control point, whereas smooth cubic curves provide a more refined transition using two control points. <arc-command>: Specified asarc [by | to] <coordinate-pair> of <length-percentage> [<length-percentage>] [<arc-sweep> | <arc-size> | rotate <angle>]. This command adds an elliptical arc curve command to the list of shape commands. It draws an elliptical arc between a starting point and an ending point. Thebyortokeyword determines whether the ending point of the curve, specified by the first<coordinate-pair>, is relative or absolute, respectively.The elliptical arc curve command defines two possible ellipses, which intersect both the starting and ending points, and each can be traced clockwise or counterclockwise, resulting in four possible arcs depending on the arc size, direction, and angle. The ofkeyword specifies the size of the ellipse from which the arc is taken: the first<length-percentage>provides the horizontal radius of the ellipse, and the second<length-percentage>provides the vertical radius.Specify the following parameters to choose which of the four arcs to use: - <arc-sweep>: Indicates whether the desired arc is the one traced around the ellipse clockwise (- cw) or counterclockwise (- ccw). If omitted, this defaults to- ccw.
- <arc-size>: Indicates whether the desired arc is the larger (- large) or smaller (- small) of the two arcs. If omitted, this defaults to- small.
- <angle>: Specifies the angle, in degrees, by which to rotate the ellipse relative to the x-axis. A positive angle rotates the ellipse clockwise, and a negative angle rotates it counterclockwise. If omitted, this defaults to- 0deg.
 Special situations are handled as follows: - If only one <length-percentage>is provided, the same value is used for both the horizontal and vertical radii, effectively creating a circle. In this case,<arc-size>and<angle>have no affect.
- If either radius is zero, the command is equivalent to a <line-command>to the ending point.
- If either radius is negative, its absolute value is used instead.
- If the horizontal and vertical radii don't describe an ellipse large enough to intersect both the starting point and ending points (after rotation by the specified <angle>), the radii are scaled up uniformly until the ellipse is just large enough to intersect both points.
- If the starting and ending points of the arc lie on exactly opposite sides of the ellipse, there is only one possible ellipse and two possible arcs. In this case, <arc-sweep>specifies the arc to choose, and<arc-size>has no effect.
 close: Adds a ClosePath command to the list of shape commands, drawing a straight line from the current position (end of the last command) to the first point in the path defined in thefrom <coordinate-pair>parameter. To close the shape without drawing a line, include a<move-command>with the originating coordinates before the close command. If theclosecommand is immediately followed by a<move-command>, it defines the starting point of the next shape or subpath.
- If only a single 
Description
The shape() function allows you to define complex shapes. It is similar to the path() shape function in several ways:
- The <fill-rule>parameter in theshape()function works exactly like the same parameter in thepath()function.
- The shape()function requires specifying one or more<shape-command>s, where each command uses an underlying path command, such as MoveTo, LineTo, and ClosePath.
However, shape() offers several advantages over using path():
- shape()uses standard CSS syntax, making it easier to create and modify shapes directly in your stylesheet. In comparison,- path()uses the SVG path syntax, which is less intuitive for those not familiar with SVG.
- shape()supports a variety of CSS units, including percentages,- rem, and- em.- path(), on the other hand, defines shapes as a single string and limits units to- px.
- shape()also allows the use of CSS math functions, like- calc(),- max(), and- abs(), providing more versatility when defining shapes.
Formal syntax
<shape()> =
shape( <'fill-rule'>? from <position> , <shape-command># )
<fill-rule> =
nonzero |
evenodd
<position> =
<position-one> |
<position-two> |
<position-four>
<shape-command> =
<move-command> |
<line-command> |
close |
<horizontal-line-command> |
<vertical-line-command> |
<curve-command> |
<smooth-command> |
<arc-command>
<position-one> =
left |
center |
right |
top |
bottom |
x-start |
x-end |
y-start |
y-end |
block-start |
block-end |
inline-start |
inline-end |
<length-percentage>
<position-two> =
[ left | center | right | x-start | x-end ] && [ top | center | bottom | y-start | y-end ] |
[ left | center | right | x-start | x-end | <length-percentage> ] [ top | center | bottom | y-start | y-end | <length-percentage> ] |
[ block-start | center | block-end ] && [ inline-start | center | inline-end ] |
[ start | center | end ]{2}
<position-four> =
[ [ left | right | x-start | x-end ] <length-percentage> ] && [ [ top | bottom | y-start | y-end ] <length-percentage> ] |
[ [ block-start | block-end ] <length-percentage> ] && [ [ inline-start | inline-end ] <length-percentage> ] |
[ [ start | end ] <length-percentage> ]{2}
<length-percentage> =
<length> |
<percentage>
Examples
>Using shape() to define a path
    This example demonstrates how the shape() function can be used in the offset-path property to define the shape of the path an element can follow.
The first shape, shape1, follows a cubic Bézier curved path defined by the curve to command. Next, the close command draws a straight line from the curve's end point back to the initial point defined in the from command. Finally, shape1 moves to its new position at 0px 150px and then proceeds along a horizontal line.
The second shape, shape2, initially follows a horizontal line, then moves back to its starting position at 50px 90px. Next, it follows a vertical line before closing the path back to the initial point.
Both shapes start with their original colors and gradually transition to hotpink by the end of the move animation, reverting to their initial color as the animation restarts. This cyclic color change provides you a visual cue about the animation's progression and restart.
.shape {
  width: 50px;
  height: 50px;
  background: #2bc4a2;
  position: absolute;
  text-align: center;
  line-height: 50px;
  animation: move 6s infinite linear;
}
.shape1 {
  offset-path: shape(
    from 30% 60px,
    curve to 180px 180px with 90px 190px,
    close,
    move by 0px 150px,
    hline by 40%
  );
}
.shape2 {
  offset-path: shape(
    from 50px 90px,
    hline to 8em,
    move to 50px 90px,
    vline by 20%,
    close
  );
}
@keyframes move {
  0% {
    offset-distance: 0%;
  }
  100% {
    offset-distance: 100%;
    background-color: hotpink;
  }
}
Result
Using shape() to define the visible part of an element
    This example demonstrates how the shape() function can be used in the clip-path property to create different shapes for the clipping region. The first shape (shape1) uses a triangle defined by straight lines. The second shape (shape2) includes curves and smooth transitions; it also illustrates the use of the <move-command> after the close command, which adds a rectangular shape to the clipping region.
.shape {
  width: 100%;
  height: 100%;
  background: #2bc4a2;
  position: absolute;
  text-align: center;
  line-height: 50px;
}
/* Triangular clipping region */
.shape1 {
  clip-path: shape(from 0% 0%, line to 100% 0%, line to 50% 100%, close);
}
/* A Heart clipping region using curve and arc transitions
   and a box using hline and vline transitions */
.shape2 {
  clip-path: shape(
    from 20px 70px,
    arc to 100px 70px of 1% cw,
    arc to 180px 70px of 1% cw,
    curve to 100px 190px with 180px 130px,
    curve to 20px 70px with 20px 130px,
    close,
    move to 150px 150px,
    hline by 40px,
    vline by 40px,
    hline by -40px,
    close
  );
}
Result
Specifications
| Specification | 
|---|
| CSS Shapes Module Level 2> # shape-function> | 
Browser compatibility
Loading…
See also
- clip-path
- offset-path
- CSS shapes module
- Overview of shapes guide
- Basic shapes guide