Merge remote-tracking branch 'origin/master'

This commit is contained in:
Weblate 2014-05-03 16:23:47 +02:00
commit e4befab193
4 changed files with 367 additions and 211 deletions

View file

@ -68,6 +68,7 @@
<category android:name="android.intent.category.CAR_DOCK" /> <category android:name="android.intent.category.CAR_DOCK" />
<category android:name="android.intent.category.DESK_DOCK" /> <category android:name="android.intent.category.DESK_DOCK" />
</intent-filter> </intent-filter>
<!-- android matches non-greedy : http://stackoverflow.com/questions/3400072/pathpattern-to-match-file-extension-does-not-work-if-a-period-exists-elsewhere-i--> <!-- android matches non-greedy : http://stackoverflow.com/questions/3400072/pathpattern-to-match-file-extension-does-not-work-if-a-period-exists-elsewhere-i-->
<!-- mimeType&host are both needed or you will either have unwanted matching or no match when needed --> <!-- mimeType&host are both needed or you will either have unwanted matching or no match when needed -->
<intent-filter> <intent-filter>
@ -78,7 +79,16 @@
<data android:scheme="file" android:host="*" android:mimeType="*/*" android:pathPattern=".*\\..*\\..*\\.gpx"/> <data android:scheme="file" android:host="*" android:mimeType="*/*" android:pathPattern=".*\\..*\\..*\\.gpx"/>
<data android:scheme="file" android:host="*" android:mimeType="*/*" android:pathPattern=".*\\..*\\..*\\..*\\.gpx"/> <data android:scheme="file" android:host="*" android:mimeType="*/*" android:pathPattern=".*\\..*\\..*\\..*\\.gpx"/>
</intent-filter> </intent-filter>
<!-- google navigation intent -->
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<data android:scheme="google.navigation"/>
</intent-filter>
<receiver android:name="net.osmand.plus.audionotes.MediaRemoteControlReceiver"> <receiver android:name="net.osmand.plus.audionotes.MediaRemoteControlReceiver">
<intent-filter> <intent-filter>
<action android:name="android.intent.action.CAMERA_BUTTON" /> <action android:name="android.intent.action.CAMERA_BUTTON" />
<action android:name="android.intent.action.MEDIA_BUTTON" /> <action android:name="android.intent.action.MEDIA_BUTTON" />
@ -118,9 +128,10 @@
<activity android:name="net.osmand.plus.activities.search.SearchBuildingByNameActivity"></activity> <activity android:name="net.osmand.plus.activities.search.SearchBuildingByNameActivity"></activity>
<activity android:name="net.osmand.plus.activities.EditPOIFilterActivity"></activity> <activity android:name="net.osmand.plus.activities.EditPOIFilterActivity"></activity>
<activity android:name="net.osmand.plus.activities.search.GeoIntentActivity"> <activity android:name="net.osmand.plus.activities.search.GeoIntentActivity">
<intent-filter><action android:name="android.intent.action.VIEW"></action> <intent-filter>
<category android:name="android.intent.category.DEFAULT"></category> <data android:scheme="geo" />
<data android:scheme="geo"></data> <action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter> </intent-filter>
<intent-filter> <intent-filter>
<data android:scheme="http" android:host="maps.google.com"/> <data android:scheme="http" android:host="maps.google.com"/>

View file

@ -1733,4 +1733,5 @@ Afghanistan, Albania, Algeria, Andorra, Angola, Anguilla, Antigua and Barbuda, A
<string name="av_camera_focus_continuous">The camera continuously tries to focus</string> <string name="av_camera_focus_continuous">The camera continuously tries to focus</string>
<string name="av_photo_play_sound">Play sound on photo shot</string> <string name="av_photo_play_sound">Play sound on photo shot</string>
<string name="av_photo_play_sound_descr">Choose whether to play a sound when shooting photos</string> <string name="av_photo_play_sound_descr">Choose whether to play a sound when shooting photos</string>
<string name="navigation_intent_invalid">Invalid format : %s</string>
</resources> </resources>

View file

