bootstrap-treetable.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. /**
  2. * 查找当前这个节点的所有节点(包含子节点),并进行折叠或者展开操作
  3. *
  4. * @param item 被点击条目的子一级条目
  5. * @param target 整个bootstrap tree table实例
  6. * @param globalCollapsedFlag 如果为true,则表示当前操作是收缩(折叠),如果是false,表示当前操作是展开
  7. * @param options 存放了一些常量,例如展开和收缩的class
  8. */
  9. function extracted($, item, target, globalCollapsedFlag, options) {
  10. var itemCodeName = $(item).find("td[name='code']").text();
  11. var subItems = target.find("tbody").find(".tg-" + itemCodeName);//下一级,改为下所有级别
  12. if (subItems.size() > 0) {
  13. $.each(subItems, function (nIndex, nItem) {
  14. extracted($, nItem, target, globalCollapsedFlag, options);
  15. });
  16. }
  17. $.each(subItems, function (pIndex, pItem) {
  18. //如果是展开,判断当前箭头是开启还是关闭
  19. var expander = $(item).find("td[name='name']").find(".treetable-expander");
  20. if (!globalCollapsedFlag) {
  21. var hasExpander = expander.hasClass(options.expanderExpandedClass);
  22. if (hasExpander) {
  23. $(pItem).css("display", "table");
  24. } else {
  25. $(pItem).css("display", "none");
  26. }
  27. } else {
  28. //如果是折叠,就把当前开着的都折叠掉
  29. $(pItem).css("display", "none");
  30. expander.removeClass(options.expanderExpandedClass);
  31. expander.addClass(options.expanderCollapsedClass);
  32. }
  33. });
  34. }
  35. (function ($) {
  36. "use strict";
  37. $.fn.bootstrapTreeTable = function (options, param) {
  38. var allData = null;//用于存放格式化后的数据
  39. // 如果是调用方法
  40. if (typeof options == 'string') {
  41. return $.fn.bootstrapTreeTable.methods[options](this, param);
  42. }
  43. // 如果是初始化组件
  44. options = $.extend({}, $.fn.bootstrapTreeTable.defaults, options || {});
  45. // 是否有radio或checkbox
  46. var hasSelectItem = false;
  47. var target = $(this);
  48. // 在外层包装一下div,样式用的bootstrap-table的
  49. var _main_div = $("<div class='bootstrap-tree-table fixed-table-container'></div>");
  50. target.before(_main_div);
  51. _main_div.append(target);
  52. target.addClass("table table-hover treetable-table table-bordered");
  53. if (options.striped) {
  54. target.addClass('table-striped');
  55. }
  56. // 工具条在外层包装一下div,样式用的bootstrap-table的
  57. if (options.toolbar) {
  58. var _tool_div = $("<div class='fixed-table-toolbar'></div>");
  59. var _tool_left_div = $("<div class='bs-bars pull-left'></div>");
  60. _tool_left_div.append($(options.toolbar));
  61. _tool_div.append(_tool_left_div);
  62. _main_div.before(_tool_div);
  63. }
  64. // 格式化数据,优化性能
  65. target.formatData = function (data) {
  66. var _root = options.rootCodeValue ? options.rootCodeValue : null
  67. $.each(data, function (index, item) {
  68. console.log(item)
  69. // 添加一个默认属性,用来判断当前节点有没有被显示
  70. item.isShow = false;
  71. // 这里兼容几种常见Root节点写法
  72. // 默认的几种判断
  73. var _defaultRootFlag = item[options.parentCode] == '0'
  74. || item[options.parentCode] == 0
  75. || item[options.parentCode] == null
  76. || item[options.parentCode] == '';
  77. if (!item[options.parentCode] || (_root ? (item[options.parentCode] == options.rootCodeValue) : _defaultRootFlag)) {
  78. if (!allData["_root_"]) {
  79. allData["_root_"] = [];
  80. }
  81. allData["_root_"].push(item);
  82. } else {
  83. if (!allData["_n_" + item[options.parentCode]]) {
  84. allData["_n_" + item[options.parentCode]] = [];
  85. }
  86. allData["_n_" + item[options.parentCode]].push(item);
  87. }
  88. });
  89. }
  90. // 得到根节点
  91. target.getRootNodes = function () {
  92. return allData["_root_"];
  93. };
  94. // 递归获取子节点并且设置子节点
  95. target.handleNode = function (parentNode, lv, tbody) {
  96. var _ls = allData["_n_" + parentNode[options.code]];
  97. var tr = target.renderRow(parentNode, _ls ? true : false, lv);
  98. tbody.append(tr);
  99. if (_ls) {
  100. $.each(_ls, function (i, item) {
  101. target.handleNode(item, (lv + 1), tbody)
  102. });
  103. }
  104. };
  105. // 绘制行
  106. target.renderRow = function (item, isP, lv) {
  107. // 标记已显示
  108. item.isShow = true;
  109. var tr = $('<tr class="tg-' + item[options.parentCode] + '"></tr>');
  110. var _icon = options.expanderCollapsedClass;
  111. if (options.expandAll) {
  112. tr.css("display", "table");
  113. _icon = options.expanderExpandedClass;
  114. } else if (options.expandFirst && lv <= 2) {
  115. tr.css("display", "table");
  116. _icon = (lv == 1) ? options.expanderExpandedClass : options.expanderCollapsedClass;
  117. } else {
  118. tr.css("display", "none");
  119. _icon = options.expanderCollapsedClass;
  120. }
  121. $.each(options.columns, function (index, column) {
  122. // 判断有没有选择列
  123. if (index == 0 && column.field == 'selectItem') {
  124. hasSelectItem = true;
  125. var td = $('<td style="text-align:center;width:36px"></td>');
  126. if (column.radio) {
  127. var _ipt = $('<input name="select_item" type="radio" value="' + item[options.id] + '"></input>');
  128. td.append(_ipt);
  129. }
  130. if (column.checkbox) {
  131. var _ipt = $('<input name="select_item" type="checkbox" value="' + item[options.id] + '"></input>');
  132. td.append(_ipt);
  133. }
  134. tr.append(td);
  135. } else {
  136. var td = $('<td title="' + item[column.field] + '" name="' + column.field + '" style="' + ((column.width) ? ('width:' + column.width) : '') + '"></td>');
  137. // 增加formatter渲染
  138. if (column.formatter) {
  139. td.html(column.formatter.call(this, item[column.field], item, index));
  140. } else {
  141. td.text(item[column.field]);
  142. }
  143. if (options.expandColumn == index) {
  144. if (!isP) {
  145. td.prepend('<span class="treetable-expander"></span>')
  146. } else {
  147. td.prepend('<span class="treetable-expander ' + _icon + '"></span>')
  148. }
  149. for (var int = 0; int < (lv - 1); int++) {
  150. td.prepend('<span class="treetable-indent"></span>')
  151. }
  152. }
  153. tr.append(td);
  154. }
  155. });
  156. return tr;
  157. }
  158. // 加载数据
  159. target.load = function (parms) {
  160. // 加载数据前先清空
  161. allData = {};
  162. // 加载数据前先清空
  163. target.html("");
  164. // 构造表头
  165. var thr = $('<tr></tr>');
  166. $.each(options.columns, function (i, item) {
  167. var th = null;
  168. // 判断有没有选择列
  169. if (i == 0 && item.field == 'selectItem') {
  170. hasSelectItem = true;
  171. th = $('<th style="width:36px"></th>');
  172. } else {
  173. th = $('<th style="' + ((item.width) ? ('width:' + item.width) : '') + '"></th>');
  174. }
  175. th.text(item.title);
  176. thr.append(th);
  177. });
  178. var thead = $('<thead class="treetable-thead"></thead>');
  179. thead.append(thr);
  180. target.append(thead);
  181. // 构造表体
  182. var tbody = $('<tbody class="treetable-tbody"></tbody>');
  183. target.append(tbody);
  184. // 添加加载loading
  185. var _loading = '<tr><td colspan="' + options.columns.length + '"><div style="display: block;text-align: center;">正在努力地加载数据中,请稍候……</div></td></tr>'
  186. tbody.html(_loading);
  187. // 默认高度
  188. if (options.height) {
  189. tbody.css("height", options.height);
  190. }
  191. $.ajax({
  192. type: options.type,
  193. url: options.url,
  194. data: parms ? parms : options.ajaxParams,
  195. dataType: "JSON",
  196. success: function (data, textStatus, jqXHR) {
  197. // 加载完数据先清空
  198. tbody.html("");
  199. if (!data || data.length <= 0) {
  200. var _empty = '<tr><td colspan="' + options.columns.length + '"><div style="display: block;text-align: center;">没有找到匹配的记录</div></td></tr>'
  201. tbody.html(_empty);
  202. return;
  203. }
  204. // 格式化数据
  205. target.formatData(data);
  206. // 开始绘制
  207. var rootNode = target.getRootNodes();
  208. if (rootNode) {
  209. $.each(rootNode, function (i, item) {
  210. target.handleNode(item, 1, tbody);
  211. });
  212. }
  213. // 下边的操作主要是为了查询时让一些没有根节点的节点显示
  214. $.each(data, function (i, item) {
  215. if (!item.isShow) {
  216. var tr = target.renderRow(item, false, 1);
  217. tbody.append(tr);
  218. }
  219. });
  220. target.append(tbody);
  221. //动态设置表头宽度
  222. thead.css("width", tbody.children(":first").css("width"));
  223. // 行点击选中事件
  224. target.find("tbody").find("tr").click(function () {
  225. if (hasSelectItem) {
  226. var _ipt = $(this).find("input[name='select_item']");
  227. if (_ipt.attr("type") == "radio") {
  228. _ipt.prop('checked', true);
  229. target.find("tbody").find("tr").removeClass("treetable-selected");
  230. $(this).addClass("treetable-selected");
  231. } else {
  232. if (_ipt.prop('checked')) {
  233. _ipt.prop('checked', false);
  234. $(this).removeClass("treetable-selected");
  235. } else {
  236. _ipt.prop('checked', true);
  237. $(this).addClass("treetable-selected");
  238. }
  239. }
  240. }
  241. });
  242. // 小图标点击事件--展开缩起
  243. target.find("tbody").find("tr").find(".treetable-expander").click(function () {
  244. var tr = $(this).parent().parent();
  245. var _code = tr.find("input[name='select_item']").val();
  246. if (options.id == options.code) {
  247. _code = tr.find("input[name='select_item']").val();
  248. } else {
  249. _code = tr.find("td[name='" + options.code + "']").text();
  250. }
  251. var _ls = target.find("tbody").find(".tg-" + _code);//下一级,改为下所有级别
  252. if (_ls && _ls.length > 0) {
  253. var _flag = $(this).hasClass(options.expanderExpandedClass);
  254. $.each(_ls, function (index, item) {
  255. //查找当前这个节点的所有节点(包含子节点),如果是折叠都显示为不显示,如果是展开,则根据当前节点的状态
  256. extracted($, item, target, _flag, options);
  257. $(item).css("display", _flag ? "none" : "table");
  258. });
  259. if (_flag) {
  260. $(this).removeClass(options.expanderExpandedClass)
  261. $(this).addClass(options.expanderCollapsedClass)
  262. } else {
  263. $(this).removeClass(options.expanderCollapsedClass)
  264. $(this).addClass(options.expanderExpandedClass)
  265. }
  266. }
  267. });
  268. },
  269. error: function (xhr, textStatus) {
  270. var _errorMsg = '<tr><td colspan="' + options.columns.length + '"><div style="display: block;text-align: center;">' + xhr.responseText + '</div></td></tr>'
  271. tbody.html(_errorMsg);
  272. },
  273. });
  274. }
  275. if (options.url) {
  276. target.load();
  277. } else {
  278. // 也可以通过defaults里面的data属性通过传递一个数据集合进来对组件进行初始化....有兴趣可以自己实现,思路和上述类似
  279. }
  280. return target;
  281. };
  282. // 组件方法封装........
  283. $.fn.bootstrapTreeTable.methods = {
  284. // 返回选中记录的id(返回的id由配置中的id属性指定)
  285. // 为了兼容bootstrap-table的写法,统一返回数组,这里只返回了指定的id
  286. getSelections: function (target, data) {
  287. // 所有被选中的记录input
  288. var _ipt = target.find("tbody").find("tr").find("input[name='select_item']:checked");
  289. var chk_value = [];
  290. // 如果是radio
  291. if (_ipt.attr("type") == "radio") {
  292. var _data = {id: _ipt.val()};
  293. var _tds = _ipt.parent().parent().find("td");
  294. _tds.each(function (_i, _item) {
  295. if (_i != 0) {
  296. _data[$(_item).attr("name")] = $(_item).text();
  297. }
  298. });
  299. chk_value.push(_data);
  300. } else {
  301. _ipt.each(function (_i, _item) {
  302. var _data = {id: $(_item).val()};
  303. var _tds = $(_item).parent().parent().find("td");
  304. _tds.each(function (_ii, _iitem) {
  305. if (_ii != 0) {
  306. _data[$(_iitem).attr("name")] = $(_iitem).text();
  307. }
  308. });
  309. chk_value.push(_data);
  310. });
  311. }
  312. return chk_value;
  313. },
  314. // 刷新记录
  315. refresh: function (target, parms) {
  316. if (parms) {
  317. target.load(parms);
  318. } else {
  319. target.load();
  320. }
  321. },
  322. // 组件的其他方法也可以进行类似封装........
  323. };
  324. $.fn.bootstrapTreeTable.defaults = {
  325. id: 'id',// 选取记录返回的值
  326. code: 'id',// 用于设置父子关系
  327. parentCode: 'parentId',// 用于设置父子关系
  328. rootCodeValue: null,//设置根节点code值----可指定根节点,默认为null,"",0,"0"
  329. data: [], // 构造table的数据集合
  330. type: "GET", // 请求数据的ajax类型
  331. url: null, // 请求数据的ajax的url
  332. ajaxParams: {}, // 请求数据的ajax的data属性
  333. expandColumn: null,// 在哪一列上面显示展开按钮
  334. expandAll: true, // 是否全部展开
  335. expandFirst: false, // 是否默认第一级展开--expandAll为false时生效
  336. striped: false, // 是否各行渐变色
  337. columns: [],
  338. toolbar: null,//顶部工具条
  339. height: 0,
  340. expanderExpandedClass: 'glyphicon glyphicon-chevron-down',// 展开的按钮的图标
  341. expanderCollapsedClass: 'glyphicon glyphicon-chevron-right'// 缩起的按钮的图标
  342. };
  343. })(jQuery);