3  * SPDX-License-Identifier: GPL-2.0-only
 
   5  * This file is part of Nominatim. (https://nominatim.org)
 
   7  * Copyright (C) 2022 by the Nominatim developer community.
 
   8  * For a full list of authors see the git log.
 
  13 require_once(CONST_LibDir.'/ClassTypes.php');
 
  16  * Detailed list of address parts for a single result
 
  21     private $aAddressLines;
 
  23     public function __construct(&$oDB, $iPlaceID, $sHousenumber, $mLangPref)
 
  25         $this->iPlaceID = $iPlaceID;
 
  27         if (is_array($mLangPref)) {
 
  28             $mLangPref = $oDB->getArraySQL($oDB->getDBQuotedList($mLangPref));
 
  31         if (!isset($sHousenumber)) {
 
  36         $sSQL .= ' get_name_by_language(name,'.$mLangPref.') as localname';
 
  37         $sSQL .= ' FROM get_addressdata('.$iPlaceID.','.$sHousenumber.')';
 
  38         $sSQL .= ' ORDER BY rank_address DESC, isaddress DESC';
 
  40         $this->aAddressLines = $oDB->getAll($sSQL);
 
  43     private static function isAddress($aLine)
 
  45         return $aLine['isaddress'] || $aLine['type'] == 'country_code';
 
  48     public function getAddressDetails($bAll = false)
 
  51             return $this->aAddressLines;
 
  54         return array_filter($this->aAddressLines, array(__CLASS__, 'isAddress'));
 
  57     public function getLocaleAddress()
 
  62         foreach ($this->aAddressLines as $aLine) {
 
  63             if ($aLine['isaddress'] && $sPrevResult != $aLine['localname']) {
 
  64                 $sPrevResult = $aLine['localname'];
 
  65                 $aParts[] = $sPrevResult;
 
  69         return join(', ', $aParts);
 
  72     public function getAddressNames()
 
  76         foreach ($this->aAddressLines as $aLine) {
 
  77             if (!self::isAddress($aLine)) {
 
  81             $sTypeLabel = ClassTypes\getLabelTag($aLine);
 
  84             if (isset($aLine['localname']) && $aLine['localname']!=='') {
 
  85                 $sName = $aLine['localname'];
 
  86             } elseif (isset($aLine['housenumber']) && $aLine['housenumber']!=='') {
 
  87                 $sName = $aLine['housenumber'];
 
  91                 && (!isset($aAddress[$sTypeLabel])
 
  92                     || $aLine['class'] == 'place')
 
  94                 $aAddress[$sTypeLabel] = $sName;
 
  96                 if (!empty($aLine['name'])) {
 
  97                     $this->addSubdivisionCode($aAddress, $aLine['admin_level'], $aLine['name']);
 
 106      * Annotates the given json with geocodejson address information fields.
 
 108      * @param array  $aJson  Json hash to add the fields to.
 
 110      * Geocodejson has the following fields:
 
 111      *  street, locality, postcode, city, district,
 
 112      *  county, state, country
 
 114      * Postcode and housenumber are added by type, district is not used.
 
 115      * All other fields are set according to address rank.
 
 117     public function addGeocodeJsonAddressParts(&$aJson)
 
 119         foreach (array_reverse($this->aAddressLines) as $aLine) {
 
 120             if (!$aLine['isaddress']) {
 
 124             if (!isset($aLine['localname']) || $aLine['localname'] == '') {
 
 128             if ($aLine['type'] == 'postcode' || $aLine['type'] == 'postal_code') {
 
 129                 $aJson['postcode'] = $aLine['localname'];
 
 133             if ($aLine['type'] == 'house_number') {
 
 134                 $aJson['housenumber'] = $aLine['localname'];
 
 138             if ($this->iPlaceID == $aLine['place_id']) {
 
 142             $iRank = (int)$aLine['rank_address'];
 
 144             if ($iRank > 25 && $iRank < 28) {
 
 145                 $aJson['street'] = $aLine['localname'];
 
 146             } elseif ($iRank >= 22 && $iRank <= 25) {
 
 147                 $aJson['locality'] = $aLine['localname'];
 
 148             } elseif ($iRank >= 17 && $iRank <= 21) {
 
 149                 $aJson['district'] = $aLine['localname'];
 
 150             } elseif ($iRank >= 13 && $iRank <= 16) {
 
 151                 $aJson['city'] = $aLine['localname'];
 
 152             } elseif ($iRank >= 10 && $iRank <= 12) {
 
 153                 $aJson['county'] = $aLine['localname'];
 
 154             } elseif ($iRank >= 5 && $iRank <= 9) {
 
 155                 $aJson['state'] = $aLine['localname'];
 
 156             } elseif ($iRank == 4) {
 
 157                 $aJson['country'] = $aLine['localname'];
 
 162     public function getAdminLevels()
 
 165         foreach (array_reverse($this->aAddressLines) as $aLine) {
 
 166             if (self::isAddress($aLine)
 
 167                 && isset($aLine['admin_level'])
 
 168                 && $aLine['admin_level'] < 15
 
 169                 && !isset($aAddress['level'.$aLine['admin_level']])
 
 171                 $aAddress['level'.$aLine['admin_level']] = $aLine['localname'];
 
 177     public function debugInfo()
 
 179         return $this->aAddressLines;
 
 182     private function addSubdivisionCode(&$aAddress, $iAdminLevel, $nameDetails)
 
 184         if (is_string($nameDetails)) {
 
 185             $nameDetails = json_decode('{' . str_replace('"=>"', '":"', $nameDetails) . '}', true);
 
 187         if (!empty($nameDetails['ISO3166-2'])) {
 
 188             $aAddress["ISO3166-2-lvl$iAdminLevel"] = $nameDetails['ISO3166-2'];