tl;dr
Use the responsive iframe generator to quickly create a responsive iframe at any aspect ratio. Check out the code below to see how you can easily create them with just CSS using aspect ratio boxes.
.iframe-container {
overflow: hidden;
/* 16:9 aspect ratio */
padding-top: 56.25%;
position: relative;
}
.iframe-container iframe {
border: 0;
height: 100%;
left: 0;
position: absolute;
top: 0;
width: 100%;
}
<div class="iframe-container">
<iframe src="//www.youtube.com/embed/KMYrIi_Mt8A" allowfullscreen></iframe>
</div>
Or you can use custom properties:
[style*="--aspect-ratio"] > :first-child {
width: 100%;
}
[style*="--aspect-ratio"] > img {
height: auto;
}
@supports (--custom:property) {
[style*="--aspect-ratio"] {
position: relative;
}
[style*="--aspect-ratio"]::before {
content: "";
display: block;
padding-bottom: calc(100% / (var(--aspect-ratio)));
}
[style*="--aspect-ratio"] > :first-child {
position: absolute;
top: 0;
left: 0;
height: 100%;
}
}
<div style="--aspect-ratio:815/419;">
</div>
<div style="--aspect-ratio:16/9;">
</div>
<!-- even single value -->
<div style="--aspect-ratio:1.4;">
</div>
Guide to Responsive iframes
You’ve spent countless hours designing and building the perfect responsive site. One problem — iframes
. Proportionally resizing these pesky little windows to another world can be frustrating. It’s easy enough to make it span 100% of its container, but there’s no attribute (yet) to make the height resize accordingly.
So how do you keep from blowing your top trying to make an iframe responsive? Hint: it doesn’t require any JavaScript, just a simple CSS technique!
First, let’s define what a “responsive iframes” actually means.
Native responsive iframes are coming! There is the experimental intrinsicsize
attribute that I could imagine being quite nice for iframes in addition to images. Plus the aspect-ratio
in CSS which could default to use the width
and height
attributes on the element.
What is a responsive iframe?
The term “responsive iframe” is a little broad. For instance, styling an iframe to use 100%
, 100vw
or 100vh
is technically making it responsive. But what if you need to adjust its height based on the width so it keeps its aspect ratio? That’s where the problem is, iframes are fluid and can’t natively adapt.
The old way of building responsive iframes usually took the form of some nasty JavaScript hack. A better, modern way uses a simple CSS technique — intrinsic ratios — to create an aspect ratio box.
What is an aspect ratio box?
An aspect ratio is basically a container that adjusts its height based on its width to always keep its aspect ratio (i.e. 16×9, 4×3, 1×1, etc.). They’re most commonly used to embed iframe videos like YouTube or Vimeo videos.
This isn’t particularly new stuff. I think the original credit goes as far back as 2009 and Thierry Koblentz’s Intrinsic Ratios and maintained popularity even for other kinds of content with articles like Uncle Dave’s Ol’ Padded Box.
What is an intrinsic ratio?
An intrinsic ratio means an element will maintain its aspect ratio when resized. Think of an img
with max-width: 100%
. Change the width of its parent and it’ll change the size while keeping the same shape (aka. its aspect ratio).
How to calculate aspect ratios?
Perfect squares and 16:9 stuff is great, but the values used for those are just simple math. An aspect ratio can be anything, and they commonly are completely arbitrary. A video or image can be cropped to any size.
So how do you figure out the padding-top
for say an image that’s 1127.34×591.44
? One way is using the CSS calc()
, like this:
padding-top: calc(591.44 / 1127.34 * 100%);
If you’re using a preprocessor like Sass, we could do the calculation ahead of time:
padding-top: 591.44px / 1127.34px * 100%;
How to create a responsive iframe?
It’s a cinch to make iframes responsive with an aspect ratio box using the intrinsic ratio technique. Or use the responsive iframe generator.
Do not use JavaScript to make iframes responsive. I cringe every-time I see someone using JS when a simple CSS solution exists — even if it’s “light-weight”, it’s not needed. Worse, they often have issues with cross-browser compatibility & bugginess. The intrinsic ratio technique is a much simpler way to implement cross-browser compliant responsive iframes.
The 3 steps to create a responsive iframe that keeps its aspect ratio:
- Create the aspect ratio box.
Add a container for the iframe, determine the aspect ratio percentage, hide the overflow, and set its position to relative.
- Position the iframe.
Set the width and height to 100% and absolutely position it to the top left.
- Optimize & style as needed.
Add some CSS to remove the iframe border, lazyload it, and remove unneeded attributes.
Create the aspect ratio box.
.iframe-container {
overflow: hidden;
/* 16:9 aspect ratio */
padding-top: 56.25%;
position: relative;
}
<div class="iframe-container"></div>
Position the iframe.
.iframe-container iframe {
height: 100%;
left: 0;
position: absolute;
top: 0;
}
<div class="iframe-container">
<iframe src="https://www.youtube.com/embed/mB1dE0FotdY" width="100%" frameborder="0" title="Responsive iframe example" allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowfullscreen />
</div>
Optimize & style as needed.
.iframe-container iframe {
border: 0;
height: 100%;
left: 0;
position: absolute;
top: 0;
width: 100%;
}
<div class="iframe-container">
<iframe src="https://www.youtube.com/embed/mB1dE0FotdY" loading="lazy" title="Responsive iframe example" allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowfullscreen />
</div>
Try resizing your browser window to see the responsive iframe in action.
Don’t forget to lazy-load your iframes. In addition to making your iframes responsive, you’ll want to lazy-load them using the loading
attribute. This improves page load times, enhances the user experience, and increases your search engine rankings. Learn more about how to lazy-load iframes.
List of ratios for responsive iframes.
Here’s a list of other aspect ratio percentages you can use when defining padding-top
.
padding-top: 56.25%; /* 16:9 aspect ratio */
padding-top: 75%; /* 4:3 aspect ratio */
padding-top: 66.66%; /* 3:2 aspect ratio */
padding-top: 62.5%; /* 8:5 aspect ratio */
padding-top: 100%; /* 1:1 aspect ratio */
What’s an aspect ratio? An aspect ratio of an element describes the proportional relationship between its width and its height. Two common video aspect ratios are 4:3 (the universal video format of the 20th century), and 16:9 (universal for HD television and European digital television, and for YouTube videos).
Responsive iframes with Sass
Sass makes it even easier to create responsive iframes. You can create a ratio
function
that’ll calculate the padding percentage needed for a particular aspect ratio, then a mixin
to generate the styles.
The responsive iframe Sass mixin.
Use this Sass mixin to create an aspect ratio box for your iframes.
/// Aspect ratio box.
///
/// @author Ben Marshall
/// @link https://www.benmarshall.me/responsive-iframes
///
/// @param {int} $width - Width in pixels.
/// @param {int} $height - Height in pixels.
///
/// @example scss - Aspect ratio box mixin
/// .iframe-container {
/// @include aspectRatioBox(834, 469);
/// }
///
/// @output CSS aspect ratio box.
/// .iframe-container {
/// overflow: hidden;
/// padding-top: 56.25%;
/// position: relative;
/// top: 0;
/// width: 100%;
/// }
///
/// .iframe-container iframe {
/// border: 0;
/// height: 100%;
/// left: 0;
/// position: absolute;
/// top: 0;
/// width: 100%;
/// }
@mixin aspectRatioBox($width, $height) {
overflow: hidden;
padding-top: percentage($height / $width);
position: relative;
iframe {
border: 0;
height: 100%;
left: 0;
position: absolute;
top: 0;
width: 100%;
}
}
The ratio
Sass function.
Here’s a handy Sass function to calculate aspect ratio percentages:
/// Calculate a ratio.
///
/// @author Ben Marshall
/// @link https://www.benmarshall.me/responsive-iframes
///
/// @param {int} $width - Width in pixels.
/// @param {int} $height - Height in pixels.
/// @return {int} The calculated ratio percent.
///
/// @example scss - Ratio function
/// ratio(834, 469)
/// // 56.25
@function ratio($width, $height) {
return percentage($height / $width);
}
CSS Framework Support
Many CSS frameworks like Bootstrap, Foundation, or Materialize have built-in styles for aspect ratio boxes. All use the same technique. Check out some of the examples below.
Responsive iframes in Bootstrap
Bootstrap 3.2+, uses the predefined class .embed-responsive
, an aspect ratio class like .embed-responsive-16by9
, and the .embed-responsive-item
for the iframe. Check out the examples below or view their Embeds documentation.
<!-- 21:9 aspect ratio -->
<div class="embed-responsive embed-responsive-21by9">
<iframe class="embed-responsive-item" src="..."></iframe>
</div>
<!-- 16:9 aspect ratio -->
<div class="embed-responsive embed-responsive-16by9">
<iframe class="embed-responsive-item" src="..."></iframe>
</div>
<!-- 4:3 aspect ratio -->
<div class="embed-responsive embed-responsive-4by3">
<iframe class="embed-responsive-item" src="..."></iframe>
</div>
<!-- 1:1 aspect ratio -->
<div class="embed-responsive embed-responsive-1by1">
<iframe class="embed-responsive-item" src="..."></iframe>
</div>
Within Bootstrap’s _variables.scss
, you can change the aspect ratios. Here’s an example of the $embed-responsive-aspect-ratios
list:
$embed-responsive-aspect-ratios: (
(21 9),
(16 9),
(4 3),
(1 1)
) !default;
Responsive iframes in Materialize
If you are using Materialize CSS, then you don’t need your own classes either. Just add the .video-container
class to your wrapper:
<div class="video-container">
<iframe src="https://www.youtube.com/embed/K1K8s-tQGqY" frameborder="0" allowfullscreen></iframe>
</div>
Responsive iframes in Foundation
<div class="responsive-embed">
<iframe src="https://www.youtube.com/embed/K1K8s-tQGqY" frameborder="0" allowfullscreen></iframe>
</div>
Aspect ratio modifier classes are set in your $responsive-embed-ratios
map in your Foundation settings file:
$responsive-embed-ratios: (
default: 16 by 9,
vertical: 9 by 16,
panorama: 256 by 81,
square: 1 by 1,
);
Responsive iframes in Semantic UI
Semantic UI provides an embed module that allows you to create aspect ratio boxes for videos, iframes, and more. See their Embed documentation for more information.
$('.url.example .ui.embed').embed();
<div class="ui embed" data-url="https://www.youtube.com/embed/O6Xo21L0ybE" data-placeholder="/images/bear-waving.jpg"></div>
Responsive iframes in Bulma
In Bulma, you can apply a specific ratio on any element by applying the has-ratio
modifier to a resizable element. Check out the example below or see their Arbitrary ratios with any element documentation.
For example, you can apply a 16by9
ratio on an iframe
.
<figure class="image is-16by9">
<iframe class="has-ratio" width="640" height="360" src="https://www.youtube.com/embed/YE7VzlLtp-4?showinfo=0" frameborder="0" allowfullscreen></iframe>
</figure>
Here’s a list of all the available Bulma aspect ration classes:
is-square
oris-1by1
– 1×1is-5by4
– 5×4is-4by3
– 4×3is-3by2
– 3×2is-5by3
– 5×3is-16by9
– 16×9is-2by1
– 2×1is-3by1
– 3×1is-4by5
– 4×5is-3by4
– 3×4is-2by3
– 2×3is-3by5
– 3×5is-9by16
– 9×16is-1by2
– 1×2is-1by3
– 1×3
What if the aspect ratio is dynamic?
This is the only time JavaScript should be used. Let’s say you have content authors creating interactives with each having different dimensions. Without knowing the aspect ratio of the iframe, it’s not easy to implement the intrinsic ratio technique.
You can overcome this problem by using JS. There’s a number of JS libraries out there (Pym.js or this jQuery plugin), or you can use this little code snippet.
function resizeAspectRatioBoxes() {
var
$this = $(this),
proportion = $this.data('proportion'),
w = $this.attr('width'),
actual_w = $this.width();
if (!proportion) {
proportion = $this.attr('height') / w;
$this.data('proportion', proportion);
}
if (actual_w != w) {
$this.css('height', Math.round(actual_w * proportion) + 'px');
}
}
$(window).resize(function() {
resizeAspectRatioBoxes();
}):
Responsive iframes are awesome.
Say Goodbye to embedded content breaking your layouts with aspect ratio boxes using the intrinsic ratio technique. No longer do you have to deal with those annoying gaps iframe containers make as content width changes. Just keep in mind these tips when building aspect ratio boxes:
Things to remember.
- First, the content within the iframe must be responsive. If not, it defeats the purpose of creating an aspect ratio box.
- Don’t forget to specify the containers
position
to berelative
. This allows the absolute positioning of theiframe
within it. - The
padding-top
value is calculated based on the aspect ratio of your content. You can calculate this value using:(height / width) * 100 = aspect ratio precent.
height
is set to0
becausepadding-bottom
gives theiframe
it’s height.- Using
overflow: hidden
is important because it ensures if any content does protrude outside of the container, it will be hidden and avoid screwing up the site’s layout. - Like with most
absolute
positioned elements, we need to set thetop
andleft
properties so theiframe
get’s put in the right place. - Finally,
width
andheight
are set to100%
so theiframe
takes up 100% of the containers’ space.
Using aspect ratio boxes is great for all kinds of content, not just iframes. We can use this same technique to make other types of embedded content responsive like Google Maps, calendars, Vimeo, and YouTube videos. Basically, anything that needs to keep its aspect ratio as the screen size changes. How are you using them on your site? Comment below.
FAQ
A responsive iframe is a iframe that “responds” to its container. For instance, 100% width, 100vw, or 100vh.
It’s a container that adjusts its height based on its width to always keep its aspect ratio (i.e. 16×9, 4×3, 1×1, etc.).
It’s simple: (height / width) * 100.
Intrinsic ratios maintains an element aspect ratio when resized. Think of an img
with max-width: 100%
. Change the width of its parent and it’ll change the size while keeping the same shape (aka. its aspect ratio).
Do you have a question about aspect ratio boxes, the intrinsic ratio technique or how to make iframes responsive? Or maybe you have another nifty technique. I wanna hear from you. Post your questions, comments, or suggestions in the comments below.
More about iframes & aspect ratio boxes.
With the numerous screen sizes, there’s a host of things to consider to ensure your site looks good no matter the device. Not only is it important to consider the responsiveness of elements, but the performance too. Check out these other articles on iframes, responsive performance, and techniques to keep things sized right.
154 comments on “Responsive iframes”.
# Sep 8, 2016
These tricks worked perfectly, thank you so much ! 🙂
# Aug 28, 2016
Thank you! Nothing except your "Proportionally Resize iframes Using JS" have helped me!
# Aug 23, 2016
Great tutorial!
But the iframe only shows up when I change the height. When the height is 0, the iframe will not display. What can I do about it?
# Oct 25, 2016
Height needs to be zero. You need to apply the padding styles for height.
# Aug 22, 2016
This will not work for my site. I have tried every solution provided but to no avail. Can anyone help with this? I am using Bootstrap.
# Aug 18, 2016
I was able to get the iframe to be responsive but the contents in the iframe are not.
# Oct 25, 2016
That’s because that URL you’re loading isn’t responsive. The page you’re loading also needs to be responsive for it to work.
# Aug 12, 2016
So if I’m using a service like Website Builder with Godaddy (I’m not a website coder), I have to have access to CSS, which I don’t. Any other way to do it?
# Aug 8, 2016
Thank you Ben ! The article was of great help !
# Aug 4, 2016
Important: Don’t forget to apply the styles above to your iframe. If you don’t, it could cause the iframe to disappear.
My iframe disappeared. I’m not sure what you mean by apply the styles to the iframe.
# May 9, 2017
I’m having the same question..
# Jul 27, 2016
Nice article!
I inherrited some iframes that have image maps inside them. These are notoriously mobile unfriendly. I’m looking at some jquery which might solve the problem, but your solution for the iframes is brilliant.
Thanks!
# Jul 13, 2016
Hello! This is wonderful idea! Thank you very much! 😀
# May 29, 2016
Your CSS example worked well for me creating a responsive embed for slides.com, which doesn’t use its own responsive script loader (like speakerdeck.com). Thanks!
# May 15, 2016
In your generateRatios mixin, you swapped the width/height in the ratio() function. Should be: ratio($width, $height)
# May 16, 2016
Thanks for the catch, updated!
# May 8, 2016
I had to view source on your example to get it (the "Don’t forget to apply the styles above to your iframe!" note was what I needed but, not being a programmer, didn’t understand). Anyway, got it, brilliant tutorial, thanks so much!
# Jul 14, 2016
Miriam, I suspect I am experiencing the same translation challenge. Can you elaborate on what it is that you did to fix the issue?
# Sep 23, 2016
I still don’t get it.
From the comment, "Don’t forget to apply the styles above to your iframe!”, I assume there’s something that needs to be done to the line, but I guess I’m even less of a coder than are you, as I don’t even know how to look at the source of his example. I right-clicked on the scrolling frame and selected "View page source", but I think that’s showing the source for this entire page, not just the scrolling window/frame, and when I search in that source-code for "
# Sep 26, 2016
After a weekend, still no success.
And as I’ve thought about it, as great a work-around as this is, it is a work-around, around a broken-ness that should not exist in the first place.
should work. If that broken-ness were fixed, we’d be done with it.
I understand there’s probably a good reason for this behavior (something about iframes not being block-level elements, blah blah blah), but intuitively, height="100%" should work.
# Sep 26, 2016
My example code got wiped out by the commenting system (with no warning, and apparently no way to edit my post to fix it – arg!). Without the code, my statement was essentially that using the width="100%" format should work.
# Apr 4, 2016
Works great Ben, thank you for sharing this solution.
# Apr 9, 2015
I followed this example exactly, including your example frame and the frame also disappears for me.
# May 20, 2015
Be sure to include the padding class if following this example word by word. If no padding-bottom exists on the container, it won’t show.
# Apr 9, 2015
It is helpful!!
You all are try this one also:-
http://shar.es/1gysdZ
# Mar 24, 2015
I had the same issue – my videos disappeared trying your code. This example worked for me: http://avexdesigns.com/responsive-youtube-embed/
# Feb 16, 2015
Hey, thanks for this information, solved the challenge I was faced with. I appreciate your taking the time to post.
# Dec 3, 2014
i had the same problem. added the div and it disappeared.
also tried the js code and it didn’t work. i’d love to get this to work.
br
# Feb 3, 2015
Most likely you’ve iframe is inheriting a style it shouldn’t have. I’ve included an example in the post to see how it works.
# Nov 6, 2014
hi,
after adding div to my page and code to css file, the iframe no longer appears 🙁 can you please advise? thank you!
# Nov 18, 2014
Can I see the code you’re using?
# Jan 15, 2015
Very nice article . Thanks a lot for putting this up.
# Feb 3, 2015
I had the same issue as marcin and changed height of div from 0% to 100% and then it worked perfectly.
# Jul 5, 2017
Had the same issue as marcin. Your solution solved the problem. Thanks Sameer!
All comments posted on 'Responsive iframes' are held for moderation and only published when on topic and not rude. Get a gold star if you actually read & follow these rules.
You may write comments in Markdown. This is the best way to post any code, inline like `<div>this</div>` or multiline blocks within triple backtick fences (```) with double new lines before and after.
Want to tell me something privately, like pointing out a typo or stuff like that? Contact Me.