2013-04-18 23:35:02 +02:00
|
|
|
package net.osmand.binary;
|
|
|
|
|
2017-08-19 11:18:49 +02:00
|
|
|
import gnu.trove.list.array.TIntArrayList;
|
|
|
|
import gnu.trove.map.hash.TIntObjectHashMap;
|
|
|
|
|
|
|
|
import java.util.Arrays;
|
2015-12-09 19:37:26 +01:00
|
|
|
import java.util.LinkedHashMap;
|
|
|
|
import java.util.Map;
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
import net.osmand.binary.BinaryMapIndexReader.MapIndex;
|
2017-08-19 12:45:06 +02:00
|
|
|
import net.osmand.binary.BinaryMapIndexReader.TagValuePair;
|
2013-04-18 23:35:02 +02:00
|
|
|
import net.osmand.render.RenderingRulesStorage;
|
2017-08-23 16:28:18 +02:00
|
|
|
import net.osmand.util.Algorithms;
|
2013-04-18 23:35:02 +02:00
|
|
|
|
|
|
|
public class BinaryMapDataObject {
|
|
|
|
protected int[] coordinates = null;
|
|
|
|
protected int[][] polygonInnerCoordinates = null;
|
|
|
|
protected boolean area = false;
|
|
|
|
protected int[] types = null;
|
|
|
|
protected int[] additionalTypes = null;
|
|
|
|
protected int objectType = RenderingRulesStorage.POINT_RULES;
|
2020-01-28 18:48:43 +01:00
|
|
|
protected int labelX;
|
|
|
|
protected int labelY;
|
2013-04-18 23:35:02 +02:00
|
|
|
|
|
|
|
protected TIntObjectHashMap<String> objectNames = null;
|
2014-06-26 01:14:30 +02:00
|
|
|
protected TIntArrayList namesOrder = null;
|
2013-04-18 23:35:02 +02:00
|
|
|
protected long id = 0;
|
|
|
|
|
|
|
|
protected MapIndex mapIndex = null;
|
|
|
|
|
|
|
|
|
|
|
|
public BinaryMapDataObject(){
|
|
|
|
}
|
|
|
|
|
2017-08-19 12:45:06 +02:00
|
|
|
|
|
|
|
|
|
|
|
public BinaryMapDataObject(long id, int[] coordinates, int[][] polygonInnerCoordinates, int objectType, boolean area,
|
2020-01-28 18:48:43 +01:00
|
|
|
int[] types, int[] additionalTypes, int labelX, int labelY){
|
2013-04-18 23:35:02 +02:00
|
|
|
this.polygonInnerCoordinates = polygonInnerCoordinates;
|
|
|
|
this.coordinates = coordinates;
|
2017-08-19 12:45:06 +02:00
|
|
|
this.additionalTypes = additionalTypes;
|
2013-04-18 23:35:02 +02:00
|
|
|
this.types = types;
|
|
|
|
this.id = id;
|
2017-08-19 12:45:06 +02:00
|
|
|
this.objectType = objectType;
|
|
|
|
this.area = area;
|
2020-01-28 18:48:43 +01:00
|
|
|
this.labelX = labelX;
|
|
|
|
this.labelY = labelY;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
protected void setCoordinates(int[] coordinates) {
|
|
|
|
this.coordinates = coordinates;
|
|
|
|
}
|
|
|
|
|
2014-03-01 12:30:25 +01:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
public String getName(){
|
|
|
|
if(objectNames == null){
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
String name = objectNames.get(mapIndex.nameEncodingType);
|
|
|
|
if(name == null){
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public TIntObjectHashMap<String> getObjectNames() {
|
|
|
|
return objectNames;
|
|
|
|
}
|
|
|
|
|
2015-12-09 19:37:26 +01:00
|
|
|
public Map<Integer, String> getOrderedObjectNames() {
|
2017-08-23 14:55:27 +02:00
|
|
|
if (namesOrder == null) {
|
|
|
|
return null;
|
|
|
|
}
|
2015-12-09 19:37:26 +01:00
|
|
|
LinkedHashMap<Integer, String> lm = new LinkedHashMap<Integer, String> ();
|
|
|
|
for (int i = 0; i < namesOrder.size(); i++) {
|
|
|
|
int nm = namesOrder.get(i);
|
|
|
|
lm.put(nm, objectNames.get(nm));
|
|
|
|
}
|
|
|
|
return lm;
|
|
|
|
}
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
public void putObjectName(int type, String name){
|
|
|
|
if(objectNames == null){
|
|
|
|
objectNames = new TIntObjectHashMap<String>();
|
2014-06-26 01:14:30 +02:00
|
|
|
namesOrder = new TIntArrayList();
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
objectNames.put(type, name);
|
2014-06-26 01:14:30 +02:00
|
|
|
namesOrder.add(type);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public int[][] getPolygonInnerCoordinates() {
|
|
|
|
return polygonInnerCoordinates;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int[] getTypes(){
|
|
|
|
return types;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean containsType(int cachedType) {
|
|
|
|
if(cachedType != -1) {
|
|
|
|
for(int i=0; i<types.length; i++){
|
|
|
|
if(types[i] == cachedType) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean containsAdditionalType(int cachedType) {
|
|
|
|
if (cachedType != -1) {
|
|
|
|
for (int i = 0; i < additionalTypes.length; i++) {
|
|
|
|
if (additionalTypes[i] == cachedType) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getNameByType(int type) {
|
|
|
|
if(type != -1 && objectNames != null) {
|
|
|
|
return objectNames.get(type);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int[] getAdditionalTypes() {
|
|
|
|
return additionalTypes;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isArea() {
|
|
|
|
return area;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isCycle(){
|
|
|
|
if(coordinates == null || coordinates.length < 2) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return coordinates[0] == coordinates[coordinates.length - 2] &&
|
|
|
|
coordinates[1] == coordinates[coordinates.length - 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setArea(boolean area) {
|
|
|
|
this.area = area;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long getId() {
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void setId(long id) {
|
|
|
|
this.id = id;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void setTypes(int[] types) {
|
|
|
|
this.types = types;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public int getSimpleLayer(){
|
|
|
|
if(mapIndex != null) {
|
2018-03-30 18:34:11 +02:00
|
|
|
if (additionalTypes != null) {
|
|
|
|
for (int i = 0; i < additionalTypes.length; i++) {
|
|
|
|
if (mapIndex.positiveLayers.contains(additionalTypes[i])) {
|
|
|
|
return 1;
|
|
|
|
} else if (mapIndex.negativeLayers.contains(additionalTypes[i])) {
|
|
|
|
return -1;
|
|
|
|
}
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-06-26 01:14:30 +02:00
|
|
|
public TIntArrayList getNamesOrder() {
|
|
|
|
return namesOrder;
|
|
|
|
}
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
public MapIndex getMapIndex() {
|
|
|
|
return mapIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setMapIndex(MapIndex mapIndex) {
|
|
|
|
this.mapIndex = mapIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getPointsLength(){
|
|
|
|
if(coordinates == null){
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return coordinates.length / 2;
|
|
|
|
}
|
|
|
|
public int getPoint31YTile(int ind) {
|
|
|
|
return coordinates[2 * ind + 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getPoint31XTile(int ind) {
|
|
|
|
return coordinates[2 * ind];
|
|
|
|
}
|
|
|
|
|
2017-08-19 12:45:06 +02:00
|
|
|
|
2017-09-03 17:32:22 +02:00
|
|
|
public boolean compareBinary(BinaryMapDataObject thatObj, int coordinatesPrecision) {
|
2017-08-19 12:45:06 +02:00
|
|
|
if(this.objectType == thatObj.objectType
|
|
|
|
&& this.id == thatObj.id
|
2017-08-19 11:18:49 +02:00
|
|
|
&& this.area == thatObj.area
|
2017-09-03 17:32:22 +02:00
|
|
|
&& compareCoordinates(this.coordinates, thatObj.coordinates, coordinatesPrecision) ) {
|
2020-01-28 18:48:43 +01:00
|
|
|
if (mapIndex == null) {
|
2017-08-29 20:53:01 +02:00
|
|
|
throw new IllegalStateException("Illegal binary object: " + id);
|
|
|
|
}
|
2020-01-28 18:48:43 +01:00
|
|
|
if (thatObj.mapIndex == null) {
|
2017-08-29 20:53:01 +02:00
|
|
|
throw new IllegalStateException("Illegal binary object: " + thatObj.id);
|
|
|
|
}
|
2017-09-02 22:24:49 +02:00
|
|
|
|
2017-08-19 12:45:06 +02:00
|
|
|
boolean equals = true;
|
2020-01-28 18:48:43 +01:00
|
|
|
if (equals) {
|
|
|
|
if (polygonInnerCoordinates == null || thatObj.polygonInnerCoordinates == null) {
|
|
|
|
equals = polygonInnerCoordinates == thatObj.polygonInnerCoordinates;
|
|
|
|
} else if (polygonInnerCoordinates.length != thatObj.polygonInnerCoordinates.length) {
|
2017-09-02 22:24:49 +02:00
|
|
|
equals = false;
|
|
|
|
} else {
|
2020-01-28 18:48:43 +01:00
|
|
|
for (int i = 0; i < polygonInnerCoordinates.length && equals; i++) {
|
|
|
|
if (polygonInnerCoordinates[i] == null || thatObj.polygonInnerCoordinates[i] == null) {
|
|
|
|
equals = polygonInnerCoordinates[i] == thatObj.polygonInnerCoordinates[i];
|
|
|
|
} else if (polygonInnerCoordinates[i].length != thatObj.polygonInnerCoordinates[i].length) {
|
2017-09-02 22:24:49 +02:00
|
|
|
equals = false;
|
|
|
|
} else {
|
2020-01-28 18:48:43 +01:00
|
|
|
equals = compareCoordinates(polygonInnerCoordinates[i], thatObj.polygonInnerCoordinates[i],
|
|
|
|
coordinatesPrecision);
|
2017-09-02 22:24:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-28 18:48:43 +01:00
|
|
|
if (equals) {
|
|
|
|
if (types == null || thatObj.types == null) {
|
|
|
|
equals = types == thatObj.types;
|
|
|
|
} else if (types.length != thatObj.types.length) {
|
2017-08-19 12:45:06 +02:00
|
|
|
equals = false;
|
|
|
|
} else {
|
2020-01-28 18:48:43 +01:00
|
|
|
for (int i = 0; i < types.length && equals; i++) {
|
2017-08-19 12:45:06 +02:00
|
|
|
TagValuePair o = mapIndex.decodeType(types[i]);
|
|
|
|
TagValuePair s = thatObj.mapIndex.decodeType(thatObj.types[i]);
|
2017-09-02 22:24:49 +02:00
|
|
|
equals = o.equals(s) && equals;
|
2017-08-19 12:45:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-28 18:48:43 +01:00
|
|
|
if (equals) {
|
|
|
|
if (additionalTypes == null || thatObj.additionalTypes == null) {
|
|
|
|
equals = additionalTypes == thatObj.additionalTypes;
|
|
|
|
} else if (additionalTypes.length != thatObj.additionalTypes.length) {
|
2017-08-19 12:45:06 +02:00
|
|
|
equals = false;
|
|
|
|
} else {
|
2020-01-28 18:48:43 +01:00
|
|
|
for (int i = 0; i < additionalTypes.length && equals; i++) {
|
2017-08-19 12:45:06 +02:00
|
|
|
TagValuePair o = mapIndex.decodeType(additionalTypes[i]);
|
|
|
|
TagValuePair s = thatObj.mapIndex.decodeType(thatObj.additionalTypes[i]);
|
|
|
|
equals = o.equals(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-28 18:48:43 +01:00
|
|
|
if (equals) {
|
|
|
|
if (namesOrder == null || thatObj.namesOrder == null) {
|
|
|
|
equals = namesOrder == thatObj.namesOrder;
|
|
|
|
} else if (namesOrder.size() != thatObj.namesOrder.size()) {
|
2017-08-19 12:45:06 +02:00
|
|
|
equals = false;
|
|
|
|
} else {
|
2020-01-28 18:48:43 +01:00
|
|
|
for (int i = 0; i < namesOrder.size() && equals; i++) {
|
2017-08-19 12:45:06 +02:00
|
|
|
TagValuePair o = mapIndex.decodeType(namesOrder.get(i));
|
|
|
|
TagValuePair s = thatObj.mapIndex.decodeType(thatObj.namesOrder.get(i));
|
|
|
|
equals = o.equals(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-28 18:48:43 +01:00
|
|
|
if (equals) {
|
2017-08-23 14:55:27 +02:00
|
|
|
// here we know that name indexes are equal & it is enough to check the value sets
|
2020-01-28 18:48:43 +01:00
|
|
|
if (objectNames == null || thatObj.objectNames == null) {
|
|
|
|
equals = objectNames == thatObj.objectNames;
|
|
|
|
} else if (objectNames.size() != thatObj.objectNames.size()) {
|
2017-08-19 12:45:06 +02:00
|
|
|
equals = false;
|
|
|
|
} else {
|
2020-01-28 18:48:43 +01:00
|
|
|
for (int i = 0; i < namesOrder.size() && equals; i++) {
|
2017-08-23 16:28:18 +02:00
|
|
|
String o = objectNames.get(namesOrder.get(i));
|
|
|
|
String s = thatObj.objectNames.get(thatObj.namesOrder.get(i));
|
|
|
|
equals = Algorithms.objectEquals(o, s);
|
|
|
|
}
|
2017-08-19 12:45:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return equals;
|
2017-08-19 11:18:49 +02:00
|
|
|
}
|
2017-09-03 01:22:03 +02:00
|
|
|
// thatObj.mapIndex.decodeType(thatObj.types[0])
|
|
|
|
// mapIndex.decodeType(types[0]) id >>7
|
2017-08-19 11:18:49 +02:00
|
|
|
return false;
|
|
|
|
}
|
2017-08-19 12:45:06 +02:00
|
|
|
|
|
|
|
|
2017-09-03 17:21:28 +02:00
|
|
|
private static boolean compareCoordinates(int[] coordinates, int[] coordinates2, int precision) {
|
|
|
|
if(precision == 0) {
|
|
|
|
return Arrays.equals(coordinates, coordinates2);
|
|
|
|
}
|
|
|
|
TIntArrayList cd = simplify(coordinates, precision);
|
|
|
|
TIntArrayList cd2 = simplify(coordinates2, precision);
|
|
|
|
return cd.equals(cd2);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private static TIntArrayList simplify(int[] c, int precision) {
|
|
|
|
int len = c.length / 2;
|
|
|
|
TIntArrayList lt = new TIntArrayList(len * 3);
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
lt.add(0);
|
|
|
|
lt.add(c[i * 2]);
|
|
|
|
lt.add(c[i * 2 + 1]);
|
|
|
|
}
|
|
|
|
lt.set(0, 1);
|
|
|
|
lt.set((len - 1) * 3, 1);
|
|
|
|
simplifyLine(lt, precision, 0, len - 1);
|
|
|
|
|
|
|
|
TIntArrayList res = new TIntArrayList(len * 2);
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
if (lt.get(i * 3) == 1) {
|
|
|
|
res.add(lt.get(i * 3 + 1));
|
|
|
|
res.add(lt.get(i * 3 + 2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static double orthogonalDistance(int x, int y, int x1, int y1, int x2, int y2) {
|
|
|
|
long A = (x - x1);
|
|
|
|
long B = (y - y1);
|
|
|
|
long C = (x2 - x1);
|
|
|
|
long D = (y2 - y1);
|
|
|
|
return Math.abs(A * D - C * B) / Math.sqrt(C * C + D * D);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void simplifyLine(TIntArrayList lt, int precision, int start, int end) {
|
|
|
|
if(start == end - 1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int x = lt.get(start*3 + 1);
|
|
|
|
int y = lt.get(start*3 + 2);
|
|
|
|
int ex = lt.get(end*3 + 1);
|
|
|
|
int ey = lt.get(end*3 + 2);
|
|
|
|
double max = 0;
|
|
|
|
int maxK = -1;
|
|
|
|
for(int k = start + 1; k < end ; k++) {
|
|
|
|
double ld = orthogonalDistance(lt.get(k*3 + 1), lt.get(k*3 + 2), x, y, ex, ey);
|
|
|
|
if(maxK == -1 || max < ld) {
|
|
|
|
maxK = k;
|
|
|
|
max = ld;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(max < precision) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
lt.set(maxK*3, 1); // keep point
|
|
|
|
simplifyLine(lt, precision, start, maxK);
|
|
|
|
simplifyLine(lt, precision, maxK, end);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-08-19 12:45:06 +02:00
|
|
|
public int[] getCoordinates() {
|
|
|
|
return coordinates;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public int getObjectType() {
|
|
|
|
return objectType;
|
|
|
|
}
|
2020-01-28 15:55:43 +01:00
|
|
|
|
|
|
|
public int[] getLabelCoordinates() {
|
|
|
|
return labelCoordinates;
|
|
|
|
}
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|