roooodcastro / GraphPanel.java
A simple Swing component to draw a Graph over a regular JPanel. Features a grid, customizable amount of hatch marks, axis labels,checking for minimum and maximum value to label correctly the Y-axis and customizable padding and label padding. Based on «Hovercraft Full Of Eels»‘s answer on StackOverflow (http://stackoverflow.com/questions/8693342/…
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters. Learn more about bidirectional Unicode characters
import java . awt . BasicStroke ; |
import java . awt . Color ; |
import java . awt . Dimension ; |
import java . awt . FontMetrics ; |
import java . awt . Graphics ; |
import java . awt . Graphics2D ; |
import java . awt . Point ; |
import java . awt . RenderingHints ; |
import java . awt . Stroke ; |
import java . util . ArrayList ; |
import java . util . List ; |
import java . util . Random ; |
import javax . swing . JFrame ; |
import javax . swing . JPanel ; |
import javax . swing . SwingUtilities ; |
/** |
* |
* @author «Hovercraft Full of Eels», «Rodrigo Azevedo» |
* |
* This is an improved version of Hovercraft Full of Eels (https://stackoverflow.com/users/522444/hovercraft-full-of-eels) |
* answer on StackOverflow: https://stackoverflow.com/a/8693635/753012 |
* |
* GitHub user @maritaria has made some performance improvements which can be found in the comment section of this Gist. |
*/ |
public class GraphPanel extends JPanel |
private int width = 800 ; |
private int height = 400 ; |
private int padding = 25 ; |
private int labelPadding = 25 ; |
private Color lineColor = new Color ( 44 , 102 , 230 , 180 ); |
private Color pointColor = new Color ( 100 , 100 , 100 , 180 ); |
private Color gridColor = new Color ( 200 , 200 , 200 , 200 ); |
private static final Stroke GRAPH_STROKE = new BasicStroke ( 2f ); |
private int pointWidth = 4 ; |
private int numberYDivisions = 10 ; |
private List < Double >scores ; |
public GraphPanel ( List < Double >scores ) |
this . scores = scores ; |
> |
@ Override |
protected void paintComponent ( Graphics g ) |
super . paintComponent ( g ); |
Graphics2D g2 = ( Graphics2D ) g ; |
g2 . setRenderingHint ( RenderingHints . KEY_ANTIALIASING , RenderingHints . VALUE_ANTIALIAS_ON ); |
double xScale = (( double ) getWidth () — ( 2 * padding ) — labelPadding ) / ( scores . size () — 1 ); |
double yScale = (( double ) getHeight () — 2 * padding — labelPadding ) / ( getMaxScore () — getMinScore ()); |
List < Point >graphPoints = new ArrayList <>(); |
for ( int i = 0 ; i < scores . size (); i ++) |
int x1 = ( int ) ( i * xScale + padding + labelPadding ); |
int y1 = ( int ) (( getMaxScore () — scores . get ( i )) * yScale + padding ); |
graphPoints . add ( new Point ( x1 , y1 )); |
> |
// draw white background |
g2 . setColor ( Color . WHITE ); |
g2 . fillRect ( padding + labelPadding , padding , getWidth () — ( 2 * padding ) — labelPadding , getHeight () — 2 * padding — labelPadding ); |
g2 . setColor ( Color . BLACK ); |
// create hatch marks and grid lines for y axis. |
for ( int i = 0 ; i < numberYDivisions + 1 ; i ++) |
int x0 = padding + labelPadding ; |
int x1 = pointWidth + padding + labelPadding ; |
int y0 = getHeight () — (( i * ( getHeight () — padding * 2 — labelPadding )) / numberYDivisions + padding + labelPadding ); |
int y1 = y0 ; |
if ( scores . size () > 0 ) |
g2 . setColor ( gridColor ); |
g2 . drawLine ( padding + labelPadding + 1 + pointWidth , y0 , getWidth () — padding , y1 ); |
g2 . setColor ( Color . BLACK ); |
String yLabel = (( int ) (( getMinScore () + ( getMaxScore () — getMinScore ()) * (( i * 1.0 ) / numberYDivisions )) * 100 )) / 100.0 + «» ; |
FontMetrics metrics = g2 . getFontMetrics (); |
int labelWidth = metrics . stringWidth ( yLabel ); |
g2 . drawString ( yLabel , x0 — labelWidth — 5 , y0 + ( metrics . getHeight () / 2 ) — 3 ); |
> |
g2 . drawLine ( x0 , y0 , x1 , y1 ); |
> |
// and for x axis |
for ( int i = 0 ; i < scores . size (); i ++) |
if ( scores . size () > 1 ) |
int x0 = i * ( getWidth () — padding * 2 — labelPadding ) / ( scores . size () — 1 ) + padding + labelPadding ; |
int x1 = x0 ; |
int y0 = getHeight () — padding — labelPadding ; |
int y1 = y0 — pointWidth ; |
if (( i % (( int ) (( scores . size () / 20.0 )) + 1 )) == 0 ) |
g2 . setColor ( gridColor ); |
g2 . drawLine ( x0 , getHeight () — padding — labelPadding — 1 — pointWidth , x1 , padding ); |
g2 . setColor ( Color . BLACK ); |
String xLabel = i + «» ; |
FontMetrics metrics = g2 . getFontMetrics (); |
int labelWidth = metrics . stringWidth ( xLabel ); |
g2 . drawString ( xLabel , x0 — labelWidth / 2 , y0 + metrics . getHeight () + 3 ); |
> |
g2 . drawLine ( x0 , y0 , x1 , y1 ); |
> |
> |
// create x and y axes |
g2 . drawLine ( padding + labelPadding , getHeight () — padding — labelPadding , padding + labelPadding , padding ); |
g2 . drawLine ( padding + labelPadding , getHeight () — padding — labelPadding , getWidth () — padding , getHeight () — padding — labelPadding ); |
Stroke oldStroke = g2 . getStroke (); |
g2 . setColor ( lineColor ); |
g2 . setStroke ( GRAPH_STROKE ); |
for ( int i = 0 ; i < graphPoints . size () - 1 ; i ++) |
int x1 = graphPoints . get ( i ). x ; |
int y1 = graphPoints . get ( i ). y ; |
int x2 = graphPoints . get ( i + 1 ). x ; |
int y2 = graphPoints . get ( i + 1 ). y ; |
g2 . drawLine ( x1 , y1 , x2 , y2 ); |
> |
g2 . setStroke ( oldStroke ); |
g2 . setColor ( pointColor ); |
for ( int i = 0 ; i < graphPoints . size (); i ++) |
int x = graphPoints . get ( i ). x — pointWidth / 2 ; |
int y = graphPoints . get ( i ). y — pointWidth / 2 ; |
int ovalW = pointWidth ; |
int ovalH = pointWidth ; |
g2 . fillOval ( x , y , ovalW , ovalH ); |
> |
> |
// @Override |
// public Dimension getPreferredSize() |
// return new Dimension(width, height); |
// > |
private double getMinScore () |
double minScore = Double . MAX_VALUE ; |
for ( Double score : scores ) |
minScore = Math . min ( minScore , score ); |
> |
return minScore ; |
> |
private double getMaxScore () |
double maxScore = Double . MIN_VALUE ; |
for ( Double score : scores ) |
maxScore = Math . max ( maxScore , score ); |
> |
return maxScore ; |
> |
public void setScores ( List < Double >scores ) |
this . scores = scores ; |
invalidate (); |
this . repaint (); |
> |
public List < Double >getScores () |
return scores ; |
> |
private static void createAndShowGui () |
List < Double >scores = new ArrayList <>(); |
Random random = new Random (); |
int maxDataPoints = 40 ; |
int maxScore = 10 ; |
for ( int i = 0 ; i < maxDataPoints ; i ++) |
scores . add (( double ) random . nextDouble () * maxScore ); |
// scores.add((double) i); |
> |
GraphPanel mainPanel = new GraphPanel ( scores ); |
mainPanel . setPreferredSize ( new Dimension ( 800 , 600 )); |
JFrame frame = new JFrame ( «DrawGraph» ); |
frame . setDefaultCloseOperation ( JFrame . EXIT_ON_CLOSE ); |
frame . getContentPane (). add ( mainPanel ); |
frame . pack (); |
frame . setLocationRelativeTo ( null ); |
frame . setVisible ( true ); |
> |
public static void main ( String [] args ) |
SwingUtilities . invokeLater ( new Runnable () |
public void run () |
createAndShowGui (); |
> |
>); |
> |
> |
Создание графика по точкам на Java
Сегодня мы научимся рисовать графики по точкам на языке Java. Для этого напишем небольшую программку, результат её вы можете увидеть на рисунке слева. Как можно увидеть, приложение в стиле минимализм — просто оси и график. При желании можно дополнить её названиями осей, пронумеровать их и так далее — этим мы займемся тогда, когда будем делать графики по функциям, а сейчас и так сойдет — нам главное разобраться со способом. А вот, собственно, и сам текст программы:
package up7; import javax.swing.*; import java.awt.*; class DrawingComponent extends JPanel < int xg[] = Up7.x; int yg[] = Up7.y; int ng = Up7.n; @Override protected void paintComponent(Graphics gh) < Graphics2D drp = (Graphics2D)gh; drp.drawLine(20, 340, 20, 20); drp.drawLine(20, 340, 460, 340); drp.drawPolyline(xg, yg, ng); >> public class Up7 extends JFrame< public static int x[] = ; public static int y[] = ; public static int n = 5; public Up7 () < super("График по точкам"); JPanel jcp = new JPanel(new BorderLayout()); setContentPane(jcp); jcp.add(new DrawingComponent (), BorderLayout.CENTER); jcp.setBackground(Color.gray); setSize(500, 400); setLocationRelativeTo(null); setDefaultCloseOperation(EXIT_ON_CLOSE); >public static void main(String[] args) < new Up7().setVisible(true); >>
Немного комментариев по листингу. В программе два класса — сама отрисовка происходит в DrawingComponent — он наследуется от JPanel, а вот значения для отрисовки задаются в классе Up7 — он наследуется от JFrame. Задаем мы три параметра — два массива со значениями по осям x, y и число точек. Точки соединяются прямыми — в этом помогает метод drawPolyline.
Вот так несложно можно создать график функции по точкам на Java. Комментарии, вопросы — на почту up777up@yandex.ru.
Автор этого материала — я — Пахолков Юрий. Я оказываю услуги по написанию программ на языках Java, C++, C# (а также консультирую по ним) и созданию сайтов. Работаю с сайтами на CMS OpenCart, WordPress, ModX и самописными. Кроме этого, работаю напрямую с JavaScript, PHP, CSS, HTML — то есть могу доработать ваш сайт или помочь с веб-программированием. Пишите сюда.
заметки, java, графики, задачи
Бесплатный https и
домен RU в подарок
About Nothing
В примере будет задействована JFreeChart — библиотека для построения и вывода на экран или в файл (jpg, svg и т.п.) графиков и диаграмм различных типов.
Я провел час в поисках библиотеки для построения графиков и остановился на JFreeChart. Она оказалась первой попавшейся мне из тех, что умеют масштабировать графики «из коробки». До неё я опробовал 3 или 4 менее тяжелых, но ни одна из них меня приятно не удивила. А теперь к делу 🙂
package javaapplication1;
import javax.swing.JFrame;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartPanel;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.plot.PlotOrientation;
import org.jfree.data.xy.XYDataset;
import org.jfree.data.xy.XYSeries;
import org.jfree.data.xy.XYSeriesCollection;
public class Main public static void main(String[] args) XYSeries series = new XYSeries("sin(a)");
for(float i = 0; i < Math.PI; i+=0.1)series.add(i, Math.sin(i));
>
XYDataset xyDataset = new XYSeriesCollection(series);
JFreeChart chart = ChartFactory
.createXYLineChart("y = sin(x)", "x", "y",
xyDataset,
PlotOrientation.VERTICAL,
true, true, true);
JFrame frame =
new JFrame("MinimalStaticChart");
// Помещаем график на фрейм
frame.getContentPane()
.add(new ChartPanel(chart));
frame.setSize(400,300);
frame.show();
>
>