Итерация по коллекции has_many в named_scope - PullRequest
2 голосов
/ 16 марта 2011

Вот мои модели:

class Message < ActiveRecord::Base
  has_many :comments

  attr_accessible :read #bool

  def unread_comments?
    comments.each { |comment| return true unless comment.read?}

    false
  end
end

class Comment < ActiveRecord::Base
  belongs_to :message

  attr_accessible :read #bool
end

Вот что я хочу сделать: я хотел бы создать named_scope в сообщении с именем unread, который в основном возвращает true, если любое из сообщенийкомментарии не прочитаны или само сообщение не прочитано.Есть ли способ, которым я могу это сделать?

Ответы [ 3 ]

3 голосов
/ 16 марта 2011
class Message < AR::Base
  ...
  def unread?
    !self.read && !self.comments.all?(&:read)
  end
end
2 голосов
/ 16 марта 2011

Как насчет этого:

named_scope :unread, :conditions => ['messages.read = ? OR comments.read = ?', 
                                      false, false],
                     :include => :comments
1 голос
/ 24 марта 2011

Почему названная область?Вы можете просто сделать

def has_unread
  !self.read || unread_comments?
end

или, если это должен быть метод класса

def self.has_unread(message)
  msg = Message.find(message.id)
  msg.read == false && msg.unread_comments?
end

Вы действительно хотите использовать именованные области, однако вызахочу что-то вроде этого:

scope :get_unreads, lambda {|message|
  { :joins => :comments,
    :conditions => {:comments => {:message_id => message.id}, :message => ['read',0]},
    :select     => "DISTINCT `clients`.*" # kill duplicates
  }
}
def self.has_unread?(message)
  self.get_unreads(message).exists? && self.find(message.id).read
end
...