Browse Source

检查检验详情模板提交

linehang 7 years ago
parent
commit
a50bb4a86e

+ 52 - 0
css/jiuzhenxiangqing.css

@ -0,0 +1,52 @@
.c-74bf00 { color: #74BF00; }
.c-fwb { font-weight: bold; }
.bc-fff { background: #FFF; }
.info-panel p { height: 36px; line-height: 36px; border-bottom: 1px solid #DCDCDC; }
.info-panel .title-icon { width: 20px; height: 20px; margin-bottom: 4px; }
.info-panel {  }
.img-list li img { width: 70px; height: 70px; }
#edit_btn img { position:fixed; top:20px; right: 20px; width: 50px; height: 50px; }
.select-image-bar{ background: transparent; border-bottom: 1px solid #909090;}
.title { font-size: 16px; }
.pswp { z-index: 9999; }
.c-arrow-r {
    position: absolute;
    top: 50%;
    right: 10px;
    margin-top: -5px;
    content: "";
    display: block;
    width: 8px;
    height: 8px;
    border: solid #cdcdcd;
    border-width: 2px 2px 0 0;
    -webkit-transform: rotate(45deg);
    -moz-transform: rotate(45deg);
    -ms-transform: rotate(45deg);
    transform: rotate(45deg);
}
.c-arrow-d {
	position: absolute;
	top: 50%;
	right: 10px;
	margin-top: -5px;
	content: "";
	display: block;
	width: 8px;
	height: 8px;
	border: solid #74BF00;
	border-width: 2px 2px 0 0;
	-webkit-transform: rotate(135deg);
	-moz-transform: rotate(135deg);
	-ms-transform: rotate(135deg);
	transform: rotate(135deg);
}
.del-icon { width: 16px; height: 16px; }
.c-btn-74bf00 {
	background-color: transparent;
	color: #74BF00;
	border: #74BF00 1px solid;
}
.c-btn-icon { width: 100%; padding: 8px 16px; border-radius: 5px; }
.c-btn-icon.active { background: #74BF00; color: #FFF; }

BIN
img/bingli_icon.png


BIN
img/binglibaogao_icon.png


BIN
img/chufang_icon.png


BIN
img/jianchabaogao_icon.png


BIN
img/jianyanbaogao_icon.png


BIN
img/jiuzhenxinxi_icon.png


BIN
img/qita_icon.png


BIN
img/weifenlei_icon.png


BIN
img/zhenduanjieguo_icon.png


+ 65 - 0
jianchajianyan-detail.html

@ -0,0 +1,65 @@
<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta name="renderer" content="webkit">
    <meta http-equiv="Cache-Control" content="no-siteapp" />
    <title>检查检验详情</title>
    <meta name="keywords" content="">
    <meta name="description" content="">
    <link rel="shortcut icon" href="favicon.ico">
    <link href="css/bootstrap.min.css?v=3.3.6" rel="stylesheet">
    <link href="css/font-awesome.min93e3.css?v=4.4.0" rel="stylesheet">
    <link href="css/animate.min.css" rel="stylesheet">
    <link href="css/plugins/bootstrap-table/bootstrap-table.min.css" rel="stylesheet">
    <link href="css/plugins/toastr/toastr.min.css" rel="stylesheet">
    <link href="js/plugins/artDialog/6.0.5/css/ui-dialog.min.css" rel="stylesheet">
    <link href="css/style.css" rel="stylesheet">
    <style>
    	body {color: #333333;font-size: 14px;padding-top: 20px;}
    	h1 {text-align: center; margin-top: 0;}
        h3 {border-bottom: 1px solid #D7DCE6; line-height: 36px;}
        /*.fc-999 {color: #999999;}*/
        ul, p {margin: 0; padding: 0;}
        li {list-style: none;}
        /*.h-180 {height: 180px;}
        .h-108 {height: 108px;}
        .w-100 {width: 100%}*/
        .page-content { padding: 5px; }
        td {text-align: center;}
    </style>
</head>
<body class="white-bg plr10">
	<div class="h45">
		<div class="demo-comtop">
			<a class="mui-action-back"></a>
			<h1><span id="selected_profile_name"></span></h1>
		</div>
	</div>
	<div class="page-content"  ms-controller="viewController">
		
	</div>
    
    <script src="js/jquery.min.js?v=2.1.4"></script>
    <script src="js/bootstrap.min.js?v=3.3.6"></script>
    <script src="js/plugins/metisMenu/jquery.metisMenu.js"></script>
    <script src="js/plugins/layer/layer.min.js"></script>
    <script src="js/plugins/slimscroll/jquery.slimscroll.min.js"></script>
    <script src="js/hplus.min.js?v=4.1.0"></script>
    <script src="js/plugins/pace/pace.min.js"></script>
    <script src="js/plugins/toastr/toastr.min.js"></script>
    <script src="js/plugins/artDialog/6.0.5/js/dialog-plus.min.js"></script>
    <script src="js/plugins/bootstrap-table/bootstrap-table.min.js"></script>
    <script src="js/plugins/bootstrap-table/bootstrap-table-mobile.min.js"></script>
    <script src="js/plugins/bootstrap-table/locale/bootstrap-table-zh-CN.min.js"></script>
    <script src="js/es6-promise.js"></script>
    <script src="js/underscore.js"></script>
    <script src="js/util.js"></script>
    <script src="js/template.js"></script>
    <script src="js/plugins/avalon/avalon.modern.js"></script>
    <script src="js/api-service.js"></script>
    <script src="js/jianchajianyan-api.js"></script>
    <script src="js/buz/jianchajianyan-detail.js"></script>
</body>
</html>

+ 192 - 0
jiuzhenxiangqing.html

@ -0,0 +1,192 @@
<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta name="renderer" content="webkit">
    <meta http-equiv="Cache-Control" content="no-siteapp" />
    <title>就诊详情</title>
    <meta name="keywords" content="">
    <meta name="description" content="">
    <link rel="shortcut icon" href="favicon.ico">
    <link href="css/bootstrap.min.css?v=3.3.6" rel="stylesheet">
    <link href="css/font-awesome.min93e3.css?v=4.4.0" rel="stylesheet">
    <link href="css/animate.min.css" rel="stylesheet">
    <link href="css/plugins/bootstrap-table/bootstrap-table.min.css" rel="stylesheet">
    <link href="css/plugins/toastr/toastr.min.css" rel="stylesheet">
    <link href="js/plugins/artDialog/6.0.5/css/ui-dialog.min.css" rel="stylesheet">
    <link href="css/style.css" rel="stylesheet">
    <link rel="stylesheet" type="text/css" href="css/jiuzhenxiangqing.css">
    <style>
    	ul {margin: 0; padding: 0;}
        li {float: left; width: 50%;list-style: none;}
    </style>
</head>
<body class="plr10">
    <div class="h45">
			<div class="demo-comtop">
				<a class="mui-action-back"></a>
				<h1 style="margin-bottom: 15px;text-align: center; font-size: 24px;">就诊详情</h1>
			</div>
		</div>
		<div class="mt10 info-panel bc-fff">
			<p class="title pl15">
				<img src="img/jiuzhenxinxi_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >就诊信息</span>
			</p>
			<ul class="pl15 pt10 pb10 c-f14 clearfix">
				<li><label>就诊时间:</label><span id="time"></span></li>
				<li><label>就诊类型:</label><span id="type"></span></li>
				<li><label>科室/医生:</label><span id="dept"></span><span id="doctor"></span></li>
				<li><label>医院:</label><span id="org"></span></li>
			</ul>
		</div>
		<div class="mt10 info-panel bc-fff">
			<p class="title pl15">
				<img src="img/zhenduanjieguo_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >诊断结果</span>
			</p>
			<div id="diagnosis" class="pl15 pt10 pb10 c-f14">
				
			</div>
		</div>
		<div data-type="病历" class="mt10 info-panel bc-fff clearfix pb10 c-hide">
			<p class="title pl15">
				<img src="img/bingli_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >病历</span>
			</p>
			<ul id="bingli_images" class="pl15 pb10 c-f14 img-list">
				
			</ul>
		</div>
		<div data-type="处方" class="mt10 info-panel bc-fff clearfix pb10 c-hide">
			<p class="title pl15">
				<img src="img/chufang_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >处方</span>
			</p>
			<ul id="chufang_images" class="pl15 pb10 c-f14 img-list">
				
			</ul>
		</div>
		<div data-type="检查报告"  class="mt10 info-panel bc-fff clearfix pb10 c-hide">
			<p class="title pl15">
				<img src="img/jianchabaogao_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >检查报告</span>
			</p>
			<ul id="jiancha_images" class="pl15 pb10 c-f14 img-list">
				
			</ul>
		</div>
		<div data-type="检验报告" class="mt10 info-panel bc-fff clearfix pb10 c-hide">
			<p class="title pl15">
				<img src="img/jianchabaogao_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >检验报告</span>
			</p>
			<ul id="jianyan_images"class="pl15 pb10 c-f14 img-list">
				
			</ul>
		</div>
		<div data-type="病理报告" class="mt10 info-panel bc-fff clearfix pb10 c-hide">
			<p class="title pl15">
				<img src="img/binglibaogao_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >病理报告</span>
			</p>
			<ul id="bingli_images" class="pl15 pb10 c-f14 img-list">
				
			</ul>
		</div>
		<div data-type="其他" class="mt10 info-panel bc-fff clearfix pb10 c-hide">
			<p class="title pl15">
				<img src="img/qita_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >其他</span>
			</p>
			<ul id="qita_images" class="pl15 pb10 c-f14 img-list">
				
			</ul>
		</div>
		<div data-type="未分类" class="mt10 info-panel bc-fff clearfix pb10 c-hide">
			<p class="title pl15">
				<img src="img/weifenlei_icon.png" class="title-icon"/>
				<span class="c-fwb pl10" >未分类</span>
			</p>
			<ul id="weifenlei_images" class="pl15 pb10 c-f14 img-list">
				
			</ul>
		</div>
		
		<!-- Root element of PhotoSwipe. Must have class pswp. -->
		<div class="pswp" tabindex="-1" role="dialog" aria-hidden="true">
		    <!-- Background of PhotoSwipe. 
		         It's a separate element as animating opacity is faster than rgba(). -->
		    <div class="pswp__bg"></div>
		
		    <!-- Slides wrapper with overflow:hidden. -->
		    <div class="pswp__scroll-wrap">
		
		        <!-- Container that holds slides. 
		            PhotoSwipe keeps only 3 of them in the DOM to save memory.
		            Don't modify these 3 pswp__item elements, data is added later on. -->
		        <div class="pswp__container">
		            <div class="pswp__item"></div>
		            <div class="pswp__item"></div>
		            <div class="pswp__item"></div>
		        </div>
		
		        <!-- Default (PhotoSwipeUI_Default) interface on top of sliding area. Can be changed. -->
		        <div class="pswp__ui pswp__ui--hidden">
		
		            <div class="pswp__top-bar" style="display: none;">
		                <!--  Controls are self-explanatory. Order can be changed. -->
		                <div class="pswp__counter"></div>
		                <button class="pswp__button pswp__button--close" title="Close (Esc)"></button>
		                <!-- Preloader demo http://codepen.io/dimsemenov/pen/yyBWoR -->
		                <!-- element will get class pswp__preloader--active when preloader is running -->
		                <div class="pswp__preloader">
		                    <div class="pswp__preloader__icn">
		                      <div class="pswp__preloader__cut">
		                        <div class="pswp__preloader__donut"></div>
		                      </div>
		                    </div>
		                </div>
		            </div>
		            <div class="pswp__caption">
		                <div class="pswp__caption__center"></div>
		            </div>
		
		        </div>
		
		    </div>
		    
		</div>
		<script type="text/html" id="li_tmpl">
			{{each list as it}}
			<li class="fl mr10 mt10">
				<img src="{{it.imgUrl | setPhoto}}" class="preview-img" data-catalog="{{it.imgType||'未分类'}}" data-sub-catalog="{{it.imgLabel}}"/>
				{{if it.imgLabel }}
				<div class="c-t-center c-909090 c-f12 mt5" >{{it.imgLabel}}</div>
				{{/if}}
			</li>
        	{{/each}}
		</script>
    
    <script src="js/jquery.min.js?v=2.1.4"></script>
    <script src="js/bootstrap.min.js?v=3.3.6"></script>
    <script src="js/plugins/metisMenu/jquery.metisMenu.js"></script>
    <script src="js/plugins/layer/layer.min.js"></script>
    <script src="js/plugins/slimscroll/jquery.slimscroll.min.js"></script>
    <script src="js/hplus.min.js?v=4.1.0"></script>
    <script src="js/plugins/pace/pace.min.js"></script>
    <script src="js/plugins/toastr/toastr.min.js"></script>
    <script src="js/plugins/artDialog/6.0.5/js/dialog-plus.min.js"></script>
    <script src="js/plugins/bootstrap-table/bootstrap-table.min.js"></script>
    <script src="js/plugins/bootstrap-table/bootstrap-table-mobile.min.js"></script>
    <script src="js/plugins/bootstrap-table/locale/bootstrap-table-zh-CN.min.js"></script>
    <script src="js/es6-promise.js"></script>
    <script src="js/underscore.js"></script>
    <script src="js/util.js"></script>
    <script src="js/template.js"></script>
    <script src="js/api-service.js"></script>
    <script src="js/jianchajianyan-api.js"></script>
    <script src="js/buz/jiuzhenxiangqing.js"></script>
</body>
</html>

+ 259 - 0
js/buz/jianchajianyan-detail.js

@ -0,0 +1,259 @@
var request = getRequest(),
    patient = request.patient,
    rowInfo = JSON.parse(window.localStorage.getItem('checkRow'));
var Profile = {
	url: '',
	link: '',
	thatBaseInfo: '',
	thatPlus: null,
	thatMui: null,
	thatPromise: null,
	thatTempalteCode: '',
	thatSerial: '',
	thatPageContent: null,
	index: 0,
	firstCode: '',
	profileRes: [],
	dc: [],
	htmlStr:['<div id="no_result_wrap">',
				'<img class="no-result-img" src="../../../images/shujuweikong_img.png" />',
				'<div class="f-fs14 c-323232 c-t-center" style="text-align:center">抱歉,暂未找到符合条件的结果。</div>',
			'</div>'].join(''),
	init: function (ops) {
		var me = this;
		if (!(!!ops && typeof ops === 'object')) {
			alert('参数有误!');
			return;
		}
		ops.cb && ops.cb.call(me);
		me.url = ops.url;
		me.link = ops.link;
		me.thatMui = ops.thatMui;
		me.thatPlus = ops.thatPlus;
		me.thatPromise = ops.thatPromise;
		me.thatBaseInfo = ops.thatBaseInfo;
		me.thatTempalteCode = ops.thatTempalteCode;
		me.thatSerial = ops.thatSerial;
		me.thatPageContent = ops.thatPageContent;
		me.index = 0;
		me.firstCode = '';
		me.profileRes = [];
		me.dc = [];
		window['dataCode']=[];
	},
	//获取模板
	templateRequest: function () {
		var me = this;
		return new me.thatPromise(function(resolve, reject) {
			me.thatMui.ajax( me.link, 
			{ dataType: 'html',
			  type:'GET', 
			  error: function(xht, type, throwErr) {
			  	me.thatMui.toast("档案模板获取失败。");
			  	me.thatPlus.nativeUI.closeWaiting();
			  },
			  success: function(html) {
			  	resolve(html);
			  }
			})
		}).catch(function(e) {
			console.error(e)
		})
	},
	//根据thatTempalteCode获取数据
	profileDataRequest: function () {
		return new Promise(function(resolve, reject) {
			var healthData = {
	      		event: rowInfo.id,
	      		patient: patient,
	      		catalog: rowInfo.catalogCode,
	      		serial: rowInfo.serial
	        }
			return janchaAPI.getHealthData({data: healthData}).then(function(res) {
				resolve(res)
			})
		}).catch(function(e) {
			console.error(e)
		})
	},
	promiseAll: function () {
		var me = this,
			fistDc = me.thatTempalteCode;
		me.thatPromise.all([ me.templateRequest(), me.profileDataRequest()]).then(function (datas) {
			var html = datas[0];
			me.profileRes.push(datas[1]);
			me.thatPageContent.html(html);
			me.firstCode = me.thatTempalteCode;
			me.dc = window['dataCode'];
			if (me.dc.length > 0) {
				me.reqDataMore(me);
			} else {
				me.changeData(fistDc);
			}
		}).catch(function(e) {
			console.error(e)
		})
	},
	//获取多个数据
	reqDataMore: function (me) {
		me.thatTempalteCode = me.dc[me.index];
		Promise.all([ me.profileDataRequest()]).then(function (datas) {
			me.profileRes.push(datas[0]);
			me.index++;
			if (me.dc.length > me.index) {
				me.reqDataMore(me);
			} else {
				me.changeData(me.firstCode);
			}
		});
	},
	
	changeData: function (fistDc) {
		var me = this,
			jsonData = [];
		me.dc.unshift(me.firstCode);
		$.each( me.profileRes, function (index) {
			jsonData.push(me.selectXmlDom(me.profileRes[index],me.dc[index]));
		});
		console.log(jsonData); 
		if (jsonData.length > 0) {
			if (jsonData.length === 1 && !!!jsonData[0]) {
				me.thatPageContent.html(me.htmlStr);
			} else {
				me.renderProfileTemplate($.extend(true,{},me.mosaicData(jsonData)));
			}
		} else {
			me.thatPageContent.html(me.htmlStr);
		}
//		plus.nativeUI.closeWaiting();
	},
	//渲染模板
	renderProfileTemplate: function (data) {
		this.thatPageContent.removeAttr('avalonctrl').attr('ms-controller',"viewController");
		var vm = avalon.define($.extend({},{"$id": "viewController"},data));
		avalon.scan();
	},
	//获取xml数据
	selectXmlDom: function (profileRes,tempalteCode) {
		var me = this,
			$dom = $(profileRes.data.replace(/<\?xml .*\?>/,'')),
			version = $dom.find(">version").length?$dom.find(">version").attr("code"):$dom.attr('version');
			$.trim(version) || (function () {
				version = $dom.find(">ClinicalDocument").length? '2.0.0.1' : '';
			})();
		if($.trim(version)) {
			var verNum = version.split('.')[0];
			var xmlToJson = window["dsXmlToJson"+tempalteCode+'_'+verNum],
				jsonData = {};
				 
			if(xmlToJson && $.isFunction(xmlToJson)) { 
				if(!profileRes.data || profileRes.data=="[]") {
					if(version=2) {
						profileRes.data = '<?xml version="1.0" encoding="utf-8"?><ClinicalDocument></ClinicalDocument>';
					} else {
						profileRes.data = '<?xml version="1.0" encoding="utf-8"?><root></root>';
					}
				}
				// xml 转 json
				if((typeof profileRes.data=='string')&&profileRes.data.constructor==String) {
					// 去除携带的脚本内容
					profileRes.data = profileRes.data.replace(/<script[^>]*?>[\s\S]*?<\/script>/ig,'');
					jsonData = xmlToJson(profileRes.data.replace(/<\?xml .*\?>/,''));
				}
			}
			return jsonData;
		} 
	},
	//数据拼接
	mosaicData: function (data) {
		for (var i = 1; i < data.length; i++) {
			data[i] && (function () {
				for (key in data[i]['data']) {
					data[0]['data'][key] = data[i]['data'][key];
				}
			})();
		}
		return data[0];
	}
};
var dataCode = [];
// 基本信息(包括userAgent、上个页面传递的数据)
var baseInfo = null,
	// 基础环境信息(包括当前webview)
	baseEnv = null;
var $profileItemList = $('.lin-sel-group'),
	// 搜索无结果时显示
	$noResultWrap = $('#no_result_wrap'),
	$selectedProfileName = $('#selected_profile_name'),
	$pageContent = $('.page-content');
// 分页查询当前页数
var curPage = 1,
	// iscroll 滚动条实例
	iscroller,
	catalogs = [];
var getCatalog = function() {
	var catalog = "";
	if(rowInfo && rowInfo.catalogCode) {
		catalog = rowInfo.catalogCode;
	}
	return catalog;
},
getCatalogName = function() {
	var name = "";
	if(rowInfo && rowInfo.label) {
		name = rowInfo.label;
	}else if(rowInfo && rowInfo.type){
		name = rowInfo.type;
	}
	return name;
},
// 初始化就诊事件列表
initProfileList = function() {
	$selectedProfileName.text(getCatalogName());
	var tempalteCode = getCatalog();
	if(tempalteCode) {
		getProfileTemplateHtml(tempalteCode);
	}
},
getProfileTemplateHtml = function(tempalteCode,serial){
	Profile.init({
		url: "http://www.xmtyw.cn/wlyy/doctor/archives/event/healthData",
		link: 'template/'+tempalteCode+'.html',
		thatMui: jQuery,
		thatPromise: Promise,
		thatBaseInfo: baseInfo,
		thatTempalteCode: tempalteCode,
		thatSerial: serial,
		thatPageContent: $pageContent,
	});
	Profile.promiseAll();
};
// 获取异步请求处理Promise对象
function getReqPromise(url, data,reqType,newWay) {
	return new Promise(function(resolve, reject) {
		sendPost(url, data,
		  	function queryFailed (xht, type, throwErr) {
				// dialog({contentType:'tipsbox', skin:'bk-popup' , content:'加载失败'}).show();
				// mui.toast("程序开了点小差, 请稍后重试!");
				if(type == "error" || type == "parsererror" || type == "null") {
					//mui.toast("程序开了点小差, 请稍后重试!");
					//throw new Error("加载失败");
					showErrorPage("程序开了点小差, 请稍后重试!");
				}
				return true;
		  	}
	  		, function success(res) {
				resolve(res); 
	  		},reqType, '', (newWay===false?false:true));
	});
}
$(function(){
    initProfileList();
});

+ 9 - 8
js/buz/jianchajianyan.js

@ -38,9 +38,14 @@ function getList(isRefresh){
                return '<span class="pr10">'+val+'</span>';
            }
        }],
        // onClickRow: function(row, $el){
        //     document.location.href = "jw-prescription-detail.html?id="+row.id+"&patient="+patient;
        // }
        onClickRow: function(row, $el){
        	window.localStorage.setItem('checkRow', JSON.stringify(row));
        	if(row.dataFrom == 2) {// json格式
            	document.location.href = "jiuzhenxiangqing.html?patient="+patient;
          	} else {// xml格式:来源基卫
          		document.location.href = "jianchajianyan-detail.html?patient="+patient;
          	}
       	}
    })
    $("#prescriptTable").bootstrapTable(options);
}
@ -51,8 +56,4 @@ function queryParams(params) {
        pageSize: params.limit,
        patient: patient
    };
}
}

+ 206 - 0
js/buz/jiuzhenxiangqing.js

@ -0,0 +1,206 @@
// 基本信息(包括userAgent、上个页面传递的数据)
var request = getRequest(),
    patient = request.patient,
    rowInfo = JSON.parse(window.localStorage.getItem('checkRow')),
    userAgent = JSON.parse(window.localStorage.getItem('userAgent'));
	
var $time = $('#time'),
	$type = $('#type'),
	$deptDoctor = $('#dept_doctor'),
	$org = $('#org'),
	$diagnosis = $('#diagnosis');
var gallery = null;
// 获取基本信息(包括userAgent、上个页面传递的数据)
var getBaseInfo = function() {
	return {
		userAgent: userAgent,
		eventInfo: rowInfo,
		patiInfo: rowInfo
	}
},
// 获取基础环境信息
getBaseEnvPromise = function () {
	
	var env = {
		webview: plus&&plus.webview.currentWebview()
	};
	return Promise.resolve().then(function(res) {
		return env;
	});
},
getDetailPromise = function() {
	var data={
		event: rowInfo.id
	}
	janchaAPI.getEventDetail({data: data}).then(function(res) {
		var data = {},
			imgsMap = {};
		if(res.status==200) {
			data = res.data;
			var eventName = getEventTypeName(data.eventType);
			$type.text(eventName);
			showDeptDoctor(data.deptName,data.doctorName);
			$time.text(data.eventDate.substr(0,10));
			if(data.orgName) {
				$org.text(data.orgName);
				$org.parent().show();
			}
			
			$diagnosis.text(data.dianosis);
			imgsMap = _.groupBy(data.eventImg, function(v) {
				return v.imgType;
			});
			_.mapObject(imgsMap,function(val, key) {
				var catalog = key? key: "未分类";
				var $wrap = $('div[data-type="'+catalog+'"]'),
					$ul = $wrap.find('.img-list').eq(0);
					html = template('li_tmpl',{list: val});
				$ul.html(html);
				$wrap.show();
			});
		}
	})
	.then(function() {
		var $imgs = $('img.preview-img');
		$imgs.on('click',function() {
			var idx = $imgs.index(this);
			var length = $imgs.length;
			updateCatalogNameTip(this);
			$('#page_counter').text('('+(idx+1)+'/'+length+')')
			previewImage(idx);
			gallery.listen('afterChange', function() {
				var imgIdx = gallery.getCurrentIndex();
				var $img = $imgs.eq(imgIdx);
				updateCatalogNameTip($img);
				$('#page_counter').text('('+(imgIdx+1)+'/'+length+')')
				updateCatalog($img);
			});
			updateCatalog(this);
		});
	})
};
function getEventTypeName(type) {
	if(type=="1") {
		return "门诊";
	} else if(type=="2"){
		return "住院";
	} else if(type == "3"){
		return "体检";
	}
}
function showDeptDoctor(dept,doctor) {
	if(!dept&&!doctor) {
		return ;
	} else if(dept&& doctor) {
		$deptDoctor.html('<label>科室/医生:</label>'+dept+'/'+doctor);
	} else if(dept) {
		$deptDoctor.html('<label>科室:</label>'+dept);
	} else if(doctor) {
		$deptDoctor.html('<label>医生:</label>'+doctor);
	}
	$deptDoctor.show();
}
function updateCatalogNameTip(img) {
	var catalog = $(img).attr('data-catalog')||"";
	var subCatalog = $(img).attr('data-sub-catalog')||"";
	var catalogName = catalog;
	if(subCatalog) {
		catalogName = catalogName + '_' + subCatalog;
	}
	$('#catalog_name').text(catalogName);
}
function updateCatalog(img) {
	var catalog = $(img).attr('data-catalog')||"";
	var subCatalog = $(img).attr('data-sub-catalog')||"";
	$('.img-catalog').find('a.c-btn-icon[data-type="'+catalog+'"]').trigger('click',[false]);
	$('.sub-catalog').find('a.c-btn-icon[data-type="'+subCatalog+'"]').trigger('click',[false]);
}
/*
 * 获取图片路径
 */
function getImgUrl(str){
	if(typeof str != 'string'){
	    return "";
	}
	if(str.length == 0){
        return "";
    }else{
        if(str.indexOf("../")>-1){
            //访问本地路径
            return str;
        }else if((str.indexOf("http://")>-1) || (str.indexOf("https://")>-1)){
            return str;
        }else{
            //服务器上的图片路径
            return imgUrlDomain + str;
        }
    }
}
function previewImage(sid) {
	var pswpElement = document.querySelectorAll('.pswp')[0];
	var w = $(window).width(), h = $(window).height();
	var items = _.map($('img.preview-img'),function(img) {
		return {
			src: $(img).attr('src'),
			w: w,
			h: h,
			initialPosition: { x: 0, y:0 }
		}
	});
	var options = {
	    // optionName: 'option value'
	    // for example:
	    index: sid||0 // start at first slide
	};
	options.mainClass = 'pswp--minimal--dark';
    options.barsSize = {top:0,bottom:0};
	options.captionEl = false;
	options.fullscreenEl = false;
	options.shareEl = false;
	options.bgOpacity = 0.85;
	options.tapToToggleControls = false;
	options.pinchToClose = false;
	options.clickToCloseNonZoomable = false;
	options.tapToClose = true;
	options.errorMsg = '<div class="pswp__error-msg">对不起,该图片加载失败</div>';
	// Initializes and opens PhotoSwipe
	gallery = new PhotoSwipe( pswpElement, PhotoSwipeUI_Default, items, options);
	gallery.init();
}
// 页面业务处理流程开始
//new Promise(function(resolve, reject) {
//
////	mui.plusReady(function() {
////		// plus已经准备好,可以往下执行
////		resolve(true);
////	});
//}).then(function() {
//	
//	// 获取基础环境信息
//	return getBaseEnvPromise().then(function(env) {
//		baseEnv = env;
//		// 获取登录医生信息
//		baseInfo = getBaseInfo();
//	})
//})
//.then(function() {
//	return getDetailPromise();
//})
//.catch(function(e) { console && console.error(e) });
getDetailPromise();
template.helper("setPhoto", function(p) {		
	return getImgUrl(p);
});

+ 9 - 0
js/jianchajianyan-api.js

