/* * Copyright 2010, 2011, 2012 mapsforge.org * * This program is free software: you can redistribute it and/or modify it under the * terms of the GNU Lesser General Public License as published by the Free Software * Foundation, either version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but WITHOUT ANY * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License along with * this program. If not, see . */ package org.oscim.theme.renderinstruction; import java.util.Locale; import org.oscim.graphics.Color; import org.oscim.graphics.Paint.FontFamily; import org.oscim.graphics.Paint.FontStyle; import org.oscim.renderer.atlas.TextureRegion; import org.oscim.theme.IRenderCallback; import org.oscim.theme.RenderThemeHandler; import org.xml.sax.Attributes; import android.graphics.Paint; import android.graphics.Paint.Align; import android.graphics.Paint.FontMetrics; import android.graphics.Paint.Style; import android.graphics.Typeface; /** * Represents a text along a polyline on the map. */ public final class Text extends RenderInstruction { /** * @param elementName * the name of the XML element. * @param attributes * the attributes of the XML element. * @param caption * ... * @return a new Text with the given rendering attributes. */ public static Text create(String elementName, Attributes attributes, boolean caption) { String textKey = null; FontFamily fontFamily = FontFamily.DEFAULT; FontStyle fontStyle = FontStyle.NORMAL; float fontSize = 0; int fill = Color.BLACK; int stroke = Color.BLACK; float strokeWidth = 0; String style = null; float dy = 0; int priority = Integer.MAX_VALUE; String symbol = null; for (int i = 0; i < attributes.getLength(); ++i) { String name = attributes.getLocalName(i); String value = attributes.getValue(i); if ("name".equals(name)) style = value; else if ("k".equals(name)) { textKey = value.intern(); } else if ("font-family".equals(name)) { fontFamily = FontFamily.valueOf(value.toUpperCase(Locale.ENGLISH)); } else if ("font-style".equals(name)) { fontStyle = FontStyle.valueOf(value.toUpperCase(Locale.ENGLISH)); } else if ("font-size".equals(name)) { fontSize = Float.parseFloat(value); } else if ("fill".equals(name)) { fill = Color.parseColor(value); } else if ("stroke".equals(name)) { stroke = Color.parseColor(value); } else if ("stroke-width".equals(name)) { strokeWidth = Float.parseFloat(value); } else if ("caption".equals(name)) { caption = Boolean.parseBoolean(value); } else if ("priority".equals(name)) { priority = Integer.parseInt(value); } else if ("dy".equals(name)) { dy = Float.parseFloat(value); } else if ("symbol".equals(name)) { symbol = value; } else { RenderThemeHandler.logUnknownAttribute(elementName, name, value, i); } } validate(elementName, textKey, fontSize, strokeWidth); return new Text(style, textKey, fontFamily, fontStyle, fontSize, fill, stroke, strokeWidth, dy, caption, symbol, priority); } private static void validate(String elementName, String textKey, float fontSize, float strokeWidth) { if (textKey == null) { throw new IllegalArgumentException("missing attribute k for element: " + elementName); } else if (fontSize < 0) { throw new IllegalArgumentException("font-size must not be negative: " + fontSize); } else if (strokeWidth < 0) { throw new IllegalArgumentException("stroke-width must not be negative: " + strokeWidth); } } public final String style; public final float fontSize; public final Paint paint; public final Paint stroke; public final String textKey; public final String symbol; public final boolean caption; public final float dy; public final int priority; public float fontHeight; public float fontDescent; public TextureRegion texture; public static Text createText(float fontSize, float strokeWidth, int fill, int outline, boolean billboard) { Text t = new Text("", "", FontFamily.DEFAULT, FontStyle.NORMAL, fontSize, fill, outline, strokeWidth, 0, billboard, null, Integer.MAX_VALUE); FontMetrics fm = t.paint.getFontMetrics(); t.fontHeight = (float) Math.ceil(Math.abs(fm.bottom) + Math.abs(fm.top)); t.fontDescent = (float) Math.ceil(Math.abs(fm.bottom)); return t; } private static int getStyle(FontStyle fontStyle) { switch (fontStyle) { case BOLD: return 1; case BOLD_ITALIC: return 3; case ITALIC: return 2; case NORMAL: return 0; } throw new IllegalArgumentException("unknown font style: " + fontStyle); } private static Typeface getTypeface(FontFamily fontFamily) { switch (fontFamily) { case DEFAULT: return Typeface.DEFAULT; case DEFAULT_BOLD: return Typeface.DEFAULT_BOLD; case MONOSPACE: return Typeface.MONOSPACE; case SANS_SERIF: return Typeface.SANS_SERIF; case SERIF: return Typeface.SERIF; } throw new IllegalArgumentException("unknown font family: " + fontFamily); } private Text(String style, String textKey, FontFamily fontFamily, FontStyle fontStyle, float fontSize, int fill, int outline, float strokeWidth, float dy, boolean caption, String symbol, int priority) { this.style = style; this.textKey = textKey; this.caption = caption; this.dy = -dy; this.priority = priority; this.symbol = symbol; //paint = Graphics.res.getPaint(); paint = new Paint(Paint.ANTI_ALIAS_FLAG); paint.setTextAlign(Align.CENTER); Typeface typeFace = Typeface.create(Text.getTypeface(fontFamily), Text.getStyle(fontStyle)); //paint.setTypeface(fontFamily, fontStyle); paint.setTypeface(typeFace); paint.setColor(fill); paint.setTextSize(fontSize); if (strokeWidth > 0) { stroke = new Paint(Paint.ANTI_ALIAS_FLAG); //stroke = Graphics.res.getPaint(); stroke.setStyle(Style.STROKE); stroke.setTextAlign(Align.CENTER); //stroke.setTypeface(fontFamily, fontStyle); stroke.setTypeface(typeFace); stroke.setColor(outline); stroke.setStrokeWidth(strokeWidth); stroke.setTextSize(fontSize); } else stroke = null; this.fontSize = fontSize; } @Override public void renderNode(IRenderCallback renderCallback) { if (caption) renderCallback.renderPointOfInterestCaption(this); } @Override public void renderWay(IRenderCallback renderCallback) { if (caption) renderCallback.renderAreaCaption(this); else renderCallback.renderWayText(this); } @Override public void scaleTextSize(float scaleFactor) { paint.setTextSize(fontSize * scaleFactor); if (stroke != null) stroke.setTextSize(fontSize * scaleFactor); FontMetrics fm = paint.getFontMetrics(); fontHeight = (float) Math.ceil(Math.abs(fm.bottom) + Math.abs(fm.top)); fontDescent = (float) Math.ceil(Math.abs(fm.bottom)); // fontHeight = paint.getFontHeight(); // fontDescent = paint.getFontDescent(); } }