jfreechart 学习笔记
1.java 2D主要绘图类描述(jdk1.5)
Graphics2D //主要绘图对象,每种图形都要通过该对象来绘制
Rectangle2D //长方形对象
Point2D //点对象
Line2D //线对象
Arc2D.Double //弧形对象
Ellipse2D //椭圆对象
Polygon //多边形对象
Paint //油漆桶对象,用来定义颜色
Stroke //画笔对象,主要用于描绘轮廓,定义线条样式
Area //几何建模对象(含几何并交差运算等方法)
GeneralPath //路径对象
1.1 sample demo eg:
/**
* 利用路径对象绘制平面图
*/
// GeneralPath bar3dRight = new GeneralPath();
// bar3dRight.moveTo((float) 100.0, (float) 100.0);
// bar3dRight.lineTo((float) 200.0, (float) 200.0);
// bar3dRight.lineTo((float) 300.0, (float) 100.0);
// bar3dRight.lineTo((float) 100.0, (float) 100.0);
// bar3dRight.closePath();
// 填充四边形的颜色
// if (itemPaint instanceof Color) {
// g2.setPaint(((Color) itemPaint).darker());
// }
/**
* 长方形对象加路径对象绘制3D柱状图
*/
//正视图
Rectangle2D r2d = new Rectangle2D.Double(
100.0, //x
100.0, //y
30.0, //width
200.0 //hight
);
//右视图
GeneralPath bar3dRight = new GeneralPath();
bar3dRight.moveTo((float) 130.0, (float) 100.0);
bar3dRight.lineTo((float) 150.0, (float) 90.0);
bar3dRight.lineTo((float) 150.0, (float) 290.0);
bar3dRight.lineTo((float) 130.0, (float) 300.0);
bar3dRight.closePath();
//俯视图
GeneralPath bar3dTop = new GeneralPath();
bar3dTop.moveTo((float) 100.0, (float) 100.0);
bar3dTop.lineTo((float) 130.0, (float) 100.0);
bar3dTop.lineTo((float) 150.0, (float) 90.0);
bar3dTop.lineTo((float) 120.0, (float) 90.0);
bar3dTop.closePath();
Paint barpaint = Color.PINK; // 实体颜色
Paint outlinepaint = Color.DARK_GRAY; // 轮廓线颜色
float dash1[] = { 10.0f };
// 虚线条描边
// Stroke stroke = new BasicStroke(1.0f,
// BasicStroke.CAP_BUTT,
// BasicStroke.JOIN_MITER,
// 10.0f, dash1, 0.0f);
// 实线条描边,宽度1.0f
Stroke stroke = new BasicStroke(1.0f);
// g2.setColor(Color.PINK);
// 先设置实体颜色
g2.setPaint(barpaint);
g2.setStroke(stroke);
g2.fill(r2d);
g2.fill(bar3dRight);
g2.fill(bar3dTop);
// 实体填充进绘图对象以后,在设置轮廓线颜色(其实是上面三个实体以外的所有图形的颜色)
g2.setPaint(outlinepaint);
g2.draw(r2d);
g2.draw(bar3dRight);
g2.draw(bar3dTop);
/**
* 利用椭圆对象绘制椭圆
*/
Ellipse2D ellipse2D = new Ellipse2D.Double(200.0, // x
100.0, // y
200.0, // width
200.0 // hight
);
Paint ellipsepaint = Color.blue;
g2.setPaint(ellipsepaint);
g2.fill(ellipse2D);
g2.draw(ellipse2D);
/**
* 利用弧形对象绘制弧形
*/
Arc2D.Double arc = new Arc2D.Double(350.0, // x
100.0, // y
200.0, // width
200.0, // hight
0.0, // start angle
90.0, // arc angle
Arc2D.PIE // pie:3.1415926
);
Arc2D.Double arc2 = new Arc2D.Double(350.0, // x
100.0, // y
200.0, // width
200.0, // hight
-60.0, // start angle
90.0, // arc angle
Arc2D.PIE // pie:3.1415926
);
Paint arcpaint1 = Color.GREEN;
Paint arcpaint2 = Color.RED;
// 设置透明度
g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER,
0.3f));
g2.setPaint(arcpaint1);
g2.fill(arc);
g2.draw(arc);
g2.setPaint(arcpaint2);
g2.fill(arc2);
g2.draw(arc2);
/**
* 图形的几何运算
*/
// Area a = new Area(arc);
// Area b = new Area(arc2);
// ab交差运算,去掉两个图形相交的部分得到的图形
// a.exclusiveOr(b);
// a交b运算
// a.intersect(b);
// a减去b运算,即差运算
// a.subtract(b);
// a并b运算
// a.add(b);
// g2.setPaint(arcpaint2);
// g2.fill(a);
// g2.draw(a);
/**
* 利用jfreechart的TextBox对象绘制文本框
*/
// 设置透明度为1,也就是不透明了
g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER,
1.0f));
TextBox textBox = new TextBox("java" // 想要显示的字符串
);
textBox.draw(g2, // 绘图对象
(float) 550.0, // x坐标
(float) 100.0, // y坐标
RectangleAnchor.CENTER // 对齐方式
);
Line2D.Double line = new Line2D.Double(522.0, // 线的第一个点的x坐标
100.0, // 线的第一个点的y坐标
500.0, // 线的第二个点的x坐标
120.0 // 线的第二个点的y坐标
);
g2.draw(line);
/**
* 写字
*/
// 字体设置
Font font = new Font(null, // 字体
Font.ITALIC + Font.BOLD, // 样式(这里设置了两种样式,黑体+斜体)
10); // 大小
g2.setFont(font);
Paint arcpaint3 = Color.BLACK;
g2.setPaint(arcpaint3);
g2.drawString("delphi", // 要写的字符串
550, // x坐标
150 // y坐标
);
/**
* 利用多边形对象绘制多边形
*/
int []xs = new int[] { 600, 620,
640, 620 };
int []ys = new int[] { 100,
100, 150,
150};
Polygon polygon = new Polygon(
xs, //多边形x坐标点的数组
ys, //多边形y坐标点的数组
4 //数组大小
);
g2.setPaint(java.awt.Color.lightGray);
g2.fill(polygon);
g2.draw(polygon);
/**
* 画网格线
*/
for(double i=0.0; i<450.0; i+=30.0){
// 网格线样式
Stroke DEFAULT_GRIDLINE_STROKE = new BasicStroke(0.5f,
BasicStroke.CAP_BUTT,
BasicStroke.JOIN_BEVEL,
0.0f,
new float[] {2.0f, 2.0f},
0.0f);
Line2D.Double gridline = new Line2D.Double(0.0,i+30.0,700.0,i+30.0);
g2.setPaint(Color.GRAY);
g2.setStroke(DEFAULT_GRIDLINE_STROKE);
g2.draw(gridline);
}
2.柱状图2D和3D是通过CategoryPlot的renderer属性来区别的
BarRenderer //2DBarRenderer
BarRenderer3D //3DBarRenderer
CategoryPlot plot = new CategoryPlot(
dataset, categoryAxis, valueAxis, renderer
);
3.3D饼图的绘制函数(现在的注释还不够全面,也可能存在错误)
/**
* Draws the plot on a Java 2D graphics device (such as the screen or a
* printer). This method is called by the {@link org.jfree.chart.JFreeChart}
* class, you don't normally need to call it yourself.
*
* @param g2
* the graphics device.
* @param plotArea
* the area within which the plot should be drawn.
* @param anchor
* the anchor point.
* @param parentState
* the state from the parent plot, if there is one.
* @param info
* collects info about the drawing (<code>null</code>
* permitted).
*/
public void draw(Graphics2D g2, Rectangle2D plotArea, Point2D anchor,
PlotState parentState, PlotRenderingInfo info) {
// adjust for insets...
RectangleInsets insets = getInsets();
insets.trim(plotArea);
Rectangle2D originalPlotArea = (Rectangle2D) plotArea.clone();
if (info != null) {
info.setPlotArea(plotArea);
info.setDataArea(plotArea);
}
Shape savedClip = g2.getClip();
g2.clip(plotArea);
// adjust the plot area by the interior spacing value
double gapPercent = getInteriorGap();
double labelPercent = 0.0;
if (getLabelGenerator() != null) {
labelPercent = getLabelGap() + getMaximumLabelWidth()
+ getLabelLinkMargin();
}
// 水平方向的间隙
double gapHorizontal = plotArea.getWidth()
* (gapPercent + labelPercent);
// 垂直方向的间隙
double gapVertical = plotArea.getHeight() * gapPercent;
// x坐标大小
double linkX = plotArea.getX() + gapHorizontal / 2;
// y坐标大小
double linkY = plotArea.getY() + gapVertical / 2;
// 图形宽度
double linkW = plotArea.getWidth() - gapHorizontal;
// 图形高度
double linkH = plotArea.getHeight() - gapVertical;
// make the link area a square if the pie chart is to be circular...
if (isCircular()) { // is circular?
double min = Math.min(linkW, linkH) / 2;
linkX = (linkX + linkX + linkW) / 2 - min;
linkY = (linkY + linkY + linkH) / 2 - min;
linkW = 2 * min;
linkH = 2 * min;
}
PiePlotState state = initialise(g2, plotArea, this, null, info);
// the explode area defines the max circle/ellipse for the exploded pie
// sections.
// it is defined by shrinking the linkArea by the linkMargin factor.
double hh = linkW * getLabelLinkMargin();
double vv = linkH * getLabelLinkMargin();
Rectangle2D explodeArea = new Rectangle2D.Double(linkX + hh / 2.0,
linkY + vv / 2.0, linkW - hh, linkH - vv);
state.setExplodedPieArea(explodeArea);
// the pie area defines the circle/ellipse for regular pie sections.
// it is defined by shrinking the explodeArea by the explodeMargin
// factor.
double maximumExplodePercent = getMaximumExplodePercent();
double percent = maximumExplodePercent / (1.0 + maximumExplodePercent);
double h1 = explodeArea.getWidth() * percent;
double v1 = explodeArea.getHeight() * percent;
Rectangle2D pieArea = new Rectangle2D.Double(explodeArea.getX() + h1
/ 2.0, explodeArea.getY() + v1 / 2.0, explodeArea.getWidth()
- h1, explodeArea.getHeight() - v1);
// 定义3D椭圆的高度
int depth = (int) (pieArea.getHeight() * this.depthFactor);
// the link area defines the dog-leg point for the linking lines to
// the labels
Rectangle2D linkArea = new Rectangle2D.Double(linkX, linkY, linkW,
linkH - depth);
state.setLinkArea(linkArea);
state.setPieArea(pieArea);
// 定义椭圆中心点
state.setPieCenterX(pieArea.getCenterX());
state.setPieCenterY(pieArea.getCenterY() - depth / 2.0);
// 定义椭圆宽的半径
state.setPieWRadius(pieArea.getWidth() / 2.0);
// 定义椭圆高的半径
state.setPieHRadius((pieArea.getHeight() - depth) / 2.0);
// 画背景长方形(500*300)区域
drawBackground(g2, plotArea);
// 获取外面传进来3D饼图的数据源
PieDataset dataset = getDataset();
// 如果数据源为空则直接返回,整个图表就一个背景长方形没有数据图形
if (DatasetUtilities.isEmptyOrNull(getDataset())) {
drawNoDataMessage(g2, plotArea);
g2.setClip(savedClip);
drawOutline(g2, plotArea);
return;
}
/**
* 如果数据源的主键的个数大于图形区域的宽度,则在图形上显示"Too many elements" 数据源示例如下: final
* DefaultPieDataset result = new DefaultPieDataset();
* result.setValue("Java", new Double(43.2));
* result.setValue("VisualBasic", new Double(10.0));
* result.setValue("C/C++", new Double(17.5));
* result.setValue("PHP", new Double(32.5));
* result.setValue("Perl", new Double(1.0));
* return result;
* 这个数据源的主键的个数为5
*/
if (dataset.getKeys().size() > plotArea.getWidth()) {
String text = "Too many elements";
Font sfont = new Font("dialog", Font.BOLD, 10);
g2.setFont(sfont);
FontMetrics fm = g2.getFontMetrics(sfont);
int stringWidth = fm.stringWidth(text);
g2
.drawString(text, (int) (plotArea.getX() + (plotArea
.getWidth() - stringWidth) / 2), (int) (plotArea
.getY() + (plotArea.getHeight() / 2)));
return;
}
// if we are drawing a perfect circle, we need to readjust the top left
// coordinates of the drawing area for the arcs to arrive at this
// effect.
// 如果我们画的是一个圆形那只要知道图形的左边距,上边距,和直径
if (isCircular()) {
// 圆形半径(取宽高中的小的那个数的一半)
double min = Math.min(plotArea.getWidth(), plotArea.getHeight()) / 2;
plotArea = new Rectangle2D.Double(
plotArea.getCenterX() - min, //左边距(新的图形的中心点x坐标)
plotArea.getCenterY() - min, //上边距(新的图形的中心点y坐标)
2 * min, //直径(图形的宽度)
2 * min //直径(图形的高度)
);
}
// 获取数据源主键列表
List sectionKeys = dataset.getKeys();
// 如果数据源主键列表为空,退出
if (sectionKeys.size() == 0) {
return;
}
// establish the coordinates of the top left corner of the drawing area
//确定pieArea区域的中心点
double arcX = pieArea.getX();
double arcY = pieArea.getY();
// g2.clip(clipArea);
Composite originalComposite = g2.getComposite();
g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER,
getForegroundAlpha()));
//把数据源每个主键对应的值加起来
double totalValue = DatasetUtilities.calculatePieDatasetTotal(dataset);
//遍历列表的时候用到
double runningTotal = 0;
//如果高度<0退出
if (depth < 0) {
return; // if depth is negative don't draw anything
}
//弧形的列表(用于添加主键的值对应的弧形)
ArrayList arcList = new ArrayList();
//定义一个弧行
Arc2D.Double arc;
//定义油漆桶,用来填充颜色
Paint paint;
Paint outlinePaint;
//用于描绘轮廓的样式
Stroke outlineStroke;
//遍历取出主键的值对应的弧形放到弧形列表
Iterator iterator = sectionKeys.iterator();
while (iterator.hasNext()) {
// 取出主键的值
Comparable currentKey = (Comparable) iterator.next();
Number dataValue = dataset.getValue(currentKey);
// 如果主键的值为空,则弧形为空
if (dataValue == null) {
arcList.add(null);
continue;
}
// 如果主键的值<=0,则弧形为空
double value = dataValue.doubleValue();
if (value <= 0) {
arcList.add(null);
continue;
}
//弧形的开始角度(从该角度算起,相当于360度的0度)
double startAngle = getStartAngle();
//方向
double direction = getDirection().getFactor();
//角度大小1(该主键对应的弧形的开始角度)
double angle1 = startAngle + (direction * (runningTotal * 360))
/ totalValue;
//角度大小2(该主键对应的弧形的结束角度)
double angle2 = startAngle
+ (direction * (runningTotal + value) * 360) / totalValue;
//如果弧形的角度大小<0.00001则该弧形为空
if (Math.abs(angle2 - angle1) > getMinimumArcAngleToDraw()) {
arcList.add(new Arc2D.Double(arcX, arcY + depth, pieArea
.getWidth(), pieArea.getHeight() - depth, angle1,
angle2 - angle1, Arc2D.PIE));
} else {
arcList.add(null);
}
//转入下个弧形(主键对应的图形)
runningTotal += value;
}
//取出剪裁的图形(500*300)
Shape oldClip = g2.getClip();
//3D饼图中上面的椭圆
Ellipse2D top = new Ellipse2D.Double(pieArea.getX(), pieArea.getY(),
pieArea.getWidth(), pieArea.getHeight() - depth);
// 3D饼图中底部的椭圆
Ellipse2D bottom = new Ellipse2D.Double(pieArea.getX(), pieArea.getY()
+ depth, pieArea.getWidth(), pieArea.getHeight() - depth);
//底部的长方形
Rectangle2D lower = new Rectangle2D.Double(top.getX(),
top.getCenterY(), pieArea.getWidth(), bottom.getMaxY()
- top.getCenterY());
//上部的长方形
Rectangle2D upper = new Rectangle2D.Double(pieArea.getX(), top.getY(),
pieArea.getWidth(), bottom.getCenterY() - top.getY());
// Area几何建模对象
Area a = new Area(top);
a.add(new Area(lower));
Area b = new Area(bottom);
b.add(new Area(upper));
Area pie = new Area(a);
// 通过几何建模对象对上面的图形进行交叉运算
pie.intersect(b);
Area front = new Area(pie);
// 相减运算
front.subtract(new Area(top));
Area back = new Area(pie);
// 相减运算
back.subtract(new Area(bottom));
// draw the bottom circle
int[] xs;
int[] ys;
outlinePaint = getSectionOutlinePaint(0);
arc = new Arc2D.Double(arcX, arcY + depth, pieArea.getWidth(), pieArea
.getHeight()
- depth, 0, 360, Arc2D.PIE);
// 画出3D饼图下部的椭圆的弧形和有该图形的高度组成的多边形
int categoryCount = arcList.size();
for (int categoryIndex = 0; categoryIndex < categoryCount; categoryIndex++) {
//取出弧形
arc = (Arc2D.Double) arcList.get(categoryIndex);
if (arc == null) {
continue;
}
paint = getSectionPaint(categoryIndex);
outlinePaint = getSectionOutlinePaint(categoryIndex);
outlineStroke = getSectionOutlineStroke(categoryIndex);
g2.setPaint(paint);
//填充弧形
g2.fill(arc);
g2.setPaint(outlinePaint);
g2.setStroke(outlineStroke);
//绘制弧形
g2.draw(arc);
g2.setPaint(paint);
//取得弧形的开始点
Point2D p1 = arc.getStartPoint();
// 定义多边形(弧形的高度组成的图形)
xs = new int[] { (int) arc.getCenterX(), (int) arc.getCenterX(),
(int) p1.getX(), (int) p1.getX() };
ys = new int[] { (int) arc.getCenterY(),
(int) arc.getCenterY() - depth, (int) p1.getY() - depth,
(int) p1.getY() };
Polygon polygon = new Polygon(
xs, //多边形x坐标点的数组
ys, //多边形y坐标点的数组
4 //数组大小
);
g2.setPaint(java.awt.Color.lightGray);
g2.fill(polygon);
g2.setPaint(outlinePaint);
g2.setStroke(outlineStroke);
//绘制该多边形
g2.draw(polygon);
g2.setPaint(paint);
}
g2.setPaint(Color.gray);
g2.fill(back);
g2.fill(front);
// cycle through once drawing only the sides at the back...
// 描出背部的边
int cat = 0;
iterator = arcList.iterator();
while (iterator.hasNext()) {
Arc2D segment = (Arc2D) iterator.next();
if (segment != null) {
paint = getSectionPaint(cat);
outlinePaint = getSectionOutlinePaint(cat);
outlineStroke = getSectionOutlineStroke(cat);
drawSide(g2, pieArea, segment, front, back, paint,
outlinePaint, outlineStroke, false, true);
}
cat++;
}
// cycle through again drawing only the sides at the front...
// 描出前面的边
cat = 0;
iterator = arcList.iterator();
while (iterator.hasNext()) {
Arc2D segment = (Arc2D) iterator.next();
if (segment != null) {
paint = getSectionPaint(cat);
outlinePaint = getSectionOutlinePaint(cat);
outlineStroke = getSectionOutlineStroke(cat);
drawSide(g2, pieArea, segment, front, back, paint,
outlinePaint, outlineStroke, true, false);
}
cat++;
}
g2.setClip(oldClip);
// 画出3D饼图上部的椭圆的弧形和相关的标签链接和底部的说明(tooltip)
Arc2D upperArc;
for (int sectionIndex = 0; sectionIndex < categoryCount; sectionIndex++) {
//取出弧形
arc = (Arc2D.Double) arcList.get(sectionIndex);
if (arc == null) {
continue;
}
//定义3D饼图上部的椭圆的弧形
upperArc = new Arc2D.Double(arcX, arcY, pieArea.getWidth(), pieArea
.getHeight()
- depth,
arc.getAngleStart(), //开始角度
arc.getAngleExtent(), //角度大小
Arc2D.PIE);
//g2图形对象所需的油漆桶
paint = getSectionPaint(sectionIndex);
//g2图形对象所需的轮廓油漆桶
outlinePaint = getSectionOutlinePaint(sectionIndex);
//g2图形对象所需的轮廓样式
outlineStroke = getSectionOutlineStroke(sectionIndex);
g2.setPaint(paint);
//往g2图形对象中填充弧形
g2.fill(upperArc);
g2.setStroke(outlineStroke);
g2.setPaint(outlinePaint);
//绘制弧形
g2.draw(upperArc);
// 为该部分弧形添加说明栏(tooltip)和url链接
Comparable currentKey = (Comparable) sectionKeys.get(sectionIndex);
if (info != null) {
EntityCollection entities = info.getOwner()
.getEntityCollection();
if (entities != null) {
String tip = null;
PieToolTipGenerator tipster = getToolTipGenerator();
if (tipster != null) {
// @mgs: using the method's return value was missing
tip = tipster.generateToolTip(dataset, currentKey);
}
String url = null;
if (getURLGenerator() != null) {
url = getURLGenerator().generateURL(dataset,
currentKey, getPieIndex());
}
PieSectionEntity entity = new PieSectionEntity(upperArc,
dataset, getPieIndex(), sectionIndex, currentKey,
tip, url);
entities.add(entity);
}
}
List keys = dataset.getKeys();
//绘制标签的长方形
Rectangle2D adjustedPlotArea = new Rectangle2D.Double(
originalPlotArea.getX(), originalPlotArea.getY(),
originalPlotArea.getWidth(), originalPlotArea.getHeight()
- depth);
//绘制标签(如:Perl=1的Textbox)
drawLabels(g2, keys, totalValue, adjustedPlotArea, linkArea, state);
}
g2.setClip(savedClip);
g2.setComposite(originalComposite);
//绘制轮廓
drawOutline(g2, originalPlotArea);
}
4.3D柱状图的绘制函数(现在的注释还不够全面,也可能存在错误)
/**
* Draws a 3D bar to represent one data item.
*
* @param g2 the graphics device.
* @param state the renderer state.
* @param dataArea the area for plotting the data.
* @param plot the plot.
* @param domainAxis the domain axis.
* @param rangeAxis the range axis.
* @param dataset the dataset.
* @param row the row index (zero-based).
* @param column the column index (zero-based).
* @param pass the pass index.
*/
public void drawItem(Graphics2D g2,
CategoryItemRendererState state,
Rectangle2D dataArea,
CategoryPlot plot,
CategoryAxis domainAxis,
ValueAxis rangeAxis,
CategoryDataset dataset,
int row,
int column,
int pass) {
// check the value we are plotting...
Number dataValue = dataset.getValue(row, column);
if (dataValue == null) {
return;
}
double value = dataValue.doubleValue();
Rectangle2D adjusted = new Rectangle2D.Double(
dataArea.getX(),
dataArea.getY() + getYOffset(),
dataArea.getWidth() - getXOffset(),
dataArea.getHeight() - getYOffset()
);
PlotOrientation orientation = plot.getOrientation();
double barW0 = calculateBarW0(
plot, orientation, adjusted, domainAxis, state, row, column
);
double[] barL0L1 = calculateBarL0L1(value);
if (barL0L1 == null) {
return; // the bar is not visible
}
RectangleEdge edge = plot.getRangeAxisEdge();
double transL0 = rangeAxis.valueToJava2D(barL0L1[0], adjusted, edge);
double transL1 = rangeAxis.valueToJava2D(barL0L1[1], adjusted, edge);
double barL0 = Math.min(transL0, transL1);
double barLength = Math.abs(transL1 - transL0);
// draw the bar...
//柱子的正视图
Rectangle2D bar = null;
if (orientation == PlotOrientation.HORIZONTAL) {
bar = new Rectangle2D.Double(
barL0, barW0, barLength, state.getBarWidth()
);
}
else {
bar = new Rectangle2D.Double(
barW0, barL0, state.getBarWidth(), barLength
);
}
Paint itemPaint = getItemPaint(row, column);
g2.setPaint(itemPaint);
g2.fill(bar);
// 柱子的俯视图上的四个点的x坐标值
double x0 = bar.getMinX();
double x1 = x0 + getXOffset(); //x0+12
double x2 = bar.getMaxX();
double x3 = x2 + getXOffset(); //x2+12
// 柱子的右视图上的四个点的y坐标值
double y0 = bar.getMinY() - getYOffset(); //正视图的最小y值-8
double y1 = bar.getMinY();
double y2 = bar.getMaxY() - getYOffset(); //正视图的最大y值-8
double y3 = bar.getMaxY();
GeneralPath bar3dRight = null;
GeneralPath bar3dTop = null;
// 柱子的右视图
if (barLength > 0.0) {
bar3dRight = new GeneralPath();
bar3dRight.moveTo((float) x2, (float) y3);
bar3dRight.lineTo((float) x2, (float) y1);
bar3dRight.lineTo((float) x3, (float) y0);
bar3dRight.lineTo((float) x3, (float) y2);
bar3dRight.closePath();
if (itemPaint instanceof Color) {
g2.setPaint(((Color) itemPaint).darker());
}
g2.fill(bar3dRight);
}
// 柱子的俯视图
bar3dTop = new GeneralPath();
bar3dTop.moveTo((float) x0, (float) y1);
bar3dTop.lineTo((float) x1, (float) y0);
bar3dTop.lineTo((float) x3, (float) y0);
bar3dTop.lineTo((float) x2, (float) y1);
bar3dTop.closePath();
g2.fill(bar3dTop);
//绘制3D柱状图
if (isDrawBarOutline()
&& state.getBarWidth() > BAR_OUTLINE_WIDTH_THRESHOLD) {
g2.setStroke(getItemOutlineStroke(row, column));
g2.setPaint(getItemOutlinePaint(row, column));
g2.draw(bar);
if (bar3dRight != null) {
g2.draw(bar3dRight);
}
if (bar3dTop != null) {
g2.draw(bar3dTop);
}
}
//绘制标签
CategoryItemLabelGenerator generator
= getItemLabelGenerator(row, column);
if (generator != null && isItemLabelVisible(row, column)) {
drawItemLabel(
g2, dataset, row, column, plot, generator, bar, (value < 0.0)
);
}
// add an item entity, if this information is being collected
EntityCollection entities = state.getEntityCollection();
if (entities != null) {
GeneralPath barOutline = new GeneralPath();
barOutline.moveTo((float) x0, (float) y3);
barOutline.lineTo((float) x0, (float) y1);
barOutline.lineTo((float) x1, (float) y0);
barOutline.lineTo((float) x3, (float) y0);
barOutline.lineTo((float) x3, (float) y2);
barOutline.lineTo((float) x2, (float) y3);
barOutline.closePath();
addItemEntity(entities, dataset, row, column, barOutline);
}
}