Subversion Repositories mkgmap

Rev

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

/*
 * Copyright (C) 2006, 2011.
 *
 * 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.reader.osm.boundary;

import java.util.ArrayList;
import java.util.BitSet;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Queue;
import java.util.concurrent.LinkedBlockingQueue;

import uk.me.parabola.imgfmt.app.Area;
import uk.me.parabola.imgfmt.app.Coord;
import uk.me.parabola.log.Logger;
import uk.me.parabola.mkgmap.reader.osm.MultiPolygonRelation;
import uk.me.parabola.mkgmap.reader.osm.Relation;
import uk.me.parabola.mkgmap.reader.osm.Way;
import uk.me.parabola.util.Java2DConverter;


public class BoundaryRelation extends MultiPolygonRelation {
        private static final Logger log = Logger
        .getLogger(BoundaryRelation.class);

        private java.awt.geom.Area outerResultArea;
       
        /** keeps the result of the multipolygon processing */
        private Boundary boundary;
       
        public BoundaryRelation(Relation other, Map<Long, Way> wayMap, Area bbox) {
                super(other, wayMap, bbox);
        }
       
        public Boundary getBoundary() {
                if (boundary == null) {
                        if (outerResultArea == null) {
                                return null;
                        }
                        boundary = new Boundary(outerResultArea, this, "r"+this.getId());
                        outerResultArea = null;
                }
                return boundary;
        }
       
        /**
         * Process the ways in this relation. Joins way with the role "outer" Adds
         * ways with the role "inner" to the way with the role "outer"
         */

        public void processElements() {
                log.info("Processing multipolygon", toBrowseURL());
       
                List<Way> allWays = getSourceWays();
               
                // join all single ways to polygons, try to close ways and remove non closed ways
                polygons = joinWays(allWays);
               
                outerWaysForLineTagging = new HashSet<>();
                outerTags = new HashMap<>();
               
                removeOutOfBbox(polygons);

                do {
                        closeWays(polygons, getMaxCloseDist());
                } while (connectUnclosedWays(polygons));

                removeUnclosedWays(polygons);

                // now only closed ways are left => polygons only

                // check if we have at least one polygon left
                boolean hasPolygons = !polygons.isEmpty();

                removeWaysOutsideBbox(polygons);

                if (polygons.isEmpty()) {
                        // do nothing
                        if (log.isInfoEnabled()) {
                                if (hasPolygons)
                                        log.info("Multipolygon", toBrowseURL(),
                                                        "is completely outside the bounding box. It is not processed.");
                                else
                                        log.info("Multipolygon " + toBrowseURL() + " does not contain a closed polygon.");
                        }
                        tagOuterWays();
                        cleanup();
                        return;
                }
               
                // the intersectingPolygons marks all intersecting/overlapping polygons
                intersectingPolygons = new HashSet<>();
               
                // check which polygons lie inside which other polygon
                createContainsMatrix(polygons);

                // unfinishedPolygons marks which polygons are not yet processed
                unfinishedPolygons = new BitSet(polygons.size());
                unfinishedPolygons.set(0, polygons.size());

                // create bitsets which polygons belong to the outer and to the inner role
                innerPolygons = new BitSet();
                taggedInnerPolygons = new BitSet();
                outerPolygons = new BitSet();
                taggedOuterPolygons = new BitSet();
               
                int wi = 0;
                for (Way w : polygons) {
                        String role = getRole(w);
                        if ("inner".equals(role)) {
                                innerPolygons.set(wi);
                                taggedInnerPolygons.set(wi);
                        } else if ("outer".equals(role)) {
                                outerPolygons.set(wi);
                                taggedOuterPolygons.set(wi);
                        } else {
                                // unknown role => it could be both
                                innerPolygons.set(wi);
                                outerPolygons.set(wi);
                        }
                        wi++;
                }

                if (outerPolygons.isEmpty()) {
                        log.warn("Multipolygon", toBrowseURL(),
                                "does not contain any way tagged with role=outer or empty role.");
                        cleanup();
                        return;
                }

                Queue<PolygonStatus> polygonWorkingQueue = new LinkedBlockingQueue<>();
                BitSet nestedOuterPolygons = new BitSet();
                BitSet nestedInnerPolygons = new BitSet();

                BitSet outmostPolygons ;
                BitSet outmostInnerPolygons = new BitSet();
                boolean outmostInnerFound;
                do {
                        outmostInnerFound = false;
                        outmostPolygons = findOutmostPolygons(unfinishedPolygons);

                        if (outmostPolygons.intersects(taggedInnerPolygons)) {
                                outmostInnerPolygons.or(outmostPolygons);
                                outmostInnerPolygons.and(taggedInnerPolygons);

                                if (log.isDebugEnabled())
                                        log.debug("wrong inner polygons: " + outmostInnerPolygons);
                                // do not process polygons tagged with role=inner but which are
                                // not contained by any other polygon
                                unfinishedPolygons.andNot(outmostInnerPolygons);
                                outmostPolygons.andNot(outmostInnerPolygons);
                                outmostInnerFound = true;
                        }
                } while (outmostInnerFound);
               
                if (!outmostPolygons.isEmpty()) {
                        polygonWorkingQueue.addAll(getPolygonStatus(outmostPolygons, "outer"));
                }

                boolean outmostPolygonProcessing = true;
               
               
                outerResultArea = new java.awt.geom.Area();
               
                while (!polygonWorkingQueue.isEmpty()) {

                        // the polygon is not contained by any other unfinished polygon
                        PolygonStatus currentPolygon = polygonWorkingQueue.poll();

                        // this polygon is now processed and should not be used by any
                        // further step
                        unfinishedPolygons.clear(currentPolygon.index);

                        BitSet polygonContains = new BitSet();
                        polygonContains.or(containsMatrix.get(currentPolygon.index));
                        // use only polygon that are contained by the polygon
                        polygonContains.and(unfinishedPolygons);
                        // polygonContains is the intersection of the unfinished and
                        // the contained polygons

                        // get the holes
                        // these are all polygons that are in the main polygon
                        // and that are not contained by any other polygon
                        boolean holesOk;
                        BitSet holeIndexes;
                        do {
                                holeIndexes = findOutmostPolygons(polygonContains);
                                holesOk = true;

                                if (currentPolygon.outer) {
                                        // for role=outer only role=inner is allowed
                                        if (holeIndexes.intersects(taggedOuterPolygons)) {
                                                BitSet addOuterNestedPolygons = new BitSet();
                                                addOuterNestedPolygons.or(holeIndexes);
                                                addOuterNestedPolygons.and(taggedOuterPolygons);
                                                nestedOuterPolygons.or(addOuterNestedPolygons);
                                                holeIndexes.andNot(addOuterNestedPolygons);
                                                // do not process them
                                                unfinishedPolygons.andNot(addOuterNestedPolygons);
                                                polygonContains.andNot(addOuterNestedPolygons);
                                               
                                                // recalculate the holes again to get all inner polygons
                                                // in the nested outer polygons
                                                holesOk = false;
                                        }
                                } else {
                                        // for role=inner both role=inner and role=outer is supported
                                        // although inner in inner is not officially allowed
                                        if (holeIndexes.intersects(taggedInnerPolygons)) {
                                                // process inner in inner but issue a warning later
                                                BitSet addInnerNestedPolygons = new BitSet();
                                                addInnerNestedPolygons.or(holeIndexes);
                                                addInnerNestedPolygons.and(taggedInnerPolygons);
                                                nestedInnerPolygons.or(addInnerNestedPolygons);
                                        }
                                }
                        } while (!holesOk);

                        ArrayList<PolygonStatus> holes = getPolygonStatus(holeIndexes,
                                (currentPolygon.outer ? "inner" : "outer"));

                        // these polygons must all be checked for holes
                        polygonWorkingQueue.addAll(holes);

                        if (currentPolygon.outer) {
                                // add the original ways to the list of ways that get the line tags of the mp
                                // the joined ways may be changed by the auto closing algorithm
                                outerWaysForLineTagging.addAll(currentPolygon.polygon.getOriginalWays());
                        }
                       
                        if (currentPolygon.outer) {
                                java.awt.geom.Area toAdd = Java2DConverter.createArea(currentPolygon.polygon.getPoints());
                                if (outerResultArea.isEmpty())
                                        outerResultArea = toAdd;
                                else
                                        outerResultArea.add(toAdd);

                                for (Way outerWay : currentPolygon.polygon.getOriginalWays()) {
                                        if (outmostPolygonProcessing) {
                                                for (Entry<String, String> tag : outerWay.getTagEntryIterator()) {
                                                        outerTags.put(tag.getKey(), tag.getValue());
                                                }
                                                outmostPolygonProcessing = false;
                                        } else {
                                                for (String tag : new ArrayList<>(outerTags.keySet())) {
                                                        if (!outerTags.get(tag).equals(outerWay.getTag(tag))) {
                                                                outerTags.remove(tag);
                                                        }
                                                }
                                        }
                                }
                        } else {
                                outerResultArea.subtract(Java2DConverter
                                                .createArea(currentPolygon.polygon.getPoints()));
                        }
                }
               
                if (hasStyleRelevantTags(this)) {
                        outerTags.clear();
                        for (Entry<String,String> mpTags : getTagEntryIterator()) {
                                if ("type".equals(mpTags.getKey())==false) {
                                        outerTags.put(mpTags.getKey(), mpTags.getValue());
                                }
                        }
                } else {
                        for (Entry<String,String> mpTags : outerTags.entrySet()) {
                                addTag(mpTags.getKey(), mpTags.getValue());
                        }
                }
               
                // Go through all original outer ways, create a copy, tag them
                // with the mp tags and mark them only to be used for polyline processing
                // This enables the style file to decide if the polygon information or
                // the simple line information should be used.
                for (Way orgOuterWay : outerWaysForLineTagging) {
//                      Way lineTagWay =  new Way(FakeIdGenerator.makeFakeId(), orgOuterWay.getPoints());
//                      lineTagWay.setName(orgOuterWay.getName());
//                      lineTagWay.addTag(STYLE_FILTER_TAG, STYLE_FILTER_LINE);
                        for (Entry<String,String> tag : outerTags.entrySet()) {
//                              lineTagWay.addTag(tag.getKey(), tag.getValue());
                               
                                // remove the tag from the original way if it has the same value
                                if (tag.getValue().equals(orgOuterWay.getTag(tag.getKey()))) {
                                        removeTagsInOrgWays(orgOuterWay, tag.getKey());
                                }
                        }
                       
//                      if (log.isDebugEnabled())
//                              log.debug("Add line way", lineTagWay.getId(), lineTagWay.toTagString());
//                      tileWayMap.put(lineTagWay.getId(), lineTagWay);
                }
               
                postProcessing();
                cleanup();
        }

        protected boolean connectUnclosedWays(List<JoinedWay> allWays) {
                List<JoinedWay> unclosed = new ArrayList<>();

                for (JoinedWay w : allWays) {
                        if (!w.hasIdenticalEndPoints()) {
                                unclosed.add(w);
                        }
                }
                // try to connect ways lying outside or on the bbox
                if (unclosed.size() >= 2) {
                        log.debug("Checking",unclosed.size(),"unclosed ways for connections outside the bbox");
                        Map<Coord, JoinedWay> outOfBboxPoints = new IdentityHashMap<>();
                       
                        // check all ways for endpoints outside or on the bbox
                        for (JoinedWay w : unclosed) {
                                Coord c1 = w.getFirstPoint();
                                Coord c2 = w.getLastPoint();
                                outOfBboxPoints.put(c1, w);
                                outOfBboxPoints.put(c2, w);
                        }
                       
                        if (outOfBboxPoints.size() < 2) {
                                log.debug(outOfBboxPoints.size(),"point outside the bbox. No connection possible.");
                                return false;
                        }
                       
                        List<ConnectionData> coordPairs = new ArrayList<>();
                        ArrayList<Coord> coords = new ArrayList<>(outOfBboxPoints.keySet());
                        for (int i = 0; i < coords.size(); i++) {
                                for (int j = i + 1; j < coords.size(); j++) {
                                        ConnectionData cd = new ConnectionData();
                                        cd.c1 = coords.get(i);
                                        cd.c2 = coords.get(j);
                                        cd.w1 = outOfBboxPoints.get(cd.c1);                                    
                                        cd.w2 = outOfBboxPoints.get(cd.c2);                                    
                                       
                                        cd.distance = cd.c1.distance(cd.c2);
                                        coordPairs.add(cd);
                                }
                        }
                       
                        if (coordPairs.isEmpty()) {
                                log.debug("All potential connections cross the bbox. No connection possible.");
                                return false;
                        } else {
                                // retrieve the connection with the minimum distance
                                ConnectionData minCon = Collections.min(coordPairs,
                                                (o1, o2) -> Double.compare(o1.distance, o2.distance));
                               
                                if (minCon.distance < getMaxCloseDist()) {

                                        if (minCon.w1 == minCon.w2) {
                                                log.debug("Close a gap in way", minCon.w1);
                                                if (minCon.imC != null)
                                                        minCon.w1.getPoints().add(minCon.imC);
                                                minCon.w1.closeWayArtificially();
                                        } else {
                                                log.debug("Connect", minCon.w1, "with", minCon.w2);
                                                if (minCon.w1.getFirstPoint() == minCon.c1) {
                                                        Collections.reverse(minCon.w1.getPoints());
                                                }
                                                if (minCon.w2.getFirstPoint() != minCon.c2) {
                                                        Collections.reverse(minCon.w2.getPoints());
                                                }

                                                minCon.w1.getPoints().addAll(minCon.w2.getPoints());
                                                minCon.w1.addWay(minCon.w2);
                                                allWays.remove(minCon.w2);
                                        }
                                        return true;
                                }
                        }
                }
                return false;
        }
       
        @Override
        protected double getMaxCloseDist() {
                double dist = 1000;
                String admString= getTag("admin_level");
               
                if ("2".equals(admString)) {
                        dist = 50000;
                } else if ("3".equals(admString)) {
                        dist = 20000;
                }else if ("4".equals(admString)) {
                        dist = 4000;
                }
                return dist;
        }
       
        private void removeOutOfBbox(List<JoinedWay> polygons) {
                ListIterator<JoinedWay> pIter = polygons.listIterator();
                while (pIter.hasNext()) {
                        JoinedWay w = pIter.next();
                        Coord first = w.getFirstPoint();
                        Coord last =  w.getLastPoint();
                        if (first != last) {
                                // the way is not closed
                                // check if one of start/endpoint is out of the bounding box
                                // in this case it is too risky to close it
                                if (!getTileBounds().contains(first) || !getTileBounds().contains(last)) {
                                        pIter.remove();
                                }
                        }
                }

        }

        @Override
        protected void cleanup() {
                super.cleanup();
                this.getElements().clear();
                ((ArrayList<?>)this.getElements()).trimToSize();
        }
}