@ -6,6 +6,8 @@ import java.util.ArrayList;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import net.osmand.Location; import net.osmand.Location;
import net.osmand.StateChangedListener; import net.osmand.StateChangedListener;
@ -335,10 +337,36 @@ public class MapActivity extends AccessibleActivity {
if (intent.getData() != null) if (intent.getData() != null)
{ {
final Uri data = intent.getData(); final Uri data = intent.getData();
if ("file".equalsIgnoreCase(data.getScheme())) final String scheme = data.getScheme();
if ("file".equals(scheme))
{ {
showImportedGpx(data.getPath()); showImportedGpx(data.getPath());
} }
else if("google.navigation".equals(scheme))
{
final String schemeSpecificPart = data.getSchemeSpecificPart();
final Matcher matcher = Pattern.compile("q=(.+?),(.+?)").matcher(schemeSpecificPart);
if (matcher.matches())
{
try
{
final double lat = Double.valueOf(matcher.group(1));
final double lon = Double.valueOf(matcher.group(2));
getMyApplication().getTargetPointsHelper().navigateToPoint(new LatLon(lat, lon), false, -1);
getMapActions().enterRoutePlanningMode(null, null);
}
catch (NumberFormatException e)
{
AccessibleToast.makeText(this, getString(R.string.navigation_intent_invalid, schemeSpecificPart), Toast.LENGTH_LONG).show(); //$NON-NLS-1$
}
}
else
{
AccessibleToast.makeText(this, getString(R.string.navigation_intent_invalid, schemeSpecificPart), Toast.LENGTH_LONG).show(); //$NON-NLS-1$
}
}
} }
} }
} }

View file

