61

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

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

有什么方法可以可靠地确定是否发生了上述任何一个错误?

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

4

6 回答 6

14

如果您可以控制 iframe 页面(并且这些页面在同一个域名上),则策略可能如下:

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

我希望这有帮助。

于 2008-12-17T19:38:50.243 回答
8

我最近遇到了这个问题,不得不求助于在父页面(包含 IFRAME 标记)上设置 Javascript 轮询操作。这个 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");
            }
        }
    }
}

}

于 2008-12-22T16:29:23.707 回答
7

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

任务: 加载 iframe 跨域内容,onLoaded在成功和onError加载错误时发出。

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

1. iframe这对我来说有点震惊,iframe 只有onload事件,它在加载和错误时调用,无法知道它是否错误。

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

3.脚本我尝试使用<script>标签加载html。遗憾的是,仅在 Chrome 中正确onload发射。onerror

当我准备放弃时,我的老同事告诉我关于 html4 tag 的事<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>没有回调规范,因此每个浏览器的行为都不同。

  • 。出错和加载时会发出load事件。
  • 火狐。发射loaderror正确。
  • 野生动物园。什么都不发射……

似乎与iframe, getEntriesByType, script... 没有什么不同。但是,我们有原生浏览器后备!所以,因为我们直接设置了fallback(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. 后备(大小 = 与 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 很烂

于 2019-03-01T22:09:54.297 回答
1

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

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

于 2009-10-09T20:02:42.857 回答
0

为您的 iframe 中加载的页面中最顶部的(正文)元素设置一个 id。

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

在这种情况下,输入 frame.src="about:blank"。确保在执行此操作之前删除负载处理程序。

于 2013-08-02T18:55:12.250 回答
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();

编辑:
所以这个方法工作正常,除了它有很多误报(拿起很多会在 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();
});
于 2017-08-22T15:20:37.227 回答