Swing:绘制渐变的饼图

 

GeometryUtil.java

import java.awt.geom.Point2D;
 
public class GeometryUtil {

    // 两点之间的距离

    public static double distanceOfPoints(Point2D p1, Point2D p2) {

        double disX = p2.getX() - p1.getX();

        double disY = p2.getY() - p1.getY();

        double dis = Math.sqrt(disX * disX + disY * disY);
 
        return dis;

    }
 
    // 两点的中点

    public static Point2D middlePoint(Point2D p1, Point2D p2) {

        double x = (p1.getX() + p2.getX()) / 2;

        double y = (p1.getY() + p2.getY()) / 2;
 
        return new Point2D.Double(x, y);

    } 

    // 在两点所在直线上,以从startPoint到endPoint为方向,离startPoint的距离disToStartPoint的点

    public static Point2D extentPoint(Point2D startPoint, Point2D endPoint, double disToStartPoint) {

        double disX = endPoint.getX() - startPoint.getX();

        double disY = endPoint.getY() - startPoint.getY();

        double dis = Math.sqrt(disX * disX + disY * disY);

        double sin = (endPoint.getY() - startPoint.getY()) / dis;

        double cos = (endPoint.getX() - startPoint.getX()) / dis;

        double deltaX = disToStartPoint * cos;

        double deltaY = disToStartPoint * sin;
 
        return new Point2D.Double(startPoint.getX() + deltaX, startPoint.getY() + deltaY);

    }

}


Pie3D.java

import java.awt.Color;
import java.awt.geom.Arc2D;
import java.awt.geom.Area;
import java.awt.geom.GeneralPath;
import java.awt.geom.Point2D;
class Pie3D {
    private Arc2D arc; // 这里的弧并不是圆上的一弧,而是椭圆的一部分.
    private Area frontSite;
    private Area leftSite;
    private Area rightSite;
    private Color color;
    private Pie3D selectedPie;
    private Point2D arcMiddle;
    private Point2D labelPosition;
    private double value;
    private int shadowDepth;
    private double selectedShiftDis; // 被选中的饼图在他的中线上移动的距离
    public Pie3D(Arc2D arc, Color color, double value) {
        this(arc, color, value, 10, 30);
    }


   public Pie3D(Arc2D arc, Color color, double value, int shadowDepth, double selectedShiftDis) {
        this.arc = arc;
        this.color = color;
        this.value = value;
        this.selectedShiftDis = selectedShiftDis;
        this.shadowDepth = shadowDepth;


       Arc2D arcBottom = new Arc2D.Double(arc.getX(), arc.getY() + shadowDepth, arc.getWidth(),
            arc.getHeight() + 0, arc.getAngleStart(), arc.getAngleExtent(), Arc2D.CHORD);
        Point2D[] topPs = getPointsOfArc(arc);
        Point2D[] bottomPs = getPointsOfArc(arcBottom);


        // Front site
        GeneralPath font = new GeneralPath();
        font.moveTo(topPs[1].getX(), topPs[1].getY());
        font.lineTo(topPs[2].getX(), topPs[2].getY());
        font.lineTo(bottomPs[2].getX(), bottomPs[2].getY());
        font.lineTo(bottomPs[1].getX(), bottomPs[1].getY());
        font.closePath();
        this.frontSite = new Area(arcBottom);
        this.frontSite.add(new Area(font));


       // Left site
        GeneralPath left = new GeneralPath();
        left.moveTo(topPs[0].getX(), topPs[0].getY());
        left.lineTo(topPs[1].getX(), topPs[1].getY());
        left.lineTo(bottomPs[1].getX(), bottomPs[1].getY());
        left.lineTo(topPs[0].getX(), topPs[0].getY() + 3);
        left.closePath();
        this.leftSite = new Area(left);


       // Right site
        GeneralPath right = new GeneralPath();
        right.moveTo(topPs[0].getX(), topPs[0].getY());
        right.lineTo(topPs[2].getX(), topPs[2].getY());
        right.lineTo(bottomPs[2].getX(), bottomPs[2].getY());
        right.lineTo(topPs[0].getX(), topPs[0].getY() + 3);
        right.closePath();
        this.rightSite = new Area(right);

        arcMiddle = calculateArcMiddle();


       // Label position: 五分之四处
        Point2D c = getPieCenter();
        // Point2D m = getChordMiddle();
        Point2D m = arcMiddle;
        double dis = GeometryUtil.distanceOfPoints(c, m) * 0.8;
        labelPosition = GeometryUtil.extentPoint(c, m, dis);
    }


    // 取得Arc上的三个点,在对Arc: center, left, right.
    public static Point2D[] getPointsOfArc(Arc2D arc) {
        Point2D center = new Point2D.Double(arc.getCenterX(), arc.getCenterY());
        Point2D left = arc.getStartPoint();
        Point2D right = arc.getEndPoint();
        Point2D[] points = new Point2D[] { center, left, right };


       return points;
    }


