All Downloads are FREE. Search and download functionalities are using the official Maven repository.

weka.gui.treevisualizer.TreeBuild Maven / Gradle / Ivy

/*
 *   This program is free software: you can redistribute it and/or modify
 *   it under the terms of the GNU General Public License as published by
 *   the Free Software Foundation, either version 3 of the License, or
 *   (at your option) any later version.
 *
 *   This program is distributed in the hope that it will be useful,
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *   GNU General Public License for more details.
 *
 *   You should have received a copy of the GNU General Public License
 *   along with this program.  If not, see .
 */

/*
 *    TreeBuild.java
 *    Copyright (C) 1999-2012 University of Waikato, Hamilton, New Zealand
 *
 */

package weka.gui.treevisualizer;

import java.awt.Color;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.Reader;
import java.io.StreamTokenizer;
import java.util.Hashtable;
import java.util.Vector;

/**
 * This class will parse a dotty file and construct a tree structure from it
 * with Edge's and Node's
 * 
 * @author Malcolm Ware ([email protected])
 * @version $Revision: 14490 $
 */
public class TreeBuild {
  // this class will parse the tree into relevant strings
  // into info objects
  // from there it will create the nodes and edges from the info objects

  /** The name of the tree, Not in use. */
  // private String m_graphName; NOT USED

  /** An array with all the nodes initially constructed into it. */
  private Vector m_aNodes;

  /** An array with all the edges initially constructed into it. */
  private Vector m_aEdges;

  /**
   * An array containing a structure that describes the node without actually
   * creating it.
   */
  private Vector m_nodes;

  /**
   * An arry containing a structure that describes the edge without actually
   * creating it.
   */
  private Vector m_edges;

  /** An object setup to take graph data. */
  private InfoObject m_grObj;

  /** An object setup to take node data. */
  private InfoObject m_noObj;

  /** An object setup to take edge data. */
  private InfoObject m_edObj;

  /** true if it is a digraph. (note that this can't build digraphs). */
  // private boolean m_digraph; NOT USED

  /** The stream to parse. */
  private StreamTokenizer m_st;

  /** A table containing all the colors. */
  private final Hashtable m_colorTable;

  /**
   * Upon construction this will only setup the color table for quick reference
   * of a color.
   */
  public TreeBuild() {
    m_colorTable = new Hashtable();

    Colors ab = new Colors();
    for (NamedColor m_col : ab.m_cols) {
      m_colorTable.put(m_col.m_name, m_col.m_col);
    }
  }

  /**
   * This will build A node structure from the dotty format passed. Don't send a
   * dotty format with multiple parents per node, and ensure that there is 1 and
   * only 1 node with no parent.
   * 
   * @param t The reader with the dotty string to be read.
   * @return The top node of the tree structure (the last node with no parent).
   */
  public Node create(Reader t) {
    m_nodes = new Vector(50, 50);
    m_edges = new Vector(50, 50);
    m_grObj = new InfoObject("graph");
    m_noObj = new InfoObject("node");
    m_edObj = new InfoObject("edge");
    // m_digraph = false; NOT USED

    m_st = new StreamTokenizer(new BufferedReader(t));
    setSyntax();

    graph();

    Node top = generateStructures();

    return top;
  }

