Как применить CSS к iframe?

942

У меня есть простая страница с некоторыми разделами iframe (для отображения ссылок на RSS). Как я могу применить тот же формат CSS от главной страницы к странице, отображаемой в iframe?

  • 61
    Это возможно, но только если домен iframe такой же, как родительский
  • 11
    gawpertron, просто чтобы уточнить, вы говорите, что если я использую контент iFrame из какого-то другого домена, который я не контролирую, я не смогу контролировать CSS для этого контента?
Показать ещё 2 комментария
Теги:
iframe

26 ответов

375

Изменить: это не работает с перекрестным доменом, если не задан соответствующий заголовок CORS.

Здесь есть две разные вещи: стиль блока iframe и стиль страницы, встроенной в iframe. Вы можете установить стиль блока iframe обычным способом:

<iframe name="iframe1" id="iframe1" src="empty.htm" 
        frameborder="0" border="0" cellspacing="0"
        style="border-style: none;width: 100%; height: 120px;"></iframe>

Стиль страницы, встроенной в iframe, должен быть задан путем включения его в дочернюю страницу:

<link type="text/css" rel="Stylesheet" href="Style/simple.css" />

Или он может быть загружен с родительской страницы с помощью Javascript:

var cssLink = document.createElement("link");
cssLink.href = "style.css"; 
cssLink.rel = "stylesheet"; 
cssLink.type = "text/css"; 
frames['iframe1'].document.head.appendChild(cssLink);
  • 27
    Пожалуйста, обратите внимание, мне кажется, что некоторые примеры, опубликованные ранее, теперь недействительны для html5. Вы можете получить доступ к содержимому фрейма следующим образом: document.getElementById("myframe").contentDocument . Однако встраивание CSS все еще не работает для меня.
  • 34
    ссылка может появиться только в заголовке
Показать ещё 11 комментариев
182

Я встретил эту проблему с Календарем Google. Я хотел стилизовать его на более темном фоне и изменить шрифт.

К счастью, URL-адрес из кода встраивания не имел никакого ограничения на прямой доступ, поэтому, используя PHP-функцию file_get_contents, можно получить весь контент со страницы. Вместо вызова URL-адреса Google можно вызвать php файл, расположенный на вашем сервере, например. google.php, который будет содержать исходное содержимое с изменениями:

$content = file_get_contents('https://www.google.com/calendar/embed?src=%23contacts%40group.v.calendar.google.com&ctz=America/Montreal');

Добавление пути к вашей таблице стилей:

$content = str_replace('</head>','<link rel="stylesheet" href="http://www.yourwebsiteurl.com/google.css" /></head>', $content);

(Это поместит вашу таблицу стилей как раз перед тегом head.)

Укажите базовый url из исходного URL в случае, если css и js называются относительно:

$content = str_replace('</title>','</title><base href="https://www.google.com/calendar/" />', $content);

Окончательный файл google.php должен выглядеть следующим образом:

<?php
$content = file_get_contents('https://www.google.com/calendar/embed?src=%23contacts%40group.v.calendar.google.com&ctz=America/Montreal');
$content = str_replace('</title>','</title><base href="https://www.google.com/calendar/" />', $content);
$content = str_replace('</head>','<link rel="stylesheet" href="http://www.yourwebsiteurl.com/google.css" /></head>', $content);
echo $content;

Затем вы изменяете код iframe embed:

<iframe src="http://www.yourwebsiteurl.com/google.php" style="border: 0" width="800" height="600" frameborder="0" scrolling="no"></iframe>

Удачи!

  • 0
    Прежде всего, я хочу сказать, что это отличное решение, однако оно не всегда работает. Если вы попробуете это с YouTube, это будет просто навсегда буфера. Есть идеи почему?
  • 46
    К вашему сведению, это в основном взлом.
Показать ещё 11 комментариев
73

Если содержимое iframe не полностью находится под вашим контролем или вы хотите получить доступ к содержимому с разных страниц с разными стилями, вы можете попробовать манипулировать им с помощью JavaScript.

var frm = frames['frame'].document;
var otherhead = frm.getElementsByTagName("head")[0];
var link = frm.createElement("link");
link.setAttribute("rel", "stylesheet");
link.setAttribute("type", "text/css");
link.setAttribute("href", "style.css");
otherhead.appendChild(link);

Обратите внимание, что в зависимости от того, какой браузер вы используете, это может работать только на страницах, обслуживаемых одним и тем же доменом.

  • 78
    Стоит отметить, что одна и та же политика происхождения прекратит эту работу, если страница находится в другом домене.
  • 2
    В том же <iframe onload="this.contentDocument.body.style.overflow='hidden';" /> но более кратко: <iframe onload="this.contentDocument.body.style.overflow='hidden';" />