@ -4,6 +4,15 @@
        getList: function(data){
            return APIService.httpGet('doctor/archives/event/report', data);
        },
        getEventDetail: function(data) { //就诊事件详情
        	return APIService.httpGet('doctor/archives/event/detail', data);
        },
        getCatalog: function(data) { //通过事件号获取档案类型列表(基卫)
        	return APIService.httpGet('doctor/archives/event/catalog', data);
        },
        getHealthData: function(data) { //获取健康档案详情(基卫)
        	return APIService.httpGet('doctor/archives/event/healthData', data);
        },
        getListTableAjaxObj: { //表格数据获取时,使用的ajax请求配置
            method: 'get',
            url: APIService.server + 'doctor/archives/event/report', 

+ 5788 - 0
js/plugins/avalon/avalon.modern.js

@ -0,0 +1,5788 @@
/*==================================================
 Copyright (c) 2013-2015 司徒正美 and other contributors
 http://www.cnblogs.com/rubylouvre/
 https://github.com/RubyLouvre
 http://weibo.com/jslouvre/
 
 Released under the MIT license
 avalon.modern.js 1.5.5 built in 2015.11.20
 support IE10+ and other browsers
 ==================================================*/
(function(global, factory) {
    if (typeof module === "object" && typeof module.exports === "object") {
        // For CommonJS and CommonJS-like environments where a proper `window`
        // is present, execute the factory and get avalon.
        // For environments that do not have a `window` with a `document`
        // (such as Node.js), expose a factory as module.exports.
        // This accentuates the need for the creation of a real `window`.
        // e.g. var avalon = require("avalon")(window);
        module.exports = global.document ? factory(global, true) : function(w) {
            if (!w.document) {
                throw new Error("Avalon requires a window with a document")
            }
            return factory(w)
        }
    } else {
        factory(global)
    }
// Pass this if window is not defined yet
}(typeof window !== "undefined" ? window : this, function(window, noGlobal){
/*********************************************************************
 *                    全局变量及方法                                  *
 **********************************************************************/
var expose = Date.now()
//http://stackoverflow.com/questions/7290086/javascript-use-strict-and-nicks-find-global-function
var DOC = window.document
var head = DOC.head //HEAD元素
head.insertAdjacentHTML("afterBegin", '<avalon ms-skip class="avalonHide"><style id="avalonStyle">.avalonHide{ display: none!important }</style></avalon>')
var ifGroup = head.firstChild
function log() {
    if (avalon.config.debug) {
// http://stackoverflow.com/questions/8785624/how-to-safely-wrap-console-log
        console.log.apply(console, arguments)
    }
}
/**
 * Creates a new object without a prototype. This object is useful for lookup without having to
 * guard against prototypically inherited properties via hasOwnProperty.
 *
 * Related micro-benchmarks:
 * - http://jsperf.com/object-create2
 * - http://jsperf.com/proto-map-lookup/2
 * - http://jsperf.com/for-in-vs-object-keys2
 */
function createMap() {
  return Object.create(null)
}
var subscribers = "$" + expose
var nullObject = {} //作用类似于noop,只用于代码防御,千万不要在它上面添加属性
var rword = /[^, ]+/g //切割字符串为一个个小块,以空格或豆号分开它们,结合replace实现字符串的forEach
var rw20g = /\w+/g
var rsvg = /^\[object SVG\w*Element\]$/
var rwindow = /^\[object (?:Window|DOMWindow|global)\]$/
var oproto = Object.prototype
var ohasOwn = oproto.hasOwnProperty
var serialize = oproto.toString
var ap = Array.prototype
var aslice = ap.slice
var W3C = window.dispatchEvent
var root = DOC.documentElement
var avalonFragment = DOC.createDocumentFragment()
var cinerator = DOC.createElement("div")
var class2type = {}
"Boolean Number String Function Array Date RegExp Object Error".replace(rword, function (name) {
    class2type["[object " + name + "]"] = name.toLowerCase()
})
function scpCompile(array){
    return Function.apply(noop,array)
}
function noop(){}
function oneObject(array, val) {
    if (typeof array === "string") {
        array = array.match(rword) || []
    }
    var result = {},
            value = val !== void 0 ? val : 1
    for (var i = 0, n = array.length; i < n; i++) {
        result[array[i]] = value
    }
    return result
}
//生成UUID http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript
var generateID = function (prefix) {
    prefix = prefix || "avalon"
    return String(Math.random() + Math.random()).replace(/\d\.\d{4}/, prefix)
}
function IE() {
    if (window.VBArray) {
        var mode = document.documentMode
        return mode ? mode : window.XMLHttpRequest ? 7 : 6
    } else {
        return NaN
    }
}
var IEVersion = IE()
avalon = function (el) { //创建jQuery式的无new 实例化结构
    return new avalon.init(el)
}
/*视浏览器情况采用最快的异步回调*/
avalon.nextTick = new function () {// jshint ignore:line
    var tickImmediate = window.setImmediate
    var tickObserver = window.MutationObserver
    if (tickImmediate) {
        return tickImmediate.bind(window)
    }
    var queue = []
    function callback() {
        var n = queue.length
        for (var i = 0; i < n; i++) {
            queue[i]()
        }
        queue = queue.slice(n)
    }
    if (tickObserver) {
        var node = document.createTextNode("avalon")
        new tickObserver(callback).observe(node, {characterData: true})// jshint ignore:line
        var bool = false
        return function (fn) {
            queue.push(fn)
            bool = !bool
            node.data = bool
        }
    }
    return function (fn) {
        setTimeout(fn, 4)
    }
}// jshint ignore:line
/*********************************************************************
 *                 avalon的静态方法定义区                              *
 **********************************************************************/
avalon.init = function (el) {
    this[0] = this.element = el
}
avalon.fn = avalon.prototype = avalon.init.prototype
avalon.type = function (obj) { //取得目标的类型
    if (obj == null) {
        return String(obj)
    }
    // 早期的webkit内核浏览器实现了已废弃的ecma262v4标准,可以将正则字面量当作函数使用,因此typeof在判定正则时会返回function
    return typeof obj === "object" || typeof obj === "function" ?
            class2type[serialize.call(obj)] || "object" :
            typeof obj
}
var isFunction = function (fn) {
    return serialize.call(fn) === "[object Function]"
}
avalon.isFunction = isFunction
avalon.isWindow = function (obj) {
    return rwindow.test(serialize.call(obj))
}
/*判定是否是一个朴素的javascript对象(Object),不是DOM对象,不是BOM对象,不是自定义类的实例*/
avalon.isPlainObject = function (obj) {
    // 简单的 typeof obj === "object"检测,会致使用isPlainObject(window)在opera下通不过
    return serialize.call(obj) === "[object Object]" && Object.getPrototypeOf(obj) === oproto
}
//与jQuery.extend方法,可用于浅拷贝,深拷贝
avalon.mix = avalon.fn.mix = function () {
    var options, name, src, copy, copyIsArray, clone,
            target = arguments[0] || {},
            i = 1,
            length = arguments.length,
            deep = false
    // 如果第一个参数为布尔,判定是否深拷贝
    if (typeof target === "boolean") {
        deep = target
        target = arguments[1] || {}
        i++
    }
    //确保接受方为一个复杂的数据类型
    if (typeof target !== "object" && !isFunction(target)) {
        target = {}
    }
    //如果只有一个参数,那么新成员添加于mix所在的对象上
    if (i === length) {
        target = this
        i--
    }
    for (; i < length; i++) {
        //只处理非空参数
        if ((options = arguments[i]) != null) {
            for (name in options) {
                src = target[name]
                copy = options[name]
                // 防止环引用
                if (target === copy) {
                    continue
                }
                if (deep && copy && (avalon.isPlainObject(copy) || (copyIsArray = Array.isArray(copy)))) {
                    if (copyIsArray) {
                        copyIsArray = false
                        clone = src && Array.isArray(src) ? src : []
                    } else {
                        clone = src && avalon.isPlainObject(src) ? src : {}
                    }
                    target[name] = avalon.mix(deep, clone, copy)
                } else if (copy !== void 0) {
                    target[name] = copy
                }
            }
        }
    }
    return target
}
function _number(a, len) { //用于模拟slice, splice的效果
    a = Math.floor(a) || 0
    return a < 0 ? Math.max(len + a, 0) : Math.min(a, len);
}
avalon.mix({
    rword: rword,
    subscribers: subscribers,
    version: 1.55,
    ui: {},
    log: log,
    slice: function (nodes, start, end) {
        return aslice.call(nodes, start, end)
    },
    noop: noop,
    /*如果不用Error对象封装一下,str在控制台下可能会乱码*/
    error: function (str, e) {
        throw new (e || Error)(str)// jshint ignore:line
    },
    /*将一个以空格或逗号隔开的字符串或数组,转换成一个键值都为1的对象*/
    oneObject: oneObject,
    /* avalon.range(10)
     => [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
     avalon.range(1, 11)
     => [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
     avalon.range(0, 30, 5)
     => [0, 5, 10, 15, 20, 25]
     avalon.range(0, -10, -1)
     => [0, -1, -2, -3, -4, -5, -6, -7, -8, -9]
     avalon.range(0)
     => []*/
    range: function (start, end, step) { // 用于生成整数数组
        step || (step = 1)
        if (end == null) {
            end = start || 0
            start = 0
        }
        var index = -1,
                length = Math.max(0, Math.ceil((end - start) / step)),
                result = new Array(length)
        while (++index < length) {
            result[index] = start
            start += step
        }
        return result
    },
    eventHooks: {},
    /*绑定事件*/
    bind: function (el, type, fn, phase) {
        var hooks = avalon.eventHooks
        var hook = hooks[type]
        if (typeof hook === "object") {
            type = hook.type || type
            phase = hook.phase || !!phase
            fn = hook.fn ? hook.fn(el, fn) : fn
        }
        el.addEventListener(type, fn, phase)
        return fn
    },
    /*卸载事件*/
    unbind: function (el, type, fn, phase) {
        var hooks = avalon.eventHooks
        var hook = hooks[type]
        var callback = fn || noop
        if (typeof hook === "object") {
            type = hook.type || type
            phase = hook.phase || !!phase
        }
        el.removeEventListener(type, callback, phase)
    },
    /*读写删除元素节点的样式*/
    css: function (node, name, value) {
        if (node instanceof avalon) {
            node = node[0]
        }
        var prop = /[_-]/.test(name) ? camelize(name) : name, fn
        name = avalon.cssName(prop) || prop
        if (value === void 0 || typeof value === "boolean") { //获取样式
            fn = cssHooks[prop + ":get"] || cssHooks["@:get"]
            if (name === "background") {
                name = "backgroundColor"
            }
            var val = fn(node, name)
            return value === true ? parseFloat(val) || 0 : val
        } else if (value === "") { //请除样式
            node.style[name] = ""
        } else { //设置样式
            if (value == null || value !== value) {
                return
            }
            if (isFinite(value) && !avalon.cssNumber[prop]) {
                value += "px"
            }
            fn = cssHooks[prop + ":set"] || cssHooks["@:set"]
            fn(node, name, value)
        }
    },
    /*遍历数组与对象,回调的第一个参数为索引或键名,第二个或元素或键值*/
    each: function (obj, fn) {
        if (obj) { //排除null, undefined
            var i = 0
            if (isArrayLike(obj)) {
                for (var n = obj.length; i < n; i++) {
                    if (fn(i, obj[i]) === false)
                        break
                }
            } else {
                for (i in obj) {
                    if (obj.hasOwnProperty(i) && fn(i, obj[i]) === false) {
                        break
                    }
                }
            }
        }
    },
    //收集元素的data-{{prefix}}-*属性,并转换为对象
    getWidgetData: function (elem, prefix) {
        var raw = avalon(elem).data()
        var result = {}
        for (var i in raw) {
            if (i.indexOf(prefix) === 0) {
                result[i.replace(prefix, "").replace(/\w/, function (a) {
                    return a.toLowerCase()
                })] = raw[i]
            }
        }
        return result
    },
    Array: {
        /*只有当前数组不存在此元素时只添加它*/
        ensure: function (target, item) {
            if (target.indexOf(item) === -1) {
                return target.push(item)
            }
        },
        /*移除数组中指定位置的元素,返回布尔表示成功与否*/
        removeAt: function (target, index) {
            return !!target.splice(index, 1).length
        },
        /*移除数组中第一个匹配传参的那个元素,返回布尔表示成功与否*/
        remove: function (target, item) {
            var index = target.indexOf(item)
            if (~index)
                return avalon.Array.removeAt(target, index)
            return false
        }
    }
})
var bindingHandlers = avalon.bindingHandlers = {}
var bindingExecutors = avalon.bindingExecutors = {}
var directives = avalon.directives = {}
avalon.directive = function (name, obj) {
    bindingHandlers[name] = obj.init = (obj.init || noop)
    bindingExecutors[name] = obj.update = (obj.update || noop)
    return directives[name] = obj
}
/*判定是否类数组,如节点集合,纯数组,arguments与拥有非负整数的length属性的纯JS对象*/
function isArrayLike(obj) {
    if (obj && typeof obj === "object") {
        var n = obj.length,
                str = serialize.call(obj)
        if (/(Array|List|Collection|Map|Arguments)\]$/.test(str)) {
            return true
        } else if (str === "[object Object]" && n === (n >>> 0)) {
            return true //由于ecma262v5能修改对象属性的enumerable,因此不能用propertyIsEnumerable来判定了
        }
    }
    return false
}
// https://github.com/rsms/js-lru
var Cache = new function() {// jshint ignore:line
    function LRU(maxLength) {
        this.size = 0
        this.limit = maxLength
        this.head = this.tail = void 0
        this._keymap = {}
    }
    var p = LRU.prototype
    p.put = function(key, value) {
        var entry = {
            key: key,
            value: value
        }
        this._keymap[key] = entry
        if (this.tail) {
            this.tail.newer = entry
            entry.older = this.tail
        } else {
            this.head = entry
        }
        this.tail = entry
        if (this.size === this.limit) {
            this.shift()
        } else {
            this.size++
        }
        return value
    }
    p.shift = function() {
        var entry = this.head
        if (entry) {
            this.head = this.head.newer
            this.head.older =
                    entry.newer =
                    entry.older =
                    this._keymap[entry.key] = void 0
             delete this._keymap[entry.key] //#1029
        }
    }
    p.get = function(key) {
        var entry = this._keymap[key]
        if (entry === void 0)
            return
        if (entry === this.tail) {
            return  entry.value
        }
        // HEAD--------------TAIL
        //   <.older   .newer>
        //  <--- add direction --
        //   A  B  C  <D>  E
        if (entry.newer) {
            if (entry === this.head) {
                this.head = entry.newer
            }
            entry.newer.older = entry.older // C <-- E.
        }
        if (entry.older) {
            entry.older.newer = entry.newer // C. --> E
        }
        entry.newer = void 0 // D --x
        entry.older = this.tail // D. --> E
        if (this.tail) {
            this.tail.newer = entry // E. <-- D
        }
        this.tail = entry
        return entry.value
    }
    return LRU
}// jshint ignore:line
/*********************************************************************
 *                           DOM 底层补丁                             *
 **********************************************************************/
//safari5+是把contains方法放在Element.prototype上而不是Node.prototype
if (!DOC.contains) {
    Node.prototype.contains = function (arg) {
        return !!(this.compareDocumentPosition(arg) & 16)
    }
}
avalon.contains = function (root, el) {
    try {
        while ((el = el.parentNode))
            if (el === root)
                return true
        return false
    } catch (e) {
        return false
    }
}
if (window.SVGElement) {
    var svgns = "http://www.w3.org/2000/svg"
    var svg = DOC.createElementNS(svgns, "svg")
    svg.innerHTML = '<circle cx="50" cy="50" r="40" fill="red" />'
    if (!rsvg.test(svg.firstChild)) {// #409
        /* jshint ignore:start */
        function enumerateNode(node, targetNode) {
            if (node && node.childNodes) {
                var nodes = node.childNodes
                for (var i = 0, el; el = nodes[i++]; ) {
                    if (el.tagName) {
                        var svg = DOC.createElementNS(svgns,
                                el.tagName.toLowerCase())
                        // copy attrs
                        ap.forEach.call(el.attributes, function (attr) {
                            svg.setAttribute(attr.name, attr.value)
                        })
                        // 递归处理子节点
                        enumerateNode(el, svg)
                        targetNode.appendChild(svg)
                    }
                }
            }
        }
        /* jshint ignore:end */
        Object.defineProperties(SVGElement.prototype, {
            "outerHTML": {//IE9-11,firefox不支持SVG元素的innerHTML,outerHTML属性
                enumerable: true,
                configurable: true,
                get: function () {
                    return new XMLSerializer().serializeToString(this)
                },
                set: function (html) {
                    var tagName = this.tagName.toLowerCase(),
                            par = this.parentNode,
                            frag = avalon.parseHTML(html)
                    // 操作的svg,直接插入
                    if (tagName === "svg") {
                        par.insertBefore(frag, this)
                        // svg节点的子节点类似
                    } else {
                        var newFrag = DOC.createDocumentFragment()
                        enumerateNode(frag, newFrag)
                        par.insertBefore(newFrag, this)
                    }
                    par.removeChild(this)
                }
            },
            "innerHTML": {
                enumerable: true,
                configurable: true,
                get: function () {
                    var s = this.outerHTML
                    var ropen = new RegExp("<" + this.nodeName + '\\b(?:(["\'])[^"]*?(\\1)|[^>])*>', "i")
                    var rclose = new RegExp("<\/" + this.nodeName + ">$", "i")
                    return  s.replace(ropen, "").replace(rclose, "")
                },
                set: function (html) {
                    if (avalon.clearHTML) {
                        avalon.clearHTML(this)
                        var frag = avalon.parseHTML(html)
                        enumerateNode(frag, this)
                    }
                }
            }
        })
    }
}
//========================= event binding ====================
var eventHooks = avalon.eventHooks
//针对firefox, chrome修正mouseenter, mouseleave(chrome30+)
if (!("onmouseenter" in root)) {
    avalon.each({
        mouseenter: "mouseover",
        mouseleave: "mouseout"
    }, function (origType, fixType) {
        eventHooks[origType] = {
            type: fixType,
            fn: function (elem, fn) {
                return function (e) {
                    var t = e.relatedTarget
                    if (!t || (t !== elem && !(elem.compareDocumentPosition(t) & 16))) {
                        delete e.type
                        e.type = origType
                        return fn.call(elem, e)
                    }
                }
            }
        }
    })
}
//针对IE9+, w3c修正animationend
avalon.each({
    AnimationEvent: "animationend",
    WebKitAnimationEvent: "webkitAnimationEnd"
}, function (construct, fixType) {
    if (window[construct] && !eventHooks.animationend) {
        eventHooks.animationend = {
            type: fixType
        }
    }
})
if (DOC.onmousewheel === void 0) {
    /* IE6-11 chrome mousewheel wheelDetla 下 -120 上 120
     firefox DOMMouseScroll detail 下3 上-3
     firefox wheel detlaY 下3 上-3
     IE9-11 wheel deltaY 下40 上-40
     chrome wheel deltaY 下100 上-100 */
    eventHooks.mousewheel = {
        type: "wheel",
        fn: function (elem, fn) {
            return function (e) {
                e.wheelDeltaY = e.wheelDelta = e.deltaY > 0 ? -120 : 120
                e.wheelDeltaX = 0
                Object.defineProperty(e, "type", {
                    value: "mousewheel"
                })
                fn.call(elem, e)
            }
        }
    }
}
/*********************************************************************
 *                           配置系统                                 *
 **********************************************************************/
function kernel(settings) {
    for (var p in settings) {
        if (!ohasOwn.call(settings, p))
            continue
        var val = settings[p]
        if (typeof kernel.plugins[p] === "function") {
            kernel.plugins[p](val)
        } else if (typeof kernel[p] === "object") {
            avalon.mix(kernel[p], val)
        } else {
            kernel[p] = val
        }
    }
    return this
}
var openTag, closeTag, rexpr, rexprg, rbind, rregexp = /[-.*+?^${}()|[\]\/\\]/g
function escapeRegExp(target) {
    //http://stevenlevithan.com/regex/xregexp/
    //将字符串安全格式化为正则表达式的源码
    return (target + "").replace(rregexp, "\\$&")
}
var plugins = {
    interpolate: function (array) {
        openTag = array[0]
        closeTag = array[1]
        if (openTag === closeTag) {
            throw new SyntaxError("openTag!==closeTag")
            var test = openTag + "test" + closeTag
            cinerator.innerHTML = test
            if (cinerator.innerHTML !== test && cinerator.innerHTML.indexOf("&lt;") > -1) {
                throw new SyntaxError("此定界符不合法")
            }
            cinerator.innerHTML = ""
        }
         kernel.openTag = openTag
            kernel.closeTag = closeTag
        var o = escapeRegExp(openTag),
                c = escapeRegExp(closeTag)
        rexpr = new RegExp(o + "(.*?)" + c)
        rexprg = new RegExp(o + "(.*?)" + c, "g")
        rbind = new RegExp(o + ".*?" + c + "|\\sms-")
    }
}
kernel.async =true
kernel.debug = true
kernel.plugins = plugins
kernel.plugins['interpolate'](["{{", "}}"])
kernel.paths = {}
kernel.shim = {}
kernel.maxRepeatSize = 100
avalon.config = kernel
function $watch(expr, binding) {
    var $events = this.$events || (this.$events = {})
    var queue = $events[expr] || ($events[expr] = [])
    if (typeof binding === "function") {
        var backup = binding
        backup.uniqueNumber = Math.random()
        binding = {
            element: root,
            type: "user-watcher",
            handler: noop,
            vmodels: [this],
            expr: expr,
            uniqueNumber: backup.uniqueNumber
        }
        binding.wildcard = /\*/.test(expr)
    }
    if (!binding.update) {
        if (/\w\.*\B/.test(expr)) {
            binding.getter = noop
            var host = this
            binding.update = function () {
                var args = this.fireArgs || []
                if (args[2])
                    binding.handler.apply(host, args)
                delete this.fireArgs
            }
            queue.sync = true
            avalon.Array.ensure(queue, binding)
        } else {
            avalon.injectBinding(binding)
        }
        if (backup) {
            binding.handler = backup
        }
    } else if (!binding.oneTime) {
        avalon.Array.ensure(queue, binding)
    }
    return function () {
        binding.update = binding.getter = binding.handler = noop
        binding.element = DOC.createElement("a")
    }
}
function $emit(key, args) {
    var event = this.$events
    if (event && event[key]) {
        if (args) {
            args[2] = key
        }
        var arr = event[key]
        notifySubscribers(arr, args)
        var parent = this.$up
        if (parent) {
            if (this.$pathname) {
                $emit.call(parent, this.$pathname + "." + key, args)//以确切的值往上冒泡
            }
            $emit.call(parent, "*." + key, args)//以模糊的值往上冒泡
        }
    } else {
        parent = this.$up
       
        if(this.$ups ){
            for(var i in this.$ups){
                $emit.call(this.$ups[i], i+"."+key, args)//以确切的值往上冒泡
            }
            return
        }
        if (parent) {
            var p = this.$pathname
            if (p === "")
                p = "*"
            var path = p + "." + key
            arr = path.split(".")
            if (arr.indexOf("*") === -1) {
                $emit.call(parent, path, args)//以确切的值往上冒泡
                arr[1] = "*"
                $emit.call(parent, arr.join("."), args)//以模糊的值往上冒泡
            } else {
                $emit.call(parent, path, args)//以确切的值往上冒泡
            }
        }
    }
}
function collectDependency(el, key) {
    do {
        if (el.$watch) {
            var e = el.$events || (el.$events = {})
            var array = e[key] || (e[key] = [])
            dependencyDetection.collectDependency(array)
            return
        }
        el = el.$up
        if (el) {
            key = el.$pathname + "." + key
        } else {
            break
        }
    } while (true)
}
function notifySubscribers(subs, args) {
    if (!subs)
        return
    if (new Date() - beginTime > 444 && typeof subs[0] === "object") {
        rejectDisposeQueue()
    }
    var users = [], renders = []
    for (var i = 0, sub; sub = subs[i++]; ) {
        if (sub.type === "user-watcher") {
            users.push(sub)
        } else {
            renders.push(sub)
        }
    }
    if (kernel.async) {
        buffer.render()//1
        for (i = 0; sub = renders[i++]; ) {
            if (sub.update) {
                var uuid = getUid(sub)
                if (!buffer.queue[uuid]) {
                    buffer.queue[uuid] = 1
                    buffer.queue.push(sub)
                }
            }
        }
    } else {
        for (i = 0; sub = renders[i++]; ) {
            if (sub.update) {
                sub.update()//最小化刷新DOM树
            }
        }
    }
    for (i = 0; sub = users[i++]; ) {
        if (args && args[2] === sub.expr || sub.wildcard) {
            sub.fireArgs = args
        }
        sub.update()
    }
}
//avalon最核心的方法的两个方法之一(另一个是avalon.scan),返回一个ViewModel(VM)
var VMODELS = avalon.vmodels = {} //所有vmodel都储存在这里
avalon.define = function (source) {
    var $id = source.$id
    if (!$id) {
        log("warning: vm必须指定$id")
    }
    var vmodel = modelFactory(source)
    vmodel.$id = $id
    return VMODELS[$id] = vmodel
}
//一些不需要被监听的属性
var $$skipArray = oneObject("$id,$watch,$fire,$events,$model,$skipArray,$active,$pathname,$up,$ups,$track,$accessors")
//如果浏览器不支持ecma262v5的Object.defineProperties或者存在BUG,比如IE8
//标准浏览器使用__defineGetter__, __defineSetter__实现
function modelFactory(source, options) {
    options = options || {}
    options.watch = true
    return observeObject(source, options)
}
//监听对象属性值的变化(注意,数组元素不是数组的属性),通过对劫持当前对象的访问器实现
//监听对象或数组的结构变化, 对对象的键值对进行增删重排, 或对数组的进行增删重排,都属于这范畴
//   通过比较前后代理VM顺序实现
function Component() {
}
function observeObject(source, options) {
    if (!source || (source.$id && source.$accessors) || (source.nodeName && source.nodeType > 0)) {
        return source
    }
    //source为原对象,不能是元素节点或null
    //options,可选,配置对象,里面有old, force, watch这三个属性
    options = options || nullObject
    var force = options.force || nullObject
    var old = options.old
    var oldAccessors = old && old.$accessors || nullObject
    var $vmodel = new Component() //要返回的对象, 它在IE6-8下可能被偷龙转凤
    var accessors = {} //监控属性
    var hasOwn = {}
    var skip = []
    var simple = []
    var $skipArray = {}
    if (source.$skipArray) {
        $skipArray = oneObject(source.$skipArray)
        delete source.$skipArray
    }
    //处理计算属性
    var computed = source.$computed
    if (computed) {
        delete source.$computed
        for (var name in computed) {
            hasOwn[name] = true;
            (function (key, value) {
                var old
                accessors[key] = {
                    get: function () {
                        return old = value.get.call(this)
                    },
                    set: function (x) {
                        if (typeof value.set === "function") {
                            var older = old
                            value.set.call(this, x)
                            var newer = this[key]
                            if (this.$fire && (newer !== older)) {
                                this.$fire(key, newer, older)
                            }
                        }
                    },
                    enumerable: true,
                    configurable: true
                }
            })(name, computed[name])// jshint ignore:line
        }
    }
    for (name in source) {
        var value = source[name]
        if (!$$skipArray[name])
            hasOwn[name] = true
        if (typeof value === "function" || (value && value.nodeName && value.nodeType > 0) ||
                (!force[name] && (name.charAt(0) === "$" || $$skipArray[name] || $skipArray[name]))) {
            skip.push(name)
        } else if (isComputed(value)) {
            log("warning:计算属性建议放在$computed对象中统一定义");
            (function (key, value) {
                var old
                accessors[key] = {
                    get: function () {
                        return old = value.get.call(this)
                    },
                    set: function (x) {
                        if (typeof value.set === "function") {
                            var older = old
                            value.set.call(this, x)
                            var newer = this[key]
                            if (this.$fire && (newer !== older)) {
                                this.$fire(key, newer, older)
                            }
                        }
                    },
                    enumerable: true,
                    configurable: true
                }
            })(name, value)// jshint ignore:line
        } else {
            simple.push(name)
            if (oldAccessors[name]) {
                accessors[name] = oldAccessors[name]
            } else {
                accessors[name] = makeGetSet(name, value)
            }
        }
    }
    accessors["$model"] = $modelDescriptor
    $vmodel = Object.defineProperties($vmodel, accessors, source)
    function trackBy(name) {
        return hasOwn[name] === true
    }
    skip.forEach(function (name) {
        $vmodel[name] = source[name]
    })
    /* jshint ignore:start */
    hideProperty($vmodel, "$ups", null)
    hideProperty($vmodel, "$id", "anonymous")
    hideProperty($vmodel, "$up", old ? old.$up : null)
    hideProperty($vmodel, "$track", Object.keys(hasOwn))
    hideProperty($vmodel, "$active", false)
    hideProperty($vmodel, "$pathname", old ? old.$pathname : "")
    hideProperty($vmodel, "$accessors", accessors)
    hideProperty($vmodel, "hasOwnProperty", trackBy)
    if (options.watch) {
        hideProperty($vmodel, "$watch", function () {
            return $watch.apply($vmodel, arguments)
        })
        hideProperty($vmodel, "$fire", function (path, a) {
            if (path.indexOf("all!") === 0) {
                var ee = path.slice(4)
                for (var i in avalon.vmodels) {
                    var v = avalon.vmodels[i]
                    v.$fire && v.$fire.apply(v, [ee, a])
                }
            } else {
                $emit.call($vmodel, path, [a])
            }
        })
    }
    /* jshint ignore:end */
    //必须设置了$active,$events
    simple.forEach(function (name) {
        var oldVal = old && old[name]
        var val = $vmodel[name] = source[name]
        if (val && typeof val === "object") {
            val.$up = $vmodel
            val.$pathname = name
        }
        $emit.call($vmodel, name,[val,oldVal])
    })
    for (name in computed) {
        value = $vmodel[name]
    }
    $vmodel.$active = true
    return $vmodel
}
/*
 新的VM拥有如下私有属性
 $id: vm.id
 $events: 放置$watch回调与绑定对象
 $watch: 增强版$watch
 $fire: 触发$watch回调
 $track:一个数组,里面包含用户定义的所有键名
 $active:boolean,false时防止依赖收集
 $model:返回一个纯净的JS对象
 $accessors:放置所有读写器的数据描述对象
 $up:返回其上级对象
 $pathname:返回此对象在上级对象的名字,注意,数组元素的$pathname为空字符串
 =============================
 $skipArray:用于指定不可监听的属性,但VM生成是没有此属性的
 */
function isComputed(val) {//speed up!
    if (val && typeof val === "object") {
        for (var i in val) {
            if (i !== "get" && i !== "set") {
                return false
            }
        }
        return  typeof val.get === "function"
    }
}
function makeGetSet(key, value) {
    var childVm, value = NaN
    return {
        get: function () {
            if (this.$active) {
                collectDependency(this, key)
            }
            return value
        },
        set: function (newVal) {
            if (value === newVal)
                return
            var oldValue = value
            childVm = observe(newVal, value)
            if (childVm) {
                value = childVm
            } else {
                childVm = void 0
                value = newVal
            }
            if (Object(childVm) === childVm) {
                childVm.$pathname = key
                childVm.$up = this
            }
            if (this.$active) {
                $emit.call(this, key, [value, oldValue])
            }
        },
        enumerable: true,
        configurable: true
    }
}
function observe(obj, old, hasReturn, watch) {
    if (Array.isArray(obj)) {
        return observeArray(obj, old, watch)
    } else if (avalon.isPlainObject(obj)) {
        if (old && typeof old === 'object') {
            var keys = Object.keys(obj)
            var keys2 = Object.keys(old)
            if (keys.join(";") === keys2.join(";")) {
                for (var i in obj) {
                    if (obj.hasOwnProperty(i)) {
                        old[i] = obj[i]
                    }
                }
                return old
            }
            old.$active = false
        }
        return observeObject(obj, {
            old: old,
            watch: watch
        })
    }
    if (hasReturn) {
        return obj
    }
}
function observeArray(array, old, watch) {
    if (old) {
        var args = [0, old.length].concat(array)
        old.splice.apply(old, args)
        return old
    } else {
        for (var i in newProto) {
            array[i] = newProto[i]
        }
        hideProperty(array, "$up", null)
        hideProperty(array, "$pathname", "")
        hideProperty(array, "$track", createTrack(array.length))
        array._ = observeObject({
            length: NaN
        }, {
            watch: true
        })
        array._.length = array.length
        array._.$watch("length", function (a, b) {
            $emit.call(array.$up, array.$pathname + ".length", [a, b])
        })
        if (watch) {
            hideProperty(array, "$watch", function () {
                return $watch.apply(array, arguments)
            })
        }
        Object.defineProperty(array, "$model", $modelDescriptor)
        for (var j = 0, n = array.length; j < n; j++) {
            var el = array[j] = observe(array[j], 0, 1, 1)
            if (Object(el) === el) {//#1077
                el.$up = array
            }
        }
        return array
    }
}
function hideProperty(host, name, value) {
    Object.defineProperty(host, name, {
        value: value,
        writable: true,
        enumerable: false,
        configurable: true
    })
}
function toJson(val) {
    var xtype = avalon.type(val)
    if (xtype === "array") {
        var array = []
        for (var i = 0; i < val.length; i++) {
            array[i] = toJson(val[i])
        }
        return array
    } else if (xtype === "object") {
        var obj = {}
        for (i in val) {
            if (val.hasOwnProperty(i)) {
                var value = val[i]
                obj[i] = value && value.nodeType ? value : toJson(value)
            }
        }
        return obj
    }
    return val
}
var $modelDescriptor = {
    get: function () {
        return toJson(this)
    },
    set: noop,
    enumerable: false,
    configurable: true
}
/*********************************************************************
 *          监控数组(与ms-each, ms-repeat配合使用)                     *
 **********************************************************************/
var arrayMethods = ['push', 'pop', 'shift', 'unshift', 'splice']
var arrayProto = Array.prototype
var newProto = {
    notify: function () {
        $emit.call(this.$up, this.$pathname)
    },
    set: function (index, val) {
        if (((index >>> 0) === index) && this[index] !== val) {
            if (index > this.length) {
                throw Error(index + "set方法的第一个参数不能大于原数组长度")
            }
            $emit.call(this.$up, this.$pathname + ".*", [val, this[index]])
            this.splice(index, 1, val)
        }
    },
    contains: function (el) { //判定是否包含
        return this.indexOf(el) !== -1
    },
    ensure: function (el) {
        if (!this.contains(el)) { //只有不存在才push
            this.push(el)
        }
        return this
    },
    pushArray: function (arr) {
        return this.push.apply(this, arr)
    },
    remove: function (el) { //移除第一个等于给定值的元素
        return this.removeAt(this.indexOf(el))
    },
    removeAt: function (index) { //移除指定索引上的元素
        if ((index >>> 0) === index) {
            return this.splice(index, 1)
        }
        return []
    },
    size: function () { //取得数组长度,这个函数可以同步视图,length不能
        return this._.length
    },
    removeAll: function (all) { //移除N个元素
        if (Array.isArray(all)) {
            for (var i = this.length - 1; i >= 0; i--) {
                if (all.indexOf(this[i]) !== -1) {
                    _splice.call(this.$track, i, 1)
                    _splice.call(this, i, 1)
                    
                }
            }
        } else if (typeof all === "function") {
            for (i = this.length - 1; i >= 0; i--) {
                var el = this[i]
                if (all(el, i)) {
                     _splice.call(this.$track, i, 1)
                    _splice.call(this, i, 1)
                   
                }
            }
        } else {
            _splice.call(this.$track, 0, this.length)
            _splice.call(this, 0, this.length)
        }
        if (!W3C) {
            this.$model = toJson(this)
        }
        this.notify()
        this._.length = this.length
    },
    clear: function () {
        return this.removeAll()
    }
}
var _splice = arrayProto.splice
arrayMethods.forEach(function (method) {
    var original = arrayProto[method]
    newProto[method] = function () {
        // 继续尝试劫持数组元素的属性
        var args = []
        for (var i = 0, n = arguments.length; i < n; i++) {
            args[i] = observe(arguments[i], 0, 1, 1)
        }
        var result = original.apply(this, args)
        addTrack(this.$track, method, args)
        if (!W3C) {
            this.$model = toJson(this)
        }
        this.notify()
        this._.length = this.length
        return result
    }
})
"sort,reverse".replace(rword, function (method) {
    newProto[method] = function () {
        var oldArray = this.concat() //保持原来状态的旧数组
        var newArray = this
        var mask = Math.random()
        var indexes = []
        var hasSort = false
        arrayProto[method].apply(newArray, arguments) //排序
        for (var i = 0, n = oldArray.length; i < n; i++) {
            var neo = newArray[i]
            var old = oldArray[i]
            if (neo === old) {
                indexes.push(i)
            } else {
                var index = oldArray.indexOf(neo)
                indexes.push(index)//得到新数组的每个元素在旧数组对应的位置
                oldArray[index] = mask    //屏蔽已经找过的元素
                hasSort = true
            }
        }
        if (hasSort) {
            sortByIndex(this.$track, indexes)
            if (!W3C) {
                this.$model = toJson(this)
            }
            this.notify()
        }
        return this
    }
})
function sortByIndex(array, indexes) {
    var map = {};
    for (var i = 0, n = indexes.length; i < n; i++) {
        map[i] = array[i]
        var j = indexes[i]
        if (j in map) {
            array[i] = map[j]
            delete map[j]
        } else {
            array[i] = array[j]
        }
    }
}
function createTrack(n) {
    var ret = []
    for (var i = 0; i < n; i++) {
        ret[i] = generateID("$proxy$each")
    }
    return ret
}
function addTrack(track, method, args) {
    switch (method) {
        case 'push':
        case 'unshift':
            args = createTrack(args.length)
            break
        case 'splice':
            if (args.length > 2) {
                // 0, 5, a, b, c --> 0, 2, 0
                // 0, 5, a, b, c, d, e, f, g--> 0, 0, 3
                var del = args[1]
                var add = args.length - 2
                // args = [args[0], Math.max(del - add, 0)].concat(createTrack(Math.max(add - del, 0)))
                args = [args[0], args[1]].concat(createTrack(args.length - 2))
            }
            break
    }
    Array.prototype[method].apply(track, args)
}
/*********************************************************************
 *                           依赖调度系统                             *
 **********************************************************************/
//检测两个对象间的依赖关系
var dependencyDetection = (function () {
    var outerFrames = []
    var currentFrame
    return {
        begin: function (binding) {
            //accessorObject为一个拥有callback的对象
            outerFrames.push(currentFrame)
            currentFrame = binding
        },
        end: function () {
            currentFrame = outerFrames.pop()
        },
        collectDependency: function (array) {
            if (currentFrame) {
                //被dependencyDetection.begin调用
                currentFrame.callback(array)
            }
        }
    };
})()
//将绑定对象注入到其依赖项的订阅数组中
var roneval = /^on$/
function returnRandom() {
    return new Date() - 0
}
avalon.injectBinding = function (binding) {
    binding.handler = binding.handler || directives[binding.type].update || noop
    binding.update = function () {
        var begin = false
        if (!binding.getter) {
            begin = true
            dependencyDetection.begin({
                callback: function (array) {
                    injectDependency(array, binding)
                }
            })
            binding.getter = parseExpr(binding.expr, binding.vmodels, binding)
            binding.observers.forEach(function (a) {
                a.v.$watch(a.p, binding)
            })
            delete binding.observers
        }
        try {
            var args = binding.fireArgs, a, b
            delete binding.fireArgs
            if (!args) {
                if (binding.type === "on") {
                    a = binding.getter + ""
                } else {
                    a = binding.getter.apply(0, binding.args)
                }
            } else {
                a = args[0]
                b = args[1]
            }
            b = typeof b === "undefined" ? binding.oldValue : b
            if (binding._filters) {
                a = filters.$filter.apply(0, [a].concat(binding._filters))
            }
            if (binding.signature) {
                var xtype = avalon.type(a)
                if (xtype !== "array" && xtype !== "object") {
                    throw Error("warning:" + binding.expr + "只能是对象或数组")
                }
                binding.xtype = xtype
                var vtrack = getProxyIds(binding.proxies || [], xtype)
                var mtrack = a.$track || (xtype === "array" ? createTrack(a.length) :
                        Object.keys(a))
                binding.track = mtrack
                if (vtrack !== mtrack.join(";")) {
                    binding.handler(a, b)
                    binding.oldValue = 1
                }
            } else if (Array.isArray(a) ? a.length !== (b && b.length) : false) {
                binding.handler(a, b)
                binding.oldValue = a.concat()
            } else if (!("oldValue" in binding) || a !== b) {
                binding.handler(a, b)
                binding.oldValue = a
            }
        } catch (e) {
            delete binding.getter
            log("warning:exception throwed in [avalon.injectBinding] ", e)
            var node = binding.element
            if (node && node.nodeType === 3) {
                node.nodeValue = openTag + (binding.oneTime ? "::" : "") + binding.expr + closeTag
            }
        } finally {
            begin && dependencyDetection.end()
        }
    }
    binding.update()
}
//将依赖项(比它高层的访问器或构建视图刷新函数的绑定对象)注入到订阅者数组
function injectDependency(list, binding) {
    if (binding.oneTime)
        return
    if (list && avalon.Array.ensure(list, binding) && binding.element) {
        injectDisposeQueue(binding, list)
        if (new Date() - beginTime > 444) {
            rejectDisposeQueue()
        }
    }
}
function getProxyIds(a, isArray) {
    var ret = []
    for (var i = 0, el; el = a[i++]; ) {
        ret.push(isArray ? el.$id : el.$key)
    }
    return ret.join(";")
}
/*********************************************************************
 *                          定时GC回收机制                             *
 **********************************************************************/
var disposeCount = 0
var disposeQueue = avalon.$$subscribers = []
var beginTime = new Date()
var oldInfo = {}
function getUid(data) { //IE9+,标准浏览器
    if (!data.uniqueNumber) {
        var elem = data.element
        if (elem) {
            if (elem.nodeType !== 1) {
                //如果是注释节点,则data.pos不存在,当一个元素下有两个注释节点就会出问题
                data.uniqueNumber = data.type + "-" + getUid(elem.parentNode) + "-" + (++disposeCount)
            } else {
                data.uniqueNumber = data.name + "-" + getUid(elem)
            }
        } else {
            data.uniqueNumber = ++disposeCount
        }
    }
    return data.uniqueNumber
}
//添加到回收列队中
function injectDisposeQueue(data, list) {
    var lists = data.lists || (data.lists = [])
    var uuid = getUid(data)
    avalon.Array.ensure(lists, list)
    list.$uuid = list.$uuid || generateID()
    if (!disposeQueue[uuid]) {
        disposeQueue[uuid] = 1
        disposeQueue.push(data)
    }
}
function rejectDisposeQueue(data) {
    var i = disposeQueue.length
    var n = i
    var allTypes = []
    var iffishTypes = {}
    var newInfo = {}
    //对页面上所有绑定对象进行分门别类, 只检测个数发生变化的类型
    while (data = disposeQueue[--i]) {
        var type = data.type
        if (newInfo[type]) {
            newInfo[type]++
        } else {
            newInfo[type] = 1
            allTypes.push(type)
        }
    }
    var diff = false
    allTypes.forEach(function (type) {
        if (oldInfo[type] !== newInfo[type]) {
            iffishTypes[type] = 1
            diff = true
        }
    })
    i = n
    if (diff) {
        while (data = disposeQueue[--i]) {
            if (data.element === null) {
                disposeQueue.splice(i, 1)
                continue
            }
            if (iffishTypes[data.type] && shouldDispose(data.element)) { //如果它没有在DOM树
                disposeQueue.splice(i, 1)
                delete disposeQueue[data.uniqueNumber]
                var lists = data.lists
                for (var k = 0, list; list = lists[k++]; ) {
                    avalon.Array.remove(lists, list)
                    avalon.Array.remove(list, data)
                }
                disposeData(data)
            }
        }
    }
    oldInfo = newInfo
    beginTime = new Date()
}
function disposeData(data) {
    delete disposeQueue[data.uniqueNumber] // 先清除,不然无法回收了
    data.element = null
    data.rollback && data.rollback()
    for (var key in data) {
        data[key] = null
    }
}
function shouldDispose(el) {
    try {//IE下,如果文本节点脱离DOM树,访问parentNode会报错
        var fireError = el.parentNode.nodeType
    } catch (e) {
        return true
    }
    if (el.ifRemove) {
        // 如果节点被放到ifGroup,才移除
        if (!root.contains(el.ifRemove) && (ifGroup === el.parentNode)) {
            el.parentNode && el.parentNode.removeChild(el)
            return true
        }
    }
    return el.msRetain ? 0 : (el.nodeType === 1 ? !root.contains(el) : !avalon.contains(root, el))
}
/************************************************************************
 *              HTML处理(parseHTML, innerHTML, clearHTML)                 *
 **************************************************************************/
//parseHTML的辅助变量
var tagHooks = new function() {// jshint ignore:line
    avalon.mix(this, {
        option: DOC.createElement("select"),
        thead: DOC.createElement("table"),
        td: DOC.createElement("tr"),
        area: DOC.createElement("map"),
        tr: DOC.createElement("tbody"),
        col: DOC.createElement("colgroup"),
        legend: DOC.createElement("fieldset"),
        _default: DOC.createElement("div"),
        "g": DOC.createElementNS("http://www.w3.org/2000/svg", "svg")
    })
    this.optgroup = this.option
    this.tbody = this.tfoot = this.colgroup = this.caption = this.thead
    this.th = this.td
}// jshint ignore:line
String("circle,defs,ellipse,image,line,path,polygon,polyline,rect,symbol,text,use").replace(rword, function(tag) {
    tagHooks[tag] = tagHooks.g //处理SVG
})
var rtagName = /<([\w:]+)/
var rxhtml = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig
var scriptTypes = oneObject(["", "text/javascript", "text/ecmascript", "application/ecmascript", "application/javascript"])
var script = DOC.createElement("script")
var rhtml = /<|&#?\w+;/
avalon.parseHTML = function(html) {
    var fragment = avalonFragment.cloneNode(false)
    if (typeof html !== "string" ) {
        return fragment
    }
    if (!rhtml.test(html)) {
        fragment.appendChild(DOC.createTextNode(html))
        return fragment
    }
    html = html.replace(rxhtml, "<$1></$2>").trim()
    var tag = (rtagName.exec(html) || ["", ""])[1].toLowerCase(),
            //取得其标签名
            wrapper = tagHooks[tag] || tagHooks._default,
            firstChild
    wrapper.innerHTML = html
    var els = wrapper.getElementsByTagName("script")
    if (els.length) { //使用innerHTML生成的script节点不会发出请求与执行text属性
        for (var i = 0, el; el = els[i++]; ) {
            if (scriptTypes[el.type]) {
                var neo = script.cloneNode(false) //FF不能省略参数
                ap.forEach.call(el.attributes, function(attr) {
                    neo.setAttribute(attr.name, attr.value)
                })// jshint ignore:line
                neo.text = el.text
                el.parentNode.replaceChild(neo, el)
            }
        }
    }
    while (firstChild = wrapper.firstChild) { // 将wrapper上的节点转移到文档碎片上!
        fragment.appendChild(firstChild)
    }
    return fragment
}
avalon.innerHTML = function(node, html) {
    var a = this.parseHTML(html)
    this.clearHTML(node).appendChild(a)
}
avalon.clearHTML = function(node) {
    node.textContent = ""
    while (node.firstChild) {
        node.removeChild(node.firstChild)
    }
    return node
}
/*********************************************************************
 *                        avalon的原型方法定义区                        *
 **********************************************************************/
function hyphen(target) {
    //转换为连字符线风格
    return target.replace(/([a-z\d])([A-Z]+)/g, "$1-$2").toLowerCase()
}
function camelize(target) {
    //转换为驼峰风格
    if (target.indexOf("-") < 0 && target.indexOf("_") < 0) {
        return target //提前判断,提高getStyle等的效率
    }
    return target.replace(/[-_][^-_]/g, function (match) {
        return match.charAt(1).toUpperCase()
    })
}
"add,remove".replace(rword, function (method) {
    avalon.fn[method + "Class"] = function (cls) {
        var el = this[0]
        //https://developer.mozilla.org/zh-CN/docs/Mozilla/Firefox/Releases/26
        if (cls && typeof cls === "string" && el && el.nodeType === 1) {
            cls.replace(/\S+/g, function (c) {
                el.classList[method](c)
            })
        }
        return this
    }
})
avalon.fn.mix({
    hasClass: function (cls) {
        var el = this[0] || {} //IE10+, chrome8+, firefox3.6+, safari5.1+,opera11.5+支持classList,chrome24+,firefox26+支持classList2.0
        return el.nodeType === 1 && el.classList.contains(cls)
    },
    toggleClass: function (value, stateVal) {
        var className, i = 0
        var classNames = String(value).split(/\s+/)
        var isBool = typeof stateVal === "boolean"
        while ((className = classNames[i++])) {
            var state = isBool ? stateVal : !this.hasClass(className)
            this[state ? "addClass" : "removeClass"](className)
        }
        return this
    },
    attr: function (name, value) {
        if (arguments.length === 2) {
            this[0].setAttribute(name, value)
            return this
        } else {
            return this[0].getAttribute(name)
        }
    },
    data: function (name, value) {
        name = "data-" + hyphen(name || "")
        switch (arguments.length) {
            case 2:
                this.attr(name, value)
                return this
            case 1:
                var val = this.attr(name)
                return parseData(val)
            case 0:
                var ret = {}
                ap.forEach.call(this[0].attributes, function (attr) {
                    if (attr) {
                        name = attr.name
                        if (!name.indexOf("data-")) {
                            name = camelize(name.slice(5))
                            ret[name] = parseData(attr.value)
                        }
                    }
                })
                return ret
        }
    },
    removeData: function (name) {
        name = "data-" + hyphen(name)
        this[0].removeAttribute(name)
        return this
    },
    css: function (name, value) {
        if (avalon.isPlainObject(name)) {
            for (var i in name) {
                avalon.css(this, i, name[i])
            }
        } else {
            var ret = avalon.css(this, name, value)
        }
        return ret !== void 0 ? ret : this
    },
    position: function () {
        var offsetParent, offset,
                elem = this[0],
                parentOffset = {
                    top: 0,
                    left: 0
                };
        if (!elem) {
            return
        }
        if (this.css("position") === "fixed") {
            offset = elem.getBoundingClientRect()
        } else {
            offsetParent = this.offsetParent() //得到真正的offsetParent
            offset = this.offset() // 得到正确的offsetParent
            if (offsetParent[0].tagName !== "HTML") {
                parentOffset = offsetParent.offset()
            }
            parentOffset.top += avalon.css(offsetParent[0], "borderTopWidth", true)
            parentOffset.left += avalon.css(offsetParent[0], "borderLeftWidth", true)
            // Subtract offsetParent scroll positions
            parentOffset.top -= offsetParent.scrollTop()
            parentOffset.left -= offsetParent.scrollLeft()
        }
        return {
            top: offset.top - parentOffset.top - avalon.css(elem, "marginTop", true),
            left: offset.left - parentOffset.left - avalon.css(elem, "marginLeft", true)
        }
    },
    offsetParent: function () {
        var offsetParent = this[0].offsetParent
        while (offsetParent && avalon.css(offsetParent, "position") === "static") {
            offsetParent = offsetParent.offsetParent;
        }
        return avalon(offsetParent || root)
    },
    bind: function (type, fn, phase) {
        if (this[0]) { //此方法不会链
            return avalon.bind(this[0], type, fn, phase)
        }
    },
    unbind: function (type, fn, phase) {
        if (this[0]) {
            avalon.unbind(this[0], type, fn, phase)
        }
        return this
    },
    val: function (value) {
        var node = this[0]
        if (node && node.nodeType === 1) {
            var get = arguments.length === 0
            var access = get ? ":get" : ":set"
            var fn = valHooks[getValType(node) + access]
            if (fn) {
                var val = fn(node, value)
            } else if (get) {
                return (node.value || "").replace(/\r/g, "")
            } else {
                node.value = value
            }
        }
        return get ? val : this
    }
})
if (root.dataset) {
    avalon.fn.data = function (name, val) {
        name = name && camelize(name)
        var dataset = this[0].dataset
        switch (arguments.length) {
            case 2:
                dataset[name] = val
                return this
            case 1:
                val = dataset[name]
                return parseData(val)
            case 0:
                var ret = createMap()
                for (name in dataset) {
                    ret[name] = parseData(dataset[name])
                }
                return ret
        }
    }
}
var rbrace = /(?:\{[\s\S]*\}|\[[\s\S]*\])$/
avalon.parseJSON = JSON.parse
function parseData(data) {
    try {
        if (typeof data === "object")
            return data
        data = data === "true" ? true :
                data === "false" ? false :
                data === "null" ? null : +data + "" === data ? +data : rbrace.test(data) ? JSON.parse(data) : data
    } catch (e) {
    }
    return data
}
avalon.fireDom = function (elem, type, opts) {
    var hackEvent = DOC.createEvent("Events");
    hackEvent.initEvent(type, true, true)
    avalon.mix(hackEvent, opts)
    elem.dispatchEvent(hackEvent)
}
avalon.each({
    scrollLeft: "pageXOffset",
    scrollTop: "pageYOffset"
}, function (method, prop) {
    avalon.fn[method] = function (val) {
        var node = this[0] || {},
                win = getWindow(node),
                top = method === "scrollTop"
        if (!arguments.length) {
            return win ? win[prop] : node[method]
        } else {
            if (win) {
                win.scrollTo(!top ? val : win[prop], top ? val : win[prop])
            } else {
                node[method] = val
            }
        }
    }
})
function getWindow(node) {
    return node.window && node.document ? node : node.nodeType === 9 ? node.defaultView : false
}
//=============================css相关==================================
var cssHooks = avalon.cssHooks = createMap()
var prefixes = ["", "-webkit-", "-moz-", "-ms-"] //去掉opera-15的支持
var cssMap = {
    "float": "cssFloat"
}
avalon.cssNumber = oneObject("animationIterationCount,animationIterationCount,columnCount,order,flex,flexGrow,flexShrink,fillOpacity,fontWeight,lineHeight,opacity,orphans,widows,zIndex,zoom")
avalon.cssName = function (name, host, camelCase) {
    if (cssMap[name]) {
        return cssMap[name]
    }
    host = host || root.style
    for (var i = 0, n = prefixes.length; i < n; i++) {
        camelCase = camelize(prefixes[i] + name)
        if (camelCase in host) {
            return (cssMap[name] = camelCase)
        }
    }
    return null
}
cssHooks["@:set"] = function (node, name, value) {
    node.style[name] = value
}
cssHooks["@:get"] = function (node, name) {
    if (!node || !node.style) {
        throw new Error("getComputedStyle要求传入一个节点 " + node)
    }
    var ret, computed = getComputedStyle(node)
    if (computed) {
        ret = name === "filter" ? computed.getPropertyValue(name) : computed[name]
        if (ret === "") {
            ret = node.style[name] //其他浏览器需要我们手动取内联样式
        }
    }
    return ret
}
cssHooks["opacity:get"] = function (node) {
    var ret = cssHooks["@:get"](node, "opacity")
    return ret === "" ? "1" : ret
}
"top,left".replace(rword, function (name) {
    cssHooks[name + ":get"] = function (node) {
        var computed = cssHooks["@:get"](node, name)
        return /px$/.test(computed) ? computed :
                avalon(node).position()[name] + "px"
    }
})
var cssShow = {
    position: "absolute",
    visibility: "hidden",
    display: "block"
}
var rdisplayswap = /^(none|table(?!-c[ea]).+)/
function showHidden(node, array) {
    //http://www.cnblogs.com/rubylouvre/archive/2012/10/27/2742529.html
    if (node.offsetWidth <= 0) { //opera.offsetWidth可能小于0
        var styles = getComputedStyle(node, null)
        if (rdisplayswap.test(styles["display"])) {
            var obj = {
                node: node
            }
            for (var name in cssShow) {
                obj[name] = styles[name]
                node.style[name] = cssShow[name]
            }
            array.push(obj)
        }
        var parent = node.parentNode
        if (parent && parent.nodeType === 1) {
            showHidden(parent, array)
        }
    }
}
"Width,Height".replace(rword, function (name) { //fix 481
    var method = name.toLowerCase(),
            clientProp = "client" + name,
            scrollProp = "scroll" + name,
            offsetProp = "offset" + name
    cssHooks[method + ":get"] = function (node, which, override) {
        var boxSizing = -4
        if (typeof override === "number") {
            boxSizing = override
        }
        which = name === "Width" ? ["Left", "Right"] : ["Top", "Bottom"]
        var ret = node[offsetProp] // border-box 0
        if (boxSizing === 2) { // margin-box 2
            return ret + avalon.css(node, "margin" + which[0], true) + avalon.css(node, "margin" + which[1], true)
        }
        if (boxSizing < 0) { // padding-box  -2
            ret = ret - avalon.css(node, "border" + which[0] + "Width", true) - avalon.css(node, "border" + which[1] + "Width", true)
        }
        if (boxSizing === -4) { // content-box -4
            ret = ret - avalon.css(node, "padding" + which[0], true) - avalon.css(node, "padding" + which[1], true)
        }
        return ret
    }
    cssHooks[method + "&get"] = function (node) {
        var hidden = [];
        showHidden(node, hidden);
        var val = cssHooks[method + ":get"](node)
        for (var i = 0, obj; obj = hidden[i++]; ) {
            node = obj.node
            for (var n in obj) {
                if (typeof obj[n] === "string") {
                    node.style[n] = obj[n]
                }
            }
        }
        return val;
    }
    avalon.fn[method] = function (value) { //会忽视其display
        var node = this[0]
        if (arguments.length === 0) {
            if (node.setTimeout) { //取得窗口尺寸,IE9后可以用node.innerWidth /innerHeight代替
                return node["inner" + name]
            }
            if (node.nodeType === 9) { //取得页面尺寸
                var doc = node.documentElement
                //FF chrome    html.scrollHeight< body.scrollHeight
                //IE 标准模式 : html.scrollHeight> body.scrollHeight
                //IE 怪异模式 : html.scrollHeight 最大等于可视窗口多一点?
                return Math.max(node.body[scrollProp], doc[scrollProp], node.body[offsetProp], doc[offsetProp], doc[clientProp])
            }
            return cssHooks[method + "&get"](node)
        } else {
            return this.css(method, value)
        }
    }
    avalon.fn["inner" + name] = function () {
        return cssHooks[method + ":get"](this[0], void 0, -2)
    }
    avalon.fn["outer" + name] = function (includeMargin) {
        return cssHooks[method + ":get"](this[0], void 0, includeMargin === true ? 2 : 0)
    }
})
avalon.fn.offset = function () { //取得距离页面左右角的坐标
    var node = this[0]
    try {
        var rect = node.getBoundingClientRect()
        // Make sure element is not hidden (display: none) or disconnected
        // https://github.com/jquery/jquery/pull/2043/files#r23981494
        if (rect.width || rect.height || node.getClientRects().length) {
            var doc = node.ownerDocument
            var root = doc.documentElement
            var win = doc.defaultView
            return {
                top: rect.top + win.pageYOffset - root.clientTop,
                left: rect.left + win.pageXOffset - root.clientLeft
            }
        }
    } catch (e) {
        return {
            left: 0,
            top: 0
        }
    }
}
//=============================val相关=======================
function getValType(elem) {
    var ret = elem.tagName.toLowerCase()
    return ret === "input" && /checkbox|radio/.test(elem.type) ? "checked" : ret
}
var valHooks = {
    "select:get": function (node, value) {
        var option, options = node.options,
                index = node.selectedIndex,
                one = node.type === "select-one" || index < 0,
                values = one ? null : [],
                max = one ? index + 1 : options.length,
                i = index < 0 ? max : one ? index : 0
        for (; i < max; i++) {
            option = options[i]
            //旧式IE在reset后不会改变selected,需要改用i === index判定
            //我们过滤所有disabled的option元素,但在safari5下,如果设置select为disable,那么其所有孩子都disable
            //因此当一个元素为disable,需要检测其是否显式设置了disable及其父节点的disable情况
            if ((option.selected || i === index) && !option.disabled) {
                value = option.value
                if (one) {
                    return value
                }
                //收集所有selected值组成数组返回
                values.push(value)
            }
        }
        return values
    },
    "select:set": function (node, values, optionSet) {
        values = [].concat(values) //强制转换为数组
        for (var i = 0, el; el = node.options[i++]; ) {
            if ((el.selected = values.indexOf(el.value) > -1)) {
                optionSet = true
            }
        }
        if (!optionSet) {
            node.selectedIndex = -1
        }
    }
}
var keyMap = {}
var keys = ["break,case,catch,continue,debugger,default,delete,do,else,false",
    "finally,for,function,if,in,instanceof,new,null,return,switch,this",
    "throw,true,try,typeof,var,void,while,with", /* 关键字*/
    "abstract,boolean,byte,char,class,const,double,enum,export,extends",
    "final,float,goto,implements,import,int,interface,long,native",
    "package,private,protected,public,short,static,super,synchronized",
    "throws,transient,volatile", /*保留字*/
    "arguments,let,yield,undefined"].join(",")
keys.replace(/\w+/g, function (a) {
    keyMap[a] = true
})
var ridentStart = /[a-z_$]/i
var rwhiteSpace = /[\s\uFEFF\xA0]/
function getIdent(input, lastIndex) {
    var result = []
    var subroutine = !!lastIndex
    lastIndex = lastIndex || 0
    //将表达式中的标识符抽取出来
    var state = "unknown"
    var variable = ""
    for (var i = 0; i < input.length; i++) {
        var c = input.charAt(i)
        if (c === "'" || c === '"') {//字符串开始
            if (state === "unknown") {
                state = c
            } else if (state === c) {//字符串结束
                state = "unknown"
            }
        } else if (c === "\\") {
            if (state === "'" || state === '"') {
                i++
            }
        } else if (ridentStart.test(c)) {//碰到标识符
            if (state === "unknown") {
                state = "variable"
                variable = c
            } else if (state === "maybePath") {
                variable = result.pop()
                variable += "." + c
                state = "variable"
            } else if (state === "variable") {
                variable += c
            }
        } else if (/\w/.test(c)) {
            if (state === "variable") {
                variable += c
            }
        } else if (c === ".") {
            if (state === "variable") {
                if (variable) {
                    result.push(variable)
                    variable = ""
                    state = "maybePath"
                }
            }
        } else if (c === "[") {
            if (state === "variable" || state === "maybePath") {
                if (variable) {//如果前面存在变量,收集它
                    result.push(variable)
                    variable = ""
                }
                var lastLength = result.length
                var last = result[lastLength - 1]
                var innerResult = getIdent(input.slice(i), i)
                if (innerResult.length) {//如果括号中存在变量,那么这里添加通配符
                    result[lastLength - 1] = last + ".*"
                    result = innerResult.concat(result)
                } else { //如果括号中的东西是确定的,直接转换为其子属性
                    var content = input.slice(i + 1, innerResult.i)
                    try {
                        var text = (scpCompile(["return " + content]))()
                        result[lastLength - 1] = last + "." + text
                    } catch (e) {
                    }
                }
                state = "maybePath"//]后面可能还接东西
                i = innerResult.i
            }
        } else if (c === "]") {
            if (subroutine) {
                result.i = i + lastIndex
                addVar(result, variable)
                return result
            }
        } else if (rwhiteSpace.test(c) && c !== "\r" && c !== "\n") {
            if (state === "variable") {
                if (addVar(result, variable)) {
                    state = "maybePath" // aaa . bbb 这样的情况
                }
                variable = ""
            }
        } else {
            addVar(result, variable)
            state = "unknown"
            variable = ""
        }
    }
    addVar(result, variable)
    return result
}
function addVar(array, element) {
    if (element && !keyMap[element]) {
        array.push(element)
        return true
    }
}
function addAssign(vars, vmodel, name, binding) {
    var ret = [],
            prefix = " = " + name + "."
    for (var i = vars.length, prop; prop = vars[--i]; ) {
        var arr = prop.split("."), a
        var first = arr[0]
        while (a = arr.shift()) {
            if (vmodel.hasOwnProperty(a)) {
                ret.push(first + prefix + first)
                binding.observers.push({
                    v: vmodel,
                    p: prop
                })
                vars.splice(i, 1)
            }
        }
    }
    return ret
}
var rproxy = /(\$proxy\$[a-z]+)\d+$/
var variablePool = new Cache(218)
//缓存求值函数,以便多次利用
var evaluatorPool = new Cache(128)
function getVars(expr) {
    expr = expr.trim()
    var ret = variablePool.get(expr)
    if (ret) {
        return ret.concat()
    }
    var array = getIdent(expr)
    var uniq = {}
    var result = []
    for (var i = 0, el; el = array[i++]; ) {
        if (!uniq[el]) {
            uniq[el] = 1
            result.push(el)
        }
    }
    return variablePool.put(expr, result).concat()
}
function parseExpr(expr, vmodels, binding) {
    var filters = binding.filters
    if (typeof filters === "string" && filters.trim() && !binding._filters) {
        binding._filters = parseFilter(filters.trim())
    }
    var vars = getVars(expr)
    var expose = new Date() - 0
    var assigns = []
    var names = []
    var args = []
    binding.observers = []
    for (var i = 0, sn = vmodels.length; i < sn; i++) {
        if (vars.length) {
            var name = "vm" + expose + "_" + i
            names.push(name)
            args.push(vmodels[i])
            assigns.push.apply(assigns, addAssign(vars, vmodels[i], name, binding))
        }
    }
    binding.args = args
    var dataType = binding.type
    var exprId = vmodels.map(function (el) {
        return String(el.$id).replace(rproxy, "$1")
    }) + expr + dataType
    var getter = evaluatorPool.get(exprId) //直接从缓存,免得重复生成
    if (getter) {
        if (dataType === "duplex") {
            var setter = evaluatorPool.get(exprId + "setter")
            binding.setter = setter.apply(setter, binding.args)
        }
        return binding.getter = getter
    }
    if (!assigns.length) {
        assigns.push("fix" + expose)
    }
    if (dataType === "duplex") {
        var nameOne = {}
        assigns.forEach(function (a) {
            var arr = a.split("=")
            nameOne[arr[0].trim()] = arr[1].trim()
        })
        expr = expr.replace(/[\$\w]+/, function (a) {
            return nameOne[a] ? nameOne[a] : a
        })
        /* jshint ignore:start */
        var fn2 = scpCompile(names.concat("'use strict';" +
                "return function(vvv){" + expr + " = vvv\n}\n"))
        /* jshint ignore:end */
        evaluatorPool.put(exprId + "setter", fn2)
        binding.setter = fn2.apply(fn2, binding.args)
    }
    if (dataType === "on") { //事件绑定
        if (expr.indexOf("(") === -1) {
            expr += ".call(this, $event)"
        } else {
            expr = expr.replace("(", ".call(this,")
        }
        names.push("$event")
        expr = "\nreturn " + expr + ";" //IE全家 Function("return ")出错,需要Function("return ;")
        var lastIndex = expr.lastIndexOf("\nreturn")
        var header = expr.slice(0, lastIndex)
        var footer = expr.slice(lastIndex)
        expr = header + "\n" + footer
    } else {
        expr = "\nreturn " + expr + ";" //IE全家 Function("return ")出错,需要Function("return ;")
    }
    /* jshint ignore:start */
    getter = scpCompile(names.concat("'use strict';\nvar " +
            assigns.join(",\n") + expr))
    /* jshint ignore:end */
    return  evaluatorPool.put(exprId, getter)
}
//========
function normalizeExpr(code) {
    var hasExpr = rexpr.test(code) //比如ms-class="width{{w}}"的情况
    if (hasExpr) {
        var array = scanExpr(code)
        if (array.length === 1) {
            return array[0].expr
        }
        return array.map(function (el) {
            return el.type ? "(" + el.expr + ")" : quote(el.expr)
        }).join(" + ")
    } else {
        return code
    }
}
avalon.normalizeExpr = normalizeExpr
avalon.parseExprProxy = parseExpr
var rthimRightParentheses = /\)\s*$/
var rthimOtherParentheses = /\)\s*\|/g
var rquoteFilterName = /\|\s*([$\w]+)/g
var rpatchBracket = /"\s*\["/g
var rthimLeftParentheses = /"\s*\(/g
function parseFilter(filters) {
    filters = filters
            .replace(rthimRightParentheses, "")//处理最后的小括号
            .replace(rthimOtherParentheses, function () {//处理其他小括号
                return "],|"
            })
            .replace(rquoteFilterName, function (a, b) { //处理|及它后面的过滤器的名字
                return "[" + quote(b)
            })
            .replace(rpatchBracket, function () {
                return '"],["'
            })
            .replace(rthimLeftParentheses, function () {
                return '",'
            }) + "]"
    /* jshint ignore:start */
    return  scpCompile(["return [" + filters + "]"])()
    /* jshint ignore:end */
}
/*********************************************************************
 *                          编译系统                                  *
 **********************************************************************/
var quote = JSON.stringify
/*********************************************************************
 *                           扫描系统                                 *
 **********************************************************************/
avalon.scan = function (elem, vmodel) {
    elem = elem || root
    var vmodels = vmodel ? [].concat(vmodel) : []
    scanTag(elem, vmodels)
}
//http://www.w3.org/TR/html5/syntax.html#void-elements
var stopScan = oneObject("area,base,basefont,br,col,command,embed,hr,img,input,link,meta,param,source,track,wbr,noscript,script,style,textarea".toUpperCase())
function checkScan(elem, callback, innerHTML) {
    var id = setTimeout(function () {
        var currHTML = elem.innerHTML
        clearTimeout(id)
        if (currHTML === innerHTML) {
            callback()
        } else {
            checkScan(elem, callback, currHTML)
        }
    })
}
function createSignalTower(elem, vmodel) {
    var id = elem.getAttribute("avalonctrl") || vmodel.$id
    elem.setAttribute("avalonctrl", id)
    if (vmodel.$events) {
        vmodel.$events.expr = elem.tagName + '[avalonctrl="' + id + '"]'
    }
}
var getBindingCallback = function (elem, name, vmodels) {
    var callback = elem.getAttribute(name)
    if (callback) {
        for (var i = 0, vm; vm = vmodels[i++]; ) {
            if (vm.hasOwnProperty(callback) && typeof vm[callback] === "function") {
                return vm[callback]
            }
        }
    }
}
function executeBindings(bindings, vmodels) {
    for (var i = 0, binding; binding = bindings[i++]; ) {
        binding.vmodels = vmodels
        directives[binding.type].init(binding)
      
        avalon.injectBinding(binding)
        if (binding.getter && binding.element.nodeType === 1) { //移除数据绑定,防止被二次解析
            //chrome使用removeAttributeNode移除不存在的特性节点时会报错 https://github.com/RubyLouvre/avalon/issues/99
            binding.element.removeAttribute(binding.name)
        }
    }
    bindings.length = 0
}
//https://github.com/RubyLouvre/avalon/issues/636
var mergeTextNodes = IEVersion && window.MutationObserver ? function (elem) {
    var node = elem.firstChild, text
    while (node) {
        var aaa = node.nextSibling
        if (node.nodeType === 3) {
            if (text) {
                text.nodeValue += node.nodeValue
                elem.removeChild(node)
            } else {
                text = node
            }
        } else {
            text = null
        }
        node = aaa
    }
} : 0
var roneTime = /^\s*::/
var rmsAttr = /ms-(\w+)-?(.*)/
var events = oneObject("animationend,blur,change,input,click,dblclick,focus,keydown,keypress,keyup,mousedown,mouseenter,mouseleave,mousemove,mouseout,mouseover,mouseup,scan,scroll,submit")
var obsoleteAttrs = oneObject("value,title,alt,checked,selected,disabled,readonly,enabled,href,src")
function bindingSorter(a, b) {
    return a.priority - b.priority
}
var rnoCollect = /^(ms-\S+|data-\S+|on[a-z]+|id|style|class)$/
var ronattr = /^on\-[\w-]+$/
function getOptionsFromTag(elem, vmodels) {
    var attributes = elem.attributes
    var ret = {}
    for (var i = 0, attr; attr = attributes[i++]; ) {
        var name = attr.name
        if (attr.specified && !rnoCollect.test(name)) {
            var camelizeName = camelize(attr.name)
            if (/^on\-[\w-]+$/.test(name)) {
                ret[camelizeName] = getBindingCallback(elem, name, vmodels) 
            } else {
                ret[camelizeName] = parseData(attr.value)
            }
        }
    }
    return ret
}
function scanAttr(elem, vmodels, match) {
    var scanNode = true
    if (vmodels.length) {
        var attributes = elem.attributes
        var bindings = []
        var uniq = {}
        for (var i = 0, attr; attr = attributes[i++]; ) {
            var name = attr.name
            if (uniq[name]) {//IE8下ms-repeat,ms-with BUG
                continue
            }
            uniq[name] = 1
            if (attr.specified) {
                if (match = name.match(rmsAttr)) {
                    //如果是以指定前缀命名的
                    var type = match[1]
                    var param = match[2] || ""
                    var value = attr.value
                    if (events[type]) {
                        param = type
                        type = "on"
                    } else if (obsoleteAttrs[type]) {
                        param = type
                        type = "attr"
                        name = "ms-" + type + "-" + param
                        log("warning!请改用" + name + "代替" + attr.name + "!")
                    }
                    if (directives[type]) {
                        var newValue = value.replace(roneTime, "")
                        var oneTime = value !== newValue
                        var binding = {
                            type: type,
                            param: param,
                            element: elem,
                            name: name,
                            expr: newValue,
                            oneTime: oneTime,
                            priority: (directives[type].priority || type.charCodeAt(0) * 10) + (Number(param.replace(/\D/g, "")) || 0)
                        }
                        if (type === "html" || type === "text") {
                            var filters = getToken(value).filters
                            binding.expr = binding.expr.replace(filters, "")
                            binding.filters = filters.replace(rhasHtml, function () {
                                binding.type = "html"
                                binding.group = 1
                                return ""
                            }).trim() // jshint ignore:line
                        } else if (type === "duplex") {
                            var hasDuplex = name
                        } else if (name === "ms-if-loop") {
                            binding.priority += 100
                        } else if (name === "ms-attr-value") {
                            var hasAttrValue = name
                        }
                        bindings.push(binding)
                    }
                }
            }
        }
        if (bindings.length) {
            bindings.sort(bindingSorter)
            if (hasDuplex && hasAttrValue && elem.type === "text") {
                log("warning!一个控件不能同时定义ms-attr-value与" + hasDuplex)
            }
            for (i = 0; binding = bindings[i]; i++) {
                type = binding.type
                if (rnoscanAttrBinding.test(type)) {
                    return executeBindings(bindings.slice(0, i + 1), vmodels)
                } else if (scanNode) {
                    scanNode = !rnoscanNodeBinding.test(type)
                }
            }
            executeBindings(bindings, vmodels)
        }
    }
    if (scanNode && !stopScan[elem.tagName]) {
        mergeTextNodes && mergeTextNodes(elem)
        scanNodeList(elem, vmodels) //扫描子孙元素
    }
}
var rnoscanAttrBinding = /^if|widget|repeat$/
var rnoscanNodeBinding = /^each|with|html|include$/
function scanNodeList(parent, vmodels) {
    var nodes = avalon.slice(parent.childNodes)
    scanNodeArray(nodes, vmodels)
}
function scanNodeArray(nodes, vmodels) {
    for (var i = 0, node; node = nodes[i++]; ) {
        switch (node.nodeType) {
            case 1:
                var elem = node
                if (!elem.msResolved && elem.parentNode && elem.parentNode.nodeType === 1) {
                    var library = isWidget(elem)
                    if (library) {
                        var widget = elem.localName ? elem.localName.replace(library + ":", "") : elem.nodeName
                        var fullName = library + ":" + camelize(widget)
                        componentQueue.push({
                            library: library,
                            element: elem,
                            fullName: fullName,
                            widget: widget,
                            vmodels: vmodels,
                            name: "widget"
                        })
                        if (avalon.components[fullName]) {
                            (function (name) {//确保所有ms-attr-name扫描完再处理
                                setTimeout(function () {
                                    avalon.component(name)
                                })
                            })(fullName)
                        }
                    }
                }
                scanTag(node, vmodels) //扫描元素节点
                if (node.msHasEvent) {
                    avalon.fireDom(node, "datasetchanged", {
                        bubble: node.msHasEvent
                    })
                }
                break
            case 3:
                if (rexpr.test(node.nodeValue)) {
                    scanText(node, vmodels, i) //扫描文本节点
                }
                break
        }
    }
}
function scanTag(elem, vmodels, node) {
    //扫描顺序  ms-skip(0) --> ms-important(1) --> ms-controller(2) --> ms-if(10) --> ms-repeat(100) 
    //--> ms-if-loop(110) --> ms-attr(970) ...--> ms-each(1400)-->ms-with(1500)--〉ms-duplex(2000)垫后        
    var a = elem.getAttribute("ms-skip")
    var b = elem.getAttributeNode("ms-important")
    var c = elem.getAttributeNode("ms-controller")
    if (typeof a === "string") {
        return
    } else if (node = b || c) {
        var newVmodel = avalon.vmodels[node.value]
        if (!newVmodel) {
            return
        }
        //ms-important不包含父VM,ms-controller相反
        vmodels = node === b ? [newVmodel] : [newVmodel].concat(vmodels)
        elem.removeAttribute(node.name) //removeAttributeNode不会刷新[ms-controller]样式规则
        elem.classList.remove(node.name)
        createSignalTower(elem, newVmodel)
    }
    scanAttr(elem, vmodels) //扫描特性节点
    if (newVmodel) {
        setTimeout(function () {
            newVmodel.$fire("ms-scan-end", elem)
        })
    }
}
var rhasHtml = /\|\s*html(?:\b|$)/,
    r11a = /\|\|/g,
    rlt = /&lt;/g,
    rgt = /&gt;/g,
    rstringLiteral = /(['"])(\\\1|.)+?\1/g
function getToken(value) {
    if (value.indexOf("|") > 0) {
        var scapegoat = value.replace(rstringLiteral, function (_) {
            return Array(_.length + 1).join("1") // jshint ignore:line
        })
        var index = scapegoat.replace(r11a, "\u1122\u3344").indexOf("|") //干掉所有短路或
        if (index > -1) {
            return {
                type: "text",
                filters: value.slice(index).trim(),
                expr: value.slice(0, index)
            }
        }
    }
    return {
        type: "text",
        expr: value,
        filters: ""
    }
}
function scanExpr(str) {
    var tokens = [],
        value, start = 0,
        stop
    do {
        stop = str.indexOf(openTag, start)
        if (stop === -1) {
            break
        }
        value = str.slice(start, stop)
        if (value) { // {{ 左边的文本
            tokens.push({
                expr: value
            })
        }
        start = stop + openTag.length
        stop = str.indexOf(closeTag, start)
        if (stop === -1) {
            break
        }
        value = str.slice(start, stop)
        if (value) { //处理{{ }}插值表达式
            tokens.push(getToken(value, start))
        }
        start = stop + closeTag.length
    } while (1)
    value = str.slice(start)
    if (value) { //}} 右边的文本
        tokens.push({
            expr: value
        })
    }
    return tokens
}
function scanText(textNode, vmodels, index) {
    var bindings = [],
    tokens = scanExpr(textNode.data)
    if (tokens.length) {
        for (var i = 0, token; token = tokens[i++];) {
            var node = DOC.createTextNode(token.expr) //将文本转换为文本节点,并替换原来的文本节点
            if (token.type) {
                token.expr = token.expr.replace(roneTime, function () {
                        token.oneTime = true
                        return ""
                    }) // jshint ignore:line
                token.element = node
                token.filters = token.filters.replace(rhasHtml, function () {
                        token.type = "html"
                        return ""
                    }) // jshint ignore:line
                token.pos = index * 1000 + i
                bindings.push(token) //收集带有插值表达式的文本
            }
            avalonFragment.appendChild(node)
        }
        textNode.parentNode.replaceChild(avalonFragment, textNode)
        if (bindings.length)
            executeBindings(bindings, vmodels)
    }
}
//使用来自游戏界的双缓冲技术,减少对视图的冗余刷新
var Buffer = function () {
    this.queue = []
}
Buffer.prototype = {
    render: function (isAnimate) {
        if (!this.locked) {
            this.locked = isAnimate ? root.offsetHeight + 10 : 1
            var me = this
            avalon.nextTick(function () {
                me.flush()
            })
        }
    },
    flush: function () {
        for (var i = 0, sub; sub = this.queue[i++]; ) {
            sub.update && sub.update()
        }
        this.locked = 0
        this.queue = []
    }
}
var buffer = new Buffer()
var componentQueue = []
var widgetList = []
var componentHooks = {
    $construct: function () {
        return avalon.mix.apply(null, arguments)
    },
    $ready: noop,
    $init: noop,
    $dispose: noop,
    $container: null,
    $childReady: noop,
    $replace: false,
    $extend: null,
    $$template: function (str) {
        return str
    }
}
avalon.components = {}
avalon.component = function (name, opts) {
    if (opts) {
        avalon.components[name] = avalon.mix({}, componentHooks, opts)
    }
    for (var i = 0, obj; obj = componentQueue[i]; i++) {
        if (name === obj.fullName) {
            componentQueue.splice(i, 1)
            i--;
            (function (host, hooks, elem, widget) {
                //如果elem已从Document里移除,直接返回
                //issuse : https://github.com/RubyLouvre/avalon2/issues/40
                if (!avalon.contains(DOC, elem)) {
                    avalon.Array.remove(componentQueue, host)
                    return
                }
                
                var dependencies = 1
                var library = host.library
                var global = avalon.libraries[library] || componentHooks
                //===========收集各种配置=======
                if (elem.getAttribute("ms-attr-identifier")) {
                    //如果还没有解析完,就延迟一下 #1155
                    return
                }
                var elemOpts = getOptionsFromTag(elem, host.vmodels)
                var vmOpts = getOptionsFromVM(host.vmodels, elemOpts.config || host.fullName)
                var $id = elemOpts.$id || elemOpts.identifier || generateID(widget)
                delete elemOpts.config
                delete elemOpts.$id
                delete elemOpts.identifier
                var componentDefinition = {}
                var parentHooks = avalon.components[hooks.$extend]
                if (parentHooks) {
                    avalon.mix(true, componentDefinition, parentHooks)
                    componentDefinition = parentHooks.$construct.call(elem, componentDefinition, {}, {})
                } else {
                    avalon.mix(true, componentDefinition, hooks)
                }
                componentDefinition = avalon.components[name].$construct.call(elem, componentDefinition, vmOpts, elemOpts)
                componentDefinition.$refs = {}
                componentDefinition.$id = $id
                //==========构建VM=========
                var keepSlot = componentDefinition.$slot
                var keepReplace = componentDefinition.$replace
                var keepContainer = componentDefinition.$container
                var keepTemplate = componentDefinition.$template
                delete componentDefinition.$slot
                delete componentDefinition.$replace
                delete componentDefinition.$container
                delete componentDefinition.$construct
                var vmodel = avalon.define(componentDefinition) || {}
                elem.msResolved = 1
                vmodel.$init(vmodel, elem)
                global.$init(vmodel, elem)
                var nodes = elem.childNodes
                //收集插入点
                var slots = {}, snode
                for (var s = 0, el; el = nodes[s++]; ) {
                    var type = el.nodeType === 1 && el.getAttribute("slot") || keepSlot
                    if (type) {
                        if (slots[type]) {
                            slots[type].push(el)
                        } else {
                            slots[type] = [el]
                        }
                    }
                }
                if (vmodel.$$template) {
                    avalon.clearHTML(elem)
                    elem.innerHTML = vmodel.$$template(keepTemplate)
                }
                for (s in slots) {
                    if (vmodel.hasOwnProperty(s)) {
                        var ss = slots[s]
                        if (ss.length) {
                            var fragment = avalonFragment.cloneNode(true)
                            for (var ns = 0; snode = ss[ns++]; ) {
                                fragment.appendChild(snode)
                            }
                            vmodel[s] = fragment
                        }
                        slots[s] = null
                    }
                }
                slots = null
                var child = elem.children[0] || elem.firstChild
                if (keepReplace) {
                    elem.parentNode.replaceChild(child, elem)
                    child.msResolved = 1
                    var cssText = elem.style.cssText
                    var className = elem.className
                    elem = host.element = child
                    elem.style.cssText = cssText
                    if (className) {
                        avalon(elem).addClass(className)
                    }
                }
                if (keepContainer) {
                    keepContainer.appendChild(elem)
                }
                avalon.fireDom(elem, "datasetchanged",
                        {library: library, vm: vmodel, childReady: 1})
                var children = 0
                var removeFn = avalon.bind(elem, "datasetchanged", function (e) {
                    if (e.childReady && e.library === library) {
                        dependencies += e.childReady
                        if (vmodel !== e.vm) {
                            vmodel.$refs[e.vm.$id] = e.vm
                            if (e.childReady === -1) {
                                children++
                                vmodel.$childReady(vmodel, elem, e)
                            }
                            e.stopPropagation()
                        }
                    }
                    if (dependencies === 0) {
                        var id1 = setTimeout(function () {
                            clearTimeout(id1)
                            vmodel.$ready(vmodel, elem, host.vmodels)
                            global.$ready(vmodel, elem, host.vmodels)
                        }, children ? Math.max(children * 17, 100) : 17)
                        avalon.unbind(elem, "datasetchanged", removeFn)
                        //==================
                        host.rollback = function () {
                            try {
                                vmodel.$dispose(vmodel, elem)
                                global.$dispose(vmodel, elem)
                            } catch (e) {
                            }
                            delete avalon.vmodels[vmodel.$id]
                        }
                        injectDisposeQueue(host, widgetList)
                        if (window.chrome) {
                            elem.addEventListener("DOMNodeRemovedFromDocument", function () {
                                setTimeout(rejectDisposeQueue)
                            })
                        }
                    }
                })
                scanTag(elem, [vmodel].concat(host.vmodels))
                avalon.vmodels[vmodel.$id] = vmodel
                if (!elem.childNodes.length) {
                    avalon.fireDom(elem, "datasetchanged", {library: library, vm: vmodel, childReady: -1})
                } else {
                    var id2 = setTimeout(function () {
                        clearTimeout(id2)
                        avalon.fireDom(elem, "datasetchanged", {library: library, vm: vmodel, childReady: -1})
                    }, 17)
                }
            })(obj, avalon.components[name], obj.element, obj.widget)// jshint ignore:line
        }
    }
}
function getOptionsFromVM(vmodels, pre) {
    if (pre) {
        for (var i = 0, v; v = vmodels[i++]; ) {
            if (v.hasOwnProperty(pre) && typeof v[pre] === "object") {
                var vmOptions = v[pre]
                return vmOptions.$model || vmOptions
                break
            }
        }
    }
    return {}
}
avalon.libraries = []
avalon.library = function (name, opts) {
    if (DOC.namespaces) {
        DOC.namespaces.add(name, 'http://www.w3.org/1999/xhtml');
    }
    avalon.libraries[name] = avalon.mix({
        $init: noop,
        $ready: noop,
        $dispose: noop
    }, opts || {})
}
avalon.library("ms")
/*
 broswer  nodeName  scopeName  localName
 IE9     ONI:BUTTON oni        button
 IE10    ONI:BUTTON undefined  oni:button
 IE8     button     oni        undefined
 chrome  ONI:BUTTON undefined  oni:button
 
 */
function isWidget(el) { //如果为自定义标签,返回UI库的名字
    if (el.scopeName && el.scopeName !== "HTML") {
        return el.scopeName
    }
    var fullName = el.nodeName.toLowerCase()
    var index = fullName.indexOf(":")
    if (index > 0) {
        return fullName.slice(0, index)
    }
}
//各种MVVM框架在大型表格下的性能测试
// https://github.com/RubyLouvre/avalon/issues/859
var bools = ["autofocus,autoplay,async,allowTransparency,checked,controls",
    "declare,disabled,defer,defaultChecked,defaultSelected",
    "contentEditable,isMap,loop,multiple,noHref,noResize,noShade",
    "open,readOnly,selected"
].join(",")
var boolMap = {}
bools.replace(rword, function (name) {
    boolMap[name.toLowerCase()] = name
})
var propMap = {//属性名映射
    "accept-charset": "acceptCharset",
    "char": "ch",
    "charoff": "chOff",
    "class": "className",
    "for": "htmlFor",
    "http-equiv": "httpEquiv"
}
var anomaly = ["accessKey,bgColor,cellPadding,cellSpacing,codeBase,codeType,colSpan",
    "dateTime,defaultValue,frameBorder,longDesc,maxLength,marginWidth,marginHeight",
    "rowSpan,tabIndex,useMap,vSpace,valueType,vAlign"
].join(",")
anomaly.replace(rword, function (name) {
    propMap[name.toLowerCase()] = name
})
var attrDir = avalon.directive("attr", {
    init: function (binding) {
        //{{aaa}} --> aaa
        //{{aaa}}/bbb.html --> (aaa) + "/bbb.html"
        binding.expr = normalizeExpr(binding.expr.trim())
        if (binding.type === "include") {
            var elem = binding.element
            effectBinding(elem, binding)
            binding.includeRendered = getBindingCallback(elem, "data-include-rendered", binding.vmodels)
            binding.includeLoaded = getBindingCallback(elem, "data-include-loaded", binding.vmodels)
            var outer = binding.includeReplace = !!avalon(elem).data("includeReplace")
            if (avalon(elem).data("includeCache")) {
                binding.templateCache = {}
            }
            binding.start = DOC.createComment("ms-include")
            binding.end = DOC.createComment("ms-include-end")
            if (outer) {
                binding.element = binding.end
                binding._element = elem
                elem.parentNode.insertBefore(binding.start, elem)
                elem.parentNode.insertBefore(binding.end, elem.nextSibling)
            } else {
                elem.insertBefore(binding.start, elem.firstChild)
                elem.appendChild(binding.end)
            }
        }
    },
    update: function (val) {
        var elem = this.element
        var attrName = this.param
        if (attrName === "href" || attrName === "src") {
            if (typeof val === "string" && !root.hasAttribute) {
                val = val.replace(/&amp;/g, "&") //处理IE67自动转义的问题
            }
            elem[attrName] = val
            if (window.chrome && elem.tagName === "EMBED") {
                var parent = elem.parentNode //#525  chrome1-37下embed标签动态设置src不能发生请求
                var comment = document.createComment("ms-src")
                parent.replaceChild(comment, elem)
                parent.replaceChild(elem, comment)
            }
        } else {
            // ms-attr-class="xxx" vm.xxx="aaa bbb ccc"将元素的className设置为aaa bbb ccc
            // ms-attr-class="xxx" vm.xxx=false  清空元素的所有类名
            // ms-attr-name="yyy"  vm.yyy="ooo" 为元素设置name属性
            var toRemove = (val === false) || (val === null) || (val === void 0)
            if (!W3C && propMap[attrName]) { //旧式IE下需要进行名字映射
                attrName = propMap[attrName]
            }
            var bool = boolMap[attrName]
            if (typeof elem[bool] === "boolean") {
                elem[bool] = !!val //布尔属性必须使用el.xxx = true|false方式设值
                if (!val) { //如果为false, IE全系列下相当于setAttribute(xxx,''),会影响到样式,需要进一步处理
                    toRemove = true
                }
            }
            if (toRemove) {
                return elem.removeAttribute(attrName)
            }
            //SVG只能使用setAttribute(xxx, yyy), VML只能使用elem.xxx = yyy ,HTML的固有属性必须elem.xxx = yyy
            var isInnate = rsvg.test(elem) ? false : (DOC.namespaces && isVML(elem)) ? true : attrName in elem.cloneNode(false)
            if (isInnate) {
                elem[attrName] = val + ""
            } else {
                elem.setAttribute(attrName, val)
            }
        }
    }
})
//这几个指令都可以使用插值表达式,如ms-src="aaa/{{b}}/{{c}}.html"
"title,alt,src,value,css,include,href".replace(rword, function (name) {
    directives[name] = attrDir
})
//根据VM的属性值或表达式的值切换类名,ms-class="xxx yyy zzz:flag"
//http://www.cnblogs.com/rubylouvre/archive/2012/12/17/2818540.html
avalon.directive("class", {
    init: function (binding) {
        var oldStyle = binding.param
        var method = binding.type
        if (!oldStyle || isFinite(oldStyle)) {
            binding.param = "" //去掉数字
            directives.effect.init(binding)
        } else {
            log('ms-' + method + '-xxx="yyy"这种用法已经过时,请使用ms-' + method + '="xxx:yyy"')
            binding.expr = '[' + quote(oldStyle) + "," + binding.expr + "]"
            binding.oldStyle = oldStyle
        }
        if (method === "hover" || method === "active") { //确保只绑定一次
            if (!binding.hasBindEvent) {
                var elem = binding.element
                var $elem = avalon(elem)
                var activate = "mouseenter" //在移出移入时切换类名
                var abandon = "mouseleave"
                if (method === "active") { //在聚焦失焦中切换类名
                    elem.tabIndex = elem.tabIndex || -1
                    activate = "mousedown"
                    abandon = "mouseup"
                    var fn0 = $elem.bind("mouseleave", function () {
                        binding.toggleClass && $elem.removeClass(binding.newClass)
                    })
                }
            }
            var fn1 = $elem.bind(activate, function () {
                binding.toggleClass && $elem.addClass(binding.newClass)
            })
            var fn2 = $elem.bind(abandon, function () {
                binding.toggleClass && $elem.removeClass(binding.newClass)
            })
            binding.rollback = function () {
                $elem.unbind("mouseleave", fn0)
                $elem.unbind(activate, fn1)
                $elem.unbind(abandon, fn2)
            }
            binding.hasBindEvent = true
        }
    },
    update: function (arr) {
        var binding = this
        var $elem = avalon(this.element)
        binding.newClass = arr[0]
        binding.toggleClass = !!arr[1]
        if (binding.oldClass && binding.newClass !== binding.oldClass) {
            $elem.removeClass(binding.oldClass)
        }
        binding.oldClass = binding.newClass
        if (binding.type === "class") {
            if (binding.oldStyle) {
                $elem.toggleClass(binding.oldStyle, !!arr[1])
            } else {
                $elem.toggleClass(binding.newClass, binding.toggleClass)
            }
        }
    }
})
"hover,active".replace(rword, function (name) {
    directives[name] = directives["class"]
})
//ms-controller绑定已经在scanTag 方法中实现
avalon.directive("css", {
    init: directives.attr.init,
    update: function (val) {
        avalon(this.element).css(this.param, val)
    }
})
avalon.directive("data", {
    priority: 100,
    update: function (val) {
        var elem = this.element
        var key = "data-" + this.param
        if (val && typeof val === "object") {
            elem[key] = val
        } else {
            elem.setAttribute(key, String(val))
        }
    }
})
//双工绑定
var rduplexType = /^(?:checkbox|radio)$/
var rduplexParam = /^(?:radio|checked)$/
var rnoduplexInput = /^(file|button|reset|submit|checkbox|radio|range)$/
var duplexBinding = avalon.directive("duplex", {
    priority: 2000,
    init: function (binding, hasCast) {
        var elem = binding.element
        var vmodels = binding.vmodels
        binding.changed = getBindingCallback(elem, "data-duplex-changed", vmodels) || noop
        var params = []
        var casting = oneObject("string,number,boolean,checked")
        if (elem.type === "radio" && binding.param === "") {
            binding.param = "checked"
        }
        binding.param.replace(rw20g, function (name) {
            if (rduplexType.test(elem.type) && rduplexParam.test(name)) {
                if (name === "radio")
                    log("ms-duplex-radio已经更名为ms-duplex-checked")
                name = "checked"
                binding.isChecked = true
                binding.xtype = "radio"
            }
            if (name === "bool") {
                name = "boolean"
                log("ms-duplex-bool已经更名为ms-duplex-boolean")
            } else if (name === "text") {
                name = "string"
                log("ms-duplex-text已经更名为ms-duplex-string")
            }
            if (casting[name]) {
                hasCast = true
            }
            avalon.Array.ensure(params, name)
        })
        if (!hasCast) {
            params.push("string")
        }
        binding.param = params.join("-")
        if (!binding.xtype) {
            binding.xtype = elem.tagName === "SELECT" ? "select" :
                    elem.type === "checkbox" ? "checkbox" :
                    elem.type === "radio" ? "radio" :
                    /^change/.test(elem.getAttribute("data-duplex-event")) ? "change" :
                    "input"
        }
        //===================绑定事件======================
        var bound = binding.bound = function (type, callback) {
            elem.addEventListener(type, callback, false)
            var old = binding.rollback
            binding.rollback = function () {
                elem.avalonSetter = null
                avalon.unbind(elem, type, callback)
                old && old()
            }
        }
        function callback(value) {
            binding.changed.call(this, value, binding)
        }
        var composing = false
        function compositionStart() {
            composing = true
        }
        function compositionEnd() {
            composing = false
        }
        var updateVModel = function (e) {
            var val = elem.value //防止递归调用形成死循环
            if (composing || val === binding.oldValue || binding.pipe === null) //处理中文输入法在minlengh下引发的BUG
                return
            var lastValue = binding.pipe(val, binding, "get")
            binding.oldValue = val
            binding.setter(lastValue)
            callback.call(elem, lastValue)
        }
        switch (binding.xtype) {
            case "radio":
                bound("click", function () {
                    var lastValue = binding.pipe(elem.value, binding, "get")
                    binding.setter(lastValue)
                    callback.call(elem, lastValue)
                })
                break
            case "checkbox":
                bound("change", function () {
                    var method = elem.checked ? "ensure" : "remove"
                    var array = binding.getter.apply(0, binding.vmodels)
                    if (!Array.isArray(array)) {
                        log("ms-duplex应用于checkbox上要对应一个数组")
                        array = [array]
                    }
                    var val = binding.pipe(elem.value, binding, "get")
                    avalon.Array[method](array, val)
                    callback.call(elem, array)
                })
                break
            case "change":
                bound("change", updateVModel)
                break
            case "input":
                bound("input", updateVModel)
                bound("keyup", updateVModel)
                if (!IEVersion) {
                    bound("compositionstart", compositionStart)
                    bound("compositionend", compositionEnd)
                    bound("DOMAutoComplete", updateVModel)
                }
                break
            case "select":
                bound("change", function () {
                    var val = avalon(elem).val() //字符串或字符串数组
                    if (Array.isArray(val)) {
                        val = val.map(function (v) {
                            return binding.pipe(v, binding, "get")
                        })
                    } else {
                        val = binding.pipe(val, binding, "get")
                    }
                    if (val + "" !== binding.oldValue) {
                        try {
                            binding.setter(val)
                        } catch (ex) {
                            log(ex)
                        }
                    }
                })
                bound("datasetchanged", function (e) {
                    if (e.bubble === "selectDuplex") {
                        var value = binding._value
                        var curValue = Array.isArray(value) ? value.map(String) : value + ""
                        avalon(elem).val(curValue)
                        elem.oldValue = curValue + ""
                        callback.call(elem, curValue)
                    }
                })
                break
        }
        if (binding.xtype === "input" && !rnoduplexInput.test(elem.type)) {
            if (elem.type !== "hidden") {
                bound("focus", function () {
                    elem.msFocus = true
                })
                bound("blur", function () {
                    elem.msFocus = false
                })
            }
            elem.avalonSetter = updateVModel //#765
            watchValueInTimer(function () {
                if (root.contains(elem)) {
                    if (!elem.msFocus) {
                        updateVModel()
                    }
                } else if (!elem.msRetain) {
                    return false
                }
            })
        }
    },
    update: function (value) {
        var elem = this.element, binding = this, curValue
        if (!this.init) {
            for (var i in avalon.vmodels) {
                var v = avalon.vmodels[i]
                v.$fire("avalon-ms-duplex-init", binding)
            }
            var cpipe = binding.pipe || (binding.pipe = pipe)
            cpipe(null, binding, "init")
            this.init = 1
        }
        switch (this.xtype) {
            case "input":
            case "change":
                curValue = this.pipe(value, this, "set")  //fix #673
                if (curValue !== this.oldValue) {
                    var fixCaret = false
                    if (elem.msFocus) {
                        try {
                            var start = elem.selectionStart
                            var end = elem.selectionEnd
                            if (start === end) {
                                var pos = start
                                fixCaret = true
                            }
                        } catch (e) {
                        }
                    }
                    elem.value = this.oldValue = curValue
                    if (fixCaret && !elem.readOnly) {
                        elem.selectionStart = elem.selectionEnd = pos
                    }
                }
                break
            case "radio":
                curValue = binding.isChecked ? !!value : value + "" === elem.value
                elem.checked = curValue
                break
            case "checkbox":
                var array = [].concat(value) //强制转换为数组
                curValue = this.pipe(elem.value, this, "get")
                elem.checked = array.indexOf(curValue) > -1
                break
            case "select":
                //必须变成字符串后才能比较
                binding._value = value
                if (!elem.msHasEvent) {
                    elem.msHasEvent = "selectDuplex"
                    //必须等到其孩子准备好才触发
                } else {
                    avalon.fireDom(elem, "datasetchanged", {
                        bubble: elem.msHasEvent
                    })
                }
                break
        }
    }
})
function fixNull(val) {
    return val == null ? "" : val
}
avalon.duplexHooks = {
    checked: {
        get: function (val, binding) {
            return !binding.oldValue
        }
    },
    string: {
        get: function (val) { //同步到VM
            return val
        },
        set: fixNull
    },
    "boolean": {
        get: function (val) {
            return val === "true"
        },
        set: fixNull
    },
    number: {
        get: function (val, binding) {
            var number = parseFloat(val)
            if (-val === -number) {
                return number
            }
            var arr = /strong|medium|weak/.exec(binding.element.getAttribute("data-duplex-number")) || ["medium"]
            switch (arr[0]) {
                case "strong":
                    return 0
                case "medium":
                    return val === "" ? "" : 0
                case "weak":
                    return val
            }
        },
        set: fixNull
    }
}
function pipe(val, binding, action, e) {
    binding.param.replace(rw20g, function (name) {
        var hook = avalon.duplexHooks[name]
        if (hook && typeof hook[action] === "function") {
            val = hook[action](val, binding)
        }
    })
    return val
}
var TimerID, ribbon = []
avalon.tick = function (fn) {
    if (ribbon.push(fn) === 1) {
        TimerID = setInterval(ticker, 60)
    }
}
function ticker() {
    for (var n = ribbon.length - 1; n >= 0; n--) {
        var el = ribbon[n]
        if (el() === false) {
            ribbon.splice(n, 1)
        }
    }
    if (!ribbon.length) {
        clearInterval(TimerID)
    }
}
var watchValueInTimer = noop
new function () { // jshint ignore:line
    try { //#272 IE9-IE11, firefox
        var setters = {}
        var aproto = HTMLInputElement.prototype
        var bproto = HTMLTextAreaElement.prototype
        function newSetter(value) { // jshint ignore:line
            setters[this.tagName].call(this, value)
            if (!this.msFocus && this.avalonSetter) {
                this.avalonSetter()
            }
        }
        var inputProto = HTMLInputElement.prototype
        Object.getOwnPropertyNames(inputProto) //故意引发IE6-8等浏览器报错
        setters["INPUT"] = Object.getOwnPropertyDescriptor(aproto, "value").set
        Object.defineProperty(aproto, "value", {
            set: newSetter
        })
        setters["TEXTAREA"] = Object.getOwnPropertyDescriptor(bproto, "value").set
        Object.defineProperty(bproto, "value", {
            set: newSetter
        })
    } catch (e) {
        //在chrome 43中 ms-duplex终于不需要使用定时器实现双向绑定了
        // http://updates.html5rocks.com/2015/04/DOM-attributes-now-on-the-prototype
        // https://docs.google.com/document/d/1jwA8mtClwxI-QJuHT7872Z0pxpZz8PBkf2bGAbsUtqs/edit?pli=1
        watchValueInTimer = avalon.tick
    }
} // jshint ignore:line
avalon.directive("effect", {
    priority: 5,
    init: function (binding) {
        var text = binding.expr,
                className,
                rightExpr
        var colonIndex = text.replace(rexprg, function (a) {
            return a.replace(/./g, "0")
        }).indexOf(":") //取得第一个冒号的位置
        if (colonIndex === -1) { // 比如 ms-class/effect="aaa bbb ccc" 的情况
            className = text
            rightExpr = true
        } else { // 比如 ms-class/effect-1="ui-state-active:checked" 的情况
            className = text.slice(0, colonIndex)
            rightExpr = text.slice(colonIndex + 1)
        }
        if (!rexpr.test(text)) {
            className = quote(className)
        } else {
            className = normalizeExpr(className)
        }
        binding.expr = "[" + className + "," + rightExpr + "]"
    },
    update: function (arr) {
        var name = arr[0]
        var elem = this.element
        if (elem.getAttribute("data-effect-name") === name) {
            return
        } else {
            elem.removeAttribute("data-effect-driver")
        }
        var inlineStyles = elem.style
        var computedStyles = window.getComputedStyle ? window.getComputedStyle(elem) : null
        var useAni = false
        if (computedStyles && (supportTransition || supportAnimation)) {
            //如果支持CSS动画
            var duration = inlineStyles[transitionDuration] || computedStyles[transitionDuration]
            if (duration && duration !== '0s') {
                elem.setAttribute("data-effect-driver", "t")
                useAni = true
            }
            if (!useAni) {
                duration = inlineStyles[animationDuration] || computedStyles[animationDuration]
                if (duration && duration !== '0s') {
                    elem.setAttribute("data-effect-driver", "a")
                    useAni = true
                }
            }
        }
        if (!useAni) {
            if (avalon.effects[name]) {
                elem.setAttribute("data-effect-driver", "j")
                useAni = true
            }
        }
        if (useAni) {
            elem.setAttribute("data-effect-name", name)
        }
    }
})
avalon.effects = {}
avalon.effect = function (name, callbacks) {
    avalon.effects[name] = callbacks
}
var supportTransition = false
var supportAnimation = false
var transitionEndEvent
var animationEndEvent
var transitionDuration = avalon.cssName("transition-duration")
var animationDuration = avalon.cssName("animation-duration")
new function () {// jshint ignore:line
    var checker = {
        'TransitionEvent': 'transitionend',
        'WebKitTransitionEvent': 'webkitTransitionEnd',
        'OTransitionEvent': 'oTransitionEnd',
        'otransitionEvent': 'otransitionEnd'
    }
    var tran
    //有的浏览器同时支持私有实现与标准写法,比如webkit支持前两种,Opera支持1、3、4
    for (var name in checker) {
        if (window[name]) {
            tran = checker[name]
            break;
        }
        try {
            var a = document.createEvent(name);
            tran = checker[name]
            break;
        } catch (e) {
        }
    }
    if (typeof tran === "string") {
        supportTransition = true
        transitionEndEvent = tran
    }
    //大致上有两种选择
    //IE10+, Firefox 16+ & Opera 12.1+: animationend
    //Chrome/Safari: webkitAnimationEnd
    //http://blogs.msdn.com/b/davrous/archive/2011/12/06/introduction-to-css3-animat ions.aspx
    //IE10也可以使用MSAnimationEnd监听,但是回调里的事件 type依然为animationend
    //  el.addEventListener("MSAnimationEnd", function(e) {
    //     alert(e.type)// animationend!!!
    // })
    checker = {
        'AnimationEvent': 'animationend',
        'WebKitAnimationEvent': 'webkitAnimationEnd'
    }
    var ani;
    for (name in checker) {
        if (window[name]) {
            ani = checker[name];
            break;
        }
    }
    if (typeof ani === "string") {
        supportTransition = true
        animationEndEvent = ani
    }
}()
var effectPool = []//重复利用动画实例
function effectFactory(el, opts) {
    if (!el || el.nodeType !== 1) {
        return null
    }
    if (opts) {
        var name = opts.effectName
        var driver = opts.effectDriver
    } else {
        name = el.getAttribute("data-effect-name")
        driver = el.getAttribute("data-effect-driver")
    }
    if (!name || !driver) {
        return null
    }
    var instance = effectPool.pop() || new Effect()
    instance.el = el
    instance.driver = driver
    instance.useCss = driver !== "j"
    if (instance.useCss) {
        opts && avalon(el).addClass(opts.effectClass)
        instance.cssEvent = driver === "t" ? transitionEndEvent : animationEndEvent
    }
    instance.name = name
    instance.callbacks = avalon.effects[name] || {}
    return instance
}
function effectBinding(elem, binding) {
    var name = elem.getAttribute("data-effect-name")
    if (name) {
        binding.effectName = name
        binding.effectDriver = elem.getAttribute("data-effect-driver")
        var stagger = +elem.getAttribute("data-effect-stagger")
        binding.effectLeaveStagger = +elem.getAttribute("data-effect-leave-stagger") || stagger
        binding.effectEnterStagger = +elem.getAttribute("data-effect-enter-stagger") || stagger
        binding.effectClass = elem.className || NaN
    }
}
function upperFirstChar(str) {
    return str.replace(/^[\S]/g, function (m) {
        return m.toUpperCase()
    })
}
var effectBuffer = new Buffer()
function Effect() {
}// 动画实例,做成类的形式,是为了共用所有原型方法
Effect.prototype = {
    contrustor: Effect,
    enterClass: function () {
        return getEffectClass(this, "enter")
    },
    leaveClass: function () {
        return getEffectClass(this, "leave")
    },
    // 共享一个函数
    actionFun: function (name, before, after) {
        if (document.hidden) {
            return
        }
        var me = this
        var el = me.el
        var isLeave = name === "leave"
        name = isLeave ? "leave" : "enter"
        var oppositeName = isLeave ? "enter" : "leave"
        callEffectHook(me, "abort" + upperFirstChar(oppositeName))
        callEffectHook(me, "before" + upperFirstChar(name))
        if (!isLeave)
            before(el) //  这里可能做插入DOM树的操作,因此必须在修改类名前执行
        var cssCallback = function (cancel) {
            el.removeEventListener(me.cssEvent, me.cssCallback)
            if (isLeave) {
                before(el) //这里可能做移出DOM树操作,因此必须位于动画之后
                avalon(el).removeClass(me.cssClass)
            } else {
                if (me.driver === "a") {
                    avalon(el).removeClass(me.cssClass)
                }
            }
            if (cancel !== true) {
                callEffectHook(me, "after" + upperFirstChar(name))
                after && after(el)
            }
            me.dispose()
        }
        if (me.useCss) {
            if (me.cssCallback) { //如果leave动画还没有完成,立即完成
                me.cssCallback(true)
            }
            me.cssClass = getEffectClass(me, name)
            me.cssCallback = cssCallback
            me.update = function () {
                el.addEventListener(me.cssEvent, me.cssCallback)
                if (!isLeave && me.driver === "t") {//transtion延迟触发
                    avalon(el).removeClass(me.cssClass)
                }
            }
            avalon(el).addClass(me.cssClass)//animation会立即触发
            effectBuffer.render(true)
            effectBuffer.queue.push(me)
        } else {
            callEffectHook(me, name, cssCallback)
        }
    },
    enter: function (before, after) {
        this.actionFun.apply(this, ["enter"].concat(avalon.slice(arguments)))
    },
    leave: function (before, after) {
        this.actionFun.apply(this, ["leave"].concat(avalon.slice(arguments)))
    },
    dispose: function () {//销毁与回收到池子中
        this.update = this.cssCallback = null
        if (effectPool.unshift(this) > 100) {
            effectPool.pop()
        }
    }
}
function getEffectClass(instance, type) {
    var a = instance.callbacks[type + "Class"]
    if (typeof a === "string")
        return a
    if (typeof a === "function")
        return a()
    return instance.name + "-" + type
}
function callEffectHook(effect, name, cb) {
    var hook = effect.callbacks[name]
    if (hook) {
        hook.call(effect, effect.el, cb)
    }
}
var applyEffect = function (el, dir/*[before, [after, [opts]]]*/) {
    var args = aslice.call(arguments, 0)
    if (typeof args[2] !== "function") {
        args.splice(2, 0, noop)
    }
    if (typeof args[3] !== "function") {
        args.splice(3, 0, noop)
    }
    var before = args[2]
    var after = args[3]
    var opts = args[4]
    var effect = effectFactory(el, opts)
    if (!effect) {
        before()
        after()
        return false
    } else {
        var method = dir ? 'enter' : 'leave'
        effect[method](before, after)
    }
}
avalon.mix(avalon.effect, {
    apply: applyEffect,
    append: function (el, parent, after, opts) {
        return applyEffect(el, 1, function () {
            parent.appendChild(el)
        }, after, opts)
    },
    before: function (el, target, after, opts) {
        return applyEffect(el, 1, function () {
            target.parentNode.insertBefore(el, target)
        }, after, opts)
    },
    remove: function (el, parent, after, opts) {
        return applyEffect(el, 0, function () {
            if (el.parentNode === parent)
                parent.removeChild(el)
        }, after, opts)
    }
})
avalon.directive("html", {
    update: function (val) {
        var binding = this
        var elem = this.element
        var isHtmlFilter = elem.nodeType !== 1
        var parent = isHtmlFilter ? elem.parentNode : elem
        if (!parent)
            return
        val = val == null ? "" : val
        if (elem.nodeType === 3) {
            var signature = generateID("html")
            parent.insertBefore(DOC.createComment(signature), elem)
            binding.element = DOC.createComment(signature + ":end")
            parent.replaceChild(binding.element, elem)
            elem = binding.element
        }
        if (typeof val !== "object") {//string, number, boolean
            var fragment = avalon.parseHTML(String(val))
        } else if (val.nodeType === 11) { //将val转换为文档碎片
            fragment = val
        } else if (val.nodeType === 1 || val.item) {
            var nodes = val.nodeType === 1 ? val.childNodes : val.item
            fragment = avalonFragment.cloneNode(true)
            while (nodes[0]) {
                fragment.appendChild(nodes[0])
            }
        }
        nodes = avalon.slice(fragment.childNodes)
        //插入占位符, 如果是过滤器,需要有节制地移除指定的数量,如果是html指令,直接清空
        if (isHtmlFilter) {
            var endValue = elem.nodeValue.slice(0, -4)
            while (true) {
                var node = elem.previousSibling
                if (!node || node.nodeType === 8 && node.nodeValue === endValue) {
                    break
                } else {
                    parent.removeChild(node)
                }
            }
            parent.insertBefore(fragment, elem)
        } else {
            avalon.clearHTML(elem).appendChild(fragment)
        }
        scanNodeArray(nodes, binding.vmodels)
    }
})
avalon.directive("if", {
    priority: 10,
    update: function (val) {
        var binding = this
        var elem = this.element
        var stamp = binding.stamp = +new Date()
        var par
        var after = function () {
            if (stamp !== binding.stamp)
                return
            binding.recoverNode = null
        }
        if (binding.recoverNode)
            binding.recoverNode() // 还原现场,有移动节点的都需要还原现场
        try {
            if (!elem.parentNode)
                return
            par = elem.parentNode
        } catch (e) {
            return
        }
        if (val) { //插回DOM树
            function alway() {// jshint ignore:line
                if (elem.getAttribute(binding.name)) {
                    elem.removeAttribute(binding.name)
                    scanAttr(elem, binding.vmodels)
                }
                binding.rollback = null
            }
            if (elem.nodeType === 8) {
                var keep = binding.keep
                var hasEffect = avalon.effect.apply(keep, 1, function () {
                    if (stamp !== binding.stamp)
                        return
                    elem.parentNode.replaceChild(keep, elem)
                    elem = binding.element = keep //这时可能为null
                    if (keep.getAttribute("_required")) {//#1044
                        elem.required = true
                        elem.removeAttribute("_required")
                    }
                    if (elem.querySelectorAll) {
                        avalon.each(elem.querySelectorAll("[_required=true]"), function (el) {
                            el.required = true
                            el.removeAttribute("_required")
                        })
                    }
                    alway()
                }, after)
                hasEffect = hasEffect === false
            }
            if (!hasEffect)
                alway()
        } else { //移出DOM树,并用注释节点占据原位置
            if (elem.nodeType === 1) {
                if (elem.required === true) {
                    elem.required = false
                    elem.setAttribute("_required", "true")
                }
                try {// 如果不支持querySelectorAll或:required,可以直接无视
                    avalon.each(elem.querySelectorAll(":required"), function (el) {
                        elem.required = false
                        el.setAttribute("_required", "true")
                    })
                } catch (e) {
                }
                var node = binding.element = DOC.createComment("ms-if"),
                        pos = elem.nextSibling
                binding.recoverNode = function () {
                    binding.recoverNode = null
                    if (node.parentNode !== par) {
                        par.insertBefore(node, pos)
                        binding.keep = elem
                    }
                }
                avalon.effect.apply(elem, 0, function () {
                    binding.recoverNode = null
                    if (stamp !== binding.stamp)
                        return
                    elem.parentNode.replaceChild(node, elem)
                    binding.keep = elem //元素节点
                    ifGroup.appendChild(elem)
                    binding.rollback = function () {
                        if (elem.parentNode === ifGroup) {
                            ifGroup.removeChild(elem)
                        }
                    }
                }, after)
            }
        }
    }
})
//ms-important绑定已经在scanTag 方法中实现
var rnoscripts = /<noscript.*?>(?:[\s\S]+?)<\/noscript>/img
var rnoscriptText = /<noscript.*?>([\s\S]+?)<\/noscript>/im
var getXHR = function () {
    return new window.XMLHttpRequest() // jshint ignore:line
}
//将所有远程加载的模板,以字符串形式存放到这里
var templatePool = avalon.templateCache = {}
function getTemplateContainer(binding, id, text) {
    var div = binding.templateCache && binding.templateCache[id]
    if (div) {
        var dom = DOC.createDocumentFragment(),
                firstChild
        while (firstChild = div.firstChild) {
            dom.appendChild(firstChild)
        }
        return dom
    }
    return avalon.parseHTML(text)
}
function nodesToFrag(nodes) {
    var frag = DOC.createDocumentFragment()
    for (var i = 0, len = nodes.length; i < len; i++) {
        frag.appendChild(nodes[i])
    }
    return frag
}
avalon.directive("include", {
    init: directives.attr.init,
    update: function (val) {
        var binding = this
        var elem = this.element
        var vmodels = binding.vmodels
        var rendered = binding.includeRendered
        var effectClass = binding.effectName && binding.effectClass // 是否开启动画
        var templateCache = binding.templateCache // 是否data-include-cache
        var outer = binding.includeReplace // 是否data-include-replace
        var loaded = binding.includeLoaded
        var target = outer ? elem.parentNode : elem
        var _ele = binding._element // data-include-replace binding.element === binding.end
        binding.recoverNodes = binding.recoverNodes || avalon.noop
        var scanTemplate = function (text) {
            var _stamp = binding._stamp = +(new Date()) // 过滤掉频繁操作
            if (loaded) {
                var newText = loaded.apply(target, [text].concat(vmodels))
                if (typeof newText === "string")
                    text = newText
            }
            if (rendered) {
                checkScan(target, function () {
                    rendered.call(target)
                }, NaN)
            }
            var lastID = binding.includeLastID || "_default" // 默认
            binding.includeLastID = val
            var leaveEl = templateCache && templateCache[lastID] || DOC.createElement(elem.tagName || binding._element.tagName) // 创建一个离场元素
            if (effectClass) {
                leaveEl.className = effectClass
                target.insertBefore(leaveEl, binding.start) // 插入到start之前,防止被错误的移动
            }
            // cache or animate,移动节点
            (templateCache || {})[lastID] = leaveEl
            var fragOnDom = binding.recoverNodes() // 恢复动画中的节点
            if (fragOnDom) {
                target.insertBefore(fragOnDom, binding.end)
            }
            while (true) {
                var node = binding.start.nextSibling
                if (node && node !== leaveEl && node !== binding.end) {
                    leaveEl.appendChild(node)
                } else {
                    break
                }
            }
            // 元素退场
            avalon.effect.remove(leaveEl, target, function () {
                if (templateCache) { // write cache
                    if (_stamp === binding._stamp)
                        ifGroup.appendChild(leaveEl)
                }
            }, binding)
            var enterEl = target,
                    before = avalon.noop,
                    after = avalon.noop
            var fragment = getTemplateContainer(binding, val, text)
            var nodes = avalon.slice(fragment.childNodes)
            if (outer && effectClass) {
                enterEl = _ele
                enterEl.innerHTML = "" // 清空
                enterEl.setAttribute("ms-skip", "true")
                target.insertBefore(enterEl, binding.end.nextSibling) // 插入到bingding.end之后避免被错误的移动
                before = function () {
                    enterEl.insertBefore(fragment, null) // 插入节点
                }
                after = function () {
                    binding.recoverNodes = avalon.noop
                    if (_stamp === binding._stamp) {
                        fragment = nodesToFrag(nodes)
                        target.insertBefore(fragment, binding.end) // 插入真实element
                        scanNodeArray(nodes, vmodels)
                    }
                    if (enterEl.parentNode === target)
                        target.removeChild(enterEl) // 移除入场动画元素
                }
                binding.recoverNodes = function () {
                    binding.recoverNodes = avalon.noop
                    return nodesToFrag(nodes)
                }
            } else {
                before = function () {// 新添加元素的动画 
                    target.insertBefore(fragment, binding.end)
                    scanNodeArray(nodes, vmodels)
                }
            }
            avalon.effect.apply(enterEl, "enter", before, after)
        }
        if (binding.param === "src") {
            if (typeof templatePool[val] === "string") {
                avalon.nextTick(function () {
                    scanTemplate(templatePool[val])
                })
            } else if (Array.isArray(templatePool[val])) { //#805 防止在循环绑定中发出许多相同的请求
                templatePool[val].push(scanTemplate)
            } else {
                var xhr = getXHR()
                xhr.onload = function () {
                    var text = xhr.responseText
                    for (var f = 0, fn; fn = templatePool[val][f++]; ) {
                        fn(text)
                    }
                    templatePool[val] = text
                }
                xhr.onerror = function () {
                    log("ms-include load [" + val + "] error")
                }
                templatePool[val] = [scanTemplate]
                xhr.open("GET", val, true)
                if ("withCredentials" in xhr) {
                    xhr.withCredentials = true
                }
                xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest")
                xhr.send(null)
            }
        } else {
            //IE系列与够新的标准浏览器支持通过ID取得元素(firefox14+)
            //http://tjvantoll.com/2012/07/19/dom-element-references-as-global-variables/
            var el = val && val.nodeType === 1 ? val : DOC.getElementById(val)
            if (el) {
                avalon.nextTick(function () {
                    scanTemplate(el.value || el.innerText || el.innerHTML)
                })
            }
        }
    }
})
var rdash = /\(([^)]*)\)/
var onDir = avalon.directive("on", {
    priority: 3000,
    init: function (binding) {
        var value = binding.expr
        binding.type = "on"
        var eventType = binding.param.replace(/-\d+$/, "") // ms-on-mousemove-10
        if (typeof onDir[eventType + "Hook"] === "function") {
            onDir[eventType + "Hook"](binding)
        }
        if (value.indexOf("(") > 0 && value.indexOf(")") > -1) {
            var matched = (value.match(rdash) || ["", ""])[1].trim()
            if (matched === "" || matched === "$event") { // aaa() aaa($event)当成aaa处理
                value = value.replace(rdash, "")
            }
        }
        binding.expr = value
    },
    update: function (callback) {
        var binding = this
        var elem = this.element
        callback = function (e) {
            var fn = binding.getter || noop
            return fn.apply(this, binding.args.concat(e))
        }
        
        var eventType = binding.param.replace(/-\d+$/, "") // ms-on-mousemove-10
        if (eventType === "scan") {
            callback.call(elem, {
                type: eventType
            })
        } else if (typeof binding.specialBind === "function") {
            binding.specialBind(elem, callback)
        } else {
            var removeFn = avalon.bind(elem, eventType, callback)
        }
        binding.rollback = function () {
            if (typeof binding.specialUnbind === "function") {
                binding.specialUnbind()
            } else {
                avalon.unbind(elem, eventType, removeFn)
            }
        }
    }
})
avalon.directive("repeat", {
    priority: 90,
    init: function (binding) {
        var type = binding.type
        binding.cache = {} //用于存放代理VM
        binding.enterCount = 0
        var elem = binding.element
        if (elem.nodeType === 1) {
            elem.removeAttribute(binding.name)
            effectBinding(elem, binding)
            binding.param = binding.param || "el"
            binding.sortedCallback = getBindingCallback(elem, "data-with-sorted", binding.vmodels)
            var rendered = getBindingCallback(elem, "data-" + type + "-rendered", binding.vmodels)
            var signature = generateID(type)
            var start = DOC.createComment(signature + ":start")
            var end = binding.element = DOC.createComment(signature + ":end")
            binding.signature = signature
            binding.start = start
            binding.template = avalonFragment.cloneNode(false)
            if (type === "repeat") {
                var parent = elem.parentNode
                parent.replaceChild(end, elem)
                parent.insertBefore(start, end)
                binding.template.appendChild(elem)
            } else {
                while (elem.firstChild) {
                    binding.template.appendChild(elem.firstChild)
                }
                elem.appendChild(start)
                elem.appendChild(end)
                parent = elem
            }
            binding.element = end
            if (rendered) {
                var removeFn = avalon.bind(parent, "datasetchanged", function () {
                    rendered.apply(parent, parent.args)
                    avalon.unbind(parent, "datasetchanged", removeFn)
                    parent.msRendered = rendered
                })
            }
        }
    },
    update: function (value, oldValue) {
        var binding = this
        var xtype = this.xtype
        this.enterCount += 1
        var init = !oldValue
        if (init) {
            binding.$outer = {}
            var check0 = "$key"
            var check1 = "$val"
            if (xtype === "array") {
                check0 = "$first"
                check1 = "$last"
            }
            for (var i = 0, v; v = binding.vmodels[i++]; ) {
                if (v.hasOwnProperty(check0) && v.hasOwnProperty(check1)) {
                    binding.$outer = v
                    break
                }
            }
        }
        var track = this.track
        if (binding.sortedCallback) { //如果有回调,则让它们排序
            var keys2 = binding.sortedCallback.call(parent, track)
            if (keys2 && Array.isArray(keys2)) {
                track = keys2
            }
        }
        var action = "move"
        binding.$repeat = value
        var fragments = []
        var transation = init && avalonFragment.cloneNode(false)
        var proxies = []
        var param = this.param
        var retain = avalon.mix({}, this.cache)
        var elem = this.element
        var length = track.length
        var parent = elem.parentNode
        for (i = 0; i < length; i++) {
            var keyOrId = track[i] //array为随机数, object 为keyName
            var proxy = retain[keyOrId]
            if (!proxy) {
                
                proxy = getProxyVM(this)
                proxy.$up = null
                if (xtype === "array") {
                    action = "add"
                    proxy.$id = keyOrId
                    var valueItem = value[i]
                    proxy[param] = valueItem //index
                    if(Object(valueItem) === valueItem){
                        valueItem.$ups = valueItem.$ups || {}
                        valueItem.$ups[param] = proxy
                    }
                } else {
                    action = "append"
                    proxy.$key = keyOrId
                    proxy.$val = value[keyOrId] //key
                }
                this.cache[keyOrId] = proxy
                var node = proxy.$anchor || (proxy.$anchor = elem.cloneNode(false))
                node.nodeValue = this.signature
                shimController(binding, transation, proxy, fragments, init && !binding.effectDriver)
                decorateProxy(proxy, binding, xtype)
            } else {
//                if (xtype === "array") {
//                    proxy[param] = value[i]
//                }
                fragments.push({})
                retain[keyOrId] = true
            }
            //重写proxy
            if (this.enterCount === 1) {// 防止多次进入,导致位置不对
                proxy.$active = false
                proxy.$oldIndex = proxy.$index
                proxy.$active = true
                proxy.$index = i
            }
            if (xtype === "array") {
                proxy.$first = i === 0
                proxy.$last = i === length - 1
                // proxy[param] = value[i]
            } else {
                proxy.$val = toJson(value[keyOrId]) // 这里是处理vm.object = newObject的情况 
            }
            proxies.push(proxy)
        }
        this.proxies = proxies
        if (init && !binding.effectDriver) {
            parent.insertBefore(transation, elem)
            fragments.forEach(function (fragment) {
                scanNodeArray(fragment.nodes || [], fragment.vmodels)
                //if(fragment.vmodels.length > 2)
                fragment.nodes = fragment.vmodels = null
            })// jshint ignore:line
        } else {
            var staggerIndex = binding.staggerIndex = 0
            for (keyOrId in retain) {
                if (retain[keyOrId] !== true) {
                    action = "del"
                    removeItem(retain[keyOrId].$anchor, binding)
                    // avalon.log("删除", keyOrId)
                    // 相当于delete binding.cache[key]
                    proxyRecycler(this.cache, keyOrId, param)
                    retain[keyOrId] = null
                }
            }
            //  console.log(effectEnterStagger)
            for (i = 0; i < length; i++) {
                proxy = proxies[i]
                keyOrId = xtype === "array" ? proxy.$id : proxy.$key
                var pre = proxies[i - 1]
                var preEl = pre ? pre.$anchor : binding.start
                if (!retain[keyOrId]) {//如果还没有插入到DOM树
                    (function (fragment, preElement) {
                        var nodes = fragment.nodes
                        var vmodels = fragment.vmodels
                        if (nodes) {
                            staggerIndex = mayStaggerAnimate(binding.effectEnterStagger, function () {
                                parent.insertBefore(fragment.content, preElement.nextSibling)
                                scanNodeArray(nodes, vmodels)
                                animateRepeat(nodes, 1, binding)
                            }, staggerIndex)
                        }
                        fragment.nodes = fragment.vmodels = null
                    })(fragments[i], preEl)// jshint ignore:line
                    // avalon.log("插入")
                } else if (proxy.$index !== proxy.$oldIndex) {
                    (function (proxy2, preElement) {
                        staggerIndex = mayStaggerAnimate(binding.effectEnterStagger, function () {
                            var curNode = removeItem(proxy2.$anchor)// 如果位置被挪动了
                            var inserted = avalon.slice(curNode.childNodes)
                            parent.insertBefore(curNode, preElement.nextSibling)
                            animateRepeat(inserted, 1, binding)
                        }, staggerIndex)
                    })(proxy, preEl)// jshint ignore:line
                    // avalon.log("移动", proxy.$oldIndex, "-->", proxy.$index)
                }
            }
        }
        if (!value.$track) {//如果是非监控对象,那么就将其$events清空,阻止其持续监听
            for (keyOrId in this.cache) {
                proxyRecycler(this.cache, keyOrId, param)
            }
        }
        //repeat --> duplex
        (function (args) {
            parent.args = args
            if (parent.msRendered) {//第一次事件触发,以后直接调用
                parent.msRendered.apply(parent, args)
            }
        })(kernel.newWatch ? arguments : [action]);
        var id = setTimeout(function () {
            clearTimeout(id)
            //触发上层的select回调及自己的rendered回调
            avalon.fireDom(parent, "datasetchanged", {
                bubble: parent.msHasEvent
            })
        })
        this.enterCount -= 1
    }
})
"with,each".replace(rword, function (name) {
    directives[name] = avalon.mix({}, directives.repeat, {
        priority: 1400
    })
})
function animateRepeat(nodes, isEnter, binding) {
    for (var i = 0, node; node = nodes[i++]; ) {
        if (node.className === binding.effectClass) {
            avalon.effect.apply(node, isEnter, noop, noop, binding)
        }
    }
}
function mayStaggerAnimate(staggerTime, callback, index) {
    if (staggerTime) {
        setTimeout(callback, (++index) * staggerTime)
    } else {
        callback()
    }
    return index
}
function removeItem(node, binding) {
    var fragment = avalonFragment.cloneNode(false)
    var last = node
    var breakText = last.nodeValue
    var staggerIndex = binding && Math.max(+binding.staggerIndex, 0)
    var nodes = avalon.slice(last.parentNode.childNodes)
    var index = nodes.indexOf(last)
    while (true) {
        var pre = nodes[--index] //node.previousSibling
        if (!pre || String(pre.nodeValue).indexOf(breakText) === 0) {
            break
        }
        if (binding && (pre.className === binding.effectClass)) {
            node = pre;
            (function (cur) {
                binding.staggerIndex = mayStaggerAnimate(binding.effectLeaveStagger, function () {
                    avalon.effect.apply(cur, 0, noop, function () {
                        fragment.appendChild(cur)
                    }, binding)
                }, staggerIndex)
            })(pre);// jshint ignore:line
        } else {
            fragment.insertBefore(pre, fragment.firstChild)
        }
    }
    fragment.appendChild(last)
    return fragment
}
function shimController(data, transation, proxy, fragments, init) {
    var content = data.template.cloneNode(true)
    var nodes = avalon.slice(content.childNodes)
    content.appendChild(proxy.$anchor)
    init && transation.appendChild(content)
    var nv = [proxy].concat(data.vmodels)
    var fragment = {
        nodes: nodes,
        vmodels: nv,
        content: content
    }
    fragments.push(fragment)
}
// {}  -->  {xx: 0, yy: 1, zz: 2} add
// {xx: 0, yy: 1, zz: 2}  -->  {xx: 0, yy: 1, zz: 2, uu: 3}
// [xx: 0, yy: 1, zz: 2}  -->  {xx: 0, zz: 1, yy: 2}
function getProxyVM(binding) {
    var agent = binding.xtype === "object" ? withProxyAgent : eachProxyAgent
    var proxy = agent(binding)
    var node = proxy.$anchor || (proxy.$anchor = binding.element.cloneNode(false))
    node.nodeValue = binding.signature
    proxy.$outer = binding.$outer
    return proxy
}
var eachProxyPool = []
function eachProxyAgent(data, proxy) {
    var itemName = data.param || "el"
    for (var i = 0, n = eachProxyPool.length; i < n; i++) {
        var candidate = eachProxyPool[i]
        if (candidate && candidate.hasOwnProperty(itemName)) {
            eachProxyPool.splice(i, 1)
            proxy = candidate
            break
        }
    }
    if (!proxy) {
        proxy = eachProxyFactory(itemName)
    }
    return proxy
}
function eachProxyFactory(itemName) {
    var source = {
        $outer: {},
        $index: 0,
        $oldIndex: 0,
        $anchor: null,
        //-----
        $first: false,
        $last: false,
        $remove: avalon.noop
    }
    source[itemName] = NaN
    var force = {
        $last: 1,
        $first: 1,
        $index: 1
    }
    force[itemName] = 1
    var proxy = modelFactory(source, {
        force: force
    })
    proxy.$id = generateID("$proxy$each")
    return proxy
}
function decorateProxy(proxy, binding, type) {
    if (type === "array") {
        proxy.$remove = function () {
            binding.$repeat.removeAt(proxy.$index)
        }
        var param = binding.param
        proxy.$watch(param, function (a) {
            var index = proxy.$index
            binding.$repeat[index] = a
        })
    } else {
        proxy.$watch("$val", function fn(a) {
            binding.$repeat[proxy.$key] = a
        })
    }
}
var withProxyPool = []
function withProxyAgent() {
    return withProxyPool.pop() || withProxyFactory()
}
function withProxyFactory() {
    var proxy = modelFactory({
        $key: "",
        $val: NaN,
        $index: 0,
        $oldIndex: 0,
        $outer: {},
        $anchor: null
    }, {
        force: {
            $key: 1,
            $val: 1,
            $index: 1
        }
    })
    proxy.$id = generateID("$proxy$with")
    return proxy
}
function proxyRecycler(cache, key, param) {
    var proxy = cache[key]
    if (proxy) {
        var proxyPool = proxy.$id.indexOf("$proxy$each") === 0 ? eachProxyPool : withProxyPool
        proxy.$outer = {}
        for (var i in proxy.$events) {
            var a = proxy.$events[i]
            if (Array.isArray(a)) {
                a.length = 0
                if (i === param) {
                    proxy[param] = NaN
                } else if (i === "$val") {
                    proxy.$val = NaN
                }
            }
        }
        if (proxyPool.unshift(proxy) > kernel.maxRepeatSize) {
            proxyPool.pop()
        }
        delete cache[key]
    }
}
/*********************************************************************
 *                         各种指令                                  *
 **********************************************************************/
//ms-skip绑定已经在scanTag 方法中实现
avalon.directive("text", {
    update: function (val) {
        var elem = this.element
        val = val == null ? "" : val //不在页面上显示undefined null
        if (elem.nodeType === 3) { //绑定在文本节点上
            try { //IE对游离于DOM树外的节点赋值会报错
                elem.data = val
            } catch (e) {
            }
        } else { //绑定在特性节点上
            elem.textContent = val
        }
    }
})
function parseDisplay(nodeName, val) {
    //用于取得此类标签的默认display值
    var key = "_" + nodeName
    if (!parseDisplay[key]) {
        var node = DOC.createElement(nodeName)
        root.appendChild(node)
        if (W3C) {
            val = getComputedStyle(node, null).display
        } else {
            val = node.currentStyle.display
        }
        root.removeChild(node)
        parseDisplay[key] = val
    }
    return parseDisplay[key]
}
avalon.parseDisplay = parseDisplay
avalon.directive("visible", {
    init: function (binding) {
        effectBinding(binding.element, binding)
    },
    update: function (val) {
        var binding = this, elem = this.element, stamp
        var noEffect = !this.effectName
        if (!this.stamp) {
            stamp = this.stamp = +new Date
            if (val) {
                elem.style.display = binding.display || ""
                if (avalon(elem).css("display") === "none") {
                    elem.style.display = binding.display = parseDisplay(elem.nodeName)
                }
            } else {
                elem.style.display = "none"
            }
            return
        }
        stamp = this.stamp = +new Date
        if (val) {
            avalon.effect.apply(elem, 1, function () {
                if (stamp !== binding.stamp)
                    return
                var driver = elem.getAttribute("data-effect-driver") || "a"
                if (noEffect) {//不用动画时走这里
                    elem.style.display = binding.display || ""
                }
                // "a", "t"
                if (driver === "a" || driver === "t") {
                    if (avalon(elem).css("display") === "none") {
                        elem.style.display = binding.display || parseDisplay(elem.nodeName)
                    }
                }
            })
        } else {
            avalon.effect.apply(elem, 0, function () {
                if (stamp !== binding.stamp)
                    return
                elem.style.display = "none"
            })
        }
    }
})
/*********************************************************************
 *                             自带过滤器                            *
 **********************************************************************/
var rscripts = /<script[^>]*>([\S\s]*?)<\/script\s*>/gim
var ron = /\s+(on[^=\s]+)(?:=("[^"]*"|'[^']*'|[^\s>]+))?/g
var ropen = /<\w+\b(?:(["'])[^"]*?(\1)|[^>])*>/ig
var rsanitize = {
    a: /\b(href)\=("javascript[^"]*"|'javascript[^']*')/ig,
    img: /\b(src)\=("javascript[^"]*"|'javascript[^']*')/ig,
    form: /\b(action)\=("javascript[^"]*"|'javascript[^']*')/ig
}
var rsurrogate = /[\uD800-\uDBFF][\uDC00-\uDFFF]/g
var rnoalphanumeric = /([^\#-~| |!])/g;
function numberFormat(number, decimals, point, thousands) {
    //form http://phpjs.org/functions/number_format/
    //number	必需,要格式化的数字
    //decimals	可选,规定多少个小数位。
    //point	可选,规定用作小数点的字符串(默认为 . )。
    //thousands	可选,规定用作千位分隔符的字符串(默认为 , ),如果设置了该参数,那么所有其他参数都是必需的。
    number = (number + '')
            .replace(/[^0-9+\-Ee.]/g, '')
    var n = !isFinite(+number) ? 0 : +number,
            prec = !isFinite(+decimals) ? 3 : Math.abs(decimals),
            sep = thousands || ",",
            dec = point || ".",
            s = '',
            toFixedFix = function(n, prec) {
                var k = Math.pow(10, prec)
                return '' + (Math.round(n * k) / k)
                        .toFixed(prec)
            }
    // Fix for IE parseFloat(0.55).toFixed(0) = 0;
    s = (prec ? toFixedFix(n, prec) : '' + Math.round(n))
            .split('.')
    if (s[0].length > 3) {
        s[0] = s[0].replace(/\B(?=(?:\d{3})+(?!\d))/g, sep)
    }
    if ((s[1] || '')
            .length < prec) {
        s[1] = s[1] || ''
        s[1] += new Array(prec - s[1].length + 1)
                .join('0')
    }
    return s.join(dec)
}
var filters = avalon.filters = {
    uppercase: function(str) {
        return str.toUpperCase()
    },
    lowercase: function(str) {
        return str.toLowerCase()
    },
    truncate: function(str, length, truncation) {
        //length,新字符串长度,truncation,新字符串的结尾的字段,返回新字符串
        length = length || 30
        truncation = typeof truncation === "string" ?  truncation : "..." 
        return str.length > length ? str.slice(0, length - truncation.length) + truncation : String(str)
    },
    $filter: function(val) {
        for (var i = 1, n = arguments.length; i < n; i++) {
            var array = arguments[i]
            var fn = avalon.filters[array[0]]
            if (typeof fn === "function") {
                var arr = [val].concat(array.slice(1))
                val = fn.apply(null, arr)
            }
        }
        return val
    },
    camelize: camelize,
    //https://www.owasp.org/index.php/XSS_Filter_Evasion_Cheat_Sheet
    //    <a href="javasc&NewLine;ript&colon;alert('XSS')">chrome</a> 
    //    <a href="data:text/html;base64, PGltZyBzcmM9eCBvbmVycm9yPWFsZXJ0KDEpPg==">chrome</a>
    //    <a href="jav	ascript:alert('XSS');">IE67chrome</a>
    //    <a href="jav&#x09;ascript:alert('XSS');">IE67chrome</a>
    //    <a href="jav&#x0A;ascript:alert('XSS');">IE67chrome</a>
    sanitize: function(str) {
        return str.replace(rscripts, "").replace(ropen, function(a, b) {
            var match = a.toLowerCase().match(/<(\w+)\s/)
            if (match) { //处理a标签的href属性,img标签的src属性,form标签的action属性
                var reg = rsanitize[match[1]]
                if (reg) {
                    a = a.replace(reg, function(s, name, value) {
                        var quote = value.charAt(0)
                        return name + "=" + quote + "javascript:void(0)" + quote// jshint ignore:line
                    })
                }
            }
            return a.replace(ron, " ").replace(/\s+/g, " ") //移除onXXX事件
        })
    },
    escape: function(str) {
        //将字符串经过 str 转义得到适合在页面中显示的内容, 例如替换 < 为 &lt 
        return String(str).
                replace(/&/g, '&amp;').
                replace(rsurrogate, function(value) {
                    var hi = value.charCodeAt(0)
                    var low = value.charCodeAt(1)
                    return '&#' + (((hi - 0xD800) * 0x400) + (low - 0xDC00) + 0x10000) + ';'
                }).
                replace(rnoalphanumeric, function(value) {
                    return '&#' + value.charCodeAt(0) + ';'
                }).
                replace(/</g, '&lt;').
                replace(/>/g, '&gt;')
    },
    currency: function(amount, symbol, fractionSize) {
        return (symbol || "\uFFE5") + numberFormat(amount, isFinite(fractionSize) ? fractionSize : 2)
    },
    number: numberFormat
}
/*
 'yyyy': 4 digit representation of year (e.g. AD 1 => 0001, AD 2010 => 2010)
 'yy': 2 digit representation of year, padded (00-99). (e.g. AD 2001 => 01, AD 2010 => 10)
 'y': 1 digit representation of year, e.g. (AD 1 => 1, AD 199 => 199)
 'MMMM': Month in year (January-December)
 'MMM': Month in year (Jan-Dec)
 'MM': Month in year, padded (01-12)
 'M': Month in year (1-12)
 'dd': Day in month, padded (01-31)
 'd': Day in month (1-31)
 'EEEE': Day in Week,(Sunday-Saturday)
 'EEE': Day in Week, (Sun-Sat)
 'HH': Hour in day, padded (00-23)
 'H': Hour in day (0-23)
 'hh': Hour in am/pm, padded (01-12)
 'h': Hour in am/pm, (1-12)
 'mm': Minute in hour, padded (00-59)
 'm': Minute in hour (0-59)
 'ss': Second in minute, padded (00-59)
 's': Second in minute (0-59)
 'a': am/pm marker
 'Z': 4 digit (+sign) representation of the timezone offset (-1200-+1200)
 format string can also be one of the following predefined localizable formats:
 
 'medium': equivalent to 'MMM d, y h:mm:ss a' for en_US locale (e.g. Sep 3, 2010 12:05:08 pm)
 'short': equivalent to 'M/d/yy h:mm a' for en_US locale (e.g. 9/3/10 12:05 pm)
 'fullDate': equivalent to 'EEEE, MMMM d,y' for en_US locale (e.g. Friday, September 3, 2010)
 'longDate': equivalent to 'MMMM d, y' for en_US locale (e.g. September 3, 2010
 'mediumDate': equivalent to 'MMM d, y' for en_US locale (e.g. Sep 3, 2010)
 'shortDate': equivalent to 'M/d/yy' for en_US locale (e.g. 9/3/10)
 'mediumTime': equivalent to 'h:mm:ss a' for en_US locale (e.g. 12:05:08 pm)
 'shortTime': equivalent to 'h:mm a' for en_US locale (e.g. 12:05 pm)
 */
new function() {// jshint ignore:line
    function toInt(str) {
        return parseInt(str, 10) || 0
    }
    function padNumber(num, digits, trim) {
        var neg = ""
        if (num < 0) {
            neg = '-'
            num = -num
        }
        num = "" + num
        while (num.length < digits)
            num = "0" + num
        if (trim)
            num = num.substr(num.length - digits)
        return neg + num
    }
    function dateGetter(name, size, offset, trim) {
        return function(date) {
            var value = date["get" + name]()
            if (offset > 0 || value > -offset)
                value += offset
            if (value === 0 && offset === -12) {
                value = 12
            }
            return padNumber(value, size, trim)
        }
    }
    function dateStrGetter(name, shortForm) {
        return function(date, formats) {
            var value = date["get" + name]()
            var get = (shortForm ? ("SHORT" + name) : name).toUpperCase()
            return formats[get][value]
        }
    }
    function timeZoneGetter(date) {
        var zone = -1 * date.getTimezoneOffset()
        var paddedZone = (zone >= 0) ? "+" : ""
        paddedZone += padNumber(Math[zone > 0 ? "floor" : "ceil"](zone / 60), 2) + padNumber(Math.abs(zone % 60), 2)
        return paddedZone
    }
    //取得上午下午
    function ampmGetter(date, formats) {
        return date.getHours() < 12 ? formats.AMPMS[0] : formats.AMPMS[1]
    }
    var DATE_FORMATS = {
        yyyy: dateGetter("FullYear", 4),
        yy: dateGetter("FullYear", 2, 0, true),
        y: dateGetter("FullYear", 1),
        MMMM: dateStrGetter("Month"),
        MMM: dateStrGetter("Month", true),
        MM: dateGetter("Month", 2, 1),
        M: dateGetter("Month", 1, 1),
        dd: dateGetter("Date", 2),
        d: dateGetter("Date", 1),
        HH: dateGetter("Hours", 2),
        H: dateGetter("Hours", 1),
        hh: dateGetter("Hours", 2, -12),
        h: dateGetter("Hours", 1, -12),
        mm: dateGetter("Minutes", 2),
        m: dateGetter("Minutes", 1),
        ss: dateGetter("Seconds", 2),
        s: dateGetter("Seconds", 1),
        sss: dateGetter("Milliseconds", 3),
        EEEE: dateStrGetter("Day"),
        EEE: dateStrGetter("Day", true),
        a: ampmGetter,
        Z: timeZoneGetter
    }
    var rdateFormat = /((?:[^yMdHhmsaZE']+)|(?:'(?:[^']|'')*')|(?:E+|y+|M+|d+|H+|h+|m+|s+|a|Z))(.*)/
    var raspnetjson = /^\/Date\((\d+)\)\/$/
    filters.date = function(date, format) {
        var locate = filters.date.locate,
                text = "",
                parts = [],
                fn, match
        format = format || "mediumDate"
        format = locate[format] || format
        if (typeof date === "string") {
            if (/^\d+$/.test(date)) {
                date = toInt(date)
            } else if (raspnetjson.test(date)) {
                date = +RegExp.$1
            } else {
                var trimDate = date.trim()
                var dateArray = [0, 0, 0, 0, 0, 0, 0]
                var oDate = new Date(0)
                //取得年月日
                trimDate = trimDate.replace(/^(\d+)\D(\d+)\D(\d+)/, function(_, a, b, c) {
                    var array = c.length === 4 ? [c, a, b] : [a, b, c]
                    dateArray[0] = toInt(array[0])     //年
                    dateArray[1] = toInt(array[1]) - 1 //月
                    dateArray[2] = toInt(array[2])     //日
                    return ""
                })
                var dateSetter = oDate.setFullYear
                var timeSetter = oDate.setHours
                trimDate = trimDate.replace(/[T\s](\d+):(\d+):?(\d+)?\.?(\d)?/, function(_, a, b, c, d) {
                    dateArray[3] = toInt(a) //小时
                    dateArray[4] = toInt(b) //分钟
                    dateArray[5] = toInt(c) //秒
                    if (d) {                //毫秒
                        dateArray[6] = Math.round(parseFloat("0." + d) * 1000)
                    }
                    return ""
                })
                var tzHour = 0
                var tzMin = 0
                trimDate = trimDate.replace(/Z|([+-])(\d\d):?(\d\d)/, function(z, symbol, c, d) {
                    dateSetter = oDate.setUTCFullYear
                    timeSetter = oDate.setUTCHours
                    if (symbol) {
                        tzHour = toInt(symbol + c)
                        tzMin = toInt(symbol + d)
                    }
                    return ""
                })
                dateArray[3] -= tzHour
                dateArray[4] -= tzMin
                dateSetter.apply(oDate, dateArray.slice(0, 3))
                timeSetter.apply(oDate, dateArray.slice(3))
                date = oDate
            }
        }
        if (typeof date === "number") {
            date = new Date(date)
        }
        if (avalon.type(date) !== "date") {
            return
        }
        while (format) {
            match = rdateFormat.exec(format)
            if (match) {
                parts = parts.concat(match.slice(1))
                format = parts.pop()
            } else {
                parts.push(format)
                format = null
            }
        }
        parts.forEach(function(value) {
            fn = DATE_FORMATS[value]
            text += fn ? fn(date, locate) : value.replace(/(^'|'$)/g, "").replace(/''/g, "'")
        })
        return text
    }
    var locate = {
        AMPMS: {
            0: "上午",
            1: "下午"
        },
        DAY: {
            0: "星期日",
            1: "星期一",
            2: "星期二",
            3: "星期三",
            4: "星期四",
            5: "星期五",
            6: "星期六"
        },
        MONTH: {
            0: "1月",
            1: "2月",
            2: "3月",
            3: "4月",
            4: "5月",
            5: "6月",
            6: "7月",
            7: "8月",
            8: "9月",
            9: "10月",
            10: "11月",
            11: "12月"
        },
        SHORTDAY: {
            "0": "周日",
            "1": "周一",
            "2": "周二",
            "3": "周三",
            "4": "周四",
            "5": "周五",
            "6": "周六"
        },
        fullDate: "y年M月d日EEEE",
        longDate: "y年M月d日",
        medium: "yyyy-M-d H:mm:ss",
        mediumDate: "yyyy-M-d",
        mediumTime: "H:mm:ss",
        "short": "yy-M-d ah:mm",
        shortDate: "yy-M-d",
        shortTime: "ah:mm"
    }
    locate.SHORTMONTH = locate.MONTH
    filters.date.locate = locate
}// jshint ignore:line
/*********************************************************************
 *                      AMD加载器                                   *
 **********************************************************************/
//https://www.devbridge.com/articles/understanding-amd-requirejs/
//http://maxogden.com/nested-dependencies.html
var modules = avalon.modules = {
    "domReady!": {
        exports: avalon,
        state: 3
    },
    "avalon": {
        exports: avalon,
        state: 4
    }
}
//Object(modules[id]).state拥有如下值 
// undefined  没有定义
// 1(send)    已经发出请求
// 2(loading) 已经被执行但还没有执行完成,在这个阶段define方法会被执行
// 3(loaded)  执行完毕,通过onload/onreadystatechange回调判定,在这个阶段checkDeps方法会执行
// 4(execute)  其依赖也执行完毕, 值放到exports对象上,在这个阶段fireFactory方法会执行
modules.exports = modules.avalon
var otherRequire = window.require
var otherDefine = window.define
var innerRequire
plugins.loader = function (builtin) {
    var flag = innerRequire && builtin
    window.require = flag ? innerRequire : otherRequire
    window.define = flag ? innerRequire.define : otherDefine
}
new function () {// jshint ignore:line
    var loadings = [] //正在加载中的模块列表
    var factorys = [] //放置define方法的factory函数
    var rjsext = /\.js$/i
    function makeRequest(name, config) {
//1. 去掉资源前缀
        var res = "js"
        name = name.replace(/^(\w+)\!/, function (a, b) {
            res = b
            return ""
        })
        if (res === "ready") {
            log("debug: ready!已经被废弃,请使用domReady!")
            res = "domReady"
        }
//2. 去掉querystring, hash
        var query = ""
        name = name.replace(rquery, function (a) {
            query = a
            return ""
        })
        //3. 去掉扩展名
        var suffix = "." + res
        var ext = /js|css/.test(suffix) ? suffix : ""
        name = name.replace(/\.[a-z0-9]+$/g, function (a) {
            if (a === suffix) {
                ext = a
                return ""
            } else {
                return a
            }
        })
        var req = avalon.mix({
            query: query,
            ext: ext,
            res: res,
            name: name,
            toUrl: toUrl
        }, config)
        req.toUrl(name)
        return req
    }
    function fireRequest(req) {
        var name = req.name
        var res = req.res
        //1. 如果该模块已经发出请求,直接返回
        var module = modules[name]
        var urlNoQuery = name && req.urlNoQuery
        if (module && module.state >= 1) {
            return name
        }
        module = modules[urlNoQuery]
        if (module && module.state >= 3) {
            innerRequire(module.deps || [], module.factory, urlNoQuery)
            return urlNoQuery
        }
        if (name && !module) {
            module = modules[urlNoQuery] = {
                id: urlNoQuery,
                state: 1 //send
            }
            var wrap = function (obj) {
                resources[res] = obj
                obj.load(name, req, function (a) {
                    if (arguments.length && a !== void 0) {
                        module.exports = a
                    }
                    module.state = 4
                    checkDeps()
                })
            }
            if (!resources[res]) {
                innerRequire([res], wrap)
            } else {
                wrap(resources[res])
            }
        }
        return name ? urlNoQuery : res + "!"
    }
//核心API之一 require
    var requireQueue = []
    var isUserFirstRequire = false
    innerRequire = avalon.require = function (array, factory, parentUrl, defineConfig) {
        if (!isUserFirstRequire) {
            requireQueue.push(avalon.slice(arguments))
            if (arguments.length <= 2) {
                isUserFirstRequire = true
                var queue = requireQueue.splice(0, requireQueue.length), args
                while (args = queue.shift()) {
                    innerRequire.apply(null, args)
                }
            }
            return
        }
        if (!Array.isArray(array)) {
            avalon.error("require方法的第一个参数应为数组 " + array)
        }
        var deps = [] // 放置所有依赖项的完整路径
        var uniq = createMap()
        var id = parentUrl || "callback" + setTimeout("1")// jshint ignore:line
        defineConfig = defineConfig || createMap()
        defineConfig.baseUrl = kernel.baseUrl
        var isBuilt = !!defineConfig.built
        if (parentUrl) {
            defineConfig.parentUrl = parentUrl.substr(0, parentUrl.lastIndexOf("/"))
            defineConfig.mapUrl = parentUrl.replace(rjsext, "")
        }
        if (isBuilt) {
            var req = makeRequest(defineConfig.defineName, defineConfig)
            id = req.urlNoQuery
        } else {
            array.forEach(function (name) {
                var req = makeRequest(name, defineConfig)
                var url = fireRequest(req) //加载资源,并返回该资源的完整地址
                if (url) {
                    if (!uniq[url]) {
                        deps.push(url)
                        uniq[url] = "司徒正美" //去重
                    }
                }
            })
        }
        var module = modules[id]
        if (!module || module.state !== 4) {
            modules[id] = {
                id: id,
                deps: isBuilt ? array.concat() : deps,
                factory: factory || noop,
                state: 3
            }
        }
        if (!module) {
            //如果此模块是定义在另一个JS文件中, 那必须等该文件加载完毕, 才能放到检测列队中
            loadings.push(id)
        }
        checkDeps()
    }
//核心API之二 require
    innerRequire.define = function (name, deps, factory) { //模块名,依赖列表,模块本身
        if (typeof name !== "string") {
            factory = deps
            deps = name
            name = "anonymous"
        }
        if (!Array.isArray(deps)) {
            factory = deps
            deps = []
        }
        var config = {
            built: !isUserFirstRequire, //用r.js打包后,所有define会放到requirejs之前
            defineName: name
        }
        var args = [deps, factory, config]
        factory.require = function (url) {
            args.splice(2, 0, url)
            if (modules[url]) {
                modules[url].state = 3 //loaded
                var isCycle = false
                try {
                    isCycle = checkCycle(modules[url].deps, url)
                } catch (e) {
                }
                if (isCycle) {
                    avalon.error(url + "模块与之前的模块存在循环依赖,请不要直接用script标签引入" + url + "模块")
                }
            }
            delete factory.require //释放内存
            innerRequire.apply(null, args) //0,1,2 --> 1,2,0
        }
//根据标准,所有遵循W3C标准的浏览器,script标签会按标签的出现顺序执行。
//老的浏览器中,加载也是按顺序的:一个文件下载完成后,才开始下载下一个文件。
//较新的浏览器中(IE8+ 、FireFox3.5+ 、Chrome4+ 、Safari4+),为了减小请求时间以优化体验,
//下载可以是并行的,但是执行顺序还是按照标签出现的顺序。
//但如果script标签是动态插入的, 就未必按照先请求先执行的原则了,目测只有firefox遵守
//唯一比较一致的是,IE10+及其他标准浏览器,一旦开始解析脚本, 就会一直堵在那里,直接脚本解析完毕
//亦即,先进入loading阶段的script标签(模块)必然会先进入loaded阶段
        var url = config.built ? "unknown" : getCurrentScript()
        if (url) {
            var module = modules[url]
            if (module) {
                module.state = 2
            }
            factory.require(url)
        } else {//合并前后的safari,合并后的IE6-9走此分支
            factorys.push(factory)
        }
    }
    //核心API之三 require.config(settings)
    innerRequire.config = kernel
    //核心API之四 define.amd 标识其符合AMD规范
    innerRequire.define.amd = modules
    //==========================对用户配置项进行再加工==========================
    var allpaths = kernel["orig.paths"] = createMap()
    var allmaps = kernel["orig.map"] = createMap()
    var allpackages = kernel["packages"] = []
    var allargs = kernel["orig.args"] = createMap()
    avalon.mix(plugins, {
        paths: function (hash) {
            avalon.mix(allpaths, hash)
            kernel.paths = makeIndexArray(allpaths)
        },
        map: function (hash) {
            avalon.mix(allmaps, hash)
            var list = makeIndexArray(allmaps, 1, 1)
            avalon.each(list, function (_, item) {
                item.val = makeIndexArray(item.val)
            })
            kernel.map = list
        },
        packages: function (array) {
            array = array.concat(allpackages)
            var uniq = createMap()
            var ret = []
            for (var i = 0, pkg; pkg = array[i++]; ) {
                pkg = typeof pkg === "string" ? {name: pkg} : pkg
                var name = pkg.name
                if (!uniq[name]) {
                    var url = joinPath(pkg.location || name, pkg.main || "main")
                    url = url.replace(rjsext, "")
                    ret.push(pkg)
                    uniq[name] = pkg.location = url
                    pkg.reg = makeMatcher(name)
                }
            }
            kernel.packages = ret.sort()
        },
        urlArgs: function (hash) {
            if (typeof hash === "string") {
                hash = {"*": hash}
            }
            avalon.mix(allargs, hash)
            kernel.urlArgs = makeIndexArray(allargs, 1)
        },
        baseUrl: function (url) {
            if (!isAbsUrl(url)) {
                var baseElement = head.getElementsByTagName("base")[0]
                if (baseElement) {
                    head.removeChild(baseElement)
                }
                var node = DOC.createElement("a")
                node.href = url
                url = node.href
                if (baseElement) {
                    head.insertBefore(baseElement, head.firstChild)
                }
            }
            if (url.length > 3)
                kernel.baseUrl = url
        },
        shim: function (obj) {
            for (var i in obj) {
                var value = obj[i]
                if (Array.isArray(value)) {
                    value = obj[i] = {
                        deps: value
                    }
                }
                if (!value.exportsFn && (value.exports || value.init)) {
                    value.exportsFn = makeExports(value)
                }
            }
            kernel.shim = obj
        }
    })
    //==============================内部方法=================================
    function checkCycle(deps, nick) {
        //检测是否存在循环依赖
        for (var i = 0, id; id = deps[i++]; ) {
            if (modules[id].state !== 4 &&
                    (id === nick || checkCycle(modules[id].deps, nick))) {
                return true
            }
        }
    }
    function checkFail(node, onError) {
        var id = trimQuery(node.src) //检测是否死链
        node.onload = node.onerror = null
        if (onError) {
            setTimeout(function () {
                head.removeChild(node)
                node = null // 处理旧式IE下的循环引用问题
            })
            log("debug: 加载 " + id + " 失败" + onError + " " + (!modules[id].state))
        } else {
            return true
        }
    }
    function checkDeps() {
        //检测此JS模块的依赖是否都已安装完毕,是则安装自身
        loop: for (var i = loadings.length, id; id = loadings[--i]; ) {
            var obj = modules[id],
                    deps = obj.deps
            if (!deps)
                continue
            for (var j = 0, key; key = deps[j]; j++) {
                if (Object(modules[key]).state !== 4) {
                    continue loop
                }
            }
            //如果deps是空对象或者其依赖的模块的状态都是4
            if (obj.state !== 4) {
                loadings.splice(i, 1) //必须先移除再安装,防止在IE下DOM树建完后手动刷新页面,会多次执行它
                fireFactory(obj.id, obj.deps, obj.factory)
                checkDeps() //如果成功,则再执行一次,以防有些模块就差本模块没有安装好
            }
        }
    }
    function loadJS(url, id, callback) {
        //通过script节点加载目标模块
        var node = DOC.createElement("script")
        node.className = subscribers //让getCurrentScript只处理类名为subscribers的script节点
        node.onload = function () {
            var factory = factorys.pop()
            factory && factory.require(id)
            if (callback) {
                callback()
            }
            log("debug: 已成功加载 " + url)
            id && loadings.push(id)
            checkDeps()
        }
        node.onerror = function () {
            checkFail(node, true)
        }
        head.insertBefore(node, head.firstChild) //chrome下第二个参数不能为null
        node.src = url //插入到head的第一个节点前,防止IE6下head标签没闭合前使用appendChild抛错
        log("debug: 正准备加载 " + url) //更重要的是IE6下可以收窄getCurrentScript的寻找范围
    }
    var resources = innerRequire.plugins = {
        //三大常用资源插件 js!, css!, text!, domReady!
        domReady: {
            load: noop
        },
        js: {
            load: function (name, req, onLoad) {
                var url = req.url
                var id = req.urlNoQuery
                var shim = kernel.shim[name.replace(rjsext, "")]
                if (shim) { //shim机制
                    innerRequire(shim.deps || [], function () {
                        var args = avalon.slice(arguments)
                        loadJS(url, id, function () {
                            onLoad(shim.exportsFn ? shim.exportsFn.apply(0, args) : void 0)
                        })
                    })
                } else {
                    loadJS(url, id)
                }
            }
        },
        css: {
            load: function (name, req, onLoad) {
                var url = req.url
                head.insertAdjacentHTML("afterBegin", '<link rel="stylesheet" href="' + url + '">')
                log("debug: 已成功加载 " + url)
                onLoad()
            }
        },
        text: {
            load: function (name, req, onLoad) {
                var url = req.url
                var xhr = getXHR()
                xhr.onload = function () {
                    var status = xhr.status;
                    if (status > 399 && status < 600) {
                        avalon.error(url + " 对应资源不存在或没有开启 CORS")
                    } else {
                        log("debug: 已成功加载 " + url)
                        onLoad(xhr.responseText)
                    }
                }
                var time = "_=" + (new Date() - 0)
                var _url = url.indexOf("?") === -1 ? url + "?" + time : url + "&" + time
                xhr.open("GET", _url, true)
                if ("withCredentials" in xhr) {//这是处理跨域
                    xhr.withCredentials = true
                }
                xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest")//告诉后端这是AJAX请求
                xhr.send()
                log("debug: 正准备加载 " + url)
            }
        }
    }
    innerRequire.checkDeps = checkDeps
    var rquery = /(\?[^#]*)$/
    function trimQuery(url) {
        return (url || "").replace(rquery, "")
    }
    function isAbsUrl(path) {
        //http://stackoverflow.com/questions/10687099/how-to-test-if-a-url-string-is-absolute-or-relative
        return  /^(?:[a-z]+:)?\/\//i.test(String(path))
    }
    function getCurrentScript() {
        // inspireb by https://github.com/samyk/jiagra/blob/master/jiagra.js
        var stack
        try {
            a.b.c() //强制报错,以便捕获e.stack
        } catch (e) { //safari5的sourceURL,firefox的fileName,它们的效果与e.stack不一样
            stack = e.stack
        }
        if (stack) {
            /**e.stack最后一行在所有支持的浏览器大致如下:
             *chrome23:
             * at http://113.93.50.63/data.js:4:1
             *firefox17:
             *@http://113.93.50.63/query.js:4
             *opera12:http://www.oldapps.com/opera.php?system=Windows_XP
             *@http://113.93.50.63/data.js:4
             *IE10:
             *  at Global code (http://113.93.50.63/data.js:4:1)
             *  //firefox4+ 可以用document.currentScript
             */
            stack = stack.split(/[@ ]/g).pop() //取得最后一行,最后一个空格或@之后的部分
            stack = stack[0] === "(" ? stack.slice(1, -1) : stack.replace(/\s/, "") //去掉换行符
            return trimQuery(stack.replace(/(:\d+)?:\d+$/i, "")) //去掉行号与或许存在的出错字符起始位置
        }
        var nodes = head.getElementsByTagName("script") //只在head标签中寻找
        for (var i = nodes.length, node; node = nodes[--i]; ) {
            if (node.className === subscribers && node.readyState === "interactive") {
                var url = node.src
                return node.className = trimQuery(url)
            }
        }
    }
    var rcallback = /^callback\d+$/
    function fireFactory(id, deps, factory) {
        var module = Object(modules[id])
        module.state = 4
        for (var i = 0, array = [], d; d = deps[i++]; ) {
            if (d === "exports") {
                var obj = module.exports || (module.exports = createMap())
                array.push(obj)
            } else {
                array.push(modules[d].exports)
            }
        }
        try {
            var ret = factory.apply(window, array)
        } catch (e) {
            log("执行[" + id + "]模块的factory抛错: ", e)
        }
        if (ret !== void 0) {
            module.exports = ret
        }
        if (rcallback.test(id)) {
            delete modules[id]
        }
        delete module.factory
        return ret
    }
    function toUrl(id) {
        if (id.indexOf(this.res + "!") === 0) {
            id = id.slice(this.res.length + 1) //处理define("css!style",[], function(){})的情况
        }
        var url = id
        //1. 是否命中paths配置项
        var usePath = 0
        var baseUrl = this.baseUrl
        var rootUrl = this.parentUrl || baseUrl
        eachIndexArray(id, kernel.paths, function (value, key) {
            url = url.replace(key, value)
            usePath = 1
        })
        //2. 是否命中packages配置项
        if (!usePath) {
            eachIndexArray(id, kernel.packages, function (value, key, item) {
                url = url.replace(item.name, item.location)
            })
        }
        //3. 是否命中map配置项
        if (this.mapUrl) {
            eachIndexArray(this.mapUrl, kernel.map, function (array) {
                eachIndexArray(url, array, function (mdValue, mdKey) {
                    url = url.replace(mdKey, mdValue)
                    rootUrl = baseUrl
                })
            })
        }
        var ext = this.ext
        if (ext && usePath && url.slice(-ext.length) === ext) {
            url = url.slice(0, -ext.length)
        }
        //4. 转换为绝对路径
        if (!isAbsUrl(url)) {
            rootUrl = this.built || /^\w/.test(url) ? baseUrl : rootUrl
            url = joinPath(rootUrl, url)
        }
        //5. 还原扩展名,query
        var urlNoQuery = url + ext
        url = urlNoQuery + this.query
        urlNoQuery = url.replace(rquery, function (a) {
          this.query = a
          return ""
        })
        //6. 处理urlArgs
        eachIndexArray(id, kernel.urlArgs, function (value) {
            url += (url.indexOf("?") === -1 ? "?" : "&") + value;
        })
        this.url = url
        return  this.urlNoQuery = urlNoQuery
    }
    function makeIndexArray(hash, useStar, part) {
        //创建一个经过特殊算法排好序的数组
        var index = hash2array(hash, useStar, part)
        index.sort(descSorterByName)
        return index
    }
    function makeMatcher(prefix) {
        return new RegExp('^' + prefix + '(/|$)')
    }
    function makeExports(value) {
        return function () {
            var ret
            if (value.init) {
                ret = value.init.apply(window, arguments)
            }
            return ret || (value.exports && getGlobal(value.exports))
        }
    }
    function hash2array(hash, useStar, part) {
        var array = [];
        for (var key in hash) {
            // if (hash.hasOwnProperty(key)) {//hash是由createMap创建没有hasOwnProperty
            var item = {
                name: key,
                val: hash[key]
            }
            array.push(item)
            item.reg = key === "*" && useStar ? /^/ : makeMatcher(key)
            if (part && key !== "*") {
                item.reg = new RegExp('\/' + key.replace(/^\//, "") + '(/|$)')
            }
            //   }
        }
        return array
    }
    function eachIndexArray(moduleID, array, matcher) {
        array = array || []
        for (var i = 0, el; el = array[i++]; ) {
            if (el.reg.test(moduleID)) {
                matcher(el.val, el.name, el)
                return false
            }
        }
    }
    // 根据元素的name项进行数组字符数逆序的排序函数
    function descSorterByName(a, b) {
        var aaa = a.name
        var bbb = b.name
        if (bbb === "*") {
            return -1
        }
        if (aaa === "*") {
            return 1
        }
        return bbb.length - aaa.length
    }
    var rdeuce = /\/\w+\/\.\./
    function joinPath(a, b) {
        if (a.charAt(a.length - 1) !== "/") {
            a += "/"
        }
        if (b.slice(0, 2) === "./") { //相对于兄弟路径
            return a + b.slice(2)
        }
        if (b.slice(0, 2) === "..") { //相对于父路径
            a += b
            while (rdeuce.test(a)) {
                a = a.replace(rdeuce, "")
            }
            return a
        }
        if (b.slice(0, 1) === "/") {
            return a + b.slice(1)
        }
        return a + b
    }
    function getGlobal(value) {
        if (!value) {
            return value
        }
        var g = window
        value.split(".").forEach(function (part) {
            g = g[part]
        })
        return g
    }
    var mainNode = DOC.scripts[DOC.scripts.length - 1]
    var dataMain = mainNode.getAttribute("data-main")
    if (dataMain) {
        plugins.baseUrl(dataMain)
        var href = kernel.baseUrl
        kernel.baseUrl = href.slice(0, href.lastIndexOf("/") + 1)
        loadJS(href.replace(rjsext, "") + ".js")
    } else {
        var loaderUrl = trimQuery(mainNode.src)
        kernel.baseUrl = loaderUrl.slice(0, loaderUrl.lastIndexOf("/") + 1)
    }
}// jshint ignore:line
/*********************************************************************
 *                    DOMReady                                         *
 **********************************************************************/
var readyList = [],
    isReady
var fireReady = function (fn) {
    isReady = true
    var require = avalon.require
    if (require && require.checkDeps) {
        modules["domReady!"].state = 4
        require.checkDeps()
    }
    while (fn = readyList.shift()) {
        fn(avalon)
    }
}
if (DOC.readyState === "complete") {
    setTimeout(fireReady) //如果在domReady之外加载
} else {
    DOC.addEventListener("DOMContentLoaded", fireReady)
}
window.addEventListener("load", fireReady)
avalon.ready = function (fn) {
    if (!isReady) {
        readyList.push(fn)
    } else {
        fn(avalon)
    }
}
avalon.config({
    loader: false
})
avalon.ready(function () {
    avalon.scan(DOC.body)
})
// Register as a named AMD module, since avalon can be concatenated with other
// files that may use define, but not via a proper concatenation script that
// understands anonymous AMD modules. A named AMD is safest and most robust
// way to register. Lowercase avalon is used because AMD module names are
// derived from file names, and Avalon is normally delivered in a lowercase
// file name. Do this after creating the global so that if an AMD module wants
// to call noConflict to hide this version of avalon, it will work.
// Note that for maximum portability, libraries that are not avalon should
// declare themselves as anonymous modules, and avoid setting a global if an
// AMD loader is present. avalon is a special case. For more information, see
// https://github.com/jrburke/requirejs/wiki/Updating-existing-libraries#wiki-anon
    if (typeof define === "function" && define.amd) {
        define("avalon", [], function() {
            return avalon
        })
    }
// Map over avalon in case of overwrite
    var _avalon = window.avalon
    avalon.noConflict = function(deep) {
        if (deep && window.avalon === avalon) {
            window.avalon = _avalon
        }
        return avalon
    }
// Expose avalon identifiers, even in AMD
// and CommonJS for browser emulators
    if (noGlobal === void 0) {
        window.avalon = avalon
    }
    return avalon
}));

+ 84 - 0
plugins/jianchajianyan.html

@ -0,0 +1,84 @@
<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta name="renderer" content="webkit">
    <meta http-equiv="Cache-Control" content="no-siteapp" />
    <title>检查检验列表</title>
    <meta name="keywords" content="">
    <meta name="description" content="">
    <link rel="shortcut icon" href="favicon.ico">
    <link href="css/bootstrap.min.css?v=3.3.6" rel="stylesheet">
    <link href="css/font-awesome.min93e3.css?v=4.4.0" rel="stylesheet">
    <link href="css/animate.min.css" rel="stylesheet">
    <link href="css/plugins/bootstrap-table/bootstrap-table.min.css" rel="stylesheet">
    <link href="css/plugins/toastr/toastr.min.css" rel="stylesheet">
    <link href="js/plugins/artDialog/6.0.5/css/ui-dialog.min.css" rel="stylesheet">
    <link href="css/style.css" rel="stylesheet">
    <style>
        .fixed-table-container{
            border:none;
        }
        .bootstrap-table .table{
            border-bottom: none;
            border-collapse: separate !important;
            border-spacing: 0 10px;
        }
        .fixed-table-container tbody td{
            border-left: none;
        }
        .bootstrap-table .table tr{
            background-color: #f5f5fa;
            margin-top: 10px;
        }
        .status-tag{
            display: inline-block;
            background: #12b7f5;
            color: #fff;
            padding: 3px 10px;
            border-radius: 5px;
        }
        .table>tbody>tr>td, .table>tbody>tr>th, .table>tfoot>tr>td, .table>tfoot>tr>th, .table>thead>tr>td, .table>thead>tr>th{
            border-top: none;
        }
    </style>
</head>
<body class="white-bg plr10">
    <table id="prescriptTable" data-mobile-responsive="true" data-show-header="false" cellspacing="10" border="0">
    </table>
    <script type="text/html" id="info_tmp">
    <div class="ui-grid pl10 ui-grid-middle mb10">
        <div class="ui-col-0" style="width: 70px"><span class="status-tag">{{type}}</span></div>
        <div class="ui-col-1">{{orgName}}</div>
    </div>
    <div class="ui-grid ui-grid-middle pl10">
        <div class="ui-col-1">
            <span>检查类型:</span>
            <span>{{label}}</span>
        </div>
    </div>
    </script>
    
    <script src="js/jquery.min.js?v=2.1.4"></script>
    <script src="js/bootstrap.min.js?v=3.3.6"></script>
    <script src="js/plugins/metisMenu/jquery.metisMenu.js"></script>
    <script src="js/plugins/layer/layer.min.js"></script>
    <script src="js/plugins/slimscroll/jquery.slimscroll.min.js"></script>
    <script src="js/hplus.min.js?v=4.1.0"></script>
    <script src="js/plugins/pace/pace.min.js"></script>
    <script src="js/plugins/toastr/toastr.min.js"></script>
    <script src="js/plugins/artDialog/6.0.5/js/dialog-plus.min.js"></script>
    <script src="js/plugins/bootstrap-table/bootstrap-table.min.js"></script>
    <script src="js/plugins/bootstrap-table/bootstrap-table-mobile.min.js"></script>
    <script src="js/plugins/bootstrap-table/locale/bootstrap-table-zh-CN.min.js"></script>
    <script src="js/es6-promise.js"></script>
    <script src="js/underscore.js"></script>
    <script src="js/util.js"></script>
    <script src="js/template.js"></script>
    <script src="js/api-service.js"></script>
    <script src="js/jianchajianyan-api.js"></script>
    <script src="js/buz/jianchajianyan.js"></script>
</body>
</html>

+ 185 - 0
template/0101.html

@ -0,0 +1,185 @@
<!--
	门诊摘要
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['姓名'] || data['性别']">
                <li ms-if="data['姓名']"><label>姓&#x3000;&#x3000;名:</label><span ms-text="data['姓名']"></span></li>
                <li ms-if="data['性别']"><label>性&#x3000;&#x3000;别:</label><span ms-text="data['性别']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['出生日期'] || data['婚姻状况']">
                <li ms-if="data['出生日期']"><label>出生日期:</label><span ms-text="data['出生日期']"></span></li>
                <li ms-if="data['婚姻状况']"><label>婚姻状况:</label><span ms-text="data['婚姻状况']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['就诊日期']">
                <li ms-if="data['就诊日期']"><label>就诊日期:</label><span ms-text="data['就诊日期']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['主治医生']">
                <li ms-if="data['主治医生']"><label>主治医生:</label><span ms-text="data['主治医生']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['科室'] || data['医院']">
                <li><label>科室/医院:</label>
                	<span ms-if="data['科室'] && data['医院']"><span ms-text="data['科室']"></span> <span ms-text="data['斜杠']"></span> <span ms-text="data['医院']"></span></span>
                	<span ms-if="data['科室'] && !(data['医院'])" ms-text="data['科室']"></span>
                	<span ms-if="!(data['科室']) && data['医院']" ms-text="data['医院']"></span>
                </li>
            </ul>
        </li>
    </ul>
    <ul class="info-list-t" ms-if="data['诊断'].length >0 || data['治疗措施']">
    	<h3 class="os-tit-o" ms-if="data['诊断'].length >0 || data['治疗措施']"><i class="icon i-visiting-result"></i><span class="os-t-o-name">就诊结果</span></h3>
        <li class="il-t-item" ms-if="data['诊断'].length >0">
            <h4 class="os-tit-t">诊断:</h4>
            <p><span ms-repeat="data['诊断']"><span ms-text="el['诊断名称']"></span>&#x3000;</span></p>
        </li>
        <li class="il-t-item" ms-if="data['治疗措施']">
            <h4 class="os-tit-t">处理意见:</h4>
            <p><span ms-text="data['治疗措施']"></span></p>
        </li>
    </ul>
    <ul class="info-list-t" ms-if="data['主诉'] || data['体格检查'] || data['辅助检查'] || data['过敏史']">
    	<h3 class="os-tit-o" style="padding-bottom:0px;margin-left: 10px;"><i class="icon i-dia-basis"></i><span class="os-t-o-name">诊断依据</span></h3>
        <li class="il-t-item" ms-if="data['主诉']">
            <h4 class="os-tit-t" ms-text="主诉:"></h4>
            <p ms-text="data['主诉']"></p>
        </li>
        <li class="il-t-item" ms-if="data['体格检查']">
            <h4 class="os-tit-t" ms-text="体格检查:"></h4>
            <p ms-text="data['体格检查']"></p>
        </li>
        <li class="il-t-item" ms-if="data['辅助检查']">
            <h4 class="os-tit-t" ms-text="辅助检查:"></h4>
            <p ms-text="data['辅助检查']"></p>
        </li>
        <li class="il-t-item" ms-if="data['专科情况']">
            <h4 class="os-tit-t" ms-text="专科情况:"></h4>
            <p ms-text="data['专科情况']"></p>
        </li>
        <li class="il-t-item" ms-if="data['过敏史']">
            <h4 class="os-tit-t" ms-text="过敏史:"></h4>
            <p ms-text="data['过敏史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['现病史']">
            <h4 class="os-tit-t" ms-text="现病史:"></h4>
            <p ms-text="data['现病史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['既往史']">
            <h4 class="os-tit-t" ms-text="既往史:"></h4>
            <p ms-text="data['既往史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['家族史']">
            <h4 class="os-tit-t" ms-text="家族史:"></h4>
            <p ms-text="data['家族史']"></p>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	dataCode = ['0111'];//新增数据请求
	function dsXmlToJson0101_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom.find('component section'),
				"病人基本信息":$xmlDom.find('recordTarget patient')
			} ;
		return {
			data:{
				"斜杠":'/',
				"姓名": $nodes["病人基本信息"].find('name').text(),
				"性别": $nodes["病人基本信息"].find('sex').text(),
				"出生日期": $nodes["病人基本信息"].find('birthDate').text(),
				"婚姻状况": $nodes["病人基本信息"].find('marriage').text(),
				"就诊日期": $nodes["基本诊疗信息"].find('entry treatTime').text(),
				"主治医生": $nodes["基本诊疗信息"].find('entry doctor name').text(),
				"科室": $nodes["基本诊疗信息"].find('entry dept').text(),
				"医院": $xmlDom.find('org').text(),
				"诊断": (function() {
						var arr = _.map($nodes["基本诊疗信息"].find('diagnosis item'),function(el) {
							var $ele = $(el); 
							return {
								"诊断名称": $ele.find('icd').text(),
							}
						});
						return arr;
				})()
			}
		}
	}
	
	function dsXmlToJson0101_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom
			} ;
		return {
			data:{
				"科室": $nodes["基本诊疗信息"].find('dept').text(),
				"主治医生": $nodes["基本诊疗信息"].find('doctor').text(),
				"诊断": (function() {
						var arr = [];
						arr = _.map($nodes["基本诊疗信息"].find('diagnosis item'),function(el) {
							var $ele = $(el); 
							return {
								"诊断名称": $ele.text(),
							}
						})
						return arr;
				})()
			}
		}
	}
	
	function dsXmlToJson0111_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom.find('component section entry'),
				"病人基本信息":$xmlDom.find('recordTarget patient')
			}
		return {
			data:{
				"主诉": $nodes["基本诊疗信息"].find('cc').text(),
				"既往史": $nodes["基本诊疗信息"].find('diseaseHistory').text(),
				"体格检查": $nodes["基本诊疗信息"].find('physicalExam').text(),
				"专科情况": $nodes["基本诊疗信息"].find('spec').text(),
				"辅助检查": $nodes["基本诊疗信息"].find('accessoryExam').text(),
				"家族史": $nodes["基本诊疗信息"].find('familyHistory').text(),
				"过敏史": $nodes["基本诊疗信息"].find('allergyDrugs').text(),
				"现病史": $nodes["基本诊疗信息"].find('currentDisease').text(),
				"治疗措施": $nodes["基本诊疗信息"].find('treatment').text(),
			}
		}
	}
	
	function dsXmlToJson0111_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom.find('component section entry'),
				"病人基本信息":$xmlDom.find('recordTarget patient')
			}
		return {
			data:{
				"姓名": $nodes["病人基本信息"].find('name').text(),
				"性别": $nodes["病人基本信息"].find('sex').text(),
				"出生日期": $nodes["病人基本信息"].find('birthTime').text(),
				"婚姻状况": $nodes["病人基本信息"].find('marriage').text(),
				"就诊日期": $xmlDom.find('inChargeDoctor time').attr('value'),
				"主诉": $nodes["基本诊疗信息"].find('cc').text(),
				"既往史": $nodes["基本诊疗信息"].find('diseaseHistory').text(),
				"体格检查": $nodes["基本诊疗信息"].find('physicalExam').text(),
				"专科情况": $nodes["基本诊疗信息"].find('spec').text(),
				"辅助检查": $nodes["基本诊疗信息"].find('accessoryExam').text(),
				"家族史": $nodes["基本诊疗信息"].find('familyHistory').text(),
				"过敏史": $nodes["基本诊疗信息"].find('allergyDrugs').text(),
				"现病史": $nodes["基本诊疗信息"].find('currentDisease').text(),
				"治疗措施": $nodes["基本诊疗信息"].find('treatment').text(),
			}
		}
	}
</script>

+ 81 - 0
template/0102.html

@ -0,0 +1,81 @@
<!--
	费用清单
-->
<div class="el-main">
    <div class="el-tit">
        <div class="el-t-l">费用总计</div>
        <div class="el-t-m" ms-text="data['费用总计']"></div>
    </div>
    <ul class="el-list">
        <li class="el-item" ms-repeat="data['门诊分类']">
            <div class="title"><span ms-text="el['分类名称']"></span><span ms-text="el['费用金额']"></span></div>
            <div class="list" ms-if="el['费用详情']" ms-repeat-item="el['费用详情']"><span ms-text="item['项目名称']"></span><span ms-text="item['总价']"></span></div>
        </li>
    </ul>
</div>
<script>
	function dsXmlToJson0102_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			fee_sum = '';//费用总计
			$nodes = {
				"费用" :$xmlDom.find('component section entry')
			} ;
		return {
			data:{
				"门诊分类":(function(){
					var arr = [];
					_.each($nodes["费用"].find('group'),function(el) {
						var $ele = $(el); 
						arr.push({
							"分类名称": $ele.find('type').text(),
							"费用金额":$ele.find('fee').text() ?parseFloat($ele.find('fee').text()).toFixed(2):'',
							"费用详情": (function() {
								var arr1 = [];
								_.each($ele.find('item'),function(it) {
									var $it = $(it);
									arr1.push({
										"项目名称": $it.find('value').text(),
										"总价": $it.find('total').text() ? parseFloat($it.find('total').text()).toFixed(2) : ''
									})
								})
								return arr1;
							})()
						})
					})
					fee_sum = _.reduce( _.pluck(arr, '费用金额'), function(a, b){ return a*1 + b*1; }, 0).toFixed(2);
					return arr;
				})(),
				"费用总计":fee_sum,
			}
		}
	}
	function dsXmlToJson0102_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			fee_sum = '',//费用总计
			$nodes = {
				"费用" :$xmlDom
			} ;
		return {
			data:{
				"门诊分类": (function() {
						var arr = [];
						_.each($nodes["费用"].find('item'),function(it) {
							var $it = $(it);
							arr.push({
								"分类名称": $it.text(),
								"费用金额":$it.attr('total') ? parseFloat($it.attr('total')).toFixed(2):'',
							})
						})
						fee_sum = _.reduce( _.pluck(arr, '费用金额'), function(a, b){ return a*1 + b*1; }, 0).toFixed(2);
						return arr;
				})(),
				"费用总计":fee_sum,
			}
		}
	}
