Package prefuse.action.layout.graph

Source Code of prefuse.action.layout.graph.BalloonTreeLayout

package prefuse.action.layout.graph;

import java.awt.geom.Point2D;
import java.util.Iterator;

import prefuse.data.Graph;
import prefuse.data.Schema;
import prefuse.data.tuple.TupleSet;
import prefuse.visual.NodeItem;


/**
* <p>Layout that computes a circular "balloon-tree" layout of a tree.
* This layout places children nodes radially around their parents, and is
* equivalent to a top-down flattened view of a ConeTree.</p>
*
* <p>The algorithm used is that of G. Melan\c{c}on and I. Herman from their
* research paper Circular Drawings of Rooted Trees, Reports of the Centre for
* Mathematics and Computer Sciences, Report Number INS-9817, 1998.</p>
*
* @author <a href="http://jheer.org">jeffrey heer</a>
*/
public class BalloonTreeLayout extends TreeLayout {

    private int m_minRadius = 2;
   
    /**
     * Create a new BalloonTreeLayout
     * @param group the data group to layout. Must resolve to a Graph
     * or Tree instance.
     */
    public BalloonTreeLayout(String group) {
        this(group, 2);
    }

    /**
     * Create a new BalloonTreeLayout
     * @param group the data group to layout. Must resolve to a Graph
     * or Tree instance.
     * @param minRadius the minimum radius to use for a layout circle
     */
    public BalloonTreeLayout(String group, int minRadius) {
        super(group);
        m_minRadius = minRadius;
    }

    /**
     * Get the minimum radius used for a layout circle.
     * @return the minimum layout radius
     */
    public int getMinRadius() {
        return m_minRadius;
    }
   
    /**
     * Set the minimum radius used for a layout circle.
     * @param minRadius the minimum layout radius
     */
    public void setMinRadius(int minRadius) {
        m_minRadius = minRadius;
    }
   
    /**
     * @see prefuse.action.Action#run(double)
     */
    public void run(double frac) {
        Graph g = (Graph)m_vis.getGroup(m_group);
        initSchema(g.getNodes());
       
        Point2D anchor = getLayoutAnchor();
        NodeItem n = getLayoutRoot();
        layout(n,anchor.getX(),anchor.getY());
    }
   
    private void layout(NodeItem n, double x, double y) {
        firstWalk(n);
        secondWalk(n,null,x,y,1,0);
    }
   
    private void firstWalk(NodeItem n) {
        Params np = getParams(n);
        np.d = 0;
        double s = 0;
        Iterator childIter = n.children();
        while ( childIter.hasNext() ) {
            NodeItem c = (NodeItem)childIter.next();
            if ( !c.isVisible() ) continue;
            firstWalk(c);
            Params cp = getParams(c);
            np.d = Math.max(np.d,cp.r);
            cp.a = Math.atan(((double)cp.r)/(np.d+cp.r));
            s += cp.a;
        }
        adjustChildren(np, s);
        setRadius(np);
    }
   
    private void adjustChildren(Params np, double s) {
        if ( s > Math.PI ) {
            np.c = Math.PI/s;
            np.f = 0;
        } else {
            np.c = 1;
            np.f = Math.PI - s;
        }
    }
   
