码迷,mamicode.com
首页 > Web开发 > 详细

js插件动态加载js、css解决方案

时间:2015-09-07 11:06:58      阅读:184      评论:0      收藏:0      [点我收藏+]

标签:

  最近因为工作需要做了一个js自动导入的插件,一开始很天真的以为动态创建个script添加到head中就ok了,试了之后才发现了问题,就是如果同时引入了多个js文件,而且后一个文件中用到了前一个文件中的变量,那就会报错,靠~~悲催了,就是说js如果动态加载(非浏览器加载因为浏览器加载时同步加载的会等待前一个js加载完成后才进行下一个js加载,这样就不会出现问题)那加载的文件是异步进行的,难怪啊!然后在网上找了些资料说用ajax同步加载,然后我试了真可以,下面就是我的代码分享出来给大家,但是注意这样加载出来的js有一个致命的弱点,就是没法再浏览器上调试- - !!!!!以后再寻找解决方案吧,先贴代码:

js主文件importComJs.js:

**
 * Created by carlos on 2015/5/19.
 * QImport 导入帮助函数,可以很方便的导入指定的js、css文件。但是需要注意的是,这样导入的js将无法调试,暂时没有找到解决方案--!
 * 使用方法先把此文件引入,必须放在head中的最前面导入,
 * <script src="../js/importComJs.js" id="QImport"></script>
 * id必须是QImport,否则无效
 * 然后加入
 * <script>
 * 	 	QImport.init(customscripts);
 * </script>
 * customscripts参数为:外加的引入文件配置,可以不填。但填写必须遵守QImport.IMPORTSCRIPTS的写法。
 * 如果不想手动初始化引入函数,可以这样写:
 * <script src="../js/importComJs.js" id="QImport" data-auto="true"></script>
 * 这里的 data-auto为true时候才会自动初始化,否则需要手动初始化。
 * 使用上自动初始化,不用担心引入顺序问题,后面的其他js、css文件按正常引用就可以了。
 * 注意:importComjs文件必须第一个引入,否则将出现引用找不到的问题。
 * 新增属性:
 * 		data-config:自定义配置文件名,默认为“importConfig.json”
 * 		data-personalconfigname:自定义个性化配置名称,在配置文件中的“personalscripts”中配置
 * 		注意:配置加载顺序是:importscripts最高、personalscripts其次、customscripts最低。如果引入的次序不对会引起空指针报错哦^^!
 */

/**
 * 同步加载js脚本
 * @param id   需要设置的<script>标签的id
 * @param url   js文件的相对路径或绝对路径
 * @return {Boolean}   返回是否加载成功,true代表成功,false代表失败
 */
