Поиск с Geocoder и Sunspot? - PullRequest
       6

Поиск с Geocoder и Sunspot?

2 голосов
/ 12 декабря 2011

Я использую гем Geocoder и Sunspot в моем приложении, и у меня есть поле с именем :search_near_address, которое должно дать пользователям возможность ввести адрес, который они хотятискать рядом с X amount of miles.То, что я пытаюсь отобразить, - это мои магазины :address, которые можно использовать для поля :search_near_address.Таким образом, пользователи могут ввести адрес (например, 451 University Avenue, Пало-Альто, Калифорния) в поле :search_near_address, и он будет искать в радиусе 50 миль.

Ответ

Sunspot 1.2.1


class Store < ActiveRecord::Base
   attr_accessible :address, :latitude, :longitude
   has_many :products
   geocoded_by :address
   after_validation :geocode
   reverse_geocoded_by :latitude, :longitude
   after_validation :reverse_geocode      
end

class Product < ActiveRecord::Base
   belongs_to :store

   searchable do # Searching with product model.
     string :search_near # For rake sunspot:reindex
     location :location
   end

   def search_near_address
 store.address if store # You may have to use the "if store".
   end

   def location
    # may need the "if store" after longitude)....
Sunspot::Util::Coordinates.new(store.latitude, store.longitude)
   end
end

class SearchController < ApplicationController

  def index
    @search = Product.search do |q| # Search with sunspot
      q.fulltext params[:search]
      q.with(:location).near(*Geocoder.coordinates(params[:search_near_address]), :precision => 4) if params[:search_near_address].present?
    end 
    @products = @search.results # Return results from Product.search block.
  end
end

# search/index/html.erb
<%= form_tag results_search_index_path, :method => 'get' do %>
    <%= text_field_tag :search, params[:search] %>
    <%= text_field_tag :search_near_address, params[:search_near_address] %>
    <%= submit_tag "Go", :name => nil %>
<% end %>

Ответы [ 2 ]

7 голосов
/ 17 декабря 2011

Во-первых, то, что сказал @m_x, правильно, вы не можете ожидать, что назначение Store # рядом с результатами для объекта @search сработает ...

Просто читать документы о геопространстве в https://github.com/sunspot/sunspot, вы можете ясно видеть, что вам не хватает нескольких вещей:

Вы должны объявить свое геопространственное поле, используя поля широты и долготы, которые требуются геокодеру:

class Product < ActiveRecord::Base
   belongs_to :store

   searchable do # for Sunspot search.
     string :search_near
     latlon(:location) { 
       Sunspot::Util::Coordinates.new(category.latitude, category.longitude)
     }
   end

   def search_near
     #I changed this because business_store will not work out.
     store.address
   end
end

Затем вы можете искать(после индексации чего-либо), например:

class SearchController < ApplicationController

  def index
    # Search with sunspot
    @search = Sunspot.search(Product) do
      fulltext params[:search]

      # The "*" pop off the elements of the array that 
      # Geocoder.coordinates returns.
      with(:location).near(*Geocoder.coordinates(params[:search_near]), 
                           :precision => 6)

      # NOTE: You could also use the in_radius method but only in the pre-release version:
      # with(:location).in_radius(*Geocoder.coordinates(params[:search_near]), 100) if params[:search_near].present?
    end

    # Return results from Product.search block.
    @products = @search.results 
  end
end

Также читайте о: точности здесь http://sunspot.github.com/docs/Sunspot/DSL/RestrictionWithNear.html#near-instance_method

О геокодере https://github.com/alexreisner/geocoder

3 голосов
/ 15 декабря 2011

Не уверен, что я полностью понимаю, что вы пытаетесь сделать с помощью вышеизложенного, но у меня похожая ситуация, когда я пытался заставить геокодер и солнечное пятно играть вместе.

Это еще не проверено, но это мое нынешнее мышление....

Event.search do
      keywords  search_terms
      any_of do
        with(:id).any_of(Event.near(coordinates,50).map(&:id))
      end
end.results

По сути, я пытаюсь объединить два набора результатов, чтобы вернуть только те результаты, которые соответствуют терминам расстояния и ключевого слова.

Надеюсь, это поможеткому-то хотя бы.

...