summaryrefslogtreecommitdiff
path: root/src/main/java/org/cristalise/gui/graph/view/DefaultDirectedEdgeRenderer.java
blob: 76958856b09513d7a666b26dd8aa2e0fce8dc3bf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
package org.cristalise.gui.graph.view;

import java.awt.Graphics2D;
import java.awt.Paint;
import java.awt.Shape;
import java.awt.geom.AffineTransform;
import java.awt.geom.GeneralPath;

import org.cristalise.kernel.graph.model.DirectedEdge;
import org.cristalise.kernel.graph.model.GraphPoint;



public class DefaultDirectedEdgeRenderer implements DirectedEdgeRenderer
{
    private GeneralPath mArrowTemplate = new GeneralPath();
    private Paint       mLinePaint     = null;


    public DefaultDirectedEdgeRenderer(Paint linePaint)
    {
        mLinePaint = linePaint;

        mArrowTemplate.moveTo(-5, 5);
        mArrowTemplate.lineTo( 0, 0);
        mArrowTemplate.lineTo( 5, 5);
    }


    @Override
	public void draw(Graphics2D g2d, DirectedEdge directedEdge)
    {
        GraphPoint      originPoint   = directedEdge.getOriginPoint();
        GraphPoint      terminusPoint = directedEdge.getTerminusPoint();
        GraphPoint      midPoint      = new GraphPoint();
        AffineTransform transform     = new AffineTransform();
        Shape           arrow         = null;

        g2d.setPaint(mLinePaint);
        g2d.drawLine(originPoint.x, originPoint.y, terminusPoint.x, terminusPoint.y);

        midPoint.x = originPoint.x + (terminusPoint.x - originPoint.x)/2;
        midPoint.y = originPoint.y + (terminusPoint.y - originPoint.y)/2;

        transform.translate(midPoint.x, midPoint.y);
        transform.rotate(calcArrowAngle(originPoint.x, originPoint.y, terminusPoint.x, terminusPoint.y));

        arrow = mArrowTemplate.createTransformedShape(transform);

        g2d.draw(arrow);
    }


    private static double calcArrowAngle(int originX, int originY, int terminusX, int terminusY) {
        double width  = terminusX - originX;
        double height = terminusY - originY;

        if((width == 0) && (height > 0))  return Math.PI;

        if((width == 0) && (height < 0))  return 0;

        if((width > 0)  && (height == 0)) return Math.PI/2.0;

        if((width < 0)  && (height == 0)) return -1.0 * Math.PI/2.0;

        if((width > 0)  && (height > 0))  return Math.PI/2.0 + Math.atan(Math.abs(height)/Math.abs(width));

        if((width > 0)  && (height < 0))  return Math.atan(Math.abs(width)/Math.abs(height));

        if((width < 0)  && (height < 0))  return -1.0 * Math.atan(Math.abs(width)/Math.abs(height));

        if((width < 0)  && (height > 0))  return -1.0 * (Math.PI/2.0 + Math.atan(Math.abs(height)/Math.abs(width)));

        return 0.0;
    }
}