</script>

+ 142 - 0
template/0111.html

@ -0,0 +1,142 @@
<!--
	门诊病历记录
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['姓名'] || data['性别']">
                <li ms-if="data['姓名']"><label>姓&#x3000;&#x3000;名:</label><span ms-text="data['姓名']"></span></li>
                <li ms-if="data['性别']"><label>性&#x3000;&#x3000;别:</label><span ms-text="data['性别']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['出生日期'] || data['婚姻状况']">
                <li ms-if="data['出生日期']"><label>出生日期:</label><span ms-text="data['出生日期']"></span></li>
                <li ms-if="data['婚姻状况']"><label>婚姻状况:</label><span ms-text="data['婚姻状况']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['就诊日期'] || data['主治医生']">
                <li ms-if="data['就诊日期']"><label>就诊日期:</label><span ms-text="data['就诊日期']"></span></li>
                <li ms-if="data['主治医生']"><label>主治医生:</label><span ms-text="data['主治医生']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['科室'] || data['医院']">
                <li><label>科室/医院:</label>
                	<span ms-if="data['科室'] && data['医院']"><span ms-text="data['科室']"></span> <span ms-text="data['斜杠']"></span> <span ms-text="data['医院']"></span></span>
                	<span ms-if="data['科室'] && !(data['医院'])" ms-text="data['科室']"></span>
                	<span ms-if="!(data['科室']) && data['医院']" ms-text="data['医院']"></span>
            </ul>
        </li>
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o" ms-if="data['主要诊断'] || data['治疗措施']" style="padding-bottom:0px;margin-left: 10px;"><i class="icon i-visiting-result"></i><span class="os-t-o-name">就诊结果</span></h3>
        <li class="il-t-item" ms-if="data['主要诊断']">
            <h4 class="os-tit-t">主要诊断:</h4>
            <p><span ms-text="data['主要诊断']"></span></p>
        </li>
        <li class="il-t-item" ms-if="data['治疗措施']">
            <h4 class="os-tit-t">处理意见:</h4>
            <p><span ms-text="data['治疗措施']"></span></p>
        </li>
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o" style="padding-bottom:0px;margin-left: 10px;"><i class="icon i-dia-basis"></i><span class="os-t-o-name">诊断依据</span></h3>
        <li class="il-t-item" ms-if="data['主诉']">
            <h4 class="os-tit-t">主诉:</h4>
            <p ms-text="data['主诉']"></p>
        </li>
        <li class="il-t-item" ms-if="data['体格检查']">
            <h4 class="os-tit-t">体格检查:</h4>
            <p ms-text="data['体格检查']"></p>
        </li>
        <li class="il-t-item" ms-if="data['辅助检查']">
            <h4 class="os-tit-t">辅助检查:</h4>
            <p ms-text="data['辅助检查']"></p>
        </li>
        <li class="il-t-item" ms-if="data['专科情况']">
            <h4 class="os-tit-t">专科情况:</h4>
            <p ms-text="data['专科情况']"></p>
        </li>
        <li class="il-t-item" ms-if="data['过敏史']">
            <h4 class="os-tit-t">过敏史:</h4>
            <p ms-text="data['过敏史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['现病史']">
            <h4 class="os-tit-t">现病史:</h4>
            <p ms-text="data['现病史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['既往史']">
            <h4 class="os-tit-t">既往史:</h4>
            <p ms-text="data['既往史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['家族史']">
            <h4 class="os-tit-t">家族史:</h4>
            <p ms-text="data['家族史']"></p>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
 
	function dsXmlToJson0111_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom.find('component section entry'),
				"病人基本信息":$xmlDom.find('recordTarget patient')
			}
		return {
			data:{
				"斜杠":'/',
				"姓名": $nodes["病人基本信息"].find('name').text(),
				"性别": $nodes["病人基本信息"].find('sex').text(),
				"出生日期": $nodes["病人基本信息"].find('birthDate').text(),
				"婚姻状况": $nodes["病人基本信息"].find('marriage').text(),
				"就诊日期": $nodes["基本诊疗信息"].find('treatTime').text(),
				"主治医生": $nodes["基本诊疗信息"].find('doctor name').text(),
				"科室": $nodes["基本诊疗信息"].find('dept').text(),
				"医院": $xmlDom.find('org').text(),
				"主诉": $nodes["基本诊疗信息"].find('cc').text(),
				"既往史": $nodes["基本诊疗信息"].find('diseaseHistory').text(),
				"体格检查": $nodes["基本诊疗信息"].find('physicalExam').text(),
				"专科情况": $nodes["基本诊疗信息"].find('spec').text(),
				"辅助检查": $nodes["基本诊疗信息"].find('accessoryExam').text(),
				"主要诊断": $nodes["基本诊疗信息"].find('diagnosis').text(),
				"家族史": $nodes["基本诊疗信息"].find('familyHistory').text(),
				"过敏史": $nodes["基本诊疗信息"].find('allergyDrugs').text(),
				"现病史": $nodes["基本诊疗信息"].find('currentDisease').text(),
				"治疗措施": $nodes["基本诊疗信息"].find('treatment').text(),
			}
		}
	}
	
	function dsXmlToJson0111_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom.find('component section entry'),
				"病人基本信息":$xmlDom.find('recordTarget patient')
			}
		return {
			data:{
				"斜杠":'/',
				"姓名": $nodes["病人基本信息"].find('name').text(),
				"性别": $nodes["病人基本信息"].find('sex').text(),
				"出生日期": $nodes["病人基本信息"].find('birthTime').text(),
				"婚姻状况": $nodes["病人基本信息"].find('marriage').text(),
				"就诊日期": $xmlDom.find('inChargeDoctor time').attr('value'),
				"主治医生": $xmlDom.find('inChargeDoctor name').text(),
				"科室": $xmlDom.find('title').text(),
				"医院": $xmlDom.find('org').text(),
				"主诉": $nodes["基本诊疗信息"].find('cc').text(),
				"既往史": $nodes["基本诊疗信息"].find('diseaseHistory').text(),
				"体格检查": $nodes["基本诊疗信息"].find('physicalExam').text(),
				"专科情况": $nodes["基本诊疗信息"].find('spec').text(),
				"辅助检查": $nodes["基本诊疗信息"].find('accessoryExam').text(),
				"主要诊断": $nodes["基本诊疗信息"].find('diagnosis').text(),
				"家族史": $nodes["基本诊疗信息"].find('familyHistory').text(),
				"过敏史": $nodes["基本诊疗信息"].find('allergyDrugs').text(),
				"现病史": $nodes["基本诊疗信息"].find('currentDisease').text(),
				"治疗措施": $nodes["基本诊疗信息"].find('treatment').text(),
			}
		}
	}
