Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Finding towns within a 10 mile radius of postcode. Google maps API

I've recently started looking at the Google Maps API to try out something new in my websites. I am currently using this code:

<?php

$postcode = $_REQUEST['postcode'];

$url = 'http://maps.googleapis.com/maps/api/geocode/xml?address='.$postcode.'&sensor=false';
$parsedXML = simplexml_load_file($url);

if($parsedXML->status != "OK") {
echo "There has been a problem: " . $parsedXML->status;
}

$myAddress = array();
foreach($parsedXML->result->address_component as $component) {
if(is_array($component->type)) $type = (string)$component->type[0];
else $type = (string)$component->type;

$myAddress[$type] = (string)$component->long_name;
}
header('Content-Type: application/json');
echo json_encode($myAddress);


?>

which simply uses a postcode that I define and searches the Google database and then returns the town, county etc.

If possible, I would like to not only show the nearest town but also any within a 5-10 mile radius. Could someone tell me how I would go about doing this please?

Thanks for any help

like image 291
Daniel Hutton Avatar asked Nov 15 '11 10:11

Daniel Hutton


People also ask

How to set the radius of a specific location on Google Maps?

In this, you can just enter the name of a specific place on Google Maps. Now, you can specify the distance of radius in miles or kilometers. This will automatically draw the specific distance radius on Google Maps for the designated location.

How to use the Circle Tool of map developers?

Consider the example of the circle tool of Map Developers that we have discussed above. In this, you can just enter the name of a specific place on Google Maps. Now, you can specify the distance of radius in miles or kilometers. This will automatically draw the specific distance radius on Google Maps for the designated location.

How to measure distance between addresses on Google Maps?

This technique will let you measure distance between any addresses on Google Maps. It doesn’t matter if it is your present location or not. If you want to measure distance in some other location, then just enter its name in the search bar to load it. In this way, you can measure distance anywhere in the world by sitting in your home.

What is the postcode radius around London in PHP?

Postcode radius script PHP/ASP ... Postcodes within a Radius of 50 miles around Postcode NW1 London. KT2 Ham. Petersham and Richmond Riverside, Kingston Upon Thames (10.5 miles) TW11 Ham. Petersham and Richmond Riverside, Teddington (10.5 miles) Road distance is calculated as the fastest route, not necessarily the shortest..


1 Answers

Update: I wrote up a more detailed blogpost about this specific subject on http://www.mullie.eu/geographic-searches/

--

Loop through all available towns using the Google Maps API to fetch their latitude & longitude. Save these somewhere (database). - Beware, Google will not accept an enormous amount of calls, so throttle your calls.

Then, when fetching a town, you can use code similar to the code below to grab the cities withing a certain range:

public static function getNearby($lat, $lng, $type = 'cities', $limit = 50, $distance = 50, $unit = 'km')
{
    // radius of earth; @note: the earth is not perfectly spherical, but this is considered the 'mean radius'
    if ($unit == 'km') $radius = 6371.009; // in kilometers
    elseif ($unit == 'mi') $radius = 3958.761; // in miles

    // latitude boundaries
    $maxLat = (float) $lat + rad2deg($distance / $radius);
    $minLat = (float) $lat - rad2deg($distance / $radius);

    // longitude boundaries (longitude gets smaller when latitude increases)
    $maxLng = (float) $lng + rad2deg($distance / $radius / cos(deg2rad((float) $lat)));
    $minLng = (float) $lng - rad2deg($distance / $radius / cos(deg2rad((float) $lat)));

    // get results ordered by distance (approx)
    $nearby = (array) FrontendDB::getDB()->retrieve('SELECT *
                                                    FROM table
                                                    WHERE lat > ? AND lat < ? AND lng > ? AND lng < ?
                                                    ORDER BY ABS(lat - ?) + ABS(lng - ?) ASC
                                                    LIMIT ?;',
                                                    array($minLat, $maxLat, $minLng, $maxLng, (float) $lat, (float) $lng, (int) $limit));

    return $nearby;
}

Notes about the above code:

  • Own database wrapper is used, so transform to mysql_query, PDO, ...
  • This will not be exact. We can't do exact spherical calculations in the DB, so we've taken the upper & lower latitude & longitude limits. This basically means that a location which is slightly further than your distance (e.g. in the far north-east, just outside of the actual radius (which actually is pretty much a circle), but still inside the max latitude & longitude (because we compare it to square limits in the database). This will just give a rough but nut 100% accurate selection of cities withing your radius.

I'll try to illustrate this:

_________________
|      / \      |
| Y  /     \    |
|  /         \  |
|(      X      )|
|  \         /  |
|    \     /    |
|______\_/______|

The above circle (somewhat) is the actual radius where you want to find locations within, based upon location X. This is too hard to accomplish straight out of your DB, so what we actually fetch from the DB is the surrounding square. As you can see, it's possible that locations (like Y) fall within these max & min boundaries, though they aren't actually withing the requested radius. These can later be filtered out through PHP though.

To tackle this last issue, you could loop all results and calculate the exact distance between both your root location, and the close matches found, to calculate if they're actually within your radius. For that, you could use this code:

public static function getDistance($lat1, $lng1, $lat2, $lng2, $unit = 'km')
{
    // radius of earth; @note: the earth is not perfectly spherical, but this is considered the 'mean radius'
    if ($unit == 'km') $radius = 6371.009; // in kilometers
    elseif ($unit == 'mi') $radius = 3958.761; // in miles

    // convert degrees to radians
    $lat1 = deg2rad((float) $lat1);
    $lng1 = deg2rad((float) $lng1);
    $lat2 = deg2rad((float) $lat2);
    $lng2 = deg2rad((float) $lng2);

    // great circle distance formula
    return $radius * acos(sin($lat1) * sin($lat2) + cos($lat1) * cos($lat2) * cos($lng1 - $lng2));
}

This will calculate the (quasi) exact distance between location X and location Y, and then you can filter out exactly those cities that were near enough to pass the rough db-fetch, but not just near enough to actually be within your bounds.

like image 77
matthiasmullie Avatar answered Sep 20 '22 13:09

matthiasmullie