package net.osmand.data; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Map.Entry; import net.osmand.Collator; import net.osmand.OsmAndCollator; import net.osmand.util.Algorithms; import net.sf.junidecode.Junidecode; public abstract class MapObject implements Comparable { public static final MapObjectComparator BY_NAME_COMPARATOR = new MapObjectComparator(); protected String name = null; protected String enName = null; /** * Looks like: {ru=Москва, dz=མོསི་ཀོ...} and does not contain values of OSM tags "name" and "name:en", * see {@link name} and {@link enName} respectively. */ protected Map names = null; protected LatLon location = null; protected int fileOffset = 0; protected Long id = null; private Object referenceFile = null; public void setId(Long id) { this.id = id; } public Long getId() { if (id != null) { return id; } return null; } public String getName() { if (name != null) { return name; } return ""; //$NON-NLS-1$ } public void setName(String name) { this.name = name; } public void setName(String lang, String name) { if (Algorithms.isEmpty(lang)) { setName(name); } else if (lang.equals("en")) { setEnName(name); } else { if (names == null) { names = new HashMap(); } names.put(lang, name); } } public Map getNamesMap(boolean includeEn) { if (!includeEn || Algorithms.isEmpty(enName)) { if (names == null) { return Collections.emptyMap(); } return names; } Map mp = new HashMap(); if (names != null) { mp.putAll(names); } mp.put("en", enName); return mp; } public List getAllNames() { List l = new ArrayList(); if (!Algorithms.isEmpty(enName)) { l.add(enName); } if (names != null) { l.addAll(names.values()); } return l; } public List getAllNames(boolean transliterate) { List l = new ArrayList(); String enName = getEnName(transliterate); if (!Algorithms.isEmpty(enName)) { l.add(enName); } if (names != null) { l.addAll(names.values()); } return l; } public void copyNames(String otherName, String otherEnName, Map otherNames, boolean overwrite) { if (!Algorithms.isEmpty(otherName) && (overwrite || Algorithms.isEmpty(name))) { name = otherName; } if (!Algorithms.isEmpty(otherEnName) && (overwrite || Algorithms.isEmpty(enName))) { enName = otherEnName; } if (!Algorithms.isEmpty(otherNames)) { if (otherNames.containsKey("name:en")) { enName = otherNames.get("name:en"); } else if (otherNames.containsKey("en")) { enName = otherNames.get("en"); } for (Entry e : otherNames.entrySet()) { String key = e.getKey(); if (key.startsWith("name:")) { key = key.substring("name:".length()); } if (names == null) { names = new HashMap(); } if (overwrite || Algorithms.isEmpty(names.get(key))) { names.put(key, e.getValue()); } } } } public void copyNames(String otherName, String otherEnName, Map otherNames) { copyNames(otherName, otherEnName, otherNames, false); } public void copyNames(MapObject s, boolean copyName, boolean copyEnName, boolean overwrite) { copyNames((copyName ? s.name : null), (copyEnName ? s.enName : null), s.names, overwrite); } public void copyNames(MapObject s) { copyNames(s, true, true, false); } public String getName(String lang) { return getName(lang, false); } public String getName(String lang, boolean transliterate) { if (lang != null && lang.length() > 0) { if (lang.equals("en")) { // ignore transliterate option here for backward compatibility return getEnName(true); } else { // get name if (names != null) { String nm = names.get(lang); if (!Algorithms.isEmpty(nm)) { return nm; } if (transliterate) { return Junidecode.unidecode(getName()); } } } } return getName(); } public String getEnName(boolean transliterate) { if (!Algorithms.isEmpty(enName)) { return this.enName; } else if (!Algorithms.isEmpty(getName()) && transliterate) { return Junidecode.unidecode(getName()); } return ""; //$NON-NLS-1$ } public void setEnName(String enName) { this.enName = enName; } public LatLon getLocation() { return location; } public void setLocation(double latitude, double longitude) { location = new LatLon(latitude, longitude); } @Override public int compareTo(MapObject o) { return OsmAndCollator.primaryCollator().compare(getName(), o.getName()); } public int getFileOffset() { return fileOffset; } public void setFileOffset(int fileOffset) { this.fileOffset = fileOffset; } public String toStringEn() { return getClass().getSimpleName() + ":" + getEnName(true); } @Override public String toString() { return getClass().getSimpleName() + " " + name + "(" + id + ")"; //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$ } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((id == null) ? 0 : id.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; MapObject other = (MapObject) obj; if (id == null) { if (other.id != null) return false; } else if (!id.equals(other.id)) return false; return true; } public static class MapObjectComparator implements Comparator { private final String l; Collator collator = OsmAndCollator.primaryCollator(); public MapObjectComparator() { this.l = null; } public MapObjectComparator(String lang) { this.l = lang; } @Override public int compare(MapObject o1, MapObject o2) { if (o1 == null ^ o2 == null) { return (o1 == null) ? -1 : 1; } else if (o1 == o2) { return 0; } else { return collator.compare(o1.getName(l), o2.getName(l)); } } public boolean areEqual(MapObject o1, MapObject o2) { if (o1 == null ^ o2 == null) { return false; } else if (o1 == o2) { return true; } else { return collator.equals(o1.getName(l), o2.getName(l)); } } } public void setReferenceFile(Object referenceFile) { this.referenceFile = referenceFile; } public Object getReferenceFile() { return referenceFile; } }