header.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. window.checkData = []; //全选 数据池
  2. window.sort=require('../utilities/sort');//排序组件
  3. window.Header = function getHeader(object) {
  4. let _targetDom = object.el ? document.getElementById(object.el) : document.getElementsByTagName("table")[0];//基点元素
  5. let _columns = object.column; //列名
  6. let _fixedTop = object.fixedTop || 0; //同级浮动元素高度,使当前元素追加该元素高度浮动
  7. let _isCheckAllBox = object.isCheckAllBox || true;//是否开启全选框
  8. let _data = object.data || []; //被排序数据
  9. let _restorationColumn = object.restorationColumn || 'id'; //恢复原数据基准字段
  10. let _is_restorationColumn_asc = object.is_restorationColumn_asc || false; //恢复原数据基准字段的排序类型 true:asc false:desc
  11. let _before = object.before;//前置元素
  12. let _name = object.name ? object.name+"." : "";//唯一名称 用于区别模块 本地存储
  13. let _isForbidDrag = object.isForbidDrag;
  14. let sortType = {}; //排序类型
  15. let columnArr = []; //列数组
  16. let moveTd = {}; //移动列
  17. let isNested = _targetDom.getElementsByTagName("table").length>0;
  18. function getTargetChildNode(dom) { //递归获取input子节点
  19. if (!dom || dom.tagName==='INPUT')return dom;
  20. return getTargetChildNode(dom.firstElementChild);
  21. }
  22. function createHeaderBefore() {//生成前置元素
  23. let tr = document.createElement("tr");
  24. tr.className = "text-center";
  25. _before.forEach(be=>{
  26. let th = document.createElement("th");
  27. if (be.colspan)th.colSpan = be.colspan;
  28. if (be.font){
  29. let font = document.createElement("span");
  30. font.className = be.font;
  31. th.appendChild(font);
  32. }
  33. if (be.value)th.appendChild(document.createTextNode(be.value));
  34. if (be.class)th.className = be.class;
  35. tr.appendChild(th);
  36. });
  37. _targetDom.insertBefore(tr, _targetDom.firstChild);
  38. }
  39. function appendFloat(div,cla="") {
  40. div.style.overflowX = "hidden";
  41. div.className="m-0 p-0";
  42. let th = document.createElement("th");
  43. th.className = "pl-2 pr-2 header-float "+cla;
  44. th.style.top = _fixedTop+"px";
  45. th.appendChild(div);
  46. return th;
  47. }
  48. function createHeader() {//生成表头列
  49. let tr = document.createElement("tr");
  50. tr.className = "text-nowrap";
  51. let firstTr = _targetDom.getElementsByTagName("tr")[0];
  52. let tds = [];
  53. if (firstTr)tds = firstTr.children;
  54. if (_isCheckAllBox){//是否开启全选
  55. let div = document.createElement("div");
  56. let check = document.createElement("input");
  57. check.type = "checkbox";
  58. check.id = "checkAll";
  59. if (firstTr)bindCheckbox(check);
  60. div.appendChild(check);
  61. tr.appendChild(appendFloat(div,"text-left"));
  62. }
  63. let i = (_isCheckAllBox && firstTr) ? 1 : 0;
  64. for (i;i<(tds.length>0 ? tds.length : _columns.length);i++){
  65. let div = document.createElement("div");
  66. let th = appendFloat(div);
  67. let column = _columns[(_isCheckAllBox && firstTr) ? i-1 : i];
  68. let wid = localStorage.getItem(_name+column.name);
  69. if (wid){
  70. div.style.width = wid+"px";
  71. th.style.minWidth = wid+"px";
  72. let trs = _targetDom.children;
  73. while(trs[0] && trs[0].tagName !== 'TR')trs = trs[0].children;
  74. if (trs[0]){
  75. for (let j=0;j<trs.length;j++){
  76. trs[j].children[i].firstChild.style.width = wid+"px";
  77. trs[j].children[i].style.minWidth = wid+"px";
  78. }
  79. }
  80. }
  81. if (column){
  82. if (column.type === 'multi') multiColumn(div,column);
  83. else defaultColumn(div,column);
  84. }
  85. tr.appendChild(th);
  86. }
  87. bindMove(tr);
  88. if (_targetDom.firstChild)_targetDom.insertBefore(tr, _targetDom.firstChild);
  89. else _targetDom.appendChild(tr);
  90. }
  91. function multiColumn(th,column) { //多列样式的生成
  92. if (column.title){
  93. let div = document.createElement("div");
  94. div.className="w-100 text-center";
  95. div.appendChild(document.createTextNode(column.title));
  96. th.appendChild(div);
  97. }
  98. let div = document.createElement("div");
  99. div.className="row text-center";
  100. if (column.rows){
  101. column.rows.forEach(row=>{
  102. let node = document.createElement("div");
  103. node.className = "col-"+(row.col ? row.col : 1);
  104. node.appendChild(document.createTextNode(row.value));
  105. div.appendChild(node);
  106. });
  107. th.appendChild(div);
  108. }
  109. }
  110. function defaultColumn(th,column) { //默认列样式的生成
  111. if (column.style)for (let key in column.style)if (column.style.hasOwnProperty(key)) th.parentElement.style[key] = column.style[key];
  112. if (column.class)th.parentElement.className += column.class;
  113. let span = document.createElement("span");
  114. span.style.display = "inline-block";
  115. if (!column.neglect){
  116. span.style.cursor = "pointer";
  117. let font = document.createElement("span");
  118. font.className = "fa fa-sort";
  119. span.appendChild(font);
  120. span.onclick = rule(column,font); //绑定排序事件
  121. }
  122. span.appendChild(document.createTextNode((column.value ? column.value : '')));
  123. th.appendChild(span);
  124. }
  125. //点击事件触发的排序规则
  126. function rule(column,columnSort) {
  127. return function () {
  128. if (!sortType[column.name]){
  129. sortType[column.name] = 'asc';
  130. columnArr.push(column.name);
  131. columnSort.className = "fa fa-sort-asc";
  132. let columnArrTemp = [];
  133. columnArrTemp.push.apply(columnArrTemp,columnArr);
  134. window.sort.sort(_data,columnArrTemp,sortType);
  135. return ;
  136. }
  137. if (sortType[column.name] === 'asc'){
  138. sortType[column.name] = 'desc';
  139. columnSort.className = "fa fa-sort-desc";
  140. let columnArrTemp = [];
  141. columnArrTemp.push.apply(columnArrTemp,columnArr);
  142. window.sort.sort(_data,columnArrTemp,sortType);
  143. return ;
  144. }
  145. if (sortType[column.name] === 'desc'){
  146. delete sortType[column.name];
  147. columnArr.some(function (name, index) {
  148. if (name === column.name){
  149. columnArr.splice(index,1);
  150. return true;
  151. }
  152. });
  153. columnSort.className = "fa fa-sort";
  154. if (columnArr.length === 0){
  155. //希尔排序
  156. let arr = [];
  157. arr.push.apply(arr,_data);
  158. let len = arr.length;
  159. for(let gap = Math.floor(len / 2); gap > 0; gap = Math.floor(gap / 2)) {
  160. for(let i = gap; i < len; i++) {
  161. let j = i;
  162. let current = arr[i];
  163. if (_is_restorationColumn_asc){
  164. while(j - gap >= 0 && Number(current[_restorationColumn]) < Number(arr[j - gap][_restorationColumn])) {
  165. arr[j] = arr[j - gap];
  166. j = j - gap;
  167. }
  168. }else{
  169. while(j - gap >= 0 && Number(current[_restorationColumn]) > Number(arr[j - gap][_restorationColumn])) {
  170. arr[j] = arr[j - gap];
  171. j = j - gap;
  172. }
  173. }
  174. arr[j] = current;
  175. }
  176. }
  177. _data.length = 0;
  178. _data.push.apply(_data,arr);
  179. return;
  180. }
  181. let columnArrTemp = [];
  182. columnArrTemp.push.apply(columnArrTemp,columnArr);
  183. window.sort.sort(_data,columnArrTemp,sortType);
  184. }
  185. }
  186. }
  187. function bindCheckbox(check) {
  188. let trs = _targetDom.children;
  189. while(trs[0].tagName !== 'TR')trs = trs[0].children;
  190. check.onchange = function () {
  191. if (event.target.checked){
  192. for (let i=0;i<trs.length;i++){
  193. let checkbox = getTargetChildNode(trs[i].children[0]);
  194. if (checkbox && !checkbox.checked){
  195. checkData.push(checkbox.value);
  196. checkbox.checked = true;
  197. }
  198. }
  199. }else{
  200. checkData = [];
  201. for (let i=0;i<trs.length;i++){
  202. let checkbox = getTargetChildNode(trs[i].children[0]);
  203. if (checkbox && checkbox.checked)checkbox.checked = false;
  204. }
  205. }
  206. };
  207. for (let i=0;i<trs.length;i++){
  208. let checkbox = getTargetChildNode(trs[i].children[0]);
  209. if (checkbox) checkbox.onchange = function () {
  210. if (event.target.checked)checkData.push(checkbox.value);
  211. else checkData.splice(checkData.indexOf(checkbox.value),1);
  212. if (checkData.length === _data.length && !check.checked)check.checked=true;
  213. if (checkData.length !== _data.length && check.checked) check.checked=false;
  214. }
  215. }
  216. }
  217. function bindMove(tr) {
  218. if (_isForbidDrag)return;
  219. for (let j=0;j<tr.children.length;j++){ //为首列绑定拖拽列宽事件
  220. tr.children[j].onmousedown = function (){
  221. if (j===0 && event.offsetX<10)return;
  222. if(event.offsetX<10)moveTd = {oldX:event.clientX,dom:tr.children[j-1],index:j-1};
  223. if (event.offsetX > tr.children[j].offsetWidth-10)moveTd = {oldX:event.clientX,dom:tr.children[j],index:j};
  224. };
  225. tr.children[j].onmousemove = function () {
  226. if (j===0 && event.offsetX<10)return;
  227. if(event.offsetX<10 || event.offsetX > tr.children[j].offsetWidth-10)
  228. this.style.cursor = 'w-resize';
  229. else this.style.cursor = 'default';
  230. }
  231. }
  232. document.onmouseup = function (){ //全局监听拖拽列宽的发生 作用域设为全局是为了防抖动
  233. if (moveTd.oldX){
  234. event.stopPropagation();
  235. moveTd.dom.style.cursor = 'default';
  236. moveTd.oldX = undefined;
  237. setTimeout(function () {
  238. let column = _columns[(_isCheckAllBox && _targetDom.firstChild) ? moveTd.index-1 : moveTd.index];
  239. if (column) localStorage.setItem(_name+column.name, moveTd.dom.offsetWidth);
  240. });
  241. }
  242. }; //移动时调整表头与表身的列宽
  243. document.onmousemove = function(){
  244. if (moveTd.oldX){
  245. event.stopPropagation();
  246. let diff = event.clientX-moveTd.oldX;
  247. let newWidth = moveTd.dom.firstChild.offsetWidth+diff+"px";
  248. if (diff!==0){
  249. let trs = _targetDom.getElementsByTagName("tr");
  250. for (let j=(_before ? 1 : 0);j<trs.length;j++){
  251. if (isNested){
  252. let table = trs[j].parentElement;
  253. while (table.tagName!=='TABLE') table = table.parentElement;
  254. if (table.id!==object.el)continue;
  255. }
  256. trs[j].children[moveTd.index].firstChild.style.width = newWidth;
  257. trs[j].children[moveTd.index].style.minWidth = newWidth;
  258. }
  259. moveTd.oldX = event.clientX;
  260. }
  261. }
  262. };
  263. }
  264. function repaintDom() {
  265. let trs = _targetDom.getElementsByTagName("tr");
  266. for (let i=0;i<trs.length;i++){
  267. if (isNested){
  268. let table = trs[i].parentElement;
  269. while (table.tagName!=='TABLE') table = table.parentElement;
  270. if (table.id!==object.el)continue;
  271. }
  272. let tds = trs[i].children;
  273. for (let j=0;j<tds.length;j++){
  274. let div = document.createElement("div");
  275. div.style.overflowX = "hidden";
  276. let count = tds[j].childNodes.length;
  277. for (let k=0;k<count;k++){div.appendChild(tds[j].childNodes[0]);}
  278. tds[j].appendChild(div);
  279. tds[j].className += " pl-2 pr-2";
  280. }
  281. }
  282. }
  283. //初始化
  284. this.init = function() {
  285. if (!_isForbidDrag)repaintDom();
  286. createHeader();
  287. if (_before)createHeaderBefore();
  288. };
  289. };