Pergunta

Eu tenho um modelo de usuário com atributos 'primeiro' e 'última' Assim, por exemplo User.first.first # => "Charlie" User.first.last # => "Brown"

Este modelo User também tem um atributo virtual 'full_name'

#user.rb
def full_name
  [first,last].join(' ')
end

def full_name=(name) #don't know what to do with people w/ middle names
  split = name.split(' ')
  self.first = split[0]
  self.last = split[1]
end

Assim, por exemplo:

User.first.full_name = "Charlie Brown" #=> "Charlie Brown"
User.first.full_name = "Homer Simpson" #=> "Home Simpson"
User.first.save
User.first.first #=> "Homer"
User.first.last #=> "Simpson"

Seria tão bom se eu poderia procurar por esse atributo virtual assim, por exemplo para a descoberta dinâmica:

User.find_by_full_name('Home Simpson') # this doesn't work

Exemplo de condições em localizar:

User.all(:conditions => ['full_name LIKE ?', query]) #this doesn't work

Estou esperando para encontrar pelo menos algumas formas de linguagem SQL que pode fazer isso; se houver um achado atributo virtual dinâmico, também, que é fonte de baunilha extra sobre o strudel. (Qualquer um que tem esta neste inverno?)

Eu também estava preocupado com um nome que está sendo procurado, por exemplo, "Holmes" só pode ser pesquisado na coluna 'primeiro', mas não o 'último' para recuperar, por exemplo, User.first.full_name #=> "Sherlock Holmes".

Eu tentei fazer uma pesquisa mais abrangente:

user.rb

def self.find_by_full_name(name) #returns an array of User model
  return all if name.blank?

  split = name.split(' ', 2)
  output = []
  if split.length > 1
    with_scope( :find => { :conditions => ['first LIKE ?', "%#{split[0]}%"] }) do
      output << all(:conditions => ['last LIKE ?', "%#{split[1]}%"])
      output.flatten!
    end
  elsif split.length == 1
    output << all(:conditions => ['first LIKE ?', "%#{split[0]}%"])
    output << all(:conditions => ['last LIKE ?', "%#{split[0]}%"])
    output.flatten!
  end
end

Por exemplo

User.find_by_full_name("John").map(&:full_name) #=> ["John Resig", "John Doe"]
User.find_by_full_name("Doe").map(&:full_name) #=> ["John Doe", "Philips Doeringer"]
User.find_by_full_name("John Doe").map(&:full_name) #=> ["John Doe"]

Mas eu apenas pensei que o método find_by_full_name aqui é um pouco pesado.

Quer dizer, se eu tivesse um full_name coluna que começa o jogo cada vez por um após salvar filtro com a concatenação de primeiro e último. Assim encontrar o nome de uma pessoa, especialmente com incoerência na memória desta pessoa, é útil. Então, se eu me lembrava 'Doe', em que a pessoa é o primeiro ou o último nome, eu sempre pode fazer um simples User.find_by_full_name ( 'Doe') para retornar o maior número possível de mais pin-lo para baixo.

E já que é uma coluna, eu posso busca-lo em um find (: condições [...]) cláusula se eu tiver que fazer algo como Project.find(:all,:include => :users, :conditions=>['users.full_name LIKE ?', query]) onde

#project.rb
has_many :assignments
has_many :users, :through=>:assignments

#user.rb
has_many :assignments
has_many :projects, :through => :assignments

#assignment.rb
belongs_to :user
belongs_to :project

Boas festas N

Foi útil?

Solução

Você pode utilizar um named_scope em sua user.rb:

named_scope :find_by_full_name, lambda {|full_name| 
  {:conditions => {:first => full_name.split(' ').first, 
     :last => full_name.split(' ').last}}
}

Em seguida, você pode fazer User.find_by_full_name('John Carver')

novidades em resposta a mudanças na exigência

named_scope :find_by_full_name, lambda {|full_name| 
  {:conditions => ["first LIKE '%?%' or last LIKE '%?%'", 
    full_name.split(' ').first, full_name.split(' ').last]}}

Outras dicas

Eu encontrei a resposta de Jim útil também. Obrigado. Eu faria uma ligeira alteração no entanto. Este código atual faz com que você perder quaisquer nomes do meio. O que tenho a seguir é um pouco confuso, mas preserva nomes do meio e sobrenomes compostos (acho que Jean-Claude van Damme). Tudo após o primeiro nome vai no campo last_name.

named_scope :find_by_full_name, lambda { |full_name| 
 {:conditions => {:first_name => full_name.split(' ').first, 
   :last_name => full_name.split(' ')[1, full_name.split(' ').length-1].join(' ')}
   }
}

É claro que qualquer forma mais limpa de fazer isso é bem-vindo.

Licenciado em: CC-BY-SA com atribuição
Não afiliado a StackOverflow
scroll top