</script>

+ 146 - 0
template/0121.html

@ -0,0 +1,146 @@
<!--
	检验详情
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['检验项目']">
                <li><label>检验项目:</label><span ms-text="data['检验项目']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['标本类型']">
                <li><label>标本类型:</label><span ms-text="data['标本类型']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['报告日期']">
                <li><label>报告时间:</label><span ms-text="data['报告日期']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['医院']">
                <li><label>医&#x3000;&#x3000;院:</label><span ms-text="data['医院']"></span></li>
            </ul>
        </li>
    </ul>
    <ul class="info-list" ms-if="data['检验结果'].length >0">
        <li class="il-item il-i-table">
            <h3 class="os-tit-o"><i class="icon i-test-results"></i><span class="os-t-o-name">检验结果</span></h3>
            <table class="id-table" cellpadding="0" cellspacing="0" border="0">
                <thead>
                    <tr>
                        <th style="width: 30%;text-align: center;font-weight: bold;">项目名称</th>
                        <th style="width: 20%;text-align: center;font-weight: bold;">检验结果</th>
                        <th style="width: 30%;text-align: center;font-weight: bold;">参考值</th>
                        <th style="width: 20%;text-align: center;font-weight: bold;">单位</th>
                    </tr>
                </thead>
                <tbody>
                    <tr ms-repeat="data['检验结果']">
                        <td ms-text="el['项目名称']"></td>
                        <td ms-text="el['结果']"></td>
                        <td ms-text="el['参考值']"></td>
                        <td ms-text="el['单位']"></td>
                    </tr>
                </tbody>
            </table>
        </li>
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o"><i class="icon i-inspectors"></i><span class="os-t-o-name">检验人员</span></h3>
        <li class="il-t-item pl0">
            <ul class="os-base-info" ms-if="data['检验时间']">
                <li><label>检验时间:</label><span ms-text="data['检验时间']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['申请科室'] || data['申请医生']">
                <li ms-if="data['申请科室']"><label>申请科室:</label><span ms-text="data['申请科室']"></span></li>
                <li ms-if="data['申请医生']"><label>申请医生:</label><span ms-text="data['申请医生']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['检验医生'] || data['审核者']">
                <li ms-if="data['检验医生']"><label>检&nbsp;&nbsp;验&nbsp;&nbsp;员:</label><span ms-text="data['检验医生']"></span></li>
                <li ms-if="data['审核者']"><label>复&nbsp;&nbsp;核&nbsp;&nbsp;人:</label><span ms-text="data['审核者']"></span></li>
            </ul>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0121_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"检验的专业类型名称" :$xmlDom.find('component section')
			} ;
		
		return {
			data:{
				"医院": $xmlDom.find('org').text(),
				"标本类型": $nodes["检验的专业类型名称"].find('entry classcode').text(),
				"检验项目": $nodes["检验的专业类型名称"].attr('displayname'),
				"检验结果": _.map($nodes["检验的专业类型名称"].find('entry observation'),function(el) {
							var $ele = $(el),
								$referRange = $ele.find('referenceRange observationRange');
							return {
								"项目名称": $ele.find('code').attr('displayname'),
								"结果": $ele.find('>value').text(),
								"参考值": (function() {
											var $low = $referRange.find('low'),
													$high = $referRange.find('high'),
													low = $.trim($low.find('value').text()),
													high = $.trim($high.find('value').text());
											if(low && high) {
												return low + '~' + high;
											} else {
												return low || high || "";
											}
								})(),
								"单位": $referRange.find('low unit').text(),
							}
						}),
				"检验时间": $nodes["检验的专业类型名称"].find('entry performer time').text(),
				"检验医生": $nodes["检验的专业类型名称"].find('entry performer name').text(),
				"审核者": $xmlDom.find('authenticator name').text(),
				"申请医生":$xmlDom.find('participant name').text(),
				"申请科室":$xmlDom.find('participant dept').text(),
			}
		}
	}
	function dsXmlToJson0121_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"检验的专业类型名称" :$xmlDom.find('component section')
			} ;
		
		return {
			data:{
				"医院": $xmlDom.find('org').text(),
				"标本类型": $nodes["检验的专业类型名称"].find('entry classcode').text(),
				"检验项目": $nodes["检验的专业类型名称"].attr('displayname'),
				"检验结果": _.map($nodes["检验的专业类型名称"].find('entry observation'),function(el) {
							var $ele = $(el),
								$referRange = $ele.find('referenceRange observationRange');
							return {
								"项目名称": $ele.find('code').attr('displayname'),
								"结果": $ele.find('>value').text(),
								"参考值": (function() {
											var $low = $referRange.find('low'),
													$high = $referRange.find('high'),
													low = $.trim($low.find('value').text()),
													high = $.trim($high.find('value').text());
											if(low && high) {
												return low + '~' + high;
											} else {
												return low || high || "";
											}
								})(),
								"单位": $referRange.find('low unit').text(),
							}
						}),
				"检验时间": $nodes["检验的专业类型名称"].find('entry performer time').text(),
				"检验医生": $nodes["检验的专业类型名称"].find('entry performer name').text(),
				"审核者": $xmlDom.find('authenticator name').text(),
				"申请医生":$xmlDom.find('participant name').text(),
				"申请科室":$xmlDom.find('participant dept').text(),
			}
		}
	}
