292 votes

Adapter l'Iframe à 100% de la hauteur restante du conteneur

Je veux concevoir une page web avec une bannière et une iframe. J'espère que l'iframe pourra remplir toute la hauteur restante de la page et être redimensionnée automatiquement lorsque le navigateur est redimensionné. Est-il possible d'y parvenir sans écrire de code JavaScript, uniquement avec des CSS ?

J'ai essayé de mettre height:100% sur l'iframe, le résultat est assez proche mais l'iframe a essayé de remplir toute la hauteur de la page, y compris le 30px hauteur de l'élément div de la bannière, ce qui fait que j'ai une barre de défilement verticale inutile. Ce n'est pas parfait.

J'ai essayé les attributs CSS margin, padding sur les DIV pour occuper toute la hauteur restante d'une page web avec succès, mais l'astuce n'a pas fonctionné sur les iframe.

 <body>
    <div style="width:100%; height:30px; background-color:#cccccc;">Banner</div>
    <iframe src="http: //www.google.com.tw" style="width:100%; height:100%;"></iframe>
</body>

272voto

Vilx- Points 37939

Mise à jour en 2019

TL;DR : Aujourd'hui, la meilleure option est - flexbox . Tout le monde le supporte bien et cela depuis des années. Allez-y et ne regardez pas en arrière. Voici un exemple de code pour flexbox :

body, html {width: 100%; height: 100%; margin: 0; padding: 0}
.row-container {display: flex; width: 100%; height: 100%; flex-direction: column; background-color: blue; overflow: hidden;}
.first-row {background-color: lime; }
.second-row { flex-grow: 1; border: none; margin: 0; padding: 0; }

<div class="row-container">
  <div class="first-row">
    <p>Some text</p>
    <p>And some more text</p>
  </div>
  <iframe src="https://jsfiddle.net/about" class="second-row"></iframe>
</div>

Le reste de cette réponse est laissé pour des raisons pédagogiques et historiques.


L'astuce consiste à comprendre de quoi sont faits les 100%. La lecture des spécifications CSS peut vous y aider.

Pour faire court, il existe un "bloc contenant", qui n'est pas nécessairement l'élément parent. En d'autres termes, c'est le premier élément de la hiérarchie qui a une position:relative ou une position:absolue. Ou l'élément body lui-même s'il n'y a rien d'autre. Ainsi, lorsque vous dites "width : 100%", le système vérifie la largeur du "bloc contenant" et définit la largeur de votre élément à la même taille. S'il y avait autre chose, le contenu du "bloc contenant" pourrait être plus grand que lui-même (donc "déborder").

La hauteur fonctionne de la même manière. A une exception près. Vous ne pouvez pas obtenir une hauteur correspondant à 100 % de la fenêtre du navigateur. L'élément de très haut niveau, par rapport auquel on peut calculer la hauteur de 100%, est l'élément body (ou html ? je ne suis pas sûr), et il s'étire juste assez pour contenir son contenu. Spécifier height:100% sur cet élément n'aura aucun effet, car il n'a pas d'"élément parent" par rapport auquel mesurer 100%. La fenêtre elle-même ne compte pas ;)

~~

Pour que quelque chose s'étende exactement sur 100 % de la fenêtre, vous avez deux possibilités :

  1. Utilisez JavaScript
  2. N'utilisez pas DOCTYPE. Ce n'est pas une bonne pratique, mais cela met les navigateurs en "mode bizarrerie", dans lequel vous pouvez faire height="100%" sur les éléments et il les étire à la taille de la fenêtre. Notez que le reste de votre page devra probablement être modifié pour s'adapter aux changements DOCTYPE.

~~

Mise à jour : Je ne suis pas sûr de ne pas m'être déjà trompé lorsque j'ai posté ceci, mais c'est certainement dépassé maintenant. Aujourd'hui, vous pouvez faire cela dans votre feuille de style : html, body { height: 100% } et il s'étendra réellement à l'ensemble de votre fenêtre d'affichage. Même avec un DOCTYPE. min-height: 100% pourrait également être utile, en fonction de votre situation.

Et je Je ne conseillerais à personne de créer un document en mode bizarrerie. plus non plus, parce que ça cause beaucoup plus de maux de tête que ça n'en résout. Chaque navigateur a un mode de fonctionnement différent, de sorte qu'il est deux fois plus difficile de faire en sorte que votre page soit cohérente d'un navigateur à l'autre. Utilisez un DOCTYPE. Toujours. De préférence le HTML5. <!DOCTYPE html> . Il est facile à retenir et fonctionne comme un charme dans tous les navigateurs, même ceux qui ont 10 ans.