  /**
   * This will go through all the found Nodes and Edges build instances of these
   * and link them together.
   * 
   * @return The node with no parent (the top of the tree).
   */
  private Node generateStructures() {
    String id, label; // ,source,target; NOT USED
    Integer shape, style;
    // int fontsize; NOT USED
    Color fontcolor, color;

    InfoObject t;
    m_aNodes = new Vector(50, 50);
    m_aEdges = new Vector(50, 50);
    for (int noa = 0; noa < m_nodes.size(); noa++) {
      t = m_nodes.elementAt(noa);
      id = t.m_id;

      if (t.m_label == null) {
        if (m_noObj.m_label == null) {
          label = "";
        } else {
          label = m_noObj.m_label;
        }
      } else {
        label = t.m_label;
      }

      if (t.m_shape == null) {
        if (m_noObj.m_shape == null) {
          shape = new Integer(2);
        } else {
          shape = getShape(m_noObj.m_shape);
        }
      } else {
        shape = getShape(t.m_shape);
      }
      if (shape == null) {
        shape = new Integer(2);
      }

      if (t.m_style == null) {
        if (m_noObj.m_style == null) {
          style = new Integer(1);
        } else {
          style = getStyle(m_noObj.m_style);
        }
      } else {
        style = getStyle(t.m_style);
      }
      if (style == null) {
        style = new Integer(1);
      }

      /*
       * NOT USED if (t.m_fontSize == null) { if (m_noObj.m_fontSize == null) {
       * fontsize = 12; } else { fontsize =
       * Integer.valueOf(m_noObj.m_fontSize).intValue(); } } else { fontsize =
       * Integer.valueOf(t.m_fontSize).intValue(); }
       */

      if (t.m_fontColor == null) {
        if (m_noObj.m_fontColor == null) {
          fontcolor = Color.black;
        } else {
          fontcolor = m_colorTable.get(m_noObj.m_fontColor.toLowerCase());
        }
      } else {
        fontcolor = m_colorTable.get(t.m_fontColor.toLowerCase());
      }
      if (fontcolor == null) {
        fontcolor = Color.black;
      }

      if (t.m_color == null) {
        if (m_noObj.m_color == null) {
          color = Color.lightGray;
        } else {
          color = m_colorTable.get(m_noObj.m_color.toLowerCase());
        }
      } else {
        color = m_colorTable.get(t.m_color.toLowerCase());
      }
      if (color == null) {
        color = Color.lightGray;
      }

      m_aNodes.addElement(new Node(label, id, style.intValue(), shape
        .intValue(), color, t.m_data));
    }

    for (int noa = 0; noa < m_edges.size(); noa++) {
      t = m_edges.elementAt(noa);
      id = t.m_id;

      if (t.m_label == null) {
        if (m_noObj.m_label == null) {
          label = "";
        } else {
          label = m_noObj.m_label;
        }
      } else {
        label = t.m_label;
      }

      if (t.m_shape == null) {
        if (m_noObj.m_shape == null) {
          shape = new Integer(2);
        } else {
          shape = getShape(m_noObj.m_shape);
        }
      } else {
        shape = getShape(t.m_shape);
      }
      if (shape == null) {
        shape = new Integer(2);
      }

      if (t.m_style == null) {
        if (m_noObj.m_style == null) {
          style = new Integer(1);
        } else {
          style = getStyle(m_noObj.m_style);
        }
      } else {
        style = getStyle(t.m_style);
      }
      if (style == null) {
        style = new Integer(1);
      }

      /*
       * NOT USED if (t.m_fontSize == null) { if (m_noObj.m_fontSize == null) {
       * fontsize = 12; NOT USEDa } else { fontsize =
       * Integer.valueOf(m_noObj.m_fontSize).intValue(); NOT USED } } else {
       * fontsize = Integer.valueOf(t.m_fontSize).intValue(); }
       */

      if (t.m_fontColor == null) {
        if (m_noObj.m_fontColor == null) {
          fontcolor = Color.black;
        } else {
          fontcolor = m_colorTable.get(m_noObj.m_fontColor.toLowerCase());
        }
      } else {
        fontcolor = m_colorTable.get(t.m_fontColor.toLowerCase());
      }
      if (fontcolor == null) {
        fontcolor = Color.black;
      }

      if (t.m_color == null) {
        if (m_noObj.m_color == null) {
          color = Color.white;
        } else {
          color = m_colorTable.get(m_noObj.m_color.toLowerCase());
        }
      } else {
        color = m_colorTable.get(t.m_color.toLowerCase());
      }
      if (color == null) {
        color = Color.white;
      }

      m_aEdges.addElement(new Edge(label, t.m_source, t.m_target));
    }

    boolean f_set, s_set;
    Node x, sour = null, targ = null;
    Edge y;
    for (int noa = 0; noa < m_aEdges.size(); noa++) {
      f_set = false;
      s_set = false;
      y = m_aEdges.elementAt(noa);
      for (int nob = 0; nob < m_aNodes.size(); nob++) {
        x = m_aNodes.elementAt(nob);
        if (x.getRefer().equals(y.getRtarget())) {
          f_set = true;
          targ = x;
        }
        if (x.getRefer().equals(y.getRsource())) {
          s_set = true;
          sour = x;
        }
        if (f_set == true && s_set == true) {
          break;
        }
      }
      if (targ != sour) {
        y.setTarget(targ);
        y.setSource(sour);
      } else {
        System.out.println("logic error");
      }
    }

    for (int noa = 0; noa < m_aNodes.size(); noa++) {
      if (m_aNodes.elementAt(noa).getParent(0) == null) {
        sour = m_aNodes.elementAt(noa);
      }
    }

    return sour;
  }