</script>

+ 132 - 0
template/0131.html

@ -0,0 +1,132 @@
<!--
	检查详情
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['检查名称']">
                <li><label>检查名称:</label><span ms-text="data['检查名称']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['检查类型'] || data['检查部位']">
                <li ms-if="data['检查类型']"><label>类&#x3000;&#x3000;型:</label><span ms-text="data['检查类型']"></span></li>
                <li ms-if="data['检查部位']"><label>部&#x3000;&#x3000;位:</label><span ms-text="data['检查部位']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['报告日期']">
                <li><label>报告时间:</label><span ms-text="data['报告日期']"></span></li> 
            </ul>
            <ul class="os-base-info" ms-if="data['医院']">
                <li><label>医&#x3000;&#x3000;院:</label><span ms-text="data['医院']"></span></li>
            </ul>
        </li>
        <li class="il-item">
        </li>
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o"><i class="icon i-ins-res"></i><span class="os-t-o-name">检查结果</span></h3>
        <li class="il-t-item jiancha-mt">
            <h4 class="os-tit-th">临床诊断:</h4>
            <p ms-text="data['临床诊断']"></p>
        </li>
        <li class="il-t-item jiancha-mt" ms-if="data['病情描述']">
            <h4 class="os-tit-t">病情描述:</h4>
            <p ms-text="data['病情描述']"></p>
        </li>
        <li class="il-t-item jiancha-mt" ms-if="data['影像所见']">
            <h4 class="os-tit-t">影像所见:</h4>
            <p ms-text="data['影像所见']"></p>
        </li>
        <li class="il-t-item jiancha-mt" ms-if="data['是否异常'] || data['影像结论']">
            <h4 class="os-tit-t">影像结论:</h4>
            <p class="ying-pb">是否异常:<span ms-text="data['是否异常']"></span></p>
            <p class="ying-pb" ms-if="data['影像结论']" ms-text="data['影像结论']"></p>
        </li>
        <!--<li class="il-t-item">
            <h4 class="os-tit-t">影像图片:</h4>
            <ul class="i-i-list">
                <li class="i-i-l-item"><img src="" alt=""></li>
            </ul>
        </li>-->
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o"><i class="icon i-inspectors"></i><span class="os-t-o-name">检查人员</span></h3>
        <li class="il-t-item pl0">
            <ul class="os-base-info" ms-if="data['执行时间']">
                <li><label>检查时间:</label><span ms-text="data['执行时间']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['申请科室'] || data['申请医生']">
                <li ms-if="data['申请科室']"><label>申请科室:</label><span ms-text="data['申请科室']"></span></li>
                <li ms-if="data['申请医生']"><label>申请医生:</label><span ms-text="data['申请医生']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['执行科室'] || data['执行医生']">
                <li ms-if="data['执行科室']"><label>执行科室:</label><span ms-text="data['执行科室']"></span></li>
                <li ms-if="data['执行医生']"><label>执行医生:</label><span ms-text="data['执行医生']"></span></li>
            </ul>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0131_2(xmlStr) {
		var $xmlDom = $(xmlStr),
				$nodes = {
					"检查项目": $xmlDom.find('component section')
				}
		
		return {
			data:{
                "医院": $xmlDom.find('org').text(),
				"申请医生": $xmlDom.find('participant name').text(),
				"申请科室": $xmlDom.find('participant dept').text(),
				"报告日期": $xmlDom.find('author time').text(),
				"检查名称": $nodes["检查项目"].attr('examname'),
				"检查部位": $nodes["检查项目"].attr('bodypartname'),
				"执行科室": $nodes["检查项目"].find('recognition entry performer assignedentity').text(),
				"执行医生": $nodes["检查项目"].find('recognition entry performer name').text(),
				"执行时间": $nodes["检查项目"].find('recognition entry performer time').text(),
				"临床诊断": $nodes["检查项目"].find('entry clinicdiagnosis').text(),
				"病情描述": $nodes["检查项目"].find('entry patientcondition').text(),
				"影像所见": $nodes["检查项目"].find('entry description').text(),
				"是否异常": function(){
								var state = $nodes["检查项目"].find('entry isabnormal').text().toUpperCase().trim();
								if(state == 'Y' || state == 'TRUE'){ state = '异常'}
								if(state == 'N' || state == 'FALSE'){ state = '正常'}
								return state;
							}()
			}
		}
	}
  function dsXmlToJson0131_1(xmlStr) {
		var $xmlDom = $(xmlStr),
				$nodes = {
					"检查项目": $xmlDom.find('component section')
				}
		
		return {
			data:{
				"医院": $xmlDom.find('org').text(),
				"申请医生": $xmlDom.find('participant name').text(),
				"申请科室": $xmlDom.find('participant dept').text(),
				"报告日期": $xmlDom.find('author time').text(),
				"检查名称": $nodes["检查项目"].attr('examname'),
				"检查部位": $nodes["检查项目"].attr('bodypartname'),
				"执行科室": $nodes["检查项目"].find('recognition entry performer assignedentity').text(),
				"执行医生": $nodes["检查项目"].find('recognition entry performer name').text(),
				"执行时间": $nodes["检查项目"].find('recognition entry performer time').text(),
				"临床诊断": $nodes["检查项目"].find('entry clinicdiagnosis').text(),
				"病情描述": $nodes["检查项目"].find('entry patientcondition').text(),
				"影像所见": $nodes["检查项目"].find('entry description').text(),
				"是否异常": function(){
								var state = $nodes["检查项目"].find('entry isabnormal').text().toUpperCase().trim();
								if(state == 'Y' || state == 'TRUE'){ state = '异常'}
								if(state == 'N' || state == 'FALSE'){ state = '正常'}
								return state;
							}()
			}
		}
	}