   public Pie3D getSelectedPie() {
        if (selectedPie == null) {

           selectedPie = createSeletecdPie();
        }


       return selectedPie;
    }


   private Pie3D createSeletecdPie() {
        // 沿中线方向移动selectedShiftDis个单位
        Point2D c = getPieCenter();
        Point2D m = getChordMiddle();
        Point2D p = GeometryUtil.extentPoint(c, m, selectedShiftDis);


       double deltaX = p.getX() - c.getX();
        double deltaY = p.getY() - c.getY();
        double x = arc.getX() + deltaX;
        double y = arc.getY() + deltaY;


       Arc2D shiftArc = (Arc2D) arc.clone();
        shiftArc.setFrame(x, y, arc.getWidth(), arc.getHeight());


       return new Pie3D(shiftArc, color, value, shadowDepth, selectedShiftDis);
    }


   public Arc2D getArc() {
        return arc;
    }


   public Area getFrontSite() {
        return frontSite;
    }


   public Area getLeftSite() {
        return leftSite;
    }


   public Area getRightSite() {
        return rightSite;
    }


   public Color getColor() {
        return color;
    }

    public void setColor(Color color) {
        this.color = color;

   }

    public Point2D getLabelPosition() {
        return labelPosition;

  }


    public void setLabelPosition(Point2D labelPosition) {
       this.labelPosition = labelPosition;
    }


   public double getValue() {
        return value;
    }


   public String getLabel() {
        return value + "%";
    }


   // 弦的中心点
    public Point2D getChordMiddle() {
        return GeometryUtil.middlePoint(arc.getStartPoint(), arc.getEndPoint());
    }


   // 饼图的圆心
    public Point2D getPieCenter() {
        return new Point2D.Double(arc.getCenterX(), arc.getCenterY());
    }


   // 弧上的中心点
    public Point2D getArcMiddle() {
        return arcMiddle;
    }


   private Point2D calculateArcMiddle() {
        // 创建一个新的弧,其扩展角度为当前弧的一半
        return new Arc2D.Double(arc.getX(), arc.getY(), arc.getWidth(), arc.getHeight(),
            arc.getAngleStart(), arc.getAngleExtent() / 2, Arc2D.PIE).getEndPoint();
    }
}


PieGradientPainter.java进行测试

import java.awt.Color;
import java.awt.FontMetrics;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.LinearGradientPaint;
import java.awt.RadialGradientPaint;
import java.awt.RenderingHints;
import java.awt.Toolkit;
import java.awt.geom.Arc2D;
import java.awt.geom.Point2D;
import java.util.ArrayList;
import java.util.List;
import javax.swing.JFrame;
import javax.swing.JPanel;


@SuppressWarnings("serial")
public class PieGradientPainter extends JPanel {
    private double[] data;
    private Color[] colors = createColors();
    private Pie3D[] pies;
    private Pie3D[] outerPies;


   public PieGradientPainter() {
        setBackground(Color.DARK_GRAY);
        data = new double[] { 10, 30, 40, 15, 25, 60 };
       int x = 0;
        int y = 0;
        int w = 300;
        int h = 300;
        int shiftAngle = -30;
        int delta = 40;
        outerPies = createPies(x, y, w, h, 0, shiftAngle, data, colors);


       x += delta / 2;
        y += delta / 2;
        w -= delta;
        h -= delta;
        pies = createPies(x, y, w, h, 0, shiftAngle, data, colors);
    }


   @Override
    protected void paintComponent(Graphics g) {
        super.paintComponent(g);
        Graphics2D g2d = (Graphics2D) g;
        g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);


       int paddingLeft = 30;
        drawLinearPie(g2d, pies, outerPies, paddingLeft, 30);


