首页 > 语言 > JavaScript > 正文

JavaScript对表格或元素按文本,数字或日期排序的方法

2024-05-06 16:20:54
字体:
来源:转载
供稿:网友

这篇文章主要介绍了JavaScript对表格或元素按文本,数字或日期排序的方法,涉及javascript页面元素操作及排序的相关技巧,需要的朋友可以参考下

本文实例讲述了JavaScript对表格或元素按文本,数字或日期排序的方法。分享给大家供大家参考。具体实现方法如下:

 

 
  1. // Sorting table columns correctly by text, number or date. There are other  
  2. // versions, plugins, etc., for this but they either are restricted to specific  
  3. // date formats, or require EVERY row-element to be given a sort attribute; mine  
  4. // can handle many different date and number formats, and even allows for specific  
  5. // cells that may not conform to the overall date/number format for that column.  
  6. // My version also enables sorting of element hierarchies: such as a DIV containing  
  7. // P-paragraphs and SPANs - this could even be an image-gallery containing prices  
  8. // or dates within spans. Very efficient as well!! 
  9. // Example: andrew.dx.am/sortgallerydel.html 
  10. // AddSortToTables(); will make the table headers clickable, to sort columns in  
  11. // ascending or descending order, for any tables with class="sortIt". 
  12. // SortTable(tbl, col); will sort the table (tbl is an id or table object) by  
  13. // the supplied column index (indexed from 0) in ascending or descending order. 
  14. // AddSortByDate(tbl, col, dateMask); enables sorting of a column by date,  
  15. // specified by a date-mask such as 'dd-mmm-yy'. 
  16. // AddSortByNumber(tbl, col); enables sorting of a column by number. This assumes a  
  17. // period . as the decimal separator (if present); it ignores any other non-numeric  
  18. // characters. 
  19. // SortElements(parentEl, childTag, colTag, colIndex); will sort (non-table)  
  20. // elements in ascending or descending order. For example, an UL containing LIs  
  21. // and SPANs. colIndex specifies which span to sort; there may be more than one in  
  22. // the LI (0 indexed). 
  23. // Example: SortElements('divid', 'p', 'span', 2); 
  24. // 3rd span within each paragraph. 
  25. // 
  26. // AddSortByDate2(parentEl, childTag, colTag, colIndex, dateMask); and  
  27. // AddSortByNumber2(parentEl, childTag, colTag, colIndex) 
  28. // provide the same feature-set as AddSortByDate and AddSortByNumber does  
  29. // for tables, but for element hierarchies. 
  30. // If there are dates or numbers in a column (or element) which don't meet the  
  31. // date-mask or number formatting necessary to sort correctly, then these individual  
  32. // elements can be given the attribute "sort" and they will still sort correctly! 
  33. // For example, with a date column <td sort="2012/12/20"> will still sort a  
  34. // cell correctly. (This format 'YYYY/MM/DD' will be converted into a Date() object.) 
  35. var MonthNames = ["January""February""March""April""May""June""July",  
  36. "August""September""October""November""December"]; 
  37. var DayNames = [ "Sunday""Monday""Tueday""Wednesday""Thursday",  
  38. "Friday""Saturday" ]; 
  39. var ShortMths = ["Jan""Feb""Mar""Apr""May""Jun""Jul""Aug",  
  40. "Sep""Oct""Nov""Dec"]; 
  41. var ShortDays = ["Sun""Mon""Tue""Wed""Thu""Fri""Sat"]; 
  42. var AddEvent = function (elem, eventType, func) { 
  43. // Helper function. 
  44. if ( elem.addEventListener ) 
  45. AddEvent = function (elem, eventType, func) { 
  46. elem.addEventListener(eventType, func, false); 
  47. }; 
  48. else if ( elem.attachEvent ) 
  49. AddEvent = function (elem, eventType, func) { 
  50. elem.attachEvent('on' + eventType, func); 
  51. }; 
  52. else 
  53. AddEvent = function (elem, eventType, func) { 
  54. elem['on' + eventType] = func; 
  55. }; 
  56. AddEvent(elem, eventType, func); 
  57. }; 
  58. // Sort methods/algorithms attributed: 
  59. var SortTable = function (tbl, col) { 
  60. // could be called directly 
  61. SortElements(tbl, 'tr''td', col); 
  62. }; 
  63. var SortElements = function (parentEl, childTag, colTag, colIndex) { 
  64. // example use: SortElements('table1','tr','td',2) 
  65. // or SortElements('list1','li') 
  66. // or SortElements('divName','p','span',3) 
  67. var i, j, cTags = {}, startAt = 0, childLen, aChild, elem, 
  68. sortBy, content, elems = [], sortedLen, frag, hdrsLen, hdr; 
  69. var parent = (typeof parentEl === 'string') ?  
  70. document.getElementById(parentEl) : parentEl; 
  71. var AscText = function (a, b) { // sort() by .data as text 
  72. var x = a.data, y = b.data, 
  73. xNum = parseFloat(x), yNum = parseFloat(y); 
  74. // check if each begin with a number.. 
  75. if ( !isNaN(xNum) && !isNaN(yNum) && (xNum - yNum) ) 
  76. return xNum - yNum; 
  77. return ((x < y) ? -1 : ((x > y) ? 1 : 0)); 
  78. }; 
  79. var DescText = function (a, b) { // sort() by .data 
  80. var x = a.data, y = b.data, 
  81. xNum = parseFloat(x), yNum = parseFloat(y); 
  82. // check if each begin with a number.. 
  83. if ( !isNaN(xNum) && !isNaN(yNum) && (yNum - xNum) ) 
  84. return yNum - xNum; 
  85. return ((x > y) ? -1 : ((x < y) ? 1 : 0)); 
  86. }; 
  87. var AscNum = function (a, b) { // used with dates as well 
  88. return a.data - b.data; 
  89. }; 
  90. var DescNum = function (a, b) { 
  91. return b.data - a.data; 
  92. }; 
  93. if (parent.nodeName.toLowerCase() == 'table') { 
  94. if ( childTag == 'tr' ) { 
  95. sortBy = parent.rows[0].cells[colIndex].sortBy || 'text'
  96. parent = parent.tBodies[0] || parent; 
  97. if ( parent.rows[0].cells[0].nodeName.toLowerCase() == 'th' ) { 
  98. startAt = 1; 
  99. cTags = parent.getElementsByTagName(childTag); 
  100. if ( typeof colIndex == 'undefined' ) { 
  101. sortBy = 'text'// sort simple lists or paragraphs as text 
  102. for (i = startAt, childLen = cTags.length; i < childLen; i++) { 
  103. // ..go forward examining each child 
  104. aChild = cTags[i]; 
  105. elem = (colTag) ? aChild.getElementsByTagName(colTag)[colIndex] : aChild; 
  106. if (elem) { 
  107. if ( !sortBy ) { // sorting non-table columns.. 
  108. sortBy = (typeof elem.numberValue != 'undefined') ? 'number' :  
  109. ((typeof elem.dateValue != 'undefined') ? 'date' : 'text'); 
  110. switch (sortBy) { 
  111. // You can supply 'sort' attributes to enable sorting of numbers, etc. 
  112. // For example, <td sort='2011/02/12'> for a date. 
  113. case 'text'
  114. content = (elem.getAttribute('sort') ||  
  115. elem.firstChild.nodeValue).toLowerCase(); 
  116. break
  117. case 'number'
  118. content = elem.numberValue; 
  119. break
  120. case 'date'
  121. content = elem.dateValue; 
  122. break
  123. default
  124. content = (elem.getAttribute('sort') ||  
  125. elem.firstChild.nodeValue).toLowerCase(); 
  126. break
  127. j = elems.length; 
  128. if ( !aChild.id )  
  129. aChild.id = 'tempSortID' + j; 
  130. elems[j] = { data: content, tempID: aChild.id }; 
  131. // The following will determine if the table/etc has already been sorted  
  132. // by the same column or tag. If so, it will sort in ascending or descending  
  133. // order. It creates custom element properties to the parent element to  
  134. // remember the previous sort details. 
  135. if ( typeof colIndex == 'undefined' ) colIndex = 0; 
  136. if ( parent.prevTag && parent.prevTag == ((typeof colTag == 'undefined') ?  
  137. childTag : colTag) ) { 
  138. if (parent.prevCol == colIndex) { 
  139. // sorting by the same column as previously 
  140. parent.prevSort = (parent.prevSort == 'asc') ? 'desc' : 'asc'
  141. else { // sorting by any other column 
  142. parent.prevCol = colIndex; 
  143. parent.prevSort = 'asc'
  144. else { 
  145. // sorting for the 1st time or by a different tag 
  146. parent.prevTag = ((typeof colTag == 'undefined') ? childTag : colTag); 
  147. parent.prevCol = colIndex; 
  148. parent.prevSort = 'asc'
  149. if ( parent.prevSort === 'desc' ) { 
  150. // 'desc' WILL BE the previous sort order.. 
  151. switch (sortBy) { 
  152. case 'text': elems.sort(DescText); break
  153. case 'number': elems.sort(DescNum); break
  154. case 'date': elems.sort(DescNum); break
  155. default: elems.sort(DescText); break
  156. else { 
  157. switch (sortBy) { 
  158. case 'text': elems.sort(AscText); break
  159. case 'number': elems.sort(AscNum); break
  160. case 'date': elems.sort(AscNum); break
  161. default: elems.sort(AscText); break
  162. frag = document.createDocumentFragment(); 
  163. for (i = 0, sortedLen = elems.length; i < sortedLen; i++) { 
  164. elem = document.getElementById(elems[i].tempID); 
  165. frag.appendChild(elem); 
  166. if ( (elem.id).substr(0,10) == 'tempSortID' ) 
  167. elem.removeAttribute('id'); 
  168. parent.appendChild(frag); 
  169. elems = null
  170. return parent.prevSort; // not currently used 
  171. }; 
  172. var AddSortToTables = function () { 
  173. // ..if table has class-name 'sortIt' 
  174. var tables = document.getElementsByTagName('table'), i, j,  
  175. tblLen, tbl, hdrs, hdrsLen; 
  176. function PreserveSortScope(a,b,c,d) { 
  177. return function () { 
  178. // assign the SortElements fn. to a table header 
  179. SortElements(a, b, c, d); 
  180. // add sorting to table headers 
  181. for ( i = 0, tblLen = tables.length; i < tblLen; i++ ) {  
  182. tbl = tables[i]; 
  183. if ( tbl.className.indexOf('sortIt') + 1) { 
  184. hdrs = tbl.getElementsByTagName('th'); 
  185. if ( hdrs ) { 
  186. for ( j = 0, hdrsLen = hdrs.length; j < hdrsLen; j++ ) { 
  187. AddEvent(hdrs[j],'click',PreserveSortScope(tbl,'tr','td',j)); 
  188. // if there's no title already, add "Click to sort" 
  189. if ( !hdrs[j].title ) hdrs[j].setAttribute('title'
  190. 'Click to sort'); 
  191. }; 
  192. var AddSortByDate = function (tbl, col, dateMask) { 
  193. // Input: the table name (or object), a column index (or array)  
  194. // and a date mask ('dd-mmm-yy') 
  195. // Adds a sortBy = 'date' property to the first row 
  196. // will ignore the first row, assuming it is a header row 
  197. var i, rLen, cell; 
  198. while ( col.length ) AddSortByDate(tbl,col.pop(),dateMask); 
  199. if ((col instanceof Array) || isNaN(col)) return
  200. var tbl = (typeof tbl === 'string') ? document.getElementById(tbl) : tbl; 
  201. tbl.rows[0].cells[col].sortBy = 'date'
  202. AddSortByDate2(tbl, 'tr''td', col, dateMask); 
  203. }; 
  204. var AddSortByDate2 = function (parentEl, childTag, colTag, colIndex, dateMask) { 
  205. var kids, startAt = 0, i, rLen, cell; 
  206. var parent = (typeof parentEl === 'string') ?  
  207. document.getElementById(parentEl) : parentEl; 
  208. if ( parent.nodeName.toLowerCase() == 'table' ) { 
  209. parent = parent.tBodies[0] || parent; 
  210. startAt = ( parent.rows[0].cells[0].nodeName.toLowerCase() == 'th' ) * 1; 
  211. kids = parent.getElementsByTagName(childTag); 
  212. for ( i = startAt, rLen = kids.length; i < rLen; i++) { 
  213. cell = kids[i].getElementsByTagName(colTag)[colIndex]; 
  214. if (cell) { 
  215. if ( typeof cell.numberValue != 'undefined' ) delete cell.numberValue; 
  216. // the above enables switching from a number to a date sort  
  217. // (although v. unlikely) 
  218. if (cell.getAttribute('sort')) { 
  219. // use sort attribute if present 
  220. cell.dateValue = new Date(cell.getAttribute('sort')); 
  221. else { 
  222. cell.dateValue = new Date(StringToDate(cell.firstChild.nodeValue,  
  223. dateMask)); 
  224. if (cell.dateValue.toString() == "NaN" || cell.dateValue.toString() ==  
  225. "Invalid Date") { 
  226. cell.dateValue = 0; 
  227. }  
  228. }; 
  229. var AddSortByNumber = function (tbl, col) { 
  230. // col is a column index or array of indices 
  231. // will ignore the first row, assuming it is a header row 
  232. var i, rLen, cell, tempNum; 
  233. while ( col.length ) AddSortByNumber(tbl,col.pop()); 
  234. if ((col instanceof Array) || isNaN(col)) return
  235. tbl = (typeof tbl === 'string') ? document.getElementById(tbl) : tbl; 
  236. tbl.rows[0].cells[col].sortBy = 'number'
  237. AddSortByNumber2(tbl,'tr','td',col); 
  238. }; 
  239. var AddSortByNumber2 = function (parentEl, childTag, colTag, colIndex) { 
  240. var kids, startAt = 0, i, rLen, cell, tempNum; 
  241. var parent = (typeof parentEl === 'string') ?  
  242. document.getElementById(parentEl) : parentEl; 
  243. if ( parent.nodeName.toLowerCase() == 'table' ) { 
  244. parent = parent.tBodies[0] || parent; 
  245. startAt = (parent.rows[0].cells[0].nodeName.toLowerCase() == 'th') * 1; 
  246. kids = parent.getElementsByTagName(childTag); 
  247. for (i = startAt, rLen = kids.length; i < rLen; i++) { 
  248. cell = kids[i].getElementsByTagName(colTag)[colIndex]; 
  249. if (cell) { 
  250. if ( typeof cell.dateValue != 'undefined' ) delete cell.dateValue; 
  251. // the above enables switching from a date to a number sort 
  252. // (although v. unlikely) 
  253. tempNum = cell.getAttribute('sort') || cell.firstChild.nodeValue; 
  254. tempNum = tempNum.replace(/[^0-9.-]/g, ''); 
  255. cell.numberValue = parseFloat(tempNum); 
  256. if (isNaN(cell.numberValue))  
  257. cell.numberValue = 0.0; 
  258. }; 
  259. var StringToDate = function (sDate, sFormat, cutOff) { 
  260. // Input: a date value as a string, it's format as a string e.g. 'dd-mmm-yy' 
  261. // Optional: a cutoff (integer) for 2 digit years. 
  262. // If no 'd' appears in the format string then the 1st of the month is assumed. 
  263. // If the year is 20 and the cut-off is 30 then the value will be converted  
  264. // to 2020; if the year is 40 then this will be converted to 1940. 
  265. // If no cut-off is supplied then '20' will be pre-pended to the year (YY). 
  266. // Output: a string in the format 'YYYY/MM/DD' or '' 
  267. // Will not attempt to convert certain combinations e.g. DMM, MDD, DDM, YYYYD. 
  268. var sParsed, fndSingle; 
  269. // sParsed will be constructed in the format 'YYYY/MM/DD' 
  270. sDate = sDate.toString().toUpperCase(); 
  271. sFormat = sFormat.toUpperCase(); 
  272. if (sFormat.search(/MMMM|MMM/) + 1) { // replace Mar/March with 03, etc. 
  273. sDate = sDate.replace(new RegExp('(' + ShortMths.join('|') + ')[A-Z]*''gi'), 
  274. function (m) { 
  275. var i = ShortMths.indexOf(m.charAt(0).toUpperCase() +  
  276. m.substr(1, 2).toLowerCase()) + 1; 
  277. return ((i < 10) ? "0" + i : "" + i).toString(); 
  278. }); 
  279. sFormat = sFormat.replace(/MMMM|MMM/g, 'MM'); 
  280. if (sFormat.search(/DDDD|DDD/) + 1) { // replace Tue/Tuesday, etc. with '' 
  281. sDate = sDate.replace(new RegExp('(' + ShortDays.join('|') + ')[A-Z]*''gi'),''); 
  282. sFormat = sFormat.replace(/DDDD|DDD/g, ''); 
  283. sDate = sDate.replace(/(^|/D)(/d)(?=/D|$)/g, function($0, $1, $2) { 
  284. // single digits 2 with 02 
  285. return $1 + '0' + $2; 
  286. }); 
  287. sFormat = sFormat.replace(/(^|[^DMY])(D|M)(?=[^DMY]|$)/g, function($0, $1, $2){ 
  288. return $1 + $2 + $2; // replace D or M with DD and MM 
  289. }); 
  290. // are there still single Ds or Ms? 
  291. fndSingle = sFormat.search(/(^|[^D])D([^D]|$)|(^|[^M])M([^M]|$)/)+1; 
  292. if ( fndSingle ) return ''// do not attempt to parse, for example, 'DMM' 
  293. sFormat = sFormat.replace(/(^|[^Y])(YY)(?=[^Y]|$)/g, function($0, $1, $2, index) { 
  294. var tempDate = sDate.substr(0, index + 1); 
  295. tempDate += (cutOff) ? ((parseInt(sDate.substr(index + 1, 2),10) > cutOff) ?  
  296. '19' : '20') : '20'
  297. tempDate += sDate.substr(index + 1); 
  298. sDate = tempDate; 
  299. return $1 + $2 + $2; 
  300. }); 
  301. sParsed = ('YYYY/MM/DD').replace(/YYYY|MM|DD/g, function(m){ 
  302. return (sFormat.indexOf(m) + 1) ?  
  303. sDate.substr(sFormat.indexOf(m), m.length) : ''
  304. }); 
  305. if (sParsed.charAt(0) == '/') { 
  306. // if no year specified, assume the current year 
  307. sParsed = (new Date().getFullYear()) + sParsed; 
  308. if (sParsed.charAt(sParsed.length - 1) == '/') { 
  309. // if no date, assume the 1st of the month 
  310. sParsed += '01'
  311. // should end up with 10 characters.. 
  312. return ( sParsed.length == 10 ) ? sParsed : ''
  313. }; 

希望本文所述对大家的javascript程序设计有所帮助。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表

图片精选