// License: GPL. For details, see LICENSE file. package org.openstreetmap.josm.data.osm.visitor.paint; import java.awt.AlphaComposite; import java.awt.BasicStroke; import java.awt.Color; import java.awt.Component; import java.awt.Dimension; import java.awt.Font; import java.awt.FontMetrics; import java.awt.Graphics2D; import java.awt.Image; import java.awt.Point; import java.awt.Polygon; import java.awt.Rectangle; import java.awt.RenderingHints; import java.awt.Shape; import java.awt.TexturePaint; import java.awt.font.FontRenderContext; import java.awt.font.GlyphVector; import java.awt.font.LineMetrics; import java.awt.font.TextLayout; import java.awt.geom.AffineTransform; import java.awt.geom.GeneralPath; import java.awt.geom.Path2D; import java.awt.geom.Point2D; import java.awt.geom.Rectangle2D; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Future; import javax.swing.AbstractButton; import javax.swing.FocusManager; import org.openstreetmap.josm.Main; import org.openstreetmap.josm.data.Bounds; import org.openstreetmap.josm.data.coor.EastNorth; import org.openstreetmap.josm.data.osm.BBox; import org.openstreetmap.josm.data.osm.Changeset; import org.openstreetmap.josm.data.osm.DataSet; import org.openstreetmap.josm.data.osm.Node; import org.openstreetmap.josm.data.osm.OsmPrimitive; import org.openstreetmap.josm.data.osm.OsmUtils; import org.openstreetmap.josm.data.osm.Relation; import org.openstreetmap.josm.data.osm.RelationMember; import org.openstreetmap.josm.data.osm.Way; import org.openstreetmap.josm.data.osm.WaySegment; import org.openstreetmap.josm.data.osm.visitor.Visitor; import org.openstreetmap.josm.data.osm.visitor.paint.relations.Multipolygon; import org.openstreetmap.josm.data.osm.visitor.paint.relations.Multipolygon.PolyData; import org.openstreetmap.josm.data.osm.visitor.paint.relations.MultipolygonCache; import org.openstreetmap.josm.gui.NavigatableComponent; import org.openstreetmap.josm.gui.mappaint.AreaElemStyle; import org.openstreetmap.josm.gui.mappaint.BoxTextElemStyle; import org.openstreetmap.josm.gui.mappaint.BoxTextElemStyle.HorizontalTextAlignment; import org.openstreetmap.josm.gui.mappaint.BoxTextElemStyle.VerticalTextAlignment; import org.openstreetmap.josm.gui.mappaint.ElemStyle; import org.openstreetmap.josm.gui.mappaint.ElemStyles; import org.openstreetmap.josm.gui.mappaint.MapImage; import org.openstreetmap.josm.gui.mappaint.MapPaintStyles; import org.openstreetmap.josm.gui.mappaint.NodeElemStyle; import org.openstreetmap.josm.gui.mappaint.NodeElemStyle.Symbol; import org.openstreetmap.josm.gui.mappaint.RepeatImageElemStyle.LineImageAlignment; import org.openstreetmap.josm.gui.mappaint.StyleCache.StyleList; import org.openstreetmap.josm.gui.mappaint.TextElement; import org.openstreetmap.josm.gui.mappaint.mapcss.MapCSSStyleSource; import org.openstreetmap.josm.gui.mappaint.mapcss.Selector; import org.openstreetmap.josm.tools.CompositeList; import org.openstreetmap.josm.tools.Geometry; import org.openstreetmap.josm.tools.Geometry.AreaAndPerimeter; import org.openstreetmap.josm.tools.ImageProvider; import org.openstreetmap.josm.tools.Pair; import org.openstreetmap.josm.tools.Utils; /** * A map renderer which renders a map according to style rules in a set of style sheets. * @since 486 */ public class StyledMapRenderer extends AbstractMapRenderer { private static final Pair THREAD_POOL = Utils.newThreadPool("mappaint.StyledMapRenderer.style_creation.numberOfThreads", "styled-map-renderer-%d", Thread.NORM_PRIORITY); /** * Iterates over a list of Way Nodes and returns screen coordinates that * represent a line that is shifted by a certain offset perpendicular * to the way direction. * * There is no intention, to handle consecutive duplicate Nodes in a * perfect way, but it is should not throw an exception. */ private class OffsetIterator implements Iterator { private final List nodes; private final double offset; private int idx; private Point prev; /* 'prev0' is a point that has distance 'offset' from 'prev' and the * line from 'prev' to 'prev0' is perpendicular to the way segment from * 'prev' to the next point. */ private int xPrev0, yPrev0; OffsetIterator(List nodes, double offset) { this.nodes = nodes; this.offset = offset; idx = 0; } @Override public boolean hasNext() { return idx < nodes.size(); } @Override public Point next() { if (Math.abs(offset) < 0.1d) return nc.getPoint(nodes.get(idx++)); Point current = nc.getPoint(nodes.get(idx)); if (idx == nodes.size() - 1) { ++idx; if (prev != null) { return new Point(xPrev0 + current.x - prev.x, yPrev0 + current.y - prev.y); } else { return current; } } Point next = nc.getPoint(nodes.get(idx+1)); int dxNext = next.x - current.x; int dyNext = next.y - current.y; double lenNext = Math.sqrt(dxNext*dxNext + dyNext*dyNext); if (lenNext == 0) { lenNext = 1; // value does not matter, because dy_next and dx_next is 0 } int xCurrent0 = current.x + (int) Math.round(offset * dyNext / lenNext); int yCurrent0 = current.y - (int) Math.round(offset * dxNext / lenNext); if (idx == 0) { ++idx; prev = current; xPrev0 = xCurrent0; yPrev0 = yCurrent0; return new Point(xCurrent0, yCurrent0); } else { int dxPrev = current.x - prev.x; int dyPrev = current.y - prev.y; // determine intersection of the lines parallel to the two segments int det = dxNext*dyPrev - dxPrev*dyNext; if (det == 0) { ++idx; prev = current; xPrev0 = xCurrent0; yPrev0 = yCurrent0; return new Point(xCurrent0, yCurrent0); } int m = dxNext*(yCurrent0 - yPrev0) - dyNext*(xCurrent0 - xPrev0); int cx = xPrev0 + (int) Math.round((double) m * dxPrev / det); int cy = yPrev0 + (int) Math.round((double) m * dyPrev / det); ++idx; prev = current; xPrev0 = xCurrent0; yPrev0 = yCurrent0; return new Point(cx, cy); } } @Override public void remove() { throw new UnsupportedOperationException(); } } private static class StyleRecord implements Comparable { private final ElemStyle style; private final OsmPrimitive osm; private final int flags; StyleRecord(ElemStyle style, OsmPrimitive osm, int flags) { this.style = style; this.osm = osm; this.flags = flags; } @Override public int compareTo(StyleRecord other) { if ((this.flags & FLAG_DISABLED) != 0 && (other.flags & FLAG_DISABLED) == 0) return -1; if ((this.flags & FLAG_DISABLED) == 0 && (other.flags & FLAG_DISABLED) != 0) return 1; int d0 = Float.compare(this.style.majorZIndex, other.style.majorZIndex); if (d0 != 0) return d0; // selected on top of member of selected on top of unselected // FLAG_DISABLED bit is the same at this point if (this.flags > other.flags) return 1; if (this.flags < other.flags) return -1; int dz = Float.compare(this.style.zIndex, other.style.zIndex); if (dz != 0) return dz; // simple node on top of icons and shapes if (this.style == NodeElemStyle.SIMPLE_NODE_ELEMSTYLE && other.style != NodeElemStyle.SIMPLE_NODE_ELEMSTYLE) return 1; if (this.style != NodeElemStyle.SIMPLE_NODE_ELEMSTYLE && other.style == NodeElemStyle.SIMPLE_NODE_ELEMSTYLE) return -1; // newer primitives to the front long id = this.osm.getUniqueId() - other.osm.getUniqueId(); if (id > 0) return 1; if (id < 0) return -1; return Float.compare(this.style.objectZIndex, other.style.objectZIndex); } } private static Map IS_GLYPH_VECTOR_DOUBLE_TRANSLATION_BUG = new HashMap<>(); /** * Check, if this System has the GlyphVector double translation bug. * * With this bug, gv.setGlyphTransform(i, trfm) has a different * effect than on most other systems, namely the translation components * ("m02" & "m12", {@link AffineTransform}) appear to be twice as large, as * they actually are. The rotation is unaffected (scale & shear not tested * so far). * * This bug has only been observed on Mac OS X, see #7841. * * After switch to Java 7, this test is a false positive on Mac OS X (see #10446), * i.e. it returns true, but the real rendering code does not require any special * handling. * It hasn't been further investigated why the test reports a wrong result in * this case, but the method has been changed to simply return false by default. * (This can be changed with a setting in the advanced preferences.) * * @param font The font to check. * @return false by default, but depends on the value of the advanced * preference glyph-bug=false|true|auto, where auto is the automatic detection * method which apparently no longer gives a useful result for Java 7. */ public static boolean isGlyphVectorDoubleTranslationBug(Font font) { Boolean cached = IS_GLYPH_VECTOR_DOUBLE_TRANSLATION_BUG.get(font); if (cached != null) return cached; String overridePref = Main.pref.get("glyph-bug", "auto"); if ("auto".equals(overridePref)) { FontRenderContext frc = new FontRenderContext(null, false, false); GlyphVector gv = font.createGlyphVector(frc, "x"); gv.setGlyphTransform(0, AffineTransform.getTranslateInstance(1000, 1000)); Shape shape = gv.getGlyphOutline(0); Main.trace("#10446: shape: "+shape.getBounds()); // x is about 1000 on normal stystems and about 2000 when the bug occurs int x = shape.getBounds().x; boolean isBug = x > 1500; IS_GLYPH_VECTOR_DOUBLE_TRANSLATION_BUG.put(font, isBug); return isBug; } else { boolean override = Boolean.parseBoolean(overridePref); IS_GLYPH_VECTOR_DOUBLE_TRANSLATION_BUG.put(font, override); return override; } } private double circum; private MapPaintSettings paintSettings; private Color highlightColorTransparent; /** * Flags used to store the primitive state along with the style. This is the normal style. *