54
var $head = $("#eFormIFrame").contents().find("head");

$head.append($("<link/>", {
    rel: "stylesheet",
    href: url,
    type: "text/css"
}));
  • 0
    @deweydb Я использовал его с кросс-доменным iframe. но я могу только попросить у тебя прощения!
  • 0
    @ramie, а вы тестировали это в нескольких браузерах? большинство браузеров выдают ошибку безопасности.
Показать ещё 1 комментарий
35

В большинстве браузеров iframe универсально обрабатывается как другая HTML-страница. Если вы хотите применить ту же таблицу стилей к содержимому iframe, просто ссылайтесь на нее с используемых там страниц.

  • 9
    @ Ханги, но ... как?
25

Если вы управляете страницей в iframe, как сказал hangy, самый простой способ - создать общий файл CSS с общими стилями, а затем просто ссылаться на него со своих html-страниц.

В противном случае вам вряд ли удастся динамически изменить стиль страницы с внешней страницы в вашем iframe. Это связано с тем, что браузеры ужесточили безопасность при использовании сценариев межкадрового доступа из-за возможного неправильного использования для подмены и других хаков.

Этот учебник может предоставить вам дополнительную информацию о сценариях iframes в целом. О скриптах с поперечными рамками объясняет ограничения безопасности с точки зрения IE.

24

Вот как применить CSS-код напрямую, не используя <link> для загрузки дополнительной таблицы стилей.

var head = jQuery("#iframe").contents().find("head");
var css = '<style type="text/css">' +
          '#banner{display:none}; ' +
          '</style>';
jQuery(head).append(css);

Это скрывает баннер на странице iframe. Спасибо за ваши предложения!

  • 0
    Нужен ли для этого iframe с идентификатором iframe ?
  • 0
    @jmalais Просто замените #iframe на #<id> или даже iframe чтобы выбрать все iframe
Показать ещё 2 комментария
20

Вышеуказанное с небольшими изменениями:

var cssLink = document.createElement("link") 
cssLink.href = "pFstylesEditor.css"; 
cssLink.rel = "stylesheet"; 
cssLink.type = "text/css"; 

//Instead of this
//frames['frame1'].document.body.appendChild(cssLink);
//Do this

var doc=document.getElementById("edit").contentWindow.document;

//If you are doing any dynamic writing do that first
doc.open();
doc.write(myData);
doc.close();

//Then append child
doc.body.appendChild(cssLink);

Работает нормально с ff3 и ie8 как минимум

  • 6
    Что такое мои данные и откуда они берутся
  • 6
    прочитайте комментарий
Показать ещё 1 комментарий
12

Если вы хотите повторно использовать CSS и JavaScript с главной страницы, возможно, вам стоит рассмотреть возможность замены <IFRAME> на загруженный контент Ajax. Теперь это более дружественное к SEO, когда поисковые роботы могут выполнять JavaScript.

Это jQuery пример, который включает другую html-страницу в ваш документ. Это гораздо больше SEO, чем iframe. Чтобы быть уверенным, что боты не индексируют включенную страницу, просто добавьте ее для запрета в robots.txt

<html>
  <header>
    <script src="/js/jquery.js" type="text/javascript"></script>
  </header>
  <body>
    <div id='include-from-outside'></div>
    <script type='text/javascript'>
      $('#include-from-outside').load('http://example.com/included.html');
    </script> 
  </body>
</html>

Вы также можете включить jQuery прямо из Google: http://code.google.com/apis/ajaxlibs/documentation/ - это означает, что необходимо автоматическое включение новых версий и значительное увеличение скорости. Кроме того, это означает, что вы должны доверять им для доставки вам только jQuery;)

  • 0
    Следует отметить, что это решение не работает, если содержимое страницы является динамическим в любом случае.
11

Следующие работали для меня.

var iframe = top.frames[name].document;
var css = '' +
          '<style type="text/css">' +
          'body{margin:0;padding:0;background:transparent}' +
          '</style>';
iframe.open();
iframe.write(css);
iframe.close();
  • 1
    Ошибка: Uncaught TypeError: Cannot read property 'open' of undefined
8

Расширение на вышеупомянутое решение jQuery, чтобы справиться с любыми задержками при загрузке содержимого фрейма.

$('iframe').each(function(){
    function injectCSS(){
        $iframe.contents().find('head').append(
            $('<link/>', { rel: 'stylesheet', href: 'iframe.css', type: 'text/css' })
        );
    }

    var $iframe = $(this);
    $iframe.on('load', injectCSS);
    injectCSS();
});
  • 1
    '$ This' должно быть '$ (this)', чтобы оно работало. Приятно, когда это происходит :)
