/** @Name:layui.table 表格操作 @Author:贤心 @License:MIT */ layui.define(['laytpl', 'laypage', 'layer', 'form', 'util'], function (exports) { "use strict"; var $ = layui.$ , laytpl = layui.laytpl , laypage = layui.laypage , layer = layui.layer , form = layui.form , util = layui.util , hint = layui.hint() , device = layui.device() //外部接口 , table = { config: { checkName: 'LAY_CHECKED' //是否选中状态的字段名 , indexName: 'LAY_TABLE_INDEX' //下标索引名 } //全局配置项 , cache: {} //数据缓存 , index: layui.table ? (layui.table.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 || options.index; if (id) { thisTable.that[id] = that; //记录当前实例对象 thisTable.config[id] = options; //记录当前实例配置项 } return { config: options , reload: function (options) { that.reload.call(that, options); } , setColsWidth: function () { that.setColsWidth.call(that); } , resize: function () { //重置表格尺寸/结构 that.resize.call(that); } } } //获取当前实例配置项 , getThisTableConfig = function (id) { var config = thisTable.config[id]; if (!config) hint.error('The ID option was not found in the table instance'); return config || null; } //解析自定义模板数据 , parseTempData = function (item3, content, tplData, text) { //表头数据、原始内容、表体数据、是否只返回文本 var str = item3.templet ? function () { return typeof item3.templet === 'function' ? item3.templet(tplData) : laytpl($(item3.templet).html() || String(content)).render(tplData) }() : content; return text ? $('<div>' + str + '</div>').text() : str; } //字符常量 , MOD_NAME = 'table', ELEM = '.layui-table', THIS = 'layui-this', SHOW = 'layui-show', HIDE = 'layui-hide', DISABLED = 'layui-disabled', NONE = 'layui-none' , ELEM_VIEW = 'layui-table-view', ELEM_TOOL = '.layui-table-tool', ELEM_BOX = '.layui-table-box', ELEM_INIT = '.layui-table-init', 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_TOTAL = '.layui-table-total', 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 ['<table cellspacing="0" cellpadding="0" border="0" class="layui-table" ' , '{{# if(d.data.skin){ }}lay-skin="{{d.data.skin}}"{{# } }} {{# if(d.data.size){ }}lay-size="{{d.data.size}}"{{# } }} {{# if(d.data.even){ }}lay-even{{# } }}>' , '<thead>' , '{{# layui.each(d.data.cols, function(i1, item1){ }}' , '<tr>' , '{{# 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 ''; }() , '{{# var isSort = !(item2.colGroup) && item2.sort; }}' , '<th data-field="{{ item2.field||i2 }}" data-key="{{d.index}}-{{i1}}-{{i2}}" {{# if( item2.parentKey){ }}data-parentkey="{{ item2.parentKey }}"{{# } }} {{# if(item2.minWidth){ }}data-minwidth="{{item2.minWidth}}"{{# } }} ' + rowCols + ' {{# if(item2.unresize || item2.colGroup){ }}data-unresize="true"{{# } }} class="{{# if(item2.hide){ }}layui-hide{{# } }}{{# if(isSort){ }} layui-unselect{{# } }}{{# if(!item2.field){ }} layui-table-col-special{{# } }}">' , '<div class="layui-table-cell laytable-cell-' , '{{# if(item2.colGroup){ }}' , 'group' , '{{# } else { }}' , '{{d.index}}-{{i1}}-{{i2}}' , '{{# if(item2.type !== "normal"){ }}' , ' laytable-cell-{{ item2.type }}' , '{{# } }}' , '{{# } }}' , '" {{#if(item2.align){}}align="{{item2.align}}"{{#}}}>' , '{{# if(item2.type === "checkbox"){ }}' //复选框 , '<input type="checkbox" name="layTableCheckbox" lay-skin="primary" lay-filter="layTableAllChoose" {{# if(item2[d.data.checkName]){ }}checked{{# }; }}>' , '{{# } else { }}' , '<span>{{item2.title||""}}</span>' , '{{# if(isSort){ }}' , '<span class="layui-table-sort layui-inline"><i class="layui-edge layui-table-sort-asc" title="升序"></i><i class="layui-edge layui-table-sort-desc" title="降序"></i></span>' , '{{# } }}' , '{{# } }}' , '</div>' , '</th>' , (options.fixed ? '{{# }; }}' : '') , '{{# }); }}' , '</tr>' , '{{# }); }}' , '</thead>' , '</table>'].join(''); } //tbody区域模板 , TPL_BODY = ['<table cellspacing="0" cellpadding="0" border="0" class="layui-table" ' , '{{# if(d.data.skin){ }}lay-skin="{{d.data.skin}}"{{# } }} {{# if(d.data.size){ }}lay-size="{{d.data.size}}"{{# } }} {{# if(d.data.even){ }}lay-even{{# } }}>' , '<tbody></tbody>' , '</table>'].join('') //主模板 , TPL_MAIN = ['<div class="layui-form layui-border-box {{d.VIEW_CLASS}}" lay-filter="LAY-table-{{d.index}}" lay-id="{{ d.data.id }}" style="{{# if(d.data.width){ }}width:{{d.data.width}}px;{{# } }} {{# if(d.data.height){ }}height:{{d.data.height}}px;{{# } }}">' , '{{# if(d.data.toolbar){ }}' , '<div class="layui-table-tool">' , '<div class="layui-table-tool-temp"></div>' , '<div class="layui-table-tool-self"></div>' , '</div>' , '{{# } }}' , '<div class="layui-table-box">' , '{{# if(d.data.loading){ }}' , '<div class="layui-table-init" style="background-color: #fff;">' , '<i class="layui-icon layui-icon-loading layui-anim layui-anim-rotate layui-anim-loop"></i>' , '</div>' , '{{# } }}' , '{{# var left, right; }}' , '<div class="layui-table-header">' , TPL_HEADER() , '</div>' , '<div class="layui-table-body layui-table-main">' , TPL_BODY , '</div>' , '{{# if(left){ }}' , '<div class="layui-table-fixed layui-table-fixed-l">' , '<div class="layui-table-header">' , TPL_HEADER({ fixed: true }) , '</div>' , '<div class="layui-table-body">' , TPL_BODY , '</div>' , '</div>' , '{{# }; }}' , '{{# if(right){ }}' , '<div class="layui-table-fixed layui-table-fixed-r">' , '<div class="layui-table-header">' , TPL_HEADER({ fixed: 'right' }) , '<div class="layui-table-mend"></div>' , '</div>' , '<div class="layui-table-body">' , TPL_BODY , '</div>' , '</div>' , '{{# }; }}' , '</div>' , '{{# if(d.data.totalRow){ }}' , '<div class="layui-table-total">' , '<table cellspacing="0" cellpadding="0" border="0" class="layui-table" ' , '{{# if(d.data.skin){ }}lay-skin="{{d.data.skin}}"{{# } }} {{# if(d.data.size){ }}lay-size="{{d.data.size}}"{{# } }} {{# if(d.data.even){ }}lay-even{{# } }}>' , '<tbody><tr><td><div class="layui-table-cell" style="visibility: hidden;">Total</div></td></tr></tbody>' , '</table>' , '</div>' , '{{# } }}' , '{{# if(d.data.page){ }}' , '<div class="layui-table-page">' , '<div id="layui-table-page{{d.index}}"></div>' , '</div>' , '{{# } }}' , '<style>' , '{{# layui.each(d.data.cols, function(i1, item1){' , 'layui.each(item1, function(i2, item2){ }}' , '.laytable-cell-{{d.index}}-{{i1}}-{{i2}}{ ' , '{{# if(item2.width){ }}' , 'width: {{item2.width}}px;' , '{{# } }}' , ' }' , '{{# });' , '}); }}' , '</style>' , '</div>'].join('') , _WIN = $(window) , _DOC = $(document) //构造器 , Class = function (options) { var that = this; that.index = ++table.index; that.config = $.extend({}, that.config, table.config, options); that.render(); }; //默认配置 Class.prototype.config = { limit: 10 //每页显示的数量 , loading: true //请求数据时,是否显示loading , cellMinWidth: 60 //所有单元格默认最小宽度 , defaultToolbar: ['filter', 'exports', 'print'] //工具栏右侧图标 , autoSort: true //是否前端自动排序。如果否,则需自主排序(通常为服务端处理好排序) , text: { none: '无数据' } }; //表格渲染 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') || that.index; //请求参数的自定义格式 options.request = $.extend({ pageName: 'page' , limitName: 'limit' }, options.request) //响应数据的自定义格式 options.response = $.extend({ statusName: 'Code' //规定数据状态的字段名称 , statusCode: 200 //规定成功的状态码 , msgName: 'Message' //规定状态信息的字段名称 , dataName: 'Result' //规定数据总数的字段名称 , totalRowName: 'totalRow' //规定数据统计的字段名称 , countName: 'Count' }, options.response); //如果 page 传入 laypage 对象 if (typeof options.page === 'object') { options.limit = options.page.limit || options.limit; options.limits = options.page.limits || options.limits; that.page = options.page.curr = options.page.curr || 1; delete options.page.elem; delete options.page.jump; } if (!options.elem[0]) return that; //高度铺满:full-差距值 if (options.height && /^full-\d+$/.test(options.height)) { that.fullHeightGap = options.height.split('-')[1]; options.height = _WIN.height() - that.fullHeightGap; } //初始化一些参数 that.setInit(); //开始插入替代元素 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; that.key = options.id || options.index; //生成替代元素 hasRender[0] && hasRender.remove(); //如果已经渲染,则Rerender othis.after(reElem); //各级容器 that.layTool = reElem.find(ELEM_TOOL); that.layBox = reElem.find(ELEM_BOX); 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.layTotal = reElem.find(ELEM_TOTAL); that.layPage = reElem.find(ELEM_PAGE); //初始化工具栏 that.renderToolbar(); //让表格平铺 that.fullSize(); //如果多级表头,则填补表头高度 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.page); //请求数据 that.events(); //事件 }; //根据列类型,定制化参数 Class.prototype.initOpts = function (item) { var that = this , options = that.config , initWidth = { checkbox: 48 , radio: 48 , space: 15 , numbers: 40 }; //让 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.setInit = function (type) { var that = this , options = that.config; options.clientWidth = 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(); }(); if (type === 'width') return options.clientWidth; //初始化列参数 layui.each(options.cols, function (i1, item1) { layui.each(item1, function (i2, item2) { //如果列参数为空,则移除 if (!item2) { item1.splice(i2, 1); return; } item2.key = i1 + '-' + i2; item2.hide = item2.hide || false; //设置列的父列索引 //如果是组合列,则捕获对应的子列 if (item2.colGroup || item2.colspan > 1) { var childIndex = 0; layui.each(options.cols[i1 + 1], function (i22, item22) { //如果子列已经被标注为{HAS_PARENT},或者子列累计 colspan 数等于父列定义的 colspan,则跳出当前子列循环 if (item22.HAS_PARENT || (childIndex > 1 && childIndex == item2.colspan)) return; item22.HAS_PARENT = true; item22.parentKey = i1 + '-' + i2; childIndex = childIndex + parseInt(item22.colspan > 1 ? item22.colspan : 1); }); item2.colGroup = true; //标注是组合列 } //根据列类型,定制化参数 that.initOpts(item2); }); }); }; //初始工具栏 Class.prototype.renderToolbar = function () { var that = this , options = that.config //添加工具栏左侧模板 var leftDefaultTemp = [ '<div class="layui-inline" lay-event="add"><i class="layui-icon layui-icon-add-1"></i></div>' , '<div class="layui-inline" lay-event="update"><i class="layui-icon layui-icon-edit"></i></div>' , '<div class="layui-inline" lay-event="delete"><i class="layui-icon layui-icon-delete"></i></div>' ].join('') , elemToolTemp = that.layTool.find('.layui-table-tool-temp'); if (options.toolbar === 'default') { elemToolTemp.html(leftDefaultTemp); } else if (typeof options.toolbar === 'string') { var toolbarHtml = $(options.toolbar).html() || ''; toolbarHtml && elemToolTemp.html( laytpl(toolbarHtml).render(options) ); } //添加工具栏右侧面板 var layout = { filter: { title: '筛选列' , layEvent: 'LAYTABLE_COLS' , icon: 'layui-icon-cols' } , exports: { title: '导出' , layEvent: 'LAYTABLE_EXPORT' , icon: 'layui-icon-export' } , print: { title: '打印' , layEvent: 'LAYTABLE_PRINT' , icon: 'layui-icon-print' } }, iconElem = []; if (typeof options.defaultToolbar === 'object') { layui.each(options.defaultToolbar, function (i, item) { var thisItem = typeof item === 'string' ? layout[item] : item; if (thisItem) { iconElem.push('<div class="layui-inline" title="' + thisItem.title + '" lay-event="' + thisItem.layEvent + '">' + '<i class="layui-icon ' + thisItem.icon + '"></i>' + '</div>'); } }); } that.layTool.find('.layui-table-tool-self').html(iconElem.join('')); } //同步表头父列的相关值 Class.prototype.setParentCol = function (hide, parentKey) { var that = this , options = that.config , parentTh = that.layHeader.find('th[data-key="' + options.index + '-' + parentKey + '"]') //获取父列元素 , parentColspan = parseInt(parentTh.attr('colspan')) || 0; if (parentTh[0]) { var arrParentKey = parentKey.split('-') , getThisCol = options.cols[arrParentKey[0]][arrParentKey[1]]; hide ? parentColspan-- : parentColspan++; parentTh.attr('colspan', parentColspan); parentTh[parentColspan < 1 ? 'addClass' : 'removeClass'](HIDE); getThisCol.colspan = parentColspan; //同步 colspan 参数 getThisCol.hide = parentColspan < 1; //同步 hide 参数 //递归,继续往上查询是否有父列 var nextParentKey = parentTh.data('parentkey'); nextParentKey && that.setParentCol(hide, nextParentKey); } }; //多级表头补丁 Class.prototype.setColsPatch = function () { var that = this , options = that.config //同步表头父列的相关值 layui.each(options.cols, function (i1, item1) { layui.each(item1, function (i2, item2) { if (item2.hide) { that.setParentCol(item2.hide, item2.parentKey); } }); }); }; //动态分配列宽 Class.prototype.setColsWidth = function () { var that = this , options = that.config , colNums = 0 //列个数 , autoColNums = 0 //自动列宽的列个数 , autoWidth = 0 //自动列分配的宽度 , countWidth = 0 //所有列总宽度和 , cntrWidth = that.setInit('width'); //统计列个数 that.eachCols(function (i, item) { item.hide || colNums++; }); //减去边框差和滚动条宽 cntrWidth = cntrWidth - function () { return (options.skin === 'line' || options.skin === 'nob') ? 2 : colNums + 1; }() - that.getScrollWidth(that.layMain[0]) - 1; //计算自动分配的宽度 var getAutoWidth = function (back) { //遍历所有列 layui.each(options.cols, function (i1, item1) { layui.each(item1, function (i2, item2) { var width = 0 , minWidth = item2.minWidth || options.cellMinWidth; //最小宽度 if (!item2) { item1.splice(i2, 1); return; } if (item2.colGroup || item2.hide) return; if (!back) { width = item2.width || 0; if (/\d+%$/.test(width)) { //列宽为百分比 width = Math.floor((parseFloat(width) / 100) * cntrWidth); width < minWidth && (width = minWidth); } else if (!width) { //列宽未填写 item2.width = width = 0; autoColNums++; } } else if (autoWidth && autoWidth < minWidth) { autoColNums--; width = minWidth; } if (item2.hide) width = 0; countWidth = countWidth + width; }); }); //如果未填充满,则将剩余宽度平分 (cntrWidth > countWidth && autoColNums) && ( autoWidth = (cntrWidth - countWidth) / autoColNums ); } getAutoWidth(); getAutoWidth(true); //重新检测分配的宽度是否低于最小列宽 //记录自动列数 that.autoColNums = autoColNums; //设置列宽 that.eachCols(function (i3, item3) { var minWidth = item3.minWidth || options.cellMinWidth; if (item3.colGroup || item3.hide) return; //给位分配宽的列平均分配宽 if (item3.width === 0) { that.getCssRule(options.index + '-' + item3.key, function (item) { item.style.width = Math.floor(autoWidth >= minWidth ? autoWidth : minWidth) + 'px'; }); } //给设定百分比的列分配列宽 else if (/\d+%$/.test(item3.width)) { that.getCssRule(options.index + '-' + item3.key, function (item) { item.style.width = Math.floor((parseFloat(item3.width) / 100) * cntrWidth) + 'px'; }); } }); //填补 Math.floor 造成的数差 var patchNums = that.layMain.width() - that.getScrollWidth(that.layMain[0]) - that.layMain.children('table').outerWidth(); if (that.autoColNums && patchNums >= -colNums && patchNums <= colNums) { var getEndTh = function (th) { var field; th = th || that.layHeader.eq(0).find('thead th:last-child') field = th.data('field'); if (!field && th.prev()[0]) { return getEndTh(th.prev()) } return th } , th = getEndTh() , key = th.data('key'); that.getCssRule(key, function (item) { var width = item.style.width || th.outerWidth(); item.style.width = (parseFloat(width) + patchNums) + 'px'; //二次校验,如果仍然出现横向滚动条(通常是 1px 的误差导致) if (that.layMain.height() - that.layMain.prop('clientHeight') > 0) { item.style.width = (parseFloat(item.style.width) - 1) + 'px'; } }); } that.loading(!0); }; //重置表格尺寸/结构 Class.prototype.resize = function () { var that = this; that.fullSize(); //让表格铺满 that.setColsWidth(); //自适应列宽 that.scrollPatch(); //滚动条补丁 if (layui.tableFilter) { layui.tableFilter.resize(that.config) } // 这是要添加的那一行 }; //表格重载 Class.prototype.reload = function (options) { var that = this; options = options || {}; delete that.haveInit; if (options.data && options.data.constructor === Array) delete that.config.data; that.config = $.extend(true, {}, that.config, options); that.render(); }; //异常提示 Class.prototype.errorView = function (html) { var that = this , elemNone = that.layMain.find('.' + NONE) , layNone = $('<div class="' + NONE + '">' + (html || 'Error') + '</div>'); if (elemNone[0]) { that.layNone.remove(); elemNone.remove(); } that.layFixed.addClass(HIDE); that.layMain.find('tbody').html(''); that.layMain.append(that.layNone = layNone); table.cache[that.key] = []; //格式化缓存数据 }; //页码 Class.prototype.page = 1; //获得数据 Class.prototype.pullData = function (curr) { var that = this , options = that.config , request = options.request , response = options.response , sort = function () { if (typeof options.initSort === 'object') { that.sort(options.initSort.field, options.initSort.type); } }; that.startTime = new Date().getTime(); //渲染开始时间 if (options.url) { //Ajax请求 var params = {}; params[request.pageName] = curr; params[request.limitName] = options.limit; //参数 var data = $.extend(params, options.where); if (options.contentType && options.contentType.indexOf("application/json") == 0) { //提交 json 格式 data = JSON.stringify(data); } that.loading(); $.ajax({ type: options.method || 'get' , url: options.url , contentType: options.contentType , data: data , dataType: 'json' , headers: options.headers || {} , success: function (res) { //如果有数据解析的回调,则获得其返回的数据 if (typeof options.parseData === 'function') { res = options.parseData(res) || res; } //检查数据格式是否符合规范 if (res[response.statusName] != response.statusCode) { that.renderForm(); that.errorView( res[response.msgName] || ('返回的数据不符合规范,正确的成功状态码应为:"' + response.statusName + '": ' + response.statusCode) ); } else { that.renderData(res, curr, res[response.countName]), sort(); options.time = (new Date().getTime() - that.startTime) + ' ms'; //耗时(接口请求+视图渲染) } that.setColsWidth(); typeof options.done === 'function' && options.done(res, curr, res[response.countName]); } , error: function (e, m) { that.errorView('数据接口请求异常:' + m); that.renderForm(); that.setColsWidth(); } }); } 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; //记录合计行数据 if (typeof options.totalRow === 'object') { res[response.totalRowName] = $.extend({}, options.totalRow); } that.renderData(res, curr, res[response.countName]), sort(); that.setColsWidth(); typeof options.done === 'function' && options.done(res, curr, res[response.countName]); } }; //遍历表头 Class.prototype.eachCols = function (callback) { var that = this; table.eachCols(null, callback, that.config.cols); return that; }; //数据渲染 Class.prototype.renderData = function (res, curr, count, sort) { var that = this , options = that.config , data = res[options.response.dataName] || [] //列表数据 , totalRowData = res[options.response.totalRowName] //合计行数据 , trs = [] , trs_fixed = [] , trs_fixed_r = [] //渲染视图 , render = function () { //后续性能提升的重点 var thisCheckedRowIndex; if (!sort && that.sortKey) { return that.sort(that.sortKey.field, that.sortKey.sort, true); } layui.each(data, function (i1, item1) { var tds = [], tds_fixed = [], tds_fixed_r = [] , numbers = i1 + options.limit * (curr - 1) + 1; //序号 if (item1.length === 0) return; if (!sort) { item1[table.config.indexName] = i1; } that.eachCols(function (i3, item3) { var field = item3.field || i3 , key = options.index + '-' + item3.key , content = item1[field]; if (content === undefined || content === null) content = ''; if (item3.colGroup) return; //td内容 var td = ['<td data-field="' + field + '" data-key="' + key + '" ' + function () { //追加各种属性 var attr = []; if (item3.edit) attr.push('data-edit="' + item3.edit + '"'); //是否允许单元格编辑 if (item3.align) attr.push('align="' + item3.align + '"'); //对齐方式 if (item3.templet) attr.push('data-content="' + content + '"'); //自定义模板 if (item3.toolbar) attr.push('data-off="true"'); //行工具列关闭单元格事件 if (item3.event) attr.push('lay-event="' + item3.event + '"'); //自定义事件 if (item3.style) attr.push('style="' + item3.style + '"'); //自定义样式 if (item3.minWidth) attr.push('data-minwidth="' + item3.minWidth + '"'); //单元格最小宽度 return attr.join(' '); }() + ' class="' + function () { //追加样式 var classNames = []; if (item3.hide) classNames.push(HIDE); //插入隐藏列样式 if (!item3.field) classNames.push('layui-table-col-special'); //插入特殊列样式 return classNames.join(' '); }() + '">' , '<div class="layui-table-cell laytable-cell-' + function () { //返回对应的CSS类标识 return item3.type === 'normal' ? key : (key + ' laytable-cell-' + item3.type); }() + '">' + function () { var tplData = $.extend(true, { LAY_INDEX: numbers }, item1) , checkName = table.config.checkName; //渲染不同风格的列 switch (item3.type) { case 'checkbox': return '<input type="checkbox" name="layTableCheckbox" lay-skin="primary" ' + function () { //如果是全选 if (item3[checkName]) { item1[checkName] = item3[checkName]; return item3[checkName] ? 'checked' : ''; } return tplData[checkName] ? 'checked' : ''; }() + '>'; break; case 'radio': if (tplData[checkName]) { thisCheckedRowIndex = i1; } return '<input type="radio" name="layTableRadio_' + options.index + '" ' + (tplData[checkName] ? 'checked' : '') + ' lay-type="layTableRadio">'; break; case 'numbers': return numbers; break; }; //解析工具列模板 if (item3.toolbar) { return laytpl($(item3.toolbar).html() || '').render(tplData); } return parseTempData(item3, content, tplData); }() , '</div></td>'].join(''); tds.push(td); if (item3.fixed && item3.fixed !== 'right') tds_fixed.push(td); if (item3.fixed === 'right') tds_fixed_r.push(td); }); trs.push('<tr data-index="' + i1 + '">' + tds.join('') + '</tr>'); trs_fixed.push('<tr data-index="' + i1 + '">' + tds_fixed.join('') + '</tr>'); trs_fixed_r.push('<tr data-index="' + i1 + '">' + tds_fixed_r.join('') + '</tr>'); }); 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('')); that.renderForm(); typeof thisCheckedRowIndex === 'number' && that.setThisRowChecked(thisCheckedRowIndex); that.syncCheckAll(); //滚动条补丁 that.haveInit ? that.scrollPatch() : setTimeout(function () { that.scrollPatch(); }, 50); that.haveInit = true; layer.close(that.tipsIndex); //同步表头父列的相关值 options.HAS_SET_COLS_PATCH || that.setColsPatch(); options.HAS_SET_COLS_PATCH = true; }; table.cache[that.key] = data; //记录数据 //显示隐藏分页栏 that.layPage[(count == 0 || (data.length === 0 && curr == 1)) ? 'addClass' : 'removeClass'](HIDE); //如果无数据 if (data.length === 0) { that.renderForm(); return that.errorView(options.text.none); } else { that.layFixed.removeClass(HIDE); } //如果执行初始排序 if (sort) { return render(); } //正常初始化数据渲染 render(); //渲染数据 that.renderTotal(data, totalRowData); //数据合计 //同步分页状态 if (options.page) { options.page = $.extend({ elem: 'layui-table-page' + options.index , count: count , limit: options.limit , limits: options.limits || [10, 20, 30, 40, 50, 60, 70, 80, 90] , groups: 3 , layout: ['prev', 'page', 'next', 'skip', 'count', 'limit'] , prev: '<i class="layui-icon"></i>' , next: '<i class="layui-icon"></i>' , jump: function (obj, first) { if (!first) { //分页本身并非需要做以下更新,下面参数的同步,主要是因为其它处理统一用到了它们 //而并非用的是 options.page 中的参数(以确保分页未开启的情况仍能正常使用) that.page = obj.curr; //更新页码 options.limit = obj.limit; //更新每页条数 that.pullData(obj.curr); } } }, options.page); options.page.count = count; //更新总条数 laypage.render(options.page); } }; //数据合计行 Class.prototype.renderTotal = function (data, totalRowData) { var that = this , options = that.config , totalNums = {}; if (!options.totalRow) return; layui.each(data, function (i1, item1) { if (item1.length === 0) return; that.eachCols(function (i3, item3) { var field = item3.field || i3 , content = item1[field]; if (item3.totalRow) { totalNums[field] = (totalNums[field] || 0) + (parseFloat(content) || 0); } }); }); that.dataTotal = {}; var tds = []; that.eachCols(function (i3, item3) { var field = item3.field || i3; //td内容 var content = function () { var text = item3.totalRowText || '' , thisTotalNum = parseFloat(totalNums[field]).toFixed(2) , tplData = {}; if (parseInt(totalNums[field]) == totalNums[field]) //是否是整数 thisTotalNum = parseFloat(totalNums[field]).toFixed(0); else thisTotalNum = parseFloat(totalNums[field]).toFixed(2); tplData[field] = thisTotalNum; thisTotalNum = parseTempData(item3, thisTotalNum, tplData); //如果直接传入了合计行数据,则不输出自动计算的结果 if (totalRowData) { return totalRowData[item3.field] || text; } else { return item3.totalRow ? (thisTotalNum || text) : text; } }() , td = ['<td data-field="' + field + '" data-key="' + options.index + '-' + item3.key + '" ' + function () { var attr = []; if (item3.align) attr.push('align="' + item3.align + '"'); //对齐方式 if (item3.style) attr.push('style="' + item3.style + '"'); //自定义样式 if (item3.minWidth) attr.push('data-minwidth="' + item3.minWidth + '"'); //单元格最小宽度 return attr.join(' '); }() + ' class="' + function () { //追加样式 var classNames = []; if (item3.hide) classNames.push(HIDE); //插入隐藏列样式 if (!item3.field) classNames.push('layui-table-col-special'); //插入特殊列样式 return classNames.join(' '); }() + '">' , '<div class="layui-table-cell laytable-cell-' + function () { //返回对应的CSS类标识 var str = (options.index + '-' + item3.key); return item3.type === 'normal' ? str : (str + ' laytable-cell-' + item3.type); }() + '">' + content , '</div></td>'].join(''); item3.field && (that.dataTotal[field] = content); tds.push(td); }); that.layTotal.find('tbody').html('<tr>' + tds.join('') + '</tr>'); }; //找到对应的列元素 Class.prototype.getColElem = function (parent, key) { var that = this , options = that.config; return parent.eq(0).find('.laytable-cell-' + (options.index + '-' + key) + ':eq(0)'); }; //渲染表单 Class.prototype.renderForm = function (type) { form.render(type, 'LAY-table-' + this.index); }; //标记当前行选中状态 Class.prototype.setThisRowChecked = function (index) { var that = this , options = that.config , ELEM_CLICK = 'layui-table-click' , tr = that.layBody.find('tr[data-index="' + index + '"]'); tr.addClass(ELEM_CLICK).siblings('tr').removeClass(ELEM_CLICK); }; //数据排序 Class.prototype.sort = function (th, type, pull, formEvent) { var that = this , field , res = {} , options = that.config , filter = options.elem.attr('lay-filter') , data = table.cache[that.key], thisData; //字段匹配 if (typeof th === 'string') { that.layHeader.find('th').each(function (i, item) { var othis = $(this) , _field = othis.data('field'); if (_field === th) { th = othis; field = _field; return false; } }); } try { var field = field || th.data('field') , key = th.data('key'); //如果欲执行的排序已在状态中,则不执行渲染 if (that.sortKey && !pull) { if (field === that.sortKey.field && type === that.sortKey.sort) { return; } } var elemSort = that.layHeader.find('th .laytable-cell-' + key).find(ELEM_SORT); that.layHeader.find('th').find(ELEM_SORT).removeAttr('lay-sort'); //清除其它标题排序状态 elemSort.attr('lay-sort', type || null); that.layFixed.find('th') } catch (e) { return hint.error('Table modules: Did not match to field'); } //记录排序索引和类型 that.sortKey = { field: field , sort: type }; //默认为前端自动排序。如果否,则需自主排序(通常为服务端处理好排序) if (options.autoSort) { if (type === 'asc') { //升序 thisData = layui.sort(data, field); } else if (type === 'desc') { //降序 thisData = layui.sort(data, field, true); } else { //清除排序 thisData = layui.sort(data, table.config.indexName); delete that.sortKey; } } res[options.response.dataName] = thisData || data; that.renderData(res, that.page, that.count, true); if (formEvent) { layui.event.call(th, MOD_NAME, 'sort(' + filter + ')', { field: field , type: type }); } }; //请求loading Class.prototype.loading = function (hide) { var that = this , options = that.config; if (options.loading) { if (hide) { that.layInit && that.layInit.remove(); delete that.layInit; that.layBox.find(ELEM_INIT).remove(); } else { that.layInit = $(['<div class="layui-table-init">' , '<i class="layui-icon layui-icon-loading layui-anim layui-anim-rotate layui-anim-loop"></i>' , '</div>'].join('')); that.layBox.append(that.layInit); } } }; //同步选中值状态 Class.prototype.setCheckData = function (index, checked) { var that = this , options = that.config , thisData = table.cache[that.key]; if (!thisData[index]) return; if (thisData[index].constructor === Array) return; thisData[index][options.checkName] = checked; }; //同步全选按钮状态 Class.prototype.syncCheckAll = function () { var that = this , options = that.config , checkAllElem = that.layHeader.find('input[name="layTableCheckbox"]') , syncColsCheck = function (checked) { that.eachCols(function (i, item) { if (item.type === 'checkbox') { item[options.checkName] = checked; } }); return checked; }; if (!checkAllElem[0]) return; if (table.checkStatus(that.key).isAll) { if (!checkAllElem[0].checked) { checkAllElem.prop('checked', true); that.renderForm('checkbox'); } syncColsCheck(true); } else { if (checkAllElem[0].checked) { checkAllElem.prop('checked', false); that.renderForm('checkbox'); } syncColsCheck(false); } }; //获取cssRule Class.prototype.getCssRule = function (key, 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-' + key)) { 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); } if (!height) return; //减去列头区域的高度 bodyHeight = parseFloat(height) - (that.layHeader.outerHeight() || 38); //此处的数字常量是为了防止容器处在隐藏区域无法获得高度的问题,暂时只对默认尺寸的表格做支持。 //减去工具栏的高度 if (options.toolbar) { bodyHeight = bodyHeight - (that.layTool.outerHeight() || 50); } //减去统计朗的高度 if (options.totalRow) { bodyHeight = bodyHeight - (that.layTotal.outerHeight() || 40); } //减去分页栏的高度 if (options.page) { bodyHeight = bodyHeight - (that.layPage.outerHeight() || 41); } that.layMain.css('height', bodyHeight - 2); }; //获取滚动条宽度 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() //表格内容器的超出宽度 //添加补丁 , addPatch = function (elem) { if (scollWidth && scollHeight) { elem = elem.eq(0); if (!elem.find('.layui-table-patch')[0]) { var patchElem = $('<th class="layui-table-patch"><div class="layui-table-cell"></div></th>'); //补丁元素 patchElem.find('div').css({ width: scollWidth }); elem.find('tr').append(patchElem); } } else { elem.find('.layui-table-patch').remove(); } } addPatch(that.layHeader); addPatch(that.layTotal); //固定列区域高度 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'); //工具栏操作事件 that.layTool.on('click', '*[lay-event]', function (e) { var othis = $(this) , events = othis.attr('lay-event') , openPanel = function (sets) { var list = $(sets.list) , panel = $('<ul class="layui-table-tool-panel"></ul>'); panel.html(list); //限制最大高度 if (options.height) { panel.css('max-height', options.height - (that.layTool.outerHeight() || 50)); } //插入元素 othis.find('.layui-table-tool-panel')[0] || othis.append(panel); that.renderForm(); panel.on('click', function (e) { layui.stope(e); }); sets.done && sets.done(panel, list) }; layui.stope(e); _DOC.trigger('table.tool.panel.remove'); layer.close(that.tipsIndex); switch (events) { case 'LAYTABLE_COLS': //筛选列 openPanel({ list: function () { var lis = []; that.eachCols(function (i, item) { if (item.field && item.type == 'normal') { lis.push('<li><input type="checkbox" name="' + item.field + '" data-key="' + item.key + '" data-parentkey="' + (item.parentKey || '') + '" lay-skin="primary" ' + (item.hide ? '' : 'checked') + ' title="' + (item.title || item.field) + '" lay-filter="LAY_TABLE_TOOL_COLS"></li>'); } }); return lis.join(''); }() , done: function () { form.on('checkbox(LAY_TABLE_TOOL_COLS)', function (obj) { var othis = $(obj.elem) , checked = this.checked , key = othis.data('key') , parentKey = othis.data('parentkey'); layui.each(options.cols, function (i1, item1) { layui.each(item1, function (i2, item2) { if (i1 + '-' + i2 === key) { var hide = item2.hide; //同步勾选列的 hide 值和隐藏样式 item2.hide = !checked; that.elem.find('*[data-key="' + options.index + '-' + key + '"]') [checked ? 'removeClass' : 'addClass'](HIDE); //根据列的显示隐藏,同步多级表头的父级相关属性值 if (hide != item2.hide) { that.setParentCol(!checked, parentKey); } //重新适配尺寸 that.resize(); } }); }); }); } }); break; case 'LAYTABLE_EXPORT': //导出 if (device.ie) { layer.tips('导出功能不支持 IE,请用 Chrome 等高级浏览器导出', this, { tips: 3 }) } else { openPanel({ list: function () { return [ '<li data-type="csv">导出到 Csv 文件</li>' , '<li data-type="xls">导出到 Excel 文件</li>' ].join('') }() , done: function (panel, list) { list.on('click', function () { var type = $(this).data('type') table.exportFile.call(that, options.id, null, type); }); } }); } break; case 'LAYTABLE_PRINT': //打印 var printWin = window.open('打印窗口', '_blank') , style = ['<style>' , 'body{font-size: 12px; color: #666;}' , 'table{width: 100%; border-collapse: collapse; border-spacing: 0;}' , 'th,td{line-height: 20px; padding: 9px 15px; border: 1px solid #ccc; text-align: left; font-size: 12px; color: #666;}' , 'a{color: #666; text-decoration:none;}' , '*.layui-hide{display: none}' , '</style>'].join('') , html = $(that.layHeader.html()); //输出表头 html.append(that.layMain.find('table').html()); //输出表体 html.append(that.layTotal.find('table').html()) //输出合计行 html.find('th.layui-table-patch').remove(); //移除补丁 html.find('.layui-table-col-special').remove(); //移除特殊列 printWin.document.write(style + html.prop('outerHTML')); printWin.document.close(); printWin.print(); printWin.close(); break; } layui.event.call(this, MOD_NAME, 'toolbar(' + filter + ')', $.extend({ event: events , config: options }, {})); }); //拖拽调整宽度 th.on('mousemove', function (e) { var othis = $(this) , oLeft = othis.offset().left , pLeft = e.clientX - oLeft; if (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 key = othis.data('key'); e.preventDefault(); dict.resizeStart = true; //开始拖拽 dict.offset = [e.clientX, e.clientY]; //记录初始坐标 that.getCssRule(key, 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; } }); //排序 th.on('click', function (e) { var othis = $(this) , elemSort = othis.find(ELEM_SORT) , nowType = elemSort.attr('lay-sort') , type; if (!elemSort[0] || resizing === 1) return resizing = 2; if (nowType === 'asc') { type = 'desc'; } else if (nowType === 'desc') { type = null; } else { type = 'asc'; } that.sort(othis, type, null, true); }).find(ELEM_SORT + ' .layui-edge ').on('click', function (e) { var othis = $(this) , index = othis.index() , field = othis.parents('th').eq(0).data('field') layui.stope(e); if (index === 0) { that.sort(field, 'asc', null, true); } else { that.sort(field, 'desc', null, true); } }); //数据行中的事件监听返回的公共对象成员 var commonMember = function (sets) { var othis = $(this) , index = othis.parents('tr').eq(0).data('index') , tr = that.layBody.find('tr[data-index="' + index + '"]') , data = table.cache[that.key] || []; data = data[index] || {}; return $.extend({ tr: tr //行元素 , data: table.clearCacheKey(data) //当前行数据 , del: function () { //删除行数据 table.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(parseTempData({ templet: templet }, value, data)); td.data('content', value); } }); } }, sets); }; //复选框选择 that.elem.on('click', 'input[name="layTableCheckbox"]+', function () { //替代元素的 click 事件 var checkbox = $(this).prev() , childs = that.layBody.find('input[name="layTableCheckbox"]') , index = checkbox.parents('tr').eq(0).data('index') , checked = checkbox[0].checked , isAll = checkbox.attr('lay-filter') === 'layTableAllChoose'; //全选 if (isAll) { childs.each(function (i, item) { item.checked = checked; that.setCheckData(i, checked); }); that.syncCheckAll(); that.renderForm('checkbox'); } else { that.setCheckData(index, checked); that.syncCheckAll(); } layui.event.call(checkbox[0], MOD_NAME, 'checkbox(' + filter + ')', commonMember.call(checkbox[0], { checked: checked , type: isAll ? 'all' : 'one' })); }); //单选框选择 that.elem.on('click', 'input[lay-type="layTableRadio"]+', function () { var radio = $(this).prev() , checked = radio[0].checked , thisData = table.cache[that.key] , index = radio.parents('tr').eq(0).data('index'); //重置数据单选属性 layui.each(thisData, function (i, item) { if (index === i) { item.LAY_CHECKED = true; } else { delete item.LAY_CHECKED; } }); that.setThisRowChecked(index); layui.event.call(this, MOD_NAME, 'radio(' + filter + ')', commonMember.call(this, { checked: checked })); }); //行事件 that.layBody.on('mouseenter', 'tr', function () { //鼠标移入行 var othis = $(this) , index = othis.index(); if (othis.data('off')) return; //不触发事件 that.layBody.find('tr:eq(' + index + ')').addClass(ELEM_HOVER) }).on('mouseleave', 'tr', function () { //鼠标移出行 var othis = $(this) , index = othis.index(); if (othis.data('off')) return; //不触发事件 that.layBody.find('tr:eq(' + index + ')').removeClass(ELEM_HOVER) }).on('click', 'tr', function () { //单击行 setRowEvent.call(this, 'row'); }).on('dblclick', 'tr', function () { //双击行 setRowEvent.call(this, 'rowDouble'); }); //创建行单击、双击事件监听 var setRowEvent = function (eventType) { var othis = $(this); if (othis.data('off')) return; //不触发事件 layui.event.call(this, MOD_NAME, eventType + '(' + filter + ')' , commonMember.call(othis.children('td')[0]) ); }; //单元格编辑 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 = table.cache[that.key][index]; data[field] = value; //更新缓存中的值 layui.event.call(this, MOD_NAME, 'edit(' + filter + ')', commonMember.call(this, { value: value , field: field })); }).on('blur', '.' + ELEM_EDIT, function () { var templet , othis = $(this) , thisElem = this , field = othis.parent().data('field') , index = othis.parents('tr').eq(0).data('index') , data = table.cache[that.key][index]; that.eachCols(function (i, item) { if (item.field == field && item.templet) { templet = item.templet; } }); othis.siblings(ELEM_CELL).html(function (value) { return parseTempData({ templet: templet }, value, data); }(thisElem.value)); othis.parent().data('content', thisElem.value); othis.remove(); }); //单元格单击事件 that.layBody.on('click', 'td', function (e) { var othis = $(this) , field = othis.data('field') , editType = othis.data('edit') , elemCell = othis.children(ELEM_CELL); if (othis.data('off')) return; //不触发事件 //显示编辑表单 if (editType) { var input = $('<input class="layui-input ' + ELEM_EDIT + '">'); input[0].value = othis.data('content') || elemCell.text(); othis.find('.' + ELEM_EDIT)[0] || othis.append(input); input.focus(); layui.stope(e); return; } }).on('mouseenter', 'td', function () { gridExpand.call(this) }).on('mouseleave', 'td', function () { gridExpand.call(this, 'hide'); }); //单元格展开图标 var ELEM_GRID = 'layui-table-grid', ELEM_GRID_DOWN = 'layui-table-grid-down', ELEM_GRID_PANEL = 'layui-table-grid-panel' , gridExpand = function (hide) { var othis = $(this) , elemCell = othis.children(ELEM_CELL); if (othis.data('off')) return; //不触发事件 if (hide) { othis.find('.layui-table-grid-down').remove(); } else if (elemCell.prop('scrollWidth') > elemCell.outerWidth()) { if (elemCell.find('.' + ELEM_GRID_DOWN)[0]) return; othis.append('<div class="' + ELEM_GRID_DOWN + '"><i class="layui-icon layui-icon-down"></i></div>'); } }; //单元格展开事件 that.layBody.on('click', '.' + ELEM_GRID_DOWN, function (e) { var othis = $(this) , td = othis.parent() , elemCell = td.children(ELEM_CELL); that.tipsIndex = layer.tips([ '<div class="layui-table-tips-main" style="margin-top: -' + (elemCell.height() + 16) + 'px;' + function () { if (options.size === 'sm') { return 'padding: 4px 15px; font-size: 12px;'; } if (options.size === 'lg') { return 'padding: 14px 15px;'; } return ''; }() + '">' , elemCell.html() , '</div>' , '<i class="layui-icon layui-table-tips-c layui-icon-close"></i>' ].join(''), elemCell[0], { tips: [3, ''] , time: -1 , anim: -1 , maxWidth: (device.ios || device.android) ? 300 : that.elem.width() / 2 , isOutAnim: false , skin: 'layui-table-tips' , success: function (layero, index) { layero.find('.layui-table-tips-c').on('click', function () { layer.close(index); }); } }); layui.stope(e); }); //行工具条操作事件 that.layBody.on('click', '*[lay-event]', function () { var othis = $(this) , index = othis.parents('tr').eq(0).data('index'); layui.event.call(this, MOD_NAME, 'tool(' + filter + ')', commonMember.call(this, { event: othis.attr('lay-event') })); that.setThisRowChecked(index); }); //同步滚动条 that.layMain.on('scroll', function () { var othis = $(this) , scrollLeft = othis.scrollLeft() , scrollTop = othis.scrollTop(); that.layHeader.scrollLeft(scrollLeft); that.layTotal.scrollLeft(scrollLeft); that.layFixed.find(ELEM_BODY).scrollTop(scrollTop); layer.close(that.tipsIndex); }); //自适应 _WIN.on('resize', function () { that.resize(); }); }; //一次性事件 ; (function () { //全局点击 _DOC.on('click', function () { _DOC.trigger('table.remove.tool.panel'); }); //工具面板移除事件 _DOC.on('table.remove.tool.panel', function () { $('.layui-table-tool-panel').remove(); }); })(); //初始化 table.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), tableData = othis.attr('lay-data'); try { tableData = new Function('return ' + tableData)(); } catch (e) { hint.error(errorTips + tableData) } 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' //偶数行背景 }, table.config, settings, tableData); 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; }); table.render(options); }); return that; }; //记录所有实例 thisTable.that = {}; //记录所有实例对象 thisTable.config = {}; //记录所有实例配置项 //遍历表头 table.eachCols = function (id, callback, cols) { var config = thisTable.config[id] || {} , arrs = [], index = 0; cols = $.extend(true, [], cols || config.cols); //重新整理表头结构 layui.each(cols, function (i1, item1) { layui.each(item1, function (i2, item2) { //如果是组合列,则捕获对应的子列 if (item2.colGroup) { var childIndex = 0; index++ item2.CHILD_COLS = []; layui.each(cols[i1 + 1], function (i22, item22) { //如果子列已经被标注为{PARENT_COL_INDEX},或者子列累计 colspan 数等于父列定义的 colspan,则跳出当前子列循环 if (item22.PARENT_COL_INDEX || (childIndex > 1 && childIndex == item2.colspan)) return; item22.PARENT_COL_INDEX = index; item2.CHILD_COLS.push(item22); childIndex = childIndex + parseInt(item22.colspan > 1 ? item22.colspan : 1); }); } if (item2.PARENT_COL_INDEX) return; //如果是子列,则不进行追加,因为已经存储在父列中 arrs.push(item2) }); }); //重新遍历列,如果有子列,则进入递归 var eachArrs = function (obj) { layui.each(obj || arrs, function (i, item) { if (item.CHILD_COLS) return eachArrs(item.CHILD_COLS); typeof callback === 'function' && callback(i, item); }); }; eachArrs(); }; //表格选中状态 table.checkStatus = function (id) { var nums = 0 , invalidNum = 0 , arr = [] , data = table.cache[id] || []; //计算全选个数 layui.each(data, function (i, item) { if (item.constructor === Array) { invalidNum++; //无效数据,或已删除的 return; } if (item[table.config.checkName]) { nums++; arr.push(table.clearCacheKey(item)); } }); return { data: arr //选中的数据 , isAll: data.length ? (nums === (data.length - invalidNum)) : false //是否全选 }; }; //表格导出 table.exportFile = function (id, data, type) { var that = this; data = data || table.clearCacheKey(table.cache[id]); type = type || 'csv'; var config = thisTable.config[id] || {} , textType = ({ csv: 'text/csv' , xls: 'application/vnd.ms-excel' })[type] , alink = document.createElement("a"); if (device.ie) return hint.error('IE_NOT_SUPPORT_EXPORTS'); alink.href = 'data:' + textType + ';charset=utf-8,\ufeff' + encodeURIComponent(function () { var dataTitle = [], dataMain = [], dataTotal = []; //表头和表体 layui.each(data, function (i1, item1) { var vals = []; if (typeof id === 'object') { //如果 id 参数直接为表头数据 layui.each(id, function (i, item) { i1 == 0 && dataTitle.push(item || ''); }); layui.each(table.clearCacheKey(item1), function (i2, item2) { vals.push('"' + (item2 || '') + '"'); }); } else { table.eachCols(id, function (i3, item3) { if (item3.field && item3.type == 'normal' && !item3.hide) { var content = item1[item3.field]; if (content === undefined || content === null) content = ''; i1 == 0 && dataTitle.push(item3.title || ''); vals.push('"' + parseTempData(item3, content, item1, 'text') + '"'); } }); } dataMain.push(vals.join(',')); }); //表合计 layui.each(that.dataTotal, function (key, value) { dataTotal.push(value); }); return dataTitle.join(',') + '\r\n' + dataMain.join('\r\n') + '\r\n' + dataTotal.join(','); }()); alink.download = (config.title || 'table_' + (config.index || '')) + '.' + type; document.body.appendChild(alink); alink.click(); document.body.removeChild(alink); }; //重置表格尺寸结构 table.resize = function (id) { //如果指定表格唯一 id,则只执行该 id 对应的表格实例 if (id) { var config = getThisTableConfig(id); //获取当前实例配置项 if (!config) return; thisTable.that[id].resize(); } else { //否则重置所有表格实例尺寸 layui.each(thisTable.that, function () { this.resize(); }); } }; //表格重载 table.reload = function (id, options) { var config = getThisTableConfig(id); //获取当前实例配置项 if (!config) return; var that = thisTable.that[id]; that.reload(options); return thisTable.call(that); }; //核心入口 table.render = function (options) { var inst = new Class(options); return thisTable.call(inst); }; //清除临时Key table.clearCacheKey = function (data) { data = $.extend({}, data); delete data[table.config.checkName]; delete data[table.config.indexName]; return data; }; //自动完成渲染 table.init(); exports(MOD_NAME, table); });