-/*******************************************************************************\r
- * Copyright (c) 2007, 2010 Association for Decentralized Information Management\r
- * in 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
+/*******************************************************************************
+ * Copyright (c) 2007, 2010 Association for Decentralized Information Management
+ * in Industry THTH ry.
+ * All rights reserved. This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution, and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ * VTT Technical Research Centre of Finland - initial API and implementation
+ *******************************************************************************/
package org.simantics.scenegraph.g2d.nodes;
-import java.awt.AlphaComposite;\r
-import java.awt.BasicStroke;\r
-import java.awt.Color;\r
-import java.awt.Font;\r
-import java.awt.FontMetrics;\r
-import java.awt.Graphics2D;\r
-import java.awt.geom.AffineTransform;\r
-import java.awt.geom.Rectangle2D;\r
-import java.util.Locale;\r
-\r
-import org.simantics.scenegraph.g2d.G2DNode;\r
-import org.simantics.scenegraph.utils.GridUtils;\r
+import java.awt.AlphaComposite;
+import java.awt.BasicStroke;
+import java.awt.Color;
+import java.awt.Font;
+import java.awt.FontMetrics;
+import java.awt.Graphics2D;
+import java.awt.geom.AffineTransform;
+import java.awt.geom.Rectangle2D;
+import java.util.Locale;
+
+import org.simantics.scenegraph.g2d.G2DNode;
+import org.simantics.scenegraph.utils.DPIUtil;
+import org.simantics.scenegraph.utils.GridUtils;
public class RulerNode extends G2DNode {
/**
*
*/
- private static final long serialVersionUID = 2490944880914577411L;\r
-\r
- /**\r
- * FIXME: this is a hack for the map UI that has to be solved some other way.\r
- */\r
- private static final boolean MAP_Y_SCALING = false;\r
-\r
- private static final Color GRAY = new Color(100, 100, 100);\r
-\r
- protected Boolean enabled = true;\r
-\r
- protected double gridSize = 1.0;\r
-\r
- @SyncField("enabled")\r
- public void setEnabled(Boolean enabled) {\r
- this.enabled = enabled;\r
- }\r
-\r
- @SyncField("gridSize")\r
- public void setGridSize(double gridSize) {\r
- if (gridSize < 1e-6)\r
- gridSize = 1e-6;\r
- this.gridSize = gridSize;\r
- }\r
-\r
+ private static final long serialVersionUID = 2490944880914577411L;
+
+ private static final Color GRAY = new Color(100, 100, 100);
+
+ protected Boolean enabled = true;
+
+ protected double gridSize = 1.0;
+
+ protected double rulerSize = 20;
+
+ @SyncField("enabled")
+ public void setEnabled(Boolean enabled) {
+ this.enabled = enabled;
+ }
+
+ @SyncField("gridSize")
+ public void setGridSize(double gridSize) {
+ if (gridSize < 1e-6)
+ gridSize = 1e-6;
+ this.gridSize = gridSize;
+ }
+
+ @SyncField("rulerSize")
+ public void setRulerSize(double rulerSize) {
+ this.rulerSize = rulerSize;
+ }
+
@Override
public void render(Graphics2D g) {
- if (!enabled)\r
- return;\r
-\r
+ if (!enabled)
+ return;
+
AffineTransform tr = g.getTransform();
double scaleX = Math.abs(tr.getScaleX());
double scaleY = Math.abs(tr.getScaleY());
double offsetY = tr.getTranslateY();
g.setTransform(new AffineTransform());
- Font rulerFont = new Font("Tahoma", Font.PLAIN, 9);
+ Font rulerFont = new Font("Tahoma", Font.PLAIN, DPIUtil.upscale(9));
//g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
g.setStroke(new BasicStroke(1));
g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, 0.8f));
- Rectangle2D vertical = new Rectangle2D.Double(bounds.getMinX(), bounds.getMinY(), bounds.getMaxX(), bounds.getMinY()+20);
+ int rulerPixelSize = (int) DPIUtil.upscale(rulerSize);
+ int u7 = DPIUtil.upscale(7);
+ int u4 = DPIUtil.upscale(4);
+ int u2 = DPIUtil.upscale(2);
+
+ Rectangle2D vertical = new Rectangle2D.Double(bounds.getMinX(), bounds.getMinY(), bounds.getMaxX(), bounds.getMinY()+rulerPixelSize);
g.fill(vertical);
- Rectangle2D horizontal = new Rectangle2D.Double(bounds.getMinX(), bounds.getMinY()+20, bounds.getMinX()+20, bounds.getMaxY());
+ Rectangle2D horizontal = new Rectangle2D.Double(bounds.getMinX(), bounds.getMinY()+rulerPixelSize, bounds.getMinX()+rulerPixelSize, bounds.getMaxY());
g.fill(horizontal);
// stepX and stepY should be something between 50 and 100
double stepX = 50;
double stepY = 50;
-\r
- stepX = GridUtils.limitedEvenGridSpacing(stepX, scaleX, 100, gridSize, true);\r
- stepY = GridUtils.limitedEvenGridSpacing(stepY, scaleY, 100, gridSize, true);\r
+
+ stepX = GridUtils.limitedEvenGridSpacing(stepX, scaleX, 100, gridSize, true);
+ stepY = GridUtils.limitedEvenGridSpacing(stepY, scaleY, 100, gridSize, true);
//while(stepX * scaleX > 100) stepX /= 2;
//while(stepY * scaleY > 100) stepY /= 2;
while(stepX * scaleX < 50) stepX *= 2;
while(stepY * scaleY < 50) stepY *= 2;
-\r
+
stepX *= scaleX;
stepY *= scaleY;
g.setColor(GRAY);
g.setFont(rulerFont);
+ FontMetrics fm = g.getFontMetrics();
double previousText = -100;
// Vertical ruler
- for(double x = offsetX%stepX-stepX; x < bounds.getMaxX(); x+=stepX) {\r
- if(x > 20) {
- String str = formatValue((x-offsetX)/scaleX);
- FontMetrics fm = g.getFontMetrics();
+ for(double x = offsetX%stepX-stepX; x < bounds.getMaxX(); x+=stepX) {
+ if(x > rulerPixelSize) {
+ double val = (x-offsetX)/scaleX / getTransform().getScaleX();
+ double modifiedValue = modifyHorizontalValue(val);
+ String str = formatValue(modifiedValue, getMaxDigits());
Rectangle2D r = fm.getStringBounds(str, g);
if((x-r.getWidth()/2) > previousText) {
g.setColor(Color.BLACK);
}
g.setColor(GRAY);
- g.drawLine((int)x, (int)bounds.getMinY()+12, (int)x, (int)bounds.getMinY()+19);
+ g.drawLine((int)x, (int)bounds.getMinY()+rulerPixelSize-1-u7, (int)x, (int)bounds.getMinY()+rulerPixelSize-1);
}
if(stepX/5 > 2) {
for(double x2 = x+stepX/5; x2 < x+stepX; x2+=stepX/5) {
- if(x2 > 20) {
- g.drawLine((int)x2, (int)bounds.getMinY()+15, (int)x2, (int)bounds.getMinY()+19);
+ if(x2 > rulerPixelSize) {
+ g.drawLine((int)x2, (int)bounds.getMinY()+rulerPixelSize-1-u4, (int)x2, (int)bounds.getMinY()+rulerPixelSize-1);
}
}
for(double x2 = x+stepX/10; x2 < x+stepX; x2+=stepX/5) {
- if(x2 > 20) {
- g.drawLine((int)x2, (int)bounds.getMinY()+17, (int)x2, (int)bounds.getMinY()+19);
+ if(x2 > rulerPixelSize) {
+ g.drawLine((int)x2, (int)bounds.getMinY()+rulerPixelSize-1-u2, (int)x2, (int)bounds.getMinY()+rulerPixelSize-1);
}
}
-
}
}
-\r
+
// Horizontal ruler
previousText = -100;
- for(double y = offsetY%stepY-stepY; y < bounds.getMaxY(); y+=stepY) {\r
- if(y > 20) {\r
- double val = (y-offsetY)/scaleY;\r
- if (MAP_Y_SCALING)\r
- val = Math.toDegrees(Math.atan(Math.sinh(Math.toRadians(val))));\r
- String str = formatValue(val);
- FontMetrics fm = g.getFontMetrics();
+ for(double y = offsetY%stepY-stepY; y < bounds.getMaxY(); y+=stepY) {
+ if(y > rulerPixelSize) {
+ double val = (y-offsetY)/scaleY / getTransform().getScaleY();
+ double modifiedValue = modifyVerticalValue(val);
+ String str = formatValue(modifiedValue, getMaxDigits());
Rectangle2D r = fm.getStringBounds(str, g);
if(y-1+r.getHeight()/2 > previousText) {
g.setColor(Color.BLACK);
previousText = y-1+r.getHeight();
}
g.setColor(GRAY);
- g.drawLine((int)bounds.getMinX()+12, (int)y, (int)bounds.getMinX()+19, (int)y);
+ g.drawLine((int)bounds.getMinX()+rulerPixelSize-1-u7, (int)y, (int)bounds.getMinX()+rulerPixelSize-1, (int)y);
}
if(stepY/5 > 2) {
for(double y2 = y+stepY/5; y2 < y+stepY; y2+=stepY/5) {
- if(y2 > 20) {
- g.drawLine((int)bounds.getMinX()+15, (int)y2, (int)bounds.getMinX()+19, (int)y2);
+ if(y2 > rulerPixelSize) {
+ g.drawLine((int)bounds.getMinX()+rulerPixelSize-1-u4, (int)y2, (int)bounds.getMinX()+rulerPixelSize-1, (int)y2);
}
}
for(double y2 = y+stepY/10; y2 < y+stepY; y2+=stepY/5) {
- if(y2 > 20) {
- g.drawLine((int)bounds.getMinX()+17, (int)y2, (int)bounds.getMinX()+19, (int)y2);
+ if(y2 > rulerPixelSize) {
+ g.drawLine((int)bounds.getMinX()+rulerPixelSize-1-u2, (int)y2, (int)bounds.getMinX()+rulerPixelSize-1, (int)y2);
}
}
}
}
g.setTransform(tr);
- }\r
+ }
+
+ /**
+ * A method for subclasses to alter the actual X-value of the ruler
+ *
+ * @param value
+ * @return possibly modified X-value
+ */
+ protected double modifyHorizontalValue(double value) {
+ return value;
+ }
+
+ /**
+ * A method for subclasses to alter the actual Y-value of the ruler
+ *
+ * @param value
+ * @return possibly modified Y-value
+ */
+ protected double modifyVerticalValue(double value) {
+ return value;
+ }
private static final transient int MAX_DIGITS = 5;
private static final transient double EPSILON = 0.01;
private static final transient String[] SI_UNIT_LARGE_PREFIXES = {
"k", "M", "G", "T", "P", "E", "Z", "Y"
};
+
+ protected int getMaxDigits() {
+ return MAX_DIGITS;
+ }
- public static String formatValue(double value) {
+ public static String formatValue(double value, int maxDigits) {
int magnitude = (int) Math.round(Math.log10(value));
//System.out.println("magnitude: " + magnitude + ", " + value);
- int allowedDecimals = MAX_DIGITS;
+ int allowedDecimals = maxDigits;
allowedDecimals -= Math.abs(magnitude);
if (allowedDecimals < 0)
allowedDecimals = 0;