(function(){
	QImport = {
		self:{
			obj:function(){
				return document.getElementById(‘QImport‘);      //获取script的id,如果项目中实在是有冲突不得不改,那就该这个“QImport”吧!
			},
			auto:function(){
				return eval(this.getAttr(‘data-auto‘));   		//是否开启自动初始化模式,默认为false
			},
			config:function(){
				return this.getAttr(‘data-config‘);       		//自定义配置文件名,默认为“importConfig.json”
			},
			personal:function(){
				return this.getAttr(‘data-personalconfigname‘)  //自定义个性化配置名称,在配置文件中的“personalscripts”中配置。
			},
			src:function(){
				return this.getAttr(‘src‘);                     //此文件位置,不多说
			},
			getAttr:function(attrName){
				var attrValue = undefined;
				try{
					attrValue = this.obj().getAttribute(attrName);
				}
				catch(e){}
				return attrValue;
			},
			relationpath:function(){
				var src = this.src();
				var re = /^[..\/]*/;
				return src.match(re)[0];
			},
			path:function(filename){
				var src = this.src();
				var temp = src.split(‘/‘);
				var path = "";
				temp.pop();
				temp.push(filename);
				return temp.join(‘/‘);
			}
		},
		autoinit:function(){
			if(this.self.auto()){
				this.init(); 
			}
		},
		getpath:function(p){
			var path = this.self.relationpath()+p;
			return path.replace(‘//‘,‘/‘); 
		},
		init:function(customscripts){						//导入文件的主方法
			this.extend(customscripts);
			var IMPORTSCRIPTS = this.IMPORTSCRIPTS;
			var head = document.getElementsByTagName("head").item(0);
			var meta = document.createElement(‘meta‘);  //自动添加web手机适应代码;
			meta.name = ‘viewport‘;
			meta.content = ‘width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no‘;
			head.insertBefore(meta, head.childNodes[0]);
			var re = /^http[s]?:\/\//i;
			for(var i=0;i<IMPORTSCRIPTS.length;i++){  
				loadpath = IMPORTSCRIPTS[i].url; 
				if(!re.test(IMPORTSCRIPTS[i].url)){
					loadpath = this.getpath(IMPORTSCRIPTS[i].url);
				}
				if(IMPORTSCRIPTS[i].type=="script"){
					var flag = this.loadJS(loadpath,loadpath,i+1)
					if(!flag){
						alert(‘loading path:"‘+loadpath+‘" failed,May be lost "http(s)://"‘);
					}	
				}
				else if(IMPORTSCRIPTS[i].type=="css"){
					var csss = document.createElement(‘link‘)
					csss.href = this.getpath(IMPORTSCRIPTS[i].url);
					csss.rel = ‘stylesheet‘;
					head.insertBefore(csss, head.childNodes[i+1]); 
				} 
			}
		},
		extend:function(customscripts){                              //自定义扩展方法,仅适用当前对象
			if(!(customscripts instanceof Array)) return;
			var IMPORTSCRIPTS = this.IMPORTSCRIPTS;
			for(var i=0;i<customscripts.length;i++){
				var flag = false;
				for(var j=0;j<IMPORTSCRIPTS.length;j++){
					if((IMPORTSCRIPTS[j].type==customscripts[i].type)&&(IMPORTSCRIPTS[j].url==customscripts[i].url)){
						flag = true;
						break;
					}
				}
				if(!flag){
					this.IMPORTSCRIPTS.push({type:customscripts[i].type,url:customscripts[i].url});
				}
			}
		},
		ajax:function(url){                                     //ajax原始方法,这里仅可以同步请求
			var  xmlHttp = null;
		    if(window.ActiveXObject)//IE
		    {
		        try {
		            //IE6以及以后版本中可以使用
		            xmlHttp = new ActiveXObject("Msxml2.XMLHTTP");
		        }
		        catch (e) {
		            //IE5.5以及以后版本可以使用
		            xmlHttp = new ActiveXObject("Microsoft.XMLHTTP");
		        }
		    }
		    else if(window.XMLHttpRequest)//Firefox,Opera 8.0+,Safari,Chrome
		    {
		        xmlHttp = new XMLHttpRequest();
		    }
		    //采用同步加载
		    xmlHttp.open("GET",url,false);
		    //发送同步请求,如果浏览器为Chrome或Opera,必须发布后才能运行,不然会报错
		    xmlHttp.send(null);
		    //4代表数据发送完毕
		    if ( xmlHttp.readyState == 4 ){
		        //0为访问的本地,200到300代表访问服务器成功,304代表没做修改访问的是缓存
		        if((xmlHttp.status >= 200 && xmlHttp.status <300) || xmlHttp.status == 0 || xmlHttp.status == 304){
		            return xmlHttp.responseText;
		        }
		        else{
		            return false;
		        }
		    }
		    else{
		        return false;
		    }
		},
		loadJS:function(id,url,rank){								//加载js文件
		    var importText = this.ajax(url);
		    if(!importText) return false;
		    var myHead = document.getElementsByTagName("HEAD").item(0);
		    var myScript = document.createElement( "script" );
		    myScript.language = "javascript";
		    myScript.type = "text/javascript";
		    myScript.id = id;
		    try{
		        //IE8以及以下不支持这种方式,需要通过text属性来设置 
		        myScript.appendChild(document.createTextNode(importText));
		    }
		    catch (ex){
		        myScript.text = importText;
		    }
		    myHead.insertBefore(myScript, myHead.childNodes[rank]);
		    return true;
		},
		template:{								//加载模板文件,现在仅在引入“jquery”的情况下可用。			
			parent:this,						//父类this引用				
			_setparent:function(){				//把父类的this赋给子方法的parent
				this.template.parent = this;
			},
			includeTagName:‘include‘,           //引入时候用的标签:<include url="head.html"></include>
			init:function(){
				this.include();
			},
			include:function(){
				var includes = document.getElementsByTagName(this.includeTagName);
				for(var i=0;i<includes.length;i++){
					var url = includes[i].getAttribute(‘url‘);
					var includeHtml = this.parent.ajax(url);
					var parent = includes[i].parentNode;
					var newEs = this.parseDom(includeHtml);
					for(var j=0;j<newEs.length;j++){
						parent.insertBefore(newEs[j], includes[i])
					}
				}
				while(true){
					if(includes.length==0) break;
					includes[0].remove();
				}
			},
			parseDom:function(str){
				var obj = document.createElement(‘div‘);
				obj.innerHTML = str;
				return obj.childNodes;
			}
		}, 
		initparent:function(){              	//子类中想用父类中的方法必须通过此方法初始化以下,还必须引入如template的parent和_setparent
			this.template._setparent.call(this);
		},
		getimportjson:function(){				//获取的配置json
			var configName = this.self.config();            //共用配置
			if(configName) this.configname = configName;
			var configpath = this.self.path(this.configname); 
			var confJsonStr = this.ajax(configpath);
			if(!confJsonStr){
				alert(‘config file path:"‘+configpath+‘" can not be loaded‘);
				return;
			}
			this.confJson = eval(‘(‘+confJsonStr+‘)‘);
			importarr = this.confJson.importscripts;
			if(importarr instanceof Array){
				this.IMPORTSCRIPTS = this.confJson.importscripts;
			}
			this.getpersonalarr();
		},
		getpersonalarr:function(){    			//获取个性化配置arr
			var personalConfigName = this.self.personal();
			if(!personalConfigName) return;
			this.personalconfigname = personalConfigName;
			var personalarr = this.confJson.personalscripts[personalConfigName];
			if(personalarr instanceof Array){
				this.extend(personalarr)
			}
		},
		confJson:{},
		IMPORTSCRIPTS:[],
		configname:‘importConfig.json‘
	}
	QImport.getimportjson();
	QImport.autoinit();
	QImport.initparent();
	try{
		$(function(){
			QImport.template.init();
		});
	}
	catch(e){}
	
})();

下面是配置文件,importConfig.json:

{
	importscripts:[
		{
			type: ‘script‘,
			url: ‘js/jquery-1.9.1/jquery-1.9.1.min.js‘
		},{
			type: ‘script‘,
			url: ‘js/mui.min.js‘
		},{
			type: ‘css‘,
			url: ‘css/mui.min.css‘
		},{
			type: ‘script‘,
			url: ‘js/json2.js‘
		}
	],
	personalscripts:{
		test:[
			{
				type:‘script‘,
				url:‘js/test.js‘
			}
		]
	}
}

注意配置文件必须和导入方法js在同一个路径底下。

以上就是我个人封装的引入js、css的代码,希望和大家交流下,有什么问题希望大家能不吝指教!

js插件动态加载js、css解决方案

标签:

原文地址:http://www.cnblogs.com/zhaojia-dream/p/4788225.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!