]> gerrit.simantics Code Review - simantics/platform.git/blobdiff - bundles/org.simantics.trend/src/org/simantics/trend/impl/TrendParticipant.java
Migrated source code from Simantics SVN
[simantics/platform.git] / bundles / org.simantics.trend / src / org / simantics / trend / impl / TrendParticipant.java
diff --git a/bundles/org.simantics.trend/src/org/simantics/trend/impl/TrendParticipant.java b/bundles/org.simantics.trend/src/org/simantics/trend/impl/TrendParticipant.java
new file mode 100644 (file)
index 0000000..66a67eb
--- /dev/null
@@ -0,0 +1,910 @@
+/*******************************************************************************\r
+ * Copyright (c) 2007, 2011 Association for Decentralized Information Management in\r
+ * Industry THTH ry.\r
+ * All rights reserved. This program and the accompanying materials\r
+ * are made available under the terms of the Eclipse Public License v1.0\r
+ * which accompanies this distribution, and is available at\r
+ * http://www.eclipse.org/legal/epl-v10.html\r
+ *\r
+ * Contributors:\r
+ *     VTT Technical Research Centre of Finland - initial API and implementation\r
+ *******************************************************************************/\r
+package org.simantics.trend.impl;\r
+\r
+import java.awt.Cursor;\r
+import java.awt.Toolkit;\r
+import java.awt.datatransfer.Clipboard;\r
+import java.awt.datatransfer.StringSelection;\r
+import java.awt.geom.Point2D;\r
+import java.awt.geom.Rectangle2D;\r
+import java.awt.print.PrinterException;\r
+import java.io.File;\r
+import java.io.IOException;\r
+\r
+import org.simantics.databoard.accessor.error.AccessorException;\r
+import org.simantics.databoard.util.Bean;\r
+import org.simantics.g2d.canvas.Hints;\r
+import org.simantics.g2d.canvas.ICanvasContext;\r
+import org.simantics.g2d.canvas.IMouseCursorHandle;\r
+import org.simantics.g2d.canvas.impl.AbstractCanvasParticipant;\r
+import org.simantics.g2d.canvas.impl.DependencyReflection.Dependency;\r
+import org.simantics.g2d.canvas.impl.HintReflection.HintListener;\r
+import org.simantics.g2d.canvas.impl.SGNodeReflection.SGCleanup;\r
+import org.simantics.g2d.chassis.ITooltipProvider;\r
+import org.simantics.g2d.participant.TimeParticipant;\r
+import org.simantics.history.HistoryException;\r
+import org.simantics.history.csv.CSVFormatter;\r
+import org.simantics.history.util.ProgressMonitor;\r
+import org.simantics.scenegraph.g2d.IG2DNode;\r
+import org.simantics.scenegraph.g2d.events.EventHandlerReflection.EventHandler;\r
+import org.simantics.scenegraph.g2d.events.MouseEvent;\r
+import org.simantics.scenegraph.g2d.events.MouseEvent.MouseButtonPressedEvent;\r
+import org.simantics.scenegraph.g2d.events.MouseEvent.MouseButtonReleasedEvent;\r
+import org.simantics.scenegraph.g2d.events.MouseEvent.MouseClickEvent;\r
+import org.simantics.scenegraph.g2d.events.MouseEvent.MouseExitEvent;\r
+import org.simantics.scenegraph.g2d.events.MouseEvent.MouseMovedEvent;\r
+import org.simantics.scenegraph.g2d.events.MouseEvent.MouseWheelMovedEvent;\r
+import org.simantics.scenegraph.g2d.events.TimeEvent;\r
+import org.simantics.scenegraph.g2d.events.command.CommandEvent;\r
+import org.simantics.scenegraph.g2d.events.command.Commands;\r
+import org.simantics.trend.configuration.ItemPlacement;\r
+import org.simantics.trend.configuration.TrendItem.Renderer;\r
+import org.simantics.trend.util.PrintUtil;\r
+import org.simantics.utils.datastructures.hints.IHintContext.Key;\r
+import org.simantics.utils.datastructures.hints.IHintContext.KeyOf;\r
+import org.simantics.utils.datastructures.hints.IHintObservable;\r
+\r
+public class TrendParticipant extends AbstractCanvasParticipant {\r
+\r
+    /**\r
+     * Key for storing chart value tip box relative position. Changed by\r
+     * {@link TrendParticipant} after user finishes dragging this class around.\r
+     * */\r
+    public static final Key KEY_VALUE_TIP_BOX_RELATIVE_POS = new KeyOf(Point2D.class);\r
+\r
+    /** Key for chart redraw interval in milliseconds (drawn only if dirty) */\r
+    public static final Key KEY_TREND_DRAW_INTERVAL = new KeyOf(Long.class);\r
+\r
+    /** The time code when trend was last redrawn */\r
+    public static final Key KEY_TREND_SHAPE_LAST = new KeyOf(Long.class);\r
+\r
+    /** Key for interval time of value scale */\r
+    public static final Key KEY_TREND_AUTOSCALE_INTERVAL = new KeyOf(Long.class);\r
+\r
+    /** Key for time of last autoscale */\r
+    public static final Key KEY_TREND_AUTOSCALE_LAST = new KeyOf(Long.class);\r
+\r
+    public static final double KEY_MOVE = 0.33;\r
+\r
+    /** Cursor when panning */\r
+    public MouseCursors cursors = new MouseCursors();\r
+\r
+    @HintListener(Class=Hints.class, Field="KEY_CONTROL_BOUNDS")\r
+    public void selectionChanged(IHintObservable sender, Key key, Object oldValue, Object newValue) {\r
+       trend.shapedirty = true;\r
+       setDirty();\r
+    }\r
+\r
+    TrendNode trend;\r
+\r
+    @Dependency TimeParticipant time;\r
+\r
+    Grab grab; \r
+\r
+    // The single item on which the mouse hovers over is set to this field. (Binary items are selectable). \r
+    public ItemNode hoveringItem;\r
+\r
+    @Override\r
+    public void addedToContext(ICanvasContext ctx) {\r
+        super.addedToContext(ctx);\r
+        time.registerForEvents(getClass());\r
+    }\r
+\r
+    public void setTrend(TrendNode node) {\r
+        trend = node;\r
+    }\r
+\r
+    public TrendNode getTrend() {\r
+        return trend;\r
+    }\r
+\r
+//    @SGInit\r
+//    public void initSG(G2DParentNode parent) {\r
+//     Set<TrendNode> nodes = NodeUtil.collectNodes(parent, TrendNode.class);\r
+//     trends.addAll(nodes);\r
+//    }\r
+\r
+    @SGCleanup\r
+    public void cleanupSG() {\r
+    }\r
+\r
+    protected void updateNode() {\r
+//     titleNode.setText(text)\r
+//        node.setEnabled(isPaintingEnabled());\r
+//        node.setGridColor(getGridColor());\r
+//        node.setGridSize(getGridSize());\r
+    }\r
+\r
+    @EventHandler(priority = 0)\r
+    public boolean handleTimeEvent(TimeEvent e) {\r
+        //System.out.println("(" + isRemoved() + ") time event: " + e.time + " (" + e.interval + ")");\r
+        if (isRemoved()) {\r
+            if (time != null)\r
+                time.unregisterForEvents(getClass());\r
+            return false;\r
+        }\r
+\r
+        long currentTime = e.time;\r
+\r
+        Long drawInterval = getHint(KEY_TREND_DRAW_INTERVAL);\r
+        if (drawInterval == null) drawInterval = 200L;\r
+        Long lastDrawTime = getHint(KEY_TREND_SHAPE_LAST);\r
+        boolean drawIntervalElapsed = lastDrawTime == null || (currentTime>=lastDrawTime+drawInterval);\r
+        if (drawIntervalElapsed) {\r
+//            System.out.println("elapsed="+drawIntervalElapsed+" currentTime="+currentTime+" lastDraw="+lastDrawTime+", interval="+drawInterval+", nextDrawTime="+(lastDrawTime+drawInterval));\r
+//            System.out.println("elapsed="+drawIntervalElapsed+" currentTime="+currentTime+" lastDraw="+lastDrawTime+", interval="+drawInterval);\r
+            setHint(KEY_TREND_SHAPE_LAST, currentTime);\r
+            trend.shapedirty |= trend.datadirty; \r
+            trend.datadirty = false;\r
+        }\r
+\r
+        // Scale time\r
+        Long autoscaleInterval = getHint(KEY_TREND_AUTOSCALE_INTERVAL);\r
+        if (autoscaleInterval == null) autoscaleInterval = 1000L;\r
+        Long autoscaleTime = getHint(KEY_TREND_AUTOSCALE_LAST);\r
+        boolean autoscale = autoscaleTime==null || (currentTime>=autoscaleTime+autoscaleInterval);\r
+        if (autoscale) {\r
+            boolean l = trend.autoscale(trend.autoscaletime, autoscale);\r
+            if (!draggingValueTip(grab)) {\r
+                trend.updateValueTipTime();\r
+            }\r
+            if (l) {\r
+                trend.layout();\r
+            }\r
+\r
+            setHint(KEY_TREND_AUTOSCALE_LAST, currentTime);\r
+        }\r
+\r
+        // IT IS DRAW TIME\r
+        if (trend.shapedirty) {\r
+//            System.out.println(currentTime+": Draw time!");\r
+            setDirty();\r
+        }\r
+\r
+        return false;\r
+    }\r
+\r
+    @EventHandler(priority = 0)\r
+    public boolean mouseClick(MouseClickEvent me) {\r
+               IG2DNode node = trend.pickNode( me.controlPosition );\r
+               \r
+       if (me.clickCount==2 && node!=null) {\r
+               trend.valueTipTime = trend.prevValueTipTime;\r
+                       if (node instanceof Plot) {\r
+                               boolean binaryArea = me.controlPosition.getY()>trend.plot.getY()+trend.plot.analogAreaHeight;\r
+                               if (binaryArea) {\r
+                                       trend.horizRuler.zoomOut();\r
+                               } else {\r
+                                       trend.zoomOut();\r
+                               }\r
+                               setDirty();\r
+                               return true;\r
+                       } else if (node instanceof HorizRuler) {\r
+                               HorizRuler hr = (HorizRuler) node;\r
+                               hr.zoomOut();\r
+                               setDirty();\r
+                               return true;\r
+                       } else if (node instanceof VertRuler) {\r
+                               VertRuler vr = (VertRuler) node;\r
+                               vr.zoomOut();\r
+                               setDirty();\r
+                               return true;\r
+                       }\r
+       }\r
+       \r
+       // Click ValueTip \r
+       if (me.clickCount==1 && me.button==MouseEvent.LEFT_BUTTON && node!=null && node instanceof Plot) {\r
+               trend.prevValueTipTime = trend.valueTipTime;\r
+                       if (trend.valueTipTime == null) {\r
+                       trend.valueTipTime = Double.isNaN(trend.mouseHoverTime)?null:trend.mouseHoverTime;\r
+                       trend.valueTipHover = true;\r
+                       } else {\r
+                               double valueTipX = trend.horizRuler.toX( trend.valueTipTime );\r
+                       double x = me.controlPosition.getX() - trend.horizRuler.getBounds().getX();\r
+                       boolean hit = x>=valueTipX-TrendLayout.VALUE_TIP_HOVER_SENSITIVITY && x<=valueTipX+TrendLayout.VALUE_TIP_HOVER_SENSITIVITY;\r
+                       if (hit) trend.valueTipTime = null;\r
+                       }\r
+                       setDirty();\r
+       }\r
+\r
+       /*\r
+       // Right click removes value tip\r
+       if (me.clickCount==1 && me.button==MouseEvent.RIGHT_BUTTON) {\r
+                       if (trend.valueTipTime != null) {\r
+                               trend.valueTipTime = null;\r
+                               setDirty();\r
+                       }               \r
+       }*/\r
+\r
+       if (me.clickCount==1 && me.button==MouseEvent.LEFT_BUTTON) {\r
+                       if (node instanceof VertRuler) {\r
+                               VertRuler vr = (VertRuler) node;\r
+                               trend.selectVertRuler( trend.vertRulers.indexOf(vr) );\r
+                               setDirty();\r
+                       } else if (node == null || node instanceof TextNode) {\r
+                               Plot p = trend.plot;\r
+                               double x = me.controlPosition.getX() - p.getX();\r
+                               double y = me.controlPosition.getY() - p.getY();\r
+                               if ( x>=0 && x<=p.getWidth() && y<=0 && y>=-Plot.DIAMOND_SIZE*2) {\r
+                                       // Click hits milestone area\r
+                                       milestoneSearch: for ( Milestone ms : trend.milestones.milestones ) {\r
+                                               double mx = trend.horizRuler.toX( ms.time );\r
+                                               if ( x>=mx-Plot.DIAMOND_SIZE && x<=mx+Plot.DIAMOND_SIZE ) {\r
+                                                       ITooltipProvider tp = getContext().getTooltipProvider();\r
+                                                       if (tp!=null) {\r
+                                                               tp.show(ms.label, ms.description);\r
+                                                       }\r
+                                                       break milestoneSearch;\r
+                                               }\r
+                                       }\r
+                               }\r
+\r
+                       }\r
+\r
+       }\r
+               return false;\r
+    }\r
+\r
+    @EventHandler(priority = 0)\r
+    public boolean mouseDown(MouseButtonPressedEvent me) {\r
+       // Start Box-Zoom\r
+       if (me.button==MouseEvent.LEFT_BUTTON) {\r
+                       IG2DNode node = trend.pickNode( me.controlPosition );   \r
+                       if (node == null || node instanceof Plot) {\r
+\r
+                // Start value box grab\r
+                {\r
+                    double x = me.controlPosition.getX() - trend.horizRuler.getBounds().getX();\r
+                    double y = me.controlPosition.getY() - trend.plot.getBounds().getY();\r
+                    if (trend.plot.valueTipBoxBounds.contains(x, y)) {\r
+                        //System.out.println("grabbed value box @ " + x + ", " + y);\r
+                        grab = new Grab();\r
+                        grab.valueBoxRelPos.setLocation(trend.spec.viewProfile.valueViewPositionX, trend.spec.viewProfile.valueViewPositionY);\r
+                        grab.valueBoxPos.setFrame(trend.plot.valueTipBoxBounds);\r
+                        grab.mousepos = new Point2D.Double( me.controlPosition.getX(), me.controlPosition.getY() );\r
+                        grab.valueBoxGrab = true;\r
+                        return true;\r
+                    }\r
+                }\r
+\r
+                               // Start Value-Tip grab\r
+                               if (trend.valueTipTime!=null) {\r
+                                       double valueTipX = trend.horizRuler.toX( trend.valueTipTime );\r
+                               double x = me.controlPosition.getX() - trend.horizRuler.getBounds().getX();\r
+                               boolean hit = x>=valueTipX-TrendLayout.VALUE_TIP_HOVER_SENSITIVITY && x<=valueTipX+TrendLayout.VALUE_TIP_HOVER_SENSITIVITY; \r
+                               if (hit) {\r
+                                       grab = new Grab();\r
+                                       grab.mousepos = new Point2D.Double( me.controlPosition.getX(), me.controlPosition.getY() );\r
+                                       grab.sx = trend.horizRuler.unitsPerPixel();\r
+                                       grab.sy = new double[ trend.vertRulers.size() ];\r
+                                       grab.valueTipGrab = true;\r
+                                       return true;\r
+                               }\r
+                               }\r
+\r
+                       // Start Box-Zoom\r
+                               boolean binaryArea = me.controlPosition.getY()>trend.plot.getY()+trend.plot.analogAreaHeight;\r
+                               boolean timeZoom = (me.stateMask & MouseEvent.SHIFT_MASK)>0 || binaryArea;\r
+                               if (trend.selection==null) {\r
+                                       trend.selection = trend.addNode("Selection", SelectionNode.class);\r
+                                       trend.selection.setZIndex( 10 );\r
+                                       trend.selection.start( me.controlPosition, binaryArea, timeZoom );\r
+                               }\r
+                       return true;\r
+                       }\r
+       }\r
+\r
+       // Start grab\r
+               IG2DNode node = trend.pickNode( me.controlPosition );\r
+       if ( (me.button==MouseEvent.MIDDLE_BUTTON) && \r
+                       node!=null && node instanceof Plot) {\r
+                       //Plot p = (Plot) node;\r
+                       TrendNode trend = (TrendNode) node.getParent();\r
+                       boolean shift = (me.stateMask & MouseEvent.SHIFT_MASK)>0;\r
+                       boolean alt = (me.stateMask & MouseEvent.ALT_MASK)>0;\r
+                       //boolean analogArea = me.controlPosition.getY() < p.getY()+p.analogAreaHeight;\r
+                       //boolean binaryArea = me.controlPosition.getY() >= p.getY()+p.analogAreaHeight;\r
+               grab = new Grab();\r
+               grab.mousepos = new Point2D.Double( me.controlPosition.getX(), me.controlPosition.getY() );\r
+               grab.sx = trend.horizRuler.unitsPerPixel();\r
+               grab.sy = new double[ trend.vertRulers.size() ];\r
+               grab.horiz = !shift || alt;\r
+               grab.vert = shift || alt;\r
+               if (grab.vert) {\r
+                       for (int i=0; i<trend.vertRulers.size(); i++)\r
+                               grab.sy[i] = trend.vertRulers.get(i).unitsPerPixel();\r
+               }\r
+               Cursor c = grab.horiz ? (grab.vert ? cursors.grab : cursors.grab_horiz) : (grab.vert ? cursors.grab_vert : cursors.grab); \r
+               grab.cursor = getContext().getMouseCursorContext().setCursor(me.mouseId, c);\r
+               grab.mouseButton = me.button;\r
+               grab.plot = true;\r
+                       trend.horizRuler.translate(0);\r
+                       setHoverTime(null);\r
+               setDirty();\r
+                       return true;\r
+       }\r
+               if ( (me.button==MouseEvent.LEFT_BUTTON||me.button==MouseEvent.MIDDLE_BUTTON) && \r
+                               node!=null && node instanceof HorizRuler) {\r
+                       HorizRuler hr = (HorizRuler) node;\r
+               grab = new Grab();\r
+               grab.cursor = getContext().getMouseCursorContext().setCursor(me.mouseId, cursors.grab_horiz);\r
+               grab.mousepos = new Point2D.Double();\r
+               grab.mousepos.setLocation( me.controlPosition );\r
+               grab.sx = hr.unitsPerPixel();\r
+               grab.horizRuler = true;\r
+               grab.mouseButton = me.button;\r
+               grab.horiz = true;\r
+                       trend.horizRuler.translate(0);\r
+               setDirty();\r
+                       return true;\r
+               }\r
+               if ( (me.button==MouseEvent.LEFT_BUTTON||me.button==MouseEvent.MIDDLE_BUTTON) && \r
+                               node!=null && node instanceof VertRuler) {\r
+                       //VertRuler vr = (VertRuler) node;\r
+               grab = new Grab();\r
+               grab.cursor = getContext().getMouseCursorContext().setCursor(me.mouseId, cursors.grab_vert);\r
+               grab.mousepos = new Point2D.Double();\r
+               grab.mousepos.setLocation( me.controlPosition );\r
+               grab.sy = new double[ trend.vertRulers.size() ];\r
+               for (int i=0; i<trend.vertRulers.size(); i++)\r
+                       grab.sy[i] = trend.vertRulers.get(i).unitsPerPixel();\r
+               grab.mouseButton = me.button;\r
+               grab.vertRuler = trend.vertRulers.indexOf(node);\r
+                       trend.selectVertRuler( grab.vertRuler );\r
+               setDirty();\r
+                       return true;\r
+       }\r
+       \r
+       return false;\r
+    }\r
+\r
+    @EventHandler(priority = 0)\r
+    public boolean mouseUp(MouseButtonReleasedEvent me) {\r
+\r
+        // Release value box grab\r
+        if (me.button==1 && draggingValueBox(grab)) {\r
+            setHint(KEY_VALUE_TIP_BOX_RELATIVE_POS, new Point2D.Double(\r
+                    trend.spec.viewProfile.valueViewPositionX,\r
+                    trend.spec.viewProfile.valueViewPositionY));\r
+            grab = null;\r
+        }\r
+\r
+        // Release value tip grab\r
+        if (me.button==1 && draggingValueTip(grab)) {\r
+            grab = null;\r
+        }\r
+\r
+       // Release Box-Zoom\r
+       if (me.button==1 && trend.selection!=null) {\r
+               Rectangle2D rect = trend.selection.rect;\r
+               \r
+               if (rect.getWidth()>1 && rect.getHeight()>1) {    \r
+                       if (trend.selection.timeZoom) {\r
+                               trend.horizRuler.zoomIn( rect.getX()-trend.horizRuler.getX(), rect.getWidth() );\r
+                       } else {\r
+                               trend.zoomIn( \r
+                                               rect.getX()-trend.plot.getX(), \r
+                                               rect.getY()-trend.plot.getY(),\r
+                                               rect.getWidth(), rect.getHeight(), true, true );\r
+                       }\r
+                       trend.layout();\r
+                               trend.shapedirty = true;\r
+                       setDirty();\r
+               }\r
+               trend.selection.delete();\r
+               trend.selection = null;\r
+\r
+               return true;\r
+       }\r
+       \r
+       if (grab!=null && grab.mouseButton==me.button) {\r
+               grab.cursor.remove();\r
+               grab = null;\r
+       }\r
+       return false;\r
+    }\r
+\r
+    @EventHandler(priority = 0)\r
+    public boolean mouseExit(MouseExitEvent me) {\r
+               trend.valueTipHover = false;\r
+       setHoverTime( null );\r
+       return false;\r
+    }\r
+\r
+    void setHoverTime(Double time) {\r
+       if ( time==null && Double.isNaN(trend.mouseHoverTime) ) return;\r
+       if ( time!=null && trend.mouseHoverTime==time ) return;\r
+       if ( time==null ) {\r
+                       trend.mouseHoverTime = Double.NaN;\r
+       } else {\r
+               trend.mouseHoverTime = time;\r
+               trend.lastMouseHoverTime = time;\r
+       }\r
+       setDirty();\r
+    }\r
+\r
+//    @EventHandler(priority = 0)\r
+//    public boolean mouseEnter(MouseExitEvent me) {\r
+//     //double x = me.controlPosition.getX() - trend.horizRuler.getBounds().getX();\r
+//     //double time = trend.horizRuler.toTime( x ) - trend.horizRuler.basetime;\r
+//     return false;\r
+//    }\r
+\r
+    @SuppressWarnings("unused")\r
+    @EventHandler(priority = 0)\r
+    public boolean mouseMoved(MouseMovedEvent me) {\r
+//        ITooltipProvider tt = getContext().getTooltipProvider();\r
+//        if (tt!=null) {\r
+//            tt.setTooltipText("Hello");\r
+//        }\r
+\r
+               IG2DNode pick = trend.pickNode( me.controlPosition );\r
+       \r
+               TrackMouseMovement: {\r
+                       if ( pick == trend.plot ) {\r
+                       double x = me.controlPosition.getX() - trend.horizRuler.getBounds().getX();\r
+                       double time = trend.horizRuler.toTime( x );\r
+                       double sx = (trend.horizRuler.end-trend.horizRuler.from) / trend.horizRuler.getWidth();\r
+                       double timeSnapTolerance = sx * 7.0; \r
+                               boolean shift = (me.stateMask & MouseEvent.SHIFT_MASK)>0;\r
+                       \r
+                               // SNAP - When shift is held\r
+                       if (shift) {\r
+                               Double snappedTime = null;\r
+                                       try {\r
+                                               snappedTime = trend.snapToValue( time, timeSnapTolerance );\r
+                                               if (snappedTime != null) time = snappedTime;\r
+                                       } catch (HistoryException e) {\r
+                                       } catch (AccessorException e) {\r
+                                       }\r
+                       }\r
+                               \r
+                               setHoverTime(time);\r
+                       }\r
+               }\r
+\r
+        // Implement value box moving while dragging.\r
+        if (draggingValueBox(grab)) {\r
+            Rectangle2D plotBox = trend.plot.getBounds();\r
+            Rectangle2D valueBoxPos = grab.valueBoxPos;\r
+            Rectangle2D valueBox = trend.plot.valueTipBoxBounds;\r
+            double dx = me.controlPosition.getX() - grab.mousepos.getX();\r
+            double dy = me.controlPosition.getY() - grab.mousepos.getY();\r
+            double margin = Plot.VALUE_TIP_BOX_PLOT_MARGIN;\r
+            double maxX = plotBox.getWidth() - margin - valueBox.getWidth();\r
+            double maxY = plotBox.getHeight() - margin - valueBox.getHeight();\r
+            double boxX = valueBoxPos.getX() + dx;\r
+            double boxY = valueBoxPos.getY() + dy;\r
+            double pw = plotBox.getWidth() - 2 * margin;\r
+            double ph = plotBox.getHeight() - 2 * margin;\r
+            double w = pw - valueBox.getWidth();\r
+            double h = ph - valueBox.getHeight();\r
+            if (w > 0 && h > 0) {\r
+                double rx = (boxX - margin) / w;\r
+                double ry = (boxY - margin) / h;\r
+                rx = Math.max(0, Math.min(rx, 1));\r
+                ry = Math.max(0, Math.min(ry, 1));\r
+                trend.spec.viewProfile.valueViewPositionX = rx;\r
+                trend.spec.viewProfile.valueViewPositionY = ry;\r
+                setDirty();\r
+            }\r
+            return false;\r
+        }\r
+\r
+        ValueToolTip: if ( pick == trend.plot) {\r
+            if (draggingValueTip(grab)) {\r
+                               // Move grabbed value-tip\r
+                               trend.valueTipTime = Double.isNaN(trend.mouseHoverTime) ? null : trend.mouseHoverTime;\r
+                               return false;\r
+                       } else {\r
+                               // Track hover color\r
+                               if (trend.valueTipTime!=null) {\r
+                                       double valueTipX = trend.horizRuler.toX( trend.valueTipTime );\r
+                               double x = me.controlPosition.getX() - trend.horizRuler.getBounds().getX();\r
+                               boolean hit = x>=valueTipX-TrendLayout.VALUE_TIP_HOVER_SENSITIVITY && x<=valueTipX+TrendLayout.VALUE_TIP_HOVER_SENSITIVITY;\r
+                               trend.valueTipHover = hit;\r
+                               }\r
+                       }\r
+       }\r
+               \r
+               // Item pick\r
+               ItemPick: {\r
+               if ( pick instanceof VertRuler ) {\r
+                       VertRuler vr = (VertRuler) pick;\r
+                       hoveringItem = null;\r
+                       for (ItemNode item : trend.analogItems) {\r
+                               if (item.item.renderer != Renderer.Analog || item.ruler==vr) {\r
+                                       if ( hoveringItem != null) break;\r
+                                       hoveringItem = item;\r
+                               }\r
+                       }\r
+               } else if ( pick == null || pick instanceof Plot ) {\r
+                       hoveringItem = trend.plot.pickItem( me.controlPosition );\r
+               } else {\r
+                       hoveringItem = null;\r
+               }\r
+       }\r
+       \r
+       // Box-Zoom\r
+       if (trend.selection!=null) {\r
+               trend.selection.setEndPoint( me.controlPosition );\r
+               setHoverTime(null);\r
+               setDirty();\r
+               return true;\r
+       }\r
+       \r
+       // Drag axis\r
+       if (grab != null) {\r
+               double dx = me.controlPosition.getX() - grab.mousepos.x;\r
+               double dy = me.controlPosition.getY() - grab.mousepos.y;\r
+               grab.mousepos.setLocation(me.controlPosition);\r
+               \r
+               if (grab.plot) {\r
+                       if (grab.horiz) {\r
+                               trend.horizRuler.translate(-dx*grab.sx);\r
+                       }\r
+                       if (grab.vert) {\r
+                               for (int i=0; i<grab.sy.length; i++) {\r
+                                       if (i>=trend.vertRulers.size()) break;\r
+                                       trend.vertRulers.get(i).translate(dy*grab.sy[i]);\r
+                               }\r
+                       }\r
+               } else if (grab.horizRuler) {\r
+                       trend.horizRuler.translate(-dx*grab.sx);\r
+               } else if (grab.vertRuler>=0) {\r
+                       if (grab.vertRuler<=trend.vertRulers.size()) {\r
+                               VertRuler vr = trend.vertRulers.get( grab.vertRuler );\r
+                               trend.selectVertRuler( grab.vertRuler );\r
+                               vr.translate(dy*grab.sy[ grab.vertRuler ]);\r
+                       }\r
+               }\r
+                       trend.layout();\r
+               trend.shapedirty = true;\r
+               setDirty();\r
+               return true;\r
+       }\r
+       return false;\r
+    }\r
+\r
+    @EventHandler(priority = 0)\r
+    public boolean mouseWheel(MouseWheelMovedEvent me) {\r
+               IG2DNode node = trend.pickNode( me.controlPosition );\r
+               if (node instanceof Plot || node instanceof HorizRuler || node instanceof VertRuler) {\r
+                       Point2D pt = node.parentToLocal( me.controlPosition );\r
+                       boolean shift = (me.stateMask & MouseEvent.SHIFT_MASK)>0;\r
+                       boolean alt = (me.stateMask & MouseEvent.ALT_MASK)>0;\r
+                       double pw = trend.plot.getWidth();\r
+                       double ph = trend.plot.analogAreaHeight;\r
+\r
+                       double r = Math.pow(0.9, me.wheelRotation);\r
+\r
+                       double w = r * pw; \r
+                       double h = r * ph;\r
+                       double rx = pt.getX() / pw;\r
+                       double ry = pt.getY() / ph;\r
+\r
+                       double zx = (pw-w)*rx;\r
+                       double zy = (ph-h)*ry;\r
+\r
+                       if (node instanceof Plot) {\r
+                               TrendNode trend = (TrendNode) node.getParent();\r
+                               if (shift||alt) {\r
+                                       for (VertRuler vr : trend.vertRulers) {\r
+                                               vr.zoomIn(zy, h);\r
+                                       }\r
+                               } \r
+                               if (!shift) {\r
+                                       trend.horizRuler.zoomIn(zx, w);\r
+                               }\r
+                       }\r
+\r
+                       if (node instanceof HorizRuler) {\r
+                               //HorizRuler hr = (HorizRuler) node;\r
+                               trend.horizRuler.zoomIn(zx, w);\r
+                       }\r
+\r
+                       if (node instanceof VertRuler) {\r
+                               VertRuler vr = (VertRuler) node;\r
+                               trend.selectVertRuler( trend.vertRulers.indexOf(vr) );\r
+                               vr.zoomIn(zy, h);\r
+                       }\r
+                       trend.shapedirty = true;\r
+                       trend.layout();\r
+                       setDirty();\r
+                       return true;\r
+               }\r
+       return false;\r
+    }\r
+\r
+    @EventHandler(priority = 0)\r
+    public boolean handleCommandEvent(CommandEvent e) {\r
+       \r
+       if (e.command == Commands.CANCEL) {\r
+               if (trend.selection != null) {\r
+                       trend.selection.delete();\r
+                       trend.selection = null;\r
+                       setDirty();\r
+                       return true;\r
+               }\r
+               \r
+               if (grab!=null) {\r
+                if (draggingValueBox(grab)) {\r
+                    trend.spec.viewProfile.valueViewPositionX = grab.valueBoxRelPos.getX();\r
+                    trend.spec.viewProfile.valueViewPositionY = grab.valueBoxRelPos.getY();\r
+                }\r
+                       if (grab.cursor!=null) grab.cursor.remove();\r
+                       grab = null;\r
+                       setDirty();\r
+                       return true;\r
+               }\r
+               \r
+               if (trend.valueTipTime != null) {\r
+                       trend.valueTipTime = null;\r
+                       setDirty();\r
+                       return true;\r
+               }\r
+               return false;\r
+       }\r
+       \r
+       if (e.command == Commands.PAN_LEFT) {\r
+                       double pw = trend.plot.getWidth();\r
+                       double ph = trend.plot.analogAreaHeight;\r
+                       double zx = -pw * KEY_MOVE;\r
+                       double zy = 0;\r
+                       trend.zoomIn(zx, zy, pw, ph, true, true);\r
+                       trend.layout();\r
+                       setDirty();\r
+       }\r
+       \r
+       if (e.command == Commands.PAN_RIGHT) {\r
+                       double pw = trend.plot.getWidth();\r
+                       double ph = trend.plot.analogAreaHeight;\r
+                       double zx = +pw * KEY_MOVE;\r
+                       double zy = 0;\r
+                       trend.zoomIn(zx, zy, pw, ph, true, true);\r
+                       trend.horizRuler.autoscroll = false;\r
+                       trend.layout();\r
+                       setDirty();\r
+       }\r
+       \r
+       if (e.command == Commands.PAN_UP) {\r
+                       double pw = trend.plot.getWidth();\r
+                       double ph = trend.plot.analogAreaHeight;\r
+                       double zx = 0;\r
+                       double zy = -ph * KEY_MOVE;\r
+                       trend.zoomIn(zx, zy, pw, ph, true, true);\r
+                       trend.horizRuler.autoscroll = false;\r
+                       trend.layout();\r
+                       setDirty();\r
+       }\r
+       \r
+       if (e.command == Commands.PAN_DOWN) {\r
+                       double pw = trend.plot.getWidth();\r
+                       double ph = trend.plot.analogAreaHeight;\r
+                       double zx = 0;\r
+                       double zy = +ph * KEY_MOVE;\r
+                       trend.zoomIn(zx, zy, pw, ph, true, true);\r
+                       trend.horizRuler.autoscroll = false;\r
+                       trend.layout();\r
+                       setDirty();\r
+       }\r
+       \r
+       // Zoom out to time window settings (VK_4)\r
+       if (e.command.equals(Commands.AUTOSCALE)) {\r
+                       trend.horizRuler.autoscroll = true;\r
+                       for (VertRuler vertRuler : trend.vertRulers) vertRuler.autoscroll = true;\r
+                       trend.zoomOut();\r
+                       trend.layout();\r
+                       setDirty();\r
+       }\r
+       \r
+       // Fit all visible (VK_1)\r
+       if (e.command.equals(Commands.ZOOM_TO_FIT)) {\r
+                       trend.readMinMaxFromEnd();\r
+                       trend.horizRuler.setFromEnd(trend.horizRuler.iFrom, trend.horizRuler.iEnd);\r
+                       trend.horizRuler.fireListener();\r
+                       int c = trend.vertRulers.size();\r
+                       for (int i=0; i<c; i++) {\r
+                               VertRuler vr = trend.vertRulers.get(c-i-1);\r
+                               double nMin = vr.iMin;\r
+                               double nMax = vr.iMax;\r
+\r
+                               double diff = nMax - nMin;\r
+                               if (diff==0.0) {\r
+                                       nMin -= 0.5;\r
+                                       nMax += 0.5;\r
+                                       diff = nMax - nMin;\r
+                               }\r
+                               double margin = diff*0.02;\r
+                               \r
+                               if (trend.itemPlacement == ItemPlacement.Stacked) {\r
+                                       nMin = nMin - (diff)*i - margin;\r
+                                       nMax = nMax + (diff)*(c-i-1) + margin;\r
+                                       \r
+                               } else {\r
+                                       nMin = vr.iMin - margin;\r
+                                       nMax = vr.iMax + margin;\r
+                               }\r
+                               \r
+                               vr.zoomTo(nMin, nMax);\r
+                       }\r
+                       trend.horizRuler.autoscroll = false;\r
+                       trend.layout();\r
+                       setDirty();\r
+                       return true;\r
+       }\r
+\r
+       // Fit horiz (VK_2)\r
+       if (e.command.equals(Commands.ZOOM_TO_FIT_HORIZ)) {\r
+                       trend.readMinMaxFromEnd();\r
+                       trend.horizRuler.setFromEnd(trend.horizRuler.iFrom, trend.horizRuler.iEnd);\r
+                       trend.layout();\r
+                       trend.horizRuler.autoscroll = false;\r
+                       setDirty();\r
+                       return true;\r
+       }\r
+\r
+       // Fit vert (VK_3)\r
+       if (e.command.equals(Commands.ZOOM_TO_FIT_VERT)) {\r
+                       trend.readMinMaxFromEnd();\r
+                       int c = trend.vertRulers.size();\r
+                       for (int i=0; i<c; i++) {\r
+                               VertRuler vr = trend.vertRulers.get(c-i-1);\r
+                               double nMin = vr.iMin;\r
+                               double nMax = vr.iMax;\r
+\r
+                               double diff = nMax - nMin;\r
+                               if (diff==0.0) {\r
+                                       nMin -= 0.5;\r
+                                       nMax += 0.5;\r
+                                       diff = nMax - nMin;\r
+                               }\r
+                               double margin = diff*0.02;\r
+                               \r
+                               if (trend.itemPlacement == ItemPlacement.Stacked) {\r
+                                       nMin = nMin - (diff)*i - margin;\r
+                                       nMax = nMax + (diff)*(c-i-1) + margin;\r
+                                       \r
+                               } else {\r
+                                       nMin = vr.iMin - margin;\r
+                                       nMax = vr.iMax + margin;\r
+                               }\r
+                               \r
+                               vr.zoomTo(nMin, nMax);\r
+                       }\r
+                       trend.layout();\r
+                       setDirty();\r
+                       return true;\r
+       }\r
+       \r
+       // +\r
+       if (e.command == Commands.ZOOM_IN) {\r
+                       double pw = trend.plot.getWidth();\r
+                       double ph = trend.plot.analogAreaHeight;\r
+                       double za = 3; // Zoom amount\r
+                       double w = (1. - (0.1*za)) * pw; \r
+                       double h = (1. - (0.1*za)) * ph;\r
+\r
+                       double zx = (pw-w)/2;\r
+                       double zy = (ph-h)/2;\r
+                       \r
+                       trend.zoomIn(zx, zy, w, h, true, true);\r
+                       trend.horizRuler.autoscroll = false;\r
+                       trend.layout();\r
+                       setDirty();\r
+                       return true;\r
+       }\r
+       \r
+       // -\r
+       if (e.command == Commands.ZOOM_OUT) {\r
+                       double pw = trend.plot.getWidth();\r
+                       double ph = trend.plot.analogAreaHeight;\r
+                       double za = -3; // Zoom amount\r
+                       double w = (1. - (0.1*za)) * pw; \r
+                       double h = (1. - (0.1*za)) * ph;\r
+\r
+                       double zx = (pw-w)/2;\r
+                       double zy = (ph-h)/2;\r
+                       \r
+                       trend.zoomIn(zx, zy, w, h, true, true);\r
+                       trend.horizRuler.autoscroll = false;\r
+                       trend.layout();\r
+                       setDirty();\r
+                       return true;\r
+       }\r
+       \r
+               // Print to printer\r
+               if (e.command == Commands.PRINT) {\r
+                       try {\r
+                               PrintUtil pu = new PrintUtil();\r
+                               pu.addTrendPage(trend);\r
+                               pu.print();\r
+                       } catch (PrinterException e1) {\r
+                               e1.printStackTrace();\r
+                       }\r
+                       return true;\r
+               }\r
+               \r
+               // Print to PDF\r
+               if (e.command == Commands.PDFPRINT) {\r
+                       PrintUtil pu = new PrintUtil();\r
+                       pu.addTrendPage(trend);\r
+                       try {\r
+                               File f = File.createTempFile("Trend", ".pdf");\r
+                               pu.printPdf(f);\r
+                               System.out.println("Printed Trend to "+f);\r
+                       } catch (IOException e1) {\r
+                               e1.printStackTrace();\r
+                       }\r
+                       return true;\r
+               }\r
+               \r
+               // Read visible values into CSV \r
+               if (e.command == Commands.COPY || e.command == Commands.EXPORT) {\r
+                       StringBuilder sb = new StringBuilder(64*1024);\r
+                       try {\r
+                               // TODO: fix to use preferences.\r
+                               CSVFormatter formatter = new CSVFormatter();\r
+                               formatter.setTimeRange(trend.horizRuler.from, trend.horizRuler.end);\r
+                               for (ItemNode i : trend.allItems)\r
+                               {\r
+                                       if (i.item.hidden) continue;\r
+                                       if (i.historyItems==null || i.historyItems.length==0) continue;\r
+                                       Bean bestQualityStream = i.historyItems[0];\r
+                                       String historyItemId = (String) bestQualityStream.getFieldUnchecked("id");\r
+                                       formatter.addItem( trend.historian, historyItemId, i.item.simpleLabel, i.item.variableReference, i.item.unit);\r
+                               }\r
+                               formatter.sort();\r
+                               try {\r
+                                   // TODO: Use a proper user-cancelable progressmonitor\r
+                                       formatter.formulate2(new ProgressMonitor.Stub(), sb);\r
+                               } catch (IOException e1) {\r
+                                       // sb cannot throw ioexception\r
+                               }\r
+                       } catch (HistoryException e1) {\r
+                               e1.toString();\r
+                       }\r
+                       Toolkit toolkit = Toolkit.getDefaultToolkit();\r
+                       Clipboard clipboard = toolkit.getSystemClipboard();\r
+                       StringSelection strSel = new StringSelection(sb.toString());\r
+                       clipboard.setContents(strSel, null);\r
+               }\r
+               return false;\r
+       }\r
+        \r
+    static class Grab {\r
+        int mouseButton;\r
+       IMouseCursorHandle cursor;\r
+       Point2D.Double mousepos = new Point2D.Double();\r
+       boolean plot = false;\r
+       boolean horizRuler = false;\r
+       boolean vert = false;\r
+       boolean horiz = false;\r
+       int vertRuler = -1;\r
+       double sx = 1;\r
+       double[] sy;\r
+        boolean valueTipGrab = false;\r
+        boolean valueBoxGrab = false;\r
+        Point2D valueBoxRelPos = new Point2D.Double(); \r
+        Rectangle2D valueBoxPos = new Rectangle2D.Double();\r
+    }\r
+\r
+    private static boolean draggingValueTip(Grab g) {\r
+        return g != null && g.valueTipGrab;\r
+    }\r
+\r
+    private static boolean draggingValueBox(Grab g) {\r
+        return g != null && g.valueBoxGrab;\r
+    }\r
+\r
+}\r