La seule exception est lorsque vous devez prendre en charge quelque chose comme IE5 ou autre. Si vous en êtes là, alors vous êtes tout seul de toute façon. Ces anciens navigateurs n'ont rien à voir avec les navigateurs d'aujourd'hui, et peu de conseils qui sont donnés ici vous aideront avec eux. Le bon côté des choses, c'est que si vous en êtes là, vous n'avez probablement qu'à supporter UN seul type de navigateur, ce qui élimine les problèmes de compatibilité.

Bonne chance !

Mise à jour 2 : Hé, ça fait longtemps ! 6 ans plus tard, de nouvelles options font leur apparition. Je viens d'avoir une discussion dans les commentaires ci-dessous, voici d'autres astuces pour vous qui fonctionnent dans les navigateurs d'aujourd'hui.

Option 1 - positionnement absolu. Sympathique et propre pour les cas où vous connaissez la hauteur précise de la première partie.

body, html {width: 100%; height: 100%; margin: 0; padding: 0}
.first-row {position: absolute;top: 0; left: 0; right: 0; height: 100px; background-color: lime;}
.second-row {position: absolute; top: 100px; left: 0; right: 0; bottom: 0; background-color: red }
.second-row iframe {display: block; width: 100%; height: 100%; border: none;}

<div class="first-row">
  <p>Some text</p>
  <p>And some more text</p>
</div>
<div class="second-row">
  <iframe src="https://jsfiddle.net/about"></iframe>
</div>

Quelques notes - le second-row est nécessaire car bottom: 0 y right: 0 ne fonctionne pas sur les iframes pour une raison quelconque. Quelque chose à voir avec le fait que in est un élément "remplacé". Mais width: 100% y height: 100% fonctionne très bien. display: block est nécessaire parce que c'est un inline par défaut et les espaces blancs commencent à créer des débordements bizarres sinon.

Option 2 - tables. Fonctionne lorsque vous ne connaissez pas la hauteur de la première partie. Vous pouvez utiliser soit la hauteur réelle <table> ou le faire de manière plus sophistiquée avec display: table . Je vais opter pour la seconde solution, car elle semble être à la mode en ce moment.

body, html {width: 100%; height: 100%; margin: 0; padding: 0}
.row-container {display: table; empty-cells: show; border-collapse: collapse; width: 100%; height: 100%;}
.first-row {display: table-row; overflow: auto; background-color: lime;}
.second-row {display: table-row; height: 100%; background-color: red; overflow: hidden }
.second-row iframe {width: 100%; height: 100%; border: none; margin: 0; padding: 0; display: block;}

<div class="row-container">
  <div class="first-row">
    <p>Some text</p>
    <p>And some more text</p>
  </div>
  <div class="second-row">
    <iframe src="https://jsfiddle.net/about"></iframe>
  </div>
</div>

Quelques notes - le overflow: auto permet de s'assurer que la ligne inclut toujours l'ensemble de son contenu. Sinon, les éléments flottants peuvent parfois déborder. Le site height: 100% sur la deuxième rangée fait en sorte qu'elle s'étende autant qu'elle le peut en comprimant la première rangée au maximum.

Recommandé : Option 3 - flexbox - La plus propre de toutes.

body, html {width: 100%; height: 100%; margin: 0; padding: 0}
.row-container {display: flex; width: 100%; height: 100%; flex-direction: column; background-color: blue; overflow: hidden;}
.first-row {background-color: lime; }
.second-row { flex-grow: 1; border: none; margin: 0; padding: 0; }

<div class="row-container">
  <div class="first-row">
    <p>Some text</p>
    <p>And some more text</p>
  </div>
  <iframe src="https://jsfiddle.net/about" class="second-row"></iframe>
</div>

Quelques notes - le overflow: hidden c'est parce que l'iframe génère toujours une sorte de débordement même avec display: block dans ce cas. Il n'est pas visible dans la vue plein écran ou dans la vue de l'ordinateur. [ ]

0 votes

Doit-on mettre cela dans la feuille de style de la page de l'IFrame, ou dans la feuille de style de la page parent qui contient l'IFrame ?

0 votes

@sproketboy - à l'origine, css était surtout un langage de formatage de texte avec une fonctionnalité de mise en page de base qui était encore réellement centrée sur le texte - il s'est avéré être meilleur que les tableaux pour la mise en page, d'où son adoption - à bien des égards, nous piratons les mises en page depuis lors

