multiPiePlot Chart に似たものはありますが、xy プロット用のものはありますか? 1 ページに 2 つまたは 3 つの xy プロットを印刷する必要があるアプリケーションがあります。複数のデータセットを同じプロットに配置できることは知っていますが、要件では、それぞれが同じページの個別のグラフでなければならないと指定されています。
2 に答える
trashgodが使用したソリューションを使用しましたが、アプリケーションで実行したい種類の印刷をサポートするようにソリューションを拡張しました。
JFreechartが通常行うように印刷できるようにしたかったのですが、グリッド内の各ページにそのグラフの倍数を配置し、ページにうまく拡大および拡大できるようにしました。この機能を使用して、下の画像に一致するプリントを生成することができました。
基本的に私はJPanelを拡張し、Printableインターフェースを実装しました。表示/印刷したいすべてのJFreechartsを含むパネルを作成します。パネルは、提供されたいくつかのレイアウト指示に基づいて、それらをグリッド形式で配置します。
パネルの印刷機能は、基本的に通常のページ形式を取りますが、その後、ページを前に指定したグリッドに分割します。次に、これらの各セルを取得し、それぞれのページ形式を作成します。次に、各JFreeChartChartPanelにそれらの各PageFormatセルに印刷するように指示します。
CustomChartPanel:
package com.company.jfreeChartCustom;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GridLayout;
import java.awt.geom.Rectangle2D;
import java.awt.print.PageFormat;
import java.awt.print.Paper;
import java.awt.print.Printable;
import java.awt.print.PrinterException;
import java.awt.print.PrinterJob;
import java.util.ArrayList;
import java.util.List;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import com.lowagie.text.Row;
/**
* The CustomChartPanel is used to display/print multiple JFreeCharts
* Users should only interact with this class with the methods defined
* below in order to get the proper functionality. Using
* inherited methods may produce unwanted display/print behavior if you
* add components other than JFreeCharts.
*
*/
public class CustomChartPanel extends JPanel implements Printable{
List<JFreeChart> charts = new ArrayList<JFreeChart>();
List<ChartPanel> panels = new ArrayList<ChartPanel>();
ChartLayoutInstructions layoutInstructions;
public CustomChartPanel(){
super();
}
public CustomChartPanel(JFreeChart chart){
super();
charts.add(chart);
}
/**
* Creates a CustomChartPanel which displays 1 or more charts in a grid-like fashion
* described by the layoutInstructions you pass in. Note that if you pass in more
* charts than there are columns specified in the ChartLayoutInstructions then excess
* charts will not be displayed or printed.
* @param charts
* @param layoutInstructions
*/
public CustomChartPanel(List<JFreeChart> charts, ChartLayoutInstructions layoutInstructions){
super();
this.layoutInstructions = layoutInstructions;
for(JFreeChart chart : charts){
this.charts.add(chart);
}
createUIComponents();
}
protected void createUIComponents(){
int size = Math.min(layoutInstructions.getColumns() * layoutInstructions.getRows(), charts.size());
this.setLayout(new GridLayout(layoutInstructions.getRows(), layoutInstructions.getColumns()));
for(int i = 0; i < size; i++ ){
System.err.println("Adding chart");
ChartPanel chartPanel = new ChartPanel(charts.get(i));
chartPanel.setMaximumDrawHeight(20000);
chartPanel.setMinimumDrawHeight(0);
chartPanel.setMaximumDrawWidth(20000);
chartPanel.setMinimumDrawWidth(0);
chartPanel.setPopupMenu(null);
panels.add(chartPanel);
this.add(chartPanel);
}
}
public void createPrintJob(){
PrinterJob job = PrinterJob.getPrinterJob();
PageFormat pf = job.defaultPage();
PageFormat pf2 = job.pageDialog(pf);
if (pf2 != pf) {
job.setPrintable(this, pf2);
if (job.printDialog()) {
try {
job.print();
}
catch (PrinterException e) {
JOptionPane.showMessageDialog(this, e);
}
}
}
}
@Override
public int print(Graphics g, PageFormat pf, int pageIndex)
throws PrinterException {
System.err.println("PRINTING");
//Divide the current page format into sections based
//on the layout instructions received in the constructor
//a new pagelayout is created for each cell in the grid
//that will then be passed along to the print method of
//each chart panel.
if(pageIndex != 0){
return NO_SUCH_PAGE;
}
List<PageFormat> pageFormats = new ArrayList<PageFormat>();
//setup all the page formats needed for the grid cells.
double x = pf.getImageableX();
double y = pf.getImageableY();
double cellWidth = pf.getImageableWidth() / layoutInstructions.getColumns();
double cellHeight = pf.getImageableHeight() / layoutInstructions.getRows();
for(int i=1; i <= layoutInstructions.getRows(); i++){
double rowOffset = (i-1)*cellHeight + y;
for(int j=1; j <= layoutInstructions.getColumns(); j++){
PageFormat format = new PageFormat();
Paper paper = new Paper();
double columnOffset = (j-1)*cellWidth + x;
paper.setImageableArea(columnOffset, rowOffset, cellWidth, cellHeight);
format.setPaper(paper);
pageFormats.add(format);
}
}
//have each chartpanel print on the graphics context using its
//particular PageFormat
int size = Math.min(pageFormats.size(), panels.size());
for(int i = 0; i < size; i++ ){
panels.get(i).print(g, pageFormats.get(i), pageIndex);
}
return PAGE_EXISTS;
}
ChartLayoutInstructions:
より高度な動作を定義するために、このクラスを自分の目的のために拡張する予定です。そのため、CustomChartPanelクラスのコンストラクターで行と列を定義するだけでなく、このクラスを使用します。
package com.company.jfreeChartCustom;
/**
* ChartLayoutInstructions are used to specify how charts should be
* layed out on screen and in print format.
*
*/
public class ChartLayoutInstructions {
int rows;
int columns;
/**
* Constructor
* @param rows number of rows in the display/print grid
* @param columns number of columns in the display/print grid
*/
public ChartLayoutInstructions(int rows, int columns, boolean allowSwap){
this.rows = Math.abs(rows);
this.columns = Math.abs(columns);
}
public int getRows() {
return rows;
}
public void setRows(int rows) {
this.rows = Math.abs(rows);
}
public int getColumns() {
return columns;
}
public void setColumns(int columns) {
this.columns = Math.abs(columns);
}
}
はい、ChartPanel
インスタンスを追加しJPanel
てGridLayout(0, 1)
、上から下へ配置するだけです。この例では、直交GridLayout(1, 0)
を使用して 3 つのパネルを横に配置します。