source: josm/trunk/src/org/openstreetmap/josm/gui/mappaint/LineElemStyle.java@ 7136

Last change on this file since 7136 was 7136, checked in by bastiK, 10 years ago

mapcss: add support for alpha info in color property, e.g.
color: #aa0022ee; or color: rgba(1.0, 0.2, 0.8, 0.8);
*opacity properties still have higher priority, if specified
explicitly

  • Property svn:eol-style set to native
File size: 14.7 KB
Line 
1// License: GPL. For details, see LICENSE file.
2package org.openstreetmap.josm.gui.mappaint;
3
4import java.awt.BasicStroke;
5import java.awt.Color;
6import java.util.Arrays;
7import java.util.Objects;
8
9import org.openstreetmap.josm.Main;
10import org.openstreetmap.josm.data.osm.Node;
11import org.openstreetmap.josm.data.osm.OsmPrimitive;
12import org.openstreetmap.josm.data.osm.Way;
13import org.openstreetmap.josm.data.osm.visitor.paint.MapPaintSettings;
14import org.openstreetmap.josm.data.osm.visitor.paint.PaintColors;
15import org.openstreetmap.josm.data.osm.visitor.paint.StyledMapRenderer;
16import org.openstreetmap.josm.gui.mappaint.mapcss.Instruction.RelativeFloat;
17import org.openstreetmap.josm.tools.Utils;
18
19public class LineElemStyle extends ElemStyle {
20
21 public static LineElemStyle createSimpleLineStyle(Color color, boolean isAreaEdge) {
22 MultiCascade mc = new MultiCascade();
23 Cascade c = mc.getOrCreateCascade("default");
24 c.put(WIDTH, Keyword.DEFAULT);
25 c.put(COLOR, color != null ? color : PaintColors.UNTAGGED.get());
26 c.put(OPACITY, 1f);
27 if (isAreaEdge) {
28 c.put(Z_INDEX, -3f);
29 }
30 return createLine(new Environment(null, mc, "default", null));
31 }
32 public static final LineElemStyle UNTAGGED_WAY = createSimpleLineStyle(null, false);
33
34 private BasicStroke line;
35 public Color color;
36 public Color dashesBackground;
37 public float offset;
38 public float realWidth; // the real width of this line in meter
39
40 private BasicStroke dashesLine;
41
42 protected enum LineType {
43 NORMAL("", 3f),
44 CASING("casing-", 2f),
45 LEFT_CASING("left-casing-", 2.1f),
46 RIGHT_CASING("right-casing-", 2.1f);
47
48 public final String prefix;
49 public final float default_major_z_index;
50
51 LineType(String prefix, float default_major_z_index) {
52 this.prefix = prefix;
53 this.default_major_z_index = default_major_z_index;
54 }
55 }
56
57 protected LineElemStyle(Cascade c, float default_major_z_index, BasicStroke line, Color color, BasicStroke dashesLine, Color dashesBackground, float offset, float realWidth) {
58 super(c, default_major_z_index);
59 this.line = line;
60 this.color = color;
61 this.dashesLine = dashesLine;
62 this.dashesBackground = dashesBackground;
63 this.offset = offset;
64 this.realWidth = realWidth;
65 }
66
67 public static LineElemStyle createLine(Environment env) {
68 return createImpl(env, LineType.NORMAL);
69 }
70
71 public static LineElemStyle createLeftCasing(Environment env) {
72 LineElemStyle leftCasing = createImpl(env, LineType.LEFT_CASING);
73 if (leftCasing != null) {
74 leftCasing.isModifier = true;
75 }
76 return leftCasing;
77 }
78
79 public static LineElemStyle createRightCasing(Environment env) {
80 LineElemStyle rightCasing = createImpl(env, LineType.RIGHT_CASING);
81 if (rightCasing != null) {
82 rightCasing.isModifier = true;
83 }
84 return rightCasing;
85 }
86
87 public static LineElemStyle createCasing(Environment env) {
88 LineElemStyle casing = createImpl(env, LineType.CASING);
89 if (casing != null) {
90 casing.isModifier = true;
91 }
92 return casing;
93 }
94
95 private static LineElemStyle createImpl(Environment env, LineType type) {
96 Cascade c = env.mc.getCascade(env.layer);
97 Cascade c_def = env.mc.getCascade("default");
98 Float width;
99 switch (type) {
100 case NORMAL:
101 width = getWidth(c, WIDTH, getWidth(c_def, WIDTH, null));
102 break;
103 case CASING:
104 Float casingWidth = c.get(type.prefix + WIDTH, null, Float.class, true);
105 if (casingWidth == null) {
106 RelativeFloat rel_casingWidth = c.get(type.prefix + WIDTH, null, RelativeFloat.class, true);
107 if (rel_casingWidth != null) {
108 casingWidth = rel_casingWidth.val / 2;
109 }
110 }
111 if (casingWidth == null)
112 return null;
113 width = getWidth(c, WIDTH, getWidth(c_def, WIDTH, null));
114 if (width == null) {
115 width = 0f;
116 }
117 width += 2 * casingWidth;
118 break;
119 case LEFT_CASING:
120 case RIGHT_CASING:
121 width = getWidth(c, type.prefix + WIDTH, null);
122 break;
123 default:
124 throw new AssertionError();
125 }
126 if (width == null)
127 return null;
128
129 float realWidth = c.get(type.prefix + REAL_WIDTH, 0f, Float.class);
130 if (realWidth > 0 && MapPaintSettings.INSTANCE.isUseRealWidth()) {
131
132 /* if we have a "width" tag, try use it */
133 String widthTag = env.osm.get("width");
134 if (widthTag == null) {
135 widthTag = env.osm.get("est_width");
136 }
137 if (widthTag != null) {
138 try {
139 realWidth = Float.valueOf(widthTag);
140 } catch(NumberFormatException nfe) {
141 Main.warn(nfe);
142 }
143 }
144 }
145
146 Float offset = c.get(OFFSET, 0f, Float.class);
147 switch (type) {
148 case NORMAL:
149 break;
150 case CASING:
151 offset += c.get(type.prefix + OFFSET, 0f, Float.class);
152 break;
153 case LEFT_CASING:
154 case RIGHT_CASING:
155 {
156 Float baseWidthOnDefault = getWidth(c_def, WIDTH, null);
157 Float baseWidth = getWidth(c, WIDTH, baseWidthOnDefault);
158 if (baseWidth == null || baseWidth < 2f) {
159 baseWidth = 2f;
160 }
161 float casingOffset = c.get(type.prefix + OFFSET, 0f, Float.class);
162 casingOffset += baseWidth / 2 + width / 2;
163 /* flip sign for the right-casing-offset */
164 if (type == LineType.RIGHT_CASING) {
165 casingOffset *= -1f;
166 }
167 offset += casingOffset;
168 break;
169 }
170 }
171
172 int alpha = 255;
173 Color color = c.get(type.prefix + COLOR, null, Color.class);
174 if (color != null) {
175 alpha = color.getAlpha();
176 }
177 if (type == LineType.NORMAL && color == null) {
178 color = c.get(FILL_COLOR, null, Color.class);
179 }
180 if (color == null) {
181 color = PaintColors.UNTAGGED.get();
182 }
183
184 Integer pAlpha = Utils.color_float2int(c.get(type.prefix + OPACITY, null, Float.class));
185 if (pAlpha != null) {
186 alpha = pAlpha;
187 }
188 color = new Color(color.getRed(), color.getGreen(), color.getBlue(), alpha);
189
190 float[] dashes = c.get(type.prefix + DASHES, null, float[].class, true);
191 if (dashes != null) {
192 boolean hasPositive = false;
193 for (float f : dashes) {
194 if (f > 0) {
195 hasPositive = true;
196 }
197 if (f < 0) {
198 dashes = null;
199 break;
200 }
201 }
202 if (!hasPositive || (dashes != null && dashes.length == 0)) {
203 dashes = null;
204 }
205 }
206 float dashesOffset = c.get(type.prefix + DASHES_OFFSET, 0f, Float.class);
207 Color dashesBackground = c.get(type.prefix + DASHES_BACKGROUND_COLOR, null, Color.class);
208 if (dashesBackground != null) {
209 pAlpha = Utils.color_float2int(c.get(type.prefix + DASHES_BACKGROUND_OPACITY, null, Float.class));
210 if (pAlpha != null) {
211 alpha = pAlpha;
212 }
213 dashesBackground = new Color(dashesBackground.getRed(), dashesBackground.getGreen(),
214 dashesBackground.getBlue(), alpha);
215 }
216
217 Integer cap = null;
218 Keyword capKW = c.get(type.prefix + "linecap", null, Keyword.class);
219 if (capKW != null) {
220 if ("none".equals(capKW.val)) {
221 cap = BasicStroke.CAP_BUTT;
222 } else if ("round".equals(capKW.val)) {
223 cap = BasicStroke.CAP_ROUND;
224 } else if ("square".equals(capKW.val)) {
225 cap = BasicStroke.CAP_SQUARE;
226 }
227 }
228 if (cap == null) {
229 cap = dashes != null ? BasicStroke.CAP_BUTT : BasicStroke.CAP_ROUND;
230 }
231
232 Integer join = null;
233 Keyword joinKW = c.get(type.prefix + "linejoin", null, Keyword.class);
234 if (joinKW != null) {
235 if ("round".equals(joinKW.val)) {
236 join = BasicStroke.JOIN_ROUND;
237 } else if ("miter".equals(joinKW.val)) {
238 join = BasicStroke.JOIN_MITER;
239 } else if ("bevel".equals(joinKW.val)) {
240 join = BasicStroke.JOIN_BEVEL;
241 }
242 }
243 if (join == null) {
244 join = BasicStroke.JOIN_ROUND;
245 }
246
247 float miterlimit = c.get(type.prefix + "miterlimit", 10f, Float.class);
248 if (miterlimit < 1f) {
249 miterlimit = 10f;
250 }
251
252 BasicStroke line = new BasicStroke(width, cap, join, miterlimit, dashes, dashesOffset);
253 BasicStroke dashesLine = null;
254
255 if (dashes != null && dashesBackground != null) {
256 float[] dashes2 = new float[dashes.length];
257 System.arraycopy(dashes, 0, dashes2, 1, dashes.length - 1);
258 dashes2[0] = dashes[dashes.length-1];
259 dashesLine = new BasicStroke(width, cap, join, miterlimit, dashes2, dashes2[0] + dashesOffset);
260 }
261
262 return new LineElemStyle(c, type.default_major_z_index, line, color, dashesLine, dashesBackground, offset, realWidth);
263 }
264
265 @Override
266 public void paintPrimitive(OsmPrimitive primitive, MapPaintSettings paintSettings, StyledMapRenderer painter, boolean selected, boolean member) {
267 Way w = (Way)primitive;
268 /* show direction arrows, if draw.segment.relevant_directions_only is not set,
269 the way is tagged with a direction key
270 (even if the tag is negated as in oneway=false) or the way is selected */
271 boolean showOrientation = !isModifier && (selected || paintSettings.isShowDirectionArrow()) && !paintSettings.isUseRealWidth();
272 boolean showOneway = !isModifier && !selected &&
273 !paintSettings.isUseRealWidth() &&
274 paintSettings.isShowOnewayArrow() && w.hasDirectionKeys();
275 boolean onewayReversed = w.reversedDirection();
276 /* head only takes over control if the option is true,
277 the direction should be shown at all and not only because it's selected */
278 boolean showOnlyHeadArrowOnly = showOrientation && !selected && paintSettings.isShowHeadArrowOnly();
279 Node lastN;
280
281 Color myDashedColor = dashesBackground;
282 BasicStroke myLine = line, myDashLine = dashesLine;
283 if (realWidth > 0 && paintSettings.isUseRealWidth() && !showOrientation) {
284 float myWidth = (int) (100 / (float) (painter.getCircum() / realWidth));
285 if (myWidth < line.getLineWidth()) {
286 myWidth = line.getLineWidth();
287 }
288 myLine = new BasicStroke(myWidth, line.getEndCap(), line.getLineJoin(),
289 line.getMiterLimit(), line.getDashArray(), line.getDashPhase());
290 if (dashesLine != null) {
291 myDashLine = new BasicStroke(myWidth, dashesLine.getEndCap(), dashesLine.getLineJoin(),
292 dashesLine.getMiterLimit(), dashesLine.getDashArray(), dashesLine.getDashPhase());
293 }
294 }
295
296 Color myColor = color;
297 if (selected) {
298 myColor = paintSettings.getSelectedColor(color.getAlpha());
299 } else if (member) {
300 myColor = paintSettings.getRelationSelectedColor(color.getAlpha());
301 } else if(w.isDisabled()) {
302 myColor = paintSettings.getInactiveColor();
303 myDashedColor = paintSettings.getInactiveColor();
304 }
305
306 painter.drawWay(w, myColor, myLine, myDashLine, myDashedColor, offset, showOrientation,
307 showOnlyHeadArrowOnly, showOneway, onewayReversed);
308
309 if(paintSettings.isShowOrderNumber() && !painter.isInactiveMode()) {
310 int orderNumber = 0;
311 lastN = null;
312 for(Node n : w.getNodes()) {
313 if(lastN != null) {
314 orderNumber++;
315 painter.drawOrderNumber(lastN, n, orderNumber, myColor);
316 }
317 lastN = n;
318 }
319 }
320 }
321
322 @Override
323 public boolean isProperLineStyle() {
324 return !isModifier;
325 }
326
327 @Override
328 public boolean equals(Object obj) {
329 if (obj == null || getClass() != obj.getClass())
330 return false;
331 if (!super.equals(obj))
332 return false;
333 final LineElemStyle other = (LineElemStyle) obj;
334 return Objects.equals(line, other.line) &&
335 Objects.equals(color, other.color) &&
336 Objects.equals(dashesLine, other.dashesLine) &&
337 Objects.equals(dashesBackground, other.dashesBackground) &&
338 offset == other.offset &&
339 realWidth == other.realWidth;
340 }
341
342 @Override
343 public int hashCode() {
344 int hash = super.hashCode();
345 hash = 29 * hash + line.hashCode();
346 hash = 29 * hash + color.hashCode();
347 hash = 29 * hash + (dashesLine != null ? dashesLine.hashCode() : 0);
348 hash = 29 * hash + (dashesBackground != null ? dashesBackground.hashCode() : 0);
349 hash = 29 * hash + Float.floatToIntBits(offset);
350 hash = 29 * hash + Float.floatToIntBits(realWidth);
351 return hash;
352 }
353
354 @Override
355 public String toString() {
356 return "LineElemStyle{" + super.toString() + "width=" + line.getLineWidth() +
357 " realWidth=" + realWidth + " color=" + Utils.toString(color) +
358 " dashed=" + Arrays.toString(line.getDashArray()) +
359 (line.getDashPhase() == 0f ? "" : " dashesOffses=" + line.getDashPhase()) +
360 " dashedColor=" + Utils.toString(dashesBackground) +
361 " linejoin=" + linejoinToString(line.getLineJoin()) +
362 " linecap=" + linecapToString(line.getEndCap()) +
363 (offset == 0 ? "" : " offset=" + offset) +
364 '}';
365 }
366
367 public String linejoinToString(int linejoin) {
368 switch (linejoin) {
369 case BasicStroke.JOIN_BEVEL: return "bevel";
370 case BasicStroke.JOIN_ROUND: return "round";
371 case BasicStroke.JOIN_MITER: return "miter";
372 default: return null;
373 }
374 }
375 public String linecapToString(int linecap) {
376 switch (linecap) {
377 case BasicStroke.CAP_BUTT: return "none";
378 case BasicStroke.CAP_ROUND: return "round";
379 case BasicStroke.CAP_SQUARE: return "square";
380 default: return null;
381 }
382 }
383}
Note: See TracBrowser for help on using the repository browser.