0 votes

@ToniLeigh Non, les tableaux sont généralement bien meilleurs pour la mise en page. Elles fonctionnent de manière cohérente sur tous les navigateurs. La seule raison pour laquelle CSS a des aspects de mise en page est qu'il a été conçu par des sociopathes.

71voto

MichAdel Points 349

Nous utilisons un JavaScript pour résoudre ce problème ; voici la source.


var buffer = 20; //scroll bar buffer
var iframe = document.getElementById('ifm');

function pageY(elem) {
    return elem.offsetParent ? (elem.offsetTop + pageY(elem.offsetParent)) : elem.offsetTop;
}

function resizeIframe() {
    var height = document.documentElement.clientHeight;
    height -= pageY(document.getElementById('ifm'))+ buffer ;
    height = (height < 0) ? 0 : height;
    document.getElementById('ifm').style.height = height + 'px';
}

// .onload doesn't work with IE8 and older.
if (iframe.attachEvent) {
    iframe.attachEvent("onload", resizeIframe);
} else {
    iframe.onload=resizeIframe;
}

window.onresize = resizeIframe;

Note : ifm est l'ID de l'iframe

pageY() a été créé par John Resig (l'auteur de jQuery).

52voto

D1SoveR Points 189

Une autre façon de procéder serait d'utiliser l'option position: fixed; sur le nœud parent.
Si je ne me trompe pas, position: fixed; lie l'élément à viewport, donc, une fois que vous donnez à ce nœud width: 100%; y height: 100%; il s'étendra sur tout l'écran. À partir de ce moment, vous pouvez mettre <iframe> et l'étendre sur l'espace restant (à la fois en largeur et en hauteur) à l'aide de la simple balise width: 100%; height: 100%; Instruction CSS.

Exemple de code


    body {
        margin: 0px;
        padding: 0px;
    }

    /* iframe's parent node */
    div#root {
        position: fixed;
        width: 100%;
        height: 100%;
    }

    /* iframe itself */
    div#root > iframe {
        display: block;
        width: 100%;
        height: 100%;
        border: none;
    }

   <html>
        <head>
            <title>iframe Test</title>
            <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
        </head>
        <body>
            <div id="root">
                <iframe src="http://stackoverflow.com/">
                    Your browser does not support inline frames.
                </iframe>
            </div>
        </body>
    </html>

1 votes

Comment est-ce que cela prend la hauteur "restante".. ?

3 votes

J'ai remarqué que vous devez aussi ajouter position: fixed à l'iframe.

1 votes

Il n'y a pas de barres de défilement visibles

39voto

ducu Points 571

Vous pouvez le faire avec DOCTYPE mais vous devez utiliser table . Regarde ça :

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<style>
*{margin:0;padding:0}
html, body {height:100%;width:100%;overflow:hidden}
table {height:100%;width:100%;table-layout:static;border-collapse:collapse}
iframe {height:100%;width:100%}

.header {border-bottom:1px solid #000}
.content {height:100%}
</style>
</head>
<body>
<table>
    <tr><td class="header"><div><h1>Header</h1></div></td></tr>
    <tr><td class="content">
        <iframe src="http://google.com/" frameborder="0"></iframe></td></tr>
</table>
</body>
</html>

1 votes

Au moins, il n'a pas besoin de js !! mais est-il sûr pour le web ?

1 votes

L'inconvénient de cette solution est que html,body{overflow:hidden;} supprimera les défilements de page.

0 votes

J'essaie d'ajouter le pied de page dans le code ci-dessus mais le pied de page ne s'affiche pas. Pouvez-vous m'aider ?

18voto

Tim Geerts Points 157

Peut-être a-t-on déjà répondu à cette question (quelques réponses ci-dessus sont des façons "correctes" de procéder), mais j'ai pensé ajouter ma solution également.

Notre iFrame est chargée dans un div, c'est pourquoi j'avais besoin d'autre chose que window.height. Et comme notre projet repose déjà largement sur jQuery, je trouve que c'est la solution la plus élégante :

$("iframe").height($("#middle").height());

Où bien sûr "#middle" est l'identifiant de la division. La seule chose supplémentaire que vous devrez faire est de rappeler ce changement de taille chaque fois que l'utilisateur redimensionne la fenêtre.

$(window).resize(function() {
    $("iframe").height($("#middle").height());
});

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X