  /**
   * This will convert the shape string to an int representing that shape.
   * 
   * @param sh The name of the shape.
   * @return An Integer representing the shape.
   */
  private Integer getShape(String sh) {
    if (sh.equalsIgnoreCase("box") || sh.equalsIgnoreCase("rectangle")) {
      return new Integer(1);
    } else if (sh.equalsIgnoreCase("oval")) {
      return new Integer(2);
    } else if (sh.equalsIgnoreCase("diamond")) {
      return new Integer(3);
    } else {
      return null;
    }
  }

  /**
   * Converts the string representing the fill style int oa number representing
   * it.
   * 
   * @param sty The name of the style.
   * @return An Integer representing the shape.
   */
  private Integer getStyle(String sty) {
    if (sty.equalsIgnoreCase("filled")) {
      return new Integer(1);
    } else {
      return null;
    }
  }

  /**
   * This will setup the syntax for the tokenizer so that it parses the string
   * properly.
   * 
   */
  private void setSyntax() {
    m_st.resetSyntax();
    m_st.eolIsSignificant(false);
    m_st.slashStarComments(true);
    m_st.slashSlashComments(true);
    // System.out.println("slash");
    m_st.whitespaceChars(0, ' ');
    m_st.wordChars(' ' + 1, '\u00ff');
    m_st.ordinaryChar('[');
    m_st.ordinaryChar(']');
    m_st.ordinaryChar('{');
    m_st.ordinaryChar('}');
    m_st.ordinaryChar('-');
    m_st.ordinaryChar('>');
    m_st.ordinaryChar('/');
    m_st.ordinaryChar('*');
    m_st.quoteChar('"');
    m_st.whitespaceChars(';', ';');
    m_st.ordinaryChar('=');
  }

  /**
   * This is the alternative syntax for the tokenizer.
   */
  private void alterSyntax() {
    m_st.resetSyntax();
    m_st.wordChars('\u0000', '\u00ff');
    m_st.slashStarComments(false);
    m_st.slashSlashComments(false);
    m_st.ordinaryChar('\n');
    m_st.ordinaryChar('\r');
  }

  /**
   * This will parse the next token out of the stream and check for certain
   * conditions.
   * 
   * @param r The error string to print out if something goes wrong.
   */
  private void nextToken(String r) {
    int t = 0;
    try {
      t = m_st.nextToken();
    } catch (IOException e) {
    }

    if (t == StreamTokenizer.TT_EOF) {
      System.out.println("eof , " + r);
    } else if (t == StreamTokenizer.TT_NUMBER) {
      System.out.println("got a number , " + r);
    }
  }

