Text on the web automatically wraps at the edge of the screen so that it doesn't overflow. Images, on the other hand, have an intrinsic size. If an image is wider than the screen, the image overflows and the user has to scroll horizontally to see all of it.
Fortunately, CSS gives you tools to stop this from happening.
Constrain your images
In your style sheet, you can use max-inline-size
to declare that images can never be rendered at a size wider than their containing element.
img {
max-inline-size: 100%;
block-size: auto;
}
You can apply the same rule to other kinds of embedded content too, like videos and iframes.
img,
video,
iframe {
max-inline-size: 100%;
block-size: auto;
}
With this rule in place, browsers automatically scale down images to fit on the screen.

Adding a block-size
value of auto
means the browser preserves your images' aspect ratio as it resizes them.
Sometimes, an image's dimensions are set by a content management system (CMS) or other content delivery system. If your design calls for a different aspect ratio from the CMS's default, you can use the aspect-ratio
property to preserve your site's design:
img {
max-inline-size: 100%;
block-size: auto;
aspect-ratio: 2/1;
}
Unfortunately, this often means the browser has to squash or stretch the image to make it fit in the intended space.

To prevent squashing and stretching, use the object-fit
property.
An object-fit
value of contain
tells the browser to preserve the image's aspect ratio, leaving empty space around the image if needed.
img {
max-inline-size: 100%;
block-size: auto;
aspect-ratio: 2/1;
object-fit: contain;
}
An object-fit
value of cover
tells the browser to preserve the image's aspect ratio, cropping the image if needed.
img {
max-inline-size: 100%;
block-size: auto;
aspect-ratio: 2/1;
object-fit: cover;
}


You can change the position of the image crop using the object-position property. This adjusts the focus of the crop, so you can make sure the most important part of the image is still visible.
img {
max-inline-size: 100%;
block-size: auto;
aspect-ratio: 2/1;
object-fit: cover;
object-position: top center;
}

object-position
to crop only one side of your image.Deliver your images
Those CSS rules tell the browser how you'd like images to be rendered. You can also provide hints in your HTML about how the browser should handle those images.
Hints for sizing
If you know your image's dimensions, always include width
and height
attributes. Even if the image is rendered at a different size because of your max-inline-size
rule, the browser still knows the width to height ratio and can set aside the right amount of space. This prevents your other content from jumping around when the image loads.
<img
src="image.png"
alt="A description of the image."
width="300"
height="200"
>
Hints for loading
Use the loading
attribute to tell the browser whether to delay loading the image until it's in or near the viewport. For images below the fold, use a value of lazy
. The browser won't load lazy images until the user has scrolled far down enough that the image is about to come into view. If the user never scrolls, the image never loads.
<img
src="image.png"
alt="A description of the image."
width="300"
height="200"
loading="lazy"
>
For a hero image above the fold, don't use loading
. If your site automatically applies the loading="lazy"
attribute, you can usually set loading
to the default value of eager
to prevent images from being lazy loaded:
<img
src="hero.jpg"
alt="A description of the image."
width="1200"
height="800"
loading="eager"
>
Fetch Priority
For important images-such as the LCP image, you can further prioritize the loading using Fetch Priority by setting the fetchpriority
attribute to high
:
<img
src="hero.jpg"
alt="A description of the image."
width="1200"
height="800"
loading="eager"
fetchpriority="high"
>
This tells the browser to fetch the image right away and at high priority, instead of waiting until the browser has finished its layout and would fetch images normally.
However, when you ask the browser to prioritize downloading one resource, like an image, the browser must de-prioritize another resource such as a script or a font file. Only set fetchpriority="high"
on an image if it's truly vital.
Hints for preloading
It's best to avoid preloading whenever possible by including all images in the initial HTML file. However, some images may be unavailable, such as images added by JavaScript or a CSS background image.
You can use preloading to get the browser to fetch these important images ahead of time. For really important images, you can combine this preloading with the fetchpriority
attribute:
<link rel="preload" href="hero.jpg" as="image" fetchpriority="high">
Again, use these attributes sparingly to avoid overriding the browser's prioritization heuristics too often. Overusing them can cause performance degradation.
Some browsers support preloading responsive images based on srcset, using the imagesrcset
and imagesizes
attributes. For example:
<link rel="preload" imagesrcset="hero_sm.jpg 1x hero_med.jpg 2x hero_lg.jpg 3x" as="image" fetchpriority="high">
By excluding the href
fallback, you can make sure browsers without srcset
support still preload the correct image.
You can't preload images in different formats based on browser support of certain formats. Attempting this can result in extra downloads that waste users' data.
Image decoding
There's also a decoding
attribute you can add to img
elements. You can tell the browser that the image can be decoded asynchronously, so it can prioritize processing other content.
<img
src="image.png"
alt="A description of the image."
width="300"
height="200"
loading="lazy"
decoding="async"
>
You can use the sync
value if the image itself is the most important piece of content to prioritize.
<img
src="hero.jpg"
alt="A description of the image."
width="1200"
height="800"
loading="eager"
decoding="sync"
>
The decoding
attribute doesn't change how fast the image decodes. It affects only whether the browser waits for this image decoding to happen before rendering other content.
In most cases this doesn't have much impact, but sometimes it can let the browser display your image or other content slightly faster. For example, for a large document with lots of elements that take time to render, and with large images that take a long time to decode, setting sync
on important images tells the browser to wait for the image and render both at once. Alternatively, you can set async
to let the browser display content faster and without waiting for the image to decode.
However, the better option is usually to try to avoid excessive DOM sizes and use responsive images to reduce decoding time, instead of using decoding
.
Responsive images with srcset
Thanks to that max-inline-size: 100%
declaration, your images can't break out of their containers. However, if a user has a small screen and a low-bandwidth network, making them download the same size images as users with larger screens wastes data.
To fix this issue, add multiple versions of the same image at different sizes, and use the srcset
attribute to tell the browser these sizes exist and when to use them.
Width descriptor
You can define a srcset
using a comma-separated list of values. Each value is the URL of an image, followed by a space, followed by some metadata about the image, called a descriptor.
In this example, the metadata describes the width of each image using the w
unit. One w
is the width of one pixel.
<img
src="small-image.png"
alt="A description of the image."
width="300"
height="200"
loading="lazy"
decoding="async"
srcset="small-image.png 300w,
medium-image.png 600w,
large-image.png 1200w"
>
The srcset
attribute complements the src
attribute instead of replacing it. You still need to have a valid src
attribute, but the browser can replace its value with one of the options listed in the srcset
. To save bandwidth, the browser only downloads the larger image if they're needed.
Sizes
If you're using the width descriptor, you must also use the sizes
attribute to give the browser more information. This tells the browser what size you expect the image to be displayed at under different conditions. Those conditions are specified in a media query.
The sizes
attribute takes a comma-separated list of media queries and image widths.
<img
src="small-image.png"
alt="A description of the image."
width="300"
height="200"
loading="lazy"
decoding="async"
srcset="small-image.png 300w,
medium-image.png 600w,
large-image.png 1200w"
sizes="(min-width: 66em) 33vw,
(min-width: 44em) 50vw,
100vw"
>
In this example, you're telling the browser that in a viewport with a width over 66em
, it should display the image no wider than one third of the screen (inside a three-column layout, for example).
For viewport widths between 44em
and 66em
, display the image at half the width of the screen (as in a two-column layout).
For anything narrower than 44em
, display the image at the full width of the screen.
This means that the biggest image won't necessarily be used for the widest screen. A wide browser window that can display a multi-column layout uses an image that fits in one column, which might be smaller than an image used for a single-column layout on a narrower screen.
Pixel density descriptor
You can also use descriptors to provide an alternate version of images to show on high-density displays, to keep images looking sharp at the higher resolutions they provide.

