javascript创建css、js,onload触发callback兼容主流浏览器的实现
http://www.fantxi.com/blog/archives/load-css-js-callback/
由于需要写个函数,既可以加载css,又可以加载js,所以对各主流浏览器对加载js、css后是否触发onload事件做了个测试。当然,为了兼容,首先要考虑的是会用到onload和onreadystatechange,但他们并不是万能的。加载js文件onload触发callback不算大问题。css比较特殊,因为Webkeit/FF下加载css不会触发onload事件。所以研究了一晚上才找到个兼容的办法,分享如下:
首先测试元素直接写在页面
- <link onload="alert('css onload')" rel="stylesheet" href="http://localhost/css/123.css"/>
- <script onload="alert('js onload')" src="http://localhost/123.js"></script>
- <link onreadystatechange="alert('css readystatechange')" rel="stylesheet" href="http://localhost/css/123.css"/>
- <script onreadystatechange="alert('js readystatechange')" src="http://localhost/123.js"></script>
CSS onload: 支持: IE6-9/OP, 不支持: FF/Webkit(SF/CM)
JS onload: 支持: IE9/OP/FF/Webkit, 不支持: IE6-8
CSS onreadystatechange: 支持: IE6-9, 不支持: OP/FF/Webkit
JS onreadystatechange: 支持: IE6-9, 不支持: OP/FF/Webkit
测试js创建元素
- var doc = document,
- head = doc.getElementsByTagName("head")[0],
- css, js;
- css = doc.createElement('link');
- css.href = 'http://localhost/123.css?2';
- css.rel = 'stylesheet';
- head.appendChild(css);
- js = doc.createElement('script');
- js.src = 'http://localhost/123.js?2';
- head.appendChild(js);
- css.onload = function(){
- alert('css onload')
- }
- js.onload = function(){
- alert('js onload')
- }
- css.onreadystatechange = function(){
- alert('css readystatechange')
- //alert(this.readyState) //IE可以得到loading/complete, OP为undefined
- }
- js.onreadystatechange = function(){
- alert('js readystatechange')
- //alert(this.readyState) //IE可以得到loading/loaded, OP为loaded
- }
CSS/JS onload:(同元素直接写在页面是一样的)
CSS onreadystatechange: 支持: IE6-9/OP, 不支持: FF/Webkit (这里有区别,OP支持js创建的css元素,但readyState为undefined)
JS onreadystatechange: 支持: IE6-9/OP, 不支持: FF/Webkit (这里有区别,OP支持js创建的js元素,readyState为loaded)
所以为了更大的兼容,onload、onreadystatechange都要写上,代码类似下面:
- // 先把js或者css加到页面: head.appendChild(node);
- // onload为IE6-9/OP下创建CSS的时候,或IE9/OP/FF/Webkit下创建JS的时候
- // onreadystatechange为IE6-9/OP下创建CSS或JS的时候
- node.onload = node.onreadystatechange = function(){
- // !this.readyState 为不支持onreadystatechange的情况,或者OP下创建CSS的情况
- // this.readyState === "loaded" 为IE/OP下创建JS的时候
- // this.readyState === "complete" 为IE下创建CSS的时候
- if (!this.readyState || this.readyState === "loaded" || this.readyState === "complete") {
- node.onload = node.onreadystatechange = null; //防止IE内存泄漏
- alert('loaded, run callback');
- }
- }
jquery、kissy的源码,判断加载成功的核心部分差不多都这样实现的。
存在的问题:
1、当浏览器同时支持onload、onreadystatechange的情况会触发上面的函数两次,
比如:
IE9加载JS的时候,会alert两次,加载CSS的时候,alert一次,注释掉“onload、readystatechange=null”,alert两次。
OP加载JS/CSS,alert一次,把“onload、readystatechange=null”注释也会alert两次。
解决:
先在外部设定个变量isLoaded = false;
"if (!this.readyState..."上面加上个判断,如果已经加载成功就返回,比如:if (isLoaded) { return; }
"node.onload =..."上面加上 isLoaded = ture;
JQ有没有加这个我忘记了、KS应该是加了类似的判断了。
2、这个方法加载JS调用callback是兼容性没问题了,但是加载CSS再callback支持情况不同了:
IE6-9/OP可以成功alert,但是FF/Webkit不支持css的onload,解决办法:
2.1、读取cssRules的length来判断是否加载成功,缺点不能跨域读取(非IE)。
- var doc = document,
- head = document.getElementsByTagName( 'head' )[0],
- link = doc.createElement('link');
- //link.href = 'http://xxx.com/123.css'; //跨域
- link.href = 'http://localhost/123.css';
- link.rel = 'stylesheet';
- head.appendChild( link );
- var sheet, cssRules;
- if ( 'sheet' in link ) { //FF/CM/OP
- sheet = 'sheet'; cssRules = 'cssRules';
- }
- else { //IE
- sheet = 'styleSheet'; cssRules = 'rules';
- }
- var _timer1 = setInterval( function() { // 通过定时器检测css是否加载成功
- try {
- if ( link[sheet] && link[sheet][cssRules].length ) { // css被成功加载
- // console.log(link[sheet][cssRules]);
- clearInterval( _timer1 ); // 清除定时器
- clearTimeout( _timer2 );
- alert('loaded, run callback'); // 加载成功执行callback
- }
- } catch( e ) {
- // FF看到的可能的报错:
- //本地:nsresult: "0x8053000f (NS_ERROR_DOM_INVALID_ACCESS_ERR)" ,因为没加载完成还不能读取,加载完毕就不会报错了
- //跨域:Security error, code: "1000" nsresult: "0x805303e8",因为不能跨域读取CSS。。。
- //关于跨域访问:FF/OP/CM都禁止,IE6-9都可以跨域读取css。
- } finally {}
- }, 20 ),
- // 创建超时定时器,如果过10秒没检测到加载成功
- _timer2 = setTimeout( function() {
- clearInterval( _timer1 ); // 清除定时器
- clearTimeout( _timer2 );
- alert('loaded ? run callback'); // 都过了这么长时间了,虽然没判断加载成功也执行callback(这里可能本身就加载失败,也可能是跨域的情况)
- }, 10000 );
- //@See: http://stackoverflow.com/questions/5537622/dynamically-loading-css-file-using-javascript-with-callback-without-jquery
2.2、创建div#hack4loaded,并添加到页面,然后定时读取div的样式,如果样式是css里面设置的就说明加载成功,此方法不存在跨域问题,所有浏览器都OK。(css里面加上一个样式: #hack4loaded{display:none})
- var doc = document,
- head = doc.getElementsByTagName('head')[0],
- link = doc.createElement('link');
- link.href = 'http://localhost/123.css';
- link.rel = 'stylesheet';
- head.appendChild( link );
- var div = document.createElement('div');
- div.id = 'hack4loaded';
- var getStyle = function(name){ //获取css里面设置的元素样式
- if(div.currentStyle){ // IE/OP
- return div.currentStyle[name];
- }else{ // FF/CM
- return div.ownerDocument.defaultView.getComputedStyle(div, null)[name];
- }
- };
- document.body.appendChild(div); // 添加到页面
- //创建定时器,读取创建的div的样式是否是已经在css里面设置好的
- var _timer1= setInterval( function() {
- //console.log(getStyle('display'));
- if ('none' === getStyle('display')) { //样式和css里面设置的一样,说明加载成功
- clearInterval( _timer1 ); // 清除定时器
- clearTimeout( _timer2 );
- div.parentNode.removeChild(div); //移除div
- alert('loaded, run callback'); // 加载成功执行callback
- }
- }, 50 ),
- // 创建超时定时器,防止css文件加载失败的情况
- _timer2 = setTimeout( function() {
- clearInterval( _timer1 ); // 清除定时器
- clearTimeout( _timer2 );
- alert('fail to load'); // 这个没加载成功的可能性很大,再就是网速太慢超时了
- }, 10000 );
所以总结如下:
javascript创建js或者css,可以兼容浏览器在onload时触发callback。
只要把上面的“node.onload = node.onreadystatechange = function()”方法,再结合2.2的创建div#hack4loaded的方法结合再一起,就可以创建一个比较完美的支持加载js、css,并可以在文件加载完毕触发callback的函数。
由于IE/OP支持css的onload,所以写这个方法的时候可以考虑非IE/OP才用2.2的方法判断css加载完成。
javascript创建css、js,onload触发callback兼容主流浏览器的实现的更多相关文章
- 重温JavaScript获取CSS样式的方法(兼容各浏览器)
众所周知,CSS样式有三种类型:行内样式.内部样式和外部样式,JavaScript获取CSS样式时分为两种情况:行内样式获取法 和 非行内样式获取法 . 一.行内样式获取相对简单,通过element. ...
- CSS 实现图片灰度效果 兼容各种浏览器
CSS 实现图片灰度效果 兼容各种浏览器如360浏览器 CSS实现图片灰度效果就是通过CSS样式让彩色图片呈现为灰色,相当于把一张图像的颜色模式调整为灰度,CSS可以通过以下几种方法来实现灰度效果. ...
- 兼容主流浏览器的CSS透明代码
透明往往能产生不错的网页视觉效果下面是兼容主流浏览器的CSS透明代码.transparent_class { filter:alpha(opacity=50); -moz-opacity:0.5; ...
- (转)兼容主流浏览器的CSS透明代码
透明往往能产生不错的网页视觉效果下面是兼容主流浏览器的CSS透明代码.transparent_class { filter:alpha(opacity=50); -moz-opacity:0.5; - ...
- javaScript与css、html常见的兼容
最近几天总是遇到兼容问题,就整理了一下javaScript和html.css出现的常见兼容.有不全面或不对的欢迎大家指正.也希望这条博客可以帮到一些刚学习的前端的朋友. 一.javaScript出现的 ...
- jQuery+CSS 简单代码实现遮罩层( 兼容主流浏览器 )
/* ** jQuery版本:jQuery-1.8.3.min.js ** 浏览器:Chrome( v31.0.1650.63 m ),IE11,Firefox( v32.0.1 ),IETester ...
- js 复制网页内容,兼容各浏览器
因需要做一个js单击,复制当前网页url的功能.使用的是如下的方法,但是只能在ie浏览器下正常使用. 方法如下: function copyURL(){ var clipBoardContent=&q ...
- 基于原生js的返回顶部组件,兼容主流浏览器
基于原生js的返回顶部插件,兼容IE8及以上.FF.chrome等主流浏览器. js文件中封装了getScrollTop()和changeScrollTop()函数分别用于获取滚动条滚动的高度和修改滚 ...
- 兼容主流浏览器的css渐变色
网页中的渐变色区域,渐变色背景,一般都是通过ps图片方法来实现,但是图片放得多了会影响网页的打开速度,本文介绍的就是用纯 CSS 实现 IE .Firefox.Chrome 和 和Safari都支持的 ...
随机推荐
- Windows 编程中恼人的各种字符以及字符指针类型
在Windows编程中,很容易见到这些数据类型:LPSTR,LPTSTR,LPCTSTR... 像很多童鞋一样,当初在学Windows编程的时候,对着些数据类型真的是丈二和尚,摸不着头脑,长时间不用就 ...
- 洗礼灵魂,修炼python(7)--元组,集合,不可变集合
前面已经把列表的基本用法讲解完 接着讲python的几大核心之--元组(tuple) 1.什么是元组? 类似列表,但为不可变对象,之前提到列表是可变对象,所谓可变对象就是支持原处修改,并且在修改前后对 ...
- 分享基于分布式Http长连接框架--代码模型
好的代码应该是方便客户端使用,代码能够自描述,规范化,大众标准化. 而且我相信代码也是有生命的,需要不断的维护它,你以什么样的态度对待它,它就会以同样的态度回敬你,所以在写代码前,先摆好自己的态度(一 ...
- FastDFS 分布式文件系统的安装与使用
跟踪服务器:192.168.152.129 (centos1) 存储服务器:192.168.152.130 (centos2) 环境:CentOS 6.6 用户:root 数据目录:/fastdfs ...
- php Socket通信
<?php error_reporting(0); $host = "0.0.0.0"; $port = 1082; $maxUser = 10; set_time_limi ...
- ajax url参数中文乱码解决方法
较好的处理办法,对js的url中的中文参数值使用两次encodeURI(),即encodeURI(encodeURI("url的中文参数值")) JS代码: var name=&q ...
- SQL Server 锁机制 悲观锁 乐观锁 实测解析
先引入一些概念,直接Copy其他Blogs中的,我就不单独写了. 一.为什么会有锁 多个用户同时对数据库的并发操作时会带来以下数据不一致的问题: 1.丢失更新 A,B两个用户读同一数据并进行修改,其中 ...
- LeetCode-2 Keys Keyboard
package Classify.DP.Medium; import org.junit.jupiter.api.Test; /** Initially on a notepad only one c ...
- Android模拟器检测常用方法
在Android开发过程中,防作弊一直是老生常谈的问题,而模拟器的检测往往是防作弊中的重要一环,接下来有关于模拟器的检测方法,和大家进行一个简单的分享. 1.传统的检测方法. 传统的检测方法主要是对模 ...
- 怎样在Win10下安装ubuntu双系统
Win10系统下安装ubuntu系统 安装前准备: 概念 在动手之前,一定要先了解双系统.系统引导.分区这3个概念,这样才能理解安装步骤,应对安装过程中的意外情况. 双系统 双系统就是开机之后,会有一 ...