+ if place
+ distance = format_distance(place.attributes["approxdistance"].to_i)
+ direction = format_direction(place.attributes["direction"].to_i)
+ placename = place.attributes["name"].to_s
+ suffix = ", #{distance} #{direction} of #{placename}"
+ else
+ suffix = ""
+ end
+ results.push({:lat => lat, :lon => lon, :zoom => zoom,
+ :prefix => prefix, :name => name, :suffix => suffix,
+ :description => description})
+ end
+
+ return { :source => "OpenStreetMap Namefinder", :url => "http://gazetteer.openstreetmap.org/namefinder/", :results => results }
+ rescue Exception => ex
+ return { :source => "OpenStreetMap Namefinder", :url => "http://gazetteer.openstreetmap.org/namefinder/", :error => "Error contacting gazetteer.openstreetmap.org: #{ex.to_s}" }
+ end
+
+ def search_geonames(query)
+ results = Array.new
+
+ # ask geonames.org
+ response = fetch_xml("http://ws.geonames.org/search?q=#{escape_query(query)}&maxRows=20")
+
+ # parse the response
+ response.elements.each("geonames/geoname") do |geoname|
+ lat = geoname.get_text("lat").to_s
+ lon = geoname.get_text("lng").to_s
+ name = geoname.get_text("name").to_s
+ country = geoname.get_text("countryName").to_s
+ results.push({:lat => lat, :lon => lon, :zoom => GEONAMES_ZOOM,
+ :name => name,
+ :suffix => ", #{country}"})
+ end
+
+ return { :source => "GeoNames", :url => "http://www.geonames.org/", :results => results }
+ rescue Exception => ex
+ return { :source => "GeoNames", :url => "http://www.geonames.org/", :error => "Error contacting ws.geonames.org: #{ex.to_s}" }
+ end
+
+ def description_osm_namefinder(types, lat, lon, max)
+ results = Array.new
+
+ # ask OSM namefinder
+ response = fetch_xml("http://gazetteer.openstreetmap.org/namefinder/search.xml?find=#{types}+near+#{lat},#{lon}&max=#{max}")
+
+ # parse the response
+ response.elements.each("searchresults/named") do |named|
+ lat = named.attributes["lat"].to_s
+ lon = named.attributes["lon"].to_s
+ zoom = named.attributes["zoom"].to_s
+ place = named.elements["place/named"] || named.elements["nearestplaces/named"]
+ type = named.attributes["info"].to_s
+ name = named.attributes["name"].to_s
+ description = named.elements["description"].to_s
+ distance = format_distance(place.attributes["approxdistance"].to_i)
+ direction = format_direction((place.attributes["direction"].to_i - 180) % 360)
+ prefix = "#{distance} #{direction} of #{type} "
+ results.push({:lat => lat, :lon => lon, :zoom => zoom,
+ :prefix => prefix.capitalize, :name => name,
+ :description => description})
+ end
+
+ return { :type => types.capitalize, :source => "OpenStreetMap Namefinder", :url => "http://gazetteer.openstreetmap.org/namefinder/", :results => results }
+ rescue Exception => ex
+ return { :type => types.capitalize, :source => "OpenStreetMap Namefinder", :url => "http://gazetteer.openstreetmap.org/namefinder/", :error => "Error contacting gazetteer.openstreetmap.org: #{ex.to_s}" }
+ end
+
+ def description_geonames(lat, lon)
+ results = Array.new
+
+ # ask geonames.org
+ response = fetch_xml("http://ws.geonames.org/countrySubdivision?lat=#{lat}&lng=#{lon}")
+
+ # parse the response
+ response.elements.each("geonames/countrySubdivision") do |geoname|
+ name = geoname.get_text("adminName1").to_s
+ country = geoname.get_text("countryName").to_s
+ results.push({:prefix => "#{name}, #{country}"})
+ end
+
+ return { :type => "Location", :source => "GeoNames", :url => "http://www.geonames.org/", :results => results }
+ rescue Exception => ex
+ return { :type => "Location", :source => "GeoNames", :url => "http://www.geonames.org/", :error => "Error contacting ws.geonames.org: #{ex.to_s}" }
+ end
+
+ def fetch_text(url)
+ return Net::HTTP.get(URI.parse(url))
+ end
+
+ def fetch_xml(url)
+ return REXML::Document.new(fetch_text(url))
+ end
+
+ def format_distance(distance)
+ return "less than 1km" if distance == 0
+ return "about #{distance}km"
+ end
+
+ def format_direction(bearing)
+ return "south-west" if bearing >= 22.5 and bearing < 67.5
+ return "south" if bearing >= 67.5 and bearing < 112.5
+ return "south-east" if bearing >= 112.5 and bearing < 157.5
+ return "east" if bearing >= 157.5 and bearing < 202.5
+ return "north-east" if bearing >= 202.5 and bearing < 247.5
+ return "north" if bearing >= 247.5 and bearing < 292.5
+ return "north-west" if bearing >= 292.5 and bearing < 337.5
+ return "west"
+ end
+
+ def count_results(results)
+ count = 0
+
+ results.each do |source|
+ count += source[:results].length if source[:results]