Use the density descriptor to describe the pixel density of the image in relation to the image in the src
attribute. The density descriptor is a number followed by the letter x, as in 1x
or 2x
.
<img
src="small-image.png"
alt="A description of the image."
width="300"
height="200"
loading="lazy"
decoding="async"
srcset="small-image.png 1x,
medium-image.png 2x,
large-image.png 3x"
>
If small-image.png
is 300 by 200 pixels in size, and medium-image.png
is 600 by 400 pixels, then medium-image.png
can have 2x
after it in the srcset
list.
You don't have to use whole numbers. If another version of the image is 450 by 300 pixels in size, you can describe it with 1.5x
.
Presentational images
Images in HTML are content. That's why you include the alt
attribute with a description of the image for screen readers and search engines.
If you embed an image that's decorative, without any meaningful content, you can use an empty alt
attribute.
<img
src="flourish.png"
alt=""
width="400"
height="50"
>
You must always include the alt
attribute, even if it's empty. An empty alt
attribute tells a screen reader that the image is presentational. A missing alt
attribute doesn't provide that information.
Ideally, presentational or decorative images are included with CSS instead of HTML. HTML is for structure. CSS is for presentation.
Background images
Use the background-image
property in CSS to load presentational images.
element {
background-image: url(flourish.png);
}
You can specify multiple image candidates using the image-set
function for background-image
.
The image-set
function in CSS works a lot like the srcset
attribute in HTML. Provide a list of images with a pixel density descriptor for each one.
element {
background-image: image-set(
small-image.png 1x,
medium-image.png 2x,
large-image.png 3x
);
}
The browser chooses the most appropriate image for the device's pixel density.
There are many factors to consider when you're adding images to your site, including:
- Reserving the right space for each image.
- Figuring out how many sizes you need.
- Deciding whether the image is content or decorative.
It's worth spending the time to get your images right. Poor image strategies can annoy and frustrate your users. A good image strategy makes your site feel snappy and sharp, regardless of the user's device or network connection.
There's one more HTML element in your toolkit to give you more control over your images: the picture
element.
Check your understanding
Test your knowledge of images.
Styles must be added for images to fit within the viewport.
When an image's height and width have been forced into an unnatural aspect ratio, which styles can help adjust how the image fits into these proportions?
object-fit
contain
and cover
.image-fit
fit-image
aspect-ratio
Putting height
and width
on your images prevents CSS from being able to style it differently.
The srcset
attribute doesn't _______ the src
attribute, it _______ it.
srcset
definitely doesn't replace the src
attribute.Missing alt
on an image is the same as an empty alt
.
alt
attribute tells a screen reader that this image is presentational.alt
signals nothing to a screen reader.