       paddingLeft = (int) (outerPies[0].getArc().getWidth()) + paddingLeft * 2;
        drawRadialPie(g2d, pies, outerPies, paddingLeft, 30);
    }


   protected void drawLinearPie(Graphics2D g2d,
                                 Pie3D[] pies,
                                 Pie3D[] outerPies,
                                 int paddingLeft,
                                 int paddingTop) {
        g2d.translate(paddingLeft, paddingTop);
        int colorIndex = colors.length - 1;
        double radius = pies[0].getArc().getWidth() / 2;
        float[] fractions = { 0.0f, 0.55f, 1.0f };
        Point2D start = new Point2D.Double(0, pies[0].getPieCenter().getY() - radius);
        Point2D end = new Point2D.Double(0, pies[0].getPieCenter().getY() + radius * 1.5);


       for (int i = 0; i < pies.length; ++i) {
            Pie3D pie = pies[i];

            colorIndex = (colorIndex + 1) % colors.length;
            Color c = colors[colorIndex];


           g2d.setColor(c);
            g2d.fill(outerPies[i].getArc());


           // Linear gradiant paint.
            Color[] cs = { c.darker().darker(), c, Color.WHITE.darker() };
            LinearGradientPaint paint = new LinearGradientPaint(start, end, fractions, cs);
            g2d.setPaint(paint);
            g2d.fill(pie.getArc());
        }


       drawPieLabel(g2d, pies);
        g2d.translate(-paddingLeft, -paddingTop);
    }
    protected void drawRadialPie(Graphics2D g2d,
                                 Pie3D[] pies,

                                Pie3D[] outerPies,
                                 int paddingLeft,
                                 int paddingTop) {
        g2d.translate(paddingLeft, paddingTop);
        int colorIndex = colors.length - 1;
        Point2D center = pies[0].getPieCenter();


       int radius = (int) pies[0].getArc().getWidth();
        float[] fractions = { 0.0f, 0.2f, 1.0f };


      for (int i = 0; i < pies.length; ++i) {
            Pie3D pie = pies[i];


            colorIndex = (colorIndex + 1) % colors.length;
           Color c = colors[colorIndex];


            g2d.setColor(c);
            g2d.fill(outerPies[i].getArc());

            // Radial gradient paint.

           Color[] cc = { c.brighter(), c, c.darker().darker().darker() };
            RadialGradientPaint paint = new RadialGradientPaint(center, radius, fractions, cc);
           g2d.setPaint(paint);

          g2d.fill(pie.getArc());
        }


       drawPieLabel(g2d, pies);
        g2d.translate(-paddingLeft, -paddingTop);
    }


   protected void drawPieLabel(Graphics2D g2d, Pie3D[] pies) {
        FontMetrics metrics = g2d.getFontMetrics();
        g2d.setColor(Color.BLACK);


       for (int i = 0; i < pies.length; ++i) {
            Pie3D p = pies[i];
            int sw = metrics.stringWidth(p.getLabel()) / 2;
            int sh = (metrics.getAscent()) / 2;
            int x = (int) (p.getLabelPosition().getX() - sw);
            int y = (int) (p.getLabelPosition().getY() + sh);
            g2d.drawString(p.getLabel(), x, y);
        }
    }


   private Color[] createColors() {

        // 返回16进制的值颜色

        List<Color> colors = new ArrayList<Color>();

        colors.add(Color.decode("#635D49"));

        colors.add(Color.decode("#4D7B20"));

        colors.add(Color.decode("#FF7321"));

        colors.add(Color.decode("#BFDD89"));

        colors.add(Color.decode("#AA6A2D"));

        colors.add(Color.decode("#9C1594"));

        colors.add(Color.decode("#00E500"));

        colors.add(Color.decode("#E2FF55"));

        colors.add(Color.decode("#D718A5"));

        colors.add(Color.decode("#BB2100"));

        colors.add(Color.decode("#D0F15A"));

        colors.add(Color.decode("#169800"));

        colors.add(Color.decode("#00DBFF"));

        colors.add(Color.decode("#00FF00"));




        return colors.toArray(new Color[0]);

    }




    public static Pie3D[] createPies(int x,

                                     int y,

                                     int w,

                                     int h,

                                     int shadowDepth,

                                     int shiftAngle,

                                     double[] data,

                                     Color[] colors) {

        double sum = 0;

        for (double d : data) {

            sum += d;

        }




        // 初始化Pies

        double arcAngle = 0;

        double startAngle = shiftAngle;

        Pie3D[] pies = new Pie3D[data.length];




        for (int i = 0; i < data.length; ++i) {

            arcAngle = data[i] * 360 / sum; // 使用百分比计算角度

            if (i + 1 == data.length) {

                arcAngle = 360 + shiftAngle - startAngle; // 保证闭合

                arcAngle = arcAngle > 0 ? arcAngle : 0;

            }




            Arc2D.Double arc = new Arc2D.Double(x, y, w, h, startAngle, arcAngle, Arc2D.PIE);




            double rate = data[i] / sum * 100 * 100 + 0.5;

            rate = ((int) rate) / 100.0;




            pies[i] = new Pie3D(arc, colors[i % colors.length], rate, shadowDepth, 30);

            startAngle += arcAngle;

        }




        return pies;

    }




    private static void createGuiAndShow() {

        JFrame frame = new JFrame("Pie with gradient effects");

        frame.getContentPane().add(new PieGradientPainter());




        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        int sw = Toolkit.getDefaultToolkit().getScreenSize().width;

        int sh = Toolkit.getDefaultToolkit().getScreenSize().height;

        int w = 690;

        int h = 390;

        int x = (sw - w) / 2;

        int y = (sh - h) / 2 - 40;

        x = x > 0 ? x : 0;

        y = y > 0 ? y : 0;

        frame.setBounds(x, y, w, h);

        frame.setVisible(true);

    }




    public static void main(String[] args) {

        createGuiAndShow();

    }

}


 

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值