package net.osmand.binary; import com.google.protobuf.CodedInputStream; import com.google.protobuf.CodedOutputStream; import com.google.protobuf.WireFormat; import net.osmand.Collator; import net.osmand.CollatorStringMatcher; import net.osmand.CollatorStringMatcher.StringMatcherMode; import net.osmand.Location; import net.osmand.OsmAndCollator; import net.osmand.PlatformUtil; import net.osmand.ResultMatcher; import net.osmand.StringMatcher; import net.osmand.binary.BinaryMapAddressReaderAdapter.AddressRegion; import net.osmand.binary.BinaryMapAddressReaderAdapter.CitiesBlock; import net.osmand.binary.BinaryMapPoiReaderAdapter.PoiRegion; import net.osmand.binary.BinaryMapRouteReaderAdapter.RouteRegion; import net.osmand.binary.BinaryMapRouteReaderAdapter.RouteSubregion; import net.osmand.binary.BinaryMapTransportReaderAdapter.TransportIndex; import net.osmand.binary.OsmandOdb.MapDataBlock; import net.osmand.binary.OsmandOdb.OsmAndMapIndex.MapDataBox; import net.osmand.binary.OsmandOdb.OsmAndMapIndex.MapEncodingRule; import net.osmand.binary.OsmandOdb.OsmAndMapIndex.MapRootLevel; import net.osmand.data.Amenity; import net.osmand.data.Building; import net.osmand.data.City; import net.osmand.data.LatLon; import net.osmand.data.MapObject; import net.osmand.data.Street; import net.osmand.data.TransportRoute; import net.osmand.data.TransportStop; import net.osmand.osm.MapPoiTypes; import net.osmand.osm.PoiCategory; import net.osmand.osm.edit.Way; import net.osmand.util.Algorithms; import net.osmand.util.MapUtils; import org.apache.commons.logging.Log; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.NodeList; import org.xml.sax.InputSource; import org.xml.sax.SAXException; import java.io.BufferedInputStream; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.io.RandomAccessFile; import java.io.Reader; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.HashMap; import java.util.Iterator; import java.util.LinkedList; import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.TreeMap; import java.util.regex.Matcher; import java.util.regex.Pattern; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import javax.xml.parsers.ParserConfigurationException; import gnu.trove.list.array.TIntArrayList; import gnu.trove.list.array.TLongArrayList; import gnu.trove.map.TIntObjectMap; import gnu.trove.map.hash.TIntObjectHashMap; import gnu.trove.map.hash.TLongObjectHashMap; import gnu.trove.set.hash.TIntHashSet; import gnu.trove.set.hash.TLongHashSet; public class BinaryMapIndexReader { public final static int TRANSPORT_STOP_ZOOM = 24; public static final int SHIFT_COORDINATES = 5; private final static Log log = PlatformUtil.getLog(BinaryMapIndexReader.class); public static boolean READ_STATS = false; public static final SearchPoiTypeFilter ACCEPT_ALL_POI_TYPE_FILTER = new SearchPoiTypeFilter() { @Override public boolean isEmpty() { return false; } @Override public boolean accept(PoiCategory type, String subcategory) { return true; } }; private final RandomAccessFile raf; protected final File file; /*private*/ int version; /*private*/ long dateCreated; // keep them immutable inside /*private*/ boolean basemap = false; /*private*/ List mapIndexes = new ArrayList(); /*private*/ List poiIndexes = new ArrayList(); /*private*/ List addressIndexes = new ArrayList(); /*private*/ List transportIndexes = new ArrayList(); /*private*/ List routingIndexes = new ArrayList(); /*private*/ List indexes = new ArrayList(); protected CodedInputStream codedIS; private final BinaryMapTransportReaderAdapter transportAdapter; private final BinaryMapPoiReaderAdapter poiAdapter; private final BinaryMapAddressReaderAdapter addressAdapter; private final BinaryMapRouteReaderAdapter routeAdapter; private static String BASEMAP_NAME = "basemap"; public BinaryMapIndexReader(final RandomAccessFile raf, File file) throws IOException { this.raf = raf; this.file = file; codedIS = CodedInputStream.newInstance(raf); codedIS.setSizeLimit(Integer.MAX_VALUE); // 2048 MB transportAdapter = new BinaryMapTransportReaderAdapter(this); addressAdapter = new BinaryMapAddressReaderAdapter(this); poiAdapter = new BinaryMapPoiReaderAdapter(this); routeAdapter = new BinaryMapRouteReaderAdapter(this); init(); } public BinaryMapIndexReader(final RandomAccessFile raf, File file, boolean init) throws IOException { this.raf = raf; this.file = file; codedIS = CodedInputStream.newInstance(raf); codedIS.setSizeLimit(Integer.MAX_VALUE); // 2048 MB transportAdapter = new BinaryMapTransportReaderAdapter(this); addressAdapter = new BinaryMapAddressReaderAdapter(this); poiAdapter = new BinaryMapPoiReaderAdapter(this); routeAdapter = new BinaryMapRouteReaderAdapter(this); if (init) { init(); } } public BinaryMapIndexReader(final RandomAccessFile raf, BinaryMapIndexReader referenceToSameFile) throws IOException { this.raf = raf; this.file = referenceToSameFile.file; codedIS = CodedInputStream.newInstance(raf); codedIS.setSizeLimit(Integer.MAX_VALUE); // 2048 MB version = referenceToSameFile.version; dateCreated = referenceToSameFile.dateCreated; transportAdapter = new BinaryMapTransportReaderAdapter(this); addressAdapter = new BinaryMapAddressReaderAdapter(this); poiAdapter = new BinaryMapPoiReaderAdapter(this); routeAdapter = new BinaryMapRouteReaderAdapter(this); mapIndexes = new ArrayList(referenceToSameFile.mapIndexes); poiIndexes = new ArrayList(referenceToSameFile.poiIndexes); addressIndexes = new ArrayList(referenceToSameFile.addressIndexes); transportIndexes = new ArrayList(referenceToSameFile.transportIndexes); routingIndexes = new ArrayList(referenceToSameFile.routingIndexes); indexes = new ArrayList(referenceToSameFile.indexes); basemap = referenceToSameFile.basemap; calculateCenterPointForRegions(); } public long getDateCreated() { return dateCreated; } private void init() throws IOException { boolean initCorrectly = false; while (true) { int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: if (!initCorrectly) { //throw new IOException("Corrupted file. It should be ended as it starts with version"); //$NON-NLS-1$ throw new IOException("Corrupt file, it should have ended as it starts with version: " + file.getName()); //$NON-NLS-1$ } return; case OsmandOdb.OsmAndStructure.VERSION_FIELD_NUMBER : version = codedIS.readUInt32(); break; case OsmandOdb.OsmAndStructure.DATECREATED_FIELD_NUMBER : dateCreated = codedIS.readInt64(); break; case OsmandOdb.OsmAndStructure.MAPINDEX_FIELD_NUMBER: MapIndex mapIndex = new MapIndex(); mapIndex.length = readInt(); mapIndex.filePointer = codedIS.getTotalBytesRead(); int oldLimit = codedIS.pushLimit(mapIndex.length); readMapIndex(mapIndex, false); basemap = basemap || mapIndex.isBaseMap(); codedIS.popLimit(oldLimit); codedIS.seek(mapIndex.filePointer + mapIndex.length); mapIndexes.add(mapIndex); indexes.add(mapIndex); break; case OsmandOdb.OsmAndStructure.ADDRESSINDEX_FIELD_NUMBER: AddressRegion region = new AddressRegion(); region.length = readInt(); region.filePointer = codedIS.getTotalBytesRead(); if(addressAdapter != null){ oldLimit = codedIS.pushLimit(region.length); addressAdapter.readAddressIndex(region); if(region.name != null){ addressIndexes.add(region); indexes.add(region); } codedIS.popLimit(oldLimit); } codedIS.seek(region.filePointer + region.length); break; case OsmandOdb.OsmAndStructure.TRANSPORTINDEX_FIELD_NUMBER: TransportIndex ind = new TransportIndex(); ind.length = readInt(); ind.filePointer = codedIS.getTotalBytesRead(); if (transportAdapter != null) { oldLimit = codedIS.pushLimit(ind.length); transportAdapter.readTransportIndex(ind); codedIS.popLimit(oldLimit); transportIndexes.add(ind); indexes.add(ind); } codedIS.seek(ind.filePointer + ind.length); break; case OsmandOdb.OsmAndStructure.ROUTINGINDEX_FIELD_NUMBER: RouteRegion routeReg = new RouteRegion(); routeReg.length = readInt(); routeReg.filePointer = codedIS.getTotalBytesRead(); if (routeAdapter != null) { oldLimit = codedIS.pushLimit(routeReg.length); routeAdapter.readRouteIndex(routeReg); codedIS.popLimit(oldLimit); routingIndexes.add(routeReg); indexes.add(routeReg); } codedIS.seek(routeReg.filePointer + routeReg.length); break; case OsmandOdb.OsmAndStructure.POIINDEX_FIELD_NUMBER: PoiRegion poiInd = new PoiRegion(); poiInd.length = readInt(); poiInd.filePointer = codedIS.getTotalBytesRead(); if (poiAdapter != null) { oldLimit = codedIS.pushLimit(poiInd.length); poiAdapter.readPoiIndex(poiInd, false); codedIS.popLimit(oldLimit); poiIndexes.add(poiInd); indexes.add(poiInd); } codedIS.seek(poiInd.filePointer + poiInd.length); break; case OsmandOdb.OsmAndStructure.VERSIONCONFIRM_FIELD_NUMBER : int cversion = codedIS.readUInt32(); calculateCenterPointForRegions(); initCorrectly = cversion == version; break; default: skipUnknownField(t); break; } } } private void calculateCenterPointForRegions() { for (AddressRegion reg : addressIndexes) { for (MapIndex map : mapIndexes) { if (Algorithms.objectEquals(reg.name, map.name)) { if (map.getRoots().size() > 0) { reg.calculatedCenter = map.getCenterLatLon(); break; } } } if(reg.calculatedCenter == null) { for (RouteRegion map : routingIndexes) { if (Algorithms.objectEquals(reg.name, map.name)) { reg.calculatedCenter = new LatLon(map.getTopLatitude() / 2 + map.getBottomLatitude() / 2, map.getLeftLongitude() / 2 + map.getRightLongitude() / 2); break; } } } } } public List getIndexes() { return indexes; } public List getMapIndexes() { return mapIndexes; } public List getRoutingIndexes() { return routingIndexes; } public boolean isBasemap() { return basemap; } public boolean containsMapData() { return mapIndexes.size() > 0; } public boolean containsPoiData() { return poiIndexes.size() > 0; } public boolean containsRouteData() { return routingIndexes.size() > 0; } public boolean containsRouteData(int left31x, int top31y, int right31x, int bottom31y, int zoom) { for (RouteRegion ri : routingIndexes) { List sr = ri.getSubregions(); for (RouteSubregion r : sr) { if (right31x >= r.left && left31x <= r.right && r.top <= bottom31y && r.bottom >= top31y) { return true; } } } return false; } public boolean containsPoiData(int left31x, int top31y, int right31x, int bottom31y) { for (PoiRegion index : poiIndexes) { if (right31x >= index.left31 && left31x <= index.right31 && index.top31 <= bottom31y && index.bottom31 >= top31y) { return true; } } return false; } public boolean containsAddressData(int left31x, int top31y, int right31x, int bottom31y) { for (AddressRegion index : addressIndexes) { if (right31x >= index.left31 && left31x <= index.right31 && index.top31 <= bottom31y && index.bottom31 >= top31y) { return true; } } return false; } public boolean containsMapData(int tile31x, int tile31y, int zoom) { for (MapIndex mapIndex : mapIndexes) { for (MapRoot root : mapIndex.getRoots()) { if (root.minZoom <= zoom && root.maxZoom >= zoom) { if (tile31x >= root.left && tile31x <= root.right && root.top <= tile31y && root.bottom >= tile31y) { return true; } } } } return false; } public boolean containsMapData(int left31x, int top31y, int right31x, int bottom31y, int zoom) { for (MapIndex mapIndex : mapIndexes) { for (MapRoot root : mapIndex.getRoots()) { if (root.minZoom <= zoom && root.maxZoom >= zoom) { if (right31x >= root.left && left31x <= root.right && root.top <= bottom31y && root.bottom >= top31y) { return true; } } } } return false; } public boolean containsAddressData() { return addressIndexes.size() > 0; } public boolean hasTransportData() { return transportIndexes.size() > 0; } public RandomAccessFile getRaf() { return raf; } public File getFile() { return file; } public String getCountryName() { List rg = getRegionNames(); if(rg.size() > 0) { return rg.get(0).split("_")[0]; } return ""; } public String getRegionName() { List rg = getRegionNames(); if (rg.size() == 0) { rg.add(file.getName()); } String ls = rg.get(0); if (ls.lastIndexOf('_') != -1) { if (ls.matches("([a-zA-Z-]+_)+([0-9]+_){2}[0-9]+\\.obf")) { Pattern osmDiffDateEnding = Pattern.compile("_([0-9]+_){2}[0-9]+\\.obf"); Matcher m = osmDiffDateEnding.matcher(ls); if (m.find()) { ls = ls.substring(0, m.start()); if (ls.lastIndexOf('_') != -1) { return ls.substring(0, ls.lastIndexOf('_')).replace('_', ' '); } else { return ls; } } } else { if (ls.contains(".")) { ls = ls.substring(0, ls.indexOf(".")); } if (ls.endsWith("_2")) { ls = ls.substring(0, ls.length() - "_2".length()); } return ls.substring(0, ls.lastIndexOf('_')).replace('_', ' '); } } return ls; } public int readByte() throws IOException { byte b = codedIS.readRawByte(); if (b < 0) { return b + 256; } else { return b; } } public final int readInt() throws IOException { int ch1 = readByte(); int ch2 = readByte(); int ch3 = readByte(); int ch4 = readByte(); return ((ch1 << 24) + (ch2 << 16) + (ch3 << 8) + ch4); } public int getVersion() { return version; } protected void skipUnknownField(int tag) throws IOException { int wireType = WireFormat.getTagWireType(tag); if (wireType == WireFormat.WIRETYPE_FIXED32_LENGTH_DELIMITED) { int length = readInt(); codedIS.skipRawBytes(length); } else { codedIS.skipField(tag); } } public TIntObjectHashMap getTransportRoutes(int[] filePointers) throws IOException { TIntObjectHashMap result = new TIntObjectHashMap(); loadTransportRoutes(filePointers, result); return result; } /** * Transport public methods */ public void loadTransportRoutes(int[] filePointers, TIntObjectHashMap result) throws IOException { Map groupPoints = new HashMap(); for (int filePointer : filePointers) { TransportIndex ind = getTransportIndex(filePointer); if (ind != null) { if (!groupPoints.containsKey(ind)) { groupPoints.put(ind, new TIntArrayList()); } groupPoints.get(ind).add(filePointer); } } Iterator> it = groupPoints.entrySet().iterator(); if (it.hasNext()) { Entry e = it.next(); TransportIndex ind = e.getKey(); TIntArrayList pointers = e.getValue(); pointers.sort(); TIntObjectHashMap stringTable = new TIntObjectHashMap(); List finishInit = new ArrayList(); for (int i = 0; i < pointers.size(); i++) { int filePointer = pointers.get(i); TransportRoute transportRoute = transportAdapter.getTransportRoute(filePointer, stringTable, false); result.put(filePointer, transportRoute); finishInit.add(transportRoute); } transportAdapter.initializeStringTable(ind, stringTable); for(TransportRoute transportRoute : finishInit ) { transportAdapter.initializeNames(false, transportRoute, stringTable); } } } public boolean transportStopBelongsTo(TransportStop s) { return getTransportIndex(s.getFileOffset()) != null; } public List getTransportIndexes() { return transportIndexes; } private TransportIndex getTransportIndex(int filePointer) { TransportIndex ind = null; for (TransportIndex i : transportIndexes) { if (i.filePointer <= filePointer && (filePointer - i.filePointer) < i.length) { ind = i; break; } } return ind; } public boolean containTransportData(double latitude, double longitude) { double x = MapUtils.getTileNumberX(TRANSPORT_STOP_ZOOM, longitude); double y = MapUtils.getTileNumberY(TRANSPORT_STOP_ZOOM, latitude); for (TransportIndex index : transportIndexes) { if (index.right >= x && index.left <= x && index.top <= y && index.bottom >= y) { return true; } } return false; } public boolean containTransportData(double topLatitude, double leftLongitude, double bottomLatitude, double rightLongitude) { double leftX = MapUtils.getTileNumberX(TRANSPORT_STOP_ZOOM, leftLongitude); double topY = MapUtils.getTileNumberY(TRANSPORT_STOP_ZOOM, topLatitude); double rightX = MapUtils.getTileNumberX(TRANSPORT_STOP_ZOOM, rightLongitude); double bottomY = MapUtils.getTileNumberY(TRANSPORT_STOP_ZOOM, bottomLatitude); for (TransportIndex index : transportIndexes) { if (index.right >= leftX && index.left <= rightX && index.top <= bottomY && index.bottom >= topY) { return true; } } return false; } public List searchTransportIndex(SearchRequest req) throws IOException { for (TransportIndex index : transportIndexes) { if (index.stopsFileLength == 0 || index.right < req.left || index.left > req.right || index.top > req.bottom || index.bottom < req.top) { continue; } if (req.stringTable != null) { req.stringTable.clear(); } codedIS.seek(index.stopsFileOffset); int oldLimit = codedIS.pushLimit(index.stopsFileLength); int offset = req.searchResults.size(); transportAdapter.searchTransportTreeBounds(0, 0, 0, 0, req); codedIS.popLimit(oldLimit); if (req.stringTable != null) { transportAdapter.initializeStringTable(index, req.stringTable); for (int i = offset; i < req.searchResults.size(); i++) { TransportStop st = req.searchResults.get(i); transportAdapter.initializeNames(req.stringTable, st); } } } if (req.numberOfVisitedObjects > 0) { log.debug("Search is done. Visit " + req.numberOfVisitedObjects + " objects. Read " + req.numberOfAcceptedObjects + " objects."); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$ log.debug("Read " + req.numberOfReadSubtrees + " subtrees. Go through " + req.numberOfAcceptedSubtrees + " subtrees."); //$NON-NLS-1$//$NON-NLS-2$//$NON-NLS-3$ } return req.getSearchResults(); } /** * Address public methods */ public List getRegionNames() { List names = new ArrayList(); for (AddressRegion r : addressIndexes) { names.add(r.name); } return names; } public LatLon getRegionCenter() { for (AddressRegion r : addressIndexes) { if (r.calculatedCenter != null) return r.calculatedCenter; } return null; } public List getCities(SearchRequest resultMatcher, int cityType) throws IOException { return getCities(resultMatcher, null, null, cityType); } public List getCities(SearchRequest resultMatcher, StringMatcher matcher, String lang, int cityType) throws IOException { List cities = new ArrayList(); for (AddressRegion r : addressIndexes) { for (CitiesBlock block : r.cities) { if (block.type == cityType) { codedIS.seek(block.filePointer); int old = codedIS.pushLimit(block.length); addressAdapter.readCities(cities, resultMatcher, matcher, r.attributeTagsTable); codedIS.popLimit(old); } } } return cities; } public List getCities(AddressRegion region, SearchRequest resultMatcher, int cityType) throws IOException { return getCities(region, resultMatcher, null, cityType); } public List getCities(AddressRegion region, SearchRequest resultMatcher, StringMatcher matcher, int cityType) throws IOException { List cities = new ArrayList(); for (CitiesBlock block : region.cities) { if (block.type == cityType) { codedIS.seek(block.filePointer); int old = codedIS.pushLimit(block.length); addressAdapter.readCities(cities, resultMatcher, matcher, region.attributeTagsTable); codedIS.popLimit(old); } } return cities; } public int preloadStreets(City c, SearchRequest resultMatcher) throws IOException { AddressRegion reg; try { reg = checkAddressIndex(c.getFileOffset()); } catch (IllegalArgumentException e) { throw new IOException(e.getMessage() + " while reading " + c + " (id: " + c.getId() + ")"); } codedIS.seek(c.getFileOffset()); int size = codedIS.readRawVarint32(); int old = codedIS.pushLimit(size); addressAdapter.readCityStreets(resultMatcher, c, reg.attributeTagsTable); codedIS.popLimit(old); return size; } private AddressRegion checkAddressIndex(int offset) { for (AddressRegion r : addressIndexes) { if (offset >= r.filePointer && offset <= (r.length + r.filePointer)) { return r; } } throw new IllegalArgumentException("Illegal offset " + offset); //$NON-NLS-1$ } public void preloadBuildings(Street s, SearchRequest resultMatcher) throws IOException { AddressRegion reg = checkAddressIndex(s.getFileOffset()); codedIS.seek(s.getFileOffset()); int size = codedIS.readRawVarint32(); int old = codedIS.pushLimit(size); City city = s.getCity(); addressAdapter.readStreet(s, resultMatcher, true, 0, 0, city != null && city.isPostcode() ? city.getName() : null, reg.attributeTagsTable); codedIS.popLimit(old); } /** * Map public methods */ private void readMapIndex(MapIndex index, boolean onlyInitEncodingRules) throws IOException { int defaultId = 1; int oldLimit; while (true) { int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: // encoding rules are required! if (onlyInitEncodingRules) { index.finishInitializingTags(); } return; case OsmandOdb.OsmAndMapIndex.NAME_FIELD_NUMBER : index.setName(codedIS.readString()); break; case OsmandOdb.OsmAndMapIndex.RULES_FIELD_NUMBER : if (onlyInitEncodingRules) { int len = codedIS.readInt32(); oldLimit = codedIS.pushLimit(len); readMapEncodingRule(index, defaultId++); codedIS.popLimit(oldLimit); } else { skipUnknownField(t); } break; case OsmandOdb.OsmAndMapIndex.LEVELS_FIELD_NUMBER : int length = readInt(); int filePointer = codedIS.getTotalBytesRead(); if (!onlyInitEncodingRules) { oldLimit = codedIS.pushLimit(length); MapRoot mapRoot = readMapLevel(new MapRoot()); mapRoot.length = length; mapRoot.filePointer = filePointer; index.getRoots().add(mapRoot); codedIS.popLimit(oldLimit); } codedIS.seek(filePointer + length); break; default: skipUnknownField(t); break; } } } private void readMapEncodingRule(MapIndex index, int id) throws IOException { int type = 0; String tags = null; String val = null; while (true) { int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: index.initMapEncodingRule(type, id, tags, val); return; case MapEncodingRule.VALUE_FIELD_NUMBER : val = codedIS.readString().intern(); break; case MapEncodingRule.TAG_FIELD_NUMBER : tags = codedIS.readString().intern(); break; case MapEncodingRule.TYPE_FIELD_NUMBER : type = codedIS.readUInt32(); break; case MapEncodingRule.ID_FIELD_NUMBER : id = codedIS.readUInt32(); break; default: skipUnknownField(t); break; } } } private MapRoot readMapLevel(MapRoot root) throws IOException { while (true) { int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: return root; case MapRootLevel.BOTTOM_FIELD_NUMBER : root.bottom = codedIS.readInt32(); break; case MapRootLevel.LEFT_FIELD_NUMBER : root.left = codedIS.readInt32(); break; case MapRootLevel.RIGHT_FIELD_NUMBER : root.right = codedIS.readInt32(); break; case MapRootLevel.TOP_FIELD_NUMBER : root.top = codedIS.readInt32(); break; case MapRootLevel.MAXZOOM_FIELD_NUMBER : root.maxZoom = codedIS.readInt32(); break; case MapRootLevel.MINZOOM_FIELD_NUMBER : root.minZoom = codedIS.readInt32(); break; case MapRootLevel.BOXES_FIELD_NUMBER : int length = readInt(); int filePointer = codedIS.getTotalBytesRead(); if (root.trees != null) { MapTree r = new MapTree(); // left, ... already initialized r.length = length; r.filePointer = filePointer; int oldLimit = codedIS.pushLimit(r.length); readMapTreeBounds(r, root.left, root.right, root.top, root.bottom); root.trees.add(r); codedIS.popLimit(oldLimit); } codedIS.seek(filePointer + length); break; case MapRootLevel.BLOCKS_FIELD_NUMBER : codedIS.skipRawBytes(codedIS.getBytesUntilLimit()); break; default: skipUnknownField(t); break; } } } private void readMapTreeBounds(MapTree tree, int aleft, int aright, int atop, int abottom) throws IOException { while (true) { int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: return; case MapDataBox.BOTTOM_FIELD_NUMBER : tree.bottom = codedIS.readSInt32() + abottom; break; case MapDataBox.LEFT_FIELD_NUMBER : tree.left = codedIS.readSInt32() + aleft; break; case MapDataBox.RIGHT_FIELD_NUMBER : tree.right = codedIS.readSInt32() + aright; break; case MapDataBox.TOP_FIELD_NUMBER : tree.top = codedIS.readSInt32() + atop; break; case MapDataBox.OCEAN_FIELD_NUMBER : if(codedIS.readBool()) { tree.ocean = Boolean.TRUE; } else { tree.ocean = Boolean.FALSE; } break; case MapDataBox.SHIFTTOMAPDATA_FIELD_NUMBER : tree.mapDataBlock = readInt() + tree.filePointer; break; default: skipUnknownField(t); break; } } } public List searchMapIndex(SearchRequest req) throws IOException { return searchMapIndex(req, null); } public List searchMapIndex(SearchRequest req, MapIndex filterMapIndex) throws IOException { req.numberOfVisitedObjects = 0; req.numberOfAcceptedObjects = 0; req.numberOfAcceptedSubtrees = 0; req.numberOfReadSubtrees = 0; List foundSubtrees = new ArrayList(); for (MapIndex mapIndex : mapIndexes) { if(filterMapIndex != null && mapIndex != filterMapIndex) { continue; } // lazy initializing rules if (mapIndex.encodingRules.isEmpty()) { codedIS.seek(mapIndex.filePointer); int oldLimit = codedIS.pushLimit(mapIndex.length); readMapIndex(mapIndex, true); codedIS.popLimit(oldLimit); } for (MapRoot index : mapIndex.getRoots()) { if (index.minZoom <= req.zoom && index.maxZoom >= req.zoom) { if (index.right < req.left || index.left > req.right || index.top > req.bottom || index.bottom < req.top) { continue; } // lazy initializing trees if (index.trees == null) { index.trees = new ArrayList(); codedIS.seek(index.filePointer); int oldLimit = codedIS.pushLimit(index.length); readMapLevel(index); codedIS.popLimit(oldLimit); } for (MapTree tree : index.trees) { if (tree.right < req.left || tree.left > req.right || tree.top > req.bottom || tree.bottom < req.top) { continue; } codedIS.seek(tree.filePointer); int oldLimit = codedIS.pushLimit(tree.length); searchMapTreeBounds(tree, index, req, foundSubtrees); codedIS.popLimit(oldLimit); } Collections.sort(foundSubtrees, new Comparator() { @Override public int compare(MapTree o1, MapTree o2) { return o1.mapDataBlock < o2.mapDataBlock ? -1 : (o1.mapDataBlock == o2.mapDataBlock ? 0 : 1); } }); for (MapTree tree : foundSubtrees) { if (!req.isCancelled()) { codedIS.seek(tree.mapDataBlock); int length = codedIS.readRawVarint32(); int oldLimit = codedIS.pushLimit(length); readMapDataBlocks(req, tree, mapIndex); codedIS.popLimit(oldLimit); } } foundSubtrees.clear(); } } } if (req.numberOfVisitedObjects > 0 && req.log) { log.info("Search is done. Visit " + req.numberOfVisitedObjects + " objects. Read " + req.numberOfAcceptedObjects + " objects."); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$ log.info("Read " + req.numberOfReadSubtrees + " subtrees. Go through " + req.numberOfAcceptedSubtrees + " subtrees."); //$NON-NLS-1$//$NON-NLS-2$//$NON-NLS-3$ } return req.getSearchResults(); } protected void readMapDataBlocks(SearchRequest req, MapTree tree, MapIndex root) throws IOException { List tempResults = null; long baseId = 0; while (true) { if (req.isCancelled()) { return; } int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: if(tempResults != null) { for(BinaryMapDataObject obj : tempResults) { req.publish(obj); } } return; case MapDataBlock.BASEID_FIELD_NUMBER: baseId = codedIS.readUInt64(); if(READ_STATS) { req.stat.addBlockHeader(MapDataBlock.BASEID_FIELD_NUMBER, 0); } break; case MapDataBlock.DATAOBJECTS_FIELD_NUMBER: int length = codedIS.readRawVarint32(); int oldLimit = codedIS.pushLimit(length); if(READ_STATS) { req.stat.lastObjectSize += length; req.stat.addBlockHeader(MapDataBlock.DATAOBJECTS_FIELD_NUMBER, length); } BinaryMapDataObject mapObject = readMapDataObject(tree, req, root); if (mapObject != null) { mapObject.setId(mapObject.getId() + baseId); if (READ_STATS) { req.publish(mapObject); } if (tempResults == null) { tempResults = new ArrayList(); } tempResults.add(mapObject); } codedIS.popLimit(oldLimit); break; case MapDataBlock.STRINGTABLE_FIELD_NUMBER: length = codedIS.readRawVarint32(); oldLimit = codedIS.pushLimit(length); if(READ_STATS) { req.stat.addBlockHeader(MapDataBlock.STRINGTABLE_FIELD_NUMBER, length); req.stat.lastBlockStringTableSize += length; } if (tempResults != null) { List stringTable = readStringTable(); for (int i = 0; i < tempResults.size(); i++) { BinaryMapDataObject rs = tempResults.get(i); if (rs.objectNames != null) { int[] keys = rs.objectNames.keys(); for (int j = 0; j < keys.length; j++) { rs.objectNames.put(keys[j], stringTable.get(rs.objectNames.get(keys[j]).charAt(0))); } } } } else { codedIS.skipRawBytes(codedIS.getBytesUntilLimit()); } codedIS.popLimit(oldLimit); break; default: skipUnknownField(t); break; } } } protected void searchMapTreeBounds(MapTree current, MapTree parent, SearchRequest req, List foundSubtrees) throws IOException { int init = 0; req.numberOfReadSubtrees++; while (true) { if (req.isCancelled()) { return; } int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); if (init == 0xf) { init = 0; // coordinates are init if (current.right < req.left || current.left > req.right || current.top > req.bottom || current.bottom < req.top) { return; } else { req.numberOfAcceptedSubtrees++; } } switch (tag) { case 0: return; case MapDataBox.BOTTOM_FIELD_NUMBER : current.bottom = codedIS.readSInt32() + parent.bottom; init |= 1; break; case MapDataBox.LEFT_FIELD_NUMBER : current.left = codedIS.readSInt32() + parent.left; init |= 2; break; case MapDataBox.RIGHT_FIELD_NUMBER : current.right = codedIS.readSInt32() + parent.right; init |= 4; break; case MapDataBox.TOP_FIELD_NUMBER : current.top = codedIS.readSInt32() + parent.top; init |= 8; break; case MapDataBox.SHIFTTOMAPDATA_FIELD_NUMBER : req.numberOfAcceptedSubtrees ++; current.mapDataBlock = readInt() + current.filePointer; foundSubtrees.add(current); break; case MapDataBox.OCEAN_FIELD_NUMBER : if(codedIS.readBool()) { current.ocean = Boolean.TRUE; } else { current.ocean = Boolean.FALSE; } req.publishOceanTile(current.ocean); break; case MapDataBox.BOXES_FIELD_NUMBER : // left, ... already initialized MapTree child = new MapTree(); child.length = readInt(); child.filePointer = codedIS.getTotalBytesRead(); int oldLimit = codedIS.pushLimit(child.length); if(current.ocean != null ){ child.ocean = current.ocean; } searchMapTreeBounds(child, current, req, foundSubtrees); codedIS.popLimit(oldLimit); codedIS.seek(child.filePointer + child.length); break; default: skipUnknownField(t); break; } } } private int MASK_TO_READ = ~((1 << SHIFT_COORDINATES) - 1); private BinaryMapDataObject readMapDataObject(MapTree tree , SearchRequest req, MapIndex root) throws IOException { int tag = WireFormat.getTagFieldNumber(codedIS.readTag()); boolean area = OsmandOdb.MapData.AREACOORDINATES_FIELD_NUMBER == tag; if (!area && OsmandOdb.MapData.COORDINATES_FIELD_NUMBER != tag) { throw new IllegalArgumentException(); } req.cacheCoordinates.clear(); int size = codedIS.readRawVarint32(); if (READ_STATS) { req.stat.lastObjectCoordinates += size; req.stat.addTagHeader(OsmandOdb.MapData.COORDINATES_FIELD_NUMBER, size); } int old = codedIS.pushLimit(size); int px = tree.left & MASK_TO_READ; int py = tree.top & MASK_TO_READ; boolean contains = false; int minX = Integer.MAX_VALUE; int maxX = 0; int minY = Integer.MAX_VALUE; int maxY = 0; req.numberOfVisitedObjects++; while (codedIS.getBytesUntilLimit() > 0) { int x = (codedIS.readSInt32() << SHIFT_COORDINATES) + px; int y = (codedIS.readSInt32() << SHIFT_COORDINATES) + py; req.cacheCoordinates.add(x); req.cacheCoordinates.add(y); px = x; py = y; if (!contains && req.left <= x && req.right >= x && req.top <= y && req.bottom >= y) { contains = true; } if (!contains) { minX = Math.min(minX, x); maxX = Math.max(maxX, x); minY = Math.min(minY, y); maxY = Math.max(maxY, y); } } if (!contains) { if (maxX >= req.left && minX <= req.right && minY <= req.bottom && maxY >= req.top) { contains = true; } } codedIS.popLimit(old); if (!contains) { codedIS.skipRawBytes(codedIS.getBytesUntilLimit()); return null; } // read List innercoordinates = null; TIntArrayList additionalTypes = null; TIntObjectHashMap stringNames = null; TIntArrayList stringOrder = null; long id = 0; boolean loop = true; while (loop) { int t = codedIS.readTag(); tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: loop = false; break; case OsmandOdb.MapData.POLYGONINNERCOORDINATES_FIELD_NUMBER: if (innercoordinates == null) { innercoordinates = new ArrayList(); } TIntArrayList polygon = new TIntArrayList(); innercoordinates.add(polygon); px = tree.left & MASK_TO_READ; py = tree.top & MASK_TO_READ; size = codedIS.readRawVarint32(); if(READ_STATS) { req.stat.lastObjectCoordinates += size; req.stat.addTagHeader(OsmandOdb.MapData.POLYGONINNERCOORDINATES_FIELD_NUMBER, size); } old = codedIS.pushLimit(size); while (codedIS.getBytesUntilLimit() > 0) { int x = (codedIS.readSInt32() << SHIFT_COORDINATES) + px; int y = (codedIS.readSInt32() << SHIFT_COORDINATES) + py; polygon.add(x); polygon.add(y); px = x; py = y; } codedIS.popLimit(old); break; case OsmandOdb.MapData.ADDITIONALTYPES_FIELD_NUMBER: additionalTypes = new TIntArrayList(); int sizeL = codedIS.readRawVarint32(); old = codedIS.pushLimit(sizeL); if(READ_STATS) { req.stat.lastObjectAdditionalTypes += sizeL; req.stat.addTagHeader(OsmandOdb.MapData.ADDITIONALTYPES_FIELD_NUMBER, sizeL); } while (codedIS.getBytesUntilLimit() > 0) { additionalTypes.add(codedIS.readRawVarint32()); } codedIS.popLimit(old); break; case OsmandOdb.MapData.TYPES_FIELD_NUMBER: req.cacheTypes.clear(); sizeL = codedIS.readRawVarint32(); old = codedIS.pushLimit(sizeL); if(READ_STATS) { req.stat.addTagHeader(OsmandOdb.MapData.TYPES_FIELD_NUMBER, sizeL); req.stat.lastObjectTypes += sizeL; } while (codedIS.getBytesUntilLimit() > 0) { req.cacheTypes.add(codedIS.readRawVarint32()); } codedIS.popLimit(old); boolean accept = true; if (req.searchFilter != null) { accept = req.searchFilter.accept(req.cacheTypes, root); } if (!accept) { codedIS.skipRawBytes(codedIS.getBytesUntilLimit()); return null; } req.numberOfAcceptedObjects++; break; case OsmandOdb.MapData.ID_FIELD_NUMBER: id = codedIS.readSInt64(); if(READ_STATS) { req.stat.addTagHeader(OsmandOdb.MapData.ID_FIELD_NUMBER, 0); req.stat.lastObjectHeaderInfo -= 1; req.stat.lastObjectIdSize += CodedOutputStream.computeSInt64SizeNoTag(id); } break; case OsmandOdb.MapData.STRINGNAMES_FIELD_NUMBER: stringNames = new TIntObjectHashMap(); stringOrder = new TIntArrayList(); sizeL = codedIS.readRawVarint32(); old = codedIS.pushLimit(sizeL); while (codedIS.getBytesUntilLimit() > 0) { int stag = codedIS.readRawVarint32(); int pId = codedIS.readRawVarint32(); stringNames.put(stag, ((char)pId)+""); stringOrder.add(stag); } codedIS.popLimit(old); if(READ_STATS) { req.stat.addTagHeader(OsmandOdb.MapData.STRINGNAMES_FIELD_NUMBER, sizeL); req.stat.lastStringNamesSize += sizeL; } break; default: skipUnknownField(t); break; } } BinaryMapDataObject dataObject = new BinaryMapDataObject(); dataObject.area = area; dataObject.coordinates = req.cacheCoordinates.toArray(); dataObject.objectNames = stringNames; dataObject.namesOrder = stringOrder; if (innercoordinates == null) { dataObject.polygonInnerCoordinates = new int[0][0]; } else { dataObject.polygonInnerCoordinates = new int[innercoordinates.size()][]; for (int i = 0; i < innercoordinates.size(); i++) { dataObject.polygonInnerCoordinates[i] = innercoordinates.get(i).toArray(); } } dataObject.types = req.cacheTypes.toArray(); if (additionalTypes != null) { dataObject.additionalTypes = additionalTypes.toArray(); } else { dataObject.additionalTypes = new int[0]; } dataObject.id = id; dataObject.area = area; dataObject.mapIndex = root; return dataObject; } public List searchAddressDataByName(SearchRequest req, List typeFilter) throws IOException { for (AddressRegion reg : addressIndexes) { if (reg.indexNameOffset != -1) { codedIS.seek(reg.indexNameOffset); int len = readInt(); int old = codedIS.pushLimit(len); addressAdapter.searchAddressDataByName(reg, req, typeFilter); codedIS.popLimit(old); } } return req.getSearchResults(); } public List searchAddressDataByName(SearchRequest req) throws IOException { return searchAddressDataByName(req, null); } public void initCategories(PoiRegion poiIndex) throws IOException { poiAdapter.initCategories(poiIndex); } public List searchPoiByName(SearchRequest req) throws IOException { if (req.nameQuery == null || req.nameQuery.length() == 0) { throw new IllegalArgumentException(); } for (PoiRegion poiIndex : poiIndexes) { poiAdapter.initCategories(poiIndex); codedIS.seek(poiIndex.filePointer); int old = codedIS.pushLimit(poiIndex.length); poiAdapter.searchPoiByName(poiIndex, req); codedIS.popLimit(old); } return req.getSearchResults(); } public Map> searchPoiCategoriesByName(String query, Map> map) throws IOException { if (query == null || query.length() == 0) { throw new IllegalArgumentException(); } Collator collator = OsmAndCollator.primaryCollator(); for (PoiRegion poiIndex : poiIndexes) { poiAdapter.initCategories(poiIndex); for (int i = 0; i < poiIndex.categories.size(); i++) { String cat = poiIndex.categories.get(i); PoiCategory catType = poiIndex.categoriesType.get(i); if (CollatorStringMatcher.cmatches(collator, cat, query, StringMatcherMode.CHECK_STARTS_FROM_SPACE)) { map.put(catType, null); } else { List subcats = poiIndex.subcategories.get(i); for (int j = 0; j < subcats.size(); j++) { if (CollatorStringMatcher.cmatches(collator, subcats.get(j), query, StringMatcherMode.CHECK_STARTS_FROM_SPACE)) { if (!map.containsKey(catType)) { map.put(catType, new ArrayList()); } List list = map.get(catType); if (list != null) { list.add(subcats.get(j)); } } } } } } return map; } public List searchPoi(SearchRequest req) throws IOException { req.numberOfVisitedObjects = 0; req.numberOfAcceptedObjects = 0; req.numberOfAcceptedSubtrees = 0; req.numberOfReadSubtrees = 0; for (PoiRegion poiIndex : poiIndexes) { poiAdapter.initCategories(poiIndex); codedIS.seek(poiIndex.filePointer); int old = codedIS.pushLimit(poiIndex.length); poiAdapter.searchPoiIndex(req.left, req.right, req.top, req.bottom, req, poiIndex); codedIS.popLimit(old); } log.info("Read " + req.numberOfReadSubtrees + " subtrees. Go through " + req.numberOfAcceptedSubtrees + " subtrees."); //$NON-NLS-1$//$NON-NLS-2$//$NON-NLS-3$ log.info("Search poi is done. Visit " + req.numberOfVisitedObjects + " objects. Read " + req.numberOfAcceptedObjects + " objects."); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$ return req.getSearchResults(); } public List searchPoi(PoiRegion poiIndex, SearchRequest req) throws IOException { req.numberOfVisitedObjects = 0; req.numberOfAcceptedObjects = 0; req.numberOfAcceptedSubtrees = 0; req.numberOfReadSubtrees = 0; poiAdapter.initCategories(poiIndex); codedIS.seek(poiIndex.filePointer); int old = codedIS.pushLimit(poiIndex.length); poiAdapter.searchPoiIndex(req.left, req.right, req.top, req.bottom, req, poiIndex); codedIS.popLimit(old); log.info("Search poi is done. Visit " + req.numberOfVisitedObjects + " objects. Read " + req.numberOfAcceptedObjects + " objects."); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$ log.info("Read " + req.numberOfReadSubtrees + " subtrees. Go through " + req.numberOfAcceptedSubtrees + " subtrees."); //$NON-NLS-1$//$NON-NLS-2$//$NON-NLS-3$ return req.getSearchResults(); } protected List readStringTable() throws IOException { List list = new ArrayList(); while (true) { int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: return list; case OsmandOdb.StringTable.S_FIELD_NUMBER : list.add(codedIS.readString()); break; default: skipUnknownField(t); break; } } } protected List getAddressIndexes() { return addressIndexes; } protected List getPoiIndexes() { return poiIndexes; } public static SearchRequest buildSearchRequest(int sleft, int sright, int stop, int sbottom, int zoom, SearchFilter searchFilter) { return buildSearchRequest(sleft, sright, stop, sbottom, zoom, searchFilter, null); } public static SearchRequest buildSearchRequest(int sleft, int sright, int stop, int sbottom, int zoom, SearchFilter searchFilter, ResultMatcher resultMatcher){ SearchRequest request = new SearchRequest(); request.left = sleft; request.right = sright; request.top = stop; request.bottom = sbottom; request.zoom = zoom; request.searchFilter = searchFilter; request.resultMatcher = resultMatcher; return request; } public static SearchRequest buildAddressRequest(ResultMatcher resultMatcher) { SearchRequest request = new SearchRequest(); request.resultMatcher = resultMatcher; return request; } public static SearchRequest buildAddressByNameRequest(ResultMatcher resultMatcher, String nameRequest, StringMatcherMode matcherMode) { SearchRequest request = new SearchRequest(); request.resultMatcher = resultMatcher; request.nameQuery = nameRequest.trim(); request.matcherMode = matcherMode; return request; } public static SearchRequest buildSearchPoiRequest(List route, double radius, SearchPoiTypeFilter poiTypeFilter, ResultMatcher resultMatcher) { SearchRequest request = new SearchRequest(); float coeff = (float) (radius / MapUtils.getTileDistanceWidth(SearchRequest.ZOOM_TO_SEARCH_POI)); TLongObjectHashMap> zooms = new TLongObjectHashMap>(); for (int i = 1; i < route.size(); i++) { Location cr = route.get(i); Location pr = route.get(i - 1); double tx = MapUtils.getTileNumberX(SearchRequest.ZOOM_TO_SEARCH_POI, cr.getLongitude()); double ty = MapUtils.getTileNumberY(SearchRequest.ZOOM_TO_SEARCH_POI, cr.getLatitude()); double px = MapUtils.getTileNumberX(SearchRequest.ZOOM_TO_SEARCH_POI, pr.getLongitude()); double py = MapUtils.getTileNumberY(SearchRequest.ZOOM_TO_SEARCH_POI, pr.getLatitude()); double topLeftX = Math.min(tx, px) - coeff; double topLeftY = Math.min(ty, py) - coeff; double bottomRightX = Math.max(tx, px) + coeff; double bottomRightY = Math.max(ty, py) + coeff; for (int x = (int) topLeftX; x <= bottomRightX; x++) { for (int y = (int) topLeftY; y <= bottomRightY; y++) { long hash = (((long) x) << SearchRequest.ZOOM_TO_SEARCH_POI) + y; if (!zooms.containsKey(hash)) { zooms.put(hash, new LinkedList()); } List ll = zooms.get(hash); ll.add(pr); ll.add(cr); } } } int sleft = Integer.MAX_VALUE, sright = 0, stop = Integer.MAX_VALUE, sbottom = 0; for (long vl : zooms.keys()) { long x = (vl >> SearchRequest.ZOOM_TO_SEARCH_POI) << (31 - SearchRequest.ZOOM_TO_SEARCH_POI); long y = (vl & ((1 << SearchRequest.ZOOM_TO_SEARCH_POI) - 1)) << (31 - SearchRequest.ZOOM_TO_SEARCH_POI); sleft = (int) Math.min(x, sleft); stop = (int) Math.min(y, stop); sbottom = (int) Math.max(y, sbottom); sright = (int) Math.max(x, sright); } request.radius = radius; request.left = sleft; request.zoom = -1; request.right = sright; request.top = stop; request.bottom = sbottom; request.tiles = zooms; request.poiTypeFilter = poiTypeFilter; request.resultMatcher = resultMatcher; return request; } public static SearchRequest buildSearchPoiRequest(int sleft, int sright, int stop, int sbottom, int zoom, SearchPoiTypeFilter poiTypeFilter, ResultMatcher matcher){ SearchRequest request = new SearchRequest(); request.left = sleft; request.right = sright; request.top = stop; request.bottom = sbottom; request.zoom = zoom; request.poiTypeFilter = poiTypeFilter; request.resultMatcher = matcher; return request; } public static SearchRequest buildSearchRouteRequest(int sleft, int sright, int stop, int sbottom, ResultMatcher matcher){ SearchRequest request = new SearchRequest(); request.left = sleft; request.right = sright; request.top = stop; request.bottom = sbottom; request.resultMatcher = matcher; return request; } public static SearchRequest buildSearchPoiRequest(int x, int y, String nameFilter, int sleft, int sright, int stop, int sbottom, ResultMatcher resultMatcher) { SearchRequest request = new SearchRequest(); request.x = x; request.y = y; request.left = sleft; request.right = sright; request.top = stop; request.bottom = sbottom; request.resultMatcher = resultMatcher; request.nameQuery = nameFilter.trim(); return request; } public static SearchRequest buildSearchTransportRequest(int sleft, int sright, int stop, int sbottom, int limit, List stops) { SearchRequest request = new SearchRequest(); if (stops != null) { request.searchResults = stops; } request.stringTable = new TIntObjectHashMap(); request.left = sleft >> (31 - TRANSPORT_STOP_ZOOM); request.right = sright >> (31 - TRANSPORT_STOP_ZOOM); request.top = stop >> (31 - TRANSPORT_STOP_ZOOM); request.bottom = sbottom >> (31 - TRANSPORT_STOP_ZOOM); request.limit = limit; return request; } public void close() throws IOException { if (codedIS != null) { raf.close(); codedIS = null; mapIndexes.clear(); addressIndexes.clear(); transportIndexes.clear(); } } public static interface SearchFilter { public boolean accept(TIntArrayList types, MapIndex index); } public static interface SearchPoiTypeFilter { public boolean accept(PoiCategory type, String subcategory); public boolean isEmpty(); } public static class MapObjectStat { public int lastStringNamesSize; public int lastObjectIdSize; public int lastObjectHeaderInfo; public int lastObjectAdditionalTypes; public int lastObjectTypes; public int lastObjectCoordinates; public int lastObjectSize; public int lastBlockStringTableSize; public int lastBlockHeaderInfo; public void addBlockHeader(int typesFieldNumber, int sizeL) { lastBlockHeaderInfo += CodedOutputStream.computeTagSize(typesFieldNumber) + CodedOutputStream.computeRawVarint32Size(sizeL); } public void addTagHeader(int typesFieldNumber, int sizeL) { lastObjectHeaderInfo += CodedOutputStream.computeTagSize(typesFieldNumber) + CodedOutputStream.computeRawVarint32Size(sizeL); } public void clearObjectStats() { lastStringNamesSize = 0; lastObjectIdSize = 0; lastObjectHeaderInfo = 0; lastObjectAdditionalTypes = 0; lastObjectTypes = 0; lastObjectCoordinates = 0; } } public static class SearchRequest { public final static int ZOOM_TO_SEARCH_POI = 16; private List searchResults = new ArrayList(); private boolean land = false; private boolean ocean = false; private ResultMatcher resultMatcher; // 31 zoom tiles // common variables int x = 0; int y = 0; int left = 0; int right = 0; int top = 0; int bottom = 0; int zoom = 15; int limit = -1; // search on the path // stores tile of 16 index and pairs (even length always) of points intersecting tile TLongObjectHashMap> tiles = null; double radius = -1; String nameQuery = null; StringMatcherMode matcherMode = StringMatcherMode.CHECK_STARTS_FROM_SPACE; SearchFilter searchFilter = null; SearchPoiTypeFilter poiTypeFilter = null; // internal read information TIntObjectHashMap stringTable = null; // cache information TIntArrayList cacheCoordinates = new TIntArrayList(); TIntArrayList cacheTypes = new TIntArrayList(); TLongArrayList cacheIdsA = new TLongArrayList(); TLongArrayList cacheIdsB = new TLongArrayList(); MapObjectStat stat = new MapObjectStat(); // TRACE INFO public boolean log = true; int numberOfVisitedObjects = 0; int numberOfAcceptedObjects = 0; int numberOfReadSubtrees = 0; int numberOfAcceptedSubtrees = 0; boolean interrupted = false; public MapObjectStat getStat() { return stat; } protected SearchRequest() { } public long getTileHashOnPath(double lat, double lon) { long x = (int) MapUtils.getTileNumberX(SearchRequest.ZOOM_TO_SEARCH_POI, lon); long y = (int) MapUtils.getTileNumberY(SearchRequest.ZOOM_TO_SEARCH_POI, lat); return (x << SearchRequest.ZOOM_TO_SEARCH_POI) | y; } public void setBBoxRadius(double lat, double lon, int radiusMeters) { double dx = MapUtils.getTileNumberX(16, lon); double half16t = MapUtils.getDistance(lat, MapUtils.getLongitudeFromTile(16, ((int) dx) + 0.5), lat, MapUtils.getLongitudeFromTile(16, (int) dx)); double cf31 = ((double) radiusMeters / (half16t * 2)) * (1 << 15); int y31 = MapUtils.get31TileNumberY(lat); int x31 = MapUtils.get31TileNumberX(lon); left = (int) (x31 - cf31); right = (int) (x31 + cf31); top = (int) (y31 - cf31); bottom = (int) (y31 + cf31); } public boolean publish(T obj) { if (resultMatcher == null || resultMatcher.publish(obj)) { searchResults.add(obj); return true; } return false; } protected void publishOceanTile(boolean ocean) { if (ocean) { this.ocean = true; } else { this.land = true; } } public List getSearchResults() { return searchResults; } public void setInterrupted(boolean interrupted) { this.interrupted = interrupted; } public boolean limitExceeded() { return limit != -1 && searchResults.size() > limit; } public boolean isCancelled() { if (this.interrupted) { return interrupted; } if (resultMatcher != null) { return resultMatcher.isCancelled(); } return false; } public boolean isOcean() { return ocean; } public boolean isLand() { return land; } public boolean intersects(int l, int t, int r, int b) { return r >= left && l <= right && t <= bottom && b >= top; } public boolean contains(int l, int t, int r, int b) { return r <= right && l >= left && b <= bottom && t >= top; } public int getLeft() { return left; } public int getRight() { return right; } public int getBottom() { return bottom; } public int getTop() { return top; } public int getZoom() { return zoom; } public void clearSearchResults() { // recreate whole list to allow GC collect old data searchResults = new ArrayList(); cacheCoordinates.clear(); cacheTypes.clear(); if(stringTable != null) { stringTable.clear(); } land = false; ocean = false; numberOfVisitedObjects = 0; numberOfAcceptedObjects = 0; numberOfReadSubtrees = 0; numberOfAcceptedSubtrees = 0; } public boolean isBboxSpecified() { return left != 0 || right != 0; } } public static class MapIndex extends BinaryIndexPart { List roots = new ArrayList(); Map> encodingRules = new HashMap>(); public TIntObjectMap decodingRules = new TIntObjectHashMap(); public int nameEncodingType = 0; public int nameEnEncodingType = -1; public int refEncodingType = -1; public int coastlineEncodingType = -1; public int coastlineBrokenEncodingType = -1; public int landEncodingType = -1; public int onewayAttribute = -1; public int onewayReverseAttribute = -1; public TIntHashSet positiveLayers = new TIntHashSet(2); public TIntHashSet negativeLayers = new TIntHashSet(2); // to speed up comparision private MapIndex referenceMapIndex; public Integer getRule(String t, String v) { Map m = encodingRules.get(t); if (m != null) { return m.get(v); } return null; } public LatLon getCenterLatLon() { if(roots.size() == 0) { return null; } MapRoot mapRoot = roots.get(roots.size() - 1); double cy = (MapUtils.get31LatitudeY(mapRoot.getBottom()) + MapUtils.get31LatitudeY(mapRoot.getTop())) / 2; double cx = (MapUtils.get31LongitudeX(mapRoot.getLeft()) + MapUtils.get31LongitudeX(mapRoot.getRight())) / 2; return new LatLon(cy, cx); } public List getRoots() { return roots; } public TagValuePair decodeType(int type) { return decodingRules.get(type); } public Integer getRule(TagValuePair tv) { Map m = encodingRules.get(tv.tag); if (m != null) { return m.get(tv.value); } return null; } public void finishInitializingTags() { int free = decodingRules.size(); coastlineBrokenEncodingType = free++; initMapEncodingRule(0, coastlineBrokenEncodingType, "natural", "coastline_broken"); if (landEncodingType == -1) { landEncodingType = free++; initMapEncodingRule(0, landEncodingType, "natural", "land"); } } public boolean isRegisteredRule(int id) { return decodingRules.containsKey(id); } public void initMapEncodingRule(int type, int id, String tag, String val) { if (!encodingRules.containsKey(tag)) { encodingRules.put(tag, new HashMap()); } encodingRules.get(tag).put(val, id); if (!decodingRules.containsKey(id)) { decodingRules.put(id, new TagValuePair(tag, val, type)); } if ("name".equals(tag)) { nameEncodingType = id; } else if ("natural".equals(tag) && "coastline".equals(val)) { coastlineEncodingType = id; } else if ("natural".equals(tag) && "land".equals(val)) { landEncodingType = id; } else if ("oneway".equals(tag) && "yes".equals(val)) { onewayAttribute = id; } else if ("oneway".equals(tag) && "-1".equals(val)) { onewayReverseAttribute = id; } else if ("ref".equals(tag)) { refEncodingType = id; } else if ("name:en".equals(tag)) { nameEnEncodingType = id; } else if ("tunnel".equals(tag)) { negativeLayers.add(id); } else if ("bridge".equals(tag)) { positiveLayers.add(id); } else if ("layer".equals(tag)) { if (val != null && !val.equals("0") && val.length() > 0) { if (val.startsWith("-")) { negativeLayers.add(id); } else { positiveLayers.add(id); } } } } public boolean isBaseMap() { return name != null && name.toLowerCase().contains(BASEMAP_NAME); } public String getPartName() { return "Map"; } public int getFieldNumber() { return OsmandOdb.OsmAndStructure.MAPINDEX_FIELD_NUMBER; } public BinaryMapDataObject adoptMapObject(BinaryMapDataObject o) { if(o.mapIndex == this || o.mapIndex == referenceMapIndex) { return o; } if(encodingRules.isEmpty()) { encodingRules.putAll(o.mapIndex.encodingRules); decodingRules.putAll(o.mapIndex.decodingRules); referenceMapIndex = o.mapIndex; return o; } TIntArrayList types = new TIntArrayList(); TIntArrayList additionalTypes = new TIntArrayList(); if (o.types != null) { for (int i = 0; i < o.types.length; i++) { TagValuePair tp = o.mapIndex.decodeType(o.types[i]); Integer r = getRule(tp); if(r != null) { types.add(r); } else { int nid = decodingRules.size() + 1; initMapEncodingRule(tp.additionalAttribute, nid, tp.tag, tp.value); types.add(nid); } } } if (o.additionalTypes != null) { for (int i = 0; i < o.additionalTypes.length; i++) { TagValuePair tp = o.mapIndex.decodeType(o.additionalTypes[i]); Integer r = getRule(tp); if(r != null) { additionalTypes.add(r); } else { int nid = decodingRules.size() + 1; initMapEncodingRule(tp.additionalAttribute, nid, tp.tag, tp.value); additionalTypes.add(nid); } } } BinaryMapDataObject bm = new BinaryMapDataObject(o.id, o.coordinates, o.polygonInnerCoordinates, o.objectType, o.area, types.toArray(), additionalTypes.isEmpty() ? null : additionalTypes.toArray()); if (o.namesOrder != null) { bm.objectNames = new TIntObjectHashMap<>(); bm.namesOrder = new TIntArrayList(); for (int i = 0; i < o.namesOrder.size(); i++) { int nameType = o.namesOrder.get(i); String name = o.objectNames.get(nameType); TagValuePair tp = o.mapIndex.decodeType(nameType); Integer r = getRule(tp); if(r != null) { bm.namesOrder.add(r); bm.objectNames.put(r, name); } else { int nid = decodingRules.size() + 1; initMapEncodingRule(tp.additionalAttribute, nid, tp.tag, tp.value); additionalTypes.add(nid); bm.objectNames.put(nid, name); } } } return bm; } } public static class TagValuePair { public String tag; public String value; public int additionalAttribute; public TagValuePair(String tag, String value, int additionalAttribute) { super(); this.tag = tag; this.value = value; this.additionalAttribute = additionalAttribute; } public boolean isAdditional() { return additionalAttribute % 2 == 1; } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + additionalAttribute; result = prime * result + ((tag == null) ? 0 : tag.hashCode()); result = prime * result + ((value == null) ? 0 : value.hashCode()); return result; } public String toSimpleString() { if (value == null) { return tag; } return tag + "-" + value; } @Override public String toString() { return "TagValuePair : " + tag + " - " + value; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; TagValuePair other = (TagValuePair) obj; if (additionalAttribute != other.additionalAttribute) return false; if (tag == null) { if (other.tag != null) return false; } else if (!tag.equals(other.tag)) return false; if (value == null) { if (other.value != null) return false; } else if (!value.equals(other.value)) return false; return true; } } public static class MapRoot extends MapTree { int minZoom = 0; int maxZoom = 0; public int getMinZoom() { return minZoom; } public int getMaxZoom() { return maxZoom; } private List trees = null; public MapZooms.MapZoomPair getMapZoom() { return new MapZooms.MapZoomPair(minZoom, maxZoom); } } private static class MapTree { int filePointer = 0; int length = 0; long mapDataBlock = 0; Boolean ocean = null; int left = 0; int right = 0; int top = 0; int bottom = 0; public int getLeft() { return left; } public int getRight() { return right; } public int getTop() { return top; } public int getBottom() { return bottom; } public int getLength() { return length; } public int getFilePointer() { return filePointer; } @Override public String toString() { return "Top Lat " + ((float) MapUtils.get31LatitudeY(top)) + " lon " + ((float) MapUtils.get31LongitudeX(left)) + " Bottom lat " + ((float) MapUtils.get31LatitudeY(bottom)) + " lon " + ((float) MapUtils.get31LongitudeX(right)); } } private static boolean testMapSearch = false; private static boolean testAddressSearch = false; private static boolean testAddressSearchName = false; private static boolean testAddressJustifySearch = false; private static boolean testPoiSearch = false; private static boolean testPoiSearchOnPath = false; private static boolean testTransportSearch = true; private static int sleft = MapUtils.get31TileNumberX(27.55079); private static int sright = MapUtils.get31TileNumberX(27.55317); private static int stop = MapUtils.get31TileNumberY(53.89378); private static int sbottom = MapUtils.get31TileNumberY(53.89276); private static int szoom = 15; private static void println(String s) { System.out.println(s); } public static void main(String[] args) throws IOException { File fl = new File(System.getProperty("maps") + "/Synthetic_test_rendering.obf"); fl = new File(System.getProperty("maps") + "/Belarus_europe_2.obf"); RandomAccessFile raf = new RandomAccessFile(fl, "r"); BinaryMapIndexReader reader = new BinaryMapIndexReader(raf, fl); println("VERSION " + reader.getVersion()); //$NON-NLS-1$ long time = System.currentTimeMillis(); if (testMapSearch) { testMapSearch(reader); } if (testAddressSearchName) { testAddressSearchByName(reader); } if (testAddressSearch) { testAddressSearch(reader); } if (testAddressJustifySearch) { testAddressJustifySearch(reader); } if (testTransportSearch) { testTransportSearch(reader); } if (testPoiSearch || testPoiSearchOnPath) { PoiRegion poiRegion = reader.getPoiIndexes().get(0); if (testPoiSearch) { testPoiSearch(reader, poiRegion); testPoiSearchByName(reader); } if (testPoiSearchOnPath) { testSearchOnthePath(reader); } } println("MEMORY " + (Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory())); //$NON-NLS-1$ println("Time " + (System.currentTimeMillis() - time)); //$NON-NLS-1$ } private static void testSearchOnthePath(BinaryMapIndexReader reader) throws IOException { float radius = 1000; final MapPoiTypes poiTypes = MapPoiTypes.getDefault(); long now = System.currentTimeMillis(); println("Searching poi on the path..."); final List locations = readGPX(new File( "/Users/victorshcherb/osmand/maps/2015-03-07_19-07_Sat.gpx")); SearchRequest req = buildSearchPoiRequest(locations, radius, new SearchPoiTypeFilter() { @Override public boolean accept(PoiCategory type, String subcategory) { if (type == poiTypes.getPoiCategoryByName("shop") && subcategory.contains("super")) { return true; } return false; } @Override public boolean isEmpty() { return false; } }, null); req.zoom = -1; List results = reader.searchPoi(req); int k = 0; println("Search done in " + (System.currentTimeMillis() - now) + " ms "); now = System.currentTimeMillis(); for (Amenity a : results) { final float dds = dist(a.getLocation(), locations); if (dds <= radius) { println("+ " + a.getType() + " " + a.getSubType() + " Dist " + dds + " (=" + (float) a.getRoutePoint().deviateDistance + ") " + a.getName() + " " + a.getLocation()); k++; } else { println(a.getType() + " " + a.getSubType() + " Dist " + dds + " " + a.getName() + " " + a.getLocation()); } } println("Filtered in " + (System.currentTimeMillis() - now) + "ms " + k + " of " + results.size()); } private static float dist(LatLon l, List locations) { float dist = Float.POSITIVE_INFINITY; for (int i = 1; i < locations.size(); i++) { dist = Math.min(dist, (float) MapUtils.getOrthogonalDistance(l.getLatitude(), l.getLongitude(), locations.get(i - 1).getLatitude(), locations.get(i - 1).getLongitude(), locations.get(i).getLatitude(), locations.get(i).getLongitude())); } return dist; } private static Reader getUTF8Reader(InputStream f) throws IOException { BufferedInputStream bis = new BufferedInputStream(f); assert bis.markSupported(); bis.mark(3); boolean reset = true; byte[] t = new byte[3]; bis.read(t); if (t[0] == ((byte) 0xef) && t[1] == ((byte) 0xbb) && t[2] == ((byte) 0xbf)) { reset = false; } if (reset) { bis.reset(); } return new InputStreamReader(bis, "UTF-8"); } private static List readGPX(File f) { List res = new ArrayList(); try { StringBuilder content = new StringBuilder(); BufferedReader reader = new BufferedReader(getUTF8Reader(new FileInputStream(f))); DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder dom = factory.newDocumentBuilder(); // { // String s = null; // boolean fist = true; // while ((s = reader.readLine()) != null) { // if (fist) { // fist = false; // } // content.append(s).append("\n"); // } // } // Document doc = dom.parse(new InputSource(new StringReader(content.toString()))); Document doc = dom.parse(new InputSource(reader)); NodeList list = doc.getElementsByTagName("trkpt"); Way w = new Way(-1); for (int i = 0; i < list.getLength(); i++) { Element item = (Element) list.item(i); try { double lon = Double.parseDouble(item.getAttribute("lon")); double lat = Double.parseDouble(item.getAttribute("lat")); final Location o = new Location(""); o.setLatitude(lat); o.setLongitude(lon); res.add(o); } catch (NumberFormatException e) { } } } catch (IOException e) { throw new RuntimeException(e); } catch (ParserConfigurationException e) { throw new RuntimeException(e); } catch (SAXException e) { throw new RuntimeException(e); } return res; } private static void testPoiSearchByName(BinaryMapIndexReader reader) throws IOException { println("Searching by name..."); SearchRequest req = buildSearchPoiRequest(0, 0, "Art", 0, Integer.MAX_VALUE, 0, Integer.MAX_VALUE, null); reader.searchPoiByName(req); for (Amenity a : req.getSearchResults()) { println(a.getType().getTranslation() + " " + a.getSubType() + " " + a.getName() + " " + a.getLocation()); } } private static void testPoiSearch(BinaryMapIndexReader reader, PoiRegion poiRegion) throws IOException { println(MapUtils.get31LongitudeX(poiRegion.left31) + " " + MapUtils.get31LongitudeX(poiRegion.right31) + " " +MapUtils.get31LatitudeY( poiRegion.bottom31 )+ " " + MapUtils.get31LatitudeY(poiRegion.top31)); for (int i = 0; i < poiRegion.categories.size(); i++) { println(poiRegion.categories.get(i)); println(" " + poiRegion.subcategories.get(i)); } SearchRequest req = buildSearchPoiRequest(sleft, sright, stop, sbottom, -1, ACCEPT_ALL_POI_TYPE_FILTER, null); List results = reader.searchPoi(req); for (Amenity a : results) { println(a.getType() + " " + a.getSubType() + " " + a.getName() + " " + a.getLocation()); } } private static void testTransportSearch(BinaryMapIndexReader reader) throws IOException { // test transport for (TransportIndex i : reader.transportIndexes) { println("Transport bounds : " + i.left + " " + i.right + " " + i.top + " " + i.bottom); } for (TransportStop s : reader.searchTransportIndex(buildSearchTransportRequest(sleft, sright, stop, sbottom, -1, null))) { println(s.getName()); TIntObjectHashMap routes = reader.getTransportRoutes(s.getReferencesToRoutes()); for (net.osmand.data.TransportRoute route : routes.valueCollection()) { println(" " + route.getRef() + " " + route.getName() + " " + route.getDistance() + " " + route.getAvgBothDistance()); StringBuilder b = new StringBuilder(); if(route.getForwardWays() == null) { continue; } for(Way w : route.getForwardWays()) { b.append(w.getNodes()).append(" "); } println(" forward ways: " + b.toString()); } } } private static void updateFrequence(Map street, String key) { if (!street.containsKey(key)) { street.put(key, 1); } else { street.put(key, street.get(key) + 1); } } int readIndexedStringTable(Collator instance, String query, String prefix, TIntArrayList list, int charMatches) throws IOException { String key = null; while (true) { int t = codedIS.readTag(); int tag = WireFormat.getTagFieldNumber(t); switch (tag) { case 0: return charMatches; case OsmandOdb.IndexedStringTable.KEY_FIELD_NUMBER : key = codedIS.readString(); if(prefix.length() > 0){ key = prefix + key; } // check query is part of key (the best matching) if(CollatorStringMatcher.cmatches(instance, key, query, StringMatcherMode.CHECK_ONLY_STARTS_WITH)){ if(query.length() >= charMatches){ if(query.length() > charMatches){ charMatches = query.length(); list.clear(); } } else { key = null; } // check key is part of query } else if (CollatorStringMatcher.cmatches(instance, query, key, StringMatcherMode.CHECK_ONLY_STARTS_WITH)) { if (key.length() >= charMatches) { if (key.length() > charMatches) { charMatches = key.length(); list.clear(); } } else { key = null; } } else { key = null; } break; case OsmandOdb.IndexedStringTable.VAL_FIELD_NUMBER : int val = readInt(); if (key != null) { list.add(val); } break; case OsmandOdb.IndexedStringTable.SUBTABLES_FIELD_NUMBER : int len = codedIS.readRawVarint32(); int oldLim = codedIS.pushLimit(len); if (key != null) { charMatches = readIndexedStringTable(instance, query, key, list, charMatches); } else { codedIS.skipRawBytes(codedIS.getBytesUntilLimit()); } codedIS.popLimit(oldLim); break; default: skipUnknownField(t); break; } } } private static void testAddressSearchByName(BinaryMapIndexReader reader) throws IOException { SearchRequest req = buildAddressByNameRequest(new ResultMatcher() { @Override public boolean publish(MapObject object) { if (object instanceof Street) { System.out.println(object + " " + ((Street) object).getCity()); } else { System.out.println(object + " " + object.getId()); } return false; } @Override public boolean isCancelled() { return false; } }, "Guy'", StringMatcherMode.CHECK_ONLY_STARTS_WITH); // req.setBBoxRadius(52.276142, 4.8608723, 15000); reader.searchAddressDataByName(req); } /** * @param reader * @throws IOException */ /** * @param reader * @throws IOException */ private static void testAddressJustifySearch(BinaryMapIndexReader reader) throws IOException { final String streetName = "Logger"; final double lat = 52.28212d; final double lon = 4.86269d; // test address index search final List streetsList = new ArrayList(); SearchRequest req = buildAddressByNameRequest(new ResultMatcher() { @Override public boolean publish(MapObject object) { if (object instanceof Street && object.getName().equalsIgnoreCase(streetName)) { if (MapUtils.getDistance(object.getLocation(), lat, lon) < 20000) { streetsList.add((Street) object); return true; } return false; } return false; } @Override public boolean isCancelled() { return false; } }, streetName, StringMatcherMode.CHECK_EQUALS_FROM_SPACE); reader.searchAddressDataByName(req); TreeMap resMap = new TreeMap(new Comparator() { @Override public int compare(MapObject o1, MapObject o2) { LatLon l1 = o1.getLocation(); LatLon l2 = o2.getLocation(); if (l1 == null || l2 == null) { return l2 == l1 ? 0 : (l1 == null ? -1 : 1); } return Double.compare(MapUtils.getDistance(l1, lat, lon), MapUtils.getDistance(l2, lat, lon)); } }); for (Street s : streetsList) { resMap.put(s, s); reader.preloadBuildings(s, null); for (Building b : s.getBuildings()) { if (MapUtils.getDistance(b.getLocation(), lat, lon) < 100) { resMap.put(b, s); } } } for (MapObject e : resMap.keySet()) { Street s = resMap.get(e); if (e instanceof Building && MapUtils.getDistance(e.getLocation(), lat, lon) < 40) { Building b = (Building) e; System.out.println(b.getName() + " " + s); } else if (e instanceof Street) { System.out.println(s + " " + ((Street) s).getCity()); } } } private static void testAddressSearch(BinaryMapIndexReader reader) throws IOException { // test address index search final Map streetFreq = new HashMap(); List cs = reader.getCities(null, BinaryMapAddressReaderAdapter.CITY_TOWN_TYPE); for (City c : cs) { int buildings = 0; reader.preloadStreets(c, null); for (Street s : c.getStreets()) { updateFrequence(streetFreq, s.getName()); reader.preloadBuildings(s, buildAddressRequest((ResultMatcher) null)); buildings += s.getBuildings().size(); println(s.getName() + " " + s.getName("ru")); } println(c.getName() + " " + c.getLocation() + " " + c.getStreets().size() + " " + buildings + " " + c.getEnName(true) + " " + c.getName("ru")); } // int[] count = new int[1]; List villages = reader.getCities(buildAddressRequest((ResultMatcher) null), BinaryMapAddressReaderAdapter.VILLAGES_TYPE); for (City v : villages) { reader.preloadStreets(v, null); for (Street s : v.getStreets()) { updateFrequence(streetFreq, s.getName()); } } System.out.println("Villages " + villages.size()); List sorted = new ArrayList(streetFreq.keySet()); Collections.sort(sorted, new Comparator() { @Override public int compare(String o1, String o2) { return -streetFreq.get(o1) + streetFreq.get(o2); } }); System.out.println(streetFreq.size()); for (String s : sorted) { System.out.println(s + " " + streetFreq.get(s)); if (streetFreq.get(s) < 10) { break; } } } private static void testMapSearch(BinaryMapIndexReader reader) throws IOException { println(reader.mapIndexes.get(0).encodingRules + ""); println("SEARCH " + sleft + " " + sright + " " + stop + " " + sbottom); reader.searchMapIndex(buildSearchRequest(sleft, sright, stop, sbottom, szoom, null, new ResultMatcher() { @Override public boolean publish(BinaryMapDataObject obj) { StringBuilder b = new StringBuilder(); b.append(obj.area ? "Area" : (obj.getPointsLength() > 1 ? "Way" : "Point")); int[] types = obj.getTypes(); b.append(" types ["); for (int j = 0; j < types.length; j++) { if (j > 0) { b.append(", "); } TagValuePair pair = obj.getMapIndex().decodeType(types[j]); if (pair == null) { throw new NullPointerException("Type " + types[j] + "was not found"); } b.append(pair.toSimpleString()).append("(").append(types[j]).append(")"); } b.append("]"); if (obj.getAdditionalTypes() != null && obj.getAdditionalTypes().length > 0) { b.append(" add_types ["); for (int j = 0; j < obj.getAdditionalTypes().length; j++) { if (j > 0) { b.append(", "); } TagValuePair pair = obj.getMapIndex().decodeType(obj.getAdditionalTypes()[j]); if (pair == null) { throw new NullPointerException("Type " + obj.getAdditionalTypes()[j] + "was not found"); } b.append(pair.toSimpleString()).append("(").append(obj.getAdditionalTypes()[j]).append(")"); } b.append("]"); } TIntObjectHashMap names = obj.getObjectNames(); if (names != null && !names.isEmpty()) { b.append(" Names ["); int[] keys = names.keys(); for (int j = 0; j < keys.length; j++) { if (j > 0) { b.append(", "); } TagValuePair pair = obj.getMapIndex().decodeType(keys[j]); if (pair == null) { throw new NullPointerException("Type " + keys[j] + "was not found"); } b.append(pair.toSimpleString()).append("(").append(keys[j]).append(")"); b.append(" - ").append(names.get(keys[j])); } b.append("]"); } b.append(" id ").append((obj.getId() >> 1)); b.append(" lat/lon : "); for (int i = 0; i < obj.getPointsLength(); i++) { float x = (float) MapUtils.get31LongitudeX(obj.getPoint31XTile(i)); float y = (float) MapUtils.get31LatitudeY(obj.getPoint31YTile(i)); b.append(x).append(" / ").append(y).append(" , "); } println(b.toString()); return false; } @Override public boolean isCancelled() { return false; } })); } public List searchRouteIndexTree(SearchRequest req, List list) throws IOException { req.numberOfVisitedObjects = 0; req.numberOfAcceptedObjects = 0; req.numberOfAcceptedSubtrees = 0; req.numberOfReadSubtrees = 0; if (routeAdapter != null) { routeAdapter.initRouteTypesIfNeeded(req, list); return routeAdapter.searchRouteRegionTree(req, list, new ArrayList()); } return Collections.emptyList(); } public void loadRouteIndexData(List toLoad, ResultMatcher matcher) throws IOException { if (routeAdapter != null) { routeAdapter.loadRouteRegionData(toLoad, matcher); } } public List loadRouteIndexData(RouteSubregion rs) throws IOException { if (routeAdapter != null) { return routeAdapter.loadRouteRegionData(rs); } return Collections.emptyList(); } public void initRouteRegion(RouteRegion routeReg) throws IOException { if (routeAdapter != null) { routeAdapter.initRouteRegion(routeReg); } } }