Subversion Repositories mkgmap

Rev

Rev 2906 | View as "text/plain" | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
 * Copyright (C) 2012.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 3 or
 * version 2 as published by the Free Software Foundation.
 *
 * 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
 * General Public License for more details.
 */


package uk.me.parabola.mkgmap.osmstyle.function;

import java.text.DecimalFormat;
import java.text.DecimalFormatSymbols;
import java.util.Locale;
import java.util.Map.Entry;

import uk.me.parabola.imgfmt.app.Coord;
import uk.me.parabola.log.Logger;
import uk.me.parabola.mkgmap.reader.osm.Element;
import uk.me.parabola.mkgmap.reader.osm.Relation;
import uk.me.parabola.mkgmap.reader.osm.Way;
import uk.me.parabola.mkgmap.scan.SyntaxException;

/**
 * Calculates the length of a way or a relation in meter. The length of a
 * relation is defined as the sum of its member lengths.
 *
 * @author WanMil
 */

public class LengthFunction extends CachedFunction {
        private static final Logger log = Logger.getLogger(LengthFunction.class);

        private final DecimalFormat nf = new DecimalFormat("0.0#####################", DecimalFormatSymbols.getInstance(Locale.US));

        public LengthFunction() {
                super(null);
        }

        protected String calcImpl(Element el) {
                double length = calcLength(el);
                return nf.format(length);
        }
       
        private double calcLength(Element el) {
                if (el instanceof Way) {
                        Way w = (Way)el;
                        double length = 0;
                        Coord prevC = null;
                        for (Coord c : w.getPoints()) {
                                if (prevC != null) {
                                        length += prevC.distance(c);
                                }
                                prevC = c;
                        }
                        return length;
                } else if (el instanceof Relation) {
                        Relation rel = (Relation)el;
                        double length = 0;
                        for (Entry<String,Element> relElem : rel.getElements()) {
                                if (relElem.getValue() instanceof Way || relElem.getValue() instanceof Relation) {
                                        if (rel == relElem.getValue()) {
                                                // avoid recursive call
                                                log.error("Relation "+rel.getId()+" contains itself as element. This is not supported.");
                                        } else {
                                                length += calcLength(relElem.getValue());
                                        }
                                }
                        }
                        return length;
                } else {
                        throw new SyntaxException("length() cannot calculate elements of type "+el.getClass().getName());
                }
        }
       
        public String getName() {
                return "length";
        }

        public boolean supportsWay() {
                return true;
        }

        public boolean supportsRelation() {
                return true;
        }
}