TaoLer/public/static/admin/modules/treeTable.js
2021-05-10 13:59:58 +08:00

2266 lines
102 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/** 树形表格3.x Created by wangfan on 2020-05-12 https://gitee.com/whvse/treetable-lay */
layui.define(['laytpl', 'form', 'util'], function (exports) {
var $ = layui.jquery;
var laytpl = layui.laytpl;
var form = layui.form;
var util = layui.util;
var device = layui.device();
var MOD_NAME = 'treeTable'; // 模块名
var _instances = {}; // 记录所有实例
/* 表格默认参数 */
var defaultOption = {
elem: undefined, // 容器
cols: undefined, // 列参数
url: undefined, // url模式请求
method: undefined, // url模式请求方式
where: undefined, // url模式请求条件
contentType: undefined, // url模式请求类型
headers: undefined, // url模式请求headers
parseData: undefined, // url模式处理请求数据
request: {pidName: 'pid'}, // url模式请求字段自定义
toolbar: undefined, // 表头工具栏
defaultToolbar: undefined, // 表头工具栏右侧按钮
width: undefined, // 容器宽度
height: undefined, // 容器高度
cellMinWidth: 90, // 单元格最小宽度
done: undefined, // 数据处理完回调
data: undefined, // 直接赋值数据
title: undefined, // 定义table大标题文件导出会用到
skin: undefined, // 表格风格
even: undefined, // 是否开启隔行变色
size: undefined, // 表格尺寸
text: {
none: '无数据' // 空数据提示
},
reqData: undefined, // 自定义加载数据方法
useAdmin: false, // 是否使用admin.ajax
tree: {
idName: 'id', // id的字段名
pidName: 'pid', // pid的字段名
childName: 'children', // children的字段名
haveChildName: 'haveChild', // 是否有children标识的字段名
haveChildReverse: false, // 是否将children标识的字段取反
openName: 'open', // 是否默认展开的字段名
iconIndex: 0, // 图标列的索引
arrowType: undefined, // 折叠箭头类型
onlyIconControl: undefined, // 仅点击图标控制展开折叠
getIcon: function (d) { // 自定义图标
var haveChild = d[this.haveChildName];
if (haveChild !== undefined) {
haveChild = haveChild === true || haveChild === 'true';
if (this.haveChildReverse) haveChild = !haveChild;
}
else if (d[this.childName]) haveChild = d[this.childName].length > 0;
if (haveChild) return '<i class="ew-tree-icon layui-icon layui-icon-layer"></i>';
else return '<i class="ew-tree-icon layui-icon layui-icon-file"></i>';
}
}
};
/* 列默认参数 */
var colDefaultOption = {
field: undefined, // 字段名
title: undefined, // 标题
width: undefined, // 宽度
minWidth: undefined, // 最小宽度
type: 'normal', // 列类型
fixed: undefined, // 固定列
hide: undefined, // 是否初始隐藏列
unresize: undefined, // 禁用拖拽列宽
style: undefined, // 单元格样式
align: undefined, // 对齐方式
colspan: undefined, // 单元格所占的列数
rowspan: undefined, // 单元格所占的行数
templet: undefined, // 自定义模板
toolbar: undefined, // 工具列
'class': undefined, // 单元格class
singleLine: undefined // 是否一行显示
};
/** TreeTable类构造方法 */
var TreeTable = function (options) {
_instances[options.elem.substring(1)] = this;
this.reload(options);
};
/**
* 根据ID查找数据
* @param id 数据条目的ID
*/
TreeTable.prototype.findDataById = function (id) {
var options = this.options;
function each(data) {
for (var i = 0; i < data.length; i++) {
if (data[i][options.tree.idName] === id) return data[i];
if (data[i][options.tree.childName]) {
var res = each(data[i][options.tree.childName]);
if (res) return res;
}
}
}
return each(options.data);
}
/** 参数设置 */
TreeTable.prototype.initOptions = function (opt) {
var that = this;
// 处理特殊列
function initCol(item) {
if (!item.INIT_OK) item = $.extend({INIT_OK: true}, colDefaultOption, item);
// 特殊列处理
if (item.type === 'space') { // 空列
if (!item.width) item.width = 15;
item.minWidth = item.width;
} else if (item.type === 'numbers') { // 序号列
if (!item.width) item.width = 40;
item.minWidth = item.width;
if (!item.singleLine) item.singleLine = false;
if (!item.unresize) item.unresize = true;
if (!item.align) item.align = 'center';
} else if (item.type === 'checkbox' || item.type === 'radio') { // 复/单选框列
if (!item.width) item.width = 48;
item.minWidth = item.width;
if (!item.singleLine) item.singleLine = false;
if (!item.unresize) item.unresize = true;
if (!item.align) item.align = 'center';
}
if (item.toolbar) item.type = 'tool';
return item;
}
// 初始化列参数
if ('Array' !== isClass(opt.cols[0])) opt.cols = [opt.cols];
// 恢复cols参数初始状态
for (var m = 0; m < opt.cols.length; m++) {
for (var n = 0; n < opt.cols[m].length; n++) {
opt.cols[m][n].INIT_OK = undefined;
opt.cols[m][n].key = undefined;
opt.cols[m][n].colGroup = undefined;
opt.cols[m][n].HAS_PARENT = undefined;
opt.cols[m][n].parentKey = undefined;
opt.cols[m][n].PARENT_COL_INDEX = undefined;
}
}
// cols参数处理
var colArrays = [], colIndex = 0;
for (var i1 = 0; i1 < opt.cols.length; i1++) {
var item1 = opt.cols[i1];
for (var i2 = 0; i2 < item1.length; i2++) {
var item2 = item1[i2];
if (!item2) {
item1.splice(i2, 1);
continue;
}
item2 = initCol(item2);
// 合并单元格处理
item2.key = i1 + '-' + i2;
var CHILD_COLS = undefined;
if (item2.colGroup || item2.colspan > 1) {
item2.colGroup = true;
item2.type = 'group';
CHILD_COLS = [];
colIndex++;
var childIndex = 0;
for (var i22 = 0; i22 < opt.cols[i1 + 1].length; i22++) {
var item22 = $.extend({INIT_OK: true}, colDefaultOption, opt.cols[i1 + 1][i22]);
if (item22.HAS_PARENT || (childIndex > 1 && childIndex == item2.colspan)) {
opt.cols[i1 + 1][i22] = item22;
continue;
}
item22.HAS_PARENT = true;
item22.parentKey = i1 + '-' + i2;
item22.key = (i1 + 1) + '-' + i22;
item22.PARENT_COL_INDEX = colIndex;
item22 = initCol(item22);
CHILD_COLS.push(item22);
childIndex = childIndex + parseInt(item22.colspan > 1 ? item22.colspan : 1);
opt.cols[i1 + 1][i22] = item22;
}
}
item2.CHILD_COLS = CHILD_COLS;
if (!item2.PARENT_COL_INDEX) colArrays.push(item2);
opt.cols[i1][i2] = item2;
}
}
this.options = $.extend(true, {}, defaultOption, opt);
this.options.colArrays = colArrays;
// url加载模式转为reqData模式
if (this.options.url) {
this.options.reqData = function (data, callback) {
if (!that.options.where) that.options.where = {};
if (data) that.options.where[that.options.request.pidName] = data[that.options.tree.idName];
(that.options.useAdmin ? layui.admin : $).ajax({
url: that.options.url,
data: that.options.contentType && that.options.contentType.indexOf('application/json') === 0 ? JSON.stringify(that.options.where) : that.options.where,
headers: that.options.headers,
type: that.options.method,
dataType: 'json',
contentType: that.options.contentType,
success: function (res) {
if (that.options.parseData) res = that.options.parseData(res);
if (res.code == 0) callback(res.data);
else callback(res.msg || '加载失败');
},
error: function (xhr) {
callback(xhr.status + ' - ' + xhr.statusText);
}
});
};
} else if (this.options.data && this.options.data.length > 0 && this.options.tree.isPidData) { // pid形式数据转children形式
this.options.data = tt.pidToChildren(this.options.data, this.options.tree.idName, this.options.tree.pidName, this.options.tree.childName);
}
// toolbar参数处理
if ('default' === this.options.toolbar) {
this.options.toolbar = [
'<div>',
' <div class="ew-tree-table-tool-item" title="添加" lay-event="add">',
' <i class="layui-icon layui-icon-add-1"></i>',
' </div>',
' <div class="ew-tree-table-tool-item" title="修改" lay-event="update">',
' <i class="layui-icon layui-icon-edit"></i>',
' </div>',
' <div class="ew-tree-table-tool-item" title="删除" lay-event="delete">',
' <i class="layui-icon layui-icon-delete"></i>',
' </div>',
'</div>'
].join('');
}
if (this.options.defaultToolbar === undefined) this.options.defaultToolbar = ['filter', 'exports', 'print'];
// 自定义图标参数处理
if (typeof this.options.tree.getIcon === 'string') {
var icon = this.options.tree.getIcon;
this.options.tree.getIcon = function (d) {
if (icon !== 'ew-tree-icon-style2') return icon;
var haveChild = d[this.haveChildName];
if (haveChild !== undefined) {
haveChild = haveChild === true || haveChild === 'true';
if (this.haveChildReverse) haveChild = !haveChild;
}
else if (d[this.childName]) haveChild = d[this.childName].length > 0;
if (haveChild) return '<i class="ew-tree-icon ew-tree-icon-folder"></i>';
else return '<i class="ew-tree-icon ew-tree-icon-file"></i>';
}
}
};
/** 初始化表格 */
TreeTable.prototype.init = function () {
var options = this.options;
var $elem = $(options.elem); // 原始表格
var tbFilter = options.elem.substring(1); // 表格的filter
// 第一次生成树表格dom
$elem.removeAttr('lay-filter');
if ($elem.next('.ew-tree-table').length === 0) {
$elem.css('display', 'none');
$elem.after([
'<div class="layui-form ew-tree-table" lay-filter="', tbFilter, '" style="', options.style || '', '">',
' <div class="ew-tree-table-tool" style="display: none;"></div>',
' <div class="ew-tree-table-head">',
' <table class="layui-table"></table>',
' </div>',
' <div class="ew-tree-table-box">',
' <table class="layui-table"></table>',
' <div class="ew-tree-table-loading">',
' <i class="layui-icon layui-icon-loading layui-anim layui-anim-rotate layui-anim-loop"></i>',
' </div>',
' <div class="ew-tree-table-empty">', options.text.none || '', '</div>',
' </div>',
'</div>'
].join(''));
}
// 获取各个组件
var components = this.getComponents();
// 基础参数设置
if (options.skin) components.$table.attr('lay-skin', options.skin);
if (options.size) components.$table.attr('lay-size', options.size);
if (options.even) components.$table.attr('lay-even', options.even);
// 头部工具栏
components.$toolbar.empty();
if (options.toolbar === false || options.toolbar === undefined) {
components.$toolbar.hide();
} else {
components.$toolbar.show();
if (typeof options.toolbar === 'string') {
laytpl($(options.toolbar).html()).render({}, function (html) {
components.$toolbar.html('<div style="display: inline-block;">' + html + '</div>');
});
}
var tbRights = ['<div class="ew-tree-table-tool-right">'];
for (var i = 0; i < options.defaultToolbar.length; i++) {
var tbItem;
if ('filter' === options.defaultToolbar[i]) {
tbItem = {title: '筛选', layEvent: 'LAYTABLE_COLS', icon: 'layui-icon-cols'};
} else if ('exports' === options.defaultToolbar[i]) {
tbItem = {title: '导出', layEvent: 'LAYTABLE_EXPORT', icon: 'layui-icon-export'};
} else if ('print' === options.defaultToolbar[i]) {
tbItem = {title: '打印', layEvent: 'LAYTABLE_PRINT', icon: 'layui-icon-print'};
} else {
tbItem = options.defaultToolbar[i];
}
if (tbItem) {
tbRights.push('<div class="ew-tree-table-tool-item"');
tbRights.push(' title="' + tbItem.title + '"');
tbRights.push(' lay-event="' + tbItem.layEvent + '">');
tbRights.push('<i class="layui-icon ' + tbItem.icon + '"></i></div>');
}
}
components.$toolbar.append(tbRights.join('') + '</div>');
}
// 固定宽度
if (options.width) {
components.$view.css('width', options.width);
components.$tHeadGroup.css('width', options.width);
components.$tBodyGroup.css('width', options.width);
}
// 表格尺寸设置
var colgroupHtml = this.resize(true);
// 生成thead
var headHtml = '<thead>' + this.renderBodyTh() + '</thead>';
// 渲染表头及空的表主体的结构
components.$tBodyGroup.children('style').remove();
if (options.height) { // 固定表头
components.$tHead.html(colgroupHtml + headHtml);
components.$tBody.html(colgroupHtml + '<tbody></tbody>');
if (options.height.indexOf('full-') === 0) { // 差值高度
var h = parseFloat(options.height.substring(5)) + components.$toolbar.outerHeight()
+ components.$tHeadGroup.outerHeight() + 1;
components.$tBodyGroup.append([
'<style>[lay-filter="', tbFilter, '"] .ew-tree-table-box {',
' height: ', getPageHeight() - h, 'px;',
' height: -moz-calc(100vh - ', h, 'px);',
' height: -webkit-calc(100vh - ', h, 'px);',
' height: calc(100vh - ', h, 'px);',
'}</style>'
].join(''));
components.$tBodyGroup.data('full', h);
components.$tBodyGroup.css('height', '');
} else { // 固定高度
components.$tBodyGroup.css('height', options.height);
components.$tBodyGroup.data('full', '');
}
components.$tHeadGroup.show();
} else {
components.$tHeadGroup.hide();
var trH = {lg: 50, sm: 30, md: 38};
components.$tBodyGroup.append([
'<style>[lay-filter="', tbFilter, '"] .ew-tree-table-box:before {',
' content: "";',
' position: absolute;',
' top: 0; left: 0; right: 0;',
' height: ' + (trH[options.size || 'md'] * options.cols.length) + 'px;',
' background-color: #f2f2f2;',
' border-bottom: 1px solid #e6e6e6;',
'}</style>'
].join(''));
components.$tBody.html(colgroupHtml + headHtml + '<tbody></tbody>');
}
form.render('checkbox', tbFilter); // 渲染表头的表单元素
// 默认隐藏列修正colspan
function patchHide($tr) {
var parentKey = $tr.data('parent'), pCol;
if (!parentKey) return;
var $parent = components.$table.children('thead').children('tr').children('[data-key="' + parentKey + '"]');
var colspan = $parent.attr('colspan') - 1;
$parent.attr('colspan', colspan);
if (colspan === 0) $parent.addClass('layui-hide');
patchHide($parent);
}
components.$table.children('thead').children('tr').children('th.layui-hide').each(function () {
patchHide($(this));
});
// 渲染数据
if (options.reqData) { // 异步加载
this.options.data = undefined;
this.renderBodyAsync();
} else if (options.data && options.data.length > 0) {
this.renderBodyData(options.data);
} else {
components.$loading.hide();
components.$empty.show();
}
};
/** 绑定各项事件 */
TreeTable.prototype.bindEvents = function () {
var that = this;
var options = this.options;
var components = this.getComponents();
var $allBody = components.$table.children('tbody');
/* 行事件公共返回对象 */
var member = function (ext) {
// 获取行dom
var $tr = $(this);
if (!$tr.is('tr')) {
var $temp = $tr.parent('tr');
if ($temp.length > 0) $tr = $temp;
else $tr = $tr.parentsUntil('tr').last().parent();
}
var data = that.getDataByTr($tr); // 行对应数据
var obj = {
tr: $tr,
data: data,
del: function () { // 删除行
var index = $tr.data('index');
var indent = parseInt($tr.data('indent'));
// 删除子级
$tr.nextAll('tr').each(function () {
if (parseInt($(this).data('indent')) <= indent) return false;
$(this).remove();
});
// 更新后面同辈的index
var indexLength = (typeof index === 'number' ? 1 : index.split('-').length);
$tr.nextAll('tr').each(function () {
var $this = $(this);
if (parseInt($this.data('indent')) < indent) return false;
var _index = $this.data('index').toString().split('-');
_index[indexLength - 1] = parseInt(_index[indexLength - 1]) - 1;
$this.data('index', _index.join('-'));
});
// 删除当前行
var $pTr = $tr.prevAll('tr');
that.del(undefined, index);
$tr.remove();
that.renderNumberCol(); // 渲染序号列
// 联动父级多选框
$pTr.each(function () {
var tInd = parseInt($(this).data('indent'));
if (tInd >= indent) return true;
that.checkParentCB($(this));
indent = tInd;
});
that.checkChooseAllCB(); // 联动全选框
if (options.data.length === 0) components.$empty.show();
updateFixedTbHead(components.$view); // 更新滚动条补丁
},
update: function (fields) { // 修改行
data = $.extend(true, data, fields);
var indent = parseInt($tr.data('indent'));
that.renderBodyTr(data, indent, undefined, $tr); // 更新界面
form.render(null, components.filter); // 渲染表单元素
that.renderNumberCol(); // 渲染序号列
// 联动父级多选框
$tr.prevAll('tr').each(function () {
var tInd = parseInt($(this).data('indent'));
if (tInd >= indent) return true;
that.checkParentCB($(this));
indent = tInd;
});
that.checkChooseAllCB(); // 联动全选框
}
};
return $.extend(obj, ext);
};
// 绑定折叠展开事件
$allBody.off('click.fold').on('click.fold', '.ew-tree-pack', function (e) {
layui.stope(e);
var $tr = $(this).parentsUntil('tr').last().parent();
if ($tr.hasClass('ew-tree-table-loading')) return; // 已是加载中
var haveChild = $tr.data('have-child');
if (haveChild !== true && haveChild !== 'true') return; // 子节点
var open = $tr.hasClass('ew-tree-table-open');
var data = that.getDataByTr($tr);
if (!open && !data[options.tree.childName]) {
that.renderBodyAsync(data, $tr);
} else {
data[options.tree.openName] = toggleRow($tr);
}
});
// 绑定lay-event事件
$allBody.off('click.tool').on('click.tool', '*[lay-event]', function (e) {
layui.stope(e);
var $this = $(this);
layui.event.call(this, MOD_NAME, 'tool(' + components.filter + ')', member.call(this, {
event: $this.attr('lay-event')
}));
});
// 绑定单选框事件
form.on('radio(' + components.radioFilter + ')', function (data) {
var d = that.getDataByTr($(data.elem).parentsUntil('tr').last().parent());
that.removeAllChecked();
d.LAY_CHECKED = true; // 同时更新数据
d.LAY_INDETERMINATE = false;
layui.event.call(this, MOD_NAME, 'checkbox(' + components.filter + ')',
{checked: true, data: d, type: 'one'});
});
// 绑定复选框事件
form.on('checkbox(' + components.checkboxFilter + ')', function (data) {
var checked = data.elem.checked;
var $cb = $(data.elem);
var $layCb = $cb.next('.layui-form-checkbox');
// 如果是半选状态,点击全选
if (!checked && $cb.hasClass('ew-form-indeterminate')) {
checked = true;
$cb.prop('checked', checked);
$layCb.addClass('layui-form-checked');
$cb.removeClass('ew-form-indeterminate');
}
var $tr = $cb.parentsUntil('tr').last().parent();
var d = that.getDataByTr($tr);
d.LAY_CHECKED = checked; // 同时更新数据
d.LAY_INDETERMINATE = false;
// 联动操作
if (d[options.tree.childName] && d[options.tree.childName].length > 0) {
that.checkSubCB($tr, checked); // 联动子级
}
var indent = parseInt($tr.data('indent'));
$tr.prevAll('tr').each(function () {
var tInd = parseInt($(this).data('indent'));
if (tInd < indent) {
that.checkParentCB($(this)); // 联动父级
indent = tInd;
}
});
that.checkChooseAllCB(); // 联动全选框
// 回调事件
layui.event.call(this, MOD_NAME, 'checkbox(' + components.filter + ')',
{checked: checked, data: d, type: 'more'});
});
// 绑定全选复选框事件
form.on('checkbox(' + components.chooseAllFilter + ')', function (data) {
var checked = data.elem.checked;
var $cb = $(data.elem);
var $layCb = $cb.next('.layui-form-checkbox');
if (!options.data || options.data.length === 0) { // 如果数据为空
$cb.prop('checked', false);
$layCb.removeClass('layui-form-checked');
$cb.removeClass('ew-form-indeterminate');
return;
}
// 如果是半选状态,点击全选
if (!checked && $cb.hasClass('ew-form-indeterminate')) {
checked = true;
$cb.prop('checked', checked);
$layCb.addClass('layui-form-checked');
$cb.removeClass('ew-form-indeterminate');
}
layui.event.call(this, MOD_NAME, 'checkbox(' + components.filter + ')', {checked: checked, type: 'all'});
that.checkSubCB(components.$tBody.children('tbody'), checked); // 联动操作
});
// 绑定行单击事件
$allBody.off('click.row').on('click.row', 'tr', function () {
layui.event.call(this, MOD_NAME, 'row(' + components.filter + ')', member.call(this, {}));
});
// 绑定行双击事件
$allBody.off('dblclick.rowDouble').on('dblclick.rowDouble', 'tr', function () {
layui.event.call(this, MOD_NAME, 'rowDouble(' + components.filter + ')', member.call(this, {}));
});
// 绑定单元格点击事件
$allBody.off('click.cell').on('click.cell', 'td', function (e) {
var $td = $(this);
var type = $td.data('type');
// 判断是否是复选框、单选框列
if (type === 'checkbox' || type === 'radio') return layui.stope(e);
var edit = $td.data('edit');
var field = $td.data('field');
if (edit) { // 开启了单元格编辑
layui.stope(e);
if ($allBody.find('.ew-tree-table-edit').length > 0) return;
var index = $td.data('index');
var indent = $td.find('.ew-tree-table-indent').length;
var d = that.getDataByTr($td.parent());
if ('text' === edit || 'number' === edit) { // 文本框
var $input = $('<input type="' + edit + '" class="layui-input ew-tree-table-edit"/>');
$input[0].value = d[field];
$td.append($input);
$input.focus();
$input.blur(function () {
var value = $(this).val();
if (value == d[field]) return $(this).remove();
var rs = layui.event.call(this, MOD_NAME, 'edit(' + components.filter + ')', member.call(this,
{value: value, field: field}));
if (rs === false) {
$(this).addClass('layui-form-danger');
$(this).focus();
} else {
d[field] = value; // 同步更新数据
var keys = $td.data('key').split('-');
that.renderBodyTd(d, indent, index, $td, options.cols[keys[0]][keys[1]]); // 更新单元格
}
});
} else {
console.error('不支持的单元格编辑类型:' + edit);
}
} else { // 回调单元格点击事件
var rs = layui.event.call(this, MOD_NAME, 'cell(' + components.filter + ')', member.call(this,
{td: $td, field: field}));
if (rs === false) layui.stope(e);
}
});
// 绑定单元格双击事件
$allBody.off('dblclick.cellDouble').on('dblclick.cellDouble', 'td', function (e) {
var $td = $(this);
var type = $td.data('type');
// 判断是否是复选框、单选框列
if (type === 'checkbox' || type === 'radio') return layui.stope(e);
var edit = $td.data('edit');
var field = $td.data('field');
if (edit) return layui.stope(e); // 开启了单元格编辑
// 回调单元格双击事件
var rs = layui.event.call(this, MOD_NAME, 'cellDouble(' + components.filter + ')', member.call(this,
{td: $td, field: field}));
if (rs === false) layui.stope(e);
});
// 绑定头部工具栏事件
components.$toolbar.off('click.toolbar').on('click.toolbar', '*[lay-event]', function (e) {
layui.stope(e);
var $this = $(this);
var event = $this.attr('lay-event');
if ('LAYTABLE_COLS' === event) that.toggleCol();
else if ('LAYTABLE_EXPORT' === event) that.exportData('show');
else if ('LAYTABLE_PRINT' === event) that.printTable();
else layui.event.call(this, MOD_NAME, 'toolbar(' + components.filter + ')', {event: event, elem: $this});
});
// 同步滚动条
components.$tBodyGroup.on('scroll', function () {
var $this = $(this);
components.$tHeadGroup.scrollLeft($this.scrollLeft());
});
// 导出数据
components.$toolbar.off('click.export').on('click.export', '.layui-table-tool-panel>[data-type]', function () {
var type = $(this).data('type');
if ('csv' === type || 'xls' === type) that.exportData(type);
});
components.$toolbar.off('click.panel').on('click.panel', '.layui-table-tool-panel', function (e) {
layui.stope(e);
});
// 筛选列
form.on('checkbox(' + components.colsToggleFilter + ')', function (data) {
that.toggleCol(data.elem.checked, undefined, data.value);
});
};
/** 获取各个组件 */
TreeTable.prototype.getComponents = function () {
var $view = $(this.options.elem).next('.ew-tree-table'); // 容器
var filter = $view.attr('lay-filter'); // 容器filter
var $tHeadGroup = $view.children('.ew-tree-table-head'); // 表头部分容器
var $tBodyGroup = $view.children('.ew-tree-table-box'); // 主体部分容器
return {
$view: $view,
filter: filter,
$tHeadGroup: $tHeadGroup,
$tBodyGroup: $tBodyGroup,
$tHead: $tHeadGroup.children('.layui-table'), // 表头表格
$tBody: $tBodyGroup.children('.layui-table'), // 主体表格
$table: $view.find('.layui-table'), // 所有表格
$toolbar: $view.children('.ew-tree-table-tool'), // 头部工具栏
$empty: $tBodyGroup.children('.ew-tree-table-empty'), // 空视图
$loading: $tBodyGroup.children('.ew-tree-table-loading'), // 加载视图
checkboxFilter: 'ew_tb_checkbox_' + filter, // 复选框filter
radioFilter: 'ew_tb_radio_' + filter, // 单选框filter
chooseAllFilter: 'ew_tb_choose_all_' + filter, // 全选按钮filter
colsToggleFilter: 'ew_tb_toggle_cols' + filter // 筛选列的filter
};
};
/**
* 遍历表头
* @param callback
* @param obj
*/
TreeTable.prototype.eachCols = function (callback, obj) {
if (!obj) obj = this.options.colArrays;
for (var i = 0; i < obj.length; i++) {
var item = obj[i];
callback && callback(i, item);
if (item.CHILD_COLS) this.eachCols(callback, item.CHILD_COLS);
}
};
/**
* 遍历数据
* @param callback
* @param data
*/
TreeTable.prototype.eachData = function (callback, data) {
if (!data) data = this.options.data;
for (var i = 0; i < data.length; i++) {
var item = data[i];
callback && callback(i, item);
if (item[this.options.tree.childName]) this.eachData(callback, item[this.options.tree.childName]);
}
};
/**
* 异步加载渲染
* @param d 父级数据
* @param $tr 父级dom
*/
TreeTable.prototype.renderBodyAsync = function (d, $tr) {
var that = this;
var options = this.options;
var components = this.getComponents();
// 显示loading
if ($tr) {
$tr.addClass('ew-tree-table-loading');
$tr.find('.ew-tree-pack').children('.ew-tree-table-arrow').addClass('layui-anim layui-anim-rotate layui-anim-loop');
} else {
components.$empty.hide();
if (options.data && options.data.length > 0) components.$loading.addClass('ew-loading-float');
components.$loading.show();
}
// 请求数据
options.reqData(d, function (data) {
if (typeof data !== 'string' && data && data.length > 0 && options.tree.isPidData) {
data = tt.pidToChildren(data, options.tree.idName, options.tree.pidName, options.tree.childName);
}
that.renderBodyData(data, d, $tr); // 渲染内容
});
};
/**
* 根据数据渲染body
* @param data 数据集合
* @param d 父级数据
* @param $tr 父级dom
*/
TreeTable.prototype.renderBodyData = function (data, d, $tr) {
var msg;
if (typeof data === 'string') {
msg = data;
data = [];
}
var that = this;
var options = this.options;
var components = this.getComponents();
// 更新到数据
if (d === undefined) options.data = data;
else d[options.tree.childName] = data;
var indent;
if ($tr) {
indent = parseInt($tr.data('indent')) + 1;
d[options.tree.openName] = true;
}
var htmlStr = this.renderBody(data, indent, d);
if ($tr) {
// 移除旧dom
$tr.nextAll('tr').each(function () {
if (parseInt($(this).data('indent')) <= (indent - 1)) return false;
$(this).remove();
});
// 渲染新dom
$tr.after(htmlStr).addClass('ew-tree-table-open');
} else {
components.$tBody.children('tbody').html(htmlStr);
}
form.render(null, components.filter); // 渲染表单元素
this.renderNumberCol(); // 渲染序号列
if ($tr) {
// 更新父级复选框状态
this.checkParentCB($tr);
$tr.prevAll('tr').each(function () {
var tInd = parseInt($(this).data('indent'));
if (tInd < (indent - 1)) {
that.checkParentCB($(this));
indent = tInd + 1;
}
});
// 移除loading
$tr.removeClass('ew-tree-table-loading');
var $arrow = $tr.find('.ew-tree-pack').children('.ew-tree-table-arrow');
$arrow.removeClass('layui-anim layui-anim-rotate layui-anim-loop');
if (msg) { // 加载失败
$tr.removeClass('ew-tree-table-open');
} else if (data && data.length === 0) { // 无子集
d[options.tree.haveChildName] = !!options.tree.haveChildReverse;
$tr.data('have-child', false);
$arrow.addClass('ew-tree-table-arrow-hide');
$arrow.next('.ew-tree-icon').after(options.tree.getIcon(d)).remove();
}
} else {
// 移除loading
components.$loading.hide();
components.$loading.removeClass('ew-loading-float');
// 显示空视图
if (data && data.length > 0) {
components.$empty.hide();
} else {
components.$empty.show();
if (msg) components.$empty.text(msg);
else components.$empty.html(options.text.none);
}
}
this.checkChooseAllCB(); // 联动全选框
updateFixedTbHead(components.$view); // 滚动条补丁
options.done && options.done(data);
};
/**
* 递归渲染表格主体部分
* @param data 数据列表
* @param indent 缩进大小
* @param parent 父级
* @param h 父级是否隐藏
* @returns {string}
*/
TreeTable.prototype.renderBody = function (data, indent, parent, h) {
var options = this.options;
if (!indent) indent = 0;
var html = '';
if (!data || data.length === 0) return html;
var hide = parent ? !parent[options.tree.openName] : undefined;
if (h) hide = h;//当所有父级存在隐藏时,隐藏所有子集
for (var i = 0; i < data.length; i++) {
var d = data[i];
d.LAY_INDEX = (parent ? parent.LAY_INDEX + '-' : '') + i;
html += this.renderBodyTr(d, indent, hide);
// 递归渲染子集
html += this.renderBody(d[options.tree.childName], indent + 1, d, h);
}
return html;
};
/**
* 渲染每一行数据
* @param d 行数据
* @param indent 缩进大小
* @param hide 是否隐藏
* @param $tr
* @returns {string}
*/
TreeTable.prototype.renderBodyTr = function (d, indent, hide, $tr) {
var that = this;
var options = this.options;
if (!indent) indent = 0;
var haveChild = d[options.tree.haveChildName];
if (options.tree.haveChildReverse) haveChild = !haveChild;
if (haveChild === undefined) haveChild = d[options.tree.childName] && d[options.tree.childName].length > 0;
if ($tr) {
$tr.data('have-child', haveChild ? 'true' : 'false');
$tr.data('indent', indent);
$tr.removeClass('ew-tree-table-loading');
}
var html = '<tr';
var classNames = '';
if (haveChild && d[options.tree.openName]) classNames += 'ew-tree-table-open';
if (hide) classNames += 'ew-tree-tb-hide';
html += (' class="' + classNames + '"');
if (haveChild) html += (' data-have-child="' + haveChild + '"');
html += (' data-index="' + d.LAY_INDEX + '"');
html += (' data-indent="' + indent + '">');
var index = 0;
this.eachCols(function (i, col) {
if (col.colGroup) return;
html += that.renderBodyTd(d, indent, index, $tr ? $tr.children('td').eq(index) : undefined, col);
index++;
});
html += '</tr>';
return html;
};
/**
* 渲染每一个单元格数据
* @param d 行数据
* @param indent 缩进大小
* @param index 第几列
* @param $td
* @param col
* @returns {string}
*/
TreeTable.prototype.renderBodyTd = function (d, indent, index, $td, col) {
if (!col||col.colGroup) return '';
var options = this.options;
var components = this.getComponents();
if (!indent) indent = 0;
// 内容填充
var content = '', cell = '', icon = '';
if (col.type === 'numbers') { // 序号列
content = '<span class="ew-tree-table-numbers"></span>';
} else if (col.type === 'checkbox') { // 复选框列
content = [
'<input type="checkbox"', d.LAY_CHECKED ? ' checked="checked"' : '',
' lay-filter="', components.checkboxFilter, '"',
' lay-skin="primary" class="ew-tree-table-checkbox',
d.LAY_INDETERMINATE ? ' ew-form-indeterminate' : '', '" />'
].join('');
} else if (col.type === 'radio') { // 单选框列
content = [
'<input type="radio"', d.LAY_CHECKED ? ' checked="checked"' : '',
' lay-filter="', components.radioFilter, '"',
' name="', components.radioFilter, '"',
' class="ew-tree-table-radio" />'
].join('');
} else if (col.templet) { // 自定义模板
if (typeof col.templet === 'function') {
content = col.templet(d);
} else if (typeof col.templet === 'string') {
laytpl($(col.templet).html()).render(d, function (html) {
content = html;
});
}
} else if (col.toolbar) { // 操作列
if (typeof col.toolbar === 'function') {
content = col.toolbar(d);
} else if (typeof col.toolbar === 'string') {
laytpl($(col.toolbar).html()).render(d, function (html) {
content = html;
});
}
} else if (col.field && d[col.field] !== undefined && d[col.field] !== null) { // 普通字段
content = util.escape(d[col.field] === 0 ? '0' : d[col.field]);
}
// 图标列处理
if (index === options.tree.iconIndex) {
// 缩进
for (var i = 0; i < indent; i++) icon += '<span class="ew-tree-table-indent"></span>';
icon += '<span class="ew-tree-pack">';
// 加箭头
var haveChild = d[options.tree.haveChildName];
if (options.tree.haveChildReverse) haveChild = !haveChild;
if (haveChild === undefined) haveChild = d[options.tree.childName] && d[options.tree.childName].length > 0;
icon += ('<i class="ew-tree-table-arrow layui-icon' + (haveChild ? '' : ' ew-tree-table-arrow-hide'));
icon += (' ' + (options.tree.arrowType || '') + '"></i>');
// 加图标
icon += options.tree.getIcon(d);
content = '<span>' + content + '</span>';
if (options.tree.onlyIconControl) content = icon + '</span>' + content;
else content = icon + content + '</span>';
}
cell = [
'<div class="ew-tree-table-cell', col.singleLine === undefined || col.singleLine ? ' single-line' : '', '"',
col.align ? ' align="' + col.align + '"' : '',
'>',
' <div class="ew-tree-table-cell-content">', content, '</div>',
' <i class="layui-icon layui-icon-close ew-tree-tips-c"></i>',
' <div class="layui-table-grid-down" style="display: none;"><i class="layui-icon layui-icon-down"></i></div>',
'</div>'
].join('');
if ($td) $td.html(cell);
var html = '<td';
if (col.field) html += (' data-field="' + col.field + '"');
if (col.edit) html += (' data-edit="' + col.edit + '"');
if (col.type) html += (' data-type="' + col.type + '"');
if (col.key) html += (' data-key="' + col.key + '"');
if (col.style) html += (' style="' + col.style + '"');
if (col['class']) html += (' class="' + col['class'] + (col.hide ? ' layui-hide' : '') + '"');
else if (col.hide) html += (' class="layui-hide"');
html += ('>' + cell + '</td>');
return html;
};
/**
* 渲染表头
* @returns {string}
*/
TreeTable.prototype.renderBodyTh = function () {
var options = this.options;
var components = this.getComponents();
var html = [];
$.each(options.cols, function (i1, item1) {
html.push('<tr>');
$.each(item1, function (i2, item2) {
html.push('<th');
if (item2.colspan) html.push(' colspan="' + item2.colspan + '"');
if (item2.rowspan) html.push(' rowspan="' + item2.rowspan + '"');
if (item2.type) html.push(' data-type="' + item2.type + '"');
if (item2.key) html.push(' data-key="' + item2.key + '"');
if (item2.parentKey) html.push(' data-parent="' + item2.parentKey + '"');
if (item2.hide) html.push(' class="layui-hide"');
html.push('>');
html.push('<div class="ew-tree-table-cell' + (item2.singleLine === undefined || item2.singleLine ? ' single-line' : '') + '"');
if (item2.thAlign || item2.align) html.push(' align="' + (item2.thAlign || item2.align) + '"');
html.push('>');
html.push('<div class="ew-tree-table-cell-content">');
// 标题
var ca = '<input type="checkbox" lay-filter="' + components.chooseAllFilter + '" lay-skin="primary" class="ew-tree-table-checkbox"/>';
if (item2.type === 'checkbox') html.push(ca);
else html.push(item2.title || '');
html.push('</div><i class="layui-icon layui-icon-close ew-tree-tips-c"></i>');
html.push('<div class="layui-table-grid-down" style="display: none;"><i class="layui-icon layui-icon-down"></i></div></div>');
// 列宽拖拽
if (!item2.colGroup && !item2.unresize) html.push('<span class="ew-tb-resize"></span>');
html.push('</th>');
});
html.push('</tr>');
});
return html.join('');
};
/** 重置表格尺寸 */
TreeTable.prototype.resize = function (returnColgroup) {
// 计算表格宽度、最小宽度、百分比宽度
var options = this.options;
var components = this.getComponents();
var minWidth = 1, width = 1, needSetWidth = true, mwPercent = 0;
this.eachCols(function (i, item) {
if (item.colGroup || item.hide) return;
if (item.width) {
width += (item.width + 1);
if (item.minWidth) {
if (item.width < item.minWidth) item.width = item.minWidth;
} else if (item.width < options.cellMinWidth) item.width = options.cellMinWidth;
} else needSetWidth = false;
if (item.width) minWidth += (item.width + 1);
else if (item.minWidth) {
minWidth += (item.minWidth + 1);
mwPercent += item.minWidth;
} else {
minWidth += (options.cellMinWidth + 1);
mwPercent += options.cellMinWidth;
}
});
if (minWidth) {
components.$tHead.css('min-width', minWidth);
components.$tBody.css('min-width', minWidth);
} else {
components.$tHead.css('min-width', 'auto');
components.$tBody.css('min-width', 'auto');
}
if (needSetWidth) {
components.$tHead.css('width', width);
components.$tBody.css('width', width);
} else {
components.$tHead.css('width', '100%');
components.$tBody.css('width', '100%');
}
// 生成colgroup
var colgroupHtml = [];
this.eachCols(function (i, item) {
if (item.colGroup || item.hide) return;
colgroupHtml.push('<col');
if (item.width) colgroupHtml.push(' width="' + item.width + '"');
else if (item.minWidth) colgroupHtml.push(' width="' + (item.minWidth / mwPercent * 100).toFixed(2) + '%"');
else colgroupHtml.push(' width="' + (options.cellMinWidth / mwPercent * 100).toFixed(2) + '%"');
if (item.type) colgroupHtml.push(' data-type="' + item.type + '"');
if (item.key) colgroupHtml.push(' data-key="' + item.key + '"');
colgroupHtml.push('/>');
});
colgroupHtml = colgroupHtml.join('');
if (returnColgroup) return '<colgroup>' + colgroupHtml + '</colgroup>';
components.$table.children('colgroup').html(colgroupHtml);
};
/** 获取行对应数据 */
TreeTable.prototype.getDataByTr = function ($tr) {
var data, index;
if (typeof $tr !== 'string' && typeof $tr !== 'number') {
if ($tr) index = $tr.data('index');
} else index = $tr;
if (index === undefined) return;
if (typeof index === 'number') index = [index];
else index = index.split('-');
for (var i = 0; i < index.length; i++) {
if (data) data = data[this.options.tree.childName][index[i]];
else data = this.options.data[index[i]];
}
return data;
};
/**
* 联动子级复选框状态
* @param $tr 当前tr的dom
* @param checked
*/
TreeTable.prototype.checkSubCB = function ($tr, checked) {
var that = this;
var components = this.getComponents();
var indent = -1, $trList;
if ($tr.is('tbody')) {
$trList = $tr.children('tr');
} else {
indent = parseInt($tr.data('indent'));
$trList = $tr.nextAll('tr');
}
$trList.each(function () {
if (parseInt($(this).data('indent')) <= indent) return false;
var $cb = $(this).children('td').find('input[lay-filter="' + components.checkboxFilter + '"]');
$cb.prop('checked', checked);
$cb.removeClass('ew-form-indeterminate');
if (checked) $cb.next('.layui-form-checkbox').addClass('layui-form-checked');
else $cb.next('.layui-form-checkbox').removeClass('layui-form-checked');
var d = that.getDataByTr($(this));
d.LAY_CHECKED = checked; // 同步更新数据
d.LAY_INDETERMINATE = false;
});
};
/**
* 联动父级复选框状态
* @param $tr 父级的dom
*/
TreeTable.prototype.checkParentCB = function ($tr) {
var options = this.options;
var components = this.getComponents();
var d = this.getDataByTr($tr);
var ckNum = 0, unCkNum = 0;
if (d[options.tree.childName]) {
function checkNum(data) {
for (var i = 0; i < data.length; i++) {
if (data[i].LAY_CHECKED) ckNum++;
else unCkNum++;
if (data[i][options.tree.childName]) checkNum(data[i][options.tree.childName]);
}
}
checkNum(d[options.tree.childName]);
}
var $cb = $tr.children('td').find('input[lay-filter="' + components.checkboxFilter + '"]');
if (ckNum > 0 && unCkNum === 0) { // 全选
$cb.prop('checked', true);
$cb.removeClass('ew-form-indeterminate');
$cb.next('.layui-form-checkbox').addClass('layui-form-checked');
d.LAY_CHECKED = true; // 同步更新数据
d.LAY_INDETERMINATE = false;
} else if (ckNum === 0 && unCkNum > 0) { // 全不选
$cb.prop('checked', false);
$cb.removeClass('ew-form-indeterminate');
$cb.next('.layui-form-checkbox').removeClass('layui-form-checked');
d.LAY_CHECKED = false; // 同步更新数据
d.LAY_INDETERMINATE = false;
} else if (ckNum > 0 && unCkNum > 0) { // 半选
$cb.prop('checked', true);
$cb.data('indeterminate', 'true');
$cb.addClass('ew-form-indeterminate');
$cb.next('.layui-form-checkbox').addClass('layui-form-checked');
d.LAY_CHECKED = true; // 同步更新数据
d.LAY_INDETERMINATE = true;
}
};
/** 联动全选复选框 */
TreeTable.prototype.checkChooseAllCB = function () {
var options = this.options;
var components = this.getComponents();
var ckNum = 0, unCkNum = 0;
function checkNum(data) {
for (var i = 0; i < data.length; i++) {
if (data[i].LAY_CHECKED) ckNum++;
else unCkNum++;
if (data[i][options.tree.childName]) checkNum(data[i][options.tree.childName]);
}
}
checkNum(options.data);
var $cb = components.$view.find('input[lay-filter="' + components.chooseAllFilter + '"]');
if (ckNum > 0 && unCkNum === 0) { // 全选
$cb.prop('checked', true);
$cb.removeClass('ew-form-indeterminate');
$cb.next('.layui-form-checkbox').addClass('layui-form-checked');
} else if ((ckNum === 0 && unCkNum > 0) || (ckNum === 0 && unCkNum === 0)) { // 全不选
$cb.prop('checked', false);
$cb.removeClass('ew-form-indeterminate');
$cb.next('.layui-form-checkbox').removeClass('layui-form-checked');
} else if (ckNum > 0 && unCkNum > 0) { // 半选
$cb.prop('checked', true);
$cb.addClass('ew-form-indeterminate');
$cb.next('.layui-form-checkbox').addClass('layui-form-checked');
}
};
/** 填充序号列 */
TreeTable.prototype.renderNumberCol = function () {
this.getComponents().$tBody.children('tbody').children('tr').each(function (i) {
$(this).children('td').find('.ew-tree-table-numbers').text(i + 1);
});
};
/** 根据id获取tr的index */
TreeTable.prototype.getIndexById = function (id) {
var options = this.options;
function each(data, pi) {
for (var i = 0; i < data.length; i++) {
if (data[i][options.tree.idName] === id) return pi !== undefined ? pi + '-' + i : i;
if (data[i][options.tree.childName]) {
var res = each(data[i][options.tree.childName], pi !== undefined ? pi + '-' + i : i);
//值不为undefined才return
if (res) return res;
}
}
}
return each(options.data);
};
/** 展开指定行 */
TreeTable.prototype.expand = function (id, cascade) {
var components = this.getComponents();
var $tr = components.$table.children('tbody').children('tr[data-index="' + this.getIndexById(id) + '"]');
if (!$tr.hasClass('ew-tree-table-open')) $tr.children('td').find('.ew-tree-pack').trigger('click');
if (cascade === false) return;
// 联动父级
var indent = parseInt($tr.data('indent'));
$tr.prevAll('tr').each(function () {
var tInd = parseInt($(this).data('indent'));
if (tInd < indent) {
if (!$(this).hasClass('ew-tree-table-open')) {
$(this).children('td').find('.ew-tree-pack').trigger('click');
}
indent = tInd;
}
});
};
/** 折叠指定行 */
TreeTable.prototype.fold = function (id) {
var components = this.getComponents();
var $tr = components.$table.children('tbody').children('tr[data-index="' + this.getIndexById(id) + '"]');
if ($tr.hasClass('ew-tree-table-open')) $tr.children('td').find('.ew-tree-pack').trigger('click');
};
/** 全部展开 */
TreeTable.prototype.expandAll = function () {
this.getComponents().$table.children('tbody').children('tr').each(function () {
if (!$(this).hasClass('ew-tree-table-open')) $(this).children('td').find('.ew-tree-pack').trigger('click');
});
};
/** 全部折叠 */
TreeTable.prototype.foldAll = function () {
this.getComponents().$table.children('tbody').children('tr').each(function () {
if ($(this).hasClass('ew-tree-table-open')) $(this).children('td').find('.ew-tree-pack').trigger('click');
});
};
/** 获取当前数据 */
TreeTable.prototype.getData = function () {
return this.options.data;
};
/** 重载表格 */
TreeTable.prototype.reload = function (opt) {
this.initOptions(this.options ? $.extend(true, this.options, opt) : opt);
this.init(); // 初始化表格
this.bindEvents(); // 绑定事件
};
/** 获取当前选中行 */
TreeTable.prototype.checkStatus = function (needIndeterminate) {
if (needIndeterminate === undefined) needIndeterminate = true;
var list = [];
this.eachData(function (i, item) {
if ((needIndeterminate || !item.LAY_INDETERMINATE) && item.LAY_CHECKED)
list.push($.extend({isIndeterminate: item.LAY_INDETERMINATE}, item));
});
return list;
};
/** 设置复/单选框选中 */
TreeTable.prototype.setChecked = function (ids) {
var that = this;
var components = this.getComponents();
var $radio = components.$table.find('input[lay-filter="' + components.radioFilter + '"]');
if ($radio.length > 0) { // 开启了单选框
$radio.each(function () {
var d = that.getDataByTr($(this).parentsUntil('tr').parent());
if (d && ids[ids.length - 1] == d[that.options.tree.idName]) {
$(this).next('.layui-form-radio').trigger('click');
return false;
}
});
} else { // 开启了复选框
components.$table.find('input[lay-filter="' + components.checkboxFilter + '"]').each(function () {
var $cb = $(this);
var $layCb = $cb.next('.layui-form-checkbox');
var checked = $cb.prop('checked');
var indeterminate = $cb.hasClass('ew-form-indeterminate');
var d = that.getDataByTr($cb.parentsUntil('tr').parent());
for (var i = 0; i < ids.length; i++) {
if (d && ids[i] == d[that.options.tree.idName]) {
if (d[that.options.tree.childName] && d[that.options.tree.childName].length > 0) continue;
if (!checked || indeterminate) $layCb.trigger('click');
}
}
});
}
};
/** 移除全部选中 */
TreeTable.prototype.removeAllChecked = function () {
this.checkSubCB(this.getComponents().$table.children('tbody'), false);
};
/** 导出 */
TreeTable.prototype.exportData = function (type) {
var components = this.getComponents();
if ('show' === type) {
components.$toolbar.find('.layui-table-tool-panel').remove();
components.$toolbar.find('[lay-event="LAYTABLE_EXPORT"]').append([
'<ul class="layui-table-tool-panel">',
' <li data-type="csv">导出到 Csv 文件</li>',
' <li data-type="xls">导出到 Excel 文件</li>',
'</ul>'
].join(''));
} else {
if (device.ie) return layer.msg('不支持ie导出');
if (!type) type = 'xls';
var head = [], body = [];
this.eachCols(function (i, item) {
if (item.type !== 'normal' || item.hide) return;
head.push(item.title || '');
});
components.$tBody.children('tbody').children('tr').each(function () {
var items = [];
$(this).children('td').each(function () {
var $this = $(this);
if ($this.data('type') !== 'normal' || $this.hasClass('layui-hide')) return true;
items.push($this.text().trim().replace(/,/g, ''));
});
body.push(items.join(','));
});
// 创建下载文件的a标签
var alink = document.createElement('a');
var content = encodeURIComponent(head.join(',') + '\r\n' + body.join('\r\n'));
var contentType = ({csv: 'text/csv', xls: 'application/vnd.ms-excel'})[type];
alink.href = 'data:' + contentType + ';charset=utf-8,\ufeff' + content;
alink.download = (this.options.title || 'table') + '.' + type;
document.body.appendChild(alink);
alink.click();
document.body.removeChild(alink);
}
};
/** 打印 */
TreeTable.prototype.printTable = function () {
var components = this.getComponents();
var head = components.$tHead.children('thead').html();
if (!head) head = components.$tBody.children('thead').html();
var body = components.$tBody.children('tbody').html();
var colgroup = components.$tBody.children('colgroup').html();
var $html = $([
'<table class="ew-tree-table-print">',
' <colgroup>', colgroup, '</colgroup>',
' <thead>', head, '</thead>',
' <tbody>', body, '</tbody>',
'</table>'
].join(''));
// 隐藏特殊列
$html.find('col[data-type="checkbox"],col[data-type="radio"],col[data-type="tool"]').remove();
$html.find('td[data-type="checkbox"],td[data-type="radio"],td[data-type="tool"],.layui-hide').remove();
function hideCol($temp) {
var parentKey = $temp.data('parent'), pCol;
if (!parentKey) return;
var $parent = $html.children('thead').children('tr').children('[data-key="' + parentKey + '"]');
var colspan = parseInt($parent.attr('colspan')) - 1;
$parent.attr('colspan', colspan);
if (colspan === 0) $parent.remove();
hideCol($parent);
}
$html.find('th[data-type="checkbox"],th[data-type="radio"],th[data-type="tool"]').each(function () {
hideCol($(this));
}).remove();
// 打印内容样式
var style = [
'<style>',
' /* 打印表格样式 */',
' .ew-tree-table-print {',
' border: none;',
' border-collapse: collapse;',
' width: 100%;',
' table-layout: fixed;',
' }',
' .ew-tree-table-print td, .ew-tree-table-print th {',
' color: #555;',
' font-size: 14px;',
' padding: 9px 15px;',
' word-break: break-all;',
' border: 1px solid #888;',
' text-align: left;',
' }',
' .ew-tree-table-print .ew-tree-table-cell {',
' min-height: 20px;',
' }',
' /* 序号列样式 */',
' .ew-tree-table-print td[data-type="numbers"], .ew-tree-table-print th[data-type="numbers"] {',
' padding-left: 0;',
' padding-right: 0;',
' }',
' /* 单/复选框列样式 */',
' .ew-tree-table-print td[data-type="tool"], .ew-tree-table-print th[data-type="tool"], ',
' .ew-tree-table-print td[data-type="checkbox"], .ew-tree-table-print th[data-type="checkbox"], ',
' .ew-tree-table-print td[data-type="radio"], .ew-tree-table-print th[data-type="radio"] {',
' border: none;',
' }',
' .ew-tree-table-print td.layui-hide + td, .ew-tree-table-print th.layui-hide + th, ',
' .ew-tree-table-print td[data-type="tool"] + td, .ew-tree-table-print th[data-type="tool"] + th, ',
' .ew-tree-table-print td[data-type="checkbox"] + td, .ew-tree-table-print th[data-type="checkbox"] + th, ',
' .ew-tree-table-print td[data-type="radio"] + td, .ew-tree-table-print th[data-type="radio"] + th {',
' border-left: none;',
' }',
' /* 不显示的元素 */',
' .layui-hide, ',
' .ew-tree-table-print td[data-type="tool"] *, .ew-tree-table-print th[data-type="tool"] *, ',
' .ew-tree-table-print td[data-type="checkbox"] *, .ew-tree-table-print th[data-type="checkbox"] *, ',
' .ew-tree-table-print td[data-type="radio"] *, .ew-tree-table-print th[data-type="radio"] *, ',
' .layui-table-grid-down, .ew-tree-tips-c, .ew-tree-icon, .ew-tree-table-arrow.ew-tree-table-arrow-hide {',
' display: none;',
' }',
' /* tree缩进 */',
' .ew-tree-table-indent {',
' padding-left: 13px;',
' }',
' /* 箭头 */',
' .ew-tree-table-arrow {',
' position: relative;',
' padding-left: 13px;',
' }',
' .ew-tree-table-arrow:before {',
' content: "";',
' border: 5px solid transparent;',
' border-top-color: #666;',
' position: absolute;',
' left: 0;',
' top: 6px;',
' }',
'</style>'
].join('');
var pWindow = window.open('', '_blank');
pWindow.focus();
var pDocument = pWindow.document;
pDocument.open();
pDocument.write($html[0].outerHTML + style);
pDocument.close();
pWindow.print();
pWindow.close();
};
/** 筛选列 */
TreeTable.prototype.toggleCol = function (show, field, key) {
var components = this.getComponents();
if (show === undefined) {
components.$toolbar.find('.layui-table-tool-panel').remove();
var cols = ['<ul class="layui-table-tool-panel">'];
this.eachCols(function (i, item) {
if (item.type !== 'normal') return;
cols.push('<li><input type="checkbox" lay-skin="primary"');
cols.push(' lay-filter="' + components.colsToggleFilter + '"');
cols.push(' value="' + item.key + '" title="' + util.escape(item.title || '') + '"');
cols.push((item.hide ? '' : ' checked') + '></li>');
});
components.$toolbar.find('[lay-event="LAYTABLE_COLS"]').append(cols.join('') + '</ul>');
form.render('checkbox', components.filter);
} else {
if (key) {
var $td = components.$table.children('tbody').children('tr').children('[data-key="' + key + '"]');
var $th = components.$table.children('thead').children('tr').children('[data-key="' + key + '"]');
if (show) {
$td.removeClass('layui-hide');
$th.removeClass('layui-hide');
} else {
$td.addClass('layui-hide');
$th.addClass('layui-hide');
}
// 同步更新数据
var ks = key.split('-');
var col = this.options.cols[ks[0]][ks[1]];
col.hide = !show;
// 更新colspan数据
function changeParent($temp) {
var parentKey = $temp.data('parent'), pCol;
if (!parentKey) return;
var $parent = components.$table.children('thead').children('tr').children('[data-key="' + parentKey + '"]');
var colspan = $parent.attr('colspan');
show ? colspan++ : colspan--;
$parent.attr('colspan', colspan);
if (colspan === 0) $parent.addClass('layui-hide');
else $parent.removeClass('layui-hide');
changeParent($parent);
}
changeParent($th);
// 同步eachCols数据
this.eachCols(function (i, item) {
if (item.key === key) item.hide = col.hide;
});
this.resize(); // 更新表格尺寸
}
}
};
/**
* 搜索数据
* @param ids 关键字或数据id集合
*/
TreeTable.prototype.filterData = function (ids) {
var components = this.getComponents();
components.$loading.show();
if (this.options.data.length > 0) components.$loading.addClass('ew-loading-float');
var $trList = components.$table.children('tbody').children('tr');
var indexList = [];
if (typeof ids === 'string') { // 关键字
$trList.each(function () {
var index = $(this).data('index');
$(this).children('td').each(function () {
if ($(this).text().indexOf(ids) !== -1) {
indexList.push(index);
return false;
}
});
});
} else {
for (var i = 0; i < ids.length; i++) {
indexList.push(this.getIndexById(ids[i]));
}
}
$trList.addClass('ew-tree-table-filter-hide');
for (var j = 0; j < indexList.length; j++) {
var $tr = $trList.filter('[data-index="' + indexList[j] + '"]');
$tr.removeClass('ew-tree-table-filter-hide');
var indent = parseInt($tr.data('indent'));
// 联动子级
$tr.nextAll('tr').each(function () {
if (parseInt($(this).data('indent')) <= indent) return false;
$(this).removeClass('ew-tree-table-filter-hide');
});
if ($tr.hasClass('ew-tree-table-open')) toggleRow($tr);
// 联动父级
$tr.prevAll('tr').each(function () {
var tInd = parseInt($(this).data('indent'));
if (tInd < indent) {
$(this).removeClass('ew-tree-table-filter-hide');
if (!$(this).hasClass('ew-tree-table-open')) toggleRow($(this));
indent = tInd;
}
});
}
// 最后再检查一遍
/*$trList.not('.ew-tree-table-filter-hide').not('.ew-tree-tb-hide').each(function () {
var index = $(this).data('index'), hide = true;
for (var k = 0; k < indexList.length; k++) {
if (indexList[k] === index) hide = false;
}
if (hide) $(this).addClass('ew-tree-table-filter-hide');
});*/
components.$loading.hide();
components.$loading.removeClass('ew-loading-float');
if (indexList.length === 0) components.$empty.show();
updateFixedTbHead(components.$view); // 更新滚动条补丁
};
/** 重置搜索 */
TreeTable.prototype.clearFilter = function () {
var components = this.getComponents();
components.$table.children('tbody').children('tr').removeClass('ew-tree-table-filter-hide');
if (this.options.data.length > 0) components.$empty.hide();
updateFixedTbHead(components.$view); // 更新滚动条补丁
};
/**
* 刷新指定父级下的节点
* @param id 父级id,空则全部刷新
* @param data 非异步模式替换的数据
*/
TreeTable.prototype.refresh = function (id, data) {
if (isClass(id) === 'Array') {
data = id;
id = undefined;
}
var components = this.getComponents();
var d, $tr;
if (id !== undefined) {
$tr = components.$table.children('tbody').children('tr[data-index="' + this.getIndexById(id) + '"]');
d = this.getDataByTr($tr);
}
if (data) { // 数据模式
if (data.length > 0) components.$loading.addClass('ew-loading-float');
components.$loading.show();
if (data.length > 0 && this.options.tree.isPidData) { // pid形式数据
this.renderBodyData(tt.pidToChildren(data, this.options.tree.idName, this.options.tree.pidName, this.options.tree.childName), d, $tr);
} else {
this.renderBodyData(data, d, $tr);
}
} else { // 异步模式
this.renderBodyAsync(d, $tr);
}
};
/** 删除数据 */
TreeTable.prototype.del = function (id, index) {
if (index === undefined) index = this.getIndexById(id);
var indexList = (typeof index === 'number' ? [index] : index.split('-'));
var d = this.options.data;
if (indexList.length > 1) {
for (var i = 0; i < indexList.length - 1; i++) {
d = d[parseInt(indexList[i])][this.options.tree.childName];
}
}
d.splice(indexList[indexList.length - 1], 1);
};
/** 更新数据 */
TreeTable.prototype.update = function (id, fields) {
$.extend(true, this.getDataByTr(this.getIndexById(id)), fields);
};
/** 折叠/展开行 */
function toggleRow($tr) {
var indent = parseInt($tr.data('indent'));
var open = $tr.hasClass('ew-tree-table-open');
if (open) { // 折叠
$tr.removeClass('ew-tree-table-open');
$tr.nextAll('tr').each(function () {
if (parseInt($(this).data('indent')) <= indent) return false;
$(this).addClass('ew-tree-tb-hide');
});
} else { // 展开
$tr.addClass('ew-tree-table-open');
var hideInd;
$tr.nextAll('tr').each(function () {
var ind = parseInt($(this).data('indent'));
if (ind <= indent) return false;
if (hideInd !== undefined && ind > hideInd) return true;
$(this).removeClass('ew-tree-tb-hide');
if (!$(this).hasClass('ew-tree-table-open')) hideInd = parseInt($(this).data('indent'));
else hideInd = undefined;
});
}
updateFixedTbHead($tr.parentsUntil('.ew-tree-table').last().parent());
return open;
}
/** 固定表头滚动条补丁 */
function updateFixedTbHead($view) {
var $headBox = $view.children('.ew-tree-table-head');
var $tbBox = $view.children('.ew-tree-table-box');
var sWidth = $tbBox.width() - $tbBox.prop('clientWidth');
$headBox.css('border-right', (sWidth > 0 ? sWidth : 0) + 'px solid #f2f2f2');
}
// 监听窗口大小改变
$(window).resize(function () {
$('.ew-tree-table').each(function () {
updateFixedTbHead($(this));
var $tbBox = $(this).children('.ew-tree-table-box');
var full = $tbBox.data('full');
if (full && device.ie && device.ie < 10) {
$tbBox.css('height', getPageHeight() - full);
}
});
});
/** 表格溢出点击展开功能 */
$(document).on('mouseenter', '.ew-tree-table-cell.single-line', function () {
var $content = $(this).children('.ew-tree-table-cell-content');
if ($content.prop('scrollWidth') > $content.outerWidth()) $(this).children('.layui-table-grid-down').show();
}).on('mouseleave', '.ew-tree-table-cell.single-line', function () {
$(this).children('.layui-table-grid-down').hide();
});
// 点击箭头展开
$(document).on('click', '.ew-tree-table-cell>.layui-table-grid-down', function (e) {
e.stopPropagation();
hideAllTdTips();
var $cell = $(this).parent();
$cell.addClass('ew-tree-tips-open');
$cell.children('.layui-table-grid-down').hide();
var tw = $cell.parent().outerWidth() + 4;
if ($cell.outerWidth() < tw) $cell.children('.ew-tree-table-cell-content').css({'width': tw, 'max-width': tw});
var $box = $cell.parents().filter('.ew-tree-table-box');
if ($box.length === 0) $box = $cell.parents().filter('.ew-tree-table-head');
if ($box.length === 0) return;
if (($cell.outerWidth() + $cell.offset().left) + 20 > $box.offset().left + $box.outerWidth()) {
$cell.addClass('ew-show-left');
}
if (($cell.outerHeight() + $cell.offset().top + 10) > $box.offset().top + $box.outerHeight()) {
$cell.addClass('ew-show-bottom');
}
});
// 点击关闭按钮关闭
$(document).on('click', '.ew-tree-table-cell>.ew-tree-tips-c', function () {
hideAllTdTips();
});
// 点击空白部分关闭
$(document).on('click', function () {
hideAllTdTips();
$('.ew-tree-table .layui-table-tool-panel').remove();
});
$(document).on('click', '.ew-tree-table-cell.ew-tree-tips-open', function (e) {
e.stopPropagation();
});
/* 关闭所有单元格溢出提示框 */
function hideAllTdTips() {
$('.ew-tree-table-cell').removeClass('ew-tree-tips-open ew-show-left ew-show-bottom');
$('.ew-tree-table-cell>.ew-tree-table-cell-content').css({'width': '', 'max-width': ''});
}
/** 拖拽调整列宽 */
$(document).on('mousedown', '.ew-tb-resize', function (e) {
layui.stope(e);
var $this = $(this);
$this.attr('move', 'true');
var key = $this.parent().data('key');
$this.data('x', e.clientX);
var w = $this.parent().parent().parent().parent().children('colgroup').children('col[data-key="' + key + '"]').attr('width');
if (!w || w.toString().indexOf('%') !== -1) w = $this.parent().outerWidth();
$this.data('width', w);
$('body').addClass('ew-tree-table-resizing');
}).on('mousemove', function (e) {
var $rs = $('.ew-tree-table .ew-tb-resize[move="true"]');
if ($rs.length === 0) return;
layui.stope(e);
var x = $rs.data('x');
var w = $rs.data('width');
var nw = parseFloat(w) + e.clientX - parseFloat(x);
if (nw <= 0) nw = 1;
// 更新实例options中的宽度
var ins = _instances[$rs.parentsUntil('.ew-tree-table').last().parent().attr('lay-filter')];
var key = $rs.parent().data('key');
var ks = key.split('-');
ins.options.cols[ks[0]][ks[1]].width = nw;
ins.eachCols(function (i, item) {
if (item.key === key) item.width = nw;
});
ins.resize();
}).on('mouseup', function (e) {
$('.ew-tree-table .ew-tb-resize[move="true"]').attr('move', 'false');
$('body').removeClass('ew-tree-table-resizing');
}).on('mouseleave', function (e) {
$('.ew-tree-table .ew-tb-resize[move="true"]').attr('move', 'false');
$('body').removeClass('ew-tree-table-resizing');
});
/** 获取顶级的pId */
function getPids(data, idName, pidName) {
var pids = [];
for (var i = 0; i < data.length; i++) {
var hasPid = false;
for (var j = 0; j < data.length; j++) {
if (data[i][pidName] == data[j][idName]) {
hasPid = true;
break;
}
}
if (!hasPid) pids.push(data[i][pidName]);
}
return pids;
}
/** 判断pId是否相等 */
function pidEquals(pId, pIds) {
if (isClass(pIds) === 'Array') {
for (var i = 0; i < pIds.length; i++)
if (pId == pIds[i]) return true;
}
return pId == pIds;
}
/** 获取变量类型 */
function isClass(o) {
if (o === null) return 'Null';
if (o === undefined) return 'Undefined';
return Object.prototype.toString.call(o).slice(8, -1);
}
/** 获取浏览器高度 */
function getPageHeight() {
return document.documentElement.clientHeight || document.body.clientHeight;
}
/** 对外提供的方法 */
var tt = {
/* 渲染 */
render: function (options) {
return new TreeTable(options);
},
/* 重载 */
reload: function (id, opt) {
_instances[id].reload(opt);
},
/* 事件监听 */
on: function (events, callback) {
return layui.onevent.call(this, MOD_NAME, events, callback);
},
/* pid转children形式 */
pidToChildren: function (data, idName, pidName, childName, pId) {
if (!childName) childName = 'children';
var newList = [];
for (var i = 0; i < data.length; i++) {
if (data[i][idName] == data[i][pidName])
return console.error('第' + i + '条数据的' + idName + '与' + pidName + '相同', data[i]);
if (pId === undefined) pId = getPids(data, idName, pidName);
if (pidEquals(data[i][pidName], pId)) {
var children = this.pidToChildren(data, idName, pidName, childName, data[i][idName]);
if (children.length > 0) data[i][childName] = children;
newList.push(data[i]);
}
}
return newList;
}
};
/** 添加样式 */
$('head').append([
'<style id="ew-tree-table-css">',
'/** 最外层容器 */',
'.ew-tree-table {',
' margin: 10px 0;',
' position: relative;',
' border: 1px solid #e6e6e6;',
' border-bottom: none;',
' border-right: none;',
'}',
'.ew-tree-table:before, .ew-tree-table:after, .ew-tree-table .ew-tree-table-head:after {',
' content: "";',
' background-color: #e6e6e6;',
' position: absolute;',
' right: 0;',
' bottom: 0;',
'}',
'.ew-tree-table:before {',
' width: 1px;',
' top: 0;',
' z-index: 1;',
'}',
'.ew-tree-table:after, .ew-tree-table .ew-tree-table-head:after {',
' height: 1px;',
' left: 0;',
'}',
'.ew-tree-table .layui-table {',
' margin: 0;',
' position: relative;',
' table-layout: fixed;',
'}',
'/** 表格 */',
'.ew-tree-table .layui-table td, .ew-tree-table .layui-table th {',
' border-top: none;',
' border-left: none;',
' padding: 0 !important;',
'}',
'.ew-tree-table .ew-tree-table-box {',
' overflow: auto;',
' position: relative;',
'}',
'.ew-tree-table .ew-tree-table-head {',
' overflow: hidden;',
' box-sizing: border-box;',
' background-color: #f2f2f2;',
' position: relative;',
'}',
'/** loading */',
'.ew-tree-table div.ew-tree-table-loading {',
' padding: 10px 0;',
' text-align: center;',
'}',
'.ew-tree-table div.ew-tree-table-loading > i {',
' color: #999;',
' font-size: 30px;',
'}',
'.ew-tree-table div.ew-tree-table-loading.ew-loading-float {',
' position: absolute;',
' top: 0;',
' left: 0;',
' right: 0;',
'}',
'/** 空数据 */',
'.ew-tree-table .ew-tree-table-empty {',
' color: #666;',
' font-size: 14px;',
' padding: 18px 0;',
' text-align: center;',
' display: none;',
'}',
'/** 单元格 */',
'.ew-tree-table-cell.ew-tree-tips-open {',
' position: absolute;',
' top: 0;',
' left: 0;',
' padding: 0;',
' z-index: 9999;',
' background-color: #fff;',
' box-shadow: 3px 3px 8px rgba(0, 0, 0, .15);',
'}',
'thead .ew-tree-table-cell.ew-tree-tips-open {',
' background-color: #f2f2f2;',
'}',
'.ew-tree-table-cell.ew-tree-tips-open.ew-show-left {',
' right: 0;',
' left: auto;',
' box-shadow: -3px 3px 8px rgba(0, 0, 0, .15);',
'}',
'.ew-tree-table-cell.ew-tree-tips-open.ew-show-bottom {',
' bottom: 0;',
' top: auto;',
' box-shadow: 3px -3px 8px rgba(0, 0, 0, .15);',
'}',
'.ew-tree-table-cell.ew-tree-tips-open.ew-show-left.ew-show-bottom {',
' box-shadow: -3px -3px 8px rgba(0, 0, 0, .15);',
'}',
'.ew-tree-table-cell > .ew-tree-tips-c {',
' position: absolute;',
' right: -6px;',
' top: -3px;',
' width: 22px;',
' height: 22px;',
' line-height: 22px;',
' font-size: 16px;',
' color: #fff;',
' background-color: #666;',
' border-radius: 50%;',
' text-align: center;',
' cursor: pointer;',
' display: none;',
'}',
'table tr:first-child .ew-tree-table-cell > .ew-tree-tips-c {',
' top: 0;',
'}',
'.ew-tree-table-cell.ew-tree-tips-open > .ew-tree-tips-c {',
' display: block;',
'}',
'.ew-tree-table-cell.ew-tree-tips-open.ew-show-left > .ew-tree-tips-c {',
' left: -6px;',
' right: auto;',
'}',
'.ew-tree-table-cell > .ew-tree-table-cell-content {',
' padding: 5px 15px;',
' line-height: 28px;',
'}',
'[lay-size="lg"] .ew-tree-table-cell > .ew-tree-table-cell-content {',
' line-height: 40px;',
'}',
'[lay-size="sm"] .ew-tree-table-cell > .ew-tree-table-cell-content {',
' padding: 1px 15px;',
'}',
'.ew-tree-table-cell.single-line > .ew-tree-table-cell-content {',
' overflow: hidden;',
' white-space: nowrap;',
' text-overflow: ellipsis;',
'}',
'.ew-tree-table-cell.ew-tree-tips-open > .ew-tree-table-cell-content {',
' overflow: auto;',
' padding: 9px 15px;',
' height: auto;',
' min-height: 100%;',
' max-height: 110px;',
' line-height: inherit;',
' max-width: 260px;',
' width: 200px;',
' width: max-content;',
' width: -moz-max-content;',
' box-sizing: border-box;',
' white-space: normal;',
'}',
'.ew-tree-table-cell > .layui-table-grid-down {',
' box-sizing: border-box;',
'}',
'/** 图标列 */',
'.ew-tree-table .ew-tree-pack {',
' cursor: pointer;',
' line-height: 16px;',
'}',
'.ew-tree-table .ew-tree-pack > .layui-icon, .ew-tree-table .ew-tree-pack > .ew-tree-icon {',
' margin-right: 5px;',
'}',
'.ew-tree-table .ew-tree-pack > * {',
' vertical-align: middle;',
'}',
'/* 缩进 */',
'.ew-tree-table .ew-tree-table-indent {',
' margin-right: 5px;',
' padding-left: 16px;',
'}',
'/* 箭头 */',
'.ew-tree-table .ew-tree-table-arrow:before {',
' content: "\\e623";',
'}',
'.ew-tree-table .ew-tree-table-open .ew-tree-table-arrow:before {',
' content: "\\e625";',
'}',
'.ew-tree-table .ew-tree-table-arrow.arrow2 {',
' font-size: 12px;',
' font-weight: 600;',
' line-height: 16px;',
' height: 16px;',
' width: 16px;',
' display: inline-block;',
' text-align: center;',
' color: #888;',
'}',
'.ew-tree-table .ew-tree-table-arrow.arrow2:before {',
' content: "\\e602";',
'}',
'.ew-tree-table .ew-tree-table-open .ew-tree-table-arrow.arrow2:before {',
' content: "\\e61a";',
'}',
'.ew-tree-table-arrow.ew-tree-table-arrow-hide {',
' visibility: hidden;',
'}',
'/* 箭头变加载中状态 */',
'.ew-tree-table tr.ew-tree-table-loading > td .ew-tree-table-arrow:before {',
' content: "\\e63d" !important;',
'}',
'.ew-tree-table tr.ew-tree-table-loading > td .ew-tree-table-arrow {',
' margin-right: 0;',
'}',
'.ew-tree-table tr.ew-tree-table-loading > td .ew-tree-table-arrow + * {',
' margin-left: 5px;',
'}',
'.ew-tree-table tr.ew-tree-table-loading * {',
' pointer-events: none !important;',
'}',
'/** 折叠行 */',
'.ew-tree-table .ew-tree-tb-hide {',
' display: none;',
'}',
'/** 特殊列调整 */',
'.ew-tree-table td[data-type="numbers"] > .ew-tree-table-cell,',
'.ew-tree-table th[data-type="numbers"] > .ew-tree-table-cell,',
'.ew-tree-table td[data-type="checkbox"] > .ew-tree-table-cell,',
'.ew-tree-table th[data-type="checkbox"] > .ew-tree-table-cell,',
'.ew-tree-table td[data-type="radio"] > .ew-tree-table-cell,',
'.ew-tree-table th[data-type="radio"] > .ew-tree-table-cell,',
'.ew-tree-table td[data-type="space"] > .ew-tree-table-cell,',
'.ew-tree-table th[data-type="space"] > .ew-tree-table-cell {',
' padding-left: 0;',
' padding-right: 0;',
'}',
'/* 单元格内表单元素样式调整 */',
'.ew-tree-table .layui-form-switch',
'.ew-tree-table .layui-form-radio {',
' margin: 0;',
'}',
'/* checkbox列调整 */',
'.ew-tree-table-checkbox + .layui-form-checkbox {',
' padding: 0;',
'}',
'.ew-tree-table-checkbox + .layui-form-checkbox > .layui-icon {',
' font-weight: 600;',
' color: transparent;',
' transition: background-color .1s linear;',
' -webkit-transition: background-color .1s linear;',
'}',
'.ew-tree-table-checkbox + .layui-form-checkbox.layui-form-checked > .layui-icon {',
' color: #fff;',
'}',
'/* checkbox半选状态 */',
'.ew-form-indeterminate + .layui-form-checkbox .layui-icon:before {',
' content: "";',
' width: 10px;',
' height: 2px;',
' background-color: #f1f1f1;',
' position: absolute;',
' top: 50%;',
' left: 50%;',
' margin: -1px 0 0 -5px;',
'}',
'/* radio列调整 */',
'.ew-tree-table-radio + .layui-form-radio {',
' margin: 0;',
' padding: 0;',
' height: 20px;',
' line-height: 20px;',
'}',
'.ew-tree-table-radio + .layui-form-radio > i {',
' margin: 0;',
' height: 20px;',
' font-size: 20px;',
' line-height: 20px;',
'}',
'/** 单元格编辑 */',
'.ew-tree-table .layui-table td[data-edit] {',
' cursor: text;',
'}',
'.ew-tree-table .ew-tree-table-edit {',
' position: absolute;',
' left: 0;',
' top: 0;',
' width: 100%;',
' height: 100%;',
' border-radius: 0;',
' box-shadow: 1px 1px 20px rgba(0, 0, 0, .15);',
'}',
'.ew-tree-table .ew-tree-table-edit:focus {',
' border-color: #5FB878 !important;',
'}',
'.ew-tree-table .ew-tree-table-edit.layui-form-danger {',
' border-color: #FF5722 !important;',
'}',
'/** 搜索数据隐藏行 */',
'.ew-tree-table tr.ew-tree-table-filter-hide {',
' display: none !important;',
'}',
'/** 头部工具栏 */',
'.ew-tree-table .ew-tree-table-tool {',
' min-height: 50px;',
' line-height: 30px;',
' padding: 10px 15px;',
' box-sizing: border-box;',
' background-color: #f2f2f2;',
' border-bottom: 1px solid #e6e6e6;',
'}',
'.ew-tree-table .ew-tree-table-tool .ew-tree-table-tool-right {',
' float: right;',
'}',
'.ew-tree-table .ew-tree-table-tool .ew-tree-table-tool-item {',
' position: relative;',
' color: #333;',
' width: 26px;',
' height: 26px;',
' line-height: 26px;',
' text-align: center;',
' margin-left: 10px;',
' display: inline-block;',
' border: 1px solid #ccc;',
' box-sizing: border-box;',
' vertical-align: middle;',
' -webkit-transition: .3s all;',
' transition: .3s all;',
' cursor: pointer;',
'}',
'.ew-tree-table .ew-tree-table-tool .ew-tree-table-tool-item:first-child {',
' margin-left: 0;',
'}',
'.ew-tree-table .ew-tree-table-tool .ew-tree-table-tool-item:hover {',
' border-color: #999;',
'}',
'.ew-tree-table .ew-tree-table-tool-right .layui-table-tool-panel {',
' left: auto;',
' right: -1px;',
' z-index: 9999;',
'}',
'/* 列宽拖拽调整 */',
'.ew-tree-table .ew-tb-resize {',
' position: absolute;',
' right: 0;',
' top: 0;',
' bottom: 0;',
' width: 10px;',
' cursor: col-resize;',
'}',
'.ew-tree-table-resizing {',
' cursor: col-resize;',
' -ms-user-select: none;',
' -moz-user-select: none;',
' -webkit-user-select: none;',
' user-select: none;',
'}',
'/* 辅助样式 */',
'.ew-tree-table .layui-form-switch {',
' margin: 0;',
'}',
'.ew-tree-table .pd-tb-0 {',
' padding-top: 0 !important;',
' padding-bottom: 0 !important;',
'}',
'.ew-tree-table .break-all {',
' word-break: break-all !important;',
'}',
'/** 扩展图标 */',
'.ew-tree-table .ew-tree-icon-folder:after, .ew-tree-table .ew-tree-icon-file:after {',
' content: "";',
' padding: 2px 10px;',
' -webkit-background-size: cover;',
' -moz-background-size: cover;',
' -o-background-size: cover;',
' background-size: cover;',
' background-repeat: no-repeat;',
' background-image: url("");',
'}',
'.ew-tree-table tr.ew-tree-table-open > td .ew-tree-icon-folder:after {',
' background-image: url("");',
'}',
'.ew-tree-table .ew-tree-icon-file:after {',
' background-image: url("");',
'}',
'</style>'
].join(''));
exports('treeTable', tt);
});