/** @Name:layui.treeGrid 树表格操作 @Author:luonianqing @License:public, 首先感谢Layui的原码分项,本源码是在table.js的基面上改进的,如有BUG,请直接在FLY上留言,代码存在冗余,因为看得不够细,没时间,项目需要,Layui没有,我就写了 去除了排序,因为树型不支持表头排序 去除了全选,因为没有意义,不支持分页,因为也没有意义,也支除了记录条数限制,下级分支有多少条就展示多少条,只支持动态数据,不支持静态 增加了行选功能.即只要单击,其它的全不选,只有结点会被选中, 增加了treeDataCache,因为树是多级目录关系 不支持,树枝选择,下面全部被选中,也许将来会支持,现在满足项目需求即可,新增了注代码 layui.use(['element', 'layer', 'form', 'upload', 'treeGrid'], function () { var treeGrid = layui.treeGrid; //很重要 var treeTable = treeGrid.render({ elem: '#treeTable' ,id:'treeTable' //这一行一定需要,和上一行不一样,这个是需要用来作数据修改和定位的 ,levelField:'level' //表示树的级别字段,从1开始,从数据Json传回时必须是1,2,3,4,5... ,paramName:'parentId' //表示树向下展开时向后端传弟的参数名称,值就是作用的paramValue ,paramValue:'id' //表示树向下展开时向后端传弟的参数名称,一定是{url}+{paramName}={paramValue},现在就是{url}+parentId={id} ,scopeField:'baseCode' //表示树的图标作用在哪个字段前面,并表示父亲需向下传递的参数,一定是{url}+{parentParame}={baseCode},该字段一定要显示 ,url: '/data.jsp' ,cols: [[ {checkbox: true, fixed: true} ,{field:'baseCode', title: '编号', width:140} ,{field:'id', title: 'ID', width:80} ,{field:'keyName', width:140 ,title: '键名称'} ,{field:'keyShortName', width:140 ,title: '键简称'} ,{field:'keyValue', width:140 ,title: '键值'} ,{field:'typeValue', width:140 ,title: '类型值'} ,{field:'note', width:140 ,title: '备注'} ,{field:'sort', width:100 ,title: '排序',fixed: 'right'} ,{field:'isPublic', width:100 ,title: '状态',fixed: 'right'} ]] ,limit:300 ,page: false ,height: 'full-200' }); }); */ layui.define(['laytpl', 'layer', 'form'], function(exports){ "use strict"; var treeDataCache; var $ = layui.$ ,laytpl = layui.laytpl ,layer = layui.layer ,form = layui.form ,hint = layui.hint() ,device = layui.device() //外部接口 ,treeGrid = { config: { checkName: 'LAY_CHECKED' //是否选中状态的字段名 ,indexName: 'LAY_TABLE_INDEX' //下标索引名 } //全局配置项 ,cache: {} //数据缓存 ,index: layui.treeGrid ? (layui.treeGrid.index + 10000) : 0 //设置全局项 ,set: function(options){ var that = this; that.config = $.extend({}, that.config, options); return that; } //事件监听 ,on: function(events, callback){ return layui.onevent.call(this, MOD_NAME, events, callback); } } //操作当前实例 ,thisTable = function(){ var that = this ,options = that.config ,id = options.id; id && (thisTable.config[id] = options); return { reload: function(options){ that.reload.call(that, options); } ,config: options } } //字符常量 ,MOD_NAME = 'treeGrid', ELEM = '.layui-table', THIS = 'layui-this', SHOW = 'layui-show', HIDE = 'layui-hide', DISABLED = 'layui-disabled', NONE = 'layui-none' ,ELEM_VIEW = 'layui-table-view', ELEM_HEADER = '.layui-table-header', ELEM_BODY = '.layui-table-body', ELEM_MAIN = '.layui-table-main', ELEM_FIXED = '.layui-table-fixed', ELEM_FIXL = '.layui-table-fixed-l', ELEM_FIXR = '.layui-table-fixed-r', ELEM_TOOL = '.layui-table-tool', ELEM_PAGE = '.layui-table-page', ELEM_SORT = '.layui-table-sort', ELEM_EDIT = 'layui-table-edit', ELEM_HOVER = 'layui-table-hover' //thead区域模板 ,TPL_HEADER = function(options){ var rowCols = '{{#if(item2.colspan){}} colspan="{{item2.colspan}}"{{#} if(item2.rowspan){}} rowspan="{{item2.rowspan}}"{{#}}}'; options = options || {}; return [''//{{# if(d.data.even){ }}lay-even{{# } }} ,'' ,'{{# layui.each(d.data.cols, function(i1, item1){ }}' ,'' ,'{{# layui.each(item1, function(i2, item2){ }}' ,'{{# if(item2.fixed && item2.fixed !== "right"){ left = true; } }}' ,'{{# if(item2.fixed === "right"){ right = true; } }}' ,function(){ if(options.fixed && options.fixed !== 'right'){ return '{{# if(item2.fixed && item2.fixed !== "right"){ }}'; } if(options.fixed === 'right'){ return '{{# if(item2.fixed === "right"){ }}'; } return ''; }() ,'' ,(options.fixed ? '{{# }; }}' : '') ,'{{# }); }}' ,'' ,'{{# }); }}' ,'' ,'
' ,'
' ,'{{# if(item2.type === "checkbox"){ }}' //复选框 ,'' ,'{{# } else { }}' ,'{{item2.title||""}}' ,'{{# } }}' ,'
' ,'
'].join(''); } //tbody区域模板 ,TPL_BODY = ['' ,'' ,'
'].join('') //主模板 ,TPL_MAIN = ['
' ,'{{# if(d.data.toolbar){ }}' ,'
' ,'{{# } }}' ,'
' ,'{{# var left, right; }}' ,'
' ,TPL_HEADER() ,'
' ,'
' ,TPL_BODY ,'
' ,'{{# if(left){ }}' ,'
' ,'
' ,TPL_HEADER({fixed: true}) ,'
' ,'
' ,TPL_BODY ,'
' ,'
' ,'{{# }; }}' ,'{{# if(right){ }}' ,'
' ,'
' ,TPL_HEADER({fixed: 'right'}) ,'
' ,'
' ,'
' ,TPL_BODY ,'
' ,'
' ,'{{# }; }}' ,'
' ,'' ,'
'].join('') ,_WIN = $(window) ,_DOC = $(document) //构造器 ,Class = function(options){ var that = this; that.index = ++treeGrid.index; that.config = $.extend({}, that.config, treeGrid.config, options); that.render(); }; //默认配置 Class.prototype.config = { limit: 10 //每页显示的数量 ,loading: true //请求数据时,是否显示loading ,cellMinWidth: 60 //所有单元格默认最小宽度 }; //表格渲染 Class.prototype.render = function(){ var that = this ,options = that.config; options.elem = $(options.elem); options.where = options.where || {}; options.id = options.id || options.elem.attr('id'); //请求参数的自定义格式 options.request = $.extend({}, options.request) //响应数据的自定义格式 options.response = $.extend({ statusName: 'code' ,statusCode: 0 ,msgName: 'msg' ,dataName: 'data' ,countName: 'count' }, options.response); if(!options.elem[0]) return that; that.setArea(); //动态分配列宽高 //开始插入替代元素 var othis = options.elem ,hasRender = othis.next('.' + ELEM_VIEW) //主容器 ,reElem = that.elem = $(laytpl(TPL_MAIN).render({ VIEW_CLASS: ELEM_VIEW ,data: options ,index: that.index //索引 })); options.index = that.index; //生成替代元素 hasRender[0] && hasRender.remove(); //如果已经渲染,则Rerender if(options.refresh) { } else { othis.after(reElem); } // console.log(othis); //各级容器 that.layHeader = reElem.find(ELEM_HEADER); that.layMain = reElem.find(ELEM_MAIN); that.layBody = reElem.find(ELEM_BODY); that.layFixed = reElem.find(ELEM_FIXED); that.layFixLeft = reElem.find(ELEM_FIXL); that.layFixRight = reElem.find(ELEM_FIXR); that.layTool = reElem.find(ELEM_TOOL); that.layPage = reElem.find(ELEM_PAGE); that.layTool.html( laytpl($(options.toolbar).html()||'').render(options) ); if(options.height) that.fullSize(); //设置body区域高度 //如果多级表头,则填补表头高度 if(options.cols.length > 1){ var th = that.layFixed.find(ELEM_HEADER).find('th'); th.height(that.layHeader.height() - 1 - parseFloat(th.css('padding-top')) - parseFloat(th.css('padding-bottom'))); } //请求数据 that.pullData(); that.events(); }; //根据列类型,定制化参数 Class.prototype.initOpts = function(item){ var that = this, options = that.config ,initWidth = { checkbox: 48 ,space: 15 }; //让 type 参数兼容旧版本 if(item.checkbox) item.type = "checkbox"; if(item.space) item.type = "space"; if(!item.type) item.type = "normal"; if(item.type !== "normal"){ item.unresize = true; item.width = item.width || initWidth[item.type]; } }; //动态分配列宽高 Class.prototype.setArea = function(){ var that = this, options = that.config ,colNums = 0 //列个数 ,autoColNums = 0 //自动列宽的列个数 ,autoWidth = 0 //自动列分配的宽度 ,countWidth = 0 //所有列总宽度和 ,cntrWidth = options.width || function(){ //获取容器宽度 //如果父元素宽度为0(一般为隐藏元素),则继续查找上层元素,直到找到真实宽度为止 var getWidth = function(parent){ var width, isNone; parent = parent || options.elem.parent() width = parent.width(); try { isNone = parent.css('display') === 'none'; } catch(e){} if(parent[0] && (!width || isNone)) return getWidth(parent.parent()); return width; }; return getWidth(); }(); //统计列个数 that.eachCols(function(){ colNums++; }); //减去边框差 cntrWidth = cntrWidth - function(){ return (options.skin === 'line' || options.skin === 'nob') ? 2 : colNums + 1; }(); //遍历所有列 layui.each(options.cols, function(i1, item1){ layui.each(item1, function(i2, item2){ var width; if(!item2){ item1.splice(i2, 1); return; } that.initOpts(item2); width = item2.width || 0; if(item2.colspan > 1) return; if(/\d+%$/.test(width)){ item2.width = width = Math.floor((parseFloat(width) / 100) * cntrWidth); } else if(!width){ //列宽未填写 item2.width = width = 0; autoColNums++; } countWidth = countWidth + width; }); }); that.autoColNums = autoColNums; //记录自动列数 //如果未填充满,则将剩余宽度平分。否则,给未设定宽度的列赋值一个默认宽 (cntrWidth > countWidth && autoColNums) && ( autoWidth = (cntrWidth - countWidth) / autoColNums ); layui.each(options.cols, function(i1, item1){ layui.each(item1, function(i2, item2){ var minWidth = item2.minWidth || options.cellMinWidth; if(item2.colspan > 1) return; if(item2.width === 0){ item2.width = Math.floor(autoWidth >= minWidth ? autoWidth : minWidth); //不能低于设定的最小宽度 } }); }); //高度铺满:full-差距值 if(options.height && /^full-\d+$/.test(options.height)){ that.fullHeightGap = options.height.split('-')[1]; options.height = _WIN.height() - that.fullHeightGap; } }; //表格重载 Class.prototype.reload = function(options){ var that = this; if(that.config.data && that.config.data.constructor === Array) delete that.config.data; that.config = $.extend({}, that.config, options); that.render(); }; //获得数据 Class.prototype.pullData = function(curr, loadIndex){ var that = this ,options = that.config ,request = options.request ,response = options.response; that.startTime = new Date().getTime(); //渲染开始时间 var parentId=0; if(options.url){ //Ajax请求 if(options.refresh) { var refreshId=options.refresh.parentNodeId; var jsonArray =$.parseJSON("{\""+options.paramName+"\":"+refreshId+"}"); options.refresh=jsonArray; parentId=refreshId; } var params = {}; $.ajax({ type: options.method || 'get' ,url: options.url ,data: $.extend(params,options.refresh, options.where) ,dataType: 'json' ,success: function(res){ if(res[response.statusName] != response.statusCode){ that.renderForm(); return that.layMain.html('
'+ (res[response.msgName] || '返回的数据状态异常') +'
'); } //console.log(options); that.renderData(parentId,res); options.time = (new Date().getTime() - that.startTime) + ' ms'; //耗时(接口请求+视图渲染) loadIndex && layer.close(loadIndex); typeof options.done === 'function' && options.done(res, curr, res[response.countName]); } ,error: function(e, m){ that.layMain.html('
数据接口请求异常
'); that.renderForm(); loadIndex && layer.close(loadIndex); } }); } else if(options.data && options.data.constructor === Array){ //已知数据 var res = {} ,startLimit = curr*options.limit - options.limit res[response.dataName] = options.data.concat().splice(startLimit, options.limit); res[response.countName] = options.data.length; that.renderData(parentId,res); typeof options.done === 'function' && options.done(res, curr, res[response.countName]); } }; //遍历表头 Class.prototype.eachCols = function(callback){ var cols = $.extend(true, [], this.config.cols) ,arrs = [], index = 0; //重新整理表头结构 layui.each(cols, function(i1, item1){ layui.each(item1, function(i2, item2){ //如果是组合列,则捕获对应的子列 if(item2.colspan > 1){ var childIndex = 0; index++ item2.CHILD_COLS = []; layui.each(cols[i1 + 1], function(i22, item22){ if(item22.PARENT_COL || childIndex == item2.colspan) return; item22.PARENT_COL = index; item2.CHILD_COLS.push(item22); childIndex = childIndex + (item22.colspan > 1 ? item22.colspan : 1); }); } if(item2.PARENT_COL) return; //如果是子列,则不进行追加,因为已经存储在父列中 arrs.push(item2) }); }); //重新遍历列,如果有子列,则进入递归 var eachArrs = function(obj){ layui.each(obj || arrs, function(i, item){ if(item.CHILD_COLS) return eachArrs(item.CHILD_COLS); callback(i, item); }); }; eachArrs(); }; //展开或者收起树 Class.prototype.exportTree=function(parentId,isExport) { var that = this; // var dbClickI=that.layMain.find('tr[data-id="'+ parentId +'"]').find('.layui-tree-head'); var dbClickI = that.layMain.find('tr[data-id="' + parentId + '"]').find('.treeHead'); var dbHtml = dbClickI.html(); if (dbHtml == undefined || (dbHtml.indexOf("folder") < 0)) { return; } if(isExport) { dbClickI.attr('isExport',"true"); //dbClickI.html(''); dbClickI.html(""); } else { dbClickI.attr('isExport', "false"); //dbClickI.html(''); dbClickI.html(""); that.layMain.find('tr[data-parent-id="'+ parentId +'"]').each( function() { var tempParentId=$(this).attr("data-id"); $(this).hide(); that.exportTree(tempParentId,false); }); that.layFixLeft.find('tr[data-parent-id="'+ parentId +'"]').each( function() { var tempParentId=$(this).attr("data-id"); $(this).hide(); that.exportTree(tempParentId,false); }); that.layFixRight.find('tr[data-parent-id="'+ parentId +'"]').each( function() { var tempParentId=$(this).attr("data-id"); $(this).hide(); that.exportTree(tempParentId,false); }); } }; //数据渲染 Class.prototype.renderData = function(parentId,res){ if(treeDataCache==null) { treeDataCache=new Object(); treeDataCache[parentId]=res[this.config.response.dataName]; } else { treeDataCache[parentId]=res[this.config.response.dataName]; } var that = this ,options = that.config ,data = treeDataCache|| [] // ,trs = [] ,trs_fixed = [] ,trs_fixed_r = [] //渲染视图 ,render = function(){ //后续性能提升的重点 //treeHead内容 var levelField=options.levelField ,paramName=options.paramName ,paramValue=options.paramValue ,scopeField=options.scopeField; //处理展开还是隐藏 var childIsExport=false; //var dbClickI=that.layMain.find('tr[data-id="'+ parentId +'"]').find('.layui-tree-head'); var dbClickI = that.layMain.find('tr[data-id="' + parentId + '"]').find('.treeHead'); if(dbClickI.length>0) { if(dbClickI.attr('isExport')=='true')//原来是展开,现在要收起 { childIsExport=false; } else { childIsExport=true; } } else { childIsExport=true; } layui.each(data[parentId], function(i1, item1){ var tds = [], tds_fixed = [], tds_fixed_r = []; if(item1.length === 0) return; that.eachCols(function(i3, item3){ var field = item3.field || i3, content = item1[field] ,cell = that.getColElem(that.layHeader, field); if(content === undefined || content === null) content = ''; if(item3.colspan > 1) return; var treeHeadImg=""; if(field===scopeField) { var level=item1[options.levelField]; var temp="" for(var i=1;i"; //treeHeadImg = temp + ""; treeHeadImg = temp + "" + (item1["HasChild"] ? "" : "") + " "; } //td内容 var td = ['' ,'
' + function(){ var tplData = $.extend(true,item1); //渲染复选框列视图 if(item3.type === 'checkbox'){ return ''; } //解析工具列模板 if(item3.toolbar){ return laytpl($(item3.toolbar).html()||'').render(tplData); } return item3.templet ? treeHeadImg+(laytpl($(item3.templet).html() || String(content)).render(tplData)) : treeHeadImg+content+""; }() ,'
'].join(''); tds.push(td); if(item3.fixed && item3.fixed !== 'right') tds_fixed.push(td); if(item3.fixed === 'right') tds_fixed_r.push(td); }); var tr = '1) { // if (level % 2 === 0) { // tr += ' style="background-image: linear-gradient(rgba(250,250,250,0.1),rgba(250,250,250,,0.1))"'; // } else { // tr += ' style="background-image: linear-gradient(rgba(0,0,0,0.1),rgba(0,0,0,0.1))"'; // } //} //if (item1[levelField] == 2) { // tr += ' style="background-image: linear-gradient(rgba(250,250,250,0.1),rgba(250,250,250,,0.1))"'; //} else if (item1[levelField] == 3) { // tr += ' style="background-image: linear-gradient(rgba(0,0,0,0.1),rgba(0,0,0,0.1))"'; //} else if (item1[levelField] == 4) { // tr += ' style="background-image: linear-gradient(rgba(0,0,0,0.3),rgba(0,0,0,0.3))"'; //} tr += '>'; ; trs.push(tr + tds.join('') + ''); trs_fixed.push(tr + tds_fixed.join('') + ''); trs_fixed_r.push(tr + tds_fixed_r.join('') + ''); }); if(parentId==0) { that.layBody.scrollTop(0); that.layMain.find('.'+ NONE).remove(); that.layMain.find('tbody').html(trs.join('')); that.layFixLeft.find('tbody').html(trs_fixed.join('')); that.layFixRight.find('tbody').html(trs_fixed_r.join('')); } else { that.layMain.find('tr[data-parent-id="'+ parentId +'"]').remove(); that.layFixLeft.find('tr[data-parent-id="'+ parentId +'"]').remove(); that.layFixRight.find('tr[data-parent-id="'+ parentId +'"]').remove(); that.layMain.find('tr[data-id="'+ parentId +'"]').after(trs.join('')); that.layFixLeft.find('tr[data-id="'+ parentId +'"]').after(trs_fixed.join('')); that.layFixRight.find('tr[data-id="'+ parentId +'"]').after(trs_fixed_r.join('')); } that.renderForm(); that.haveInit ? that.scrollPatch() : setTimeout(function(){ that.scrollPatch(); }, 50); that.haveInit = true; layer.close(that.tipsIndex); that.exportTree(parentId,childIsExport); }; that.key = options.id || options.index; treeGrid.cache[that.key] = treeDataCache; //记录数据 if(data.length === 0){ that.renderForm(); that.layFixed.remove(); that.layMain.find('tbody').html(''); that.layMain.find('.'+ NONE).remove(); return that.layMain.append('
无数据
'); } render(); }; //找到对应的列元素 Class.prototype.getColElem = function(parent, field){ var that = this ,options = that.config; return parent.eq(0).find('.laytable-cell-'+ (options.index + '-' + field) + ':eq(0)'); }; //渲染表单 Class.prototype.renderForm = function(type){ form.render(type, 'LAY-table-'+ this.index); } //请求loading Class.prototype.loading = function(){ var that = this ,options = that.config; if(options.loading && options.url){ return layer.msg('数据请求中', { icon: 16 ,offset: [ that.elem.offset().top + that.elem.height()/2 - 35 - _WIN.scrollTop() + 'px' ,that.elem.offset().left + that.elem.width()/2 - 90 - _WIN.scrollLeft() + 'px' ] ,time: -1 ,anim: -1 ,fixed: false }); } }; //同步选中值状态 Class.prototype.setCheckData = function(index, checked){ var that = this ,options = that.config ,thisData = treeGrid.cache[that.key]; if(!thisData[index.split("_")[0]][index.split("_")[1]]) return; if(thisData[index.split("_")[0]][index.split("_")[1]].constructor === Array) return; else { thisData[index.split("_")[0]][index.split("_")[1]][options.checkName]=checked; } }; //获取cssRule Class.prototype.getCssRule = function(field, callback){ var that = this ,style = that.elem.find('style')[0] ,sheet = style.sheet || style.styleSheet || {} ,rules = sheet.cssRules || sheet.rules; layui.each(rules, function(i, item){ if(item.selectorText === ('.laytable-cell-'+ that.index +'-'+ field)){ return callback(item), true; } }); }; //铺满表格主体高度 Class.prototype.fullSize = function(){ var that = this ,options = that.config ,height = options.height, bodyHeight; if(that.fullHeightGap){ height = _WIN.height() - that.fullHeightGap; if(height < 135) height = 135; that.elem.css('height', height); } //tbody区域高度 bodyHeight = parseFloat(height) - parseFloat(that.layHeader.height()) - 1; if(options.toolbar){ bodyHeight = bodyHeight - that.layTool.outerHeight(); } that.layMain.css('height', bodyHeight); }; //获取滚动条宽度 Class.prototype.getScrollWidth = function(elem){ var width = 0; if(elem){ width = elem.offsetWidth - elem.clientWidth; } else { elem = document.createElement('div'); elem.style.width = '100px'; elem.style.height = '100px'; elem.style.overflowY = 'scroll'; document.body.appendChild(elem); width = elem.offsetWidth - elem.clientWidth; document.body.removeChild(elem); } return width; }; //滚动条补丁 Class.prototype.scrollPatch = function(){ var that = this ,layMainTable = that.layMain.children('table') ,scollWidth = that.layMain.width() - that.layMain.prop('clientWidth') //纵向滚动条宽度 ,scollHeight = that.layMain.height() - that.layMain.prop('clientHeight') //横向滚动条高度 ,getScrollWidth = that.getScrollWidth(that.layMain[0]) //获取主容器滚动条宽度,如果有的话 ,outWidth = layMainTable.outerWidth() - that.layMain.width(); //表格内容器的超出宽度 //如果存在自动列宽,则要保证绝对填充满,并且不能出现横向滚动条 if(that.autoColNums && outWidth < 5 && !that.scrollPatchWStatus){ var th = that.layHeader.eq(0).find('thead th:last-child') ,field = th.data('field'); that.getCssRule(field, function(item){ var width = item.style.width || th.outerWidth(); item.style.width = (parseFloat(width) - getScrollWidth - outWidth) + 'px'; //二次校验,如果仍然出现横向滚动条 if(that.layMain.height() - that.layMain.prop('clientHeight') > 0){ item.style.width = parseFloat(item.style.width) - 1 + 'px'; } that.scrollPatchWStatus = true; }); } if(scollWidth && scollHeight){ if(!that.elem.find('.layui-table-patch')[0]){ var patchElem = $('
'); //补丁元素 patchElem.find('div').css({ width: scollWidth }); that.layHeader.eq(0).find('thead tr').append(patchElem) } } else { that.layHeader.eq(0).find('.layui-table-patch').remove(); } //固定列区域高度 var mainHeight = that.layMain.height() ,fixHeight = mainHeight - scollHeight; that.layFixed.find(ELEM_BODY).css('height', layMainTable.height() > fixHeight ? fixHeight : 'auto'); //表格宽度小于容器宽度时,隐藏固定列 that.layFixRight[outWidth > 0 ? 'removeClass' : 'addClass'](HIDE); //操作栏 that.layFixRight.css('right', scollWidth - 1); }; //事件处理 Class.prototype.events = function(){ var that = this ,options = that.config ,_BODY = $('body') ,dict = {} ,th = that.layHeader.find('th') ,resizing ,ELEM_CELL = '.layui-table-cell' ,filter = options.elem.attr('lay-filter'); //拖拽调整宽度 th.on('mousemove', function(e){ var othis = $(this) ,oLeft = othis.offset().left ,pLeft = e.clientX - oLeft; if(othis.attr('colspan') > 1 || othis.data('unresize') || dict.resizeStart){ return; } dict.allowResize = othis.width() - pLeft <= 10; //是否处于拖拽允许区域 _BODY.css('cursor', (dict.allowResize ? 'col-resize' : '')); }).on('mouseleave', function(){ var othis = $(this); if(dict.resizeStart) return; _BODY.css('cursor', ''); }).on('mousedown', function(e){ var othis = $(this); if(dict.allowResize){ var field = othis.data('field'); e.preventDefault(); dict.resizeStart = true; //开始拖拽 dict.offset = [e.clientX, e.clientY]; //记录初始坐标 that.getCssRule(field, function(item){ var width = item.style.width || othis.outerWidth(); dict.rule = item; dict.ruleWidth = parseFloat(width); dict.minWidth = othis.data('minwidth') || options.cellMinWidth; }); } }); //拖拽中 _DOC.on('mousemove', function(e){ if(dict.resizeStart){ e.preventDefault(); if(dict.rule){ var setWidth = dict.ruleWidth + e.clientX - dict.offset[0]; if(setWidth < dict.minWidth) setWidth = dict.minWidth; dict.rule.style.width = setWidth + 'px'; layer.close(that.tipsIndex); } resizing = 1 } }).on('mouseup', function(e){ if(dict.resizeStart){ dict = {}; _BODY.css('cursor', ''); that.scrollPatch(); } if(resizing === 2){ resizing = null; } }); //复选框选择 that.elem.on('click', 'input[name="layTableCheckbox"]+', function(){ var checkbox = $(this).prev() ,index = checkbox.parents('tr').eq(0).data('index') ,checked = checkbox[0].checked; //单选 that.setCheckData(index, checked); layui.event.call(this, MOD_NAME, 'checkbox('+ filter +')', { checked: checked ,data: treeGrid.cache[that.key] ? (treeGrid.cache[that.key][index.split("_")[0]][index.split("_")[1]] || {}) : {} ,type: 'one' }); }); // that.elem.on('click', 'i.layui-tree-head', function() that.elem.on('click', 'i.treeHead', function () { var othis = $(this), othisHtml = othis.html(); if (othisHtml.indexOf("folder") > 0&& !othis.find(".fa").hasClass('fa-folder-o') ) { var index = othis.parents('tr').eq(0).data('index') , tr = that.layBody.find('tr[data-index="' + index + '"]') , request = options.request , response = options.response , data = treeGrid.cache[that.key][index.split("_")[0]][index.split("_")[1]]; var levelField = options.levelField , paramName = options.paramName , paramValue = options.paramValue , scopeField = options.scopeField; var tempValue = data[paramValue]; $.ajax({ type: options.method || 'get' , url: options.url , data: paramName + "=" + tempValue , dataType: 'json' , success: function (res) { if (res[response.statusName] != response.statusCode) { that.renderForm(); return that.layMain.html('
' + (res[response.msgName] || '返回的数据状态异常') + '
'); } that.renderData(data[paramValue], res); options.time = (new Date().getTime() - that.startTime) + ' ms'; //耗时(接口请求+视图渲染) } , error: function (e, m) { that.layMain.html('
数据接口请求异常
'); that.renderForm(); } }); } }); //单行点击事件 luo that.elem.on('click', 'td', function() { var othis = $(this); var fieldIndex= othis.attr("data-field"); if(!(/^\+?[1-9][0-9]*$/.test(fieldIndex))) //排除checkBox点击事件, 因为td 点击只能如果只要不是checkBox点击,checkBox 点击一定是数字 { var childs = that.layBody.find('input[name="layTableCheckbox"]'); var index = othis.parents('tr').eq(0).data('index'); childs.each(function(i, item) { if(item.id==="tableCheckBox"+index) { item.checked = true; treeGrid.cache[that.key][index.split("_")[0]][index.split("_")[1]][options.checkName]=true; } else { var otherItemId=item.id.substring("tableCheckBox".length,item.id.length); item.checked = false; treeGrid.cache[that.key][otherItemId.split("_")[0]][otherItemId.split("_")[1]][options.checkName]=false; } }); that.renderForm(); layui.event.call(this, MOD_NAME, 'checkbox('+ filter +')', { checked: true ,data: treeGrid.cache[that.key] ? (treeGrid.cache[that.key][index.split("_")[0]][index.split("_")[1]] || {}) : {} ,type: 'one' }); } }); that.elem.on('dblclick', 'td', function() { var othis = $(this), fieldIndex= othis.attr("data-field"); if ((/^\+?[1-9][0-9]*$/.test(fieldIndex))) return; if (othis.closest("tr").find("i.treeHead").find("i.fa").hasClass('fa-folder-o')) return; var index = othis.parents('tr').eq(0).data('index') ,tr = that.layBody.find('tr[data-index="'+ index +'"]') ,request = options.request ,response = options.response ,data = treeGrid.cache[that.key][index.split("_")[0]][index.split("_")[1]]; var levelField=options.levelField ,paramName=options.paramName ,paramValue=options.paramValue ,scopeField=options.scopeField; var tempValue=data[paramValue]; $.ajax({ type: options.method || 'get' ,url: options.url ,data: paramName+"="+tempValue ,dataType: 'json' ,success: function(res){ if(res[response.statusName] != response.statusCode){ that.renderForm(); return that.layMain.html('
'+ (res[response.msgName] || '返回的数据状态异常') +'
'); } that.renderData(data[paramValue],res); options.time = (new Date().getTime() - that.startTime) + ' ms'; //耗时(接口请求+视图渲染) } ,error: function(e, m){ that.layMain.html('
数据接口请求异常
'); that.renderForm(); } }); }); //行事件 that.layBody.on('mouseenter', 'tr', function(){ var othis = $(this) ,index = othis.index(); that.layBody.find('tr:eq('+ index +')').addClass(ELEM_HOVER) }).on('mouseleave', 'tr', function(){ var othis = $(this) ,index = othis.index(); that.layBody.find('tr:eq('+ index +')').removeClass(ELEM_HOVER) }); //单元格编辑 that.layBody.on('change', '.'+ELEM_EDIT, function(){ var othis = $(this) ,value = this.value ,field = othis.parent().data('field') ,index = othis.parents('tr').eq(0).data('index') ,data = treeGrid.cache[that.key][index.split("_")[0]][index.split("_")[1]]; data[field] = value; //更新缓存中的值 layui.event.call(this, MOD_NAME, 'edit('+ filter +')', { value: value ,data: data ,field: field }); }).on('blur', '.'+ELEM_EDIT, function(){ var templet ,othis = $(this) ,field = othis.parent().data('field') ,index = othis.parents('tr').eq(0).data('index') ,data = treeGrid.cache[that.key][index]; that.eachCols(function(i, item){ if(item.field == field && item.templet){ templet = item.templet; } }); othis.siblings(ELEM_CELL).html( templet ? laytpl($(templet).html() || this.value).render(data) : this.value ); othis.parent().data('content', this.value); othis.remove(); }); //单元格事件 that.layBody.on('click', 'td', function(){ var othis = $(this) ,field = othis.data('field') ,editType = othis.data('edit') ,elemCell = othis.children(ELEM_CELL); layer.close(that.tipsIndex); if(othis.data('off')) return; //显示编辑表单 if(editType){ if(editType === 'select') { //选择框 //var select = $(''); //othis.find('.'+ELEM_EDIT)[0] || othis.append(select); } else { //输入框 var input = $(''); input[0].value = othis.data('content') || elemCell.text(); othis.find('.'+ELEM_EDIT)[0] || othis.append(input); input.focus(); } return; } //如果出现省略,则可查看更多 if(elemCell.find('.layui-form-switch,.layui-form-checkbox')[0]) return; //限制不出现更多(暂时) if(Math.round(elemCell.prop('scrollWidth')) > Math.round(elemCell.outerWidth())){ that.tipsIndex = layer.tips([ '
' ,elemCell.html() ,'
' ,'' ].join(''), elemCell[0], { tips: [3, ''] ,time: -1 ,anim: -1 ,maxWidth: (device.ios || device.android) ? 300 : 600 ,isOutAnim: false ,skin: 'layui-table-tips' ,success: function(layero, index){ layero.find('.layui-table-tips-c').on('click', function(){ layer.close(index); }); } }); } }); //工具条操作事件 that.layBody.on('click', '*[lay-event]', function(){ var othis = $(this) ,index = othis.parents('tr').eq(0).data('index') ,tr = that.layBody.find('tr[data-index="'+ index +'"]') ,ELEM_CLICK = 'layui-table-click' ,data = treeGrid.cache[that.key][index.split("_")[0]][index.split("_")[1]]; layui.event.call(this, MOD_NAME, 'tool('+ filter +')', { data: treeGrid.clearCacheKey(data) ,event: othis.attr('lay-event') ,tr: tr ,del: function(){ treeGrid.cache[that.key][index] = []; tr.remove(); that.scrollPatch(); } ,update: function(fields){ fields = fields || {}; layui.each(fields, function(key, value){ if(key in data){ var templet, td = tr.children('td[data-field="'+ key +'"]'); data[key] = value; that.eachCols(function(i, item2){ if(item2.field == key && item2.templet){ templet = item2.templet; } }); td.children(ELEM_CELL).html( templet ? laytpl($(templet).html() || value).render(data) : value ); td.data('content', value); } }); } }); tr.addClass(ELEM_CLICK).siblings('tr').removeClass(ELEM_CLICK); }); //同步滚动条 that.layMain.on('scroll', function(){ var othis = $(this) ,scrollLeft = othis.scrollLeft() ,scrollTop = othis.scrollTop(); that.layHeader.scrollLeft(scrollLeft); that.layFixed.find(ELEM_BODY).scrollTop(scrollTop); layer.close(that.tipsIndex); }); _WIN.on('resize', function(){ //自适应 that.fullSize(); that.scrollPatch(); }); }; //初始化 treeGrid.init = function(filter, settings){ settings = settings || {}; var that = this ,elemTable = filter ? $('table[lay-filter="'+ filter +'"]') : $(ELEM + '[lay-data]') ,errorTips = 'Table element property lay-data configuration item has a syntax error: '; //遍历数据表格 elemTable.each(function(){ var othis = $(this), treeGridData = othis.attr('lay-data'); try{ treeGridData = new Function('return '+ treeGridData)(); } catch(e){ hint.error(errorTips + treeGridData) } var cols = [], options = $.extend({ elem: this ,cols: [] ,data: [] ,skin: othis.attr('lay-skin') //风格 ,size: othis.attr('lay-size') //尺寸 ,even: typeof othis.attr('lay-even') === 'string' //偶数行背景 }, treeGrid.config, settings, treeGridData); filter && othis.hide(); //获取表头数据 othis.find('thead>tr').each(function(i){ options.cols[i] = []; $(this).children().each(function(ii){ var th = $(this), itemData = th.attr('lay-data'); try{ itemData = new Function('return '+ itemData)(); } catch(e){ return hint.error(errorTips + itemData) } var row = $.extend({ title: th.text() ,colspan: th.attr('colspan') || 0 //列单元格 ,rowspan: th.attr('rowspan') || 0 //行单元格 }, itemData); if(row.colspan < 2) cols.push(row); options.cols[i].push(row); }); }); //获取表体数据 othis.find('tbody>tr').each(function(i1){ var tr = $(this), row = {}; //如果定义了字段名 tr.children('td').each(function(i2, item2){ var td = $(this) ,field = td.data('field'); if(field){ return row[field] = td.html(); } }); //如果未定义字段名 layui.each(cols, function(i3, item3){ var td = tr.children('td').eq(i3); row[item3.field] = td.html(); }); options.data[i1] = row; }); treeGrid.render(options); }); return that; }; treeGrid.refreshNode = function (id, parentId, options) { $("#treeHead"+parentId).attr("isexport","false"); $("#treeHead"+parentId).click(); }; //表格选中状态 treeGrid.checkStatus = function(id){ var nums = 0 ,invalidNum = 0 ,arr = [] ,data = treeGrid.cache[id] || []; //计算全选个数 layui.each(data, function(i, item) { layui.each(item, function(j, itemTemp) { if (itemTemp.constructor === Array) { invalidNum++; //无效数据,或已删除的 return; } if (itemTemp[treeGrid.config.checkName]) { nums++; arr.push(treeGrid.clearCacheKey(itemTemp)); } }); }); return { data: arr //选中的数据 ,isAll: data.length ? (nums === (data.length - invalidNum)) : false //是否全选 }; }; //表格重载 thisTable.config = {}; treeGrid.reload = function(id, options){ var config = thisTable.config[id]; options = options || {}; if(!config) return hint.error('The ID option was not found in the treeGrid instance'); if(options.data && options.data.constructor === Array) delete config.data; return treeGrid.render($.extend(true, {}, config, options)); }; //核心入口 treeGrid.render = function(options){ var inst = new Class(options); return thisTable.call(inst); }; //清除临时Key treeGrid.clearCacheKey = function(data){ data = $.extend({}, data); delete data[treeGrid.config.checkName]; delete data[treeGrid.config.indexName]; return data; }; //自动完成渲染 treeGrid.init(); exports(MOD_NAME, treeGrid); });