</script>

+ 165 - 0
template/0141.html

@ -0,0 +1,165 @@
<!--
	门诊处方
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['就诊日期']">
                <li ms-if="data['就诊日期']"><label>开方时间:</label><span ms-text="data['就诊日期']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['主治医生']">
                <li ms-if="data['主治医生']"><label>开方医生:</label><span ms-text="data['主治医生']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['科室'] || data['医院']">
                <li><label>科室/医院:</label>
                	<span ms-if="data['科室'] && data['医院']"><span ms-text="data['科室']"></span> <span ms-text="data['斜杠']"></span> <span ms-text="data['医院']"></span></span>
                	<span ms-if="data['科室'] && !(data['医院'])" ms-text="data['科室']"></span>
                	<span ms-if="!(data['科室']) && data['医院']" ms-text="data['医院']"></span>
                </li>
            </ul>
        </li>
    </ul>
    <ul class="info-list-t" ms-if="data['诊断'].length >0">
    	<h3 class="os-tit-o"><i class="icon i-visiting-result"></i><span class="os-t-o-name">临床诊断</span></h3>
        <li class="il-t-item">
            <p><span ms-repeat="data['诊断']"><span ms-text="el['诊断名称']"></span>&#x3000;</span></p>
        </li>
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o o-t-o-nb"><i class="icon Rp"></i><span class="os-t-o-name">Rp</span></h3>
        <li class="il-t-item list-mt" ms-repeat="data['门诊用药记录']" ms-repeat-item="el['用药子项']">
            <h4 class="os-tit-f" ms-if="item['药品名称']" ms-text="item['药品名称']"></h4>
            <ul class="os-base-info pl0" ms-if="el['用药方法'] || item['次剂量'] || el['频率']">
                <li><label>用法:</label><span class="pd-r" ms-if="el['用药方法']" ms-text="el['用药方法']"></span><span ms-if="item['次剂量']" class="pd-r"><span ms-if="item['次剂量']">每次</span><span ms-text="item['次剂量']"></span></span><span class="pd-r" ms-if="el['频率']" ms-text="el['频率']"></span></li>
            </ul>
            <ul class="os-base-info pl0" ms-if="item['药品规格'] || item['总剂量']">
            	<li ms-if="item['总剂量']" class="flex1"><label>总量:</label><span ms-text="item['总剂量']"></span></li>
                <li ms-if="item['药品规格']" class="flex2"><label>规格:</label><span ms-text="item['药品规格']"></span></li>
            </ul>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	dataCode = ['0101'];//新增数据请求
	function dsXmlToJson0141_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"用药记录" :$xmlDom.find('component section entry')
			} ;
		var resultData = []; 
		return {
			data:{
				"门诊用药记录": _.map($nodes["用药记录"].find('group'),function(el) {
					var $ele = $(el); 
					var $freq = $ele.find("freq");//频率
					var $method = $ele.find("method");//方法
					var $medicineArr = $ele.find("medicine");//用药子项
					return {
					 	"频率": $freq.text(),
					 	"用药方法":$method.text(),
					 	"用药子项": (function() {
					 		return _.map($medicineArr, function(secEl){
							 			var $eleItem = $(secEl);
							 			return {
										"药品名称": $eleItem.find('value').text().trim() ? $eleItem.find('value').text().trim(): $eleItem.find('custome').text().trim(),
										"药品规格": $eleItem.find('spec').text().trim(),
										"总剂量":$eleItem.find('totalQuantity').text()? parseFloat($eleItem.find('totalQuantity').text()).toFixed(0)+ $eleItem.find('totalUnit').text():'' ,
										"次剂量":$eleItem.find('doseQuantity').text()? parseFloat($eleItem.find('doseQuantity').text()).toFixed(2)+ $eleItem.find('doseUnit').text():'' 
									}
					 			})
					 	})()
					}	 
				})
			}
		}
	}
  function dsXmlToJson0141_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"用药记录" :$xmlDom
			} ;
		var resultData = []; 
		return {
			data:{
				"门诊用药记录": _.map($nodes["用药记录"].find('group'),function(el) {
					var $ele = $(el); 
					var $freq = $ele.attr("freq");//频率
					var $method = $ele.attr("method");//方法
					var $medicineArr = $ele.find("medicine");//用药子项
					 return {
					 	"频率": $freq,
					 	"用药方法":$method,
					 	"用药子项": (function() {
					 		return _.map($medicineArr, function(secEl){
					 			var $eleItem = $(secEl);
					 			return {
									"药品名称": $eleItem.text().trim(),
									"药品规格": $eleItem.attr('spec').trim(),
									"总剂量":$eleItem.attr('total_quantity')? parseFloat($eleItem.attr('total_quantity')).toFixed(0)+ $eleItem.attr('total_unit'):'' ,
									"次剂量":$eleItem.attr('dose_quantity')? parseFloat($eleItem.attr('dose_quantity')).toFixed(2)+ $eleItem.attr('dose_unit'):'' 
								}
					 		})
					 	})()
					 } 
				})
			}
		}
	}
  
  	function dsXmlToJson0101_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom.find('component section'),
				"病人基本信息":$xmlDom.find('recordTarget patient')
			} ;
		return {
			data:{
				"斜杠":'/',
				"就诊日期": $nodes["基本诊疗信息"].find('entry treatTime').text(),
				"主治医生": $nodes["基本诊疗信息"].find('entry doctor name').text(),
				"科室": $nodes["基本诊疗信息"].find('entry dept').text(),
				"医院": $xmlDom.find('org').text(),
				"诊断": (function() {
						var arr = _.map($nodes["基本诊疗信息"].find('diagnosis item'),function(el) {
							var $ele = $(el); 
							return {
								"诊断名称": $ele.find('icd').text(),
							}
						});
						return arr;
				})()
			}
		}
	}
	
	function dsXmlToJson0101_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"基本诊疗信息" :$xmlDom
			} ;
		return {
			data:{
				"科室": $nodes["基本诊疗信息"].find('dept').text(),
				"主治医生": $nodes["基本诊疗信息"].find('doctor').text(),
				"诊断": (function() {
						var arr = [];
						arr = _.map($nodes["基本诊疗信息"].find('diagnosis item'),function(el) {
							var $ele = $(el); 
							return {
								"诊断名称": $ele.text(),
							}
						})
						return arr;
				})()
			}
		}
	}
