forked from hans/Nominatim
Merge pull request #487 from lonvia/unify-reverse
Remove duplicate reverse geocoding code
This commit is contained in:
@@ -1,5 +1,6 @@
|
|||||||
<?php
|
<?php
|
||||||
require_once(CONST_BasePath.'/lib/PlaceLookup.php');
|
require_once(CONST_BasePath.'/lib/PlaceLookup.php');
|
||||||
|
require_once(CONST_BasePath.'/lib/ReverseGeocode.php');
|
||||||
|
|
||||||
class Geocode
|
class Geocode
|
||||||
{
|
{
|
||||||
@@ -1696,9 +1697,16 @@
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Just interpret as a reverse geocode
|
// Just interpret as a reverse geocode
|
||||||
$iPlaceID = geocodeReverse((float)$this->aNearPoint[0], (float)$this->aNearPoint[1]);
|
$oReverse = new ReverseGeocode($this->oDB);
|
||||||
if ($iPlaceID)
|
$oReverse->setLatLon((float)$this->aNearPoint[0], (float)$this->aNearPoint[1]);
|
||||||
$aSearchResults = $this->getDetails(array($iPlaceID));
|
$oReverse->setZoom(18);
|
||||||
|
|
||||||
|
$aLookup = $oReverse->lookup(false);
|
||||||
|
|
||||||
|
if (CONST_Debug) var_dump("Reverse search", $aLookup);
|
||||||
|
|
||||||
|
if ($aLookup['place_id'])
|
||||||
|
$aSearchResults = $this->getDetails(array($aLookup['place_id'] => -1));
|
||||||
else
|
else
|
||||||
$aSearchResults = array();
|
$aSearchResults = array();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -108,7 +108,7 @@
|
|||||||
|
|
||||||
// returns { place_id =>, type => '(osm|tiger)' }
|
// returns { place_id =>, type => '(osm|tiger)' }
|
||||||
// fails if no place was found
|
// fails if no place was found
|
||||||
function lookup()
|
function lookup($bDoInterpolation = true)
|
||||||
{
|
{
|
||||||
$sPointSQL = 'ST_SetSRID(ST_Point('.$this->fLon.','.$this->fLat.'),4326)';
|
$sPointSQL = 'ST_SetSRID(ST_Point('.$this->fLon.','.$this->fLat.'),4326)';
|
||||||
$iMaxRank = $this->iMaxRank;
|
$iMaxRank = $this->iMaxRank;
|
||||||
@@ -155,7 +155,7 @@
|
|||||||
$bIsInUnitedStates = ($aPlace['calculated_country_code'] == 'us');
|
$bIsInUnitedStates = ($aPlace['calculated_country_code'] == 'us');
|
||||||
}
|
}
|
||||||
// if a street or house was found, look in interpolation lines table
|
// if a street or house was found, look in interpolation lines table
|
||||||
if ($iMaxRank_orig >= 28 && $aPlace && $aPlace['rank_search'] >= 26)
|
if ($bDoInterpolation && $iMaxRank_orig >= 28 && $aPlace && $aPlace['rank_search'] >= 26)
|
||||||
{
|
{
|
||||||
// if a house was found, search the interpolation line that is at least as close as the house
|
// if a house was found, search the interpolation line that is at least as close as the house
|
||||||
$sSQL = 'SELECT place_id, parent_place_id, 30 as rank_search, ST_line_locate_point(linegeo,'.$sPointSQL.') as fraction';
|
$sSQL = 'SELECT place_id, parent_place_id, 30 as rank_search, ST_line_locate_point(linegeo,'.$sPointSQL.') as fraction';
|
||||||
@@ -220,7 +220,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Only street found? If it's in the US we can check TIGER data for nearest housenumber
|
// Only street found? If it's in the US we can check TIGER data for nearest housenumber
|
||||||
if (CONST_Use_US_Tiger_Data && $bIsInUnitedStates && $iMaxRank_orig >= 28 && $iPlaceID && ($aPlace['rank_search'] == 26 || $aPlace['rank_search'] == 27 ))
|
if (CONST_Use_US_Tiger_Data && $bDoInterpolation && $bIsInUnitedStates && $iMaxRank_orig >= 28 && $iPlaceID && ($aPlace['rank_search'] == 26 || $aPlace['rank_search'] == 27 ))
|
||||||
{
|
{
|
||||||
$fSearchDiam = 0.001;
|
$fSearchDiam = 0.001;
|
||||||
$sSQL = 'SELECT place_id,parent_place_id,30 as rank_search, ST_line_locate_point(linegeo,'.$sPointSQL.') as fraction';
|
$sSQL = 'SELECT place_id,parent_place_id,30 as rank_search, ST_line_locate_point(linegeo,'.$sPointSQL.') as fraction';
|
||||||
|
|||||||
84
lib/lib.php
84
lib/lib.php
@@ -680,90 +680,6 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function geocodeReverse($fLat, $fLon, $iZoom=18)
|
|
||||||
{
|
|
||||||
$oDB =& getDB();
|
|
||||||
|
|
||||||
$sPointSQL = "ST_SetSRID(ST_Point($fLon,$fLat),4326)";
|
|
||||||
|
|
||||||
// Zoom to rank, this could probably be calculated but a lookup gives fine control
|
|
||||||
$aZoomRank = array(
|
|
||||||
0 => 2, // Continent / Sea
|
|
||||||
1 => 2,
|
|
||||||
2 => 2,
|
|
||||||
3 => 4, // Country
|
|
||||||
4 => 4,
|
|
||||||
5 => 8, // State
|
|
||||||
6 => 10, // Region
|
|
||||||
7 => 10,
|
|
||||||
8 => 12, // County
|
|
||||||
9 => 12,
|
|
||||||
10 => 17, // City
|
|
||||||
11 => 17,
|
|
||||||
12 => 18, // Town / Village
|
|
||||||
13 => 18,
|
|
||||||
14 => 22, // Suburb
|
|
||||||
15 => 22,
|
|
||||||
16 => 26, // Street, TODO: major street?
|
|
||||||
17 => 26,
|
|
||||||
18 => 30, // or >, Building
|
|
||||||
19 => 30, // or >, Building
|
|
||||||
);
|
|
||||||
$iMaxRank = isset($aZoomRank[$iZoom])?$aZoomRank[$iZoom]:28;
|
|
||||||
|
|
||||||
// Find the nearest point
|
|
||||||
$fSearchDiam = 0.0001;
|
|
||||||
$iPlaceID = null;
|
|
||||||
$aArea = false;
|
|
||||||
$fMaxAreaDistance = 1;
|
|
||||||
while(!$iPlaceID && $fSearchDiam < $fMaxAreaDistance)
|
|
||||||
{
|
|
||||||
$fSearchDiam = $fSearchDiam * 2;
|
|
||||||
|
|
||||||
// If we have to expand the search area by a large amount then we need a larger feature
|
|
||||||
// then there is a limit to how small the feature should be
|
|
||||||
if ($fSearchDiam > 2 && $iMaxRank > 4) $iMaxRank = 4;
|
|
||||||
if ($fSearchDiam > 1 && $iMaxRank > 9) $iMaxRank = 8;
|
|
||||||
if ($fSearchDiam > 0.8 && $iMaxRank > 10) $iMaxRank = 10;
|
|
||||||
if ($fSearchDiam > 0.6 && $iMaxRank > 12) $iMaxRank = 12;
|
|
||||||
if ($fSearchDiam > 0.2 && $iMaxRank > 17) $iMaxRank = 17;
|
|
||||||
if ($fSearchDiam > 0.1 && $iMaxRank > 18) $iMaxRank = 18;
|
|
||||||
if ($fSearchDiam > 0.008 && $iMaxRank > 22) $iMaxRank = 22;
|
|
||||||
if ($fSearchDiam > 0.001 && $iMaxRank > 26) $iMaxRank = 26;
|
|
||||||
|
|
||||||
$sSQL = 'select place_id,parent_place_id from placex';
|
|
||||||
$sSQL .= ' WHERE ST_DWithin('.$sPointSQL.', geometry, '.$fSearchDiam.')';
|
|
||||||
$sSQL .= ' and rank_search != 28 and rank_search >= '.$iMaxRank;
|
|
||||||
$sSQL .= ' and (name is not null or housenumber is not null)';
|
|
||||||
$sSQL .= ' and class not in (\'waterway\')';
|
|
||||||
$sSQL .= ' and (ST_GeometryType(geometry) not in (\'ST_Polygon\',\'ST_MultiPolygon\') ';
|
|
||||||
$sSQL .= ' OR ST_DWithin('.$sPointSQL.', ST_Centroid(geometry), '.$fSearchDiam.'))';
|
|
||||||
$sSQL .= ' ORDER BY ST_distance('.$sPointSQL.', geometry) ASC limit 1';
|
|
||||||
//var_dump($sSQL);
|
|
||||||
$aPlace = chksql($oDB->getRow($sSQL));
|
|
||||||
$iPlaceID = $aPlace['place_id'];
|
|
||||||
}
|
|
||||||
|
|
||||||
// The point we found might be too small - use the address to find what it is a child of
|
|
||||||
if ($iPlaceID)
|
|
||||||
{
|
|
||||||
$sSQL = "select address_place_id from place_addressline where cached_rank_address <= $iMaxRank and place_id = $iPlaceID order by cached_rank_address desc,isaddress desc,distance desc limit 1";
|
|
||||||
$iPlaceID = chksql($oDB->getOne($sSQL));
|
|
||||||
|
|
||||||
if ($iPlaceID && $aPlace['place_id'] && $iMaxRank < 28)
|
|
||||||
{
|
|
||||||
$sSQL = "select address_place_id from place_addressline where cached_rank_address <= $iMaxRank and place_id = ".$aPlace['place_id']." order by cached_rank_address desc,isaddress desc,distance desc";
|
|
||||||
$iPlaceID = chksql($oDB->getOne($sSQL));
|
|
||||||
}
|
|
||||||
if (!$iPlaceID)
|
|
||||||
{
|
|
||||||
$iPlaceID = $aPlace['place_id'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $iPlaceID;
|
|
||||||
}
|
|
||||||
|
|
||||||
function addQuotes($s)
|
function addQuotes($s)
|
||||||
{
|
{
|
||||||
return "'".$s."'";
|
return "'".$s."'";
|
||||||
|
|||||||
@@ -18,9 +18,9 @@ Feature: Reverse geocoding
|
|||||||
| xml | 4
|
| xml | 4
|
||||||
When looking up coordinates 53.9788769,13.0830313
|
When looking up coordinates 53.9788769,13.0830313
|
||||||
And results contain valid boundingboxes
|
And results contain valid boundingboxes
|
||||||
|
|
||||||
Scenario: Reverse geocoding for odd interpolated housenumber
|
Scenario: Reverse geocoding for odd interpolated housenumber
|
||||||
|
|
||||||
Scenario: Reverse geocoding for even interpolated housenumber
|
Scenario: Reverse geocoding for even interpolated housenumber
|
||||||
|
|
||||||
@Tiger
|
@Tiger
|
||||||
@@ -32,9 +32,10 @@ Feature: Reverse geocoding
|
|||||||
And exactly 1 result is returned
|
And exactly 1 result is returned
|
||||||
And result addresses contain
|
And result addresses contain
|
||||||
| ID | house_number | road | postcode | country_code
|
| ID | house_number | road | postcode | country_code
|
||||||
| 0 | 7096 | Kings Estate Drive | 84128 | us
|
| 0 | 709. | Kings Estate Drive | 84128 | us
|
||||||
And result 0 has not attributes osm_id,osm_type
|
And results contain
|
||||||
|
| osm_type
|
||||||
|
| tiger
|
||||||
|
|
||||||
@Tiger
|
@Tiger
|
||||||
Scenario: No TIGER house number for zoom < 18
|
Scenario: No TIGER house number for zoom < 18
|
||||||
|
|||||||
Reference in New Issue
Block a user