* Not used in any public interfaces. */ private static final int FLAG_NORMAL = 0; /** * A primitive with {@link OsmPrimitive#isDisabled()} */ private static final int FLAG_DISABLED = 1; /** * A primitive with {@link OsmPrimitive#isMemberOfSelected()} */ private static final int FLAG_MEMBER_OF_SELECTED = 2; /** * A primitive with {@link OsmPrimitive#isSelected()} */ private static final int FLAG_SELECTED = 4; /** * A primitive with {@link OsmPrimitive#isOuterMemberOfSelected()} */ private static final int FLAG_OUTERMEMBER_OF_SELECTED = 8; private static final double PHI = Math.toRadians(20); private static final double cosPHI = Math.cos(PHI); private static final double sinPHI = Math.sin(PHI); private Collection highlightWaySegments; // highlight customization fields private int highlightLineWidth; private int highlightPointRadius; private int widerHighlight; private int highlightStep; //flag that activate wider highlight mode private boolean useWiderHighlight; private boolean useStrokes; private boolean showNames; private boolean showIcons; private boolean isOutlineOnly; private boolean isUnclosedAreaHighlight; private double unclosedAreaHighlightWidth; private double partialFillThreshold; private Font orderFont; private boolean leftHandTraffic; private Object antialiasing; /** * Constructs a new {@code StyledMapRenderer}. * * @param g the graphics context. Must not be null. * @param nc the map viewport. Must not be null. * @param isInactiveMode if true, the paint visitor shall render OSM objects such that they * look inactive. Example: rendering of data in an inactive layer using light gray as color only. * @throws IllegalArgumentException if {@code g} is null * @throws IllegalArgumentException if {@code nc} is null */ public StyledMapRenderer(Graphics2D g, NavigatableComponent nc, boolean isInactiveMode) { super(g, nc, isInactiveMode); if (nc != null) { Component focusOwner = FocusManager.getCurrentManager().getFocusOwner(); useWiderHighlight = !(focusOwner instanceof AbstractButton || focusOwner == nc); } } private Polygon buildPolygon(Point center, int radius, int sides) { return buildPolygon(center, radius, sides, 0.0); } private static Polygon buildPolygon(Point center, int radius, int sides, double rotation) { Polygon polygon = new Polygon(); for (int i = 0; i < sides; i++) { double angle = ((2 * Math.PI / sides) * i) - rotation; int x = (int) Math.round(center.x + radius * Math.cos(angle)); int y = (int) Math.round(center.y + radius * Math.sin(angle)); polygon.addPoint(x, y); } return polygon; } private void displaySegments(GeneralPath path, GeneralPath orientationArrows, GeneralPath onewayArrows, GeneralPath onewayArrowsCasing, Color color, BasicStroke line, BasicStroke dashes, Color dashedColor) { g.setColor(isInactiveMode ? inactiveColor : color); if (useStrokes) { g.setStroke(line); } g.draw(path); if (!isInactiveMode && useStrokes && dashes != null) { g.setColor(dashedColor); g.setStroke(dashes); g.draw(path); } if (orientationArrows != null) { g.setColor(isInactiveMode ? inactiveColor : color); g.setStroke(new BasicStroke(line.getLineWidth(), line.getEndCap(), BasicStroke.JOIN_MITER, line.getMiterLimit())); g.draw(orientationArrows); } if (onewayArrows != null) { g.setStroke(new BasicStroke(1, line.getEndCap(), BasicStroke.JOIN_MITER, line.getMiterLimit())); g.fill(onewayArrowsCasing); g.setColor(isInactiveMode ? inactiveColor : backgroundColor); g.fill(onewayArrows); } if (useStrokes) { g.setStroke(new BasicStroke()); } } /** * Displays text at specified position including its halo, if applicable. * * @param gv Text's glyphs to display. If {@code null}, use text from {@code s} instead. * @param s text to display if {@code gv} is {@code null} * @param x X position * @param y Y position * @param disabled {@code true} if element is disabled (filtered out) * @param text text style to use */ private void displayText(GlyphVector gv, String s, int x, int y, boolean disabled, TextElement text) { if (isInactiveMode || disabled) { g.setColor(inactiveColor); if (gv != null) { g.drawGlyphVector(gv, x, y); } else { g.setFont(text.font); g.drawString(s, x, y); } } else if (text.haloRadius != null) { g.setStroke(new BasicStroke(2*text.haloRadius, BasicStroke.CAP_BUTT, BasicStroke.JOIN_ROUND)); g.setColor(text.haloColor); Shape textOutline; if (gv == null) { if (s.isEmpty()) return; FontRenderContext frc = g.getFontRenderContext(); TextLayout tl = new TextLayout(s, text.font, frc); textOutline = tl.getOutline(AffineTransform.getTranslateInstance(x, y)); } else { textOutline = gv.getOutline(x, y); } g.draw(textOutline); g.setStroke(new BasicStroke()); g.setColor(text.color); g.fill(textOutline); } else { g.setColor(text.color); if (gv != null) { g.drawGlyphVector(gv, x, y); } else { g.setFont(text.font); g.drawString(s, x, y); } } } /** * Worker function for drawing areas. * * @param osm the primitive * @param path the path object for the area that should be drawn; in case * of multipolygons, this can path can be a complex shape with one outer * polygon and one or more inner polygons * @param color The color to fill the area with. * @param fillImage The image to fill the area with. Overrides color. * @param extent if not null, area will be filled partially; specifies, how * far to fill from the boundary towards the center of the area; * if null, area will be filled completely * @param unclosedHighlight true, if the fact that the way / multipolygon is not * properly closed should be highlighted; this parameter is only used * for partial fill ({@code extent != null}), otherwise it is ignored * @param disabled If this should be drawn with a special disabled style. * @param text The text to write on the area. */ protected void drawArea(OsmPrimitive osm, Path2D.Double path, Color color, MapImage fillImage, Float extent, boolean unclosedHighlight, boolean disabled, TextElement text) { Shape area = path.createTransformedShape(nc.getAffineTransform()); if (!isOutlineOnly) { g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_OFF); if (fillImage == null) { if (isInactiveMode) { g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, 0.33f)); } g.setColor(color); if (extent == null) { g.fill(area); } else { if (unclosedHighlight) { g.setStroke(new BasicStroke((int) (unclosedAreaHighlightWidth / 100 * extent), BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER)); g.draw(area); } else { Shape clip = g.getClip(); g.clip(area); g.setStroke(new BasicStroke(2 * extent, BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER)); g.draw(area); g.setClip(clip); } } } else { TexturePaint texture = new TexturePaint(fillImage.getImage(disabled), new Rectangle(0, 0, fillImage.getWidth(), fillImage.getHeight())); g.setPaint(texture); Float alpha = fillImage.getAlphaFloat(); if (!Utils.equalsEpsilon(alpha, 1f)) { g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha)); } if (extent == null) { g.fill(area); } else { if (unclosedHighlight) { g.setStroke(new BasicStroke((int) (unclosedAreaHighlightWidth / 100 * extent), BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER)); g.draw(area); } else { Shape clip = g.getClip(); BasicStroke stroke = new BasicStroke(2 * extent, BasicStroke.CAP_BUTT, BasicStroke.JOIN_MITER); g.clip(stroke.createStrokedShape(area)); g.fill(area); g.setClip(clip); } } g.setPaintMode(); } g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, antialiasing); } drawAreaText(osm, text, area); } private void drawAreaText(OsmPrimitive osm, TextElement text, Shape area) { if (text != null && isShowNames()) { // abort if we can't compose the label to be rendered if (text.labelCompositionStrategy == null) return; String name = text.labelCompositionStrategy.compose(osm); if (name == null) return; Rectangle pb = area.getBounds(); FontMetrics fontMetrics = g.getFontMetrics(orderFont); // if slow, use cache Rectangle2D nb = fontMetrics.getStringBounds(name, g); // if slow, approximate by strlen()*maxcharbounds(font) // Using the Centroid is Nicer for buildings like: +--------+ // but this needs to be fast. As most houses are | 42 | // boxes anyway, the center of the bounding box +---++---+ // will have to do. ++ // Centroids are not optimal either, just imagine a U-shaped house. // quick check to see if label box is smaller than primitive box if (pb.width >= nb.getWidth() && pb.height >= nb.getHeight()) { final double w = pb.width - nb.getWidth(); final double h = pb.height - nb.getHeight(); final int x2 = pb.x + (int) (w/2.0); final int y2 = pb.y + (int) (h/2.0); final int nbw = (int) nb.getWidth(); final int nbh = (int) nb.getHeight(); Rectangle centeredNBounds = new Rectangle(x2, y2, nbw, nbh); // slower check to see if label is displayed inside primitive shape boolean labelOK = area.contains(centeredNBounds); if (!labelOK) { // if center position (C) is not inside osm shape, try naively some other positions as follows: final int x1 = pb.x + (int) (w/4.0); final int x3 = pb.x + (int) (3*w/4.0); final int y1 = pb.y + (int) (h/4.0); final int y3 = pb.y + (int) (3*h/4.0); // +-----------+ // | 5 1 6 | // | 4 C 2 | // | 8 3 7 | // +-----------+ Rectangle[] candidates = new Rectangle[] { new Rectangle(x2, y1, nbw, nbh), new Rectangle(x3, y2, nbw, nbh), new Rectangle(x2, y3, nbw, nbh), new Rectangle(x1, y2, nbw, nbh), new Rectangle(x1, y1, nbw, nbh), new Rectangle(x3, y1, nbw, nbh), new Rectangle(x3, y3, nbw, nbh), new Rectangle(x1, y3, nbw, nbh) }; // Dumb algorithm to find a better placement. We could surely find a smarter one but it should // solve most of building issues with only few calculations (8 at most) for (int i = 0; i < candidates.length && !labelOK; i++) { centeredNBounds = candidates[i]; labelOK = area.contains(centeredNBounds); } } if (labelOK) { Font defaultFont = g.getFont(); int x = (int) (centeredNBounds.getMinX() - nb.getMinX()); int y = (int) (centeredNBounds.getMinY() - nb.getMinY()); displayText(null, name, x, y, osm.isDisabled(), text); g.setFont(defaultFont); } else if (Main.isDebugEnabled()) { Main.debug("Couldn't find a correct label placement for "+osm+" / "+name); } } } } /** * Draws a multipolygon area. * @param r The multipolygon relation * @param color The color to fill the area with. * @param fillImage The image to fill the area with. Overrides color. * @param extent if not null, area will be filled partially; specifies, how * far to fill from the boundary towards the center of the area; * if null, area will be filled completely * @param disabled If this should be drawn with a special disabled style. * @param text The text to write on the area. */ public void drawArea(Relation r, Color color, MapImage fillImage, Float extent, boolean disabled, TextElement text) { Multipolygon multipolygon = MultipolygonCache.getInstance().get(nc, r); if (!r.isDisabled() && !multipolygon.getOuterWays().isEmpty()) { for (PolyData pd : multipolygon.getCombinedPolygons()) { Path2D.Double p = pd.get(); if (!isAreaVisible(p)) { continue; } boolean unclosedHighlight = false; if (extent != null) { if (pd.isClosed()) { AreaAndPerimeter ap = pd.getAreaAndPerimeter(); // if partial fill would only leave a small gap in the center ... if (ap.getPerimeter() * extent * circum / 100 > partialFillThreshold / 100 * ap.getArea()) { // ... turn it off and fill completely extent = null; } } else { unclosedHighlight = isUnclosedAreaHighlight; } } drawArea(r, p, pd.selected ? paintSettings.getRelationSelectedColor(color.getAlpha()) : color, fillImage, extent, unclosedHighlight, disabled, text); } } } /** * Draws an area defined by a way. They way does not need to be closed, but it should. * @param w The way. * @param color The color to fill the area with. * @param fillImage The image to fill the area with. Overrides color. * @param extent if not null, area will be filled partially; specifies, how * far to fill from the boundary towards the center of the area; * if null, area will be filled completely * @param disabled If this should be drawn with a special disabled style. * @param text The text to write on the area. */ public void drawArea(Way w, Color color, MapImage fillImage, Float extent, boolean disabled, TextElement text) { if (extent != null && w.isClosed()) { AreaAndPerimeter ap = Geometry.getAreaAndPerimeter(w.getNodes()); // if partial fill would only leave a small gap in the center ... if (ap.getPerimeter() * extent * circum / 100 > partialFillThreshold / 100 * ap.getArea()) { // ... turn it off and fill completely extent = null; } } drawArea(w, getPath(w), color, fillImage, extent, isUnclosedAreaHighlight && !w.isClosed(), disabled, text); } public void drawBoxText(Node n, BoxTextElemStyle bs) { if (!isShowNames() || bs == null) return; Point p = nc.getPoint(n); TextElement text = bs.text; String s = text.labelCompositionStrategy.compose(n); if (s == null) return; Font defaultFont = g.getFont(); g.setFont(text.font); int x = p.x + text.xOffset; int y = p.y + text.yOffset; /** * * left-above __center-above___ right-above * left-top| |right-top * | | * left-center| center-center |right-center * | | * left-bottom|_________________|right-bottom * left-below center-below right-below * */ Rectangle box = bs.getBox(); if (bs.hAlign == HorizontalTextAlignment.RIGHT) { x += box.x + box.width + 2; } else { FontRenderContext frc = g.getFontRenderContext(); Rectangle2D bounds = text.font.getStringBounds(s, frc); int textWidth = (int) bounds.getWidth(); if (bs.hAlign == HorizontalTextAlignment.CENTER) { x -= textWidth / 2; } else if (bs.hAlign == HorizontalTextAlignment.LEFT) { x -= -box.x + 4 + textWidth; } else throw new AssertionError(); } if (bs.vAlign == VerticalTextAlignment.BOTTOM) { y += box.y + box.height; } else { FontRenderContext frc = g.getFontRenderContext(); LineMetrics metrics = text.font.getLineMetrics(s, frc); if (bs.vAlign == VerticalTextAlignment.ABOVE) { y -= -box.y + metrics.getDescent(); } else if (bs.vAlign == VerticalTextAlignment.TOP) { y -= -box.y - metrics.getAscent(); } else if (bs.vAlign == VerticalTextAlignment.CENTER) { y += (metrics.getAscent() - metrics.getDescent()) / 2; } else if (bs.vAlign == VerticalTextAlignment.BELOW) { y += box.y + box.height + metrics.getAscent() + 2; } else throw new AssertionError(); } displayText(null, s, x, y, n.isDisabled(), text); g.setFont(defaultFont); } /** * Draw an image along a way repeatedly. * * @param way the way * @param pattern the image * @param disabled If this should be drawn with a special disabled style. * @param offset offset from the way * @param spacing spacing between two images * @param phase initial spacing * @param align alignment of the image. The top, center or bottom edge can be aligned with the way. */ public void drawRepeatImage(Way way, MapImage pattern, boolean disabled, double offset, double spacing, double phase, LineImageAlignment align) { final int imgWidth = pattern.getWidth(); final double repeat = imgWidth + spacing; final int imgHeight = pattern.getHeight(); Point lastP = null; double currentWayLength = phase % repeat; if (currentWayLength < 0) { currentWayLength += repeat; } int dy1, dy2; switch (align) { case TOP: dy1 = 0; dy2 = imgHeight; break; case CENTER: dy1 = -imgHeight / 2; dy2 = imgHeight + dy1; break; case BOTTOM: dy1 = -imgHeight; dy2 = 0; break; default: throw new AssertionError(); } OffsetIterator it = new OffsetIterator(way.getNodes(), offset); while (it.hasNext()) { Point thisP = it.next(); if (lastP != null) { final double segmentLength = thisP.distance(lastP); final double dx = thisP.x - lastP.x; final double dy = thisP.y - lastP.y; // pos is the position from the beginning of the current segment // where an image should be painted double pos = repeat - (currentWayLength % repeat); AffineTransform saveTransform = g.getTransform(); g.translate(lastP.x, lastP.y); g.rotate(Math.atan2(dy, dx)); // draw the rest of the image from the last segment in case it // is cut off if (pos > spacing) { // segment is too short for a complete image if (pos > segmentLength + spacing) { g.drawImage(pattern.getImage(disabled), 0, dy1, (int) segmentLength, dy2, (int) (repeat - pos), 0, (int) (repeat - pos + segmentLength), imgHeight, null); // rest of the image fits fully on the current segment } else { g.drawImage(pattern.getImage(disabled), 0, dy1, (int) (pos - spacing), dy2, (int) (repeat - pos), 0, imgWidth, imgHeight, null); } } // draw remaining images for this segment while (pos < segmentLength) { // cut off at the end? if (pos + imgWidth > segmentLength) { g.drawImage(pattern.getImage(disabled), (int) pos, dy1, (int) segmentLength, dy2, 0, 0, (int) segmentLength - (int) pos, imgHeight, null); } else { g.drawImage(pattern.getImage(disabled), (int) pos, dy1, nc); } pos += repeat; } g.setTransform(saveTransform); currentWayLength += segmentLength; } lastP = thisP; } } @Override public void drawNode(Node n, Color color, int size, boolean fill) { if (size <= 0 && !n.isHighlighted()) return; Point p = nc.getPoint(n); if (n.isHighlighted()) { drawPointHighlight(p, size); } if (size > 1) { if ((p.x < 0) || (p.y < 0) || (p.x > nc.getWidth()) || (p.y > nc.getHeight())) return; int radius = size / 2; if (isInactiveMode || n.isDisabled()) { g.setColor(inactiveColor); } else { g.setColor(color); } if (fill) { g.fillRect(p.x-radius-1, p.y-radius-1, size + 1, size + 1); } else { g.drawRect(p.x-radius-1, p.y-radius-1, size, size); } } } public void drawNodeIcon(Node n, MapImage img, boolean disabled, boolean selected, boolean member, double theta) { Point p = nc.getPoint(n); final int w = img.getWidth(), h = img.getHeight(); if (n.isHighlighted()) { drawPointHighlight(p, Math.max(w, h)); } float alpha = img.getAlphaFloat(); if (!Utils.equalsEpsilon(alpha, 1f)) { g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha)); } g.rotate(theta, p.x, p.y); g.drawImage(img.getImage(disabled), p.x - w/2 + img.offsetX, p.y - h/2 + img.offsetY, nc); g.rotate(-theta, p.x, p.y); g.setPaintMode(); if (selected || member) { Color color; if (disabled) { color = inactiveColor; } else if (selected) { color = selectedColor; } else { color = relationSelectedColor; } g.setColor(color); g.drawRect(p.x - w/2 + img.offsetX - 2, p.y - h/2 + img.offsetY - 2, w + 4, h + 4); } } public void drawNodeSymbol(Node n, Symbol s, Color fillColor, Color strokeColor) { Point p = nc.getPoint(n); int radius = s.size / 2; if (n.isHighlighted()) { drawPointHighlight(p, s.size); } if (fillColor != null) { g.setColor(fillColor); switch (s.symbol) { case SQUARE: g.fillRect(p.x - radius, p.y - radius, s.size, s.size); break; case CIRCLE: g.fillOval(p.x - radius, p.y - radius, s.size, s.size); break; case TRIANGLE: g.fillPolygon(buildPolygon(p, radius, 3, Math.PI / 2)); break; case PENTAGON: g.fillPolygon(buildPolygon(p, radius, 5, Math.PI / 2)); break; case HEXAGON: g.fillPolygon(buildPolygon(p, radius, 6)); break; case HEPTAGON: g.fillPolygon(buildPolygon(p, radius, 7, Math.PI / 2)); break; case OCTAGON: g.fillPolygon(buildPolygon(p, radius, 8, Math.PI / 8)); break; case NONAGON: g.fillPolygon(buildPolygon(p, radius, 9, Math.PI / 2)); break; case DECAGON: g.fillPolygon(buildPolygon(p, radius, 10)); break; default: throw new AssertionError(); } } if (s.stroke != null) { g.setStroke(s.stroke); g.setColor(strokeColor); switch (s.symbol) { case SQUARE: g.drawRect(p.x - radius, p.y - radius, s.size - 1, s.size - 1); break; case CIRCLE: g.drawOval(p.x - radius, p.y - radius, s.size - 1, s.size - 1); break; case TRIANGLE: g.drawPolygon(buildPolygon(p, radius, 3, Math.PI / 2)); break; case PENTAGON: g.drawPolygon(buildPolygon(p, radius, 5, Math.PI / 2)); break; case HEXAGON: g.drawPolygon(buildPolygon(p, radius, 6)); break; case HEPTAGON: g.drawPolygon(buildPolygon(p, radius, 7, Math.PI / 2)); break; case OCTAGON: g.drawPolygon(buildPolygon(p, radius, 8, Math.PI / 8)); break; case NONAGON: g.drawPolygon(buildPolygon(p, radius, 9, Math.PI / 2)); break; case DECAGON: g.drawPolygon(buildPolygon(p, radius, 10)); break; default: throw new AssertionError(); } g.setStroke(new BasicStroke()); } } /** * Draw a number of the order of the two consecutive nodes within the * parents way * * @param n1 First node of the way segment. * @param n2 Second node of the way segment. * @param orderNumber The number of the segment in the way. * @param clr The color to use for drawing the text. */ public void drawOrderNumber(Node n1, Node n2, int orderNumber, Color clr) { Point p1 = nc.getPoint(n1); Point p2 = nc.getPoint(n2); drawOrderNumber(p1, p2, orderNumber, clr); } /** * highlights a given GeneralPath using the settings from BasicStroke to match the line's * style. Width of the highlight is hard coded. * @param path path to draw * @param line line style */ private void drawPathHighlight(GeneralPath path, BasicStroke line) { if (path == null) return; g.setColor(highlightColorTransparent); float w = line.getLineWidth() + highlightLineWidth; if (useWiderHighlight) w += widerHighlight; while (w >= line.getLineWidth()) { g.setStroke(new BasicStroke(w, line.getEndCap(), line.getLineJoin(), line.getMiterLimit())); g.draw(path); w -= highlightStep; } } /** * highlights a given point by drawing a rounded rectangle around it. Give the * size of the object you want to be highlighted, width is added automatically. */ private void drawPointHighlight(Point p, int size) { g.setColor(highlightColorTransparent); int s = size + highlightPointRadius; if (useWiderHighlight) s += widerHighlight; while (s >= size) { int r = (int) Math.floor(s/2d); g.fillRoundRect(p.x-r, p.y-r, s, s, r, r); s -= highlightStep; } } public void drawRestriction(Image img, Point pVia, double vx, double vx2, double vy, double vy2, double angle, boolean selected) { // rotate image with direction last node in from to, and scale down image to 16*16 pixels Image smallImg = ImageProvider.createRotatedImage(img, angle, new Dimension(16, 16)); int w = smallImg.getWidth(null), h = smallImg.getHeight(null); g.drawImage(smallImg, (int) (pVia.x+vx+vx2)-w/2, (int) (pVia.y+vy+vy2)-h/2, nc); if (selected) { g.setColor(isInactiveMode ? inactiveColor : relationSelectedColor); g.drawRect((int) (pVia.x+vx+vx2)-w/2-2, (int) (pVia.y+vy+vy2)-h/2-2, w+4, h+4); } } public void drawRestriction(Relation r, MapImage icon, boolean disabled) { Way fromWay = null; Way toWay = null; OsmPrimitive via = null; /* find the "from", "via" and "to" elements */ for (RelationMember m : r.getMembers()) { if (m.getMember().isIncomplete()) return; else { if (m.isWay()) { Way w = m.getWay(); if (w.getNodesCount() < 2) { continue; } switch(m.getRole()) { case "from": if (fromWay == null) { fromWay = w; } break; case "to": if (toWay == null) { toWay = w; } break; case "via": if (via == null) { via = w; } } } else if (m.isNode()) { Node n = m.getNode(); if ("via".equals(m.getRole()) && via == null) { via = n; } } } } if (fromWay == null || toWay == null || via == null) return; Node viaNode; if (via instanceof Node) { viaNode = (Node) via; if (!fromWay.isFirstLastNode(viaNode)) return; } else { Way viaWay = (Way) via; Node firstNode = viaWay.firstNode(); Node lastNode = viaWay.lastNode(); Boolean onewayvia = Boolean.FALSE; String onewayviastr = viaWay.get("oneway"); if (onewayviastr != null) { if ("-1".equals(onewayviastr)) { onewayvia = Boolean.TRUE; Node tmp = firstNode; firstNode = lastNode; lastNode = tmp; } else { onewayvia = OsmUtils.getOsmBoolean(onewayviastr); if (onewayvia == null) { onewayvia = Boolean.FALSE; } } } if (fromWay.isFirstLastNode(firstNode)) { viaNode = firstNode; } else if (!onewayvia && fromWay.isFirstLastNode(lastNode)) { viaNode = lastNode; } else return; } /* find the "direct" nodes before the via node */ Node fromNode; if (fromWay.firstNode() == via) { fromNode = fromWay.getNode(1); } else { fromNode = fromWay.getNode(fromWay.getNodesCount()-2); } Point pFrom = nc.getPoint(fromNode); Point pVia = nc.getPoint(viaNode); /* starting from via, go back the "from" way a few pixels (calculate the vector vx/vy with the specified length and the direction away from the "via" node along the first segment of the "from" way) */ double distanceFromVia = 14; double dx = pFrom.x >= pVia.x ? pFrom.x - pVia.x : pVia.x - pFrom.x; double dy = pFrom.y >= pVia.y ? pFrom.y - pVia.y : pVia.y - pFrom.y; double fromAngle; if (dx == 0) { fromAngle = Math.PI/2; } else { fromAngle = Math.atan(dy / dx); } double fromAngleDeg = Math.toDegrees(fromAngle); double vx = distanceFromVia * Math.cos(fromAngle); double vy = distanceFromVia * Math.sin(fromAngle); if (pFrom.x < pVia.x) { vx = -vx; } if (pFrom.y < pVia.y) { vy = -vy; } /* go a few pixels away from the way (in a right angle) (calculate the vx2/vy2 vector with the specified length and the direction 90degrees away from the first segment of the "from" way) */ double distanceFromWay = 10; double vx2 = 0; double vy2 = 0; double iconAngle = 0; if (pFrom.x >= pVia.x && pFrom.y >= pVia.y) { if (!leftHandTraffic) { vx2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg - 90)); vy2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg - 90)); } else { vx2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg + 90)); vy2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg + 90)); } iconAngle = 270+fromAngleDeg; } if (pFrom.x < pVia.x && pFrom.y >= pVia.y) { if (!leftHandTraffic) { vx2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg)); vy2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg)); } else { vx2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg + 180)); vy2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg + 180)); } iconAngle = 90-fromAngleDeg; } if (pFrom.x < pVia.x && pFrom.y < pVia.y) { if (!leftHandTraffic) { vx2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg + 90)); vy2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg + 90)); } else { vx2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg - 90)); vy2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg - 90)); } iconAngle = 90+fromAngleDeg; } if (pFrom.x >= pVia.x && pFrom.y < pVia.y) { if (!leftHandTraffic) { vx2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg + 180)); vy2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg + 180)); } else { vx2 = distanceFromWay * Math.sin(Math.toRadians(fromAngleDeg)); vy2 = distanceFromWay * Math.cos(Math.toRadians(fromAngleDeg)); } iconAngle = 270-fromAngleDeg; } drawRestriction(icon.getImage(disabled), pVia, vx, vx2, vy, vy2, iconAngle, r.isSelected()); } /** * Draws a text along a given way. * @param way The way to draw the text on. * @param text The text definition (font/.../text content) to draw. */ public void drawTextOnPath(Way way, TextElement text) { if (way == null || text == null) return; String name = text.getString(way); if (name == null || name.isEmpty()) return; FontMetrics fontMetrics = g.getFontMetrics(text.font); Rectangle2D rec = fontMetrics.getStringBounds(name, g); Rectangle bounds = g.getClipBounds(); Polygon poly = new Polygon(); Point lastPoint = null; Iterator it = way.getNodes().iterator(); double pathLength = 0; long dx, dy; // find half segments that are long enough to draw text on (don't draw text over the cross hair in the center of each segment) List longHalfSegmentStart = new ArrayList<>(); // start point of half segment (as length along the way) List longHalfSegmentEnd = new ArrayList<>(); // end point of half segment (as length along the way) List longHalfsegmentQuality = new ArrayList<>(); // quality factor (off screen / partly on screen / fully on screen) while (it.hasNext()) { Node n = it.next(); Point p = nc.getPoint(n); poly.addPoint(p.x, p.y); if (lastPoint != null) { dx = p.x - lastPoint.x; dy = p.y - lastPoint.y; double segmentLength = Math.sqrt(dx*dx + dy*dy); if (segmentLength > 2*(rec.getWidth()+4)) { Point center = new Point((lastPoint.x + p.x)/2, (lastPoint.y + p.y)/2); double q = 0; if (bounds != null) { if (bounds.contains(lastPoint) && bounds.contains(center)) { q = 2; } else if (bounds.contains(lastPoint) || bounds.contains(center)) { q = 1; } } longHalfSegmentStart.add(pathLength); longHalfSegmentEnd.add(pathLength + segmentLength / 2); longHalfsegmentQuality.add(q); q = 0; if (bounds != null) { if (bounds.contains(center) && bounds.contains(p)) { q = 2; } else if (bounds.contains(center) || bounds.contains(p)) { q = 1; } } longHalfSegmentStart.add(pathLength + segmentLength / 2); longHalfSegmentEnd.add(pathLength + segmentLength); longHalfsegmentQuality.add(q); } pathLength += segmentLength; } lastPoint = p; } if (rec.getWidth() > pathLength) return; double t1, t2; if (!longHalfSegmentStart.isEmpty()) { if (way.getNodesCount() == 2) { // For 2 node ways, the two half segments are exactly the same size and distance from the center. // Prefer the first one for consistency. longHalfsegmentQuality.set(0, longHalfsegmentQuality.get(0) + 0.5); } // find the long half segment that is closest to the center of the way // candidates with higher quality value are preferred double bestStart = Double.NaN; double bestEnd = Double.NaN; double bestDistanceToCenter = Double.MAX_VALUE; double bestQuality = -1; for (int i = 0; i < longHalfSegmentStart.size(); i++) { double start = longHalfSegmentStart.get(i); double end = longHalfSegmentEnd.get(i); double dist = Math.abs(0.5 * (end + start) - 0.5 * pathLength); if (longHalfsegmentQuality.get(i) > bestQuality || (dist < bestDistanceToCenter && Utils.equalsEpsilon(longHalfsegmentQuality.get(i), bestQuality))) { bestStart = start; bestEnd = end; bestDistanceToCenter = dist; bestQuality = longHalfsegmentQuality.get(i); } } double remaining = bestEnd - bestStart - rec.getWidth(); // total space left and right from the text // The space left and right of the text should be distributed 20% - 80% (towards the center), // but the smaller space should not be less than 7 px. // However, if the total remaining space is less than 14 px, then distribute it evenly. double smallerSpace = Math.min(Math.max(0.2 * remaining, 7), 0.5 * remaining); if ((bestEnd + bestStart)/2 < pathLength/2) { t2 = bestEnd - smallerSpace; t1 = t2 - rec.getWidth(); } else { t1 = bestStart + smallerSpace; t2 = t1 + rec.getWidth(); } } else { // doesn't fit into one half-segment -> just put it in the center of the way t1 = pathLength/2 - rec.getWidth()/2; t2 = pathLength/2 + rec.getWidth()/2; } t1 /= pathLength; t2 /= pathLength; double[] p1 = pointAt(t1, poly, pathLength); double[] p2 = pointAt(t2, poly, pathLength); if (p1 == null || p2 == null) return; double angleOffset; double offsetSign; double tStart; if (p1[0] < p2[0] && p1[2] < Math.PI/2 && p1[2] > -Math.PI/2) { angleOffset = 0; offsetSign = 1; tStart = t1; } else { angleOffset = Math.PI; offsetSign = -1; tStart = t2; } List gvs = Utils.getGlyphVectorsBidi(name, text.font, g.getFontRenderContext()); double gvOffset = 0; for (GlyphVector gv : gvs) { double gvWidth = gv.getLogicalBounds().getBounds2D().getWidth(); for (int i = 0; i < gv.getNumGlyphs(); ++i) { Rectangle2D rect = gv.getGlyphLogicalBounds(i).getBounds2D(); double t = tStart + offsetSign * (gvOffset + rect.getX() + rect.getWidth()/2) / pathLength; double[] p = pointAt(t, poly, pathLength); if (p != null) { AffineTransform trfm = AffineTransform.getTranslateInstance(p[0] - rect.getX(), p[1]); trfm.rotate(p[2]+angleOffset); double off = -rect.getY() - rect.getHeight()/2 + text.yOffset; trfm.translate(-rect.getWidth()/2, off); if (isGlyphVectorDoubleTranslationBug(text.font)) { // scale the translation components by one half AffineTransform tmp = AffineTransform.getTranslateInstance(-0.5 * trfm.getTranslateX(), -0.5 * trfm.getTranslateY()); tmp.concatenate(trfm); trfm = tmp; } gv.setGlyphTransform(i, trfm); } } displayText(gv, null, 0, 0, way.isDisabled(), text); gvOffset += gvWidth; } } /** * draw way. This method allows for two draw styles (line using color, dashes using dashedColor) to be passed. * @param way The way to draw * @param color The base color to draw the way in * @param line The line style to use. This is drawn using color. * @param dashes The dash style to use. This is drawn using dashedColor. null if unused. * @param dashedColor The color of the dashes. * @param offset The offset * @param showOrientation show arrows that indicate the technical orientation of * the way (defined by order of nodes) * @param showHeadArrowOnly True if only the arrow at the end of the line but not those on the segments should be displayed. * @param showOneway show symbols that indicate the direction of the feature, * e.g. oneway street or waterway * @param onewayReversed for oneway=-1 and similar */ public void drawWay(Way way, Color color, BasicStroke line, BasicStroke dashes, Color dashedColor, float offset, boolean showOrientation, boolean showHeadArrowOnly, boolean showOneway, boolean onewayReversed) { GeneralPath path = new GeneralPath(); GeneralPath orientationArrows = showOrientation ? new GeneralPath() : null; GeneralPath onewayArrows = showOneway ? new GeneralPath() : null; GeneralPath onewayArrowsCasing = showOneway ? new GeneralPath() : null; Rectangle bounds = g.getClipBounds(); if (bounds != null) { // avoid arrow heads at the border bounds.grow(100, 100); } double wayLength = 0; Point lastPoint = null; boolean initialMoveToNeeded = true; List wayNodes = way.getNodes(); if (wayNodes.size() < 2) return; // only highlight the segment if the way itself is not highlighted if (!way.isHighlighted() && highlightWaySegments != null) { GeneralPath highlightSegs = null; for (WaySegment ws : highlightWaySegments) { if (ws.way != way || ws.lowerIndex < offset) { continue; } if (highlightSegs == null) { highlightSegs = new GeneralPath(); } Point p1 = nc.getPoint(ws.getFirstNode()); Point p2 = nc.getPoint(ws.getSecondNode()); highlightSegs.moveTo(p1.x, p1.y); highlightSegs.lineTo(p2.x, p2.y); } drawPathHighlight(highlightSegs, line); } Iterator it = new OffsetIterator(wayNodes, offset); while (it.hasNext()) { Point p = it.next(); if (lastPoint != null) { Point p1 = lastPoint; Point p2 = p; /** * Do custom clipping to work around openjdk bug. It leads to * drawing artefacts when zooming in a lot. (#4289, #4424) * (Looks like int overflow.) */ LineClip clip = new LineClip(p1, p2, bounds); if (clip.execute()) { if (!p1.equals(clip.getP1())) { p1 = clip.getP1(); path.moveTo(p1.x, p1.y); } else if (initialMoveToNeeded) { initialMoveToNeeded = false; path.moveTo(p1.x, p1.y); } p2 = clip.getP2(); path.lineTo(p2.x, p2.y); /* draw arrow */ if (showHeadArrowOnly ? !it.hasNext() : showOrientation) { final double segmentLength = p1.distance(p2); if (segmentLength != 0) { final double l = (10. + line.getLineWidth()) / segmentLength; final double sx = l * (p1.x - p2.x); final double sy = l * (p1.y - p2.y); orientationArrows.moveTo(p2.x + cosPHI * sx - sinPHI * sy, p2.y + sinPHI * sx + cosPHI * sy); orientationArrows.lineTo(p2.x, p2.y); orientationArrows.lineTo(p2.x + cosPHI * sx + sinPHI * sy, p2.y - sinPHI * sx + cosPHI * sy); } } if (showOneway) { final double segmentLength = p1.distance(p2); if (segmentLength != 0) { final double nx = (p2.x - p1.x) / segmentLength; final double ny = (p2.y - p1.y) / segmentLength; final double interval = 60; // distance from p1 double dist = interval - (wayLength % interval); while (dist < segmentLength) { for (int i = 0; i < 2; ++i) { float onewaySize = i == 0 ? 3f : 2f; GeneralPath onewayPath = i == 0 ? onewayArrowsCasing : onewayArrows; // scale such that border is 1 px final double fac = -(onewayReversed ? -1 : 1) * onewaySize * (1 + sinPHI) / (sinPHI * cosPHI); final double sx = nx * fac; final double sy = ny * fac; // Attach the triangle at the incenter and not at the tip. // Makes the border even at all sides. final double x = p1.x + nx * (dist + (onewayReversed ? -1 : 1) * (onewaySize / sinPHI)); final double y = p1.y + ny * (dist + (onewayReversed ? -1 : 1) * (onewaySize / sinPHI)); onewayPath.moveTo(x, y); onewayPath.lineTo(x + cosPHI * sx - sinPHI * sy, y + sinPHI * sx + cosPHI * sy); onewayPath.lineTo(x + cosPHI * sx + sinPHI * sy, y - sinPHI * sx + cosPHI * sy); onewayPath.lineTo(x, y); } dist += interval; } } wayLength += segmentLength; } } } lastPoint = p; } if (way.isHighlighted()) { drawPathHighlight(path, line); } displaySegments(path, orientationArrows, onewayArrows, onewayArrowsCasing, color, line, dashes, dashedColor); } /** * Gets the "circum". This is the distance on the map in meters that 100 screen pixels represent. * @return The "circum" */ public double getCircum() { return circum; } @Override public void getColors() { super.getColors(); this.highlightColorTransparent = new Color(highlightColor.getRed(), highlightColor.getGreen(), highlightColor.getBlue(), 100); this.backgroundColor = PaintColors.getBackgroundColor(); } @Override public void getSettings(boolean virtual) { super.getSettings(virtual); paintSettings = MapPaintSettings.INSTANCE; circum = nc.getDist100Pixel(); leftHandTraffic = Main.pref.getBoolean("mappaint.lefthandtraffic", false); useStrokes = paintSettings.getUseStrokesDistance() > circum; showNames = paintSettings.getShowNamesDistance() > circum; showIcons = paintSettings.getShowIconsDistance() > circum; isOutlineOnly = paintSettings.isOutlineOnly(); isUnclosedAreaHighlight = paintSettings.isUnclosedAreaHighlight(); unclosedAreaHighlightWidth = paintSettings.getUnclosedAreaHighlightWidth(); partialFillThreshold = paintSettings.getPartialFillThreshold(); orderFont = new Font(Main.pref.get("mappaint.font", "Droid Sans"), Font.PLAIN, Main.pref.getInteger("mappaint.fontsize", 8)); antialiasing = Main.pref.getBoolean("mappaint.use-antialiasing", true) ? RenderingHints.VALUE_ANTIALIAS_ON : RenderingHints.VALUE_ANTIALIAS_OFF; g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, antialiasing); Object textAntialiasing; switch (Main.pref.get("mappaint.text-antialiasing", "default")) { case "on": textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_ON; break; case "off": textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_OFF; break; case "gasp": textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_GASP; break; case "lcd-hrgb": textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_LCD_HRGB; break; case "lcd-hbgr": textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_LCD_HBGR; break; case "lcd-vrgb": textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_LCD_VRGB; break; case "lcd-vbgr": textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_LCD_VBGR; break; default: textAntialiasing = RenderingHints.VALUE_TEXT_ANTIALIAS_DEFAULT; } g.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, textAntialiasing); highlightLineWidth = Main.pref.getInteger("mappaint.highlight.width", 4); highlightPointRadius = Main.pref.getInteger("mappaint.highlight.radius", 7); widerHighlight = Main.pref.getInteger("mappaint.highlight.bigger-increment", 5); highlightStep = Main.pref.getInteger("mappaint.highlight.step", 4); } private static Path2D.Double getPath(Way w) { Path2D.Double path = new Path2D.Double(); boolean initial = true; for (Node n : w.getNodes()) { EastNorth p = n.getEastNorth(); if (p != null) { if (initial) { path.moveTo(p.getX(), p.getY()); initial = false; } else { path.lineTo(p.getX(), p.getY()); } } } if (w.isClosed()) { path.closePath(); } return path; } private boolean isAreaVisible(Path2D.Double area) { Rectangle2D bounds = area.getBounds2D(); if (bounds.isEmpty()) return false; Point2D p = nc.getPoint2D(new EastNorth(bounds.getX(), bounds.getY())); if (p.getX() > nc.getWidth()) return false; if (p.getY() < 0) return false; p = nc.getPoint2D(new EastNorth(bounds.getX() + bounds.getWidth(), bounds.getY() + bounds.getHeight())); if (p.getX() < 0) return false; if (p.getY() > nc.getHeight()) return false; return true; } public boolean isInactiveMode() { return isInactiveMode; } public boolean isShowIcons() { return showIcons; } public boolean isShowNames() { return showNames; } private static double[] pointAt(double t, Polygon poly, double pathLength) { double totalLen = t * pathLength; double curLen = 0; long dx, dy; double segLen; // Yes, it is inefficient to iterate from the beginning for each glyph. // Can be optimized if it turns out to be slow. for (int i = 1; i < poly.npoints; ++i) { dx = poly.xpoints[i] - poly.xpoints[i-1]; dy = poly.ypoints[i] - poly.ypoints[i-1]; segLen = Math.sqrt(dx*dx + dy*dy); if (totalLen > curLen + segLen) { curLen += segLen; continue; } return new double[] { poly.xpoints[i-1]+(totalLen - curLen)/segLen*dx, poly.ypoints[i-1]+(totalLen - curLen)/segLen*dy, Math.atan2(dy, dx)}; } return null; } /** * Computes the flags for a given OSM primitive. * @param primitive The primititve to compute the flags for. * @param checkOuterMember true if we should also add {@link #FLAG_OUTERMEMBER_OF_SELECTED} * @return The flag. */ public static int computeFlags(OsmPrimitive primitive, boolean checkOuterMember) { if (primitive.isDisabled()) { return FLAG_DISABLED; } else if (primitive.isSelected()) { return FLAG_SELECTED; } else if (checkOuterMember && primitive.isOuterMemberOfSelected()) { return FLAG_OUTERMEMBER_OF_SELECTED; } else if (primitive.isMemberOfSelected()) { return FLAG_MEMBER_OF_SELECTED; } else { return FLAG_NORMAL; } } private class ComputeStyleListWorker implements Callable>, Visitor { private final List input; private final int from; private final int to; private final List output; private final ElemStyles styles = MapPaintStyles.getStyles(); private final boolean drawArea = circum <= Main.pref.getInteger("mappaint.fillareas", 10000000); private final boolean drawMultipolygon = drawArea && Main.pref.getBoolean("mappaint.multipolygon", true); private final boolean drawRestriction = Main.pref.getBoolean("mappaint.restriction", true); /** * Constructs a new {@code ComputeStyleListWorker}. * @param input the primitives to process * @param from first index of input to use * @param to last index + 1 * @param output the list of styles to which styles will be added */ ComputeStyleListWorker(final List input, int from, int to, List output) { this.input = input; this.from = from; this.to = to; this.output = output; this.styles.setDrawMultipolygon(drawMultipolygon); } @Override public List call() throws Exception { MapCSSStyleSource.STYLE_SOURCE_LOCK.readLock().lock(); try { for (int i = from; i < to; i++) { OsmPrimitive osm = input.get(i); if (osm.isDrawable()) { osm.accept(this); } } return output; } finally { MapCSSStyleSource.STYLE_SOURCE_LOCK.readLock().unlock(); } } @Override public void visit(Node n) { add(n, computeFlags(n, false)); } @Override public void visit(Way w) { add(w, computeFlags(w, true)); } @Override public void visit(Relation r) { add(r, computeFlags(r, true)); } @Override public void visit(Changeset cs) { throw new UnsupportedOperationException(); } public void add(Node osm, int flags) { StyleList sl = styles.get(osm, circum, nc); for (ElemStyle s : sl) { output.add(new StyleRecord(s, osm, flags)); } } public void add(Relation osm, int flags) { StyleList sl = styles.get(osm, circum, nc); for (ElemStyle s : sl) { if (drawMultipolygon && drawArea && s instanceof AreaElemStyle && (flags & FLAG_DISABLED) == 0) { output.add(new StyleRecord(s, osm, flags)); } else if (drawRestriction && s instanceof NodeElemStyle) { output.add(new StyleRecord(s, osm, flags)); } } } public void add(Way osm, int flags) { StyleList sl = styles.get(osm, circum, nc); for (ElemStyle s : sl) { if (!(drawArea && (flags & FLAG_DISABLED) == 0) && s instanceof AreaElemStyle) { continue; } output.add(new StyleRecord(s, osm, flags)); } } } private class ConcurrentTasksHelper { private final List allStyleElems; ConcurrentTasksHelper(List allStyleElems) { this.allStyleElems = allStyleElems; } void process(List prims) { final List tasks = new ArrayList<>(); final int bucketsize = Math.max(100, prims.size()/THREAD_POOL.a/3); final int noBuckets = (prims.size() + bucketsize - 1) / bucketsize; final boolean singleThread = THREAD_POOL.a == 1 || noBuckets == 1; for (int i = 0; i < noBuckets; i++) { int from = i*bucketsize; int to = Math.min((i+1)*bucketsize, prims.size()); List target = singleThread ? allStyleElems : new ArrayList(to - from); tasks.add(new ComputeStyleListWorker(prims, from, to, target)); } if (singleThread) { try { for (ComputeStyleListWorker task : tasks) { task.call(); } } catch (Exception ex) { throw new RuntimeException(ex); } } else if (!tasks.isEmpty()) { try { for (Future> future : THREAD_POOL.b.invokeAll(tasks)) { allStyleElems.addAll(future.get()); } } catch (InterruptedException | ExecutionException ex) { throw new RuntimeException(ex); } } } } @Override public void render(final DataSet data, boolean renderVirtualNodes, Bounds bounds) { BBox bbox = bounds.toBBox(); getSettings(renderVirtualNodes); boolean benchmark = Main.isTraceEnabled() || Main.pref.getBoolean("mappaint.render.benchmark", false); data.getReadLock().lock(); try { highlightWaySegments = data.getHighlightedWaySegments(); long timeStart = 0, timePhase1 = 0, timeFinished; if (benchmark) { timeStart = System.currentTimeMillis(); System.err.print("BENCHMARK: rendering "); } List nodes = data.searchNodes(bbox); List ways = data.searchWays(bbox); List relations = data.searchRelations(bbox); final List allStyleElems = new ArrayList<>(nodes.size()+ways.size()+relations.size()); ConcurrentTasksHelper helper = new ConcurrentTasksHelper(allStyleElems); // Need to process all relations first. // Reason: Make sure, ElemStyles.getStyleCacheWithRange is // not called for the same primitive in parallel threads. // (Could be synchronized, but try to avoid this for // performance reasons.) helper.process(relations); helper.process(new CompositeList<>(nodes, ways)); if (benchmark) { timePhase1 = System.currentTimeMillis(); System.err.print("phase 1 (calculate styles): " + Utils.getDurationString(timePhase1 - timeStart)); } Collections.sort(allStyleElems); // TODO: try parallel sort when switching to Java 8 for (StyleRecord r : allStyleElems) { r.style.paintPrimitive( r.osm, paintSettings, StyledMapRenderer.this, (r.flags & FLAG_SELECTED) != 0, (r.flags & FLAG_OUTERMEMBER_OF_SELECTED) != 0, (r.flags & FLAG_MEMBER_OF_SELECTED) != 0 ); } if (benchmark) { timeFinished = System.currentTimeMillis(); System.err.println("; phase 2 (draw): " + Utils.getDurationString(timeFinished - timePhase1) + "; total: " + Utils.getDurationString(timeFinished - timeStart) + " (scale: " + circum + " zoom level: " + Selector.GeneralSelector.scale2level(circum) + ')'); } drawVirtualNodes(data, bbox); } finally { data.getReadLock().unlock(); } } }