Team:Paris Bettencourt/js/CollapsibleLists

From 2012.igem.org

/*

CollapsibleLists.js

An object allowing lists to dynamically expand and collapse

Created by Stephen Morley - http://code.stephenmorley.org/ - and released under the terms of the CC0 1.0 Universal legal code:

http://creativecommons.org/publicdomain/zero/1.0/legalcode

  • /

// create the CollapsibleLists object var CollapsibleLists =

   new function(){
     /* Makes all lists with the class 'collapsibleList' collapsible. The
      * parameter is:
      *
      * doNotRecurse - true if sub-lists should not be made collapsible
      */
     this.apply = function(doNotRecurse){
       // loop over the unordered lists
       var uls = document.getElementsByTagName('ul');
       for (var index = 0; index < uls.length; index ++){
         // check whether this list should be made collapsible
         if (uls[index].className.match(/(^| )collapsibleList( |$)/)){
           // make this list collapsible
           this.applyTo(uls[index], true);
           // check whether sub-lists should also be made collapsible
           if (!doNotRecurse){
             // add the collapsibleList class to the sub-lists
             var subUls = uls[index].getElementsByTagName('ul');
             for (var subIndex = 0; subIndex < subUls.length; subIndex ++){
               subUls[subIndex].className += ' collapsibleList';
             }
           }
         }
       }
     };
     /* Makes the specified list collapsible. The parameters are:
      *
      * node         - the list element
      * doNotRecurse - true if sub-lists should not be made collapsible
      */
     this.applyTo = function(node, doNotRecurse){
       // loop over the list items within this node
       var lis = node.getElementsByTagName('li');
       for (var index = 0; index < lis.length; index ++){
         // check whether this list item should be collapsible
         if (!doNotRecurse || node == lis[index].parentNode){
           // prevent text from being selected unintentionally
           if (lis[index].addEventListener){
             lis[index].addEventListener(
                 'mousedown', function (e){ e.preventDefault(); }, false);
           }else{
             lis[index].attachEvent(
                 'onselectstart', function(){ event.returnValue = false; });
           }
           // add the click listener
           if (lis[index].addEventListener){
             lis[index].addEventListener(
                 'click', createClickListener(lis[index]), false);
           }else{
             lis[index].attachEvent(
                 'onclick', createClickListener(lis[index]));
           }
           // close the unordered lists within this list item
           toggle(lis[index]);
         }
       }
     };
     /* Returns a function that toggles the display status of any unordered
      * list elements within the specified node. The parameter is:
      *
      * node - the node containing the unordered list elements
      */
     function createClickListener(node){
       // return the function
       return function(e){
         // ensure the event object is defined
         if (!e) e = window.event;
         // find the list item containing the target of the event
         var li = (e.target ? e.target : e.srcElement);
         while (li.nodeName != 'LI') li = li.parentNode;
         // toggle the state of the node if it was the target of the event
         if (li == node) toggle(node);
       };
     }
     /* Opens or closes the unordered list elements directly within the
      * specified node. The parameter is:
      *
      * node - the node containing the unordered list elements
      */
     function toggle(node){
       // determine whether to open or close the unordered lists
       var open = node.className.match(/(^| )collapsibleListClosed( |$)/);
       // loop over the unordered list elements with the node
       var uls = node.getElementsByTagName('ul');
       for (var index = 0; index < uls.length; index ++){
         // find the parent list item of this unordered list
         var li = uls[index];
         while (li.nodeName != 'LI') li = li.parentNode;
         // style the unordered list if it is directly within this node
         if (li == node) uls[index].style.display = (open ? 'block' : 'none');
       }
       // remove the current class from the node
       node.className =
           node.className.replace(
               /(^| )collapsibleList(Open|Closed)( |$)/, );
       // if the node contains unordered lists, set its class
       if (uls.length > 0){
         node.className += ' collapsibleList' + (open ? 'Open' : 'Closed');
       }
     }
   }();