</script>

+ 121 - 0
template/0201.html

@ -0,0 +1,121 @@
<!--
	住院诊疗基本信息
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['住院科室'].trim()">
                <li><label>住院科室:</label><span ms-text="data['住院科室']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['住院次数'].trim()">
                <li><label>住院次数:</label><span ms-text="data['住院次数']"></span></li>
            </ul>
        </li>
    </ul>
    <ul class="info-list-t" ms-if="(data['症状'] && data['症状'].length > 0) && (!!data['症状'][0]['症状名称'].trim() || !!data['症状'][0]['发病日期'].trim())">
    	<li style="padding-left: 10px;">
            <h3 class="os-tit-o"><i class="icon i-dia-basis"></i><span class="os-t-o-name">症状</span></h3>
    	</li>
        <li class="il-t-item" ms-repeat="data['症状']">
        	<span ms-if="el['症状名称'].trim()">
	            <h4 class="os-tit-th">症状名称:</h4>
	            <p ms-html="el['症状名称']"></p>
        	</span>
        	<span ms-if="el['发病日期'].trim()">
	            <h4 class="os-tit-th">发病日期:</h4>
	            <p ms-html="el['发病日期']"></p>
        	</span>
        </li>
    </ul>
    <ul class="info-list-t" ms-if="(data['入院诊断'] && data['入院诊断'].length > 0) && (!!data['入院诊断'][0]['入院诊断名称'].trim() || !!data['入院诊断'][0]['确诊日期'].trim())">
    	<li style="padding-left: 10px;">
            <h3 class="os-tit-o"><i class="icon i-dia-basis"></i><span class="os-t-o-name">入院诊断</span></h3>
    	</li>
        <li class="il-t-item" ms-repeat="data['入院诊断']">
        	<span ms-if="el['入院诊断名称'] && el['入院诊断名称'].length > 0">
	            <h4 class="os-tit-th">入院诊断名称:</h4>
	            <p ms-html="el['入院诊断名称']"></p>
        	</span>
        	<span ms-if="el['确诊日期'] && el['确诊日期'].length > 0">
	            <h4 class="os-tit-th">确诊日期:</h4>
	            <p ms-html="el['确诊日期']"></p>
        	</span>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0201_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"住院基本诊疗信息" :$xmlDom.find('component section')
			} ;
		return {
			data:{
				"住院科室": $nodes["住院基本诊疗信息"].find('entry in dept').text() || " ",
				"住院次数": $nodes["住院基本诊疗信息"].find('entry residence serial').text() || " ",
				"症状": (function() {
					var $items = $nodes["住院基本诊疗信息"].find('entry symptom item');
					if($items.length==0) {
						return [{"症状名称": "", "发病日期": ""}];
					}
					return _.map($items,function(el) {
							var $el = $(el);
							return {
								"症状名称": $el.text(),
								"发病日期": $nodes["住院基本诊疗信息"].find('entry onsettime').text()
							}
					})
				})(),
				"入院诊断": 
					(function() {
						var $items = $nodes["住院基本诊疗信息"].find('entry indiagnosis item');
						if($items.length == 0) {
							return [{"入院诊断名称": "", "确诊日期":""}]
						}
						return _.map($items,function(el) {
								var $el = $(el);
								return {
									"入院诊断名称": $el.find('icd').text(),
									"确诊日期": ""
								}
						});
					})()
			}
		}
	}
	function dsXmlToJson0201_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"住院基本诊疗信息" :$xmlDom
			} ;
		return {
			data:{
				"住院科室": $nodes["住院基本诊疗信息"].find('indept').text() || " ",
				"住院次数": $nodes["住院基本诊疗信息"].find('residence').attr('serial') || " ",
				"症状": (function() {
					return [{"症状名称": "", "发病日期": ""}];
				})(),
				"入院诊断": (function() {
						var $items = $nodes["住院基本诊疗信息"].find('indiagnosis item');
						if($items.length == 0) {
							return [{"入院诊断名称": "", "确诊日期":""}]
						}
						return _.map($items,function(el) {
								var $el = $(el);
								return {
									"入院诊断名称": $el.find('icd').text(),
									"确诊日期": ""
								}
						});
					})()
			}
		}
	}
</script>

+ 144 - 0
template/0202.html

@ -0,0 +1,144 @@
<!--
	费用明细
-->
<div class="el-main">
    <div class="el-tit">
        <div class="el-t-l">费用总计</div>
        <div class="el-t-m" ms-text="data['总费用']"></div>
    </div>
    
    <ul class="el-list" ms-if="data['费用'] && data['费用'].length > 0">
        <li class="el-item" ms-repeat="data['费用']">
            <p ms-if="el['分类']">
            	<div ms-repeat-el="el['分类']">
            		<div class="el-title" ms-if="el['分类']['分类名称'] && el['分类']['分类名称'].trim()">
            			<span ms-text="el['分类']['分类名称']"></span>
            			<span ms-text="el['分类']['费用']"></span>
            		</div>
            		<div ms-if="el['分类']['详情']">
	            		<p ms-repeat-elem="el['分类']['详情']">
	            			<span class="el-list" ms-if="elem['项目名称'] && elem['项目名称'].length > 0">
            					<b ms-text="elem['项目名称']"></b>
            					<b ms-text="elem['总价']"></b>
	            			</span>
	            		</p>
            		</div>
            	</div>
            </p>
            <p ms-if="!el['分类']">
            	<div class="el-title noline" ms-if="el['项目名称'].trim()"><span ms-text="el['项目名称']"></span><span ms-text="el['总价']"></span></div>
            </p>
        </li>
    </ul>
</div>
<script>
 
	function dsXmlToJson0202_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"费用" :$xmlDom.find('component section entry')
			},
			countMon = 0;
		return {
			data:{
				"费用": (function() {
						var arr = [];
						_.each($nodes["费用"].find('group'),function(el,index) {
							var $ele = $(el); 
							
							arr[index] = {
								'分类': {
									'分类名称': $ele.find('type').text(),
									'费用': $ele.find('fee').text() ? parseFloat($ele.find('fee').text()).toFixed(2) : '',
									'详情': []
								}
							};
							_.each($ele.find('item'),function(it,ind) {
								var $it = $(it);
								arr[index]['分类']['详情'][ind] = {
									"扣费时间": $it.find('date').text(), // 该字段实际与提供的文档不符
									"项目名称": $it.find('value').text(),
									"规格": $it.find('spec').text(), 	// 提供的文档没有提到该字段
									"数量": $it.find('quantity').text() ? parseFloat($it.find('quantity').text()).toFixed(0) + $it.find('unit').text() : '',
									"单价": $it.find('price').text() ? parseFloat($it.find('price').text()).toFixed(2) : '',
									"总价": $it.find('total').text(),
									"备注": $it.find('notes').text()
								};
							});
						});
						if(arr.length==0) {
//							arr = [{
//									"扣费时间": " ",
//									"项目名称": " ",
//									"数量": " ",
//									"单价": " ",
//									"总价": " ",
//									"备注": " "
//							}]
							arr = []
						} else {
							countMon = (function () {
								var c = 0;
								console.log(arr);
								_.each(arr,function (item,index) {
									var n = item['分类']['费用'];
									c += parseFloat(n);
								});
								return c.toFixed(2);
							})();
						}
						return arr;
				})(),
				'总费用':countMon
			}
		}
	}
	function dsXmlToJson0202_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"费用" :$xmlDom
			} ;
		return {
			data:{
				"费用": (function() {
						var arr = [];
						_.each($nodes["费用"].find('item'),function(it) {
							var $it = $(it);
							arr.push({
								"项目名称": $it.text(),
								"扣费时间": $it.attr('date'), 
								"规格": $it.attr('spec'), 
								"数量": $it.attr('quantity') ? parseFloat($it.attr('quantity')).toFixed(2) + $it.attr('unit') : '',
								"单价": $it.attr('price') ? parseFloat($it.attr('price')).toFixed(2) : '',
								"总价": $it.attr('total'),
								"备注": $it.text('notes')
							});
						})
				
						if(arr.length==0) {
//							arr = [{	
//									"项目名称":" ",
//									"扣费时间":" ",
//									"规格": " ",
//									"数量": " ",
//									"单价": " ",
//									"总价": " ",
//									"备注": " "
//							}]
							arr = []
						} else {
							countMon = (_.reduce( _.pluck(arr, '总价'), function(a, b){ return a*1 + b*1; }, 0)).toFixed(2);
						}
						return arr;
				})(),
				'总费用':countMon
			}
		}
	}
</script>

+ 360 - 0
template/0211.html

@ -0,0 +1,360 @@
<!--
	住院病案首页
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-pat-info"></i><span class="os-t-o-name">患者信息</span></h3>
            <ul class="os-base-info" ms-if="data['姓名'].length > 0 || data['性别'].length > 0">
                <li ms-if="data['姓名'].length > 0"><label>姓&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;名:</label><span ms-text="data['姓名']"></span></li>
                <li ms-if="data['性别'].length > 0"><label>性&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;别:</label><span ms-text="data['性别']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['出生日期'].length > 0 || data['婚姻状态'].length > 0">
                <li ms-if="data['出生日期'].length > 0"><label>出生日期:</label><span ms-text="data['出生日期'] | date('yyyy-MM-dd')"></span></li>
                <li ms-if="data['婚姻状态'].length > 0"><label>婚姻状况:</label><span ms-text="data['婚姻状态']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['血型'].length > 0 || data['Rh血型'].length > 0">
                <li ms-if="data['血型'].length > 0"><label>血&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;型:</label><span ms-text="data['血型']"></span></li>
                <li ms-if="data['Rh血型'].length > 0"><label>Rh:</label><span ms-text="data['Rh血型']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="(data['新生儿体重'] || data['新生儿入院体重'])">
                <li><label>新生儿体重:</label><span>{{data['新生儿体重']}}g</span></li>
                <li><label>新生儿入院体重:</label><span>{{data['新生儿入院体重']}}g</span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['过敏药物'].length > 0">
                <li><label>过敏药物:</label><span ms-text="data['过敏药物']"></span></li>
            </ul>
        </li>
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-hos-sum"></i><span class="os-t-o-name">住院概要</span></h3>
            <ul class="os-base-info" ms-if="data['入院日期'].length > 0">
                <li><label>入院日期:</label><span ms-text="data['入院日期'] | date('yyyy-MM-dd')"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['入院途径'].length > 0">
                <li><label>入院途径:</label><span ms-text="data['入院途径']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['入院科室'].length > 0">
                <li><label>入院科室:</label><span ms-text="data['入院科室']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['转科科室'].length > 0">
                <li><label>转科科室:</label><span ms-text="data['转科科室']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['出院日期'].length > 0">
                <li><label>出院日期:</label><span ms-text="data['出院日期'] | date('yyyy-MM-dd')"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['出院科室'].length > 0">
                <li><label>出院科室:</label><span ms-text="data['出院科室']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['主要诊断'].length > 0">
                <li><label>主要诊断:</label><span ms-text="data['主要诊断']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['病例诊断'].length > 0">
                <li><label>病例诊断:</label><span ms-text="data['病例诊断']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['门(急)诊诊断'].length > 0">
                <li><label>门(急)诊诊断:</label><span ms-text="data['门(急)诊诊断']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['其他诊断'].length > 0">
                <li><label>其他诊断:</label><span ms-text="data['其他诊断']"></span></li>
            </ul>
        </li>
        <li class="il-item il-i-table" ms-if="data['手术信息'] && data['手术信息'].length > 0">
            <h3 class="os-tit-o"><i class="icon i-ope-rec"></i><span class="os-t-o-name">手术记录</span></h3>
            <table class="hp-table" cellpadding="0" cellspacing="0" border="0">
                <thead>
                    <tr>
                        <th style="width: 15%">日期</th>
                        <th style="width: 50%">手术名称</th>
                        <th style="width: 15%">切口愈合等级</th>
                        <th style="width: 20%">麻醉方式</th>
                    </tr>
                </thead>
                <tbody>
                    <tr ms-repeat="data['手术信息']">
                        <td>{{el['日期'] | date('yyyy-MM-dd')}}</td>
                        <td>{{el['名称']}}</td>
                        <td>{{el['切口愈合等级']}}</td>
                        <td>{{el['麻醉方式']}}</td>
                    </tr>
                </tbody>
            </table>
        </li>
        <!--<li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-dia-rec"></i><span class="os-t-o-name">症状记录</span></h3>
            <ul class="hp-list">
                <li><span>发病时间</span><span>症状记录</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
            </ul>
        </li>
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-zdjl"></i><span class="os-t-o-name">诊断记录</span></h3>
            <ul class="hp-list">
                <li><span>发病时间</span><span>症状记录</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
                <li><span>2017-03-27 10:55</span><span>马来酸非尼拉敏盐酸萘甲菲林</span></li>
            </ul>
        </li>-->
        <li class="il-item" ms-if="data['科主任'].length > 0 || data['主任(副主任)医师'].length > 0 || data['主治医师'].length > 0 || data['住院医师'].length > 0 ||data['责任护士'].length > 0 || data['进修医师'].length > 0">
            <h3 class="os-tit-o"><i class="icon i-doc-info"></i><span class="os-t-o-name">医生信息</span></h3>
            <ul class="os-base-info" ms-if="data['科主任'].length > 0 || data['主任(副主任)医师'].length > 0">
                <li ms-if="data['科主任'].length > 0"><label>科&nbsp;&nbsp;主&nbsp;&nbsp;任:</label><span ms-text="data['科主任']"></span></li>
                <li ms-if="data['主任(副主任)医师'].length > 0"><label>主任(副主任)医师:</label><span ms-text="data['主任(副主任)医师']"></span></li>
            </ul>	
            <ul class="os-base-info" ms-if="data['主治医师'].length > 0 || data['住院医师'].length > 0">
                <li ms-if="data['主治医师'].length > 0"><label>主治医师:</label><span ms-text="data['主治医师']"></span></li>
                <li ms-if="data['住院医师'].length > 0"><label>住院医师:</label><span ms-text="data['住院医师']"></span></li>
            </ul>	
            <ul class="os-base-info" ms-if="data['责任护士'].length > 0 || data['进修医师'].length > 0">
                <li ms-if="data['责任护士'].length > 0"><label>责任护士:</label><span ms-text="data['责任护士']"></span></li>
                <li ms-if="data['进修医师'].length > 0"><label>进修医师:</label><span ms-text="data['进修医师']"></span></li>
            </ul>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0211_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"病人信息": $xmlDom.find('recordTarget patient'),
				"病案信息": $xmlDom.find('component section entry')
			} ;
		return {
			data:{
//				"医疗付费方式": $nodes["病案信息"].find('payStyle').text(),
				"住院次数": $nodes["病案信息"].find('residence serial').text(),
				"病案号":$nodes["病案信息"].find('residence code').text(),
				"姓名": $nodes["病人信息"].find('name').text() || "&nbsp;",
				"性别": $nodes["病人信息"].find('sex').text() || "&nbsp;",
				"年龄": $nodes["病人信息"].find('age').text() || "&nbsp;",
				"不足一周岁的年龄": "",
				"新生儿出生体重": "",
				"新生儿入院体重": "",
				"国籍": $nodes["病人信息"].find('country').text(),
				"民族": $nodes["病人信息"].find('nation').text(),
				"婚姻状态": $nodes["病人信息"].find('marriage').text(),
				"职业": $nodes["病人信息"].find('work').text(),
				"籍贯": "",
				"出生地": $nodes["病人信息"].find('birthAdd').text(),
				"出生日期": $nodes["病人信息"].find('birthDate').text(),
				"户口地址": $nodes["病人信息"].find('registeredAddress').text(),
				"邮编": $nodes["病人信息"].find('postalCode').text(),
				"身份证号": $nodes["病人信息"].find('idno').text(),
				"单位": $nodes["病人信息"].find('organization').text(),
				"单位电话": "",
				"单位邮编": "",
				"联系人姓名": $xmlDom.find('contactPerson name').text(),
				"联系人关系": $xmlDom.find('contactPerson type').text(),
				"地址": $nodes["病人信息"].find('address').text(),
				"电话": $nodes["病人信息"].find('telephone').text(),
				"入院日期": $nodes["病案信息"].find('inTime').attr('value'),
				"入院科室": $nodes["病案信息"].find('inDept').text(),
				"转科科室": $nodes["病案信息"].find('moveDept').text(),
				"出院日期": $nodes["病案信息"].find('outTime').attr('value'),
				"出院科室": $nodes["病案信息"].find('outDept').text(),
				"实际住院天数": $nodes["病案信息"].find('inHospitalDay').text(),
				"入院途径": $nodes["病案信息"].find('inClass').text(),
				"门(急)诊诊断": $nodes["病案信息"].find('clinicDiagnosis').text(),
				"主要诊断": "",
				"其他诊断": "",
				"病例诊断": "",
				"病理号": $nodes["病案信息"].find('pathologyDiagnosis').attr('extension'),
				"疾病编码": "",
				"损伤、中毒的外部原因": "",
				"药物过敏标志": "",
				"过敏药物": $nodes["病案信息"].find('allergyDrugs').text(),
				"死亡患者尸检": $nodes["病案信息"].find('autopsy').text(),
				"血型": $nodes["病案信息"].find('bloodType').text(),
				"Rh血型": $nodes["病案信息"].find('bloodRh').text(),
				"科主任": $xmlDom.find('deptOfficer name').text(),
				"主任(副主任)医师": $xmlDom.find('directorDoctor name').text(),
				"主治医师": $xmlDom.find('inChargeDoctor name').text(),
				"住院医师": $xmlDom.find('residentDoctor name').text(),
				"责任护士": "",
				"进修医师": $xmlDom.find('learningDoctor name').text(),
				"实习医师": $xmlDom.find('intern name').text(),
				"编码员": $xmlDom.find('cataloger name').text(),
				"质控医师": $xmlDom.find('qcDoctor name').text(),
				"质控护士": $xmlDom.find('qcNurse name').text(),
				"质控医师签名时间": $xmlDom.find('qcDoctor time').attr('value'),
				"病案质量": $nodes["病案信息"].find('quality').text(),
				"手术信息": 
					(function() {
						var $operations = $nodes["病案信息"].find('operation item');
						if($operations.length == 0) {
//							return [{
//								"编码": "&nbsp;","日期": "&nbsp;","名称": "&nbsp;","手术级别": " ","手术医师": " ",
//								"Ⅰ助": " ","Ⅱ助": " ","切口愈合等级": " ","麻醉方式": " ","麻醉医师": " "
//							}]
							return [];
						}
						return _.map($operations,function(el) {
							var $el = $(el);
							return {
								"编码": $el.find('oper').attr('code'),
								"日期": $el.find('time').text(),
								"名称": $el.find('oper').text(),
								"手术级别": " ",
								"手术医师": $el.find('surgeon').text(),
								"Ⅰ助": $el.find('assistant1').text(),
								"Ⅱ助": $el.find('assistant2').text(),
								"切口愈合等级": $el.find('incisionAndHeal').text(),
								"麻醉方式": $el.find('anesthesia').text(),
								"麻醉医师": " "
							}
						});
					})()
				,
				"总费用": $nodes["病案信息"].find('fee total').text(),
				"床位费": $nodes["病案信息"].find('fee bed').text(),
				"护理费": $nodes["病案信息"].find('fee tend').text(),
				"西药费": $nodes["病案信息"].find('fee westmedicine').text(),
				"中成药费": $nodes["病案信息"].find('fee cnmedicine1').text(),
				"中草药费": $nodes["病案信息"].find('fee cnmedicine2').text(),
				"放射费": $nodes["病案信息"].find('fee radiation').text(),
				"化验费": $nodes["病案信息"].find('fee assay').text(),
				"输氧费": $nodes["病案信息"].find('fee oxygen').text(),
				"输血费": $nodes["病案信息"].find('fee blood').text(),
				"诊疗费": $nodes["病案信息"].find('fee diagnosis').text(),
				"治疗费": $nodes["病案信息"].find('fee treatment').text(),
				"手术费": $nodes["病案信息"].find('fee operation').text(),
				"接生费": $nodes["病案信息"].find('fee deliver').text(),
				"检查费": $nodes["病案信息"].find('fee check').text(),
				"麻醉费": $nodes["病案信息"].find('fee anaes').text(),
				"婴儿费": $nodes["病案信息"].find('fee baby').text(),
				"陪床费": $nodes["病案信息"].find('fee extraBed').text(),
				"其它费用1": $nodes["病案信息"].find('fee other1').text(),
				"其它费用2": $nodes["病案信息"].find('fee other2').text(),
				"其它费用3": $nodes["病案信息"].find('fee other3').text(),
				"其它费用4": $nodes["病案信息"].find('fee other4').text()
			}
		}
	}
  function dsXmlToJson0211_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"病人信息": $xmlDom.find('recordTarget patient'),
				"病案信息": $xmlDom.find('component section entry')
			} ;
		return {
			data:{
//				"医疗付费方式": $nodes["病案信息"].find('payStyle').text(),
				"住院次数": $nodes["病案信息"].find('residence serial').text(),
				"病案号":$nodes["病案信息"].find('residence code').text(),
				"姓名": $nodes["病人信息"].find('name').text() || "&nbsp;",
				"性别": $nodes["病人信息"].find('sex').text() || "&nbsp;",
				"年龄": $nodes["病人信息"].find('age').text() || "&nbsp;",
				"不足一周岁的年龄": "",
				"新生儿出生体重": "",
				"新生儿入院体重": "",
				"国籍": $nodes["病人信息"].find('country').text(),
				"民族": $nodes["病人信息"].find('nation').text(),
				"婚姻状态": $nodes["病人信息"].find('marriage').text(),
				"职业": $nodes["病人信息"].find('work').text(),
				"籍贯": "",
				"出生地": $nodes["病人信息"].find('birthAdd').text(),
				"出生日期": $nodes["病人信息"].find('birthDate').text(),
				"户口地址": $nodes["病人信息"].find('registeredAddress').text(),
				"邮编": $nodes["病人信息"].find('postalCode').text(),
				"身份证号": $nodes["病人信息"].find('idno').text(),
				"单位": $nodes["病人信息"].find('organization').text(),
				"单位电话": "",
				"单位邮编": "",
				"联系人姓名": $xmlDom.find('contactPerson name').text(),
				"联系人关系": $xmlDom.find('contactPerson type').text(),
				"地址": $nodes["病人信息"].find('address').text(),
				"电话": $nodes["病人信息"].find('telephone').text(),
				"入院日期": $nodes["病案信息"].find('inTime').attr('value'),
				"入院科室": $nodes["病案信息"].find('inDept').text(),
				"转科科室": $nodes["病案信息"].find('moveDept').text(),
				"出院日期": $nodes["病案信息"].find('outTime').attr('value'),
				"出院科室": $nodes["病案信息"].find('outDept').text(),
				"实际住院天数": $nodes["病案信息"].find('inHospitalDay').text(),
				"入院途径": $nodes["病案信息"].find('inClass').text(),
				"门(急)诊诊断": $nodes["病案信息"].find('clinicDiagnosis').text(),
				"主要诊断": "",
				"其他诊断": "",
				"病例诊断": "",
				"病理号": $nodes["病案信息"].find('pathologyDiagnosis').attr('extension'),
				"疾病编码": "",
				"损伤、中毒的外部原因": "",
				"药物过敏标志": "",
				"过敏药物": $nodes["病案信息"].find('allergyDrugs').text(),
				"死亡患者尸检": $nodes["病案信息"].find('autopsy').text(),
				"血型": $nodes["病案信息"].find('bloodType').text(),
				"Rh血型": $nodes["病案信息"].find('bloodRh').text(),
				"科主任": $xmlDom.find('deptOfficer name').text(),
				"主任(副主任)医师": $xmlDom.find('directorDoctor name').text(),
				"主治医师": $xmlDom.find('inChargeDoctor name').text(),
				"住院医师": $xmlDom.find('residentDoctor name').text(),
				"责任护士": "",
				"进修医师": $xmlDom.find('learningDoctor name').text(),
				"实习医师": $xmlDom.find('intern name').text(),
				"编码员": $xmlDom.find('cataloger name').text(),
				"质控医师": $xmlDom.find('qcDoctor name').text(),
				"质控护士": $xmlDom.find('qcNurse name').text(),
				"质控医师签名时间": $xmlDom.find('qcDoctor time').attr('value'),
				"病案质量": $nodes["病案信息"].find('quality').text(),
				"手术信息": 
					(function() {
						var $operations = $nodes["病案信息"].find('operation item');
						if($operations.length == 0) {
//							return [{
//								"编码": "&nbsp;","日期": "&nbsp;","名称": "&nbsp;","手术级别": " ","手术医师": " ",
//								"Ⅰ助": " ","Ⅱ助": " ","切口愈合等级": " ","麻醉方式": " ","麻醉医师": " "
//							}]
							return [];
						}
						return _.map($operations,function(el) {
							var $el = $(el);
							return {
								"编码": $el.find('oper').attr('code'),
								"日期": $el.find('time').text(),
								"名称": $el.find('oper').text(),
								"手术级别": " ",
								"手术医师": $el.find('surgeon').text(),
								"Ⅰ助": $el.find('assistant1').text(),
								"Ⅱ助": $el.find('assistant2').text(),
								"切口愈合等级": $el.find('incisionAndHeal').text(),
								"麻醉方式": $el.find('anesthesia').text(),
								"麻醉医师": " "
							}
						});
					})()
				,
				"总费用": $nodes["病案信息"].find('fee total').text(),
				"床位费": $nodes["病案信息"].find('fee bed').text(),
				"护理费": $nodes["病案信息"].find('fee tend').text(),
				"西药费": $nodes["病案信息"].find('fee westmedicine').text(),
				"中成药费": $nodes["病案信息"].find('fee cnmedicine1').text(),
				"中草药费": $nodes["病案信息"].find('fee cnmedicine2').text(),
				"放射费": $nodes["病案信息"].find('fee radiation').text(),
				"化验费": $nodes["病案信息"].find('fee assay').text(),
				"输氧费": $nodes["病案信息"].find('fee oxygen').text(),
				"输血费": $nodes["病案信息"].find('fee blood').text(),
				"诊疗费": $nodes["病案信息"].find('fee diagnosis').text(),
				"治疗费": $nodes["病案信息"].find('fee treatment').text(),
				"手术费": $nodes["病案信息"].find('fee operation').text(),
				"接生费": $nodes["病案信息"].find('fee deliver').text(),
				"检查费": $nodes["病案信息"].find('fee check').text(),
				"麻醉费": $nodes["病案信息"].find('fee anaes').text(),
				"婴儿费": $nodes["病案信息"].find('fee baby').text(),
				"陪床费": $nodes["病案信息"].find('fee extraBed').text(),
				"其它费用1": $nodes["病案信息"].find('fee other1').text(),
				"其它费用2": $nodes["病案信息"].find('fee other2').text(),
				"其它费用3": $nodes["病案信息"].find('fee other3').text(),
				"其它费用4": $nodes["病案信息"].find('fee other4').text()
			}
		}
	}
</script>

+ 227 - 0
template/0212.html

@ -0,0 +1,227 @@
<!--
	入院记录
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['入院时间'].length > 0">
                <li><label>入院日期:</label><span ms-text="data['入院时间']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['出院时间'] && data['出院时间'].length > 0">
                <li><label>出院日期:</label><span ms-text="data['出院时间'] | date('yyyy-MM-dd HH:mm:ss')"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['床号'].length > 0 || data['住院号'].length > 0">
                <li ms-if="data['床号'].length > 0"><label>床&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;号:</label><span ms-text="data['床号']"></span></li>
                <li ms-if="data['住院号'].length > 0"><label>住&nbsp;&nbsp;院&nbsp;&nbsp;号::</label><span ms-text="data['住院号']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['科室'].length > 0">
                <li><label>科室/医院:</label><span ms-text="data['科室']"></span></li>
            </ul>
        </li>
    </ul>
    <ul class="info-list-t">
    	<li style="padding-left: 10px;">
            <h3 class="os-tit-o"><i class="icon i-hos-sum"></i><span class="os-t-o-name">入院详情</span></h3>
    	</li>
        <li class="il-t-item" ms-if="data['主诉'].length > 0">
            <h4 class="os-tit-th">主诉:</h4>
            <p ms-text="data['主诉']"></p>
        </li>
        <li class="il-t-item" ms-if="data['入院情况'].length > 0">
            <h4 class="os-tit-t">入院情况:</h4>
            <p ms-text="data['入院情况']"></p>
        </li>
        <li class="il-t-item" ms-if="data['入院诊断'].length > 0">
            <h4 class="os-tit-t">入院诊断:</h4>
            <p ms-text="data['入院诊断']"></p>
        </li>
        <li class="il-t-item">
            <h4 class="os-tit-t">体格检查:</h4>
            <p>
            	<span ms-if="data['体温'].length > 0">体温:<span ms-text="data['体温']"></span>℃、</span>
            	<span ms-if="data['脉搏'].length > 0">脉搏:<span ms-text="data['脉搏']"></span>次/分、</span>
            	<span ms-if="data['呼吸'].length > 0">呼吸:<span ms-text="data['呼吸']"></span>次/分</span>
            </p>
            <p>
            	<span ms-if="data['体温'].length > 0">心率:<span ms-text="data['体温']"></span>次/分、</span>
            	<span ms-if="data['血压'].length > 0">血压:<span ms-text="data['血压']"></span>mmhg</span>
            </p>
            <p ms-if="data['一般状况'].length > 0">
            	<span>一般状况:<span ms-text="data['一般状况']"></span></span>
            </p>
            <p ms-if="data['皮肤黏膜'].length > 0">
            	<span>皮肤黏膜:<span ms-text="data['皮肤黏膜']"></span></span>
            </p>
            <p ms-if="data['淋巴结'].length > 0">
            	<span>淋巴结:<span ms-text="data['淋巴结']"></span></span>
            </p>
            <p ms-if="data['头部'].length > 0">
            	<span>头部:<span ms-text="data['头部']"></span></span>
            </p>
            <p ms-if="data['颈部'].length > 0">
            	<span>颈部:<span ms-text="data['颈部']"></span></span>
            </p>
            <p ms-if="data['胸部'].length > 0">
            	<span>胸部:<span ms-text="data['胸部']"></span></span>
            </p>
            <p ms-if="data['肺'].length > 0">
            	<span>肺:<span ms-text="data['肺']"></span></span>
            </p>
            <p ms-if="data['周围血管'].length > 0">
            	<span>周围血管:<span ms-text="data['周围血管']"></span></span>
            </p>
            <p ms-if="data['腹部'].length > 0">
            	<span>腹部:<span ms-text="data['腹部']"></span></span>
            </p>
            <p ms-if="data['生殖器'].length > 0">
            	<span>生殖器:<span ms-text="data['生殖器']"></span></span>
            </p>
            <p ms-if="data['肛门直肠'].length > 0">
            	<span>肛门直肠:<span ms-text="data['肛门直肠']"></span></span>
            </p>
            <p ms-if="data['颈椎四肢'].length > 0">
            	<span>颈椎四肢:<span ms-text="data['颈椎四肢']"></span></span>
            </p>
            <p ms-if="data['神经系统'].length > 0">
            	<span>神经系统:<span ms-text="data['神经系统']"></span></span>
            </p>
        </li>
    </ul>
    <ul class="info-list-t">
    	<li style="padding-left: 10px;">
            <h3 class="os-tit-o"><i class="icon i-dia-basis"></i><span class="os-t-o-name">患者病史</span></h3>
    	</li>
        <li class="il-t-item" ms-if="data['过敏史'] && data['过敏史'].length > 0">
            <h4 class="os-tit-th">过敏史:</h4>
            <p ms-text="data['过敏史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['现病史'] && data['现病史'].length > 0">
            <h4 class="os-tit-t">现病史:</h4>
            <p ms-text="data['现病史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['个人史'] && data['个人史'].length > 0">
            <h4 class="os-tit-t">个人史:</h4>
            <p ms-text="data['个人史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['既往史'] && data['既往史'].length > 0">
            <h4 class="os-tit-t">既往史:</h4>
            <p ms-text="data['既往史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['婚育史'] && data['婚育史'].length > 0">
            <h4 class="os-tit-t">婚育史:</h4>
            <p ms-text="data['婚育史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['月经史'] && data['月经史'].length > 0">
            <h4 class="os-tit-t">月经史:</h4>
            <p ms-text="data['月经史']"></p>
        </li>
        <li class="il-t-item" ms-if="data['家族史'] && data['家族史'].length > 0">
            <h4 class="os-tit-t">家族史:</h4>
            <p ms-text="data['家族史']"></p>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0212_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				
			} ;
		var resultData = []; 
		return {
			data:{
				"科室": $xmlDom.find('component section entry inDept').text(),
				"床号": $xmlDom.find('component section entry bed').text(),
				"住院号": "",
				"姓名": $xmlDom.find('recordTarget patient name').text(),
				"性别": $xmlDom.find('recordTarget patient sex').text(),
				"出生日期": $xmlDom.find('recordTarget patient birthDate').text(),
				"婚姻": $xmlDom.find('recordTarget marriage').text(),
				"民族": $xmlDom.find('recordTarget nation').text(),
				"职业":  $xmlDom.find('recordTarget patient work ').text(),
				"入院时间": $xmlDom.find('component section entry inTime').attr('value'),
				"出院时间": $xmlDom.find('component section entry outTime').attr('value'),
				"主诉": $xmlDom.find('component section entry cc').text(),
				"入院情况": $xmlDom.find('component section entry inCondition').text(),
				"入院诊断": $xmlDom.find('component section entry inDiagnosis').text(),
				"既往史": $xmlDom.find('component section entry diseaseHistory').text(),
				"个人史": $xmlDom.find('component section entry personalHistory').text(),
				"婚育史": $xmlDom.find('component section entry marriageHistory').text(),
				"月经史": $xmlDom.find('component section entry mensesHistory').text(),
				"家族史": $xmlDom.find('component section entry familyHistory').text(),
				"体温": $xmlDom.find('component section entry physicalExam temperature').text(),
				"脉搏": $xmlDom.find('component section entry physicalExam pulse').text(),
				"呼吸": $xmlDom.find('component section entry physicalExam respiration').text(),
				"血压": $xmlDom.find('component section entry physicalExam diastolicPressure').text()+'--'+$xmlDom.find('component section entry physicalExam systolicPressure').text(),
				"一般状况": $xmlDom.find('component section entry physicalExam status').text(),
				"皮肤黏膜": $xmlDom.find('component section entry physicalExam skin').text(),
				"淋巴结": $xmlDom.find('component section entry physicalExam lymph').text(),
				"头部": $xmlDom.find('component section entry physicalExam head').text(),
				"颈部": $xmlDom.find('component section entry physicalExam neck').text(),
				"胸部": $xmlDom.find('component section entry physicalExam chest').text(),
				"肺": $xmlDom.find('component section entry physicalExam lung').text(),
				"心": $xmlDom.find('component section entry physicalExam heart').text(),
				"周围血管": $xmlDom.find('component section entry physicalExam peripheralVessels').text(),
				"腹部": $xmlDom.find('component section entry physicalExam abdomen').text(),
				"生殖器": $xmlDom.find('component section entry physicalExam genitalia').text(),
				"肛门直肠": $xmlDom.find('component section entry physicalExam anus').text(),
				"颈椎四肢": $xmlDom.find('component section entry physicalExam acantha').text(),
				"神经系统": $xmlDom.find('component section entry physicalExam neural').text()
			}
		}
	}
  function dsXmlToJson0212_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				
			} ;
		var resultData = []; 
		return {
			data:{
				"科室": $xmlDom.find('component section entry inDept').text(),
				"床号": $xmlDom.find('component section entry bed').text(),
				"住院号": "",
				"姓名": $xmlDom.find('recordTarget patient name').text(),
				"性别": $xmlDom.find('recordTarget patient sex').text(),
				"出生日期": $xmlDom.find('recordTarget patient birthDate').text(),
				"婚姻": $xmlDom.find('recordTarget marriage').text(),
				"民族": $xmlDom.find('recordTarget nation').text(),
				"职业":  $xmlDom.find('recordTarget patient work ').text(),
				"入院时间": $xmlDom.find('component section entry inTime').attr('value'),
				"出院时间": $xmlDom.find('component section entry outTime').attr('value'),
				"主诉": $xmlDom.find('component section entry cc').text(),
				"入院情况": $xmlDom.find('component section entry inCondition').text(),
				"入院诊断": $xmlDom.find('component section entry inDiagnosis').text(),
				"既往史": $xmlDom.find('component section entry diseaseHistory').text(),
				"个人史": $xmlDom.find('component section entry personalHistory').text(),
				"婚育史": $xmlDom.find('component section entry marriageHistory').text(),
				"月经史": $xmlDom.find('component section entry mensesHistory').text(),
				"家族史": $xmlDom.find('component section entry familyHistory').text(),
				"体温": $xmlDom.find('component section entry physicalExam temperature').text(),
				"脉搏": $xmlDom.find('component section entry physicalExam pulse').text(),
				"呼吸": $xmlDom.find('component section entry physicalExam respiration').text(),
				"血压": $xmlDom.find('component section entry physicalExam diastolicPressure').text()+'--'+$xmlDom.find('component section entry physicalExam systolicPressure').text(),
				"一般状况": $xmlDom.find('component section entry physicalExam status').text(),
				"皮肤黏膜": $xmlDom.find('component section entry physicalExam skin').text(),
				"淋巴结": $xmlDom.find('component section entry physicalExam lymph').text(),
				"头部": $xmlDom.find('component section entry physicalExam head').text(),
				"颈部": $xmlDom.find('component section entry physicalExam neck').text(),
				"胸部": $xmlDom.find('component section entry physicalExam chest').text(),
				"肺": $xmlDom.find('component section entry physicalExam lung').text(),
				"心": $xmlDom.find('component section entry physicalExam heart').text(),
				"周围血管": $xmlDom.find('component section entry physicalExam peripheralVessels').text(),
				"腹部": $xmlDom.find('component section entry physicalExam abdomen').text(),
				"生殖器": $xmlDom.find('component section entry physicalExam genitalia').text(),
				"肛门直肠": $xmlDom.find('component section entry physicalExam anus').text(),
				"颈椎四肢": $xmlDom.find('component section entry physicalExam acantha').text(),
				"神经系统": $xmlDom.find('component section entry physicalExam neural').text()
			}
		}
	}