8

Моя компактная версия:

<script type="text/javascript">
$(window).load(function () {
    var frame = $('iframe').get(0);
    if (frame != null) {
        var frmHead = $(frame).contents().find('head');
        if (frmHead != null) {
            frmHead.append($('style, link[rel=stylesheet]').clone()); // clone existing css link
            //frmHead.append($("<link/>", { rel: "stylesheet", href: "/styles/style.css", type: "text/css" })); // or create css link yourself
        }
    }   
});
</script>

Однако иногда iframe не готов к загрузке окна, поэтому необходимо использовать таймер.

Готовый к использованию код (с таймером):

<script type="text/javascript">
var frameListener;
$(window).load(function () {
    frameListener = setInterval("frameLoaded()", 50);
});
function frameLoaded() {
    var frame = $('iframe').get(0);
    if (frame != null) {
        var frmHead = $(frame).contents().find('head');
        if (frmHead != null) {
            clearInterval(frameListener); // stop the listener
            frmHead.append($('style, link[rel=stylesheet]').clone()); // clone existing css link
            //frmHead.append($("<link/>", { rel: "stylesheet", href: "/styles/style.css", type: "text/css" })); // or create css link yourself
        }
    }
}
</script>

... и ссылка jQuery:

<script src="https://ajax.aspnetcdn.com/ajax/jQuery/jquery-1.9.1.min.js" type="text/javascript"></script>
6

Когда вы говорите "doc.open()", это означает, что вы можете написать любой HTML-тег внутри iframe, поэтому вы должны написать все основные теги для HTML-страницы, и если вы хотите иметь ссылку CSS в своей голове iframe просто напишите iframe с ссылкой CSS. Я приведу вам пример:

doc.open();

doc.write('<!DOCTYPE html><html><head><meta charset="utf-8"/><meta http-quiv="Content-Type" content="text/html; charset=utf-8"/><title>Print Frame</title><link rel="stylesheet" type="text/css" href="/css/print.css"/></head><body><table id="' + gridId + 'Printable' + '" class="print" >' + out + '</table></body></html>');

doc.close();
5

Вы не сможете стирать содержимое iframe таким образом. Мое предложение было бы использовать серверный скриптинг (PHP, ASP или Perl script) или найти онлайн-сервис, который преобразует фид в код JavaScript. Единственный способ сделать это - это сделать, если вы можете сделать serveride include.

  • 25
    Осторожнее, когда говоришь, что что-то сделать нельзя, а на самом деле это просто сложно
3

use может попробовать следующее:

$('iframe').load( function() {
     $('iframe').contents().find("head")
     .append($("<style type='text/css'>  .my-class{display:none;}  </style>"));
  });
  • 3
    Если ваш iframe имеет другое происхождение, механизм CORS не допустит этот обходной путь.
  • 0
    Что если в iframe-url CORS включен для всех местоположений?
3

Другие ответы здесь, похоже, используют ссылки jQuery и CSS.

В этом коде используется ванильный JavaScript. Он создает новый элемент <style>. Он устанавливает текстовое содержимое этого элемента как строку, содержащую новый CSS. И он добавляет этот элемент непосредственно в заголовок документа iframe.

var iframe = document.getElementById('the-iframe');
var style = document.createElement('style');
style.textContent =
  '.some-class-name {' +
  '  some-style-name: some-value;' +
  '}' 
;
iframe.contentDocument.head.appendChild(style);
2

Если у вас есть доступ к странице iframe и вы хотите, чтобы другой CSS применялся на нем только при загрузке через iframe на вашей странице, здесь я нашел решение для таких вещей

это работает, даже если iframe загружает другой домен

проверить postMessage()

plan, отправьте css в iframe как сообщение, например

iframenode.postMessage('h2{color:red;}','*');

* - отправить это сообщение независимо от того, в каком домене он находится в iframe

и получите сообщение в iframe и добавьте полученное сообщение (CSS) в этот заголовок документа.

код для добавления на страницу iframe

window.addEventListener('message',function(e){

    if(e.data == 'send_user_details')
    document.head.appendChild('<style>'+e.data+'</style>');

});
2

Я нашел другое решение, чтобы поместить стиль в основной html, как этот

<style id="iframestyle">
    html {
        color: white;
        background: black;
    }
</style>
<style>
    html {
        color: initial;
        background: initial;
    }
    iframe {
        border: none;
    }
</style>

а затем в iframe это сделать (см. js onload)

<iframe  onload="iframe.document.head.appendChild(ifstyle)" name="log" src="/upgrading.log"></iframe>

и в js

<script>
    ifstyle = document.getElementById('iframestyle')
    iframe = top.frames["log"];
</script>

