package net.sf.openrocket.gui.plot; import java.awt.Color; import java.awt.Cursor; import java.awt.Point; import java.awt.event.MouseEvent; import java.awt.event.MouseWheelEvent; import java.awt.event.MouseWheelListener; import java.awt.geom.Point2D; import java.awt.geom.Rectangle2D; import java.io.Serializable; import javax.swing.BorderFactory; import org.jfree.chart.ChartPanel; import org.jfree.chart.ChartRenderingInfo; import org.jfree.chart.JFreeChart; import org.jfree.chart.plot.PiePlot; import org.jfree.chart.plot.Plot; import org.jfree.chart.plot.PlotOrientation; import org.jfree.chart.plot.PlotRenderingInfo; import org.jfree.chart.plot.Zoomable; import com.jogamp.newt.event.InputEvent; /** * Custom implementation of JFreeChart's ChartPanel which changes the mouse handling. * * Changed mouse drag (left click + move) to pan the image. * * Changed mouse wheel handling. wheel zooms. Alt+wheel zooms only domain axis. * * @author kruland * */ public class SimulationChart extends ChartPanel { private Point2D panLast; private Point startPoint; private double panW; private double panH; private enum Interaction { ZOOM }; private Interaction interaction = null; private MouseWheelHandler mouseWheelHandler = null; public SimulationChart(JFreeChart chart) { super(chart, /* properties */false, /* save */true, /* print */false, /* zoom */true, /* tooltips */true); this.setMouseWheelEnabled(true); this.setEnforceFileExtensions(true); this.setInitialDelay(500); this.setBorder(BorderFactory.createLineBorder(Color.GRAY, 1)); } @Override public boolean isMouseWheelEnabled() { return mouseWheelHandler != null; } @Override public void setMouseWheelEnabled(boolean flag) { if (flag && mouseWheelHandler == null) { this.mouseWheelHandler = new MouseWheelHandler(); this.addMouseWheelListener(this.mouseWheelHandler); } else if (!flag && mouseWheelHandler != null) { this.removeMouseWheelListener(this.mouseWheelHandler); this.mouseWheelHandler = null; } } @Override public void mousePressed(MouseEvent e) { if (e.getButton() == MouseEvent.BUTTON3) { Rectangle2D screenDataArea = getScreenDataArea(e.getX(), e.getY()); if (screenDataArea != null && screenDataArea.contains(e.getPoint())) { this.panW = screenDataArea.getWidth(); this.panH = screenDataArea.getHeight(); this.panLast = e.getPoint(); this.startPoint = e.getPoint(); } interaction = Interaction.ZOOM; setCursor(Cursor.getPredefinedCursor(Cursor.NE_RESIZE_CURSOR)); } else { interaction = null; super.mousePressed(e); } } @Override public void mouseDragged(MouseEvent e) { if (interaction == null) { super.mouseDragged(e); return; } if (panLast == null) { return; } double dx = e.getX() - this.panLast.getX(); double dy = e.getY() - this.panLast.getY(); if (dx == 0.0 && dy == 0.0) { return; } double wPercent = -dx / this.panW; double hPercent = dy / this.panH; boolean old = this.getChart().getPlot().isNotify(); this.getChart().getPlot().setNotify(false); switch (interaction) { case ZOOM: Zoomable pz = (Zoomable) this.getChart().getPlot(); double zoomx = 1 + 2 * wPercent; double zoomy = 1 + 2 * hPercent; Point2D anchor = SimulationChart.this.translateScreenToJava2D(startPoint); if (pz.getOrientation() == PlotOrientation.VERTICAL) { pz.zoomDomainAxes(zoomx, this.getChartRenderingInfo().getPlotInfo(), anchor, true); pz.zoomRangeAxes(zoomy, this.getChartRenderingInfo().getPlotInfo(), anchor, true); } else { pz.zoomRangeAxes(zoomx, this.getChartRenderingInfo().getPlotInfo(), anchor, true); pz.zoomDomainAxes(zoomy, this.getChartRenderingInfo().getPlotInfo(), anchor, true); } break; } this.panLast = e.getPoint(); this.getChart().getPlot().setNotify(old); } @Override public void mouseReleased(MouseEvent e) { if (interaction == null) { super.mouseReleased(e); return; } if (this.panLast != null) { this.panLast = null; setCursor(Cursor.getDefaultCursor()); } interaction = null; } /** * * Hacked up copy of MouseWheelHandler from JFreechart. This version * has the special ability to only zoom on the domain if the alt key is pressed. * * A class that handles mouse wheel events for the {@link ChartPanel} class. * Mouse wheel event support was added in JDK 1.4, so this class will be omitted * from JFreeChart if you build it using JDK 1.3. * * @since 1.0.13 */ class MouseWheelHandler implements MouseWheelListener, Serializable { /** The zoom factor. */ double zoomFactor; /** * Creates a new instance for the specified chart panel. * * @param chartPanel the chart panel (<code>null</code> not permitted). */ public MouseWheelHandler() { this.zoomFactor = 0.10; } /** * Returns the current zoom factor. The default value is 0.10 (ten * percent). * * @return The zoom factor. * * @see #setZoomFactor(double) */ public double getZoomFactor() { return this.zoomFactor; } /** * Sets the zoom factor. * * @param zoomFactor the zoom factor. * * @see #getZoomFactor() */ public void setZoomFactor(double zoomFactor) { this.zoomFactor = zoomFactor; } /** * Handles a mouse wheel event from the underlying chart panel. * * @param e the event. */ public void mouseWheelMoved(MouseWheelEvent e) { JFreeChart chart = SimulationChart.this.getChart(); if (chart == null) { return; } Plot plot = chart.getPlot(); if (plot instanceof Zoomable) { Zoomable zoomable = (Zoomable) plot; handleZoomable(zoomable, e); } else if (plot instanceof PiePlot) { PiePlot pp = (PiePlot) plot; pp.handleMouseWheelRotation(e.getWheelRotation()); } } /** * Handle the case where a plot implements the {@link Zoomable} interface. * * @param zoomable the zoomable plot. * @param e the mouse wheel event. */ private void handleZoomable(Zoomable zoomable, MouseWheelEvent e) { // don't zoom unless the mouse pointer is in the plot's data area ChartRenderingInfo info = SimulationChart.this.getChartRenderingInfo(); PlotRenderingInfo pinfo = info.getPlotInfo(); Point2D p = SimulationChart.this.translateScreenToJava2D(e.getPoint()); if (!pinfo.getDataArea().contains(p)) { return; } Plot plot = (Plot) zoomable; // do not notify while zooming each axis boolean notifyState = plot.isNotify(); plot.setNotify(false); int clicks = e.getWheelRotation(); double zf = 1.0 + this.zoomFactor; if (clicks < 0) { zf = 1.0 / zf; } if (SimulationChart.this.isDomainZoomable()) { zoomable.zoomDomainAxes(zf, pinfo, p, true); } boolean domainOnly = (e.getModifiers() & InputEvent.CTRL_MASK) != 0; if (SimulationChart.this.isRangeZoomable() && !domainOnly) { zoomable.zoomRangeAxes(zf, pinfo, p, true); } plot.setNotify(notifyState); // this generates the change event too } } }