</script>

+ 147 - 0
template/0213.html

@ -0,0 +1,147 @@
<!--
	出院小结
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['姓名'].length > 0 || data['性别'].length > 0">
                <li ms-if="data['姓名'].length > 0"><label>姓&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;名:</label><span ms-text="data['姓名']"></span></li>
                <li ms-if="data['性别'].length > 0"><label>性&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;别:</label><span ms-text="data['性别']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['民族'].length > 0 || data['婚姻'].length > 0">
                <li ms-if="data['民族'].length > 0"><label>民&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;族:</label><span ms-text="data['民族']"></span></li>
                <li ms-if="data['婚姻'].length > 0"><label>婚姻状况:</label><span ms-text="data['婚姻']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['入院时间'].length > 0">
                <li><label>入院日期:</label><span ms-text="data['入院时间'] | date('yyyy-MM-dd')"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['入院时间'].length > 0">
                <li><label>出院日期:</label><span ms-text="data['出院时间'] | date('yyyy-MM-dd')"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['住院天数'].length > 0">
                <li><label>住院天数:</label><span ms-text="data['住院天数']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['床号'].length > 0 || data['住院号'].length > 0">
                <li ms-if="data['床号'].length > 0"><label>床&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;号:</label><span ms-text="data['床号']"></span></li>
                <li ms-if="data['住院号'].length > 0"><label>住&nbsp;&nbsp;院&nbsp;&nbsp;号::</label><span ms-text="data['住院号']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['科室'].length > 0">
                <li><label>科室/医院:</label><span ms-text="data['科室']"></span></li>
            </ul>
        </li>
    </ul>
    <ul class="info-list-t">
    	<li style="padding-left: 10px;">
            <h3 class="os-tit-o"><i class="icon i-hos-sum"></i><span class="os-t-o-name">具体内容</span></h3>
    	</li>
        <li class="il-t-item" ms-if="data['入院情况'].length > 0">
            <h4 class="os-tit-th">入院情况:</h4>
            <p ms-text="data['入院情况']"></p>
        </li>
        <li class="il-t-item" ms-if="data['入院诊断'].length > 0">
            <h4 class="os-tit-t">入院诊断:</h4>
            <p ms-text="data['入院诊断']"></p>
        </li>
        <li class="il-t-item" ms-if="data['出院诊断']">
            <h4 class="os-tit-t">出院诊断:</h4>
            <p ms-text="data['出院诊断']"></p>
        </li>
        <li class="il-t-item" ms-if="data['诊疗经过'].length > 0">
            <h4 class="os-tit-t">诊疗经过:</h4>
            <p ms-text="data['诊疗经过']"></p>
        </li>
        <li class="il-t-item" ms-if="data['出院情况'].length > 0">
            <h4 class="os-tit-t">出院情况:</h4>
            <p ms-text="data['出院情况']"></p>
        </li>
        <li class="il-t-item" ms-if="data['出院医嘱'].length > 0">
            <h4 class="os-tit-t">出院医嘱:</h4>
            <p ms-text="data['出院医嘱']"></p>
        </li>
        <li class="il-t-item" ms-if="data['治疗结果'].length > 0">
            <h4 class="os-tit-t">治疗结果:</h4>
            <p ms-text="data['治疗结果']"></p>
        </li>
        <li class="il-t-item" ms-if="data['住院医师'].length > 0">
            <h4 class="os-tit-t">住院医师:</h4>
            <p ms-text="data['住院医师']"></p>
        </li>
        <li class="il-t-item" ms-if="data['主治医师'].length > 0">
            <h4 class="os-tit-t">主治医师:</h4>
            <p ms-text="data['主治医师']"></p>
        </li>
        <li class="il-t-item" ms-if="data['时间'].length > 0">
            <h4 class="os-tit-t">时间:</h4>
            <p ms-text="data['时间'] | date('yyyy-MM-dd HH:mm:ss')"></p>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0213_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
			
			} ;
		var resultData = []; 
		return {
			data:{
				"科室": $xmlDom.find('component section entry inDept').text(),
				"床号": $xmlDom.find('component section entry bed').text(),
				"住院号": "",
				"姓名": $xmlDom.find('recordTarget patient name').text(),
				"性别": $xmlDom.find('recordTarget patient sex').text(),
				"民族": $xmlDom.find('recordTarget nation').text(),
				"婚姻": $xmlDom.find('recordTarget marriage').text(),
				"入院时间": $xmlDom.find('component section entry inTime').attr('value'),
				"出院时间": $xmlDom.find('component section entry outTime').attr('value'),
				"住院天数": $xmlDom.find('component section entry inHospitalDay').text(),
				"入院情况": $xmlDom.find('component section entry inCondition').text(),
				"入院诊断": $xmlDom.find('component section entry inDiagnosis').text(),
				"出院诊断": $xmlDom.find('component section entry outDiagnosis').text(),
				"诊疗经过": $xmlDom.find('component section entry remedialCourse').text(),
				"出院情况": $xmlDom.find('component section entry outCondition').text(),
				"出院医嘱": $xmlDom.find('component section entry outOrder').text(),
				"治疗结果": $xmlDom.find('component section entry remedialResult').text(),
				"住院医师": $xmlDom.find('residentDoctor name').text(),
				"主治医师": $xmlDom.find('inChargeDoctor name').text(),
				"时间": $xmlDom.find('inChargeDoctor time').attr('value')
			}
		}
	}
  function dsXmlToJson0213_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
			
			} ;
		var resultData = []; 
		return {
			data:{
				"科室": $xmlDom.find('component section entry inDept').text(),
				"床号": $xmlDom.find('component section entry bed').text(),
				"住院号": "",
				"姓名": $xmlDom.find('recordTarget patient name').text(),
				"性别": $xmlDom.find('recordTarget patient sex').text(),
				"民族": $xmlDom.find('recordTarget nation').text(),
				"婚姻": $xmlDom.find('recordTarget marriage').text(),
				"入院时间": $xmlDom.find('component section entry inTime').attr('value'),
				"出院时间": $xmlDom.find('component section entry outTime').attr('value'),
				"住院天数": $xmlDom.find('component section entry inHospitalDay').text(),
				"入院情况": $xmlDom.find('component section entry inCondition').text(),
				"入院诊断": $xmlDom.find('component section entry inDiagnosis').text(),
				"出院诊断": $xmlDom.find('component section entry outDiagnosis').text(),
				"诊疗经过": $xmlDom.find('component section entry remedialCourse').text(),
				"出院情况": $xmlDom.find('component section entry outCondition').text(),
				"出院医嘱": $xmlDom.find('component section entry outOrder').text(),
				"治疗结果": $xmlDom.find('component section entry remedialResult').text(),
				"住院医师": $xmlDom.find('residentDoctor name').text(),
				"主治医师": $xmlDom.find('inChargeDoctor name').text(),
				"时间": $xmlDom.find('inChargeDoctor time').attr('value')
			}
		}
	}
</script>

+ 94 - 0
template/0214.html

@ -0,0 +1,94 @@
<!--
	长期医嘱
-->
<div class="os-main">
    <div class="lto-con" ms-repeat="data['长期医嘱']">
        <h3 class="lto-tit"><span ms-text="el['下嘱时间'] | date('yyyy-MM-dd HH:mm:ss')"></span>~<span ms-text="el['停嘱时间'] | date('yyyy-MM-dd HH:mm:ss')"></span></h3>
        <p class="lto-info" ms-if="el['医嘱内容'].length>0"><span>医嘱内容:</span>{{el['医嘱内容']}}</p>
        <p class="lto-info"><span>频&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;次:</span>{{el['频次']}}</p>
        <p class="lto-info" ms-if="el['下嘱医生'].length>0"><span>下嘱医生:</span>{{el['下嘱医生']}}</p>
    </div>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0214_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"长期医嘱子项" :$xmlDom.find('component section entry order item')
			} ;
		return {
			data:{
				"长期医嘱": (function() {
					var $items = $nodes["长期医嘱子项"];
					if($items.length==0) {
						return [{
										"医嘱内容": "",
										"频次": " ",
										"下嘱医生": " ",
										"检验护士": " ",
										"下嘱时间": " ",
										"停嘱时间": " ",
										"停嘱医生": " ",
										"停嘱护士": " "}];
					}
					return _.map($items,function(el) {
							var $el = $(el);
							return {
								"医嘱内容": $el.find('orderName').text(),
								"频次": $el.find('freq').text(),
								"下嘱医生": $el.find('orderDoctor').text(),
								"检验护士": $el.find('checkNurse').text(),
								"下嘱时间": $el.find('orderTime').text(),
								"停嘱时间": $el.find('stopTime').text(),
								"停嘱医生": $el.find('stopDoctor').text(),
								"停嘱护士": $el.find('stopNurse').text()
							}
					})
				})()
			}
		}
	}
	function dsXmlToJson0214_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"长期医嘱子项" :$xmlDom
			} ;
		return {
			data:{
				"长期医嘱": (function() {
					var $items = $nodes["长期医嘱子项"].find('item');
					if($items.length==0) {
						return [{
										"医嘱内容": "",
										"频次": " ",
										"下嘱医生": " ",
										"检验护士": " ",
										"下嘱时间": " ",
										"停嘱时间": " ",
										"停嘱医生": " ",
										"停嘱护士": " "}];
					}
					return _.map($items,function(el) {
							var $el = $(el);
							return {
								"医嘱内容": $el.attr('order_name'),
								"频次": $el.attr('freq'),
								"下嘱医生": $el.attr('order_doctor'),
								"检验护士": $el.attr('check_nurse'),
								"下嘱时间": $el.attr('order_time'),
								"停嘱时间": $el.attr('stop_time'),
								"停嘱医生": $el.attr('stop_doctor'),
								"停嘱护士": $el.attr('stop_nurse')
							}
					})
				})()
			}
		}
	}
</script>

+ 95 - 0
template/0215.html

@ -0,0 +1,95 @@
<!--
	临时医嘱
-->
<div class="os-main">
    <div class="lto-con" ms-repeat="data['临时医嘱']">
    	<p style="padding-top: 15px;"></p>
        <p class="lto-info" ms-if="el['下嘱时间'].length>0"><span>下嘱时间:</span>{{el['下嘱时间'] | date('yyyy-MM-dd HH:mm:ss')}}</p>
        <p class="lto-info" ms-if="el['医嘱内容'].length>0"><span>医嘱内容:</span>{{el['医嘱内容']}}</p>
        <p class="lto-info" ms-if="el['下嘱医生'].length>0"><span>下嘱医生:</span>{{el['下嘱医生']}}</p>
        <p class="lto-info" ms-if="el['检验护士'].length>0"><span>检验护士:</span>{{el['检验护士']}}</p>
    </div>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0215_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"临时医嘱子项" :$xmlDom.find('component section entry order item')
			} ;
		return {
			data:{
				"临时医嘱": (function() {
					var $items = $nodes["临时医嘱子项"];
					if($items.length==0) {
						return [{
										"医嘱内容": "&nbsp;",
										"频次": "&nbsp;",
										"下嘱医生": "&nbsp;",
										"检验护士": "&nbsp;",
										"下嘱时间": "&nbsp;",
										"执行护士": "&nbsp;",
										"执行时间": "&nbsp;"
										}];
					}
					return _.map($items,function(el) {
							var $el = $(el);
							return {
								"医嘱内容": $el.find('orderName').text(),
								"频次": $el.find('freq').text(),
								"下嘱医生": $el.find('orderDoctor').text(),
								"检验护士": $el.find('checkNurse').text(),
								"下嘱时间": $el.find('orderTime').text(),
								"执行护士": $el.find('execNurse').text(),
								"执行时间": $el.find('execTime').text()
							}
					})
				})()
			}
		}
	}
	function dsXmlToJson0215_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"临时医嘱子项" :$xmlDom.find('item')
			} ;
		return {
			data:{
				"临时医嘱": (function() {
					var $items = $nodes["临时医嘱子项"];
					if($items.length==0) {
						return [{
										"医嘱内容": "&nbsp;",
										// "频次": " ",
										"下嘱医生": "&nbsp;",
										"检验护士": "&nbsp;",
										"下嘱时间": "&nbsp;",
										"执行护士": "&nbsp;",
										"执行时间": "&nbsp;"
										}];
					}
					return _.map($items,function(el) {
							var $el = $(el);
							return {
								"医嘱内容": $el.attr('std_code'),
								// "频次": $el.attr('freq').text(),
								"下嘱医生": $el.attr('order_doctor'),
								"检验护士": $el.attr('check_nurse'),
								"下嘱时间": $el.attr('order_time'),
								"执行护士": $el.attr('exec_nurse'),
								"执行时间": $el.attr('exec_time')
							}
					})
				})()
			}
		}
	}
</script>

+ 126 - 0
template/0216.html

@ -0,0 +1,126 @@
<!--
	住院手术记录
-->
 <div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['记录者'] && data['记录者'].length > 0">
                <li><label>记&nbsp;&nbsp;录&nbsp;&nbsp;者:</label><span ms-text="data['记录者']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['记录时间'] && data['记录时间'].length > 0">
                <li><label>记录时间:</label><span ms-text="data['记录时间'] | date('yyyy-MM-dd HH:mm:ss')"></span></li>
            </ul>
        </li>
    </ul>
    <ul class="info-list-t">
    	<li style="padding-left: 10px;">
            <h3 class="os-tit-o"><i class="icon i-dia-basis"></i><span class="os-t-o-name">手术具体信息</span></h3>
    	</li>
        <li class="il-t-item" ms-if="data['手术操作名称'] && data['手术操作名称'].length > 0">
            <h4 class="os-tit-th">手术操作名称:</h4>
            <p ms-text="data['手术操作名称']"></p>
        </li>
        <li class="il-t-item" ms-if="data['手术时间'] && data['手术时间'].length > 0">
            <h4 class="os-tit-t">手术时间:</h4>
            <p ms-text="data['手术时间'] | date('yyyy-MM-dd HH:mm:ss')"></p>
        </li>
        <li class="il-t-item" ms-if="data['手术编码'] && data['手术编码'].length > 0">
            <h4 class="os-tit-t">手术编码:</h4>
            <p ms-text="data['手术编码']"></p>
        </li>
        <li class="il-t-item" ms-if="data['术前诊断'] && data['术前诊断'].length > 0">
            <h4 class="os-tit-t">术前诊断:</h4>
            <p ms-text="data['术前诊断']"></p>
        </li>
        <li class="il-t-item" ms-if="data['诊断ICD10'] && data['诊断ICD10'].length > 0">
            <h4 class="os-tit-t">诊断ICD10:</h4>
            <p ms-text="data['诊断ICD10']"></p>
        </li>
        <li class="il-t-item" ms-if="data['术中诊断'] && data['术中诊断'].length > 0">
            <h4 class="os-tit-t">术中诊断:</h4>
            <p ms-text="data['术中诊断']"></p>
        </li>
        <li class="il-t-item" ms-if="data['麻醉方法'] && data['麻醉方法'].length > 0">
            <h4 class="os-tit-t">麻醉方法:</h4>
            <p ms-text="data['麻醉方法']"></p>
        </li>
        <li class="il-t-item" ms-if="data['手术经过'] && data['手术经过'].length > 0">
            <h4 class="os-tit-t">手术经过:</h4>
            <p ms-text="data['手术经过']"></p>
        </li>
    </ul>
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-doc-info"></i><span class="os-t-o-name">手术人员</span></h3>
            <ul class="os-base-info" ms-if="(data['手术医师'] || data['麻醉医师']) && (data['手术医师'].length > 0 || data['麻醉医师'].length > 0)">
                <li><label>手术医师:</label><span ms-text="data['手术医师']"></span></li>
                <li><label>麻醉医师:</label><span ms-text="data['麻醉医师']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="(data['Ⅰ助'] || data['Ⅱ助']) && (data['Ⅰ助'].length > 0 || data['Ⅱ助'].length > 0)">
                <li><label>Ⅰ助:</label><span ms-text="data['Ⅰ助']"></span></li>
                <li><label>Ⅱ助:</label><span ms-text="data['Ⅱ助']"></span></li>
            </ul>
        </li>
   </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
    
<script>
	function dsXmlToJson0216_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				
			} ;
		var resultData = []; 
		return {
			data:{
				"记录者": $xmlDom.find('author name').text(),
				"记录时间": $xmlDom.find('author time').text(),
				"手术医师": $xmlDom.find('surgeon name').text(),
				"Ⅰ助": $xmlDom.find('assistant1 name').text(),
				"Ⅱ助": $xmlDom.find('assistant2 name').text(),
				"麻醉医师": $xmlDom.find('anesthesiologist name').text(),
				"手术操作名称": $xmlDom.find('component section entry operation').text(),
				"手术时间": $xmlDom.find('component section entry time').text(),
				"手术编码": $xmlDom.find('component section entry operation').attr('code'),
				"术前诊断": $xmlDom.find('component section entry preDiagnosis').text(),
				"诊断ICD10": $xmlDom.find('component section entry preDiagnosis').attr('code'),
				"术中诊断": $xmlDom.find('component section entry midDiagnosis').text(),
				"诊断ICD10": $xmlDom.find('component section entry midDiagnosis').attr('code'),
				"麻醉方法": $xmlDom.find('component section entry anesthesia').text(),
				"手术经过": $xmlDom.find('component section entry operationCourse').text()
			}
		}
	}
  function dsXmlToJson0216_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				
			} ;
		var resultData = []; 
		return {
			data:{
				"记录者": $xmlDom.find('author name').text(),
				"记录时间": $xmlDom.find('author time').text(),
				"手术医师": $xmlDom.find('surgeon name').text(),
				"Ⅰ助": $xmlDom.find('assistant1 name').text(),
				"Ⅱ助": $xmlDom.find('assistant2 name').text(),
				"麻醉医师": $xmlDom.find('anesthesiologist name').text(),
				"手术操作名称": $xmlDom.find('component section entry operation').text(),
				"手术时间": $xmlDom.find('component section entry time').text(),
				"手术编码": $xmlDom.find('component section entry operation').attr('code'),
				"术前诊断": $xmlDom.find('component section entry preDiagnosis').text(),
				"诊断ICD10": $xmlDom.find('component section entry preDiagnosis').attr('code'),
				"术中诊断": $xmlDom.find('component section entry midDiagnosis').text(),
				"诊断ICD10": $xmlDom.find('component section entry midDiagnosis').attr('code'),
				"麻醉方法": $xmlDom.find('component section entry anesthesia').text(),
				"手术经过": $xmlDom.find('component section entry operationCourse').text()
			}
		}
	}
</script>

+ 146 - 0
template/0221.html

@ -0,0 +1,146 @@
<!--
	检验详情
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['检验项目']">
                <li><label>检验项目:</label><span ms-text="data['检验项目']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['标本类型']">
                <li><label>标本类型:</label><span ms-text="data['标本类型']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['报告日期']">
                <li><label>报告时间:</label><span ms-text="data['报告日期']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['医院']">
                <li><label>医&#x3000;&#x3000;院:</label><span ms-text="data['医院']"></span></li>
            </ul>
        </li>
    </ul>
    <ul class="info-list" ms-if="data['检验结果'].length >0">
        <li class="il-item il-i-table">
            <h3 class="os-tit-o"><i class="icon i-test-results"></i><span class="os-t-o-name">检验结果</span></h3>
            <table class="id-table" cellpadding="0" cellspacing="0" border="0">
                <thead>
                    <tr>
                        <th style="width: 30%;text-align: center;font-weight: bold;">项目名称</th>
                        <th style="width: 20%;text-align: center;font-weight: bold;">检验结果</th>
                        <th style="width: 30%;text-align: center;font-weight: bold;">参考值</th>
                        <th style="width: 20%;text-align: center;font-weight: bold;">单位</th>
                    </tr>
                </thead>
                <tbody>
                    <tr ms-repeat="data['检验结果']">
                        <td ms-text="el['项目名称']"></td>
                        <td ms-text="el['结果']"></td>
                        <td ms-text="el['参考值']"></td>
                        <td ms-text="el['单位']"></td>
                    </tr>
                </tbody>
            </table>
        </li>
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o"><i class="icon i-inspectors"></i><span class="os-t-o-name">检验人员</span></h3>
        <li class="il-t-item pl0">
            <ul class="os-base-info" ms-if="data['检验时间']">
                <li><label>检验时间:</label><span ms-text="data['检验时间']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['申请科室'] || data['申请医生']">
                <li ms-if="data['申请科室']"><label>申请科室:</label><span ms-text="data['申请科室']"></span></li>
                <li ms-if="data['申请医生']"><label>申请医生:</label><span ms-text="data['申请医生']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['检验医生'] || data['审核者']">
                <li ms-if="data['检验医生']"><label>检&nbsp;&nbsp;验&nbsp;&nbsp;员:</label><span ms-text="data['检验医生']"></span></li>
                <li ms-if="data['审核者']"><label>复&nbsp;&nbsp;核&nbsp;&nbsp;人:</label><span ms-text="data['审核者']"></span></li>
            </ul>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0221_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"检验的专业类型名称" :$xmlDom.find('component section')
			} ;
		
		return {
			data:{
				"医院": $xmlDom.find('org').text(),
				"标本类型": $nodes["检验的专业类型名称"].find('entry classcode').text(),
				"检验项目": $nodes["检验的专业类型名称"].attr('displayname'),
				"检验结果": _.map($nodes["检验的专业类型名称"].find('entry observation'),function(el) {
							var $ele = $(el),
								$referRange = $ele.find('referenceRange observationRange');
							return {
								"项目名称": $ele.find('code').attr('displayname'),
								"结果": $ele.find('>value').text(),
								"参考值": (function() {
											var $low = $referRange.find('low'),
													$high = $referRange.find('high'),
													low = $.trim($low.find('value').text()),
													high = $.trim($high.find('value').text());
											if(low && high) {
												return low + '~' + high;
											} else {
												return low || high || "";
											}
								})(),
								"单位": $referRange.find('low unit').text(),
							}
						}),
				"检验时间": $nodes["检验的专业类型名称"].find('entry performer time').text(),
				"检验医生": $nodes["检验的专业类型名称"].find('entry performer name').text(),
				"审核者": $xmlDom.find('authenticator name').text(),
				"申请医生":$xmlDom.find('participant name').text(),
				"申请科室":$xmlDom.find('participant dept').text(),
			}
		}
	}
	function dsXmlToJson0221_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"检验的专业类型名称" :$xmlDom.find('component section')
			} ;
		
		return {
			data:{
				"医院": $xmlDom.find('org').text(),
				"标本类型": $nodes["检验的专业类型名称"].find('entry classcode').text(),
				"检验项目": $nodes["检验的专业类型名称"].attr('displayname'),
				"检验结果": _.map($nodes["检验的专业类型名称"].find('entry observation'),function(el) {
							var $ele = $(el),
								$referRange = $ele.find('referenceRange observationRange');
							return {
								"项目名称": $ele.find('code').attr('displayname'),
								"结果": $ele.find('>value').text(),
								"参考值": (function() {
											var $low = $referRange.find('low'),
													$high = $referRange.find('high'),
													low = $.trim($low.find('value').text()),
													high = $.trim($high.find('value').text());
											if(low && high) {
												return low + '~' + high;
											} else {
												return low || high || "";
											}
								})(),
								"单位": $referRange.find('low unit').text(),
							}
						}),
				"检验时间": $nodes["检验的专业类型名称"].find('entry performer time').text(),
				"检验医生": $nodes["检验的专业类型名称"].find('entry performer name').text(),
				"审核者": $xmlDom.find('authenticator name').text(),
				"申请医生":$xmlDom.find('participant name').text(),
				"申请科室":$xmlDom.find('participant dept').text(),
			}
		}
	}
</script>

+ 132 - 0
template/0231.html

@ -0,0 +1,132 @@
<!--
	检查详情
-->
<div class="os-main">
    <ul class="info-list">
        <li class="il-item">
            <h3 class="os-tit-o"><i class="icon i-ess-info"></i><span class="os-t-o-name">基本信息</span></h3>
            <ul class="os-base-info" ms-if="data['检查名称']">
                <li><label>检查名称:</label><span ms-text="data['检查名称']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['检查类型'] || data['检查部位']">
                <li ms-if="data['检查类型']"><label>类&#x3000;&#x3000;型:</label><span ms-text="data['检查类型']"></span></li>
                <li ms-if="data['检查部位']"><label>部&#x3000;&#x3000;位:</label><span ms-text="data['检查部位']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['报告日期']">
                <li><label>报告时间:</label><span ms-text="data['报告日期']"></span></li> 
            </ul>
            <ul class="os-base-info" ms-if="data['医院']">
                <li><label>医&#x3000;&#x3000;院:</label><span ms-text="data['医院']"></span></li>
            </ul>
        </li>
        <li class="il-item">
        </li>
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o"><i class="icon i-ins-res"></i><span class="os-t-o-name">检查结果</span></h3>
        <li class="il-t-item jiancha-mt">
            <h4 class="os-tit-th">临床诊断:</h4>
            <p ms-text="data['临床诊断']"></p>
        </li>
        <li class="il-t-item jiancha-mt" ms-if="data['病情描述']">
            <h4 class="os-tit-t">病情描述:</h4>
            <p ms-text="data['病情描述']"></p>
        </li>
        <li class="il-t-item jiancha-mt" ms-if="data['影像所见']">
            <h4 class="os-tit-t">影像所见:</h4>
            <p ms-text="data['影像所见']"></p>
        </li>
        <li class="il-t-item jiancha-mt" ms-if="data['是否异常'] || data['影像结论']">
            <h4 class="os-tit-t">影像结论:</h4>
            <p class="ying-pb">是否异常:<span ms-text="data['是否异常']"></span></p>
            <p class="ying-pb" ms-if="data['影像结论']" ms-text="data['影像结论']"></p>
        </li>
        <!--<li class="il-t-item">
            <h4 class="os-tit-t">影像图片:</h4>
            <ul class="i-i-list">
                <li class="i-i-l-item"><img src="" alt=""></li>
            </ul>
        </li>-->
    </ul>
    <ul class="info-list-t">
    	<h3 class="os-tit-o"><i class="icon i-inspectors"></i><span class="os-t-o-name">检查人员</span></h3>
        <li class="il-t-item pl0">
            <ul class="os-base-info" ms-if="data['执行时间']">
                <li><label>检查时间:</label><span ms-text="data['执行时间']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['申请科室'] || data['申请医生']">
                <li ms-if="data['申请科室']"><label>申请科室:</label><span ms-text="data['申请科室']"></span></li>
                <li ms-if="data['申请医生']"><label>申请医生:</label><span ms-text="data['申请医生']"></span></li>
            </ul>
            <ul class="os-base-info" ms-if="data['执行科室'] || data['执行医生']">
                <li ms-if="data['执行科室']"><label>执行科室:</label><span ms-text="data['执行科室']"></span></li>
                <li ms-if="data['执行医生']"><label>执行医生:</label><span ms-text="data['执行医生']"></span></li>
            </ul>
        </li>
    </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0231_2(xmlStr) {
		var $xmlDom = $(xmlStr),
				$nodes = {
					"检查项目": $xmlDom.find('component section')
				}
		
		return {
			data:{
                "医院": $xmlDom.find('org').text(),
				"申请医生": $xmlDom.find('participant name').text(),
				"申请科室": $xmlDom.find('participant dept').text(),
				"报告日期": $xmlDom.find('author time').text(),
				"检查名称": $nodes["检查项目"].attr('examname'),
				"检查部位": $nodes["检查项目"].attr('bodypartname'),
				"执行科室": $nodes["检查项目"].find('recognition entry performer assignedentity').text(),
				"执行医生": $nodes["检查项目"].find('recognition entry performer name').text(),
				"执行时间": $nodes["检查项目"].find('recognition entry performer time').text(),
				"临床诊断": $nodes["检查项目"].find('entry clinicdiagnosis').text(),
				"病情描述": $nodes["检查项目"].find('entry patientcondition').text(),
				"影像所见": $nodes["检查项目"].find('entry description').text(),
				"是否异常": function(){
								var state = $nodes["检查项目"].find('entry isabnormal').text().toUpperCase().trim();
								if(state == 'Y' || state == 'TRUE'){ state = '异常'}
								if(state == 'N' || state == 'FALSE'){ state = '正常'}
								return state;
							}()
			}
		}
	}
  function dsXmlToJson0231_1(xmlStr) {
		var $xmlDom = $(xmlStr),
				$nodes = {
					"检查项目": $xmlDom.find('component section')
				}
		
		return {
			data:{
				"医院": $xmlDom.find('org').text(),
				"申请医生": $xmlDom.find('participant name').text(),
				"申请科室": $xmlDom.find('participant dept').text(),
				"报告日期": $xmlDom.find('author time').text(),
				"检查名称": $nodes["检查项目"].attr('examname'),
				"检查部位": $nodes["检查项目"].attr('bodypartname'),
				"执行科室": $nodes["检查项目"].find('recognition entry performer assignedentity').text(),
				"执行医生": $nodes["检查项目"].find('recognition entry performer name').text(),
				"执行时间": $nodes["检查项目"].find('recognition entry performer time').text(),
				"临床诊断": $nodes["检查项目"].find('entry clinicdiagnosis').text(),
				"病情描述": $nodes["检查项目"].find('entry patientcondition').text(),
				"影像所见": $nodes["检查项目"].find('entry description').text(),
				"是否异常": function(){
								var state = $nodes["检查项目"].find('entry isabnormal').text().toUpperCase().trim();
								if(state == 'Y' || state == 'TRUE'){ state = '异常'}
								if(state == 'N' || state == 'FALSE'){ state = '正常'}
								return state;
							}()
			}
		}
	}
</script>

+ 88 - 0
template/0241.html

@ -0,0 +1,88 @@
<!--
	住院用药
-->
<div class="os-main">
	<ul class="info-list-t">
    	<h3 class="os-tit-o o-t-o-nb"><i class="icon Rp"></i><span class="os-t-o-name">Rp</span></h3>
        <li class="il-t-item list-mt" ms-repeat="data['门诊用药记录']" ms-repeat-item="el['用药子项']">
            <h4 class="os-tit-f" ms-if="item['药品名称']" ms-text="item['药品名称']"></h4>
            <ul class="os-base-info pl0" ms-if="el['用药方法'] || item['次剂量'] || el['频率']">
                <li><label>用法:</label><span class="pd-r" ms-if="el['用药方法']" ms-text="el['用药方法']"></span><span ms-if="item['次剂量']" class="pd-r"><span ms-if="item['次剂量']">每次</span><span ms-text="item['次剂量']"></span></span><span class="pd-r" ms-if="el['频率']" ms-text="el['频率']"></span></li>
            </ul>
            <ul class="os-base-info pl0" ms-if="item['药品规格'] || item['总剂量']">
            	<li ms-if="item['总剂量']" class="flex1"><label>总量:</label><span ms-text="item['总剂量']"></span></li>
                <li ms-if="item['药品规格']" class="flex2"><label>规格:</label><span ms-text="item['药品规格']"></span></li>
            </ul>
        </li>
   </ul>
    <p class="p-bottom">详情内容以医院出具纸质病例为准</p>
</div>
<script>
	function dsXmlToJson0241_2(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"用药记录" :$xmlDom.find('component section entry')
			} ;
		var resultData = []; 
		return {
			data:{
				"门诊用药记录": _.map($nodes["用药记录"].find('group'),function(el) {
					var $ele = $(el); 
					var $freq = $ele.find("freq");//频率
					var $method = $ele.find("method");//方法
					var $medicineArr = $ele.find("medicine");//用药子项
					return {
					 	"频率": $freq.text(),
					 	"用药方法":$method.text(),
					 	"用药子项": (function() {
					 		return _.map($medicineArr, function(secEl){
							 			var $eleItem = $(secEl);
							 			return {
										"药品名称": $eleItem.find('value').text().trim() ? $eleItem.find('value').text().trim(): $eleItem.find('custome').text().trim(),
										"药品规格": $eleItem.find('spec').text().trim(),
										"总剂量":$eleItem.find('totalQuantity').text()? parseFloat($eleItem.find('totalQuantity').text()).toFixed(0)+ $eleItem.find('totalUnit').text():'',
										"次剂量":$eleItem.find('doseQuantity').text()? parseFloat($eleItem.find('doseQuantity').text()).toFixed(2)+ $eleItem.find('doseUnit').text():''
									}
					 			})
					 	})()
					}	 
				})
			}
		}
	}
  function dsXmlToJson0241_1(xmlStr) {
		var $xmlDom = $(xmlStr),
			$nodes = {
				"用药记录" :$xmlDom
			} ;
		var resultData = []; 
		return {
			data:{
				"门诊用药记录": _.map($nodes["用药记录"].find('group'),function(el) {
					var $ele = $(el); 
					var $freq = $ele.attr("freq");//频率
					var $method = $ele.attr("method");//方法
					var $medicineArr = $ele.find("medicine");//用药子项
					 return {
					 	"频率": $freq,
					 	"用药方法":$method,
					 	"用药子项": (function() {
					 		return _.map($medicineArr, function(secEl){
					 			var $eleItem = $(secEl);
					 			return {
									"药品名称": $eleItem.text().trim(),
									"药品规格": $eleItem.attr('spec').trim(),
									"总剂量":$eleItem.attr('total_quantity')? parseFloat($eleItem.attr('total_quantity')).toFixed(0)+ $eleItem.attr('total_unit'):'',
									"次剂量":$eleItem.attr('dose_quantity')? parseFloat($eleItem.attr('dose_quantity')).toFixed(2)+ $eleItem.attr('dose_unit'):''
								}
					 		})
					 	})()
					 } 
				})
			}
		}
	}
</script>