jquery.treegrid.extension.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. (function($) {
  2. "use strict";
  3. $.fn.bootstrapTreeTable = function(options, param) {
  4. // 如果是调用方法
  5. if (typeof options == 'string') {
  6. return $.fn.bootstrapTreeTable.methods[options](this, param);
  7. }
  8. // 如果是初始化组件
  9. options = $.extend({}, $.fn.bootstrapTreeTable.defaults, options || {});
  10. // 是否有radio或checkbox
  11. var hasSelectItem = false;
  12. var target = $(this);
  13. // 在外层包装一下div,样式用的bootstrap-table的
  14. var _main_div = $("<div class='fixed-table-container'></div>");
  15. target.before(_main_div);
  16. _main_div.append(target);
  17. target.addClass("table table-hover treegrid-table table-bordered");
  18. if (options.striped) {
  19. target.addClass('table-striped');
  20. }
  21. // 工具条在外层包装一下div,样式用的bootstrap-table的
  22. if(options.toolbar){
  23. var _tool_div = $("<div class='fixed-table-toolbar'></div>");
  24. var _tool_left_div = $("<div class='bs-bars pull-left'></div>");
  25. _tool_left_div.append($(options.toolbar));
  26. _tool_div.append(_tool_left_div);
  27. _main_div.before(_tool_div);
  28. }
  29. // 得到根节点
  30. target.getRootNodes = function(data) {
  31. // 指定Root节点值
  32. var _root = options.rootCodeValue?options.rootCodeValue:null
  33. var result = [];
  34. $.each(data, function(index, item) {
  35. // 这里兼容几种常见Root节点写法
  36. // 默认的几种判断
  37. var _defaultRootFlag = item[options.parentCode] == '0'
  38. || item[options.parentCode] == 0
  39. || item[options.parentCode] == null
  40. || item[options.parentCode] == '';
  41. if (!item[options.parentCode] || (_root?(item[options.parentCode] == options.rootCodeValue):_defaultRootFlag)){
  42. result.push(item);
  43. }
  44. // 添加一个默认属性,用来判断当前节点有没有被显示
  45. item.isShow = false;
  46. });
  47. return result;
  48. };
  49. var j = 0;
  50. // 递归获取子节点并且设置子节点
  51. target.getChildNodes = function(data, parentNode, parentIndex, tbody) {
  52. $.each(data, function(i, item) {
  53. if (item[options.parentCode] == parentNode[options.code]) {
  54. var tr = $('<tr></tr>');
  55. var nowParentIndex = (parentIndex + (j++) + 1);
  56. tr.addClass('treegrid-' + nowParentIndex);
  57. tr.addClass('treegrid-parent-' + parentIndex);
  58. target.renderRow(tr,item);
  59. item.isShow = true;
  60. tbody.append(tr);
  61. target.getChildNodes(data, item, nowParentIndex, tbody)
  62. }
  63. });
  64. };
  65. // 绘制行
  66. target.renderRow = function(tr,item){
  67. $.each(options.columns, function(index, column) {
  68. // 判断有没有选择列
  69. if(index==0&&column.field=='selectItem'){
  70. hasSelectItem = true;
  71. var td = $('<td style="text-align:center;width:36px"></td>');
  72. if(column.radio){
  73. var _ipt = $('<input name="select_item" type="radio" value="'+item[options.id]+'"></input>');
  74. td.append(_ipt);
  75. }
  76. if(column.checkbox){
  77. var _ipt = $('<input name="select_item" type="checkbox" value="'+item[options.id]+'"></input>');
  78. td.append(_ipt);
  79. }
  80. tr.append(td);
  81. }else{
  82. var td = $('<td style="'+((column.width)?('width:'+column.width):'')+'"></td>');
  83. // 列样式
  84. if(column.class){
  85. td.addClass(column.class);
  86. }
  87. // 增加formatter渲染
  88. if (column.formatter) {
  89. td.html(column.formatter.call(this, '', item, index));
  90. } else {
  91. td.text(item[column.field]);
  92. }
  93. tr.append(td);
  94. }
  95. });
  96. }
  97. // 加载数据
  98. target.load = function(parms){
  99. // 加载数据前先清空
  100. target.html("");
  101. // 构造表头
  102. var thr = $('<tr></tr>');
  103. $.each(options.columns, function(i, item) {
  104. var th = null;
  105. // 判断有没有选择列
  106. if(i==0&&item.field=='selectItem'){
  107. hasSelectItem = true;
  108. th = $('<th style="width:36px"></th>');
  109. }else{
  110. th = $('<th style="padding:10px;'+((item.width)?('width:'+item.width):'')+'"></th>');
  111. }
  112. th.text(item.title);
  113. // 列样式
  114. if(item.class){
  115. th.addClass(item.class);
  116. }
  117. thr.append(th);
  118. });
  119. var thead = $('<thead class="treegrid-thead"></thead>');
  120. thead.append(thr);
  121. target.append(thead);
  122. // 构造表体
  123. var tbody = $('<tbody class="treegrid-tbody"></tbody>');
  124. target.append(tbody);
  125. // 添加加载loading
  126. var _loading = '<tr><td colspan="'+options.columns.length+'"><div style="display: block;text-align: center;">正在努力地加载数据中,请稍候……</div></td></tr>'
  127. tbody.html(_loading);
  128. // 默认高度
  129. if(options.height){
  130. tbody.css("height",options.height);
  131. }
  132. $.ajax({
  133. type : options.type,
  134. url : options.url,
  135. data : parms?parms:options.ajaxParams,
  136. dataType : "JSON",
  137. success : function(data, textStatus, jqXHR) {
  138. // 加载完数据先清空
  139. tbody.html("");
  140. if(!data||data.length<=0){
  141. var _empty = '<tr><td colspan="'+options.columns.length+'"><div style="display: block;text-align: center;">没有找到匹配的记录</div></td></tr>'
  142. tbody.html(_empty);
  143. return;
  144. }
  145. var rootNode = target.getRootNodes(data);
  146. $.each(rootNode, function(i, item) {
  147. var tr = $('<tr></tr>');
  148. tr.addClass('treegrid-' + (j + "_" + i));
  149. target.renderRow(tr,item);
  150. item.isShow = true;
  151. tbody.append(tr);
  152. target.getChildNodes(data, item, (j + "_" + i), tbody);
  153. });
  154. // 下边的操作主要是为了查询时让一些没有根节点的节点显示
  155. $.each(data, function(i, item) {
  156. if(!item.isShow){
  157. var tr = $('<tr></tr>');
  158. tr.addClass('treegrid-' + (j + "_" + i));
  159. target.renderRow(tr,item);
  160. tbody.append(tr);
  161. }
  162. });
  163. target.append(tbody);
  164. // 初始化treegrid
  165. target.treegrid({
  166. treeColumn: options.expandColumn?options.expandColumn:(hasSelectItem?1:0),//如果有radio或checkbox默认第二列层级显示,当前是在用户未设置的提前下
  167. expanderExpandedClass : options.expanderExpandedClass,
  168. expanderCollapsedClass : options.expanderCollapsedClass
  169. });
  170. if (!options.expandAll) {
  171. target.treegrid('collapseAll');
  172. }
  173. //动态设置表头宽度
  174. thead.css("width", tbody.children(":first").css("width"));
  175. // 行点击选中事件
  176. target.find("tbody").find("tr").click(function(){
  177. if(hasSelectItem){
  178. var _ipt = $(this).find("input[name='select_item']");
  179. if(_ipt.attr("type")=="radio"){
  180. _ipt.prop('checked',true);
  181. target.find("tbody").find("tr").removeClass("treegrid-selected");
  182. $(this).addClass("treegrid-selected");
  183. }else{
  184. if(_ipt.prop('checked')){
  185. _ipt.prop('checked',false);
  186. $(this).removeClass("treegrid-selected");
  187. }else{
  188. _ipt.prop('checked',true);
  189. $(this).addClass("treegrid-selected");
  190. }
  191. }
  192. }
  193. });
  194. },
  195. error:function(xhr,textStatus){
  196. var _errorMsg = '<tr><td colspan="'+options.columns.length+'"><div style="display: block;text-align: center;">'+xhr.responseText+'</div></td></tr>'
  197. tbody.html(_errorMsg);
  198. debugger;
  199. },
  200. });
  201. }
  202. if (options.url) {
  203. target.load();
  204. } else {
  205. // 也可以通过defaults里面的data属性通过传递一个数据集合进来对组件进行初始化....有兴趣可以自己实现,思路和上述类似
  206. }
  207. return target;
  208. };
  209. // 组件方法封装........
  210. $.fn.bootstrapTreeTable.methods = {
  211. // 返回选中记录的id(返回的id由配置中的id属性指定)
  212. // 为了兼容bootstrap-table的写法,统一返回数组,这里只返回了指定的id
  213. getSelections : function(target, data) {
  214. // 所有被选中的记录input
  215. var _ipt = target.find("tbody").find("tr").find("input[name='select_item']:checked");
  216. var chk_value =[];
  217. // 如果是radio
  218. if(_ipt.attr("type")=="radio"){
  219. chk_value.push({id:_ipt.val()});
  220. }else{
  221. _ipt.each(function(_i,_item){
  222. chk_value.push({id:$(_item).val()});
  223. });
  224. }
  225. return chk_value;
  226. },
  227. // 刷新记录
  228. refresh : function(target, parms) {
  229. if(parms){
  230. target.load(parms);
  231. }else{
  232. target.load();
  233. }
  234. },
  235. // 组件的其他方法也可以进行类似封装........
  236. };
  237. $.fn.bootstrapTreeTable.defaults = {
  238. id : 'id',// 选取记录返回的值
  239. code : 'code',// 用于设置父子关系
  240. parentCode : 'parentId',// 用于设置父子关系
  241. rootCodeValue: null,//设置根节点code值----可指定根节点,默认为null,"",0,"0"
  242. data : [], // 构造table的数据集合
  243. type : "GET", // 请求数据的ajax类型
  244. url : null, // 请求数据的ajax的url
  245. ajaxParams : {}, // 请求数据的ajax的data属性
  246. expandColumn : null,// 在哪一列上面显示展开按钮
  247. expandAll : true, // 是否全部展开
  248. striped : false, // 是否各行渐变色
  249. columns : [],
  250. toolbar: null,//顶部工具条
  251. height: 0,
  252. expanderExpandedClass : 'glyphicon glyphicon-chevron-down',// 展开的按钮的图标
  253. expanderCollapsedClass : 'glyphicon glyphicon-chevron-right'// 缩起的按钮的图标
  254. };
  255. })(jQuery);