/* * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2019, Red Hat Inc. All rights reserved. * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * The contents of this file are subject to the terms of either the Universal Permissive License * v 1.0 as shown at http://oss.oracle.com/licenses/upl * * or the following license: * * Redistribution and use in source and binary forms, with or without modification, are permitted * provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this list of conditions * and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, this list of * conditions and the following disclaimer in the documentation and/or other materials provided with * the distribution. * * 3. Neither the name of the copyright holder nor the names of its contributors may be used to * endorse or promote products derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY * WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package org.openjdk.jmc.ui.misc; import java.awt.Graphics2D; import org.eclipse.swt.SWT; import org.eclipse.swt.custom.SashForm; import org.eclipse.swt.events.MouseAdapter; import org.eclipse.swt.events.MouseEvent; import org.eclipse.swt.events.MouseMoveListener; import org.eclipse.swt.events.PaintEvent; import org.eclipse.swt.events.PaintListener; import org.eclipse.swt.graphics.Point; import org.eclipse.swt.graphics.Rectangle; import org.eclipse.swt.widgets.Canvas; import org.eclipse.swt.widgets.Composite; import org.openjdk.jmc.common.unit.IQuantity; import org.openjdk.jmc.common.unit.IRange; import org.openjdk.jmc.ui.charts.AWTChartToolkit; import org.openjdk.jmc.ui.charts.SubdividedQuantityRange; import org.openjdk.jmc.ui.charts.XYChart; import org.openjdk.jmc.ui.misc.PatternFly.Palette; public class TimelineCanvas extends Canvas { private static final int RANGE_INDICATOR_HEIGHT = 10; private static final int RANGE_INDICATOR_Y_OFFSET = 25; private int x1; private int x2; private int xOffset; private AwtCanvas awtCanvas; private ChartCanvas chartCanvas; private Graphics2D g2d; private IRange chartRange; private Rectangle dragRect; private Rectangle indicatorRect; private Rectangle timelineRect; private SashForm sashForm; private SubdividedQuantityRange xTickRange; private XYChart chart; public TimelineCanvas(Composite parent, ChartCanvas chartCanvas, SashForm sashForm) { super(parent, SWT.NONE); this.chartCanvas = chartCanvas; this.sashForm = sashForm; awtCanvas = new AwtCanvas(); addPaintListener(new TimelineCanvasPainter()); DragDetector dragDetector = new DragDetector(); addMouseListener(dragDetector); addMouseMoveListener(dragDetector); } private int calculateXOffset() { return sashForm.getChildren()[0].getSize().x + sashForm.getSashWidth(); } public void renderRangeIndicator(int x1, int x2) { this.x1 = x1; this.x2 = x2; this.redraw(); } public void setXTickRange(SubdividedQuantityRange xTickRange) { this.xTickRange = xTickRange; } public void setChart(XYChart chart) { this.chart = chart; chartRange = chart.getVisibleRange(); } private class TimelineCanvasPainter implements PaintListener { @Override public void paintControl(PaintEvent e) { xOffset = chartCanvas.translateDisplayToImageXCoordinates(calculateXOffset()); Rectangle rect = getClientArea(); g2d = awtCanvas.getGraphics(rect.width, rect.height); // Draw the background Point adjusted = chartCanvas.translateDisplayToImageCoordinates(rect.width, rect.height); g2d.setColor(Palette.PF_BLACK_100.getAWTColor()); g2d.fillRect(0, 0, adjusted.x, adjusted.y); // Draw the horizontal axis if (xTickRange != null) { g2d.setColor(Palette.PF_BLACK.getAWTColor()); AWTChartToolkit.drawAxis(g2d, xTickRange, 0, false, 1, false, xOffset); } // Draw the range indicator indicatorRect = dragRect != null ? dragRect : new Rectangle( x1 + xOffset, chartCanvas.translateDisplayToImageYCoordinates(RANGE_INDICATOR_Y_OFFSET), x2 - x1, chartCanvas.translateDisplayToImageYCoordinates(RANGE_INDICATOR_HEIGHT)); dragRect = null; g2d.setPaint(Palette.PF_ORANGE_400.getAWTColor()); g2d.fillRect(indicatorRect.x, indicatorRect.y, indicatorRect.width, indicatorRect.height); Point totalSize = sashForm.getChildren()[1].getSize(); adjusted = chartCanvas.translateDisplayToImageCoordinates(totalSize.x, totalSize.y); timelineRect = new Rectangle( xOffset, chartCanvas.translateDisplayToImageYCoordinates(RANGE_INDICATOR_Y_OFFSET), adjusted.x, chartCanvas.translateDisplayToImageYCoordinates(RANGE_INDICATOR_HEIGHT)); g2d.setPaint(Palette.PF_BLACK_600.getAWTColor()); g2d.drawRect(timelineRect.x, timelineRect.y, timelineRect.width, timelineRect.height); awtCanvas.paint(e, 0, 0); } } private class DragDetector extends MouseAdapter implements MouseMoveListener { boolean isDrag = false; Point currentSelection; Point lastSelection; @Override public void mouseDown(MouseEvent e) { e.x = chartCanvas.translateDisplayToImageXCoordinates(e.x); e.y = chartCanvas.translateDisplayToImageYCoordinates(e.y); if (isDrag || e.button == 1 && timelineRect.contains(e.x, e.y)) { isDrag = true; currentSelection = new Point(e.x, e.y); } } @Override public void mouseUp(MouseEvent e) { isDrag = false; chart.setIsZoomPanDrag(false); } @Override public void mouseMove(MouseEvent e) { e.x = chartCanvas.translateDisplayToImageXCoordinates(e.x); e.y = chartCanvas.translateDisplayToImageYCoordinates(e.y); if (timelineRect.contains(e.x, e.y)) { setCursor(getDisplay().getSystemCursor(SWT.CURSOR_HAND)); } else { setCursor(getDisplay().getSystemCursor(SWT.CURSOR_ARROW)); } if (isDrag) { lastSelection = currentSelection; chart.setIsZoomPanDrag(true); currentSelection = new Point(e.x, e.y); int xdiff = currentSelection.x - lastSelection.x; updateTimelineIndicatorFromDrag(xdiff); } } private void updateTimelineIndicatorFromDrag(int xdiff) { if (xdiff != 0 && (indicatorRect.x + xdiff) >= timelineRect.x && (indicatorRect.x + xdiff + indicatorRect.width) <= timelineRect.x + timelineRect.width) { indicatorRect.x = indicatorRect.x + xdiff; SubdividedQuantityRange xAxis = new SubdividedQuantityRange(chartRange.getStart(), chartRange.getEnd(), timelineRect.width, 1); chart.setVisibleRange(xAxis.getQuantityAtPixel(indicatorRect.x - xOffset), xAxis.getQuantityAtPixel(indicatorRect.x - xOffset + indicatorRect.width)); dragRect = indicatorRect; chartCanvas.redrawChart(); } } } }