Subversion Repositories mkgmap

Rev

Rev 4551 | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
 * Copyright (C) 2011-2014.
 *
 * 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;

import java.awt.Polygon;
import java.awt.Rectangle;
import java.awt.geom.Area;
import java.awt.geom.Line2D;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.BitSet;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.IdentityHashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.ListIterator;
import java.util.Locale;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Queue;
import java.util.Set;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.logging.Level;

import uk.me.parabola.imgfmt.app.Coord;
import uk.me.parabola.log.Logger;
import uk.me.parabola.util.Java2DConverter;

/**
 * Representation of an OSM Multipolygon Relation.<br/>
 * The different way of the multipolygon are joined to polygons and inner
 * polygons are cut out from the outer polygons.
 *
 * @author WanMil
 */

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

        public static final String STYLE_FILTER_TAG = "mkgmap:stylefilter";
        public static final String STYLE_FILTER_LINE = "polyline";
        public static final String STYLE_FILTER_POLYGON = "polygon";
       
        /** A tag that is set with value true on each polygon that is created by the mp processing. */
        public static final short TKM_MP_CREATED = TagDict.getInstance().xlate("mkgmap:mp_created");
        private static final short TKM_MP_ROLE = TagDict.getInstance().xlate("mkgmap:mp_role");
        private static final short TKM_CACHE_AREA_SIZEKEY = TagDict.getInstance().xlate("mkgmap:cache_area_size");
        private final Map<Long, Way> tileWayMap;
        private final Map<Long, String> roleMap = new HashMap<>();
 
        private Map<Long, Way> mpPolygons = new LinkedHashMap<>();
       
       
        protected ArrayList<BitSet> containsMatrix;
        protected ArrayList<JoinedWay> polygons;
        protected Set<JoinedWay> intersectingPolygons;
       
        protected double largestSize;
        protected JoinedWay largestOuterPolygon;
       
        protected Set<Way> outerWaysForLineTagging;
        protected Map<String, String> outerTags;

        private final uk.me.parabola.imgfmt.app.Area tileBounds;
        private Area tileArea;
       
        private Coord cOfG = null;
       
        // the sum of all outer polygons area size
        private double mpAreaSize = 0;
       
        /**
         * A point that has a lower or equal squared distance from
         * a line is treated as if it lies one the line.<br/>
         * 1.0d is very exact. 2.0d covers rounding problems when converting
         * OSM locations to mkgmap internal format. A larger value
         * is more tolerant against imprecise OSM data.
         */

        private static final double OVERLAP_TOLERANCE_DISTANCE = 2.0d;
       
        /**
         * Create an instance based on an existing relation. We need to do this
         * because the type of the relation is not known until after all its tags
         * are read in.
         *
         * @param other
         *            The relation to base this one on.
         * @param wayMap
         *            Map of all ways.
         * @param bbox
         *            The bounding box of the tile
         */

        public MultiPolygonRelation(Relation other, Map<Long, Way> wayMap,
                        uk.me.parabola.imgfmt.app.Area bbox) {
                this.tileWayMap = wayMap;
                this.tileBounds = bbox;
                // create an Area for the bbox to clip the polygons
                tileArea = Java2DConverter.createBoundsArea(tileBounds);

                setId(other.getId());
                copyTags(other);
                this.setTagsIncomplete(other.getTagsIncomplete());

                if (log.isDebugEnabled()) {
                        log.debug("Construct multipolygon", toBrowseURL(), toTagString());
                }

                for (Map.Entry<String, Element> pair : other.getElements()) {
                        String role = pair.getKey();
                        Element el = pair.getValue();
                        if (log.isDebugEnabled()) {
                                log.debug(" ", role, el.toBrowseURL(), el.toTagString());
                        }
                        if (roleMap.containsKey(el.getId()) )
                                log.warn("repeated member with id ", el.getId(), "in multipolygon relation",this.getId(),"is ignored");
                        else {
                                addElement(role, el);
                                roleMap.put(el.getId(), role);
                        }
                }
        }
       

        /**
         * Retrieves the center point of this multipolygon. This is set in the
         * {@link #processElements()} methods so it returns <code>null</code>
         * before that. It can also return <code>null</code> in case the
         * multipolygon could not be processed.<br/>
         * The returned point may lie outside the multipolygon area. It is just
         * the center point of it.
         *
         * @return the center point of this multipolygon (maybe <code>null</code>)
         */

        public Coord getCofG() {
                return cOfG;
        }
       
        /**
         * Retrieves the mp role of the given element.
         *
         * @param element
         *            the element
         * @return the role of the element
         */

        protected String getRole(Element element) {
                String role = roleMap.get(element.getId());
                if (role != null && ("outer".equals(role) || "inner".equals(role))) {
                        return role;
                }

                return null;
        }

        /**
         * Try to join the two ways.
         *
         * @param joinWay
         *            the way to which tempWay is added in case both ways could be
         *            joined and checkOnly is false.
         * @param tempWay
         *            the way to be added to joinWay
         * @param checkOnly
         *            <code>true</code> checks only and does not perform the join
         *            operation
         * @return <code>true</code> if tempWay way is (or could be) joined to
         *         joinWay
         */

        private static boolean joinWays(JoinedWay joinWay, JoinedWay tempWay, boolean checkOnly) {
                boolean reverseTempWay = false;
                int insIdx = -1;
                int firstTmpIdx = 1;
                boolean joinable = false;
               
                // use == or equals as comparator??
                if (joinWay.getFirstPoint() == tempWay.getFirstPoint()) {
                        insIdx = 0;
                        reverseTempWay = true;
                        firstTmpIdx = 1;
                        joinable = true;
                } else if (joinWay.getLastPoint() == tempWay.getFirstPoint()) {
                        insIdx = joinWay.getPoints().size();
                        firstTmpIdx = 1;
                        joinable = true;
                } else if (joinWay.getFirstPoint() == tempWay.getLastPoint()) {
                        insIdx = 0;
                        firstTmpIdx = 0;
                        joinable = true;
                } else if (joinWay.getLastPoint() == tempWay.getLastPoint()) {
                        insIdx = joinWay.getPoints().size();
                        reverseTempWay = true;
                        firstTmpIdx = 0;
                        joinable = true;
                }
               
                if (!checkOnly && joinable){
                        int lastIdx = tempWay.getPoints().size();
                        if (firstTmpIdx == 0) {
                                // the last temp point is already contained in the joined way - do not copy it
                                lastIdx--;
                        }
                                       
                        List<Coord> tempCoords = tempWay.getPoints().subList(firstTmpIdx,lastIdx);
                       
                        if (reverseTempWay) {
                                // the remp coords need to be reversed so copy the list
                                tempCoords = new ArrayList<>(tempCoords);
                                // and reverse it
                                Collections.reverse(tempCoords);
                        }
                       
                        joinWay.getPoints().addAll(insIdx, tempCoords);
                        joinWay.addWay(tempWay);
                }
                return joinable;
        }

        /**
         * Combine a list of way segments to a list of maximally joined ways
         *
         * @param segments
         *            a list of closed or unclosed ways
         * @return a list of closed ways
         */

        protected ArrayList<JoinedWay> joinWays(List<Way> segments) {
                // TODO check if the closed polygon is valid and implement a backtracking algorithm to get other combinations

                ArrayList<JoinedWay> joinedWays = new ArrayList<>();
                if (segments == null || segments.isEmpty()) {
                        return joinedWays;
                }

                // go through all segments and categorize them to closed and unclosed
                // list
                ArrayList<JoinedWay> unclosedWays = new ArrayList<>();
                for (Way orgSegment : segments) {
                        JoinedWay jw = new JoinedWay(orgSegment);
                        roleMap.put(jw.getId(), getRole(orgSegment));
                        if (orgSegment.isClosed()) {
                                if (!orgSegment.isComplete()) {
                                        // the way is closed in planet but some points are missing in this tile
                                        // we can close it artificially
                                        if (log.isDebugEnabled())
                                                log.debug("Close incomplete but closed polygon:",orgSegment);
                                        jw.closeWayArtificially();
                                }
                                assert  jw.hasIdenticalEndPoints() : "way is not closed";
                                joinedWays.add(jw);
                        } else {
                                unclosedWays.add(jw);
                        }
                }

                while (!unclosedWays.isEmpty()) {
                        JoinedWay joinWay = unclosedWays.remove(0);

                        // check if the current way is already closed or if it is the last
                        // way
                        if (joinWay.hasIdenticalEndPoints() || unclosedWays.isEmpty()) {
                                joinedWays.add(joinWay);
                                continue;
                        }

                        boolean joined = false;

                        // if we have a way that could be joined but which has a wrong role
                        // then store it here and check in the end if it's working
                        JoinedWay wrongRoleWay = null;
                        String joinRole = getRole(joinWay);

                        // go through all ways and check if there is a way that can be
                        // joined with it
                        // in this case join the two ways
                        // => add all points of tempWay to joinWay, remove tempWay and put
                        // joinWay to the beginning of the list
                        // (not optimal but understandable - can be optimized later)
                        for (JoinedWay tempWay : unclosedWays) {
                                if (tempWay.hasIdenticalEndPoints()) {
                                        continue;
                                }

                                String tempRole = getRole(tempWay);
                                // if a role is not 'inner' or 'outer' then it is used as
                                // universal
                                // check if the roles of the ways are matching
                                if ((!"outer".equals(joinRole) && !"inner".equals(joinRole))
                                                || (!"outer".equals(tempRole) && !"inner".equals(tempRole))
                                                || (joinRole != null && joinRole.equals(tempRole))) {
                                        // the roles are matching => try to join both ways
                                        joined = joinWays(joinWay, tempWay, false);
                                } else {
                                        // the roles are not matching => test if both ways would
                                        // join

                                        // as long as we don't have an alternative way with wrong
                                        // role
                                        // or if the alternative way is shorter then check if
                                        // the way with the wrong role could be joined
                                        if (wrongRoleWay == null || wrongRoleWay.getPoints().size() < tempWay.getPoints().size()
                                                        && joinWays(joinWay, tempWay, true)) {
                                                // save this way => maybe we will use it in the end
                                                // if we don't find any other way
                                                wrongRoleWay = tempWay;
                                        }
                                }

                                if (joined) {
                                        // we have joined the way
                                        unclosedWays.remove(tempWay);
                                        break;
                                }
                        }

                        if (!joined && wrongRoleWay != null) {

                                log.warn("Join ways with different roles. Multipolygon: "
                                                + toBrowseURL());
                                log.warn("Way1 Role:", getRole(joinWay));
                                logWayURLs(Level.WARNING, "-", joinWay);
                                log.warn("Way2 Role:", getRole(wrongRoleWay));
                                logWayURLs(Level.WARNING, "-", wrongRoleWay);

                                joined = joinWays(joinWay, wrongRoleWay, false);
                                if (joined) {
                                        // we have joined the way
                                        unclosedWays.remove(wrongRoleWay);
                                        break;
                                }
                        }

                        if (joined) {
                                if (joinWay.hasIdenticalEndPoints()) {
                                        // it's closed => don't process it again
                                        joinedWays.add(joinWay);
                                } else if (unclosedWays.isEmpty()) {
                                        // no more ways to join with
                                        // it's not closed but we cannot join it more
                                        joinedWays.add(joinWay);
                                } else {
                                        // it is not yet closed => process it once again
                                        unclosedWays.add(0, joinWay);
                                }
                        } else {
                                // it's not closed but we cannot join it more
                                joinedWays.add(joinWay);
                        }
                }

                return joinedWays;
        }

        /**
         * Try to close all unclosed ways in the given list of ways.
         *
         * @param wayList
         *            a list of ways
         * @param maxCloseDist max distance between ends for artificial close
         *
         */

        protected void closeWays(ArrayList<JoinedWay> wayList, double maxCloseDist) {
                for (JoinedWay way : wayList) {
                        if (way.hasIdenticalEndPoints() || way.getPoints().size() < 3) {
                                continue;
                        }
                        Coord p1 = way.getFirstPoint();
                        Coord p2 = way.getLastPoint();

                        if (!tileBounds.insideBoundary(p1) && !tileBounds.insideBoundary(p2)
                        // both points lie outside the bbox or on the bbox
                        // check if both points are on the same side of the bounding box
                                        && (p1.getLatitude() <= tileBounds.getMinLat() && p2.getLatitude() <= tileBounds.getMinLat())
                                        || (p1.getLatitude() >= tileBounds.getMaxLat() && p2.getLatitude() >= tileBounds.getMaxLat())
                                        || (p1.getLongitude() <= tileBounds.getMinLong() && p2.getLongitude() <= tileBounds.getMinLong())
                                        || (p1.getLongitude() >= tileBounds.getMaxLong() && p2.getLongitude() >= tileBounds.getMaxLong())) {
                                // they are on the same side outside of the bbox
                                // so just close them without worrying about if
                                // they intersect itself because the intersection also
                                // is outside the bbox
                                way.closeWayArtificially();
                                log.info("Endpoints of way", way, "are both outside the bbox. Closing it directly.");
                                continue;
                        }
                       
                        Line2D closingLine = new Line2D.Float(p1.getLongitude(), p1
                                        .getLatitude(), p2.getLongitude(), p2.getLatitude());

                        boolean intersects = false;
                        Coord lastPoint = null;
                        // don't use the first and the last point
                        // the closing line can intersect only in one point or complete.
                        // Both isn't interesting for this check
                        for (Coord thisPoint : way.getPoints().subList(1, way.getPoints().size() - 1)) {
                                if (lastPoint != null && closingLine.intersectsLine(lastPoint.getLongitude(), lastPoint.getLatitude(),
                                                thisPoint.getLongitude(), thisPoint.getLatitude())) {
                                        intersects = true;
                                        break;
                                }
                                lastPoint = thisPoint;
                        }

                        if (!intersects) {
                                // close the polygon
                                // the new way segment does not intersect the rest of the polygon
                                boolean doClose = true;
                                if (maxCloseDist > 0) {
                                        // calc the distance to close
                                        double closeDist = way.getFirstPoint().distance(way.getLastPoint());
                                        doClose = closeDist < maxCloseDist;
                                }
                                if (doClose) {
                                        if (log.isInfoEnabled()) {
                                                log.info("Closing way", way);
                                                log.info("from", way.getFirstPoint().toOSMURL());
                                                log.info("to", way.getLastPoint().toOSMURL());
                                        }
                                        // mark this ways as artificially closed
                                        way.closeWayArtificially();
                                }
                        }
                }
        }

       
        protected static class ConnectionData {
                public Coord c1;
                public Coord c2;
                public JoinedWay w1;
                public JoinedWay w2;
                // sometimes the connection of both points cannot be done directly but with an intermediate point
                public Coord imC;
                public double distance;
                public ConnectionData() {
                       
                }
        }
       
        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();
                                if (!tileBounds.insideBoundary(c1)) {
                                        log.debug("Point", c1, "of way", w.getId(), "outside bbox");
                                        outOfBboxPoints.put(c1, w);
                                }

                                if (!tileBounds.insideBoundary(c2)) {
                                        log.debug("Point", c2, "of way", w.getId(), "outside bbox");
                                        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);                                    
                                       
                                        if (lineCutsBbox(cd.c1, cd.c2)) {
                                                // Check if the way can be closed with one additional point
                                                // outside the bounding box.
                                                // The additional point is combination of the coords of both endpoints.
                                                // It works if the lines from the endpoints to the additional point does
                                                // not cut the bounding box.
                                                // This can be removed when the splitter guarantees to provide logical complete
                                                // multi-polygons.
                                                Coord edgePoint1 = new Coord(cd.c1.getLatitude(), cd.c2.getLongitude());
                                                Coord edgePoint2 = new Coord(cd.c2.getLatitude(), cd.c1.getLongitude());

                                                if (!lineCutsBbox(cd.c1, edgePoint1) && !lineCutsBbox(edgePoint1, cd.c2)) {
                                                        cd.imC = edgePoint1;
                                                } else if (!lineCutsBbox(cd.c1, edgePoint2) && !lineCutsBbox(edgePoint2, cd.c2)) {
                                                        cd.imC = edgePoint1;
                                                } else {
                                                        // both endpoints are on opposite sides of the bounding box
                                                        // automatically closing such points would create wrong polygons in most cases
                                                        continue;
                                                }
                                                cd.distance = cd.c1.distance(cd.imC) + cd.imC.distance(cd.c2);
                                        } else {
                                                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.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;
        }

       
        /**
         * Removes all non closed ways from the given list.
         * <code>!{@link Way#hasIdenticalEndPoints()}</code>)
         *
         * @param wayList
         *            list of ways
         */

        protected void removeUnclosedWays(ArrayList<JoinedWay> wayList) {
                Iterator<JoinedWay> it = wayList.iterator();
                boolean firstWarn = true;
                while (it.hasNext()) {
                        JoinedWay tempWay = it.next();
                        if (!tempWay.hasIdenticalEndPoints()) {
                                // warn only if the way intersects the bounding box
                                boolean inBbox = tempWay.intersects(tileBounds);
                                if (inBbox) {
                                        if (firstWarn) {
                                                log.warn(
                                                        "Cannot join the following ways to closed polygons. Multipolygon",
                                                        toBrowseURL(), toTagString());
                                                firstWarn = false;
                                        }
                                        logWayURLs(Level.WARNING, "- way:", tempWay);
                                        logFakeWayDetails(Level.WARNING, tempWay);
                                }

                                it.remove();
                               
                                if (inBbox) {
                                        String role = getRole(tempWay);
                                        if (role == null || "".equals(role) || "outer".equals(role)) {
                                                // anyhow add the ways to the list for line tagging
                                                outerWaysForLineTagging.addAll(tempWay.getOriginalWays());
                                        }
                                }
                        }
                }
        }

        /**
         * Removes all ways that are completely outside the bounding box.
         * This reduces error messages from problems on the tile bounds.
         * @param wayList list of ways
         */

        protected void removeWaysOutsideBbox(ArrayList<JoinedWay> wayList) {
                ListIterator<JoinedWay> wayIter = wayList.listIterator();
                while (wayIter.hasNext()) {
                        JoinedWay w = wayIter.next();
                        boolean remove = true;
                        // check all points
                        for (Coord c : w.getPoints()) {
                                if (tileBounds.contains(c)) {
                                        // if one point is in the bounding box the way should not be removed
                                        remove = false;
                                        break;
                                }
                        }

                        // check if the polygon contains the complete bounding box
                        if (remove && w.getBounds().contains(tileArea.getBounds())) {
                                remove = false;
                        }
                       
                        if (remove) {
                                if (log.isDebugEnabled()) {
                                        log.debug("Remove way", w.getId(),
                                                "because it is completely outside the bounding box.");
                                }
                                wayIter.remove();
                        }
                }
        }

        /**
         * Find all polygons that are not contained by any other polygon.
         *
         * @param candidates
         *            all polygons that should be checked
         * @param roleFilter
         *            an additional filter
         * @return all polygon indexes that are not contained by any other polygon
         */

        private BitSet findOutmostPolygons(BitSet candidates, BitSet roleFilter) {
                BitSet realCandidates = ((BitSet) candidates.clone());
                realCandidates.and(roleFilter);
                return findOutmostPolygons(realCandidates);
        }

        /**
         * Finds all polygons that are not contained by any other polygons and that match
         * to the given role. All polygons with index given by <var>candidates</var>
         * are used.
         *
         * @param candidates
         *            indexes of the polygons that should be used
         * @return the bits of all outermost polygons are set to true
         */

        protected BitSet findOutmostPolygons(BitSet candidates) {
                BitSet outmostPolygons = new BitSet();

                // go through all candidates and check if they are contained by any
                // other candidate
                for (int candidateIndex = candidates.nextSetBit(0); candidateIndex >= 0; candidateIndex = candidates
                                .nextSetBit(candidateIndex + 1)) {
                        // check if the candidateIndex polygon is not contained by any
                        // other candidate polygon
                        boolean isOutmost = true;
                        for (int otherCandidateIndex = candidates.nextSetBit(0); otherCandidateIndex >= 0; otherCandidateIndex = candidates
                                        .nextSetBit(otherCandidateIndex + 1)) {
                                if (contains(otherCandidateIndex, candidateIndex)) {
                                        // candidateIndex is not an outermost polygon because it is
                                        // contained by the otherCandidateIndex polygon
                                        isOutmost = false;
                                        break;
                                }
                        }
                        if (isOutmost) {
                                // this is an outermost polygon
                                // put it to the bitset
                                outmostPolygons.set(candidateIndex);
                        }
                }

                return outmostPolygons;
        }

        protected ArrayList<PolygonStatus> getPolygonStatus(BitSet outmostPolygons,
                        String defaultRole) {
                ArrayList<PolygonStatus> polygonStatusList = new ArrayList<>();
                for (int polyIndex = outmostPolygons.nextSetBit(0); polyIndex >= 0; polyIndex = outmostPolygons
                                .nextSetBit(polyIndex + 1)) {
                        // polyIndex is the polygon that is not contained by any other
                        // polygon
                        JoinedWay polygon = polygons.get(polyIndex);
                        String role = getRole(polygon);
                        // if the role is not explicitly set use the default role
                        if (role == null || "".equals(role)) {
                                role = defaultRole;
                        }
                        polygonStatusList.add(new PolygonStatus("outer".equals(role), polyIndex, polygon));
                }
                // sort by role and then by number of points, this improves performance
                // in the routines which add the polygons to areas
                if (polygonStatusList.size() > 2) {
                        polygonStatusList.sort((o1, o2) -> {
                                if (o1.outer != o2.outer)
                                        return (o1.outer) ? -1 : 1;
                                return o1.polygon.getPoints().size() - o2.polygon.getPoints().size();
                        });
                }
                return polygonStatusList;
        }

        /**
         * Creates a list of all original ways of the multipolygon.
         * @return all source ways
         */

        protected List<Way> getSourceWays() {
                ArrayList<Way> allWays = new ArrayList<>();

                for (Map.Entry<String, Element> entry : getElements()) {
                        if (entry.getValue() instanceof Way) {
                                if (((Way) entry.getValue()).getPoints().isEmpty()) {
                                        log.warn("Way", entry.getValue(), "has no points and cannot be used for the multipolygon",
                                                        toBrowseURL());
                                } else {
                                        allWays.add((Way) entry.getValue());
                                }
                        } else if (!(entry.getValue() instanceof Node)
                                        || (!"admin_centre".equals(entry.getKey()) && !"label".equals(entry.getKey()))) {
                                log.warn("Non way member in role", entry.getKey(), entry.getValue().toBrowseURL(),
                                                "in multipolygon", toBrowseURL(), toTagString());
                        }
                }
                return allWays;
        }
       
       
        // unfinishedPolygons marks which polygons are not yet processed
        protected BitSet unfinishedPolygons;

        // create bitsets which polygons belong to the outer and to the inner role
        protected BitSet innerPolygons;
        protected BitSet taggedInnerPolygons;
        protected BitSet outerPolygons;
        protected BitSet taggedOuterPolygons;

        /**
         * 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();
               
                // check if it makes sense to process the mp
                if (!isMpProcessable(allWays)) {
                        log.info("Do not process multipolygon", getId(), "because it has no style relevant tags.");
                        return;
                }

               
                // join all single ways to polygons, try to close ways and remove non closed ways
                polygons = joinWays(allWays);
               
                outerWaysForLineTagging = new HashSet<>();
                outerTags = new HashMap<>();
               
                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()) {
                                log.info("Multipolygon", toBrowseURL(),
                                                hasPolygons ? "is completely outside the bounding box. It is not processed."
                                                                : "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) {
                        w.setFullArea(w.getFullArea()); // trigger setting area before start cutting...
                        // do like this to disguise function with side effects
                        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;
               
       
                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());
                        }
                       
                        // calculate the size of the polygon
                        double outerAreaSize = currentPolygon.polygon.getSizeOfArea();
                        if (outerAreaSize > largestSize) {
                                // subtract the holes
                                for (PolygonStatus hole : holes) {
                                        outerAreaSize -= hole.polygon.getSizeOfArea();
                                }
                                // is it still larger than the largest known polygon?
                                if (outerAreaSize > largestSize) {
                                        largestOuterPolygon = currentPolygon.polygon;
                                        largestSize = outerAreaSize;
                                }
                        }
                       
                        // check if the polygon is an outer polygon or
                        // if there are some holes
                        boolean processPolygon = currentPolygon.outer || (!holes.isEmpty());

                        if (processPolygon) {
                                List<Way> singularOuterPolygons;
                                if (holes.isEmpty()) {
                                        singularOuterPolygons = Collections
                                                        .singletonList((Way) new JoinedWay(currentPolygon.polygon));
                                } else {
                                        List<Way> innerWays = new ArrayList<>(holes.size());
                                        for (PolygonStatus polygonHoleStatus : holes) {
                                                innerWays.add(polygonHoleStatus.polygon);
                                        }

                                        MultiPolygonCutter cutter = new MultiPolygonCutter(this, tileArea);
                                        singularOuterPolygons = cutter.cutOutInnerPolygons(currentPolygon.polygon, innerWays);
                                }
                               
                                if (!singularOuterPolygons.isEmpty()) {
                                        // handle the tagging
                                        if (currentPolygon.outer && hasStyleRelevantTags(this)) {
                                                // use the tags of the multipolygon
                                                for (Way p : singularOuterPolygons) {
                                                        // overwrite all tags
                                                        p.copyTags(this);
                                                        p.deleteTag("type");
                                                }
                                                // remove the multipolygon tags in the original ways of the current polygon
                                                removeTagsInOrgWays(this, currentPolygon.polygon);
                                        } else {
                                                // use the tags of the original ways
                                                currentPolygon.polygon.mergeTagsFromOrgWays();
                                                for (Way p : singularOuterPolygons) {
                                                        // overwrite all tags
                                                        p.copyTags(currentPolygon.polygon);
                                                }
                                                // remove the current polygon tags in its original ways
                                                removeTagsInOrgWays(currentPolygon.polygon, currentPolygon.polygon);
                                        }
                               
                                        if (currentPolygon.outer && outmostPolygonProcessing) {
                                                // this is the outer most polygon - copy its tags. They will be used
                                                // later for tagging of the lines

                                                // all cut polygons have the same tags - copy them from the first polygon
                                                Way outerWay = singularOuterPolygons.get(0);
                                                for (Entry<String, String> tag : outerWay.getTagEntryIterator()) {
                                                        outerTags.put(tag.getKey(), tag.getValue());
                                                }
                                                outmostPolygonProcessing = false;
                                        }
                                       
                                        long fullArea = currentPolygon.polygon.getFullArea();
                                        for (Way mpWay : singularOuterPolygons) {
                                                // put the cut out polygons to the
                                                // final way map
                                                if (log.isDebugEnabled())
                                                        log.debug(mpWay.getId(), mpWay.toTagString());
                                       
                                                mpWay.setFullArea(fullArea);
                                                // mark this polygons so that only polygon style rules are applied
                                                mpWay.addTag(STYLE_FILTER_TAG, STYLE_FILTER_POLYGON);
                                                mpWay.addTag(TKM_MP_CREATED, "true");
                                               
                                                if (currentPolygon.outer) {
                                                        mpWay.addTag(TKM_MP_ROLE, "outer");
                                                        if (isAreaSizeCalculated())
                                                                mpAreaSize += calcAreaSize(mpWay.getPoints());
                                                } else {
                                                        mpWay.addTag(TKM_MP_ROLE, "inner");
                                                }
                                               
                                                getMpPolygons().put(mpWay.getId(), mpWay);
                                        }
                                }
                        }
                }
               
                if (log.isLoggable(Level.WARNING) && (outmostInnerPolygons.cardinality() + unfinishedPolygons.cardinality()
                                + nestedOuterPolygons.cardinality() + nestedInnerPolygons.cardinality() >= 1)) {
                        log.warn("Multipolygon", toBrowseURL(), toTagString(), "contains errors.");

                        BitSet outerUnusedPolys = new BitSet();
                        outerUnusedPolys.or(unfinishedPolygons);
                        outerUnusedPolys.or(outmostInnerPolygons);
                        outerUnusedPolys.or(nestedOuterPolygons);
                        outerUnusedPolys.or(nestedInnerPolygons);
                        outerUnusedPolys.or(unfinishedPolygons);
                        // use only the outer polygons
                        outerUnusedPolys.and(outerPolygons);
                        for (JoinedWay w : getWaysFromPolygonList(outerUnusedPolys)) {
                                outerWaysForLineTagging.addAll(w.getOriginalWays());
                        }
                       
                        runIntersectionCheck(unfinishedPolygons);
                        runOutmostInnerPolygonCheck(outmostInnerPolygons);
                        runNestedOuterPolygonCheck(nestedOuterPolygons);
                        runNestedInnerPolygonCheck(nestedInnerPolygons);
                        runWrongInnerPolygonCheck(unfinishedPolygons, innerPolygons);

                        // we have at least one polygon that could not be processed
                        // Probably we have intersecting or overlapping polygons
                        // one possible reason is if the relation overlaps the tile
                        // bounds
                        // => issue a warning
                        List<JoinedWay> lostWays = getWaysFromPolygonList(unfinishedPolygons);
                        for (JoinedWay w : lostWays) {
                                log.warn("Polygon", w, "is not processed due to an unknown reason.");
                                logWayURLs(Level.WARNING, "-", w);
                        }
                }

                if (!hasStyleRelevantTags(this)) {
                        // add tags to the multipolygon that are taken from the outer ways
                        // they may be required by some hooks (e.g. Area2POIHook)
                        for (Entry<String, String> tags : outerTags.entrySet()) {
                                addTag(tags.getKey(), tags.getValue());
                        }
                }
                String mpAreaSizeStr = null;
                if (isAreaSizeCalculated()) {
                        // calculate tag value for mkgmap:cache_area_size only once
                        mpAreaSizeStr = String.format(Locale.US, "%.3f", mpAreaSize);
                }
                // 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(getOriginalId(), orgOuterWay.getPoints());
                        lineTagWay.markAsGeneratedFrom(this);
                        lineTagWay.addTag(STYLE_FILTER_TAG, STYLE_FILTER_LINE);
                        lineTagWay.addTag(TKM_MP_CREATED, "true");
                        if (mpAreaSizeStr != null) {
                                // assign the area size of the whole multipolygon to all outer polygons
                                lineTagWay.addTag(TKM_CACHE_AREA_SIZEKEY, mpAreaSizeStr);
                        }
                        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 double getMaxCloseDist() {
                return -1; //
        }


        protected void postProcessing() {
               
                if (isAreaSizeCalculated()) {
                        // assign the area size of the whole multipolygon to all outer polygons
                        String mpAreaSizeStr = String.format(Locale.US, "%.3f", mpAreaSize);
                        addTag(TKM_CACHE_AREA_SIZEKEY, mpAreaSizeStr);
                        for (Way w : mpPolygons.values()) {
                                if ("outer".equals(w.getTag(TKM_MP_ROLE))) {
                                        w.addTag(TKM_CACHE_AREA_SIZEKEY, mpAreaSizeStr);
                                }
                        }
                }

                for (Way w : mpPolygons.values()) {
                        w.deleteTag("mkgmap:mp_role");
                }
                // copy all polygons created by the multipolygon algorithm to the global way map
                tileWayMap.putAll(mpPolygons);
               
                if (largestOuterPolygon != null) {
                        // check if the mp contains a node with role "label"
                        for (Map.Entry<String, Element> entry : getElements()) {
                                if (entry.getValue() instanceof Node && "label".equals(entry.getKey())) {
                                        // yes => use the label node as reference point
                                        cOfG = ((Node) entry.getValue()).getLocation();
                                        break;
                                }
                        }
                       
                        if (cOfG == null) {
                                // use the center of the largest polygon as reference point
                                cOfG = largestOuterPolygon.getCofG();
                        }
                }
        }
       
        private void runIntersectionCheck(BitSet unfinishedPolys) {
                if (intersectingPolygons.isEmpty()) {
                        // nothing to do
                        return;
                }

                log.warn("Some polygons are intersecting. This is not allowed in multipolygons.");

                boolean oneOufOfBbox = false;
                for (JoinedWay polygon : intersectingPolygons) {
                        int pi = polygons.indexOf(polygon);
                        unfinishedPolys.clear(pi);

                        boolean outOfBbox = false;
                        for (Coord c : polygon.getPoints()) {
                                if (!tileBounds.contains(c)) {
                                        outOfBbox = true;
                                        oneOufOfBbox = true;
                                        break;
                                }
                        }

                        logWayURLs(Level.WARNING, (outOfBbox ? "*" : "-"), polygon);
                }
               
                for (JoinedWay polygon : intersectingPolygons) {
                        // print out the details of the original ways
                        logFakeWayDetails(Level.WARNING, polygon);
                }
               
                if (oneOufOfBbox) {
                        log.warn("Some of these intersections/overlaps may be caused by incomplete data on bounding box edges (*).");
                }
        }

        private void runNestedOuterPolygonCheck(BitSet nestedOuterPolygons) {
                // just print out warnings
                // the check has been done before
                for (int wiIndex = nestedOuterPolygons.nextSetBit(0); wiIndex >= 0; wiIndex = nestedOuterPolygons
                                .nextSetBit(wiIndex + 1)) {
                        JoinedWay outerWay = polygons.get(wiIndex);
                        log.warn("Polygon",     outerWay, "carries role outer but lies inside an outer polygon. Potentially its role should be inner.");
                        logFakeWayDetails(Level.WARNING, outerWay);
                }
        }
       
        private void runNestedInnerPolygonCheck(BitSet nestedInnerPolygons) {
                // just print out warnings
                // the check has been done before
                for (int wiIndex = nestedInnerPolygons.nextSetBit(0); wiIndex >= 0; wiIndex = nestedInnerPolygons
                                .nextSetBit(wiIndex + 1)) {
                        JoinedWay innerWay = polygons.get(wiIndex);
                        log.warn("Polygon",     innerWay, "carries role", getRole(innerWay), "but lies inside an inner polygon. Potentially its role should be outer.");
                        logFakeWayDetails(Level.WARNING, innerWay);
                }
        }      
       
        private void runOutmostInnerPolygonCheck(BitSet outmostInnerPolygons) {
                // just print out warnings
                // the check has been done before
                for (int wiIndex = outmostInnerPolygons.nextSetBit(0); wiIndex >= 0; wiIndex = outmostInnerPolygons
                                .nextSetBit(wiIndex + 1)) {
                        JoinedWay innerWay = polygons.get(wiIndex);
                        log.warn("Polygon",     innerWay, "carries role", getRole(innerWay), "but is not inside any other polygon. Potentially it does not belong to this multipolygon.");
                        logFakeWayDetails(Level.WARNING, innerWay);
                }
        }

        private void runWrongInnerPolygonCheck(BitSet unfinishedPolygons, BitSet innerPolygons) {
                // find all unfinished inner polygons that are not contained by any
                BitSet wrongInnerPolygons = findOutmostPolygons(unfinishedPolygons, innerPolygons);
                if (log.isDebugEnabled()) {
                        log.debug("unfinished", unfinishedPolygons);
                        log.debug("inner", innerPolygons);
                        // other polygon
                        log.debug("wrong", wrongInnerPolygons);
                }
                if (!wrongInnerPolygons.isEmpty()) {
                        // we have an inner polygon that is not contained by any outer polygon
                        // check if
                        for (int wiIndex = wrongInnerPolygons.nextSetBit(0); wiIndex >= 0; wiIndex = wrongInnerPolygons
                                        .nextSetBit(wiIndex + 1)) {
                                BitSet containedPolygons = new BitSet();
                                containedPolygons.or(unfinishedPolygons);
                                containedPolygons.and(containsMatrix.get(wiIndex));

                                JoinedWay innerWay = polygons.get(wiIndex);
                                if (containedPolygons.isEmpty()) {
                                        log.warn("Polygon",     innerWay, "carries role", getRole(innerWay),
                                                "but is not inside any outer polygon. Potentially it does not belong to this multipolygon.");
                                        logFakeWayDetails(Level.WARNING, innerWay);
                                } else {
                                        log.warn("Polygon",     innerWay, "carries role", getRole(innerWay),
                                                "but is not inside any outer polygon. Potentially the roles are interchanged with the following",
                                                (containedPolygons.cardinality() > 1 ? "ways" : "way"), ".");

                                        for (int wrIndex = containedPolygons.nextSetBit(0); wrIndex >= 0; wrIndex = containedPolygons
                                                        .nextSetBit(wrIndex + 1)) {
                                                logWayURLs(Level.WARNING, "-", polygons.get(wrIndex));
                                                unfinishedPolygons.set(wrIndex);
                                                wrongInnerPolygons.set(wrIndex);
                                        }
                                        logFakeWayDetails(Level.WARNING, innerWay);
                                }

                                unfinishedPolygons.clear(wiIndex);
                                wrongInnerPolygons.clear(wiIndex);
                        }
                }
        }

        protected void cleanup() {
                mpPolygons = null;
                roleMap.clear();
                containsMatrix = null;
                polygons = null;
                tileArea = null;
                intersectingPolygons = null;
                outerWaysForLineTagging = null;
                outerTags = null;
               
                unfinishedPolygons = null;
                innerPolygons = null;
                taggedInnerPolygons = null;
                outerPolygons = null;
                taggedOuterPolygons = null;
               
                largestOuterPolygon = null;
        }

        /**
         * Retrieves if the given element contains tags that may be relevant
         * for style processing. If it has no relevant tag it will probably be
         * dropped by the style.
         *
         * @param element the OSM element
         * @return <code>true</code> has style relevant tags
         */

        protected boolean hasStyleRelevantTags(Element element) {
                if (element instanceof MultiPolygonRelation && ((MultiPolygonRelation) element).getTagsIncomplete()) {
                        return true;
                }
               
                for (Map.Entry<String, String> tagEntry : element.getTagEntryIterator()) {
                        String tagName = tagEntry.getKey();
                        // all tags are style relevant
                        // except: type (for relations), mkgmap:*
                        boolean isStyleRelevant = !(element instanceof Relation && "typ".equals(tagName))
                                        && !tagName.startsWith("mkgmap:");
                        if (isStyleRelevant) {
                                return true;
                        }
                }
                return false;
        }
       
        /**
         * Checks if this mp should be processed or if it is needless to process it
         * because there is no result.
         * @param ways the list of ways of the mp
         * @return <code>true</code> the mp processing will have a result;
         *                 <code>false</code> the mp processing will fail
         */

        private boolean isMpProcessable(Collection<Way> ways) {
                // Check if the multipolygon itself or the member ways have a
                // tag. If not it does not make sense to process the mp because
                // the output will not change anything
                if (hasStyleRelevantTags(this)) {
                        return true;
                }

                for (Way w : ways) {
                        if (hasStyleRelevantTags(w)) {
                                return true;
                        }
                }
                return false;
        }

        /**
         * Creates a matrix which polygon contains which polygon. A polygon does not
         * contain itself.
         *
         * @param polygonList
         *            a list of polygons
         */

        protected void createContainsMatrix(List<JoinedWay> polygonList) {
                containsMatrix = new ArrayList<>();
                for (int i = 0; i < polygonList.size(); i++) {
                        containsMatrix.add(new BitSet());
                }

                long t1 = System.currentTimeMillis();

                if (log.isDebugEnabled())
                        log.debug("createContainsMatrix listSize:", polygonList.size());

                // use this matrix to check which matrix element has been
                // calculated
                ArrayList<BitSet> finishedMatrix = new ArrayList<>(polygonList
                                .size());

                for (int i = 0; i < polygonList.size(); i++) {
                        BitSet matrixRow = new BitSet();
                        // a polygon does not contain itself
                        matrixRow.set(i);
                        finishedMatrix.add(matrixRow);
                }

                for (int rowIndex = 0; rowIndex < polygonList.size(); rowIndex++) {
                        JoinedWay potentialOuterPolygon = polygonList.get(rowIndex);
                        BitSet containsColumns = containsMatrix.get(rowIndex);
                        BitSet finishedCol = finishedMatrix.get(rowIndex);
                       
                        // the polygon need to be created only sometimes
                        // so use a lazy creation to improve performance
                        WayAndLazyPolygon lazyPotOuterPolygon = new WayAndLazyPolygon(potentialOuterPolygon);

                        // get all non calculated columns of the matrix
                        for (int colIndex = finishedCol.nextClearBit(0); colIndex >= 0
                                        && colIndex < polygonList.size(); colIndex = finishedCol
                                        .nextClearBit(colIndex + 1)) {

                                JoinedWay innerPolygon = polygonList.get(colIndex);

                                if (potentialOuterPolygon.getBounds().intersects(
                                                innerPolygon.getBounds()))
                                {
                                        boolean contains = contains(lazyPotOuterPolygon, innerPolygon);
                                       
                                        if (contains) {
                                                containsColumns.set(colIndex);

                                                // we also know that the inner polygon does not contain the
                                                // outer polygon
                                                // so we can set the finished bit for this matrix
                                                // element
                                                finishedMatrix.get(colIndex).set(rowIndex);

                                                // additionally we know that the outer polygon contains all
                                                // polygons that are contained by the inner polygon
                                                containsColumns.or(containsMatrix.get(colIndex));
                                                finishedCol.or(containsColumns);
                                        }
                                } else {
                                        // both polygons do not intersect
                                        // we can flag both matrix elements as finished
                                        finishedMatrix.get(colIndex).set(rowIndex);
                                        finishedMatrix.get(rowIndex).set(colIndex);
                                }
                                // this matrix element is calculated now
                                finishedCol.set(colIndex);
                        }
                }

                if (log.isDebugEnabled()) {
                        long t2 = System.currentTimeMillis();
                        log.debug("createMatrix for", polygonList.size(), "polygons took",
                                (t2 - t1), "ms");

                        log.debug("Containsmatrix:");
                        int i = 0;
                        boolean noContained = true;
                        for (BitSet b : containsMatrix) {
                                if (!b.isEmpty()) {
                                        log.debug(i,"contains",b);
                                        noContained = false;
                                }
                                i++;
                        }
                        if (noContained) {
                                log.debug("Matrix is empty");
                        }
                }
        }

       
        /**
         * This is a helper class that creates a high precision polygon for a way
         * on request only.
         */

        private static class WayAndLazyPolygon {
                private final JoinedWay way;
                private Polygon polygon;
               
                public WayAndLazyPolygon(JoinedWay way) {
                        this.way = way;
                }

                public final JoinedWay getWay() {
                        return this.way;
                }

                public final Polygon getPolygon() {
                        if (this.polygon == null) {
                                this.polygon = Java2DConverter.createHighPrecPolygon(this.way.getPoints());
                        }
                        return this.polygon;
                }
        }
       
        /**
         * Checks if the polygon with polygonIndex1 contains the polygon with polygonIndex2.
         *
         * @return true if polygon(polygonIndex1) contains polygon(polygonIndex2)
         */

        private boolean contains(int polygonIndex1, int polygonIndex2) {
                return containsMatrix.get(polygonIndex1).get(polygonIndex2);
        }

        /**
         * Checks if polygon1 contains polygon2.
         *
         * @param polygon1
         *            a closed way
         * @param polygon2
         *            a 2nd closed way
         * @return true if polygon1 contains polygon2
         */

        private boolean contains(WayAndLazyPolygon polygon1, JoinedWay polygon2) {
                if (!polygon1.getWay().hasIdenticalEndPoints()) {
                        return false;
                }
                // check if the bounds of polygon2 are completely inside/enclosed the bounds
                // of polygon1
                if (!polygon1.getWay().getBounds().contains(polygon2.getBounds())) {
                        return false;
                }

                // check first if one point of polygon2 is in polygon1

                // ignore intersections outside the bounding box
                // so it is necessary to check if there is at least one
                // point of polygon2 in polygon1 ignoring all points outside the bounding box
                boolean onePointContained = false;
                boolean allOnLine = true;
                for (Coord px : polygon2.getPoints()) {
                        if (polygon1.getPolygon().contains(px.getHighPrecLon(), px.getHighPrecLat())){
                                // there's one point that is in polygon1 and in the bounding
                                // box => polygon1 may contain polygon2
                                onePointContained = true;
                                if (!locatedOnLine(px, polygon1.getWay().getPoints())) {
                                        allOnLine = false;
                                        break;
                                }
                        } else if (tileBounds.contains(px) && !locatedOnLine(px, polygon1.getWay().getPoints())) {
                                // there's one point that is not in polygon1 but inside the
                                // bounding box => polygon1 does not contain polygon2
                                return false;
                        }
                }
               
                if (allOnLine) {
                        onePointContained = false;
                        // all points of polygon2 lie on lines of polygon1
                        // => the middle of each line polygon must NOT lie outside polygon1
                        ArrayList<Coord> middlePoints2 = new ArrayList<>(polygon2.getPoints().size());
                        Coord p1 = null;
                        for (Coord p2 : polygon2.getPoints()) {
                                if (p1 != null) {
                                        Coord pm = p1.makeBetweenPoint(p2, 0.5);
                                        middlePoints2.add(pm);
                                }
                                p1 = p2;
                        }
                       
                        for (Coord px : middlePoints2) {
                                if (polygon1.getPolygon().contains(px.getHighPrecLon(), px.getHighPrecLat())){
                                        // there's one point that is in polygon1 and in the bounding
                                        // box => polygon1 may contain polygon2
                                        onePointContained = true;
                                        break;
                                } else if (tileBounds.contains(px) && !locatedOnLine(px, polygon1.getWay().getPoints())) {
                                        // there's one point that is not in polygon1 but inside the
                                        // bounding box => polygon1 does not contain polygon2
                                        return false;
                                }
                        }                      
                }

                if (!onePointContained) {
                        // no point of polygon2 is in polygon1 => polygon1 does not contain polygon2
                        return false;
                }
               
                Iterator<Coord> it1 = polygon1.getWay().getPoints().iterator();
                Coord p11 = it1.next();

                while (it1.hasNext()) {
                        Coord p12 = p11;
                        p11 = it1.next();

                        if (!polygon2.linePossiblyIntersectsWay(p11, p12)) {
                                // don't check it - this segment of the outer polygon
                                // definitely does not intersect the way
                                continue;
                        }

                        int lonMin = Math.min(p11.getLongitude(), p12.getLongitude());
                        int lonMax = Math.max(p11.getLongitude(), p12.getLongitude());
                        int latMin = Math.min(p11.getLatitude(), p12.getLatitude());
                        int latMax = Math.max(p11.getLatitude(), p12.getLatitude());

                        // check all lines of way1 and way2 for intersections
                        Iterator<Coord> it2 = polygon2.getPoints().iterator();
                        Coord p21 = it2.next();

                        // for speedup we divide the area around the second line into
                        // a 3x3 matrix with lon(-1,0,1) and lat(-1,0,1).
                        // -1 means below min lon/lat of bbox line p1_1-p1_2
                        // 0 means inside the bounding box of the line p1_1-p1_2
                        // 1 means above max lon/lat of bbox line p1_1-p1_2
                        int lonField = p21.getLongitude() < lonMin ? -1 : p21.getLongitude() > lonMax ? 1 : 0;
                        int latField = p21.getLatitude() < latMin ? -1 : p21.getLatitude() > latMax ? 1 : 0;

                        int prevLonField = lonField;
                        int prevLatField = latField;

                        while (it2.hasNext()) {
                                Coord p22 = p21;
                                p21 = it2.next();

                                int changes = 0;
                                // check if the field of the 3x3 matrix has changed
                                if ((lonField >= 0 && p11.getLongitude() < lonMin)
                                                || (lonField <= 0 && p11.getLongitude() > lonMax)) {
                                        changes++;
                                        lonField = p11.getLongitude() < lonMin ? -1 : p11.getLongitude() > lonMax ? 1 : 0;
                                }
                                if ((latField >= 0 && p11.getLatitude() < latMin)
                                                || (latField <= 0 && p11.getLatitude() > latMax)) {
                                        changes++;
                                        latField = p11.getLatitude() < latMin ? -1 : p11.getLatitude() > latMax ? 1 : 0;
                                }

                                // an intersection is possible if
                                // latField and lonField has changed
                                // or if we come from or go to the inner matrix field
                                boolean intersectionPossible = (changes == 2)
                                                || (latField == 0 && lonField == 0)
                                                || (prevLatField == 0 && prevLonField == 0);

                                boolean intersects = intersectionPossible && linesCutEachOther(p11, p12, p21, p22);
                               
                                if (intersects) {
                                        if ((polygon1.getWay().isClosedArtificially() && !it1.hasNext())
                                                        || (polygon2.isClosedArtificially() && !it2.hasNext())) {
                                                // don't care about this intersection
                                                // one of the polygons is closed by this mp code and the
                                                // closing segment causes the intersection
                                                log.info("Polygon", polygon1, "may contain polygon", polygon2,
                                                        ". Ignoring artificial generated intersection.");
                                        } else if ((!tileBounds.contains(p11))
                                                        || (!tileBounds.contains(p12))
                                                        || (!tileBounds.contains(p21))
                                                        || (!tileBounds.contains(p22))) {
                                                // at least one point is outside the bounding box
                                                // we ignore the intersection because the ways may not
                                                // be complete
                                                // due to removals of the tile splitter or osmosis
                                                log.info("Polygon", polygon1, "may contain polygon", polygon2,
                                                        ". Ignoring because at least one point is outside the bounding box.");
                                        } else {
                                                // store them in the intersection polygons set
                                                // the error message will be printed out in the end of
                                                // the mp handling
                                                intersectingPolygons.add(polygon1.getWay());
                                                intersectingPolygons.add(polygon2);
                                                return false;
                                        }
                                }

                                prevLonField = lonField;
                                prevLatField = latField;
                        }
                }

                // don't have any intersection
                // => polygon1 contains polygon2
                return true;
        }

        /**
         * Checks if the point p is located on one line of the given points.
         * @param p a point
         * @param points a list of points; all consecutive points are handled as lines
         * @return true if p is located on one line given by points
         */

        private static boolean locatedOnLine(Coord p, List<Coord> points) {
                Coord cp1 = null;
                for (Coord cp2 : points) {
                        if (p.highPrecEquals(cp2)) {
                                return true;
                        }

                        try {
                                if (cp1 == null // first init
                                                || p.getHighPrecLon() < Math.min(cp1.getHighPrecLon(), cp2.getHighPrecLon())
                                                || p.getHighPrecLon() > Math.max(cp1.getHighPrecLon(), cp2.getHighPrecLon())
                                                || p.getHighPrecLat() < Math.min(cp1.getHighPrecLat(), cp2.getHighPrecLat())
                                                || p.getHighPrecLat() > Math.max(cp1.getHighPrecLat(), cp2.getHighPrecLat())) {
                                        continue;
                                }

                                double dist = Line2D.ptSegDistSq(cp1.getHighPrecLon(), cp1.getHighPrecLat(),
                                                cp2.getHighPrecLon(), cp2.getHighPrecLat(),
                                                p.getHighPrecLon(), p.getHighPrecLat());

                                if (dist <= OVERLAP_TOLERANCE_DISTANCE) {
                                        log.debug("Point", p, "is located on line between", cp1, "and",
                                                cp2, ". Distance:", dist);
                                        return true;
                                }
                        } finally {
                                cp1 = cp2;
                        }
                }
                return false;
        }

        private boolean lineCutsBbox(Coord p1, Coord p2) {
                Coord nw = new Coord(tileBounds.getMaxLat(), tileBounds.getMinLong());
                Coord sw = new Coord(tileBounds.getMinLat(), tileBounds.getMinLong());
                Coord se = new Coord(tileBounds.getMinLat(), tileBounds.getMaxLong());
                Coord ne = new Coord(tileBounds.getMaxLat(), tileBounds.getMaxLong());
                return linesCutEachOther(nw, sw, p1, p2)
                                || linesCutEachOther(sw, se, p1, p2)
                                || linesCutEachOther(se, ne, p1, p2)
                                || linesCutEachOther(ne, nw, p1, p2);
        }

        /**
         * XXX: This code presumes that certain tests were already done!
         * Check if the line p1_1 to p1_2 cuts line p2_1 to p2_2 in two pieces and vice versa.
         * This is a form of intersection check where it is allowed that one line ends on the
         * other line or that the two lines overlap.
         * @param p11 first point of line 1
         * @param p12 second point of line 1
         * @param p21 first point of line 2
         * @param p22 second point of line 2
         * @return true if both lines intersect somewhere in the middle of each other
         */

        private static boolean linesCutEachOther(Coord p11, Coord p12, Coord p21, Coord p22) {
                long width1 = (long) p12.getHighPrecLon() - p11.getHighPrecLon();
                long width2 = (long) p22.getHighPrecLon() - p21.getHighPrecLon();

                long height1 = (long) p12.getHighPrecLat() - p11.getHighPrecLat();
                long height2 = (long) p22.getHighPrecLat() - p21.getHighPrecLat();

                long denominator = ((height2 * width1) - (width2 * height1));
                if (denominator == 0) {
                        // the lines are parallel
                        // they might overlap but this is ok for this test
                        return false;
                }
               
                long x1Mx3 = (long) p11.getHighPrecLon() - p21.getHighPrecLon();
                long y1My3 = (long) p11.getHighPrecLat() - p21.getHighPrecLat();

                double isx = (double) ((width2 * y1My3) - (height2 * x1Mx3)) / denominator;
                if (isx <= 0 || isx >= 1) {
                        return false;
                }
               
                double isy = (double) ((width1 * y1My3) - (height1 * x1Mx3)) / denominator;

                return (isy > 0 &&  isy < 1);
        }

        private List<JoinedWay> getWaysFromPolygonList(BitSet selection) {
                if (selection.isEmpty()) {
                        return Collections.emptyList();
                }
                List<JoinedWay> wayList = new ArrayList<>(selection
                                .cardinality());
                for (int i = selection.nextSetBit(0); i >= 0; i = selection.nextSetBit(i + 1)) {
                        wayList.add(polygons.get(i));
                }
                return wayList;
        }

        private static void logWayURLs(Level level, String preMsg, Way way) {
                if (log.isLoggable(level)) {
                        if (way instanceof JoinedWay) {
                                if (((JoinedWay) way).getOriginalWays().isEmpty()) {
                                        log.warn("Way", way, "does not contain any original ways");
                                }
                                for (Way segment : ((JoinedWay) way).getOriginalWays()) {
                                        if (preMsg == null || preMsg.length() == 0) {
                                                log.log(level, segment.toBrowseURL());
                                        } else {
                                                log.log(level, preMsg, segment.toBrowseURL());
                                        }
                                }
                        } else {
                                if (preMsg == null || preMsg.length() == 0) {
                                        log.log(level, way.toBrowseURL());
                                } else {
                                        log.log(level, preMsg, way.toBrowseURL());
                                }
                        }
                }
        }
       
        /**
         * Logs the details of the original ways of a way with a fake id. This is
         * primarily necessary for the sea multipolygon because it consists of
         * faked ways only. In this case logging messages can be improved by the
         * start and end points of the faked ways.
         * @param logLevel the logging level
         * @param fakeWay a way composed by other ways with faked ids
         */

        private void logFakeWayDetails(Level logLevel, JoinedWay fakeWay) {
                if (!log.isLoggable(logLevel)) {
                        return;
                }
               
                // only log if this is an artificial multipolygon
                if (!FakeIdGenerator.isFakeId(getId())) {
                        return;
                }
               
                boolean containsOrgFakeWay = false;
                for (Way orgWay : fakeWay.getOriginalWays()) {
                        if (FakeIdGenerator.isFakeId(orgWay.getId())) {
                                containsOrgFakeWay = true;
                        }
                }
               
                if (!containsOrgFakeWay) {
                        return;
                }
               
                // the fakeWay consists only of other faked ways
                // there should be more information about these ways
                // so that it is possible to retrieve the original
                // OSM ways
                // => log the start and end points
               
                for (Way orgWay : fakeWay.getOriginalWays()) {
                        log.log(logLevel, " Way",orgWay.getId(),"is composed of other artificial ways. Details:");
                        log.log(logLevel, "  Start:",orgWay.getFirstPoint().toOSMURL());
                        if (orgWay.hasEqualEndPoints()) {
                                // the way is closed so start and end are equal - log the point in the middle of the way
                                int mid = orgWay.getPoints().size()/2;
                                log.log(logLevel, "  Mid:  ",orgWay.getPoints().get(mid).toOSMURL());
                        } else {
                                log.log(logLevel, "  End:  ",orgWay.getLastPoint().toOSMURL());
                        }
                }              
        }

        protected void tagOuterWays() {
                Map<String, String> tags;
                if (hasStyleRelevantTags(this)) {
                        tags = new HashMap<>();
                        for (Entry<String, String> relTag : getTagEntryIterator()) {
                                tags.put(relTag.getKey(), relTag.getValue());
                        }
                } else {
                        tags = JoinedWay.getMergedTags(outerWaysForLineTagging);
                }
               
               
                // 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(getOriginalId(), orgOuterWay.getPoints());
                        lineTagWay.markAsGeneratedFrom(this);
                        lineTagWay.addTag(STYLE_FILTER_TAG, STYLE_FILTER_LINE);
                        lineTagWay.addTag(TKM_MP_CREATED, "true");
                        for (Entry<String, String> tag : tags.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);
                }
        }
       
       
        /**
         * Marks all tags of the original ways of the given JoinedWay that are also
         * contained in the given tagElement for removal.
         *
         * @param tagElement
         *            an element contains the tags to be removed
         * @param way
         *            a joined way
         */

        private void removeTagsInOrgWays(Element tagElement, JoinedWay way) {
                for (Entry<String, String> tag : tagElement.getTagEntryIterator()) {
                        removeTagInOrgWays(way, tag.getKey(), tag.getValue());
                }
        }

        /**
         * Mark the given tag of all original ways of the given JoinedWay.
         *
         * @param way
         *            a joined way
         * @param tagname
         *            the tag to be removed
         * @param tagvalue
         *            the value of the tag to be removed
         */

        private void removeTagInOrgWays(JoinedWay way, String tagname, String tagvalue) {
                for (Way w : way.getOriginalWays()) {
                        if (w instanceof JoinedWay) {
                                // remove the tags recursively
                                removeTagInOrgWays((JoinedWay) w, tagname, tagvalue);
                                continue;
                        }

                        if (tagvalue.equals(w.getTag(tagname))) {
                                if (log.isDebugEnabled()) {
                                        log.debug("Will remove", tagname + "=" + w.getTag(tagname), "from way", w.getId(), w.toTagString());
                                }
                                removeTagsInOrgWays(w, tagname);
                        }
                }
        }
       
        protected void removeTagsInOrgWays(Way way, String tag) {
                if (tag == null || tag.isEmpty()) {
                        return;
                }
                String tagsToRemove = way.getTag(ElementSaver.TKM_REMOVETAGS);
               
                if (tagsToRemove == null) {
                        tagsToRemove = tag;
                } else if (tag.equals(tagsToRemove)) {
                        return;
                } else {
                        String[] keys = tagsToRemove.split(";");
                        if (Arrays.asList(keys).contains(tag)) {
                                return;
                        }
                        tagsToRemove += ";" + tag;
                }
                way.addTag(ElementSaver.TKM_REMOVETAGS, tagsToRemove);
        }
       
        /**
         * Flag if the area size of the mp should be calculated and added as tag.
         * @return {@code true} area size should be calculated; {@code false} area size should not be calculated
         */

        protected boolean isAreaSizeCalculated() {
                return true;
        }

        protected Map<Long, Way> getTileWayMap() {
                return tileWayMap;
        }

        protected Map<Long, Way> getMpPolygons() {
                return mpPolygons;
        }

        protected uk.me.parabola.imgfmt.app.Area getTileBounds() {
                return tileBounds;
        }
       
        /**
         * Calculates a unitless number that gives a value for the size
         * of the area. The calculation does not correct to any earth
         * coordinate system. It uses the simple rectangular coordinate
         * system of garmin coordinates.
         *
         * @param polygon the points of the area
         * @return the size of the area (unitless)
         */

        public static double calcAreaSize(List<Coord> polygon) {
                if (polygon.size() < 4 || polygon.get(0) != polygon.get(polygon.size() - 1)) {
                        return 0; // line or not closed
                }
                long area = 0;
                Iterator<Coord> polyIter = polygon.iterator();
                Coord c2 = polyIter.next();
                while (polyIter.hasNext()) {
                        Coord c1 = c2;
                        c2 = polyIter.next();
                        area += (long) (c2.getHighPrecLon() + c1.getHighPrecLon())
                                        * (c1.getHighPrecLat() - c2.getHighPrecLat());
                }
                //  convert from high prec to value in map units
                double areaSize = (double) area / (2 * (1 << Coord.DELTA_SHIFT) * (1 << Coord.DELTA_SHIFT));  
                return Math.abs(areaSize);
        }


        /**
         * This is a helper class that gives access to the original
         * segments of a joined way.
         */

        public static final class JoinedWay extends Way {
                private final List<Way> originalWays;
                private boolean closedArtificially;

                private int minLat;
                private int maxLat;
                private int minLon;
                private int maxLon;
                private Rectangle bounds;

                public JoinedWay(Way originalWay) {
                        super(originalWay.getOriginalId(), originalWay.getPoints());
                        markAsGeneratedFrom(originalWay);
                        originalWays = new ArrayList<>();
                        addWay(originalWay);

                        // we have to initialize the min/max values
                        Coord c0 = originalWay.getFirstPoint();
                        minLat = maxLat = c0.getLatitude();
                        minLon = maxLon = c0.getLongitude();

                        updateBounds(originalWay.getPoints());
                }

                public void addPoint(int index, Coord point) {
                        getPoints().add(index, point);
                        updateBounds(point);
                }

                @Override
                public void addPoint(Coord point) {
                        super.addPoint(point);
                        updateBounds(point);
                }

                private void updateBounds(List<Coord> pointList) {
                        for (Coord c : pointList) {
                                updateBounds(c.getLatitude(), c.getLongitude());
                        }
                }

                private void updateBounds(JoinedWay other) {
                        updateBounds(other.minLat,other.minLon);
                        updateBounds(other.maxLat,other.maxLon);
                }

                private void updateBounds(int lat, int lon) {
                        if (lat < minLat) {
                                minLat = lat;
                                bounds = null;
                        } else if (lat > maxLat) {
                                maxLat = lat;
                                bounds = null;
                        }

                        if (lon < minLon) {
                                minLon = lon;
                                bounds = null;
                        } else if (lon > maxLon) {
                                maxLon = lon;
                                bounds = null;
                        }

                       
                }
                private void updateBounds(Coord point) {
                        updateBounds(point.getLatitude(), point.getLongitude());
                }
               
                /**
                 * Checks if this way intersects the given bounding box at least with
                 * one point.
                 *
                 * @param bbox
                 *            the bounding box
                 * @return <code>true</code> if this way intersects or touches the
                 *         bounding box; <code>false</code> else
                 */

                public boolean intersects(uk.me.parabola.imgfmt.app.Area bbox) {
                        return (maxLat >= bbox.getMinLat()
                                        && minLat <= bbox.getMaxLat()
                                        && maxLon >= bbox.getMinLong()
                                        && minLon <= bbox.getMaxLong());
                }

                public Rectangle getBounds() {
                        if (bounds == null) {
                                // note that we increase the rectangle by 1 because intersects
                                // checks
                                // only the interior
                                bounds = new Rectangle(minLon - 1, minLat - 1, maxLon - minLon
                                                + 2, maxLat - minLat + 2);
                        }

                        return bounds;
                }

                public boolean linePossiblyIntersectsWay(Coord p1, Coord p2) {
                        return getBounds().intersectsLine(p1.getLongitude(),
                                        p1.getLatitude(), p2.getLongitude(), p2.getLatitude());
                }

                public void addWay(Way way) {
                        if (way instanceof JoinedWay) {
                                for (Way w : ((JoinedWay) way).getOriginalWays()) {
                                        addWay(w);
                                }
                                updateBounds((JoinedWay) way);
                        } else {
                                if (log.isDebugEnabled()) {
                                        log.debug("Joined", this.getId(), "with", way.getId());
                                }
                                this.originalWays.add(way);
                        }
                }

                public void closeWayArtificially() {
                        addPoint(getPoints().get(0));
                        closedArtificially = true;
                }

                public boolean isClosedArtificially() {
                        return closedArtificially;
                }

                public static Map<String, String> getMergedTags(Collection<Way> ways) {
                        Map<String, String> mergedTags = new HashMap<>();
                        boolean first = true;
                        for (Way way : ways) {
                                if (first) {
                                        // the tags of the first way are copied completely
                                        for (Map.Entry<String, String> tag : way.getTagEntryIterator()) {
                                                mergedTags.put(tag.getKey(), tag.getValue());
                                        }
                                        first = false;
                                } else {
                                        // for all other ways all non matching tags are removed
                                        ArrayList<String> tagsToRemove = null;
                                        for (Map.Entry<String, String> tag : mergedTags.entrySet()) {
                                                String wayTagValue = way.getTag(tag.getKey());
                                                if (wayTagValue != null && !tag.getValue().equals(wayTagValue)) {
                                                        // the tags are different
                                                        if (tagsToRemove == null) {
                                                                tagsToRemove = new ArrayList<>();
                                                        }
                                                        tagsToRemove.add(tag.getKey());
                                                }
                                        }
                                        if (tagsToRemove != null) {
                                                for (String tag : tagsToRemove) {
                                                        mergedTags.remove(tag);
                                                }
                                        }
                                }
                        }
                        return mergedTags;
                }
               
                /**
                 * Tags this way with a merge of the tags of all original ways.
                 */

                public void mergeTagsFromOrgWays() {
                        if (log.isDebugEnabled()) {
                                log.debug("Way", getId(), "merge tags from", getOriginalWays().size(), "ways");
                        }
                        removeAllTags();
                       
                        Map<String, String> mergedTags = getMergedTags(getOriginalWays());
                        for (Entry<String, String> tag : mergedTags.entrySet()) {
                                addTag(tag.getKey(), tag.getValue());
                        }
                }

                public List<Way> getOriginalWays() {
                        return originalWays;
                }
               
                /**
                 * Retrieves a measurement of the area covered by this polygon. The
                 * returned value has no unit. It is just a rough comparable value
                 * because it uses a rectangular coordinate system without correction.
                 * @return size of the covered areas (0 if the way is not closed)
                 */

                public double getSizeOfArea() {
                        return MultiPolygonRelation.calcAreaSize(getPoints());
                }

                @Override
                public String toString() {
                        StringBuilder sb = new StringBuilder(200);
                        sb.append(getId());
                        sb.append("(");
                        sb.append(getPoints().size());
                        sb.append("P)(");
                        boolean first = true;
                        for (Way w : getOriginalWays()) {
                                if (first) {
                                        first = false;
                                } else {
                                        sb.append(",");
                                }
                                sb.append(w.getId());
                                sb.append("[");
                                sb.append(w.getPoints().size());
                                sb.append("P]");
                        }
                        sb.append(")");
                        return sb.toString();
                }
        }

        public static class PolygonStatus {
                public final boolean outer;
                public final int index;
                public final JoinedWay polygon;

                public PolygonStatus(boolean outer, int index, JoinedWay polygon) {
                        this.outer = outer;
                        this.index = index;
                        this.polygon = polygon;
                }
               
                public String toString() {
                        return polygon + "_" + outer;
                }
        }
}