{ Map Viewer Geolocation Engine for geonames.org Copyright (C) 2011 Maciej Kaczkowski / keit.co This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. } unit mvGeoNames; interface uses SysUtils, Classes, StrUtils, mvTypes, mvDownloadEngine; type TNameFoundEvent = procedure (const AName: string; const ADescr: String; const ALoc: TRealPoint) of object; TStringArray = array of string; TResRec = record Name: String; Descr: String; Loc: TRealPoint; end; { TMVGeoNames } TMVGeoNames = class(TComponent) private FLocationName: string; FInResTable: Boolean; FInDataRows: Boolean; FNamePending: Boolean; FLongitudePending: Boolean; FLatitudePending: Boolean; FCol: Integer; FCountry: String; FSmall: Boolean; FFirstLocation: TResRec; FFoundLocation: TResRec; FOnNameFound: TNameFoundEvent; procedure FoundTagHandler(NoCaseTag, {%H-}ActualTag: string); procedure FoundTextHandler(AText: String); function Parse(AStr: PChar): TRealPoint; // function RemoveTag(const str: String): TStringArray; public function Search(ALocationName: String; ADownloadEngine: TMvCustomDownloadEngine): TRealPoint; published property LocationName: string read FLocationName; property OnNameFound: TNameFoundEvent read FOnNameFound write FOnNameFound; end; implementation uses FastHtmlParser; const SEARCH_URL = 'http://geonames.org/search.html?q=%s'; //&country=%s'; function CleanLocationName(x: string): string; var i: Integer; begin Result := ''; for i := 1 to Length(x) do begin if x[i] in ['A'..'Z', 'a'..'z', '0'..'9'] then Result := Result + x[i] else Result := Result + '+' end; end; { TMVGeoNames } procedure TMvGeoNames.FoundTagHandler(NoCaseTag, ActualTag: String); begin if not FInResTable and (NoCaseTag = '') then begin FInResTable := true; FInDataRows := false; FNamePending := false; FLatitudePending := false; FLongitudePending := false; FSmall := false; end else if FInResTable and (NoCaseTag = '
') then FInResTable := false; if FInResTable then begin if NoCaseTag = '' then FInDataRows := true; if FInDataRows then begin if NoCaseTag = '' then begin FCol := 0; with FFoundLocation do begin Name := ''; Descr := ''; Loc.Lon := 0; Loc.Lat := 0; end; end; if NoCaseTag = '' then inc(FCol); if FCol = 2 then begin if not FNamePending and (pos('') then FNamePending := false; if not FLatitudePending and (NoCaseTag = '') then FLatitudePending := true else if FLatitudePending and (NoCaseTag = '') then FLatitudePending := false; if not FLongitudePending and (NoCaseTag = '') then FLongitudePending := true else if FLongitudePending and (NoCasetag = '') then FLongitudePending := false; end; if FCol = 3 then if not FSmall and (NoCaseTag = '') then FSmall := true else if FSmall and (NoCaseTag = '') then FSmall := false; if NoCaseTag = '' then begin if (FFirstLocation.Name = '') then FFirstLocation := FFoundLocation; if Assigned(FOnNameFound) and (FFoundLocation.Name <> '') then with FFoundLocation do FOnNameFound(Name, Descr, Loc); end; end; end; end; procedure TMvGeoNames.FoundTextHandler(AText: String); var code: Integer; begin if not FInDataRows or (AText = #10) then exit; if FNamePending then FFoundLocation.Name := AText else if FLatitudePending then val(AText, FFoundLocation.Loc.Lat, code) else if FLongitudePending then val(AText, FFoundLocation.Loc.Lon, code) else if (FCol = 3) and not FSmall then FCountry := FCountry + AText else if FCol = 4 then begin if FFoundLocation.Descr = '' then FFoundLocation.Descr := AText else FFoundLocation.Descr := FFoundLocation.Descr + ', ' + aText; if FCountry <> '' then FFoundLocation.Name := FFoundLocation.Name + ' (' + FCountry + ')'; FCountry := ''; end; end; function TMVGeonames.Parse(AStr: PChar): TRealPoint; var parser: THtmlParser; begin FFirstLocation.Name := ''; parser := THtmlParser.Create(AStr); try parser.OnFoundTag := @FoundTagHandler; parser.OnFoundText := @FoundTextHandler; parser.Exec; Result := FFirstLocation.Loc; finally parser.Free; end; end; function TMVGeoNames.Search(ALocationName: String; ADownloadEngine: TMvCustomDownloadEngine): TRealPoint; var s: string; function gs(id: string; Start: integer): string; var i: Integer; ln: Integer; begin Result := ''; ln := Length(s); i := PosEx(id, s,start) + Length(id); while (s[i] <> '<') and (i < ln) do begin if s[i] = '.' then Result := Result + FormatSettings.DecimalSeparator else Result := Result + s[i]; Inc(i); end; end; var ms: TMemoryStream; url: String; begin FLocationName := ALocationName; ms := TMemoryStream.Create; try url := Format(SEARCH_URL, [CleanLocationName(FLocationName)]); ADownloadEngine.DownloadFile(url, ms); ms.Position := 0; SetLength(s, ms.Size); ms.Read(s[1], ms.Size); finally ms.Free; end; Result := Parse(PChar(s)); end; end.