Greasy Fork is available in English.

Table Sort for TorrentsMD

A script for making table sortable

//
// ==UserScript==
// @name          Table Sort for TorrentsMD
// @namespace     Table Sorter
// @description   A script for making table sortable
// @include     *torrentsmd.*/browse.php*
// @include     *torrentsmoldova.*/browse.php*
// @include     *torrentsmd.*/search.php*
// @include     *torrentsmoldova.*/search.php*

// @version 0.0.1.20140607094600
// ==/UserScript==

if (CODEBOX===undefined){
    var CODEBOX={};
}

CODEBOX.tabsorter = {
    'processTable' : function(tableElement){
        if ( !checkTableStructureOk(tableElement) ){
            return null;
        }

     // State is initialised here
        var rowObjects           = [];
        var currentSortColIndex  = null;
        var currentSortAscending = null;
        var linkTds              = [];
        var rowWithArrows, rowWithOptions;
        var colCount             = 0;
        var sortTypes = {
            'NONE' : {
                'text' : 'none',
                'fn'   : sortRestore
            },
            'ALPHA' : {
                'text' : 'alpha',
                'fn'   : sortColByAlpha
            },
            'NUM' : {
                'text' : 'num',
                'fn'   : sortColByNum
            }
        };
        var currentSortType = sortTypes.NONE;


        function sortRestore(){
            rowObjects.sort(function(a,b){
                return a.getOriginalIndex()    - b.getOriginalIndex();
            });
            redraw();
        }

        function sortColByAlpha(col, ascending){
            if (col<colCount){
                var multiplier = ascending ? 1 : -1;
                rowObjects.sort(function(a,b){
                    var aItem = a.getCellValue(col);
                    aItem = aItem ? aItem.toLowerCase() : aItem;

                    var bItem = b.getCellValue(col);
                    bItem = bItem ? bItem.toLowerCase() : bItem;

                    if (aItem){
                        if (bItem){
                            if (aItem > bItem){
                                return 1 * multiplier;
                            } else if (aItem < bItem){
                                return -1 * multiplier;
                            } else {
                                return 0;
                            }
                        } else {
                            return 1 * multiplier;
                        }
                    } else {
                        if (bItem){
                            return -1 * multiplier;
                        } else {
                            return 0;
                        }
                    }
                    CODEBOX.assert(false);
                });
            }
            redraw();
        }

        function sortColByNum(col, ascending){
            function makeNumFromText(txt){
                return parseFloat(txt.replace(/,/g, '').replace(/[^0-9\.+\-]+/, ' '));
            }
            if (col<colCount){
                var multiplier = ascending ? 1 : -1;
                rowObjects.sort(function(a,b){
                    var aNum = makeNumFromText(a.getCellValue(col));
                    if (isNaN(aNum)){
                     // Make non-numeric values go to the bottom of the list
                        return Infinity;
                    }
                    var bNum = makeNumFromText(b.getCellValue(col));
                    if (isNaN(bNum)){
                     // Make non-numeric values go to the bottom of the list
                        return -Infinity;
                    }
                    return (aNum - bNum) * multiplier;
                });
            }
            redraw();
        }

     // Populate rowObjects array
        CODEBOX.forEach(tableElement.getElementsByTagName("TR"), function(item, i){
            var rowObject = buildRowObject(item, i);
            if (rowObject){
                rowObjects.push(rowObject);
                colCount = ( (colCount < rowObject.getCellCount()) ? rowObject.getCellCount() : colCount);
            }
        });

        setupSortRows();


        function checkTableStructureOk(tableElement){
            CODEBOX.assert(tableElement);

         // tables that contain other tables are not processed...
             var nestedTables = tableElement.getElementsByTagName("TABLE");
             if (nestedTables.length > 0){
                 return false;
             }

         // tables that have less than 2 rows are not processed...
             var rows = tableElement.getElementsByTagName("TR");
             if (rows.length < 2){
                 return false;
             }

         // tables that don't have any text nodes are not processed...
              function checkForTextNodeChildren(item){
                  CODEBOX.assert(item);
                  var childNodes = item.childNodes;

                  if (!childNodes){
                      return false;
                  }

                  var childNode;
                  var TEXT_NODE_TYPE = 3;
                  var TRIM_REGEX     = /^\s+|\s+$/;
                  for (var i=0, l=childNodes.length; i<l; i++){
                      childNode = childNodes[i];
                      if (childNode.nodeType===TEXT_NODE_TYPE){
                          var text = childNode.nodeValue;
                          text = text.replace(TRIM_REGEX, '');
                          if (text.length > 0){
                              return true;
                          }
                      } else {
                          if ( checkForTextNodeChildren(childNode) ){
                              return true;
                          }
                      }
                  }
                  return false;
              }
              if (!checkForTextNodeChildren(tableElement)){
                  return false;
              }

            return true;
        }

        function buildRowObject(trElement, index){
            if (trElement.parentNode.tagName==='THEAD'){
             // Any rows inside a header are ignored
                return null;
            }

            var tdElements = trElement.getElementsByTagName("TD");
            if (tdElements.length===0){
             // Rows without any TDs are ignored, these are usually headers with THs instead
                return null;
            }

            var obj = {};

            obj.getTr = function(){
                return trElement;
            };

            var parent = trElement.parentNode;
            var cellToElementMappingArray = [];
            var cellIndex=0;

         /* Produces a mapping from 'virtual' cells to TD elements
            so if the third cell has colspan='3' then virtual cells 2,3 and 4 all map to TD 2
             |A|B|  C  |
             [0,1,2,2,2] */
             CODEBOX.forEach(tdElements, function(item, i){
                var colSpan = +(item.colSpan);
                if (isNaN(colSpan) || colSpan===0){
                    colSpan = 1;
                }
                CODEBOX.assert(colSpan);
                for(var j=0; j<colSpan; j++){
                    cellToElementMappingArray[cellIndex++] = i;
                }
             });

            obj.getCellCount = function(){
                return cellToElementMappingArray.length;
            };

            obj.getCellValue = function(index){
                if (index >= cellToElementMappingArray.length){
                    return null;
                } else {
                    var tdIndex = cellToElementMappingArray[index];
                    CODEBOX.assert( tdIndex < tdElements.length );

                    var cellContent = tdElements[tdIndex].innerHTML;
                 // This might contain markup, eg <span class='tdText>cell text</span>, so clean it out
                     var STRIP_MARKUP_REGEX = /<[^<]*>/g;
                    cellContent = cellContent.replace(STRIP_MARKUP_REGEX, '');

                    return cellContent;
                }
            };

            obj.getOriginalIndex = function(){
                return index;
            };

            obj.removeFromParent = function(){
                parent.removeChild(trElement);
            };
            obj.attachToParent = function(){
                parent.appendChild(trElement);
            };

            return obj;
        }


        function setSortType(sortType){
            currentSortType = sortType;

         // Adjust the styles of the sort-type links
            CODEBOX.forEachObject(sortTypes, function(thisSortType){
                var isSelectedSortType = (thisSortType===currentSortType);
                
                thisSortType.link.style.color          = (isSelectedSortType ? 'black' : 'gray');
                thisSortType.link.style.textDecoration = (isSelectedSortType ? '' : 'underline');
                thisSortType.link.style.cursor         = (isSelectedSortType ? '' : 'pointer'  );
            });
            
         // Hide the sort arrows if we aren't sorting
            rowWithArrows.style.display = (sortType===sortTypes.NONE) ? 'none' : '';
            
         // Reset any 'selected' arrow from the previous sort
            if (currentSortColIndex !== null){
                updateLinkCell(currentSortColIndex, false, false);
            }
        }

        function setupSortRows(){
            var firstRow = rowObjects[0].getTr();

         // Add a new row for the options
             rowWithOptions = document.createElement("TR");
             firstRow.parentNode.insertBefore(rowWithOptions, firstRow);

            var optionsTd = document.createElement("TD");
            optionsTd.style.fontSize = '0.7em';
            optionsTd.colSpan        = colCount;

            var optionsSpan = document.createElement("SPAN");
            optionsSpan.style.paddingLeft = '1em';
            optionsSpan.innerHTML         = 'Sort Type: ';

            var optionAlphaSortLink = document.createElement("A");
            optionAlphaSortLink.style.paddingLeft    = '0.5em';
            optionAlphaSortLink.style.textDecoration = 'underline';
            optionAlphaSortLink.style.cursor         = 'pointer';
            optionAlphaSortLink.innerHTML            = 'alpha';
            optionAlphaSortLink.addEventListener('click',
                    function(e){
                        setSortType( sortTypes.ALPHA );
                    }, false);
            sortTypes.ALPHA.link = optionAlphaSortLink;

            var optionNoSortLink = document.createElement("A");
            optionNoSortLink.style.paddingLeft    = '0.5em';
            optionNoSortLink.style.textDecoration = 'underline';
            optionNoSortLink.style.cursor         = 'pointer';
            optionNoSortLink.innerHTML            = 'none';
            optionNoSortLink.addEventListener('click',
                    function(e){
                        setSortType( sortTypes.NONE );
                        sortTypes.NONE.fn();
                    }, false);
            sortTypes.NONE.link = optionNoSortLink;

            var optionNumSortLink = document.createElement("A");
            optionNumSortLink.style.paddingLeft    = '0.5em';
            optionNumSortLink.style.textDecoration = 'underline';
            optionNumSortLink.style.cursor         = 'pointer';
            optionNumSortLink.innerHTML            = 'num';
            optionNumSortLink.addEventListener('click',
                    function(e){
                        setSortType( sortTypes.NUM );
                    }, false);
            sortTypes.NUM.link = optionNumSortLink;

            optionsTd.appendChild(optionsSpan);
            optionsTd.appendChild(optionAlphaSortLink);
            optionsTd.appendChild(optionNumSortLink);
            optionsTd.appendChild(optionNoSortLink);
            rowWithOptions.appendChild(optionsTd);

         // Add a new row for the sort arrows
             rowWithArrows = document.createElement("TR");
             firstRow.parentNode.insertBefore(rowWithArrows, firstRow);
             var linkTd, linkElement;
             for(var i=0; i<colCount; i++){
                linkTd = document.createElement("TD");
                linkTd.style.fontSize   = '0.7em';
                linkTd.style.fontWeight = 'bold';
                linkTd.style.textAlign  = 'center';

                linkTds.push(linkTd);

                linkElement = document.createElement("A");
                linkTd.appendChild(linkElement);

                rowWithArrows.appendChild(linkTd);

                setCellImages(i, false, false);
             }

             setSortType(sortTypes.NONE);
        }

        function updateLinkCell(colIndex, isAscending, isDescending){
            if (currentSortColIndex === null){
             // No column is currently selected, so we don't need to reset anything
                setCellImages(colIndex, isAscending, isDescending);

            } else {
             // There is already a column selected
                if ( colIndex == currentSortColIndex){
                 // The new column is the same as the current one...
                    if (currentSortAscending != null && ((currentSortAscending && isAscending) || (!currentSortAscending && isDescending))){
                         // The new sort order the the same as the current one, so do nothing
                    } else {
                     // The column is the same, but the order is different
                        setCellImages(colIndex, isAscending, isDescending);
                    }
                } else {
                 // The new column is different to the current one so reset the current one
                    setCellImages(currentSortColIndex, false, false);
                 // Set up the new one
                     setCellImages(colIndex, isAscending, isDescending);
                }
            }
        }

        function setCellImages(colIndex, isAscending, isDescending){
            var DATA_URL_UP         = "";
            var DATA_URL_UP_FADED   = "";
            var DATA_URL_DOWN       = "";
            var DATA_URL_DOWN_FADED = "";

            var linkCell = linkTds[colIndex];

         // Remove the current images...
            while(linkCell.firstChild){
                linkCell.removeChild(linkCell.firstChild);
            }

            var upArrowImage = document.createElement('IMG');
            upArrowImage.src = (isAscending ? DATA_URL_UP : DATA_URL_UP_FADED);
            upArrowImage.style.cursor = 'pointer';
            upArrowImage.addEventListener('click',
                    function(e){
                        updateLinkCell(colIndex, true, false);
                        currentSortColIndex  = colIndex;
                        currentSortAscending = true;
                        currentSortType.fn(colIndex, true);
                    }, false);

            var downArrowImage = document.createElement('IMG');
            downArrowImage.src = (isDescending ? DATA_URL_DOWN : DATA_URL_DOWN_FADED);
            downArrowImage.style.cursor = 'pointer';
            downArrowImage.addEventListener('click',
                    function(e){
                        updateLinkCell(colIndex, false, true);
                        currentSortColIndex  = colIndex;
                        currentSortAscending = false;
                        currentSortType.fn(colIndex, false);
                    }, false);

            linkCell.appendChild(upArrowImage);
            linkCell.appendChild(downArrowImage);
        }


        function redraw(){
            CODEBOX.forEach(rowObjects, function(item){
                item.removeFromParent();
            });

            CODEBOX.forEach(rowObjects, function(item){
                item.attachToParent();
            });
        }

    }
};


