165 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /*
 | |
|   +--------------------------------------------------------------------+
 | |
|   | CiviCRM version 4.7                                                |
 | |
|   +--------------------------------------------------------------------+
 | |
|   | Copyright CiviCRM LLC (c) 2004-2017                                |
 | |
|   +--------------------------------------------------------------------+
 | |
|   | This file is a part of CiviCRM.                                    |
 | |
|   |                                                                    |
 | |
|   | CiviCRM is free software; you can copy, modify, and distribute it  |
 | |
|   | under the terms of the GNU Affero General Public License           |
 | |
|   | Version 3, 19 November 2007 and the CiviCRM Licensing Exception.   |
 | |
|   |                                                                    |
 | |
|   | CiviCRM is distributed in the hope that it will be useful, but     |
 | |
|   | WITHOUT ANY WARRANTY; without even the implied warranty of         |
 | |
|   | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.               |
 | |
|   | See the GNU Affero General Public License for more details.        |
 | |
|   |                                                                    |
 | |
|   | You should have received a copy of the GNU Affero General Public   |
 | |
|   | License and the CiviCRM Licensing Exception along                  |
 | |
|   | with this program; if not, contact CiviCRM LLC                     |
 | |
|   | at info[AT]civicrm[DOT]org. If you have questions about the        |
 | |
|   | GNU Affero General Public License or the licensing of CiviCRM,     |
 | |
|   | see the CiviCRM license FAQ at http://civicrm.org/licensing        |
 | |
|   +--------------------------------------------------------------------+
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  * @package CRM
 | |
|  * @copyright CiviCRM LLC (c) 2004-2017
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * Class that uses google geocoder
 | |
|  */
 | |
| class CRM_Utils_Geocode_Google {
 | |
| 
 | |
|   /**
 | |
|    * Server to retrieve the lat/long
 | |
|    *
 | |
|    * @var string
 | |
|    */
 | |
|   static protected $_server = 'maps.googleapis.com';
 | |
| 
 | |
|   /**
 | |
|    * Uri of service.
 | |
|    *
 | |
|    * @var string
 | |
|    */
 | |
|   static protected $_uri = '/maps/api/geocode/xml?sensor=false&address=';
 | |
| 
 | |
|   /**
 | |
|    * Function that takes an address object and gets the latitude / longitude for this
 | |
|    * address. Note that at a later stage, we could make this function also clean up
 | |
|    * the address into a more valid format
 | |
|    *
 | |
|    * @param array $values
 | |
|    * @param bool $stateName
 | |
|    *
 | |
|    * @return bool
 | |
|    *   true if we modified the address, false otherwise
 | |
|    */
 | |
|   public static function format(&$values, $stateName = FALSE) {
 | |
|     // we need a valid country, else we ignore
 | |
|     if (empty($values['country'])) {
 | |
|       return FALSE;
 | |
|     }
 | |
| 
 | |
|     $config = CRM_Core_Config::singleton();
 | |
| 
 | |
|     $add = '';
 | |
| 
 | |
|     if (!empty($values['street_address'])) {
 | |
|       $add = urlencode(str_replace('', '+', $values['street_address']));
 | |
|       $add .= ',+';
 | |
|     }
 | |
| 
 | |
|     $city = CRM_Utils_Array::value('city', $values);
 | |
|     if ($city) {
 | |
|       $add .= '+' . urlencode(str_replace('', '+', $city));
 | |
|       $add .= ',+';
 | |
|     }
 | |
| 
 | |
|     if (!empty($values['state_province']) || (!empty($values['state_province_id']) && $values['state_province_id'] != 'null')) {
 | |
|       if (!empty($values['state_province_id'])) {
 | |
|         $stateProvince = CRM_Core_DAO::getFieldValue('CRM_Core_DAO_StateProvince', $values['state_province_id']);
 | |
|       }
 | |
|       else {
 | |
|         if (!$stateName) {
 | |
|           $stateProvince = CRM_Core_DAO::getFieldValue('CRM_Core_DAO_StateProvince',
 | |
|             $values['state_province'],
 | |
|             'name',
 | |
|             'abbreviation'
 | |
|           );
 | |
|         }
 | |
|         else {
 | |
|           $stateProvince = $values['state_province'];
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // dont add state twice if replicated in city (happens in NZ and other countries, CRM-2632)
 | |
|       if ($stateProvince != $city) {
 | |
|         $add .= '+' . urlencode(str_replace('', '+', $stateProvince));
 | |
|         $add .= ',+';
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (!empty($values['postal_code'])) {
 | |
|       $add .= '+' . urlencode(str_replace('', '+', $values['postal_code']));
 | |
|       $add .= ',+';
 | |
|     }
 | |
| 
 | |
|     if (!empty($values['country'])) {
 | |
|       $add .= '+' . urlencode(str_replace('', '+', $values['country']));
 | |
|     }
 | |
| 
 | |
|     if (!empty($config->geoAPIKey)) {
 | |
|       $add .= '&key=' . urlencode($config->geoAPIKey);
 | |
|     }
 | |
| 
 | |
|     $query = 'https://' . self::$_server . self::$_uri . $add;
 | |
| 
 | |
|     require_once 'HTTP/Request.php';
 | |
|     $request = new HTTP_Request($query);
 | |
|     $request->sendRequest();
 | |
|     $string = $request->getResponseBody();
 | |
| 
 | |
|     libxml_use_internal_errors(TRUE);
 | |
|     $xml = @simplexml_load_string($string);
 | |
|     CRM_Utils_Hook::geocoderFormat('Google', $values, $xml);
 | |
|     if ($xml === FALSE) {
 | |
|       // account blocked maybe?
 | |
|       CRM_Core_Error::debug_var('Geocoding failed.  Message from Google:', $string);
 | |
|       return FALSE;
 | |
|     }
 | |
| 
 | |
|     if (isset($xml->status)) {
 | |
|       if ($xml->status == 'OK' &&
 | |
|         is_a($xml->result->geometry->location,
 | |
|           'SimpleXMLElement'
 | |
|         )
 | |
|       ) {
 | |
|         $ret = $xml->result->geometry->location->children();
 | |
|         if ($ret->lat && $ret->lng) {
 | |
|           $values['geo_code_1'] = (float) $ret->lat;
 | |
|           $values['geo_code_2'] = (float) $ret->lng;
 | |
|           return TRUE;
 | |
|         }
 | |
|       }
 | |
|       elseif ($xml->status == 'ZERO_RESULTS') {
 | |
|         // reset the geo code values if we did not get any good values
 | |
|         $values['geo_code_1'] = $values['geo_code_2'] = 'null';
 | |
|         return FALSE;
 | |
|       }
 | |
|       else {
 | |
|         CRM_Core_Error::debug_var("Geocoding failed. Message from Google: ({$xml->status})", (string ) $xml->error_message);
 | |
|         $values['geo_code_1'] = $values['geo_code_2'] = 'null';
 | |
|         $values['geo_code_error'] = $xml->status;
 | |
|         return FALSE;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
| }
 |