  /**
   * Parses the top of the dotty stream that has the graph information.
   * 
   */
  private void graph() {
    nextToken("expected 'digraph'");

    if (m_st.sval.equalsIgnoreCase("digraph")) {
      // m_digraph = true; NOT USED
    } else {
      System.out.println("expected 'digraph'");
    }

    nextToken("expected a Graph Name");
    if (m_st.sval != null) {
      // m_graphName = m_st.sval; NOT USED
    } else {
      System.out.println("expected a Graph Name");
    }

    nextToken("expected '{'");

    if (m_st.ttype == '{') {
      stmtList();
    } else {
      System.out.println("expected '{'");
    }
  }

  /**
   * This is one of the states, this one is where new items can be defined or
   * the structure can end.
   * 
   */
  private void stmtList() {
    boolean flag = true;
    while (flag) {
      nextToken("expects a STMT_LIST item or '}'");
      if (m_st.ttype == '}') {
        flag = false;
      } else if (m_st.sval.equalsIgnoreCase("graph")
        || m_st.sval.equalsIgnoreCase("node")
        || m_st.sval.equalsIgnoreCase("edge")) {
        m_st.pushBack();
        attrStmt();
      } else {
        nodeId(m_st.sval, 0);
      }
    }
  }

  /**
   * This will deal specifically with a new object such as graph , node , edge.
   * 
   */
  private void attrStmt() {

    nextToken("expected 'graph' or 'node' or 'edge'");

    if (m_st.sval.equalsIgnoreCase("graph")) {
      nextToken("expected a '['");
      if (m_st.ttype == '[') {
        attrList(m_grObj);
      } else {
        System.out.println("expected a '['");
      }
    } else if (m_st.sval.equalsIgnoreCase("node")) {
      nextToken("expected a '['");
      if (m_st.ttype == '[') {
        attrList(m_noObj);
      } else {
        System.out.println("expected a '['");
      }
    } else if (m_st.sval.equalsIgnoreCase("edge")) {
      nextToken("expected a '['");
      if (m_st.ttype == '[') {
        attrList(m_edObj);
      } else {
        System.out.println("expected a '['");
      }

    } else {
      System.out.println("expected 'graph' or 'node' or 'edge'");
    }
  }

  /**
   * Generates a new InfoObject with the specified name and either does further
   * processing if applicable Otherwise it is an edge and will deal with that.
   * 
   * @param s The ID string.
   * @param t Not sure!.
   */
  private void nodeId(String s, int t) {

    nextToken("error occurred in node_id");

    if (m_st.ttype == '}') {
      // creates a node if t is zero
      if (t == 0) {
        m_nodes.addElement(new InfoObject(s));
      }
      m_st.pushBack();
    } else if (m_st.ttype == '-') {
      nextToken("error occurred checking for an edge");
      if (m_st.ttype == '>') {
        edgeStmt(s);
      } else {
        System.out.println("error occurred checking for an edge");
      }
    } else if (m_st.ttype == '[') {
      // creates a node if t is zero and sends it to attr
      if (t == 0) {
        m_nodes.addElement(new InfoObject(s));
        attrList(m_nodes.lastElement());
      } else {
        attrList(m_edges.lastElement());
      }
    } else if (m_st.sval != null) {
      // creates a node if t is zero
      if (t == 0) {
        m_nodes.addElement(new InfoObject(s));
      }
      m_st.pushBack();
    } else {
      System.out.println("error occurred in node_id");
    }
  }

  /**
   * This will get the target of the edge.
   * 
   * @param i The source of the edge.
   */
  private void edgeStmt(String i) {
    nextToken("error getting target of edge");

    if (m_st.sval != null) {
      m_edges.addElement(new InfoObject("an edge ,no id"));
      m_edges.lastElement().m_source = i;
      m_edges.lastElement().m_target = m_st.sval;
      nodeId(m_st.sval, 1);
    } else {
      System.out.println("error getting target of edge");
    }
  }