CODEBOX.addNamespace = function (namespaceName){
    function addNamespaceToObject(obj, namespace){
        if (!obj){
            obj = CODEBOX;
        }
        if (obj[namespace]===undefined){
            obj[namespace] = {};
        }
        return obj[namespace];
    }

    var namespaceParts = namespaceName.split('.');
    var parentNamespace = null;
    var thisPart;

    for( var i=0; i<namespaceParts.length; i++) {
        thisPart = namespaceParts[i];
        parentNamespace = addNamespaceToObject(parentNamespace, thisPart);
    }
};

CODEBOX.addNamespace('consts');
CODEBOX.consts.assert = true;
CODEBOX.consts.dialogOnAssert = true;

CODEBOX.assert = function( expression, msg, location ){
    if (CODEBOX.consts.assert){
        if (!expression){
            var errMsg = 'ASSERTION FAILED';
            if (msg){
                errMsg += ': ' + msg;
                if (location){
                    errMsg += ' at ' + location;
                }
            }
            if (CODEBOX.consts.dialogOnAssert){
                alert(errMsg);
            }
            throw new Error(errMsg);
        }
    }
};

CODEBOX.forEach = function(arr, fnDoThis){
    var len = arr.length;
    for (var i=0; i < len; i++){
        fnDoThis(arr[i], i);
    }
};

CODEBOX.forEachObject = function(obj, fnDoThis){
    for (var item in obj){
        if (obj.hasOwnProperty(item)){
            fnDoThis(obj[item]);
        }
    }
};

var allTables = window.document.documentElement.getElementsByTagName("TABLE");
CODEBOX.forEach(allTables, function(table){
    CODEBOX.tabsorter.processTable(table);
});