ncbo/bioportal_web_ui

View on GitHub
public/widgets/jquery.ncbo.tree.js

Summary

Maintainability
F
1 wk
Test Coverage
/*
* jQuery SimpleTree Drag&Drop plugin
* Update on 22th May 2008
* Version 0.3
*
* Licensed under BSD <http://en.wikipedia.org/wiki/BSD_License>
* Copyright (c) 2008, Peter Panov <panov@elcat.kg>, IKEEN Group http://www.ikeen.com
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*     * Redistributions of source code must retain the above copyright
*       notice, this list of conditions and the following disclaimer.
*     * Redistributions in binary form must reproduce the above copyright
*       notice, this list of conditions and the following disclaimer in the
*       documentation and/or other materials provided with the distribution.
*     * Neither the name of the Peter Panov, IKEEN Group nor the
*       names of its contributors may be used to endorse or promote products
*       derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY Peter Panov, IKEEN Group ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL Peter Panov, IKEEN Group BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/

(function($) {
  $.fn.NCBOTree = function(opt) {
    var OPTIONS;
    var ROOT_ID = "roots";

    OPTIONS = {
      autoclose:         false,
      beforeExpand:      false,
      afterExpand:       false,
      afterExpandError:  false,
      afterSelect:       false,
      afterJumpToClass:  false,
      timeout:           999999,
      treeClass:         "ncboTree",
      autocompleteClass: "ncboAutocomplete",
      width:             350,
      ncboAPIURL:        "http://data.bioontology.org",
      ncboUIURL:         "http://bioportal.bioontology.org",
      apikey:            null,
      ontology:          null,
      startingClass:     null,
      startingRoot:      ROOT_ID,
      defaultRoot:       ROOT_ID
    };

    OPTIONS = $.extend(OPTIONS, opt);

    // Required options
    if (OPTIONS.apikey == null)
      throw new Error("You must provide an API Key for NCBO Tree Widget to operate");

    if (OPTIONS.ontology == null)
      throw new Error("You must provide an ontology id for NCBO Tree Widget to operate");

    setupTree = function(container, option) {
      var $TREE_CONTAINER = container;
      var TREE = $("<ul>").append($("<li>").addClass("root"));
      var OPTIONS = option;
      var ROOT = $('.root', TREE);
      var mousePressed = false;
      TREE.css("width", OPTIONS.width);

      // Empty out the tree container
      $TREE_CONTAINER.html("");

      // Only set starting root when something other than roots is selected
      var startingRoot = (OPTIONS.startingRoot == OPTIONS.defaultRoot) ? null : OPTIONS.startingRoot;

      // Add the autocomplete
      var autocompleteContainer = $("<div>").addClass(OPTIONS.autocompleteClass).addClass("ncboTree");
      var input = $("<input>")
        .addClass(OPTIONS.autocompleteClass)
        .css("width", OPTIONS.width)
        .attr("placeholder", "Search for class...");
      autocompleteContainer.append(input);
      input.NCBOAutocomplete({
        url: OPTIONS.ncboAPIURL + "/search",
        searchParameter: "q",
        resultAttribute: "collection",
        property: "prefLabel",
        searchTextSuffix: "*",
        searchFromRoot: startingRoot,
        onSelect: function(item, searchInput) {
          TREE.jumpToClass(item["@id"]);
          searchInput.val("");
        },
        minCharacters: 3,
        additionalParameters: {
          apikey: OPTIONS.apikey,
          no_context: true,
          ontologies: OPTIONS.ontology
        }
      });
      $TREE_CONTAINER.append(autocompleteContainer);

      // Add the actual tree
      $TREE_CONTAINER.append(TREE);

      // Add provided class
      TREE.addClass(OPTIONS.treeClass);

      // format the nodes to match what simpleTree is expecting
      TREE.formatNodes = function(nodes) {
        var holder = $("<span>");
        var ul = $("<ul>")

        // Sort by prefLabel
        nodes.sort(function(a, b){
          var aName = a.prefLabel.toLowerCase();
          var bName = b.prefLabel.toLowerCase();
          return ((aName < bName) ? -1 : ((aName > bName) ? 1 : 0));
        });

        $.each(nodes, function(index, node){
          var li = $("<li>");
          var a = $("<a>").attr("href", TREE.determineHTTPS(node.links.self)).html(node.prefLabel);
          a.attr("data-id", encodeURIComponent(node["@id"]));

          ul.append(li.append(a));

          var hasChildrenNotExpanded = typeof node.children !== 'undefined' && node.childrenCount > 0 && node.children.length == 0;
          if (node.childrenCount > 0 && typeof node.children === 'undefined' || hasChildrenNotExpanded) {
            var ajax_ul = $("<ul>").addClass("ajax");
            var ajax_li = $("<li>");
            var ajax_a = $("<a>").attr("href", node.links.children);
            li.append(ajax_ul.append(ajax_li.append(ajax_a)));
          } else if (typeof node.children !== 'undefined' && node.children.length > 0) {
            var child_ul = TREE.formatNodes(node.children);
            li.attr("class", "folder-open")
            li.append(child_ul);
          }
        });

        holder.append(ul)
        return holder.html();
      }

      TREE.findRootNode = function(nodes) {
        var startingRoot = (OPTIONS.startingRoot == OPTIONS.defaultRoot) ? null : OPTIONS.startingRoot;
        if (startingRoot == null) {return nodes;}

        var foundNode = false;
        var searchQueue = nodes;

        while (searchQueue.length > 0 || foundNode == false) {
          var node = searchQueue.shift();
          if (node["@id"] == startingRoot) {
            foundNode = [node];
          } else if (typeof node.children !== 'undefined' && node.children.length > 0) {
            searchQueue = searchQueue.concat(node.children);
          }
        }

        return foundNode;
      }

      TREE.jumpToClass = function(cls, callback) {
        ROOT.html($("<span>").html("Loading...").css("font-size", "smaller"));
        $.ajax({
          url: TREE.determineHTTPS(OPTIONS.ncboAPIURL) + "/ontologies/" + OPTIONS.ontology + "/classes/" + encodeURIComponent(cls) + "/tree",
          data: {apikey: OPTIONS.apikey, include: "prefLabel,childrenCount", no_context: true},
          contentType: 'json',
          crossDomain: true,
          success: function(roots){
            roots = TREE.findRootNode(roots);
            ROOT.html(TREE.formatNodes(roots));
            TREE.setTreeNodes(ROOT, false);

            if (typeof callback == 'function') {
              callback();
            }

            TREE.selectClass(cls);
            if (typeof OPTIONS.afterJumpToClass == 'function') {
              OPTIONS.afterJumpToClass(cls);
            }
            $TREE_CONTAINER.trigger("afterJumpToClass", cls);
          }
        });
      }

      TREE.selectClass = function(cls){
        var foundClass = $($(this).find("a[data-id='" + encodeURIComponent(cls) + "']"));
        $(TREE.find("a.active")[0]).removeClass("active");
        foundClass.addClass("active");
      }

      TREE.closeNearby = function(obj) {
        $(obj).siblings().filter('.folder-open, .folder-open-last').each(function(){
          var childUl = $('>ul',this);
          var className = this.className;
          this.className = className.replace('open', 'close');
          childUl.hide();
        });
      };

      TREE.nodeToggle = function(obj) {
        var childUl = $('>ul',obj);
        if (childUl.is(':visible')) {
          obj.className = obj.className.replace('open','close');
          childUl.hide();
        } else {
          obj.className = obj.className.replace('close','open');
          childUl.show();
          if (OPTIONS.autoclose)
            TREE.closeNearby(obj);
          if (childUl.is('.ajax'))
            TREE.setAjaxNodes(childUl, obj.id);
        }
      };

      TREE.setAjaxNodes = function(node, parentId, successCallback, errorCallback) {
        if (typeof OPTIONS.beforeExpand == 'function') {
          OPTIONS.beforeExpand(node);
        }
        $TREE_CONTAINER.trigger("beforeExpand", node);

        var url = $.trim($('a', node).attr("href"));
        if (url) {
          $.ajax({
            type: "GET",
            url: url,
            data: {apikey: OPTIONS.apikey, include: "prefLabel,childrenCount", no_context: true},
            crossDomain: true,
            contentType: 'json',
            timeout: OPTIONS.timeout,
            success: function(response) {
              var nodes = TREE.formatNodes(response.collection)
              node.removeAttr('class');
              node.html(nodes);
              $.extend(node, {url:url});
              TREE.setTreeNodes(node, true);
              if (typeof OPTIONS.afterExpand == 'function') {
                OPTIONS.afterExpand(node);
              }
              $TREE_CONTAINER.trigger("afterExpand", node);
              if (typeof successCallback == 'function') {
                successCallback(node);
              }
            },
            error: function(response) {
              if (typeof OPTIONS.afterExpandError == 'function') {
                OPTIONS.afterExpandError(node);
              }
              if (typeof errorCallback == 'function') {
                errorCallback(node);
              }
              $TREE_CONTAINER.trigger("afterExpandError", node);
            }
          });
        }
      };

      TREE.setTreeNodes = function(obj, useParent) {
        obj = useParent ? obj.parent() : obj;
        $('li>a', obj).addClass('text').bind('selectstart', function() {
          return false;
        }).click(function(){
          var parent = $(this).parent();
          var selectedNode = $(this);
          $('.active', TREE).attr('class', 'text');
          if (this.className == 'text') {
            this.className = 'active';
          }
          if (typeof OPTIONS.afterSelect == 'function') {
            OPTIONS.afterSelect(decodeURIComponent(selectedNode.data("id")), selectedNode.text(), selectedNode);
          }
          $TREE_CONTAINER.trigger("afterSelect", [decodeURIComponent(selectedNode.data("id")), selectedNode.text(), selectedNode]);
          return false;
        }).bind("contextmenu",function(){
          $('.active', TREE).attr('class', 'text');
          if (this.className == 'text') {
            this.className = 'active';
          }
          if (typeof OPTIONS.afterContextMenu == 'function') {
            OPTIONS.afterContextMenu(parent);
          }
          return false;
        }).mousedown(function(event) {
          mousePressed = true;
          cloneNode = $(this).parent().clone();
          var LI = $(this).parent();
          return false;
        });

        $('li', obj).each(function(i) {
          var className = this.className;
          var open = false;
          var cloneNode=false;
          var LI = this;
          var childNode = $('>ul',this);
          if (childNode.size() > 0){
            var setClassName = 'folder-';
            if (className && className.indexOf('open') >= 0) {
              setClassName = setClassName + 'open';
              open = true;
            } else {
              setClassName = setClassName+'close';
            }
            this.className = setClassName + ($(this).is(':last-child') ? '-last' : '');

            if (!open || className.indexOf('ajax') >= 0)
              childNode.hide();

            TREE.setTrigger(this);
          } else {
            var setClassName = 'doc';
            this.className = setClassName + ($(this).is(':last-child') ? '-last' : '');
          }
        }).before('<li class="line">&nbsp;</li>')
          .filter(':last-child')
          .after('<li class="line-last"></li>');
      };

      TREE.setTrigger = function(node) {
        $('>a',node).before('<img class="trigger" src="" border=0>');
        var trigger = $('>.trigger', node);
        trigger.click(function(event){
          TREE.nodeToggle(node);
        });
        // TODO: $.browser was removed in jQuery 1.9, check IE compatability
        // if (!$.browser.msie) {
        //   trigger.css('float','left');
        // }
      };

      TREE.checkNodeIsLast = function(node) {
        if (node.className.indexOf('last')>=0) {
          node.className = node.className.replace('-last','');
        }
      };

      TREE.checkLineIsLast = function(line) {
        if (line.className.indexOf('last')>=0) {
          var prev = $(line).prev();
          if (prev.size() > 0) {
            prev[0].className = prev[0].className.replace('-last','');
          }
          dragNode_source[0].className+='-last';
        }
      };

      TREE.convertToFolder = function(node) {
        node[0].className = node[0].className.replace('doc','folder-open');
        node.append('<ul><li class="line-last"></li></ul>');
        TREE.setTrigger(node[0]);
        TREE.setEventLine($('.line, .line-last', node));
      };

      TREE.convertToDoc = function(node) {
        $('>ul', node).remove();
        $('img', node).remove();
        node[0].className = node[0].className.replace(/folder-(open|close)/gi , 'doc');
      };

      TREE.addNode = function(id, text, link, callback) {
        var temp_node = $('<li><ul><li id="'+id+'"><a href="'+link+'">'+text+'</a></li></ul></li>');
        TREE.setTreeNodes(temp_node);
        dragNode_destination = TREE.getSelected();
        dragNode_source = $('.doc-last',temp_node);
        TREE.moveNodeToFolder(dragNode_destination);
        temp_node.remove();
        if (typeof(callback) == 'function') {
          callback(dragNode_destination, dragNode_source);
        }
      };

      TREE.delNode = function(callback) {
        dragNode_source = TREE.getSelected();
        TREE.checkNodeIsLast(dragNode_source[0]);
        dragNode_source.prev().remove();
        dragNode_source.remove();
        if (typeof(callback) == 'function') {
          callback(dragNode_destination);
        }
      };

      TREE.determineHTTPS = function(url) {
        return url.replace("http:", ('https:' == document.location.protocol ? 'https:' : 'http:'));
      }

      // Populate roots and init tree
      TREE.init = function() {
        if (OPTIONS.startingClass !== null) {
          TREE.jumpToClass(OPTIONS.startingClass);
          OPTIONS.startingClass = null;
        } else {
          ROOT.html($("<span>").html("Loading...").css("font-size", "smaller"));
          $.ajax({
            url: TREE.determineHTTPS(OPTIONS.ncboAPIURL) + "/ontologies/" + OPTIONS.ontology + "/classes/" + encodeURIComponent(OPTIONS.startingRoot),
            data: {apikey: OPTIONS.apikey, include: "prefLabel,childrenCount", no_context: true},
            contentType: 'json',
            crossDomain: true,
            success: function(roots){
              // Flatten potentially nested arrays
              roots = $.map([roots], function(n){
                return n;
              });
              ROOT.html(TREE.formatNodes(roots));
              TREE.setTreeNodes(ROOT, false);
            }
          });
        }
      };

      return TREE;
    }

    // Returns the original object(s) so they can be chained
    return this.each(function() {
      var $this = $(this);

      // These methods will be added to the tree object
      // they are essentially "public". Doing the following
      // gives you a handle to inspect the tree from:
      // var tree = $("#my_div").NCBOTree(opts);
      // And then you can do: tree.selectedClass()
      $.extend(this, {
        selectedClass: function(){
          var cls = $($(this).find("a.active")[0]);
          if (cls.length == 0) {
            return null;
          } else {
            return {
              id: decodeURIComponent(cls.data("id")),
              prefLabel: cls.html(),
              URL: cls.attr("href")
            };
          }
        },

        selectClass: function(cls){
          var foundClass = $($(this).find("a[data-id='" + encodeURIComponent(cls) + "']"));
          $($(this).find("a.active")[0]).removeClass("active");
          foundClass.addClass("active");
        },

        jumpToClass: function(cls, callback){
          TREE.jumpToClass(cls, callback);
        },

        changeOntology: function(ont){
          var newTree = $("<ul>").append($("<li>").addClass("root"));
          setupTree($TREE_CONTAINER, newTree, OPTIONS);
          OPTIONS.ontology = ont;
          TREE.init();
        }
      });

      // Add the autocomplete code
      $.ajax({
        url: OPTIONS.ncboUIURL.replace("http:", ('https:' == document.location.protocol ? 'https:' : 'http:')) + "/widgets/jquery.ncbo.autocomplete.js",
        type: "GET",
        crossDomain: true,
        dataType: "script",
        success: function(){
          var tree = setupTree($this, OPTIONS);
          tree.init();
        }
      });
    });
  }
}(jQuery));