Это может быть не лучшее решение, и его, безусловно, можно улучшить, но это еще один вариант, если вы хотите сохранить тег "style" в родительском окне

2

Мы можем вставить тег стиля в iframe. Добавлено также здесь...

<style type="text/css" id="cssID">
.className
{
    background-color: red;
}
</style>

<iframe id="iFrameID"></iframe>

<script type="text/javascript">
    $(function () {
        $("#iFrameID").contents().find("head")[0].appendChild(cssID);
        //Or $("#iFrameID").contents().find("head")[0].appendChild($('#cssID')[0]);
    });
</script>
  • 0
    Это не работает Кажется, правильно вставить тег стиля, но внутри него нет содержимого и нет идентификатора.
  • 0
    не работает в случае кросс-домена
1

Я думаю, что самый простой способ - добавить еще один div в том же месте, что и iframe, затем

сделать свой z-index больше, чем контейнер iframe, так что вы можете легко просто создать свой собственный div. Если вам нужно щелкнуть по нему, просто используйте pointer-events:none в своем собственном div, поэтому iframe будет работать, если вам нужно щелкнуть по нему;)

Я надеюсь, что это поможет кому-то;)

0

В качестве альтернативы вы можете использовать технологию CSS-in-JS, например ниже lib:

https://github.com/cssobj/cssobj

Он может вставлять JS-объект как CSS в iframe, динамически

0

Здесь есть две вещи внутри домена

  1. Раздел iFrame
  2. Страница загружена внутри рамки

Итак, вы хотите оформить эти два раздела следующим образом:

1. Стиль для раздела iFrame

Он может создавать стили с использованием CSS с этим уважаемым id или именем class. Вы можете просто стилизовать его в родительских таблицах стилей.

<style>
#my_iFrame{
height: 300px;
width: 100%;
position:absolute;
top:0;
left:0;
border: 1px black solid;
}
</style>

<iframe name='iframe1' id="my_iFrame" src="#" cellspacing="0"></iframe>

2. Стиль страницы, загруженной в iFrame

Эти стили могут быть загружены с родительской страницы с помощью Javascript

var cssFile  = document.createElement("link") 
cssFile.rel  = "stylesheet"; 
cssFile.type = "text/css"; 
cssFile.href = "iFramePage.css"; 

затем установите этот CSS файл в соответствующий раздел iFrame

//to Load in the Body Part
frames['my_iFrame'].document.body.appendChild(cssFile); 
//to Load in the Head Part
frames['my_iFrame'].document.head.appendChild(cssFile);

Здесь вы также можете редактировать заголовок страницы внутри iFrame также

var $iFrameHead = $("#my_iFrame").contents().find("head");
$iFrameHead.append(
   $("<link/>",{ 
      rel: "stylesheet", 
      href: urlPath, 
      type: "text/css" }
     ));
0

var link1 = document.createElement('link');
    link1.type = 'text/css';
    link1.rel = 'stylesheet';
    link1.href = "../../assets/css/normalize.css";
window.frames['richTextField'].document.body.appendChild(link1);
  • 1
    Я проверял этот ответ много раз, что здесь такое richTextField ?
  • 1
    это имя iframe
Показать ещё 2 комментария
0

Существует замечательный script, который заменяет node самой версией iframe. CodePen Демо

Изображение 2603

Примеры использования:

// Single node
var component = document.querySelector('.component');
var iframe = iframify(component);

// Collection of nodes
var components = document.querySelectorAll('.component');
var iframes = Array.prototype.map.call(components, function (component) {
  return iframify(component, {});
});

// With options
var component = document.querySelector('.component');
var iframe = iframify(component, {
  headExtra: '<style>.component { color: red; }</style>',
  metaViewport: '<meta name="viewport" content="width=device-width">'
});
  • 0
    Почему вы хотите это сделать?!
-1

Лучший способ может быть следующим (если вы контролируете целевой сайт):

1) установите ссылку iframe с параметром style, например:

http://your_site.com/target.php?style=a%7Bcolor%3Ared%7D

(последняя фраза a{color:red}, закодированная функцией urlencode.

2) установите страницу получателя target.php следующим образом:

<head>
..........
<style><? echo urldecode($_GET['style']);?> </style>
..........
  • 0
    Предупреждение: это инъекция во всей красе.
-16

Хорошо, я выполнил следующие шаги:

  • Div с классом для хранения iframe
  • Добавьте iframe в div.
  • В файле CSS
divClass { width: 500px; height: 500px; }
divClass iframe { width: 100%; height: 100%; }

Это работает в IE 6. Если вы работаете в других браузерах, проверьте!

  • 9
    нужно контролировать div внутри iframe, это не

Ещё вопросы

Сообщество Overcoder
Наверх
Меню