    private void setRadius(Params np) {
        np.r = Math.max(np.d,m_minRadius) + 2*np.d;
    }
   
/*
    private void setRadius(NodeItem n, ParamBlock np) {
        int numChildren = n.getChildCount();
        double p  = Math.PI;
        double fs = (numChildren==0 ? 0 : np.f/numChildren);
        double pr = 0;
        double bx = 0, by = 0;
        Iterator childIter = n.getChildren();
        while ( childIter.hasNext() ) {
            NodeItem c = (NodeItem)childIter.next();
            ParamBlock cp = getParams(c);
            p += pr + cp.a + fs;
            bx += (cp.r)*Math.cos(p);
            by += (cp.r)*Math.sin(p);
            pr = cp.a;
        }
        if ( numChildren != 0 ) {
            bx /= numChildren;
            by /= numChildren;
        }
        np.rx = -bx;
        np.ry = -by;
       
        p = Math.PI;
        pr = 0;
        np.r = 0;
        childIter = n.getChildren();
        while ( childIter.hasNext() ) {
            NodeItem c = (NodeItem)childIter.next();
            ParamBlock cp = getParams(c);
            p += pr + cp.a + fs;
            double x = cp.r*Math.cos(p)-bx;
            double y = cp.r*Math.sin(p)-by;
            double d = Math.sqrt(x*x+y*y) + cp.r;
            np.r = Math.max(np.r, (int)Math.round(d));
            pr = cp.a;
        }
        if ( np.r == 0 )
            np.r = m_minRadius + 2*np.d;
    } //
    private void secondWalk2(NodeItem n, NodeItem r,
            double x, double y, double l, double t)
    {
        ParamBlock np = getParams(n);
        double cost = Math.cos(t);
        double sint = Math.sin(t);
        double nx = x + l*(np.rx*cost-np.ry*sint);
        double ny = y + l*(np.rx*sint+np.ry*cost);
        setLocation(n,r,nx,ny);
        double dd = l*np.d;
        double p  = Math.PI;
        double fs = np.f / (n.getChildCount()+1);
        double pr = 0;
        Iterator childIter = n.getChildren();
        while ( childIter.hasNext() ) {
            NodeItem c = (NodeItem)childIter.next();
            ParamBlock cp = getParams(c);
            double aa = np.c * cp.a;
            double rr = np.d * Math.tan(aa)/(1-Math.tan(aa));
            p += pr + aa + fs;
            double xx = (l*rr+dd)*Math.cos(p)+np.rx;
            double yy = (l*rr+dd)*Math.sin(p)+np.ry;
            double x2 = xx*cost - yy*sint;
            double y2 = xx*sint + yy*cost;
            pr = aa;
            secondWalk2(c, n, x+x2, y+y2, l*rr/cp.r, p);
        }
    } //
*/
   
    private void secondWalk(NodeItem n, NodeItem r,
            double x, double y, double l, double t)
    {
        setX(n, r, x);
        setY(n, r, y);
       
        Params np = getParams(n);
        int numChildren = 0;
        Iterator childIter = n.children();
        while ( childIter.hasNext() ) {
            NodeItem c = (NodeItem)childIter.next();
            if ( c.isVisible() ) ++numChildren;
        }
        double dd = l*np.d;
        double p  = t + Math.PI;
        double fs = (numChildren==0 ? 0 : np.f/numChildren);
        double pr = 0;
        childIter = n.children();
        while ( childIter.hasNext() ) {
            NodeItem c = (NodeItem)childIter.next();
            if ( !c.isVisible() ) continue;
            Params cp = getParams(c);
            double aa = np.c * cp.a;
            double rr = np.d * Math.tan(aa)/(1-Math.tan(aa));
            p += pr + aa + fs;
            double xx = (l*rr+dd)*Math.cos(p);
            double yy = (l*rr+dd)*Math.sin(p);
            pr = aa;
            secondWalk(c, n, x+xx, y+yy, l*np.c/*l*rr/cp.r*/, p);
        }
    }
   
    // ------------------------------------------------------------------------
    // Parameters
   
    /**
     * The data field in which the parameters used by this layout are stored.
     */
    public static final String PARAMS = "_balloonTreeLayoutParams";
    /**
     * The schema for the parameters used by this layout.
     */
    public static final Schema PARAMS_SCHEMA = new Schema();
    static {
        PARAMS_SCHEMA.addColumn(PARAMS, Params.class);
    }
   
    private void initSchema(TupleSet ts) {
        try {
            ts.addColumns(PARAMS_SCHEMA);
        } catch ( IllegalArgumentException iae ) {}
    }
   
    private Params getParams(NodeItem n) {
        Params np = (Params)n.get(PARAMS);
        if ( np == null ) {
            np = new Params();
            n.set(PARAMS, np);
        }
        return np;
    }
   
    /**
     * Wrapper class holding parameters used for each node in this layout.
     */
    public static class Params {
        public int d;
        public int r;
        public double rx, ry;
        public double a;
        public double c;
        public double f;
    }

} // end of class BalloonTreeLayout
TOP

Related Classes of prefuse.action.layout.graph.BalloonTreeLayout

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.