Make "nearby users" show all those within 50km rather than all those

within a degree of latitude and longitude.
This commit is contained in:
Tom Hughes 2007-06-20 17:04:29 +00:00
parent 788161010f
commit b982bc6b19
4 changed files with 48 additions and 9 deletions

View file

@ -59,15 +59,23 @@ class User < ActiveRecord::Base
return el1 return el1
end end
def nearby(lat_range=1, lon_range=1) def nearby(radius = 50)
if self.home_lon and self.home_lat if self.home_lon and self.home_lat
nearby = User.find(:all, :conditions => "#{self.home_lon} > home_lon - #{lon_range} and #{self.home_lon} < home_lon + #{lon_range} and #{self.home_lat} > home_lat - #{lat_range} and #{self.home_lat} < home_lat + #{lat_range} and data_public = 1 and id != #{self.id}") gc = OSM::GreatCircle.new(self.home_lat, self.home_lon)
bounds = gc.bounds(radius)
nearby = User.find(:all, :conditions => "home_lat between #{bounds[:minlat]} and #{bounds[:maxlat]} and home_lon between #{bounds[:minlon]} and #{bounds[:maxlon]} and data_public = 1 and id != #{self.id}")
nearby.delete_if { |u| gc.distance(u.home_lat, u.home_lon) > radius }
nearby.sort! { |u1,u2| gc.distance(u1.home_lat, u1.home_lon) <=> gc.distance(u2.home_lat, u2.home_lon) }
else else
nearby = [] nearby = []
end end
return nearby return nearby
end end
def distance(nearby_user)
return OSM::GreatCircle.new(self.home_lat, self.home_lon).distance(nearby_user.home_lat, nearby_user.home_lon)
end
def self.has_messages? def self.has_messages?
if Message.fdhjklsafind_by_to_user_id(self.id) if Message.fdhjklsafind_by_to_user_id(self.id)
return true return true

View file

@ -34,13 +34,15 @@
<table> <table>
<tr> <tr>
<th>Name</th> <th>Name</th>
<th>Distance</th>
<th>Contact</th> <th>Contact</th>
</tr> </tr>
<% @user.nearby(1,1).each do |nearby| %> <% @user.nearby.each do |nearby| %>
<% nearest_str += "nearest.push( { 'display_name' : '#{nearby.display_name}', 'home_lat' : #{nearby.home_lat}, 'home_lon' : #{nearby.home_lon} } );\n" %> <% nearest_str += "nearest.push( { 'display_name' : '#{nearby.display_name}', 'home_lat' : #{nearby.home_lat}, 'home_lon' : #{nearby.home_lon} } );\n" %>
<tr> <tr>
<td><%= link_to nearby.display_name, :controller => 'user', :action => 'view', :display_name => nearby.display_name %></td> <td><%= link_to nearby.display_name, :controller => 'user', :action => 'view', :display_name => nearby.display_name %></td>
<td><%= @user.distance(nearby).round %>km away</td>
<td><%= link_to 'send message', :controller => 'message', :action => 'new', :user_id => nearby.id %></td> <td><%= link_to 'send message', :controller => 'message', :action => 'new', :user_id => nearby.id %></td>
</tr> </tr>
<%end%> <%end%>

View file

@ -75,9 +75,10 @@
There are no users who admit to mapping nearby. There are no users who admit to mapping nearby.
<% else %> <% else %>
<table id="nearbyusers"> <table id="nearbyusers">
<% @this_user.nearby(1,1).each do |nearby| %> <% @this_user.nearby.each do |nearby| %>
<tr> <tr>
<td class="username"><%= link_to nearby.display_name, :controller => 'user', :action => 'view', :display_name => nearby.display_name %></td> <td class="username"><%= link_to nearby.display_name, :controller => 'user', :action => 'view', :display_name => nearby.display_name %></td>
<td><%= @this_user.distance(nearby).round %>km away</td>
<td class="message">(<%= link_to 'send message', :controller => 'message', :action => 'new', :user_id => nearby.id %>)</td> <td class="message">(<%= link_to 'send message', :controller => 'message', :action => 'new', :user_id => nearby.id %>)</td>
</tr> </tr>
<%end%> <%end%>

View file

@ -241,6 +241,34 @@ module OSM
end end
class GreatCircle
include Math
# initialise with a base position
def initialize(lat, lon)
@lat = lat * PI / 180
@lon = lon * PI / 180
end
# get the distance from the base position to a given position
def distance(lat, lon)
lat = lat * PI / 180
lon = lon * PI / 180
return 6372.795 * 2 * asin(sqrt(sin((lat - @lat) / 2) ** 2 + cos(@lat) * cos(lat) * sin((lon - @lon)/2) ** 2))
end
# get the worst case bounds for a given radius from the base position
def bounds(radius)
latradius = 2 * asin(sqrt(sin(radius / 6372.795 / 2) ** 2))
lonradius = 2 * asin(sqrt(sin(radius / 6372.795 / 2) ** 2 / cos(@lat) ** 2))
minlat = (@lat - latradius) * 180 / PI
maxlat = (@lat + latradius) * 180 / PI
minlon = (@lon - lonradius) * 180 / PI
maxlon = (@lon + lonradius) * 180 / PI
return { :minlat => minlat, :maxlat => maxlat, :minlon => minlon, :maxlon => maxlon }
end
end
class GeoRSS class GeoRSS
def initialize(feed_title='OpenStreetMap GPS Traces', feed_description='OpenStreetMap GPS Traces', feed_url='http://www.openstreetmap.org/traces/') def initialize(feed_title='OpenStreetMap GPS Traces', feed_description='OpenStreetMap GPS Traces', feed_url='http://www.openstreetmap.org/traces/')
@doc = XML::Document.new @doc = XML::Document.new