检测无法加载iframe的内容

时间:2008-12-17 19:20:59

标签: javascript html firefox-addon frames xulrunner

我可以使用 load 事件检测何时加载了iframe的内容。不幸的是,就我的目的而言,这有两个问题:

  • 如果加载页面时出错(404/500等),则永远不会触发加载事件。
  • 如果某些图像或其他依赖项无法加载,则会像往常一样触发load事件。

是否有某种方法可以确定是否发生了上述任何一种错误?

我正在编写一个基于Mozilla / XULRunner的半网络半桌面应用程序,因此欢迎仅适用于Mozilla的解决方案。

6 个答案:

答案 0 :(得分:14)

如果您可以控制iframe页面(并且页面位于同一域名中),则策略可以如下所示:

  • 在父文档中,初始化变量var iFrameLoaded = false;
  • 加载iframe文档时,请在父项中将此变量设置为true,例如,从iframe文档调用父项的函数(setIFrameLoaded();
  • 使用iFrameLoaded对象检查timer标志(将计时器设置为您的首选超时限制) - 如果标志仍为false,则可以判断iframe未定期加载。

我希望这会有所帮助。

答案 1 :(得分:8)

我最近遇到了这个问题,不得不求助于在父页面上设置Javascript轮询操作(包含IFRAME标记)。此JavaScript函数检查IFRAME的内容,以查找仅应存在于GOOD响应中的显式元素。当然,这假定您不必处理违反“同源政策”的问题。

而不是检查可能从许多不同的网络资源生成的所有可能的错误。我只是检查了一个我知道应该是一个良好响应的恒定正元素。

在预定时间和/或尝试检测到预期元素失败后,JavaScript会修改IFRAME的SRC属性(从我的Servlet请求)一个用户友好错误页面,而不是显示典型的HTTP错误消息。 JavaScript也可以轻松地修改SRC属性以产生完全不同的请求。

function checkForContents(){
var contents=document.getElementById('myiframe').contentWindow.document
if(contents){
    alert('found contents of myiframe:' + contents);
    if(contents.documentElement){
        if(contents.documentElement.innerHTML){
            alert("Found contents: " +contents.documentElement.innerHTML);
            if(contents.documentElement.innerHTML.indexOf("FIND_ME") > -1){
                openMediumWindow("woot.html", "mypopup");
            }
        }
    }
}

}

答案 2 :(得分:1)

我认为针对错误页面触发了pageshow事件。或者,如果您是从chrome执行此操作,那么请检查您的进度监听器的请求,看看它是否是HTTP通道,在这种情况下您可以检索状态代码。

至于页面依赖性,我认为你只能通过添加一个捕获onerror事件监听器从chrome中做到这一点,即使这样,它也只能在元素中找到错误,而不是CSS背景或其他图像。

答案 3 :(得分:1)

这是一个很晚的答案,但我会将其留给需要它的人。

任务加载iframe跨域内容,如果成功则发出onLoaded,如果发生错误则发出onError

这是我可以开发的跨浏览器最独立的解决方案。但是首先,我将简要介绍一下我使用的其他方法以及它们为什么不好。

1。 iframe 令我有些震惊,因为iframe仅具有onload事件,并且在加载和出错时都会被调用,无法知道它是否出错。

2。 performance.getEntriesByType('resource')。此方法返回已加载的资源。听起来像我们需要的。但是可惜的是,firefox总是在资源数组中添加资源,无论它是加载还是失败。无法通过Resource实例知道它是否成功。照常。顺便说一句,此方法在ios <11中不起作用。

3。脚本,我尝试使用<script>标签加载html。不幸的是,仅在Chrome中正确发出onloadonerror

当我准备放弃时,我的一位老同事告诉我有关html4标签<object>的信息。它与<iframe>标记类似,但是在未加载内容时会回退。这听起来像我们所需要的!可悲的是,它并不像听起来那样容易。

代码部分

var obj = document.createElement('object');

// we need to specify a callback (i will mention why later)
obj.innerHTML = '<div style="height:5px"><div/>'; // fallback

obj.style.display = 'block'; // so height=5px will work
obj.style.visibility = 'hidden'; // to hide before loaded

obj.data = src;

此后,我们可以像为<object>那样为iframe设置一些属性。唯一的区别是,我们应该使用<params>而不是属性,但是它们的名称和值是相同的。

for (var prop in params) {
    if (params.hasOwnProperty(prop)) {
        var param = document.createElement('param');
        param.name = prop;
        param.value = params[prop];

        obj.appendChild(param);
    }
}

现在,困难的部分。像许多相同元素一样,<object>没有回调规范,因此每个浏览器的行为都不同。

  • Chrome 。错误且加载时会发出load事件。
  • Firefox 。正确发出loaderror
  • Safari 。什么都没发出。...

似乎与iframegetEntriesByTypescript没什么不同。... 但是,我们有本机浏览器后备功能!因此,由于我们直接设置了后备(innerHtml),因此可以判断是否<object>已加载

function isReallyLoaded(obj) {
    return obj.offsetHeight !== 5; // fallback height
}
/**
 * Chrome calls always, Firefox on load
 */
obj.onload = function() {
    isReallyLoaded(obj) ? onLoaded() : onError();
};

/**
 * Firefox on error
 */
obj.onerror = function() {
    onError();
};

但是Safari怎么办?好老的setTimeout

var interval = function() {
    if (isLoaded) { // some flag
        return;
    }

    if (hasResult(obj)) {
        if (isReallyLoaded(obj)) {
            onLoaded();
        } else {
            onError();
        }
    }

    setTimeout(interval, 100);
};

function hasResult(obj) {
    return obj.offsetHeight > 0;
}

是的。事实是,<object>失败时在规范行为中未提及:

  1. 尝试加载(大小= 0)
  2. 失败(大小=任何)
  3. Fallback(大小= innnerHtml中的值)

因此,代码需要一点增强

var interval = function() {
    if (isLoaded) { // some flag
        return;
    }

    if (hasResult(obj)) {
        if (isReallyLoaded(obj)) {
            interval.count++;
            // needs less then 400ms to fallback
            interval.count > 4 && onLoadedResult(obj, onLoaded);
        } else {
            onErrorResult(obj, onError);
        }
    }

    setTimeout(interval, 100);
};
interval.count = 0;

setTimeout(interval, 100);

好了,开始加载

document.body.appendChild(obj);

仅此。我试图详细解释代码,因此看起来并不那么愚蠢。

PS WebDev很烂

答案 4 :(得分:0)

拥有iframe中正在加载的页面中最顶层(正文)元素的ID。

在iframe的Load处理程序上,检查getElementById()是否返回非null值。 如果是,iframe已成功加载。否则就失败了。

在这种情况下,将frame.src =“about:blank”。确保在执行此操作之前删除了loadhandler。

答案 5 :(得分:0)

我没有准确回答你的问题,但是我的搜索结果将我带到了这里,所以我发帖以防万一其他人对我有类似的疑问。

它并没有完全使用加载事件,但它可以检测网站是否可访问和可调用(如果是,那么理论上应该加载iFrame)。

起初,我想像其他人一样做一个AJAX调用,除了它最初对我没用,因为我使用了jQuery。如果你执行XMLHttpRequest,它可以很好地工作:

var url = http://url_to_test.com/
var xhttp = new XMLHttpRequest();
xhttp.onreadystatechange = function() {
    if (this.readyState == 4 && this.status != 200) {
        console.log("iframe failed to load");
    }
};
xhttp.open("GET", url, true);
xhttp.send();

修改
因此,这种方法可以正常工作,除非它有很多假阴性(由于交叉原始的malarky而拾取了很多会在iframe中显示的东西)。我解决这个问题的方法是在服务器上执行CURL / Web请求,然后检查响应标头a)如果网站存在,b)标头是否设置了x-frame-options

如果您运行自己的网络服务器,这不是问题,因为您可以自己进行api调用。

我在node.js中的实现:

app.get('/iframetest',function(req,res){ //Call using /iframetest?url=url - needs to be stripped of http:// or https://
   var url = req.query.url; 
    var request = require('https').request({host: url}, function(response){ //This does an https request - require('http') if you want to do a http request
        var headers = response.headers;
        if (typeof headers["x-frame-options"] != 'undefined') {
            res.send(false); //Headers don't allow iframe
        } else {
            res.send(true); //Headers don't disallow iframe
        }
    });
    request.on('error',function(e){
       res.send(false); //website unavailable
    });
    request.end();
});