@ -1,5 +1,6 @@
package net.osmand.plus.activities.search; package net.osmand.plus.activities.search;
import android.os.AsyncTask;
import gnu.trove.map.hash.TLongObjectHashMap; import gnu.trove.map.hash.TLongObjectHashMap;
import java.util.ArrayList; import java.util.ArrayList;
@ -7,6 +8,8 @@ import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import net.osmand.ResultMatcher; import net.osmand.ResultMatcher;
import net.osmand.access.AccessibleToast; import net.osmand.access.AccessibleToast;
@ -47,53 +50,104 @@ public class GeoIntentActivity extends OsmandListActivity {
private ProgressDialog startProgressDialog; private ProgressDialog startProgressDialog;
@Override @Override
protected void onCreate(Bundle savedInstanceState) { protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
setContentView(R.layout.search_address_offline); setContentView(R.layout.search_address_offline);
getSupportActionBar().setTitle(R.string.search_osm_offline); getSupportActionBar().setTitle(R.string.search_osm_offline);
startProgressDialog = new ProgressDialog(this); startProgressDialog = new ProgressDialog(this);
getMyApplication().checkApplicationIsBeingInitialized(this, startProgressDialog); getMyApplication().checkApplicationIsBeingInitialized(this, startProgressDialog);
location = getMyApplication().getSettings().getLastKnownMapLocation(); location = getMyApplication().getSettings().getLastKnownMapLocation();
final Intent intent = getIntent(); final Intent intent = getIntent();
if (intent != null) { if (intent != null)
progressDlg = ProgressDialog.show(this, {
getString(R.string.searching), final ProgressDialog progress = ProgressDialog.show(GeoIntentActivity.this, getString(R.string.searching), getString(R.string.searching_address));
getString(R.string.searching_address)); final GeoIntentTask task = new GeoIntentTask(progress, intent);
final Thread searcher = new Thread(new Runnable() {
progress.setOnCancelListener(new OnCancelListener()
{
@Override @Override
public void run() { public void onCancel(DialogInterface dialog)
try { {
while(getMyApplication().isApplicationInitializing()) { task.cancel(true);
Thread.sleep(200);
}
Collection<? extends MapObject> results = extract(
intent.getData()).execute();
// show the first result on map, and populate the list!
if (!results.isEmpty()) {
showResult(0, new ArrayList<MapObject>(results));
} else {
showResult(R.string.search_nothing_found, null);
}
} catch (Exception e) {
e.printStackTrace();
showResult(R.string.error_doing_search, null);
} finally {
if (progressDlg != null) {
progressDlg.dismiss();
}
}
}
}, "SearchingAddress");
searcher.start();
progressDlg.setOnCancelListener(new OnCancelListener() {
@Override
public void onCancel(DialogInterface dialog) {
searcher.interrupt();
} }
}); });
progressDlg.setCancelable(true); progress.setCancelable(true);
task.execute();
} }
// finish(); }
private class GeoIntentTask extends AsyncTask<Void,Void, ExecutionResult>
{
private final ProgressDialog progress;
private final Intent intent;
private GeoIntentTask(final ProgressDialog progress, final Intent intent)
{
this.progress = progress;
this.intent = intent;
}
@Override
protected void onPreExecute()
{
}
@Override
protected ExecutionResult doInBackground(Void... nothing)
{
try
{
while (getMyApplication().isApplicationInitializing())
{
Thread.sleep(200);
}
return extract(intent.getScheme(), intent.getData()).execute();
}
catch (Exception e)
{
return null;
}
}
@Override
protected void onPostExecute(ExecutionResult result)
{
progress.dismiss();
if (result != null)
{
if (result.isEmpty())
{
AccessibleToast.makeText(GeoIntentActivity.this, getString(R.string.search_nothing_found), Toast.LENGTH_LONG).show();
}
else
{
if (result.hasZoom())
{
getMyApplication().getSettings().setLastKnownMapZoom(result.getZoom());
}
final List<MapObject> places = new ArrayList<MapObject>(result.getMapObjects());
setListAdapter(new MapObjectAdapter(places));
if (places.size() == 1)
{
onListItemClick(
getListView(),
getListAdapter().getView(0, null, null),
0,
getListAdapter().getItemId(0)
);
}
}
}
else
{
AccessibleToast.makeText(GeoIntentActivity.this, getString(R.string.search_offline_geo_error, intent.getData()), Toast.LENGTH_LONG).show();
}
}
} }
@Override @Override
@ -104,26 +158,7 @@ public class GeoIntentActivity extends OsmandListActivity {
return super.onCreateDialog(id); return super.onCreateDialog(id);
} }
private void showResult(final int warning, final List<MapObject> places) { private class MapObjectAdapter extends ArrayAdapter<MapObject> {
runOnUiThread(new Runnable() {
@Override
public void run() {
if (places == null) {
AccessibleToast.makeText(GeoIntentActivity.this, getString(warning),
Toast.LENGTH_LONG).show();
} else {
setListAdapter(new MapObjectAdapter(places));
if (places.size() == 1) {
onListItemClick(getListView(), getListAdapter()
.getView(0, null, null), 0, getListAdapter()
.getItemId(0));
}
}
}
});
}
class MapObjectAdapter extends ArrayAdapter<MapObject> {
public MapObjectAdapter(List<MapObject> places) { public MapObjectAdapter(List<MapObject> places) {
super(GeoIntentActivity.this, super(GeoIntentActivity.this,
@ -190,71 +225,114 @@ public class GeoIntentActivity extends OsmandListActivity {
} }
/** /**
* geo:latitude,longitude<BR> * Extracts information from geo and map intents:
* geo:latitude,longitude?z=zoom<BR>
* geo:0,0?q=my+street+address<BR>
* geo:0,0?q=business+near+city
* *
* @param data * geo:47.6,-122.3<br/>
* geo:47.6,-122.3?z=11<br/>
* geo:0,0?q=34.99,-106.61(Treasure)<br/>
* geo:0,0?q=1600+Amphitheatre+Parkway%2C+CA<br/>
*
* @param scheme The intent scheme
* @param data The intent uri
* @return * @return
*/ */
private MyService extract(Uri data) { private MyService extract(final String scheme, final Uri data)
if ("http".equalsIgnoreCase(data.getScheme()) && "maps.google.com".equals(data.getHost())) { {
if ("http".equals(scheme))
{
String q = null; String q = null;
String parameter = data.getQueryParameter("q"); String parameter = data.getQueryParameter("q");
if (parameter == null) { if (parameter == null)
{
parameter = data.getQueryParameter("daddr"); parameter = data.getQueryParameter("daddr");
} }
if(parameter != null) { if (parameter != null)
{
q = parameter.split(" ")[0]; q = parameter.split(" ")[0];
} }
if (q.indexOf(',') != -1) { if (q.indexOf(',') != -1)
{
int i = q.indexOf(','); int i = q.indexOf(',');
String lat = q.substring(0, i); String lat = q.substring(0, i);
String lon = q.substring(i + 1); String lon = q.substring(i + 1);
if (lat.indexOf(':') != -1) { if (lat.indexOf(':') != -1)
{
i = lat.indexOf(':'); i = lat.indexOf(':');
lat = lat.substring(i + 1); lat = lat.substring(i + 1);
} }
try { try
{
double llat = Double.parseDouble(lat.trim()); double llat = Double.parseDouble(lat.trim());
double llon = Double.parseDouble(lon.trim()); double llon = Double.parseDouble(lon.trim());
return new GeoPointSearch(llat, llon); return new GeoPointSearch(llat, llon);
} catch (NumberFormatException e) {
showErrorMessage(q);
} }
} else { catch (NumberFormatException e)
showErrorMessage(q); {
return null;
} }
} else if (data.getSchemeSpecificPart().indexOf("0,0?") != -1) { }
// it is 0,0? that means a search else
return new GeoAddressSearch(data.getQuery()); {
} else { return null;
String geo = data.getSchemeSpecificPart(); }
if(geo == null) { }
showErrorMessage(""); if ("geo".equals(scheme))
} else { {
int latIndex = geo.indexOf(','); //geo:
if (latIndex > 0) { final String schemeSpecific = data.getSchemeSpecificPart();
int lonIndex = geo.indexOf(',', latIndex + 1); if (schemeSpecific == null)
int altIndex = geo.indexOf(';', latIndex + 1); {
int paramIndex = geo.indexOf('?', latIndex + 1); return null;
paramIndex = paramIndex > 0 ? paramIndex : geo.length(); }
altIndex = altIndex > 0 && altIndex < paramIndex ? altIndex : paramIndex; if (schemeSpecific.startsWith("0,0?q="))
lonIndex = lonIndex > 0 && lonIndex < altIndex ? lonIndex : altIndex; {
try { //geo:0,0?q=34.99,-106.61(Treasure)
double lat = Double.parseDouble(geo.substring(0, latIndex).trim()); //geo:0,0?q=1600+Amphitheatre+Parkway%2C+CA
double lon = Double.parseDouble(geo.substring(latIndex + 1, lonIndex).trim()); final String query = schemeSpecific.substring("0,0?q=".length());
final Matcher matcher = Pattern.compile("([\\-0-9.]+),([\\-0-9.]+)(?:,[\\-0-9.]+)?\\((.+?)\\)").matcher(query);
if (matcher.matches())
{
final double lat = Double.valueOf(matcher.group(1));
final double lon = Double.valueOf(matcher.group(2));
final String name = matcher.group(3);
return new GeoPointSearch(lat, lon, name);
}
else
{
//we suppose it's a search
return new GeoAddressSearch(query);
}
}
else
{
//geo:47.6,-122.3
//geo:47.6,-122.3?z=11
//allow for http://tools.ietf.org/html/rfc5870 (geo uri) , just ignore everything after ';'
final String pattern = "([\\-0-9.]+),([\\-0-9.]+)(?:,([\\-0-9.]+))?(?:\\?z=([0-9]+))?(?:;.*)?";
final Matcher matcher = Pattern.compile(pattern).matcher(schemeSpecific);
if (matcher.matches())
{
final double lat = Double.valueOf(matcher.group(1));
final double lon = Double.valueOf(matcher.group(2));
if (matcher.group(4) == null)
{
return new GeoPointSearch(lat, lon); return new GeoPointSearch(lat, lon);
} catch (NumberFormatException e) {
showErrorMessage(geo);
} }
} else { else
showErrorMessage(geo); {
return new GeoPointSearch(lat, lon, Integer.valueOf(matcher.group(4)));
}
}
else
{
return null;
} }
} }
} }
return new Empty(); return null;
} }
private final class GeoAddressSearch implements MyService { private final class GeoAddressSearch implements MyService {
@ -318,14 +396,14 @@ public class GeoIntentActivity extends OsmandListActivity {
} }
@Override @Override
public Collection<? extends MapObject> execute() { public ExecutionResult execute() {
if (elements.isEmpty()) { if (elements.isEmpty()) {
return Collections.emptyList(); return ExecutionResult.EMPTY;
} }
List<String> q = new ArrayList<String>(elements); List<String> q = new ArrayList<String>(elements);
MapObject geo = checkGeoPoint(); MapObject geo = checkGeoPoint();
if(geo != null) { if(geo != null) {
return Collections.singleton(geo); return new ExecutionResult(Collections.singleton(geo));
} }
// now try to search the City, Street, Etc.. if Street is not found, // now try to search the City, Street, Etc.. if Street is not found,
@ -357,7 +435,7 @@ public class GeoIntentActivity extends OsmandListActivity {
} }
} }
if(cityIds.isEmpty()) { if(cityIds.isEmpty()) {
return allStreets; return new ExecutionResult(allStreets);
} }
final List<MapObject> connectedStreets = new ArrayList<MapObject>(); final List<MapObject> connectedStreets = new ArrayList<MapObject>();
Iterator<Street> p = allStreets.iterator(); Iterator<Street> p = allStreets.iterator();
@ -382,16 +460,16 @@ public class GeoIntentActivity extends OsmandListActivity {
List<MapObject> all = new ArrayList<MapObject>(); List<MapObject> all = new ArrayList<MapObject>();
all.addAll(cityIds.valueCollection()); all.addAll(cityIds.valueCollection());
all.addAll(allStreets); all.addAll(allStreets);
return all; return new ExecutionResult(all);
} else { } else {
// add all other results to connected streets // add all other results to connected streets
connectedStreets.addAll(cityIds.valueCollection()); connectedStreets.addAll(cityIds.valueCollection());
return connectedStreets; return new ExecutionResult(connectedStreets);
} }
} }
private Collection<RegionAddressRepository> limitSearchToCountries(List<String> q) { private Collection<RegionAddressRepository> limitSearchToCountries(List<String> q) {
ResourceManager resourceManager = resourceManager(); ResourceManager resourceManager = getMyApplication().getResourceManager();
List<RegionAddressRepository> foundCountries = new ArrayList<RegionAddressRepository>(); List<RegionAddressRepository> foundCountries = new ArrayList<RegionAddressRepository>();
RegionAddressRepository country; RegionAddressRepository country;
Iterator<String> it = q.iterator(); Iterator<String> it = q.iterator();
@ -414,61 +492,99 @@ public class GeoIntentActivity extends OsmandListActivity {
} }
private ResourceManager resourceManager() {
return getMyApplication().getResourceManager();
}
private void showErrorMessage(final String geo) {
runOnUiThread(new Runnable() {
@Override
public void run() {
AccessibleToast.makeText(GeoIntentActivity.this,
getString(R.string.search_offline_geo_error, geo),
Toast.LENGTH_LONG);
}
});
}
private static class Empty implements MyService {
@Override
public Collection<MapObject> execute() {
return Collections.emptyList();
}
}
private static class GeoPointSearch implements MyService { private static class GeoPointSearch implements MyService {
private MapObject point; private final MapObject point;
/** private final int zoom;
* geo:latitude,longitude geo:latitude,longitude?z=zoom
*/ public GeoPointSearch(double lat , double lon)
public GeoPointSearch(double lat , double lon ) { {
// TODO zoom is omited for now this(lat, lon, ExecutionResult.NO_ZOOM);
point = new Amenity();
((Amenity)point).setType(AmenityType.USER_DEFINED);
((Amenity)point).setSubType("");
point.setLocation(lat, lon);
point.setName("Lat: " + lat + ",Lon:" + lon);
} }
public GeoPointSearch(double lat , double lon, int zoom)
{
this(lat, lon, "Lat: " + lat + ",Lon: " + lon, zoom);
}
public GeoPointSearch(double lat , double lon, String name )
{
this(lat, lon, name,ExecutionResult.NO_ZOOM);
}
public GeoPointSearch(double lat , double lon, String name, int zoom )
{
final Amenity amenity = new Amenity();
amenity.setLocation(lat, lon);
amenity.setName(name);
amenity.setType(AmenityType.USER_DEFINED);
amenity.setSubType("");
this.point = amenity;
this.zoom = zoom;
}
@Override @Override
public Collection<? extends MapObject> execute() { public ExecutionResult execute() {
if (point != null) { if (point != null) {
return Collections.singletonList(point); return new ExecutionResult(Collections.singletonList(point), zoom);
} else { } else {
return Collections.emptyList(); return ExecutionResult.EMPTY;
} }
} }
} }
private interface MyService { private static class ExecutionResult
{
public static final int NO_ZOOM = -1;
public static final ExecutionResult EMPTY = new ExecutionResult(new ArrayList<MapObject>(), NO_ZOOM);
public Collection<? extends MapObject> execute(); private final Collection<? extends MapObject> mapObjects;
private final int zoom;
public ExecutionResult(final Collection<? extends MapObject> mapObjects)
{
this(mapObjects, NO_ZOOM);
} }
public ExecutionResult(final Collection<? extends MapObject> mapObjects, final int zoom)
{
this.mapObjects = mapObjects;
this.zoom = zoom;
}
public boolean isEmpty()
{
return mapObjects.isEmpty();
}
public boolean hasZoom()
{
return zoom != NO_ZOOM;
}
public Collection<? extends MapObject> getMapObjects()
{
return mapObjects;
}
public int getZoom()
{
return zoom;
}
@Override
public String toString()
{
return "ExecutionResult{" +
"mapObjects=" + mapObjects +
", zoom=" + zoom +
'}';
}
}
private static interface MyService
{
public ExecutionResult execute();
}
} }