  /**
   * This will parse all the items in the attrib list for an object.
   * 
   * @param a The object that the attribs apply to.
   */
  private void attrList(InfoObject a) {
    boolean flag = true;

    while (flag) {
      nextToken("error in attr_list");
      // System.out.println(st.sval);
      if (m_st.ttype == ']') {
        flag = false;
      } else if (m_st.sval.equalsIgnoreCase("color")) {
        nextToken("error getting color");
        if (m_st.ttype == '=') {
          nextToken("error getting color");
          if (m_st.sval != null) {
            a.m_color = m_st.sval;
          } else {
            System.out.println("error getting color");
          }
        } else {
          System.out.println("error getting color");
        }
      } else if (m_st.sval.equalsIgnoreCase("fontcolor")) {
        nextToken("error getting font color");
        if (m_st.ttype == '=') {
          nextToken("error getting font color");
          if (m_st.sval != null) {
            a.m_fontColor = m_st.sval;
          } else {
            System.out.println("error getting font color");
          }
        } else {
          System.out.println("error getting font color");
        }
      } else if (m_st.sval.equalsIgnoreCase("fontsize")) {
        nextToken("error getting font size");
        if (m_st.ttype == '=') {
          nextToken("error getting font size");
          if (m_st.sval != null) {
          } else {
            System.out.println("error getting font size");
          }
        } else {
          System.out.println("error getting font size");
        }
      } else if (m_st.sval.equalsIgnoreCase("label")) {
        nextToken("error getting label");
        if (m_st.ttype == '=') {
          nextToken("error getting label");
          if (m_st.sval != null) {
            a.m_label = m_st.sval;
          } else {
            System.out.println("error getting label");
          }
        } else {
          System.out.println("error getting label");
        }
      } else if (m_st.sval.equalsIgnoreCase("shape")) {
        nextToken("error getting shape");
        if (m_st.ttype == '=') {
          nextToken("error getting shape");
          if (m_st.sval != null) {
            a.m_shape = m_st.sval;
          } else {
            System.out.println("error getting shape");
          }
        } else {
          System.out.println("error getting shape");
        }
      } else if (m_st.sval.equalsIgnoreCase("style")) {
        nextToken("error getting style");
        if (m_st.ttype == '=') {
          nextToken("error getting style");
          if (m_st.sval != null) {
            a.m_style = m_st.sval;
          } else {
            System.out.println("error getting style");
          }
        } else {
          System.out.println("error getting style");
        }
      } else if (m_st.sval.equalsIgnoreCase("data")) {
        nextToken("error getting data");
        if (m_st.ttype == '=') {
          // data has a special data string that can have anything
          // this is delimited by a single comma on an otherwise empty line
          alterSyntax();
          a.m_data = new String("");

          while (true) {
            nextToken("error getting data");
            if (m_st.sval != null && a.m_data != null && m_st.sval.equals(",")) {
              break;
            } else if (m_st.sval != null) {
              a.m_data = a.m_data.concat(m_st.sval);
            } else if (m_st.ttype == '\r') {
              a.m_data = a.m_data.concat("\r");
            } else if (m_st.ttype == '\n') {
              a.m_data = a.m_data.concat("\n");
            } else {
              System.out.println("error getting data");
            }
          }
          setSyntax();
        } else {
          System.out.println("error getting data");
        }
      }
    }
  }

  // special class for use in creating the tree

  /**
   * This is an internal class used to keep track of the info for the objects
   * before they are actually created.
   */
  private class InfoObject {
    /** The ID string for th object. */
    public String m_id;

    /** The color name for the object. */
    public String m_color;

    /** The font color for the object. not in use. */
    public String m_fontColor;

    /** The label for the object. */
    public String m_label;

    /** The shape name of for the object. */
    public String m_shape;

    /** The backstyle name for the object. */
    public String m_style;

    /** The source ID of the object. */
    public String m_source;

    /** The target ID of the object. */
    public String m_target;

    /** The data for this object. */
    public String m_data;

    /**
     * This will construct a new InfoObject with the specified ID string.
     */
    public InfoObject(String i) {
      m_id = i;
      m_color = null;
      m_fontColor = null;
      m_label = null;
      m_shape = null;
      m_style = null;
      m_source = null;
      m_target = null;
      m_data = null;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy